43#include "input_output/FGXMLElement.h"
44#include "input_output/string_utilities.h"
55 : nRows(NRows), nCols(1)
59 Data.push_back(std::numeric_limits<double>::quiet_NaN());
60 Data.push_back(std::numeric_limits<double>::quiet_NaN());
66FGTable::FGTable(
int NRows,
int NCols)
67 : nRows(NRows), nCols(NCols)
71 Data.push_back(std::numeric_limits<double>::quiet_NaN());
78 : PropertyManager(t.PropertyManager)
83 internal = t.internal;
85 lookupProperty[0] = t.lookupProperty[0];
86 lookupProperty[1] = t.lookupProperty[1];
87 lookupProperty[2] = t.lookupProperty[2];
90 Tables.reserve(t.Tables.size());
91 for(
const auto &t: t.Tables)
92 Tables.push_back(std::make_unique<FGTable>(*t));
99unsigned int FindNumColumns(
const string& test_line)
103 unsigned int nCols=0;
104 while ((position = test_line.find_first_not_of(
" \t", position)) != string::npos) {
106 position = test_line.find_first_of(
" \t", position);
114 const std::string& Prefix)
115 : PropertyManager(pm)
124 if (call_type ==
"internal") {
126 }
else if (!call_type.empty()) {
128 <<
" An unknown table type attribute is listed: " << call_type
137 unsigned int dimension = 0;
146 <<
fgred <<
" This table specifies both 'internal' call type" << endl
147 <<
" and specific lookup properties via the 'independentVar' element." << endl
148 <<
" These are mutually exclusive specifications. The 'internal'" << endl
149 <<
" attribute will be ignored." <<
fgdef << endl << endl;
153 while (axisElement) {
154 string property_string = axisElement->
GetDataLine();
155 if (property_string.find(
"#") != string::npos) {
156 if (is_number(Prefix)) {
157 property_string = replace(property_string,
"#",Prefix);
162 PropertyManager, axisElement);
164 if (lookup_axis ==
string(
"row")) {
165 lookupProperty[eRow] = node;
166 dimension = std::max(dimension, 1u);
167 }
else if (lookup_axis ==
string(
"column")) {
168 lookupProperty[eColumn] = node;
169 dimension = std::max(dimension, 2u);
170 }
else if (lookup_axis ==
string(
"table")) {
171 lookupProperty[eTable] = node;
172 dimension = std::max(dimension, 3u);
173 }
else if (!lookup_axis.empty()) {
174 throw BaseException(
"Lookup table axis specification not understood: " + lookup_axis);
176 lookupProperty[eRow] = node;
177 dimension = std::max(dimension, 1u);
182 }
else if (internal) {
192 unsigned int nColumns = FindNumColumns(tableData->
GetDataLine(0));
194 unsigned int nColumns1 = FindNumColumns(tableData->
GetDataLine(1));
195 if (nColumns1 == nColumns + 1) {
197 nColumns = nColumns1;
204 for(
unsigned int i=1; i<nLines; ++i) {
205 if (FindNumColumns(tableData->
GetDataLine(i)) != nColumns) {
207 <<
"Invalid number of columns in line "
216 if (dimension == 1 && nColumns != 2) {
218 <<
"Too many columns for a 1D table" << endl;
226 if (brkpt_string.empty()) {
230 <<
"No independentVars found, and table is not marked as internal,"
231 <<
" nor is it a 3D table." << endl;
232 throw BaseException(
"No independent variable found for table.");
237 if (brkpt_string.empty()) {
249 <<
"FGTable: <tableData> elements are missing" << endl;
250 throw BaseException(
"FGTable: <tableData> elements are missing");
253 std::cerr << tableData->
ReadFrom() <<
"<tableData> is empty." << endl;
258 if (!internal && brkpt_string.empty()) {
261 if (!lookupProperty[eTable]) {
263 <<
"FGTable: missing lookup axis \"table\"";
264 throw BaseException(
"FGTable: missing lookup axis \"table\"");
268 if (!lookupProperty[eColumn]) {
270 <<
"FGTable: missing lookup axis \"column\"";
271 throw BaseException(
"FGTable: missing lookup axis \"column\"");
275 if (!lookupProperty[eRow]) {
277 <<
"FGTable: missing lookup axis \"row\"";
291 if (line.find_first_not_of(
"0123456789.-+eE \t\n") != string::npos) {
292 cerr <<
" In file " << tableData->
GetFileName() << endl
293 <<
" Illegal character found in line "
294 << tableData->
GetLineNumber() + i + 1 <<
": " << endl << line << endl;
306 Data.push_back(std::numeric_limits<double>::quiet_NaN());
307 Data.push_back(std::numeric_limits<double>::quiet_NaN());
315 Data.push_back(std::numeric_limits<double>::quiet_NaN());
323 Data.push_back(std::numeric_limits<double>::quiet_NaN());
327 Tables.push_back(std::make_unique<FGTable>(PropertyManager, tableData));
329 Tables.back()->lookupProperty[eRow] = lookupProperty[eRow];
330 Tables.back()->lookupProperty[eColumn] = lookupProperty[eColumn];
352 for (
unsigned int b=2; b<=Tables.size(); ++b) {
353 if (Data[b] <= Data[b-1]) {
356 <<
" FGTable: breakpoint lookup is not monotonically increasing" << endl
357 <<
" in breakpoint " << b;
358 if (nameel != 0) std::cerr <<
" of table in " << nameel->
GetAttributeValue(
"name");
359 std::cerr <<
":" <<
reset << endl
360 <<
" " << Data[b] <<
"<=" << Data[b-1] << endl;
361 throw BaseException(
"Breakpoint lookup is not monotonically increasing");
368 for (
unsigned int c=2; c<=nCols; ++c) {
369 if (Data[c] <= Data[c-1]) {
372 <<
" FGTable: column lookup is not monotonically increasing" << endl
373 <<
" in column " << c;
374 if (nameel != 0) std::cerr <<
" of table in " << nameel->
GetAttributeValue(
"name");
375 std::cerr <<
":" <<
reset << endl
376 <<
" " << Data[c] <<
"<=" << Data[c-1] << endl;
377 throw BaseException(
"FGTable: column lookup is not monotonically increasing");
384 for (
size_t r=2; r<=nRows; ++r) {
385 if (Data[r*(nCols+1)]<=Data[(r-1)*(nCols+1)]) {
388 <<
" FGTable: row lookup is not monotonically increasing" << endl
390 if (nameel != 0) std::cerr <<
" of table in " << nameel->
GetAttributeValue(
"name");
391 std::cerr <<
":" <<
reset << endl
392 <<
" " << Data[r*(nCols+1)] <<
"<=" << Data[(r-1)*(nCols+1)] << endl;
393 throw BaseException(
"FGTable: row lookup is not monotonically increasing");
401 if (Data.size() != 2*nRows+2) missingData(el, 2*nRows, Data.size()-2);
404 if (Data.size() !=
static_cast<size_t>(nRows+1)*(nCols+1))
405 missingData(el, (nRows+1)*(nCols+1)-1, Data.size()-1);
408 if (Data.size() != nRows+1) missingData(el, nRows, Data.size()-1);
417 if (debug_lvl & 1) Print();
422void FGTable::missingData(
Element *el,
unsigned int expected_size,
size_t actual_size)
426 <<
" FGTable: Missing data";
427 if (!Name.empty()) std::cerr <<
" in table " << Name;
428 std::cerr <<
":" <<
reset << endl
429 <<
" Expecting " << expected_size <<
" elements while "
430 << actual_size <<
" elements were provided." << endl;
440 if (!Name.empty() && !internal) {
441 string tmp = PropertyManager->mkPropertyName(Name,
false);
443 if (node && node->isTied())
444 PropertyManager->Untie(node);
452double FGTable::GetElement(
unsigned int r,
unsigned int c)
const
454 assert(r <= nRows && c <= nCols);
456 assert(Data.size() == nRows+1);
459 assert(Data.size() == (nCols+1)*(nRows+1));
460 return Data[r*(nCols+1)+c];
471 assert(lookupProperty[eRow]);
472 return GetValue(lookupProperty[eRow]->getDoubleValue());
474 assert(lookupProperty[eRow]);
475 assert(lookupProperty[eColumn]);
476 return GetValue(lookupProperty[eRow]->getDoubleValue(),
477 lookupProperty[eColumn]->getDoubleValue());
479 assert(lookupProperty[eRow]);
480 assert(lookupProperty[eColumn]);
481 assert(lookupProperty[eTable]);
482 return GetValue(lookupProperty[eRow]->getDoubleValue(),
483 lookupProperty[eColumn]->getDoubleValue(),
484 lookupProperty[eTable]->getDoubleValue());
487 return std::numeric_limits<double>::quiet_NaN();
496 assert(Data.size() == 2*nRows+2);
501 else if (key >= Data[2*nRows])
502 return Data[2*nRows+1];
507 while (Data[2*r] < key) r++;
509 double x0 = Data[2*r-2];
510 double Span = Data[2*r] - x0;
512 double Factor = (key - x0) / Span;
513 assert(Factor >= 0.0 && Factor <= 1.0);
515 double y0 = Data[2*r-1];
516 return Factor*(Data[2*r+1] - y0) + y0;
523 if (nCols == 1)
return GetValue(rowKey);
525 assert(Type == tt2D);
526 assert(Data.size() == (nCols+1)*(nRows+1));
529 while(Data[c] < colKey && c < nCols) c++;
530 double x0 = Data[c-1];
531 double Span = Data[c] - x0;
533 double cFactor =
Constrain(0.0, (colKey - x0) / Span, 1.0);
536 double y0 = Data[(nCols+1)+c-1];
537 return cFactor*(Data[(nCols+1)+c] - y0) + y0;
541 while(Data[r*(nCols+1)] < rowKey && r < nRows) r++;
542 x0 = Data[(r-1)*(nCols+1)];
543 Span = Data[r*(nCols+1)] - x0;
545 double rFactor =
Constrain(0.0, (rowKey - x0) / Span, 1.0);
546 double col1temp = rFactor*Data[r*(nCols+1)+c-1]+(1.0-rFactor)*Data[(r-1)*(nCols+1)+c-1];
547 double col2temp = rFactor*Data[r*(nCols+1)+c]+(1.0-rFactor)*Data[(r-1)*(nCols+1)+c];
549 return cFactor*(col2temp-col1temp)+col1temp;
556 assert(Type == tt3D);
557 assert(Data.size() == nRows+1);
560 if(tableKey <= Data[1])
561 return Tables[0]->GetValue(rowKey, colKey);
562 else if (tableKey >= Data[nRows])
563 return Tables[nRows-1]->GetValue(rowKey, colKey);
568 while (Data[r] < tableKey) r++;
570 double x0 = Data[r-1];
571 double Span = Data[r] - x0;
573 double Factor = (tableKey - x0) / Span;
574 assert(Factor >= 0.0 && Factor <= 1.0);
576 double y0 = Tables[r-2]->GetValue(rowKey, colKey);
577 return Factor*(Tables[r-1]->GetValue(rowKey, colKey) - y0) + y0;
582double FGTable::GetMinValue(
void)
const
584 assert(Type == tt1D);
585 assert(Data.size() == 2*nRows+2);
587 double minValue = HUGE_VAL;
589 for(
unsigned int i=1; i<=nRows; ++i)
590 minValue = std::min(minValue, Data[2*i+1]);
600 assert(Type != tt3D);
613 assert(Type != tt3D);
617 size_t n = Data.size();
618 if (Type == tt2D && nCols > 1 && n >= 3 && n <= nCols+1) {
619 if (Data.at(n-1) <= Data.at(n-2))
620 throw BaseException(
"FGTable: column lookup is not monotonically increasing");
624 size_t row = (n-1) / (nCols+1);
625 if (row >=2 && row*(nCols+1) == n-1) {
626 if (Data.at(row*(nCols+1)) <= Data.at((row-1)*(nCols+1)))
627 throw BaseException(
"FGTable: row lookup is not monotonically increasing");
635void FGTable::Print(
void)
637 ios::fmtflags flags = cout.setf(ios::fixed);
642 cout <<
" 1 dimensional table with " << nRows <<
" rows." << endl;
645 cout <<
" 2 dimensional table with " << nRows <<
" rows, " << nCols <<
" columns." << endl;
648 cout <<
" 3 dimensional table with " << nRows <<
" breakpoints, "
649 << Tables.size() <<
" tables." << endl;
652 unsigned int startCol=1, startRow=1;
659 if (Type == tt2D) startRow = 0;
661 for (
unsigned int r=startRow; r<=nRows; r++) {
670 for (
unsigned int c=startCol; c<=nCols; c++) {
671 cout << Data[p++] <<
"\t";
674 Tables[r-1]->Print();
684void FGTable::bind(Element* el,
const string& Prefix)
686 if ( !Name.empty() && !internal) {
687 if (!Prefix.empty()) {
688 if (is_number(Prefix)) {
689 if (Name.find(
"#") != string::npos) {
690 Name = replace(Name,
"#", Prefix);
692 cerr << el->ReadFrom()
693 <<
"Malformed table name with number: " << Prefix
694 <<
" and property name: " << Name
695 <<
" but no \"#\" sign for substitution." << endl;
696 throw BaseException(
"Missing \"#\" sign for substitution");
699 Name = Prefix +
"/" + Name;
702 string tmp = PropertyManager->mkPropertyName(Name,
false);
704 if (PropertyManager->HasNode(tmp)) {
705 FGPropertyNode* _property = PropertyManager->GetNode(tmp);
706 if (_property->isTied()) {
707 cerr << el->ReadFrom()
708 <<
"Property " << tmp <<
" has already been successfully bound (late)." << endl;
709 throw BaseException(
"Failed to bind the property to an existing already tied node.");
735void FGTable::Debug(
int from)
737 if (debug_lvl <= 0)
return;
742 if (debug_lvl & 2 ) {
743 if (from == 0) cout <<
"Instantiated: FGTable" << endl;
744 if (from == 1) cout <<
"Destroyed: FGTable" << endl;
746 if (debug_lvl & 4 ) {
748 if (debug_lvl & 8 ) {
750 if (debug_lvl & 16) {
752 if (debug_lvl & 64) {
Element * FindElement(const std::string &el="")
Searches for a specified element.
double GetAttributeValueAsNumber(const std::string &key)
Retrieves an attribute value as a double precision real number.
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.
Element * GetParent(void)
Returns a pointer to the parent of an element.
unsigned int GetNumDataLines(void)
Returns the number of lines of data stored.
unsigned int GetNumElements(void)
Returns the number of child elements for 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...
Element * FindNextElement(const std::string &el="")
Searches for the next element as specified.
static char fgred[6]
red text
static constexpr double Constrain(double min, double value, double max)
Constrain a value between a minimum and a maximum value.
static char fgdef[6]
default text
static char reset[5]
resets text properties
static char highint[5]
highlights text
Class wrapper for property handling.
Represents a property value which can use late binding.
void operator<<(std::istream &)
Read the table in.
FGTable(const FGTable &table)
This is the very important copy constructor.
double GetValue(void) const
Get the current table value.