/************************************************************************ ** ** @file vcontainer.h ** @author Roman Telezhynskyi ** @date November 15, 2013 ** ** @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) 2013-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 . ** *************************************************************************/ #ifndef VCONTAINER_H #define VCONTAINER_H #include #include #include #include #include #include #include #include #include #include #include #include #include #include "../ifc/exception/vexceptionbadid.h" #include "../vgeometry/vabstractcubicbezierpath.h" #include "../vgeometry/vabstractcurve.h" #include "../vgeometry/vgobject.h" #include "../vmisc/def.h" #include "variables/vinternalvariable.h" #include "vpiece.h" #include "vpiecepath.h" #include "vtranslatevars.h" class VEllipticalArc; class VMeasurement; class VIncrement; class VLengthLine; class VCurveLength; class VCurveCLength; class VLineAngle; class VArcRadius; class VCurveAngle; class VPieceArea; QT_WARNING_PUSH QT_WARNING_DISABLE_GCC("-Weffc++") QT_WARNING_DISABLE_INTEL(2021) QT_WARNING_DISABLE_GCC("-Wnon-virtual-dtor") class VContainerData final : public QSharedData //-V690 { public: VContainerData(const VTranslateVars *trVars, const Unit *patternUnit, const QString &nspace); VContainerData(const VContainerData &data) = default; virtual ~VContainerData(); // NOLINTNEXTLINE(misc-non-private-member-variables-in-classes) QHash> calculationObjects{}; // NOLINTNEXTLINE(misc-non-private-member-variables-in-classes) QSharedPointer>> modelingObjects{ QSharedPointer>>::create()}; /** * @brief variables container for measurements, increments, lines lengths, lines angles, arcs lengths, curve lengths */ // NOLINTNEXTLINE(misc-non-private-member-variables-in-classes) QHash> variables{}; // NOLINTNEXTLINE(misc-non-private-member-variables-in-classes) QSharedPointer> pieces{QSharedPointer>::create()}; // NOLINTNEXTLINE(misc-non-private-member-variables-in-classes) QSharedPointer> piecePaths{QSharedPointer>::create()}; const VTranslateVars *trVars; // NOLINT(misc-non-private-member-variables-in-classes) const Unit *patternUnit; // NOLINT(misc-non-private-member-variables-in-classes) /** @brief nspace namespace for static variables */ QString nspace; // NOLINT(misc-non-private-member-variables-in-classes) private: Q_DISABLE_ASSIGN_MOVE(VContainerData) // NOLINT }; QT_WARNING_POP /** * @brief The VContainer class container of all variables. */ class VContainer { Q_DECLARE_TR_FUNCTIONS(VContainer) // NOLINT public: VContainer(const VTranslateVars *trVars, const Unit *patternUnit, const QString &nspace); VContainer(const VContainer &data); ~VContainer(); friend class VContainerData; auto operator=(const VContainer &data) -> VContainer &; #ifdef Q_COMPILER_RVALUE_REFS VContainer(VContainer &&data) noexcept; auto operator=(VContainer &&data) noexcept -> VContainer &; #endif static auto UniqueNamespace() -> QString; template auto GeometricObject(const quint32 &id) const -> const QSharedPointer; auto GetGObject(quint32 id) const -> const QSharedPointer; static auto GetFakeGObject(quint32 id) -> const QSharedPointer; auto GetPiece(quint32 id) const -> VPiece; auto GetPiecePath(quint32 id) const -> VPiecePath; auto GetPieceForPiecePath(quint32 id) const -> quint32; template auto GetVariable(const QString &name) const -> QSharedPointer; auto getId() const -> quint32; auto getNextId() const -> quint32; void UpdateId(quint32 newId) const; static void UpdateId(quint32 newId, const QString &nspace); void RegisterUniqueName(VGObject *obj) const; void RegisterUniqueName(const QSharedPointer &obj) const; auto AddGObject(VGObject *obj) -> quint32; auto AddGObject(const QSharedPointer &obj) -> quint32; auto AddPiece(const VPiece &detail) -> quint32; auto AddPiecePath(const VPiecePath &path) -> quint32; void AddLine(const quint32 &firstPointId, const quint32 &secondPointId); void AddArc(const QSharedPointer &arc, const quint32 &id, const quint32 &parentId = NULL_ID); void AddSpline(const QSharedPointer &curve, quint32 id, quint32 parentId = NULL_ID); void AddCurveWithSegments(const QSharedPointer &curve, const quint32 &id, quint32 parentId = NULL_ID); template void AddUniqueVariable(T *var); template void AddUniqueVariable(const QSharedPointer &var); template void AddVariable(T *var); template void AddVariable(const QSharedPointer &var); void RemoveVariable(const QString &name); void RemovePiece(quint32 id); template void UpdateGObject(quint32 id, T *obj); template void UpdateGObject(quint32 id, const QSharedPointer &obj); void UpdatePiece(quint32 id, const VPiece &detail); void UpdatePiecePath(quint32 id, const VPiecePath &path); void Clear(); void ClearForFullParse(); void ClearGObjects(); void ClearCalculationGObjects(); void ClearVariables(const VarType &type = VarType::Unknown); void ClearVariables(const QVector &types); void ClearUniqueNames() const; void ClearUniqueIncrementNames() const; void ClearExceptUniqueIncrementNames() const; void RemoveIncrement(const QString &name); void FillPiecesAreas(Unit unit); auto CalculationGObjects() const -> const QHash> *; auto DataPieces() const -> const QHash *; auto DataVariables() const -> const QHash> *; auto DataMeasurements() const -> const QMap>; auto DataMeasurementsWithSeparators() const -> const QMap>; auto DataIncrements() const -> const QMap>; auto DataIncrementsWithSeparators() const -> const QMap>; auto DataLengthLines() const -> const QMap>; auto DataLengthCurves() const -> const QMap>; auto DataCurvesCLength() const -> const QMap>; auto DataAngleLines() const -> const QMap>; auto DataRadiusesArcs() const -> const QMap>; auto DataAnglesCurves() const -> const QMap>; auto DataPieceArea() const -> const QMap>; auto IsUnique(const QString &name) const -> bool; static auto IsUnique(const QString &name, const QString &nspace) -> bool; auto AllUniqueNames() const -> QStringList; static auto AllUniqueNames(const QString &nspace) -> QStringList; auto GetPatternUnit() const -> const Unit *; auto GetTrVars() const -> const VTranslateVars *; private: /** * @brief _id current id. New object will have value +1. For empty class equal 0. */ static QMap _id; static QMap> uniqueNames; static QMap copyCounter; QSharedDataPointer d; void AddCurve(const QSharedPointer &curve, const quint32 &id, quint32 parentId = NULL_ID); template void AddVariable(const QSharedPointer &var, const QString &name); #if QT_VERSION < QT_VERSION_CHECK(6, 0, 0) template auto qHash(const QSharedPointer &p, uint seed = 0) -> uint; #else template auto qHash(const QSharedPointer &p, size_t seed = 0) -> size_t; #endif template void UpdateObject(const quint32 &id, const QSharedPointer &point); template auto DataVar(const VarType &type) const -> const QMap>; static void ClearNamespace(const QString &nspace); }; Q_DECLARE_TYPEINFO(VContainer, Q_MOVABLE_TYPE); // NOLINT /* * Defintion of templated member functions of VContainer */ //--------------------------------------------------------------------------------------------------------------------- template auto VContainer::GeometricObject(const quint32 &id) const -> const QSharedPointer { if (id == NULL_ID) { throw VExceptionBadId(tr("Can't find object"), id); } QSharedPointer gObj; if (d->calculationObjects.contains(id)) { gObj = d->calculationObjects.value(id); } else if (d->modelingObjects->contains(id)) { gObj = d->modelingObjects->value(id); } else { throw VExceptionBadId(tr("Can't find object"), id); } QSharedPointer obj = qSharedPointerDynamicCast(gObj); SCASSERT(obj.isNull() == false) return obj; } //--------------------------------------------------------------------------------------------------------------------- /** * @brief GetVariable return varible by name * @param name variable's name * @return variable */ template auto VContainer::GetVariable(const QString &name) const -> QSharedPointer { SCASSERT(name.isEmpty() == false) if (d->variables.contains(name)) { try { QSharedPointer value = qSharedPointerDynamicCast(d->variables.value(name)); SCASSERT(value.isNull() == false) return value; } catch (const std::bad_alloc &) { throw VExceptionBadId(QCoreApplication::translate("VContainer", "Can't cast object"), name); } } else { throw VExceptionBadId(QCoreApplication::translate("VContainer", "Can't find object"), name); } } //--------------------------------------------------------------------------------------------------------------------- template void VContainer::AddUniqueVariable(T *var) { AddUniqueVariable(QSharedPointer(var)); } //--------------------------------------------------------------------------------------------------------------------- template void VContainer::AddUniqueVariable(const QSharedPointer &var) { AddVariable(var); if (d->variables.contains(var->GetName())) { uniqueNames[d->nspace].insert(var->GetName()); } } //--------------------------------------------------------------------------------------------------------------------- template void VContainer::AddVariable(T *var) { AddVariable(QSharedPointer(var)); } //--------------------------------------------------------------------------------------------------------------------- template void VContainer::AddVariable(const QSharedPointer &var) { SCASSERT(not var->GetName().isEmpty()) AddVariable(var, var->GetName()); if (not var->GetAlias().isEmpty()) { AddVariable(var, var->GetAlias()); } } //--------------------------------------------------------------------------------------------------------------------- template void VContainer::AddVariable(const QSharedPointer &var, const QString &name) { if (name.isEmpty()) { return; } if (d->variables.contains(name)) { if (d->variables.value(name)->GetType() == var->GetType()) { QSharedPointer v = qSharedPointerDynamicCast(d->variables.value(name)); if (v.isNull()) { throw VExceptionBadId(tr("Can't cast object."), name); } *v = *var; } else { throw VExceptionBadId(tr("Can't find object. Type mismatch."), name); } } else { d->variables.insert(name, var); } } //--------------------------------------------------------------------------------------------------------------------- #if QT_VERSION < QT_VERSION_CHECK(6, 0, 0) template auto VContainer::qHash(const QSharedPointer &p, uint seed) -> uint #else template auto VContainer::qHash(const QSharedPointer &p, size_t seed) -> size_t #endif { return qHash(p.data(), seed); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief UpdateGObject update GObject by id * @param id id of existing GObject * @param obj object */ template void VContainer::UpdateGObject(quint32 id, T *obj) { SCASSERT(obj != nullptr) UpdateGObject(id, QSharedPointer(obj)); } //--------------------------------------------------------------------------------------------------------------------- template void VContainer::UpdateGObject(quint32 id, const QSharedPointer &obj) { SCASSERT(not obj.isNull()) UpdateObject(id, obj); RegisterUniqueName(obj); } //--------------------------------------------------------------------------------------------------------------------- /** * @brief UpdateObject update object in container * @param id id of existing object * @param point object */ template void VContainer::UpdateObject(const quint32 &id, const QSharedPointer &point) { Q_ASSERT_X(id != NULL_ID, Q_FUNC_INFO, "id == 0"); //-V654 //-V712 SCASSERT(point.isNull() == false) point->setId(id); if (d->calculationObjects.contains(id) && point->getMode() == Draw::Calculation) { QSharedPointer obj = qSharedPointerDynamicCast(d->calculationObjects.value(id)); if (obj.isNull()) { throw VExceptionBadId(tr("Can't cast object"), id); } *obj = *point; } else if (d->modelingObjects->contains(id) && point->getMode() == Draw::Modeling) { QSharedPointer obj = qSharedPointerDynamicCast(d->modelingObjects->value(id)); if (obj.isNull()) { throw VExceptionBadId(tr("Can't cast object"), id); } *obj = *point; } else if (point->getMode() == Draw::Calculation) { d->calculationObjects.insert(id, point); } else if (point->getMode() == Draw::Modeling) { d->modelingObjects->insert(id, point); } else { qWarning("Can't update an object with mode 'Layout'"); return; } UpdateId(id); } #endif // VCONTAINER_H