2024-05-30 18:33:05 +02:00
|
|
|
import bpy
|
|
|
|
from bpy.types import Operator
|
|
|
|
import mathutils
|
|
|
|
from mathutils import Vector, Matrix, geometry
|
|
|
|
from bpy_extras import view3d_utils
|
|
|
|
from . import utils
|
|
|
|
|
|
|
|
# def get_layer_list(self, context):
|
|
|
|
# '''return (identifier, name, description) of enum content'''
|
|
|
|
# if not context:
|
|
|
|
# return [('None', 'None','None')]
|
|
|
|
# if not context.object:
|
|
|
|
# return [('None', 'None','None')]
|
2024-11-11 15:56:43 +01:00
|
|
|
# return [(l.name, l.name, '') for l in context.object.data.layers] # if l != context.object.data.layers.active
|
2024-05-30 18:33:05 +02:00
|
|
|
|
|
|
|
## in Class
|
|
|
|
# bl_property = "layers_enum"
|
|
|
|
|
|
|
|
# layers_enum : bpy.props.EnumProperty(
|
|
|
|
# name="Send Material To Layer",
|
|
|
|
# description="Send active material to layer",
|
|
|
|
# items=get_layer_list,
|
|
|
|
# options={'HIDDEN'},
|
|
|
|
# )
|
|
|
|
|
|
|
|
class GPTB_OT_move_material_to_layer(Operator) :
|
|
|
|
bl_idname = "gp.move_material_to_layer"
|
|
|
|
bl_label = 'Move Material To Layer'
|
|
|
|
bl_description = 'Move active material to an existing or new layer'
|
|
|
|
bl_options = {"REGISTER", "UNDO", "INTERNAL"}
|
|
|
|
|
|
|
|
layer_name : bpy.props.StringProperty(
|
|
|
|
name='Layer Name', default='', options={'SKIP_SAVE'})
|
2024-07-16 17:57:27 +02:00
|
|
|
|
|
|
|
copy : bpy.props.BoolProperty(
|
|
|
|
name='Copy to layer', default=False,
|
|
|
|
description='Copy strokes to layer instead of moving',
|
|
|
|
options={'SKIP_SAVE'})
|
2024-05-30 18:33:05 +02:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def poll(cls, context):
|
2024-11-11 15:35:39 +01:00
|
|
|
return context.object and context.object.type == 'GREASEPENCIL'
|
2024-05-30 18:33:05 +02:00
|
|
|
|
|
|
|
def invoke(self, context, event):
|
|
|
|
if self.layer_name:
|
|
|
|
return self.execute(context)
|
|
|
|
if not len(context.object.data.layers):
|
|
|
|
self.report({'WARNING'}, 'No layers on current GP object')
|
|
|
|
return {'CANCELLED'}
|
|
|
|
|
|
|
|
mat = context.object.data.materials[context.object.active_material_index]
|
|
|
|
self.mat_name = mat.name
|
|
|
|
|
|
|
|
# wm.invoke_search_popup(self)
|
|
|
|
return context.window_manager.invoke_props_dialog(self, width=250)
|
|
|
|
|
|
|
|
def draw(self, context):
|
|
|
|
layout = self.layout
|
|
|
|
# layout.operator_context = "INVOKE_DEFAULT"
|
2024-07-16 17:57:27 +02:00
|
|
|
layout.prop(self, 'copy', text='Copy Strokes')
|
|
|
|
action_label = 'Copy' if self.copy else 'Move'
|
|
|
|
layout.label(text=f'{action_label} material "{self.mat_name}" to layer:', icon='MATERIAL')
|
2024-05-30 18:33:05 +02:00
|
|
|
|
|
|
|
col = layout.column()
|
|
|
|
col.prop(self, 'layer_name', text='', icon='ADD')
|
|
|
|
# if self.layer_name:
|
|
|
|
# col.label(text='Ok/Enter to create new layer', icon='INFO')
|
|
|
|
|
|
|
|
col.separator()
|
|
|
|
for l in reversed(context.object.data.layers):
|
|
|
|
|
|
|
|
icon = 'GREASEPENCIL' if l == context.object.data.layers.active else 'BLANK1'
|
|
|
|
row = col.row()
|
|
|
|
row.alignment = 'LEFT'
|
2024-11-11 15:56:43 +01:00
|
|
|
op = col.operator('gp.move_material_to_layer', text=l.name, icon=icon, emboss=False)
|
|
|
|
op.layer_name = l.name
|
2024-07-16 17:57:27 +02:00
|
|
|
op.copy = self.copy
|
2024-05-30 18:33:05 +02:00
|
|
|
|
|
|
|
def execute(self, context):
|
|
|
|
if not self.layer_name:
|
|
|
|
print('Out')
|
|
|
|
return {'CANCELLED'}
|
|
|
|
|
|
|
|
## Active + selection
|
2024-11-11 15:35:39 +01:00
|
|
|
pool = [o for o in bpy.context.selected_objects if o.type == 'GREASEPENCIL']
|
2024-05-30 18:33:05 +02:00
|
|
|
if not context.object in pool:
|
|
|
|
pool.append(context.object)
|
|
|
|
|
|
|
|
mat = context.object.data.materials[context.object.active_material_index]
|
|
|
|
|
|
|
|
print(f'Moving strokes using material "{mat.name}" on {len(pool)} object(s)')
|
|
|
|
# import time
|
|
|
|
# t = time.time() # Dbg
|
|
|
|
total = 0
|
|
|
|
oct = 0
|
2024-12-02 14:51:43 +01:00
|
|
|
|
2024-05-30 18:33:05 +02:00
|
|
|
for ob in pool:
|
|
|
|
mat_index = next((i for i, ms in enumerate(ob.material_slots) if ms.material and ms.material == mat), None)
|
|
|
|
if mat_index is None:
|
|
|
|
print(f'/!\ {ob.name} has no Material {mat.name} in stack')
|
|
|
|
continue
|
|
|
|
|
|
|
|
gpl = ob.data.layers
|
|
|
|
|
|
|
|
if not (target_layer := gpl.get(self.layer_name)):
|
|
|
|
target_layer = gpl.new(self.layer_name)
|
|
|
|
|
|
|
|
## List existing frames
|
|
|
|
key_dict = {f.frame_number : f for f in target_layer.frames}
|
|
|
|
|
|
|
|
### Move Strokes to a new key (or existing key if comming for yet another layer)
|
|
|
|
fct = 0
|
|
|
|
sct = 0
|
2024-12-02 14:51:43 +01:00
|
|
|
for layer in gpl:
|
|
|
|
if layer == target_layer:
|
2024-05-30 18:33:05 +02:00
|
|
|
## ! infinite loop if target layer is included
|
|
|
|
continue
|
2024-12-02 14:51:43 +01:00
|
|
|
for fr in layer.frames:
|
2024-05-30 18:33:05 +02:00
|
|
|
## skip if no stroke has active material
|
2024-12-02 14:51:43 +01:00
|
|
|
if not next((s for s in fr.drawing.strokes if s.material_index == mat_index), None):
|
2024-05-30 18:33:05 +02:00
|
|
|
continue
|
|
|
|
## Get/Create a destination frame and keep a reference to it
|
2024-12-02 14:51:43 +01:00
|
|
|
if not (dest_key := key_dict.get(fr.frame_number)):
|
|
|
|
dest_key = target_layer.frames.new(fr.frame_number)
|
2024-05-30 18:33:05 +02:00
|
|
|
key_dict[dest_key.frame_number] = dest_key
|
|
|
|
|
2024-12-02 14:51:43 +01:00
|
|
|
print(f'{ob.name} : frame {fr.frame_number}')
|
2024-05-30 18:33:05 +02:00
|
|
|
## Replicate strokes in dest_keys
|
|
|
|
stroke_to_delete = []
|
2024-12-02 14:51:43 +01:00
|
|
|
for s_idx, s in enumerate(fr.drawing.strokes):
|
2024-05-30 18:33:05 +02:00
|
|
|
if s.material_index == mat_index:
|
|
|
|
utils.copy_stroke_to_frame(s, dest_key)
|
2024-12-02 14:51:43 +01:00
|
|
|
stroke_to_delete.append(s_idx)
|
2024-05-30 18:33:05 +02:00
|
|
|
|
|
|
|
## Debug
|
|
|
|
# if time.time() - t > 10:
|
|
|
|
# print('TIMEOUT')
|
|
|
|
# return {'CANCELLED'}
|
|
|
|
|
|
|
|
sct += len(stroke_to_delete)
|
|
|
|
|
2024-12-02 14:51:43 +01:00
|
|
|
## Remove from source frame (fr)
|
2024-07-16 17:57:27 +02:00
|
|
|
if not self.copy:
|
2024-12-02 14:51:43 +01:00
|
|
|
# print('Removing frames') # Dbg
|
|
|
|
if stroke_to_delete:
|
|
|
|
fr.drawing.remove_strokes(indices=stroke_to_delete)
|
2024-05-30 18:33:05 +02:00
|
|
|
|
2024-12-02 14:51:43 +01:00
|
|
|
## ? Remove frame if layer is empty ? -> probably not, otherwise will show previous frame
|
2024-05-30 18:33:05 +02:00
|
|
|
|
|
|
|
fct += 1
|
2024-12-02 14:51:43 +01:00
|
|
|
|
2024-05-30 18:33:05 +02:00
|
|
|
|
|
|
|
if fct:
|
|
|
|
oct += 1
|
|
|
|
print(f'{ob.name}: Moved {fct} frames -> {sct} Strokes') # Dbg
|
|
|
|
|
|
|
|
total += fct
|
|
|
|
|
|
|
|
report_type = 'INFO' if total else 'WARNING'
|
2024-07-16 17:57:27 +02:00
|
|
|
if self.copy:
|
|
|
|
self.report({report_type}, f'Copied {total} frames accross {oct} object(s)')
|
|
|
|
else:
|
|
|
|
self.report({report_type}, f'Moved {total} frames accross {oct} object(s)')
|
2024-05-30 18:33:05 +02:00
|
|
|
return {'FINISHED'}
|
|
|
|
|
|
|
|
|
|
|
|
# def menu_duplicate_and_send_to_layer(self, context):
|
2024-11-14 19:09:52 +01:00
|
|
|
# if context.space_data.ui_mode == 'GPENCIL':
|
2024-05-30 18:33:05 +02:00
|
|
|
# self.layout.operator_context = 'INVOKE_REGION_WIN'
|
|
|
|
# self.layout.operator('gp.duplicate_send_to_layer', text='Move Keys To Layer').delete_source = True
|
|
|
|
# self.layout.operator('gp.duplicate_send_to_layer', text='Copy Keys To Layer')
|
|
|
|
|
|
|
|
classes = (
|
|
|
|
GPTB_OT_move_material_to_layer,
|
|
|
|
)
|
|
|
|
|
|
|
|
def register():
|
|
|
|
for cls in classes:
|
|
|
|
bpy.utils.register_class(cls)
|
|
|
|
|
|
|
|
def unregister():
|
|
|
|
for cls in reversed(classes):
|
|
|
|
bpy.utils.unregister_class(cls)
|