14.Compositor

本文介绍Ogre3D合成器框架的应用实例,通过像素着色器实现视口内容合成。示例中展示了如何利用CompositorListener调整像素着色器参数,实现分屏效果。代码覆盖.cpp、.material及.compositor文件。

终于到了合成器框架~对合成器我的理解就是一个针对某个ViewPort的pixelShader......嗯

下面是一个合成器的例子:

compositor Compositor5
{
	technique
	{
		texture scene target_width target_height PF_R8G8B8
		target scene
		{
			input previous
		}
		target scene
		{
			pass render_quad
			{
				material Ogre3DBeginnersGuide/Comp2
				input 0 scene
			}
		}
		target_output
		{
			input none
			pass render_quad
			{
				material Ogre3DBeginnersGuide/Comp3
				input 0 scene
			}
		}
	}
}
说到底,合成器效果主要还是看PixelShader~下面直接上最后一个例子的代码吧~用到了CompositorListener,使得可以修改PXShader中的参数。假如加上FrameListener,就可以实时控制参数了,不过要注意此时应该使用notifyMaterialRender。

最后一个例子是分屏处理~

.CPP文件

#include "Ogre\ExampleApplication.h"

class CompositorListener2:public Ogre::CompositorInstance::Listener
{
public:
	void notifyMaterialSetup(uint32 pass_id, MaterialPtr &mat)
	{
		mat ->getBestTechnique() ->getPass(pass_id) ->getFragmentProgramParameters() ->setNamedConstant("factors", Ogre::Vector3(1, 0, 0));
	}
};

class CompositorListener3:public Ogre::CompositorInstance::Listener
{
public:
	void notifyMaterialSetup(uint32 pass_id, MaterialPtr &mat)
	{
		mat ->getBestTechnique() ->getPass(pass_id) ->getFragmentProgramParameters() ->setNamedConstant("factors", Ogre::Vector3(0, 1, 0));
	}
};

class CompositorListener4:public Ogre::CompositorInstance::Listener
{
public:
	void notifyMaterialSetup(uint32 pass_id, MaterialPtr &mat)
	{
		mat ->getBestTechnique() ->getPass(pass_id) ->getFragmentProgramParameters() ->setNamedConstant("factors", Ogre::Vector3(0, 0, 1));
	}
};

class Example1:public ExampleApplication
{
public:
	Example1()
	{
		compListener = NULL;
		compListener2 = NULL;
		compListener3 = NULL;
	}

	~Example1()
	{
		delete compListener;
		delete compListener2;
		delete compListener3;
	}

	void createScene()
	{
		Ogre::SceneNode* node = mSceneMgr ->getRootSceneNode() ->createChildSceneNode();
		Ogre::Entity* ent = mSceneMgr ->createEntity("Sinbad.mesh");
		node ->attachObject(ent);

		Ogre::CompositorManager::getSingleton().addCompositor(vp, "Compositor9");
		Ogre::CompositorManager::getSingleton().setCompositorEnabled(vp, "Compositor9", true);
		Ogre::CompositorInstance* comp = Ogre::CompositorManager::getSingleton().getCompositorChain(vp) ->getCompositor("Compositor9");
		compListener = new CompositorListener2();
		comp ->addListener(compListener);

		Ogre::CompositorManager::getSingleton().addCompositor(vp2, "Compositor9");
		Ogre::CompositorManager::getSingleton().setCompositorEnabled(vp2, "Compositor9", true);
		Ogre::CompositorInstance* comp2 = Ogre::CompositorManager::getSingleton().getCompositorChain(vp2) ->getCompositor("Compositor9");
		compListener2 = new CompositorListener3();
		comp2 ->addListener(compListener2);

		Ogre::CompositorManager::getSingleton().addCompositor(vp3, "Compositor9");
		Ogre::CompositorManager::getSingleton().setCompositorEnabled(vp3, "Compositor9", true);
		Ogre::CompositorInstance* comp3 = Ogre::CompositorManager::getSingleton().getCompositorChain(vp3) ->getCompositor("Compositor9");
		compListener3 = new CompositorListener4();
		comp3 ->addListener(compListener3);
	}

	void createCamera()
	{
		mCamera = mSceneMgr ->createCamera("MyCamera1");
		mCamera ->setPosition(0, 10, 20);
		mCamera ->lookAt(0, 0, 0);
		mCamera ->setNearClipDistance(5);
	}

	void createViewports()
	{
		vp = mWindow ->addViewport(mCamera, 0, 0.0, 0.0, 0.5, 0.5);
		vp ->setBackgroundColour(ColourValue(0.0f, 0.0f, 0.0f));
		
		vp2 = mWindow ->addViewport(mCamera, 1, 0.5, 0.0, 0.5, 0.5);
		vp2 ->setBackgroundColour(ColourValue(0.0f, 0.0f, 0.0f));

		vp3 = mWindow ->addViewport(mCamera, 2, 0.0, 0.5, 0.5, 0.5);
		vp3 ->setBackgroundColour(ColourValue(0.0f, 0.0f, 0.0f));

		vp4 = mWindow ->addViewport(mCamera, 3, 0.5, 0.5, 0.5, 0.5);
		vp4 ->setBackgroundColour(ColourValue(0.0f, 0.0f, 0.0f));

		mCamera ->setAspectRatio(Real(vp ->getActualWidth()) / Real(vp ->getActualHeight()));
	}

private:
	Ogre::Camera* mCamera2;
	Ogre::Viewport* vp;
	Ogre::Viewport* vp2;
	Ogre::Viewport* vp3;
	Ogre::Viewport* vp4;

	CompositorListener2* compListener;
	CompositorListener3* compListener2;
	CompositorListener4* compListener3;
};

int main(void)
{
	Example1 app;
	app.go();
}
.material文件

fragment_program MyFragmentShader10 cg
{
	source Ogre3DBeginnersGuideShaders.cg
	entry_point MyFragmentShader10
	profiles ps_2_0 arbfp1

	default_params
	{
		param_named factors float4 1 1 1 0
	}
}

material Ogre3DBeginnersGuide/Comp7
{
	technique
	{
		pass
		{
			texture_unit
			{
			}
			fragment_program_ref MyFragmentShader10
			{
			}
		}
	}
}
.compositor

compositor Compositor9
{
	technique
	{
		texture scene target_width target_height PF_R8G8B8
		target scene
		{
			input previous
		}
		target_output
		{
			input none
			pass render_quad
			{
				material Ogre3DBeginnersGuide/Comp7
				input 0 scene
			}
		}
	}
}
.cg

void MyFragmentShader10(float2 uv : TEXCOORD0,
			out float4 color : COLOR,
			uniform sampler2D texture,
			uniform float4 factors)
{
	color = tex2D(texture, uv);
	color *= factors;
}
#主窗口 - PyQt6 版本 import os import sys import logging import faulthandler # 用于捕获崩溃信息 import ctypes # Windows API 调用支持 from typing import Optional from PyQt6.QtCore import Qt, pyqtSignal, PYQT_VERSION_STR from PyQt6.QtGui import QAction from PyQt6.QtWidgets import QDockWidget, QWidget, QVBoxLayout, QLabel, QMainWindow, QStatusBar, QPushButton, \ QHBoxLayout, QTabWidget, QComboBox, QToolBar, QApplication, QMessageBox, QFileDialog from ui.template_editor import TemplateCanvas # 启用故障处理程序,帮助调试崩溃问题 faulthandler.enable() # 设置日志系统,记录运行时信息 logging.basicConfig( level=logging.DEBUG, format='%(asctime)s - %(name)s - %(levelname)s - %(message)s', handlers=[ logging.FileHandler("app_debug.log"), # 输出到文件 logging.StreamHandler() # 输出到控制台 ] ) logger = logging.getLogger(__name__) logger.info("应用程序启动") logger.info(f"Python 路径: {sys.executable}") logger.info(f"参数: {sys.argv}") logger.info(f"系统路径: {sys.path}") logger.info(f"工作目录: {os.getcwd()}") logger.info(f"PyQt6 版本: {'未导入' if 'PYQT_VERSION_STR' not in globals() else PYQT_VERSION_STR}") # 检查是否运行在虚拟环境中 if not hasattr(sys, 'real_prefix') and not hasattr(sys, 'base_prefix'): logging.warning("警告: 未在虚拟环境中运行,建议使用虚拟环境") # 设置调试标志和环境变量 os.environ['QT_DEBUG_PLUGINS'] = '1' os.environ['QTWEBENGINE_CHROMIUM_FLAGS'] = '--disable-gpu --no-sandbox' # Windows 错误报告设置 if sys.platform == 'win32': ctypes.windll.kernel32.SetErrorMode.argtypes = [ctypes.c_uint] ctypes.windll.kernel32.SetErrorMode(ctypes.c_uint(0x0001 | 0x0002)) # 设置堆栈保护 os.environ['QT_FATAL_CRITICALS'] = '1' os.environ['QT_FATAL_WARNINGS'] = '1' class BasePropertiesPanel(QWidget): """属性面板基类,定义通用信号与接口""" propertyChanged = pyqtSignal(str, object) # 属性变化信号 def __init__(self, parent=None): super().__init__(parent) self.region = None self.setup_ui() def setup_ui(self): pass def set_region(self, region): """绑定当前编辑区域对象""" self.region = region self.update_ui_from_region() def update_ui_from_region(self): """根据区域数据更新 UI""" pass def update_region_from_ui(self): """根据 UI 数据更新区域属性""" pass class TextPropertiesPanel(BasePropertiesPanel): """文本属性面板 UI 实现""" def setup_ui(self): layout = QVBoxLayout(self) layout.addWidget(QLabel("文本属性面板")) self.setLayout(layout) def update_ui_from_region(self): if self.region: pass class QRPropertiesPanel(BasePropertiesPanel): """二维码属性面板 UI 实现""" def setup_ui(self): layout = QVBoxLayout(self) layout.addWidget(QLabel("二维码属性面板")) self.setLayout(layout) class LogoPropertiesPanel(BasePropertiesPanel): """Logo属性面板 UI 实现""" def setup_ui(self): layout = QVBoxLayout(self) layout.addWidget(QLabel("Logo属性面板")) self.setLayout(layout) class MainWindow(QMainWindow): def __init__(self, data_path: Optional[str] = None): super().__init__() self.logger = logging.getLogger(__name__) self.logger.info("主窗口初始化开始") self.setWindowTitle("自动打印系统") self.resize(800, 600) self.data_path = data_path or os.path.join(os.path.dirname(__file__), "../../../data") # 初始化所有实例属性为 None self.left_layout = None self.btn_upload = None self.btn_add_text = None self.btn_add_qr = None self.template_canvas = None self.props_tabs = None self.text_props_panel = None self.qr_props_panel = None self.logo_props_panel = None self.right_layout = None self.btn_import_data = None self.btn_save_config = None self.data_mapper = None self.field_mapping_widget = None self.history_combo = None self.btn_history = None self.btn_generate = None self.btn_rerun = None self.project_manager = None self.font_manager = None self.compositor = None # 初始化控件 self.init_components() try: # 创建项目管理器 from src.auto_print_system.core.project_manager import ProjectManager self.project_manager = ProjectManager(self.data_path) # 创建字体管理器 from src.auto_print_system.core.font_manager import FontManager self.font_manager = FontManager(self.data_path) # 创建合成引擎 from src.auto_print_system.core.file_compositor import FileCompositor self.compositor = FileCompositor(self.data_path, self.font_manager) except ImportError as e: self.logger.error(f"初始化核心组件失败: {str(e)}", exc_info=True) QMessageBox.critical(self, "错误", f"初始化核心组件失败:\n{str(e)}") # 设置状态栏 self.status_bar = QStatusBar() self.setStatusBar(self.status_bar) self.status_bar.showMessage("就绪") self.current_project = None self.logo_path = None self.spot_color_path = None # 加载样式 self.apply_styles() def init_components(self) -> None: """初始化所有界面组件""" self.logger.info("开始初始化组件") central_widget = QWidget() self.setCentralWidget(central_widget) main_layout = QVBoxLayout(central_widget) # 左侧模板编辑区 left_dock = QDockWidget("模板编辑区", self) left_widget = QWidget() self.left_layout = QVBoxLayout(left_widget) btn_layout = QHBoxLayout() self.btn_upload = QPushButton("上传模板") self.btn_add_text = QPushButton("添加文本框") self.btn_add_qr = QPushButton("添加二维码区") self.btn_upload.clicked.connect(self.upload_template) self.btn_add_text.clicked.connect(self.add_text_region) self.btn_add_qr.clicked.connect(self.add_qr_region) btn_layout.addWidget(self.btn_upload) btn_layout.addWidget(self.btn_add_text) btn_layout.addWidget(self.btn_add_qr) self.left_layout.addLayout(btn_layout) try: from src.auto_print_system.ui.template_editor import TemplateCanvas self.template_canvas = TemplateCanvas(data_path=self.data_path) self.template_canvas.setAlignment(Qt.AlignmentFlag.AlignCenter) self.template_canvas.setStyleSheet("background-color: #f0f0f0; border: 1px solid #ccc;") self.left_layout.addWidget(self.template_canvas) except ImportError as e: self.logger.error(f"初始化模板画布失败: {str(e)}", exc_info=True) self.template_canvas = None # 属性面板 self.props_tabs = QTabWidget() self.text_props_panel = TextPropertiesPanel() self.qr_props_panel = QRPropertiesPanel() self.logo_props_panel = LogoPropertiesPanel() self.props_tabs.addTab(self.text_props_panel, "文本属性") self.props_tabs.addTab(self.qr_props_panel, "二维码属性") self.props_tabs.addTab(self.logo_props_panel, "Logo属性") self.left_layout.addWidget(self.props_tabs) left_dock.setWidget(left_widget) self.addDockWidget(Qt.DockWidgetArea.LeftDockWidgetArea, left_dock) # 右侧数据映射区 right_dock = QDockWidget("数据映射区", self) right_widget = QWidget() self.right_layout = QVBoxLayout(right_widget) data_btn_layout = QHBoxLayout() self.btn_import_data = QPushButton("导入数据") self.btn_save_config = QPushButton("保存配置") self.btn_import_data.clicked.connect(self.import_data) self.btn_save_config.clicked.connect(self.save_config) data_btn_layout.addWidget(self.btn_import_data) data_btn_layout.addWidget(self.btn_save_config) self.right_layout.addLayout(data_btn_layout) try: from src.auto_print_system.ui.data_mapper import DataMapperWidget self.data_mapper = DataMapperWidget(self) self.right_layout.addWidget(self.data_mapper, 1) except ImportError as e: self.logger.error(f"初始化数据映射器失败: {str(e)}", exc_info=True) self.field_mapping_widget = QLabel("拖拽字段到模板区域进行映射") self.field_mapping_widget.setStyleSheet("min-height: 100px; background-color: #fff; border: 1px dashed #999;") self.right_layout.addWidget(self.field_mapping_widget) right_dock.setWidget(right_widget) self.addDockWidget(Qt.DockWidgetArea.RightDockWidgetArea, right_dock) # 底部操作区 bottom_widget = QWidget() bottom_layout = QHBoxLayout(bottom_widget) self.history_combo = QComboBox() self.btn_history = QPushButton("管理") self.btn_generate = QPushButton("开始合成") self.btn_rerun = QPushButton("执行翻单") self.btn_rerun.setStyleSheet("background-color: #2196F3; color: white; font-weight: bold; padding: 8px 16px;") self.btn_rerun.clicked.connect(self.rerun_project) self.btn_rerun.setEnabled(False) # 设置按钮初始状态为不可点击 bottom_layout.addWidget(QLabel("翻单历史:")) bottom_layout.addWidget(self.history_combo) bottom_layout.addWidget(self.btn_history) bottom_layout.addSpacing(20) bottom_layout.addWidget(self.btn_generate) bottom_layout.addWidget(self.btn_rerun) main_layout.addWidget(bottom_widget) # 翻单按钮状态初始化 self.create_menus() self.create_toolbar() self.update_rerun_button_state() # ===== 功能槽函数 ===== def update_rerun_button_state(self) -> None: """根据项目状态更新翻单按钮的可用性""" # 检查是否满足翻单条件: # 1. 当前有有效项目 # 2. 模板已加载 # 3. 数据映射已完成 can_rerun = ( self.current_project is not None and self.template_canvas is not None and self.template_canvas.template_item is not None and self.data_mapper is not None and self.data_mapper.is_mapping_complete() ) if self.btn_rerun: self.btn_rerun.setEnabled(can_rerun) else: self.logger.warning("翻单按钮未初始化,无法更新状态") def rerun_project(self) -> None: """执行翻单操作""" if self.btn_rerun is not None: self.btn_rerun.setEnabled(False) # 防止多次点击 else: logging.error("btn_rerun 控件未正确初始化") return # ... 其他业务逻辑 ... def create_menus(self) -> None: """创建菜单栏""" menubar = self.menuBar() file_menu = menubar.addMenu('文件') new_action = QAction('新建项目', self) open_action = QAction('打开项目', self) save_action = QAction('保存项目', self) exit_action = QAction('退出', self) exit_action.triggered.connect(self.close) file_menu.addAction(new_action) file_menu.addAction(open_action) file_menu.addAction(save_action) file_menu.addSeparator() file_menu.addAction(exit_action) edit_menu = menubar.addMenu('编辑') edit_menu.addAction('撤销') edit_menu.addAction('重做') def create_toolbar(self) -> None: """创建工具栏""" toolbar = QToolBar("主工具栏") self.addToolBar(toolbar) toolbar.addAction("打开模板", self.upload_template) toolbar.addAction("导入数据", self.import_data) toolbar.addSeparator() toolbar.addAction("添加文本", self.add_text_region) toolbar.addAction("添加二维码", self.add_qr_region) toolbar.addSeparator() toolbar.addAction("开始合成", self.generate_output) def apply_styles(self) -> None: """应用基本样式表""" self.setStyleSheet(""" QMainWindow { background-color: #f5f5f5; } QPushButton { padding: 5px 10px; border: 1px solid #ccc; border-radius: 4px; } QPushButton:hover { background-color: #e9e9e9; } """) def upload_template(self) -> None: """上传模板文件""" self.logger.debug("上传模板操作被触发") file_dialog = QFileDialog(self) file_dialog.setNameFilter("模板文件 (*.pdf *.png *.jpg *.jpeg)") file_dialog.setWindowTitle("选择模板文件") if file_dialog.exec(): template_path = file_dialog.selectedFiles()[0] self.logger.info(f"选择了模板文件: {template_path}") try: if self.template_canvas: success = self.template_canvas.load_template(template_path) if success: self.status_bar.showMessage(f"模板加载成功: {os.path.basename(template_path)}") else: self.status_bar.showMessage("模板加载失败,请检查文件格式") self.logger.error(f"模板加载失败: {template_path}") else: self.logger.error("模板画布未初始化") self.status_bar.showMessage("模板画布未初始化,无法加载模板") except Exception as e: self.logger.error(f"模板加载失败: {str(e)}", exc_info=True) self.status_bar.showMessage("模板加载失败,请检查文件格式或权限") def add_text_region(self) -> None: """添加文本区域""" self.logger.debug("添加文本区域操作被触发") if self.template_canvas: try: self.template_canvas.set_mode(TemplateCanvas.MODE_TEXT) self.status_bar.showMessage("已切换到【添加文本区域】模式") except Exception as e: self.logger.error(f"设置文本区域模式失败: {str(e)}", exc_info=True) self.status_bar.showMessage("设置模式失败,请查看日志") else: self.logger.error("模板画布未初始化") self.status_bar.showMessage("模板画布未初始化,无法添加文本区域") def add_qr_region(self) -> None: """添加二维码区域""" self.logger.debug("添加二维码区域操作被触发") if self.template_canvas: try: self.template_canvas.set_mode(TemplateCanvas.MODE_QR) self.status_bar.showMessage("添加二维码区域模式激活") except Exception as e: self.logger.error(f"设置二维码区域模式失败: {str(e)}", exc_info=True) self.status_bar.showMessage("设置二维码区域模式失败,请重试") else: self.logger.error("模板画布未初始化") self.status_bar.showMessage("模板画布未初始化,无法添加二维码区域") def import_data(self) -> None: """导入数据文件""" self.logger.debug("导入数据操作被触发") file_dialog = QFileDialog(self) file_dialog.setNameFilter("数据文件 (*.csv *.xlsx *.xls)") file_dialog.setWindowTitle("选择数据文件") if file_dialog.exec(): data_path = file_dialog.selectedFiles()[0] self.logger.info(f"选择了数据文件: {data_path}") try: if self.data_mapper: self.data_mapper.import_data() self.status_bar.showMessage(f"数据文件已导入: {os.path.basename(data_path)}") else: self.logger.error("数据映射器未初始化") self.status_bar.showMessage("数据映射器未初始化,无法导入数据文件") except Exception as e: self.logger.error(f"导入数据失败: {str(e)}", exc_info=True) self.status_bar.showMessage("导入数据失败,请检查文件格式或权限") if self.data_mapper: self.data_mapper.import_data() self.update_rerun_button_state() #更新按钮状态 def save_config(self) -> None: """保存配置文件""" self.logger.debug("保存配置操作被触发") try: if self.data_mapper: mapping_config = self.data_mapper.get_mapping() if mapping_config: # 假设配置文件保存到 JSON 文件 config_path = os.path.join(self.data_path, "config.json") with open(config_path, "w", encoding="utf-8") as config_file: import json json.dump(mapping_config, config_file, ensure_ascii=False, indent=4) self.logger.info(f"配置文件已保存至: {config_path}") self.status_bar.showMessage("配置文件保存成功") else: self.logger.warning("无映射配置可保存") self.status_bar.showMessage("无映射配置可保存") else: self.logger.error("数据映射器未初始化") self.status_bar.showMessage("数据映射器未初始化,无法保存配置") except Exception as e: self.logger.error(f"保存配置失败: {str(e)}", exc_info=True) self.status_bar.showMessage("保存配置失败,请检查文件权限") def generate_output(self) -> None: """生成输出文件""" self.logger.debug("开始合成操作被触发") if not self.current_project: self.logger.error("未选择项目,无法生成输出文件") self.status_bar.showMessage("未选择项目,无法生成输出文件") return try: # 检查是否有有效的模板和数据映射 if not self.template_canvas or not self.template_canvas.template_item: self.logger.error("无有效模板,无法生成输出文件") self.status_bar.showMessage("无有效模板,无法生成输出文件") return if not self.data_mapper or not self.data_mapper.data_frame: self.logger.error("无有效数据,无法生成输出文件") self.status_bar.showMessage("无有效数据,无法生成输出文件") return # 开始合成操作 self.status_bar.showMessage("正在生成输出文件...") self.logger.info("开始合成输出文件") output_path = os.path.join(self.data_path, "output.pdf") self.compositor.compose(output_path) self.status_bar.showMessage("输出文件生成成功") self.logger.info(f"输出文件已生成至: {output_path}") except Exception as e: self.logger.error(f"生成输出文件失败: {str(e)}", exc_info=True) self.status_bar.showMessage("生成输出文件失败,请检查设置") def rerun_project(self) -> None: """执行翻单操作""" self.logger.debug("执行翻单操作被触发") if self.btn_rerun is not None: self.btn_rerun.setEnabled(False) # 防止多次点击 else: self.logger.error("btn_rerun 控件未正确初始化") return if not self.current_project: self.logger.error("未选择项目,无法执行翻单") self.status_bar.showMessage("未选择项目,无法执行翻单") return try: # 检查是否有有效的模板和数据映射 if not self.template_canvas or not self.template_canvas.template_item: self.logger.error("无有效模板,无法执行翻单") self.status_bar.showMessage("无有效模板,无法执行翻单") return if not self.data_mapper or not self.data_mapper.data_frame: self.logger.error("无有效数据,无法执行翻单") self.status_bar.showMessage("无有效数据,无法执行翻单") return # 执行翻单操作 self.status_bar.showMessage("正在执行翻单...") self.logger.info("开始执行翻单") output_path = os.path.join(self.data_path, "rerun_output.pdf") self.compositor.compose(output_path) self.status_bar.showMessage("翻单执行成功") self.logger.info(f"翻单输出文件已生成至: {output_path}") # 恢复按钮状态 self.btn_rerun.setEnabled(True) except Exception as e: self.logger.error(f"翻单执行失败: {str(e)}", exc_info=True) self.status_bar.showMessage("翻单执行失败,请检查设置") # 恢复按钮状态 self.btn_rerun.setEnabled(True) # 测试运行 if __name__ == "__main__": app = QApplication(sys.argv) window = MainWindow() window.show() sys.exit(app.exec()) 运行后的结果是:D:\PycharmProjects\Panelsystem\.venv\Scripts\python.exe D:\PycharmProjects\Panelsystem\src\auto_print_system\ui\main_window.py 2025-07-14 17:36:56,007 - __main__ - INFO - 应用程序启动 2025-07-14 17:36:56,007 - __main__ - INFO - Python 路径: D:\PycharmProjects\Panelsystem\.venv\Scripts\python.exe 2025-07-14 17:36:56,008 - __main__ - INFO - 参数: ['D:\\PycharmProjects\\Panelsystem\\src\\auto_print_system\\ui\\main_window.py'] 2025-07-14 17:36:56,008 - __main__ - INFO - 系统路径: ['D:\\PycharmProjects\\Panelsystem\\src\\auto_print_system\\ui', 'D:\\PycharmProjects\\Panelsystem', 'D:\\PycharmProjects\\Panelsystem\\src', 'D:\\PycharmProjects\\Panelsystem\\src\\auto_print_system', 'C:\\Users\\rr417341\\AppData\\Local\\Programs\\Python\\Python311\\python311.zip', 'C:\\Users\\rr417341\\AppData\\Local\\Programs\\Python\\Python311\\DLLs', 'C:\\Users\\rr417341\\AppData\\Local\\Programs\\Python\\Python311\\Lib', 'D:\\PycharmProjects\\Panelsystem\\.venv\\Scripts', 'D:\\PycharmProjects\\Panelsystem\\.venv', 'D:\\PycharmProjects\\Panelsystem\\.venv\\Lib\\site-packages'] 2025-07-14 17:36:56,008 - __main__ - INFO - 工作目录: D:\PycharmProjects\Panelsystem\src\auto_print_system\ui 2025-07-14 17:36:56,008 - __main__ - INFO - PyQt6 版本: 6.4.2 D:\PycharmProjects\Panelsystem\src\auto_print_system\ui\main_window.py:111: DeprecationWarning: sipPyTypeDict() is deprecated, the extension module should use sipPyTypeDictRef() instead super().__init__() 2025-07-14 17:36:56,089 - __main__ - INFO - 主窗口初始化开始 2025-07-14 17:36:56,090 - __main__ - INFO - 开始初始化组件 D:\PycharmProjects\Panelsystem\src\auto_print_system\ui\main_window.py:60: DeprecationWarning: sipPyTypeDict() is deprecated, the extension module should use sipPyTypeDictRef() instead super().__init__(parent) Traceback (most recent call last): File "D:\PycharmProjects\Panelsystem\src\auto_print_system\ui\main_window.py", line 557, in <module> window = MainWindow() ^^^^^^^^^^^^ File "D:\PycharmProjects\Panelsystem\src\auto_print_system\ui\main_window.py", line 145, in __init__ self.init_components() File "D:\PycharmProjects\Panelsystem\src\auto_print_system\ui\main_window.py", line 283, in init_components self.update_rerun_button_state() File "D:\PycharmProjects\Panelsystem\src\auto_print_system\ui\main_window.py", line 293, in update_rerun_button_state self.current_project is not None and ^^^^^^^^^^^^^^^^^^^^ AttributeError: 'MainWindow' object has no attribute 'current_project'. Did you mean: 'rerun_project'?
07-15
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值