44#include "FGCondition.h"
45#include "FGPropertyValue.h"
46#include "input_output/FGXMLElement.h"
47#include "input_output/FGPropertyManager.h"
48#include "FGParameterValue.h"
59FGCondition::FGCondition(Element* element, std::shared_ptr<FGPropertyManager> PropertyManager)
60 : Logic(elUndef), TestParam1(nullptr), TestParam2(nullptr),
63 string logic = element->GetAttributeValue(
"logic");
65 if (logic ==
"OR") Logic = eOR;
66 else if (logic ==
"AND") Logic = eAND;
68 throw BaseException(
"FGCondition: unrecognized LOGIC token:'" + logic +
"'");
74 assert(Logic != elUndef);
76 for (
unsigned int i=0; i<element->GetNumDataLines(); i++) {
77 string data = element->GetDataLine(i);
78 conditions.push_back(make_shared<FGCondition>(data, PropertyManager, element));
81 Element* condition_element = element->GetElement();
82 const string& elName = element->GetName();
84 while (condition_element) {
85 string tagName = condition_element->GetName();
87 if (tagName != elName) {
88 throw BaseException(
"FGCondition: unrecognized TAG:'" + tagName +
"' in the condition statement.");
91 conditions.push_back(make_shared<FGCondition>(condition_element, PropertyManager));
92 condition_element = element->GetNextElement();
95 if (conditions.empty())
throw BaseException(
"Empty conditional");
104FGCondition::FGCondition(
const string& test, std::shared_ptr<FGPropertyManager> PropertyManager,
106 : Logic(elUndef), TestParam1(nullptr), TestParam2(nullptr),
109 static constexpr array<pair<const char*, enum eComparison>, 18> mComparison {{
130 vector<string> test_strings = split(test,
' ');
132 if (test_strings.size() == 3) {
133 TestParam1 =
new FGPropertyValue(test_strings[0], PropertyManager, el);
134 conditional = test_strings[1];
135 TestParam2 =
new FGParameterValue(test_strings[2], PropertyManager, el);
138 s <<
" Conditional test is invalid: \"" << test
139 <<
"\" has " << test_strings.size() <<
" elements in the "
140 <<
"test condition.\n";
141 throw BaseException(s.str());
144 assert(Comparison == ecUndef);
145 for(
auto& elm: mComparison) {
146 if (conditional == elm.first) {
147 Comparison = elm.second;
152 if (Comparison == ecUndef) {
153 throw BaseException(
"FGCondition: Comparison operator: \""+conditional
154 +
"\" does not exist. Please check the conditional.");
160bool FGCondition::Evaluate(
void)
const
169 for (
auto& cond: conditions) {
170 if (!cond->Evaluate()) pass =
false;
176 for (
auto& cond: conditions) {
177 if (cond->Evaluate()) pass =
true;
183 double value1 = TestParam1->GetValue();
184 double value2 = TestParam2->GetValue();
186 switch (Comparison) {
188 pass = value1 == value2;
191 pass = value1 != value2;
194 pass = value1 > value2;
197 pass = value1 >= value2;
200 pass = value1 < value2;
203 pass = value1 <= value2;
216void FGCondition::PrintCondition(
string indent)
const
220 if (!conditions.empty()) {
225 cerr <<
"unset logic for test condition" << endl;
228 scratch = indent +
"if all of the following are true: {";
231 scratch = indent +
"if any of the following are true: {";
234 scratch =
" UNKNOWN";
235 cerr <<
"Unknown logic for test condition" << endl;
237 cout << scratch << endl;
239 for (
auto& cond: conditions) {
240 cond->PrintCondition(indent +
" ");
244 cout << indent <<
"}";
247 cout << indent << TestParam1->GetName() <<
" " << conditional
248 <<
" " << TestParam2->GetName();
271void FGCondition::Debug(
int from)
273 if (debug_lvl <= 0)
return;
280 if (debug_lvl & 2 ) {
281 if (from == 0) cout <<
"Instantiated: FGCondition" << endl;
282 if (from == 1) cout <<
"Destroyed: FGCondition" << endl;
284 if (debug_lvl & 4 ) {
286 if (debug_lvl & 8 ) {
288 if (debug_lvl & 16) {
290 if (debug_lvl & 64) {