-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathui_scene.py
546 lines (454 loc) · 23.5 KB
/
ui_scene.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
import bpy
from bpy.types import (
Context,
UILayout,
Menu,
Panel,
Object,
Mesh,
Armature,
)
from bpy.props import BoolProperty
from typing import cast
from collections import defaultdict
from .registration import register_module_classes_factory, OperatorBase
from .extensions import ScenePropertyGroup, ObjectPropertyGroup, MmdShapeKeySettings, SceneBuildSettings, WindowManagerPropertyGroup
from .op_build_avatar import BuildAvatarOp
from .ui_object import ObjectBuildSettingsAdd
from . import ui_common
from .context_collection_ops import (
PropCollectionType,
ContextCollectionOperatorBase,
CollectionAddBase,
CollectionRemoveBase,
CollectionDuplicateBase,
)
from . import utils
class SceneBuildSettingsMenu(Menu):
bl_idname = "scene_build_menu"
bl_label = "Build Settings Specials"
def draw(self, context):
pass
class ScenePanel(Panel):
bl_idname = "scene_panel"
bl_label = "Avatar Builder"
bl_space_type = "VIEW_3D"
bl_region_type = "UI"
bl_category = "Avatar Builder"
# Before MMD Shape Mapping Panel by default
bl_order = 0
@staticmethod
def draw_mmd(layout: UILayout, mmd_settings: MmdShapeKeySettings):
layout.prop(mmd_settings, 'do_remap')
if mmd_settings.do_remap:
col = layout.column()
col.use_property_split = True
col.prop(mmd_settings, 'limit_to_body')
col.prop(mmd_settings, 'remap_to')
col.prop(mmd_settings, 'mode')
col.prop(mmd_settings, 'avoid_double_activation')
def draw(self, context: Context):
layout = self.layout
layout.use_property_decorate = False
group = ScenePropertyGroup.get_group(context.scene)
col = layout.column()
if group.is_export_scene:
col.label(text=context.scene.name)
col.operator(DeleteExportScene.bl_idname, icon='TRASH')
else:
col.label(text="Scene Settings Groups")
group.draw_search(col,
new=SceneBuildSettingsAddMenu.bl_idname,
new_is_menu=True,
unlink=SceneBuildSettingsRemove.bl_idname,
name_prop='name_prop',
icon='SETTINGS')
# [ Add | Remove ] [Select | Deselect]
# [Enable | Disable] [ Purge Orphaned ]
buttons_col = col.column(align=True)
columns = buttons_col.column_flow(columns=2, align=False)
row = columns.row(align=True)
row.operator(AddSelectedToSceneSettings.bl_idname)
row.operator(RemoveSelectedFromSceneSettings.bl_idname)
row = columns.row(align=True)
row.operator(SelectObjectsInSceneSettings.bl_idname)
row.operator(DeselectObjectsInSceneSettings.bl_idname)
columns = buttons_col.column_flow(columns=2, align=False)
row = columns.row(align=True)
row.operator(EnableSelectedFromSceneSettings.bl_idname)
row.operator(DisableSelectedFromSceneSettings.bl_idname)
row = columns.row(align=True)
row.operator(SceneBuildSettingsPurge.bl_idname)
col = layout.column()
scene_settings = group.active
if scene_settings:
scene_toggles = WindowManagerPropertyGroup.get_group(context.window_manager).ui_toggles.scene
draw_general, _, _ = ui_common.draw_expandable_header(col, scene_toggles, 'general')
if draw_general:
box = col.box()
box_col = box.column()
box_col.prop(scene_settings, 'limit_to_collection')
box_col.prop(scene_settings, 'ignore_hidden_objects')
box_col.prop(scene_settings, 'reduce_to_two_meshes')
if scene_settings.reduce_to_two_meshes:
sub = box_col.column()
sub.use_property_split = True
sub.alert = not scene_settings.shape_keys_mesh_name
sub.prop(scene_settings, 'shape_keys_mesh_name', icon="MESH_DATA", text="Shape keys")
sub.alert = not scene_settings.no_shape_keys_mesh_name
sub.prop(scene_settings, 'no_shape_keys_mesh_name', icon="MESH_DATA", text="No shape keys")
sub.alert = False
self.draw_mmd(box_col, scene_settings.mmd_settings)
box_col.prop(scene_settings, 'do_limit_total')
if scene_settings.do_limit_total:
sub = box_col.column()
sub.use_property_split = True
sub.prop(scene_settings, 'limit_num_groups')
draw_fixes, _, _ = ui_common.draw_expandable_header(col, scene_toggles, 'fixes')
if draw_fixes:
box = col.box()
box_col = box.column()
fix_settings = scene_settings.fix_settings
box_col.prop(fix_settings, 'sync_mesh_vertices_to_reference_key')
box_col.prop(fix_settings, 'remove_nan_uvs')
# And finally the button for actually running Build Avatar
col.operator(BuildAvatarOp.bl_idname, icon='SHADERFX')
class SceneBuildSettingsBase(ContextCollectionOperatorBase):
@classmethod
def get_collection(cls, context: Context) -> PropCollectionType:
return ScenePropertyGroup.get_group(context.scene).collection
@classmethod
def get_active_index(cls, context: Context) -> int:
return ScenePropertyGroup.get_group(context.scene).active_index
@classmethod
def set_active_index(cls, context: Context, value: int):
ScenePropertyGroup.get_group(context.scene).active_index = value
_op_builder = SceneBuildSettingsBase.op_builder(
class_name_prefix='SceneBuildSettings',
bl_idname_prefix='scene_build_settings',
element_label="scene build settings",
)
@_op_builder.add.decorate
class SceneBuildSettingsAdd(CollectionAddBase[SceneBuildSettings], SceneBuildSettingsBase):
# Position is irrelevant for SceneBuildSettings
_use_positional_description = False
def set_new_item_name(self, data, added: SceneBuildSettings):
if self.name:
added.name_prop = self.name
else:
# Rename if not unique and ensure that the internal name is also set
orig_name = added.name_prop
added_name = utils.get_unique_name(orig_name, data, number_separator=' ', min_number_digits=1)
if added_name != orig_name:
# Assigning the prop will also update the internal name
added.name_prop = added_name
else:
added.name = added_name
def execute(self, context: Context) -> set[str]:
no_elements_to_start_with = not self.get_collection(context)
result = super().execute(context)
# If there weren't any settings to start with, and we just added new settings, we want to cause a UI redraw for
# currently displayed Properties areas that are showing Object Properties of an Object with a type that can be
# built. This is so that the .poll of the Object Panel gets called again, making the Panel appear due to the
# fact that there are now some settings that exist.
if no_elements_to_start_with and 'FINISHED' in result:
ui_common.redraw_object_properties_panels(context)
return result
class _DuplicateBase(CollectionDuplicateBase[SceneBuildSettings], SceneBuildSettingsBase):
# Position is irrelevant for SceneBuildSettings
_use_positional_description = False
def set_new_item_name(self, data: PropCollectionType, added: SceneBuildSettings):
desired_name = self.name
if desired_name:
# Since we're duplicating some existing settings, it's probably best that we don't force the name of the
# duplicated settings to the desired name and instead pick a unique name using the desired name as the base
duplicate_name = utils.get_unique_name(desired_name, data)
else:
source = data[self.index_being_duplicated]
source_name = source.name
# Get the first unique name using source_name as the base name
duplicate_name = utils.get_unique_name(source_name, data)
# Set the name for the duplicated element (we've guaranteed that it's unique, so no need to propagate to other
# settings to ensure uniqueness)
added.set_name_no_propagate(duplicate_name)
@_op_builder.duplicate.decorate
class SceneBuildSettingsDuplicate(_DuplicateBase):
"""Duplicate the active settings (will not duplicate the settings on Objects)"""
class SceneBuildSettingsDuplicateDeep(_DuplicateBase):
"""Duplicate the active settings and also duplicate matching settings on all Objects in the Scene
(orphaned settings matching the name of the duplicate settings will be overwritten)"""
bl_label = "Deep Copy"
bl_idname = 'scene_build_settings_deep_copy'
def modify_newly_created(self, context: Context, data: PropCollectionType, added: SceneBuildSettings):
# super call to perform the duplication and automatic naming
super().modify_newly_created(context, data, added)
# Get the SceneBuildSettings that were duplicated and their name
original = data[self.index_being_duplicated]
original_name = original.name
# Get the name of the new, duplicate SceneBuildSettings
added_name = added.name
for obj in context.scene.objects:
object_settings_col = ObjectPropertyGroup.get_group(obj).collection
# The ObjectBuildSettings matching the SceneBuildSettings being duplicated
orig_object_settings = object_settings_col.get(original_name)
if orig_object_settings:
# Normally there won't be any existing ObjectBuildSettings with the name of the duplicate, since those
# settings would have to have been orphaned prior to the creation of the duplicate. If there is existing
# settings, we'll replace them.
existing_orphaned_settings = object_settings_col.get(added_name)
if existing_orphaned_settings:
duplicate_object_settings = existing_orphaned_settings
else:
# Create new, empty/default settings
duplicate_object_settings = object_settings_col.add()
# Copy all properties across to the duplicate
utils.id_prop_group_copy(orig_object_settings, duplicate_object_settings)
# Since copying also replaces the name properties (and we haven't set the name properties if the
# duplicate is newly created), we need to set the name properties of the duplicate.
# The name is basically guaranteed to be unique since either it wasn't in the collection (guaranteed
# unique) or the duplicate already had the name (should be unique if the addon is working correctly to
# ensure the names are always unique)
duplicate_object_settings.set_name_no_propagate(added_name)
class SceneBuildSettingsAddMenu(Menu):
bl_idname = 'scene_build_settings_add'
bl_label = 'Add'
def draw(self, context: Context):
layout = self.layout
layout.operator(SceneBuildSettingsAdd.bl_idname)
layout.operator(SceneBuildSettingsDuplicate.bl_idname)
layout.operator(SceneBuildSettingsDuplicateDeep.bl_idname)
@_op_builder.remove.decorate
class SceneBuildSettingsRemove(CollectionRemoveBase, SceneBuildSettingsBase):
bl_options = {'UNDO', 'REGISTER'}
also_remove_from_objects: BoolProperty(
name="Remove from Objects",
description="Also remove the settings from all Objects in the Scene",
default=True,
)
def execute(self, context: Context) -> set[str]:
# Optionally remove the settings from all Objects in the Scene
if self.also_remove_from_objects:
collection = self.get_collection(context)
active_index = self.get_active_index(context)
active_name = collection[active_index].name
for obj in context.scene.objects:
object_group_col = ObjectPropertyGroup.get_group(obj).collection
idx = object_group_col.find(active_name)
if idx != -1:
object_group_col.remove(idx)
# Will remove the SceneBuildSettings
super().execute(context)
if not self.get_collection(context):
# If we've just removed the last settings, tell any Object Properties regions to redraw so that they update
# for the fact that there are no longer any settings, meaning the Panel in Object Properties shouldn't be
# drawn anymore
ui_common.redraw_object_properties_panels(context)
return {'FINISHED'}
class SceneBuildSettingsPurge(OperatorBase):
"""Remove orphaned Build Settings from all Objects in every Scene."""
bl_idname = "scene_build_settings_purge"
bl_label = "Purge Orphaned"
bl_options = {'REGISTER', 'UNDO'}
def execute(self, context: Context) -> set[str]:
total_num_settings_removed = 0
num_objects_removed_from = 0
# Objects could be in multiple Scenes, so we need to find the possible SceneBuildSettings for each Object in
# each Scene
non_orphan_settings_per_object: dict[Object, set[str]] = defaultdict(set)
for scene in bpy.data.scenes:
scene_property_group = ScenePropertyGroup.get_group(scene)
# Get the names of all SceneBuildSettings in this Scene
settings_in_scene = {spg.name for spg in scene_property_group.collection}
# Only need to look through the Objects in the Scene if there is at least one SceneBuildSettings
if settings_in_scene:
# Iterate through every Object in this Scene
for obj in scene.objects:
# Only need to check Objects of the allowed types
if obj.type in ObjectPropertyGroup.ALLOWED_TYPES:
# Add the set of names of settings in this Scene to set of all names for this Object
non_orphan_settings_per_object[obj].update(settings_in_scene)
# Iterate through all found Objects, removing any ObjectBuildSettings that are not in the set of names for the
# Object being iterated
for obj, non_orphan_groups in non_orphan_settings_per_object.items():
object_group = ObjectPropertyGroup.get_group(obj)
# Get the collection of ObjectBuildSettings
settings_col = object_group.collection
# Iterate in reverse so that we can remove settings without affecting the indices of settings we are yet to
# iterate.
num_settings_removed = 0
for idx, settings in utils.enumerate_reversed(settings_col):
# If the name of the ObjectBuildSettings doesn't match any of the SceneBuildSettings for this Object,
# remove the ObjectBuildSettings
settings_name = settings.name
if settings_name not in non_orphan_groups:
settings_col.remove(idx)
num_settings_removed += 1
num_remaining_settings = len(settings_col)
if object_group.active_index >= num_remaining_settings:
object_group.active_index = max(0, num_remaining_settings - 1)
if num_settings_removed != 0:
total_num_settings_removed += num_settings_removed
num_objects_removed_from += 1
self.report({'INFO'}, f"Removed {total_num_settings_removed} settings from {num_objects_removed_from} Objects.")
if total_num_settings_removed != 0:
# Cause a UI redraw
ui_common.redraw_object_properties_panels(context)
return {'FINISHED'}
class DeleteExportScene(OperatorBase):
"""Delete the current Export Scene, all Objects in it and their data"""
bl_idname = "delete_export_scene"
bl_label = "Delete Export Scene"
bl_options = {'REGISTER', 'UNDO'}
@classmethod
def poll(cls, context) -> bool:
if not bpy.ops.scene.delete.poll():
return cls.poll_fail("Current Scene cannot be deleted (is it the only Scene?)")
if not ScenePropertyGroup.get_group(context.scene).is_export_scene:
return cls.poll_fail("Current Scene is not an export Scene")
return True
def execute(self, context: Context) -> set[str]:
export_scene = context.scene
obj: Object
for obj in export_scene.objects:
# Deleting data also deletes any objects using that data when do_unlink=True (default value)
data = obj.data
if obj.type == 'MESH':
data = cast(Mesh, data)
shape_keys = data.shape_keys
if shape_keys:
obj.shape_key_clear()
bpy.data.meshes.remove(data)
elif obj.type == 'ARMATURE':
bpy.data.armatures.remove(cast(Armature, data))
else:
bpy.data.objects.remove(obj)
group = ScenePropertyGroup.get_group(export_scene)
original_scene_name = group.export_scene_source_scene
# Switching the scene to the original scene before deleting seems to crash blender sometimes????
# Another workaround seems to be to delete the objects after the scene has been deleted instead of before
# If this is somehow the only scene, deleting isn't possible
if bpy.ops.scene.delete.poll():
bpy.ops.scene.delete()
if original_scene_name:
original_scene = bpy.data.scenes.get(original_scene_name)
if original_scene:
context.window.scene = original_scene
return {'FINISHED'}
class _ActiveSceneSettingsOp(OperatorBase):
@classmethod
def poll(cls, context: Context) -> bool:
scene = context.scene
if scene is None:
return cls.poll_fail("No Scene")
if ScenePropertyGroup.get_group(context.scene).active is None:
return cls.poll_fail("No active Scene Settings")
return True
class _ObjectSelectionInSceneBase(_ActiveSceneSettingsOp):
@classmethod
def poll(cls, context: Context) -> bool:
if not super().poll(context):
return False
if context.mode != 'OBJECT':
return cls.poll_fail("Must be in Object mode")
return True
class SelectObjectsInSceneSettings(_ObjectSelectionInSceneBase):
"""Select objects in the active scene settings"""
bl_idname = "select_objects_in_scene_settings"
bl_label = "Select"
bl_options = {'REGISTER', 'UNDO'}
include_disabled: bpy.props.BoolProperty(
name="Include Disabled",
description="Include objects where the active scene settings are currently disabled",
default=False,
)
def execute(self, context: Context) -> set[str]:
active = ScenePropertyGroup.get_group(context.scene).active
if active:
active_group_name = active.name
vl = context.view_layer
for obj in context.visible_objects:
if not obj.select_get(view_layer=vl):
object_settings = ObjectPropertyGroup.get_group(obj).collection
if active_group_name in object_settings:
if self.include_disabled or object_settings[active_group_name].include_in_build:
obj.select_set(state=True, view_layer=vl)
return {'FINISHED'}
class DeselectObjectsInSceneSettings(_ObjectSelectionInSceneBase):
"""Deselect objects in the active scene settings"""
bl_idname = "deselect_objects_in_scene_settings"
bl_label = "Deselect"
bl_options = {'REGISTER', 'UNDO'}
include_disabled: bpy.props.BoolProperty(
name="Include Disabled",
description="Include objects where the active scene settings are currently disabled",
default=False,
)
def execute(self, context: Context) -> set[str]:
active = ScenePropertyGroup.get_group(context.scene).active
if active:
active_group_name = active.name
vl = context.view_layer
for obj in context.visible_objects:
if obj.select_get(view_layer=vl):
object_settings = ObjectPropertyGroup.get_group(obj).collection
if active_group_name in object_settings:
if self.include_disabled or object_settings[active_group_name].include_in_build:
obj.select_set(state=False, view_layer=vl)
return {'FINISHED'}
class AddSelectedToSceneSettings(_ActiveSceneSettingsOp):
"""Add the selected objects to the active scene settings if they do not already exist"""
bl_idname = "add_selected_to_scene_settings"
bl_label = "Add"
bl_options = {'REGISTER', 'UNDO'}
def execute(self, context: Context) -> set[str]:
active = ScenePropertyGroup.get_group(context.scene).active
if active:
active_group_name = active.name
for obj in context.selected_objects:
object_group = ObjectPropertyGroup.get_group(obj)
object_settings = object_group.collection
if active_group_name not in object_settings:
added = object_settings.add()
ObjectBuildSettingsAdd.set_new_item_name_static(object_settings, added, active_group_name)
return {'FINISHED'}
class RemoveSelectedFromSceneSettings(_ActiveSceneSettingsOp):
"""Remove the selected objects from the active scene settings"""
bl_idname = "remove_selected_from_scene_settings"
bl_label = "Remove"
bl_options = {'REGISTER', 'UNDO'}
def execute(self, context: Context) -> set[str]:
active = ScenePropertyGroup.get_group(context.scene).active
if active:
active_group_name = active.name
for obj in context.selected_objects:
object_group = ObjectPropertyGroup.get_group(obj)
object_settings_col = object_group.collection
idx = object_settings_col.find(active_group_name)
if idx != -1:
object_settings_col.remove(idx)
return {'FINISHED'}
class _IncludeSelectedFromSceneSettingsBase(_ActiveSceneSettingsOp):
_include: bool
def execute(self, context: Context) -> set[str]:
active = ScenePropertyGroup.get_group(context.scene).active
if active:
active_group_name = active.name
for obj in context.selected_objects:
object_settings = ObjectPropertyGroup.get_group(obj).collection.get(active_group_name)
if object_settings is not None:
object_settings.include_in_build = self._include
return {'FINISHED'}
class EnableSelectedFromSceneSettings(_IncludeSelectedFromSceneSettingsBase):
"""Enable the active scene settings on the selected objects if the settings exist"""
bl_idname = "enable_selected_from_scene_settings"
bl_label = "Enable"
bl_options = {'REGISTER', 'UNDO'}
_include = True
class DisableSelectedFromSceneSettings(_IncludeSelectedFromSceneSettingsBase):
"""Disable the active scene settings on the selected objects if the settings exist"""
bl_idname = "disable_selected_from_scene_settings"
bl_label = "Disable"
bl_options = {'REGISTER', 'UNDO'}
_include = False
del _op_builder
register_module_classes_factory(__name__, globals())