git-svn-id: https://free-cad.svn.sourceforge.net/svnroot/free-cad/trunk@5000 e8eeb9e2-ec13-0410-a4a9-efa5cf37419d
520 lines
14 KiB
C++
520 lines
14 KiB
C++
/***************************************************************************
|
|
* Copyright (c) 2007 *
|
|
* Joachim Zettler <Joachim.Zettler@gmx.de> *
|
|
* Adapted by Joachim Zettler to use with a WireExplorer made *
|
|
* by Stephane Routelous *
|
|
* *
|
|
* This file is part of the FreeCAD CAx development system. *
|
|
* *
|
|
* This library is free software; you can redistribute it and/or *
|
|
* modify it under the terms of the GNU Library General Public *
|
|
* License as published by the Free Software Foundation; either *
|
|
* version 2 of the License, or (at your option) any later version. *
|
|
* *
|
|
* This library is distributed in the hope that it will be useful, *
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of *
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
|
|
* GNU Library General Public License for more details. *
|
|
* *
|
|
* You should have received a copy of the GNU Library General Public *
|
|
* License along with this library; see the file COPYING.LIB. If not, *
|
|
* write to the Free Software Foundation, Inc., 59 Temple Place, *
|
|
* Suite 330, Boston, MA 02111-1307, USA *
|
|
* *
|
|
***************************************************************************/
|
|
|
|
//
|
|
|
|
#include "PreCompiled.h"
|
|
|
|
#include "BRepAdaptor_CompCurve2.h"
|
|
#include <BRepAdaptor_Curve.hxx>
|
|
#include <BRepAdaptor_HCurve.hxx>
|
|
#include <BRepAdaptor_HCompCurve.hxx>
|
|
#include <BRepAdaptor_HArray1OfCurve.hxx>
|
|
#include <TColStd_HArray1OfReal.hxx>
|
|
#include <BRep_Tool.hxx>
|
|
|
|
#include <TopAbs_Orientation.hxx>
|
|
|
|
#include <GCPnts_AbscissaPoint.hxx>
|
|
#include <ElCLib.hxx>
|
|
#include <TopExp.hxx>
|
|
#include <TopoDS_Vertex.hxx>
|
|
#include "WireExplorer.h"
|
|
|
|
BRepAdaptor_CompCurve2::BRepAdaptor_CompCurve2()
|
|
{
|
|
}
|
|
|
|
BRepAdaptor_CompCurve2::BRepAdaptor_CompCurve2(const TopoDS_Wire& W,
|
|
const Standard_Boolean AC)
|
|
{
|
|
Initialize(W, AC);
|
|
}
|
|
|
|
BRepAdaptor_CompCurve2::BRepAdaptor_CompCurve2(const TopoDS_Wire& W,
|
|
const Standard_Boolean AC,
|
|
const Standard_Real First,
|
|
const Standard_Real Last,
|
|
const Standard_Real Tol)
|
|
{
|
|
Initialize(W, AC, First, Last, Tol);
|
|
}
|
|
|
|
void BRepAdaptor_CompCurve2::Initialize(const TopoDS_Wire& W,
|
|
const Standard_Boolean AC)
|
|
{
|
|
Standard_Integer ii, NbEdge;
|
|
|
|
TopoDS_Edge E;
|
|
|
|
myWire = W;
|
|
WireExplorer wexp(myWire);
|
|
PTol = 0.0;
|
|
IsbyAC = AC;
|
|
|
|
for (NbEdge=0, wexp.Init();wexp.More(); wexp.Next())
|
|
if (! BRep_Tool::Degenerated(wexp.Current())) NbEdge++;
|
|
|
|
if (NbEdge == 0) return;
|
|
|
|
CurIndex = (NbEdge+1)/2;
|
|
myCurves = new BRepAdaptor_HArray1OfCurve(1,NbEdge);
|
|
myKnots = new (TColStd_HArray1OfReal) (1,NbEdge+1);
|
|
myKnots->SetValue(1, 0.);
|
|
|
|
for (ii=0, wexp.Init();wexp.More(); wexp.Next()) {
|
|
E = wexp.Current();
|
|
if (! BRep_Tool::Degenerated(E)) {
|
|
ii++;
|
|
myCurves->ChangeValue(ii).Initialize(E);
|
|
if (AC) {
|
|
myKnots->SetValue(ii+1, myKnots->Value(ii));
|
|
myKnots->ChangeValue(ii+1) +=
|
|
GCPnts_AbscissaPoint::Length(myCurves->ChangeValue(ii));
|
|
}
|
|
else myKnots->SetValue(ii+1, (Standard_Real)ii);
|
|
}
|
|
}
|
|
|
|
Forward = Standard_True; // Defaut ; Les Edge Reverse seront parcourue
|
|
// a rebourt.
|
|
if((NbEdge > 2) || ((NbEdge==2) && (!myWire.Closed())) ) {
|
|
TopAbs_Orientation Or = myCurves->Value(1).Edge().Orientation();
|
|
Standard_Boolean B;
|
|
TopoDS_Vertex VI, VL;
|
|
B = TopExp::CommonVertex(myCurves->Value(1).Edge(),
|
|
myCurves->Value(2).Edge(),
|
|
VI);
|
|
VL = TopExp::LastVertex(myCurves->Value(1).Edge());
|
|
if (VI.IsSame(VL)) { // On Garde toujours le sens de parcout
|
|
if (Or == TopAbs_REVERSED)
|
|
Forward = Standard_False;
|
|
}
|
|
else {// On renverse toujours le sens de parcout
|
|
if (Or != TopAbs_REVERSED)
|
|
Forward = Standard_False;
|
|
}
|
|
}
|
|
|
|
TFirst = 0;
|
|
TLast = myKnots->Value(myKnots->Length());
|
|
myPeriod = TLast - TFirst;
|
|
if (NbEdge == 1) {
|
|
Periodic = myCurves->Value(1).IsPeriodic();
|
|
}
|
|
else {
|
|
Periodic = Standard_False;
|
|
}
|
|
}
|
|
|
|
void BRepAdaptor_CompCurve2::Initialize(const TopoDS_Wire& W,
|
|
const Standard_Boolean AC,
|
|
const Standard_Real First,
|
|
const Standard_Real Last,
|
|
const Standard_Real Tol)
|
|
{
|
|
Initialize(W, AC);
|
|
TFirst = First;
|
|
TLast = Last;
|
|
PTol = Tol;
|
|
|
|
// Trim des courbes extremes.
|
|
Handle (BRepAdaptor_HCurve) HC;
|
|
Standard_Integer i1, i2;
|
|
Standard_Real f=TFirst, l=TLast, d;
|
|
i1 = i2 = CurIndex;
|
|
Prepare(f, d, i1);
|
|
Prepare(l, d, i2);
|
|
CurIndex = (i1+i2)/2; // Petite optimisation
|
|
if (i1==i2) {
|
|
if (l > f)
|
|
HC = Handle(BRepAdaptor_HCurve)::DownCast(myCurves->Value(i1).Trim(f, l, PTol));
|
|
else
|
|
HC = Handle(BRepAdaptor_HCurve)::DownCast(myCurves->Value(i1).Trim(l, f, PTol));
|
|
myCurves->SetValue(i1, HC->ChangeCurve());
|
|
}
|
|
else {
|
|
const BRepAdaptor_Curve& c1 = myCurves->Value(i1);
|
|
const BRepAdaptor_Curve& c2 = myCurves->Value(i2);
|
|
Standard_Real k;
|
|
|
|
k = c1.LastParameter();
|
|
if (k>f)
|
|
HC = Handle(BRepAdaptor_HCurve)::DownCast(c1.Trim(f, k, PTol));
|
|
else
|
|
HC = Handle(BRepAdaptor_HCurve)::DownCast(c1.Trim(k, f, PTol));
|
|
myCurves->SetValue(i1, HC->ChangeCurve());
|
|
|
|
k = c2.FirstParameter();
|
|
if (k<=l)
|
|
HC = Handle(BRepAdaptor_HCurve)::DownCast(c2.Trim(k, l, PTol));
|
|
else
|
|
HC = Handle(BRepAdaptor_HCurve)::DownCast(c2.Trim(l, k, PTol));
|
|
myCurves->SetValue(i2, HC->ChangeCurve());
|
|
}
|
|
}
|
|
|
|
|
|
void BRepAdaptor_CompCurve2::SetPeriodic(const Standard_Boolean isPeriodic)
|
|
{
|
|
if (myWire.Closed()) {
|
|
Periodic = isPeriodic;
|
|
}
|
|
}
|
|
|
|
|
|
const TopoDS_Wire& BRepAdaptor_CompCurve2::Wire() const
|
|
{
|
|
return myWire;
|
|
}
|
|
|
|
void BRepAdaptor_CompCurve2::Edge(const Standard_Real U,
|
|
TopoDS_Edge& E,
|
|
Standard_Real& UonE) const
|
|
{
|
|
Standard_Real d;
|
|
Standard_Integer index = CurIndex;
|
|
UonE = U;
|
|
Prepare(UonE, d, index);
|
|
E = myCurves->Value(index).Edge();
|
|
}
|
|
|
|
Standard_Real BRepAdaptor_CompCurve2::FirstParameter() const
|
|
{
|
|
return TFirst;
|
|
}
|
|
|
|
Standard_Real BRepAdaptor_CompCurve2::LastParameter() const
|
|
{
|
|
return TLast;
|
|
}
|
|
|
|
GeomAbs_Shape BRepAdaptor_CompCurve2::Continuity() const
|
|
{
|
|
if ( myCurves->Length() > 1) return GeomAbs_C0;
|
|
return myCurves->Value(1).Continuity();
|
|
}
|
|
|
|
Standard_Integer BRepAdaptor_CompCurve2::NbIntervals(const GeomAbs_Shape S)
|
|
{
|
|
Standard_Integer NbInt, ii;
|
|
for (ii=1, NbInt=0; ii<=myCurves->Length(); ii++)
|
|
NbInt += myCurves->ChangeValue(ii).NbIntervals(S);
|
|
|
|
return NbInt;
|
|
}
|
|
|
|
void BRepAdaptor_CompCurve2::Intervals(TColStd_Array1OfReal& T,
|
|
const GeomAbs_Shape S)
|
|
{
|
|
Standard_Integer ii, jj, kk, n;
|
|
Standard_Real f, F, delta;
|
|
|
|
// Premiere courbe (sens de parcourt de le edge)
|
|
n = myCurves->ChangeValue(1).NbIntervals(S);
|
|
Handle(TColStd_HArray1OfReal) Ti = new (TColStd_HArray1OfReal) (1, n+1);
|
|
myCurves->ChangeValue(1).Intervals(Ti->ChangeArray1(), S);
|
|
InvPrepare(1, f, delta);
|
|
F = myKnots->Value(1);
|
|
if (delta < 0) {
|
|
//sens de parcourt inverser
|
|
for (kk=1,jj=Ti->Length(); jj>0; kk++, jj--)
|
|
T(kk) = F + (Ti->Value(jj)-f)*delta;
|
|
}
|
|
else {
|
|
for (kk=1; kk<=Ti->Length(); kk++)
|
|
T(kk) = F + (Ti->Value(kk)-f)*delta;
|
|
}
|
|
|
|
// et les suivante
|
|
for (ii=2; ii<=myCurves->Length(); ii++) {
|
|
n = myCurves->ChangeValue(ii).NbIntervals(S);
|
|
if (n != Ti->Length()-1) Ti = new (TColStd_HArray1OfReal) (1, n+1);
|
|
myCurves->ChangeValue(ii).Intervals(Ti->ChangeArray1(), S);
|
|
InvPrepare(ii, f, delta);
|
|
F = myKnots->Value(ii);
|
|
if (delta < 0) {
|
|
//sens de parcourt inverser
|
|
for (jj=Ti->Length()-1; jj>0; kk++, jj--)
|
|
T(kk) = F + (Ti->Value(jj)-f)*delta;
|
|
}
|
|
else {
|
|
for (jj=2; jj<=Ti->Length(); kk++, jj++)
|
|
T(kk) = F + (Ti->Value(jj)-f)*delta;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
Standard_Boolean BRepAdaptor_CompCurve2::IsClosed() const
|
|
{
|
|
return myWire.Closed();
|
|
}
|
|
|
|
Standard_Boolean BRepAdaptor_CompCurve2::IsPeriodic() const
|
|
{
|
|
return Periodic;
|
|
|
|
}
|
|
|
|
Standard_Real BRepAdaptor_CompCurve2::Period() const
|
|
{
|
|
return myPeriod;
|
|
}
|
|
|
|
gp_Pnt BRepAdaptor_CompCurve2::Value(const Standard_Real U) const
|
|
{
|
|
Standard_Real u = U, d;
|
|
Standard_Integer index = CurIndex;
|
|
Prepare(u, d, index);
|
|
return myCurves->Value(index).Value(u);
|
|
}
|
|
|
|
void BRepAdaptor_CompCurve2::D0(const Standard_Real U,
|
|
gp_Pnt& P) const
|
|
{
|
|
Standard_Real u = U, d;
|
|
Standard_Integer index = CurIndex;
|
|
Prepare(u, d, index);
|
|
myCurves->Value(index).D0(u, P);
|
|
}
|
|
|
|
void BRepAdaptor_CompCurve2::D1(const Standard_Real U,
|
|
gp_Pnt& P,
|
|
gp_Vec& V) const
|
|
{
|
|
Standard_Real u = U, d;
|
|
Standard_Integer index = CurIndex;
|
|
Prepare(u, d, index);
|
|
myCurves->Value(index).D1(u, P, V);
|
|
V*=d;
|
|
}
|
|
|
|
void BRepAdaptor_CompCurve2::D2(const Standard_Real U,
|
|
gp_Pnt& P,
|
|
gp_Vec& V1,
|
|
gp_Vec& V2) const
|
|
{
|
|
Standard_Real u = U, d;
|
|
Standard_Integer index = CurIndex;
|
|
Prepare(u, d, index);
|
|
myCurves->Value(index).D2(u, P, V1, V2);
|
|
V1*=d;
|
|
V2 *= d*d;
|
|
}
|
|
|
|
void BRepAdaptor_CompCurve2::D3(const Standard_Real U,
|
|
gp_Pnt& P,gp_Vec& V1,
|
|
gp_Vec& V2,
|
|
gp_Vec& V3) const
|
|
{
|
|
Standard_Real u = U, d;
|
|
Standard_Integer index = CurIndex;
|
|
Prepare(u, d, index);
|
|
myCurves->Value(index).D3(u, P, V1, V2, V3);
|
|
V1*=d;
|
|
V2 *= d*d;
|
|
V3 *= d*d*d;
|
|
}
|
|
|
|
gp_Vec BRepAdaptor_CompCurve2::DN(const Standard_Real U,
|
|
const Standard_Integer N) const
|
|
{
|
|
Standard_Real u = U, d;
|
|
Standard_Integer index = CurIndex;
|
|
Prepare(u, d, index);
|
|
|
|
return (myCurves->Value(index).DN(u, N) * Pow(d, N));
|
|
}
|
|
|
|
Standard_Real BRepAdaptor_CompCurve2::Resolution(const Standard_Real R3d) const
|
|
{
|
|
Standard_Real Res = 1.e200, r;
|
|
Standard_Integer ii, L = myCurves->Length();
|
|
for (ii=1; ii<=L; ii++) {
|
|
r = myCurves->Value(ii).Resolution(R3d);
|
|
if (r < Res) Res = r;
|
|
}
|
|
return Res;
|
|
}
|
|
|
|
GeomAbs_CurveType BRepAdaptor_CompCurve2::GetType() const
|
|
{
|
|
return GeomAbs_OtherCurve; //temporaire
|
|
// if ( myCurves->Length() > 1) return GeomAbs_OtherCurve;
|
|
// return myCurves->Value(1).GetType();
|
|
}
|
|
|
|
gp_Lin BRepAdaptor_CompCurve2::Line() const
|
|
{
|
|
return myCurves->Value(1).Line();
|
|
}
|
|
|
|
gp_Circ BRepAdaptor_CompCurve2::Circle() const
|
|
{
|
|
return myCurves->Value(1).Circle();
|
|
}
|
|
|
|
gp_Elips BRepAdaptor_CompCurve2::Ellipse() const
|
|
{
|
|
return myCurves->Value(1).Ellipse();
|
|
}
|
|
|
|
gp_Hypr BRepAdaptor_CompCurve2::Hyperbola() const
|
|
{
|
|
return myCurves->Value(1).Hyperbola();
|
|
}
|
|
|
|
gp_Parab BRepAdaptor_CompCurve2::Parabola() const
|
|
{
|
|
return myCurves->Value(1).Parabola();
|
|
}
|
|
|
|
Standard_Integer BRepAdaptor_CompCurve2::Degree() const
|
|
{
|
|
return myCurves->Value(1).Degree();
|
|
}
|
|
|
|
Standard_Boolean BRepAdaptor_CompCurve2::IsRational() const
|
|
{
|
|
return myCurves->Value(1).IsRational();
|
|
}
|
|
|
|
Standard_Integer BRepAdaptor_CompCurve2::NbPoles() const
|
|
{
|
|
return myCurves->Value(1).NbPoles();
|
|
}
|
|
|
|
Standard_Integer BRepAdaptor_CompCurve2::NbKnots() const
|
|
{
|
|
return myCurves->Value(1).NbKnots();
|
|
}
|
|
|
|
Handle(Geom_BezierCurve) BRepAdaptor_CompCurve2::Bezier() const
|
|
{
|
|
return myCurves->Value(1).Bezier();
|
|
}
|
|
|
|
Handle(Geom_BSplineCurve) BRepAdaptor_CompCurve2::BSpline() const
|
|
{
|
|
return myCurves->Value(1).BSpline();
|
|
}
|
|
|
|
//=======================================================================
|
|
//function : Prepare
|
|
//purpose :
|
|
// Lorsque le parametre est pres de un "noeud" on determine la loi en
|
|
// fonction du signe de tol:
|
|
// - negatif -> Loi precedente au noeud.
|
|
// - positif -> Loi consecutive au noeud.
|
|
//=======================================================================
|
|
|
|
void BRepAdaptor_CompCurve2::Prepare(Standard_Real& W,
|
|
Standard_Real& Delta,
|
|
Standard_Integer& CurIndex) const
|
|
{
|
|
Standard_Real f,l, Wtest, Eps;
|
|
Standard_Integer ii;
|
|
if (W-TFirst < TLast-W) { Eps = PTol; }
|
|
else { Eps = -PTol;}
|
|
|
|
|
|
Wtest = W+Eps; //Decalage pour discriminer les noeuds
|
|
if(Periodic){
|
|
Wtest = ElCLib::InPeriod(Wtest,
|
|
0,
|
|
myPeriod);
|
|
W = Wtest-Eps;
|
|
}
|
|
|
|
// Recheche de le index
|
|
Standard_Boolean Trouve = Standard_False;
|
|
if (myKnots->Value(CurIndex) > Wtest) {
|
|
for (ii=CurIndex-1; ii>0 && !Trouve; ii--)
|
|
if (myKnots->Value(ii)<= Wtest) {
|
|
CurIndex = ii;
|
|
Trouve = Standard_True;
|
|
}
|
|
if (!Trouve) CurIndex = 1; // En dehors des bornes ...
|
|
}
|
|
|
|
else if (myKnots->Value(CurIndex+1) <= Wtest) {
|
|
for (ii=CurIndex+1; ii<=myCurves->Length() && !Trouve; ii++)
|
|
if (myKnots->Value(ii+1)> Wtest) {
|
|
CurIndex = ii;
|
|
Trouve = Standard_True;
|
|
}
|
|
if (!Trouve) CurIndex = myCurves->Length(); // En dehors des bornes ...
|
|
}
|
|
|
|
// Reverse ?
|
|
const TopoDS_Edge& E = myCurves->Value(CurIndex).Edge();
|
|
TopAbs_Orientation Or = E.Orientation();
|
|
Standard_Boolean Reverse;
|
|
Reverse = (Forward && (Or == TopAbs_REVERSED)) ||
|
|
(!Forward && (Or != TopAbs_REVERSED));
|
|
|
|
// Calcul du parametre local
|
|
BRep_Tool::Range(E, f, l);
|
|
Delta = myKnots->Value(CurIndex+1) - myKnots->Value(CurIndex);
|
|
if (Delta > PTol*1.e-9) Delta = (l-f)/Delta;
|
|
|
|
if (Reverse) {
|
|
Delta *= -1;
|
|
W = l + (W-myKnots->Value(CurIndex)) * Delta;
|
|
}
|
|
else {
|
|
W = f + (W-myKnots->Value(CurIndex)) * Delta;
|
|
}
|
|
}
|
|
|
|
void BRepAdaptor_CompCurve2::InvPrepare(const Standard_Integer index,
|
|
Standard_Real& First,
|
|
Standard_Real& Delta) const
|
|
{
|
|
// Reverse ?
|
|
const TopoDS_Edge& E = myCurves->Value(index).Edge();
|
|
TopAbs_Orientation Or = E.Orientation();
|
|
Standard_Boolean Reverse;
|
|
Reverse = (Forward && (Or == TopAbs_REVERSED)) ||
|
|
(!Forward && (Or != TopAbs_REVERSED));
|
|
|
|
// Calcul des parametres de reparametrisation
|
|
// tel que : T = Ti + (t-First)*Delta
|
|
Standard_Real f, l;
|
|
BRep_Tool::Range(E, f, l);
|
|
Delta = myKnots->Value(index+1) - myKnots->Value(index);
|
|
if (l-f > PTol*1.e-9) Delta /= (l-f);
|
|
|
|
if (Reverse) {
|
|
Delta *= -1;
|
|
First = l;
|
|
}
|
|
else {
|
|
First = f;
|
|
}
|
|
}
|