gp_render/OP_check_scene.py

142 lines
4.5 KiB
Python

import bpy, re
from . import fn
def check_broken_modifier_target(pool=None, reports=None):
if not reports:
reports = []
if not pool:
pool = [o for o in bpy.context.scene.objects if o.type == 'GPENCIL']
for o in pool:
lay_name_list = [l.info for l in o.data.layers]
for m in o.grease_pencil_modifiers:
if not hasattr(m, 'layer'):
continue
if not m.layer in lay_name_list:
reports.append(f'Broken modifier target :{o.name} > {m.name} > {m.layer}')
# else:
# print(f'Modifier target :{o.name} > {m.name} > ok')
return reports
def check_layer_state(pool=None, reports=None):
if not reports:
reports = []
if not pool:
pool = [o for o in bpy.context.scene.objects if o.type == 'GPENCIL']
for ob in pool:
layers = ob.data.layers
for l in layers:
# if l.mask_layers:
# if not any(not x.hide for x in l.mask_layers):
# # all masks disable
# pass
## just list masks
# state = '' if l.use_mask_layer else ' (disabled)'
# reports.append(f'{ob.name} > {l.info} masks{state}:')
# for ml in l.mask_layers:
# mlstate = ' (disabled)' if ml.hide else ''
# mlinvert = ' <>' if ml.invert else ''
# reports.append(f' - {ml.name}{mlstate}{mlinvert}')
if l.opacity != 1:
reports.append(f'{ob.name} > {l.info} > opacity {l.opacity}')
# if l.use_lights:
# reports.append(f'-> use lights !')
if l.blend_mode != 'REGULAR':
reports.append(f'{ob.name} > {l.info} > blend mode "{l.blend_mode}" !')
return reports
def check_file_output_numbering(reports=None):
if not reports:
reports = []
prenum = re.compile(r'\d{3}_')
file_outs = []
for S in bpy.data.scenes:
if S.name == 'Scene' or not S.node_tree or not S.use_nodes:
continue
file_outs += [n for n in S.node_tree.nodes if n.type == 'OUTPUT_FILE']
used=False
if not file_outs:
reports.append('No file output nodes found')
return reports
for fo in file_outs:
if not prenum.match(fo.base_path.split('/')[-1]):
reports.append(f'No object numbering : node {fo.name}')
pct = 0
for fs in fo.file_slots:
if not prenum.match(fs.path.split('/')[0]):
pct += 1
if pct:
reports.append(f'{pct}/{len(fo.file_slots)} slots not numbered: node {fo.name}')
return reports
class GPEXP_OT_check_render_scene(bpy.types.Operator):
bl_idname = "gp.check_render_scene"
bl_label = "Check render scene"
bl_description = "Auto check render scene"
bl_options = {"REGISTER"} # , "UNDO"
# clear_unused_view_layers : bpy.props.BoolProperty(name="Clear unused view layers",
# description="Delete view layer that aren't used in the nodetree anymore",
# default=True)
@classmethod
def poll(cls, context):
return True
def invoke(self, context, event):
return self.execute(context)
return context.window_manager.invoke_props_dialog(self)
def draw(self, context):
layout = self.layout
# layout.prop(self, 'clear_unused_view_layers')
def execute(self, context):
reports = []
# check gp modifiers
broken_mods = check_broken_modifier_target()
if broken_mods:
reports.append('GP modifiers targets:')
reports += broken_mods
# check layers
layer_state = check_layer_state()
if layer_state:
if reports: reports.append('')
reports.append('Layers State:')
reports += layer_state
# check file output numbering
numbering_problems = check_file_output_numbering()
if numbering_problems:
if reports: reports.append('')
reports.append('File output numbering:')
reports += numbering_problems
if not reports:
self.report({'INFO'}, 'All OK !')
else:
fn.show_message_box(_message=reports, _title='Potential Problems list')
return {"FINISHED"}
classes=(
GPEXP_OT_check_render_scene,
)
def register():
for cls in classes:
bpy.utils.register_class(cls)
def unregister():
for cls in reversed(classes):
bpy.utils.unregister_class(cls)