BallanceBlenderHelper/ballance_blender_plugin/NAMES_rename_system.py

563 lines
20 KiB
Python
Raw Normal View History

2022-04-04 22:04:11 +08:00
import bpy
from . import UTILS_constants, UTILS_functions, UTILS_virtools_prop
2022-04-04 22:04:11 +08:00
class rename_system_props(bpy.types.Operator):
name_standard: bpy.props.EnumProperty(
name="Name Standard",
2022-04-05 20:45:00 +08:00
description="Choose your name standard",
2022-04-04 22:04:11 +08:00
items=(
("YYC", "YYC Tools Chains", "YYC Tools Chains name standard."),
("IMENGYU", "Imengyu Ballance", "Auto grouping name standard for Imengyu/Ballance")
),
)
def invoke(self, context, event):
wm = context.window_manager
return wm.invoke_props_dialog(self)
def draw(self, context):
layout = self.layout
layout.prop(self, "name_standard")
2022-04-05 15:55:42 +08:00
class BALLANCE_OT_rename_by_group(rename_system_props):
"""Rename object by Virtools groups"""
bl_idname = "ballance.rename_by_group"
bl_label = "Rename by Group"
2022-04-04 22:04:11 +08:00
bl_options = {'UNDO'}
def execute(self, context):
2022-04-05 21:16:53 +08:00
_rename_core(_NameStandard.CKGROUP, _NameStandard.cvt_std_from_str_to_int(self.name_standard))
2022-04-04 22:04:11 +08:00
return {'FINISHED'}
class BALLANCE_OT_convert_name(rename_system_props):
"""Convert name from one name standard to another one."""
bl_idname = "ballance.convert_name"
bl_label = "Convert Name"
bl_options = {'UNDO'}
2022-04-05 21:16:53 +08:00
dest_name_standard: bpy.props.EnumProperty(
name="Destination Name Standard",
description="Choose your name standard",
items=(
("YYC", "YYC Tools Chains", "YYC Tools Chains name standard."),
("IMENGYU", "Imengyu Ballance", "Auto grouping name standard for Imengyu/Ballance")
),
)
2022-04-04 22:04:11 +08:00
def execute(self, context):
2022-04-05 21:16:53 +08:00
_rename_core(
_NameStandard.cvt_std_from_str_to_int(self.name_standard),
_NameStandard.cvt_std_from_str_to_int(self.dest_name_standard))
2022-04-04 22:04:11 +08:00
return {'FINISHED'}
2022-04-05 21:16:53 +08:00
# rewrite draw func
def draw(self, context):
layout = self.layout
layout.prop(self, "name_standard")
layout.prop(self, "dest_name_standard")
2022-04-04 22:04:11 +08:00
class BALLANCE_OT_auto_grouping(rename_system_props):
"""Auto Grouping object according to specific name standard."""
bl_idname = "ballance.auto_grouping"
bl_label = "Auto Grouping"
bl_options = {'UNDO'}
def execute(self, context):
2022-04-05 21:16:53 +08:00
_rename_core(_NameStandard.cvt_std_from_str_to_int(self.name_standard), _NameStandard.CKGROUP)
2022-04-04 22:04:11 +08:00
return {'FINISHED'}
2022-04-05 20:45:00 +08:00
# ==========================================
# rename misc funcs
class _RenameErrorType():
ERROR = 0
WARNING = 1
INFO = 2
@staticmethod
def cvt_err_from_int_to_str(err_t):
if err_t == _RenameErrorType.ERROR:
return "ERROR"
elif err_t == _RenameErrorType.WARNING:
return "WARNING"
elif err_t == _RenameErrorType.INFO:
return "INFO"
else:
raise Exception("Unknow error type.")
class _RenameErrorItem():
def __init__(self, err_t, description):
self.err_type = err_t
self.description = description
def get_presentation(self):
return "[{}]\t{}".format(_RenameErrorType.cvt_err_from_int_to_str(self.err_type), self.description)
class _RenameErrorReporter():
def __init__(self):
self.err_container: list[_RenameErrorItem] = []
def add_error(self, description):
self.err_container.append(_RenameErrorItem(_RenameErrorType.ERROR, description))
def add_warning(self, description):
self.err_container.append(_RenameErrorItem(_RenameErrorType.WARNING, description))
def add_info(self, description):
self.err_container.append(_RenameErrorItem(_RenameErrorType.INFO, description))
def can_report(self):
return len(self.err_container) != 0
def report(self, header):
print(header)
for i in self.err_container:
print('\t' + i.get_presentation())
def clear(self):
self.err_container.clear()
2022-04-05 15:55:42 +08:00
class _ObjectBasicType():
2022-04-04 22:04:11 +08:00
COMPONENT = 0
2022-04-05 15:55:42 +08:00
FLOOR = 1
RAIL = 2
WOOD = 3
STOPPER = 4
DEPTH_CUBE = 5
DECORATION = 6
LEVEL_START = 7
LEVEL_END = 8
CHECKPOINT = 9
RESETPOINT = 10
class _NameStandard():
2022-04-05 21:16:53 +08:00
CKGROUP = 0
YYC = 1
IMENGYU = 2
2022-04-05 15:55:42 +08:00
@staticmethod
def cvt_std_from_str_to_int(std_str):
if std_str == "YYC":
return _NameStandard.YYC
elif std_str == "IMENGYU":
return _NameStandard.IMENGYU
else:
raise Exception("Unknow name standard.")
class _NameInfoHelper():
2022-04-05 21:16:53 +08:00
def __init__(self, _basic_type):
2022-04-04 22:04:11 +08:00
self.basic_type = _basic_type
# extra field notes:
2022-04-05 15:55:42 +08:00
# COMPONENT:
# component_type(string)
# sector(int)
# CHECKPOINT, RESETPOINT:
# sector(int)(following Ballance index, checkpoint starts with 1)
2022-04-04 22:04:11 +08:00
2022-04-05 21:16:53 +08:00
def _get_selected_objects():
return bpy.context.view_layer.active_layer_collection.collection.objects
2022-04-05 15:55:42 +08:00
2022-04-05 20:45:00 +08:00
def _get_sector_from_ckgroup(group_set):
# this counter is served for stupid
# multi-sector-grouping accident.
2022-04-05 15:55:42 +08:00
counter = 0
last_matched_sector = ''
for i in group_set:
regex_result = UTILS_constants.rename_regexCKGroupSector.match(i)
if regex_result is not None:
last_matched_sector = regex_result.group(1)
counter += 1
if counter != 1:
return None
2022-04-04 22:04:11 +08:00
else:
2022-04-05 15:55:42 +08:00
return last_matched_sector
2022-04-04 22:04:11 +08:00
2022-04-05 20:45:00 +08:00
# ==========================================
# rename core funcs
# NOTE: the implement of this function are copied from
# BallanceVirtoolsHelper/bvh/features/mapping/grouping.cpp
2022-04-06 14:18:45 +08:00
# ---
# YYC Tools Chains name standard is Ballance-compatible name standard.
# So this functions also serving for `_get_name_info_from_group` function
# to help get sector field from PC/PR elements. In ordinary call(external call)
# The final error output should be outputed normally. But in the call from
2022-04-06 14:18:45 +08:00
# `_get_name_info_from_group`, this function should not output any error.
# So parameter `call_internal` is served for this work. In common it is False
# to let function output error str normally. But only set it to True in
# the call from `_get_name_info_from_group` to disable error output.
def _get_name_info_from_yyc_name(obj_name, err_reporter: _RenameErrorReporter, call_internal = False):
2022-04-05 20:45:00 +08:00
# check component first
regex_result = UTILS_constants.rename_regexYYCComponent.match(obj_name)
if regex_result is not None:
data = _NameInfoHelper(_ObjectBasicType.COMPONENT)
data.component_type = regex_result.group(1)
data.sector = int(regex_result.group(2))
return data
# check PC PR elements
2022-04-05 21:16:53 +08:00
regex_result = UTILS_constants.rename_regexYYCPC.match(obj_name)
2022-04-05 20:45:00 +08:00
if regex_result is not None:
data = _NameInfoHelper(_ObjectBasicType.CHECKPOINT)
data.sector = int(regex_result.group(1))
return data
2022-04-05 21:16:53 +08:00
regex_result = UTILS_constants.rename_regexYYCPR.match(obj_name)
2022-04-05 20:45:00 +08:00
if regex_result is not None:
data = _NameInfoHelper(_ObjectBasicType.RESETPOINT)
data.sector = int(regex_result.group(1))
return data
# check other unique elements
if obj_name == "PS_FourFlames_01":
return _NameInfoHelper(_ObjectBasicType.LEVEL_START)
if obj_name == "PE_Balloon_01":
return _NameInfoHelper(_ObjectBasicType.LEVEL_END)
# process floors
if obj_name.startswith("A_Floor"):
return _NameInfoHelper(_ObjectBasicType.FLOOR)
if obj_name.startswith("A_Wood"):
return _NameInfoHelper(_ObjectBasicType.WOOD)
if obj_name.startswith("A_Rail"):
return _NameInfoHelper(_ObjectBasicType.RAIL)
if obj_name.startswith("A_Stopper"):
return _NameInfoHelper(_ObjectBasicType.STOPPER)
# process others
if obj_name.startswith("DepthCubes"):
return _NameInfoHelper(_ObjectBasicType.DEPTH_CUBE)
if obj_name.startswith("D_"):
return _NameInfoHelper(_ObjectBasicType.DECORATION)
2022-04-06 14:18:45 +08:00
# only output in external calling
if not call_internal:
err_reporter.add_error("Name match lost.")
2022-04-06 14:18:45 +08:00
2022-04-05 20:45:00 +08:00
return None
2022-04-04 22:04:11 +08:00
def _get_name_info_from_imengyu_name(obj_name, err_reporter: _RenameErrorReporter):
2022-04-05 20:45:00 +08:00
# check component first
regex_result = UTILS_constants.rename_regexImengyuComponent.match(obj_name)
if regex_result is not None:
data = _NameInfoHelper(_ObjectBasicType.COMPONENT)
data.component_type = regex_result.group(1)
data.sector = int(regex_result.group(2))
return data
# check PC PR elements
2022-04-05 21:16:53 +08:00
regex_result = UTILS_constants.rename_regexImengyuPCRComp.match(obj_name)
2022-04-05 20:45:00 +08:00
if regex_result is not None:
eles_name = regex_result.group(1)
if eles_name == 'PC_CheckPoint':
data = _NameInfoHelper(_ObjectBasicType.CHECKPOINT)
elif eles_name == 'PR_ResetPoint':
data = _NameInfoHelper(_ObjectBasicType.RESETPOINT)
data.sector = int(regex_result.group(2))
return data
# check other unique elements
if obj_name == "PS_LevelStart":
return _NameInfoHelper(_ObjectBasicType.LEVEL_START)
if obj_name == "PE_LevelEnd":
return _NameInfoHelper(_ObjectBasicType.LEVEL_END)
# process floors
if obj_name.startswith("S_Floors"):
return _NameInfoHelper(_ObjectBasicType.FLOOR)
if obj_name.startswith("S_FloorWoods"):
return _NameInfoHelper(_ObjectBasicType.WOOD)
if obj_name.startswith("S_FloorRails"):
return _NameInfoHelper(_ObjectBasicType.RAIL)
if obj_name.startswith("S_FloorStopper"):
return _NameInfoHelper(_ObjectBasicType.STOPPER)
# process others
if obj_name.startswith("DepthTestCubes"):
return _NameInfoHelper(_ObjectBasicType.DEPTH_CUBE)
if obj_name.startswith("O_"):
return _NameInfoHelper(_ObjectBasicType.DECORATION)
err_reporter.add_error("Name match lost.")
2022-04-05 20:45:00 +08:00
return None
2022-04-04 22:04:11 +08:00
def _get_name_info_from_group(obj, err_reporter: _RenameErrorReporter):
group_list = UTILS_virtools_prop.get_virtools_group_data(obj)
if len(group_list) == 0:
2022-04-05 15:55:42 +08:00
# name it as a decoration
return _NameInfoHelper(_ObjectBasicType.DECORATION)
2022-04-04 22:04:11 +08:00
2022-04-05 15:55:42 +08:00
group_set = set(group_list)
2022-04-04 22:04:11 +08:00
2022-04-05 15:55:42 +08:00
# try to filter unique elements first
2022-04-05 21:16:53 +08:00
set_result = UTILS_constants.rename_uniqueComponentsGroupName.intersection(group_set)
2022-04-05 15:55:42 +08:00
if len(set_result) == 1:
# get it
gotten_group_name = (list(set_result))[0]
if gotten_group_name == 'PS_Levelstart':
return _NameInfoHelper(_ObjectBasicType.LEVEL_START)
elif gotten_group_name == 'PE_Levelende':
return _NameInfoHelper(_ObjectBasicType.LEVEL_END)
elif gotten_group_name == 'PC_Checkpoints' or gotten_group_name == 'PR_Resetpoints':
# these type's data should be gotten from its name
# use _get_name_info_from_yyc_name to get it
# _get_name_info_from_yyc_name is Ballance-compatible name standard
data = _get_name_info_from_yyc_name(obj.name, err_reporter, call_internal=True)
2022-04-05 21:16:53 +08:00
if data is None:
err_reporter.add_error("PC_Checkpoints or PR_Resetpoints detected. But couldn't get sector from name.")
2022-04-05 21:16:53 +08:00
return None
2022-04-05 15:55:42 +08:00
if data.basic_type != _ObjectBasicType.CHECKPOINT and data.basic_type != _ObjectBasicType.RESETPOINT:
# check whether it is checkpoint or resetpoint
# if not, it mean that we got error data from name
# return None instead
err_reporter.add_error("PC_Checkpoints or PR_Resetpoints detected. But name is illegal.")
2022-04-05 15:55:42 +08:00
return None
# otherwise return data
return data
else:
err_reporter.add_error("The match of Unique Component lost.")
2022-04-05 15:55:42 +08:00
return None
elif len(set_result) != 0:
# must be a weird grouping, report it
err_reporter.add_error("A Multi-grouping Unique Component.")
2022-04-05 15:55:42 +08:00
return None
# distinguish normal elements
set_result = UTILS_constants.rename_normalComponentsGroupName.intersection(group_set)
if len(set_result) == 1:
# get it
# now try get its sector
gotten_elements = (tuple(set_result))[0]
2022-04-05 20:45:00 +08:00
gotten_sector = _get_sector_from_ckgroup(group_set)
2022-04-05 15:55:42 +08:00
if gotten_sector is None:
# fail to get sector
err_reporter.add_error("Component detected. But couldn't get sector from CKGroup data.")
2022-04-05 15:55:42 +08:00
return None
data = _NameInfoHelper(_ObjectBasicType.COMPONENT)
data.component_type = gotten_elements
data.sector = int(gotten_sector)
return data
elif len(set_result) != 0:
# must be a weird grouping, report it
err_reporter.add_error("A Multi-grouping Component.")
2022-04-05 15:55:42 +08:00
return None
# distinguish road
if 'Phys_FloorRails' in group_set:
# rail
return _NameInfoHelper(_ObjectBasicType.RAIL)
elif 'Phys_Floors' in group_set:
# distinguish it between Floor and Wood
floor_result =UTILS_constants.rename_floorGroupTester.intersection(group_set)
rail_result = UTILS_constants.rename_woodGroupTester.intersection(group_set)
if len(floor_result) > 0 and len(rail_result) == 0:
return _NameInfoHelper(_ObjectBasicType.FLOOR)
elif len(floor_result) == 0 and len(rail_result) > 0:
return _NameInfoHelper(_ObjectBasicType.WOOD)
else:
err_reporter.add_warning("Can't distinguish object between Floors and Rails. Suppose it is Floors.")
2022-04-05 15:55:42 +08:00
return _NameInfoHelper(_ObjectBasicType.FLOOR)
elif 'Phys_FloorStopper' in group_set:
return _NameInfoHelper(_ObjectBasicType.STOPPER)
elif 'DepthTestCubes' in group_set:
return _NameInfoHelper(_ObjectBasicType.DEPTH_CUBE)
# no matched
err_reporter.add_error("Group match lost.")
2022-04-05 15:55:42 +08:00
return None
def _set_for_yyc_name(obj, name_info, err_reporter: _RenameErrorReporter):
2022-04-05 15:55:42 +08:00
basic_type = name_info.basic_type
2022-04-05 20:45:00 +08:00
if basic_type == _ObjectBasicType.DECORATION:
2022-04-05 15:55:42 +08:00
obj.name = "D_"
elif basic_type == _ObjectBasicType.LEVEL_START:
obj.name = "PS_FourFlames_01"
elif basic_type == _ObjectBasicType.LEVEL_END:
obj.name = "PE_Balloon_01"
elif basic_type == _ObjectBasicType.RESETPOINT:
obj.name = "PR_Resetpoint_{:0>2d}".format(name_info.sector)
elif basic_type == _ObjectBasicType.CHECKPOINT:
obj.name = "PC_TwoFlames_{:0>2d}".format(name_info.sector)
elif basic_type == _ObjectBasicType.DEPTH_CUBE:
obj.name = "DepthCubes_"
2022-04-04 22:04:11 +08:00
2022-04-05 15:55:42 +08:00
elif basic_type == _ObjectBasicType.FLOOR:
obj.name = "A_Floor_"
elif basic_type == _ObjectBasicType.WOOD:
obj.name = "A_Wood_"
elif basic_type == _ObjectBasicType.RAIL:
obj.name = "A_Rail_"
elif basic_type == _ObjectBasicType.STOPPER:
obj.name = "A_Stopper_"
elif basic_type == _ObjectBasicType.COMPONENT:
obj.name = "{}_{:0>2d}_".format(name_info.component_type, name_info.sector)
def _set_for_imengyu_name(obj, name_info, err_reporter: _RenameErrorReporter):
2022-04-05 15:55:42 +08:00
basic_type = name_info.basic_type
2022-04-05 20:45:00 +08:00
if basic_type == _ObjectBasicType.DECORATION:
2022-04-05 15:55:42 +08:00
obj.name = "O_"
elif basic_type == _ObjectBasicType.LEVEL_START:
obj.name = "PS_LevelStart"
elif basic_type == _ObjectBasicType.LEVEL_END:
obj.name = "PE_LevelEnd"
elif basic_type == _ObjectBasicType.RESETPOINT:
obj.name = "PR_ResetPoint:{:d}".format(name_info.sector)
elif basic_type == _ObjectBasicType.CHECKPOINT:
obj.name = "PC_CheckPoint:{:d}".format(name_info.sector + 1)
elif basic_type == _ObjectBasicType.DEPTH_CUBE:
obj.name = "DepthTestCubes"
elif basic_type == _ObjectBasicType.FLOOR:
obj.name = "S_Floors"
elif basic_type == _ObjectBasicType.WOOD:
obj.name = "S_FloorWoods"
elif basic_type == _ObjectBasicType.RAIL:
obj.name = "S_FloorRails"
elif basic_type == _ObjectBasicType.STOPPER:
obj.name = "S_FloorStopper"
elif basic_type == _ObjectBasicType.COMPONENT:
2022-04-05 21:16:53 +08:00
obj.name = "{}:{}:{:d}".format(name_info.component_type, obj.name.replace(":", "_"), name_info.sector)
2022-04-05 15:55:42 +08:00
2022-04-05 20:45:00 +08:00
# NOTE: the implement of this function are copied from
# BallanceVirtoolsHelper/bvh/features/mapping/grouping.cpp
def _set_for_group(obj, name_info, err_reporter: _RenameErrorReporter):
2022-04-05 20:45:00 +08:00
gps = []
basic_type = name_info.basic_type
if basic_type == _ObjectBasicType.DECORATION:
# decoration do not need grouping
pass
elif basic_type == _ObjectBasicType.LEVEL_START:
gps.append("PS_Levelstart")
elif basic_type == _ObjectBasicType.LEVEL_END:
gps.append("PE_Levelende")
elif basic_type == _ObjectBasicType.RESETPOINT:
gps.append("PC_Checkpoints")
elif basic_type == _ObjectBasicType.CHECKPOINT:
gps.append("PR_Resetpoints")
elif basic_type == _ObjectBasicType.DEPTH_CUBE:
gps.append("DepthTestCubes")
elif basic_type == _ObjectBasicType.FLOOR:
gps.append("Phys_Floors")
gps.append("Sound_HitID_01")
gps.append("Sound_RollID_01")
gps.append("Shadow")
elif basic_type == _ObjectBasicType.WOOD:
gps.append("Phys_FloorRails")
gps.append("Sound_HitID_03")
gps.append("Sound_RollID_03")
elif basic_type == _ObjectBasicType.RAIL:
gps.append("Phys_Floors")
gps.append("Sound_HitID_02")
gps.append("Sound_RollID_02")
gps.append("Shadow")
elif basic_type == _ObjectBasicType.STOPPER:
gps.append("Phys_FloorStopper")
elif basic_type == _ObjectBasicType.COMPONENT:
gps.append(name_info.component_type)
# set compabitility for 999 sector loader
if (name_info.sector == 9):
gps.append("Sector_9")
else:
gps.append("Sector_{:0>2d}".format(name_info.sector))
# apply to custom property
UTILS_virtools_prop.fill_virtools_group_data(obj, tuple(gps))
2022-04-04 22:04:11 +08:00
2022-04-05 21:16:53 +08:00
# ==========================================
# assemble funcs
def _get_data(obj, standard, err_reporter: _RenameErrorReporter):
2022-04-05 21:16:53 +08:00
if standard == _NameStandard.YYC:
return _get_name_info_from_yyc_name(obj.name, err_reporter)
2022-04-05 21:16:53 +08:00
elif standard == _NameStandard.IMENGYU:
return _get_name_info_from_imengyu_name(obj.name, err_reporter)
2022-04-05 21:16:53 +08:00
elif standard == _NameStandard.CKGROUP:
return _get_name_info_from_group(obj, err_reporter)
2022-04-05 21:16:53 +08:00
else:
raise Exception("Unknow standard")
def _set_data(obj, name_info, standard, err_reporter: _RenameErrorReporter):
2022-04-05 21:16:53 +08:00
if standard == _NameStandard.YYC:
return _set_for_yyc_name(obj, name_info, err_reporter)
2022-04-05 21:16:53 +08:00
elif standard == _NameStandard.IMENGYU:
return _set_for_imengyu_name(obj, name_info, err_reporter)
2022-04-05 21:16:53 +08:00
elif standard == _NameStandard.CKGROUP:
return _set_for_group(obj, name_info, err_reporter)
2022-04-05 21:16:53 +08:00
else:
raise Exception("Unknow standard")
def _rename_core(source_std, dest_std):
2022-04-06 14:18:45 +08:00
if source_std == dest_std:
# if source == dest
# we do not to do anything
return
# create fail counter and error reporter
2022-04-05 21:16:53 +08:00
failed_obj_counter = 0
all_obj_counter = 0
err_reporter = _RenameErrorReporter()
2022-04-05 21:16:53 +08:00
2022-04-06 14:18:45 +08:00
print('============')
print('Rename System Report')
2022-04-06 14:18:45 +08:00
print('------------')
2022-04-05 21:16:53 +08:00
for obj in _get_selected_objects():
# set counter and name
2022-04-05 21:16:53 +08:00
all_obj_counter += 1
old_name = new_name = obj.name
# get data
info = _get_data(obj, source_std, err_reporter)
# do operation according to whether getting data successfully
2022-04-05 21:16:53 +08:00
if info is None:
failed_obj_counter += 1
else:
_set_data(obj, info, dest_std, err_reporter)
# refresh obj name
new_name = obj.name
# report result
if err_reporter.can_report():
if new_name == old_name:
report_header = 'For object "{}"'.format(new_name)
else:
report_header = 'For object "{}" (Old name: "{}")'.format(new_name, old_name)
err_reporter.report(report_header)
# clear report
err_reporter.clear()
2022-04-05 21:16:53 +08:00
print('------------')
print('All / Failed - {} / {}'.format(all_obj_counter, failed_obj_counter))
2022-04-06 14:18:45 +08:00
print('============')
2022-04-05 21:16:53 +08:00
UTILS_functions.show_message_box(
('Rename system report',
'View console to get more detail',
'All: {}'.format(all_obj_counter),
'Failed: {}'.format(failed_obj_counter)),
"Info",
"INFO"
)