/************************************************************************ ** ** @file vmeasurements.cpp ** @author Roman Telezhynskyi ** @date 14 7, 2015 ** ** @brief ** @copyright ** This source code is part of the Valentina project, a pattern making ** program, whose allow create and modeling patterns of clothing. ** Copyright (C) 2015 Valentina project ** All Rights Reserved. ** ** Valentina is free software: you can redistribute it and/or modify ** it under the terms of the GNU General Public License as published by ** the Free Software Foundation, either version 3 of the License, or ** (at your option) any later version. ** ** Valentina is distributed in the hope that it will be useful, ** but WITHOUT ANY WARRANTY; without even the implied warranty of ** MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the ** GNU General Public License for more details. ** ** You should have received a copy of the GNU General Public License ** along with Valentina. If not, see . ** *************************************************************************/ #include "vmeasurements.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "../ifc/exception/vexceptionemptyparameter.h" #include "../ifc/xml/vvitconverter.h" #include "../ifc/xml/vvstconverter.h" #include "../ifc/ifcdef.h" #include "../qmuparser/qmutokenparser.h" #include "../qmuparser/qmuparsererror.h" #include "../vpatterndb/calculator.h" #include "../vpatterndb/variables/vmeasurement.h" #include "../vpatterndb/vcontainer.h" #include "../vpatterndb/measurements.h" #include "../vpatterndb/pmsystems.h" #include "../vmisc/projectversion.h" const QString VMeasurements::TagVST = QStringLiteral("vst"); const QString VMeasurements::TagVIT = QStringLiteral("vit"); const QString VMeasurements::TagBodyMeasurements = QStringLiteral("body-measurements"); const QString VMeasurements::TagNotes = QStringLiteral("notes"); const QString VMeasurements::TagPersonal = QStringLiteral("personal"); const QString VMeasurements::TagCustomer = QStringLiteral("customer"); const QString VMeasurements::TagBirthDate = QStringLiteral("birth-date"); const QString VMeasurements::TagGender = QStringLiteral("gender"); const QString VMeasurements::TagPMSystem = QStringLiteral("pm_system"); const QString VMeasurements::TagEmail = QStringLiteral("email"); const QString VMeasurements::TagReadOnly = QStringLiteral("read-only"); const QString VMeasurements::TagMeasurement = QStringLiteral("m"); const QString VMeasurements::TagDimensions = QStringLiteral("dimensions"); const QString VMeasurements::TagDimension = QStringLiteral("dimension"); const QString VMeasurements::TagRestrictions = QStringLiteral("restrictions"); const QString VMeasurements::TagRestriction = QStringLiteral("restriction"); const QString VMeasurements::TagCorrections = QStringLiteral("corrections"); const QString VMeasurements::TagCorrection = QStringLiteral("correction"); const QString VMeasurements::TagLabels = QStringLiteral("labels"); const QString VMeasurements::TagLabel = QStringLiteral("label"); const QString VMeasurements::AttrBase = QStringLiteral("base"); const QString VMeasurements::AttrValue = QStringLiteral("value"); const QString VMeasurements::AttrShiftA = QStringLiteral("shiftA"); const QString VMeasurements::AttrShiftB = QStringLiteral("shiftB"); const QString VMeasurements::AttrShiftC = QStringLiteral("shiftC"); const QString VMeasurements::AttrCorrection = QStringLiteral("correction"); const QString VMeasurements::AttrCoordinates = QStringLiteral("coordinates"); const QString VMeasurements::AttrExclude = QStringLiteral("exclude"); const QString VMeasurements::AttrSpecialUnits = QStringLiteral("specialUnits"); const QString VMeasurements::AttrDescription = QStringLiteral("description"); const QString VMeasurements::AttrName = QStringLiteral("name"); const QString VMeasurements::AttrFullName = QStringLiteral("full_name"); const QString VMeasurements::AttrMin = QStringLiteral("min"); const QString VMeasurements::AttrMax = QStringLiteral("max"); const QString VMeasurements::AttrStep = QStringLiteral("step"); const QString VMeasurements::AttrCircumference = QStringLiteral("circumference"); const QString VMeasurements::AttrFullCircumference = QStringLiteral("fullCircumference"); const QString VMeasurements::AttrLabel = QStringLiteral("label"); const QString VMeasurements::AttrDimension = QStringLiteral("dimension"); const QString VMeasurements::GenderMale = QStringLiteral("male"); const QString VMeasurements::GenderFemale = QStringLiteral("female"); const QString VMeasurements::GenderUnknown = QStringLiteral("unknown"); const QString VMeasurements::DimensionN = QStringLiteral("n"); const QString VMeasurements::DimensionX = QStringLiteral("x"); const QString VMeasurements::DimensionY = QStringLiteral("y"); const QString VMeasurements::DimensionW = QStringLiteral("w"); const QString VMeasurements::DimensionZ = QStringLiteral("z"); namespace { Q_GLOBAL_STATIC_WITH_ARGS(const QString, defBirthDate, (QLatin1String("1800-01-01"))) //--------------------------------------------------------------------------------------------------------------------- QString FileComment() { return QString("Measurements created with Valentina v%1 (https://smart-pattern.com.ua/).") .arg(APP_VERSION_STR); } } //--------------------------------------------------------------------------------------------------------------------- VMeasurements::VMeasurements(VContainer *data) : data(data), type(MeasurementsType::Unknown) { SCASSERT(data != nullptr) } //--------------------------------------------------------------------------------------------------------------------- VMeasurements::VMeasurements(Unit unit, VContainer *data) : data(data), type(MeasurementsType::Individual) { SCASSERT(data != nullptr) CreateEmptyIndividualFile(unit); } //--------------------------------------------------------------------------------------------------------------------- VMeasurements::VMeasurements(Unit unit, const QVector &dimensions, VContainer *data) : data(data), type(MeasurementsType::Multisize) { SCASSERT(data != nullptr) CreateEmptyMultisizeFile(unit, dimensions); } //--------------------------------------------------------------------------------------------------------------------- void VMeasurements::setXMLContent(const QString &fileName) { VDomDocument::setXMLContent(fileName); type = ReadType(); m_units = ReadUnits(); m_dimensions = ReadDimensions(); } //--------------------------------------------------------------------------------------------------------------------- bool VMeasurements::SaveDocument(const QString &fileName, QString &error) { // Update comment with Valentina version QDomNode commentNode = documentElement().firstChild(); if (commentNode.isComment()) { QDomComment comment = commentNode.toComment(); comment.setData(FileComment()); } return VDomDocument::SaveDocument(fileName, error); } //--------------------------------------------------------------------------------------------------------------------- void VMeasurements::AddEmpty(const QString &name, const QString &formula) { const QDomElement element = MakeEmpty(name, formula); const QDomNodeList list = elementsByTagName(TagBodyMeasurements); list.at(0).appendChild(element); } //--------------------------------------------------------------------------------------------------------------------- void VMeasurements::AddEmptyAfter(const QString &after, const QString &name, const QString &formula) { const QDomElement element = MakeEmpty(name, formula); const QDomElement sibling = FindM(after); const QDomNodeList list = elementsByTagName(TagBodyMeasurements); if (sibling.isNull()) { list.at(0).appendChild(element); } else { list.at(0).insertAfter(element, sibling); } } //--------------------------------------------------------------------------------------------------------------------- void VMeasurements::Remove(const QString &name) { const QDomNodeList list = elementsByTagName(TagBodyMeasurements); list.at(0).removeChild(FindM(name)); } //--------------------------------------------------------------------------------------------------------------------- void VMeasurements::MoveTop(const QString &name) { const QDomElement node = FindM(name); if (not node.isNull()) { const QDomNodeList mList = elementsByTagName(TagMeasurement); if (mList.size() >= 2) { const QDomNode top = mList.at(0); if (not top.isNull()) { const QDomNodeList list = elementsByTagName(TagBodyMeasurements); list.at(0).insertBefore(node, top); } } } } //--------------------------------------------------------------------------------------------------------------------- void VMeasurements::MoveUp(const QString &name) { const QDomElement node = FindM(name); if (not node.isNull()) { const QDomElement prSibling = node.previousSiblingElement(TagMeasurement); if (not prSibling.isNull()) { const QDomNodeList list = elementsByTagName(TagBodyMeasurements); list.at(0).insertBefore(node, prSibling); } } } //--------------------------------------------------------------------------------------------------------------------- void VMeasurements::MoveDown(const QString &name) { const QDomElement node = FindM(name); if (not node.isNull()) { const QDomElement nextSibling = node.nextSiblingElement(TagMeasurement); if (not nextSibling.isNull()) { const QDomNodeList list = elementsByTagName(TagBodyMeasurements); list.at(0).insertAfter(node, nextSibling); } } } //--------------------------------------------------------------------------------------------------------------------- void VMeasurements::MoveBottom(const QString &name) { const QDomElement node = FindM(name); if (not node.isNull()) { const QDomNodeList mList = elementsByTagName(TagMeasurement); if (mList.size() >= 2) { const QDomNode bottom = mList.at(mList.size()-1); if (not bottom.isNull()) { const QDomNodeList list = elementsByTagName(TagBodyMeasurements); list.at(0).insertAfter(node, bottom); } } } } //--------------------------------------------------------------------------------------------------------------------- auto VMeasurements::Units() const -> Unit { return m_units; } //--------------------------------------------------------------------------------------------------------------------- void VMeasurements::StoreNames(bool store) { m_keepNames = store; } //--------------------------------------------------------------------------------------------------------------------- void VMeasurements::ReadMeasurements(qreal baseA, qreal baseB, qreal baseC) const { // For conversion values we must first calculate all data in measurement file's unit. // That's why we need two containers: one for converted values, second for real data. // Container for values in measurement file's unit QSharedPointer tempData; if (type == MeasurementsType::Individual) { tempData = QSharedPointer::create(data->GetTrVars(), data->GetPatternUnit(), VContainer::UniqueNamespace()); } const QDomNodeList list = elementsByTagName(TagMeasurement); for (int i=0; i < list.size(); ++i) { const QDomElement dom = list.at(i).toElement(); const QString name = GetParametrString(dom, AttrName).simplified(); const QString description = GetParametrEmptyString(dom, AttrDescription); const QString fullName = GetParametrEmptyString(dom, AttrFullName); const bool specialUnits = GetParametrBool(dom, AttrSpecialUnits, falseStr); QSharedPointer meash; QSharedPointer tempMeash; if (type == MeasurementsType::Multisize) { qreal base = GetParametrDouble(dom, AttrBase, QChar('0')); qreal shiftA = GetParametrDouble(dom, AttrShiftA, QChar('0')); qreal shiftB = GetParametrDouble(dom, AttrShiftB, QChar('0')); qreal shiftC = GetParametrDouble(dom, AttrShiftC, QChar('0')); QMap corrections = ReadCorrections(dom); qreal convertedBaseA = DimensionABase(); qreal convertedBaseB = DimensionBBase(); qreal convertedBaseC = DimensionCBase(); qreal convertedStepA = DimensionAStep(); qreal convertedStepB = DimensionBStep(); qreal convertedStepC = DimensionCStep(); if (not specialUnits) { base = UnitConvertor(base, Units(), *data->GetPatternUnit()); shiftA = UnitConvertor(shiftA, Units(), *data->GetPatternUnit()); shiftB = UnitConvertor(shiftB, Units(), *data->GetPatternUnit()); shiftC = UnitConvertor(shiftC, Units(), *data->GetPatternUnit()); QMutableMapIterator iterator(corrections); while (iterator.hasNext()) { iterator.next(); iterator.setValue(UnitConvertor(iterator.value(), Units(), *data->GetPatternUnit())); } convertedBaseA = UnitConvertor(convertedBaseA, Units(), *data->GetPatternUnit()); convertedBaseB = UnitConvertor(convertedBaseB, Units(), *data->GetPatternUnit()); convertedBaseC = UnitConvertor(convertedBaseC, Units(), *data->GetPatternUnit()); convertedStepA = UnitConvertor(convertedStepA, Units(), *data->GetPatternUnit()); convertedStepB = UnitConvertor(convertedStepB, Units(), *data->GetPatternUnit()); convertedStepC = UnitConvertor(convertedStepC, Units(), *data->GetPatternUnit()); } meash = QSharedPointer::create(static_cast(i), name, convertedBaseA, convertedBaseB, convertedBaseC, base); meash->SetBaseA(baseA); meash->SetBaseB(baseB); meash->SetBaseC(baseC); meash->SetShiftA(shiftA); meash->SetShiftB(shiftB); meash->SetShiftC(shiftC); meash->SetStepA(convertedStepA); meash->SetStepB(convertedStepB); meash->SetStepC(convertedStepC); meash->SetSpecialUnits(specialUnits); meash->SetCorrections(corrections); meash->SetGuiText(fullName); meash->SetDescription(description); } else { const IMD dimension = VMeasurements::StrToIMD(GetParametrString(dom, AttrDimension, VMeasurements::IMDToStr(IMD::N))); const QString formula = GetParametrString(dom, AttrValue, QChar('0')); bool ok = false; qreal value = EvalFormula(tempData.data(), formula, &ok); tempMeash = QSharedPointer::create(tempData.data(), static_cast(i), name, value, formula, ok); if (not specialUnits) { value = UnitConvertor(value, Units(), *data->GetPatternUnit()); } meash = QSharedPointer::create(data, static_cast(i), name, value, formula, ok); meash->SetGuiText(fullName); meash->SetDescription(description); meash->SetSpecialUnits(specialUnits); meash->SetDimension(dimension); } if (m_keepNames) { if (not tempData.isNull()) { tempData->AddUniqueVariable(tempMeash); } data->AddUniqueVariable(meash); } else { if (not tempData.isNull()) { tempData->AddVariable(tempMeash); } data->AddVariable(meash); } } } //--------------------------------------------------------------------------------------------------------------------- void VMeasurements::ClearForExport() { const QDomNodeList list = elementsByTagName(TagMeasurement); for (int i=0; i < list.size(); ++i) { QDomElement domElement = list.at(i).toElement(); if (domElement.isNull() == false) { if (qmu::QmuTokenParser::IsSingle(domElement.attribute(AttrValue))) { SetAttribute(domElement, AttrValue, QChar('0')); } } } } //--------------------------------------------------------------------------------------------------------------------- MeasurementsType VMeasurements::Type() const { return type; } //--------------------------------------------------------------------------------------------------------------------- qreal VMeasurements::DimensionABase() const { if (type == MeasurementsType::Multisize) { const auto dimensions = Dimensions(); if (not dimensions.isEmpty()) { return dimensions.first()->BaseValue(); } } return 0; } //--------------------------------------------------------------------------------------------------------------------- qreal VMeasurements::DimensionBBase() const { if (type == MeasurementsType::Multisize) { const auto dimensions = Dimensions(); if (not dimensions.isEmpty() && dimensions.size() >= 2) { return dimensions.values().at(1)->BaseValue(); } } return 0; } //--------------------------------------------------------------------------------------------------------------------- qreal VMeasurements::DimensionCBase() const { if (type == MeasurementsType::Multisize) { const auto dimensions = Dimensions(); if (not dimensions.isEmpty() && dimensions.size() == 3) { return dimensions.last()->BaseValue(); } } return 0; } //--------------------------------------------------------------------------------------------------------------------- qreal VMeasurements::DimensionAStep() const { if (type == MeasurementsType::Multisize) { const auto dimensions = Dimensions(); if (not dimensions.isEmpty()) { return dimensions.first()->Step(); } } return 0; } //--------------------------------------------------------------------------------------------------------------------- qreal VMeasurements::DimensionBStep() const { if (type == MeasurementsType::Multisize) { const auto dimensions = Dimensions(); if (not dimensions.isEmpty() && dimensions.size() >= 2) { return dimensions.values().at(1)->Step(); } } return 0; } //--------------------------------------------------------------------------------------------------------------------- qreal VMeasurements::DimensionCStep() const { if (type == MeasurementsType::Multisize) { const auto dimensions = Dimensions(); if (not dimensions.isEmpty() && dimensions.size() == 3) { return dimensions.last()->Step(); } } return 0; } //--------------------------------------------------------------------------------------------------------------------- QString VMeasurements::Notes() const { return UniqueTagText(TagNotes, QString()); } //--------------------------------------------------------------------------------------------------------------------- void VMeasurements::SetNotes(const QString &text) { if (not IsReadOnly()) { setTagText(TagNotes, text); } } //--------------------------------------------------------------------------------------------------------------------- QString VMeasurements::Customer() const { return UniqueTagText(TagCustomer, QString()); } //--------------------------------------------------------------------------------------------------------------------- void VMeasurements::SetCustomer(const QString &text) { if (not IsReadOnly()) { setTagText(TagCustomer, text); } } //--------------------------------------------------------------------------------------------------------------------- QDate VMeasurements::BirthDate() const { return QDate::fromString(UniqueTagText(TagBirthDate, *defBirthDate), "yyyy-MM-dd"); } //--------------------------------------------------------------------------------------------------------------------- void VMeasurements::SetBirthDate(const QDate &date) { if (not IsReadOnly()) { setTagText(TagBirthDate, date.toString("yyyy-MM-dd")); } } //--------------------------------------------------------------------------------------------------------------------- GenderType VMeasurements::Gender() const { return StrToGender(UniqueTagText(TagGender, GenderUnknown)); } //--------------------------------------------------------------------------------------------------------------------- void VMeasurements::SetGender(const GenderType &gender) { if (not IsReadOnly()) { setTagText(TagGender, GenderToStr(gender)); } } //--------------------------------------------------------------------------------------------------------------------- QString VMeasurements::PMSystem() const { return UniqueTagText(TagPMSystem, ClearPMCode(p998_S)); } //--------------------------------------------------------------------------------------------------------------------- void VMeasurements::SetPMSystem(const QString &system) { if (not IsReadOnly()) { setTagText(TagPMSystem, ClearPMCode(system)); } } //--------------------------------------------------------------------------------------------------------------------- QString VMeasurements::Email() const { return UniqueTagText(TagEmail, QString()); } //--------------------------------------------------------------------------------------------------------------------- void VMeasurements::SetEmail(const QString &text) { if (not IsReadOnly()) { setTagText(TagEmail, text); } } //--------------------------------------------------------------------------------------------------------------------- bool VMeasurements::IsReadOnly() const { return UniqueTagText(TagReadOnly, falseStr) == trueStr; } //--------------------------------------------------------------------------------------------------------------------- void VMeasurements::SetReadOnly(bool ro) { setTagText(TagReadOnly, ro ? trueStr : falseStr); } //--------------------------------------------------------------------------------------------------------------------- bool VMeasurements::IsFullCircumference() const { QDomElement dimenstionsTag = documentElement().firstChildElement(TagDimensions); if (not dimenstionsTag.isNull()) { return GetParametrBool(dimenstionsTag, AttrFullCircumference, falseStr); } else { qDebug()<<"Can't read full circumference "< corrections = ReadCorrections(mElement); const QString hash = VMeasurement::CorrectionHash(baseA, baseB, baseC); if (not qFuzzyIsNull(value)) { corrections.insert(hash, value); } else { if (corrections.contains(hash)) { corrections.remove(hash); } } WriteCorrections(mElement, corrections); } else { qWarning() << tr("Can't find measurement '%1'").arg(name); } } //--------------------------------------------------------------------------------------------------------------------- void VMeasurements::SetMDescription(const QString &name, const QString &text) { QDomElement node = FindM(name); if (not node.isNull()) { SetAttribute(node, AttrDescription, text); } else { qWarning() << tr("Can't find measurement '%1'").arg(name); } } //--------------------------------------------------------------------------------------------------------------------- void VMeasurements::SetMFullName(const QString &name, const QString &text) { QDomElement node = FindM(name); if (not node.isNull()) { SetAttribute(node, AttrFullName, text); } else { qWarning() << tr("Can't find measurement '%1'").arg(name); } } //--------------------------------------------------------------------------------------------------------------------- void VMeasurements::SetMDimension(const QString &name, IMD type) { ClearDimension(type); QDomElement node = FindM(name); if (not node.isNull()) { SetAttributeOrRemoveIf(node, AttrDimension, VMeasurements::IMDToStr(type), type == IMD::N); } else { qWarning() << tr("Can't find measurement '%1'").arg(name); } } //--------------------------------------------------------------------------------------------------------------------- QString VMeasurements::MeasurementForDimension(IMD type) const { const QString d = VMeasurements::IMDToStr(type); QDomNodeList list = elementsByTagName(TagMeasurement); for (int i=0; i < list.size(); ++i) { const QDomElement domElement = list.at(i).toElement(); if (domElement.isNull() == false) { if (domElement.attribute(AttrDimension) == d) { return domElement.attribute(AttrName); } } } return QString(); } //--------------------------------------------------------------------------------------------------------------------- auto VMeasurements::Dimensions() const -> VDimensions { return m_dimensions; } //--------------------------------------------------------------------------------------------------------------------- auto VMeasurements::GetRestrictions() const -> QMap { QMap restrictions; const QDomNodeList list = elementsByTagName(TagRestriction); for (int i=0; i < list.size(); ++i) { const QDomElement res = list.at(i).toElement(); QString coordinates = GetParametrString(res, AttrCoordinates); const qreal min = GetParametrDouble(res, AttrMin, QChar('0')); const qreal max = GetParametrDouble(res, AttrMax, QChar('0')); const QString exclude = GetParametrEmptyString(res, AttrExclude); restrictions.insert(coordinates, VDimensionRestriction(min, max, exclude)); } return restrictions; } //--------------------------------------------------------------------------------------------------------------------- void VMeasurements::SetRestrictions(const QMap &restrictions) { QDomElement root = documentElement(); QDomElement restrictionsTag = root.firstChildElement(TagRestrictions); if (restrictionsTag.isNull()) { qDebug() << "Can't find restrictions tag"; } RemoveAllChildren(restrictionsTag); auto i = restrictions.constBegin(); while (i != restrictions.constEnd()) { QDomElement restrictionTag = createElement(TagRestriction); SetAttribute(restrictionTag, AttrCoordinates, i.key()); SetAttributeOrRemoveIf(restrictionTag, AttrMin, i.value().GetMin(), qFuzzyIsNull(i.value().GetMin())); SetAttributeOrRemoveIf(restrictionTag, AttrMax, i.value().GetMax(), qFuzzyIsNull(i.value().GetMax())); SetAttributeOrRemoveIf(restrictionTag, AttrExclude, i.value().GetExcludeString(), i.value().GetExcludeString().isEmpty()); restrictionsTag.appendChild(restrictionTag); ++i; } } //--------------------------------------------------------------------------------------------------------------------- auto VMeasurements::Restriction(qreal base, qreal base2) const -> VDimensionRestriction { const QMap restrictions = GetRestrictions(); const QString hash = VMeasurement::CorrectionHash(base, base2, 0); return restrictions.value(hash); } //--------------------------------------------------------------------------------------------------------------------- void VMeasurements::SetDimensionLabels(const QMap &labels) { const QDomNodeList list = elementsByTagName(TagDimension); for (int i=0; i < list.size(); ++i) { QDomElement dom = list.at(i).toElement(); const MeasurementDimension type = StrToDimensionType(GetParametrString(dom, AttrType)); if (labels.contains(type)) { SaveDimesionLabels(dom, labels.value(type)); } } m_dimensions = ReadDimensions(); // Refresh cache } //--------------------------------------------------------------------------------------------------------------------- QString VMeasurements::GenderToStr(const GenderType &sex) { switch (sex) { case GenderType::Male: return GenderMale; case GenderType::Female: return GenderFemale; case GenderType::Unknown: default: return GenderUnknown; } } //--------------------------------------------------------------------------------------------------------------------- GenderType VMeasurements::StrToGender(const QString &sex) { const QStringList genders = QStringList() << GenderMale << GenderFemale << GenderUnknown; switch (genders.indexOf(sex)) { case 0: // GenderMale return GenderType::Male; case 1: // GenderFemale return GenderType::Female; case 2: // GenderUnknown default: return GenderType::Unknown; } } //--------------------------------------------------------------------------------------------------------------------- QString VMeasurements::DimensionTypeToStr(const MeasurementDimension &type) { switch (type) { case MeasurementDimension::Y: return DimensionY; case MeasurementDimension::W: return DimensionW; case MeasurementDimension::Z: return DimensionZ; case MeasurementDimension::X: default: return DimensionX; } } //--------------------------------------------------------------------------------------------------------------------- MeasurementDimension VMeasurements::StrToDimensionType(const QString &type) { const QStringList dimensions = QStringList{DimensionX, DimensionY, DimensionW, DimensionZ}; switch (dimensions.indexOf(type)) { case 1: // DimensionY return MeasurementDimension::Y; case 2: // DimensionW return MeasurementDimension::W; case 3: // DimensionZ return MeasurementDimension::Z; case 0: // DimensionX default: return MeasurementDimension::X; } } //--------------------------------------------------------------------------------------------------------------------- QString VMeasurements::IMDToStr(const IMD &type) { switch (type) { case IMD::X: return DimensionX; case IMD::Y: return DimensionY; case IMD::W: return DimensionW; case IMD::Z: return DimensionZ; case IMD::N: default: return DimensionN; } } //--------------------------------------------------------------------------------------------------------------------- IMD VMeasurements::StrToIMD(const QString &type) { const QStringList dimensions = QStringList{DimensionN, DimensionX, DimensionY, DimensionW, DimensionZ}; switch (dimensions.indexOf(type)) { case 1: // DimensionX return IMD::X; case 2: // DimensionY return IMD::Y; case 3: // DimensionW return IMD::W; case 4: // DimensionZ return IMD::Z; case 0: // DimensionN default: return IMD::N; } } //--------------------------------------------------------------------------------------------------------------------- QString VMeasurements::IMDName(IMD type) { switch(type) { case IMD::N: return tr("None"); case IMD::X: return tr("Height"); case IMD::Y: return tr("Size"); case IMD::W: return tr("Waist"); case IMD::Z: return tr("Hip"); default: return QString(); } } //--------------------------------------------------------------------------------------------------------------------- QStringList VMeasurements::ListAll() const { QStringList listNames; const QDomNodeList list = elementsByTagName(TagMeasurement); for (int i=0; i < list.size(); ++i) { const QDomElement domElement = list.at(i).toElement(); if (domElement.isNull() == false) { listNames.append(domElement.attribute(AttrName)); } } return listNames; } //--------------------------------------------------------------------------------------------------------------------- QStringList VMeasurements::ListKnown() const { QStringList listNames; const QStringList list = ListAll(); for (int i=0; i < list.size(); ++i) { if (list.at(i).indexOf(CustomMSign) != 0) { listNames.append(list.at(i)); } } return listNames; } //--------------------------------------------------------------------------------------------------------------------- bool VMeasurements::IsDefinedKnownNamesValid() const { QStringList names = AllGroupNames(); QSet set; for (const auto &var : names) { set.insert(var); } names = ListKnown(); for (const auto &var : qAsConst(names)) { if (not set.contains(var)) { return false; } } return true; } //--------------------------------------------------------------------------------------------------------------------- VContainer *VMeasurements::GetData() const { return data; } //--------------------------------------------------------------------------------------------------------------------- void VMeasurements::CreateEmptyMultisizeFile(Unit unit, const QVector &dimensions) { this->clear(); QDomElement mElement = this->createElement(TagVST); mElement.appendChild(createComment(FileComment())); mElement.appendChild(CreateElementWithText(TagVersion, VVSTConverter::MeasurementMaxVerStr)); mElement.appendChild(CreateElementWithText(TagReadOnly, falseStr)); mElement.appendChild(createElement(TagNotes)); mElement.appendChild(CreateElementWithText(TagUnit, UnitsToStr(unit))); mElement.appendChild(CreateElementWithText(TagPMSystem, ClearPMCode(p998_S))); mElement.appendChild(CreateDimensions(dimensions)); mElement.appendChild(createElement(TagRestrictions)); mElement.appendChild(createElement(TagBodyMeasurements)); this->appendChild(mElement); insertBefore(createProcessingInstruction(QStringLiteral("xml"), QStringLiteral("version=\"1.0\" encoding=\"UTF-8\"")), this->firstChild()); // Cache data m_units = unit; m_dimensions = ReadDimensions(); } //--------------------------------------------------------------------------------------------------------------------- void VMeasurements::CreateEmptyIndividualFile(Unit unit) { this->clear(); QDomElement mElement = this->createElement(TagVIT); mElement.appendChild(createComment(FileComment())); mElement.appendChild(CreateElementWithText(TagVersion, VVITConverter::MeasurementMaxVerStr)); mElement.appendChild(CreateElementWithText(TagReadOnly, falseStr)); mElement.appendChild(createElement(TagNotes)); mElement.appendChild(CreateElementWithText(TagUnit, UnitsToStr(unit))); mElement.appendChild(CreateElementWithText(TagPMSystem, ClearPMCode(p998_S))); QDomElement personal = createElement(TagPersonal); personal.appendChild(createElement(TagCustomer)); personal.appendChild(CreateElementWithText(TagBirthDate, *defBirthDate)); personal.appendChild(CreateElementWithText(TagGender, GenderToStr(GenderType::Unknown))); personal.appendChild(createElement(TagEmail)); mElement.appendChild(personal); mElement.appendChild(createElement(TagBodyMeasurements)); this->appendChild(mElement); insertBefore(createProcessingInstruction(QStringLiteral("xml"), QStringLiteral("version=\"1.0\" encoding=\"UTF-8\"")), this->firstChild()); // Cache data m_units = unit; m_dimensions = ReadDimensions(); } //--------------------------------------------------------------------------------------------------------------------- QDomElement VMeasurements::CreateDimensions(const QVector &dimensions) { QDomElement dimensionsTag = createElement(TagDimensions); for(auto &dimension : dimensions) { QDomElement dimensionTag = createElement(TagDimension); SetAttribute(dimensionTag, AttrType, DimensionTypeToStr(dimension->Type())); SetAttribute(dimensionTag, AttrBase, dimension->BaseValue()); SetAttribute(dimensionTag, AttrMin, dimension->MinValue()); SetAttribute(dimensionTag, AttrMax, dimension->MaxValue()); SetAttribute(dimensionTag, AttrStep, dimension->Step()); if (dimension->Type() == MeasurementDimension::Y) { SetAttribute(dimensionTag, AttrCircumference, dimension->IsCircumference()); } dimensionsTag.appendChild(dimensionTag); } return dimensionsTag; } //--------------------------------------------------------------------------------------------------------------------- qreal VMeasurements::UniqueTagAttr(const QString &tag, const QString &attr, qreal defValue) const { const qreal defVal = UnitConvertor(defValue, Unit::Cm, Units()); const QDomNodeList nodeList = this->elementsByTagName(tag); if (nodeList.isEmpty()) { return defVal; } else { const QDomNode domNode = nodeList.at(0); if (domNode.isNull() == false && domNode.isElement()) { const QDomElement domElement = domNode.toElement(); if (domElement.isNull() == false) { return GetParametrDouble(domElement, attr, QString("%1").arg(defVal)); } } } return defVal; } //--------------------------------------------------------------------------------------------------------------------- QDomElement VMeasurements::MakeEmpty(const QString &name, const QString &formula) { QDomElement element = createElement(TagMeasurement); SetAttribute(element, AttrName, name); if (type == MeasurementsType::Multisize) { SetAttribute(element, AttrBase, QChar('0')); } else { SetAttribute(element, AttrValue, formula.isEmpty() ? QChar('0') : formula); } return element; } //--------------------------------------------------------------------------------------------------------------------- QDomElement VMeasurements::FindM(const QString &name) const { if (name.isEmpty()) { qWarning() << tr("The measurement name is empty!"); return QDomElement(); } QDomNodeList list = elementsByTagName(TagMeasurement); for (int i=0; i < list.size(); ++i) { const QDomElement domElement = list.at(i).toElement(); if (domElement.isNull() == false) { const QString parameter = domElement.attribute(AttrName); if (parameter == name) { return domElement; } } } return QDomElement(); } //--------------------------------------------------------------------------------------------------------------------- MeasurementsType VMeasurements::ReadType() const { QDomElement root = documentElement(); if (root.tagName() == TagVST) { return MeasurementsType::Multisize; } else if (root.tagName() == TagVIT) { return MeasurementsType::Individual; } else { return MeasurementsType::Unknown; } } //--------------------------------------------------------------------------------------------------------------------- auto VMeasurements::ReadUnits() const -> Unit { Unit units = StrToUnits(UniqueTagText(TagUnit, unitCM)); if (units == Unit::Px) { units = Unit::Cm; } return units; } //--------------------------------------------------------------------------------------------------------------------- auto VMeasurements::ReadDimensions() const -> VDimensions { if (type != MeasurementsType::Multisize) { return {}; } VDimensions dimensions; const Unit units = Units(); const QDomNodeList list = elementsByTagName(TagDimension); for (int i=0; i < list.size(); ++i) { const QDomElement dom = list.at(i).toElement(); const MeasurementDimension type = StrToDimensionType(GetParametrString(dom, AttrType)); const qreal min = GetParametrDouble(dom, AttrMin, QChar('0')); const qreal max = GetParametrDouble(dom, AttrMax, QChar('0')); const qreal step = GetParametrDouble(dom, AttrStep, QStringLiteral("-1")); const qreal base = GetParametrDouble(dom, AttrBase, QChar('0')); const DimesionLabels labels = ReadDimensionLabels(dom); if (type == MeasurementDimension::X) { auto dimension = QSharedPointer::create(units, min, max, step); dimension->SetBaseValue(base); dimension->SetLabels(labels); dimensions.insert(type, dimension); } else if (type == MeasurementDimension::Y) { auto dimension = QSharedPointer::create(units, min, max, step); dimension->SetBaseValue(base); dimension->SetCircumference(GetParametrBool(dom, AttrCircumference, trueStr)); dimension->SetLabels(labels); dimensions.insert(type, dimension); } else if (type == MeasurementDimension::W) { auto dimension = QSharedPointer::create(units, min, max, step); dimension->SetBaseValue(base); dimension->SetLabels(labels); dimensions.insert(type, dimension); } else if (type == MeasurementDimension::Z) { auto dimension = QSharedPointer::create(units, min, max, step); dimension->SetBaseValue(base); dimension->SetLabels(labels); dimensions.insert(type, dimension); } } return dimensions; } //--------------------------------------------------------------------------------------------------------------------- qreal VMeasurements::EvalFormula(VContainer *data, const QString &formula, bool *ok) const { if (formula.isEmpty()) { *ok = true; return 0; } else { try { QScopedPointer cal(new Calculator()); const qreal result = cal->EvalFormula(data->DataVariables(), formula); (qIsInf(result) || qIsNaN(result)) ? *ok = false : *ok = true; return result; } catch (qmu::QmuParserError &e) { Q_UNUSED(e) *ok = false; return 0; } } } //--------------------------------------------------------------------------------------------------------------------- QString VMeasurements::ClearPMCode(const QString &code) const { QString clear = code; const int index = clear.indexOf(QLatin1Char('p')); if (index == 0) { clear.remove(0, 1); } return clear; } //--------------------------------------------------------------------------------------------------------------------- QMap VMeasurements::ReadCorrections(const QDomElement &mElement) const { if (mElement.isNull()) { return QMap(); } QDomElement correctionsTag = mElement.firstChildElement(TagCorrections); if (correctionsTag.isNull()) { return QMap(); } QMap corrections; QDomNode correctionTag = correctionsTag.firstChild(); while (not correctionTag.isNull()) { if (correctionTag.isElement()) { const QDomElement c = correctionTag.toElement(); const QString hash = GetParametrString(c, AttrCoordinates); const qreal correction = GetParametrDouble(c, AttrCorrection, QChar('0')); corrections.insert(hash, correction); } correctionTag = correctionTag.nextSibling(); } return corrections; } //--------------------------------------------------------------------------------------------------------------------- void VMeasurements::WriteCorrections(QDomElement &mElement, const QMap &corrections) { QDomElement correctionsTag = mElement.firstChildElement(TagCorrections); if (not corrections.isEmpty()) { if (not correctionsTag.isNull()) { RemoveAllChildren(correctionsTag); } else { correctionsTag = createElement(TagCorrections); mElement.appendChild(correctionsTag); } QMap::const_iterator i = corrections.constBegin(); while (i != corrections.constEnd()) { QDomElement correctionTag = createElement(TagCorrection); SetAttribute(correctionTag, AttrCorrection, i.value()); SetAttribute(correctionTag, AttrCoordinates, i.key()); correctionsTag.appendChild(correctionTag); ++i; } } else { if (not correctionsTag.isNull()) { mElement.removeChild(correctionsTag); } } } //--------------------------------------------------------------------------------------------------------------------- void VMeasurements::SaveDimesionLabels(QDomElement &dElement, const DimesionLabels &labels) { if (dElement.isNull()) { qDebug() << "Invalid dimension tag"; } QDomElement labelsTag = dElement.firstChildElement(TagLabels); if (not labels.isEmpty()) { if (not labelsTag.isNull()) { RemoveAllChildren(labelsTag); } else { labelsTag = createElement(TagLabels); dElement.appendChild(labelsTag); } DimesionLabels::const_iterator i = labels.constBegin(); while (i != labels.constEnd()) { if (not i.value().isEmpty()) { QDomElement labelTag = createElement(TagLabel); SetAttribute(labelTag, AttrValue, i.key()); SetAttribute(labelTag, AttrLabel, i.value()); labelsTag.appendChild(labelTag); } ++i; } } else { if (not labelsTag.isNull()) { dElement.removeChild(labelsTag); } } } //--------------------------------------------------------------------------------------------------------------------- DimesionLabels VMeasurements::ReadDimensionLabels(const QDomElement &dElement) const { if (dElement.isNull()) { return DimesionLabels(); } QDomElement labelsTag = dElement.firstChildElement(TagLabels); if (labelsTag.isNull()) { return DimesionLabels(); } DimesionLabels labels; QDomNode labelTag = labelsTag.firstChild(); while (not labelTag.isNull()) { if (labelTag.isElement()) { const QDomElement l = labelTag.toElement(); const qreal value = GetParametrDouble(l, AttrValue, QChar('0')); const QString label = GetParametrEmptyString(l, AttrLabel); if (value > 0 && not label.isEmpty()) { labels.insert(value, label); } } labelTag = labelTag.nextSibling(); } return labels; } //--------------------------------------------------------------------------------------------------------------------- void VMeasurements::ClearDimension(IMD type) { const QString d = VMeasurements::IMDToStr(type); QDomNodeList list = elementsByTagName(TagMeasurement); for (int i=0; i < list.size(); ++i) { QDomElement domElement = list.at(i).toElement(); if (domElement.isNull() == false) { if (domElement.attribute(AttrDimension) == d) { domElement.removeAttribute(AttrDimension); } } } }