22

我最近才开始编程,尤其是 Python (PyQt)。我有我的主要QMainWindow课程。但我想将它与 UI 小部件分开,以便所有 Windows 的东西(菜单、工具栏、常用按钮)都在 中QMainWindow,但所有程序/UI 特定的小部件(按钮、组合框、图像、复选框等)都在单独的QWidget类中。但我不确定我这样做是否正确。

  1. 我的布局有问题 - 一些不可见的东西覆盖了菜单,因此它们不能被鼠标点击,我想我没有正确地将我的 UI 小部件添加到主窗口

这是我的做法:

class MyMainWindow(QMainWindow):
    def __init__(self, parent = None):
        super(MyMainWindow, self).__init__(parent)

        self.main_widget = QWidget(self)
        ...
        self.form_widget = FormWidget(self) 
        #This is my UI widget

        self.main_layout = QVBoxLayout(self.main_widget)
        self.main_layout.sizeConstraint = QLayout.SetDefaultConstraint
        self.main_layout.addWidget(self.form_widget.main_widget) 
        #form_widget has its own main_widget where I put all other widgets onto

        self.main_widget.setLayout(self.main_layout)
        self.setCentralWidget(self.main_widget)
  1. 我见过其他 Python 程序,其中应用程序被分解为许多小代码文件(据我了解,主类中的所有内容都是不可读或难以管理的)。

你对将代码分解成小块有什么建议?怎么做比较好?或者对于 UI,它可以都在一个大的地方?我应该把 UI 代码/类分解成单独的文件吗?

谢谢你。

[解决了]

我发现了我的错误 - 我从 UI 小部件类中删除了 main_widget(现在所有 UI 小部件都直接放在 UI 类小部件本身上)并且只这样做:

self.main_layout.addWidget(self.form_widget)

菜单不再有问题

4

3 回答 3

27

你在寻找这样的东西吗?我不太确定你main_widget是什么

from PyQt4.QtCore import *
from PyQt4.QtGui  import *

import sys

class MyMainWindow(QMainWindow):

    def __init__(self, parent=None):

        super(MyMainWindow, self).__init__(parent)
        self.form_widget = FormWidget(self) 
        self.setCentralWidget(self.form_widget) 


class FormWidget(QWidget):

    def __init__(self, parent):        
        super(FormWidget, self).__init__(parent)
        self.layout = QVBoxLayout(self)

        self.button1 = QPushButton("Button 1")
        self.layout.addWidget(self.button1)

        self.button2 = QPushButton("Button 2")
        self.layout.addWidget(self.button2)

        self.setLayout(self.layout)

app = QApplication([])
foo = MyMainWindow()
foo.show()
sys.exit(app.exec_())
于 2012-01-11T06:33:21.363 回答
11

我建议使用 Qt Designer 来创建尽可能多的 UI。

您会发现以这种方式尝试布局等要容易得多,并且它会自动将大部分与 UI 相关的内容与您的应用程序逻辑的其余部分分开。对主窗口执行此操作,也对任何对话框执行此操作,无论多么简单。

然后使用pyuic4从所有ui文件中编译python模块,并将它们全部放在自己的子包中。

我建议在编译文件-w时使用该标志。ui这将生成一个可以直接子类化的简单包装 UI 类。

所以你的主窗口最终看起来像这样:

from ui.mainwindow import MainWindowUI

class MainWindow(MainWindowUI):
    def __init__(self):
        super(MainWindow, self).__init__()
        # connect signals...
        # do other setup stuff...

请注意,Qt Designer 中添加的所有小部件现在都可以作为MainWindow实例的属性直接访问。

在开发后期之前,我不会担心将您的应用程序分解为更小的模块。结果可能没有必要——但如果确实如此,那么一旦应用程序开始变得更加复杂,如何做到这一点就会变得更加明显。

没有硬性规定——每个项目都是不同的。

于 2012-01-11T20:23:07.447 回答
10
import sys
from PyQt4 import QtCore, QtGui


class MainWindow(QtGui.QMainWindow):

    def __init__(self, parent=None):
        super(MainWindow, self).__init__(parent)
        self.form_widget = FormWidget(self)
        _widget = QtGui.QWidget()
        _layout = QtGui.QVBoxLayout(_widget)
        _layout.addWidget(self.form_widget)
        self.setCentralWidget(_widget)

class FormWidget(QtGui.QWidget):

    def __init__(self, parent):
        super(FormWidget, self).__init__(parent)
        self.__controls()
        self.__layout()

    def __controls(self):
        self.label = QtGui.QLabel("Name for backdrop")
        self.txted = QtGui.QLineEdit()
        self.lbled = QtGui.QLabel("Select a readNode")
        self.cmbox = QtGui.QComboBox()

    def __layout(self):
        self.vbox = QtGui.QVBoxLayout()
        self.hbox = QtGui.QHBoxLayout()
        self.h2Box = QtGui.QHBoxLayout()

        self.hbox.addWidget(self.label)
        self.hbox.addWidget(self.txted)

        self.h2Box.addWidget(self.lbled)
        self.h2Box.addWidget(self.cmbox)

        self.vbox.addLayout(self.hbox)
        self.vbox.addLayout(self.h2Box)
        self.setLayout(self.vbox)

def main():
    app = QtGui.QApplication(sys.argv)
    win = MainWindow()
    win.show()
    app.exec_()

if __name__ == '__main__':
    sys.exit(main()) 

正确方法!!!

于 2013-06-18T08:42:39.793 回答