35#include "FGXMLElement.h"
37#include "input_output/string_utilities.h"
47bool Element::converterIsInitialized =
false;
48map <string, map <string, double> > Element::convert;
61 if (!converterIsInitialized) {
62 converterIsInitialized =
true;
65 convert[
"M"][
"FT"] = 3.2808399;
66 convert[
"FT"][
"M"] = 1.0/convert[
"M"][
"FT"];
67 convert[
"CM"][
"FT"] = 0.032808399;
68 convert[
"FT"][
"CM"] = 1.0/convert[
"CM"][
"FT"];
69 convert[
"KM"][
"FT"] = 3280.8399;
70 convert[
"FT"][
"KM"] = 1.0/convert[
"KM"][
"FT"];
71 convert[
"FT"][
"IN"] = 12.0;
72 convert[
"IN"][
"FT"] = 1.0/convert[
"FT"][
"IN"];
73 convert[
"IN"][
"M"] = convert[
"IN"][
"FT"] * convert[
"FT"][
"M"];
74 convert[
"M"][
"IN"] = convert[
"M"][
"FT"] * convert[
"FT"][
"IN"];
76 convert[
"M2"][
"FT2"] = convert[
"M"][
"FT"]*convert[
"M"][
"FT"];
77 convert[
"FT2"][
"M2"] = 1.0/convert[
"M2"][
"FT2"];
78 convert[
"CM2"][
"FT2"] = convert[
"CM"][
"FT"]*convert[
"CM"][
"FT"];
79 convert[
"FT2"][
"CM2"] = 1.0/convert[
"CM2"][
"FT2"];
80 convert[
"M2"][
"IN2"] = convert[
"M"][
"IN"]*convert[
"M"][
"IN"];
81 convert[
"IN2"][
"M2"] = 1.0/convert[
"M2"][
"IN2"];
82 convert[
"FT2"][
"IN2"] = 144.0;
83 convert[
"IN2"][
"FT2"] = 1.0/convert[
"FT2"][
"IN2"];
85 convert[
"IN3"][
"CC"] = 16.387064;
86 convert[
"CC"][
"IN3"] = 1.0/convert[
"IN3"][
"CC"];
87 convert[
"FT3"][
"IN3"] = 1728.0;
88 convert[
"IN3"][
"FT3"] = 1.0/convert[
"FT3"][
"IN3"];
89 convert[
"M3"][
"FT3"] = 35.3146667;
90 convert[
"FT3"][
"M3"] = 1.0/convert[
"M3"][
"FT3"];
91 convert[
"LTR"][
"IN3"] = 61.0237441;
92 convert[
"IN3"][
"LTR"] = 1.0/convert[
"LTR"][
"IN3"];
93 convert[
"GAL"][
"FT3"] = 0.133681;
94 convert[
"FT3"][
"GAL"] = 1.0/convert[
"GAL"][
"FT3"];
95 convert[
"IN3"][
"GAL"] = convert[
"IN3"][
"FT3"]*convert[
"FT3"][
"GAL"];
96 convert[
"LTR"][
"GAL"] = convert[
"LTR"][
"IN3"]*convert[
"IN3"][
"GAL"];
97 convert[
"M3"][
"GAL"] = 1000.*convert[
"LTR"][
"GAL"];
98 convert[
"CC"][
"GAL"] = convert[
"CC"][
"IN3"]*convert[
"IN3"][
"GAL"];
100 convert[
"LBS"][
"KG"] = 0.45359237;
101 convert[
"KG"][
"LBS"] = 1.0/convert[
"LBS"][
"KG"];
102 convert[
"SLUG"][
"KG"] = 14.59390;
103 convert[
"KG"][
"SLUG"] = 1.0/convert[
"SLUG"][
"KG"];
105 convert[
"SLUG*FT2"][
"KG*M2"] = 1.35594;
106 convert[
"KG*M2"][
"SLUG*FT2"] = 1.0/convert[
"SLUG*FT2"][
"KG*M2"];
108 convert[
"RAD"][
"DEG"] = 180.0/M_PI;
109 convert[
"DEG"][
"RAD"] = 1.0/convert[
"RAD"][
"DEG"];
111 convert[
"RAD/SEC"][
"DEG/SEC"] = convert[
"RAD"][
"DEG"];
112 convert[
"DEG/SEC"][
"RAD/SEC"] = 1.0/convert[
"RAD/SEC"][
"DEG/SEC"];
114 convert[
"LBS/FT"][
"N/M"] = 14.5939;
115 convert[
"N/M"][
"LBS/FT"] = 1.0/convert[
"LBS/FT"][
"N/M"];
117 convert[
"LBS/FT/SEC"][
"N/M/SEC"] = 14.5939;
118 convert[
"N/M/SEC"][
"LBS/FT/SEC"] = 1.0/convert[
"LBS/FT/SEC"][
"N/M/SEC"];
120 convert[
"LBS/FT2/SEC2"][
"N/M2/SEC2"] = 47.880259;
121 convert[
"N/M2/SEC2"][
"LBS/FT2/SEC2"] = 1.0/convert[
"LBS/FT2/SEC2"][
"N/M2/SEC2"];
123 convert[
"WATTS"][
"HP"] = 0.001341022;
124 convert[
"HP"][
"WATTS"] = 1.0/convert[
"WATTS"][
"HP"];
126 convert[
"N"][
"LBS"] = 0.22482;
127 convert[
"LBS"][
"N"] = 1.0/convert[
"N"][
"LBS"];
129 convert[
"KTS"][
"FT/SEC"] = 1.6878098571;
130 convert[
"FT/SEC"][
"KTS"] = 1.0/convert[
"KTS"][
"FT/SEC"];
131 convert[
"M/S"][
"FT/S"] = 3.2808399;
132 convert[
"M/S"][
"KTS"] = convert[
"M/S"][
"FT/S"]/convert[
"KTS"][
"FT/SEC"];
133 convert[
"M/SEC"][
"FT/SEC"] = 3.2808399;
134 convert[
"FT/S"][
"M/S"] = 1.0/convert[
"M/S"][
"FT/S"];
135 convert[
"M/SEC"][
"FT/SEC"] = 3.2808399;
136 convert[
"FT/SEC"][
"M/SEC"] = 1.0/convert[
"M/SEC"][
"FT/SEC"];
137 convert[
"KM/SEC"][
"FT/SEC"] = 3280.8399;
138 convert[
"FT/SEC"][
"KM/SEC"] = 1.0/convert[
"KM/SEC"][
"FT/SEC"];
140 convert[
"FT*LBS"][
"N*M"] = 1.35581795;
141 convert[
"N*M"][
"FT*LBS"] = 1/convert[
"FT*LBS"][
"N*M"];
143 convert[
"M4*SEC/KG"][
"FT4*SEC/SLUG"] = convert[
"M"][
"FT"]*convert[
"M"][
"FT"]*
144 convert[
"M"][
"FT"]*convert[
"M"][
"FT"]/convert[
"KG"][
"SLUG"];
145 convert[
"FT4*SEC/SLUG"][
"M4*SEC/KG"] =
146 1.0/convert[
"M4*SEC/KG"][
"FT4*SEC/SLUG"];
148 convert[
"INHG"][
"PSF"] = 70.7180803;
149 convert[
"PSF"][
"INHG"] = 1.0/convert[
"INHG"][
"PSF"];
150 convert[
"ATM"][
"INHG"] = 29.9246899;
151 convert[
"INHG"][
"ATM"] = 1.0/convert[
"ATM"][
"INHG"];
152 convert[
"PSI"][
"INHG"] = 2.03625437;
153 convert[
"INHG"][
"PSI"] = 1.0/convert[
"PSI"][
"INHG"];
154 convert[
"INHG"][
"PA"] = 3386.0;
155 convert[
"PA"][
"INHG"] = 1.0/convert[
"INHG"][
"PA"];
156 convert[
"LBS/FT2"][
"N/M2"] = 14.5939/convert[
"FT"][
"M"];
157 convert[
"N/M2"][
"LBS/FT2"] = 1.0/convert[
"LBS/FT2"][
"N/M2"];
158 convert[
"LBS/FT2"][
"PA"] = convert[
"LBS/FT2"][
"N/M2"];
159 convert[
"PA"][
"LBS/FT2"] = 1.0/convert[
"LBS/FT2"][
"PA"];
161 convert[
"KG/MIN"][
"LBS/MIN"] = convert[
"KG"][
"LBS"];
162 convert[
"KG/SEC"][
"LBS/SEC"] = convert[
"KG"][
"LBS"];
163 convert [
"N/SEC"][
"LBS/SEC"] = 0.224808943;
164 convert [
"LBS/SEC"][
"N/SEC"] = 1.0/convert [
"N/SEC"][
"LBS/SEC"];
166 convert[
"LBS/HP*HR"][
"KG/KW*HR"] = 0.6083;
167 convert[
"KG/KW*HR"][
"LBS/HP*HR"] = 1.0/convert[
"LBS/HP*HR"][
"KG/KW*HR"];
169 convert[
"KG/L"][
"LBS/GAL"] = 8.3454045;
170 convert[
"LBS/GAL"][
"KG/L"] = 1.0/convert[
"KG/L"][
"LBS/GAL"];
172 convert[
"FT3/SEC2"][
"M3/SEC2"] = convert[
"FT3"][
"M3"];
173 convert[
"M3/SEC2"][
"FT3/SEC2"] = convert[
"M3"][
"FT3"];
176 convert[
"M"][
"M"] = 1.00;
177 convert[
"KM"][
"KM"] = 1.00;
178 convert[
"FT"][
"FT"] = 1.00;
179 convert[
"IN"][
"IN"] = 1.00;
181 convert[
"M2"][
"M2"] = 1.00;
182 convert[
"FT2"][
"FT2"] = 1.00;
184 convert[
"IN3"][
"IN3"] = 1.00;
185 convert[
"CC"][
"CC"] = 1.0;
186 convert[
"M3"][
"M3"] = 1.0;
187 convert[
"FT3"][
"FT3"] = 1.0;
188 convert[
"LTR"][
"LTR"] = 1.0;
189 convert[
"GAL"][
"GAL"] = 1.0;
191 convert[
"KG"][
"KG"] = 1.00;
192 convert[
"LBS"][
"LBS"] = 1.00;
194 convert[
"KG*M2"][
"KG*M2"] = 1.00;
195 convert[
"SLUG*FT2"][
"SLUG*FT2"] = 1.00;
197 convert[
"DEG"][
"DEG"] = 1.00;
198 convert[
"RAD"][
"RAD"] = 1.00;
200 convert[
"DEG/SEC"][
"DEG/SEC"] = 1.00;
201 convert[
"RAD/SEC"][
"RAD/SEC"] = 1.00;
203 convert[
"LBS/FT"][
"LBS/FT"] = 1.00;
204 convert[
"N/M"][
"N/M"] = 1.00;
206 convert[
"LBS/FT/SEC"][
"LBS/FT/SEC"] = 1.00;
207 convert[
"N/M/SEC"][
"N/M/SEC"] = 1.00;
209 convert[
"LBS/FT2/SEC2"][
"LBS/FT2/SEC2"] = 1.00;
210 convert[
"N/M2/SEC2"][
"N/M2/SEC2"] = 1.00;
212 convert[
"HP"][
"HP"] = 1.00;
213 convert[
"WATTS"][
"WATTS"] = 1.00;
215 convert[
"N"][
"N"] = 1.00;
217 convert[
"FT/SEC"][
"FT/SEC"] = 1.00;
218 convert[
"KTS"][
"KTS"] = 1.00;
219 convert[
"M/S"][
"M/S"] = 1.0;
220 convert[
"M/SEC"][
"M/SEC"] = 1.0;
221 convert[
"KM/SEC"][
"KM/SEC"] = 1.0;
223 convert[
"FT*LBS"][
"FT*LBS"] = 1.00;
224 convert[
"N*M"][
"N*M"] = 1.00;
226 convert[
"M4*SEC/KG"][
"M4*SEC/KG"] = 1.0;
227 convert[
"FT4*SEC/SLUG"][
"FT4*SEC/SLUG"] = 1.0;
229 convert[
"PSI"][
"PSI"] = 1.00;
230 convert[
"PSF"][
"PSF"] = 1.00;
231 convert[
"INHG"][
"INHG"] = 1.00;
232 convert[
"ATM"][
"ATM"] = 1.0;
233 convert[
"PA"][
"PA"] = 1.0;
234 convert[
"N/M2"][
"N/M2"] = 1.00;
235 convert[
"LBS/FT2"][
"LBS/FT2"] = 1.00;
237 convert[
"LBS/SEC"][
"LBS/SEC"] = 1.00;
238 convert[
"KG/MIN"][
"KG/MIN"] = 1.0;
239 convert[
"LBS/MIN"][
"LBS/MIN"] = 1.0;
240 convert[
"N/SEC"][
"N/SEC"] = 1.0;
242 convert[
"LBS/HP*HR"][
"LBS/HP*HR"] = 1.0;
243 convert[
"KG/KW*HR"][
"KG/KW*HR"] = 1.0;
245 convert[
"KG/L"][
"KG/L"] = 1.0;
246 convert[
"LBS/GAL"][
"LBS/GAL"] = 1.0;
248 convert[
"FT3/SEC2"][
"FT3/SEC2"] = 1.0;
249 convert[
"M3/SEC2"][
"M3/SEC2"] = 1.0;
251 convert[
"VOLTS"][
"VOLTS"] = 1.0;
252 convert[
"OHMS"][
"OHMS"] = 1.0;
253 convert[
"AMPERES"][
"AMPERES"] = 1.0;
261 for (
unsigned int i = 0; i < children.size(); ++i)
279 attributes[key] = value;
290 if (attribute.empty()) {
292 s <<
ReadFrom() <<
"Expecting numeric attribute value, but got no data";
293 cerr << s.str() << endl;
294 throw length_error(s.str());
299 number = atof_locale_c(attribute);
303 cerr << s.str() << endl;
315 if (children.size() > el) {
329 if (children.size() > element_index+1) {
331 return children[element_index];
342 if (!data_lines.empty())
return data_lines[i];
343 else return string(
"");
350 if (data_lines.size() == 1) {
353 number = atof_locale_c(data_lines[0]);
357 cerr << s.str() << endl;
362 }
else if (data_lines.empty()) {
364 s <<
ReadFrom() <<
"Expected numeric value, but got no data";
365 cerr << s.str() << endl;
366 throw length_error(s.str());
368 cerr <<
ReadFrom() <<
"Attempting to get single data value in element "
369 <<
"<" << name <<
">" << endl
370 <<
" from multiple lines:" << endl;
371 for(
unsigned int i=0; i<data_lines.size(); ++i)
372 cerr << data_lines[i] << endl;
374 s <<
ReadFrom() <<
"Attempting to get single data value in element "
375 <<
"<" << name <<
">"
376 <<
" from multiple lines (" << data_lines.size() <<
").";
377 throw length_error(s.str());
385 unsigned int number_of_elements=0;
388 number_of_elements++;
391 return number_of_elements;
398 if (el.empty() && children.size() >= 1) {
402 for (
unsigned int i=0; i<children.size(); i++) {
403 if (el == children[i]->
GetName()) {
417 if (element_index < children.size()) {
418 return children[element_index++];
424 for (
unsigned int i=element_index; i<children.size(); i++) {
425 if (el == children[i]->
GetName()) {
441 value = DisperseValue(element, value);
445 s <<
ReadFrom() <<
"Attempting to get non-existent element " << el;
446 cerr << s.str() << endl;
447 throw length_error(s.str());
467 cerr <<
ReadFrom() <<
"Attempting to get non-existent element " << el <<
" ;returning false"
493 s <<
ReadFrom() <<
"Attempting to get non-existent element " << el;
494 cerr << s.str() << endl;
495 throw length_error(s.str());
500 if (!supplied_units.empty()) {
501 if (convert.find(supplied_units) == convert.end()) {
503 s << element->
ReadFrom() <<
"Supplied unit: \"" << supplied_units
504 <<
"\" does not exist (typo?).";
505 cerr << s.str() << endl;
506 throw invalid_argument(s.str());
508 if (convert[supplied_units].find(target_units) == convert[supplied_units].end()) {
510 s << element->
ReadFrom() <<
"Supplied unit: \"" << supplied_units
511 <<
"\" cannot be converted to " << target_units;
512 cerr << s.str() << endl;
513 throw invalid_argument(s.str());
520 if ((supplied_units ==
"RAD") && (fabs(value) > 2 * M_PI)) {
522 << value <<
" RAD is outside the range [ -2*M_PI RAD ; +2*M_PI RAD ]"
525 if ((supplied_units ==
"DEG") && (fabs(value) > 360.0)) {
527 << value <<
" DEG is outside the range [ -360 DEG ; +360 DEG ]"
532 if (!supplied_units.empty()) {
533 value *= convert[supplied_units][target_units];
536 if ((target_units ==
"RAD") && (fabs(value) > 2 * M_PI)) {
538 << value <<
" RAD is outside the range [ -2*M_PI RAD ; +2*M_PI RAD ]"
541 if ((target_units ==
"DEG") && (fabs(value) > 360.0)) {
543 << value <<
" DEG is outside the range [ -360 DEG ; +360 DEG ]"
547 value = DisperseValue(element, value, supplied_units, target_units);
555 const string& supplied_units,
556 const string& target_units)
562 s <<
ReadFrom() <<
"Attempting to get non-existent element " << el;
563 cerr << s.str() << endl;
564 throw length_error(s.str());
567 if (!supplied_units.empty()) {
568 if (convert.find(supplied_units) == convert.end()) {
570 s << element->
ReadFrom() <<
"Supplied unit: \"" << supplied_units
571 <<
"\" does not exist (typo?).";
572 cerr << s.str() << endl;
573 throw invalid_argument(s.str());
575 if (convert[supplied_units].find(target_units) == convert[supplied_units].end()) {
577 s << element->
ReadFrom() <<
"Supplied unit: \"" << supplied_units
578 <<
"\" cannot be converted to " << target_units;
579 cerr << s.str() << endl;
580 throw invalid_argument(s.str());
585 if (!supplied_units.empty()) {
586 value *= convert[supplied_units][target_units];
589 value = DisperseValue(element, value, supplied_units, target_units);
603 if (!supplied_units.empty()) {
604 if (convert.find(supplied_units) == convert.end()) {
606 s <<
ReadFrom() <<
"Supplied unit: \"" << supplied_units
607 <<
"\" does not exist (typo?).";
608 cerr << s.str() << endl;
609 throw invalid_argument(s.str());
611 if (convert[supplied_units].find(target_units) == convert[supplied_units].end()) {
613 s <<
ReadFrom() <<
"Supplied unit: \"" << supplied_units
614 <<
"\" cannot be converted to " << target_units;
615 cerr << s.str() << endl;
616 throw invalid_argument(s.str());
624 if (!supplied_units.empty()) value *= convert[supplied_units][target_units];
625 triplet(1) = DisperseValue(item, value, supplied_units, target_units);
635 if (!supplied_units.empty()) value *= convert[supplied_units][target_units];
636 triplet(2) = DisperseValue(item, value, supplied_units, target_units);
645 if (!supplied_units.empty()) value *= convert[supplied_units][target_units];
646 triplet(3) = DisperseValue(item, value, supplied_units, target_units);
656double Element::DisperseValue(
Element *e,
double val,
const std::string& supplied_units,
657 const std::string& target_units)
661 bool disperse =
false;
663 char* num = getenv(
"JSBSIM_DISPERSE");
665 disperse = (atoi(num) == 1);
669 std::cerr <<
"Could not process JSBSIM_DISPERSE environment variable: Assumed NO dispersions." << endl;
674 if (!supplied_units.empty()) disp *= convert[supplied_units][target_units];
676 RandomNumberGenerator generator;
678 if (attType ==
"gaussian" || attType ==
"gaussiansigned") {
679 double grn = generator.GetNormalRandomNumber();
680 if (attType ==
"gaussian")
681 value = val + disp*grn;
683 value = (val + disp*grn)*FGJSBBase::sign(grn);
684 }
else if (attType ==
"uniform" || attType ==
"uniformsigned") {
685 double urn = generator.GetUniformRandomNumber();
686 if (attType ==
"uniform")
687 value = val + disp * urn;
689 value = (val + disp * urn)*FGJSBBase::sign(urn);
692 s <<
ReadFrom() <<
"Unknown dispersion type" << attType;
693 cerr << s.str() << endl;
694 throw domain_error(s.str());
705 unsigned int i, spaces;
708 for (spaces=0; spaces<=level; spaces++) cout <<
" ";
709 cout <<
"Element Name: " << name;
711 map<string, string>::iterator it;
712 for (it = attributes.begin(); it != attributes.end(); ++it)
713 cout <<
" " << it->first <<
" = " << it->second;
716 for (i=0; i<data_lines.size(); i++) {
717 for (spaces=0; spaces<=level; spaces++) cout <<
" ";
718 cout << data_lines[i] << endl;
720 for (i=0; i<children.size(); i++) {
721 children[i]->Print(level);
729 attributes[name] = value;
736 string::size_type string_start = d.find_first_not_of(
" \t");
737 if (string_start != string::npos && string_start > 0) {
738 d.erase(0,string_start);
740 data_lines.push_back(d);
747 ostringstream message;
753 return message.str();
760 map<string, string>::iterator it;
762 for (it=el->attributes.begin(); it != el->attributes.end(); ++it) {
763 if (attributes.find(it->first) == attributes.end())
764 attributes[it->first] = it->second;
766 if (FGJSBBase::debug_lvl > 0 && (attributes[it->first] != it->second))
767 cout << el->
ReadFrom() <<
" Attribute '" << it->first <<
"' is overridden in file "
769 <<
" The value '" << attributes[it->first] <<
"' will be used instead of '"
770 << it->second <<
"'." << endl;
Element * FindElement(const std::string &el="")
Searches for a specified element.
const std::string & GetName(void) const
Retrieves the element name.
bool SetAttributeValue(const std::string &key, const std::string &value)
Modifies an attribute.
double GetAttributeValueAsNumber(const std::string &key)
Retrieves an attribute value as a double precision real number.
FGColumnVector3 FindElementTripletConvertTo(const std::string &target_units)
Composes a 3-element column vector for the supplied location or orientation.
Element * GetElement(unsigned int el=0)
Returns a pointer to the element requested by index.
Element(const std::string &nm)
Constructor.
bool FindElementValueAsBoolean(const std::string &el="")
Searches for the named element and returns the data belonging to it as a bool.
std::string GetAttributeValue(const std::string &key)
Retrieves an attribute.
int GetLineNumber(void) const
Returns the line number at which the element has been defined.
std::string GetDataLine(unsigned int i=0)
Gets a line of data belonging to an element.
const std::string & GetFileName(void) const
Returns the name of the file in which the element has been read.
unsigned int GetNumElements(void)
Returns the number of child elements for this element.
void MergeAttributes(Element *el)
Merges the attributes of the current element with another element.
void AddAttribute(const std::string &name, const std::string &value)
Stores an attribute belonging to this element.
std::string ReadFrom(void) const
Return a string that contains a description of the location where the current XML element was read fr...
std::string FindElementValue(const std::string &el="")
Searches for the named element and returns the string data belonging to it.
void Print(unsigned int level=0)
Prints the element.
double FindElementValueAsNumberConvertTo(const std::string &el, const std::string &target_units)
Searches for the named element and converts and returns the data belonging to it.
Element * FindNextElement(const std::string &el="")
Searches for the next element as specified.
double GetDataAsNumber(void)
Converts the element data to a number.
bool HasAttribute(const std::string &key)
Determines if an element has the supplied attribute.
void SetParent(Element *p)
This function sets the value of the parent class attribute to the supplied Element pointer.
Element * GetNextElement(void)
Returns a pointer to the next element in the list.
double FindElementValueAsNumberConvertFromTo(const std::string &el, const std::string &supplied_units, const std::string &target_units)
Searches for the named element and converts and returns the data belonging to it.
double FindElementValueAsNumber(const std::string &el="")
Searches for the named element and returns the data belonging to it as a number.
void AddData(std::string d)
Stores data belonging to this element.
~Element(void)
Destructor.
This class implements a 3 element column vector.