python - 创建 Qt 显示

标签 python pyqt pyqt5 qgraphicsview qgraphicsscene

我正在制作一个项目,该项目将在字段中使用箭头,如下所示。

enter image description here

我希望对象在接收到一些移动数据(例如来自卫星的坐标)时移动。

问题是:

  • 如何创建对象箭头?
  • 我应该使用哪些库?我应该使用 QGraphicsScene 类来移动它吗?

我从来没有真正使用过这种类型的 Qt 对象。我开始使用 PyQt5 用 Python 进行编码,但由于缺乏使用这种语言的库的示例 - 我开始学习 C++ 以更好地理解 Qt5 库。

最佳答案

Qt 的优点是它允许您使用低级工具(如 QPainter)到高级工具(如 QGraphicsItem、QML 项目)创建视觉元素。在这种情况下,最简单的事情是使用基于 QGraphicsPathItem 的 QGraphicsItem 并重写 Paint 方法进行自定义绘制。对于移动的情况,您只需使用 setPos() 方法,但要获得平滑的移动,则必须使用 QVariantAnimation。

import random

from PyQt5 import QtCore, QtGui, QtWidgets


class ArrowItem(QtWidgets.QGraphicsPathItem):
    def __init__(self, parent=None):
        super().__init__(parent)
        self._length = -1
        self._points = QtCore.QPointF(), QtCore.QPointF(), QtCore.QPointF()

        self.length = 40.0

    @property
    def length(self):
        return self._length

    @length.setter
    def length(self, l):
        self._length = l

        pos_top = QtCore.QPointF(0, l * 4 / 5)
        pos_left = QtCore.QPointF(-l * 3 / 5, -l / 5)
        pos_right = QtCore.QPointF(
            l * 3 / 5,
            -l / 5,
        )

        path = QtGui.QPainterPath()
        path.moveTo(pos_top)
        path.lineTo(pos_right)
        path.lineTo(pos_left)

        self.setPath(path)

        self._points = pos_top, pos_left, pos_right

    def paint(self, painter, option, widget):

        pos_top, pos_left, pos_right = self._points

        left_color = QtGui.QColor("#cc0000")
        right_color = QtGui.QColor("#ff0000")
        bottom_color = QtGui.QColor("#661900")

        path_left = QtGui.QPainterPath()
        path_left.lineTo(pos_top)
        path_left.lineTo(pos_left)

        path_right = QtGui.QPainterPath()
        path_right.lineTo(pos_top)
        path_right.lineTo(pos_right)

        path_bottom = QtGui.QPainterPath()
        path_bottom.lineTo(pos_left)
        path_bottom.lineTo(pos_right)

        painter.setPen(QtGui.QColor("black"))
        painter.setBrush(left_color)
        painter.drawPath(path_left)
        painter.setBrush(right_color)
        painter.drawPath(path_right)
        painter.setBrush(bottom_color)
        painter.drawPath(path_bottom)

    def moveTo(self, next_position, duration=100):
        self._animation = QtCore.QVariantAnimation()
        self._animation.setStartValue(self.pos())
        self._animation.setEndValue(next_position)
        self._animation.setDuration(duration)
        self._animation.start(QtCore.QAbstractAnimation.DeleteWhenStopped)
        self._animation.valueChanged.connect(self.setPos)


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

        self.scene = QtWidgets.QGraphicsScene()
        view = QtWidgets.QGraphicsView()
        view.setRenderHints(QtGui.QPainter.Antialiasing)
        view.setScene(self.scene)
        view.scale(1, -1)
        button = QtWidgets.QPushButton("Click me")

        central_widget = QtWidgets.QWidget()
        lay = QtWidgets.QVBoxLayout(central_widget)
        lay.addWidget(view)
        lay.addWidget(button)
        self.setCentralWidget(central_widget)
        self.resize(640, 480)

        self.arrow_item = ArrowItem()
        self.scene.addItem(self.arrow_item)

        button.clicked.connect(self.handle_clicked)

    def handle_clicked(self):
        x, y = random.sample(range(300), 2)
        print("next position:", x, y)
        self.arrow_item.moveTo(QtCore.QPointF(x, y))


if __name__ == "__main__":
    app = QtWidgets.QApplication([])

    w = MainWindow()
    w.show()

    app.exec_()

注意:PyQt5(以及 PySide2)提供了许多用 python 编写的示例,这些示例是 C++ 示例的翻译,因此我建议您下载源代码( PyQt5PySide2 )并研究它们。我还有a repo以及一些示例的翻译。

关于python - 创建 Qt 显示,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/63642395/

相关文章:

java - 用于查找规范覆盖或最小数量的函数依赖项的程序

python - 没有 session 的 Tensorflow eval() 或将变量移动到另一个 session

python - 如何在 QTextEdit 中创建完全透明

python - 来自托盘图标的弹出窗口在窗口关闭时不会关闭程序

python - pyqt5 tablewidgets 检索数据

python - 如何安装 six.moves.xmlrpc_client?

python - 没有名为 'Ocr' 的模块

python - 如何使用未舍入的变量值更新 PyQt5 标签?

python - 装饰器添加了一个意想不到的参数

python - 删除PyQt5中动态创建的按钮