yyc12345
f4d3e48be2
- add TinyMutex to resolve the issue that we can not operate the virtools group infos of 2 individual objects. * use this new class for all class need resource mutex, such as ballance element, bme materials and etc. - add CollectionVisitor to resolve that blender have bad document and type hint for runtime bpy.types.CollectionProperty. * now all visit to CollectionProperty are delegated by this class.
308 lines
11 KiB
Python
308 lines
11 KiB
Python
import bpy
|
|
import typing, enum, copy
|
|
from . import PROP_virtools_material, PROP_virtools_texture
|
|
from . import UTIL_ballance_texture, UTIL_functions, UTIL_icons_manager
|
|
|
|
#region BME Material Presets
|
|
|
|
class _BMEMaterialPreset():
|
|
## Associated Ballance texture file name, including file extension.
|
|
mTexName: str
|
|
## Predefined mtl preset in virtools material module
|
|
mRawMtl: PROP_virtools_material.RawVirtoolsMaterial
|
|
|
|
def __init__(self, texname: str, rawmtl: PROP_virtools_material.RawVirtoolsMaterial):
|
|
self.mTexName = texname
|
|
self.mRawMtl = rawmtl
|
|
|
|
_g_BMEMaterialPresets: dict[str, _BMEMaterialPreset] = {
|
|
'FloorSide': _BMEMaterialPreset(
|
|
'Floor_Side.bmp',
|
|
PROP_virtools_material.get_virtools_material_preset(PROP_virtools_material.MaterialPresetType.FloorSide).mData
|
|
),
|
|
'LightingFloorTopBorder': _BMEMaterialPreset(
|
|
'Floor_Top_Border.bmp',
|
|
PROP_virtools_material.get_virtools_material_preset(PROP_virtools_material.MaterialPresetType.FloorSide).mData
|
|
),
|
|
'LightingFloorTopBorderless': _BMEMaterialPreset(
|
|
'Floor_Top_Borderless.bmp',
|
|
PROP_virtools_material.get_virtools_material_preset(PROP_virtools_material.MaterialPresetType.FloorSide).mData
|
|
),
|
|
|
|
'FloorTopBorder': _BMEMaterialPreset(
|
|
'Floor_Top_Border.bmp',
|
|
PROP_virtools_material.get_virtools_material_preset(PROP_virtools_material.MaterialPresetType.FloorTop).mData
|
|
),
|
|
'FloorTopBorderless': _BMEMaterialPreset(
|
|
'Floor_Top_Borderless.bmp',
|
|
PROP_virtools_material.get_virtools_material_preset(PROP_virtools_material.MaterialPresetType.FloorTop).mData
|
|
),
|
|
'FloorTopFlat': _BMEMaterialPreset(
|
|
'Floor_Top_Flat.bmp',
|
|
PROP_virtools_material.get_virtools_material_preset(PROP_virtools_material.MaterialPresetType.FloorTop).mData
|
|
),
|
|
'FloorTopProfil': _BMEMaterialPreset(
|
|
'Floor_Top_Profil.bmp',
|
|
PROP_virtools_material.get_virtools_material_preset(PROP_virtools_material.MaterialPresetType.FloorTop).mData
|
|
),
|
|
'FloorTopProfilFlat': _BMEMaterialPreset(
|
|
'Floor_Top_ProfilFlat.bmp',
|
|
PROP_virtools_material.get_virtools_material_preset(PROP_virtools_material.MaterialPresetType.FloorTop).mData
|
|
),
|
|
|
|
'BallPaper': _BMEMaterialPreset(
|
|
'Ball_Paper.bmp',
|
|
PROP_virtools_material.get_virtools_material_preset(PROP_virtools_material.MaterialPresetType.TrafoPaper).mData
|
|
),
|
|
|
|
'BallStone': _BMEMaterialPreset(
|
|
'Ball_Stone.bmp',
|
|
PROP_virtools_material.get_virtools_material_preset(PROP_virtools_material.MaterialPresetType.TraforWoodStone).mData
|
|
),
|
|
'BallWood': _BMEMaterialPreset(
|
|
'Ball_Wood.bmp',
|
|
PROP_virtools_material.get_virtools_material_preset(PROP_virtools_material.MaterialPresetType.TraforWoodStone).mData
|
|
),
|
|
|
|
'Rail': _BMEMaterialPreset(
|
|
'Rail_Environment.bmp',
|
|
PROP_virtools_material.get_virtools_material_preset(PROP_virtools_material.MaterialPresetType.Rail).mData
|
|
),
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region BME Material Define & Visitor
|
|
|
|
class BBP_PG_bme_material(bpy.types.PropertyGroup):
|
|
bme_material_name: bpy.props.StringProperty(
|
|
name = "Name",
|
|
default = ""
|
|
) # type: ignore
|
|
|
|
material_ptr: bpy.props.PointerProperty(
|
|
name = "Material",
|
|
type = bpy.types.Material
|
|
) # type: ignore
|
|
|
|
def get_bme_materials(scene: bpy.types.Scene) -> UTIL_functions.CollectionVisitor[BBP_PG_bme_material]:
|
|
return UTIL_functions.CollectionVisitor(scene.bme_materials)
|
|
|
|
#endregion
|
|
|
|
#region Material Preset Loader
|
|
|
|
def _load_bme_material_preset(mtl: bpy.types.Material, preset_name: str) -> None:
|
|
# get preset first
|
|
preset: _BMEMaterialPreset = _g_BMEMaterialPresets[preset_name]
|
|
|
|
# get raw mtl and do a shallow copy
|
|
# because we will change something later. but do not want to affect preset self.
|
|
raw_mtl: PROP_virtools_material.RawVirtoolsMaterial = copy.copy(preset.mRawMtl)
|
|
|
|
# load ballance texture
|
|
blctex: bpy.types.Image = UTIL_ballance_texture.load_ballance_texture(preset.mTexName)
|
|
# apply texture props
|
|
PROP_virtools_texture.set_raw_virtools_texture(blctex, PROP_virtools_texture.get_ballance_texture_preset(preset.mTexName))
|
|
# set loaded texture to shallow copied raw mtl
|
|
raw_mtl.mTexture = blctex
|
|
|
|
# set raw mtl
|
|
PROP_virtools_material.set_raw_virtools_material(mtl, raw_mtl)
|
|
# apply vt mtl to blender mtl
|
|
PROP_virtools_material.apply_to_blender_material(mtl)
|
|
|
|
#endregion
|
|
|
|
#region BME Material Operation Help Class & Functions
|
|
|
|
class BMEMaterialsHelper():
|
|
"""
|
|
The helper of BME materials processing.
|
|
|
|
All BME materials operations, including getting or setting, must be manipulated by this class.
|
|
You should NOT operate BME Materials property (in Scene) directly.
|
|
|
|
This class should only have 1 instance at the same time. This class support `with` syntax to achieve this.
|
|
This class frequently used in creating BME meshes.
|
|
"""
|
|
__mSingletonMutex: typing.ClassVar[UTIL_functions.TinyMutex[bpy.types.Scene]] = UTIL_functions.TinyMutex()
|
|
__mIsValid: bool
|
|
__mAssocScene: bpy.types.Scene
|
|
__mMaterialMap: dict[str, bpy.types.Material]
|
|
|
|
def __init__(self, assoc: bpy.types.Scene):
|
|
self.__mMaterialMap = {}
|
|
self.__mAssocScene = assoc
|
|
self.__mIsValid = False
|
|
|
|
# check singleton
|
|
BMEMaterialsHelper.__mSingletonMutex.lock(self.__mAssocScene)
|
|
# set validation and read ballance elements property
|
|
self.__mIsValid = True
|
|
self.__read_from_bme_materials()
|
|
|
|
def is_valid(self) -> bool:
|
|
return self.__mIsValid
|
|
|
|
def __enter__(self):
|
|
return self
|
|
|
|
def __exit__(self, exc_type, exc_value, traceback):
|
|
self.dispose()
|
|
|
|
def dispose(self) -> None:
|
|
if self.is_valid():
|
|
# write to ballance elements property and reset validation
|
|
self.__write_to_bme_materials()
|
|
self.__mIsValid = False
|
|
BMEMaterialsHelper.__mSingletonMutex.unlock(self.__mAssocScene)
|
|
|
|
def get_material(self, preset_name: str) -> bpy.types.Material:
|
|
if not self.is_valid():
|
|
raise UTIL_functions.BBPException('calling invalid BMEMaterialsHelper')
|
|
|
|
# get exist one
|
|
mtl: bpy.types.Material | None = self.__mMaterialMap.get(preset_name, None)
|
|
if mtl is not None:
|
|
return mtl
|
|
|
|
# if no existing one, create new one
|
|
new_mtl_name: str = 'BME' + preset_name
|
|
new_mtl: bpy.types.Material = bpy.data.materials.new(new_mtl_name)
|
|
|
|
_load_bme_material_preset(new_mtl, preset_name)
|
|
self.__mMaterialMap[preset_name] = new_mtl
|
|
return new_mtl
|
|
|
|
def __write_to_bme_materials(self) -> None:
|
|
mtls = get_bme_materials(self.__mAssocScene)
|
|
mtls.clear()
|
|
|
|
for preset_name, mtl in self.__mMaterialMap.items():
|
|
item: BBP_PG_bme_material = mtls.add()
|
|
item.bme_material_name = preset_name
|
|
item.material_ptr = mtl
|
|
|
|
def __read_from_bme_materials(self) -> None:
|
|
mtls = get_bme_materials(self.__mAssocScene)
|
|
self.__mMaterialMap.clear()
|
|
|
|
for item in mtls:
|
|
# check requirements
|
|
if item.material_ptr is None: continue
|
|
# add into map
|
|
self.__mMaterialMap[item.bme_material_name] = item.material_ptr
|
|
|
|
def reset_bme_materials(scene: bpy.types.Scene) -> None:
|
|
invalid_idx: list[int] = []
|
|
mtls = get_bme_materials(scene)
|
|
|
|
# re-load all elements
|
|
index: int = 0
|
|
item: BBP_PG_bme_material
|
|
for item in mtls:
|
|
# load or record invalid entry
|
|
if item.material_ptr is None:
|
|
invalid_idx.append(index)
|
|
else:
|
|
_load_bme_material_preset(item.material_ptr, item.bme_material_name)
|
|
|
|
# inc counter
|
|
index += 1
|
|
|
|
# remove invalid one with reversed order
|
|
invalid_idx.reverse()
|
|
for idx in invalid_idx:
|
|
mtls.remove(idx)
|
|
|
|
#endregion
|
|
|
|
#region BME Materials Representation
|
|
|
|
class BBP_UL_bme_materials(bpy.types.UIList):
|
|
def draw_item(self, context, layout: bpy.types.UILayout, data, item: BBP_PG_bme_material, icon, active_data, active_propname):
|
|
# check requirements
|
|
if item.material_ptr is None: return
|
|
# draw list item
|
|
layout.label(text = item.bme_material_name, translate = False)
|
|
layout.label(text = item.material_ptr.name, translate = False, icon = 'MATERIAL')
|
|
|
|
class BBP_OT_reset_bme_materials(bpy.types.Operator):
|
|
"""Reset all BME Materials to Default Settings."""
|
|
bl_idname = "bbp.reset_bme_materials"
|
|
bl_label = "Reset BME Materials"
|
|
bl_options = {'UNDO'}
|
|
|
|
@classmethod
|
|
def poll(cls, context):
|
|
return context.scene is not None
|
|
|
|
def execute(self, context):
|
|
reset_bme_materials(context.scene)
|
|
# show a window to let user know, not silence
|
|
UTIL_functions.message_box(
|
|
('Reset OK.', ),
|
|
"Reset Result",
|
|
UTIL_icons_manager.BlenderPresetIcons.Info.value
|
|
)
|
|
return {'FINISHED'}
|
|
|
|
class BBP_PT_bme_materials(bpy.types.Panel):
|
|
"""Show BME Materials Properties."""
|
|
bl_label = "BME Materials"
|
|
bl_idname = "BBP_PT_bme_materials"
|
|
bl_space_type = 'PROPERTIES'
|
|
bl_region_type = 'WINDOW'
|
|
bl_context = "scene"
|
|
|
|
@classmethod
|
|
def poll(cls, context):
|
|
return context.scene is not None
|
|
|
|
def draw(self, context):
|
|
layout: bpy.types.UILayout = self.layout
|
|
target: bpy.types.Scene = context.scene
|
|
col = layout.column()
|
|
|
|
# show restore operator
|
|
opercol = col.column()
|
|
opercol.operator(BBP_OT_reset_bme_materials.bl_idname, icon='LOOP_BACK')
|
|
|
|
# show list but not allowed to edit
|
|
listcol = col.column()
|
|
listcol.enabled = False
|
|
listcol.template_list(
|
|
"BBP_UL_bme_materials", "",
|
|
target, "bme_materials",
|
|
target, "active_bme_materials",
|
|
# default row height is a half of the count of all presets
|
|
# limit the max row height to the the count of all presets
|
|
rows = len(_g_BMEMaterialPresets) // 2,
|
|
maxrows = len(_g_BMEMaterialPresets),
|
|
)
|
|
|
|
#endregion
|
|
|
|
def register() -> None:
|
|
# register all classes
|
|
bpy.utils.register_class(BBP_PG_bme_material)
|
|
bpy.utils.register_class(BBP_UL_bme_materials)
|
|
bpy.utils.register_class(BBP_OT_reset_bme_materials)
|
|
bpy.utils.register_class(BBP_PT_bme_materials)
|
|
|
|
# add into scene metadata
|
|
bpy.types.Scene.bme_materials = bpy.props.CollectionProperty(type = BBP_PG_bme_material)
|
|
bpy.types.Scene.active_bme_materials = bpy.props.IntProperty()
|
|
|
|
def unregister() -> None:
|
|
# del from scene metadata
|
|
del bpy.types.Scene.active_bme_materials
|
|
del bpy.types.Scene.bme_materials
|
|
|
|
bpy.utils.unregister_class(BBP_PT_bme_materials)
|
|
bpy.utils.unregister_class(BBP_OT_reset_bme_materials)
|
|
bpy.utils.unregister_class(BBP_UL_bme_materials)
|
|
bpy.utils.unregister_class(BBP_PG_bme_material)
|
|
|