Source code for cam.strategy

"""Fabex 'strategy.py' © 2012 Vilem Novak

Strategy functionality of Fabex - e.g. Cutout, Parallel, Spiral, Waterline
The functions here are called with operators defined in 'ops.py'
"""

from math import (
    ceil,
    pi,
    radians,
    sqrt,
    tan,
)
import sys
import time

import shapely
from shapely.geometry import polygon as spolygon
from shapely.geometry import Point  # Double check this import!
from shapely import geometry as sgeometry
from shapely import affinity

import bpy
from bpy_extras import object_utils
from mathutils import Euler, Vector


from .bridges import use_bridges
from .cam_chunk import (
    CamPathChunk,
    curve_to_chunks,
    limit_chunks,
    shapely_to_chunks,
    sample_chunks_n_axis,
    silhouette_offset,
    get_object_silhouette,
    get_object_outline,
    get_operation_silhouette,
    sort_chunks,
)
from .collision import cleanup_bullet_collision
from .constants import SHAPELY
from .exception import CamException

from .operators.curve_create_ops import generate_crosshatch

from .utilities.chunk_utils import (
    chunks_refine,
    optimize_chunk,
    chunks_refine_threshold,
    parent_child_distance,
    parent_child_poly,
    set_chunks_z,
    extend_chunks_5_axis,
)
from .utilities.compare_utils import check_equal, unique
from .utilities.geom_utils import circle, helix
from .utilities.operation_utils import get_operation_sources
from .utilities.shapely_utils import shapely_to_curve
from .utilities.simple_utils import (
    activate,
    delete_object,
    join_multiple,
    progress,
    remove_multiple,
    subdivide_short_lines,
)


# add pocket op for medial axis and profile cut inside to clean unremoved material
[docs] def add_pocket(maxdepth, sname, new_cutter_diameter): """Add a pocket operation for the medial axis and profile cut. This function first deselects all objects in the scene and then checks for any existing medial pocket objects, deleting them if found. It verifies whether a medial pocket operation already exists in the camera operations. If it does not exist, it creates a new pocket operation with the specified parameters. The function also modifies the selected object's silhouette offset based on the new cutter diameter. Args: maxdepth (float): The maximum depth of the pocket to be created. sname (str): The name of the object to which the pocket will be added. new_cutter_diameter (float): The diameter of the new cutter to be used. """ bpy.ops.object.select_all(action="DESELECT") s = bpy.context.scene mpocket_exists = False for ob in s.objects: # delete old medial pocket if ob.name.startswith("medial_poc"): ob.select_set(True) bpy.ops.object.delete() for op in s.cam_operations: # verify medial pocket operation exists if op.name == "MedialPocket": mpocket_exists = True ob = bpy.data.objects[sname] ob.select_set(True) bpy.context.view_layer.objects.active = ob silhouette_offset(ob, -new_cutter_diameter / 2, 1, 2) bpy.context.active_object.name = "medial_pocket" m_ob = bpy.context.view_layer.objects.active bpy.ops.object.origin_set(type="ORIGIN_GEOMETRY", center="BOUNDS") m_ob.location.z = maxdepth if not mpocket_exists: # create a pocket operation if it does not exist already s.cam_operations.add() o = s.cam_operations[-1] o.object_name = "medial_pocket" s.cam_active_operation = len(s.cam_operations) - 1 o.name = "MedialPocket" o.filename = o.name o.strategy = "POCKET" o.use_layers = False o.material.estimate_from_model = False o.material.size[2] = -maxdepth
# cutout strategy is completely here:
[docs] async def cutout(o): """Perform a cutout operation based on the provided parameters. This function calculates the necessary cutter offset based on the cutter type and its parameters. It processes a list of objects to determine how to cut them based on their geometry and the specified cutting type. The function handles different cutter types such as 'VCARVE', 'CYLCONE', 'BALLCONE', and 'BALLNOSE', applying specific calculations for each. It also manages the layering and movement strategies for the cutting operation, including options for lead-ins, ramps, and bridges. Args: o (object): An object containing parameters for the cutout operation, including cutter type, diameter, depth, and other settings. Returns: None: This function does not return a value but performs operations on the provided object. """ max_depth = check_min_z(o) cutter_angle = radians(o.cutter_tip_angle / 2) c_offset = o.cutter_diameter / 2 # cutter offset print("Cutter Type:", o.cutter_type) print("Max Depth:", max_depth) if o.cutter_type == "VCARVE": c_offset = -max_depth * tan(cutter_angle) elif o.cutter_type == "CYLCONE": c_offset = -max_depth * tan(cutter_angle) + o.cylcone_diameter / 2 elif o.cutter_type == "BALLCONE": c_offset = -max_depth * tan(cutter_angle) + o.ball_radius elif o.cutter_type == "BALLNOSE": r = o.cutter_diameter / 2 print("Cutter Radius:", r) print("Skin: ", o.skin) if -max_depth < r: c_offset = sqrt(r**2 - (r + max_depth) ** 2) print("Offset:", c_offset) if c_offset > o.cutter_diameter / 2: c_offset = o.cutter_diameter / 2 c_offset += o.skin # add skin for profile if o.straight: join = 2 else: join = 1 print("Operation: Cutout") offset = True for ob in o.objects: if ob.type == "CURVE": if ob.data.splines and ob.data.splines[0].type == "BEZIER": activate(ob) bpy.ops.object.curve_remove_doubles(merge_distance=0.0001, keep_bezier=True) else: bpy.ops.object.curve_remove_doubles() # make sure all polylines are at least three points long subdivide_short_lines(ob) if o.cut_type == "ONLINE" and o.onlycurves: # is separate to allow open curves :) print("Separate") chunksFromCurve = [] for ob in o.objects: chunksFromCurve.extend(curve_to_chunks(ob, o.use_modifiers)) else: chunksFromCurve = [] if o.cut_type == "ONLINE": p = get_object_outline(0, o, True) else: offset = True if o.cut_type == "INSIDE": offset = False p = get_object_outline(c_offset, o, offset) if o.outlines_count > 1: for i in range(1, o.outlines_count): chunksFromCurve.extend(shapely_to_chunks(p, -1)) path_distance = o.distance_between_paths if o.cut_type == "INSIDE": path_distance *= -1 p = p.buffer( distance=path_distance, resolution=o.optimisation.circle_detail, join_style=join, mitre_limit=2, ) chunksFromCurve.extend(shapely_to_chunks(p, -1)) if o.outlines_count > 1 and o.movement.insideout == "OUTSIDEIN": chunksFromCurve.reverse() chunksFromCurve = limit_chunks(chunksFromCurve, o) if not o.dont_merge: parent_child_poly(chunksFromCurve, chunksFromCurve, o) if o.outlines_count == 1: chunksFromCurve = await sort_chunks(chunksFromCurve, o) if (o.movement.type == "CLIMB" and o.movement.spindle_rotation == "CCW") or ( o.movement.type == "CONVENTIONAL" and o.movement.spindle_rotation == "CW" ): for ch in chunksFromCurve: ch.reverse() if o.cut_type == "INSIDE": # there would bee too many conditions above, # so for now it gets reversed once again when inside cutting. for ch in chunksFromCurve: ch.reverse() layers = get_layers(o, o.max_z, check_min_z(o)) extendorder = [] if o.first_down: # each shape gets either cut all the way to bottom, # or every shape gets cut 1 layer, then all again. has to create copies, # because same chunks are worked with on more layers usually for chunk in chunksFromCurve: dir_switch = False # needed to avoid unnecessary lifting of cutter with open chunks # and movement set to "MEANDER" for layer in layers: chunk_copy = chunk.copy() if dir_switch: chunk_copy.reverse() extendorder.append([chunk_copy, layer]) if (not chunk.closed) and o.movement.type == "MEANDER": dir_switch = not dir_switch else: for layer in layers: for chunk in chunksFromCurve: extendorder.append([chunk.copy(), layer]) for chl in extendorder: # Set Z for all chunks chunk = chl[0] layer = chl[1] print(layer[1]) chunk.set_z(layer[1]) chunks = [] if o.use_bridges: # add bridges to chunks print("Using Bridges") remove_multiple(o.name + "_cut_bridges") print("Old Briddge Cut Removed") bridgeheight = min(o.max.z, o.min.z + abs(o.bridges_height)) for chl in extendorder: chunk = chl[0] layer = chl[1] if layer[1] < bridgeheight: use_bridges(chunk, o) if o.profile_start > 0: print("Cutout Change Profile Start") for chl in extendorder: chunk = chl[0] if chunk.closed: chunk.change_path_start(o) # Lead in if o.lead_in > 0.0 or o.lead_out > 0: print("Cutout Lead-in") for chl in extendorder: chunk = chl[0] if chunk.closed: chunk.break_path_for_leadin_leadout(o) chunk.lead_contour(o) if o.movement.ramp: # add ramps or simply add chunks for chl in extendorder: chunk = chl[0] layer = chl[1] if o.movement.zig_zag_ramp: chunk.ramp_zig_zag(layer[0], layer[1], o) chunks.append(chunk) else: if chunk.closed: chunk.ramp_contour(layer[0], layer[1], o) chunks.append(chunk) else: chunk.ramp_zig_zag(layer[0], layer[1], o) chunks.append(chunk) else: for chl in extendorder: chunks.append(chl[0]) chunks_to_mesh(chunks, o)
[docs] async def curve(o): """Process and convert curve objects into mesh chunks. This function takes an operation object and processes the curves contained within it. It first checks if all objects are curves; if not, it raises an exception. The function then converts the curves into chunks, sorts them, and refines them. If layers are to be used, it applies layer information to the chunks, adjusting their Z-offsets accordingly. Finally, it converts the processed chunks into a mesh. Args: o (Operation): An object containing operation parameters, including a list of objects, flags for layer usage, and movement constraints. Returns: None: This function does not return a value; it performs operations on the input. Raises: CamException: If not all objects in the operation are curves. """ print("Operation: Curve") pathSamples = [] get_operation_sources(o) if not o.onlycurves: raise CamException("All Objects Must Be Curves for This Operation.") for ob in o.objects: # make sure all polylines are at least three points long subdivide_short_lines(ob) # make the chunks from curve here pathSamples.extend(curve_to_chunks(ob)) # sort before sampling pathSamples = await sort_chunks(pathSamples, o) pathSamples = chunks_refine(pathSamples, o) # simplify # layers here if o.use_layers: layers = get_layers(o, o.max_z, round(check_min_z(o), 6)) # layers is a list of lists [[0.00,l1],[l1,l2],[l2,l3]] containg the start and end of each layer extendorder = [] chunks = [] for layer in layers: for ch in pathSamples: # include layer information to chunk list extendorder.append([ch.copy(), layer]) for chl in extendorder: # Set offset Z for all chunks according to the layer information, chunk = chl[0] layer = chl[1] print("Layer: " + str(layer[1])) chunk.offset_z(o.max_z * 2 - o.min_z + layer[1]) chunk.clamp_z(o.min_z) # safety to not cut lower than minz # safety, not higher than free movement height chunk.clamp_max_z(o.movement.free_height) for ( chl ) in extendorder: # strip layer information from extendorder and transfer them to chunks chunks.append(chl[0]) chunks_to_mesh(chunks, o) # finish by converting to mesh else: # no layers, old curve for ch in pathSamples: ch.clamp_z(o.min_z) # safety to not cut lower than minz # safety, not higher than free movement height ch.clamp_max_z(o.movement.free_height) chunks_to_mesh(pathSamples, o)
[docs] async def project_curve(s, o): """Project a curve onto another curve object. This function takes a source object and a target object, both of which are expected to be curve objects. It projects the points of the source curve onto the target curve, adjusting the start and end points based on specified extensions. The resulting projected points are stored in the source object's path samples. Args: s (object): The source object containing the curve to be projected. o (object): An object containing references to the curve objects involved in the projection. Returns: None: This function does not return a value; it modifies the source object's path samples in place. Raises: CamException: If the target curve is not of type 'CURVE'. """ print("Operation: Projected Curve") pathSamples = [] chunks = [] ob = bpy.data.objects[o.curve_source] pathSamples.extend(curve_to_chunks(ob)) targetCurve = s.objects[o.curve_target] from cam import cam_chunk if targetCurve.type != "CURVE": raise CamException("Projection Target and Source Have to Be Curve Objects!") if 1: extend_up = 0.1 extend_down = 0.04 tsamples = curve_to_chunks(targetCurve) for chi, ch in enumerate(pathSamples): cht = tsamples[chi].get_points() ch.depth = 0 ch_points = ch.get_points() for i, s in enumerate(ch_points): # move the points a bit ep = Vector(cht[i]) sp = Vector(ch_points[i]) # extend startpoint vecs = sp - ep vecs.normalize() vecs *= extend_up sp += vecs ch.startpoints.append(sp) # extend endpoint vece = sp - ep vece.normalize() vece *= extend_down ep -= vece ch.endpoints.append(ep) ch.rotations.append((0, 0, 0)) vec = sp - ep ch.depth = min(ch.depth, -vec.length) ch_points[i] = sp.copy() ch.set_points(ch_points) layers = get_layers(o, 0, ch.depth) chunks.extend(sample_chunks_n_axis(o, pathSamples, layers)) chunks_to_mesh(chunks, o)
[docs] async def pocket(o): """Perform pocketing operation based on the provided parameters. This function executes a pocketing operation using the specified parameters from the object `o`. It calculates the cutter offset based on the cutter type and depth, processes curves, and generates the necessary chunks for the pocketing operation. The function also handles various movement types and optimizations, including helix entry and retract movements. Args: o (object): An object containing parameters for the pocketing Returns: None: The function modifies the scene and generates geometry based on the pocketing operation. """ if o.straight: join = 2 else: join = 1 print("Operation: Pocket") scene = bpy.context.scene remove_multiple("3D_poc") max_depth = check_min_z(o) + o.skin cutter_angle = radians(o.cutter_tip_angle / 2) c_offset = o.cutter_diameter / 2 if o.cutter_type == "VCARVE": c_offset = -max_depth * tan(cutter_angle) elif o.cutter_type == "CYLCONE": c_offset = -max_depth * tan(cutter_angle) + o.cylcone_diameter / 2 elif o.cutter_type == "BALLCONE": c_offset = -max_depth * tan(cutter_angle) + o.ball_radius if c_offset > o.cutter_diameter / 2: c_offset = o.cutter_diameter / 2 c_offset += o.skin # add skin print("Cutter Offset", c_offset) obname = o.object_name c_ob = bpy.data.objects[obname] for ob in o.objects: if ob.type == "CURVE": if ob.data.splines and ob.data.splines[0].type == "BEZIER": activate(ob) bpy.ops.object.curve_remove_doubles(merge_distance=0.0001, keep_bezier=True) else: bpy.ops.object.curve_remove_doubles() chunksFromCurve = [] angle = radians(o.parallel_pocket_angle) distance = o.distance_between_paths offset = -c_offset pocket_shape = "" n_angle = angle - pi / 2 pr = get_object_outline(0, o, False) if o.pocket_type == "PARALLEL": if o.parallel_pocket_contour: offset = -(c_offset + distance / 2) p = pr.buffer( -c_offset, resolution=o.optimisation.circle_detail, join_style=join, mitre_limit=2 ) nchunks = shapely_to_chunks(p, o.min.z) chunksFromCurve.extend(nchunks) crosshatch_result = generate_crosshatch( bpy.context, angle, distance, offset, pocket_shape, join, c_ob ) nchunks = shapely_to_chunks(crosshatch_result, o.min.z) chunksFromCurve.extend(nchunks) if o.parallel_pocket_crosshatch: crosshatch_result = generate_crosshatch( bpy.context, n_angle, distance, offset, pocket_shape, join, c_ob ) nchunks = shapely_to_chunks(crosshatch_result, o.min.z) chunksFromCurve.extend(nchunks) else: p = pr.buffer( -c_offset, resolution=o.optimisation.circle_detail, join_style=join, mitre_limit=2 ) approxn = (min(o.max.x - o.min.x, o.max.y - o.min.y) / o.distance_between_paths) / 2 print("Approximative:" + str(approxn)) print(o.name) i = 0 chunks = [] lastchunks = [] centers = None firstoutline = p # for testing in the end. prest = p.buffer(-c_offset, o.optimisation.circle_detail) while not p.is_empty: if o.pocket_to_curve: # make a curve starting with _3dpocket shapely_to_curve("3dpocket", p, 0.0) nchunks = shapely_to_chunks(p, o.min.z) # print("nchunks") pnew = p.buffer( -o.distance_between_paths, o.optimisation.circle_detail, join_style=join, mitre_limit=2, ) if pnew.is_empty: # test if the last curve will leave material pt = p.buffer( -c_offset, o.optimisation.circle_detail, join_style=join, mitre_limit=2 ) if not pt.is_empty: pnew = pt # print("pnew") nchunks = limit_chunks(nchunks, o) chunksFromCurve.extend(nchunks) parent_child_distance(lastchunks, nchunks, o) lastchunks = nchunks percent = int(i / approxn * 100) progress("Outlining Polygons ", percent) p = pnew i += 1 # if (o.poc)#TODO inside outside! if (o.movement.type == "CLIMB" and o.movement.spindle_rotation == "CW") or ( o.movement.type == "CONVENTIONAL" and o.movement.spindle_rotation == "CCW" ): for ch in chunksFromCurve: ch.reverse() chunksFromCurve = await sort_chunks(chunksFromCurve, o) chunks = [] layers = get_layers(o, o.max_z, check_min_z(o)) for l in layers: lchunks = set_chunks_z(chunksFromCurve, l[1]) if o.movement.ramp: for ch in lchunks: ch.zstart = l[0] ch.zend = l[1] # helix_enter first try here TODO: check if helix radius is not out of operation area. if o.movement.helix_enter: helix_radius = ( c_offset * o.movement.helix_diameter * 0.01 ) # 90 percent of cutter radius helix_circumference = helix_radius * pi * 2 revheight = helix_circumference * tan(o.movement.ramp_in_angle) for chi, ch in enumerate(lchunks): if not chunksFromCurve[chi].children: # TODO:intercept closest next point when it should stay low p = ch.get_point(0) # first thing to do is to check if helix enter can really enter. checkc = circle(helix_radius + c_offset, o.optimisation.circle_detail) checkc = affinity.translate(checkc, p[0], p[1]) covers = False for poly in o.silhouette.geoms: if poly.contains(checkc): covers = True break if covers: revolutions = (l[0] - p[2]) / revheight # print(revolutions) h = helix(helix_radius, o.optimisation.circle_detail, l[0], p, revolutions) # invert helix if not the typical direction if ( o.movement.type == "CONVENTIONAL" and o.movement.spindle_rotation == "CW" ) or (o.movement.type == "CLIMB" and o.movement.spindle_rotation == "CCW"): nhelix = [] for v in h: nhelix.append((2 * p[0] - v[0], v[1], v[2])) h = nhelix ch.extend(h, at_index=0) # ch.points = h + ch.points else: o.info.warnings += "Helix entry did not fit! \n " ch.closed = True ch.ramp_zig_zag(l[0], l[1], o) # Arc retract here first try: # TODO: check for entry and exit point before actual computing... will be much better. if o.movement.retract_tangential: # TODO: fix this for CW and CCW! for chi, ch in enumerate(lchunks): # print(chunksFromCurve[chi]) # print(chunksFromCurve[chi].parents) if chunksFromCurve[chi].parents == [] or len(chunksFromCurve[chi].parents) == 1: revolutions = 0.25 v1 = Vector(ch.get_point(-1)) i = -2 v2 = Vector(ch.get_point(i)) v = v1 - v2 while v.length == 0: i = i - 1 v2 = Vector(ch.get_point(i)) v = v1 - v2 v.normalize() rotangle = Vector((v.x, v.y)).angle_signed(Vector((1, 0))) e = Euler((0, 0, pi / 2.0)) # TODO:#CW CLIMB! v.rotate(e) p = v1 + v * o.movement.retract_radius center = p p = (p.x, p.y, p.z) # progress(str((v1,v,p))) h = helix( o.movement.retract_radius, o.optimisation.circle_detail, p[2] + o.movement.retract_height, p, revolutions, ) # angle to rotate whole retract move e = Euler((0, 0, rotangle + pi)) rothelix = [] c = [] # polygon for outlining and checking collisions. for p in h: # rotate helix to go from tangent of vector v1 = Vector(p) v = v1 - center v.x = -v.x # flip it here first... v.rotate(e) p = center + v rothelix.append(p) c.append((p[0], p[1])) c = sgeometry.Polygon(c) # print('çoutline') # print(c) coutline = c.buffer(c_offset, o.optimisation.circle_detail) # print(h) # print('çoutline') # print(coutline) # polyToMesh(coutline,0) rothelix.reverse() covers = False for poly in o.silhouette.geoms: if poly.contains(coutline): covers = True break if covers: ch.extend(rothelix) chunks.extend(lchunks) if o.movement.ramp: for ch in chunks: ch.ramp_zig_zag(ch.zstart, ch.get_point(0)[2], o) if o.first_down: if o.pocket_option == "OUTSIDE": chunks.reverse() chunks = await sort_chunks(chunks, o) if o.pocket_to_curve: # make curve instead of a path join_multiple("3dpocket") else: chunks_to_mesh(chunks, o) # make normal pocket path
[docs] async def drill(o): """Perform a drilling operation on the specified objects. This function iterates through the objects in the provided context, activating each object and applying transformations. It duplicates the objects and processes them based on their type (CURVE or MESH). For CURVE objects, it calculates the bounding box and center points of the splines and bezier points, and generates chunks based on the specified drill type. For MESH objects, it generates chunks from the vertices. The function also manages layers and chunk depths for the drilling operation. Args: o (object): An object containing properties and methods required for the drilling operation, including a list of objects to drill, drill type, and depth parameters. Returns: None: This function does not return a value but performs operations that modify the state of the Blender context. """ print("Operation: Drill") chunks = [] for ob in o.objects: activate(ob) bpy.ops.object.duplicate_move( OBJECT_OT_duplicate={"linked": False, "mode": "TRANSLATION"}, TRANSFORM_OT_translate={ "value": (0, 0, 0), "constraint_axis": (False, False, False), "orient_type": "GLOBAL", "mirror": False, "use_proportional_edit": False, "proportional_edit_falloff": "SMOOTH", "proportional_size": 1, "snap": False, "snap_target": "CLOSEST", "snap_point": (0, 0, 0), "snap_align": False, "snap_normal": (0, 0, 0), "texture_space": False, "release_confirm": False, }, ) # bpy.ops.collection.objects_remove_all() bpy.ops.object.parent_clear(type="CLEAR_KEEP_TRANSFORM") ob = bpy.context.active_object if ob.type == "CURVE": ob.data.dimensions = "3D" try: bpy.ops.object.transform_apply(location=True, rotation=False, scale=False) bpy.ops.object.transform_apply(location=False, rotation=True, scale=False) bpy.ops.object.transform_apply(location=False, rotation=False, scale=True) except: pass l = ob.location if ob.type == "CURVE": for c in ob.data.splines: maxx, minx, maxy, miny, maxz, minz = -10000, 10000, -10000, 10000, -10000, 10000 for p in c.points: if o.drill_type == "ALL_POINTS": chunks.append(CamPathChunk([(p.co.x + l.x, p.co.y + l.y, p.co.z + l.z)])) minx = min(p.co.x, minx) maxx = max(p.co.x, maxx) miny = min(p.co.y, miny) maxy = max(p.co.y, maxy) minz = min(p.co.z, minz) maxz = max(p.co.z, maxz) for p in c.bezier_points: if o.drill_type == "ALL_POINTS": chunks.append(CamPathChunk([(p.co.x + l.x, p.co.y + l.y, p.co.z + l.z)])) minx = min(p.co.x, minx) maxx = max(p.co.x, maxx) miny = min(p.co.y, miny) maxy = max(p.co.y, maxy) minz = min(p.co.z, minz) maxz = max(p.co.z, maxz) cx = (maxx + minx) / 2 cy = (maxy + miny) / 2 cz = (maxz + minz) / 2 center = (cx, cy) aspect = (maxx - minx) / (maxy - miny) if ( 1.3 > aspect > 0.7 and o.drill_type == "MIDDLE_SYMETRIC" ) or o.drill_type == "MIDDLE_ALL": chunks.append(CamPathChunk([(center[0] + l.x, center[1] + l.y, cz + l.z)])) elif ob.type == "MESH": for v in ob.data.vertices: chunks.append(CamPathChunk([(v.co.x + l.x, v.co.y + l.y, v.co.z + l.z)])) delete_object(ob) # delete temporary object with applied transforms layers = get_layers(o, o.max_z, check_min_z(o)) chunklayers = [] for layer in layers: for chunk in chunks: # If using object for minz then use z from points in object if o.min_z_from == "OBJECT": z = chunk.get_point(0)[2] else: # using operation minz z = o.min_z # only add a chunk layer if the chunk z point is in or lower than the layer if z <= layer[0]: if z <= layer[1]: z = layer[1] # perform peck drill newchunk = chunk.copy() newchunk.set_z(z) chunklayers.append(newchunk) # retract tool to maxz (operation depth start in ui) newchunk = chunk.copy() newchunk.set_z(o.max_z) chunklayers.append(newchunk) chunklayers = await sort_chunks(chunklayers, o) chunks_to_mesh(chunklayers, o)
[docs] async def medial_axis(o): """Generate the medial axis for a given operation. This function computes the medial axis of the specified operation, which involves processing various cutter types and their parameters. It starts by removing any existing medial mesh, then calculates the maximum depth based on the cutter type and its properties. The function refines curves and computes the Voronoi diagram for the points derived from the operation's silhouette. It filters points and edges based on their positions relative to the computed shapes, and generates a mesh representation of the medial axis. Finally, it handles layers and optionally adds a pocket operation if specified. Args: o (Operation): An object containing parameters for the operation, including cutter type, dimensions, and other relevant properties. Returns: dict: A dictionary indicating the completion status of the operation. Raises: CamException: If an unsupported cutter type is provided or if the input curve is not closed. """ print("Operation: Medial Axis") remove_multiple("medialMesh") from .voronoi import Site, compute_voronoi_diagram chunks = [] gpoly = spolygon.Polygon() angle = o.cutter_tip_angle slope = tan(pi * (90 - angle / 2) / 180) # angle in degrees # slope = tan((pi-angle)/2) #angle in radian new_cutter_diameter = o.cutter_diameter m_o_ob = o.object_name if o.cutter_type == "VCARVE": angle = o.cutter_tip_angle # start the max depth calc from the "start depth" of the operation. maxdepth = o.max_z - slope * o.cutter_diameter / 2 - o.skin # don't cut any deeper than the "end depth" of the operation. if maxdepth < o.min_z: maxdepth = o.min_z # the effective cutter diameter can be reduced from it's max # since we will be cutting shallower than the original maxdepth # without this, the curve is calculated as if the diameter was at the original maxdepth and we get the bit # pulling away from the desired cut surface new_cutter_diameter = (maxdepth - o.max_z) / (-slope) * 2 elif o.cutter_type == "BALLNOSE": maxdepth = -new_cutter_diameter / 2 - o.skin else: raise CamException("Only Ballnose and V-carve Cutters Are Supported for Medial Axis.") # remember resolutions of curves, to refine them, # otherwise medial axis computation yields too many branches in curved parts resolutions_before = [] for ob in o.objects: if ob.type == "CURVE": if ob.data.splines and ob.data.splines[0].type == "BEZIER": activate(ob) bpy.ops.object.curve_remove_doubles(merge_distance=0.0001, keep_bezier=True) else: bpy.ops.object.curve_remove_doubles() for ob in o.objects: if ob.type == "CURVE" or ob.type == "FONT": resolutions_before.append(ob.data.resolution_u) if ob.data.resolution_u < 64: ob.data.resolution_u = 64 polys = get_operation_silhouette(o) if isinstance(polys, list): if len(polys) == 1 and isinstance(polys[0], shapely.MultiPolygon): mpoly = polys[0] else: mpoly = sgeometry.MultiPolygon(polys) elif isinstance(polys, shapely.MultiPolygon): # just a multipolygon mpoly = polys else: raise CamException("Failed Getting Object Silhouette. Is Input Curve Closed?") mpoly_boundary = mpoly.boundary ipol = 0 for poly in mpoly.geoms: ipol = ipol + 1 schunks = shapely_to_chunks(poly, -1) schunks = chunks_refine_threshold( schunks, o.medial_axis_subdivision, o.medial_axis_threshold ) # chunks_refine(schunks,o) verts = [] for ch in schunks: verts.extend(ch.get_points()) # for pt in ch.get_points(): # # pvoro = Site(pt[0], pt[1]) # verts.append(pt) # (pt[0], pt[1]), pt[2]) # verts= points#[[vert.x, vert.y, vert.z] for vert in vertsPts] nDupli, nZcolinear = unique(verts) nVerts = len(verts) print(str(nDupli) + " Duplicate Points Ignored") print(str(nZcolinear) + " Z Colinear Points Excluded") if nVerts < 3: print("Not Enough Points") return {"FINISHED"} # Check colinear xValues = [pt[0] for pt in verts] yValues = [pt[1] for pt in verts] if check_equal(xValues) or check_equal(yValues): print("Points Are Colinear") return {"FINISHED"} # Create diagram print("Tesselation... (" + str(nVerts) + " Points)") xbuff, ybuff = 5, 5 # % zPosition = 0 vertsPts = [Point(vert[0], vert[1], vert[2]) for vert in verts] # vertsPts= [Point(vert[0], vert[1]) for vert in verts] pts, edgesIdx = compute_voronoi_diagram( vertsPts, xbuff, ybuff, polygonsOutput=False, formatOutput=True ) # pts=[[pt[0], pt[1], zPosition] for pt in pts] newIdx = 0 vertr = [] filteredPts = [] print("Filter Points") ipts = 0 for p in pts: ipts = ipts + 1 if ipts % 500 == 0: sys.stdout.write("\r") # the exact output you're looking for: prog_message = f"Points: {ipts} / {len(pts)} {round(100 * ipts / len(pts))}%\n" sys.stdout.write(prog_message) sys.stdout.flush() if not poly.contains(sgeometry.Point(p)): vertr.append((True, -1)) else: vertr.append((False, newIdx)) if o.cutter_type == "VCARVE": # start the z depth calc from the "start depth" of the operation. z = o.max_z - mpoly.boundary.distance(sgeometry.Point(p)) * slope if z < maxdepth: z = maxdepth elif o.cutter_type == "BALL" or o.cutter_type == "BALLNOSE": d = mpoly_boundary.distance(sgeometry.Point(p)) r = new_cutter_diameter / 2.0 if d >= r: z = -r else: # print(r, d) z = -r + sqrt(r * r - d * d) else: z = 0 # # print(mpoly.distance(sgeometry.Point(0,0))) # if(z!=0):print(z) filteredPts.append((p[0], p[1], z)) newIdx += 1 print("Filter Edges") filteredEdgs = [] ledges = [] for e in edgesIdx: do = True # p1 = pts[e[0]] # p2 = pts[e[1]] # print(p1,p2,len(vertr)) if vertr[e[0]][0]: # exclude edges with allready excluded points do = False elif vertr[e[1]][0]: do = False if do: filteredEdgs.append((vertr[e[0]][1], vertr[e[1]][1])) ledges.append( sgeometry.LineString((filteredPts[vertr[e[0]][1]], filteredPts[vertr[e[1]][1]])) ) # print(ledges[-1].has_z) bufpoly = poly.buffer(-new_cutter_diameter / 2, resolution=64) lines = shapely.ops.linemerge(ledges) # print(lines.type) if bufpoly.geom_type == "Polygon" or bufpoly.geom_type == "MultiPolygon": lines = lines.difference(bufpoly) chunks.extend(shapely_to_chunks(bufpoly, maxdepth)) chunks.extend(shapely_to_chunks(lines, 0)) # generate a mesh from the medial calculations if o.add_mesh_for_medial: shapely_to_curve("medialMesh", lines, 0.0) bpy.ops.object.convert(target="MESH") oi = 0 for ob in o.objects: if ob.type == "CURVE" or ob.type == "FONT": ob.data.resolution_u = resolutions_before[oi] oi += 1 # bpy.ops.object.join() chunks = await sort_chunks(chunks, o) layers = get_layers(o, o.max_z, o.min.z) chunklayers = [] for layer in layers: for chunk in chunks: if chunk.is_below_z(layer[0]): newchunk = chunk.copy() newchunk.clamp_z(layer[1]) chunklayers.append(newchunk) if o.first_down: chunklayers = await sort_chunks(chunklayers, o) if o.add_mesh_for_medial: # make curve instead of a path join_multiple("medialMesh") chunks_to_mesh(chunklayers, o) # add pocket operation for medial if add pocket checked if o.add_pocket_for_medial: # o.add_pocket_for_medial = False # export medial axis parameter to pocket op add_pocket(maxdepth, m_o_ob, new_cutter_diameter)
[docs] def get_layers(operation, startdepth, enddepth): """Returns a list of layers bounded by start depth and end depth. This function calculates the layers between the specified start and end depths based on the step down value defined in the operation. If the operation is set to use layers, it computes the number of layers by dividing the difference between start and end depths by the step down value. The function raises an exception if the start depth is lower than the end depth. Args: operation (object): An object that contains the properties `use_layers`, `stepdown`, and `maxz` which are used to determine how layers are generated. startdepth (float): The starting depth for layer calculation. enddepth (float): The ending depth for layer calculation. Returns: list: A list of layers, where each layer is represented as a list containing the start and end depths of that layer. Raises: CamException: If the start depth is lower than the end depth. """ if startdepth < enddepth: raise CamException( "Start Depth Is Lower than End Depth. " "if You Have Set a Custom Depth End, It Must Be Lower than Depth Start, " "and Should Usually Be Negative. Set This in the CAM Operation Area Panel." ) if operation.use_layers: layers = [] n = ceil((startdepth - enddepth) / operation.stepdown) print(f"Start Depth: {startdepth}") print(f"End Depth: {enddepth}") print(f"Layers: {n}") layerstart = operation.max_z for x in range(0, n): layerend = round(max(startdepth - ((x + 1) * operation.stepdown), enddepth), 6) if int(layerstart * 10**8) != int(layerend * 10**8): # it was possible that with precise same end of operation, # last layer was done 2x on exactly same level... layers.append([layerstart, layerend]) layerstart = layerend else: layers = [[round(startdepth, 6), round(enddepth, 6)]] return layers
[docs] def chunks_to_mesh(chunks, o): """Convert sampled chunks into a mesh path for a given optimization object. This function takes a list of sampled chunks and converts them into a mesh path based on the specified optimization parameters. It handles different machine axes configurations and applies optimizations as needed. The resulting mesh is created in the Blender context, and the function also manages the lifting and dropping of the cutter based on the chunk positions. Args: chunks (list): A list of chunk objects to be converted into a mesh. o (object): An object containing optimization parameters and settings. Returns: None: The function creates a mesh in the Blender context but does not return a value. """ t = time.time() s = bpy.context.scene m = s.cam_machine verts = [] free_height = o.movement.free_height # o.max.z + if o.machine_axes == "3": if m.use_position_definitions: origin = (m.starting_position.x, m.starting_position.y, m.starting_position.z) # dhull else: origin = (0, 0, free_height) verts = [origin] if o.machine_axes != "3": verts_rotations = [] # (0,0,0) if (o.machine_axes == "5" and o.strategy_5_axis == "INDEXED") or ( o.machine_axes == "4" and o.strategy_4_axis == "INDEXED" ): extend_chunks_5_axis(chunks, o) if o.array: nchunks = [] for x in range(0, o.array_x_count): for y in range(0, o.array_y_count): print(x, y) for ch in chunks: ch = ch.copy() ch.shift(x * o.array_x_distance, y * o.array_y_distance, 0) nchunks.append(ch) chunks = nchunks progress("Building Paths from Chunks") e = 0.0001 lifted = True for chi in range(0, len(chunks)): ch = chunks[chi] # print(chunks) # print (ch) # TODO: there is a case where parallel+layers+zigzag ramps send empty chunks here... if ch.count() > 0: # print(len(ch.points)) nverts = [] if o.optimisation.optimize: ch = optimize_chunk(ch, o) # lift and drop if ( lifted ): # did the cutter lift before? if yes, put a new position above of the first point of next chunk. if ( o.machine_axes == "3" or (o.machine_axes == "5" and o.strategy_5_axis == "INDEXED") or (o.machine_axes == "4" and o.strategy_4_axis == "INDEXED") ): v = (ch.get_point(0)[0], ch.get_point(0)[1], free_height) else: # otherwise, continue with the next chunk without lifting/dropping v = ch.startpoints[0] # startpoints=retract points verts_rotations.append(ch.rotations[0]) verts.append(v) # add whole chunk verts.extend(ch.get_points()) # add rotations for n-axis if o.machine_axes != "3": verts_rotations.extend(ch.rotations) lift = True # check if lifting should happen if chi < len(chunks) - 1 and chunks[chi + 1].count() > 0: # TODO: remake this for n axis, and this check should be somewhere else... last = Vector(ch.get_point(-1)) first = Vector(chunks[chi + 1].get_point(0)) vect = first - last if ( o.machine_axes == "3" and (o.strategy == "PARALLEL" or o.strategy == "CROSS") and vect.z == 0 and vect.length < o.distance_between_paths * 2.5 ) or (o.machine_axes == "4" and vect.length < o.distance_between_paths * 2.5): # case of neighbouring paths lift = False # case of stepdown by cutting. if abs(vect.x) < e and abs(vect.y) < e: lift = False if lift: if ( o.machine_axes == "3" or (o.machine_axes == "5" and o.strategy_5_axis == "INDEXED") or (o.machine_axes == "4" and o.strategy_4_axis == "INDEXED") ): v = (ch.get_point(-1)[0], ch.get_point(-1)[1], free_height) else: v = ch.startpoints[-1] verts_rotations.append(ch.rotations[-1]) verts.append(v) lifted = lift # print(verts_rotations) if o.optimisation.use_exact and not o.optimisation.use_opencamlib: cleanup_bullet_collision(o) print(time.time() - t) t = time.time() # actual blender object generation starts here: edges = [] for a in range(0, len(verts) - 1): edges.append((a, a + 1)) oname = "cam_path_{}".format(o.name) mesh = bpy.data.meshes.new(oname) mesh.name = oname mesh.from_pydata(verts, edges, []) if oname in s.objects: s.objects[oname].data = mesh ob = s.objects[oname] else: ob = object_utils.object_data_add(bpy.context, mesh, operator=None) if o.machine_axes != "3": # store rotations into shape keys, only way to store large arrays with correct floating point precision # - object/mesh attributes can only store array up to 32000 intems. ob.shape_key_add() ob.shape_key_add() shapek = mesh.shape_keys.key_blocks[1] shapek.name = "rotations" print(len(shapek.data)) print(len(verts_rotations)) # TODO: optimize this. this is just rewritten too many times... for i, co in enumerate(verts_rotations): shapek.data[i].co = co print(time.time() - t) ob.location = (0, 0, 0) o.path_object_name = oname # parent the path object to source object if object mode if (o.geometry_source == "OBJECT") and o.parent_path_to_object: activate(o.objects[0]) ob.select_set(state=True, view_layer=None) bpy.ops.object.parent_set(type="OBJECT", keep_transform=True) else: ob.select_set(state=True, view_layer=None)
[docs] def check_min_z(o): """Check the minimum value based on the specified condition. This function evaluates the 'minz_from' attribute of the input object 'o'. If 'minz_from' is set to 'MATERIAL', it returns the value of 'min.z'. Otherwise, it returns the value of 'minz'. Args: o (object): An object that has attributes 'minz_from', 'min', and 'minz'. Returns: The minimum value, which can be either 'o.min.z' or 'o.min_z' depending on the condition. """ if o.min_z_from == "MATERIAL": return o.min.z else: return o.min_z