blender/release/scripts/startup/bl_operators/nla.py

171 lines
5.1 KiB
Python
Raw Normal View History

2010-04-01 09:29:35 +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 compliant>
import bpy
2010-04-04 14:52:15 +00:00
2010-04-01 09:29:35 +00:00
def pose_info():
from mathutils import Matrix
2010-04-01 09:29:35 +00:00
info = {}
obj = bpy.context.object
pose = obj.pose
pose_items = pose.bones.items()
for name, pbone in pose_items:
binfo = {}
bone = pbone.bone
binfo["parent"] = getattr(bone.parent, "name", None)
binfo["bone"] = bone
binfo["pbone"] = pbone
binfo["matrix_local"] = bone.matrix_local.copy()
try:
2011-02-05 07:04:23 +00:00
binfo["matrix_local_inv"] = binfo["matrix_local"].inverted()
2010-04-01 09:29:35 +00:00
except:
binfo["matrix_local_inv"] = Matrix()
2010-04-04 14:52:15 +00:00
2010-04-01 09:29:35 +00:00
binfo["matrix"] = bone.matrix.copy()
binfo["matrix_pose"] = pbone.matrix.copy()
try:
2011-02-05 07:04:23 +00:00
binfo["matrix_pose_inv"] = binfo["matrix_pose"].inverted()
2010-04-01 09:29:35 +00:00
except:
binfo["matrix_pose_inv"] = Matrix()
2010-04-04 14:52:15 +00:00
2010-04-01 09:29:35 +00:00
print(binfo["matrix_pose"])
info[name] = binfo
for name, pbone in pose_items:
binfo = info[name]
binfo_parent = binfo.get("parent", None)
if binfo_parent:
binfo_parent = info[binfo_parent]
2010-04-04 14:52:15 +00:00
2010-04-01 09:29:35 +00:00
matrix = binfo["matrix_pose"]
rest_matrix = binfo["matrix_local"]
if binfo_parent:
2010-04-04 14:52:15 +00:00
matrix = binfo_parent["matrix_pose_inv"] * matrix
rest_matrix = binfo_parent["matrix_local_inv"] * rest_matrix
2010-04-01 09:29:35 +00:00
2011-02-05 07:04:23 +00:00
matrix = rest_matrix.inverted() * matrix
2010-04-01 09:29:35 +00:00
binfo["matrix_key"] = matrix.copy()
return info
2010-04-01 21:44:56 +00:00
def bake(frame_start, frame_end, step=1, only_selected=False):
2010-04-01 09:29:35 +00:00
scene = bpy.context.scene
obj = bpy.context.object
pose = obj.pose
info_ls = []
2010-04-04 14:52:15 +00:00
2010-04-01 21:44:56 +00:00
frame_range = range(frame_start, frame_end + 1, step)
2010-04-01 09:29:35 +00:00
# could spped this up by applying steps here too...
for f in frame_range:
scene.frame_set(f)
2010-04-04 14:52:15 +00:00
2010-04-01 09:29:35 +00:00
info = pose_info()
info_ls.append(info)
f += 1
action = bpy.data.actions.new("Action")
bpy.context.object.animation_data.action = action
pose_items = pose.bones.items()
for name, pbone in pose_items:
if only_selected and not pbone.bone.select:
2010-04-01 09:29:35 +00:00
continue
for f in frame_range:
2010-04-04 14:52:15 +00:00
matrix = info_ls[int((f - frame_start) / step)][name]["matrix_key"]
2010-04-01 09:29:35 +00:00
2011-02-05 07:04:23 +00:00
#pbone.location = matrix.to_translation()
#pbone.rotation_quaternion = matrix.to_quaternion()
pbone.matrix_basis = matrix
pbone.keyframe_insert("location", -1, f, name)
2010-04-04 14:52:15 +00:00
2010-04-01 09:29:35 +00:00
rotation_mode = pbone.rotation_mode
2010-04-04 14:52:15 +00:00
2010-04-01 09:29:35 +00:00
if rotation_mode == 'QUATERNION':
pbone.keyframe_insert("rotation_quaternion", -1, f, name)
2010-04-01 09:29:35 +00:00
elif rotation_mode == 'AXIS_ANGLE':
pbone.keyframe_insert("rotation_axis_angle", -1, f, name)
2010-09-07 15:17:42 +00:00
else: # euler, XYZ, ZXY etc
pbone.keyframe_insert("rotation_euler", -1, f, name)
2010-04-01 09:29:35 +00:00
pbone.keyframe_insert("scale", -1, f, name)
2010-04-01 09:29:35 +00:00
return action
from bpy.props import IntProperty, BoolProperty
2010-04-01 09:29:35 +00:00
class BakeAction(bpy.types.Operator):
'''Bake animation to an Action'''
2010-04-01 09:29:35 +00:00
bl_idname = "nla.bake"
bl_label = "Bake Action"
bl_options = {'REGISTER', 'UNDO'}
2010-04-01 21:44:56 +00:00
frame_start = IntProperty(name="Start Frame",
2010-04-01 09:29:35 +00:00
description="Start frame for baking",
default=1, min=1, max=300000)
2010-04-01 21:44:56 +00:00
frame_end = IntProperty(name="End Frame",
2010-04-01 09:29:35 +00:00
description="End frame for baking",
default=250, min=1, max=300000)
step = IntProperty(name="Frame Step",
description="Frame Step",
default=1, min=1, max=120)
only_selected = BoolProperty(name="Only Selected",
default=True)
def execute(self, context):
action = bake(self.frame_start, self.frame_end, self.step, self.only_selected)
2010-04-04 14:52:15 +00:00
2010-04-01 09:29:35 +00:00
# basic cleanup, could move elsewhere
for fcu in action.fcurves:
keyframe_points = fcu.keyframe_points
i = 1
while i < len(fcu.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
return {'FINISHED'}
def invoke(self, context, event):
wm = context.window_manager
2010-04-01 09:29:35 +00:00
return wm.invoke_props_dialog(self)