From bc67757515c7d59cedc382ea389568ba7d26dd0c Mon Sep 17 00:00:00 2001 From: apl Date: Thu, 23 Oct 2014 14:13:46 +0400 Subject: [PATCH] 0024965: Problem in local selection mode with selected objects staying in the viewer Added method to clear outdated selection of entity owners on recompute (update) of selection. Added test case to check selection behavior in local context. Cosmetic corrections --- src/AIS/AIS_InteractiveContext.cxx | 45 +++++++---- src/AIS/AIS_LocalContext.cdl | 16 ++-- src/AIS/AIS_LocalContext.cxx | 6 +- src/AIS/AIS_LocalContext_1.cxx | 119 +++++++++++++++++++++++------ tests/bugs/vis/bug24965 | 62 +++++++++++++++ 5 files changed, 200 insertions(+), 48 deletions(-) create mode 100644 tests/bugs/vis/bug24965 diff --git a/src/AIS/AIS_InteractiveContext.cxx b/src/AIS/AIS_InteractiveContext.cxx index e8e408aae9..a39f44ca88 100644 --- a/src/AIS/AIS_InteractiveContext.cxx +++ b/src/AIS/AIS_InteractiveContext.cxx @@ -1241,26 +1241,36 @@ void AIS_InteractiveContext::RecomputePrsOnly(const Handle(AIS_InteractiveObject //function : RecomputeSelectionOnly //purpose : //======================================================================= -void AIS_InteractiveContext::RecomputeSelectionOnly(const Handle(AIS_InteractiveObject)& anIObj) +void AIS_InteractiveContext::RecomputeSelectionOnly (const Handle(AIS_InteractiveObject)& theIO) { - if(anIObj.IsNull()) return; - mgrSelector->RecomputeSelection(anIObj); + if (theIO.IsNull()) + { + return; + } + mgrSelector->RecomputeSelection (theIO); - - TColStd_ListOfInteger LI; - TColStd_ListIteratorOfListOfInteger Lit; - ActivatedModes(anIObj,LI); - if(!HasOpenedContext()){ - if(!myObjects.IsBound(anIObj)) return; - - if (myObjects(anIObj)->GraphicStatus() == AIS_DS_Displayed) + if (HasOpenedContext()) + { + for (Standard_Integer aContextIdx = 1; aContextIdx <= myLocalContexts.Extent(); aContextIdx++) { - for(Lit.Initialize(LI);Lit.More();Lit.Next()) - { - mgrSelector->Activate(anIObj,Lit.Value(),myMainSel); - } + myLocalContexts (aContextIdx)->ClearOutdatedSelection (theIO, Standard_False); } + return; + } + + if (!myObjects.IsBound (theIO) || + myObjects (theIO)->GraphicStatus() != AIS_DS_Displayed) + { + return; + } + + TColStd_ListOfInteger aModes; + ActivatedModes (theIO, aModes); + TColStd_ListIteratorOfListOfInteger aModesIter (aModes); + for (; aModesIter.More(); aModesIter.Next()) + { + mgrSelector->Activate (theIO, aModesIter.Value(), myMainSel); } } @@ -1287,6 +1297,11 @@ void AIS_InteractiveContext::Update (const Handle(AIS_InteractiveObject)& theIOb mgrSelector->Update(theIObj); + for (Standard_Integer aContextIdx = 1; aContextIdx <= myLocalContexts.Extent(); aContextIdx++) + { + myLocalContexts (aContextIdx)->ClearOutdatedSelection (theIObj, Standard_False); + } + if (theUpdateViewer) { if (!myObjects.IsBound (theIObj)) diff --git a/src/AIS/AIS_LocalContext.cdl b/src/AIS/AIS_LocalContext.cdl index 977ffee1da..ec645e4611 100644 --- a/src/AIS/AIS_LocalContext.cdl +++ b/src/AIS/AIS_LocalContext.cdl @@ -332,13 +332,17 @@ is ---Purpose: Clears local context selection. -- @param toUpdateViewer [in] if TRUE the viewer will be updated. - ClearSelected (me : mutable; - theIO : InteractiveObject from AIS; - toUpdateViewer : Boolean from Standard = Standard_True); - ---Purpose: Removes an interactive object from the local context selection. - -- Method deselects all associated entity owners. + ClearOutdatedSelection (me : mutable; + theIO : InteractiveObject from AIS; + toClearDeactivated : Boolean from Standard); + ---Purpose: Clears outdated selection and detection of owners for the + -- interactive object. Use this method if selection structures + -- of the interactive object have changed. The method unhilights + -- and removes outdated entity owners from lists of selected + -- and detected owners. -- @param theIO [in] the interactive object. - -- @param toUpdateViewer [in] if TRUE the viewer will be updated. + -- @param toClearDeactivated [in] pass TRUE to treat deactivated + -- entity owners as 'outdated' when clearing the selection. ---Category: GET THE DETECTED diff --git a/src/AIS/AIS_LocalContext.cxx b/src/AIS/AIS_LocalContext.cxx index 230606cd11..aee453fdd9 100644 --- a/src/AIS/AIS_LocalContext.cxx +++ b/src/AIS/AIS_LocalContext.cxx @@ -333,8 +333,7 @@ Erase(const Handle(AIS_InteractiveObject)& anInteractive) UpdateSort(); - // Remove object from current selection of local context - ClearSelected (anInteractive, Standard_False); + ClearOutdatedSelection (anInteractive, Standard_True); return status; } @@ -503,8 +502,7 @@ Standard_Boolean AIS_LocalContext::Remove(const Handle(AIS_InteractiveObject)& a UpdateSort(); - // Remove object from current selection of local context - ClearSelected (aSelectable, Standard_False); + ClearOutdatedSelection (aSelectable, Standard_True); return Standard_True; } diff --git a/src/AIS/AIS_LocalContext_1.cxx b/src/AIS/AIS_LocalContext_1.cxx index cc826ecf70..3bbeb805ae 100644 --- a/src/AIS/AIS_LocalContext_1.cxx +++ b/src/AIS/AIS_LocalContext_1.cxx @@ -72,6 +72,8 @@ #include #include +#include +#include #include #include #include @@ -915,28 +917,78 @@ void AIS_LocalContext::ClearSelected (const Standard_Boolean updateviewer) } //================================================== -// Function: ClearSelected +// Function: ClearOutdatedSelection // Purpose : //================================================== -void AIS_LocalContext::ClearSelected (const Handle(AIS_InteractiveObject)& theIO, - const Standard_Boolean toUpdateViewer) +void AIS_LocalContext::ClearOutdatedSelection (const Handle(AIS_InteractiveObject)& theIO, + const Standard_Boolean toClearDeactivated) { - // Keep last detected object for lastindex initialization. - Handle(SelectMgr_EntityOwner) aLastPicked = myMainVS->OnePicked(); + // 1. Collect selectable entities + SelectMgr_IndexedMapOfOwner aValidOwners; - // Remove the interactive object from detected sequence - for (Standard_Integer anIdx = 1; anIdx <= myAISDetectedSeq.Length(); ++anIdx) + const TColStd_ListOfInteger& aModes = SelectionModes (theIO); + + TColStd_ListIteratorOfListOfInteger aModeIter (aModes); + for (; aModeIter.More(); aModeIter.Next()) { - Handle(AIS_InteractiveObject) aDetectedIO = myAISDetectedSeq.Value (anIdx); - if (!aDetectedIO.IsNull() && aDetectedIO == theIO) + int aMode = aModeIter.Value(); + if (!theIO->HasSelection(aMode)) { - myAISDetectedSeq.Remove (anIdx--); + continue; + } + + if (toClearDeactivated && !mySM->IsActivated (theIO, myMainVS, aMode)) + { + continue; + } + + Handle(SelectMgr_Selection) aSelection = theIO->Selection(aMode); + for (aSelection->Init(); aSelection->More(); aSelection->Next()) + { + Handle(SelectBasics_SensitiveEntity) anEntity = aSelection->Sensitive(); + if (anEntity.IsNull()) + { + continue; + } + + Handle(SelectMgr_EntityOwner) anOwner = + Handle(SelectMgr_EntityOwner)::DownCast (anEntity->OwnerId()); + + if (anOwner.IsNull()) + { + continue; + } + + aValidOwners.Add(anOwner); } } - Standard_Integer aHilightMode = theIO->HasHilightMode() ? theIO->HilightMode() : 0; + // 2. Refresh context's detection and selection and keep only active owners + // Keep last detected object for lastindex initialization. + Handle(SelectMgr_EntityOwner) aLastPicked = myMainVS->OnePicked(); - // Remove entity owners from AIS_Selection + // Remove entity owners from detected sequences + for (Standard_Integer anIdx = 1; anIdx <= myDetectedSeq.Length(); ++anIdx) + { + Handle(SelectMgr_EntityOwner) anOwner = myMainVS->Picked (myDetectedSeq (anIdx)); + if (anOwner.IsNull() || anOwner->Selectable() != theIO || aValidOwners.Contains (anOwner)) + { + continue; + } + + myDetectedSeq.Remove (anIdx--); + + if (anIdx < myCurDetected) + { + myCurDetected--; + } + } + myCurDetected = Max (myCurDetected, 1); + + Standard_Boolean isAISRemainsDetected = Standard_False; + + // 3. Remove entity owners from AIS_Selection + const Handle(V3d_Viewer)& aViewer = myCTX->CurrentViewer(); Handle(AIS_Selection) aSelection = AIS_Selection::Selection (mySelName.ToCString()); AIS_NListTransient::Iterator anIter (aSelection->Objects()); AIS_NListTransient aRemoveEntites; @@ -948,22 +1000,27 @@ void AIS_LocalContext::ClearSelected (const Handle(AIS_InteractiveObject)& theIO continue; } - aRemoveEntites.Append (anOwner); - - if (IsSelected (anOwner)) + if (aValidOwners.Contains (anOwner)) { - anOwner->Unhilight (myMainPM, aHilightMode); + isAISRemainsDetected = Standard_True; + } + + aRemoveEntites.Append (anOwner); + anOwner->SetSelected (Standard_False); + for (aViewer->InitActiveViews(); aViewer->MoreActiveViews(); aViewer->NextActiveViews()) + { + Unhilight (anOwner, aViewer->ActiveView()); } } + AIS_NListTransient::Iterator anIterRemove (aRemoveEntites); for (; anIterRemove.More(); anIterRemove.Next()) { aSelection->Select (anIterRemove.Value()); } - // Remove entity owners from myMapOfOwner + // 4. Remove entity owners from myMapOfOwner SelectMgr_IndexedMapOfOwner anOwnersToKeep; - const Handle(V3d_Viewer)& aViewer = myCTX->CurrentViewer(); for (Standard_Integer anIdx = 1; anIdx <= myMapOfOwner.Extent(); anIdx++) { Handle(SelectMgr_EntityOwner) anOwner = myMapOfOwner (anIdx); @@ -972,7 +1029,7 @@ void AIS_LocalContext::ClearSelected (const Handle(AIS_InteractiveObject)& theIO continue; } - if (anOwner->Selectable() != theIO) + if (anOwner->Selectable() != theIO || aValidOwners.Contains (anOwner)) { anOwnersToKeep.Add (anOwner); } @@ -988,9 +1045,25 @@ void AIS_LocalContext::ClearSelected (const Handle(AIS_InteractiveObject)& theIO myMapOfOwner.Assign (anOwnersToKeep); mylastindex = myMapOfOwner.FindIndex (aLastPicked); - if (toUpdateViewer) + if (!isAISRemainsDetected) { - aViewer->Update(); + // Remove the interactive object from detected sequences + for (Standard_Integer anIdx = 1; anIdx <= myAISDetectedSeq.Length(); ++anIdx) + { + Handle(AIS_InteractiveObject) aDetectedIO = myAISDetectedSeq.Value (anIdx); + if (aDetectedIO.IsNull() || aDetectedIO != theIO) + { + continue; + } + + myAISDetectedSeq.Remove (anIdx--); + + if (anIdx < myAISCurDetected) + { + myAISCurDetected--; + } + } + myAISCurDetected = Max (myAISCurDetected, 1); } } @@ -1361,7 +1434,7 @@ Standard_Integer AIS_LocalContext::HilightNextDetected (const Handle(V3d_View)& { myCurDetected = 1; } - Handle(SelectMgr_EntityOwner) anOwner = myMainVS->Picked (myCurDetected); + Handle(SelectMgr_EntityOwner) anOwner = myMainVS->Picked (myDetectedSeq (myCurDetected)); if (anOwner.IsNull()) { return 0; @@ -1386,7 +1459,7 @@ Standard_Integer AIS_LocalContext::HilightPreviousDetected (const Handle(V3d_Vie { myCurDetected = 1; } - Handle(SelectMgr_EntityOwner) anOwner = myMainVS->Picked (myCurDetected); + Handle(SelectMgr_EntityOwner) anOwner = myMainVS->Picked (myDetectedSeq (myCurDetected)); if (anOwner.IsNull()) { return 0; diff --git a/tests/bugs/vis/bug24965 b/tests/bugs/vis/bug24965 new file mode 100644 index 0000000000..fad47aba96 --- /dev/null +++ b/tests/bugs/vis/bug24965 @@ -0,0 +1,62 @@ +puts "============" +puts "OCC24966" +puts "============" +puts "" +#################################################################################### +# Problem in local selection mode with selected objects staying in the viewer +# Outdated entity owners remain selected after recompute of presentation in +# local context. +# The following cases are tested: +# o Recompute should not clear per-object selection (neutral selection context). +# o Recompute should clear per-owner selection (local selection context). +#################################################################################### + +set check_recomputed_x 227 +set check_recomputed_y 269 +set check_untouched_x 239 +set check_untouched_y 309 + +vinit View1 +vpoint p1 100 100 0 +vpoint p2 150 150 0 +vpoint p3 100 150 0 +vplane pln1 p1 p2 p3 +veraseall +vdisplay pln1 p1 p2 p3 +vselect 0 0 2500 2500 +vchangeplane pln1 x=100 y=100 z=200 dx=0.707 dy=0.707 dz=0.707 +vfit + +checkcolor $check_recomputed_x $check_recomputed_y 0.8 0.8 0.8 +if { $stat != 1 } { + puts "Error : Neutral selection of updated object is erased." +} + +checkcolor $check_untouched_x $check_untouched_y 0.8 0.8 0.8 +if { $stat != 1 } { + puts "Error : Neutral selection of unmodified object is erased." +} + +vinit View2 +vpoint p4 100 100 0 +vpoint p5 150 150 0 +vpoint p6 100 150 0 +vplane pln2 p4 p5 p6 +veraseall +vdisplay -local pln2 p4 p5 p6 +vselect 0 0 2500 2500 +vchangeplane pln2 x=100 y=100 z=200 dx=0.707 dy=0.707 dz=0.707 +vfit + +checkcolor $check_recomputed_x $check_recomputed_y 0 1 0 +if { $stat != 1 } { + puts "Error : Local (renewed) selection of updated object is not updated." +} + +checkcolor $check_untouched_x $check_untouched_y 0.8 0.8 0.8 +if { $stat != 1 } { + puts "Error : Local selection of unmodified object is erased." +} + +vdump ${imagedir}/${casename}_View1.png +vdump ${imagedir}/${casename}_View2.png