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

PySide 在各种小部件上都很丰富。 没有工具包可以向程序员提供在其应用中可能需要的所有小部件。 工具箱通常仅提供最常见的窗口小部件,例如按钮,文本窗口小部件,滑块等。如果需要更专业的窗口小部件,则必须自己创建。

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

刻录小部件

这是一个小部件,我们可以在 Nero,K3B 或其他 CD/DVD 刻录软件中看到。 我们完全从头开始创建小部件。 它基于最小的QtGui.QWidget小部件。

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

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

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

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

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

  1. self.setMinimumSize(1, 30)

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

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

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

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

我们动态绘制小部件。 窗口越大,刻录小部件越大。 反之亦然。 这就是为什么我们必须计算在其上绘制自定义窗口小部件的窗口小部件的大小的原因。 直到参数确定要绘制的总大小。 该值来自滑块小部件。 它占整个面积的一部分。 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信号。 该参数是滑块的当前值。 该值随后用于计算刻录小部件的容量。 然后将自定义窗口小部件重新粉刷。

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

图:刻录小部件

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