1
0
mirror of https://git.dev.opencascade.org/repos/occt.git synced 2025-04-05 18:16:23 +03:00

0025631: Bounding box is too big for the face.

Fixed start and finish indexes of poles in case of trimmed bspline surface.

Test case for issue CR25631
This commit is contained in:
aml 2015-01-29 15:09:51 +03:00 committed by bugmaster
parent 4366363b04
commit 9bf0740bed
2 changed files with 211 additions and 32 deletions

View File

@ -23,10 +23,13 @@
#include <Adaptor3d_HSurface.hxx>
#include <GeomAbs_SurfaceType.hxx>
#include <BndLib.hxx>
#include <BSplCLib.hxx>
#include <gp_Pnt.hxx>
#include <gp_Pln.hxx>
#include <ElSLib.hxx>
#include <TColgp_Array2OfPnt.hxx>
#include <TColStd_Array1OfReal.hxx>
#include <TColStd_Array1OfInteger.hxx>
#include <Geom_BSplineSurface.hxx>
#include <Geom_BezierSurface.hxx>
@ -178,13 +181,44 @@ static void TreatInfinitePlane(const gp_Pln &aPlane,
aB.Enlarge(aTol);
}
// Modified by skv - Fri Aug 27 12:29:04 2004 OCC6503 End
// Compute start and finish indexes used in convex hull.
// theMinIdx - minimum poles index, that can be used.
// theMaxIdx - maximum poles index, that can be used.
// theShiftCoeff - shift between flatknots array and poles array.
// This vaule should be equal to 1 in case of non periodic BSpline,
// and (degree + 1) - mults(the lowest index).
void ComputePolesIndexes(const TColStd_Array1OfReal &theFlatKnots,
const Standard_Integer theDegree,
const Standard_Real theMin,
const Standard_Real theMax,
const Standard_Integer theMinIdx,
const Standard_Integer theMaxIdx,
const Standard_Integer theShiftCoeff,
Standard_Integer &theOutMinIdx,
Standard_Integer &theOutMaxIdx)
{
// Compute first and last used flat knots.
for(Standard_Integer aKnotIdx = theFlatKnots.Lower();
aKnotIdx < theFlatKnots.Upper();
aKnotIdx++)
{
if (theFlatKnots(aKnotIdx) <= theMin)
theOutMinIdx = aKnotIdx;
if (theFlatKnots(theFlatKnots.Upper() - aKnotIdx + theFlatKnots.Lower()) >= theMax)
theOutMaxIdx = theFlatKnots.Upper() - aKnotIdx + theFlatKnots.Lower();
}
theOutMinIdx = Max(theOutMinIdx - 2 * theDegree + 2 - theShiftCoeff, theMinIdx);
theOutMaxIdx = Min(theOutMaxIdx - 2 + theDegree + 1 - theShiftCoeff, theMaxIdx);
}
// Modified by skv - Fri Aug 27 12:29:04 2004 OCC6503 End
//=======================================================================
//function : Add
//purpose :
//=======================================================================
void BndLib_AddSurface::Add(const Adaptor3d_Surface& S,
const Standard_Real UMin,
const Standard_Real UMax,
@ -268,23 +302,131 @@ void BndLib_AddSurface::Add(const Adaptor3d_Surface& S,
case GeomAbs_BezierSurface:
case GeomAbs_BSplineSurface:
{
Standard_Boolean isUseConvexHullAlgorithm = Standard_True;
Standard_Real PTol = Precision::Parametric(Precision::Confusion());
if (Abs(UMin-S.FirstUParameter()) < PTol &&
Abs(VMin-S.FirstVParameter()) < PTol &&
Abs(UMax-S.LastUParameter ()) < PTol &&
Abs(VMax-S.LastVParameter ()) < PTol ) {
// Borders of underlying geometry.
Standard_Real anUMinParam = UMin, anUMaxParam = UMax,// BSpline case.
aVMinParam = VMin, aVMaxParam = VMax;
if (Type == GeomAbs_BezierSurface)
{
// Bezier surface:
// All of poles used for any parameter,
// thats why in case of trimmed parameters handled by grid algorithm.
if (Abs(UMin-S.FirstUParameter()) > PTol ||
Abs(VMin-S.FirstVParameter()) > PTol ||
Abs(UMax-S.LastUParameter ()) > PTol ||
Abs(VMax-S.LastVParameter ()) > PTol )
{
// Borders not equal to topology borders.
isUseConvexHullAlgorithm = Standard_False;
}
}
else
{
// BSpline:
// If Umin, Vmin, Umax, Vmax lies inside geometry bounds then:
// use convex hull algorithm,
// if Umin, VMin, Umax, Vmax lies outside then:
// use grid algorithm on analytic continuation (default case).
S.BSpline()->Bounds(anUMinParam, anUMaxParam, aVMinParam, aVMaxParam);
if ( (UMin - anUMinParam) < -PTol ||
(VMin - aVMinParam) < -PTol ||
(UMax - anUMaxParam) > PTol ||
(VMax - aVMaxParam) > PTol )
{
// Out of geometry borders.
isUseConvexHullAlgorithm = Standard_False;
}
}
if (isUseConvexHullAlgorithm)
{
TColgp_Array2OfPnt Tp(1,S.NbUPoles(),1,S.NbVPoles());
if (Type == GeomAbs_BezierSurface) {
Standard_Integer UMinIdx = 0, UMaxIdx = 0;
Standard_Integer VMinIdx = 0, VMaxIdx = 0;
if (Type == GeomAbs_BezierSurface)
{
S.Bezier()->Poles(Tp);
UMinIdx = Tp.LowerRow();
UMaxIdx = Tp.UpperRow();
VMinIdx = Tp.LowerCol();
VMaxIdx = Tp.UpperCol();
}
else {
else
{
S.BSpline()->Poles(Tp);
UMinIdx = Tp.LowerRow();
UMaxIdx = Tp.UpperRow();
VMinIdx = Tp.LowerCol();
VMaxIdx = Tp.UpperCol();
if (UMin > anUMinParam ||
UMax < anUMaxParam)
{
Standard_Integer anUFlatKnotsCount = S.BSpline()->NbUPoles() + S.BSpline()->UDegree() + 1;
Standard_Integer aShift = 1;
if (S.BSpline()->IsUPeriodic())
{
TColStd_Array1OfInteger aMults(1, S.BSpline()->NbUKnots());
S.BSpline()->UMultiplicities(aMults);
anUFlatKnotsCount = BSplCLib::KnotSequenceLength(aMults, S.BSpline()->UDegree(), Standard_True);
aShift = S.BSpline()->UDegree() + 1 - S.BSpline()->UMultiplicity(1);
}
for (Standard_Integer i = Tp.LowerRow();i<=Tp.UpperRow();i++) {
for (Standard_Integer j = Tp.LowerCol();j<=Tp.UpperCol();j++) {
TColStd_Array1OfReal anUFlatKnots(1, anUFlatKnotsCount);
S.BSpline()->UKnotSequence(anUFlatKnots);
ComputePolesIndexes(anUFlatKnots,
S.BSpline()->UDegree(),
UMin, UMax,
UMinIdx, UMaxIdx, // Min and Max Indexes
aShift,
UMinIdx, UMaxIdx); // the Output indexes
}
if (VMin > aVMinParam ||
VMax < aVMaxParam)
{
Standard_Integer anVFlatKnotsCount = S.BSpline()->NbVPoles() + S.BSpline()->VDegree() + 1;
Standard_Integer aShift = 1;
if (S.BSpline()->IsVPeriodic())
{
TColStd_Array1OfInteger aMults(1, S.BSpline()->NbVKnots());
S.BSpline()->VMultiplicities(aMults);
anVFlatKnotsCount = BSplCLib::KnotSequenceLength(aMults, S.BSpline()->VDegree(), Standard_True);
aShift = S.BSpline()->VDegree() + 1 - S.BSpline()->VMultiplicity(1);
}
TColStd_Array1OfReal anVFlatKnots(1, anVFlatKnotsCount);
S.BSpline()->VKnotSequence(anVFlatKnots);
ComputePolesIndexes(anVFlatKnots,
S.BSpline()->VDegree(),
VMin, VMax,
VMinIdx, VMaxIdx, // Min and Max Indexes
aShift,
VMinIdx, VMaxIdx); // the Output indexes
}
}
// Use poles to build convex hull.
for (Standard_Integer i = UMinIdx; i <= UMaxIdx; i++)
{
for (Standard_Integer j = VMinIdx; j <= VMaxIdx; j++)
{
B.Add(Tp(i,j));
}
}
B.Enlarge(Tol);
break;
}

37
tests/bugs/moddata_3/bug25631 Executable file
View File

@ -0,0 +1,37 @@
puts "========="
puts "OCC25631"
puts "========="
puts ""
#####################################################################
## Bounding box is too big for the face.
#####################################################################
restore [locate_data_file bug25631_fbx.brep] sh1
set rr [bounding sh1]
regexp { *([-0-9.+eE]+) +([-0-9.+eE]+) +([-0-9.+eE]+) +([-0-9.+eE]+) +([-0-9.+eE]+) +([-0-9.+eE]+)} $rr full v1_x v1_y v1_z v2_x v2_y v2_z
set tol_abs 1.0e-4
set tol_rel 0.0001
set expected_v1_x 102.04999989999993
checkreal "v1_x" ${v1_x} ${expected_v1_x} ${tol_abs} ${tol_rel}
set expected_v1_y -12.576503364721431
checkreal "v1_y" ${v1_y} ${expected_v1_y} ${tol_abs} ${tol_rel}
set expected_v1_z -12.267407382031644
checkreal "v1_z" ${v1_z} ${expected_v1_z} ${tol_abs} ${tol_rel}
set expected_v2_x 145.65000009999983
checkreal "v2_x" ${v2_x} ${expected_v2_x} ${tol_abs} ${tol_rel}
set expected_v2_y 1.0883692081680807
checkreal "v2_y" ${v2_y} ${expected_v2_y} ${tol_abs} ${tol_rel}
set expected_v2_z 1.4146362604116396
checkreal "v2_z" ${v2_z} ${expected_v2_z} ${tol_abs} ${tol_rel}
smallview
fit
set only_screen_axo 1