mirror of
https://git.dev.opencascade.org/repos/occt.git
synced 2025-04-05 18:16:23 +03:00
0025729: algorith BRepOffset_MakeOffset(...) produces wrong result for join type Intersection
Test case for issue CR25729
This commit is contained in:
parent
2c12770c38
commit
b0091bc929
@ -45,8 +45,17 @@
|
|||||||
|
|
||||||
#include <Precision.hxx>
|
#include <Precision.hxx>
|
||||||
#include <gp.hxx>
|
#include <gp.hxx>
|
||||||
|
#include <BRepAdaptor_Surface.hxx>
|
||||||
|
#include <Adaptor3d_Surface.hxx>
|
||||||
|
//
|
||||||
|
static void Correct2dPoint(const Adaptor3d_Surface& theS, gp_Pnt2d& theP2d);
|
||||||
|
//
|
||||||
|
static BRepOffset_Type DefineConnectType(const TopoDS_Edge& E,
|
||||||
|
const TopoDS_Face& F1,
|
||||||
|
const TopoDS_Face& F2,
|
||||||
|
const Standard_Real SinTol,
|
||||||
|
const Standard_Boolean CorrectPoint);
|
||||||
|
//
|
||||||
static void CorrectOrientationOfTangent(gp_Vec& TangVec,
|
static void CorrectOrientationOfTangent(gp_Vec& TangVec,
|
||||||
const TopoDS_Vertex& aVertex,
|
const TopoDS_Vertex& aVertex,
|
||||||
const TopoDS_Edge& anEdge)
|
const TopoDS_Edge& anEdge)
|
||||||
@ -55,7 +64,6 @@ static void CorrectOrientationOfTangent(gp_Vec& TangVec,
|
|||||||
if (aVertex.IsSame(Vlast))
|
if (aVertex.IsSame(Vlast))
|
||||||
TangVec.Reverse();
|
TangVec.Reverse();
|
||||||
}
|
}
|
||||||
|
|
||||||
//=======================================================================
|
//=======================================================================
|
||||||
//function : BRepOffset_Analyse
|
//function : BRepOffset_Analyse
|
||||||
//purpose :
|
//purpose :
|
||||||
@ -85,99 +93,32 @@ BRepOffset_Analyse::BRepOffset_Analyse(const TopoDS_Shape& S,
|
|||||||
//=======================================================================
|
//=======================================================================
|
||||||
|
|
||||||
static void EdgeAnalyse(const TopoDS_Edge& E,
|
static void EdgeAnalyse(const TopoDS_Edge& E,
|
||||||
const TopoDS_Face& F1,
|
const TopoDS_Face& F1,
|
||||||
const TopoDS_Face& F2,
|
const TopoDS_Face& F2,
|
||||||
const Standard_Real SinTol,
|
const Standard_Real SinTol,
|
||||||
BRepOffset_ListOfInterval& LI)
|
BRepOffset_ListOfInterval& LI)
|
||||||
{
|
{
|
||||||
|
|
||||||
|
|
||||||
TopLoc_Location L;
|
|
||||||
Standard_Real f,l;
|
Standard_Real f,l;
|
||||||
|
BRep_Tool::Range(E, F1, f, l);
|
||||||
Handle (Geom_Surface) S1 = BRep_Tool::Surface(F1);
|
BRepOffset_Interval I;
|
||||||
Handle (Geom_Surface) S2 = BRep_Tool::Surface(F2);
|
I.First(f); I.Last(l);
|
||||||
Handle (Geom2d_Curve) C1 = BRep_Tool::CurveOnSurface(E,F1,f,l);
|
//
|
||||||
Handle (Geom2d_Curve) C2 = BRep_Tool::CurveOnSurface(E,F2,f,l);
|
|
||||||
|
|
||||||
BRepAdaptor_Curve C(E);
|
|
||||||
f = C.FirstParameter();
|
|
||||||
l = C.LastParameter();
|
|
||||||
|
|
||||||
// Tangent if the regularity is at least G1.
|
// Tangent if the regularity is at least G1.
|
||||||
if (BRep_Tool::HasContinuity(E,F1,F2)) {
|
if (BRep_Tool::HasContinuity(E,F1,F2)) {
|
||||||
if (BRep_Tool::Continuity(E,F1,F2) > GeomAbs_C0) {
|
if (BRep_Tool::Continuity(E,F1,F2) > GeomAbs_C0) {
|
||||||
BRepOffset_Interval I;
|
|
||||||
I.First(f); I.Last(l);
|
|
||||||
I.Type(BRepOffset_Tangent);
|
I.Type(BRepOffset_Tangent);
|
||||||
LI.Append(I);
|
LI.Append(I);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// First stage : Type determined by one of ends.
|
//
|
||||||
// Calculate normals and tangents on the curves and surface.
|
BRepOffset_Type aType = DefineConnectType(E, F1, F2, SinTol, Standard_False);
|
||||||
// normals are oriented outwards.
|
if(aType != BRepOffset_Tangent)
|
||||||
|
{
|
||||||
Standard_Real ParOnC = 0.5*(f+l);
|
aType = DefineConnectType(E, F1, F2, SinTol, Standard_True);
|
||||||
gp_Vec T1 = C.DN(ParOnC,1).Transformed(L.Transformation());
|
|
||||||
if (T1.SquareMagnitude() > gp::Resolution()) {
|
|
||||||
T1.Normalize();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (BRepOffset_Tool::OriEdgeInFace(E,F1) == TopAbs_REVERSED) {
|
|
||||||
T1.Reverse();
|
|
||||||
}
|
|
||||||
if (F1.Orientation() == TopAbs_REVERSED) T1.Reverse();
|
|
||||||
|
|
||||||
gp_Pnt2d P = C1->Value(ParOnC);
|
|
||||||
gp_Pnt P3;
|
|
||||||
gp_Vec D1U,D1V;
|
|
||||||
|
|
||||||
S1->D1(P.X(),P.Y(),P3,D1U,D1V);
|
|
||||||
gp_Vec DN1(D1U^D1V);
|
|
||||||
if (F1.Orientation() == TopAbs_REVERSED) DN1.Reverse();
|
|
||||||
|
|
||||||
P = C2->Value(ParOnC);
|
|
||||||
S2->D1(P.X(),P.Y(),P3,D1U,D1V);
|
|
||||||
gp_Vec DN2(D1U^D1V);
|
|
||||||
if (F2.Orientation() == TopAbs_REVERSED) DN2.Reverse();
|
|
||||||
|
|
||||||
DN1.Normalize();
|
|
||||||
DN2.Normalize();
|
|
||||||
|
|
||||||
gp_Vec ProVec = DN1^DN2;
|
|
||||||
Standard_Real NormProVec = ProVec.Magnitude();
|
|
||||||
|
|
||||||
BRepOffset_Interval I;
|
|
||||||
I.First(f); I.Last(l);
|
|
||||||
|
|
||||||
if (Abs(NormProVec) < SinTol) {
|
|
||||||
// plane
|
|
||||||
if (DN1.Dot(DN2) > 0) {
|
|
||||||
//Tangent
|
|
||||||
I.Type(BRepOffset_Tangent);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
//Mixed not finished!
|
|
||||||
#ifdef OCCT_DEBUG
|
|
||||||
cout <<" faces locally mixed"<<endl;
|
|
||||||
#endif
|
|
||||||
I.Type(BRepOffset_Convex);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
if (NormProVec > gp::Resolution())
|
|
||||||
ProVec.Normalize();
|
|
||||||
Standard_Real Prod = T1.Dot(DN1^DN2);
|
|
||||||
if (Prod > 0.) {
|
|
||||||
//
|
|
||||||
I.Type(BRepOffset_Convex);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
//reenters
|
|
||||||
I.Type(BRepOffset_Concave);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
I.Type(aType);
|
||||||
LI.Append(I);
|
LI.Append(I);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -540,3 +481,130 @@ void BRepOffset_Analyse::AddFaces (const TopoDS_Face& Face,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//=======================================================================
|
||||||
|
//function : Correct2dPoint
|
||||||
|
//purpose :
|
||||||
|
//=======================================================================
|
||||||
|
void Correct2dPoint(const Adaptor3d_Surface& theS, gp_Pnt2d& theP2d)
|
||||||
|
{
|
||||||
|
const Standard_Real coeff = 0.01;
|
||||||
|
Standard_Real eps;
|
||||||
|
Standard_Real u1, u2, v1, v2;
|
||||||
|
if(theS.GetType() >= GeomAbs_BezierSurface)
|
||||||
|
{
|
||||||
|
u1 = theS.FirstUParameter();
|
||||||
|
u2 = theS.LastUParameter();
|
||||||
|
v1 = theS.FirstVParameter();
|
||||||
|
v2 = theS.LastVParameter();
|
||||||
|
if(!(Precision::IsInfinite(u1) || Precision::IsInfinite(u2)))
|
||||||
|
{
|
||||||
|
eps = Max(coeff*(u2-u1), Precision::PConfusion());
|
||||||
|
if(Abs(theP2d.X()-u1) < eps)
|
||||||
|
{
|
||||||
|
theP2d.SetX(u1 + eps);
|
||||||
|
}
|
||||||
|
if(Abs(theP2d.X()-u2) < eps)
|
||||||
|
{
|
||||||
|
theP2d.SetX(u2 - eps);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if(!(Precision::IsInfinite(v1) || Precision::IsInfinite(v2)))
|
||||||
|
{
|
||||||
|
eps = Max(coeff*(v2-v1), Precision::PConfusion());
|
||||||
|
if(Abs(theP2d.Y()-v1) < eps)
|
||||||
|
{
|
||||||
|
theP2d.SetY(v1 + eps);
|
||||||
|
}
|
||||||
|
if(Abs(theP2d.Y()-v2) < eps)
|
||||||
|
{
|
||||||
|
theP2d.SetY(v2 - eps);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//=======================================================================
|
||||||
|
//function : DefineConnectType
|
||||||
|
//purpose :
|
||||||
|
//=======================================================================
|
||||||
|
BRepOffset_Type DefineConnectType(const TopoDS_Edge& E,
|
||||||
|
const TopoDS_Face& F1,
|
||||||
|
const TopoDS_Face& F2,
|
||||||
|
const Standard_Real SinTol,
|
||||||
|
const Standard_Boolean CorrectPoint)
|
||||||
|
{
|
||||||
|
TopLoc_Location L;
|
||||||
|
Standard_Real f,l;
|
||||||
|
|
||||||
|
BRepAdaptor_Surface S1(F1), S2(F2);
|
||||||
|
Handle (Geom2d_Curve) C1 = BRep_Tool::CurveOnSurface(E,F1,f,l);
|
||||||
|
Handle (Geom2d_Curve) C2 = BRep_Tool::CurveOnSurface(E,F2,f,l);
|
||||||
|
|
||||||
|
BRepAdaptor_Curve C(E);
|
||||||
|
f = C.FirstParameter();
|
||||||
|
l = C.LastParameter();
|
||||||
|
//
|
||||||
|
Standard_Real ParOnC = 0.5*(f+l);
|
||||||
|
gp_Vec T1 = C.DN(ParOnC,1).Transformed(L.Transformation());
|
||||||
|
if (T1.SquareMagnitude() > gp::Resolution()) {
|
||||||
|
T1.Normalize();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (BRepOffset_Tool::OriEdgeInFace(E,F1) == TopAbs_REVERSED) {
|
||||||
|
T1.Reverse();
|
||||||
|
}
|
||||||
|
if (F1.Orientation() == TopAbs_REVERSED) T1.Reverse();
|
||||||
|
|
||||||
|
gp_Pnt2d P = C1->Value(ParOnC);
|
||||||
|
gp_Pnt P3;
|
||||||
|
gp_Vec D1U,D1V;
|
||||||
|
|
||||||
|
if(CorrectPoint)
|
||||||
|
Correct2dPoint(S1, P);
|
||||||
|
//
|
||||||
|
S1.D1(P.X(),P.Y(),P3,D1U,D1V);
|
||||||
|
gp_Vec DN1(D1U^D1V);
|
||||||
|
if (F1.Orientation() == TopAbs_REVERSED) DN1.Reverse();
|
||||||
|
|
||||||
|
P = C2->Value(ParOnC);
|
||||||
|
if(CorrectPoint)
|
||||||
|
Correct2dPoint(S2, P);
|
||||||
|
S2.D1(P.X(),P.Y(),P3,D1U,D1V);
|
||||||
|
gp_Vec DN2(D1U^D1V);
|
||||||
|
if (F2.Orientation() == TopAbs_REVERSED) DN2.Reverse();
|
||||||
|
|
||||||
|
DN1.Normalize();
|
||||||
|
DN2.Normalize();
|
||||||
|
|
||||||
|
gp_Vec ProVec = DN1^DN2;
|
||||||
|
Standard_Real NormProVec = ProVec.Magnitude();
|
||||||
|
|
||||||
|
if (Abs(NormProVec) < SinTol) {
|
||||||
|
// plane
|
||||||
|
if (DN1.Dot(DN2) > 0) {
|
||||||
|
//Tangent
|
||||||
|
return BRepOffset_Tangent;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
//Mixed not finished!
|
||||||
|
#ifdef OCCT_DEBUG
|
||||||
|
cout <<" faces locally mixed"<<endl;
|
||||||
|
#endif
|
||||||
|
return BRepOffset_Convex;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
if (NormProVec > gp::Resolution())
|
||||||
|
ProVec.Normalize();
|
||||||
|
Standard_Real Prod = T1.Dot(DN1^DN2);
|
||||||
|
if (Prod > 0.) {
|
||||||
|
//
|
||||||
|
return BRepOffset_Convex;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
//reenters
|
||||||
|
return BRepOffset_Concave;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -83,7 +83,7 @@
|
|||||||
|
|
||||||
#ifdef DRAW
|
#ifdef DRAW
|
||||||
#include <DBRep.hxx>
|
#include <DBRep.hxx>
|
||||||
Standard_IMPORT extern Standard_Boolean AffichInt2d;
|
Standard_Boolean Inter2dAffichInt2d;
|
||||||
static Standard_Integer NbF2d = 0;
|
static Standard_Integer NbF2d = 0;
|
||||||
static Standard_Integer NbE2d = 0;
|
static Standard_Integer NbE2d = 0;
|
||||||
static Standard_Integer NbNewVertices = 0;
|
static Standard_Integer NbNewVertices = 0;
|
||||||
@ -245,7 +245,7 @@ static void Store (const TopoDS_Edge& E1,
|
|||||||
}
|
}
|
||||||
|
|
||||||
#ifdef DRAW
|
#ifdef DRAW
|
||||||
if (AffichInt2d) {
|
if (Inter2dAffichInt2d) {
|
||||||
if (!OnE1 && !OnE2) {
|
if (!OnE1 && !OnE2) {
|
||||||
char name[256];
|
char name[256];
|
||||||
sprintf(name,"VV_%d",NbNewVertices++);
|
sprintf(name,"VV_%d",NbNewVertices++);
|
||||||
@ -272,7 +272,7 @@ static void EdgeInter(const TopoDS_Face& F,
|
|||||||
Standard_Boolean WithOri)
|
Standard_Boolean WithOri)
|
||||||
{
|
{
|
||||||
#ifdef DRAW
|
#ifdef DRAW
|
||||||
if (AffichInt2d) {
|
if (Inter2dAffichInt2d) {
|
||||||
char name[256];
|
char name[256];
|
||||||
sprintf(name,"E2d_%d_%d",NbF2d,NbE2d++);
|
sprintf(name,"E2d_%d_%d",NbF2d,NbE2d++);
|
||||||
DBRep::Set(name,E1);
|
DBRep::Set(name,E1);
|
||||||
@ -543,7 +543,7 @@ static void RefEdgeInter(const TopoDS_Face& F,
|
|||||||
gp_Pnt& Pref)
|
gp_Pnt& Pref)
|
||||||
{
|
{
|
||||||
#ifdef DRAW
|
#ifdef DRAW
|
||||||
if (AffichInt2d) {
|
if (Inter2dAffichInt2d) {
|
||||||
char name[256];
|
char name[256];
|
||||||
sprintf(name,"E2d_%d_%d",NbF2d,NbE2d++);
|
sprintf(name,"E2d_%d_%d",NbF2d,NbE2d++);
|
||||||
DBRep::Set(name,E1);
|
DBRep::Set(name,E1);
|
||||||
|
@ -129,6 +129,9 @@
|
|||||||
Standard_Boolean ChronBuild = Standard_False;
|
Standard_Boolean ChronBuild = Standard_False;
|
||||||
Standard_Integer NbAE = 0;
|
Standard_Integer NbAE = 0;
|
||||||
Standard_Integer NbAF = 0;
|
Standard_Integer NbAF = 0;
|
||||||
|
Standard_Integer NVP = 0;
|
||||||
|
Standard_Integer NVM = 0;
|
||||||
|
Standard_Integer NVN = 0;
|
||||||
static OSD_Chronometer Clock;
|
static OSD_Chronometer Clock;
|
||||||
char name[100];
|
char name[100];
|
||||||
|
|
||||||
|
@ -1405,6 +1405,7 @@ void BRepOffset_Offset::Init(const TopoDS_Vertex& Vertex,
|
|||||||
for (it.Initialize(LEdge); it.More(); it.Next()) {
|
for (it.Initialize(LEdge); it.More(); it.Next()) {
|
||||||
sprintf(name,"EOnSph_%d_%d",NbOFFSET,NbEdges++);
|
sprintf(name,"EOnSph_%d_%d",NbOFFSET,NbEdges++);
|
||||||
#ifdef DRAW
|
#ifdef DRAW
|
||||||
|
const TopoDS_Shape& CurE = it.Value();
|
||||||
DBRep::Set(name, CurE);
|
DBRep::Set(name, CurE);
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
@ -2342,9 +2342,6 @@ void BiTgte_Blend::ComputeShape()
|
|||||||
|
|
||||||
BRep_Builder B;
|
BRep_Builder B;
|
||||||
|
|
||||||
#ifdef DRAW
|
|
||||||
Standard_Integer NbNT = 1;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// Maj of the Map of created.
|
// Maj of the Map of created.
|
||||||
// Update edges that do not change in the resulting shape
|
// Update edges that do not change in the resulting shape
|
||||||
|
17
tests/bugs/modalg_5/bug25729
Normal file
17
tests/bugs/modalg_5/bug25729
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
puts "========="
|
||||||
|
puts "OCC25729"
|
||||||
|
puts "========="
|
||||||
|
puts ""
|
||||||
|
###############################################
|
||||||
|
# algorith BRepOffset_MakeOffset(...) produces wrong result for join type Intersection
|
||||||
|
###############################################
|
||||||
|
|
||||||
|
restore [locate_data_file bug25729_source_shape.brep] s
|
||||||
|
|
||||||
|
offsetparameter 1.e-7 i i
|
||||||
|
offsetload s 0.14
|
||||||
|
offsetperform result
|
||||||
|
|
||||||
|
checkshape result
|
||||||
|
|
||||||
|
set 2dviewer 1
|
Loading…
x
Reference in New Issue
Block a user