1
0
mirror of https://git.dev.opencascade.org/repos/occt.git synced 2025-04-03 17:56:21 +03:00

0031987: Visualization - Slow rectangular selection on models with big number of sensitives

SelectMgr_SensitiveEntitySet now stores a map of registered owners with a counter of registered entities.
SelectMgr_ViewerSelector::traverseObject() now reads the number of sensitive entities for specific owner
from SelectMgr_SensitiveEntitySet instead of re-computing it every time.
This commit is contained in:
age 2020-12-08 13:32:07 +03:00 committed by bugmaster
parent c74e3dc300
commit 266877a7c3
3 changed files with 65 additions and 14 deletions

View File

@ -41,7 +41,12 @@ void SelectMgr_SensitiveEntitySet::Append (const Handle(SelectMgr_SensitiveEntit
theEntity->ResetSelectionActiveStatus();
return;
}
mySensitives.Add (theEntity);
const Standard_Integer anExtent = mySensitives.Extent();
if (mySensitives.Add (theEntity) > anExtent)
{
addOwner (theEntity->BaseSensitive()->OwnerId());
}
MarkDirty();
}
@ -59,7 +64,12 @@ void SelectMgr_SensitiveEntitySet::Append (const Handle(SelectMgr_Selection)& th
aSelEntIter.Value()->ResetSelectionActiveStatus();
continue;
}
mySensitives.Add (aSelEntIter.Value());
const Standard_Integer anExtent = mySensitives.Extent();
if (mySensitives.Add (aSelEntIter.Value()) > anExtent)
{
addOwner (aSelEntIter.Value()->BaseSensitive()->OwnerId());
}
}
MarkDirty();
}
@ -85,6 +95,7 @@ void SelectMgr_SensitiveEntitySet::Remove (const Handle(SelectMgr_Selection)& th
}
mySensitives.RemoveLast();
removeOwner (aSelEntIter.Value()->BaseSensitive()->OwnerId());
}
MarkDirty();
@ -144,3 +155,37 @@ const Handle(SelectMgr_SensitiveEntity)& SelectMgr_SensitiveEntitySet::GetSensit
{
return mySensitives.FindKey (theIndex + 1);
}
//=======================================================================
// function : addOwner
// purpose :
//=======================================================================
void SelectMgr_SensitiveEntitySet::addOwner (const Handle(SelectMgr_EntityOwner)& theOwner)
{
if (!theOwner.IsNull())
{
if (Standard_Integer* aNumber = myOwnersMap.ChangeSeek (theOwner))
{
++(*aNumber);
}
else
{
myOwnersMap.Bind (theOwner, 1);
}
}
}
//=======================================================================
// function : removeOwner
// purpose :
//=======================================================================
void SelectMgr_SensitiveEntitySet::removeOwner (const Handle(SelectMgr_EntityOwner)& theOwner)
{
if (Standard_Integer* aNumber = !theOwner.IsNull() ? myOwnersMap.ChangeSeek (theOwner) : NULL)
{
if (--(*aNumber) == 0)
{
myOwnersMap.UnBind (theOwner);
}
}
}

View File

@ -17,13 +17,16 @@
#define _SelectMgr_SensitiveEntitySet_HeaderFile
#include <BVH_PrimitiveSet3d.hxx>
#include <NCollection_DataMap.hxx>
#include <NCollection_IndexedMap.hxx>
#include <Select3D_BndBox3d.hxx>
#include <Select3D_BVHBuilder3d.hxx>
#include <SelectMgr_EntityOwner.hxx>
#include <SelectMgr_SensitiveEntity.hxx>
#include <SelectMgr_Selection.hxx>
typedef NCollection_IndexedMap<Handle(SelectMgr_SensitiveEntity)> SelectMgr_IndexedMapOfHSensitive;
typedef NCollection_DataMap<Handle(SelectMgr_EntityOwner), Standard_Integer> SelectMgr_MapOfOwners;
//! This class is used to store all calculated sensitive entites of one selectable
//! object. It provides an interface for building BVH tree which is used to speed-up
@ -73,9 +76,21 @@ public:
//! Returns map of entities.
const SelectMgr_IndexedMapOfHSensitive& Sensitives() const { return mySensitives; }
//! Returns map of owners.
const SelectMgr_MapOfOwners& Owners() const { return myOwnersMap; }
protected:
//! Adds entity owner to the map of owners (or increases its counter if it is already there).
Standard_EXPORT void addOwner (const Handle(SelectMgr_EntityOwner)& theOwner);
//! Decreases counter of owner in the map of owners (or removes it from the map if counter == 0).
Standard_EXPORT void removeOwner (const Handle(SelectMgr_EntityOwner)& theOwner);
private:
SelectMgr_IndexedMapOfHSensitive mySensitives; //!< Map of entities and its corresponding index in BVH
SelectMgr_MapOfOwners myOwnersMap; //!< Map of entity owners and its corresponding number of sensitives
};
#endif // _SelectMgr_SensitiveEntitySet_HeaderFile

View File

@ -581,19 +581,10 @@ void SelectMgr_ViewerSelector::traverseObject (const Handle(SelectMgr_Selectable
const SelectMgr_SortCriterion& aCriterion = mystored.FindFromIndex (aStoredIter);
const Handle(SelectMgr_EntityOwner)& anOwner = aCriterion.Entity->OwnerId();
Standard_Integer aNbOwnerEntities = 0;
for (SelectMgr_IndexedMapOfHSensitive::Iterator aSensIter (anEntitySet->Sensitives()); aSensIter.More(); aSensIter.Next())
anEntitySet->Owners().Find (anOwner, aNbOwnerEntities);
if (aNbOwnerEntities > aCriterion.NbOwnerMatches)
{
if (aSensIter.Value()->BaseSensitive()->OwnerId() == anOwner)
{
if (++aNbOwnerEntities > aCriterion.NbOwnerMatches)
{
// Remove from index map.
// Considering NCollection_IndexedDataMap implementation, the values for lower indexes will not be modified.
// Hence, just keep iterating in backward direction.
mystored.RemoveFromIndex (aStoredIter);
break;
}
}
mystored.RemoveFromIndex (aStoredIter);
}
}
}