PyQt 如何創建自定義QWidget

開發環境

Win7  PyCharm  Python3.5.1  PyQt5

主要文件:

|-- main.py
|-- res
| `-- fish.jpg
`-- ui
 `-- app_widget.py

main.py

import sys

from PyQt5.QtWidgets import QApplication

from ui.app_widget import AppQWidget

if __name__ == '__main__':
 app = QApplication(sys.argv)
 w = AppQWidget()
 w.show()

 sys.exit(app.exec_())

app_main_window.py

自定義瞭一個居中顯示的窗口,關閉時彈確認框

from PyQt5.QtCore import QCoreApplication
from PyQt5.QtGui import QIcon
from PyQt5.QtWidgets import QWidget, QPushButton, QDesktopWidget, QMessageBox


class AppQWidget(QWidget):
 """
 A custom QWidget by Rust Fisher
 """

 def __init__(self):
  super().__init__()
  self.init_ui()

 def init_ui(self):
  # self.setGeometry(300, 300, 400, 200) # 相當於move和resize
  self.resize(300, 200)
  self.move_to_center()
  self.setWindowTitle('Demo1')
  self.setWindowIcon(QIcon('res/fish.jpg'))

  btn1 = QPushButton('Quit', self)
  btn1.setToolTip('Click to quit')
  btn1.resize(btn1.sizeHint())
  btn1.move(200, 150)
  btn1.clicked.connect(QCoreApplication.instance().quit) # cannot locate function connect

 def closeEvent(self, event):
  reply = QMessageBox.question(self, 'Message',
          'Are you sure to quit now?',
          QMessageBox.Yes | QMessageBox.No,
          QMessageBox.No)
  if reply == QMessageBox.Yes:
   event.accept()
  else:
   event.ignore()

 def move_to_center(self):
  qr = self.frameGeometry()
  cp = QDesktopWidget().availableGeometry().center() # got center info here
  qr.moveCenter(cp)
  self.move(qr.topLeft()) # 應用窗口的左上方的點到qr矩形的左上方的點,因此居中顯示在我們的屏幕上

Tips

多控件可以存在list中

存在一起,需要對整體操作時直接遍歷列表

 # 同組的控件可以存在同一個list中
 self.cb_list = [
  self.ma.i2cCB,
  self.ma.mipiCB,
  self.ma.eepromCB,
  self.ma.tem_sensorCB,
  self.ma.lensCB,
  self.ma.vcmCB,
  self.ma.mirrorCB,
  self.ma.mirrorCaliCB, ]

 self.test_count_et_list = [
  self.ma.i2cCountEt,
  self.ma.mipiCountEt,
  self.ma.eepromCountEt,
  self.ma.tem_sensorCountEt,
  self.ma.lensCountEt,
  self.ma.vcmCountEt,
  self.ma.mirrorCountEt,
  self.ma.mirrorCaliCountEt,
 ]

# 需要操作某組控件時 直接遍歷列表
def _click_test_item_cb(self):
 """ Update [choose all checkbox] by all test item state """
 choose_all = True
 for cb in self.cb_list:
  choose_all = choose_all & cb.isChecked()
 self.ma.selecteAllCB.setChecked(choose_all)

QApplication與QWidget

QApplication是一個單例,在QWidget中可以通過QApplication.instance()獲取到對象

實際上在實例化QApplication前就使用QtGui.QWidget()是會報錯的

>>> QtGui.QWidget()
QWidget: Must construct a QApplication before a QPaintDevice

參考 How QApplication() and QWidget() objects are connected in PySide/PyQt?

在我們自定義的QMainWindow中,也可以直接獲取到QApplication的實例。

class RustMainWindow(QMainWindow):
 """ This is the main class """

 def _trigger_english(self):
  print "Change to English", QApplication.instance()

# Change to English <PyQt4.QtGui.QApplication object at 0x02ABE3A0>

註意widget持有外部對象引用的問題

如果在程序啟動的地方將引用交給widget,退出時會造成應用無法關閉的問題(類似內存泄漏)。

if __name__ == '__main__':
 app = QApplication(sys.argv)
 # 這裡把app交給瞭MainWindow,MainWindow關閉時是無法正常退出應用的
 main_d = RustMainWindow(app) # 不建議這麼做
 main_d.show()
 sys.exit(app.exec_())

以上就是PyQt 如何創建自定義QWidget的詳細內容,更多關於PyQt 創建自定義QWidget的資料請關註WalkonNet其它相關文章!

推薦閱讀: