mirror of
https://git.dev.opencascade.org/repos/occt.git
synced 2025-04-05 18:16:23 +03:00
0026029: Visualization - Poor performance of connected objects
Fix performance issue with SelectMgr_SelectableObjectSet.
This commit is contained in:
parent
9319499b57
commit
d4aaad5b82
@ -32,6 +32,7 @@ Graphic3d_ShaderVariable.cxx
|
|||||||
Graphic3d_ShaderVariable.lxx
|
Graphic3d_ShaderVariable.lxx
|
||||||
Graphic3d_MapOfStructure.hxx
|
Graphic3d_MapOfStructure.hxx
|
||||||
Graphic3d_MapIteratorOfMapOfStructure.hxx
|
Graphic3d_MapIteratorOfMapOfStructure.hxx
|
||||||
|
Graphic3d_IndexedMapOfAddress.hxx
|
||||||
Graphic3d_TypeOfShaderObject.hxx
|
Graphic3d_TypeOfShaderObject.hxx
|
||||||
Graphic3d_DataStructure.pxx
|
Graphic3d_DataStructure.pxx
|
||||||
Graphic3d_StructureManager.pxx
|
Graphic3d_StructureManager.pxx
|
||||||
|
@ -493,6 +493,7 @@ is
|
|||||||
imported transient class NMapOfTransient;
|
imported transient class NMapOfTransient;
|
||||||
|
|
||||||
imported MapOfStructure;
|
imported MapOfStructure;
|
||||||
|
imported IndexedMapOfAddress;
|
||||||
imported SequenceOfDisplayedStructures;
|
imported SequenceOfDisplayedStructures;
|
||||||
|
|
||||||
---------------------------------
|
---------------------------------
|
||||||
|
23
src/Graphic3d/Graphic3d_IndexedMapOfAddress.hxx
Normal file
23
src/Graphic3d/Graphic3d_IndexedMapOfAddress.hxx
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
// Created on: 2015-04-2015
|
||||||
|
// Created by: Denis BOGOLEPOV
|
||||||
|
// Copyright (c) 2015 OPEN CASCADE SAS
|
||||||
|
//
|
||||||
|
// This file is part of Open CASCADE Technology software library.
|
||||||
|
//
|
||||||
|
// This library is free software; you can redistribute it and/or modify it under
|
||||||
|
// the terms of the GNU Lesser General Public License version 2.1 as published
|
||||||
|
// by the Free Software Foundation, with special exception defined in the file
|
||||||
|
// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT
|
||||||
|
// distribution for complete text of the license and disclaimer of any warranty.
|
||||||
|
//
|
||||||
|
// Alternatively, this file may be used under the terms of Open CASCADE
|
||||||
|
// commercial license or contractual agreement.
|
||||||
|
|
||||||
|
#ifndef _Graphic3d_IndexedMapOfAddress
|
||||||
|
#define _Graphic3d_IndexedMapOfAddress
|
||||||
|
|
||||||
|
#include <NCollection_DataMap.hxx>
|
||||||
|
|
||||||
|
typedef NCollection_DataMap<const Standard_Address, Standard_Integer> Graphic3d_IndexedMapOfAddress;
|
||||||
|
|
||||||
|
#endif // _Graphic3d_IndexedMapOfAddress
|
@ -40,42 +40,42 @@ class Structure from Graphic3d inherits TShared
|
|||||||
|
|
||||||
uses
|
uses
|
||||||
|
|
||||||
Array2OfReal from TColStd,
|
Array2OfReal from TColStd,
|
||||||
SequenceOfAddress from TColStd,
|
SequenceOfAddress from TColStd,
|
||||||
|
|
||||||
Color from Quantity,
|
Color from Quantity,
|
||||||
|
|
||||||
GenId from Aspect,
|
GenId from Aspect,
|
||||||
TypeOfHighlightMethod from Aspect,
|
TypeOfHighlightMethod from Aspect,
|
||||||
|
|
||||||
DataStructureManager from Graphic3d,
|
DataStructureManager from Graphic3d,
|
||||||
AspectFillArea3d from Graphic3d,
|
AspectFillArea3d from Graphic3d,
|
||||||
AspectLine3d from Graphic3d,
|
AspectLine3d from Graphic3d,
|
||||||
AspectMarker3d from Graphic3d,
|
AspectMarker3d from Graphic3d,
|
||||||
AspectText3d from Graphic3d,
|
AspectText3d from Graphic3d,
|
||||||
CStructure from Graphic3d,
|
CStructure from Graphic3d,
|
||||||
CStructure from Graphic3d,
|
GraphicDriver from Graphic3d,
|
||||||
GraphicDriver from Graphic3d,
|
Group from Graphic3d,
|
||||||
Group from Graphic3d,
|
SequenceOfGroup from Graphic3d,
|
||||||
SequenceOfGroup from Graphic3d,
|
SequenceOfStructure from Graphic3d,
|
||||||
SequenceOfStructure from Graphic3d,
|
HSequenceOfStructure from Graphic3d,
|
||||||
HSequenceOfStructure from Graphic3d,
|
MapOfStructure from Graphic3d,
|
||||||
MapOfStructure from Graphic3d,
|
IndexedMapOfAddress from Graphic3d,
|
||||||
StructureManager from Graphic3d,
|
StructureManager from Graphic3d,
|
||||||
StructureManagerPtr from Graphic3d,
|
StructureManagerPtr from Graphic3d,
|
||||||
TypeOfComposition from Graphic3d,
|
TypeOfComposition from Graphic3d,
|
||||||
TypeOfConnection from Graphic3d,
|
TypeOfConnection from Graphic3d,
|
||||||
TypeOfPrimitive from Graphic3d,
|
TypeOfPrimitive from Graphic3d,
|
||||||
TypeOfStructure from Graphic3d,
|
TypeOfStructure from Graphic3d,
|
||||||
Vector from Graphic3d,
|
Vector from Graphic3d,
|
||||||
Vertex from Graphic3d,
|
Vertex from Graphic3d,
|
||||||
TransModeFlags from Graphic3d,
|
TransModeFlags from Graphic3d,
|
||||||
ZLayerId from Graphic3d,
|
ZLayerId from Graphic3d,
|
||||||
Pnt from gp,
|
Pnt from gp,
|
||||||
SequenceOfHClipPlane from Graphic3d,
|
SequenceOfHClipPlane from Graphic3d,
|
||||||
BndBox4f from Graphic3d,
|
BndBox4f from Graphic3d,
|
||||||
BndBox4d from Graphic3d,
|
BndBox4d from Graphic3d,
|
||||||
Box from Bnd
|
Box from Bnd
|
||||||
|
|
||||||
raises
|
raises
|
||||||
|
|
||||||
@ -957,6 +957,38 @@ is
|
|||||||
---C++: return const &
|
---C++: return const &
|
||||||
---C++: inline
|
---C++: inline
|
||||||
|
|
||||||
|
AppendDescendant (me : mutable;
|
||||||
|
theDescendant : Address from Standard)
|
||||||
|
returns Boolean from Standard
|
||||||
|
is protected;
|
||||||
|
---Level: Internal
|
||||||
|
---Purpose: Appends new descendant structure.
|
||||||
|
---Category: Private methods
|
||||||
|
|
||||||
|
RemoveDescendant (me : mutable;
|
||||||
|
theDescendant : Address from Standard)
|
||||||
|
returns Boolean from Standard
|
||||||
|
is protected;
|
||||||
|
---Level: Internal
|
||||||
|
---Purpose: Removes the given descendant structure.
|
||||||
|
---Category: Private methods
|
||||||
|
|
||||||
|
AppendAncestor (me : mutable;
|
||||||
|
theAncestor : Address from Standard)
|
||||||
|
returns Boolean from Standard
|
||||||
|
is protected;
|
||||||
|
---Level: Internal
|
||||||
|
---Purpose: Appends new ancestor structure.
|
||||||
|
---Category: Private methods
|
||||||
|
|
||||||
|
RemoveAncestor (me : mutable;
|
||||||
|
theAncestor : Address from Standard)
|
||||||
|
returns Boolean from Standard
|
||||||
|
is protected;
|
||||||
|
---Level: Internal
|
||||||
|
---Purpose: Removes the given ancestor structure.
|
||||||
|
---Category: Private methods
|
||||||
|
|
||||||
fields
|
fields
|
||||||
|
|
||||||
--
|
--
|
||||||
@ -968,14 +1000,20 @@ fields
|
|||||||
-- It is a sequence of groups of primitives.
|
-- It is a sequence of groups of primitives.
|
||||||
--
|
--
|
||||||
|
|
||||||
-- the associated low-level structure
|
-- the associated low-level structure
|
||||||
myCStructure : CStructure from Graphic3d;
|
myCStructure : CStructure from Graphic3d;
|
||||||
|
|
||||||
-- the structures to which the structure is attached
|
-- the structures to which the structure is attached
|
||||||
myAncestors : SequenceOfAddress from TColStd;
|
myAncestors : SequenceOfAddress from TColStd;
|
||||||
|
|
||||||
-- the structures attached to the structure
|
-- the structures attached to the structure
|
||||||
myDescendants : SequenceOfAddress from TColStd;
|
myDescendants : SequenceOfAddress from TColStd;
|
||||||
|
|
||||||
|
-- the map of structures to which the structure is attached
|
||||||
|
myAncestorMap : IndexedMapOfAddress from Graphic3d;
|
||||||
|
|
||||||
|
-- the map of structures attached to the structure
|
||||||
|
myDescendantMap : IndexedMapOfAddress from Graphic3d;
|
||||||
|
|
||||||
-- the highlight method of the structure
|
-- the highlight method of the structure
|
||||||
myHighlightColor : Color from Quantity;
|
myHighlightColor : Color from Quantity;
|
||||||
|
@ -1312,6 +1312,90 @@ void Graphic3d_Structure::Descendants (Graphic3d_MapOfStructure& theSet) const
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//=============================================================================
|
||||||
|
//function : AppendDescendant
|
||||||
|
//purpose :
|
||||||
|
//=============================================================================
|
||||||
|
Standard_Boolean Graphic3d_Structure::AppendDescendant (const Standard_Address theDescendant)
|
||||||
|
{
|
||||||
|
if (myDescendantMap.IsBound (theDescendant))
|
||||||
|
{
|
||||||
|
return Standard_False; // already connected
|
||||||
|
}
|
||||||
|
|
||||||
|
myDescendantMap.Bind (theDescendant, myDescendants.Length() + 1);
|
||||||
|
myDescendants.Append (theDescendant);
|
||||||
|
|
||||||
|
return Standard_True;
|
||||||
|
}
|
||||||
|
|
||||||
|
//=============================================================================
|
||||||
|
//function : RemoveDescendant
|
||||||
|
//purpose :
|
||||||
|
//=============================================================================
|
||||||
|
Standard_Boolean Graphic3d_Structure::RemoveDescendant (const Standard_Address theDescendant)
|
||||||
|
{
|
||||||
|
Standard_Integer aStructIdx;
|
||||||
|
if (!myDescendantMap.Find (theDescendant, aStructIdx))
|
||||||
|
{
|
||||||
|
return Standard_False;
|
||||||
|
}
|
||||||
|
|
||||||
|
myDescendantMap.UnBind (theDescendant);
|
||||||
|
|
||||||
|
if (aStructIdx != myDescendants.Length())
|
||||||
|
{
|
||||||
|
myDescendants.Exchange (aStructIdx, myDescendants.Length());
|
||||||
|
myDescendantMap.Bind (myDescendants (aStructIdx), aStructIdx);
|
||||||
|
}
|
||||||
|
|
||||||
|
myDescendants.Remove (myDescendants.Length());
|
||||||
|
|
||||||
|
return Standard_True;
|
||||||
|
}
|
||||||
|
|
||||||
|
//=============================================================================
|
||||||
|
//function : AppendAncestor
|
||||||
|
//purpose :
|
||||||
|
//=============================================================================
|
||||||
|
Standard_Boolean Graphic3d_Structure::AppendAncestor (const Standard_Address theAncestor)
|
||||||
|
{
|
||||||
|
if (myAncestorMap.IsBound (theAncestor))
|
||||||
|
{
|
||||||
|
return Standard_False; // already connected
|
||||||
|
}
|
||||||
|
|
||||||
|
myAncestorMap.Bind (theAncestor, myAncestors.Length() + 1);
|
||||||
|
myAncestors.Append (theAncestor);
|
||||||
|
|
||||||
|
return Standard_True;
|
||||||
|
}
|
||||||
|
|
||||||
|
//=============================================================================
|
||||||
|
//function : RemoveAncestor
|
||||||
|
//purpose :
|
||||||
|
//=============================================================================
|
||||||
|
Standard_Boolean Graphic3d_Structure::RemoveAncestor (const Standard_Address theAncestor)
|
||||||
|
{
|
||||||
|
Standard_Integer aStructIdx;
|
||||||
|
if (!myAncestorMap.Find (theAncestor, aStructIdx))
|
||||||
|
{
|
||||||
|
return Standard_False;
|
||||||
|
}
|
||||||
|
|
||||||
|
myAncestorMap.UnBind (theAncestor);
|
||||||
|
|
||||||
|
if (aStructIdx != myAncestors.Length())
|
||||||
|
{
|
||||||
|
myAncestors.Exchange (aStructIdx, myAncestors.Length());
|
||||||
|
myAncestorMap.Bind (myAncestors (aStructIdx), aStructIdx);
|
||||||
|
}
|
||||||
|
|
||||||
|
myAncestors.Remove (myAncestors.Length());
|
||||||
|
|
||||||
|
return Standard_True;
|
||||||
|
}
|
||||||
|
|
||||||
//=============================================================================
|
//=============================================================================
|
||||||
//function : Connect
|
//function : Connect
|
||||||
//purpose :
|
//purpose :
|
||||||
@ -1320,7 +1404,10 @@ void Graphic3d_Structure::Connect (const Handle(Graphic3d_Structure)& theStructu
|
|||||||
const Graphic3d_TypeOfConnection theType,
|
const Graphic3d_TypeOfConnection theType,
|
||||||
const Standard_Boolean theWithCheck)
|
const Standard_Boolean theWithCheck)
|
||||||
{
|
{
|
||||||
if (IsDeleted()) return;
|
if (IsDeleted())
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
// cycle detection
|
// cycle detection
|
||||||
if (theWithCheck
|
if (theWithCheck
|
||||||
@ -1329,47 +1416,34 @@ void Graphic3d_Structure::Connect (const Handle(Graphic3d_Structure)& theStructu
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
switch (theType)
|
const Standard_Address aStructure = theStructure.operator->();
|
||||||
|
|
||||||
|
if (theType == Graphic3d_TOC_DESCENDANT)
|
||||||
{
|
{
|
||||||
case Graphic3d_TOC_DESCENDANT:
|
if (!AppendDescendant (aStructure))
|
||||||
{
|
{
|
||||||
const Standard_Integer aNbDesc = myDescendants.Length();
|
|
||||||
for (Standard_Integer anIter = 1; anIter <= aNbDesc; ++anIter)
|
|
||||||
{
|
|
||||||
if (myDescendants.Value (anIter) == theStructure.operator->())
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
myDescendants.Append (theStructure.operator->());
|
|
||||||
CalculateBoundBox();
|
|
||||||
theStructure->Connect (this, Graphic3d_TOC_ANCESTOR);
|
|
||||||
|
|
||||||
GraphicConnect (theStructure);
|
|
||||||
myStructureManager->Connect (this, theStructure);
|
|
||||||
|
|
||||||
Update();
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
case Graphic3d_TOC_ANCESTOR:
|
|
||||||
|
CalculateBoundBox();
|
||||||
|
theStructure->Connect (this, Graphic3d_TOC_ANCESTOR);
|
||||||
|
|
||||||
|
GraphicConnect (theStructure);
|
||||||
|
myStructureManager->Connect (this, theStructure);
|
||||||
|
|
||||||
|
Update();
|
||||||
|
}
|
||||||
|
else // Graphic3d_TOC_ANCESTOR
|
||||||
|
{
|
||||||
|
if (!AppendAncestor (aStructure))
|
||||||
{
|
{
|
||||||
const Standard_Integer aNbAnces = myAncestors.Length();
|
|
||||||
for (Standard_Integer anIter = 1; anIter <= aNbAnces; ++anIter)
|
|
||||||
{
|
|
||||||
if (myAncestors.Value (anIter) == theStructure.operator->())
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
myAncestors.Append (theStructure.operator->());
|
|
||||||
CalculateBoundBox();
|
|
||||||
theStructure->Connect (this, Graphic3d_TOC_DESCENDANT);
|
|
||||||
|
|
||||||
// myGraphicDriver->Connect is called in case if connection between parent and child
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
CalculateBoundBox();
|
||||||
|
theStructure->Connect (this, Graphic3d_TOC_DESCENDANT);
|
||||||
|
|
||||||
|
// myStructureManager->Connect is called in case if connection between parent and child
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1379,37 +1453,29 @@ void Graphic3d_Structure::Connect (const Handle(Graphic3d_Structure)& theStructu
|
|||||||
//=============================================================================
|
//=============================================================================
|
||||||
void Graphic3d_Structure::Disconnect (const Handle(Graphic3d_Structure)& theStructure)
|
void Graphic3d_Structure::Disconnect (const Handle(Graphic3d_Structure)& theStructure)
|
||||||
{
|
{
|
||||||
if (IsDeleted()) return;
|
if (IsDeleted())
|
||||||
|
|
||||||
const Standard_Integer aNbDesc = myDescendants.Length();
|
|
||||||
for (Standard_Integer anIter = 1; anIter <= aNbDesc; ++anIter)
|
|
||||||
{
|
{
|
||||||
if (myDescendants.Value (anIter) == theStructure.operator->())
|
return;
|
||||||
{
|
|
||||||
myDescendants.Remove (anIter);
|
|
||||||
theStructure->Disconnect (this);
|
|
||||||
|
|
||||||
GraphicDisconnect (theStructure);
|
|
||||||
myStructureManager->Disconnect (this, theStructure);
|
|
||||||
|
|
||||||
CalculateBoundBox();
|
|
||||||
|
|
||||||
Update();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const Standard_Integer aNbAnces = myAncestors.Length();
|
const Standard_Address aStructure = theStructure.operator->();
|
||||||
for (Standard_Integer anIter = 1; anIter <= aNbAnces; ++anIter)
|
|
||||||
|
if (RemoveDescendant (aStructure))
|
||||||
{
|
{
|
||||||
if (myAncestors.Value (anIter) == theStructure.operator->())
|
theStructure->Disconnect (this);
|
||||||
{
|
|
||||||
myAncestors.Remove (anIter);
|
GraphicDisconnect (theStructure);
|
||||||
theStructure->Disconnect (this);
|
myStructureManager->Disconnect (this, theStructure);
|
||||||
CalculateBoundBox();
|
|
||||||
// no call of myGraphicDriver->Disconnect in case of an ancestor
|
CalculateBoundBox();
|
||||||
return;
|
Update();
|
||||||
}
|
}
|
||||||
|
else if (RemoveAncestor (aStructure))
|
||||||
|
{
|
||||||
|
theStructure->Disconnect (this);
|
||||||
|
CalculateBoundBox();
|
||||||
|
|
||||||
|
// no call of myStructureManager->Disconnect in case of an ancestor
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1680,34 +1746,13 @@ gp_Pnt Graphic3d_Structure::TransformPersistencePoint() const
|
|||||||
void Graphic3d_Structure::Remove (const Standard_Address thePtr,
|
void Graphic3d_Structure::Remove (const Standard_Address thePtr,
|
||||||
const Graphic3d_TypeOfConnection theType)
|
const Graphic3d_TypeOfConnection theType)
|
||||||
{
|
{
|
||||||
switch (theType)
|
if (theType == Graphic3d_TOC_DESCENDANT)
|
||||||
{
|
{
|
||||||
case Graphic3d_TOC_DESCENDANT:
|
RemoveDescendant (thePtr);
|
||||||
{
|
}
|
||||||
const Standard_Integer aNbDesc = myDescendants.Length();
|
else
|
||||||
for (Standard_Integer anIter = 1; anIter <= aNbDesc; ++anIter)
|
{
|
||||||
{
|
RemoveAncestor (thePtr);
|
||||||
if (myDescendants.Value (anIter) == thePtr)
|
|
||||||
{
|
|
||||||
myDescendants.Remove (anIter);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case Graphic3d_TOC_ANCESTOR:
|
|
||||||
{
|
|
||||||
const Standard_Integer aNbAncestors = myAncestors.Length();
|
|
||||||
for (Standard_Integer anIter = 1; anIter <= aNbAncestors; ++anIter)
|
|
||||||
{
|
|
||||||
if (myAncestors.Value (anIter) == thePtr)
|
|
||||||
{
|
|
||||||
myAncestors.Remove (anIter);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -42,8 +42,7 @@ OpenGl_Group::OpenGl_Group (const Handle(Graphic3d_Structure)& theStruct)
|
|||||||
myAspectText(NULL),
|
myAspectText(NULL),
|
||||||
myFirst(NULL),
|
myFirst(NULL),
|
||||||
myLast(NULL),
|
myLast(NULL),
|
||||||
myIsRaytracable (Standard_False),
|
myIsRaytracable (Standard_False)
|
||||||
myModificationState (0)
|
|
||||||
{
|
{
|
||||||
Handle(OpenGl_Structure) aStruct = Handle(OpenGl_Structure)::DownCast (myStructure->CStructure());
|
Handle(OpenGl_Structure) aStruct = Handle(OpenGl_Structure)::DownCast (myStructure->CStructure());
|
||||||
if (aStruct == NULL)
|
if (aStruct == NULL)
|
||||||
@ -116,11 +115,10 @@ void OpenGl_Group::UpdateAspectFace (const Standard_Boolean theIsGlobal)
|
|||||||
|
|
||||||
if (myIsRaytracable)
|
if (myIsRaytracable)
|
||||||
{
|
{
|
||||||
++myModificationState;
|
|
||||||
OpenGl_Structure* aStruct = GlStruct();
|
OpenGl_Structure* aStruct = GlStruct();
|
||||||
if (aStruct != NULL)
|
if (aStruct != NULL)
|
||||||
{
|
{
|
||||||
aStruct->UpdateStateWithAncestorStructures();
|
aStruct->UpdateStateIfRaytracable (Standard_False);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -302,14 +300,12 @@ void OpenGl_Group::AddElement (OpenGl_Element* theElem)
|
|||||||
|
|
||||||
if (OpenGl_Raytrace::IsRaytracedElement (aNode))
|
if (OpenGl_Raytrace::IsRaytracedElement (aNode))
|
||||||
{
|
{
|
||||||
myModificationState++;
|
|
||||||
myIsRaytracable = Standard_True;
|
myIsRaytracable = Standard_True;
|
||||||
|
|
||||||
OpenGl_Structure* aStruct = GlStruct();
|
OpenGl_Structure* aStruct = GlStruct();
|
||||||
if (aStruct != NULL)
|
if (aStruct != NULL)
|
||||||
{
|
{
|
||||||
aStruct->UpdateStateWithAncestorStructures();
|
aStruct->UpdateStateIfRaytracable (Standard_False);
|
||||||
aStruct->SetRaytracableWithAncestorStructures();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -366,6 +362,8 @@ void OpenGl_Group::Clear (const Standard_Boolean theToUpdateStructureMgr)
|
|||||||
|
|
||||||
Release (aCtx);
|
Release (aCtx);
|
||||||
Graphic3d_Group::Clear (theToUpdateStructureMgr);
|
Graphic3d_Group::Clear (theToUpdateStructureMgr);
|
||||||
|
|
||||||
|
myIsRaytracable = Standard_False;
|
||||||
}
|
}
|
||||||
|
|
||||||
// =======================================================================
|
// =======================================================================
|
||||||
|
@ -104,9 +104,6 @@ public:
|
|||||||
//! Returns OpenGL face aspect.
|
//! Returns OpenGL face aspect.
|
||||||
const OpenGl_AspectFace* AspectFace() const { return myAspectFace; }
|
const OpenGl_AspectFace* AspectFace() const { return myAspectFace; }
|
||||||
|
|
||||||
//! Returns modification state for ray-tracing.
|
|
||||||
Standard_Size ModificationState() const { return myModificationState; }
|
|
||||||
|
|
||||||
//! Is the group ray-tracable (contains ray-tracable elements)?
|
//! Is the group ray-tracable (contains ray-tracable elements)?
|
||||||
Standard_Boolean IsRaytracable() const { return myIsRaytracable; }
|
Standard_Boolean IsRaytracable() const { return myIsRaytracable; }
|
||||||
|
|
||||||
@ -128,7 +125,6 @@ protected:
|
|||||||
OpenGl_ElementNode* myLast;
|
OpenGl_ElementNode* myLast;
|
||||||
|
|
||||||
Standard_Boolean myIsRaytracable;
|
Standard_Boolean myIsRaytracable;
|
||||||
Standard_Size myModificationState;
|
|
||||||
|
|
||||||
public:
|
public:
|
||||||
|
|
||||||
|
@ -516,10 +516,9 @@ namespace OpenGl_Raytrace
|
|||||||
// function : IsRaytracedGroup
|
// function : IsRaytracedGroup
|
||||||
// purpose : Checks to see if the group contains ray-trace geometry
|
// purpose : Checks to see if the group contains ray-trace geometry
|
||||||
// =======================================================================
|
// =======================================================================
|
||||||
Standard_Boolean IsRaytracedGroup (const OpenGl_Group *theGroup)
|
Standard_Boolean IsRaytracedGroup (const OpenGl_Group* theGroup)
|
||||||
{
|
{
|
||||||
const OpenGl_ElementNode* aNode;
|
for (const OpenGl_ElementNode* aNode = theGroup->FirstNode(); aNode != NULL; aNode = aNode->next)
|
||||||
for (aNode = theGroup->FirstNode(); aNode != NULL; aNode = aNode->next)
|
|
||||||
{
|
{
|
||||||
if (IsRaytracedElement (aNode))
|
if (IsRaytracedElement (aNode))
|
||||||
{
|
{
|
||||||
@ -535,17 +534,12 @@ namespace OpenGl_Raytrace
|
|||||||
// =======================================================================
|
// =======================================================================
|
||||||
Standard_Boolean IsRaytracedStructure (const OpenGl_Structure* theStructure)
|
Standard_Boolean IsRaytracedStructure (const OpenGl_Structure* theStructure)
|
||||||
{
|
{
|
||||||
for (OpenGl_Structure::GroupIterator aGroupIter (theStructure->DrawGroups());
|
for (OpenGl_Structure::GroupIterator anIter (theStructure->DrawGroups()); anIter.More(); anIter.Next())
|
||||||
aGroupIter.More(); aGroupIter.Next())
|
|
||||||
{
|
{
|
||||||
if (aGroupIter.Value()->IsRaytracable())
|
if (anIter.Value()->IsRaytracable())
|
||||||
return Standard_True;
|
{
|
||||||
}
|
|
||||||
for (OpenGl_ListOfStructure::Iterator anIts (theStructure->ConnectedStructures());
|
|
||||||
anIts.More(); anIts.Next())
|
|
||||||
{
|
|
||||||
if (IsRaytracedStructure (anIts.Value()))
|
|
||||||
return Standard_True;
|
return Standard_True;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return Standard_False;
|
return Standard_False;
|
||||||
}
|
}
|
||||||
|
@ -119,17 +119,18 @@ public:
|
|||||||
// =======================================================================
|
// =======================================================================
|
||||||
OpenGl_Structure::OpenGl_Structure (const Handle(Graphic3d_StructureManager)& theManager)
|
OpenGl_Structure::OpenGl_Structure (const Handle(Graphic3d_StructureManager)& theManager)
|
||||||
: Graphic3d_CStructure (theManager),
|
: Graphic3d_CStructure (theManager),
|
||||||
myTransformation(NULL),
|
myTransformation (NULL),
|
||||||
myTransPers(NULL),
|
myTransPers (NULL),
|
||||||
myAspectLine(NULL),
|
myAspectLine (NULL),
|
||||||
myAspectFace(NULL),
|
myAspectFace (NULL),
|
||||||
myAspectMarker(NULL),
|
myAspectMarker (NULL),
|
||||||
myAspectText(NULL),
|
myAspectText (NULL),
|
||||||
myHighlightColor(NULL),
|
myHighlightColor (NULL),
|
||||||
myIsRaytracable (Standard_False),
|
myInstancedStructure (NULL),
|
||||||
myModificationState (0),
|
myIsRaytracable (Standard_False),
|
||||||
myIsCulled (Standard_True),
|
myModificationState (0),
|
||||||
myIsMirrored (Standard_False)
|
myIsCulled (Standard_True),
|
||||||
|
myIsMirrored (Standard_False)
|
||||||
{
|
{
|
||||||
//
|
//
|
||||||
}
|
}
|
||||||
@ -189,9 +190,9 @@ void OpenGl_Structure::UpdateTransformation()
|
|||||||
|
|
||||||
matcpy (myTransformation->mat, &Graphic3d_CStructure::Transformation[0][0]);
|
matcpy (myTransformation->mat, &Graphic3d_CStructure::Transformation[0][0]);
|
||||||
|
|
||||||
if (myIsRaytracable)
|
if (IsRaytracable())
|
||||||
{
|
{
|
||||||
UpdateStateWithAncestorStructures();
|
++myModificationState;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -236,9 +237,9 @@ void OpenGl_Structure::SetAspectFace (const CALL_DEF_CONTEXTFILLAREA& theAspect)
|
|||||||
}
|
}
|
||||||
myAspectFace->SetAspect (theAspect);
|
myAspectFace->SetAspect (theAspect);
|
||||||
|
|
||||||
if (myIsRaytracable)
|
if (IsRaytracable())
|
||||||
{
|
{
|
||||||
UpdateStateWithAncestorStructures();
|
++myModificationState;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -365,111 +366,41 @@ void OpenGl_Structure::clearHighlightColor (const Handle(OpenGl_Context)& theGlC
|
|||||||
// =======================================================================
|
// =======================================================================
|
||||||
void OpenGl_Structure::OnVisibilityChanged()
|
void OpenGl_Structure::OnVisibilityChanged()
|
||||||
{
|
{
|
||||||
if (myIsRaytracable)
|
if (IsRaytracable())
|
||||||
{
|
{
|
||||||
UpdateStateWithAncestorStructures();
|
++myModificationState;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// =======================================================================
|
// =======================================================================
|
||||||
// function : RegisterAncestorStructure
|
// function : IsRaytracable
|
||||||
// purpose :
|
// purpose :
|
||||||
// =======================================================================
|
// =======================================================================
|
||||||
void OpenGl_Structure::RegisterAncestorStructure (const OpenGl_Structure* theStructure) const
|
Standard_Boolean OpenGl_Structure::IsRaytracable() const
|
||||||
{
|
{
|
||||||
for (OpenGl_ListOfStructure::Iterator anIt (myAncestorStructures); anIt.More(); anIt.Next())
|
if (!myGroups.IsEmpty())
|
||||||
{
|
{
|
||||||
if (anIt.Value() == theStructure)
|
return myIsRaytracable; // geometry structure
|
||||||
{
|
}
|
||||||
return;
|
else if (myInstancedStructure != NULL)
|
||||||
}
|
{
|
||||||
|
return myInstancedStructure->IsRaytracable(); // instance structure
|
||||||
}
|
}
|
||||||
|
|
||||||
myAncestorStructures.Append (theStructure);
|
return Standard_False; // has no any groups or structures
|
||||||
}
|
}
|
||||||
|
|
||||||
// =======================================================================
|
// =======================================================================
|
||||||
// function : UnregisterAncestorStructure
|
// function : UpdateRaytracableState
|
||||||
// purpose :
|
// purpose :
|
||||||
// =======================================================================
|
// =======================================================================
|
||||||
void OpenGl_Structure::UnregisterAncestorStructure (const OpenGl_Structure* theStructure) const
|
void OpenGl_Structure::UpdateStateIfRaytracable (const Standard_Boolean toCheck) const
|
||||||
{
|
{
|
||||||
for (OpenGl_ListOfStructure::Iterator anIt (myAncestorStructures); anIt.More(); anIt.Next())
|
myIsRaytracable = !toCheck || OpenGl_Raytrace::IsRaytracedStructure (this);
|
||||||
|
|
||||||
|
if (IsRaytracable())
|
||||||
{
|
{
|
||||||
if (anIt.Value() == theStructure)
|
++myModificationState;
|
||||||
{
|
|
||||||
myAncestorStructures.Remove (anIt);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// =======================================================================
|
|
||||||
// function : UnregisterFromAncestorStructure
|
|
||||||
// purpose :
|
|
||||||
// =======================================================================
|
|
||||||
void OpenGl_Structure::UnregisterFromAncestorStructure() const
|
|
||||||
{
|
|
||||||
for (OpenGl_ListOfStructure::Iterator anIta (myAncestorStructures); anIta.More(); anIta.Next())
|
|
||||||
{
|
|
||||||
OpenGl_Structure* anAncestor = const_cast<OpenGl_Structure*> (anIta.ChangeValue());
|
|
||||||
|
|
||||||
for (OpenGl_ListOfStructure::Iterator anIts (anAncestor->myConnected); anIts.More(); anIts.Next())
|
|
||||||
{
|
|
||||||
if (anIts.Value() == this)
|
|
||||||
{
|
|
||||||
anAncestor->myConnected.Remove (anIts);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// =======================================================================
|
|
||||||
// function : UpdateStateWithAncestorStructures
|
|
||||||
// purpose :
|
|
||||||
// =======================================================================
|
|
||||||
void OpenGl_Structure::UpdateStateWithAncestorStructures() const
|
|
||||||
{
|
|
||||||
myModificationState++;
|
|
||||||
|
|
||||||
for (OpenGl_ListOfStructure::Iterator anIt (myAncestorStructures); anIt.More(); anIt.Next())
|
|
||||||
{
|
|
||||||
anIt.Value()->UpdateStateWithAncestorStructures();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// =======================================================================
|
|
||||||
// function : UpdateRaytracableWithAncestorStructures
|
|
||||||
// purpose :
|
|
||||||
// =======================================================================
|
|
||||||
void OpenGl_Structure::UpdateRaytracableWithAncestorStructures() const
|
|
||||||
{
|
|
||||||
myIsRaytracable = OpenGl_Raytrace::IsRaytracedStructure (this);
|
|
||||||
|
|
||||||
if (!myIsRaytracable)
|
|
||||||
{
|
|
||||||
for (OpenGl_ListOfStructure::Iterator anIt (myAncestorStructures); anIt.More(); anIt.Next())
|
|
||||||
{
|
|
||||||
anIt.Value()->UpdateRaytracableWithAncestorStructures();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// =======================================================================
|
|
||||||
// function : SetRaytracableWithAncestorStructures
|
|
||||||
// purpose :
|
|
||||||
// =======================================================================
|
|
||||||
void OpenGl_Structure::SetRaytracableWithAncestorStructures() const
|
|
||||||
{
|
|
||||||
myIsRaytracable = Standard_True;
|
|
||||||
|
|
||||||
for (OpenGl_ListOfStructure::Iterator anIt (myAncestorStructures); anIt.More(); anIt.Next())
|
|
||||||
{
|
|
||||||
if (!anIt.Value()->IsRaytracable())
|
|
||||||
{
|
|
||||||
anIt.Value()->SetRaytracableWithAncestorStructures();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -479,17 +410,17 @@ void OpenGl_Structure::SetRaytracableWithAncestorStructures() const
|
|||||||
// =======================================================================
|
// =======================================================================
|
||||||
void OpenGl_Structure::Connect (Graphic3d_CStructure& theStructure)
|
void OpenGl_Structure::Connect (Graphic3d_CStructure& theStructure)
|
||||||
{
|
{
|
||||||
OpenGl_Structure* aStruct = (OpenGl_Structure* )&theStructure;
|
OpenGl_Structure* aStruct = static_cast<OpenGl_Structure*> (&theStructure);
|
||||||
Disconnect (theStructure);
|
|
||||||
myConnected.Append (aStruct);
|
Standard_ASSERT_RAISE (myInstancedStructure == NULL || myInstancedStructure == aStruct,
|
||||||
|
"Error! Instanced structure is already defined");
|
||||||
|
|
||||||
|
myInstancedStructure = aStruct;
|
||||||
|
|
||||||
if (aStruct->IsRaytracable())
|
if (aStruct->IsRaytracable())
|
||||||
{
|
{
|
||||||
UpdateStateWithAncestorStructures();
|
UpdateStateIfRaytracable (Standard_False);
|
||||||
SetRaytracableWithAncestorStructures();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
aStruct->RegisterAncestorStructure (this);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// =======================================================================
|
// =======================================================================
|
||||||
@ -498,22 +429,15 @@ void OpenGl_Structure::Connect (Graphic3d_CStructure& theStructure)
|
|||||||
// =======================================================================
|
// =======================================================================
|
||||||
void OpenGl_Structure::Disconnect (Graphic3d_CStructure& theStructure)
|
void OpenGl_Structure::Disconnect (Graphic3d_CStructure& theStructure)
|
||||||
{
|
{
|
||||||
OpenGl_Structure* aStruct = (OpenGl_Structure* )&theStructure;
|
OpenGl_Structure* aStruct = static_cast<OpenGl_Structure*> (&theStructure);
|
||||||
for (OpenGl_ListOfStructure::Iterator anIter (myConnected); anIter.More(); anIter.Next())
|
|
||||||
|
if (myInstancedStructure == aStruct)
|
||||||
{
|
{
|
||||||
// Check for the given structure
|
myInstancedStructure = NULL;
|
||||||
if (anIter.Value() == aStruct)
|
|
||||||
|
if (aStruct->IsRaytracable())
|
||||||
{
|
{
|
||||||
myConnected.Remove (anIter);
|
UpdateStateIfRaytracable();
|
||||||
|
|
||||||
if (aStruct->IsRaytracable())
|
|
||||||
{
|
|
||||||
UpdateStateWithAncestorStructures();
|
|
||||||
UpdateRaytracableWithAncestorStructures();
|
|
||||||
}
|
|
||||||
|
|
||||||
aStruct->UnregisterAncestorStructure (this);
|
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -545,12 +469,14 @@ void OpenGl_Structure::RemoveGroup (const Handle(Graphic3d_Group)& theGroup)
|
|||||||
// Check for the given group
|
// Check for the given group
|
||||||
if (aGroupIter.Value() == theGroup)
|
if (aGroupIter.Value() == theGroup)
|
||||||
{
|
{
|
||||||
|
const Standard_Boolean wasRaytracable =
|
||||||
|
static_cast<const OpenGl_Group&> (*theGroup).IsRaytracable();
|
||||||
|
|
||||||
theGroup->Clear (Standard_False);
|
theGroup->Clear (Standard_False);
|
||||||
|
|
||||||
if (((OpenGl_Group* )theGroup.operator->())->IsRaytracable())
|
if (wasRaytracable)
|
||||||
{
|
{
|
||||||
UpdateStateWithAncestorStructures();
|
UpdateStateIfRaytracable();
|
||||||
UpdateRaytracableWithAncestorStructures();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
myGroups.Remove (aGroupIter);
|
myGroups.Remove (aGroupIter);
|
||||||
@ -588,8 +514,7 @@ void OpenGl_Structure::Clear (const Handle(OpenGl_Context)& theGlCtx)
|
|||||||
|
|
||||||
if (aRaytracableGroupDeleted)
|
if (aRaytracableGroupDeleted)
|
||||||
{
|
{
|
||||||
UpdateStateWithAncestorStructures();
|
myIsRaytracable = Standard_False;
|
||||||
UpdateRaytracableWithAncestorStructures();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Is2dText = Standard_False;
|
Is2dText = Standard_False;
|
||||||
@ -695,12 +620,10 @@ void OpenGl_Structure::Render (const Handle(OpenGl_Workspace) &theWorkspace) con
|
|||||||
if (myHighlightColor)
|
if (myHighlightColor)
|
||||||
theWorkspace->HighlightColor = myHighlightColor;
|
theWorkspace->HighlightColor = myHighlightColor;
|
||||||
|
|
||||||
// Render connected structures
|
// Render instanced structure (if exists)
|
||||||
OpenGl_ListOfStructure::Iterator anIter (myConnected);
|
if (myInstancedStructure != NULL)
|
||||||
while (anIter.More())
|
|
||||||
{
|
{
|
||||||
anIter.Value()->RenderGeometry (theWorkspace);
|
myInstancedStructure->RenderGeometry (theWorkspace);
|
||||||
anIter.Next();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Set up plane equations for non-structure transformed global model-view matrix
|
// Set up plane equations for non-structure transformed global model-view matrix
|
||||||
@ -816,9 +739,6 @@ void OpenGl_Structure::Release (const Handle(OpenGl_Context)& theGlCtx)
|
|||||||
OpenGl_Element::Destroy (theGlCtx.operator->(), myAspectMarker);
|
OpenGl_Element::Destroy (theGlCtx.operator->(), myAspectMarker);
|
||||||
OpenGl_Element::Destroy (theGlCtx.operator->(), myAspectText);
|
OpenGl_Element::Destroy (theGlCtx.operator->(), myAspectText);
|
||||||
clearHighlightColor (theGlCtx);
|
clearHighlightColor (theGlCtx);
|
||||||
|
|
||||||
// Remove from connected list of ancestor
|
|
||||||
UnregisterFromAncestorStructure();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// =======================================================================
|
// =======================================================================
|
||||||
|
@ -175,8 +175,8 @@ public:
|
|||||||
//! and will lead to broken visualization due to loosed data.
|
//! and will lead to broken visualization due to loosed data.
|
||||||
Standard_EXPORT void ReleaseGlResources (const Handle(OpenGl_Context)& theGlCtx);
|
Standard_EXPORT void ReleaseGlResources (const Handle(OpenGl_Context)& theGlCtx);
|
||||||
|
|
||||||
//! Returns list of connected OpenGL structures.
|
//! Returns instanced OpenGL structure.
|
||||||
const OpenGl_ListOfStructure& ConnectedStructures() const { return myConnected; }
|
const OpenGl_Structure* InstancedStructure() const { return myInstancedStructure; }
|
||||||
|
|
||||||
//! Returns OpenGL face aspect.
|
//! Returns OpenGL face aspect.
|
||||||
const OpenGl_AspectFace* AspectFace() const { return myAspectFace; }
|
const OpenGl_AspectFace* AspectFace() const { return myAspectFace; }
|
||||||
@ -194,29 +194,14 @@ public:
|
|||||||
void ResetModificationState() const { myModificationState = 0; }
|
void ResetModificationState() const { myModificationState = 0; }
|
||||||
|
|
||||||
//! Is the structure ray-tracable (contains ray-tracable elements)?
|
//! Is the structure ray-tracable (contains ray-tracable elements)?
|
||||||
Standard_Boolean IsRaytracable() const { return myIsRaytracable; }
|
Standard_Boolean IsRaytracable() const;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
|
||||||
Standard_EXPORT virtual ~OpenGl_Structure();
|
Standard_EXPORT virtual ~OpenGl_Structure();
|
||||||
|
|
||||||
//! Registers ancestor connected structure (for updating ray-tracing state).
|
|
||||||
void RegisterAncestorStructure (const OpenGl_Structure* theStructure) const;
|
|
||||||
|
|
||||||
//! Unregisters ancestor connected structure (for updating ray-tracing state).
|
|
||||||
void UnregisterAncestorStructure (const OpenGl_Structure* theStructure) const;
|
|
||||||
|
|
||||||
//! Unregisters structure from ancestor structure (for updating ray-tracing state).
|
|
||||||
void UnregisterFromAncestorStructure() const;
|
|
||||||
|
|
||||||
//! Updates modification state for structure and its parents.
|
|
||||||
void UpdateStateWithAncestorStructures() const;
|
|
||||||
|
|
||||||
//! Updates ray-tracable status for structure and its parents.
|
//! Updates ray-tracable status for structure and its parents.
|
||||||
void UpdateRaytracableWithAncestorStructures() const;
|
void UpdateStateIfRaytracable (const Standard_Boolean toCheck = Standard_True) const;
|
||||||
|
|
||||||
//! Sets ray-tracable status for structure and its parents.
|
|
||||||
void SetRaytracableWithAncestorStructures() const;
|
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
|
||||||
@ -230,15 +215,14 @@ protected:
|
|||||||
Handle(OpenGl_Group) myHighlightBox;
|
Handle(OpenGl_Group) myHighlightBox;
|
||||||
TEL_COLOUR* myHighlightColor;
|
TEL_COLOUR* myHighlightColor;
|
||||||
|
|
||||||
OpenGl_ListOfStructure myConnected;
|
OpenGl_Structure* myInstancedStructure;
|
||||||
|
|
||||||
mutable OpenGl_ListOfStructure myAncestorStructures;
|
mutable Standard_Boolean myIsRaytracable;
|
||||||
mutable Standard_Boolean myIsRaytracable;
|
mutable Standard_Size myModificationState;
|
||||||
mutable Standard_Size myModificationState;
|
|
||||||
|
|
||||||
mutable Standard_Boolean myIsCulled; //!< A status specifying is structure needs to be rendered after BVH tree traverse.
|
mutable Standard_Boolean myIsCulled; //!< A status specifying is structure needs to be rendered after BVH tree traverse.
|
||||||
|
|
||||||
Standard_Boolean myIsMirrored; //!< Used to tell OpenGl to interpret polygons in clockwise order.
|
Standard_Boolean myIsMirrored; //!< Used to tell OpenGl to interpret polygons in clockwise order.
|
||||||
|
|
||||||
public:
|
public:
|
||||||
|
|
||||||
|
@ -448,6 +448,31 @@ protected: //! @name data types related to ray-tracing
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
//! Describes state of OpenGL structure.
|
||||||
|
struct StructState
|
||||||
|
{
|
||||||
|
Standard_Size StructureState;
|
||||||
|
Standard_Size InstancedState;
|
||||||
|
|
||||||
|
//! Creates new structure state.
|
||||||
|
StructState (const Standard_Size theStructureState = 0,
|
||||||
|
const Standard_Size theInstancedState = 0)
|
||||||
|
: StructureState (theStructureState),
|
||||||
|
InstancedState (theInstancedState)
|
||||||
|
{
|
||||||
|
//
|
||||||
|
}
|
||||||
|
|
||||||
|
//! Creates new structure state.
|
||||||
|
StructState (const OpenGl_Structure* theStructure)
|
||||||
|
{
|
||||||
|
StructureState = theStructure->ModificationState();
|
||||||
|
|
||||||
|
InstancedState = theStructure->InstancedStructure() != NULL ?
|
||||||
|
theStructure->InstancedStructure()->ModificationState() : 0;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
protected: //! @name methods related to ray-tracing
|
protected: //! @name methods related to ray-tracing
|
||||||
|
|
||||||
//! Updates 3D scene geometry for ray-tracing.
|
//! Updates 3D scene geometry for ray-tracing.
|
||||||
@ -671,7 +696,7 @@ protected: //! @name fields related to ray-tracing
|
|||||||
Standard_Integer myUniformLocations[2][OpenGl_RT_NbVariables];
|
Standard_Integer myUniformLocations[2][OpenGl_RT_NbVariables];
|
||||||
|
|
||||||
//! State of OpenGL structures reflected to ray-tracing.
|
//! State of OpenGL structures reflected to ray-tracing.
|
||||||
std::map<const OpenGl_Structure*, Standard_Size> myStructureStates;
|
std::map<const OpenGl_Structure*, StructState> myStructureStates;
|
||||||
|
|
||||||
//! PrimitiveArray to TriangleSet map for scene partial update.
|
//! PrimitiveArray to TriangleSet map for scene partial update.
|
||||||
std::map<Standard_Size, OpenGl_TriangleSet*> myArrayToTrianglesMap;
|
std::map<Standard_Size, OpenGl_TriangleSet*> myArrayToTrianglesMap;
|
||||||
|
@ -157,7 +157,7 @@ Standard_Boolean OpenGl_View::updateRaytraceGeometry (const RaytraceUpdateMode
|
|||||||
else if (theMode == OpenGl_GUM_REBUILD)
|
else if (theMode == OpenGl_GUM_REBUILD)
|
||||||
{
|
{
|
||||||
// Actualize the hash map of structures - remove out-of-date records
|
// Actualize the hash map of structures - remove out-of-date records
|
||||||
std::map<const OpenGl_Structure*, Standard_Size>::iterator anIter = myStructureStates.begin();
|
std::map<const OpenGl_Structure*, StructState>::iterator anIter = myStructureStates.begin();
|
||||||
|
|
||||||
while (anIter != myStructureStates.end())
|
while (anIter != myStructureStates.end())
|
||||||
{
|
{
|
||||||
@ -209,14 +209,18 @@ Standard_Boolean OpenGl_View::toUpdateStructure (const OpenGl_Structure* theStru
|
|||||||
return Standard_False; // did not contain ray-trace elements
|
return Standard_False; // did not contain ray-trace elements
|
||||||
}
|
}
|
||||||
|
|
||||||
std::map<const OpenGl_Structure*, Standard_Size>::iterator aStructState = myStructureStates.find (theStructure);
|
std::map<const OpenGl_Structure*, StructState>::iterator aStructState = myStructureStates.find (theStructure);
|
||||||
|
|
||||||
if (aStructState != myStructureStates.end())
|
if (aStructState == myStructureStates.end() || aStructState->second.StructureState != theStructure->ModificationState())
|
||||||
{
|
{
|
||||||
return aStructState->second != theStructure->ModificationState();
|
return Standard_True;
|
||||||
|
}
|
||||||
|
else if (theStructure->InstancedStructure() != NULL)
|
||||||
|
{
|
||||||
|
return aStructState->second.InstancedState != theStructure->InstancedStructure()->ModificationState();
|
||||||
}
|
}
|
||||||
|
|
||||||
return Standard_True;
|
return Standard_False;
|
||||||
}
|
}
|
||||||
|
|
||||||
// =======================================================================
|
// =======================================================================
|
||||||
@ -354,7 +358,7 @@ Standard_Boolean OpenGl_View::addRaytraceStructure (const OpenGl_Structure*
|
|||||||
{
|
{
|
||||||
if (!theStructure->IsVisible())
|
if (!theStructure->IsVisible())
|
||||||
{
|
{
|
||||||
myStructureStates[theStructure] = theStructure->ModificationState();
|
myStructureStates[theStructure] = StructState (theStructure);
|
||||||
|
|
||||||
return Standard_True;
|
return Standard_True;
|
||||||
}
|
}
|
||||||
@ -388,16 +392,15 @@ Standard_Boolean OpenGl_View::addRaytraceStructure (const OpenGl_Structure*
|
|||||||
theStructure->Transformation()->mat ? aStructTransform : NULL, theGlContext);
|
theStructure->Transformation()->mat ? aStructTransform : NULL, theGlContext);
|
||||||
|
|
||||||
// Process all connected OpenGL structures
|
// Process all connected OpenGL structures
|
||||||
for (OpenGl_ListOfStructure::Iterator anIts (theStructure->ConnectedStructures()); anIts.More(); anIts.Next())
|
const OpenGl_Structure* anInstanced = theStructure->InstancedStructure();
|
||||||
|
|
||||||
|
if (anInstanced != NULL && anInstanced->IsRaytracable())
|
||||||
{
|
{
|
||||||
if (anIts.Value()->IsRaytracable())
|
aResult &= addRaytraceGroups (anInstanced, aStructMatID,
|
||||||
{
|
theStructure->Transformation()->mat ? aStructTransform : NULL, theGlContext);
|
||||||
aResult &= addRaytraceGroups (anIts.Value(), aStructMatID,
|
|
||||||
theStructure->Transformation()->mat ? aStructTransform : NULL, theGlContext);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
myStructureStates[theStructure] = theStructure->ModificationState();
|
myStructureStates[theStructure] = StructState (theStructure);
|
||||||
|
|
||||||
return aResult;
|
return aResult;
|
||||||
}
|
}
|
||||||
|
@ -34,10 +34,10 @@ SelectMgr_SelectableObjectSet::SelectMgr_SelectableObjectSet()
|
|||||||
//=======================================================================
|
//=======================================================================
|
||||||
void SelectMgr_SelectableObjectSet::Append (const Handle(SelectMgr_SelectableObject)& theObject)
|
void SelectMgr_SelectableObjectSet::Append (const Handle(SelectMgr_SelectableObject)& theObject)
|
||||||
{
|
{
|
||||||
myObjects.Append (theObject);
|
if (Size() < myObjects.Add (theObject))
|
||||||
myObjectIdxs.Append (myObjects.Size());
|
{
|
||||||
|
MarkDirty();
|
||||||
MarkDirty();
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
//=======================================================================
|
//=======================================================================
|
||||||
@ -47,19 +47,15 @@ void SelectMgr_SelectableObjectSet::Append (const Handle(SelectMgr_SelectableObj
|
|||||||
//=======================================================================
|
//=======================================================================
|
||||||
void SelectMgr_SelectableObjectSet::Remove (const Handle(SelectMgr_SelectableObject)& theObject)
|
void SelectMgr_SelectableObjectSet::Remove (const Handle(SelectMgr_SelectableObject)& theObject)
|
||||||
{
|
{
|
||||||
for (Standard_Integer anObjectIdx = 1; anObjectIdx <= myObjects.Size(); ++anObjectIdx)
|
const Standard_Integer anIndex = myObjects.FindIndex (theObject);
|
||||||
|
|
||||||
|
if (anIndex != 0)
|
||||||
{
|
{
|
||||||
if (myObjects.Value (anObjectIdx) == theObject)
|
Swap (anIndex - 1, Size() - 1);
|
||||||
{
|
|
||||||
myObjects.Remove (anObjectIdx);
|
myObjects.RemoveLast();
|
||||||
myObjectIdxs.Clear();
|
|
||||||
for (Standard_Integer anObjIdxsIter = 1; anObjIdxsIter <= myObjects.Size(); ++anObjIdxsIter)
|
MarkDirty();
|
||||||
{
|
|
||||||
myObjectIdxs.Append (anObjIdxsIter);
|
|
||||||
}
|
|
||||||
MarkDirty();
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -88,11 +84,9 @@ Standard_Real SelectMgr_SelectableObjectSet::Center (const Standard_Integer theI
|
|||||||
const Standard_Integer theAxis) const
|
const Standard_Integer theAxis) const
|
||||||
{
|
{
|
||||||
Select3D_BndBox3d aBndBox = Box (theIndex);
|
Select3D_BndBox3d aBndBox = Box (theIndex);
|
||||||
Standard_Real aCenter = theAxis == 0 ? (aBndBox.CornerMin().x() + aBndBox.CornerMax().x()) * 0.5 :
|
|
||||||
(theAxis == 1 ? (aBndBox.CornerMin().y() + aBndBox.CornerMax().y()) * 0.5 :
|
|
||||||
(aBndBox.CornerMin().z() + aBndBox.CornerMax().z()) * 0.5);
|
|
||||||
|
|
||||||
return aCenter;
|
return (aBndBox.CornerMin()[theAxis] +
|
||||||
|
aBndBox.CornerMax()[theAxis]) * 0.5;
|
||||||
}
|
}
|
||||||
|
|
||||||
//=======================================================================
|
//=======================================================================
|
||||||
@ -102,10 +96,15 @@ Standard_Real SelectMgr_SelectableObjectSet::Center (const Standard_Integer theI
|
|||||||
void SelectMgr_SelectableObjectSet::Swap (const Standard_Integer theIndex1,
|
void SelectMgr_SelectableObjectSet::Swap (const Standard_Integer theIndex1,
|
||||||
const Standard_Integer theIndex2)
|
const Standard_Integer theIndex2)
|
||||||
{
|
{
|
||||||
Standard_Integer anObjectIdx1 = myObjectIdxs.Value (theIndex1 + 1);
|
const Standard_Integer aIndex1 = theIndex1 + 1;
|
||||||
Standard_Integer anObjectIdx2 = myObjectIdxs.Value (theIndex2 + 1);
|
const Standard_Integer aIndex2 = theIndex2 + 1;
|
||||||
myObjectIdxs.ChangeValue (theIndex1 + 1) = anObjectIdx2;
|
|
||||||
myObjectIdxs.ChangeValue (theIndex2 + 1) = anObjectIdx1;
|
Handle(SelectMgr_SelectableObject) anObject1 = myObjects.FindKey (aIndex1);
|
||||||
|
Handle(SelectMgr_SelectableObject) anObject2 = myObjects.FindKey (aIndex2);
|
||||||
|
|
||||||
|
myObjects.Substitute (aIndex1, EMPTY_OBJ);
|
||||||
|
myObjects.Substitute (aIndex2, anObject1);
|
||||||
|
myObjects.Substitute (aIndex1, anObject2);
|
||||||
}
|
}
|
||||||
|
|
||||||
//=======================================================================
|
//=======================================================================
|
||||||
@ -114,18 +113,16 @@ void SelectMgr_SelectableObjectSet::Swap (const Standard_Integer theIndex1,
|
|||||||
//=======================================================================
|
//=======================================================================
|
||||||
Standard_Integer SelectMgr_SelectableObjectSet::Size() const
|
Standard_Integer SelectMgr_SelectableObjectSet::Size() const
|
||||||
{
|
{
|
||||||
return myObjectIdxs.Size();
|
return myObjects.Size();
|
||||||
}
|
}
|
||||||
|
|
||||||
//=======================================================================
|
//=======================================================================
|
||||||
// function : GetObjectById
|
// function : GetObjectById
|
||||||
// purpose : Returns object from set by theIndex given
|
// purpose : Returns object from set by theIndex given
|
||||||
//=======================================================================
|
//=======================================================================
|
||||||
const Handle(SelectMgr_SelectableObject)& SelectMgr_SelectableObjectSet::GetObjectById
|
const Handle(SelectMgr_SelectableObject)& SelectMgr_SelectableObjectSet::GetObjectById (const Standard_Integer theIndex) const
|
||||||
(const Standard_Integer theIndex) const
|
|
||||||
{
|
{
|
||||||
Standard_Integer anIdx = myObjectIdxs.Value (theIndex + 1);
|
return myObjects.FindKey (theIndex + 1);
|
||||||
return myObjects.Value (anIdx);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
//=======================================================================
|
//=======================================================================
|
||||||
@ -134,11 +131,5 @@ const Handle(SelectMgr_SelectableObject)& SelectMgr_SelectableObjectSet::GetObje
|
|||||||
//=======================================================================
|
//=======================================================================
|
||||||
Standard_Boolean SelectMgr_SelectableObjectSet::Contains (const Handle(SelectMgr_SelectableObject)& theObject) const
|
Standard_Boolean SelectMgr_SelectableObjectSet::Contains (const Handle(SelectMgr_SelectableObject)& theObject) const
|
||||||
{
|
{
|
||||||
for (Standard_Integer anObjectIdx = 1; anObjectIdx <= myObjects.Size(); ++anObjectIdx)
|
return myObjects.Contains (theObject);
|
||||||
{
|
|
||||||
if (myObjects.Value (anObjectIdx) == theObject)
|
|
||||||
return Standard_True;
|
|
||||||
}
|
|
||||||
|
|
||||||
return Standard_False;
|
|
||||||
}
|
}
|
||||||
|
@ -23,18 +23,23 @@
|
|||||||
#include <SelectMgr_SelectableObject.hxx>
|
#include <SelectMgr_SelectableObject.hxx>
|
||||||
#include <SelectMgr_VectorTypes.hxx>
|
#include <SelectMgr_VectorTypes.hxx>
|
||||||
|
|
||||||
|
#include <NCollection_IndexedMap.hxx>
|
||||||
|
|
||||||
//! The purpose of this class is to organize all selectable objects into
|
//! The purpose of this class is to organize all selectable objects into
|
||||||
//! data structure, allowing to build BVH tree. For selectable objects
|
//! data structure, allowing to build BVH tree. For selectable objects
|
||||||
//! binned BVH builder is used with 32 bins and 1 element per leaf.
|
//! binned BVH builder is used with 32 bins and 1 element per leaf.
|
||||||
class SelectMgr_SelectableObjectSet : public BVH_PrimitiveSet<Standard_Real, 3>
|
class SelectMgr_SelectableObjectSet : public BVH_PrimitiveSet<Standard_Real, 3>
|
||||||
{
|
{
|
||||||
|
Handle(SelectMgr_SelectableObject) EMPTY_OBJ;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
|
|
||||||
//! Creates new empty objects set and initializes BVH tree
|
//! Creates new empty objects set and initializes BVH tree
|
||||||
//! builder to Binned builder with 1 element per list
|
//! builder to Binned builder with 1 element per list
|
||||||
SelectMgr_SelectableObjectSet();
|
SelectMgr_SelectableObjectSet();
|
||||||
|
|
||||||
virtual ~SelectMgr_SelectableObjectSet() {};
|
//! Releases resources of selectable object set.
|
||||||
|
virtual ~SelectMgr_SelectableObjectSet() { }
|
||||||
|
|
||||||
//! Adds new object to the set and marks BVH tree for rebuild
|
//! Adds new object to the set and marks BVH tree for rebuild
|
||||||
void Append (const Handle(SelectMgr_SelectableObject)& theObject);
|
void Append (const Handle(SelectMgr_SelectableObject)& theObject);
|
||||||
@ -65,8 +70,7 @@ public:
|
|||||||
|
|
||||||
private:
|
private:
|
||||||
|
|
||||||
NCollection_Sequence<Handle(SelectMgr_SelectableObject)> myObjects;
|
NCollection_IndexedMap<Handle(SelectMgr_SelectableObject)> myObjects;
|
||||||
NCollection_Sequence<Standard_Integer> myObjectIdxs;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif // _SelectMgr_SelectableObjectSet_HeaderFile
|
#endif // _SelectMgr_SelectableObjectSet_HeaderFile
|
||||||
|
@ -3834,7 +3834,7 @@ static Standard_Integer VConnect (Draw_Interpretor& /*di*/,
|
|||||||
//===============================================================================================
|
//===============================================================================================
|
||||||
//function : VConnectTo
|
//function : VConnectTo
|
||||||
//purpose : Creates and displays AIS_ConnectedInteractive object from input object and location
|
//purpose : Creates and displays AIS_ConnectedInteractive object from input object and location
|
||||||
//Draw arg : vconnectto name Xo Yo Zo object
|
//Draw arg : vconnectto name Xo Yo Zo object [-nodisplay|-noupdate|-update]
|
||||||
//===============================================================================================
|
//===============================================================================================
|
||||||
|
|
||||||
static Standard_Integer VConnectTo (Draw_Interpretor& /*di*/,
|
static Standard_Integer VConnectTo (Draw_Interpretor& /*di*/,
|
||||||
@ -3843,6 +3843,7 @@ static Standard_Integer VConnectTo (Draw_Interpretor& /*di*/,
|
|||||||
{
|
{
|
||||||
// Check the viewer
|
// Check the viewer
|
||||||
Handle(AIS_InteractiveContext) aContext = ViewerTest::GetAISContext();
|
Handle(AIS_InteractiveContext) aContext = ViewerTest::GetAISContext();
|
||||||
|
ViewerTest_AutoUpdater anUpdateTool (aContext, ViewerTest::CurrentView());
|
||||||
if (aContext.IsNull())
|
if (aContext.IsNull())
|
||||||
{
|
{
|
||||||
std::cout << "vconnect error : call vinit before\n";
|
std::cout << "vconnect error : call vinit before\n";
|
||||||
@ -3925,10 +3926,15 @@ static Standard_Integer VConnectTo (Draw_Interpretor& /*di*/,
|
|||||||
anArg.LowerCase();
|
anArg.LowerCase();
|
||||||
if (anArg == "-nodisplay")
|
if (anArg == "-nodisplay")
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
|
if (!anUpdateTool.parseRedrawMode (anArg))
|
||||||
|
{
|
||||||
|
std::cout << "Warning! Unknown argument '" << anArg << "' passed, -nodisplay|-noupdate|-update expected at this point.\n";
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Display connected object
|
// Display connected object
|
||||||
TheAISContext()->Display (aConnected);
|
TheAISContext()->Display (aConnected, Standard_False);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@ -6026,7 +6032,7 @@ void ViewerTest::ObjectCommands(Draw_Interpretor& theCommands)
|
|||||||
__FILE__, VConnect, group);
|
__FILE__, VConnect, group);
|
||||||
|
|
||||||
theCommands.Add("vconnectto",
|
theCommands.Add("vconnectto",
|
||||||
"vconnectto : instance_name Xo Yo Zo object [-nodisplay]"
|
"vconnectto : instance_name Xo Yo Zo object [-nodisplay|-noupdate|-update]"
|
||||||
" Makes an instance 'instance_name' of 'object' with position (Xo Yo Zo)."
|
" Makes an instance 'instance_name' of 'object' with position (Xo Yo Zo)."
|
||||||
"\n\t\t: -nodisplay - only creates interactive object, but not displays it",
|
"\n\t\t: -nodisplay - only creates interactive object, but not displays it",
|
||||||
__FILE__, VConnectTo,group);
|
__FILE__, VConnectTo,group);
|
||||||
|
59
tests/bugs/vis/bug26029
Normal file
59
tests/bugs/vis/bug26029
Normal file
@ -0,0 +1,59 @@
|
|||||||
|
puts "============"
|
||||||
|
puts "0026029: Visualization - Poor performance of connected objects"
|
||||||
|
puts "============"
|
||||||
|
puts ""
|
||||||
|
###########################################################
|
||||||
|
# Time spent on computation of large number of connected objects
|
||||||
|
# should grow linearly with the number of objects
|
||||||
|
###########################################################
|
||||||
|
|
||||||
|
pload MODELING VISUALIZATION
|
||||||
|
psphere s 0.5
|
||||||
|
tclean s
|
||||||
|
incmesh s 0.1
|
||||||
|
trinfo s
|
||||||
|
|
||||||
|
vinit View1
|
||||||
|
vclear
|
||||||
|
vaxo
|
||||||
|
#vcaps -vbo 0
|
||||||
|
vsetdispmode 1
|
||||||
|
vdefaults -defl 1.0 -autoTriang off
|
||||||
|
vdisplay s
|
||||||
|
|
||||||
|
# disable output of commands
|
||||||
|
decho off
|
||||||
|
|
||||||
|
set aNb1 100
|
||||||
|
|
||||||
|
# display 100x100 connected instances of single presentation
|
||||||
|
puts "Creating [expr $aNb1*$aNb1] instances..."
|
||||||
|
set t [time {for {set i 0} {$i < $aNb1} {incr i} {for {set j 0} {$j < $aNb1} {incr j} {vconnectto s_${i}_${j} ${i} ${j} 0 s -noupdate}}}]
|
||||||
|
set d1 [lindex $t 0]
|
||||||
|
puts "Done in $d1 microseconds!\n"
|
||||||
|
vclear
|
||||||
|
|
||||||
|
set aNb2 200
|
||||||
|
|
||||||
|
# display 200x200 connected instances of single presentation
|
||||||
|
puts "Creating [expr $aNb2*$aNb2] instances..."
|
||||||
|
set t [time {for {set i 0} {$i < $aNb2} {incr i} {for {set j 0} {$j < $aNb2} {incr j} {vconnectto s_${i}_${j} ${i} ${j} 0 s -noupdate}}}]
|
||||||
|
set d2 [lindex $t 0]
|
||||||
|
puts "Done in $d2 microseconds!\n"
|
||||||
|
vclear
|
||||||
|
|
||||||
|
# compare two CPU times: the ratio should be quasi-linear
|
||||||
|
set expected_ratio [expr double($aNb2 * $aNb2) / double($aNb1 * $aNb1)]
|
||||||
|
set actual_ratio [expr double($d2) / double($d1)]
|
||||||
|
puts "Comparing CPU time for the two test cases..."
|
||||||
|
puts "============================================="
|
||||||
|
puts "Expected ratio: $expected_ratio"
|
||||||
|
puts "Actual ratio: $actual_ratio"
|
||||||
|
puts "============================================="
|
||||||
|
|
||||||
|
# Allow 50% deviation from linear growth
|
||||||
|
if {[expr $actual_ratio / $expected_ratio] > 1.5} {
|
||||||
|
puts "Error: non-linear time growth detected!"
|
||||||
|
} else {
|
||||||
|
puts "Test passed!"
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user