-// Copyright (C) 2014-20xx CEA/DEN, EDF R&D
-
-#include <ModuleBase_WidgetValidated.h>
-#include <ModuleBase_FilterFactory.h>
-#include <ModuleBase_IViewer.h>
-#include <SelectMgr_ListIteratorOfListOfFilter.hxx>
-
-#include <ModelAPI_Session.h>
-#include <ModelAPI_Validator.h>
-#include <ModelAPI_ResultValidator.h>
-
-#include <QWidget>
-
-ModuleBase_WidgetValidated::ModuleBase_WidgetValidated(QWidget* theParent,
- const Config_WidgetAPI* theData,
- const std::string& theParentId)
- : ModuleBase_ModelWidget(theParent, theData, theParentId)
-{
-}
-
-ModuleBase_WidgetValidated::~ModuleBase_WidgetValidated()
-{
-}
-
-//********************************************************************
-bool ModuleBase_WidgetValidated::isValid(ObjectPtr theObj, GeomShapePtr theShape) const
-{
- SessionPtr aMgr = ModelAPI_Session::get();
- ModelAPI_ValidatorsFactory* aFactory = aMgr->validators();
- std::list<ModelAPI_Validator*> aValidators;
- std::list<std::list<std::string> > anArguments;
- aFactory->validators(parentID(), attributeID(), aValidators, anArguments);
-
- // Check the type of selected object
- std::list<ModelAPI_Validator*>::iterator aValidator = aValidators.begin();
- bool isValid = true;
- for (; aValidator != aValidators.end(); aValidator++) {
- const ModelAPI_ResultValidator* aResValidator =
- dynamic_cast<const ModelAPI_ResultValidator*>(*aValidator);
- if (aResValidator) {
- isValid = false;
- if (aResValidator->isValid(theObj)) {
- isValid = true;
- break;
- }
- }
- }
- return isValid;
-}
-
-void ModuleBase_WidgetValidated::activateFilters(ModuleBase_IWorkshop* theWorkshop,
- const bool toActivate) const
-{
- ModuleBase_IViewer* aViewer = theWorkshop->viewer();
-
- // apply filters loaded from the XML definition of the widget
- ModuleBase_FilterFactory* aFactory = theWorkshop->selectionFilters();
- SelectMgr_ListOfFilter aFactoryFilters;
- aFactory->filters(parentID(), attributeID(), aFactoryFilters);
- SelectMgr_ListIteratorOfListOfFilter aFactoryIt(aFactoryFilters);
- for (; aFactoryIt.More(); aFactoryIt.Next()) {
- Handle(SelectMgr_Filter) aSelFilter = aFactoryIt.Value();
- if (aSelFilter.IsNull())
- continue;
- if (toActivate)
- aViewer->addSelectionFilter(aSelFilter);
- else
- aViewer->removeSelectionFilter(aSelFilter);
- }
-}
-
-void ModuleBase_WidgetValidated::selectionFilters(ModuleBase_IWorkshop* theWorkshop,
- SelectMgr_ListOfFilter& theFilters) const
-{
- ModuleBase_FilterFactory* aFactory = theWorkshop->selectionFilters();
- SelectMgr_ListOfFilter aFilters;
- aFactory->filters(parentID(), attributeID(), aFilters);
- SelectMgr_ListIteratorOfListOfFilter aIt(aFilters);
- for (; aIt.More(); aIt.Next()) {
- Handle(SelectMgr_Filter) aSelFilter = aIt.Value();
- if (aSelFilter.IsNull())
- continue;
-
- theFilters.Append(aSelFilter);
- }
-}
+// Copyright (C) 2014-20xx CEA/DEN, EDF R&D\r
+\r
+#include <ModuleBase_WidgetValidated.h>\r
+#include <ModuleBase_FilterFactory.h>\r
+#include <ModuleBase_IViewer.h>\r
+#include <ModuleBase_ISelection.h>\r
+#include <ModuleBase_WidgetSelectorStore.h>\r
+#include <ModuleBase_ViewerPrs.h>\r
+\r
+#include <Events_InfoMessage.h>\r
+\r
+#include <ModelAPI_Session.h>\r
+#include <ModelAPI_Validator.h>\r
+#include <ModelAPI_AttributeValidator.h>\r
+#include <ModelAPI_Events.h>\r
+#include <ModelAPI_ResultCompSolid.h>\r
+#include <ModelAPI_Tools.h>\r
+\r
+#include <SelectMgr_ListIteratorOfListOfFilter.hxx>\r
+#include <SelectMgr_EntityOwner.hxx>\r
+#include <StdSelect_BRepOwner.hxx>\r
+\r
+#include <Events_Loop.h>\r
+\r
+#include <QWidget>\r
+\r
+//#define DEBUG_VALID_STATE\r
+\r
+ModuleBase_WidgetValidated::ModuleBase_WidgetValidated(QWidget* theParent,\r
+ ModuleBase_IWorkshop* theWorkshop,\r
+ const Config_WidgetAPI* theData)\r
+: ModuleBase_ModelWidget(theParent, theData),\r
+ myWorkshop(theWorkshop), myIsInValidate(false)\r
+{\r
+ myAttributeStore = new ModuleBase_WidgetSelectorStore();\r
+}\r
+\r
+ModuleBase_WidgetValidated::~ModuleBase_WidgetValidated()\r
+{\r
+ delete myAttributeStore;\r
+}\r
+\r
+//********************************************************************\r
+ObjectPtr ModuleBase_WidgetValidated::findPresentedObject(const AISObjectPtr& theAIS) const\r
+{\r
+ return myPresentedObject;\r
+}\r
+\r
+//********************************************************************\r
+void ModuleBase_WidgetValidated::clearValidatedCash()\r
+{\r
+#ifdef DEBUG_VALID_STATE\r
+ qDebug("clearValidatedCash");\r
+#endif\r
+ myValidPrs.Clear();\r
+ myInvalidPrs.Clear();\r
+}\r
+\r
+//********************************************************************\r
+void ModuleBase_WidgetValidated::storeAttributeValue(const AttributePtr& theAttribute)\r
+{\r
+ myIsInValidate = true;\r
+ myAttributeStore->storeAttributeValue(theAttribute, myWorkshop);\r
+}\r
+\r
+//********************************************************************\r
+void ModuleBase_WidgetValidated::restoreAttributeValue(const AttributePtr& theAttribute,\r
+ const bool theValid)\r
+{\r
+ myIsInValidate = false;\r
+ myAttributeStore->restoreAttributeValue(theAttribute, myWorkshop);\r
+}\r
+\r
+//********************************************************************\r
+bool ModuleBase_WidgetValidated::isValidInFilters(const ModuleBase_ViewerPrsPtr& thePrs)\r
+{\r
+ bool aValid = true;\r
+ Handle(SelectMgr_EntityOwner) anOwner = thePrs->owner();\r
+\r
+ // if an owner is null, the selection happens in the Object browser.\r
+ // creates a selection owner on the base of object shape and the object AIS object\r
+ if (anOwner.IsNull() && thePrs->owner().IsNull() && thePrs->object().get()) {\r
+ ResultPtr aResult = myWorkshop->selection()->getResult(thePrs);\r
+ GeomShapePtr aShape = aResult.get() ? aResult->shape() : GeomShapePtr();\r
+ // some results have no shape, e.g. the parameter one. So, they should not be validated\r
+ if (aShape.get()) {\r
+ const TopoDS_Shape aTDShape = aShape->impl<TopoDS_Shape>();\r
+ Handle(AIS_InteractiveObject) anIO = myWorkshop->selection()->getIO(thePrs);\r
+ anOwner = new StdSelect_BRepOwner(aTDShape, anIO);\r
+ myPresentedObject = aResult;\r
+ }\r
+ else\r
+ aValid = false; // only results with a shape can be filtered\r
+ }\r
+ // checks the owner by the AIS context activated filters\r
+ if (!anOwner.IsNull()) {\r
+ // the widget validator filter should be active, but during check by preselection\r
+ // it is not yet activated, so we need to activate/deactivate it manually\r
+ bool isActivated = isFilterActivated();\r
+ if (!isActivated)\r
+ activateFilters(true);\r
+\r
+ Handle(AIS_InteractiveContext) aContext = myWorkshop->viewer()->AISContext();\r
+ if (!aContext.IsNull()) {\r
+ const SelectMgr_ListOfFilter& aFilters = aContext->Filters();\r
+ SelectMgr_ListIteratorOfListOfFilter anIt(aFilters);\r
+ for (; anIt.More() && aValid; anIt.Next()) {\r
+ Handle(SelectMgr_Filter) aFilter = anIt.Value();\r
+ aValid = aFilter->IsOk(anOwner);\r
+ }\r
+ }\r
+ if (!isActivated)\r
+ activateFilters(false);\r
+ }\r
+\r
+ // removes created owner\r
+ if (!anOwner.IsNull() && anOwner != thePrs->owner()) {\r
+ anOwner.Nullify();\r
+ myPresentedObject = ObjectPtr();\r
+ }\r
+ return aValid;\r
+}\r
+\r
+//********************************************************************\r
+AttributePtr ModuleBase_WidgetValidated::attribute() const\r
+{\r
+ return myFeature->attribute(attributeID());\r
+}\r
+\r
+//********************************************************************\r
+bool ModuleBase_WidgetValidated::isValidSelection(const ModuleBase_ViewerPrsPtr& theValue)\r
+{\r
+ bool aValid = false;\r
+ if (getValidState(theValue, aValid)) {\r
+ return aValid;\r
+ }\r
+ aValid = isValidSelectionCustom(theValue);\r
+ if (aValid)\r
+ aValid = isValidSelectionForAttribute(theValue, attribute());\r
+\r
+ storeValidState(theValue, aValid);\r
+ return aValid;\r
+}\r
+\r
+//********************************************************************\r
+bool ModuleBase_WidgetValidated::isValidSelectionForAttribute(const ModuleBase_ViewerPrsPtr& theValue,\r
+ const AttributePtr& theAttribute)\r
+{\r
+ bool aValid = false;\r
+\r
+ // stores the current values of the widget attribute\r
+ bool isFlushesActived, isAttributeSetInitializedBlocked;\r
+\r
+ blockAttribute(theAttribute, true, isFlushesActived, isAttributeSetInitializedBlocked);\r
+\r
+ storeAttributeValue(theAttribute);\r
+\r
+ // saves the owner value to the widget attribute\r
+ aValid = setSelectionCustom(theValue);\r
+ if (aValid)\r
+ // checks the attribute validity\r
+ aValid = isValidAttribute(theAttribute);\r
+\r
+ // restores the current values of the widget attribute\r
+ restoreAttributeValue(theAttribute, aValid);\r
+\r
+ blockAttribute(theAttribute, false, isFlushesActived, isAttributeSetInitializedBlocked);\r
+ /// NDS: The following rows are commented for issue #1452 (to be removed after debug)\r
+ /// This is not correct to perform it here because it might cause update selection and\r
+ /// the selection mechanizm will be circled: use the scenario of the bug with preselected point.\r
+ // In particular case the results are deleted and called as redisplayed inside of this\r
+ // highlight-selection, to they must be flushed as soon as possible.\r
+ // Example: selection of group-vertices subshapes with shift pressend on body. Without\r
+ // these 4 lines below the application crashes because of left presentations on\r
+ // removed results still in the viewer.\r
+ /*static Events_ID aDeletedEvent = Events_Loop::eventByName(EVENT_OBJECT_DELETED);\r
+ static Events_ID aRedispEvent = Events_Loop::eventByName(EVENT_OBJECT_TO_REDISPLAY);\r
+ Events_Loop::loop()->flush(aDeletedEvent);\r
+ Events_Loop::loop()->flush(aRedispEvent);\r
+ */\r
+ return aValid;\r
+}\r
+\r
+//********************************************************************\r
+bool ModuleBase_WidgetValidated::isValidSelectionCustom(const ModuleBase_ViewerPrsPtr& thePrs)\r
+{\r
+ return true;\r
+}\r
+\r
+//********************************************************************\r
+bool ModuleBase_WidgetValidated::isValidAttribute(const AttributePtr& theAttribute) const\r
+{\r
+ SessionPtr aMgr = ModelAPI_Session::get();\r
+ ModelAPI_ValidatorsFactory* aFactory = aMgr->validators();\r
+ std::string aValidatorID;\r
+ Events_InfoMessage anError;\r
+ return aFactory->validate(theAttribute, aValidatorID, anError);\r
+}\r
+\r
+bool ModuleBase_WidgetValidated::isFilterActivated() const\r
+{\r
+ bool isActivated = false;\r
+\r
+ Handle(SelectMgr_Filter) aSelFilter = myWorkshop->validatorFilter();\r
+ ModuleBase_IViewer* aViewer = myWorkshop->viewer();\r
+\r
+ return aViewer->hasSelectionFilter(aSelFilter);\r
+}\r
+\r
+bool ModuleBase_WidgetValidated::activateFilters(const bool toActivate)\r
+{\r
+ ModuleBase_IViewer* aViewer = myWorkshop->viewer();\r
+\r
+ Handle(SelectMgr_Filter) aSelFilter = myWorkshop->validatorFilter();\r
+ bool aHasSelectionFilter = aViewer->hasSelectionFilter(aSelFilter);\r
+\r
+ if (toActivate)\r
+ aViewer->addSelectionFilter(aSelFilter);\r
+ else {\r
+ aViewer->removeSelectionFilter(aSelFilter);\r
+ clearValidatedCash();\r
+ }\r
+\r
+ return aHasSelectionFilter;\r
+}\r
+\r
+//********************************************************************\r
+void ModuleBase_WidgetValidated::blockAttribute(const AttributePtr& theAttribute,\r
+ const bool& theToBlock,\r
+ bool& isFlushesActived,\r
+ bool& isAttributeSetInitializedBlocked)\r
+{\r
+ Events_Loop* aLoop = Events_Loop::loop();\r
+ DataPtr aData = myFeature->data();\r
+ if (theToBlock) {\r
+ // blocks the flush signals to avoid the temporary objects visualization in the viewer\r
+ // they should not be shown in order to do not lose highlight by erasing them\r
+ isFlushesActived = aLoop->activateFlushes(false);\r
+\r
+ aData->blockSendAttributeUpdated(true);\r
+ isAttributeSetInitializedBlocked = theAttribute->blockSetInitialized(true);\r
+ }\r
+ else {\r
+ aData->blockSendAttributeUpdated(false, false);\r
+ theAttribute->blockSetInitialized(isAttributeSetInitializedBlocked);\r
+ aLoop->activateFlushes(isFlushesActived);\r
+ }\r
+}\r
+\r
+//********************************************************************\r
+void ModuleBase_WidgetValidated::storeValidState(const ModuleBase_ViewerPrsPtr& theValue, const bool theValid)\r
+{\r
+ GeomShapePtr aShape = theValue.get() ? theValue->shape() : GeomShapePtr();\r
+ if (aShape.get()) {\r
+ if (theValid) {\r
+ const TopoDS_Shape& aTDShape = aShape->impl<TopoDS_Shape>();\r
+ bool aValidPrsContains = myValidPrs.IsBound(aTDShape) &&\r
+ theValue.get()->isEqual(myValidPrs.Find(aTDShape).get());\r
+ if (!aValidPrsContains) {\r
+ #ifdef LIST_OF_VALID_PRS\r
+ myValidPrs.append(theValue);\r
+ #else\r
+ myValidPrs.Bind(aTDShape, theValue);\r
+ #endif\r
+ // the commented code will be useful when the valid state of the presentation\r
+ // will be changable between activate/deactivate. Currently it does not happen.\r
+ //if (anInvalidPrs)\r
+ // myInvalidPrs.removeOne(theValue);\r
+ }\r
+ }\r
+ else { // !theValid\r
+ if (aShape.get()) {\r
+ const TopoDS_Shape& aTDShape = aShape->impl<TopoDS_Shape>();\r
+ bool anIValidPrsContains = myInvalidPrs.IsBound(aTDShape) &&\r
+ theValue.get()->isEqual(myInvalidPrs.Find(aTDShape).get());\r
+ if (!anIValidPrsContains) {\r
+ #ifdef LIST_OF_VALID_PRS\r
+ myInvalidPrs.append(theValue);\r
+ #else\r
+ myInvalidPrs.Bind(aTDShape, theValue);\r
+ #endif\r
+ //if (!aValidPrs)\r
+ // myValidPrs.removeOne(theValue);\r
+ }\r
+ }\r
+ }\r
+ }\r
+ #ifdef DEBUG_VALID_STATE\r
+ qDebug(QString("storeValidState: myValidPrs.size() = %1, myInvalidPrs.size() = %2").arg(myValidPrs.count())\r
+ .arg(myInvalidPrs.count()).toStdString().c_str());\r
+ #endif\r
+}\r
+\r
+//********************************************************************\r
+bool ModuleBase_WidgetValidated::getValidState(const ModuleBase_ViewerPrsPtr& theValue, bool& theValid)\r
+{\r
+ if (!theValue.get())\r
+ return false;\r
+\r
+#ifdef LIST_OF_VALID_PRS\r
+ bool aValidPrsContains = myValidPrs.contains(theValue);\r
+ bool anInvalidPrsContains = myInvalidPrs.contains(theValue);\r
+#else\r
+ GeomShapePtr aShape = theValue.get() ? theValue->shape() : GeomShapePtr();\r
+ if (!aShape.get())\r
+ return false;\r
+\r
+ const TopoDS_Shape& aTDShape = aShape->impl<TopoDS_Shape>();\r
+ bool aValidPrsContains = myValidPrs.IsBound(aTDShape) &&\r
+ theValue.get()->isEqual(myValidPrs.Find(aTDShape).get());\r
+\r
+ bool anInvalidPrsContains = myInvalidPrs.IsBound(aTDShape) &&\r
+ theValue.get()->isEqual(myInvalidPrs.Find(aTDShape).get());\r
+ /*\r
+ bool aValidPrsContains = false, anInvalidPrsContains = false;\r
+ GeomShapePtr aShape = theValue.get() ? theValue->shape() : GeomShapePtr();\r
+ if (aShape.get()) {\r
+ aValidPrsContains = myValidPrs.contains(aShape);\r
+ anInvalidPrsContains = myInvalidPrs.contains(aShape);\r
+\r
+ if (aValidPrsContains)\r
+ aValidPrsContains = theValue == myValidPrs[aShape];\r
+ else\r
+ anInvalidPrsContains = theValue == myInvalidPrs[aShape];*/\r
+#endif\r
+\r
+ if (aValidPrsContains)\r
+ theValid = true;\r
+ else if (anInvalidPrsContains)\r
+ theValid = false;\r
+\r
+ return aValidPrsContains || anInvalidPrsContains;\r
+}\r
+\r
+//********************************************************************\r
+QList<ModuleBase_ViewerPrsPtr> ModuleBase_WidgetValidated::getFilteredSelected()\r
+{\r
+ QList<ModuleBase_ViewerPrsPtr> aSelected = myWorkshop->selection()->getSelected(\r
+ ModuleBase_ISelection::Viewer);\r
+\r
+ QList<ModuleBase_ViewerPrsPtr> anOBSelected = myWorkshop->selection()->getSelected(\r
+ ModuleBase_ISelection::Browser);\r
+ // filter the OB presentations\r
+ filterPresentations(anOBSelected);\r
+ if (!anOBSelected.isEmpty())\r
+ ModuleBase_ISelection::appendSelected(anOBSelected, aSelected);\r
+\r
+ filterCompSolids(aSelected);\r
+\r
+ return aSelected;\r
+}\r
+\r
+//********************************************************************\r
+void ModuleBase_WidgetValidated::filterPresentations(QList<ModuleBase_ViewerPrsPtr>& theValues)\r
+{\r
+ QList<ModuleBase_ViewerPrsPtr> aValidatedValues;\r
+\r
+ QList<ModuleBase_ViewerPrsPtr>::const_iterator anIt = theValues.begin(), aLast = theValues.end();\r
+ bool isDone = false;\r
+ for (; anIt != aLast; anIt++) {\r
+ if (isValidInFilters(*anIt))\r
+ aValidatedValues.append(*anIt);\r
+ }\r
+ if (aValidatedValues.size() != theValues.size()) {\r
+ theValues.clear();\r
+ theValues = aValidatedValues;\r
+ }\r
+}\r
+\r
+//********************************************************************\r
+void ModuleBase_WidgetValidated::filterCompSolids(QList<ModuleBase_ViewerPrsPtr>& theValues)\r
+{\r
+ std::set<ResultCompSolidPtr> aCompSolids;\r
+ QList<ModuleBase_ViewerPrsPtr> aValidatedValues;\r
+\r
+ // Collect compsolids.\r
+ QList<ModuleBase_ViewerPrsPtr>::const_iterator anIt = theValues.begin(), aLast = theValues.end();\r
+ for (; anIt != aLast; anIt++) {\r
+ const ModuleBase_ViewerPrsPtr& aViewerPrs = *anIt;\r
+ ObjectPtr anObject = aViewerPrs->object();\r
+ ResultCompSolidPtr aResultCompSolid = std::dynamic_pointer_cast<ModelAPI_ResultCompSolid>(anObject);\r
+ if(aResultCompSolid.get()) {\r
+ aCompSolids.insert(aResultCompSolid);\r
+ }\r
+ }\r
+\r
+ // Filter sub-solids of compsolids.\r
+ anIt = theValues.begin();\r
+ for (; anIt != aLast; anIt++) {\r
+ const ModuleBase_ViewerPrsPtr& aViewerPrs = *anIt;\r
+ ObjectPtr anObject = aViewerPrs->object();\r
+ ResultPtr aResult = std::dynamic_pointer_cast<ModelAPI_Result>(anObject);\r
+ ResultCompSolidPtr aResCompSolidPtr = ModelAPI_Tools::compSolidOwner(aResult);\r
+ if(aResCompSolidPtr.get() && (aCompSolids.find(aResCompSolidPtr) != aCompSolids.end())) {\r
+ // Skip sub-solid of compsolid.\r
+ continue;\r
+ } else {\r
+ aValidatedValues.append(*anIt);\r
+ }\r
+ }\r
+\r
+ if (aValidatedValues.size() != theValues.size()) {\r
+ theValues.clear();\r
+ theValues = aValidatedValues;\r
+ }\r
+}\r