2013-01-23 05:56:49 +00:00
|
|
|
# ##### 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 LICENSE BLOCK #####
|
|
|
|
|
|
|
|
# <pep8-80 compliant>
|
|
|
|
|
|
|
|
import bpy
|
|
|
|
from bpy.types import Operator
|
|
|
|
from bpy.props import IntProperty
|
|
|
|
from bpy.props import EnumProperty
|
|
|
|
|
|
|
|
|
|
|
|
class CopyRigidbodySettings(Operator):
|
|
|
|
'''Copy Rigid Body settings from active object to selected'''
|
|
|
|
bl_idname = "rigidbody.object_settings_copy"
|
2013-01-27 18:14:24 +00:00
|
|
|
bl_label = "Copy Rigid Body Settings"
|
2013-01-23 05:56:49 +00:00
|
|
|
bl_options = {'REGISTER', 'UNDO'}
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def poll(cls, context):
|
2013-01-23 07:52:31 +00:00
|
|
|
obj = context.object
|
2013-01-23 05:56:49 +00:00
|
|
|
return (obj and obj.rigid_body)
|
|
|
|
|
|
|
|
def execute(self, context):
|
|
|
|
obj = context.object
|
2013-01-28 12:15:50 +00:00
|
|
|
scene = context.scene
|
2013-01-23 05:56:49 +00:00
|
|
|
|
|
|
|
# deselect all but mesh objects
|
|
|
|
for o in context.selected_objects:
|
|
|
|
if o.type != 'MESH':
|
|
|
|
o.select = False
|
|
|
|
|
2013-01-28 12:15:50 +00:00
|
|
|
objects = context.selected_objects
|
|
|
|
if objects:
|
2013-01-23 05:56:49 +00:00
|
|
|
# add selected objects to active one groups and recalculate
|
|
|
|
bpy.ops.group.objects_add_active()
|
2013-01-28 12:15:50 +00:00
|
|
|
scene.frame_set(scene.frame_current)
|
2013-01-23 05:56:49 +00:00
|
|
|
|
|
|
|
# copy settings
|
2013-01-28 12:15:50 +00:00
|
|
|
for o in objects:
|
2013-01-23 05:56:49 +00:00
|
|
|
if o.rigid_body is None:
|
|
|
|
continue
|
|
|
|
|
|
|
|
o.rigid_body.type = obj.rigid_body.type
|
|
|
|
o.rigid_body.kinematic = obj.rigid_body.kinematic
|
|
|
|
o.rigid_body.mass = obj.rigid_body.mass
|
|
|
|
o.rigid_body.collision_shape = obj.rigid_body.collision_shape
|
|
|
|
o.rigid_body.use_margin = obj.rigid_body.use_margin
|
|
|
|
o.rigid_body.collision_margin = obj.rigid_body.collision_margin
|
|
|
|
o.rigid_body.friction = obj.rigid_body.friction
|
|
|
|
o.rigid_body.restitution = obj.rigid_body.restitution
|
|
|
|
o.rigid_body.use_deactivation = obj.rigid_body.use_deactivation
|
|
|
|
o.rigid_body.start_deactivated = obj.rigid_body.start_deactivated
|
|
|
|
o.rigid_body.deactivate_linear_velocity = obj.rigid_body.deactivate_linear_velocity
|
|
|
|
o.rigid_body.deactivate_angular_velocity = obj.rigid_body.deactivate_angular_velocity
|
|
|
|
o.rigid_body.linear_damping = obj.rigid_body.linear_damping
|
|
|
|
o.rigid_body.angular_damping = obj.rigid_body.angular_damping
|
|
|
|
o.rigid_body.collision_groups = obj.rigid_body.collision_groups
|
|
|
|
|
|
|
|
return {'FINISHED'}
|
|
|
|
|
|
|
|
|
|
|
|
class BakeToKeyframes(Operator):
|
|
|
|
'''Bake rigid body transformations of selected objects to keyframes'''
|
|
|
|
bl_idname = "rigidbody.bake_to_keyframes"
|
|
|
|
bl_label = "Bake To Keyframes"
|
|
|
|
bl_options = {'REGISTER', 'UNDO'}
|
|
|
|
|
|
|
|
frame_start = IntProperty(
|
|
|
|
name="Start Frame",
|
|
|
|
description="Start frame for baking",
|
|
|
|
min=0, max=300000,
|
|
|
|
default=1,
|
|
|
|
)
|
|
|
|
frame_end = IntProperty(
|
|
|
|
name="End Frame",
|
|
|
|
description="End frame for baking",
|
|
|
|
min=1, max=300000,
|
|
|
|
default=250,
|
|
|
|
)
|
|
|
|
step = IntProperty(
|
|
|
|
name="Frame Step",
|
|
|
|
description="Frame Step",
|
|
|
|
min=1, max=120,
|
|
|
|
default=1,
|
|
|
|
)
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def poll(cls, context):
|
2013-01-23 07:52:31 +00:00
|
|
|
obj = context.object
|
2013-01-23 05:56:49 +00:00
|
|
|
return (obj and obj.rigid_body)
|
|
|
|
|
|
|
|
def execute(self, context):
|
|
|
|
bake = []
|
2013-01-28 12:15:50 +00:00
|
|
|
objects = []
|
2013-01-23 07:52:31 +00:00
|
|
|
scene = context.scene
|
2013-01-23 05:56:49 +00:00
|
|
|
frame_orig = scene.frame_current
|
|
|
|
frames = list(range(self.frame_start, self.frame_end + 1, self.step))
|
|
|
|
|
|
|
|
# filter objects selection
|
2013-01-23 07:52:31 +00:00
|
|
|
for obj in context.selected_objects:
|
|
|
|
if not obj.rigid_body or obj.rigid_body.type != 'ACTIVE':
|
|
|
|
obj.select = False
|
2013-01-23 05:56:49 +00:00
|
|
|
|
2013-01-28 12:15:50 +00:00
|
|
|
objects = context.selected_objects
|
2013-01-23 05:56:49 +00:00
|
|
|
|
2013-01-28 12:15:50 +00:00
|
|
|
if objects:
|
2013-01-23 05:56:49 +00:00
|
|
|
# store transformation data
|
|
|
|
for f in list(range(self.frame_start, self.frame_end + 1)):
|
|
|
|
scene.frame_set(f)
|
|
|
|
if f in frames:
|
|
|
|
mat = {}
|
2013-01-28 12:15:50 +00:00
|
|
|
for i, obj in enumerate(objects):
|
2013-01-23 07:52:31 +00:00
|
|
|
mat[i] = obj.matrix_world.copy()
|
2013-01-23 05:56:49 +00:00
|
|
|
bake.append(mat)
|
|
|
|
|
|
|
|
# apply transformations as keyframes
|
|
|
|
for i, f in enumerate(frames):
|
|
|
|
scene.frame_set(f)
|
2013-01-28 12:15:50 +00:00
|
|
|
obj_prev = objects[0]
|
|
|
|
for j, obj in enumerate(objects):
|
2013-01-23 05:56:49 +00:00
|
|
|
mat = bake[i][j]
|
|
|
|
|
2013-01-23 07:52:31 +00:00
|
|
|
obj.location = mat.to_translation()
|
2013-01-23 05:56:49 +00:00
|
|
|
|
2013-01-23 07:52:31 +00:00
|
|
|
rot_mode = obj.rotation_mode
|
2013-01-23 05:56:49 +00:00
|
|
|
if rot_mode == 'QUATERNION':
|
2013-01-23 07:52:31 +00:00
|
|
|
obj.rotation_quaternion = mat.to_quaternion()
|
2013-01-23 05:56:49 +00:00
|
|
|
elif rot_mode == 'AXIS_ANGLE':
|
|
|
|
# this is a little roundabout but there's no better way right now
|
|
|
|
aa = mat.to_quaternion().to_axis_angle()
|
2013-01-23 07:52:31 +00:00
|
|
|
obj.rotation_axis_angle = (aa[1], ) + aa[0][:]
|
2013-01-23 05:56:49 +00:00
|
|
|
else: # euler
|
|
|
|
# make sure euler rotation is compatible to previous frame
|
2013-01-23 07:52:31 +00:00
|
|
|
obj.rotation_euler = mat.to_euler(rot_mode, obj_prev.rotation_euler)
|
2013-01-23 05:56:49 +00:00
|
|
|
|
2013-01-23 07:52:31 +00:00
|
|
|
obj_prev = obj
|
2013-01-23 05:56:49 +00:00
|
|
|
|
|
|
|
bpy.ops.anim.keyframe_insert(type='BUILTIN_KSI_LocRot', confirm_success=False)
|
|
|
|
|
|
|
|
# remove baked objects from simulation
|
|
|
|
bpy.ops.rigidbody.objects_remove()
|
|
|
|
|
|
|
|
# clean up keyframes
|
2013-01-28 12:15:50 +00:00
|
|
|
for obj in objects:
|
2013-01-23 07:52:31 +00:00
|
|
|
action = obj.animation_data.action
|
2013-01-23 05:56:49 +00:00
|
|
|
for fcu in action.fcurves:
|
|
|
|
keyframe_points = fcu.keyframe_points
|
|
|
|
i = 1
|
|
|
|
# remove unneeded keyframes
|
|
|
|
while i < len(keyframe_points) - 1:
|
|
|
|
val_prev = keyframe_points[i - 1].co[1]
|
|
|
|
val_next = keyframe_points[i + 1].co[1]
|
|
|
|
val = keyframe_points[i].co[1]
|
|
|
|
|
|
|
|
if abs(val - val_prev) + abs(val - val_next) < 0.0001:
|
|
|
|
keyframe_points.remove(keyframe_points[i])
|
|
|
|
else:
|
|
|
|
i += 1
|
|
|
|
# use linear interpolation for better visual results
|
|
|
|
for keyframe in keyframe_points:
|
|
|
|
keyframe.interpolation = 'LINEAR'
|
|
|
|
|
|
|
|
# return to the frame we started on
|
|
|
|
scene.frame_set(frame_orig)
|
|
|
|
|
|
|
|
return {'FINISHED'}
|
|
|
|
|
|
|
|
def invoke(self, context, event):
|
|
|
|
scene = context.scene
|
|
|
|
self.frame_start = scene.frame_start
|
|
|
|
self.frame_end = scene.frame_end
|
|
|
|
|
|
|
|
wm = context.window_manager
|
|
|
|
return wm.invoke_props_dialog(self)
|
|
|
|
|
2013-01-23 05:56:56 +00:00
|
|
|
|
|
|
|
class ConnectRigidBodies(Operator):
|
2013-01-28 11:56:01 +00:00
|
|
|
'''Create rigid body constraints between selected and active rigid bodies'''
|
2013-01-23 05:56:56 +00:00
|
|
|
bl_idname = "rigidbody.connect"
|
2013-01-27 18:14:24 +00:00
|
|
|
bl_label = "Connect Rigid Bodies"
|
2013-01-23 05:56:56 +00:00
|
|
|
bl_options = {'REGISTER', 'UNDO'}
|
|
|
|
|
|
|
|
con_type = EnumProperty(
|
|
|
|
name="Type",
|
2013-01-27 18:14:24 +00:00
|
|
|
description="Type of generated constraint",
|
|
|
|
# XXX Would be nice to get icons too, but currently not possible ;)
|
|
|
|
items=tuple((e.identifier, e.name, e.description, e. value)
|
|
|
|
for e in bpy.types.RigidBodyConstraint.bl_rna.properties["type"].enum_items),
|
2013-01-23 05:56:56 +00:00
|
|
|
default='FIXED',)
|
|
|
|
|
|
|
|
pivot_type = EnumProperty(
|
|
|
|
name="Location",
|
|
|
|
description="Constraint pivot location",
|
|
|
|
items=(('CENTER', "Center", "Pivot location is between the constrained rigid bodies"),
|
|
|
|
('ACTIVE', "Active", "Pivot location is at the active object position"),
|
2013-01-27 18:14:24 +00:00
|
|
|
('SELECTED', "Selected", "Pivot location is at the selected object position")),
|
2013-01-23 05:56:56 +00:00
|
|
|
default='CENTER',)
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def poll(cls, context):
|
2013-01-23 07:52:31 +00:00
|
|
|
obj = context.object
|
2013-01-28 12:15:50 +00:00
|
|
|
objects = context.selected_objects
|
|
|
|
return (obj and obj.rigid_body and (len(objects) > 1))
|
2013-01-23 05:56:56 +00:00
|
|
|
|
|
|
|
def execute(self, context):
|
|
|
|
|
2013-01-28 12:15:50 +00:00
|
|
|
objects = context.selected_objects
|
2013-01-23 07:52:31 +00:00
|
|
|
obj_act = context.active_object
|
2013-01-23 05:56:56 +00:00
|
|
|
|
2013-01-28 12:15:50 +00:00
|
|
|
for obj in objects:
|
2013-01-23 07:52:31 +00:00
|
|
|
if obj == obj_act:
|
2013-01-23 05:56:56 +00:00
|
|
|
continue
|
|
|
|
if self.pivot_type == 'ACTIVE':
|
2013-01-23 07:52:31 +00:00
|
|
|
loc = obj_act.location
|
2013-01-23 05:56:56 +00:00
|
|
|
elif self.pivot_type == 'SELECTED':
|
2013-01-23 07:52:31 +00:00
|
|
|
loc = obj.location
|
2013-01-23 05:56:56 +00:00
|
|
|
else:
|
2013-01-23 07:52:31 +00:00
|
|
|
loc = (obj_act.location + obj.location) / 2.0
|
2013-01-23 05:56:56 +00:00
|
|
|
bpy.ops.object.add(type='EMPTY', view_align=False, enter_editmode=False, location=loc)
|
2013-01-25 06:26:38 +00:00
|
|
|
bpy.ops.rigidbody.constraint_add()
|
2013-01-23 07:52:31 +00:00
|
|
|
con = context.active_object.rigid_body_constraint
|
2013-01-23 05:56:56 +00:00
|
|
|
con.type = self.con_type
|
2013-01-23 07:52:31 +00:00
|
|
|
con.object1 = obj_act
|
|
|
|
con.object2 = obj
|
2013-01-28 11:56:01 +00:00
|
|
|
# restore selection
|
|
|
|
bpy.ops.object.select_all(action='DESELECT')
|
|
|
|
for obj in objs:
|
|
|
|
obj.select = True;
|
|
|
|
bpy.context.scene.objects.active = obj_act
|
2013-01-23 05:56:56 +00:00
|
|
|
|
|
|
|
return {'FINISHED'}
|