/**************************************************************************** * * (c) 2009-2016 QGROUNDCONTROL PROJECT * * QGroundControl is licensed according to the terms in the file * COPYING.md in the root of the source code directory. * ****************************************************************************/ #include "FactMetaData.h" #include "SettingsManager.h" #include "JsonHelper.h" #include "QGCApplication.h" #include #include #include #include #include #include // Conversion Constants // Time const qreal FactMetaData::UnitConsts_s::secondsPerHour = 3600.0; // Velocity const qreal FactMetaData::UnitConsts_s::knotsToKPH = 1.852; // exact, hence weird base for knotsToMetersPerSecond // Length const qreal FactMetaData::UnitConsts_s::milesToMeters = 1609.344; const qreal FactMetaData::UnitConsts_s::feetToMeters = 0.3048; const qreal FactMetaData::UnitConsts_s::inchesToCentimeters = 2.54; // Built in translations for all Facts const FactMetaData::BuiltInTranslation_s FactMetaData::_rgBuiltInTranslations[] = { { "centi-degrees", "deg", FactMetaData::_centiDegreesToDegrees, FactMetaData::_degreesToCentiDegrees }, { "radians", "deg", FactMetaData::_radiansToDegrees, FactMetaData::_degreesToRadians }, { "gimbal-degrees", "deg", FactMetaData::_mavlinkGimbalDegreesToUserGimbalDegrees, FactMetaData::_userGimbalDegreesToMavlinkGimbalDegrees }, { "norm", "%", FactMetaData::_normToPercent, FactMetaData::_percentToNorm }, }; // Translations driven by app settings const FactMetaData::AppSettingsTranslation_s FactMetaData::_rgAppSettingsTranslations[] = { { "m", "m", false, UnitsSettings::DistanceUnitsMeters, FactMetaData::_defaultTranslator, FactMetaData::_defaultTranslator }, { "meters", "meters", false, UnitsSettings::DistanceUnitsMeters, FactMetaData::_defaultTranslator, FactMetaData::_defaultTranslator }, { "cm/px", "cm/px", false, UnitsSettings::DistanceUnitsMeters, FactMetaData::_defaultTranslator, FactMetaData::_defaultTranslator }, { "m/s", "m/s", true, UnitsSettings::SpeedUnitsMetersPerSecond, FactMetaData::_defaultTranslator, FactMetaData::_defaultTranslator }, { "m^2", "m^2", false, UnitsSettings::AreaUnitsSquareMeters, FactMetaData::_defaultTranslator, FactMetaData::_defaultTranslator }, { "m", "ft", false, UnitsSettings::DistanceUnitsFeet, FactMetaData::_metersToFeet, FactMetaData::_feetToMeters }, { "meters", "ft", false, UnitsSettings::DistanceUnitsFeet, FactMetaData::_metersToFeet, FactMetaData::_feetToMeters }, { "cm/px", "in/px", false, UnitsSettings::DistanceUnitsFeet, FactMetaData::_centimetersToInches, FactMetaData::_inchesToCentimeters }, { "m^2", "km^2", false, UnitsSettings::AreaUnitsSquareKilometers, FactMetaData::_squareMetersToSquareKilometers, FactMetaData::_squareKilometersToSquareMeters }, { "m^2", "ha", false, UnitsSettings::AreaUnitsHectares, FactMetaData::_squareMetersToHectares, FactMetaData::_hectaresToSquareMeters }, { "m^2", "ft^2", false, UnitsSettings::AreaUnitsSquareFeet, FactMetaData::_squareMetersToSquareFeet, FactMetaData::_squareFeetToSquareMeters }, { "m^2", "ac", false, UnitsSettings::AreaUnitsAcres, FactMetaData::_squareMetersToAcres, FactMetaData::_acresToSquareMeters }, { "m^2", "mi^2", false, UnitsSettings::AreaUnitsSquareMiles, FactMetaData::_squareMetersToSquareMiles, FactMetaData::_squareMilesToSquareMeters }, { "m/s", "ft/s", true, UnitsSettings::SpeedUnitsFeetPerSecond, FactMetaData::_metersToFeet, FactMetaData::_feetToMeters }, { "m/s", "mph", true, UnitsSettings::SpeedUnitsMilesPerHour, FactMetaData::_metersPerSecondToMilesPerHour, FactMetaData::_milesPerHourToMetersPerSecond }, { "m/s", "km/h", true, UnitsSettings::SpeedUnitsKilometersPerHour, FactMetaData::_metersPerSecondToKilometersPerHour, FactMetaData::_kilometersPerHourToMetersPerSecond }, { "m/s", "kn", true, UnitsSettings::SpeedUnitsKnots, FactMetaData::_metersPerSecondToKnots, FactMetaData::_knotsToMetersPerSecond }, }; const char* FactMetaData::_decimalPlacesJsonKey = "decimalPlaces"; const char* FactMetaData::_nameJsonKey = "name"; const char* FactMetaData::_typeJsonKey = "type"; const char* FactMetaData::_shortDescriptionJsonKey = "shortDescription"; const char* FactMetaData::_longDescriptionJsonKey = "longDescription"; const char* FactMetaData::_unitsJsonKey = "units"; const char* FactMetaData::_defaultValueJsonKey = "defaultValue"; const char* FactMetaData::_mobileDefaultValueJsonKey = "mobileDefaultValue"; const char* FactMetaData::_minJsonKey = "min"; const char* FactMetaData::_maxJsonKey = "max"; FactMetaData::FactMetaData(QObject* parent) : QObject(parent) , _type(valueTypeInt32) , _decimalPlaces(unknownDecimalPlaces) , _rawDefaultValue(0) , _defaultValueAvailable(false) , _group("*Default Group") , _rawMax(_maxForType()) , _maxIsDefaultForType(true) , _rawMin(_minForType()) , _minIsDefaultForType(true) , _rawTranslator(_defaultTranslator) , _cookedTranslator(_defaultTranslator) , _rebootRequired(false) , _increment(std::numeric_limits::quiet_NaN()) { } FactMetaData::FactMetaData(ValueType_t type, QObject* parent) : QObject(parent) , _type(type) , _decimalPlaces(unknownDecimalPlaces) , _rawDefaultValue(0) , _defaultValueAvailable(false) , _group("*Default Group") , _rawMax(_maxForType()) , _maxIsDefaultForType(true) , _rawMin(_minForType()) , _minIsDefaultForType(true) , _rawTranslator(_defaultTranslator) , _cookedTranslator(_defaultTranslator) , _rebootRequired(false) , _increment(std::numeric_limits::quiet_NaN()) { } FactMetaData::FactMetaData(const FactMetaData& other, QObject* parent) : QObject(parent) { *this = other; } const FactMetaData& FactMetaData::operator=(const FactMetaData& other) { _decimalPlaces = other._decimalPlaces; _rawDefaultValue = other._rawDefaultValue; _defaultValueAvailable = other._defaultValueAvailable; _bitmaskStrings = other._bitmaskStrings; _bitmaskValues = other._bitmaskValues; _enumStrings = other._enumStrings; _enumValues = other._enumValues; _group = other._group; _longDescription = other._longDescription; _rawMax = other._rawMax; _maxIsDefaultForType = other._maxIsDefaultForType; _rawMin = other._rawMin; _minIsDefaultForType = other._minIsDefaultForType; _name = other._name; _shortDescription = other._shortDescription; _type = other._type; _rawUnits = other._rawUnits; _cookedUnits = other._cookedUnits; _rawTranslator = other._rawTranslator; _cookedTranslator = other._cookedTranslator; _rebootRequired = other._rebootRequired; _increment = other._increment; return *this; } QVariant FactMetaData::rawDefaultValue(void) const { if (_defaultValueAvailable) { return _rawDefaultValue; } else { qWarning() << "Attempt to access unavailable default value"; return QVariant(0); } } void FactMetaData::setRawDefaultValue(const QVariant& rawDefaultValue) { if (_type == valueTypeString || (_rawMin <= rawDefaultValue && rawDefaultValue <= _rawMax)) { _rawDefaultValue = rawDefaultValue; _defaultValueAvailable = true; } else { qWarning() << "Attempt to set default value which is outside min/max range"; } } void FactMetaData::setRawMin(const QVariant& rawMin) { if (rawMin >= _minForType()) { _rawMin = rawMin; _minIsDefaultForType = false; } else { qWarning() << "Attempt to set min below allowable value for fact: " << name() << ", value attempted: " << rawMin << ", type: " << type() << ", min for type: " << _minForType(); _rawMin = _minForType(); } } void FactMetaData::setRawMax(const QVariant& rawMax) { if (rawMax > _maxForType()) { qWarning() << "Attempt to set max above allowable value"; _rawMax = _maxForType(); } else { _rawMax = rawMax; _maxIsDefaultForType = false; } } QVariant FactMetaData::_minForType(void) const { switch (_type) { case valueTypeUint8: return QVariant(std::numeric_limits::min()); case valueTypeInt8: return QVariant(std::numeric_limits::min()); case valueTypeUint16: return QVariant(std::numeric_limits::min()); case valueTypeInt16: return QVariant(std::numeric_limits::min()); case valueTypeUint32: return QVariant(std::numeric_limits::min()); case valueTypeInt32: return QVariant(std::numeric_limits::min()); case valueTypeFloat: return QVariant(-std::numeric_limits::max()); case valueTypeDouble: return QVariant(-std::numeric_limits::max()); case valueTypeString: return QVariant(); case valueTypeBool: return QVariant(0); case valueTypeElapsedTimeInSeconds: return QVariant(0.0); } // Make windows compiler happy, even switch is full cased return QVariant(); } QVariant FactMetaData::_maxForType(void) const { switch (_type) { case valueTypeUint8: return QVariant(std::numeric_limits::max()); case valueTypeInt8: return QVariant(std::numeric_limits::max()); case valueTypeUint16: return QVariant(std::numeric_limits::max()); case valueTypeInt16: return QVariant(std::numeric_limits::max()); case valueTypeUint32: return QVariant(std::numeric_limits::max()); case valueTypeInt32: return QVariant(std::numeric_limits::max()); case valueTypeFloat: return QVariant(std::numeric_limits::max()); case valueTypeElapsedTimeInSeconds: case valueTypeDouble: return QVariant(std::numeric_limits::max()); case valueTypeString: return QVariant(); case valueTypeBool: return QVariant(1); } // Make windows compiler happy, even switch is full cased return QVariant(); } bool FactMetaData::convertAndValidateRaw(const QVariant& rawValue, bool convertOnly, QVariant& typedValue, QString& errorString) { bool convertOk = false; errorString.clear(); switch (type()) { case FactMetaData::valueTypeInt8: case FactMetaData::valueTypeInt16: case FactMetaData::valueTypeInt32: typedValue = QVariant(rawValue.toInt(&convertOk)); if (!convertOnly && convertOk) { if (typedValue < rawMin() || typedValue > rawMax()) { errorString = QString("Value must be within %1 and %2").arg(cookedMin().toInt()).arg(cookedMax().toInt()); } } break; case FactMetaData::valueTypeUint8: case FactMetaData::valueTypeUint16: case FactMetaData::valueTypeUint32: typedValue = QVariant(rawValue.toUInt(&convertOk)); if (!convertOnly && convertOk) { if (typedValue < rawMin() || typedValue > rawMax()) { errorString = QString("Value must be within %1 and %2").arg(cookedMin().toUInt()).arg(cookedMax().toUInt()); } } break; case FactMetaData::valueTypeFloat: typedValue = QVariant(rawValue.toFloat(&convertOk)); if (!convertOnly && convertOk) { if (typedValue < rawMin() || typedValue > rawMax()) { errorString = QString("Value must be within %1 and %2").arg(cookedMin().toFloat()).arg(cookedMax().toFloat()); } } break; case FactMetaData::valueTypeElapsedTimeInSeconds: case FactMetaData::valueTypeDouble: typedValue = QVariant(rawValue.toDouble(&convertOk)); if (!convertOnly && convertOk) { if (typedValue < rawMin() || typedValue > rawMax()) { errorString = QString("Value must be within %1 and %2").arg(cookedMin().toDouble()).arg(cookedMax().toDouble()); } } break; case FactMetaData::valueTypeString: convertOk = true; typedValue = QVariant(rawValue.toString()); break; case FactMetaData::valueTypeBool: convertOk = true; typedValue = QVariant(rawValue.toBool()); break; } if (!convertOk) { errorString += "Invalid number"; } return convertOk && errorString.isEmpty(); } bool FactMetaData::convertAndValidateCooked(const QVariant& cookedValue, bool convertOnly, QVariant& typedValue, QString& errorString) { bool convertOk = false; errorString.clear(); switch (type()) { case FactMetaData::valueTypeInt8: case FactMetaData::valueTypeInt16: case FactMetaData::valueTypeInt32: typedValue = QVariant(cookedValue.toInt(&convertOk)); if (!convertOnly && convertOk) { if (cookedMin() > typedValue || typedValue > cookedMax()) { errorString = QString("Value must be within %1 and %2").arg(cookedMin().toInt()).arg(cookedMax().toInt()); } } break; case FactMetaData::valueTypeUint8: case FactMetaData::valueTypeUint16: case FactMetaData::valueTypeUint32: typedValue = QVariant(cookedValue.toUInt(&convertOk)); if (!convertOnly && convertOk) { if (cookedMin() > typedValue || typedValue > cookedMax()) { errorString = QString("Value must be within %1 and %2").arg(cookedMin().toUInt()).arg(cookedMax().toUInt()); } } break; case FactMetaData::valueTypeFloat: typedValue = QVariant(cookedValue.toFloat(&convertOk)); if (!convertOnly && convertOk) { if (cookedMin() > typedValue || typedValue > cookedMax()) { errorString = QString("Value must be within %1 and %2").arg(cookedMin().toFloat()).arg(cookedMax().toFloat()); } } break; case FactMetaData::valueTypeElapsedTimeInSeconds: case FactMetaData::valueTypeDouble: typedValue = QVariant(cookedValue.toDouble(&convertOk)); if (!convertOnly && convertOk) { if (cookedMin() > typedValue || typedValue > cookedMax()) { errorString = QString("Value must be within %1 and %2").arg(cookedMin().toDouble()).arg(cookedMax().toDouble()); } } break; case FactMetaData::valueTypeString: convertOk = true; typedValue = QVariant(cookedValue.toString()); break; case FactMetaData::valueTypeBool: convertOk = true; typedValue = QVariant(cookedValue.toBool()); break; } if (!convertOk) { errorString += "Invalid number"; } return convertOk && errorString.isEmpty(); } void FactMetaData::setBitmaskInfo(const QStringList& strings, const QVariantList& values) { if (strings.count() != values.count()) { qWarning() << "Count mismatch strings:values" << strings.count() << values.count(); return; } _bitmaskStrings = strings; _bitmaskValues = values; setBuiltInTranslator(); } void FactMetaData::addBitmaskInfo(const QString& name, const QVariant& value) { _bitmaskStrings << name; _bitmaskValues << value; } void FactMetaData::setEnumInfo(const QStringList& strings, const QVariantList& values) { if (strings.count() != values.count()) { qWarning() << "Count mismatch strings:values" << strings.count() << values.count(); return; } _enumStrings = strings; _enumValues = values; setBuiltInTranslator(); } void FactMetaData::addEnumInfo(const QString& name, const QVariant& value) { _enumStrings << name; _enumValues << value; } void FactMetaData::setTranslators(Translator rawTranslator, Translator cookedTranslator) { _rawTranslator = rawTranslator; _cookedTranslator = cookedTranslator; } void FactMetaData::setBuiltInTranslator(void) { if (_enumStrings.count()) { // No translation if enum setTranslators(_defaultTranslator, _defaultTranslator); _cookedUnits = _rawUnits; return; } else { for (size_t i=0; irawUnits == _rawUnits.toLower()) { _cookedUnits = pBuiltInTranslation->cookedUnits; setTranslators(pBuiltInTranslation->rawTranslator, pBuiltInTranslation->cookedTranslator); return; } } } // Translator not yet set, try app settings translators _setAppSettingsTranslators(); } QVariant FactMetaData::_degreesToRadians(const QVariant& degrees) { return QVariant(qDegreesToRadians(degrees.toDouble())); } QVariant FactMetaData::_radiansToDegrees(const QVariant& radians) { return QVariant(qRadiansToDegrees(radians.toDouble())); } QVariant FactMetaData::_centiDegreesToDegrees(const QVariant& centiDegrees) { return QVariant(centiDegrees.toReal() / 100.0); } QVariant FactMetaData::_degreesToCentiDegrees(const QVariant& degrees) { return QVariant(qRound(degrees.toReal() * 100.0)); } QVariant FactMetaData::_userGimbalDegreesToMavlinkGimbalDegrees(const QVariant& userGimbalDegrees) { // User facing gimbal degree values are from 0 (level) to 90 (straight down) // Mavlink gimbal degree values are from 0 (level) to -90 (straight down) return userGimbalDegrees.toDouble() * -1.0; } QVariant FactMetaData::_mavlinkGimbalDegreesToUserGimbalDegrees(const QVariant& mavlinkGimbalDegrees) { // User facing gimbal degree values are from 0 (level) to 90 (straight down) // Mavlink gimbal degree values are from 0 (level) to -90 (straight down) return mavlinkGimbalDegrees.toDouble() * -1.0; } QVariant FactMetaData::_metersToFeet(const QVariant& meters) { return QVariant(meters.toDouble() * 1.0/constants.feetToMeters); } QVariant FactMetaData::_feetToMeters(const QVariant& feet) { return QVariant(feet.toDouble() * constants.feetToMeters); } QVariant FactMetaData::_squareMetersToSquareKilometers(const QVariant& squareMeters) { return QVariant(squareMeters.toDouble() * 0.000001); } QVariant FactMetaData::_squareKilometersToSquareMeters(const QVariant& squareKilometers) { return QVariant(squareKilometers.toDouble() * 1000000.0); } QVariant FactMetaData::_squareMetersToHectares(const QVariant& squareMeters) { return QVariant(squareMeters.toDouble() * 0.0001); } QVariant FactMetaData::_hectaresToSquareMeters(const QVariant& hectares) { return QVariant(hectares.toDouble() * 1000.0); } QVariant FactMetaData::_squareMetersToSquareFeet(const QVariant& squareMeters) { return QVariant(squareMeters.toDouble() * 10.7639); } QVariant FactMetaData::_squareFeetToSquareMeters(const QVariant& squareFeet) { return QVariant(squareFeet.toDouble() * 0.0929); } QVariant FactMetaData::_squareMetersToAcres(const QVariant& squareMeters) { return QVariant(squareMeters.toDouble() * 0.000247105); } QVariant FactMetaData::_acresToSquareMeters(const QVariant& acres) { return QVariant(acres.toDouble() * 4046.86); } QVariant FactMetaData::_squareMetersToSquareMiles(const QVariant& squareMeters) { return QVariant(squareMeters.toDouble() * 3.86102e-7); } QVariant FactMetaData::_squareMilesToSquareMeters(const QVariant& squareMiles) { return QVariant(squareMiles.toDouble() * 258999039.98855); } QVariant FactMetaData::_metersPerSecondToMilesPerHour(const QVariant& metersPerSecond) { return QVariant((metersPerSecond.toDouble() * 1.0/constants.milesToMeters) * constants.secondsPerHour); } QVariant FactMetaData::_milesPerHourToMetersPerSecond(const QVariant& milesPerHour) { return QVariant((milesPerHour.toDouble() * constants.milesToMeters) / constants.secondsPerHour); } QVariant FactMetaData::_metersPerSecondToKilometersPerHour(const QVariant& metersPerSecond) { return QVariant((metersPerSecond.toDouble() / 1000.0) * constants.secondsPerHour); } QVariant FactMetaData::_kilometersPerHourToMetersPerSecond(const QVariant& kilometersPerHour) { return QVariant((kilometersPerHour.toDouble() * 1000.0) / constants.secondsPerHour); } QVariant FactMetaData::_metersPerSecondToKnots(const QVariant& metersPerSecond) { return QVariant(metersPerSecond.toDouble() * constants.secondsPerHour / (1000.0 * constants.knotsToKPH)); } QVariant FactMetaData::_knotsToMetersPerSecond(const QVariant& knots) { return QVariant(knots.toDouble() * (1000.0 * constants.knotsToKPH / constants.secondsPerHour)); } QVariant FactMetaData::_percentToNorm(const QVariant& percent) { return QVariant(percent.toDouble() / 100.0); } QVariant FactMetaData::_normToPercent(const QVariant& normalized) { return QVariant(normalized.toDouble() * 100.0); } QVariant FactMetaData::_centimetersToInches(const QVariant& centimeters) { return QVariant(centimeters.toDouble() * 1.0/constants.inchesToCentimeters); } QVariant FactMetaData::_inchesToCentimeters(const QVariant& inches) { return QVariant(inches.toDouble() * constants.inchesToCentimeters); } void FactMetaData::setRawUnits(const QString& rawUnits) { _rawUnits = rawUnits; _cookedUnits = rawUnits; setBuiltInTranslator(); } FactMetaData::ValueType_t FactMetaData::stringToType(const QString& typeString, bool& unknownType) { QStringList knownTypeStrings; QList knownTypes; unknownType = false; knownTypeStrings << QStringLiteral("Uint8") << QStringLiteral("Int8") << QStringLiteral("Uint16") << QStringLiteral("Int16") << QStringLiteral("Uint32") << QStringLiteral("Int32") << QStringLiteral("Float") << QStringLiteral("Double") << QStringLiteral("String") << QStringLiteral("Bool") << QStringLiteral("ElapsedSeconds"); knownTypes << valueTypeUint8 << valueTypeInt8 << valueTypeUint16 << valueTypeInt16 << valueTypeUint32 << valueTypeInt32 << valueTypeFloat << valueTypeDouble << valueTypeString << valueTypeBool << valueTypeElapsedTimeInSeconds; for (int i=0; irawUnits == _rawUnits.toLower() && ((pAppSettingsTranslation->speed && pAppSettingsTranslation->speedOrDistanceUnits == qgcApp()->toolbox()->settingsManager()->unitsSettings()->speedUnits()->rawValue().toUInt()) || (!pAppSettingsTranslation->speed && pAppSettingsTranslation->speedOrDistanceUnits == qgcApp()->toolbox()->settingsManager()->unitsSettings()->distanceUnits()->rawValue().toUInt()))) { _cookedUnits = pAppSettingsTranslation->cookedUnits; setTranslators(pAppSettingsTranslation->rawTranslator, pAppSettingsTranslation->cookedTranslator); return; } } } } const FactMetaData::AppSettingsTranslation_s* FactMetaData::_findAppSettingsDistanceUnitsTranslation(const QString& rawUnits) { for (size_t i=0; irawUnits == rawUnits && (!pAppSettingsTranslation->speed && pAppSettingsTranslation->speedOrDistanceUnits == qgcApp()->toolbox()->settingsManager()->unitsSettings()->distanceUnits()->rawValue().toUInt())) { return pAppSettingsTranslation; } } return NULL; } const FactMetaData::AppSettingsTranslation_s* FactMetaData::_findAppSettingsAreaUnitsTranslation(const QString& rawUnits) { for (size_t i=0; irawUnits == rawUnits && (!pAppSettingsTranslation->speed && pAppSettingsTranslation->speedOrDistanceUnits == qgcApp()->toolbox()->settingsManager()->unitsSettings()->areaUnits()->rawValue().toUInt()) ) { return pAppSettingsTranslation; } } return NULL; } QVariant FactMetaData::metersToAppSettingsDistanceUnits(const QVariant& meters) { const AppSettingsTranslation_s* pAppSettingsTranslation = _findAppSettingsDistanceUnitsTranslation("m"); if (pAppSettingsTranslation) { return pAppSettingsTranslation->rawTranslator(meters); } else { return meters; } } QVariant FactMetaData::appSettingsDistanceUnitsToMeters(const QVariant& distance) { const AppSettingsTranslation_s* pAppSettingsTranslation = _findAppSettingsDistanceUnitsTranslation("m"); if (pAppSettingsTranslation) { return pAppSettingsTranslation->cookedTranslator(distance); } else { return distance; } } QString FactMetaData::appSettingsDistanceUnitsString(void) { const AppSettingsTranslation_s* pAppSettingsTranslation = _findAppSettingsDistanceUnitsTranslation("m"); if (pAppSettingsTranslation) { return pAppSettingsTranslation->cookedUnits; } else { return QStringLiteral("m"); } } QVariant FactMetaData::squareMetersToAppSettingsAreaUnits(const QVariant& squareMeters) { const AppSettingsTranslation_s* pAppSettingsTranslation = _findAppSettingsAreaUnitsTranslation("m^2"); if (pAppSettingsTranslation) { return pAppSettingsTranslation->rawTranslator(squareMeters); } else { return squareMeters; } } QVariant FactMetaData::appSettingsAreaUnitsToSquareMeters(const QVariant& area) { const AppSettingsTranslation_s* pAppSettingsTranslation = _findAppSettingsAreaUnitsTranslation("m^2"); if (pAppSettingsTranslation) { return pAppSettingsTranslation->cookedTranslator(area); } else { return area; } } QString FactMetaData::appSettingsAreaUnitsString(void) { const AppSettingsTranslation_s* pAppSettingsTranslation = _findAppSettingsAreaUnitsTranslation("m^2"); if (pAppSettingsTranslation) { return pAppSettingsTranslation->cookedUnits; } else { return QStringLiteral("m^2"); } } int FactMetaData::decimalPlaces(void) const { int actualDecimalPlaces = defaultDecimalPlaces; int incrementDecimalPlaces = unknownDecimalPlaces; // First determine decimal places from increment double increment = _rawTranslator(this->increment()).toDouble(); if (!qIsNaN(increment)) { double integralPart; // Get the fractional part only increment = fabs(modf(increment, &integralPart)); if (increment == 0.0) { // No fractional part, so no decimal places incrementDecimalPlaces = 0; } else { incrementDecimalPlaces = -ceil(log10(increment)); } } // Correct decimal places is the larger of the two, increment or meta data value if (incrementDecimalPlaces != unknownDecimalPlaces && _decimalPlaces == unknownDecimalPlaces) { actualDecimalPlaces = incrementDecimalPlaces; } else { // Adjust decimal places for cooked translation int settingsDecimalPlaces = _decimalPlaces == unknownDecimalPlaces ? defaultDecimalPlaces : _decimalPlaces; double ctest = _rawTranslator(1.0).toDouble(); settingsDecimalPlaces += -log10(ctest); settingsDecimalPlaces = qMin(25, settingsDecimalPlaces); settingsDecimalPlaces = qMax(0, settingsDecimalPlaces); actualDecimalPlaces = qMax(settingsDecimalPlaces, incrementDecimalPlaces); } return actualDecimalPlaces; } FactMetaData* FactMetaData::createFromJsonObject(const QJsonObject& json, QObject* metaDataParent) { QString errorString; // Make sure we have the required keys QStringList requiredKeys; requiredKeys << _nameJsonKey << _typeJsonKey; if (!JsonHelper::validateRequiredKeys(json, requiredKeys, errorString)) { qWarning() << errorString; return new FactMetaData(valueTypeUint32, metaDataParent); } // Validate key types QStringList keys; QList types; keys << _nameJsonKey << _decimalPlacesJsonKey << _typeJsonKey << _shortDescriptionJsonKey << _longDescriptionJsonKey << _unitsJsonKey << _minJsonKey << _maxJsonKey; types << QJsonValue::String << QJsonValue::Double << QJsonValue::String << QJsonValue::String << QJsonValue::String << QJsonValue::String << QJsonValue::Double << QJsonValue::Double; if (!JsonHelper::validateKeyTypes(json, keys, types, errorString)) { qWarning() << errorString; return new FactMetaData(valueTypeUint32, metaDataParent); } bool unknownType; FactMetaData::ValueType_t type = FactMetaData::stringToType(json[_typeJsonKey].toString(), unknownType); if (unknownType) { qWarning() << "Unknown type" << json[_typeJsonKey].toString(); return new FactMetaData(valueTypeUint32, metaDataParent); } FactMetaData* metaData = new FactMetaData(type, metaDataParent); metaData->_name = json[_nameJsonKey].toString(); QStringList enumValues, enumStrings; if (JsonHelper::parseEnum(json, enumStrings, enumValues, errorString)) { for (int i=0; iconvertAndValidateRaw(enumValues[i], false /* validate */, enumVariant, errorString)) { metaData->addEnumInfo(enumStrings[i], enumVariant); } else { qWarning() << "Invalid enum value, name:" << metaData->name() << " type:" << metaData->type() << " value:" << enumValues[i] << " error:" << errorString; } } } else { qWarning() << errorString; } metaData->setDecimalPlaces(json[_decimalPlacesJsonKey].toInt(0)); metaData->setShortDescription(json[_shortDescriptionJsonKey].toString()); metaData->setLongDescription(json[_longDescriptionJsonKey].toString()); if (json.contains(_unitsJsonKey)) { metaData->setRawUnits(json[_unitsJsonKey].toString()); } #ifdef __mobile__ if (json.contains(_mobileDefaultValueJsonKey)) { metaData->setRawDefaultValue(json[_mobileDefaultValueJsonKey].toVariant()); } else if (json.contains(_defaultValueJsonKey)) { metaData->setRawDefaultValue(json[_defaultValueJsonKey].toVariant()); } #else if (json.contains(_defaultValueJsonKey)) { metaData->setRawDefaultValue(json[_defaultValueJsonKey].toVariant()); } #endif if (json.contains(_minJsonKey)) { QVariant typedValue; QString errorString; metaData->convertAndValidateRaw(json[_minJsonKey].toVariant(), true /* convertOnly */, typedValue, errorString); metaData->setRawMin(typedValue); } if (json.contains(_maxJsonKey)) { QVariant typedValue; QString errorString; metaData->convertAndValidateRaw(json[_maxJsonKey].toVariant(), true /* convertOnly */, typedValue, errorString); metaData->setRawMax(typedValue); } return metaData; } QMap FactMetaData::createMapFromJsonFile(const QString& jsonFilename, QObject* metaDataParent) { QMap metaDataMap; QFile jsonFile(jsonFilename); if (!jsonFile.open(QIODevice::ReadOnly | QIODevice::Text)) { qWarning() << "Unable to open file" << jsonFilename << jsonFile.errorString(); return metaDataMap; } QByteArray bytes = jsonFile.readAll(); jsonFile.close(); QJsonParseError jsonParseError; QJsonDocument doc = QJsonDocument::fromJson(bytes, &jsonParseError); if (jsonParseError.error != QJsonParseError::NoError) { qWarning() << "Unable to parse json document" << jsonFilename << jsonParseError.errorString(); return metaDataMap; } if (!doc.isArray()) { qWarning() << "json document is not array"; return metaDataMap; } QJsonArray jsonArray = doc.array(); for (int i=0; iname())) { qWarning() << QStringLiteral("Duplicate fact name:") << metaData->name(); } else { metaDataMap[metaData->name()] = metaData; } } return metaDataMap; } QVariant FactMetaData::cookedMax(void) const { // We have to be careful with cooked min/max. Running the raw values through the translator could flip min and max. QVariant cookedMax = _rawTranslator(_rawMax); QVariant cookedMin = _rawTranslator(_rawMin); if (cookedMax < cookedMin) { // We need to flip return cookedMin; } else { return cookedMax; } } QVariant FactMetaData::cookedMin(void) const { // We have to be careful with cooked min/max. Running the raw values through the translator could flip min and max. QVariant cookedMax = _rawTranslator(_rawMax); QVariant cookedMin = _rawTranslator(_rawMin); if (cookedMax < cookedMin) { // We need to flip return cookedMax; } else { return cookedMin; } }