interactive mode
parent
02a62406d9
commit
660c3c4d76
|
@ -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, 8, 4),
|
"version": (0, 8, 5),
|
||||||
"blender": (4, 0, 2),
|
"blender": (4, 0, 2),
|
||||||
"location": "Sidebar > Gpencil Tab > Interpolate",
|
"location": "Sidebar > Gpencil Tab > Interpolate",
|
||||||
"description": "Interpolate Grease pencil strokes over 3D",
|
"description": "Interpolate Grease pencil strokes over 3D",
|
||||||
|
|
|
@ -21,6 +21,8 @@ class GP_OT_interpolate_stroke_base(bpy.types.Operator):
|
||||||
bl_description = 'Interpolate Stroke based on user bound triangle'
|
bl_description = 'Interpolate Stroke based on user bound triangle'
|
||||||
bl_options = {'REGISTER', 'UNDO'}
|
bl_options = {'REGISTER', 'UNDO'}
|
||||||
|
|
||||||
|
interactive : bpy.props.BoolProperty(name='Interactive', default=False, options={'SKIP_SAVE'})
|
||||||
|
|
||||||
next : bpy.props.BoolProperty(name='Next', default=True, options={'SKIP_SAVE'})
|
next : bpy.props.BoolProperty(name='Next', default=True, options={'SKIP_SAVE'})
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
|
@ -38,22 +40,23 @@ class GP_OT_interpolate_stroke_base(bpy.types.Operator):
|
||||||
else:
|
else:
|
||||||
return f"Interpolate Stroke Backward"
|
return f"Interpolate Stroke Backward"
|
||||||
|
|
||||||
def apply_and_store(self):
|
def apply_and_store(self, attrs):
|
||||||
'''individual item in self.store_list: (prop, attr, [new_val])'''
|
'''individual item in attrs: (prop, attr, [new_val])'''
|
||||||
for item in self.store_list:
|
for item in attrs:
|
||||||
prop, attr = item[:2]
|
prop, attr = item[:2]
|
||||||
self.store.append( (prop, attr, getattr(prop, attr)) )
|
self.stored_attrs.append( (prop, attr, getattr(prop, attr)) )
|
||||||
if len(item) >= 3:
|
if len(item) >= 3:
|
||||||
setattr(prop, attr, item[2])
|
setattr(prop, attr, item[2])
|
||||||
|
|
||||||
def restore(self):
|
def restore(self):
|
||||||
for prop, attr, old_val in self.store:
|
for prop, attr, old_val in self.stored_attrs:
|
||||||
setattr(prop, attr, old_val)
|
setattr(prop, attr, old_val)
|
||||||
|
|
||||||
def exit(self, context, status='INFO', text=None, cancelled=False):
|
def exit(self, context, status='INFO', text=None, cancelled=False):
|
||||||
context.area.header_text_set(None)
|
context.area.header_text_set(None)
|
||||||
wm = context.window_manager
|
wm = context.window_manager
|
||||||
wm.progress_end() # Pgs
|
if self.report_progress:
|
||||||
|
wm.progress_end() # Pgs
|
||||||
if self.timer:
|
if self.timer:
|
||||||
wm.event_timer_remove(self.timer)
|
wm.event_timer_remove(self.timer)
|
||||||
self.restore()
|
self.restore()
|
||||||
|
@ -65,8 +68,8 @@ class GP_OT_interpolate_stroke_base(bpy.types.Operator):
|
||||||
## Remove Plane and it's collection after use
|
## Remove Plane and it's collection after use
|
||||||
if self.plane is not None:
|
if self.plane is not None:
|
||||||
bpy.data.objects.remove(self.plane)
|
bpy.data.objects.remove(self.plane)
|
||||||
if self.toolcol is not None:
|
if self.tool_col is not None:
|
||||||
bpy.data.collections.remove(self.toolcol)
|
bpy.data.collections.remove(self.tool_col)
|
||||||
|
|
||||||
cancel_state = '(Stopped!) ' if cancelled else ''
|
cancel_state = '(Stopped!) ' if cancelled else ''
|
||||||
mess = f'{cancel_state}{self.loop_count} interpolated frame(s) ({time()-self.start:.3f}s)'
|
mess = f'{cancel_state}{self.loop_count} interpolated frame(s) ({time()-self.start:.3f}s)'
|
||||||
|
@ -84,33 +87,36 @@ class GP_OT_interpolate_stroke_base(bpy.types.Operator):
|
||||||
def get_stroke_to_interpolate(self, context):
|
def get_stroke_to_interpolate(self, context):
|
||||||
## Get strokes to interpolate
|
## Get strokes to interpolate
|
||||||
tgt_strokes = [s for s in self.gp.data.layers.active.active_frame.strokes if s.select]
|
tgt_strokes = [s for s in self.gp.data.layers.active.active_frame.strokes if s.select]
|
||||||
|
|
||||||
## If nothing selected in sculpt/paint, Select all before triggering
|
## If nothing selected in sculpt/paint, Select all before triggering
|
||||||
if not tgt_strokes and context.mode in ('SCULPT_GPENCIL', 'PAINT_GPENCIL'):
|
if not tgt_strokes and context.mode in ('SCULPT_GPENCIL', 'PAINT_GPENCIL'):
|
||||||
for s in self.gp.data.layers.active.active_frame.strokes:
|
for s in self.gp.data.layers.active.active_frame.strokes:
|
||||||
s.select = True
|
s.select = True
|
||||||
tgt_strokes = self.gp.data.layers.active.active_frame.strokes
|
tgt_strokes = self.gp.data.layers.active.active_frame.strokes
|
||||||
|
|
||||||
if not tgt_strokes:
|
if tgt_strokes:
|
||||||
return self.exit(context, status='ERROR', text='No stroke selected!')
|
return tgt_strokes
|
||||||
|
|
||||||
|
return self.exit(context, status='ERROR', text='No stroke selected!')
|
||||||
|
|
||||||
return tgt_strokes
|
|
||||||
|
|
||||||
## For now, operators have their own invoke
|
## For now, operators have their own invoke
|
||||||
def invoke(self, context, event):
|
def invoke(self, context, event):
|
||||||
self.debug = False
|
self.debug = False
|
||||||
self.store_list = []
|
self.stored_attrs = [] # context manager store/restore
|
||||||
self.store = []
|
self.loop_count = 0 # frames list iterator
|
||||||
self.loop_count = 0
|
|
||||||
self.start = time()
|
self.start = time()
|
||||||
self.scan_time = None
|
self.scan_time = None # to print time at exit in debug mode
|
||||||
self.plane = None
|
self.plane = None # 3D Plane for bone interpolation
|
||||||
self.toolcol = None
|
self.tool_col = None # collection containing 3D plane
|
||||||
self.gp = context.object
|
self.gp = context.object
|
||||||
self.settings = context.scene.gp_interpo_settings
|
self.settings = context.scene.gp_interpo_settings
|
||||||
self.frames_to_jump = None
|
self.frames_to_jump = None
|
||||||
self.cancelled = False
|
self.cancelled = False
|
||||||
self.timer = None
|
self.timer = None
|
||||||
|
self.timer_event = 'TIMER'
|
||||||
|
self.report_progress = (self.settings.use_animation and not self.interactive)
|
||||||
|
self.interpolated_keys = {context.scene.frame_current}
|
||||||
|
|
||||||
## Remove interpolation_plane collection ! (unseen, but can be hit)
|
## Remove interpolation_plane collection ! (unseen, but can be hit)
|
||||||
if interp_plane := bpy.data.objects.get('interpolation_plane'):
|
if interp_plane := bpy.data.objects.get('interpolation_plane'):
|
||||||
|
@ -118,11 +124,6 @@ class GP_OT_interpolate_stroke_base(bpy.types.Operator):
|
||||||
if interp_col := bpy.data.collections.get('interpolation_tool'):
|
if interp_col := bpy.data.collections.get('interpolation_tool'):
|
||||||
bpy.data.collections.remove(interp_col)
|
bpy.data.collections.remove(interp_col)
|
||||||
|
|
||||||
## Determine on what key/keys to jump
|
|
||||||
self.frames_to_jump = following_keys(forward=self.next, animation=self.settings.use_animation)
|
|
||||||
if not len(self.frames_to_jump):
|
|
||||||
return self.exit(context, status='WARNING', text='No keyframe available in this direction')
|
|
||||||
|
|
||||||
## Change active layer if strokes are selected only on this layer
|
## Change active layer if strokes are selected only on this layer
|
||||||
layers = [l for l in self.gp.data.layers
|
layers = [l for l in self.gp.data.layers
|
||||||
if (not l.lock and l.active_frame)
|
if (not l.lock and l.active_frame)
|
||||||
|
@ -137,8 +138,23 @@ class GP_OT_interpolate_stroke_base(bpy.types.Operator):
|
||||||
## Set active layer
|
## Set active layer
|
||||||
self.gp.data.layers.active = layers[0]
|
self.gp.data.layers.active = layers[0]
|
||||||
|
|
||||||
# TODO: Expose timer (in preferences ?) to let user more time to see result between frames
|
|
||||||
self.timer = context.window_manager.event_timer_add(0.05, window=context.window)
|
# if self.interactive:
|
||||||
|
# ## TODO: Allow even if 0 keys are available
|
||||||
|
# ## disable timer event detection
|
||||||
|
# self.timer_event = None
|
||||||
|
# # Add available keys in other direction then sort
|
||||||
|
# self.frames_to_jump += following_keys(forward=not self.next, animation=self.settings.use_animation or self.interactive)
|
||||||
|
# self.frames_to_jump.sort()
|
||||||
|
|
||||||
|
if not self.interactive:
|
||||||
|
## Determine on what key/keys to jump
|
||||||
|
self.frames_to_jump = following_keys(forward=self.next, animation=self.settings.use_animation or self.interactive)
|
||||||
|
if not len(self.frames_to_jump):
|
||||||
|
return self.exit(context, status='WARNING', text='No keyframe available in this direction')
|
||||||
|
|
||||||
|
# TODO: Expose timer (in preferences ?) to let user more time to see result between frames
|
||||||
|
self.timer = context.window_manager.event_timer_add(0.04, window=context.window)
|
||||||
|
|
||||||
## Converted to modal from "operator_single"
|
## Converted to modal from "operator_single"
|
||||||
|
|
||||||
|
@ -185,7 +201,7 @@ class GP_OT_interpolate_stroke(GP_OT_interpolate_stroke_base):
|
||||||
if state := super().invoke(context, event):
|
if state := super().invoke(context, event):
|
||||||
return state
|
return state
|
||||||
|
|
||||||
scn = bpy.context.scene
|
scn = context.scene
|
||||||
|
|
||||||
origin = scn.camera.matrix_world.to_translation()
|
origin = scn.camera.matrix_world.to_translation()
|
||||||
|
|
||||||
|
@ -209,13 +225,13 @@ class GP_OT_interpolate_stroke(GP_OT_interpolate_stroke_base):
|
||||||
|
|
||||||
## Ensure collection and plane exists
|
## Ensure collection and plane exists
|
||||||
# get/create collection
|
# get/create collection
|
||||||
self.toolcol = bpy.data.collections.get('interpolation_tool')
|
self.tool_col = bpy.data.collections.get('interpolation_tool')
|
||||||
if not self.toolcol:
|
if not self.tool_col:
|
||||||
self.toolcol = bpy.data.collections.new('interpolation_tool')
|
self.tool_col = bpy.data.collections.new('interpolation_tool')
|
||||||
|
|
||||||
if self.toolcol.name not in bpy.context.scene.collection.children:
|
if self.tool_col.name not in bpy.context.scene.collection.children:
|
||||||
bpy.context.scene.collection.children.link(self.toolcol)
|
bpy.context.scene.collection.children.link(self.tool_col)
|
||||||
self.toolcol.hide_viewport = True # needed ?
|
self.tool_col.hide_viewport = True # needed ?
|
||||||
|
|
||||||
# get/create meshplane
|
# get/create meshplane
|
||||||
self.plane = bpy.data.objects.get('interpolation_plane')
|
self.plane = bpy.data.objects.get('interpolation_plane')
|
||||||
|
@ -223,8 +239,8 @@ class GP_OT_interpolate_stroke(GP_OT_interpolate_stroke_base):
|
||||||
self.plane = create_plane(name='interpolation_plane')
|
self.plane = create_plane(name='interpolation_plane')
|
||||||
self.plane.select_set(False)
|
self.plane.select_set(False)
|
||||||
|
|
||||||
if self.plane.name not in self.toolcol.objects:
|
if self.plane.name not in self.tool_col.objects:
|
||||||
self.toolcol.objects.link(self.plane)
|
self.tool_col.objects.link(self.plane)
|
||||||
target_obj = self.plane
|
target_obj = self.plane
|
||||||
|
|
||||||
elif self.settings.method == 'GEOMETRY':
|
elif self.settings.method == 'GEOMETRY':
|
||||||
|
@ -243,14 +259,14 @@ class GP_OT_interpolate_stroke(GP_OT_interpolate_stroke_base):
|
||||||
target_obj = override
|
target_obj = override
|
||||||
|
|
||||||
## Prepare context manager
|
## Prepare context manager
|
||||||
self.store_list = [
|
attrs = [
|
||||||
# (context.view_layer.objects, 'active', self.gp),
|
# (context.view_layer.objects, 'active', self.gp),
|
||||||
(context.tool_settings, 'use_keyframe_insert_auto', True),
|
(context.tool_settings, 'use_keyframe_insert_auto', True),
|
||||||
# (bpy.context.scene.render, 'simplify_subdivision', 0),
|
# (bpy.context.scene.render, 'simplify_subdivision', 0),
|
||||||
]
|
]
|
||||||
|
|
||||||
## Set everything in SETUP list
|
## Set everything in SETUP list
|
||||||
self.apply_and_store()
|
self.apply_and_store(attrs)
|
||||||
|
|
||||||
if self.settings.method == 'BONE':
|
if self.settings.method == 'BONE':
|
||||||
## replace plane
|
## replace plane
|
||||||
|
@ -307,27 +323,52 @@ class GP_OT_interpolate_stroke(GP_OT_interpolate_stroke_base):
|
||||||
|
|
||||||
# Jump frame and paste
|
# Jump frame and paste
|
||||||
|
|
||||||
context.window_manager.progress_begin(self.frames_to_jump[0], self.frames_to_jump[-1]) # Pgs
|
if self.report_progress:
|
||||||
|
context.window_manager.progress_begin(self.frames_to_jump[0], self.frames_to_jump[-1]) # Pgs
|
||||||
context.window_manager.modal_handler_add(self)
|
context.window_manager.modal_handler_add(self)
|
||||||
context.area.header_text_set('Starting interpolation | Esc: Cancel')
|
context.area.header_text_set('Starting interpolation | Esc: Cancel')
|
||||||
|
|
||||||
return {'RUNNING_MODAL'}
|
return {'RUNNING_MODAL'}
|
||||||
|
|
||||||
def modal(self, context, event):
|
def modal(self, context, event):
|
||||||
frame_num = len(self.frames_to_jump)
|
scn = context.scene
|
||||||
percentage = (self.loop_count) / (frame_num) * 100
|
|
||||||
context.area.header_text_set(f'Interpolation {percentage:.0f}% {self.loop_count + 1}/{frame_num} | Esc: Cancel')
|
|
||||||
|
|
||||||
if event.type in {'RIGHTMOUSE', 'ESC'}:
|
if self.interactive:
|
||||||
context.area.header_text_set(f'Cancelling')
|
frame = None
|
||||||
|
self.loop_count = 0 # Reset to keep inifinite loop
|
||||||
|
prev_frame_num = following_keys(forward=False)
|
||||||
|
prev_frame_num = prev_frame_num[0] if prev_frame_num else None
|
||||||
|
next_frame_num = following_keys(forward=True)
|
||||||
|
next_frame_num = next_frame_num[0] if next_frame_num else None
|
||||||
|
context.area.header_text_set(f'Interpolate: {prev_frame_num} < {context.scene.frame_current} > {next_frame_num} | Esc: Finish')
|
||||||
|
|
||||||
|
if event.type == 'LEFT_ARROW' and event.value == 'PRESS':
|
||||||
|
frame = prev_frame_num
|
||||||
|
if event.type == 'RIGHT_ARROW' and event.value == 'PRESS':
|
||||||
|
frame = next_frame_num
|
||||||
|
if frame is None:
|
||||||
|
self.report({'WARNING'}, 'No frame to jump to in this direction!')
|
||||||
|
|
||||||
|
else:
|
||||||
|
frame_num = len(self.frames_to_jump)
|
||||||
|
percentage = (self.loop_count) / (frame_num) * 100
|
||||||
|
context.area.header_text_set(f'Interpolation {percentage:.0f}% {self.loop_count + 1}/{frame_num} | Esc: Cancel')
|
||||||
|
|
||||||
|
if event.type in {'RIGHTMOUSE', 'ESC', 'RET'}:
|
||||||
return self.exit(context, status='WARNING', text='Cancelling', cancelled=True)
|
return self.exit(context, status='WARNING', text='Cancelling', cancelled=True)
|
||||||
|
|
||||||
## -- LOOPTIMER
|
## -- LOOPTIMER
|
||||||
if event.type == 'TIMER':
|
if event.type == self.timer_event or (self.interactive and frame is not None):
|
||||||
f = self.frames_to_jump[self.loop_count]
|
if not self.interactive:
|
||||||
bpy.context.window_manager.progress_update(f) # Pgs
|
frame = self.frames_to_jump[self.loop_count]
|
||||||
scn = bpy.context.scene
|
scn.frame_set(frame)
|
||||||
scn.frame_set(f)
|
if frame in self.interpolated_keys:
|
||||||
|
self.report({'INFO'}, f'SKIP {frame} (already interpolated)')
|
||||||
|
return {'RUNNING_MODAL'}
|
||||||
|
print(f'-> {frame}')
|
||||||
|
if self.report_progress:
|
||||||
|
context.window_manager.progress_update(frame) # Pgs
|
||||||
|
|
||||||
origin = scn.camera.matrix_world.to_translation()
|
origin = scn.camera.matrix_world.to_translation()
|
||||||
plane_co, plane_no = get_gp_draw_plane(self.gp)
|
plane_co, plane_no = get_gp_draw_plane(self.gp)
|
||||||
bpy.ops.gpencil.select_all(action='DESELECT')
|
bpy.ops.gpencil.select_all(action='DESELECT')
|
||||||
|
@ -385,12 +426,15 @@ class GP_OT_interpolate_stroke(GP_OT_interpolate_stroke_base):
|
||||||
bpy.ops.gpencil.select_all(action='SELECT')
|
bpy.ops.gpencil.select_all(action='SELECT')
|
||||||
for stroke in other_strokes:
|
for stroke in other_strokes:
|
||||||
stroke.select = False
|
stroke.select = False
|
||||||
|
|
||||||
self.loop_count += 1
|
|
||||||
if self.loop_count >= len(self.frames_to_jump):
|
|
||||||
return self.exit(context)
|
|
||||||
|
|
||||||
bpy.ops.wm.redraw_timer(type='DRAW_WIN_SWAP', iterations=1)
|
if self.interactive:
|
||||||
|
self.interpolated_keys.add(frame)
|
||||||
|
else:
|
||||||
|
self.loop_count += 1
|
||||||
|
if self.loop_count >= len(self.frames_to_jump):
|
||||||
|
return self.exit(context)
|
||||||
|
|
||||||
|
# bpy.ops.wm.redraw_timer(type='DRAW_WIN_SWAP', iterations=1)
|
||||||
|
|
||||||
return {'RUNNING_MODAL'}
|
return {'RUNNING_MODAL'}
|
||||||
|
|
||||||
|
|
6
ui.py
6
ui.py
|
@ -40,9 +40,9 @@ class GP_PT_interpolate(bpy.types.Panel):
|
||||||
direction_button_row.operator(ops_id, text=prev_text, icon=prev_icon).next = False
|
direction_button_row.operator(ops_id, text=prev_text, icon=prev_icon).next = False
|
||||||
direction_button_row.operator(ops_id, text=next_text, icon=next_icon).next = True
|
direction_button_row.operator(ops_id, text=next_text, icon=next_icon).next = True
|
||||||
|
|
||||||
## Button for interactive mode
|
## Button for interactive mode (icons: SNAP_INCREMENT, ACTION_TWEAK, CON_ACTION, CENTER_ONLY)
|
||||||
# interactive_mode_row = row.row()
|
interactive_mode_row = row.row()
|
||||||
# interactive_mode_row.operator(ops_id, text='', icon='ACTION_TWEAK') # SNAP_INCREMENT, ACTION_TWEAK, CON_ACTION, CENTER_ONLY
|
interactive_mode_row.operator(ops_id, text='', icon='ACTION_TWEAK').interactive = True
|
||||||
|
|
||||||
|
|
||||||
col.prop(settings, 'use_animation', text='Animation')
|
col.prop(settings, 'use_animation', text='Animation')
|
||||||
|
|
Loading…
Reference in New Issue