blendercam/scripts/addons/cam/collision.py

340 wiersze
15 KiB
Python

# blender CAM collision.py (c) 2012 Vilem Novak
#
# ***** BEGIN GPL LICENSE BLOCK *****
#
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# of the License, or (at your option) any later version.
#
# 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 General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software Foundation,
# Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
#
# ***** END GPL LICENCE BLOCK *****
import bpy
import time
from cam import simple
from cam.simple import *
BULLET_SCALE = 10000
# this is a constant for scaling the rigidbody collision world for higher precision from bullet library
CUTTER_OFFSET = (-5 * BULLET_SCALE, -5 * BULLET_SCALE, -5 * BULLET_SCALE)
# the cutter object has to be present in the scene , so we need to put it aside for sweep collisions,
# otherwise it collides itself.
def getCutterBullet(o):
"""cutter for rigidbody simulation collisions
note that everything is 100x bigger for simulation precision."""
s = bpy.context.scene
if s.objects.get('cutter') is not None:
c = s.objects['cutter']
activate(c)
type = o.cutter_type
if type == 'END':
bpy.ops.mesh.primitive_cylinder_add(vertices=32, radius= o.cutter_diameter / 2,
depth= o.cutter_diameter, end_fill_type='NGON',
align='WORLD', enter_editmode=False, location = CUTTER_OFFSET,
rotation=(0, 0, 0))
cutter = bpy.context.active_object
cutter.scale *= BULLET_SCALE
bpy.ops.object.transform_apply(location=False, rotation=False, scale=True)
bpy.ops.object.origin_set(type='GEOMETRY_ORIGIN', center='BOUNDS')
bpy.ops.rigidbody.object_add(type='ACTIVE')
cutter = bpy.context.active_object
cutter.rigid_body.collision_shape = 'CYLINDER'
elif type == 'BALLNOSE':
# ballnose ending used mainly when projecting from sides.
# the actual collision shape is capsule in this case.
bpy.ops.mesh.primitive_ico_sphere_add(subdivisions = 3, radius = o.cutter_diameter / 2,
align='WORLD', enter_editmode=False,
location = CUTTER_OFFSET, rotation=(0, 0, 0))
cutter = bpy.context.active_object
cutter.scale *= BULLET_SCALE
bpy.ops.object.transform_apply(location=False, rotation=False, scale=True)
bpy.ops.object.origin_set(type='GEOMETRY_ORIGIN', center='BOUNDS')
bpy.ops.rigidbody.object_add(type='ACTIVE')
cutter = bpy.context.active_object
#cutter.dimensions.z = 0.2 * BULLET_SCALE # should be sufficient for now... 20 cm.
cutter.rigid_body.collision_shape = 'CAPSULE'
#bpy.ops.object.transform_apply(location=False, rotation=False, scale=True)
elif type == 'VCARVE':
angle = o.cutter_tip_angle
s = math.tan(math.pi * (90 - angle / 2) / 180) / 2 # angles in degrees
cone_d = o.cutter_diameter * s
bpy.ops.mesh.primitive_cone_add(vertices=32, radius1 = o.cutter_diameter / 2, radius2=0,
depth = cone_d, end_fill_type='NGON',
align='WORLD', enter_editmode=False, location = CUTTER_OFFSET,
rotation=(math.pi, 0, 0))
cutter = bpy.context.active_object
cutter.scale *= BULLET_SCALE
bpy.ops.object.transform_apply(location=False, rotation=False, scale=True)
bpy.ops.object.origin_set(type='GEOMETRY_ORIGIN', center='BOUNDS')
bpy.ops.rigidbody.object_add(type='ACTIVE')
cutter = bpy.context.active_object
cutter.rigid_body.collision_shape = 'CONE'
elif type == 'CYLCONE':
angle = o.cutter_tip_angle
s = math.tan(math.pi * (90 - angle / 2) / 180) / 2 # angles in degrees
cylcone_d =(o.cutter_diameter - o.cylcone_diameter) * s
bpy.ops.mesh.primitive_cone_add(vertices = 32, radius1 = o.cutter_diameter / 2,
radius2 = o.cylcone_diameter / 2,
depth = cylcone_d, end_fill_type='NGON',
align = 'WORLD', enter_editmode = False,
location = CUTTER_OFFSET,rotation = (math.pi, 0, 0))
cutter = bpy.context.active_object
cutter.scale *= BULLET_SCALE
bpy.ops.object.transform_apply(location=False, rotation=False, scale=True)
bpy.ops.object.origin_set(type='GEOMETRY_ORIGIN', center='BOUNDS')
bpy.ops.rigidbody.object_add(type='ACTIVE')
cutter = bpy.context.active_object
cutter.rigid_body.collision_shape = 'CONVEX_HULL'
cutter.location = CUTTER_OFFSET
elif type == 'BALLCONE':
angle = math.radians(o.cutter_tip_angle)/2
cutter_R = o.cutter_diameter/2
Ball_R = o.ball_radius/math.cos(angle)
conedepth = (cutter_R - o.ball_radius)/math.tan(angle)
bpy.ops.curve.simple(align='WORLD', location=(0, 0, 0), rotation=(0, 0, 0),
Simple_Type='Point', use_cyclic_u=False)
oy = Ball_R
for i in range(1 , 10):
ang = -i * (math.pi/2-angle) / 9
qx = math.sin(ang) * oy
qy = oy - math.cos(ang) * oy
bpy.ops.curve.vertex_add(location=(qx , qy , 0))
conedepth += qy
bpy.ops.curve.vertex_add(location=(-cutter_R , conedepth , 0))
#bpy.ops.curve.vertex_add(location=(0 , conedepth , 0))
bpy.ops.object.editmode_toggle()
bpy.ops.object.convert(target='MESH')
bpy.ops.transform.rotate(value = -math.pi / 2, orient_axis = 'X')
bpy.ops.object.transform_apply(location=True, rotation=True, scale=True)
ob = bpy.context.active_object
ob.name = "BallConeTool"
ob_scr = ob.modifiers.new(type='SCREW', name = 'scr')
ob_scr.angle = math.radians(-360)
ob_scr.steps = 32
ob_scr.merge_threshold = 0
ob_scr.use_merge_vertices = True
bpy.ops.object.modifier_apply(modifier='scr')
bpy.data.objects['BallConeTool'].select_set(True)
cutter = bpy.context.active_object
cutter.scale *= BULLET_SCALE
bpy.ops.object.transform_apply(location=False, rotation=False, scale=True)
bpy.ops.object.origin_set(type='GEOMETRY_ORIGIN', center='BOUNDS')
bpy.ops.rigidbody.object_add(type='ACTIVE')
cutter.location = CUTTER_OFFSET
cutter.rigid_body.collision_shape = 'CONVEX_HULL'
cutter.location = CUTTER_OFFSET
elif type == 'CUSTOM':
cutob = bpy.data.objects[o.cutter_object_name]
activate(cutob)
bpy.ops.object.duplicate()
bpy.ops.rigidbody.object_add(type='ACTIVE')
cutter = bpy.context.active_object
scale = o.cutter_diameter / cutob.dimensions.x
cutter.scale *= BULLET_SCALE * scale
bpy.ops.object.transform_apply(location=False, rotation=False, scale=True)
bpy.ops.object.origin_set(type='GEOMETRY_ORIGIN', center='BOUNDS')
# print(cutter.dimensions,scale)
bpy.ops.rigidbody.object_add(type='ACTIVE')
cutter.rigid_body.collision_shape = 'CONVEX_HULL'
cutter.location = CUTTER_OFFSET
cutter.name = 'cam_cutter'
o.cutter_shape = cutter
return cutter
def subdivideLongEdges(ob, threshold):
print('subdividing long edges')
m = ob.data
scale = (ob.scale.x + ob.scale.y + ob.scale.z) / 3
subdivides = []
n = 1
iter = 0
while n > 0:
n = 0
for i, e in enumerate(m.edges):
v1 = m.vertices[e.vertices[0]].co
v2 = m.vertices[e.vertices[1]].co
vec = v2 - v1
l = vec.length
if l * scale > threshold:
n += 1
subdivides.append(i)
if n > 0:
print(len(subdivides))
bpy.ops.object.editmode_toggle()
# bpy.ops.mesh.quads_convert_to_tris(quad_method='BEAUTY', ngon_method='BEAUTY')
# bpy.ops.mesh.tris_convert_to_quads()
bpy.ops.mesh.select_all(action='DESELECT')
bpy.ops.mesh.select_mode(use_extend=False, use_expand=False, type='EDGE')
bpy.ops.object.editmode_toggle()
for i in subdivides:
m.edges[i].select = True
bpy.ops.object.editmode_toggle()
bpy.ops.mesh.subdivide(smoothness=0)
if iter == 0:
bpy.ops.mesh.select_all(action='SELECT')
bpy.ops.mesh.quads_convert_to_tris(quad_method='SHORTEST_DIAGONAL', ngon_method='BEAUTY')
bpy.ops.mesh.select_all(action='DESELECT')
bpy.ops.object.editmode_toggle()
ob.update_from_editmode()
iter += 1
# n=0
#
def prepareBulletCollision(o):
"""prepares all objects needed for sampling with bullet collision"""
progress('preparing collisions')
print(o.name)
active_collection = bpy.context.view_layer.active_layer_collection.collection
t = time.time()
s = bpy.context.scene
s.gravity = (0, 0, 0)
# cleanup rigidbodies wrongly placed somewhere in the scene
for ob in bpy.context.scene.objects:
if ob.rigid_body is not None and (bpy.data.objects.find('machine') > -1
and ob.name not in bpy.data.objects['machine'].objects):
activate(ob)
bpy.ops.rigidbody.object_remove()
for collisionob in o.objects:
bpy.context.view_layer.objects.active = collisionob
collisionob.select_set(state=True)
bpy.ops.object.duplicate(linked=False)
collisionob = bpy.context.active_object
if collisionob.type == 'CURVE' or collisionob.type == 'FONT': # support for curve objects collision
if collisionob.type == 'CURVE':
odata = collisionob.data.dimensions
collisionob.data.dimensions = '2D'
bpy.ops.object.convert(target='MESH', keep_original=False)
if o.use_modifiers:
depsgraph = bpy.context.evaluated_depsgraph_get()
mesh_owner = collisionob.evaluated_get(depsgraph)
newmesh = mesh_owner.to_mesh()
oldmesh = collisionob.data
collisionob.modifiers.clear()
collisionob.data = bpy.data.meshes.new_from_object(mesh_owner.evaluated_get(depsgraph),
preserve_all_data_layers=True,
depsgraph=depsgraph)
bpy.data.meshes.remove(oldmesh)
# subdivide long edges here:
if o.optimisation.exact_subdivide_edges:
subdivideLongEdges(collisionob, o.cutter_diameter * 2)
bpy.ops.rigidbody.object_add(type='ACTIVE')
# using active instead of passive because of performance.TODO: check if this works also with 4axis...
collisionob.rigid_body.collision_shape = 'MESH'
collisionob.rigid_body.kinematic = True
# this fixed a serious bug and gave big speedup, rbs could move since they are now active...
collisionob.rigid_body.collision_margin = o.skin * BULLET_SCALE
bpy.ops.transform.resize(value=(BULLET_SCALE, BULLET_SCALE, BULLET_SCALE),
constraint_axis=(False, False, False), orient_type='GLOBAL', mirror=False,
use_proportional_edit=False, proportional_edit_falloff='SMOOTH', proportional_size=1,
texture_space=False, release_confirm=False)
collisionob.location = collisionob.location * BULLET_SCALE
bpy.ops.object.transform_apply(location=True, rotation=True, scale=True)
bpy.context.view_layer.objects.active = collisionob
if active_collection in collisionob.users_collection:
active_collection.objects.unlink(collisionob)
getCutterBullet(o)
# machine objects scaling up to simulation scale
if bpy.data.objects.find('machine') > -1:
for ob in bpy.data.objects['machine'].objects:
activate(ob)
bpy.ops.transform.resize(value=(BULLET_SCALE, BULLET_SCALE, BULLET_SCALE),
constraint_axis=(False, False, False), orient_type='GLOBAL',
mirror=False, use_proportional_edit=False, proportional_edit_falloff='SMOOTH',
proportional_size=1, texture_space=False,
release_confirm=False)
ob.location = ob.location * BULLET_SCALE
# stepping simulation so that objects are up to date
bpy.context.scene.frame_set(0)
bpy.context.scene.frame_set(1)
bpy.context.scene.frame_set(2)
progress(time.time() - t)
def cleanupBulletCollision(o):
if bpy.data.objects.find('machine') > -1:
machinepresent = True
else:
machinepresent = False
for ob in bpy.context.scene.objects:
if ob.rigid_body is not None and not (machinepresent and ob.name in bpy.data.objects['machine'].objects):
delob(ob)
# machine objects scaling up to simulation scale
if machinepresent:
for ob in bpy.data.objects['machine'].objects:
activate(ob)
bpy.ops.transform.resize(value=(1.0 / BULLET_SCALE, 1.0 / BULLET_SCALE, 1.0 / BULLET_SCALE),
constraint_axis=(False, False, False), orient_type='GLOBAL',
mirror=False, use_proportional_edit=False, proportional_edit_falloff='SMOOTH',
proportional_size=1, texture_space=False,
release_confirm=False)
ob.location = ob.location / BULLET_SCALE
def getSampleBullet(cutter, x, y, radius, startz, endz):
"""collision test for 3 axis milling. Is simplified compared to the full 3d test"""
scene = bpy.context.scene
pos = scene.rigidbody_world.convex_sweep_test(cutter, (x * BULLET_SCALE, y * BULLET_SCALE, startz * BULLET_SCALE),
(x * BULLET_SCALE, y * BULLET_SCALE, endz * BULLET_SCALE))
# radius is subtracted because we are interested in cutter tip position, this gets collision object center
if pos[3] == 1:
return (pos[0][2] - radius) / BULLET_SCALE
else:
return endz - 10
def getSampleBulletNAxis(cutter, startpoint, endpoint, rotation, cutter_compensation):
"""fully 3d collision test for NAxis milling"""
cutterVec = Vector((0, 0, 1)) * cutter_compensation
# cutter compensation vector - cutter physics object has center in the middle, while cam needs the tip position.
cutterVec.rotate(Euler(rotation))
start = (startpoint * BULLET_SCALE + cutterVec).to_tuple()
end = (endpoint * BULLET_SCALE + cutterVec).to_tuple()
pos = bpy.context.scene.rigidbody_world.convex_sweep_test(cutter, start, end)
if pos[3] == 1:
pos = Vector(pos[0])
# rescale and compensate from center to tip.
res = pos / BULLET_SCALE - cutterVec / BULLET_SCALE
return res
else:
return None