-
Notifications
You must be signed in to change notification settings - Fork 299
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
docs give examples of tuples where they should be Vectors #256
Comments
Many of the methods will implicitly convert a tuple to a Vector so that either can be passed by the caller. Either we missed that in DirectionMinMaxSelector, or it was dropped in the transition to PythonOCC. |
Here's an example: cadquery/cadquery/occ_impl/shapes.py Line 330 in f9d9ae0
|
Haha @jmwright, you're too quick, I was in the middle of writing the following when you commented. Rather than change the docs I would suggest we consistently handle tuples or Vectors. But what is the consistent CadQuery way of doing this? We check type for tuple here: cadquery/cadquery/occ_impl/shapes.py Lines 481 to 501 in e69b2f8
Request tuple in docs and init a Vector here: cadquery/cadquery/selectors.py Lines 62 to 94 in e69b2f8
But the Vector class handles being initiated with a Personally I want to check for the attribute the method needs. eg. def __init__(self, vector, directionMax=True, tolerance=0.0001):
self.vector = vector if hasattr(vector, 'dot') else Vector(vector) edit: |
Also found Lines 989 to 1013 in e69b2f8
If you follow the code, |
@marcus7070 The inconsistent handling of tuples and Vectors is something that it would be nice to fix. If a Vector can be constructed with either a tuple or a Vector, why add the |
@jmwright Yeah, 99% redundant. Vector init actually grabs an XYZ tuple from the argument and reconstructs a new Vector, so cadquery/cadquery/occ_impl/geom.py Lines 11 to 51 in 3cd466a
|
Ok, yeah. The |
Awesome, I might have a go at this when I get some time. |
I've got an idea that I may as well document here, in case anyone else picks this up before I do. Throughout CadQuery we accept tuples in the place of Vectors and type check or initialise Vectors in each function. There is a lot of code duplication and different implementations of this. It seems like a worthwhile task to centralise that code into one method we apply everywhere. IMHO a decorator makes sense for this job. How should devs indicate that a particular argument represents a point in space and can be either a tuple or a Vector? Type hinting seems to make the most sense to me. Here is a patch that implements it: diff --git a/cadquery/cq.py b/cadquery/cq.py
index 25dc1d3..7577c4c 100644
--- a/cadquery/cq.py
+++ b/cadquery/cq.py
@@ -19,6 +19,9 @@
import math
from copy import copy
+from functools import wraps
+from inspect import signature
+import typing
from . import (
Vector,
Plane,
@@ -33,6 +36,11 @@ from . import (
exporters,
)
+Numeric = typing.Union[int, float]
+Tuple2d = typing.Tuple[Numeric, Numeric]
+Tuple3d = typing.Tuple[Numeric, Numeric, Numeric]
+Pnt = typing.Union[Tuple2d, Tuple3d, Vector]
+
class CQContext(object):
"""
@@ -54,6 +62,25 @@ class CQContext(object):
self.tolerance = 0.0001 # user specified tolerance
+def tup_to_vec(f):
+ """
+ A decorator that looks for arguments type hinted as Pnts and if they are
+ not Vectors, converts them to Vectors. It is expected that any argument
+ that is not a Vector will be a tuple, but since this is Python we do not
+ type check.
+ """
+ sig = signature(f)
+ @wraps(f)
+ def wraps_f(*args, **kwargs):
+ bound_args = sig.bind(*args, **kwargs)
+ for name, param in sig.parameters.items():
+ val = bound_args.arguments[name]
+ if param.annotation == Pnt and not isinstance(val, Vector):
+ bound_args.arguments[name] = Vector(val)
+ return f(*bound_args.args, **bound_args.kwargs)
+ return wraps_f
+
+
class CQ(object):
"""
Provides enhanced functionality for a wrapped CAD primitive.
@@ -856,7 +883,8 @@ class CQ(object):
return self.each(_rot, False)
- def rotate(self, axisStartPoint, axisEndPoint, angleDegrees):
+ @tup_to_vec
+ def rotate(self, axisStartPoint: Pnt, axisEndPoint: Pnt, angleDegrees):
"""
Returns a copy of all of the items on the stack rotated through and angle around the axis
of rotation.
@@ -873,7 +901,8 @@ class CQ(object):
[o.rotate(axisStartPoint, axisEndPoint, angleDegrees) for o in self.objects]
)
- def mirror(self, mirrorPlane="XY", basePointVector=(0, 0, 0)):
+ @tup_to_vec
+ def mirror(self, mirrorPlane="XY", basePointVector: Pnt = (0, 0, 0)):
"""
Mirror a single CQ object. This operation is the same as in the FreeCAD PartWB's mirroring I'm fairly confident I've defined those types incorrectly, but the code works. Since that decorator inspects arguments, it would be pretty simple to add a try/except around the wrapped function and look for the common mistake of leaving the brackets off a tuple. If the Pnt argument and the one or two after it are all numeric types, append a suggestion to the error message that the user may have left the brackets off a tuple. Anyhow, I haven't convinced myself that the extra dependencies (and adding partial type hinting) are worth it yet. I'm also not very experienced with type hinting. So I'm not going to make a draft PR yet, I'll just carry on with some other work and think about this for a while. |
Seems like we want to add type hints to CadQuery anyway: #247 |
cadquery/cadquery/selectors.py
Lines 287 to 338 in e69b2f8
The docs suggest
DirectionMinMaxSelector((0, 0, 1), True)
, but this code:fails with
AttributeError: 'tuple' object has no attribute 'wrapped'
.Changing the tuple to a Vector works:
The text was updated successfully, but these errors were encountered: