cahnged actions name and custom panel tab
0.7.1 - customizable panel category name - Change generated action name: - `expanded` -> `baked` - `autogen` -> `pinned`master
parent
cf8fdc0f71
commit
427a3ad4b2
|
@ -0,0 +1,63 @@
|
||||||
|
# Changelog
|
||||||
|
|
||||||
|
0.7.1
|
||||||
|
|
||||||
|
- customizable panel category name
|
||||||
|
- Change generated action name:
|
||||||
|
- `expanded` -> `baked`
|
||||||
|
- `autogen` -> `pinned`
|
||||||
|
|
||||||
|
0.7.0
|
||||||
|
|
||||||
|
- auto-detect foot to use for path animation
|
||||||
|
- button to go back and forth between curve edit and armature pose mode
|
||||||
|
- UI revamp showing better separation of tool categories
|
||||||
|
|
||||||
|
0.6.0
|
||||||
|
|
||||||
|
- World paste and `Jump next frame`
|
||||||
|
- more compact and improved ui for world paste
|
||||||
|
- fix errors in some operators
|
||||||
|
|
||||||
|
0.5.0
|
||||||
|
|
||||||
|
- pin feet working
|
||||||
|
|
||||||
|
0.4.2
|
||||||
|
|
||||||
|
- context manager for `expand cycle step` store / restore
|
||||||
|
|
||||||
|
0.4.1
|
||||||
|
|
||||||
|
- update fcurve after bake
|
||||||
|
|
||||||
|
0.4.0
|
||||||
|
|
||||||
|
- better curve creation
|
||||||
|
|
||||||
|
0.3.5
|
||||||
|
|
||||||
|
- partly working
|
||||||
|
|
||||||
|
0.3.3
|
||||||
|
|
||||||
|
- total wip
|
||||||
|
|
||||||
|
0.3.0
|
||||||
|
|
||||||
|
- Rework interface
|
||||||
|
- Add manual bone pinning: simple world space copy/paste + pose and jump prev/next
|
||||||
|
- Faster pin feets
|
||||||
|
- bool prop to disable end/curve stepping (interpolation as linear if needed)
|
||||||
|
- Switch action back to new curve when re-snapping to ground
|
||||||
|
- fix shrinkwrap
|
||||||
|
- fix rebake with linear
|
||||||
|
- error message when no fcurve has anim cycle
|
||||||
|
|
||||||
|
0.2.0:
|
||||||
|
|
||||||
|
- first working version
|
||||||
|
|
||||||
|
0.1.0:
|
||||||
|
|
||||||
|
- initial commit, halfway there
|
|
@ -81,8 +81,8 @@ def find_best_foot(ob):
|
||||||
return ('ERROR', f'No action active on {ob.name}')
|
return ('ERROR', f'No action active on {ob.name}')
|
||||||
|
|
||||||
# use original action as ref
|
# use original action as ref
|
||||||
if '_expanded' in act.name:
|
if '_baked' in act.name:
|
||||||
base_act_name = act.name.split('_expanded')[0]
|
base_act_name = act.name.split('_baked')[0]
|
||||||
base_act = bpy.data.actions.get(base_act_name)
|
base_act = bpy.data.actions.get(base_act_name)
|
||||||
if base_act:
|
if base_act:
|
||||||
act = base_act
|
act = base_act
|
||||||
|
@ -161,8 +161,8 @@ def anim_path_from_translate():
|
||||||
return ('ERROR', f'No action active on {ob.name}')
|
return ('ERROR', f'No action active on {ob.name}')
|
||||||
|
|
||||||
# use original action as ref
|
# use original action as ref
|
||||||
if '_expanded' in act.name:
|
if '_baked' in act.name:
|
||||||
base_act_name = act.name.split('_expanded')[0]
|
base_act_name = act.name.split('_baked')[0]
|
||||||
base_act = bpy.data.actions.get(base_act_name)
|
base_act = bpy.data.actions.get(base_act_name)
|
||||||
if base_act:
|
if base_act:
|
||||||
act = base_act
|
act = base_act
|
||||||
|
@ -362,7 +362,7 @@ class UAC_OT_adjust_animation_length(bpy.types.Operator):
|
||||||
self.report({'ERROR'}, f'No action on {ob.name} data')
|
self.report({'ERROR'}, f'No action on {ob.name} data')
|
||||||
return {"CANCELLED"}
|
return {"CANCELLED"}
|
||||||
|
|
||||||
# if '_expanded' in self.act.name:
|
# if '_baked' in self.act.name:
|
||||||
# self.report({'WARNING'}, f'Action is expanded')
|
# self.report({'WARNING'}, f'Action is expanded')
|
||||||
|
|
||||||
const = root.constraints.get("Follow Path")
|
const = root.constraints.get("Follow Path")
|
||||||
|
|
|
@ -14,7 +14,7 @@ def bake_cycle(on_selection=True):
|
||||||
print('ERROR', 'active is not an armature type')
|
print('ERROR', 'active is not an armature type')
|
||||||
return
|
return
|
||||||
|
|
||||||
act = fn.set_expanded_action(obj)
|
act = fn.set_baked_action(obj)
|
||||||
if not act:
|
if not act:
|
||||||
return
|
return
|
||||||
|
|
||||||
|
@ -107,7 +107,7 @@ def bake_cycle(on_selection=True):
|
||||||
ct += 1
|
ct += 1
|
||||||
|
|
||||||
if ct_fcu == ct_no_cycle: # skipped because no cycle exists
|
if ct_fcu == ct_no_cycle: # skipped because no cycle exists
|
||||||
rexpand = re.compile(r'_expanded\.?\d{0,3}$')
|
rexpand = re.compile(r'_baked\.?\d{0,3}$')
|
||||||
if rexpand.search(act.name):
|
if rexpand.search(act.name):
|
||||||
# is an autogenerated one
|
# is an autogenerated one
|
||||||
org_action_name = rexpand.sub('', act.name)
|
org_action_name = rexpand.sub('', act.name)
|
||||||
|
@ -246,7 +246,7 @@ def pin_down_feets():
|
||||||
debug = fn.get_addon_prefs().debug
|
debug = fn.get_addon_prefs().debug
|
||||||
scn = bpy.context.scene
|
scn = bpy.context.scene
|
||||||
# Delete current action if its not the main one
|
# Delete current action if its not the main one
|
||||||
# create a new '_autogen' one
|
# create a new '_pinned' one
|
||||||
act = fn.set_generated_action(obj)
|
act = fn.set_generated_action(obj)
|
||||||
if not act:
|
if not act:
|
||||||
return ('ERROR', f'No action on {obj.name}')
|
return ('ERROR', f'No action on {obj.name}')
|
||||||
|
|
57
README.md
57
README.md
|
@ -52,61 +52,4 @@ Bonus:
|
||||||
- align curve to root
|
- align curve to root
|
||||||
- Put curve forward motion on bones modifier's offset value as negative time offset (instead of using curve )
|
- Put curve forward motion on bones modifier's offset value as negative time offset (instead of using curve )
|
||||||
-->
|
-->
|
||||||
---
|
|
||||||
|
|
||||||
## Changelog:
|
|
||||||
|
|
||||||
0.7.0
|
|
||||||
|
|
||||||
- auto-detect foot to use for path animation
|
|
||||||
- button to go back and forth between curve edit and armature pose mode
|
|
||||||
- UI revamp showing better separation of tool categories
|
|
||||||
|
|
||||||
0.6.0
|
|
||||||
|
|
||||||
- World paste and `Jump next frame`
|
|
||||||
- more compact and improved ui for world paste
|
|
||||||
- fix errors in some operators
|
|
||||||
|
|
||||||
0.5.0
|
|
||||||
|
|
||||||
- pin feet working
|
|
||||||
|
|
||||||
0.4.2
|
|
||||||
|
|
||||||
- context manager for `expand cycle step` store / restore
|
|
||||||
|
|
||||||
0.4.1
|
|
||||||
|
|
||||||
- update fcurve after bake
|
|
||||||
|
|
||||||
0.4.0
|
|
||||||
|
|
||||||
- better curve creation
|
|
||||||
|
|
||||||
0.3.5
|
|
||||||
|
|
||||||
- partly working
|
|
||||||
|
|
||||||
0.3.3
|
|
||||||
|
|
||||||
- total wip
|
|
||||||
|
|
||||||
0.3.0
|
|
||||||
|
|
||||||
- Rework interface
|
|
||||||
- Add manual bone pinning: simple world space copy/paste + pose and jump prev/next
|
|
||||||
- Faster pin feets
|
|
||||||
- bool prop to disable end/curve stepping (interpolation as linear if needed)
|
|
||||||
- Switch action back to new curve when re-snapping to ground
|
|
||||||
- fix shrinkwrap
|
|
||||||
- fix rebake with linear
|
|
||||||
- error message when no fcurve has anim cycle
|
|
||||||
|
|
||||||
0.2.0:
|
|
||||||
|
|
||||||
- first working version
|
|
||||||
|
|
||||||
0.1.0:
|
|
||||||
|
|
||||||
- initial commit, halfway there
|
|
|
@ -4,7 +4,7 @@ bl_info = {
|
||||||
"name": "Unfold Anim Cycle",
|
"name": "Unfold Anim Cycle",
|
||||||
"description": "Anim tools to develop walk/run cycles along a curve",
|
"description": "Anim tools to develop walk/run cycles along a curve",
|
||||||
"author": "Samuel Bernou",
|
"author": "Samuel Bernou",
|
||||||
"version": (0, 7, 0),
|
"version": (0, 7, 1),
|
||||||
"blender": (3, 0, 0),
|
"blender": (3, 0, 0),
|
||||||
"location": "View3D",
|
"location": "View3D",
|
||||||
"warning": "WIP",
|
"warning": "WIP",
|
||||||
|
@ -34,6 +34,7 @@ else:
|
||||||
from . import panels
|
from . import panels
|
||||||
|
|
||||||
import bpy
|
import bpy
|
||||||
|
from . import fn
|
||||||
|
|
||||||
mods = (
|
mods = (
|
||||||
properties,
|
properties,
|
||||||
|
@ -47,12 +48,16 @@ mods = (
|
||||||
panels,
|
panels,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
def register():
|
def register():
|
||||||
if bpy.app.background:
|
if bpy.app.background:
|
||||||
return
|
return
|
||||||
for module in mods:
|
for module in mods:
|
||||||
module.register()
|
module.register()
|
||||||
|
|
||||||
|
panels.update_panel(fn.get_addon_prefs(), bpy.context)
|
||||||
|
|
||||||
def unregister():
|
def unregister():
|
||||||
if bpy.app.background:
|
if bpy.app.background:
|
||||||
return
|
return
|
||||||
|
|
12
fn.py
12
fn.py
|
@ -107,14 +107,14 @@ def get_obj_action(obj):
|
||||||
return act
|
return act
|
||||||
|
|
||||||
def set_generated_action(obj):
|
def set_generated_action(obj):
|
||||||
'''Backup object action and return a new action suffixed '_autogen'
|
'''Backup object action and return a new action suffixed '_pinned'
|
||||||
associated with the object
|
associated with the object
|
||||||
'''
|
'''
|
||||||
print(helper())
|
print(helper())
|
||||||
act = get_obj_action(obj)
|
act = get_obj_action(obj)
|
||||||
if not act: return
|
if not act: return
|
||||||
|
|
||||||
regen = re.compile(r'_autogen\.?\d{0,3}$')
|
regen = re.compile(r'_pinned\.?\d{0,3}$')
|
||||||
|
|
||||||
if regen.search(act.name):
|
if regen.search(act.name):
|
||||||
# is an autogenerated one
|
# is an autogenerated one
|
||||||
|
@ -132,18 +132,18 @@ def set_generated_action(obj):
|
||||||
# backup action before doing anything crazy
|
# backup action before doing anything crazy
|
||||||
act.use_fake_user = True
|
act.use_fake_user = True
|
||||||
new_act = act.copy()
|
new_act = act.copy()
|
||||||
new_act.name = act.name + '_autogen'
|
new_act.name = act.name + '_pinned'
|
||||||
obj.animation_data.action = new_act
|
obj.animation_data.action = new_act
|
||||||
return new_act
|
return new_act
|
||||||
|
|
||||||
|
|
||||||
def set_expanded_action(obj):
|
def set_baked_action(obj):
|
||||||
'''Backup object action and return a new action
|
'''Backup object action and return a new action
|
||||||
associated with the object
|
associated with the object
|
||||||
'''
|
'''
|
||||||
print(helper())
|
print(helper())
|
||||||
|
|
||||||
rexpand = re.compile(r'_expanded\.?\d{0,3}$')
|
rexpand = re.compile(r'_baked\.?\d{0,3}$')
|
||||||
act = obj.animation_data.action
|
act = obj.animation_data.action
|
||||||
|
|
||||||
if rexpand.search(act.name):
|
if rexpand.search(act.name):
|
||||||
|
@ -162,7 +162,7 @@ def set_expanded_action(obj):
|
||||||
# backup action before doing anything crazy
|
# backup action before doing anything crazy
|
||||||
act.use_fake_user = True
|
act.use_fake_user = True
|
||||||
new_act = act.copy()
|
new_act = act.copy()
|
||||||
new_act.name = act.name + '_expanded'
|
new_act.name = act.name + '_baked'
|
||||||
obj.animation_data.action = new_act
|
obj.animation_data.action = new_act
|
||||||
return new_act
|
return new_act
|
||||||
|
|
||||||
|
|
19
panels.py
19
panels.py
|
@ -4,7 +4,7 @@ from . import fn
|
||||||
class UAC_PT_walk_cycle_anim_panel(bpy.types.Panel):
|
class UAC_PT_walk_cycle_anim_panel(bpy.types.Panel):
|
||||||
bl_space_type = "VIEW_3D"
|
bl_space_type = "VIEW_3D"
|
||||||
bl_region_type = "UI"
|
bl_region_type = "UI"
|
||||||
bl_category = "Anim"
|
bl_category = "Walk"
|
||||||
bl_label = "Walk Cycle Anim"
|
bl_label = "Walk Cycle Anim"
|
||||||
|
|
||||||
def draw(self, context):
|
def draw(self, context):
|
||||||
|
@ -86,12 +86,16 @@ class UAC_PT_walk_cycle_anim_panel(bpy.types.Panel):
|
||||||
# Pin feet
|
# Pin feet
|
||||||
col.operator('anim.pin_feets', text='Pin feets', icon='PINNED')
|
col.operator('anim.pin_feets', text='Pin feets', icon='PINNED')
|
||||||
|
|
||||||
|
## show a dropdown allowing to go back to unpinned, unbaked version of the animation
|
||||||
|
# if ob.type == 'ARMATURE':
|
||||||
|
# pass
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
class UAC_PT_anim_tools_panel(bpy.types.Panel):
|
class UAC_PT_anim_tools_panel(bpy.types.Panel):
|
||||||
bl_space_type = "VIEW_3D"
|
bl_space_type = "VIEW_3D"
|
||||||
bl_region_type = "UI"
|
bl_region_type = "UI"
|
||||||
bl_category = "Anim"
|
bl_category = "Walk"
|
||||||
bl_label = "Tools"
|
bl_label = "Tools"
|
||||||
|
|
||||||
def draw(self, context):
|
def draw(self, context):
|
||||||
|
@ -120,12 +124,21 @@ class UAC_PT_anim_tools_panel(bpy.types.Panel):
|
||||||
row.scale_x = 2
|
row.scale_x = 2
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
classes=(
|
classes=(
|
||||||
UAC_PT_walk_cycle_anim_panel,
|
UAC_PT_walk_cycle_anim_panel,
|
||||||
UAC_PT_anim_tools_panel,
|
UAC_PT_anim_tools_panel,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
## Addons Preferences Update Panel
|
||||||
|
def update_panel(self, context):
|
||||||
|
for cls in classes:
|
||||||
|
try:
|
||||||
|
bpy.utils.unregister_class(cls)
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
cls.bl_category = self.category#fn.get_addon_prefs().category
|
||||||
|
bpy.utils.register_class(cls)
|
||||||
|
|
||||||
def register():
|
def register():
|
||||||
for cls in classes:
|
for cls in classes:
|
||||||
bpy.utils.register_class(cls)
|
bpy.utils.register_class(cls)
|
||||||
|
|
|
@ -1,10 +1,17 @@
|
||||||
import bpy
|
import bpy
|
||||||
|
from .panels import update_panel
|
||||||
|
|
||||||
class UAC_addon_prefs(bpy.types.AddonPreferences):
|
class UAC_addon_prefs(bpy.types.AddonPreferences):
|
||||||
## can be just __name__ if prefs are in the __init__ mainfile
|
## can be just __name__ if prefs are in the __init__ mainfile
|
||||||
# Else need the splitext '__name__ = addonname.subfile' (or use a static name)
|
# Else need the splitext '__name__ = addonname.subfile' (or use a static name)
|
||||||
bl_idname = __name__.split('.')[0] # or with: os.path.splitext(__name__)[0]
|
bl_idname = __name__.split('.')[0] # or with: os.path.splitext(__name__)[0]
|
||||||
|
|
||||||
|
category : bpy.props.StringProperty(
|
||||||
|
name="Category",
|
||||||
|
description="Choose a name for the category of the panel",
|
||||||
|
default="Walk",
|
||||||
|
update=update_panel)
|
||||||
|
|
||||||
# some_bool_prop to display in the addon pref
|
# some_bool_prop to display in the addon pref
|
||||||
debug : bpy.props.IntProperty(
|
debug : bpy.props.IntProperty(
|
||||||
name='Debug',
|
name='Debug',
|
||||||
|
@ -37,6 +44,9 @@ class UAC_addon_prefs(bpy.types.AddonPreferences):
|
||||||
layout = self.layout
|
layout = self.layout
|
||||||
layout.use_property_split = True
|
layout.use_property_split = True
|
||||||
|
|
||||||
|
box = layout.box()
|
||||||
|
box.prop(self, "category")
|
||||||
|
|
||||||
box = layout.box()
|
box = layout.box()
|
||||||
box.label(text='Curve creation parameters:')
|
box.label(text='Curve creation parameters:')
|
||||||
box.prop(self, "tgt_bone")
|
box.prop(self, "tgt_bone")
|
||||||
|
|
Loading…
Reference in New Issue