2021-04-05 01:35:12 +02:00
|
|
|
import bpy
|
2021-04-06 18:30:25 +02:00
|
|
|
from . import fn
|
2021-04-05 01:35:12 +02:00
|
|
|
|
2022-04-20 12:02:19 +02:00
|
|
|
class AW_PT_walk_cycle_anim_panel(bpy.types.Panel):
|
2021-04-05 01:35:12 +02:00
|
|
|
bl_space_type = "VIEW_3D"
|
|
|
|
bl_region_type = "UI"
|
2022-04-12 11:21:00 +02:00
|
|
|
bl_category = "Walk"
|
2022-04-20 12:02:19 +02:00
|
|
|
bl_label = "Auto Walk"
|
2021-04-05 01:35:12 +02:00
|
|
|
|
|
|
|
def draw(self, context):
|
|
|
|
layout = self.layout
|
2022-04-11 19:46:22 +02:00
|
|
|
prefs = fn.get_addon_prefs()
|
|
|
|
ob = context.object
|
|
|
|
|
2022-03-29 18:46:33 +02:00
|
|
|
settings = context.scene.anim_cycle_settings
|
2022-04-11 19:46:22 +02:00
|
|
|
tweak = settings.tweak
|
2022-03-29 18:46:33 +02:00
|
|
|
# need to know root orientation forward)
|
|
|
|
## know direction to evaluate feet moves
|
|
|
|
## Define Constraint axis (depend on root orientation)
|
2022-04-11 11:41:56 +02:00
|
|
|
|
|
|
|
# layout.prop(settings, "forward_axis") # plain prop
|
2022-04-13 18:38:03 +02:00
|
|
|
col = layout.column()
|
|
|
|
row = col.row()
|
2022-04-11 11:41:56 +02:00
|
|
|
row.label(text='Forward Axis')
|
|
|
|
row.prop(settings, "forward_axis", text='')
|
2022-04-20 12:02:19 +02:00
|
|
|
row.operator('autowalk.open_addon_prefs', icon='PREFERENCES', text='')
|
|
|
|
col.operator("autowalk.autoset_axis", text='Auto-Set Axis') # maybe check for fcruve cycle at the end of autoset axis ? (like a check)
|
|
|
|
col.operator("autowalk.create_cycles_modifiers", text='Add Cycles Modifiers', icon='GRAPH')
|
2022-04-12 19:24:47 +02:00
|
|
|
|
|
|
|
pb = None
|
|
|
|
constrained = False
|
|
|
|
if ob and ob.type == 'ARMATURE':
|
|
|
|
pb = ob.pose.bones.get(prefs.tgt_bone)
|
|
|
|
if pb:
|
|
|
|
follow = pb.constraints.get('Follow Path')
|
|
|
|
if follow and follow.target:
|
|
|
|
constrained = True
|
|
|
|
|
|
|
|
if not settings.path_to_follow and not constrained:
|
2022-04-20 12:02:19 +02:00
|
|
|
layout.operator('autowalk.create_curve_path', text='Create Curve at Root Position', icon='CURVE_BEZCURVE')
|
2022-04-19 15:33:06 +02:00
|
|
|
if (w_co := context.scene.collection.children.get('walk_markers')) and w_co.objects:
|
|
|
|
row=layout.row(align=True)
|
|
|
|
row.label(text=f"1st Point Frame {w_co.objects[0].name.split('_')[-1]}")
|
2022-04-20 12:02:19 +02:00
|
|
|
row.operator('autowalk.remove_a_b_step', text='', icon='X') # Remove
|
|
|
|
layout.operator('autowalk.create_a_b_step', text='Set 2nd Point', icon='CURVE_PATH')
|
2022-04-19 15:33:06 +02:00
|
|
|
else:
|
2022-04-20 12:02:19 +02:00
|
|
|
layout.operator('autowalk.create_a_b_step', text='Set 1st Point', icon='CURVE_PATH')
|
2022-04-19 15:33:06 +02:00
|
|
|
|
2022-04-12 19:24:47 +02:00
|
|
|
else:
|
2022-04-20 12:02:19 +02:00
|
|
|
layout.operator('autowalk.edit_curve', text='Edit Curve', icon='OUTLINER_DATA_CURVE') # FORCE_CURVE
|
2022-03-31 17:07:04 +02:00
|
|
|
|
2022-04-12 19:24:47 +02:00
|
|
|
elif ob and ob.type == 'CURVE':
|
|
|
|
if context.mode in ('OBJECT', 'EDIT_CURVE') \
|
|
|
|
and settings.path_to_follow \
|
|
|
|
and ob == settings.path_to_follow:
|
2022-04-20 12:02:19 +02:00
|
|
|
layout.operator('autowalk.object_from_curve', text='Back To Object', icon='LOOP_BACK')
|
2022-04-12 19:24:47 +02:00
|
|
|
|
2022-04-11 19:46:22 +02:00
|
|
|
box = layout.box()
|
|
|
|
expand_icon = 'TRIA_DOWN' if tweak else 'TRIA_RIGHT'
|
|
|
|
box.prop(settings, 'tweak', text='Curve Options', icon=expand_icon)
|
2022-04-13 18:38:03 +02:00
|
|
|
|
2022-04-11 19:46:22 +02:00
|
|
|
if tweak:
|
|
|
|
#-# path and ground objects
|
2022-04-13 18:38:03 +02:00
|
|
|
box.prop(settings, "tgt_bone", text='Bone')
|
2022-04-11 19:46:22 +02:00
|
|
|
box.prop_search(settings, "path_to_follow", context.scene, "objects")
|
|
|
|
box.prop_search(settings, "gnd", context.scene, "objects")
|
2021-04-06 18:30:25 +02:00
|
|
|
|
2022-04-11 19:46:22 +02:00
|
|
|
row = box.row()
|
2022-04-27 14:54:05 +02:00
|
|
|
if ob and ob.type == 'ARMATURE':
|
|
|
|
row.operator('autowalk.snap_curve_to_ground', text='Snap Curve To Ground', icon='SNAP_ON')
|
|
|
|
elif ob and ob.type == 'CURVE':
|
|
|
|
row.operator('autowalk.snap_selected_curve', text='Snap Selected Curve To Ground', icon='SNAP_ON')
|
|
|
|
else:
|
|
|
|
row.label(text='Select curve or armature to snap', icon='INFO')
|
|
|
|
|
2022-04-11 19:46:22 +02:00
|
|
|
row.active = bool(settings.gnd)
|
|
|
|
|
|
|
|
|
|
|
|
# Determine if already has a constraint (a bit too much condition in a panel...)
|
2022-04-12 19:24:47 +02:00
|
|
|
|
2022-04-11 19:46:22 +02:00
|
|
|
if ob:
|
|
|
|
if ob.type == 'ARMATURE':
|
|
|
|
pb = ob.pose.bones.get(prefs.tgt_bone)
|
|
|
|
if pb:
|
|
|
|
follow = pb.constraints.get('Follow Path')
|
|
|
|
if follow and follow.target:
|
2022-04-20 17:54:49 +02:00
|
|
|
row = box.row()
|
|
|
|
row.label(text=f'{pb.name} -> {follow.target.name}', icon='CON_FOLLOWPATH')
|
|
|
|
row.operator('autowalk.remove_follow_path', text='', icon='X')
|
2022-04-11 19:46:22 +02:00
|
|
|
constrained = True
|
|
|
|
## Put this in a setting popup or submenu
|
|
|
|
# if context.mode == 'POSE':
|
|
|
|
if not constrained:
|
2022-04-20 12:02:19 +02:00
|
|
|
box.operator('autowalk.create_follow_path', text='Add follow path constraint', icon='CON_FOLLOWPATH')
|
2022-04-11 19:46:22 +02:00
|
|
|
|
2021-04-05 01:35:12 +02:00
|
|
|
|
2022-04-11 19:46:22 +02:00
|
|
|
box = layout.box()
|
|
|
|
col=box.column()
|
|
|
|
col.label(text='Motion:')
|
2022-04-20 17:54:49 +02:00
|
|
|
row = col.row(align=True)
|
|
|
|
row.prop(settings, "start_frame", text='Start')
|
|
|
|
row.prop(settings, "end_frame", text='End')
|
2022-04-21 14:25:43 +02:00
|
|
|
col.operator('autowalk.animate_path', text='Move On Curve', icon='ANIM') # Animate Forward Motion
|
2021-04-05 01:35:12 +02:00
|
|
|
|
2022-04-11 19:46:22 +02:00
|
|
|
row=col.row()
|
2022-04-21 14:25:43 +02:00
|
|
|
row.operator('autowalk.adjust_animation_length', text='Adjust Speed', icon='MOD_TIME') # Adjust Forward Speed
|
2021-04-05 01:35:12 +02:00
|
|
|
|
2021-04-06 18:30:25 +02:00
|
|
|
## Bake cycle (on selected)
|
2022-04-11 19:46:22 +02:00
|
|
|
box = layout.box()
|
|
|
|
col=box.column()
|
2022-04-13 18:38:03 +02:00
|
|
|
col.label(text='Action:')
|
|
|
|
|
2022-04-11 19:46:22 +02:00
|
|
|
row=col.row()
|
2022-03-29 18:46:33 +02:00
|
|
|
row.prop(settings, "linear", text='Linear')
|
|
|
|
row.prop(settings, "expand_on_selected_bones")
|
2022-03-31 17:07:04 +02:00
|
|
|
|
|
|
|
txt = 'Bake keys' if settings.linear else 'Bake keys and step path'
|
2022-04-20 12:02:19 +02:00
|
|
|
col.operator('autowalk.bake_cycle_and_step', text=txt, icon='SHAPEKEY_DATA')
|
2021-04-05 01:35:12 +02:00
|
|
|
|
2022-10-15 19:13:50 +02:00
|
|
|
# Custom Axis Pinning
|
|
|
|
subox = col.box()
|
|
|
|
subox.prop(settings, "custom_pin", text='Custom Pinning')
|
|
|
|
if settings.custom_pin:
|
|
|
|
row=subox.row(align=True)
|
|
|
|
row.label(text='Location')
|
|
|
|
row.prop(settings, 'pin_loc_x', text='X', toggle=True)
|
|
|
|
row.prop(settings, 'pin_loc_y', text='Y', toggle=True)
|
|
|
|
row.prop(settings, 'pin_loc_z', text='Z', toggle=True)
|
|
|
|
row=subox.row(align=True)
|
|
|
|
row.label(text='Rotation')
|
|
|
|
row.prop(settings, 'pin_rot_x', text='X', toggle=True)
|
|
|
|
row.prop(settings, 'pin_rot_y', text='Y', toggle=True)
|
|
|
|
row.prop(settings, 'pin_rot_z', text='Z', toggle=True)
|
|
|
|
|
2021-04-05 01:35:12 +02:00
|
|
|
# Pin feet
|
2022-04-20 12:02:19 +02:00
|
|
|
col.operator('autowalk.pin_feets', text='Pin feets', icon='PINNED')
|
2022-04-12 11:21:00 +02:00
|
|
|
|
|
|
|
## show a dropdown allowing to go back to unpinned, unbaked version of the animation
|
2022-04-12 19:24:47 +02:00
|
|
|
if ob and ob.type == 'ARMATURE':
|
2022-04-13 18:38:03 +02:00
|
|
|
anim = ob.animation_data
|
|
|
|
if anim and anim.action and not anim.use_tweak_mode:
|
|
|
|
# skipped if in NLA tweak mode because anim.is_property_readonly('action') = True
|
|
|
|
if 'baked' in anim.action.name or 'pinned' in anim.action.name:
|
|
|
|
col=box.column()
|
2022-04-20 12:02:19 +02:00
|
|
|
col.operator('autowalk.step_back_actions', text='Use Previous Actions', icon= 'ACTION')
|
2021-04-05 01:35:12 +02:00
|
|
|
|
2023-03-08 17:59:49 +01:00
|
|
|
class AW_MT_wrap_animation_help(bpy.types.Menu):
|
|
|
|
# bl_idname = "OBJECT_MT_custom_menu"
|
|
|
|
bl_label = "Wrap Animation Infos"
|
|
|
|
|
|
|
|
def draw(self, context):
|
|
|
|
layout = self.layout
|
|
|
|
col = layout.column()
|
|
|
|
# col.label(text='Wrap animation:', icon='INFO')
|
|
|
|
col.label(text='Apply offset from ref bone to root bone')
|
|
|
|
col.label(text='on all selected pose bones')
|
|
|
|
|
|
|
|
col.separator()
|
|
|
|
col.label(text='Example:')
|
|
|
|
col.label(text='Root is the bone to return to. Often root/walk/world bone')
|
|
|
|
col.label(text='In most cases, Ref bone will be the bottom spine bone')
|
|
|
|
col.label(text='Applying with spine selected will align spine with root on X-Y axis')
|
|
|
|
|
|
|
|
col.separator()
|
|
|
|
col.label(text='Note: If resulted animation is broken', icon='ERROR')
|
|
|
|
col.label(text='try applying with only one selected bone at a time')
|
|
|
|
|
|
|
|
|
|
|
|
class AW_PT_wrap_animation_panel(bpy.types.Panel):
|
|
|
|
bl_space_type = "VIEW_3D"
|
|
|
|
bl_region_type = "UI"
|
|
|
|
bl_category = "Walk"
|
|
|
|
bl_label = "Wrap Anim"
|
|
|
|
bl_options = {'DEFAULT_CLOSED'}
|
|
|
|
|
|
|
|
def draw(self, context):
|
|
|
|
layout = self.layout
|
|
|
|
layout.use_property_split = True
|
|
|
|
col = layout.column()
|
|
|
|
if not context.object or context.object.type != 'ARMATURE':
|
|
|
|
col.label(text='Active object must be of Armature type')
|
|
|
|
return
|
|
|
|
|
|
|
|
settings = context.scene.anim_cycle_settings
|
|
|
|
|
|
|
|
## As pose bone search picker
|
|
|
|
# col.prop_search(context.object, "pose_bone", context.object.pose, "bones")
|
|
|
|
# col.prop_search(settings, "wrap_root_bone", context.object.pose, "bones")
|
|
|
|
# col.prop_search(settings, "wrap_ref_bone", context.object.pose, "bones")
|
|
|
|
|
|
|
|
## As strings
|
|
|
|
# col.label(text='offset selection from reference bone to Root')
|
|
|
|
row = col.row()
|
|
|
|
row.label(text='Enter bones names')
|
|
|
|
row.operator("wm.call_menu", text="", icon='QUESTION').name = "AW_MT_wrap_animation_help"
|
|
|
|
|
|
|
|
col.prop(settings, "wrap_root_bone", text='Root')
|
|
|
|
col.prop(settings, "wrap_ref_bone", text='Ref')
|
|
|
|
col.operator('autowalk.wrap_animation', text='Wrap Animation') # , icon=''
|
|
|
|
|
2022-04-20 12:02:19 +02:00
|
|
|
class AW_PT_anim_tools_panel(bpy.types.Panel):
|
2021-04-06 18:30:25 +02:00
|
|
|
bl_space_type = "VIEW_3D"
|
|
|
|
bl_region_type = "UI"
|
2022-04-12 11:21:00 +02:00
|
|
|
bl_category = "Walk"
|
2022-04-21 14:25:43 +02:00
|
|
|
bl_label = "Pin Tools"
|
2021-04-06 18:30:25 +02:00
|
|
|
|
|
|
|
def draw(self, context):
|
|
|
|
layout = self.layout
|
2022-04-21 14:25:43 +02:00
|
|
|
## FIXME: need to fix vertices get
|
|
|
|
# layout.operator('autowalk.contact_to_ground', text='Ground selected feet', icon='SNAP_OFF')
|
|
|
|
|
2021-04-08 19:25:05 +02:00
|
|
|
row = layout.row()
|
2022-04-27 12:01:39 +02:00
|
|
|
row.operator('pose.world_space_copy', text='Copy Pose', icon='COPYDOWN')
|
2022-04-21 14:25:43 +02:00
|
|
|
|
2022-04-20 12:02:19 +02:00
|
|
|
# row.operator('autowalk.world_space_paste', text='Paste', icon='PASTEDOWN')
|
2022-04-11 11:41:56 +02:00
|
|
|
# row = layout.row(align=False)
|
|
|
|
|
|
|
|
## multi buttons
|
|
|
|
# row.label(text='Paste and jump:')
|
|
|
|
# row = layout.row(align=True)
|
2022-04-20 12:02:19 +02:00
|
|
|
# row.operator('autowalk.world_space_paste_next', text='Prev key', icon='PREV_KEYFRAME').prev = True # Paste Prev key
|
|
|
|
# row.operator('autowalk.world_space_paste_next', text='Next key', icon='NEXT_KEYFRAME').prev = False # Paste Next key
|
2022-04-11 11:41:56 +02:00
|
|
|
# row = layout.row(align=True)
|
2022-04-20 12:02:19 +02:00
|
|
|
# row.operator('autowalk.world_space_paste_next_frame', text='Prev frame', icon='FRAME_PREV').prev = True # Paste Prev frame
|
|
|
|
# row.operator('autowalk.world_space_paste_next_frame', text='Next frame', icon='FRAME_NEXT').prev = False # Paste Next frame
|
2022-04-11 11:41:56 +02:00
|
|
|
|
|
|
|
row = layout.row(align=True)
|
2022-04-27 12:01:39 +02:00
|
|
|
row.operator('pose.world_space_paste_next_frame', text='', icon='FRAME_PREV').prev = True
|
|
|
|
row.operator('pose.world_space_paste_next', text='', icon='PREV_KEYFRAME').prev = True
|
|
|
|
row.operator('pose.world_space_paste', text='', icon='PASTEDOWN')
|
|
|
|
row.operator('pose.world_space_paste_next', text='', icon='NEXT_KEYFRAME').prev = False
|
|
|
|
row.operator('pose.world_space_paste_next_frame', text='', icon='FRAME_NEXT').prev = False
|
2022-04-11 11:41:56 +02:00
|
|
|
row.scale_x = 2
|
2021-04-08 19:25:05 +02:00
|
|
|
|
2022-04-20 12:02:19 +02:00
|
|
|
class AW_PT_nla_tools_panel(bpy.types.Panel):
|
2022-04-13 18:38:03 +02:00
|
|
|
bl_space_type = "NLA_EDITOR"
|
|
|
|
bl_region_type = "UI"
|
|
|
|
bl_category = "Strip"
|
|
|
|
bl_label = "Walk Tools"
|
|
|
|
|
|
|
|
def draw(self, context):
|
|
|
|
layout = self.layout
|
|
|
|
# layout.label(text='Retime Tools')
|
2023-03-08 17:59:49 +01:00
|
|
|
settings = context.scene.anim_cycle_settings
|
2022-04-21 17:44:35 +02:00
|
|
|
row = layout.row(align=True)
|
|
|
|
row.operator('autowalk.nla_key_speed', text='Set/Update Time Keys', icon='TIME')
|
|
|
|
row.operator('autowalk.nla_remove_key_speed', text='', icon='X')
|
2022-04-13 18:38:03 +02:00
|
|
|
|
2021-04-05 01:35:12 +02:00
|
|
|
|
|
|
|
classes=(
|
2022-04-20 12:02:19 +02:00
|
|
|
AW_PT_walk_cycle_anim_panel,
|
|
|
|
AW_PT_anim_tools_panel,
|
|
|
|
AW_PT_nla_tools_panel,
|
2023-03-08 17:59:49 +01:00
|
|
|
AW_MT_wrap_animation_help,
|
|
|
|
AW_PT_wrap_animation_panel,
|
2022-04-13 18:38:03 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
classes_override_category =(
|
2022-04-20 12:02:19 +02:00
|
|
|
AW_PT_walk_cycle_anim_panel,
|
|
|
|
AW_PT_anim_tools_panel,
|
2021-04-05 01:35:12 +02:00
|
|
|
)
|
|
|
|
|
2022-04-12 11:21:00 +02:00
|
|
|
## Addons Preferences Update Panel
|
|
|
|
def update_panel(self, context):
|
2022-04-13 18:38:03 +02:00
|
|
|
for cls in classes_override_category: # classes
|
2022-04-12 11:21:00 +02:00
|
|
|
try:
|
|
|
|
bpy.utils.unregister_class(cls)
|
|
|
|
except:
|
|
|
|
pass
|
2022-04-13 18:38:03 +02:00
|
|
|
cls.bl_category = self.category # fn.get_addon_prefs().category
|
2022-04-12 11:21:00 +02:00
|
|
|
bpy.utils.register_class(cls)
|
|
|
|
|
2021-04-05 01:35:12 +02:00
|
|
|
def register():
|
|
|
|
for cls in classes:
|
|
|
|
bpy.utils.register_class(cls)
|
|
|
|
|
|
|
|
def unregister():
|
|
|
|
for cls in reversed(classes):
|
|
|
|
bpy.utils.unregister_class(cls)
|