/*************************************************************************** * Copyright (c) 2015 WandererFan * * * * 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" #ifndef _PreComp_ # include # include # include #include # include # include # include # include # include #include #include #include #include #include #include #include #include #include #include #include #include #include #endif #include #include #include #include #include #include "DrawUtil.h" using namespace TechDraw; /*static*/ int DrawUtil::getIndexFromName(std::string geomName) { boost::regex re("\\d+$"); // one of more digits at end of string boost::match_results what; boost::match_flag_type flags = boost::match_default; char* endChar; std::string::const_iterator begin = geomName.begin(); std::string::const_iterator end = geomName.end(); std::stringstream ErrorMsg; if (!geomName.empty()) { if (boost::regex_search(begin, end, what, re, flags)) { return int (std::strtol(what.str().c_str(), &endChar, 10)); //TODO: use std::stoi() in c++11 } else { ErrorMsg << "getIndexFromName: malformed geometry name - " << geomName; throw Base::Exception(ErrorMsg.str()); } } else { throw Base::Exception("getIndexFromName - empty geometry name"); } } std::string DrawUtil::getGeomTypeFromName(std::string geomName) { boost::regex re("^[a-zA-Z]*"); //one or more letters at start of string boost::match_results what; boost::match_flag_type flags = boost::match_default; std::string::const_iterator begin = geomName.begin(); std::string::const_iterator end = geomName.end(); std::stringstream ErrorMsg; if (!geomName.empty()) { if (boost::regex_search(begin, end, what, re, flags)) { return what.str(); //TODO: use std::stoi() in c++11 } else { ErrorMsg << "In getGeomTypeFromName: malformed geometry name - " << geomName; throw Base::Exception(ErrorMsg.str()); } } else { throw Base::Exception("getGeomTypeFromName - empty geometry name"); } } std::string DrawUtil::makeGeomName(std::string geomType, int index) { std::stringstream newName; newName << geomType << index; return newName.str(); } bool DrawUtil::isSamePoint(TopoDS_Vertex v1, TopoDS_Vertex v2) { bool result = false; gp_Pnt p1 = BRep_Tool::Pnt(v1); gp_Pnt p2 = BRep_Tool::Pnt(v2); if (p1.IsEqual(p2,Precision::Confusion())) { result = true; } return result; } bool DrawUtil::isZeroEdge(TopoDS_Edge e) { TopoDS_Vertex vStart = TopExp::FirstVertex(e); TopoDS_Vertex vEnd = TopExp::LastVertex(e); bool result = isSamePoint(vStart,vEnd); if (result) { //closed edge will have same V's but non-zero length GProp_GProps props; BRepGProp::LinearProperties(e, props); double len = props.Mass(); if (len > Precision::Confusion()) { result = false; } } return result; } //! assumes 2d on XY //! quick angle for straight edges double DrawUtil::angleWithX(TopoDS_Edge e, bool reverse) { double result = 0; gp_Pnt gstart = BRep_Tool::Pnt(TopExp::FirstVertex(e)); Base::Vector3d start(gstart.X(),gstart.Y(),gstart.Z()); gp_Pnt gend = BRep_Tool::Pnt(TopExp::LastVertex(e)); Base::Vector3d end(gend.X(),gend.Y(),gend.Z()); Base::Vector3d u; if (reverse) { u = start - end; } else { u = end - start; } result = atan2(u.y,u.x); if (result < 0) { result += 2.0 * M_PI; } return result; } double DrawUtil::angleWithX(TopoDS_Edge e, TopoDS_Vertex v) { double result = 0; double param = 0; //find tangent @ v double adjust = 1.0; //occ tangent points in direction of curve. at lastVert we need to reverse it. BRepAdaptor_Curve adapt(e); if (isFirstVert(e,v)) { param = adapt.FirstParameter(); } else if (isLastVert(e,v)) { param = adapt.LastParameter(); adjust = -1; } else { //TARFU Base::Console().Message("Error: DU::angleWithX - v is neither first nor last \n"); //must be able to get non-terminal point parm from curve/ } Base::Vector3d uVec(0.0,0.0,0.0); gp_Dir uDir; BRepLProp_CLProps prop(adapt,param,2,Precision::Confusion()); if (prop.IsTangentDefined()) { prop.Tangent(uDir); uVec = Base::Vector3d(uDir.X(),uDir.Y(),uDir.Z()) * adjust; } else { //this bit is a little sketchy gp_Pnt gstart = BRep_Tool::Pnt(TopExp::FirstVertex(e)); Base::Vector3d start(gstart.X(),gstart.Y(),gstart.Z()); gp_Pnt gend = BRep_Tool::Pnt(TopExp::LastVertex(e)); Base::Vector3d end(gend.X(),gend.Y(),gend.Z()); if (isFirstVert(e,v)) { uVec = end - start; } else if (isLastVert(e,v)) { uVec = end - start; } else { gp_Pnt errPnt = BRep_Tool::Pnt(v); Base::Console().Warning("angleWithX: Tangent not defined at (%.3f,%.3f,%.3f)\n",errPnt.X(),errPnt.Y(),errPnt.Z()); //throw ?????? } } result = atan2(uVec.y,uVec.x); if (result < 0) { //map from [-PI:PI] to [0:2PI] result += 2.0 * M_PI; } return result; } bool DrawUtil::isFirstVert(TopoDS_Edge e, TopoDS_Vertex v) { bool result = false; TopoDS_Vertex first = TopExp::FirstVertex(e); if (isSamePoint(first,v)) { result = true; } return result; } bool DrawUtil::isLastVert(TopoDS_Edge e, TopoDS_Vertex v) { bool result = false; TopoDS_Vertex last = TopExp::LastVertex(e); if (isSamePoint(last,v)) { result = true; } return result; } bool DrawUtil::fpCompare(const double& d1, const double& d2) { bool result = false; if (std::fabs(d1 - d2) < FLT_EPSILON) { result = true; } return result; } Base::Vector3d DrawUtil::vertex2Vector(const TopoDS_Vertex& v) { gp_Pnt gp = BRep_Tool::Pnt(v); Base::Vector3d result(gp.X(),gp.Y(),gp.Z()); return result; } std::string DrawUtil::formatVector(const Base::Vector3d& v) { std::string result; std::stringstream builder; builder << " (" << v.x << "," << v.y << "," << v.z << ") "; result = builder.str(); return result; } //! compare 2 vectors for sorting purposes ( -1 -> v1 v1 == v2, 1 -> v1 > v2) int DrawUtil::vectorCompare(const Base::Vector3d& v1, const Base::Vector3d& v2) { int result = 0; if (v1 == v2) { return result; } if (v1.x < v2.x) { result = -1; } else if (DrawUtil::fpCompare(v1.x, v2.x)) { if (v1.y < v2.y) { result = -1; } else if (DrawUtil::fpCompare(v1.y, v2.y)) { if (v1.z < v2.z) { result = -1; } else { result = 1; } } else { result = 1; //v2y > v1y } } else { result = 1; //v1x > v2x } return result; } //based on Function provided by Joe Dowsett, 2014 double DrawUtil::sensibleScale(double working_scale) { double result = 1.0; //which gives the largest scale for which the min_space requirements can be met, but we want a 'sensible' scale, rather than 0.28457239... //eg if working_scale = 0.115, then we want to use 0.1, similarly 7.65 -> 5, and 76.5 -> 50 float exponent = std::floor(std::log10(working_scale)); //if working_scale = a * 10^b, what is b? working_scale *= std::pow(10, -exponent); //now find what 'a' is. //int choices = 10; float valid_scales[2][10] = {{1.0, 1.25, 2.0, 2.5, 3.75, 5.0, 7.5, 10.0, 50.0, 100.0}, //equate to 1:10, 1:8, 1:5, 1:4, 3:8, 1:2, 3:4, 1:1 // .1 .125 .375 .75 {1.0, 1.5 , 2.0, 3.0, 4.0 , 5.0, 8.0, 10.0, 50.0, 100.0}}; //equate to 1:1, 3:2, 2:1, 3:1, 4:1, 5:1, 8:1, 10:1 // 1.5:1 //int i = choices - 1; int i = 9; while (valid_scales[(exponent >= 0)][i] > working_scale) //choose closest value smaller than 'a' from list. i -= 1; //choosing top list if exponent -ve, bottom list for +ve exponent //now have the appropriate scale, reapply the *10^b result = valid_scales[(exponent >= 0)][i] * pow(10, exponent); return result; } //============================ // various debugging routines. void DrawUtil::dumpVertexes(const char* text, const TopoDS_Shape& s) { Base::Console().Message("DUMP - %s\n",text); TopExp_Explorer expl(s, TopAbs_VERTEX); int i; for (i = 1 ; expl.More(); expl.Next(),i++) { const TopoDS_Vertex& v = TopoDS::Vertex(expl.Current()); gp_Pnt pnt = BRep_Tool::Pnt(v); Base::Console().Message("v%d: (%.3f,%.3f,%.3f)\n",i,pnt.X(),pnt.Y(),pnt.Z()); } } void DrawUtil::countFaces(const char* text, const TopoDS_Shape& s) { TopTools_IndexedMapOfShape mapOfFaces; TopExp::MapShapes(s, TopAbs_FACE, mapOfFaces); int num = mapOfFaces.Extent(); Base::Console().Message("COUNT - %s has %d Faces\n",text,num); } //count # of unique Wires in shape. void DrawUtil::countWires(const char* text, const TopoDS_Shape& s) { TopTools_IndexedMapOfShape mapOfWires; TopExp::MapShapes(s, TopAbs_WIRE, mapOfWires); int num = mapOfWires.Extent(); Base::Console().Message("COUNT - %s has %d wires\n",text,num); } void DrawUtil::countEdges(const char* text, const TopoDS_Shape& s) { TopTools_IndexedMapOfShape mapOfEdges; TopExp::MapShapes(s, TopAbs_EDGE, mapOfEdges); int num = mapOfEdges.Extent(); Base::Console().Message("COUNT - %s has %d edges\n",text,num); } void DrawUtil::dump1Vertex(const char* text, const TopoDS_Vertex& v) { Base::Console().Message("DUMP - dump1Vertex - %s\n",text); gp_Pnt pnt = BRep_Tool::Pnt(v); Base::Console().Message("%s: (%.3f,%.3f,%.3f)\n",text,pnt.X(),pnt.Y(),pnt.Z()); } void DrawUtil::dumpEdge(char* label, int i, TopoDS_Edge e) { BRepAdaptor_Curve adapt(e); double start = BRepLProp_CurveTool::FirstParameter(adapt); double end = BRepLProp_CurveTool::LastParameter(adapt); BRepLProp_CLProps propStart(adapt,start,0,Precision::Confusion()); const gp_Pnt& vStart = propStart.Value(); BRepLProp_CLProps propEnd(adapt,end,0,Precision::Confusion()); const gp_Pnt& vEnd = propEnd.Value(); //Base::Console().Message("%s edge:%d start:(%.3f,%.3f,%.3f)/%0.3f end:(%.2f,%.3f,%.3f)/%.3f\n",label,i, // vStart.X(),vStart.Y(),vStart.Z(),start,vEnd.X(),vEnd.Y(),vEnd.Z(),end); Base::Console().Message("%s edge:%d start:(%.3f,%.3f,%.3f) end:(%.2f,%.3f,%.3f)\n",label,i, vStart.X(),vStart.Y(),vStart.Z(),vEnd.X(),vEnd.Y(),vEnd.Z()); } const char* DrawUtil::printBool(bool b) { return (b ? "True" : "False"); } QString DrawUtil::qbaToDebug(const QByteArray & line) { QString s; uchar c; for ( int i=0 ; i < line.size() ; i++ ){ c = line[i]; if (( c >= 0x20) && (c <= 126) ) { s.append(QChar::fromLatin1(c)); } else { s.append(QString::fromUtf8("<%1>").arg(c, 2, 16, QChar::fromLatin1('0'))); } } return s; } //==================================