# SPDX-License-Identifier: LGPL-2.1-or-later # *************************************************************************** # * Copyright (c) 2009, 2010 Yorik van Havre * # * Copyright (c) 2009, 2010 Ken Cline * # * Copyright (c) 2020 FreeCAD Developers * # * * # * This program is free software; you can redistribute it and/or modify * # * it under the terms of the GNU Lesser General Public License (LGPL) * # * as published by the Free Software Foundation; either version 2 of * # * the License, or (at your option) any later version. * # * for detail see the LICENCE text file. * # * * # * This program 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 program; if not, write to the Free Software * # * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 * # * USA * # * * # *************************************************************************** """Provides the object code for the BSpline object.""" ## @package bspline # \ingroup draftobjects # \brief Provides the object code for the BSpline object. ## \addtogroup draftobjects # @{ from PySide.QtCore import QT_TRANSLATE_NOOP import FreeCAD as App from draftobjects.base import DraftObject from draftutils import gui_utils from draftutils import params class BSpline(DraftObject): """The BSpline object""" def __init__(self, obj): super().__init__(obj, "BSpline") _tip = QT_TRANSLATE_NOOP("App::Property", "The points of the B-spline") obj.addProperty("App::PropertyVectorList", "Points", "Draft", _tip, locked=True) _tip = QT_TRANSLATE_NOOP("App::Property", "If the B-spline is closed or not") obj.addProperty("App::PropertyBool", "Closed", "Draft", _tip, locked=True) _tip = QT_TRANSLATE_NOOP("App::Property", "Create a face if this B-spline is closed") obj.addProperty("App::PropertyBool", "MakeFace", "Draft", _tip, locked=True) _tip = QT_TRANSLATE_NOOP("App::Property", "The area of this object") obj.addProperty("App::PropertyArea", "Area", "Draft", _tip, locked=True) obj.MakeFace = params.get_param("MakeFaceMode") obj.Closed = False obj.Points = [] self.assureProperties(obj) def onDocumentRestored(self, obj): super().onDocumentRestored(obj) gui_utils.restore_view_object(obj, vp_module="view_bspline", vp_class="ViewProviderBSpline") def assureProperties(self, obj): # for Compatibility with older versions if not hasattr(obj, "Parameterization"): _tip = QT_TRANSLATE_NOOP("App::Property", "Parameterization factor") obj.addProperty("App::PropertyFloat", "Parameterization", "Draft", _tip, locked=True) obj.Parameterization = 1.0 self.knotSeq = [] def parameterization(self, pts, a, closed): """Computes a knot Sequence for a set of points. fac (0-1) : parameterization factor fac = 0 -> Uniform / fac=0.5 -> Centripetal / fac=1.0 -> Chord-Length """ if closed: # we need to add the first point as the end point pts.append(pts[0]) params = [0] for i in range(1, len(pts)): p = pts[i].sub(pts[i - 1]) pl = pow(p.Length, a) params.append(params[-1] + pl) return params def onChanged(self, fp, prop): self.props_changed_store(prop) if prop == "Parameterization": if fp.Parameterization < 0.0: fp.Parameterization = 0.0 if fp.Parameterization > 1.0: fp.Parameterization = 1.0 def execute(self, obj): if self.props_changed_placement_only() or not obj.Points: obj.positionBySupport() self.props_changed_clear() return import Part self.assureProperties(obj) self.knotSeq = self.parameterization(obj.Points, obj.Parameterization, obj.Closed) plm = obj.Placement if obj.Closed and (len(obj.Points) > 2): if obj.Points[0] == obj.Points[-1]: # should not occur, but OCC will crash _err = QT_TRANSLATE_NOOP( "Draft", "_BSpline.createGeometry: " "Closed with same first/last Point. Geometry not updated.", ) App.Console.PrintError(_err + "\n") return spline = Part.BSplineCurve() spline.interpolate(obj.Points, PeriodicFlag=True, Parameters=self.knotSeq) # DNC: bug fix: convert to face if closed shape = Part.Wire(spline.toShape()) # Creating a face from a closed spline cannot be expected to always work # Usually, if the spline is not flat the call of Part.Face() fails try: if hasattr(obj, "MakeFace"): if obj.MakeFace: shape = Part.Face(shape) else: shape = Part.Face(shape) except Part.OCCError: pass obj.Shape = shape if hasattr(obj, "Area") and hasattr(shape, "Area"): obj.Area = shape.Area else: spline = Part.BSplineCurve() spline.interpolate(obj.Points, PeriodicFlag=False, Parameters=self.knotSeq) shape = spline.toShape() obj.Shape = shape if hasattr(obj, "Area") and hasattr(shape, "Area"): obj.Area = shape.Area obj.Placement = plm obj.positionBySupport() self.props_changed_clear() # Alias for compatibility with v0.18 and earlier _BSpline = BSpline ## @}