From 1c0e7479c88d19c587ef17db968f92b2d1268581 Mon Sep 17 00:00:00 2001 From: AU Date: Thu, 26 Dec 2024 23:17:25 +0100 Subject: [PATCH] Final shape / free func tweaks (#1731) * Add sample * Mypy fix * Isoline constructon * Add isolines and return params when sampling * Extend check * Test extended check output * Mypy fix * Func API * Use the official import in the docs * Add face * Handle trimming and all possible geometries * Initial tests * test _adaptor_curve_to_edge * Fix test * Be consistent with Sketch * Punctuation * Update cadquery/occ_impl/shapes.py Co-authored-by: Jeremy Wright --------- Co-authored-by: Jeremy Wright --- cadquery/func.py | 47 +++++++++++++ cadquery/occ_impl/geom.py | 6 +- cadquery/occ_impl/shapes.py | 129 +++++++++++++++++++++++++++++++++-- doc/free-func.rst | 14 ++-- tests/test_free_functions.py | 33 +++++++++ tests/test_shapes.py | 35 ++++++++++ 6 files changed, 250 insertions(+), 14 deletions(-) create mode 100644 cadquery/func.py diff --git a/cadquery/func.py b/cadquery/func.py new file mode 100644 index 000000000..cb7050983 --- /dev/null +++ b/cadquery/func.py @@ -0,0 +1,47 @@ +from .occ_impl.geom import Vector, Plane, Location +from .occ_impl.shapes import ( + Shape, + Vertex, + Edge, + Wire, + Face, + Shell, + Solid, + CompSolid, + Compound, + wire, + face, + shell, + solid, + compound, + vertex, + segment, + polyline, + polygon, + rect, + spline, + circle, + ellipse, + plane, + box, + cylinder, + sphere, + torus, + cone, + text, + fuse, + cut, + intersect, + split, + fill, + clean, + cap, + fillet, + chamfer, + extrude, + revolve, + offset, + sweep, + loft, + check, +) diff --git a/cadquery/occ_impl/geom.py b/cadquery/occ_impl/geom.py index 14e1bc96c..89bd90c08 100644 --- a/cadquery/occ_impl/geom.py +++ b/cadquery/occ_impl/geom.py @@ -1,6 +1,6 @@ from math import pi, radians, degrees -from typing import overload, Sequence, Union, Tuple, Type, Optional +from typing import overload, Sequence, Union, Tuple, Type, Optional, Iterator from OCP.gp import ( gp_Vec, @@ -236,6 +236,10 @@ def __eq__(self, other: "Vector") -> bool: # type: ignore[override] else False ) + def __iter__(self) -> Iterator[float]: + + yield from (self.x, self.y, self.z) + def toPnt(self) -> gp_Pnt: return gp_Pnt(self.wrapped.XYZ()) diff --git a/cadquery/occ_impl/shapes.py b/cadquery/occ_impl/shapes.py index dc1ae574f..cedc4809a 100644 --- a/cadquery/occ_impl/shapes.py +++ b/cadquery/occ_impl/shapes.py @@ -231,6 +231,8 @@ GeomAbs_C2, GeomAbs_Intersection, GeomAbs_JoinType, + GeomAbs_IsoType, + GeomAbs_CurveType, ) from OCP.BRepOffsetAPI import BRepOffsetAPI_MakeFilling from OCP.BRepOffset import BRepOffset_MakeOffset, BRepOffset_Mode @@ -244,7 +246,11 @@ from OCP.ShapeAnalysis import ShapeAnalysis_FreeBounds, ShapeAnalysis_Wire from OCP.TopTools import TopTools_HSequenceOfShape -from OCP.GCPnts import GCPnts_AbscissaPoint +from OCP.GCPnts import ( + GCPnts_AbscissaPoint, + GCPnts_QuasiUniformAbscissa, + GCPnts_QuasiUniformDeflection, +) from OCP.GeomFill import ( GeomFill_Frenet, @@ -281,6 +287,10 @@ from OCP.BinTools import BinTools +from OCP.Adaptor3d import Adaptor3d_IsoCurve, Adaptor3d_Curve + +from OCP.GeomAdaptor import GeomAdaptor_Surface + from math import pi, sqrt, inf, radians, cos import warnings @@ -1900,7 +1910,8 @@ def _curve_and_param( def positionAt( self: Mixin1DProtocol, d: float, mode: ParamMode = "length", ) -> Vector: - """Generate a position along the underlying curve. + """ + Generate a position along the underlying curve. :param d: distance or parameter value :param mode: position calculation mode (default: length) @@ -1914,7 +1925,8 @@ def positionAt( def positions( self: Mixin1DProtocol, ds: Iterable[float], mode: ParamMode = "length", ) -> List[Vector]: - """Generate positions along the underlying curve + """ + Generate positions along the underlying curve. :param ds: distance or parameter values :param mode: position calculation mode (default: length) @@ -1923,6 +1935,35 @@ def positions( return [self.positionAt(d, mode) for d in ds] + def sample( + self: Mixin1DProtocol, n: Union[int, float] + ) -> Tuple[List[Vector], List[float]]: + """ + Sample a curve based on a number of points or deflection. + + :param n: Number of positions or deflection + :return: A list of Vectors and a list of parameters. + """ + + gcpnts: Union[GCPnts_QuasiUniformAbscissa, GCPnts_QuasiUniformDeflection] + + if isinstance(n, int): + crv = self._geomAdaptor() + gcpnts = GCPnts_QuasiUniformAbscissa(crv, n + 1 if crv.IsClosed() else n) + else: + crv = self._geomAdaptor() + gcpnts = GCPnts_QuasiUniformDeflection(crv, n) + + N_pts = gcpnts.NbPoints() + + params = [ + gcpnts.Parameter(i) + for i in range(1, N_pts if crv.IsClosed() else N_pts + 1) + ] + pnts = [Vector(crv.Value(p)) for p in params] + + return pnts, params + def locationAt( self: Mixin1DProtocol, d: float, @@ -1930,7 +1971,8 @@ def locationAt( frame: FrameMode = "frenet", planar: bool = False, ) -> Location: - """Generate a location along the underlying curve. + """ + Generate a location along the underlying curve. :param d: distance or parameter value :param mode: position calculation mode (default: length) @@ -1973,7 +2015,8 @@ def locations( frame: FrameMode = "frenet", planar: bool = False, ) -> List[Location]: - """Generate location along the curve + """ + Generate locations along the curve. :param ds: distance or parameter values :param mode: position calculation mode (default: length) @@ -3188,6 +3231,35 @@ def trim(self, u0: Real, u1: Real, v0: Real, v1: Real, tol: Real = 1e-6) -> "Fac return self.__class__(bldr.Shape()) + def isoline(self, param: Real, direction: Literal["u", "v"] = "v") -> Edge: + """ + Construct an isoline. + """ + + u1, u2, v1, v2 = self._uvBounds() + + if direction == "u": + iso = GeomAbs_IsoType.GeomAbs_IsoU + p1, p2 = v1, v2 + else: + iso = GeomAbs_IsoType.GeomAbs_IsoV + p1, p2 = u1, u2 + + adaptor = Adaptor3d_IsoCurve( + GeomAdaptor_Surface(self._geomAdaptor()), iso, param + ) + + return Edge(_adaptor_curve_to_edge(adaptor, p1, p2)) + + def isolines( + self, params: Iterable[Real], direction: Literal["u", "v"] = "v" + ) -> List[Edge]: + """ + Construct multiple isolines. + """ + + return [self.isoline(p, direction) for p in params] + class Shell(Shape): """ @@ -4622,6 +4694,14 @@ def _shapes_to_toptools_list(s: Iterable[Shape]) -> TopTools_ListOfShape: return rv +def _toptools_list_to_shapes(tl: TopTools_ListOfShape) -> List[Shape]: + """ + Convert a TopTools list (OCCT specific) to a compound. + """ + + return [_normalize(Shape.cast(el)) for el in tl] + + _geomabsshape_dict = dict( C0=GeomAbs_Shape.GeomAbs_C0, C1=GeomAbs_Shape.GeomAbs_C1, @@ -4656,6 +4736,34 @@ def _to_parametrization(name: str) -> Approx_ParametrizationType: return _parametrization_dict[name.lower()] +def _adaptor_curve_to_edge(crv: Adaptor3d_Curve, p1: float, p2: float) -> TopoDS_Edge: + + GCT = GeomAbs_CurveType + + t = crv.GetType() + + if t == GCT.GeomAbs_BSplineCurve: + bldr = BRepBuilderAPI_MakeEdge(crv.BSpline(), p1, p2) + elif t == GCT.GeomAbs_BezierCurve: + bldr = BRepBuilderAPI_MakeEdge(crv.Bezier(), p1, p2) + elif t == GCT.GeomAbs_Circle: + bldr = BRepBuilderAPI_MakeEdge(crv.Circle(), p1, p2) + elif t == GCT.GeomAbs_Line: + bldr = BRepBuilderAPI_MakeEdge(crv.Line(), p1, p2) + elif t == GCT.GeomAbs_Ellipse: + bldr = BRepBuilderAPI_MakeEdge(crv.Ellipse(), p1, p2) + elif t == GCT.GeomAbs_Hyperbola: + bldr = BRepBuilderAPI_MakeEdge(crv.Hyperbola(), p1, p2) + elif t == GCT.GeomAbs_Parabola: + bldr = BRepBuilderAPI_MakeEdge(crv.Parabola(), p1, p2) + elif t == GCT.GeomAbs_OffsetCurve: + bldr = BRepBuilderAPI_MakeEdge(crv.OffsetCurve(), p1, p2) + else: + raise ValueError(r"{t} is not a supported curve type") + + return bldr.Edge() + + #%% alternative constructors @@ -5675,7 +5783,7 @@ def loft( #%% diagnotics -def check(s: Shape) -> bool: +def check(s: Shape, results: Optional[List[Tuple[List[Shape], Any]]] = None) -> bool: """ Check if a shape is valid. """ @@ -5688,4 +5796,13 @@ def check(s: Shape) -> bool: rv = analyzer.IsValid() + # output detailed results if requested + if results is not None: + results.clear() + + for r in analyzer.Result(): + results.append( + (_toptools_list_to_shapes(r.GetFaultyShapes1()), r.GetCheckStatus()) + ) + return rv diff --git a/doc/free-func.rst b/doc/free-func.rst index be8085834..f6cf7f0d0 100644 --- a/doc/free-func.rst +++ b/doc/free-func.rst @@ -21,7 +21,7 @@ The purpose of this section is to demonstrate how to construct Shape objects usi .. cadquery:: :height: 600px - from cadquery.occ_impl.shapes import * + from cadquery.func import * dh = 2 r = 1 @@ -97,7 +97,7 @@ Various 1D, 2D and 3D primitives are supported. .. cadquery:: - from cadquery.occ_impl.shapes import * + from cadquery.func import * e = segment((0,0), (0,1)) @@ -119,7 +119,7 @@ One can for example union multiple solids at once by first combining them into a .. cadquery:: - from cadquery.occ_impl.shapes import * + from cadquery.func import * c1 = cylinder(1, 2) c2 = cylinder(0.5, 3) @@ -158,7 +158,7 @@ Constructing complex shapes from simple shapes is possible in various contexts. .. cadquery:: - from cadquery.occ_impl.shapes import * + from cadquery.func import * e1 = segment((0,0), (1,0)) e2 = segment((1,0), (1,1)) @@ -196,7 +196,7 @@ Free function API currently supports :meth:`~cadquery.occ_impl.shapes.extrude`, .. cadquery:: - from cadquery.occ_impl.shapes import * + from cadquery.func import * r = rect(1,0.5) f = face(r, circle(0.2).moved(0.2), rect(0.2, 0.4).moved(-0.2)) @@ -229,7 +229,7 @@ Placement and creation of arrays is possible using :meth:`~cadquery.Shape.move` .. cadquery:: - from cadquery.occ_impl.shapes import * + from cadquery.func import * locs = [(0,-1,0), (0,1,0)] @@ -246,7 +246,7 @@ The free function API has extensive text creation capabilities including text on .. cadquery:: - from cadquery.occ_impl.shapes import * + from cadquery.func import * from math import pi diff --git a/tests/test_free_functions.py b/tests/test_free_functions.py index 091e2d405..ab7b1cb26 100644 --- a/tests/test_free_functions.py +++ b/tests/test_free_functions.py @@ -35,15 +35,19 @@ Location, Shape, Compound, + Edge, _get_one_wire, _get_wires, _get, _get_one, _get_edges, + _adaptor_curve_to_edge, check, Vector, ) +from OCP.BOPAlgo import BOPAlgo_CheckStatus + from pytest import approx, raises from math import pi @@ -97,6 +101,29 @@ def test_utils(): list(_get_edges(fill(circle(1)))) +def test_adaptor_curve_to_edge(): + + from OCP.gp import gp_Hypr, gp_Parab, gp_Ax2 + from OCP.BRepBuilderAPI import BRepBuilderAPI_MakeEdge + from OCP.TopoDS import TopoDS_Edge + + # make some dummy edges with different geometries + lin = segment(Vector(), Vector(0, 1)) + bez = Edge.makeBezier([Vector(), Vector(0, 0, 1)]) + spl = spline([Vector(), Vector(0, 0, 1)]) + circ = circle(1) + el = ellipse(2, 1) + off = wire(el).offset2D(-0.1, kind="tangent")[0].Edges()[0] + hypr = Edge(BRepBuilderAPI_MakeEdge(gp_Hypr(gp_Ax2(), 2, 1)).Edge()) + parab = Edge(BRepBuilderAPI_MakeEdge(gp_Parab()).Edge()) + + # smoke test + for s in (lin, bez, spl, circ, el, off, hypr, parab): + e = _adaptor_curve_to_edge(s._geomAdaptor().Curve(), 0, 1) + + assert isinstance(e, TopoDS_Edge) + + #%% constructors @@ -657,6 +684,7 @@ def test_export(): # %% diagnostics def test_check(): + # correct shape s1 = box(1, 1, 1) assert check(s1) @@ -664,3 +692,8 @@ def test_check(): s2 = sweep(rect(1, 1), segment((0, 0), (1, 1))) assert not check(s2) + + res = [] + + assert not check(s2, res) + assert res[0][1] == BOPAlgo_CheckStatus.BOPAlgo_SelfIntersect diff --git a/tests/test_shapes.py b/tests/test_shapes.py index 7bd95f602..4dc027a48 100644 --- a/tests/test_shapes.py +++ b/tests/test_shapes.py @@ -10,6 +10,8 @@ plane, torus, Shape, + cylinder, + ellipse, ) from pytest import approx, raises @@ -136,3 +138,36 @@ def test_bin_import_export(): with raises(Exception): Shape.importBin(BytesIO()) + + +def test_sample(): + + e = ellipse(10, 1) + s = segment((0, 0), (1, 0)) + + pts1, params1 = e.sample(10) # equidistant + pts2, params2 = e.sample(0.1) # deflection based + pts3, params3 = s.sample(10) # equidistant, open + + assert len(pts1) == len(params1) + assert len(pts1) == 10 # e is closed + + assert len(pts2) == len(params2) + assert len(pts2) == 16 + + assert len(pts3) == len(params3) + assert len(pts3) == 10 # s is open + + +def test_isolines(): + + c = cylinder(1, 2).faces("%CYLINDER") + + isos_v = c.isolines([0, 1]) + isos_u = c.isolines([0, 1], "u") + + assert len(isos_u) == 2 + assert len(isos_v) == 2 + + assert isos_u[0].Length() == approx(2) + assert isos_v[0].Length() == approx(pi)