-
Notifications
You must be signed in to change notification settings - Fork 8
/
lods.py
433 lines (324 loc) · 13.9 KB
/
lods.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
"""LOD Management system."""
import bpy
from bpy.types import (
Context,
Mesh,
PropertyGroup,
Object,
)
from bpy.props import (
EnumProperty,
PointerProperty,
StringProperty,
BoolProperty,
)
from typing import Callable, Optional
from .sollumz_properties import (
SollumType,
LODLevel,
LODLevelEnumItems,
FRAGMENT_TYPES,
DRAWABLE_TYPES,
SOLLUMZ_UI_NAMES,
BOUND_TYPES,
BOUND_POLYGON_TYPES,
)
from .tools.blenderhelper import lod_level_enum_flag_prop_factory
from .sollumz_helper import find_sollumz_parent
from .icons import icon_manager
class LODLevelProps(PropertyGroup):
def on_lod_level_enter(self):
"""Called when the LOD level switches to this level."""
obj: Object = self.id_data
if self.has_mesh:
# Update the object current mesh to this LOD mesh
obj.data = self.mesh_ref
self.mesh_ref = None # keep a single ref to the mesh, in Object.data
if obj.name in bpy.context.view_layer.objects:
obj.hide_set(not self.has_mesh)
def on_lod_level_exit(self):
"""Called when the LOD level switches aways from this level."""
obj: Object = self.id_data
# Store a reference to the mesh
self.mesh_ref = obj.data if self.has_mesh else None
def _get_mesh_name(self) -> str:
m = self.mesh
return m.name if m is not None else ""
def _set_mesh_name(self, value: str) -> str:
self.mesh = bpy.data.meshes.get(value, None)
# Wrapper property so we can modify the LOD meshes from the UI without keeping a reference to the mesh of the
# active LOD.
mesh_name: StringProperty(get=_get_mesh_name, set=_set_mesh_name)
# Only set on non-active LOD levels, used to keep a reference to the mesh data-block so Blender doesn't remove it
# during garbage collection. For the active LOD, the mesh is the current object.data mesh.
# Blender expects a single reference to the mesh (in object.data) for many operations, otherwise asks
# the user to duplicate the mesh and then object.data becomes out-of-sync with Sollumz LODs.
mesh_ref: PointerProperty(type=Mesh) # DO NOT MODIFY DIRECTLY OUTSIDE THIS CLASS, use .mesh or .mesh_name
# Whether this LOD actually has a mesh. Needed because when a level that doesn't have a mesh becomes the active LOD,
# object.data keeps the previous mesh and instead the object is hidden, so when switching away we wouldn't know if
# object.data is actually our mesh or not.
has_mesh: BoolProperty(default=False) # DO NOT MODIFY DIRECTLY OUTSIDE THIS CLASS, use .mesh or .mesh_name
@property
def mesh(self) -> Optional[Mesh]:
"""Gets the mesh of this LOD level, or ``None`` if there is no mesh."""
if not self.has_mesh:
return None
obj: Object = self.id_data
lods: LODLevels = obj.sz_lods
if lods.active_lod_level == self.level:
return obj.data
else:
return self.mesh_ref
@mesh.setter
def mesh(self, value: Optional[Mesh]):
"""Sets the mesh of this LOD level. Set to ``None`` to remove the mesh."""
obj: Object = self.id_data
lods: LODLevels = obj.sz_lods
self.has_mesh = value is not None
if lods.active_lod_level == self.level:
self.mesh_ref = None
if self.has_mesh:
obj.data = value
if obj.name in bpy.context.view_layer.objects:
obj.hide_set(not self.has_mesh)
else:
self.mesh_ref = value
@property
def level(self) -> LODLevel:
obj: Object = self.id_data
lods: LODLevels = obj.sz_lods
if lods.very_high == self:
return LODLevel.VERYHIGH
if lods.high == self:
return LODLevel.HIGH
if lods.medium == self:
return LODLevel.MEDIUM
if lods.low == self:
return LODLevel.LOW
if lods.very_low == self:
return LODLevel.VERYLOW
assert False, "LODLevelProps for unknown LOD level"
class LODLevels(PropertyGroup):
def on_lod_level_update(self, context: Context):
prev_lod = self.get_lod(self.active_lod_level_prev)
curr_lod = self.get_lod(self.active_lod_level)
prev_lod.on_lod_level_exit()
curr_lod.on_lod_level_enter()
self.active_lod_level_prev = self.active_lod_level
active_lod_level: EnumProperty(items=LODLevelEnumItems, update=on_lod_level_update)
active_lod_level_prev: EnumProperty(items=LODLevelEnumItems)
very_high: PointerProperty(type=LODLevelProps)
high: PointerProperty(type=LODLevelProps)
medium: PointerProperty(type=LODLevelProps)
low: PointerProperty(type=LODLevelProps)
very_low: PointerProperty(type=LODLevelProps)
@property
def active_lod(self) -> LODLevelProps:
return self.get_lod(self.active_lod_level)
def get_lod(self, lod_level: LODLevel) -> LODLevelProps:
match lod_level:
case LODLevel.VERYHIGH:
return self.very_high
case LODLevel.HIGH:
return self.high
case LODLevel.MEDIUM:
return self.medium
case LODLevel.LOW:
return self.low
case LODLevel.VERYLOW:
return self.very_low
case _:
assert False, f"Unknown LOD level '{lod_level}'"
def set_highest_lod_active(self):
for lod_level in LODLevel:
lod = self.get_lod(lod_level)
if lod.mesh is not None:
self.active_lod_level = lod_level
return
class SetLodLevelHelper:
"""Helper class for setting the LOD level of Sollumz objects."""
LOD_LEVEL: LODLevel = LODLevel.HIGH
bl_description = "Set the viewing level for the selected Fragment/Drawable"
@classmethod
def poll(cls, context):
active_obj = context.view_layer.objects.active
return active_obj is not None and find_sollumz_parent(active_obj)
def execute(self, context):
active_obj = context.view_layer.objects.active
obj = find_sollumz_parent(active_obj)
set_all_lods(obj, self.LOD_LEVEL)
return {"FINISHED"}
class SOLLUMZ_OT_SET_LOD_VERY_HIGH(bpy.types.Operator, SetLodLevelHelper):
bl_idname = "sollumz.set_lod_very_high"
bl_label = "Very High"
LOD_LEVEL = LODLevel.VERYHIGH
class SOLLUMZ_OT_SET_LOD_HIGH(bpy.types.Operator, SetLodLevelHelper):
bl_idname = "sollumz.set_lod_high"
bl_label = "High"
LOD_LEVEL = LODLevel.HIGH
class SOLLUMZ_OT_SET_LOD_MED(bpy.types.Operator, SetLodLevelHelper):
bl_idname = "sollumz.set_lod_med"
bl_label = "Medium"
LOD_LEVEL = LODLevel.MEDIUM
class SOLLUMZ_OT_SET_LOD_LOW(bpy.types.Operator, SetLodLevelHelper):
bl_idname = "sollumz.set_lod_low"
bl_label = "Low"
LOD_LEVEL = LODLevel.LOW
class SOLLUMZ_OT_SET_LOD_VLOW(bpy.types.Operator, SetLodLevelHelper):
bl_idname = "sollumz.set_lod_vlow"
bl_label = "Very Low"
LOD_LEVEL = LODLevel.VERYLOW
class SOLLUMZ_OT_HIDE_OBJECT(bpy.types.Operator, SetLodLevelHelper):
bl_idname = "sollumz.hide_object"
bl_label = "Hidden"
def execute(self, context):
active_obj = context.view_layer.objects.active
obj = find_sollumz_parent(active_obj)
obj_hidden = obj.sollumz_obj_is_hidden
do_hide = not obj_hidden
obj.sollumz_obj_is_hidden = do_hide
obj.hide_set(do_hide)
for child in obj.children_recursive:
active_lod = child.sz_lods.active_lod
if child.sollum_type != SollumType.DRAWABLE_MODEL or active_lod.mesh is None:
continue
child.hide_set(do_hide)
return {"FINISHED"}
class SOLLUMZ_OT_HIDE_COLLISIONS(bpy.types.Operator):
bl_idname = "sollumz.hide_collisions"
bl_label = "Hide Collisions"
bl_description = "Hide all collisions in the scene"
def execute(self, context):
context.scene.sollumz_show_collisions = False
set_collision_visibility(False)
return {"FINISHED"}
class SOLLUMZ_OT_SHOW_COLLISIONS(bpy.types.Operator):
bl_idname = "sollumz.show_collisions"
bl_label = "Show Collisions"
bl_description = "Show all collisions in the scene"
def execute(self, context):
context.scene.sollumz_show_collisions = True
set_collision_visibility(True)
return {"FINISHED"}
class SOLLUMZ_OT_SHOW_SHATTERMAPS(bpy.types.Operator):
bl_idname = "sollumz.show_shattermaps"
bl_label = "Show Shattermaps"
bl_description = "Show all shattermaps in the scene"
def execute(self, context):
context.scene.sollumz_show_shattermaps = True
set_shattermaps_visibility(True)
return {"FINISHED"}
class SOLLUMZ_OT_HIDE_SHATTERMAPS(bpy.types.Operator):
bl_idname = "sollumz.hide_shattermaps"
bl_label = "Hide Shattermaps"
bl_description = "Hide all shattermaps in the scene"
def execute(self, context):
context.scene.sollumz_show_shattermaps = False
set_shattermaps_visibility(False)
return {"FINISHED"}
class SOLLUMZ_OT_copy_lod(bpy.types.Operator):
bl_idname = "sollumz.copy_lod"
bl_label = "Copy LOD"
bl_description = "Copy the current LOD level into the specified LOD level"
bl_options = {"REGISTER", "UNDO"}
copy_lod_levels: lod_level_enum_flag_prop_factory(default={LODLevel.HIGH})
@classmethod
def poll(self, context):
aobj = context.active_object
return aobj is not None and aobj.sz_lods.active_lod.mesh is not None
def draw(self, context):
self.layout.props_enum(self, "copy_lod_levels")
def execute(self, context):
aobj = context.active_object
lods = aobj.sz_lods
active_lod = lods.active_lod
active_lod_mesh = active_lod.mesh
for lod_level in self.copy_lod_levels:
lod = lods.get_lod(lod_level)
if lod is None:
return {"CANCELLED"}
if lod.mesh is not None:
self.report({"INFO"}, f"{SOLLUMZ_UI_NAMES[lod_level]} already has a mesh!")
return {"CANCELLED"}
lod.mesh = active_lod_mesh.copy()
return {"FINISHED"}
def invoke(self, context, event):
return context.window_manager.invoke_props_dialog(self)
class SOLLUMZ_PT_LOD_LEVEL_PANEL(bpy.types.Panel):
bl_label = "Sollumz LODs"
bl_idname = "SOLLUMZ_PT_LOD_LEVEL_PANEL"
bl_space_type = "PROPERTIES"
bl_region_type = "WINDOW"
bl_context = "data"
@classmethod
def poll(cls, context):
active_obj = context.view_layer.objects.active
return active_obj is not None and active_obj.type == "MESH" and active_obj.sollum_type in [*FRAGMENT_TYPES, *DRAWABLE_TYPES]
def draw_header(self, context):
icon_manager.icon_label("sollumz_icon", self)
def draw(self, context):
layout = self.layout
active_obj = context.view_layer.objects.active
lods = active_obj.sz_lods
layout.enabled = active_obj.mode == "OBJECT"
row = layout.row()
col = row.column(align=True)
for lod_level in (
LODLevel.VERYHIGH,
LODLevel.HIGH,
LODLevel.MEDIUM,
LODLevel.LOW,
LODLevel.VERYLOW,
):
lod = lods.get_lod(lod_level)
lod_split = col.split(align=True, factor=0.3)
lod_split.prop_enum(lods, "active_lod_level", lod_level)
lod_split.prop_search(lod, "mesh_name", bpy.data, "meshes", text="")
row.operator(SOLLUMZ_OT_copy_lod.bl_idname, icon="COPYDOWN", text="")
def set_collision_visibility(is_visible: bool):
"""Set visibility of all collision objects in the scene"""
for obj in bpy.context.view_layer.objects:
obj_is_collision = obj.sollum_type in BOUND_TYPES or obj.sollum_type in BOUND_POLYGON_TYPES
if not obj_is_collision:
continue
obj.hide_set(not is_visible)
def set_shattermaps_visibility(is_visible: bool):
"""Set visibility of all shattermap objects in the scene"""
for obj in bpy.context.view_layer.objects:
if obj.sollum_type != SollumType.SHATTERMAP:
continue
obj.hide_set(not is_visible)
def set_all_lods(obj: bpy.types.Object, lod_level: LODLevel):
"""Set LOD levels of all of children of ``obj``"""
obj.sollumz_obj_is_hidden = False
obj.hide_set(False)
for child in obj.children_recursive:
if child.type == "MESH" and child.sollum_type == SollumType.DRAWABLE_MODEL:
child.sz_lods.active_lod_level = lod_level
continue
def operates_on_lod_level(func: Callable):
"""Decorator for functions that operate on a particular LOD level of an object.
Will automatically set the LOD level to ``lod_level`` at the beginning of execution
and will set it back to the original LOD level at the end."""
def wrapper(model_obj: bpy.types.Object, lod_level: LODLevel, *args, **kwargs):
current_lod_level = model_obj.sz_lods.active_lod_level
was_hidden = model_obj.hide_get()
model_obj.sz_lods.active_lod_level = lod_level
res = func(model_obj, lod_level, *args, **kwargs)
# Set the lod level back to what it was
model_obj.sz_lods.active_lod_level = current_lod_level
model_obj.hide_set(was_hidden)
return res
return wrapper
def register():
bpy.types.Object.sz_lods = bpy.props.PointerProperty(type=LODLevels)
bpy.types.Object.sollumz_obj_is_hidden = bpy.props.BoolProperty()
bpy.types.Scene.sollumz_show_collisions = bpy.props.BoolProperty(default=True)
bpy.types.Scene.sollumz_show_shattermaps = bpy.props.BoolProperty(default=True)
bpy.types.Scene.sollumz_copy_lod_level = bpy.props.EnumProperty(items=LODLevelEnumItems)
def unregister():
del bpy.types.Object.sz_lods
del bpy.types.Object.sollumz_obj_is_hidden
del bpy.types.Scene.sollumz_show_collisions
del bpy.types.Scene.sollumz_show_shattermaps
del bpy.types.Scene.sollumz_copy_lod_level