mirror of
https://git.dev.opencascade.org/repos/occt.git
synced 2025-04-06 18:26:22 +03:00
Macro NO_CXX_EXCEPTION was removed from code. Method Raise() was replaced by explicit throw statement. Method Standard_Failure::Caught() was replaced by normal C++mechanism of exception transfer. Method Standard_Failure::Caught() is deprecated now. Eliminated empty constructors. Updated samples. Eliminate empty method ChangeValue from NCollection_Map class. Removed not operable methods from NCollection classes.
561 lines
17 KiB
C++
561 lines
17 KiB
C++
// Created on: 1999-09-21
|
|
// Created by: Edward AGAPOV
|
|
// Copyright (c) 1999-1999 Matra Datavision
|
|
// Copyright (c) 1999-2014 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.
|
|
|
|
|
|
#include <Adaptor3d_HCurve.hxx>
|
|
#include <GeomAdaptor_HSurfaceOfRevolution.hxx>
|
|
#include <ElCLib.hxx>
|
|
#include <Extrema_ExtPElC.hxx>
|
|
#include <Extrema_ExtPRevS.hxx>
|
|
#include <Extrema_GenExtPS.hxx>
|
|
#include <Extrema_POnCurv.hxx>
|
|
#include <Extrema_POnSurf.hxx>
|
|
#include <gp_Ax1.hxx>
|
|
#include <gp_Ax2.hxx>
|
|
#include <gp_Lin.hxx>
|
|
#include <gp_Pln.hxx>
|
|
#include <gp_Pnt.hxx>
|
|
#include <gp_Trsf.hxx>
|
|
#include <gp_Vec.hxx>
|
|
#include <Precision.hxx>
|
|
#include <Standard_OutOfRange.hxx>
|
|
#include <Standard_Type.hxx>
|
|
#include <StdFail_NotDone.hxx>
|
|
|
|
IMPLEMENT_STANDARD_RTTIEXT(Extrema_ExtPRevS,Standard_Transient)
|
|
|
|
static gp_Ax2 GetPosition (const GeomAdaptor_SurfaceOfRevolution& S)//const Handle(Adaptor_HCurve)& C)
|
|
{
|
|
Handle(Adaptor3d_HCurve) C = S.BasisCurve();
|
|
|
|
switch (C->GetType()) {
|
|
|
|
case GeomAbs_Line: {
|
|
gp_Lin L = C->Line();
|
|
gp_Dir N = S.AxeOfRevolution().Direction();
|
|
if (N.IsParallel(L.Direction(), Precision::Angular())) {
|
|
gp_Vec OO (L.Location(), S.AxeOfRevolution().Location());
|
|
if (OO.Magnitude() <= gp::Resolution()) {
|
|
OO = gp_Vec(L.Location(), ElCLib::Value(100,L));
|
|
if (N.IsParallel(OO, Precision::Angular()))
|
|
return gp_Ax2(); // Line and axe of revolution coinside
|
|
}
|
|
N ^= OO;
|
|
}
|
|
else {
|
|
N ^= L.Direction();
|
|
}
|
|
return gp_Ax2 (L.Location(), N, L.Direction());
|
|
}
|
|
case GeomAbs_Circle:
|
|
return C->Circle().Position();
|
|
case GeomAbs_Ellipse:
|
|
return C->Ellipse().Position();
|
|
case GeomAbs_Hyperbola:
|
|
return C->Hyperbola().Position();
|
|
case GeomAbs_Parabola:
|
|
return C->Parabola().Position();
|
|
default:
|
|
return gp_Ax2();
|
|
}
|
|
}
|
|
|
|
//=======================================================================
|
|
//function : HasSingularity
|
|
//purpose :
|
|
//=======================================================================
|
|
|
|
static Standard_Boolean HasSingularity(const GeomAdaptor_SurfaceOfRevolution& S)
|
|
{
|
|
|
|
const Handle(Adaptor3d_HCurve) C = S.BasisCurve();
|
|
gp_Dir N = S.AxeOfRevolution().Direction();
|
|
gp_Pnt P = S.AxeOfRevolution().Location();
|
|
|
|
gp_Lin L(P, N);
|
|
|
|
P = C->Value(C->FirstParameter());
|
|
|
|
if(L.SquareDistance(P) < Precision::SquareConfusion()) return Standard_True;
|
|
|
|
P = C->Value(C->LastParameter());
|
|
|
|
if(L.SquareDistance(P) < Precision::SquareConfusion()) return Standard_True;
|
|
|
|
return Standard_False;
|
|
}
|
|
|
|
//=============================================================================
|
|
|
|
static void PerformExtPElC (Extrema_ExtPElC& E,
|
|
const gp_Pnt& P,
|
|
const Handle(Adaptor3d_HCurve)& C,
|
|
const Standard_Real Tol)
|
|
{
|
|
switch (C->GetType()) {
|
|
case GeomAbs_Hyperbola:
|
|
E.Perform(P, C->Hyperbola(), Tol, -Precision::Infinite(),Precision::Infinite());
|
|
return;
|
|
case GeomAbs_Line:
|
|
E.Perform(P, C->Line(), Tol, -Precision::Infinite(),Precision::Infinite());
|
|
return;
|
|
case GeomAbs_Circle:
|
|
E.Perform(P, C->Circle(), Tol, 0.0, 2.0 * M_PI);
|
|
return;
|
|
case GeomAbs_Ellipse:
|
|
E.Perform(P, C->Ellipse(), Tol, 0.0, 2.0 * M_PI);
|
|
return;
|
|
case GeomAbs_Parabola:
|
|
E.Perform(P, C->Parabola(), Tol, -Precision::Infinite(),Precision::Infinite());
|
|
return;
|
|
default:
|
|
return ;
|
|
}
|
|
}
|
|
|
|
//=======================================================================
|
|
//function : IsCaseAnalyticallyComputable
|
|
//purpose :
|
|
//=======================================================================
|
|
|
|
static Standard_Boolean IsCaseAnalyticallyComputable
|
|
(const GeomAbs_CurveType& theType,
|
|
const gp_Ax2& theCurvePos,
|
|
const gp_Ax1& AxeOfRevolution)
|
|
{
|
|
// check type
|
|
switch (theType) {
|
|
case GeomAbs_Line:
|
|
case GeomAbs_Circle:
|
|
case GeomAbs_Ellipse:
|
|
case GeomAbs_Hyperbola:
|
|
case GeomAbs_Parabola:
|
|
break;
|
|
default:
|
|
return Standard_False;
|
|
}
|
|
// the axe of revolution must be in the plane of the curve.
|
|
gp_Pln pl(theCurvePos.Location(), theCurvePos.Direction());
|
|
gp_Pnt p1 = AxeOfRevolution.Location();
|
|
Standard_Real dist = 100., dist2 = dist * dist;
|
|
Standard_Real aThreshold = Precision::Angular() * Precision::Angular() * dist2;
|
|
gp_Pnt p2 = AxeOfRevolution.Location().XYZ() + dist * AxeOfRevolution.Direction().XYZ();
|
|
|
|
if((pl.SquareDistance(p1) < aThreshold) &&
|
|
(pl.SquareDistance(p2) < aThreshold))
|
|
return Standard_True;
|
|
return Standard_False;
|
|
// gp_Vec V (AxeOfRevolution.Location(),theCurvePos.Location());
|
|
// if (Abs( V * theCurvePos.Direction()) <= gp::Resolution())
|
|
// return Standard_True;
|
|
// else
|
|
// return Standard_False;
|
|
}
|
|
|
|
//=======================================================================
|
|
//function : IsOriginalPnt
|
|
//purpose :
|
|
//=======================================================================
|
|
|
|
static Standard_Boolean IsOriginalPnt (const gp_Pnt& P,
|
|
const Extrema_POnSurf* Points,
|
|
const Standard_Integer NbPoints)
|
|
{
|
|
for (Standard_Integer i=1; i<=NbPoints; i++) {
|
|
if (Points[i-1].Value().IsEqual(P, Precision::Confusion())) {
|
|
return Standard_False;
|
|
}
|
|
}
|
|
return Standard_True;
|
|
}
|
|
//=======================================================================
|
|
//function : IsExtremum
|
|
//purpose :
|
|
//=======================================================================
|
|
|
|
static Standard_Boolean IsExtremum (const Standard_Real U, const Standard_Real V,
|
|
const gp_Pnt& P, const Adaptor3d_SurfacePtr& S,
|
|
gp_Pnt& E, Standard_Real& Dist2,
|
|
const Standard_Boolean IsVSup,
|
|
const Standard_Boolean IsMin)
|
|
{
|
|
E = S->Value(U,V);
|
|
Dist2 = P.SquareDistance(E);
|
|
if (IsMin)
|
|
return (Dist2 < P.SquareDistance(S->Value(U+1,V)) &&
|
|
Dist2 < P.SquareDistance(S->Value(U-1,V)) &&
|
|
Dist2 < P.SquareDistance(S->Value(U, IsVSup ? V-1 : V+1)));
|
|
else
|
|
return (Dist2 > P.SquareDistance(S->Value(U+1,V)) &&
|
|
Dist2 > P.SquareDistance(S->Value(U-1,V)) &&
|
|
Dist2 > P.SquareDistance(S->Value(U, IsVSup ? V-1 : V+1)));
|
|
}
|
|
//=======================================================================
|
|
//function : Extrema_ExtPRevS
|
|
//purpose :
|
|
//=======================================================================
|
|
|
|
Extrema_ExtPRevS::Extrema_ExtPRevS()
|
|
{
|
|
myDone = Standard_False;
|
|
}
|
|
//=======================================================================
|
|
//function : Extrema_ExtPRevS
|
|
//purpose :
|
|
//=======================================================================
|
|
|
|
Extrema_ExtPRevS::Extrema_ExtPRevS (const gp_Pnt& theP,
|
|
const Handle(GeomAdaptor_HSurfaceOfRevolution)& theS,
|
|
const Standard_Real theUmin,
|
|
const Standard_Real theUsup,
|
|
const Standard_Real theVmin,
|
|
const Standard_Real theVsup,
|
|
const Standard_Real theTolU,
|
|
const Standard_Real theTolV)
|
|
{
|
|
Initialize (theS,
|
|
theUmin,
|
|
theUsup,
|
|
theVmin,
|
|
theVsup,
|
|
theTolU,
|
|
theTolV);
|
|
|
|
Perform (theP);
|
|
}
|
|
//=======================================================================
|
|
//function : Extrema_ExtPRevS
|
|
//purpose :
|
|
//=======================================================================
|
|
|
|
Extrema_ExtPRevS::Extrema_ExtPRevS (const gp_Pnt& theP,
|
|
const Handle(GeomAdaptor_HSurfaceOfRevolution)& theS,
|
|
const Standard_Real theTolU,
|
|
const Standard_Real theTolV)
|
|
{
|
|
Initialize (theS,
|
|
theS->FirstUParameter(),
|
|
theS->LastUParameter(),
|
|
theS->FirstVParameter(),
|
|
theS->LastVParameter(),
|
|
theTolU,
|
|
theTolV);
|
|
|
|
Perform (theP);
|
|
}
|
|
//=======================================================================
|
|
//function : Initialize
|
|
//purpose :
|
|
//=======================================================================
|
|
|
|
void Extrema_ExtPRevS::Initialize (const Handle(GeomAdaptor_HSurfaceOfRevolution)& theS,
|
|
const Standard_Real theUmin,
|
|
const Standard_Real theUsup,
|
|
const Standard_Real theVmin,
|
|
const Standard_Real theVsup,
|
|
const Standard_Real theTolU,
|
|
const Standard_Real theTolV)
|
|
{
|
|
myvinf = theVmin;
|
|
myvsup = theVsup;
|
|
mytolv = theTolV;
|
|
|
|
Handle(Adaptor3d_HCurve) anACurve = theS->BasisCurve();
|
|
|
|
if (myS != theS)
|
|
{
|
|
myS = theS;
|
|
myPosition = GetPosition (theS->ChangeSurface());
|
|
myIsAnalyticallyComputable =
|
|
IsCaseAnalyticallyComputable (anACurve->GetType(), myPosition, theS->AxeOfRevolution());
|
|
}
|
|
|
|
if (!myIsAnalyticallyComputable)
|
|
{
|
|
Standard_Integer aNbu = 32, aNbv = 32;
|
|
|
|
if (HasSingularity (theS->ChangeSurface()))
|
|
{
|
|
aNbv = 100;
|
|
}
|
|
|
|
myExtPS.Initialize (theS->ChangeSurface(),
|
|
aNbu,
|
|
aNbv,
|
|
theUmin,
|
|
theUsup,
|
|
theVmin,
|
|
theVsup,
|
|
theTolU,
|
|
theTolV);
|
|
}
|
|
}
|
|
//=======================================================================
|
|
//function : Perform
|
|
//purpose :
|
|
//=======================================================================
|
|
|
|
void Extrema_ExtPRevS::Perform(const gp_Pnt& P)
|
|
{
|
|
myDone = Standard_False;
|
|
myNbExt = 0;
|
|
|
|
if (!myIsAnalyticallyComputable) {
|
|
|
|
myExtPS.Perform(P);
|
|
myDone = myExtPS.IsDone();
|
|
myNbExt = myExtPS.NbExt();
|
|
return;
|
|
}
|
|
|
|
Handle(Adaptor3d_HCurve) anACurve = myS->BasisCurve();
|
|
|
|
gp_Ax1 Ax = myS->AxeOfRevolution();
|
|
gp_Vec Dir = Ax.Direction(), Z = myPosition.Direction();
|
|
gp_Pnt O = Ax.Location();
|
|
|
|
Standard_Real OPdir = gp_Vec(O,P).Dot(Dir);
|
|
gp_Pnt Pp = P.Translated(Dir.Multiplied(-OPdir));
|
|
if (O.IsEqual(Pp,Precision::Confusion())) // P is on the AxeOfRevolution
|
|
return;
|
|
|
|
Standard_Real U,V;
|
|
gp_Pnt P1, Ppp;
|
|
Standard_Real OPpz = gp_Vec(O,Pp).Dot(Z);
|
|
if (Abs(OPpz) <= gp::Resolution()) {
|
|
Ppp = Pp;
|
|
U = 0;
|
|
}
|
|
else {
|
|
Ppp = Pp.Translated(Z.Multiplied(-OPpz));
|
|
if (O.IsEqual(Ppp,Precision::Confusion()))
|
|
U = M_PI/2;
|
|
else {
|
|
U = gp_Vec(O,Ppp).AngleWithRef(gp_Vec(O,Pp),Dir);
|
|
}
|
|
}
|
|
|
|
gp_Vec OPpp (O,Ppp), OPq (O, myS->Value(M_PI/2,0));
|
|
if (U != M_PI/2) {
|
|
if (Abs(OPq.Magnitude()) <= gp::Resolution())
|
|
OPq = gp_Vec(O, myS->Value(M_PI/2,anACurve->LastParameter()/10));
|
|
if (OPpp.AngleWithRef(OPq,Dir) < 0)
|
|
U += M_PI;
|
|
}
|
|
|
|
gp_Trsf T;
|
|
T.SetRotation(Ax, -U);
|
|
P1 = P.Transformed(T);
|
|
|
|
gp_Pnt E;
|
|
Standard_Real Dist2;
|
|
Standard_Integer i;
|
|
|
|
Extrema_ExtPElC anExt;
|
|
PerformExtPElC(anExt, P1, anACurve, mytolv);
|
|
|
|
if (anExt.IsDone()) {
|
|
myDone = Standard_True;
|
|
for (i=1; i<=anExt.NbExt(); i++) {
|
|
Extrema_POnCurv POC=anExt.Point(i);
|
|
V = POC.Parameter();
|
|
if (V > myvsup) {
|
|
// if ( !IsExtremum (U, V = myvsup, P, myS, E, Dist2,
|
|
// Standard_True, anExt.IsMin(i))) continue;
|
|
Standard_Real newV = myvsup;
|
|
|
|
if((anACurve->GetType() == GeomAbs_Circle) ||
|
|
(anACurve->GetType() == GeomAbs_Ellipse)) {
|
|
newV = ElCLib::InPeriod(V, myvinf, myvinf + 2. * M_PI);
|
|
|
|
if (newV > myvsup) {
|
|
newV -= 2. * M_PI;
|
|
|
|
if (newV + mytolv < myvinf) {
|
|
newV = myvsup;
|
|
} else if (newV < myvinf) {
|
|
newV = myvinf;
|
|
}
|
|
}
|
|
}
|
|
V = newV;
|
|
|
|
if ( !IsExtremum (U, V, P, &(myS->ChangeSurface()), E, Dist2,
|
|
Standard_True, anExt.IsMin(i))) {
|
|
continue;
|
|
}
|
|
} else if (V < myvinf) {
|
|
// if ( !IsExtremum (U, V = myvinf, P, myS, E, Dist2,
|
|
// Standard_False, anExt.IsMin(i))) continue;
|
|
|
|
Standard_Real newV = myvinf;
|
|
|
|
if((anACurve->GetType() == GeomAbs_Circle) ||
|
|
(anACurve->GetType() == GeomAbs_Ellipse)) {
|
|
newV = ElCLib::InPeriod(V, myvsup - 2. * M_PI, myvsup);
|
|
|
|
if(newV < myvinf) {
|
|
newV += 2. * M_PI;
|
|
|
|
if (newV - mytolv > myvsup) {
|
|
newV = myvinf;
|
|
} else if (newV > myvsup) {
|
|
newV = myvsup;
|
|
}
|
|
}
|
|
}
|
|
V = newV;
|
|
|
|
if ( !IsExtremum (U, V, P, &(myS->ChangeSurface()), E, Dist2,
|
|
Standard_False, anExt.IsMin(i))) continue;
|
|
} else {
|
|
E = myS->Value(U,V);
|
|
Dist2 = P.SquareDistance(E);
|
|
}
|
|
if (IsOriginalPnt(E, myPoint, myNbExt)) {
|
|
myPoint[myNbExt] = Extrema_POnSurf(U,V,E);
|
|
mySqDist[myNbExt] = Dist2;
|
|
myNbExt++;
|
|
}
|
|
}
|
|
}
|
|
T.SetRotation(Ax, M_PI);
|
|
P1.Transform(T);
|
|
|
|
PerformExtPElC(anExt, P1, anACurve, mytolv);
|
|
if (anExt.IsDone()) {
|
|
myDone = Standard_True;
|
|
|
|
U += M_PI;
|
|
|
|
for (i=1; i<=anExt.NbExt(); i++) {
|
|
Extrema_POnCurv POC=anExt.Point(i);
|
|
V = POC.Parameter();
|
|
if (V > myvsup) {
|
|
// if ( !IsExtremum (U, V = myvsup, P, myS, E, Dist2,
|
|
// Standard_True, anExt.IsMin(i))) continue;
|
|
|
|
Standard_Real newV = myvsup;
|
|
|
|
if((anACurve->GetType() == GeomAbs_Circle) ||
|
|
(anACurve->GetType() == GeomAbs_Ellipse)) {
|
|
newV = ElCLib::InPeriod(V, myvinf, myvinf + 2. * M_PI);
|
|
|
|
if (newV > myvsup) {
|
|
newV -= 2. * M_PI;
|
|
|
|
if (newV + mytolv < myvinf) {
|
|
newV = myvsup;
|
|
} else if (newV < myvinf) {
|
|
newV = myvinf;
|
|
}
|
|
}
|
|
}
|
|
V = newV;
|
|
|
|
if ( !IsExtremum (U, V, P, &(myS->ChangeSurface()), E, Dist2,
|
|
Standard_True, anExt.IsMin(i))) continue;
|
|
} else if (V < myvinf) {
|
|
// if ( !IsExtremum (U, V = myvinf, P, myS, E, Dist2,
|
|
// Standard_False, anExt.IsMin(i))) continue;
|
|
Standard_Real newV = myvinf;
|
|
|
|
if((anACurve->GetType() == GeomAbs_Circle) ||
|
|
(anACurve->GetType() == GeomAbs_Ellipse)) {
|
|
newV = ElCLib::InPeriod(V, myvsup - 2. * M_PI, myvsup);
|
|
|
|
if(newV < myvinf) {
|
|
newV += 2. * M_PI;
|
|
|
|
if (newV - mytolv > myvsup) {
|
|
newV = myvinf;
|
|
} else if (newV > myvsup) {
|
|
newV = myvsup;
|
|
}
|
|
}
|
|
}
|
|
V = newV;
|
|
|
|
if ( !IsExtremum (U, V, P, &(myS->ChangeSurface()), E, Dist2,
|
|
Standard_False, anExt.IsMin(i))) continue;
|
|
} else {
|
|
E = myS->Value(U,V);
|
|
Dist2 = P.SquareDistance(E);
|
|
}
|
|
if (IsOriginalPnt(E, myPoint, myNbExt)) {
|
|
myPoint[myNbExt] = Extrema_POnSurf(U,V,E);
|
|
mySqDist[myNbExt] = Dist2;
|
|
myNbExt++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
//=======================================================================
|
|
//function : IsDone
|
|
//purpose :
|
|
//=======================================================================
|
|
|
|
Standard_Boolean Extrema_ExtPRevS::IsDone() const
|
|
{
|
|
return myDone;
|
|
}
|
|
|
|
|
|
//=======================================================================
|
|
//function : NbExt
|
|
//purpose :
|
|
//=======================================================================
|
|
|
|
Standard_Integer Extrema_ExtPRevS::NbExt() const
|
|
{
|
|
if (!IsDone()) { throw StdFail_NotDone(); }
|
|
return myNbExt;
|
|
}
|
|
|
|
|
|
//=======================================================================
|
|
//function : Value
|
|
//purpose :
|
|
//=======================================================================
|
|
|
|
Standard_Real Extrema_ExtPRevS::SquareDistance(const Standard_Integer N) const
|
|
{
|
|
if (!IsDone()) { throw StdFail_NotDone(); }
|
|
if ((N < 1) || (N > myNbExt)) { throw Standard_OutOfRange(); }
|
|
if (myIsAnalyticallyComputable)
|
|
return mySqDist[N-1];
|
|
else
|
|
return myExtPS.SquareDistance(N);
|
|
}
|
|
//=======================================================================
|
|
//function : Point
|
|
//purpose :
|
|
//=======================================================================
|
|
|
|
const Extrema_POnSurf& Extrema_ExtPRevS::Point(const Standard_Integer N) const
|
|
{
|
|
if (!IsDone()) { throw StdFail_NotDone(); }
|
|
if ((N < 1) || (N > myNbExt)) { throw Standard_OutOfRange(); }
|
|
if (myIsAnalyticallyComputable)
|
|
return myPoint[N-1];
|
|
else
|
|
return myExtPS.Point(N);
|
|
}
|
|
|
|
|