X-Git-Url: http://git.salome-platform.org/gitweb/?a=blobdiff_plain;f=src%2FModuleBase%2FModuleBase_Tools.cpp;h=a96e15ff07179c97dffa4719e96d4fd52423d30e;hb=d588149317acbc2bd8e6ceb00f8e82e806571775;hp=10dd13c0e74c94ce30c0b9f2886baba93b7b268e;hpb=7578a7f30f735d48925d993d1e16216939b1e7ed;p=modules%2Fshaper.git diff --git a/src/ModuleBase/ModuleBase_Tools.cpp b/src/ModuleBase/ModuleBase_Tools.cpp old mode 100755 new mode 100644 index 10dd13c0e..a96e15ff0 --- a/src/ModuleBase/ModuleBase_Tools.cpp +++ b/src/ModuleBase/ModuleBase_Tools.cpp @@ -1,8 +1,21 @@ -// Copyright (C) 2014-20xx CEA/DEN, EDF R&D - -// File: ModuleBase_Tools.cpp -// Created: 11 July 2014 -// Author: Vitaly Smetannikov +// Copyright (C) 2014-2019 CEA/DEN, EDF R&D +// +// This library is free software; you can redistribute it and/or +// modify it under the terms of the GNU Lesser General Public +// License as published by the Free Software Foundation; either +// version 2.1 of the License, or (at your option) any later version. +// +// This library 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 +// Lesser General Public License for more details. +// +// You should have received a copy of the GNU Lesser General Public +// License along with this library; if not, write to the Free Software +// Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA +// +// See http://www.salome-platform.org/ or email : webmaster.salome@opencascade.com +// #include "ModuleBase_Tools.h" @@ -11,6 +24,10 @@ #include #include #include +#include +#include +#include +#include #include #include @@ -19,22 +36,40 @@ #include #include #include +#include +#include +#include +#include +#include #include #include #include -#include #include #include #include #include +#include + +#include +#include #include +#include +#include +#include #include -#include +#include +#include #include +#include + +#include +#include + +#include #include #include @@ -44,15 +79,31 @@ #include #include #include +#include +#include +#include +#include +#include #include #include +#ifdef WIN32 +#pragma warning(disable : 4996) // for getenv +#endif + const double tolerance = 1e-7; +const double DEFAULT_DEVIATION_COEFFICIENT = 1.e-4; //#define DEBUG_ACTIVATE_WINDOW //#define DEBUG_SET_FOCUS +#ifdef WIN32 +# define FSEP "\\" +#else +# define FSEP "/" +#endif + namespace ModuleBase_Tools { //****************************************************************** @@ -91,7 +142,10 @@ void zeroMargins(QLayout* theLayout) void activateWindow(QWidget* theWidget, const QString& theInfo) { - theWidget->activateWindow(); + if (theWidget) { + theWidget->activateWindow(); + theWidget->raise(); + } #ifdef DEBUG_ACTIVATE_WINDOW qDebug(QString("activateWindow: %1").arg(theInfo).toStdString().c_str()); @@ -100,8 +154,10 @@ void activateWindow(QWidget* theWidget, const QString& theInfo) void setFocus(QWidget* theWidget, const QString& theInfo) { + activateWindow(theWidget); theWidget->setFocus(); - + // rectangle of focus is not visible on tool button widgets + theWidget->update(); #ifdef DEBUG_SET_FOCUS qDebug(QString("setFocus: %1").arg(theInfo).toStdString().c_str()); #endif @@ -119,46 +175,50 @@ void setShadowEffect(QWidget* theWidget, const bool isSetEffect) else { QGraphicsEffect* anEffect = theWidget->graphicsEffect(); if(anEffect) - anEffect->deleteLater(); + anEffect->deleteLater(); theWidget->setGraphicsEffect(NULL); } } QPixmap composite(const QString& theAdditionalIcon, const QString& theIcon) { - QImage anIcon(theIcon); + QImage anIcon = ModuleBase_IconFactory::loadImage(theIcon); QImage anAditional(theAdditionalIcon); + return composite(anAditional, anIcon); +} - if (anIcon.isNull()) +QPixmap composite(const QImage& theAdditionalIcon, QImage& theIcon) +{ + if (theIcon.isNull()) return QPixmap(); - int anAddWidth = anAditional.width(); - int anAddHeight = anAditional.height(); + int anAddWidth = theAdditionalIcon.width(); + int anAddHeight = theAdditionalIcon.height(); - int aWidth = anIcon.width(); - int aHeight = anIcon.height(); + int aWidth = theIcon.width(); + int aHeight = theIcon.height(); - int aStartWidthPos = aWidth - anAddWidth - 1; - int aStartHeightPos = aHeight - anAddHeight - 1; + int aStartWidthPos = aWidth - anAddWidth; + int aStartHeightPos = aHeight - anAddHeight; for (int i = 0; i < anAddWidth && i + aStartWidthPos < aWidth; i++) { for (int j = 0; j < anAddHeight && j + aStartHeightPos < aHeight; j++) { - if (qAlpha(anAditional.pixel(i, j)) > 0) - anIcon.setPixel(i + aStartWidthPos, j + aStartHeightPos, anAditional.pixel(i, j)); + if (qAlpha(theAdditionalIcon.pixel(i, j)) > 0) + theIcon.setPixel(i + aStartWidthPos, j + aStartHeightPos, theAdditionalIcon.pixel(i, j)); } } - return QPixmap::fromImage(anIcon); + return QPixmap::fromImage(theIcon); } QPixmap lighter(const QString& theIcon, const int theLighterValue) { - QImage anIcon(theIcon); + QImage anIcon = ModuleBase_IconFactory::loadImage(theIcon); if (anIcon.isNull()) return QPixmap(); - QImage aResult(theIcon); + QImage aResult = ModuleBase_IconFactory::loadImage(theIcon); for (int i = 0; i < anIcon.width(); i++) { for (int j = 0; j < anIcon.height(); j++) @@ -177,7 +237,7 @@ QPixmap lighter(const QString& theIcon, const int theLighterValue) void setSpinText(ModuleBase_ParamSpinBox* theSpin, const QString& theText) { - if (theSpin->text() == theText) + if (theSpin->text() == theText) return; // In order to avoid extra text setting because it will // reset cursor position in control @@ -224,6 +284,30 @@ void setSpinValue(ModuleBase_ParamIntSpinBox* theSpin, int theValue) theSpin->blockSignals(isBlocked); } +QAction* createAction(const QIcon& theIcon, const QString& theText, + QObject* theParent, const QObject* theReceiver, + const char* theMember, const QString& theToolTip, + const QString& theStatusTip) +{ + QAction* anAction = new QAction(theIcon, theText, theParent); + anAction->setToolTip(theToolTip.isEmpty() ? theText : theToolTip); + anAction->setStatusTip(!theStatusTip.isEmpty() ? theStatusTip : + (!theToolTip.isEmpty() ? theToolTip : theText)); + if (theReceiver) + QObject::connect(anAction, SIGNAL(triggered(bool)), theReceiver, theMember); + + return anAction; +} + +#ifdef _DEBUG +QString objectName(const ObjectPtr& theObj) +{ + if (!theObj.get()) + return ""; + + return theObj->data()->name().c_str(); +} + QString objectInfo(const ObjectPtr& theObj, const bool isUseAttributesInfo) { QString aFeatureStr = "feature"; @@ -252,98 +336,148 @@ QString objectInfo(const ObjectPtr& theObj, const bool isUseAttributesInfo) if (aFeature.get()) { aFeatureStr.append(QString(": %1").arg(aFeature->getKind().c_str()).toStdString().c_str()); if (aFeature->data()->isValid()) { - aFeatureStr.append(QString(", name=%1").arg(aFeature->data()->name().c_str()).toStdString() + aFeatureStr.append(QString(", name=%1").arg(theObj->data()->name().c_str()).toStdString() .c_str()); } if (isUseAttributesInfo) { - std::list anAttrs = aFeature->data()->attributes(""); - std::list::const_iterator anIt = anAttrs.begin(), aLast = anAttrs.end(); - QStringList aValues; - for(; anIt != aLast; anIt++) { - AttributePtr anAttr = *anIt; - QString aValue = "not defined"; - std::string aType = anAttr->attributeType(); - if (aType == GeomDataAPI_Point2D::typeId()) { - std::shared_ptr aPoint = std::dynamic_pointer_cast( - anAttr); - if (aPoint.get()) - aValue = QString("(%1, %2)").arg(aPoint->x()).arg(aPoint->y()); - } - else if (aType == ModelAPI_AttributeRefAttr::typeId()) { - } - - aValues.push_back(QString("%1: %2").arg(anAttr->id().c_str()).arg(aValue).toStdString().c_str()); - } - if (!aValues.empty()) - aFeatureStr.append(QString(", attributes: %1").arg(aValues.join(", ").toStdString().c_str())); + std::set > anAttributes; + std::string aPointsInfo = ModelGeomAlgo_Point2D::getPontAttributesInfo(aFeature, + anAttributes).c_str(); + if (!aPointsInfo.empty()) + aFeatureStr.append(QString(", attributes: %1") + .arg(aPointsInfo.c_str()).toStdString().c_str()); } } return aFeatureStr; } +#endif -typedef QMap ShapeTypes; -static ShapeTypes MyShapeTypes; +typedef QMap ShapeTypes; +static ShapeTypes myShapeTypes; -TopAbs_ShapeEnum shapeType(const QString& theType) +int shapeType(const QString& theType) { - if (MyShapeTypes.count() == 0) { - MyShapeTypes["face"] = TopAbs_FACE; - MyShapeTypes["faces"] = TopAbs_FACE; - MyShapeTypes["vertex"] = TopAbs_VERTEX; - MyShapeTypes["vertices"] = TopAbs_VERTEX; - MyShapeTypes["wire"] = TopAbs_WIRE; - MyShapeTypes["edge"] = TopAbs_EDGE; - MyShapeTypes["edges"] = TopAbs_EDGE; - MyShapeTypes["shell"] = TopAbs_SHELL; - MyShapeTypes["solid"] = TopAbs_SOLID; - MyShapeTypes["solids"] = TopAbs_SOLID; - MyShapeTypes["objects"] = TopAbs_SHAPE; + if (myShapeTypes.count() == 0) { + myShapeTypes["compound"] = TopAbs_COMPOUND; + myShapeTypes["compounds"] = TopAbs_COMPOUND; + myShapeTypes["compsolid"] = TopAbs_COMPSOLID; + myShapeTypes["compsolids"] = TopAbs_COMPSOLID; + myShapeTypes["solid"] = TopAbs_SOLID; + myShapeTypes["solids"] = TopAbs_SOLID; + myShapeTypes["shell"] = TopAbs_SHELL; + myShapeTypes["shells"] = TopAbs_SHELL; + myShapeTypes["face"] = TopAbs_FACE; + myShapeTypes["faces"] = TopAbs_FACE; + myShapeTypes["wire"] = TopAbs_WIRE; + myShapeTypes["wires"] = TopAbs_WIRE; + myShapeTypes["edge"] = TopAbs_EDGE; + myShapeTypes["edges"] = TopAbs_EDGE; + myShapeTypes["vertex"] = TopAbs_VERTEX; + myShapeTypes["vertices"] = TopAbs_VERTEX; + myShapeTypes["object"] = ModuleBase_ResultPrs::Sel_Result; + myShapeTypes["objects"] = ModuleBase_ResultPrs::Sel_Result; } QString aType = theType.toLower(); - if (MyShapeTypes.contains(aType)) - return MyShapeTypes[aType]; - Events_Error::send("Shape type defined in XML is not implemented!"); + if(myShapeTypes.contains(aType)) + return myShapeTypes[aType]; + Events_InfoMessage("ModuleBase_Tools", "Shape type defined in XML is not implemented!").send(); return TopAbs_SHAPE; } void checkObjects(const QObjectPtrList& theObjects, bool& hasResult, bool& hasFeature, - bool& hasParameter, bool& hasCompositeOwner) + bool& hasParameter, bool& hasCompositeOwner, bool& hasResultInHistory, + bool& hasFolder) { hasResult = false; hasFeature = false; hasParameter = false; hasCompositeOwner = false; + hasResultInHistory = false; + hasFolder = false; foreach(ObjectPtr aObj, theObjects) { FeaturePtr aFeature = std::dynamic_pointer_cast(aObj); ResultPtr aResult = std::dynamic_pointer_cast(aObj); + FolderPtr aFolder = std::dynamic_pointer_cast(aObj); ResultParameterPtr aConstruction = std::dynamic_pointer_cast(aResult); + FieldStepPtr aStep = std::dynamic_pointer_cast(aObj); - hasResult |= (aResult.get() != NULL); + hasResult |= ((aResult.get() != NULL) || (aStep.get() != NULL)); hasFeature |= (aFeature.get() != NULL); + hasFolder |= (aFolder.get() != NULL); hasParameter |= (aConstruction.get() != NULL); - if (hasFeature) + if (hasFeature) hasCompositeOwner |= (ModelAPI_Tools::compositeOwner(aFeature) != NULL); + else if (aResult.get()) + hasCompositeOwner |= (ModelAPI_Tools::bodyOwner(aResult) != NULL); + + if (!hasResultInHistory && aResult.get()) { + FeaturePtr aFeature = ModelAPI_Feature::feature(aResult); + hasResultInHistory = aFeature.get() && aFeature->isInHistory(); + } + if (hasFeature && hasResult && hasParameter && hasCompositeOwner) break; } } +/*bool setDefaultDeviationCoefficient(std::shared_ptr theGeomShape) +{ + if (!theGeomShape.get()) + return false; + // if the shape could not be exploded on faces, it contains only wires, edges, and vertices + // correction of deviation for them should not influence to the application performance + GeomAPI_ShapeExplorer anExp(theGeomShape, GeomAPI_Shape::FACE); + bool anEmpty = anExp.empty(); + return !anExp.more(); +}*/ + +/*void setDefaultDeviationCoefficient(const std::shared_ptr& theResult, + const Handle(Prs3d_Drawer)& theDrawer) +{ + if (!theResult.get()) + return; + bool aUseDeviation = false; + + std::string aResultGroup = theResult->groupName(); + if (aResultGroup == ModelAPI_ResultConstruction::group()) + aUseDeviation = true; + else if (aResultGroup == ModelAPI_ResultBody::group()) { + GeomShapePtr aGeomShape = theResult->shape(); + if (aGeomShape.get()) + aUseDeviation = setDefaultDeviationCoefficient(aGeomShape); + } + if (aUseDeviation) + theDrawer->SetDeviationCoefficient(DEFAULT_DEVIATION_COEFFICIENT); +} +*/ void setDefaultDeviationCoefficient(const TopoDS_Shape& theShape, const Handle(Prs3d_Drawer)& theDrawer) { if (theShape.IsNull()) return; - TopAbs_ShapeEnum aType = theShape.ShapeType(); - if ((aType == TopAbs_EDGE) || (aType == TopAbs_WIRE)) - theDrawer->SetDeviationCoefficient(1.e-4); + + std::shared_ptr aGeomShape(new GeomAPI_Shape()); + aGeomShape->setImpl(new TopoDS_Shape(theShape)); + + // if the shape could not be exploded on faces, it contains only wires, edges, and vertices + // correction of deviation for them should not influence to the application performance + GeomAPI_ShapeExplorer anExp(aGeomShape, GeomAPI_Shape::FACE); + bool isConstruction = !anExp.more(); + + double aDeflection; + if (isConstruction) + aDeflection = Config_PropManager::real("Visualization", "construction_deflection"); + else + aDeflection = Config_PropManager::real("Visualization", "body_deflection"); + + theDrawer->SetDeviationCoefficient(aDeflection); } Quantity_Color color(const std::string& theSection, - const std::string& theName, - const std::string& theDefault) + const std::string& theName) { - std::vector aColor = Config_PropManager::color(theSection, theName, theDefault); + std::vector aColor = Config_PropManager::color(theSection, theName); return Quantity_Color(aColor[0] / 255., aColor[1] / 255., aColor[2] / 255., Quantity_TOC_RGB); } @@ -352,17 +486,20 @@ ObjectPtr getObject(const AttributePtr& theAttribute) ObjectPtr anObject; std::string anAttrType = theAttribute->attributeType(); if (anAttrType == ModelAPI_AttributeRefAttr::typeId()) { - AttributeRefAttrPtr anAttr = std::dynamic_pointer_cast(theAttribute); + AttributeRefAttrPtr anAttr = + std::dynamic_pointer_cast(theAttribute); if (anAttr != NULL && anAttr->isObject()) anObject = anAttr->object(); } if (anAttrType == ModelAPI_AttributeSelection::typeId()) { - AttributeSelectionPtr anAttr = std::dynamic_pointer_cast(theAttribute); + AttributeSelectionPtr anAttr = + std::dynamic_pointer_cast(theAttribute); if (anAttr != NULL) anObject = anAttr->context(); } if (anAttrType == ModelAPI_AttributeReference::typeId()) { - AttributeReferencePtr anAttr = std::dynamic_pointer_cast(theAttribute); + AttributeReferencePtr anAttr = + std::dynamic_pointer_cast(theAttribute); if (anAttr.get() != NULL) anObject = anAttr->value(); } @@ -395,6 +532,18 @@ TopAbs_ShapeEnum getCompoundSubType(const TopoDS_Shape& theShape) return aShapeType; } +TopoDS_Shape getSelectedShape(const std::shared_ptr& thePrs) +{ + if (thePrs->shape().get()) + return thePrs->shape()->impl(); + + Handle(StdSelect_BRepOwner) anOwner = Handle(StdSelect_BRepOwner)::DownCast(thePrs->owner()); + if (!anOwner.IsNull()) + return anOwner->Shape(); + + return TopoDS_Shape(); +} + void getParameters(QStringList& theParameters) { theParameters.clear(); @@ -419,7 +568,8 @@ void getParameters(QStringList& theParameters) } } -std::string findGreedAttribute(ModuleBase_IWorkshop* theWorkshop, const FeaturePtr& theFeature) +std::string findGreedAttribute(ModuleBase_IWorkshop* theWorkshop, + const FeaturePtr& theFeature) { std::string anAttributeId; @@ -433,22 +583,98 @@ std::string findGreedAttribute(ModuleBase_IWorkshop* theWorkshop, const FeatureP return anAttributeId; } -void setObject(const AttributePtr& theAttribute, const ObjectPtr& theObject, - const GeomShapePtr& theShape, ModuleBase_IWorkshop* theWorkshop, +bool hasObject(const AttributePtr& theAttribute, const ObjectPtr& theObject, + const std::shared_ptr& theShape, + ModuleBase_IWorkshop* theWorkshop, const bool theTemporarily) { + bool aHasObject = false; if (!theAttribute.get()) - return; + return aHasObject; + + std::string aType = theAttribute->attributeType(); + if (aType == ModelAPI_AttributeReference::typeId()) { + AttributeReferencePtr aRef = + std::dynamic_pointer_cast(theAttribute); + ObjectPtr aObject = aRef->value(); + aHasObject = aObject && aObject->isSame(theObject); + //if (!(aObject && aObject->isSame(theObject))) { + // aRef->setValue(theObject); + //} + } else if (aType == ModelAPI_AttributeRefAttr::typeId()) { + AttributeRefAttrPtr aRefAttr = + std::dynamic_pointer_cast(theAttribute); + + AttributePtr anAttribute = theWorkshop->module()->findAttribute(theObject, theShape); + if (anAttribute.get()) { + //aRefAttr->setAttr(anAttribute); + } + else { + ObjectPtr aObject = aRefAttr->object(); + aHasObject = aObject && aObject->isSame(theObject); + //if (!(aObject && aObject->isSame(theObject))) { + // aRefAttr->setObject(theObject); + //} + } + } else if (aType == ModelAPI_AttributeSelection::typeId()) { + /*AttributeSelectionPtr aSelectAttr = + std::dynamic_pointer_cast(theAttribute); + ResultPtr aResult = std::dynamic_pointer_cast(theObject); + if (aSelectAttr.get() != NULL) { + aSelectAttr->setValue(aResult, theShape, theTemporarily); + }*/ + } + if (aType == ModelAPI_AttributeSelectionList::typeId()) { + AttributeSelectionListPtr aSelectionListAttr = + std::dynamic_pointer_cast(theAttribute); + ResultPtr aResult = std::dynamic_pointer_cast(theObject); + aHasObject = aSelectionListAttr->isInList(aResult, theShape, theTemporarily); + } + else if (aType == ModelAPI_AttributeRefList::typeId()) { + AttributeRefListPtr aRefListAttr = + std::dynamic_pointer_cast(theAttribute); + aHasObject = aRefListAttr->isInList(theObject); + //if (!theCheckIfAttributeHasObject || !aRefListAttr->isInList(theObject)) + // aRefListAttr->append(theObject); + } + else if (aType == ModelAPI_AttributeRefAttrList::typeId()) { + AttributeRefAttrListPtr aRefAttrListAttr = + std::dynamic_pointer_cast(theAttribute); + AttributePtr anAttribute = theWorkshop->module()->findAttribute(theObject, theShape); + if (anAttribute.get()) { + aHasObject = aRefAttrListAttr->isInList(anAttribute); + //if (!theCheckIfAttributeHasObject || !aRefAttrListAttr->isInList(anAttribute)) + // aRefAttrListAttr->append(anAttribute); + } + else { + aHasObject = aRefAttrListAttr->isInList(theObject); + //if (!theCheckIfAttributeHasObject || !aRefAttrListAttr->isInList(theObject)) + // aRefAttrListAttr->append(theObject); + } + } + return aHasObject; +} + +bool setObject(const AttributePtr& theAttribute, const ObjectPtr& theObject, + const GeomShapePtr& theShape, ModuleBase_IWorkshop* theWorkshop, + const bool theTemporarily, const bool theCheckIfAttributeHasObject) +{ + if (!theAttribute.get()) + return false; + + bool isDone = true; std::string aType = theAttribute->attributeType(); if (aType == ModelAPI_AttributeReference::typeId()) { - AttributeReferencePtr aRef = std::dynamic_pointer_cast(theAttribute); + AttributeReferencePtr aRef = + std::dynamic_pointer_cast(theAttribute); ObjectPtr aObject = aRef->value(); if (!(aObject && aObject->isSame(theObject))) { aRef->setValue(theObject); } } else if (aType == ModelAPI_AttributeRefAttr::typeId()) { - AttributeRefAttrPtr aRefAttr = std::dynamic_pointer_cast(theAttribute); + AttributeRefAttrPtr aRefAttr = + std::dynamic_pointer_cast(theAttribute); AttributePtr anAttribute = theWorkshop->module()->findAttribute(theObject, theShape); if (anAttribute.get()) @@ -462,36 +688,46 @@ void setObject(const AttributePtr& theAttribute, const ObjectPtr& theObject, } else if (aType == ModelAPI_AttributeSelection::typeId()) { AttributeSelectionPtr aSelectAttr = std::dynamic_pointer_cast(theAttribute); - ResultPtr aResult = std::dynamic_pointer_cast(theObject); if (aSelectAttr.get() != NULL) { - aSelectAttr->setValue(aResult, theShape, theTemporarily); + aSelectAttr->setValue(theObject, theShape, theTemporarily); } } if (aType == ModelAPI_AttributeSelectionList::typeId()) { AttributeSelectionListPtr aSelectionListAttr = std::dynamic_pointer_cast(theAttribute); - ResultPtr aResult = std::dynamic_pointer_cast(theObject); - if (!aSelectionListAttr->isInList(aResult, theShape, theTemporarily)) - aSelectionListAttr->append(aResult, theShape, theTemporarily); + if (!theCheckIfAttributeHasObject || + !aSelectionListAttr->isInList(theObject, theShape, theTemporarily)) + aSelectionListAttr->append(theObject, theShape, theTemporarily); } else if (aType == ModelAPI_AttributeRefList::typeId()) { - AttributeRefListPtr aRefListAttr = std::dynamic_pointer_cast(theAttribute); - if (!aRefListAttr->isInList(theObject)) - aRefListAttr->append(theObject); + AttributeRefListPtr aRefListAttr = + std::dynamic_pointer_cast(theAttribute); + if (!theCheckIfAttributeHasObject || !aRefListAttr->isInList(theObject)) { + if (theObject.get()) + aRefListAttr->append(theObject); + else + isDone = false; + } } else if (aType == ModelAPI_AttributeRefAttrList::typeId()) { - AttributeRefAttrListPtr aRefAttrListAttr = std::dynamic_pointer_cast(theAttribute); + AttributeRefAttrListPtr aRefAttrListAttr = + std::dynamic_pointer_cast(theAttribute); AttributePtr anAttribute = theWorkshop->module()->findAttribute(theObject, theShape); if (anAttribute.get()) { - if (!aRefAttrListAttr->isInList(anAttribute)) + if (!theCheckIfAttributeHasObject || !aRefAttrListAttr->isInList(anAttribute)) aRefAttrListAttr->append(anAttribute); } else { - if (!aRefAttrListAttr->isInList(theObject)) - aRefAttrListAttr->append(theObject); + if (!theCheckIfAttributeHasObject || !aRefAttrListAttr->isInList(theObject)) { + if (theObject.get()) + aRefAttrListAttr->append(theObject); + else + isDone = false; + } } } + return isDone; } GeomShapePtr getShape(const AttributePtr& theAttribute, ModuleBase_IWorkshop* theWorkshop) @@ -503,7 +739,8 @@ GeomShapePtr getShape(const AttributePtr& theAttribute, ModuleBase_IWorkshop* th std::string aType = theAttribute->attributeType(); if (aType == ModelAPI_AttributeReference::typeId()) { } else if (aType == ModelAPI_AttributeRefAttr::typeId()) { - AttributeRefAttrPtr aRefAttr = std::dynamic_pointer_cast(theAttribute); + AttributeRefAttrPtr aRefAttr = + std::dynamic_pointer_cast(theAttribute); if (aRefAttr.get() && !aRefAttr->isObject()) { AttributePtr anAttribute = aRefAttr->attr(); aShape = theWorkshop->module()->findShape(anAttribute); @@ -513,6 +750,8 @@ GeomShapePtr getShape(const AttributePtr& theAttribute, ModuleBase_IWorkshop* th (theAttribute); aShape = aSelectAttr->value(); } + else // Geom2D point processing + aShape = theWorkshop->module()->findShape(theAttribute); return aShape; } @@ -520,7 +759,13 @@ void flushUpdated(ObjectPtr theObject) { blockUpdateViewer(true); + // Fix the problem of not previewed results of constraints applied. Flush Create/Delete + // (for the sketch result) to start processing of the sketch in the solver. + // TODO: these flushes should be moved in a separate method provided by Model + Events_Loop::loop()->flush(Events_Loop::eventByName(EVENT_OBJECT_CREATED)); + Events_Loop::loop()->flush(Events_Loop::eventByName(EVENT_VISUAL_ATTRIBUTES)); Events_Loop::loop()->flush(Events_Loop::eventByName(EVENT_OBJECT_UPDATED)); + Events_Loop::loop()->flush(Events_Loop::eventByName(EVENT_OBJECT_DELETED)); blockUpdateViewer(false); } @@ -562,7 +807,7 @@ QString wrapTextByWords(const QString& theValue, QWidget* theWidget, int n = aWords.count(); QString aLine; for (int i = 0; i < n; i++) { - QString aLineExt = aLine + " " + aWords[i]; + QString aLineExt = i == 0 ? aWords[i] : aLine + " " + aWords[i]; qreal anWidthNonExt = tfm.boundingRect(aLine).width(); qreal anWidthExt = tfm.boundingRect(aLineExt).width(); qreal aDeltaNonExt = fabs(anWidthNonExt-aGoldWidth); @@ -583,6 +828,512 @@ QString wrapTextByWords(const QString& theValue, QWidget* theWidget, return aResult; } +//************************************************************** +QLocale doubleLocale() +{ + // VSR 01/07/2010: Disable thousands separator for spin box + // (to avoid inconsistency of double-2-string and string-2-double conversion) + QLocale aLocale; + aLocale.setNumberOptions(aLocale.numberOptions() | + QLocale::OmitGroupSeparator | + QLocale::RejectGroupSeparator); + return aLocale; +} + +//************************************************************** +void refsToFeatureInFeatureDocument(const ObjectPtr& theObject, + std::set& theRefFeatures) +{ + FeaturePtr aFeature = ModelAPI_Feature::feature(theObject); + if (aFeature.get()) { + DocumentPtr aFeatureDoc = aFeature->document(); + // 1. find references in the current document + aFeatureDoc->refsToFeature(aFeature, theRefFeatures, false); + } +} + + +//************************************************************** +/*bool isSubOfComposite(const ObjectPtr& theObject) +{ + bool isSub = false; + std::set aRefFeatures; + refsToFeatureInFeatureDocument(theObject, aRefFeatures); + std::set::const_iterator anIt = aRefFeatures.begin(), + aLast = aRefFeatures.end(); + for (; anIt != aLast && !isSub; anIt++) { + isSub = isSubOfComposite(theObject, *anIt); + } + return isSub; +}*/ + +//************************************************************** +/*bool isSubOfComposite(const ObjectPtr& theObject, const FeaturePtr& theFeature) +{ + bool isSub = false; + CompositeFeaturePtr aComposite = std::dynamic_pointer_cast(theFeature); + if (aComposite.get()) { + isSub = aComposite->isSub(theObject); + // the recursive is possible, the parameters are sketch circle and extrusion cut. They are + // separated by composite sketch feature + if (!isSub) { + int aNbSubs = aComposite->numberOfSubs(); + for (int aSub = 0; aSub < aNbSubs && !isSub; aSub++) { + isSub = isSubOfComposite(theObject, aComposite->subFeature(aSub)); + } + } + } + return isSub; +}*/ + +//************************************************************** +ResultPtr firstResult(const ObjectPtr& theObject) +{ + ResultPtr aResult = std::dynamic_pointer_cast(theObject); + if (!aResult.get()) { + FeaturePtr aFeature = std::dynamic_pointer_cast(theObject); + if (aFeature.get()) + aResult = aFeature->firstResult(); + } + return aResult; +} + +//************************************************************** +bool isFeatureOfResult(const FeaturePtr& theFeature, const std::string& theGroupOfResult) +{ + bool isResult = false; + + if (!theFeature->data()->isValid()) + return isResult; + + ResultPtr aFirstResult = theFeature->firstResult(); + if (!aFirstResult.get()) + return isResult; + + return aFirstResult->groupName() == theGroupOfResult; +} + +//************************************************************** +bool hasModuleDocumentFeature(const std::set& theFeatures) +{ + bool aFoundModuleDocumentObject = false; + DocumentPtr aModuleDoc = ModelAPI_Session::get()->moduleDocument(); + + std::set::const_iterator anIt = theFeatures.begin(), aLast = theFeatures.end(); + for (; anIt != aLast && !aFoundModuleDocumentObject; anIt++) { + FeaturePtr aFeature = *anIt; + ResultPtr aResult = ModuleBase_Tools::firstResult(aFeature); + if (aResult.get() && aResult->groupName() == ModelAPI_ResultPart::group()) + continue; + aFoundModuleDocumentObject = aFeature->document() == aModuleDoc; + } + + return aFoundModuleDocumentObject; +} + +//************************************************************** +bool askToDelete(const std::set theFeatures, + const std::map >& theReferences, + QWidget* theParent, + std::set& theReferencesToDelete, + const std::string& thePrefixInfo) +{ + QString aNotActivatedDocWrn; + std::string aNotActivatedNames; + if (!ModelAPI_Tools::allDocumentsActivated(aNotActivatedNames)) { + if (ModuleBase_Tools::hasModuleDocumentFeature(theFeatures)) + aNotActivatedDocWrn = + QObject::tr("Selected objects can be used in Part documents which are not loaded: %1.\n") + .arg(aNotActivatedNames.c_str()); + } + + std::set aFeaturesRefsTo; + std::set aFeaturesRefsToParameter; + std::set aParameterFeatures; + QStringList aPartFeatureNames; + std::set::const_iterator anIt = theFeatures.begin(), + aLast = theFeatures.end(); + // separate features to references to parameter features and references to others + for (; anIt != aLast; anIt++) { + FeaturePtr aFeature = *anIt; + if (theReferences.find(aFeature) == theReferences.end()) + continue; + + if (isFeatureOfResult(aFeature, ModelAPI_ResultPart::group())) + aPartFeatureNames.append(aFeature->name().c_str()); + + std::set aRefFeatures; + std::set aRefList = theReferences.at(aFeature); + std::set::const_iterator aRefIt = aRefList.begin(), aRefLast = aRefList.end(); + for (; aRefIt != aRefLast; aRefIt++) { + FeaturePtr aRefFeature = *aRefIt; + if (theFeatures.find(aRefFeature) == theFeatures.end() && // it is not selected + aRefFeatures.find(aRefFeature) == aRefFeatures.end()) // it is not added + aRefFeatures.insert(aRefFeature); + } + + if (isFeatureOfResult(aFeature, ModelAPI_ResultParameter::group())) { + aFeaturesRefsToParameter.insert(aRefFeatures.begin(), aRefFeatures.end()); + aParameterFeatures.insert(aFeature); + } + else { + theReferencesToDelete.insert(aRefFeatures.begin(), aRefFeatures.end()); + } + } + + std::set aFeaturesRefsToParameterOnly; + anIt = aFeaturesRefsToParameter.begin(); + aLast = aFeaturesRefsToParameter.end(); + // separate features to references to parameter features and references to others + QStringList aParamFeatureNames; + for (; anIt != aLast; anIt++) { + FeaturePtr aFeature = *anIt; + if (theReferencesToDelete.find(aFeature) == theReferencesToDelete.end()) { + aFeaturesRefsToParameterOnly.insert(aFeature); + aParamFeatureNames.append(aFeature->name().c_str()); + } + } + aParamFeatureNames.sort(); + QStringList anOtherFeatureNames; + anIt = theReferencesToDelete.begin(); + aLast = theReferencesToDelete.end(); + for (; anIt != aLast; anIt++) { + FeaturePtr aFeature = *anIt; + if (aFeature->getKind() == "RemoveResults") + continue; // skip the remove results feature mentioning: result will be removed anyway + if (isFeatureOfResult(aFeature, ModelAPI_ResultPart::group())) + aPartFeatureNames.append(aFeature->name().c_str()); + else + anOtherFeatureNames.append(aFeature->name().c_str()); + } + aPartFeatureNames.sort(); + anOtherFeatureNames.sort(); + + bool aCanReplaceParameters = !aFeaturesRefsToParameterOnly.empty(); + + QMessageBox aMessageBox(theParent); + aMessageBox.setWindowTitle(QObject::tr("Delete features")); + aMessageBox.setIcon(QMessageBox::Warning); + aMessageBox.setStandardButtons(QMessageBox::No | QMessageBox::Yes); + aMessageBox.setDefaultButton(QMessageBox::No); + + QString aText; + if (!thePrefixInfo.empty()) + aText = thePrefixInfo.c_str(); + QString aSep = ", "; + if (!aPartFeatureNames.empty()) { + aText += QString(QObject::tr("The following parts will be deleted: %1.\n")) + .arg(aPartFeatureNames.join(aSep)); + } + if (!aNotActivatedDocWrn.isEmpty()) + aText += aNotActivatedDocWrn; + if (!anOtherFeatureNames.empty()) { + const char* aMsg = "Features are used in the following features: %1.\nThese " + "features will be deleted.\n"; + aText += QString(QObject::tr(aMsg)) + .arg(anOtherFeatureNames.join(aSep)); + } + if (!aParamFeatureNames.empty()) { + const char* aMsg = "Parameters are used directly and through a sequence " + "of dependencies in the following features: %1.\nThese features will " + "be deleted.\nOr parameters could be replaced by their values.\n"; + aText += QString(QObject::tr(aMsg)) + .arg(aParamFeatureNames.join(aSep)); + QPushButton *aReplaceButton = + aMessageBox.addButton(QObject::tr("Replace"), QMessageBox::ActionRole); + } + + if (!aText.isEmpty()) { + aText += "Would you like to continue?"; + aMessageBox.setText(aText); + aMessageBox.exec(); + QMessageBox::ButtonRole aButtonRole = aMessageBox.buttonRole(aMessageBox.clickedButton()); + + if (aButtonRole == QMessageBox::NoRole) + return false; + + if (aButtonRole == QMessageBox::ActionRole) { + foreach (FeaturePtr aObj, aParameterFeatures) + ModelAPI_ReplaceParameterMessage::send(aObj, 0); + } + else + theReferencesToDelete.insert(aFeaturesRefsToParameterOnly.begin(), + aFeaturesRefsToParameterOnly.end()); + } + return true; +} + +//************************************************************** +void convertToFeatures(const QObjectPtrList& theObjects, std::set& theFeatures) +{ + QObjectPtrList::const_iterator anIt = theObjects.begin(), aLast = theObjects.end(); + for(; anIt != aLast; anIt++) { + ObjectPtr anObject = *anIt; + FeaturePtr aFeature = std::dynamic_pointer_cast(anObject); + // for parameter result, use the corresponded reature to be removed + if (!aFeature.get() && anObject->groupName() == ModelAPI_ResultParameter::group()) { + aFeature = ModelAPI_Feature::feature(anObject); + } + if (aFeature.get()) + theFeatures.insert(aFeature); + } +} + +//************************************************************** +void convertToFolders(const QObjectPtrList& theObjects, + std::set& theFolders) +{ + QObjectPtrList::const_iterator anIt = theObjects.begin(), aLast = theObjects.end(); + for(; anIt != aLast; anIt++) { + ObjectPtr anObject = *anIt; + FolderPtr aFeature = std::dynamic_pointer_cast(anObject); + if (aFeature.get()) + theFolders.insert(aFeature); + } +} + + +//************************************************************** +QString translate(const Events_InfoMessage& theMessage) +{ + QString aMessage; + + if (!theMessage.empty()) { + std::string aStr = Config_Translator::translate(theMessage); + if (!aStr.empty()) { + std::string aCodec = Config_Translator::codec(theMessage); + aMessage = QTextCodec::codecForName(aCodec.c_str())->toUnicode(aStr.c_str()); + } + } + + return aMessage; +} + +QString translate(const std::string& theContext, const std::string& theMessage) +{ + QString aMessage; + + if (!theMessage.empty()) { + std::string aStr = Config_Translator::translate(theContext, theMessage); + if (!aStr.empty()) { + std::string aCodec = Config_Translator::codec(theContext); + aMessage = QTextCodec::codecForName(aCodec.c_str())->toUnicode(aStr.c_str()); + } + } + + return aMessage; +} + +void setPointBallHighlighting(AIS_Shape* theAIS) +{ + static Handle(Image_AlienPixMap) aPixMap; + if(aPixMap.IsNull()) { + // Load icon for the presentation + std::string aFile; + char* anEnv = getenv("SHAPER_ROOT_DIR"); + if(anEnv) { + aFile = std::string(anEnv) + + FSEP + "share" + FSEP + "salome" + FSEP + "resources" + FSEP + "shaper"; + } else { + anEnv = getenv("CADBUILDER_ROOT_DIR"); + if (anEnv) + aFile = std::string(anEnv) + FSEP + "resources"; + } + + aFile += FSEP; + static const std::string aMarkerName = "marker_dot.png"; + aFile += aMarkerName; + aPixMap = new Image_AlienPixMap(); + if(!aPixMap->Load(aFile.c_str())) { + // The icon for constraint is not found + static const std::string aMsg = + "Error: Point market not found by path: \"" + aFile + "\". Falling back."; + //Events_InfoMessage("ModuleBase_Tools::setPointBallHighlighting", aMsg).send(); + } + } + + Handle(Graphic3d_AspectMarker3d) anAspect; + Handle(Prs3d_Drawer) aDrawer = theAIS->DynamicHilightAttributes(); + if (aDrawer.IsNull()) { + if (ModuleBase_IViewer::DefaultHighlightDrawer.IsNull()) + return; + aDrawer = new Prs3d_Drawer(*ModuleBase_IViewer::DefaultHighlightDrawer); + if (!aDrawer->HasOwnPointAspect()) { + aDrawer->SetPointAspect(new Prs3d_PointAspect(Aspect_TOM_BALL, Quantity_NOC_BLACK, 2.0)); + } + } + if(aDrawer->HasOwnPointAspect()) { + Handle(Prs3d_PointAspect) aPntAspect = aDrawer->PointAspect(); + if(aPixMap->IsEmpty()) { + anAspect = aPntAspect->Aspect(); + anAspect->SetType(Aspect_TOM_BALL); + } else { + if(aPixMap->Format() == Image_PixMap::ImgGray) { + aPixMap->SetFormat (Image_PixMap::ImgAlpha); + } else if(aPixMap->Format() == Image_PixMap::ImgGrayF) { + aPixMap->SetFormat (Image_PixMap::ImgAlphaF); + } + anAspect = new Graphic3d_AspectMarker3d(aPixMap); + aPntAspect->SetAspect(anAspect); + } + aDrawer->SetPointAspect(aPntAspect); + theAIS->SetDynamicHilightAttributes(aDrawer); + } +} + +FeaturePtr createParameter(const QString& theText) +{ + FeaturePtr aParameter; + QStringList aList = theText.split("="); + if (aList.count() != 2) { + return aParameter; + } + QString aParamName = aList.at(0).trimmed(); + + if (isNameExist(aParamName, FeaturePtr())) { + return aParameter; + } + + if (!ModelAPI_Expression::isVariable(aParamName.toStdString())) { + return aParameter; + } + + QString aExpression = aList.at(1).trimmed(); + if (aExpression.isEmpty()) { + return aParameter; + } + + SessionPtr aMgr = ModelAPI_Session::get(); + std::shared_ptr aDoc = aMgr->activeDocument(); + + aParameter = aDoc->addFeature("Parameter"); + if (aParameter.get()) { + AttributeStringPtr aNameAttr = aParameter->string("variable"); + aNameAttr->setValue(aParamName.toStdString()); + + AttributeStringPtr aExprAttr = aParameter->string("expression"); + aExprAttr->setValue(aExpression.toStdString()); + aParameter->execute(); + + Events_Loop::loop()->flush(Events_Loop::eventByName(EVENT_OBJECT_CREATED)); + Events_Loop::loop()->flush(Events_Loop::eventByName(EVENT_OBJECT_UPDATED)); + } + return aParameter; +} + +void editParameter(FeaturePtr theParam, const QString& theText) +{ + QStringList aList = theText.split("="); + QString aParamName = aList.at(0).trimmed(); + + QString aExpression = aList.at(1).trimmed(); + if (aExpression.isEmpty()) { + return; + } + + if (isNameExist(aParamName, theParam)) { + return; + } + AttributeStringPtr aNameAttr = theParam->string("variable"); + aNameAttr->setValue(aParamName.toStdString()); + + AttributeStringPtr aExprAttr = theParam->string("expression"); + aExprAttr->setValue(aExpression.toStdString()); + theParam->execute(); + + Events_Loop::loop()->flush(Events_Loop::eventByName(EVENT_OBJECT_UPDATED)); +} + +bool isNameExist(const QString& theName, FeaturePtr theIgnoreParameter) +{ + SessionPtr aMgr = ModelAPI_Session::get(); + std::shared_ptr aDoc = aMgr->activeDocument(); + FeaturePtr aParamFeature; + int aNbFeatures = aDoc->numInternalFeatures(); + std::string aName = theName.toStdString(); + for (int i = 0; i < aNbFeatures; i++) { + aParamFeature = aDoc->internalFeature(i); + if (aParamFeature && aParamFeature->getKind() == "Parameter") { + if ((theIgnoreParameter != aParamFeature) && (aParamFeature->name() == aName)) + return true; + } + } + return false; +} + +FeaturePtr findParameter(const QString& theName) +{ + SessionPtr aMgr = ModelAPI_Session::get(); + std::shared_ptr aDoc = aMgr->activeDocument(); + FeaturePtr aParamFeature; + int aNbFeatures = aDoc->numInternalFeatures(); + std::string aName = theName.toStdString(); + for (int i = 0; i < aNbFeatures; i++) { + aParamFeature = aDoc->internalFeature(i); + if (aParamFeature && aParamFeature->getKind() == "Parameter") { + if (aParamFeature->name() == aName) + return aParamFeature; + } + } + return FeaturePtr(); +} + + +//******************************************************************** +std::string generateName(const AttributePtr& theAttribute, + ModuleBase_IWorkshop* theWorkshop) +{ + std::string aName; + if (theAttribute.get() != NULL) { + ModuleBase_Operation* anOperation = theWorkshop->currentOperation(); + + FeaturePtr aFeature = ModelAPI_Feature::feature(theAttribute->owner()); + if (aFeature.get()) { + std::string aXmlCfg, aDescription; + theWorkshop->module()->getXMLRepresentation(aFeature->getKind(), aXmlCfg, aDescription); + + ModuleBase_WidgetFactory aFactory(aXmlCfg, theWorkshop); + std::string anAttributeTitle; + aFactory.getAttributeTitle(theAttribute->id(), anAttributeTitle); + + std::stringstream aStreamName; + aStreamName << theAttribute->owner()->data()->name() << "/" << anAttributeTitle.c_str(); + aName = aStreamName.str(); + } + } + return aName; +} + +bool isSameShape(const TopoDS_Shape& theShape1, const TopoDS_Shape& theShape2) +{ + // In case of compound we cannot rely on simple comparison method. + // If the compound is generated by Group feature then this compound is alwais new. + // So, we have to compare content of these compounds + if (theShape1.ShapeType() != theShape2.ShapeType()) + return false; + + if (theShape1.ShapeType() != TopAbs_COMPOUND) + return theShape1.IsSame(theShape2); + + TopoDS_Iterator aIt1(theShape1); + TopoDS_Iterator aIt2(theShape2); + + for (; aIt1.More() && aIt2.More(); aIt1.Next(), aIt2.Next()) { + if (!(aIt1.Value()).IsSame(aIt2.Value())) + return false; + } + return true; +} + +qreal currentPixelRatio() +{ + QWindowList aWnds = qApp->topLevelWindows(); + if (aWnds.size() > 0) + return aWnds.first()->devicePixelRatio(); + return qApp->primaryScreen()->devicePixelRatio(); +} + + } // namespace ModuleBase_Tools