animation mode to create all keys at once

master
pullusb 2023-12-07 12:16:18 +01:00
parent a56a9ea537
commit 77e3049d5d
4 changed files with 77 additions and 43 deletions

View File

@ -1,7 +1,7 @@
bl_info = { bl_info = {
"name": "gp interpolate", "name": "gp interpolate",
"author": "Christophe Seux, Samuel Bernou", "author": "Christophe Seux, Samuel Bernou",
"version": (0, 1, 2), "version": (0, 2, 0),
"blender": (3, 6, 0), "blender": (3, 6, 0),
"location": "Sidebar > Gpencil Tab > Interpolate", "location": "Sidebar > Gpencil Tab > Interpolate",
"description": "Interpolate Grease pencil strokes over 3D", "description": "Interpolate Grease pencil strokes over 3D",

View File

@ -20,13 +20,26 @@ from mathutils.geometry import (barycentric_transform,
tessellate_polygon) tessellate_polygon)
def following_key(forward=True): def following_keys(forward=True, all_keys=True) -> list:# -> list[int] | list | None:
'''return a lsit of int or an empty list'''
direction = 1 if forward else -1 direction = 1 if forward else -1
cur_frame = bpy.context.scene.frame_current cur_frame = bpy.context.scene.frame_current
settings = bpy.context.scene.gp_interpo_settings settings = bpy.context.scene.gp_interpo_settings
if settings.mode == 'FRAME': if settings.mode == 'FRAME':
return cur_frame + (settings.padding * direction) if all_keys:
scn = bpy.context.scene
if forward:
limit = scn.frame_preview_end if scn.use_preview_range else scn.frame_end
else:
limit = scn.frame_preview_start if scn.use_preview_range else scn.frame_start
limit += direction # offset by one for limit to be in range
return list(range(cur_frame, limit, settings.padding * direction))
else:
return [cur_frame + (settings.padding * direction)]
elif settings.mode == 'GPKEY': elif settings.mode == 'GPKEY':
layers = bpy.context.object.data.layers layers = bpy.context.object.data.layers
@ -42,18 +55,26 @@ def following_key(forward=True):
frames = [k.co.x for fc in arm.animation_data.action.fcurves for k in fc.keyframe_points] frames = [k.co.x for fc in arm.animation_data.action.fcurves for k in fc.keyframe_points]
if not frames: if not frames:
return return []
# Sort frames (invert if looking backward)
frames.sort(reversed=not forward)
if all_keys:
frames = list(set(frames))
if forward:
frame_list = [int(f) for f in frames if f > cur_frame]
else:
frame_list = [int(f) for f in frames if f < cur_frame]
return frame_list
frames.sort()
if forward: if forward:
new = next((f for f in frames if f > cur_frame), None) new = next((f for f in frames if f > cur_frame), None)
else: else:
below = [f for f in frames if f < cur_frame] new = next((f for f in frames if f < cur_frame), None)
if not below: if new is None:
return return []
new = below[-1] return [int(new)]
return int(new)
## TODO: add bake animation to empty for later GP layer parenting ## TODO: add bake animation to empty for later GP layer parenting
@ -88,9 +109,9 @@ class GP_OT_interpolate_stroke(bpy.types.Operator):
# auto_key_status = context.tool_settings.use_keyframe_insert_auto # auto_key_status = context.tool_settings.use_keyframe_insert_auto
# context.tool_settings.use_keyframe_insert_auto = True # context.tool_settings.use_keyframe_insert_auto = True
## Determine on what key to jump ## Determine on what key/keys to jump
frame_to_jump = following_key(forward=self.next) frames_to_jump = following_keys(forward=self.next, all_keys=settings.use_animation)
if frame_to_jump is None: if not len(frames_to_jump):
self.report({'WARNING'}, 'No keyframe available in this direction') self.report({'WARNING'}, 'No keyframe available in this direction')
return {'CANCELLED'} return {'CANCELLED'}
@ -214,38 +235,45 @@ class GP_OT_interpolate_stroke(bpy.types.Operator):
# Copy stroke selection, jump frame and paste # Copy stroke selection, jump frame and paste
bpy.ops.gpencil.copy() bpy.ops.gpencil.copy()
scn.frame_set(frame_to_jump)
plan_co, plane_no = get_gp_draw_plane(gp)
bpy.ops.gpencil.paste()
if settings.method == 'BONE': wm = bpy.context.window_manager # Pgs
bone_plane = plane_on_bone(settings.target_rig.pose.bones.get(settings.target_bone),
arm=settings.target_rig,
set_rotation=settings.use_bone_rotation,
mesh=True)
dg = bpy.context.evaluated_depsgraph_get() wm.progress_begin(frames_to_jump[0], frames_to_jump[-1]) # Pgs
matrix_inv = np.array(gp.matrix_world.inverted(), dtype='float64')#.inverted() for f in frames_to_jump:
new_strokes = gp.data.layers.active.active_frame.strokes[-len(strokes_data):] wm.progress_update(f) # Pgs
scn.frame_set(f)
plan_co, plane_no = get_gp_draw_plane(gp)
bpy.ops.gpencil.paste()
for new_stroke, stroke_data in zip(new_strokes, strokes_data): if settings.method == 'BONE':
world_co_3d = [] # np.array(len()dtype='float64')#np. bone_plane = plane_on_bone(settings.target_rig.pose.bones.get(settings.target_bone),
for stroke, point_co, object_hit, hit_location, tri_a, tri_indices in stroke_data: arm=settings.target_rig,
eval_ob = object_hit.evaluated_get(dg) set_rotation=settings.use_bone_rotation,
tri_b = [eval_ob.data.vertices[i].co for i in tri_indices] mesh=True)
tri_b = matrix_transform(tri_b, eval_ob.matrix_world)
new_loc = barycentric_transform(hit_location, *tri_a, *tri_b) dg = bpy.context.evaluated_depsgraph_get()
world_co_3d.append(new_loc) matrix_inv = np.array(gp.matrix_world.inverted(), dtype='float64')#.inverted()
new_strokes = gp.data.layers.active.active_frame.strokes[-len(strokes_data):]
# Reproject on plane for new_stroke, stroke_data in zip(new_strokes, strokes_data):
new_world_co_3d = [intersect_line_plane(origin, p, plan_co, plane_no) for p in world_co_3d] world_co_3d = [] # np.array(len()dtype='float64')#np.
new_local_co_3d = matrix_transform(new_world_co_3d, matrix_inv) for stroke, point_co, object_hit, hit_location, tri_a, tri_indices in stroke_data:
eval_ob = object_hit.evaluated_get(dg)
tri_b = [eval_ob.data.vertices[i].co for i in tri_indices]
tri_b = matrix_transform(tri_b, eval_ob.matrix_world)
nb_points = len(new_stroke.points) new_loc = barycentric_transform(hit_location, *tri_a, *tri_b)
new_stroke.points.foreach_set('co', new_local_co_3d.reshape(nb_points*3)) world_co_3d.append(new_loc)
new_stroke.points.update()
# Reproject on plane
new_world_co_3d = [intersect_line_plane(origin, p, plan_co, plane_no) for p in world_co_3d]
new_local_co_3d = matrix_transform(new_world_co_3d, matrix_inv)
nb_points = len(new_stroke.points)
new_stroke.points.foreach_set('co', new_local_co_3d.reshape(nb_points*3))
new_stroke.points.update()
wm.progress_end() # Pgs
## Reset autokey status ## Reset autokey status
# context.tool_settings.use_keyframe_insert_auto = auto_key_status # (Done in context manager) # context.tool_settings.use_keyframe_insert_auto = auto_key_status # (Done in context manager)
@ -284,8 +312,8 @@ class GP_OT_interpolate_stroke(bpy.types.Operator):
context.tool_settings.use_keyframe_insert_auto = True context.tool_settings.use_keyframe_insert_auto = True
## Determine on what key to jump ## Determine on what key to jump
frame_to_jump = following_key(forward=self.next) frames_to_jump = following_keys(forward=self.next)
if frame_to_jump is None: if frames_to_jump is None:
self.report({'WARNING'}, 'No keyframe available in this direction') self.report({'WARNING'}, 'No keyframe available in this direction')
return {'CANCELLED'} return {'CANCELLED'}
@ -376,7 +404,7 @@ class GP_OT_interpolate_stroke(bpy.types.Operator):
bpy.ops.gpencil.copy() bpy.ops.gpencil.copy()
scn.frame_set(frame_to_jump) scn.frame_set(frames_to_jump)
plan_co, plane_no = get_gp_draw_plane(gp) plan_co, plane_no = get_gp_draw_plane(gp)

View File

@ -24,6 +24,11 @@ class GP_PG_interpolate_settings(PropertyGroup):
description='Select method for interpolating strokes' description='Select method for interpolating strokes'
) )
use_animation : BoolProperty(
name='Animatation',
default=True,
description='Apply the interpolation on the remaining range')
search_range : FloatProperty( search_range : FloatProperty(
name="Search Range", name="Search Range",
description="Search range size when points are out of mesh", description="Search range size when points are out of mesh",

1
ui.py
View File

@ -30,6 +30,7 @@ class GP_PT_interpolate(bpy.types.Panel):
col.prop(settings, 'method', text='Method') col.prop(settings, 'method', text='Method')
col.prop(settings, 'use_animation', text='Animation')
if settings.method == 'BONE': if settings.method == 'BONE':
col = layout.column(align=True) col = layout.column(align=True)