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

0033791: Shape Healing - ShapeCustom not take location of source shape for the cached context and misses root one

Pass ShapeBuild_ReShape to recursive call to retrieve complete change history.
Update history of changes by the source shape (if changed), not only by its subshapes.
Check the context for a cached shape using a reference shape without location.
This commit is contained in:
oan 2024-08-08 19:33:52 +01:00 committed by dpasukhi
parent 677f383561
commit 2736652117

View File

@ -36,6 +36,68 @@
#include <Message_ProgressScope.hxx> #include <Message_ProgressScope.hxx>
namespace
{
//=======================================================================
//function : UpdateHistoryShape
//purpose : Updates ShapeBuild_ReShape by the info of the given shape
//=======================================================================
bool UpdateHistoryShape (const TopoDS_Shape& theShape,
const BRepTools_Modifier& theModifier,
const Handle(ShapeBuild_ReShape)& theReShape)
{
TopoDS_Shape aResult;
try
{
OCC_CATCH_SIGNALS
aResult = theModifier.ModifiedShape (theShape);
}
catch (Standard_NoSuchObject const&)
{
// the sub shape isn't in the map
aResult.Nullify();
}
if (!aResult.IsNull() && !theShape.IsSame (aResult))
{
theReShape->Replace (theShape, aResult);
return true;
}
return false;
}
//=======================================================================
//function : UpdateHistory
//purpose : Recursively updates ShapeBuild_ReShape to add information of all sub-shapes
//=======================================================================
void UpdateHistory (const TopoDS_Shape& theShape,
const BRepTools_Modifier& theModifier,
const Handle(ShapeBuild_ReShape)& theReShape)
{
for (TopoDS_Iterator theIterator (theShape, Standard_False); theIterator.More(); theIterator.Next())
{
const TopoDS_Shape& aCurrent = theIterator.Value();
if (UpdateHistoryShape (aCurrent, theModifier, theReShape))
{
UpdateHistory (aCurrent, theModifier, theReShape);
}
}
}
//=======================================================================
//function : UpdateShapeBuild
//purpose : Recursively updates ShapeBuild_ReShape to add information of all sub-shapes
//=======================================================================
void UpdateShapeBuild (const TopoDS_Shape& theShape,
const BRepTools_Modifier& theModifier,
const Handle(ShapeBuild_ReShape)& theReShape)
{
UpdateHistoryShape (theShape, theModifier, theReShape);
UpdateHistory (theShape, theModifier, theReShape);
}
}
//======================================================================= //=======================================================================
//function : ApplyModifier //function : ApplyModifier
//purpose : static //purpose : static
@ -58,25 +120,35 @@ TopoDS_Shape ShapeCustom::ApplyModifier (const TopoDS_Shape &S,
BRep_Builder B; BRep_Builder B;
B.MakeCompound ( C ); B.MakeCompound ( C );
SF.Location (TopLoc_Location());
Standard_Integer aShapeCount = SF.NbChildren(); Standard_Integer aShapeCount = SF.NbChildren();
Message_ProgressScope aPS(theProgress, "Applying Modifier For Solids", aShapeCount); Message_ProgressScope aPS(theProgress, "Applying Modifier For Solids", aShapeCount);
for ( TopoDS_Iterator it(SF); it.More() && aPS.More(); it.Next()) { for (TopoDS_Iterator it(SF); it.More() && aPS.More(); it.Next())
TopoDS_Shape shape = it.Value(); {
TopLoc_Location L = shape.Location(), nullLoc;
shape.Location ( nullLoc );
TopoDS_Shape res;
Message_ProgressRange aRange = aPS.Next(); Message_ProgressRange aRange = aPS.Next();
if ( context.IsBound ( shape ) )
res = context.Find ( shape ).Oriented ( shape.Orientation() ); TopoDS_Shape shape = it.Value();
else TopoDS_Shape aShapeNoLoc = it.Value();
res = ApplyModifier ( shape, M, context ,MD, aRange); aShapeNoLoc.Location (TopLoc_Location());
if ( ! res.IsSame ( shape ) ) { TopoDS_Shape res;
context.Bind ( shape, res ); if (context.Find (aShapeNoLoc, res))
{
res.Orientation (shape.Orientation());
res.Location (shape.Location(), Standard_False);
}
else
{
res = ApplyModifier (shape, M, context, MD, aRange, aReShape);
}
if ( !res.IsSame (shape) )
{
context.Bind (aShapeNoLoc, res.Located (TopLoc_Location()));
locModified = Standard_True; locModified = Standard_True;
} }
res.Location ( L, Standard_False );
B.Add ( C, res ); B.Add (C, res);
} }
if ( !aPS.More() ) if ( !aPS.More() )
@ -85,9 +157,16 @@ TopoDS_Shape ShapeCustom::ApplyModifier (const TopoDS_Shape &S,
return S; return S;
} }
if ( ! locModified ) return S; if ( !locModified )
{
return S;
}
context.Bind ( SF, C ); context.Bind ( SF, C );
return C.Oriented ( S.Orientation() );
C.Orientation (S.Orientation());
C.Location (S.Location(), Standard_False);
return C;
} }
Message_ProgressScope aPS(theProgress, "Modify the Shape", 1); Message_ProgressScope aPS(theProgress, "Modify the Shape", 1);
@ -98,26 +177,7 @@ TopoDS_Shape ShapeCustom::ApplyModifier (const TopoDS_Shape &S,
if ( !aPS.More() || !MD.IsDone() ) return S; if ( !aPS.More() || !MD.IsDone() ) return S;
if ( !aReShape.IsNull() ) if ( !aReShape.IsNull() )
{ {
for(TopoDS_Iterator theIterator(SF,Standard_False);theIterator.More();theIterator.Next()) UpdateShapeBuild ( SF, MD, aReShape );
{
const TopoDS_Shape & current = theIterator.Value();
TopoDS_Shape result;
try
{
OCC_CATCH_SIGNALS
result = MD.ModifiedShape( current );
}
catch (Standard_NoSuchObject const&)
{
// the sub shape isn't in the map
result.Nullify();
}
if (!result.IsNull() && !current.IsSame(result))
{
aReShape->Replace(current, result);
}
}
} }
return MD.ModifiedShape(SF).Oriented(S.Orientation()); return MD.ModifiedShape(SF).Oriented(S.Orientation());