More Tools
- Added Apply Transforms - Added Separate by materials and loose parts.
This commit is contained in:
@@ -0,0 +1,37 @@
|
||||
import bpy
|
||||
from bpy.types import Context, Operator
|
||||
from ..core.register import register_wrap
|
||||
from ..core.common import get_selected_armature, is_valid_armature, get_all_meshes
|
||||
from ..functions.translations import t
|
||||
|
||||
@register_wrap
|
||||
class ApplyTransforms(Operator):
|
||||
bl_idname = "avatar_toolkit.apply_transforms"
|
||||
bl_label = t("Tools.apply_transforms.label")
|
||||
bl_description = t("Tools.apply_transforms.desc")
|
||||
bl_options = {'REGISTER', 'UNDO'}
|
||||
|
||||
@classmethod
|
||||
def poll(cls, context: Context) -> bool:
|
||||
return get_selected_armature(context) is not None
|
||||
|
||||
def execute(self, context: Context) -> set[str]:
|
||||
armature = get_selected_armature(context)
|
||||
if not is_valid_armature(armature):
|
||||
self.report({'ERROR'}, t("Tools.apply_transforms.invalid_armature"))
|
||||
return {'CANCELLED'}
|
||||
|
||||
bpy.ops.object.mode_set(mode='OBJECT')
|
||||
bpy.ops.object.select_all(action='DESELECT')
|
||||
|
||||
armature.select_set(True)
|
||||
context.view_layer.objects.active = armature
|
||||
|
||||
meshes = get_all_meshes(context)
|
||||
for mesh in meshes:
|
||||
mesh.select_set(True)
|
||||
|
||||
bpy.ops.object.transform_apply(location=True, rotation=True, scale=True)
|
||||
|
||||
self.report({'INFO'}, t("Tools.apply_transforms.success"))
|
||||
return {'FINISHED'}
|
||||
@@ -0,0 +1,45 @@
|
||||
import bpy
|
||||
from bpy.types import Context, Operator
|
||||
from ..core.register import register_wrap
|
||||
from ..core.common import get_selected_armature, is_valid_armature, select_current_armature
|
||||
from ..functions.translations import t
|
||||
|
||||
@register_wrap
|
||||
class SeparateByMaterials(Operator):
|
||||
bl_idname = "avatar_toolkit.separate_by_materials"
|
||||
bl_label = t("Tools.separate_by_materials.label")
|
||||
bl_description = t("Tools.separate_by_materials.desc")
|
||||
bl_options = {'REGISTER', 'UNDO'}
|
||||
|
||||
@classmethod
|
||||
def poll(cls, context: Context) -> bool:
|
||||
return context.active_object and context.active_object.type == 'MESH'
|
||||
|
||||
def execute(self, context: Context) -> set[str]:
|
||||
obj = context.active_object
|
||||
bpy.ops.object.mode_set(mode='EDIT')
|
||||
bpy.ops.mesh.select_all(action='SELECT')
|
||||
bpy.ops.mesh.separate(type='MATERIAL')
|
||||
bpy.ops.object.mode_set(mode='OBJECT')
|
||||
self.report({'INFO'}, t("Tools.separate_by_materials.success"))
|
||||
return {'FINISHED'}
|
||||
|
||||
@register_wrap
|
||||
class SeparateByLooseParts(Operator):
|
||||
bl_idname = "avatar_toolkit.separate_by_loose_parts"
|
||||
bl_label = t("Tools.separate_by_loose_parts.label")
|
||||
bl_description = t("Tools.separate_by_loose_parts.desc")
|
||||
bl_options = {'REGISTER', 'UNDO'}
|
||||
|
||||
@classmethod
|
||||
def poll(cls, context: Context) -> bool:
|
||||
return context.active_object and context.active_object.type == 'MESH'
|
||||
|
||||
def execute(self, context: Context) -> set[str]:
|
||||
obj = context.active_object
|
||||
bpy.ops.object.mode_set(mode='EDIT')
|
||||
bpy.ops.mesh.select_all(action='SELECT')
|
||||
bpy.ops.mesh.separate(type='LOOSE')
|
||||
bpy.ops.object.mode_set(mode='OBJECT')
|
||||
self.report({'INFO'}, t("Tools.separate_by_loose_parts.success"))
|
||||
return {'FINISHED'}
|
||||
@@ -108,6 +108,16 @@
|
||||
"Tools.no_armature_selected": "No armature selected",
|
||||
"Tools.select_armature": "Please select an armature",
|
||||
"Tools.tools_title.label": "Tools:",
|
||||
"Tools.separate_by_materials.label": "Separate by Materials",
|
||||
"Tools.separate_by_materials.desc": "Separate the selected mesh by materials",
|
||||
"Tools.separate_by_materials.success": "Mesh separated by materials successfully",
|
||||
"Tools.separate_by_loose_parts.label": "Separate by Loose Parts",
|
||||
"Tools.separate_by_loose_parts.desc": "Separate the selected mesh by loose parts",
|
||||
"Tools.separate_by_loose_parts.success": "Mesh separated by loose parts successfully",
|
||||
"Tools.apply_transforms.label": "Apply Transforms",
|
||||
"Tools.apply_transforms.desc": "Apply position, rotation, and scale to the armature and its meshes",
|
||||
"Tools.apply_transforms.invalid_armature": "Invalid armature selected",
|
||||
"Tools.apply_transforms.success": "Transforms applied successfully to armature and meshes",
|
||||
"VisemePanel.create_visemes": "Create Visemes",
|
||||
"VisemePanel.creating_viseme": "Creating viseme: {viseme_name}",
|
||||
"VisemePanel.creating_viseme_detail": "Creating viseme: {viseme_name}",
|
||||
|
||||
@@ -108,6 +108,16 @@
|
||||
"Tools.no_armature_selected": "アーマチュアが選択されていません",
|
||||
"Tools.select_armature": "アーマチュアを選択してください",
|
||||
"Tools.tools_title.label": "ツール:",
|
||||
"Tools.separate_by_materials.label": "マテリアルで分離",
|
||||
"Tools.separate_by_materials.desc": "選択されたメッシュをマテリアルごとに分離します",
|
||||
"Tools.separate_by_materials.success": "メッシュがマテリアルごとに正常に分離されました",
|
||||
"Tools.separate_by_loose_parts.label": "バラバラの部分で分離",
|
||||
"Tools.separate_by_loose_parts.desc": "選択されたメッシュをバラバラの部分ごとに分離します",
|
||||
"Tools.separate_by_loose_parts.success": "メッシュがバラバラの部分ごとに正常に分離されました",
|
||||
"Tools.apply_transforms.label": "トランスフォームを適用",
|
||||
"Tools.apply_transforms.desc": "アーマチュアとそのメッシュに位置、回転、スケールを適用します",
|
||||
"Tools.apply_transforms.invalid_armature": "無効なアーマチュアが選択されています",
|
||||
"Tools.apply_transforms.success": "アーマチュアとメッシュにトランスフォームが正常に適用されました",
|
||||
"VisemePanel.create_visemes": "ビセームを作成",
|
||||
"VisemePanel.creating_viseme": "ビセームを作成中:{viseme_name}",
|
||||
"VisemePanel.creating_viseme_detail": "ビセームを作成中:{viseme_name}",
|
||||
|
||||
@@ -5,6 +5,8 @@ from bpy.types import Context
|
||||
from ..functions.digitigrade_legs import CreateDigitigradeLegs
|
||||
from ..functions.translations import t
|
||||
from ..core.common import get_selected_armature
|
||||
from ..functions.seperate_by import SeparateByMaterials, SeparateByLooseParts
|
||||
from ..functions.additional_tools import ApplyTransforms
|
||||
|
||||
@register_wrap
|
||||
class AvatarToolkitToolsPanel(bpy.types.Panel):
|
||||
@@ -29,5 +31,12 @@ class AvatarToolkitToolsPanel(bpy.types.Panel):
|
||||
row.operator("avatar_toolkit.convert_to_resonite", text=t("Tools.convert_to_resonite.label"), icon='SCENE_DATA')
|
||||
row = layout.row(align=True)
|
||||
row.operator(CreateDigitigradeLegs.bl_idname, text=t("Tools.create_digitigrade_legs.label"), icon='BONE_DATA')
|
||||
layout.separator()
|
||||
row = layout.row(align=True)
|
||||
row.operator(SeparateByMaterials.bl_idname, text=t("Tools.separate_by_materials.label"), icon='MATERIAL')
|
||||
row = layout.row(align=True)
|
||||
row.operator(SeparateByLooseParts.bl_idname, text=t("Tools.separate_by_loose_parts.label"), icon='OUTLINER_OB_MESH')
|
||||
row = layout.row(align=True)
|
||||
row.operator(ApplyTransforms.bl_idname, text=t("Tools.apply_transforms.label"), icon='OBJECT_ORIGIN')
|
||||
else:
|
||||
layout.label(text=t("Tools.select_armature"), icon='ERROR')
|
||||
|
||||
Reference in New Issue
Block a user