/************************************************************************ ** ** @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 #include "../vmisc/def.h" #include "../ifc/exception/vexceptionbadid.h" #include "../ifc/ifcdef.h" #include "../vgeometry/vabstractcubicbezierpath.h" #include "../vgeometry/vabstractcurve.h" #include "../vgeometry/vgobject.h" #include "../vmisc/diagnostic.h" #include "variables.h" #include "variables/vinternalvariable.h" #include "vpiece.h" #include "vpiecepath.h" #include "vtranslatevars.h" class VEllipticalArc; QT_WARNING_PUSH QT_WARNING_DISABLE_GCC("-Weffc++") QT_WARNING_DISABLE_INTEL(2021) QT_WARNING_DISABLE_GCC("-Wnon-virtual-dtor") class VContainerData : public QSharedData //-V690 { public: VContainerData(const VTranslateVars *trVars, const Unit *patternUnit, const QString &nspace) : calculationObjects(QHash >()), modelingObjects(QSharedPointer>>::create()), variables(QHash > ()), pieces(QSharedPointer>::create()), piecePaths(QSharedPointer>::create()), trVars(trVars), patternUnit(patternUnit), nspace(nspace) {} VContainerData(const VContainerData &data) : QSharedData(data), calculationObjects(data.calculationObjects), modelingObjects(data.modelingObjects), variables(data.variables), pieces(data.pieces), piecePaths(data.piecePaths), trVars(data.trVars), patternUnit(data.patternUnit), nspace(data.nspace) {} virtual ~VContainerData(); QHash > calculationObjects; QSharedPointer>> modelingObjects; /** * @brief variables container for measurements, increments, lines lengths, lines angles, arcs lengths, curve lengths */ QHash> variables; QSharedPointer> pieces; QSharedPointer> piecePaths; const VTranslateVars *trVars; const Unit *patternUnit; /** @brief nspace namespace for static variables */ QString nspace; private: Q_DISABLE_ASSIGN(VContainerData) }; QT_WARNING_POP /** * @brief The VContainer class container of all variables. */ class VContainer { Q_DECLARE_TR_FUNCTIONS(VContainer) public: VContainer(const VTranslateVars *trVars, const Unit *patternUnit, const QString &nspace); VContainer(const VContainer &data); ~VContainer(); friend class VContainerData; VContainer &operator=(const VContainer &data); #ifdef Q_COMPILER_RVALUE_REFS VContainer(const VContainer &&data) Q_DECL_NOTHROW; VContainer &operator=(VContainer &&data) Q_DECL_NOTHROW; #endif static QString UniqueNamespace(); template const QSharedPointer GeometricObject(const quint32 &id) const; const QSharedPointer GetGObject(quint32 id) const; static const QSharedPointer GetFakeGObject(quint32 id); VPiece GetPiece(quint32 id) const; VPiecePath GetPiecePath(quint32 id) const; quint32 GetPieceForPiecePath(quint32 id) const; template QSharedPointer GetVariable(const QString &name) const; quint32 getId() const; quint32 getNextId() const; void UpdateId(quint32 newId) const; static void UpdateId(quint32 newId, const QString &nspace); quint32 AddGObject(VGObject *obj); quint32 AddGObject(const QSharedPointer &obj); quint32 AddPiece(const VPiece &detail); quint32 AddPiecePath(const VPiecePath &path); 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); const QHash > *CalculationGObjects() const; const QHash *DataPieces() const; const QHash> *DataVariables() const; const QMap > DataMeasurements() const; const QMap > DataIncrements() const; const QMap > DataIncrementsWithSeparators() const; const QMap > DataLengthLines() const; const QMap > DataLengthCurves() const; const QMap > DataCurvesCLength() const; const QMap > DataAngleLines() const; const QMap > DataRadiusesArcs() const; const QMap > DataAnglesCurves() const; bool IsUnique(const QString &name) const; static bool IsUnique(const QString &name, const QString &nspace); QStringList AllUniqueNames() const; static QStringList AllUniqueNames(const QString &nspace); const Unit *GetPatternUnit() const; const VTranslateVars *GetTrVars() const; 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); template uint qHash( const QSharedPointer &p ); template void UpdateObject(const quint32 &id, const QSharedPointer &point); template const QMap > DataVar(const VarType &type) const; static void ClearNamespace(const QString &nspace); }; Q_DECLARE_TYPEINFO(VContainer, Q_MOVABLE_TYPE); /* * Defintion of templated member functions of VContainer */ //--------------------------------------------------------------------------------------------------------------------- template const QSharedPointer VContainer::GeometricObject(const quint32 &id) const { 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 QSharedPointer VContainer::GetVariable(const QString &name) const { 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(tr("Can't cast object"), name); } } else { throw VExceptionBadId(tr("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); } } //--------------------------------------------------------------------------------------------------------------------- template uint VContainer::qHash( const QSharedPointer &p ) { return qHash( p.data() ); } //--------------------------------------------------------------------------------------------------------------------- /** * @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); uniqueNames[d->nspace].insert(obj->name()); } //--------------------------------------------------------------------------------------------------------------------- /** * @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