原文: http://zetcode.com/gui/pyqt5/customwidgets/

PyQt5 具有丰富的小部件集。 但是,没有工具包可以向程序员提供在其应用中可能需要的所有小部件。 工具箱通常仅提供最常见的窗口小部件,例如按钮,文本窗口小部件或滑块。 如果需要更专业的小部件,我们必须自己创建它。

使用工具箱提供的绘图工具创建自定义窗口小部件。 有两种基本的可能性:程序员可以修改或增强现有的小部件,或者可以从头开始创建自定义小部件。

刻录小部件

这是一个小部件,我们可以在 Nero,K3B 或其他 CD/DVD 刻录软件中看到。

customwidget.py

  1. #!/usr/bin/python3
  2. # -*- coding: utf-8 -*-
  3. """
  4. ZetCode PyQt5 tutorial
  5. In this example, we create a custom widget.
  6. Author: Jan Bodnar
  7. Website: zetcode.com
  8. Last edited: August 2017
  9. """
  10. from PyQt5.QtWidgets import (QWidget, QSlider, QApplication,
  11. QHBoxLayout, QVBoxLayout)
  12. from PyQt5.QtCore import QObject, Qt, pyqtSignal
  13. from PyQt5.QtGui import QPainter, QFont, QColor, QPen
  14. import sys
  15. class Communicate(QObject):
  16. updateBW = pyqtSignal(int)
  17. class BurningWidget(QWidget):
  18. def __init__(self):
  19. super().__init__()
  20. self.initUI()
  21. def initUI(self):
  22. self.setMinimumSize(1, 30)
  23. self.value = 75
  24. self.num = [75, 150, 225, 300, 375, 450, 525, 600, 675]
  25. def setValue(self, value):
  26. self.value = value
  27. def paintEvent(self, e):
  28. qp = QPainter()
  29. qp.begin(self)
  30. self.drawWidget(qp)
  31. qp.end()
  32. def drawWidget(self, qp):
  33. MAX_CAPACITY = 700
  34. OVER_CAPACITY = 750
  35. font = QFont('Serif', 7, QFont.Light)
  36. qp.setFont(font)
  37. size = self.size()
  38. w = size.width()
  39. h = size.height()
  40. step = int(round(w / 10))
  41. till = int(((w / OVER_CAPACITY) * self.value))
  42. full = int(((w / OVER_CAPACITY) * MAX_CAPACITY))
  43. if self.value >= MAX_CAPACITY:
  44. qp.setPen(QColor(255, 255, 255))
  45. qp.setBrush(QColor(255, 255, 184))
  46. qp.drawRect(0, 0, full, h)
  47. qp.setPen(QColor(255, 175, 175))
  48. qp.setBrush(QColor(255, 175, 175))
  49. qp.drawRect(full, 0, till-full, h)
  50. else:
  51. qp.setPen(QColor(255, 255, 255))
  52. qp.setBrush(QColor(255, 255, 184))
  53. qp.drawRect(0, 0, till, h)
  54. pen = QPen(QColor(20, 20, 20), 1,
  55. Qt.SolidLine)
  56. qp.setPen(pen)
  57. qp.setBrush(Qt.NoBrush)
  58. qp.drawRect(0, 0, w-1, h-1)
  59. j = 0
  60. for i in range(step, 10*step, step):
  61. qp.drawLine(i, 0, i, 5)
  62. metrics = qp.fontMetrics()
  63. fw = metrics.width(str(self.num[j]))
  64. qp.drawText(i-fw/2, h/2, str(self.num[j]))
  65. j = j + 1
  66. class Example(QWidget):
  67. def __init__(self):
  68. super().__init__()
  69. self.initUI()
  70. def initUI(self):
  71. OVER_CAPACITY = 750
  72. sld = QSlider(Qt.Horizontal, self)
  73. sld.setFocusPolicy(Qt.NoFocus)
  74. sld.setRange(1, OVER_CAPACITY)
  75. sld.setValue(75)
  76. sld.setGeometry(30, 40, 150, 30)
  77. self.c = Communicate()
  78. self.wid = BurningWidget()
  79. self.c.updateBW[int].connect(self.wid.setValue)
  80. sld.valueChanged[int].connect(self.changeValue)
  81. hbox = QHBoxLayout()
  82. hbox.addWidget(self.wid)
  83. vbox = QVBoxLayout()
  84. vbox.addStretch(1)
  85. vbox.addLayout(hbox)
  86. self.setLayout(vbox)
  87. self.setGeometry(300, 300, 390, 210)
  88. self.setWindowTitle('Burning widget')
  89. self.show()
  90. def changeValue(self, value):
  91. self.c.updateBW.emit(value)
  92. self.wid.repaint()
  93. if __name__ == '__main__':
  94. app = QApplication(sys.argv)
  95. ex = Example()
  96. sys.exit(app.exec_())

在我们的示例中,我们有一个QSlider和一个自定义小部件。 滑块控制自定义窗口小部件。 此小部件以图形方式显示了介质的总容量和可供我们使用的可用空间。 我们的自定义窗口小部件的最小值是 1,最大值是OVER_CAPACITY。 如果达到值MAX_CAPACITY,我们将开始绘制红色。 这通常表示过度燃烧。

刻录小部件位于窗口的底部。 这可以通过使用一个QHBoxLayout和一个QVBoxLayout来实现。

  1. class BurningWidget(QWidget):
  2. def __init__(self):
  3. super().__init__()

它基于QWidget小部件的刻录小部件。

  1. self.setMinimumSize(1, 30)

我们更改小部件的最小大小(高度)。 默认值对我们来说有点小。

  1. font = QFont('Serif', 7, QFont.Light)
  2. qp.setFont(font)

我们使用的字体比默认字体小。 这更适合我们的需求。

  1. size = self.size()
  2. w = size.width()
  3. h = size.height()
  4. step = int(round(w / 10))
  5. till = int(((w / OVER_CAPACITY) * self.value))
  6. full = int(((w / OVER_CAPACITY) * MAX_CAPACITY))

我们动态绘制小部件。 窗口越大,刻录的窗口小部件越大,反之亦然。 这就是为什么我们必须计算在其上绘制自定义窗口小部件的窗口小部件的大小的原因。 till参数确定要绘制的总大小。 该值来自滑块小部件。 它占整个面积的一部分。 full参数确定我们开始用红色绘制的点。

实际图纸包括三个步骤。 我们绘制黄色或红色和黄色矩形。 然后,我们绘制垂直线,将小部件分为几个部分。 最后,我们画出表示介质容量的数字。

  1. metrics = qp.fontMetrics()
  2. fw = metrics.width(str(self.num[j]))
  3. qp.drawText(i-fw/2, h/2, str(self.num[j]))

我们使用字体指标来绘制文本。 我们必须知道文本的宽度,以便使其围绕垂直线居中。

  1. def changeValue(self, value):
  2. self.c.updateBW.emit(value)
  3. self.wid.repaint()

当我们移动滑块时,将调用changeValue()方法。 在方法内部,我们发送带有参数的自定义updateBW信号。 该参数是滑块的当前值。 该值随后用于计算刻录小部件的容量。 然后将自定义窗口小部件重新粉刷。

PyQt5 中的自定义小部件 - 图1

图:刻录小部件

在 PyQt5 教程的这一部分中,我们创建了一个自定义小部件。