479 lines
18 KiB
Python
479 lines
18 KiB
Python
#!/usr/bin/env python
|
|
# -*- coding: utf-8 -*-
|
|
|
|
import os
|
|
import sys
|
|
import maya.cmds as cmds
|
|
import maya.OpenMayaUI as omui
|
|
from shiboken2 import wrapInstance
|
|
import traceback
|
|
|
|
# 添加项目根目录到 Python 路径
|
|
ROOT_DIR = os.path.dirname(os.path.dirname(__file__))
|
|
if ROOT_DIR not in sys.path:
|
|
sys.path.insert(0, ROOT_DIR)
|
|
from config import data
|
|
from dna_utils import DNAManager
|
|
QtCore, QtGui, QtWidgets = data.Qt()
|
|
|
|
#===================================== 2. Global Variables =====================================
|
|
TOOL_NAME = data.TOOL_NAME
|
|
TOOL_VERSION = data.TOOL_VERSION
|
|
TOOL_AUTHOR = data.TOOL_AUTHOR
|
|
TOOL_LANG = data.TOOL_LANG
|
|
TOOL_WSCL_NAME = data.TOOL_WSCL_NAME
|
|
TOOL_HELP_URL = data.TOOL_HELP_URL
|
|
SCRIPTS_PATH = data.SCRIPTS_PATH
|
|
ICONS_PATH = data.ICONS_PATH
|
|
TOOL_MAIN_SCRIPT = data.TOOL_MAIN_SCRIPT
|
|
TOOL_MOD_FILENAME = data.TOOL_MOD_FILENAME
|
|
TOOL_ICON = data.TOOL_ICON
|
|
TOOL_COMMAND_ICON = data.TOOL_COMMAND_ICON
|
|
DNA_PATH = data.DNA_PATH
|
|
DNA_IMG_PATH = data.DNA_IMG_PATH
|
|
DNALIB_PATH = data.DNALIB_PATH
|
|
BUILDER_PATH = data.BUILDER_PATH
|
|
|
|
main_window = None
|
|
|
|
class MetaFusionWindow(QtWidgets.QMainWindow):
|
|
def __init__(self, parent=None):
|
|
try:
|
|
super(MetaFusionWindow, self).__init__(parent)
|
|
self.setWindowTitle("MetaFusion")
|
|
self.resize(800, 600)
|
|
|
|
# 设置窗口标志
|
|
if parent is None:
|
|
# 独立窗口模式
|
|
self.setWindowFlags(QtCore.Qt.WindowStaysOnTopHint)
|
|
else:
|
|
# 嵌入模式
|
|
self.setWindowFlags(QtCore.Qt.Widget)
|
|
|
|
# 加载样式表
|
|
self.load_stylesheet()
|
|
|
|
# 创建UI
|
|
self.setup_ui()
|
|
except Exception as e:
|
|
cmds.warning(f"窗口初始化失败: {str(e)}")
|
|
self.safe_shutdown()
|
|
|
|
def load_stylesheet(self):
|
|
"""加载QSS样式表"""
|
|
style_file = os.path.join(os.path.dirname(os.path.dirname(__file__)), "resources", "styles", "style.qss")
|
|
if os.path.exists(style_file):
|
|
with open(style_file, 'r', encoding='utf-8') as f:
|
|
self.setStyleSheet(f.read())
|
|
|
|
def setup_ui(self):
|
|
"""设置UI结构"""
|
|
# 创建中心部件
|
|
central_widget = QtWidgets.QWidget()
|
|
self.setCentralWidget(central_widget)
|
|
|
|
# 创建主布局
|
|
main_layout = QtWidgets.QVBoxLayout(central_widget)
|
|
|
|
# 创建菜单栏
|
|
self.create_menu_bar()
|
|
|
|
# 创建工具栏
|
|
self.create_tool_bar()
|
|
|
|
# 创建标签页
|
|
self.create_tabs()
|
|
|
|
def create_menu_bar(self):
|
|
"""创建菜单栏"""
|
|
menubar = self.menuBar()
|
|
|
|
# 文件菜单
|
|
file_menu = menubar.addMenu("文件")
|
|
self.add_menu_action(file_menu, "打开DNA", "open.png", self.on_open_dna)
|
|
self.add_menu_action(file_menu, "保存DNA", "save.png", self.on_save_dna)
|
|
self.add_menu_action(file_menu, "加载当前项目的DNA", "open.png", self.on_load_project_dna)
|
|
self.add_menu_action(file_menu, "修改混合目标名称", "rename.png", self.on_rename_blend_target)
|
|
self.add_menu_action(file_menu, "重置混合目标名称", "resetname.png", self.on_reset_blend_target)
|
|
self.add_menu_action(file_menu, "导出FBX", "export.png", self.on_export_fbx)
|
|
file_menu.addSeparator()
|
|
self.add_menu_action(file_menu, "退出", "exit.png", self.close)
|
|
|
|
# 编辑菜单
|
|
edit_menu = menubar.addMenu("编辑")
|
|
self.add_menu_action(edit_menu, "创建RL4节点", "connect.png", self.create_rl4_node)
|
|
self.add_menu_action(edit_menu, "删除RL4节点", "disconnect.png", self.delete_rl4_node)
|
|
self.add_menu_action(edit_menu, "镜像左至右", "mirrorL.png", self.mirror_left_to_right)
|
|
self.add_menu_action(edit_menu, "镜像右至左", "mirrorR.png", self.mirror_right_to_left)
|
|
self.add_menu_action(edit_menu, "姿势由A型转T型", "pose_A_To_T.png", self.pose_A_to_T)
|
|
self.add_menu_action(edit_menu, "姿势由T型转A型", "pose_T_To_A.png", self.pose_T_to_A)
|
|
self.add_menu_action(edit_menu, "传输LOD贴图", "locator.png", self.transfer_lod_texture)
|
|
self.add_menu_action(edit_menu, "设置关节颜色", "color.png", self.set_joint_color)
|
|
self.add_menu_action(edit_menu, "取消全部标记", "unmark_all.png", self.unmark_all)
|
|
self.add_menu_action(edit_menu, "重建所有目标", "rebuildTargets.png", self.rebuild_targets)
|
|
self.add_menu_action(edit_menu, "为所有表情设置关键帧", "bakeAnimation.png", self.bake_all_animations)
|
|
self.add_menu_action(edit_menu, "烘焙所有表情的关键帧", "centerCurrentTime.png", self.bake_all_keyframes)
|
|
|
|
# 工具菜单
|
|
tool_menu = menubar.addMenu("工具")
|
|
self.add_menu_action(tool_menu, "导出蒙皮", "export_skin.png", self.export_skin)
|
|
self.add_menu_action(tool_menu, "导入蒙皮", "import_skin.png", self.import_skin)
|
|
self.add_menu_action(tool_menu, "拷贝装皮", "copy_skin.png", self.copy_skin)
|
|
self.add_menu_action(tool_menu, "RBF变形器", "blendShape.png", self.create_blend_shape)
|
|
self.add_menu_action(tool_menu, "快速绑定服装", "clothing_weight.png", self.quick_bind_clothing)
|
|
self.add_menu_action(tool_menu, "克隆混合变形", "blendShape.png", self.clone_blend_shape)
|
|
self.add_menu_action(tool_menu, "UV传递点序", "repair_vertex_order.png", self.repair_vertex_order)
|
|
self.add_menu_action(tool_menu, "面部生成控制器", "controller.png", self.create_face_controller)
|
|
self.add_menu_action(tool_menu, "提取52BS", "ARKit52.png", self.extract_52BS)
|
|
self.add_menu_action(tool_menu, "关节轴向修复", "joint.png", self.repair_joint_axis)
|
|
self.add_menu_action(tool_menu, "生成身体控制器", "create_body_ctrl.png", self.create_body_controller)
|
|
self.add_menu_action(tool_menu, "导入面部动画", "import_face_anim.png", self.import_face_animation)
|
|
self.add_menu_action(tool_menu, "导入身体动画", "import_body_anim.png", self.import_body_animation)
|
|
|
|
# 语言菜单
|
|
lang_menu = menubar.addMenu("语言")
|
|
self.add_menu_action(lang_menu, "中文", "chinese.png", self.set_chinese)
|
|
self.add_menu_action(lang_menu, "English", "english.png", self.set_english)
|
|
|
|
# 帮助菜单
|
|
help_menu = menubar.addMenu("帮助")
|
|
self.add_menu_action(help_menu, "帮助文档", "help.png", self.show_help_document)
|
|
self.add_menu_action(help_menu, "关于", "warning.png", self.show_about_dialog)
|
|
|
|
def add_menu_action(self, menu, text, icon, callback):
|
|
"""通用菜单项添加方法"""
|
|
action = QtWidgets.QAction(QtGui.QIcon(os.path.join(ICONS_PATH, icon)), text, self)
|
|
action.triggered.connect(callback)
|
|
menu.addAction(action)
|
|
return action
|
|
|
|
def create_tool_bar(self):
|
|
"""创建工具栏"""
|
|
toolbar = self.addToolBar("主工具栏")
|
|
toolbar.setMovable(False)
|
|
|
|
# 添加工具栏按钮
|
|
toolbar.addAction(QtGui.QIcon(os.path.join(ICONS_PATH, "save.png")), "保存DNA")
|
|
toolbar.addAction(QtGui.QIcon(os.path.join(ICONS_PATH, "open.png")), "加载当前项目DNA")
|
|
|
|
def create_tabs(self):
|
|
"""创建标签页"""
|
|
self.tab_widget = QtWidgets.QTabWidget()
|
|
self.centralWidget().layout().addWidget(self.tab_widget)
|
|
|
|
# 创建四个主要标签页
|
|
self.model_tab = QtWidgets.QWidget()
|
|
self.rig_tab = QtWidgets.QWidget()
|
|
self.adjust_tab = QtWidgets.QWidget()
|
|
self.define_tab = QtWidgets.QWidget()
|
|
|
|
# 添加标签页到标签页控件
|
|
self.tab_widget.addTab(self.model_tab, "模型")
|
|
self.tab_widget.addTab(self.rig_tab, "绑定")
|
|
self.tab_widget.addTab(self.adjust_tab, "调整")
|
|
self.tab_widget.addTab(self.define_tab, "定义")
|
|
|
|
# 设置各个标签页的内容
|
|
self.setup_model_tab()
|
|
self.setup_rig_tab()
|
|
self.setup_adjust_tab()
|
|
self.setup_define_tab()
|
|
|
|
def setup_model_tab(self):
|
|
"""设置模型标签页内容"""
|
|
layout = QtWidgets.QVBoxLayout(self.model_tab)
|
|
|
|
# 创建 LOD 分栏
|
|
self.lod_tabs = QtWidgets.QTabWidget()
|
|
layout.addWidget(self.lod_tabs)
|
|
|
|
# 添加 LOD 分页
|
|
for i in range(8):
|
|
lod_tab = QtWidgets.QWidget()
|
|
lod_layout = QtWidgets.QVBoxLayout(lod_tab)
|
|
|
|
# 添加删除按钮
|
|
delete_btn = QtWidgets.QPushButton(QtGui.QIcon(os.path.join(ICONS_PATH, "delete.png")), "删除")
|
|
lod_layout.addWidget(delete_btn)
|
|
|
|
# 添加模型输入框和加载按钮
|
|
for part in ["头部", "牙齿", "牙龈", "左眼", "右眼", "虹膜", "睫毛", "眼睑", "软骨", "身体"]:
|
|
part_layout = QtWidgets.QHBoxLayout()
|
|
part_input = QtWidgets.QLineEdit()
|
|
load_btn = QtWidgets.QPushButton(QtGui.QIcon(os.path.join(ICONS_PATH, "target.png")), "加载")
|
|
part_layout.addWidget(part_input)
|
|
part_layout.addWidget(load_btn)
|
|
lod_layout.addLayout(part_layout)
|
|
|
|
self.lod_tabs.addTab(lod_tab, f"LOD{i}")
|
|
|
|
# 添加 LOD 功能按钮
|
|
lod_func_layout = QtWidgets.QHBoxLayout()
|
|
lod_func_layout.addWidget(QtWidgets.QPushButton(QtGui.QIcon(os.path.join(ICONS_PATH, "load_meshes.png")), "自定加载模型"))
|
|
lod_func_layout.addWidget(QtWidgets.QPushButton(QtGui.QIcon(os.path.join(ICONS_PATH, "standardized_naming.png")), "标准化命名"))
|
|
lod_func_layout.addWidget(QtWidgets.QPushButton(QtGui.QIcon(os.path.join(ICONS_PATH, "automatic_groupingg.png")), "自动分组"))
|
|
layout.addLayout(lod_func_layout)
|
|
|
|
# 添加模型工具
|
|
model_tool_layout = QtWidgets.QHBoxLayout()
|
|
model_tool_layout.addWidget(QtWidgets.QComboBox()) # 拓扑结构下拉菜单
|
|
model_tool_layout.addWidget(QtWidgets.QComboBox()) # 选择LOD下拉菜单
|
|
model_tool_layout.addWidget(QtWidgets.QPushButton(QtGui.QIcon(os.path.join(ICONS_PATH, "polySplitVertex.png")), "模型分离"))
|
|
model_tool_layout.addWidget(QtWidgets.QPushButton(QtGui.QIcon(os.path.join(ICONS_PATH, "supplement_meshes.png")), "生成面部配件"))
|
|
model_tool_layout.addWidget(QtWidgets.QPushButton(QtGui.QIcon(os.path.join(ICONS_PATH, "repair_normals.png")), "修复法线"))
|
|
model_tool_layout.addWidget(QtWidgets.QPushButton(QtGui.QIcon(os.path.join(ICONS_PATH, "repair_vertex_order.png")), "修复点序"))
|
|
model_tool_layout.addWidget(QtWidgets.QPushButton(QtGui.QIcon(os.path.join(ICONS_PATH, "polyChipOff.png")), "修复接缝"))
|
|
layout.addLayout(model_tool_layout)
|
|
|
|
def setup_rig_tab(self):
|
|
"""设置绑定标签页内容"""
|
|
layout = QtWidgets.QVBoxLayout(self.rig_tab)
|
|
|
|
# 添加 DNA 浏览器
|
|
self.dna_browser = QtWidgets.QListWidget()
|
|
self.dna_browser.setIconSize(QtCore.QSize(64, 64))
|
|
layout.addWidget(self.dna_browser)
|
|
|
|
# 添加 DNA 图标缩放滑块
|
|
self.dna_scale_slider = QtWidgets.QSlider(QtCore.Qt.Horizontal)
|
|
self.dna_scale_slider.setMinimum(50)
|
|
self.dna_scale_slider.setMaximum(200)
|
|
self.dna_scale_slider.setValue(100)
|
|
self.dna_scale_slider.valueChanged.connect(self.on_dna_scale_changed)
|
|
layout.addWidget(self.dna_scale_slider)
|
|
|
|
# 添加导入/导出按钮
|
|
button_layout = QtWidgets.QHBoxLayout()
|
|
import_btn = QtWidgets.QPushButton("导入设置")
|
|
export_btn = QtWidgets.QPushButton("导出设置")
|
|
import_btn.clicked.connect(self.on_import_settings)
|
|
export_btn.clicked.connect(self.on_export_settings)
|
|
button_layout.addWidget(import_btn)
|
|
button_layout.addWidget(export_btn)
|
|
layout.addLayout(button_layout)
|
|
|
|
# 初始化 DNA 浏览器
|
|
self.init_dna_browser()
|
|
|
|
def init_dna_browser(self):
|
|
"""初始化 DNA 浏览器"""
|
|
# 清空现有内容
|
|
self.dna_browser.clear()
|
|
|
|
# 获取 DNA 文件列表
|
|
dna_files = os.listdir(DNA_PATH)
|
|
img_files = os.listdir(DNA_IMG_PATH)
|
|
|
|
# 添加 DNA 项目
|
|
for dna_file in dna_files:
|
|
item = QtWidgets.QListWidgetItem(dna_file)
|
|
# 查找对应的图片
|
|
img_name = os.path.splitext(dna_file)[0] + ".png"
|
|
if img_name in img_files:
|
|
item.setIcon(QtGui.QIcon(os.path.join(data.DNA_IMG_PATH, img_name)))
|
|
self.dna_browser.addItem(item)
|
|
|
|
def on_dna_scale_changed(self, value):
|
|
"""处理 DNA 图标缩放"""
|
|
size = int(64 * (value / 100))
|
|
self.dna_browser.setIconSize(QtCore.QSize(size, size))
|
|
|
|
def on_import_settings(self):
|
|
pass
|
|
|
|
def on_export_settings(self):
|
|
"""导出设置"""
|
|
# TODO: 实现导出设置功能
|
|
pass
|
|
|
|
def setup_adjust_tab(self):
|
|
"""设置调整标签页内容"""
|
|
layout = QtWidgets.QVBoxLayout(self.adjust_tab)
|
|
|
|
# 添加BlendShape列表
|
|
blend_list = QtWidgets.QListWidget()
|
|
layout.addWidget(blend_list)
|
|
|
|
# 添加滑块组
|
|
slider_layout = QtWidgets.QVBoxLayout()
|
|
for i in range(5):
|
|
slider = QtWidgets.QSlider(QtCore.Qt.Horizontal)
|
|
slider_layout.addWidget(slider)
|
|
layout.addLayout(slider_layout)
|
|
|
|
# 添加按钮组
|
|
button_layout = QtWidgets.QHBoxLayout()
|
|
edit_btn = QtWidgets.QPushButton("编辑BlendShape")
|
|
save_btn = QtWidgets.QPushButton("保存设置")
|
|
button_layout.addWidget(edit_btn)
|
|
button_layout.addWidget(save_btn)
|
|
layout.addLayout(button_layout)
|
|
|
|
def setup_define_tab(self):
|
|
"""设置定义标签页内容"""
|
|
layout = QtWidgets.QVBoxLayout(self.define_tab)
|
|
|
|
# 添加DNA编辑区域
|
|
self.dna_edit = QtWidgets.QTextEdit()
|
|
layout.addWidget(self.dna_edit)
|
|
|
|
# 添加按钮组
|
|
button_layout = QtWidgets.QHBoxLayout()
|
|
self.load_btn = QtWidgets.QPushButton("载入DNA")
|
|
self.save_btn = QtWidgets.QPushButton("保存DNA")
|
|
self.export_btn = QtWidgets.QPushButton("导出FBX")
|
|
|
|
# 连接信号槽
|
|
self.load_btn.clicked.connect(self.on_load_dna)
|
|
self.save_btn.clicked.connect(self.on_save_dna)
|
|
self.export_btn.clicked.connect(self.on_export_fbx)
|
|
|
|
button_layout.addWidget(self.load_btn)
|
|
button_layout.addWidget(self.save_btn)
|
|
button_layout.addWidget(self.export_btn)
|
|
layout.addLayout(button_layout)
|
|
|
|
# 创建DNA管理器
|
|
self.dna_manager = DNAManager()
|
|
|
|
def on_load_dna(self):
|
|
pass
|
|
|
|
def on_save_dna(self):
|
|
pass
|
|
|
|
def on_export_fbx(self):
|
|
pass
|
|
|
|
def on_open_dna(self):
|
|
pass
|
|
|
|
def on_load_project_dna(self):
|
|
pass
|
|
|
|
def on_rename_blend_target(self):
|
|
pass
|
|
|
|
def on_reset_blend_target(self):
|
|
pass
|
|
|
|
def create_rl4_node(self):
|
|
pass
|
|
|
|
def delete_rl4_node(self):
|
|
pass
|
|
|
|
def mirror_left_to_right(self):
|
|
pass
|
|
|
|
def mirror_right_to_left(self):
|
|
pass
|
|
|
|
def pose_A_to_T(self):
|
|
pass
|
|
|
|
def pose_T_to_A(self):
|
|
pass
|
|
|
|
def transfer_lod_texture(self):
|
|
pass
|
|
|
|
def set_joint_color(self):
|
|
pass
|
|
|
|
def unmark_all(self):
|
|
pass
|
|
|
|
def rebuild_targets(self):
|
|
pass
|
|
|
|
def bake_all_animations(self):
|
|
pass
|
|
|
|
def bake_all_keyframes(self):
|
|
pass
|
|
|
|
def safe_shutdown(self):
|
|
pass
|
|
|
|
|
|
# ===================================== 显示主窗口 =====================================
|
|
def get_maya_window():
|
|
"""获取 Maya 主窗口"""
|
|
import maya.OpenMayaUI as omui
|
|
from shiboken2 import wrapInstance
|
|
|
|
maya_main_window_ptr = omui.MQtUtil.mainWindow()
|
|
return wrapInstance(int(maya_main_window_ptr), QtWidgets.QWidget)
|
|
|
|
def dock_to_maya():
|
|
"""将窗口嵌入到 Maya 的 Dock 面板"""
|
|
try:
|
|
# 先清理可能存在的旧控件
|
|
if cmds.workspaceControl("MetaFusionDock", exists=True):
|
|
cmds.deleteUI("MetaFusionDock")
|
|
|
|
# 创建新的Dock控件
|
|
dock_control = cmds.workspaceControl(
|
|
"MetaFusionDock",
|
|
label=TOOL_NAME,
|
|
tabToControl=["AttributeEditor", -1],
|
|
initialWidth=1400,
|
|
minimumWidth=1000,
|
|
minimumHeight=800,
|
|
widthProperty="free",
|
|
heightProperty="free"
|
|
)
|
|
|
|
# 获取Dock控件指针
|
|
maya_dock_ptr = omui.MQtUtil.findControl(dock_control)
|
|
if not maya_dock_ptr:
|
|
raise RuntimeError("无法获取Dock控件指针")
|
|
|
|
# 获取Maya主窗口并创建实例
|
|
maya_main_window = get_maya_window()
|
|
main_window = MetaFusionWindow(parent=maya_main_window)
|
|
|
|
# 嵌入到Dock
|
|
maya_dock_widget = wrapInstance(int(maya_dock_ptr), QtWidgets.QWidget)
|
|
maya_dock_widget.layout().addWidget(main_window)
|
|
|
|
return main_window
|
|
|
|
except Exception as e:
|
|
error_msg = f"Dock嵌入失败: {str(e)}\n{''.join(traceback.format_exc())}"
|
|
cmds.warning(error_msg)
|
|
return None
|
|
|
|
def show():
|
|
"""显示主窗口"""
|
|
global main_window
|
|
|
|
try:
|
|
# 清理旧实例
|
|
if main_window:
|
|
main_window.close()
|
|
cmds.deleteUI("MetaFusionDock")
|
|
except:
|
|
pass
|
|
|
|
# 尝试嵌入 Dock
|
|
main_window = dock_to_maya()
|
|
|
|
# 备用方案:独立窗口模式
|
|
if not main_window:
|
|
cmds.warning("Dock 模式失败,使用独立窗口模式")
|
|
main_window = MetaFusionWindow()
|
|
main_window.show()
|
|
|
|
return main_window
|
|
|
|
# ===================================== 主函数 =====================================
|
|
|
|
if __name__ == "__main__":
|
|
app = QtWidgets.QApplication([])
|
|
window = show()
|
|
app.exec_()
|
|
|