| 
									
										
										
										
											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'} |