簡體   English   中英

如何在 Qt 中使用 QGraphicsItem 中的 itemChange

[英]How to use itemChange from QGraphicsItem in Qt

我有自定義橢圓QGraphicsItem類和自定義線類。 在現場,我假設兩個橢圓和它們之間由一條線連接。 橢圓有一個指向這條線的指針並且是可移動的。 我的問題是我不知道如何使用QGraphicsItem itemChange() 我想建立隨着橢圓運動而變化的連接。 因此,我想使用itemChange()方法以始終位於橢圓中心的方式更改線坐標。 我從QGraphicsItem::itemChange()閱讀文檔,但我不知道如何在我的情況下使用它。

正如其他人已經指出的那樣,您需要覆蓋(重新實現)類中的方法。

下面是一個完整的工作示例,演示了這一點:

#include "Dialog.h"
#include <QApplication>
#include <QGraphicsScene>
#include <QGraphicsView>
#include <QGraphicsEllipseItem>
#include <QGraphicsLineItem>


class CustomElipse : public QGraphicsEllipseItem
{
public:
    CustomElipse (const QRectF& rect) : QGraphicsEllipseItem(rect) {
        setFlag(QGraphicsItem::ItemIsMovable);
        setFlag(QGraphicsItem::ItemSendsScenePositionChanges);
    }

    void addLine(QGraphicsLineItem *line, bool isPoint1) {
        this->line = line;
        isP1 = isPoint1;
    }

    QVariant itemChange(GraphicsItemChange change, const QVariant &value)
    {
        if (change == ItemPositionChange && scene()) {
            // value is the new position.
            QPointF newPos = value.toPointF();

            moveLineToCenter(newPos);
        }
        return QGraphicsItem::itemChange(change, value);
    }

    void moveLineToCenter(QPointF newPos) {
        // Converts the elipse position (top-left)
        // to its center position
        int xOffset = rect().x() + rect().width()/2;
        int yOffset = rect().y() + rect().height()/2;

        QPointF newCenterPos = QPointF(newPos.x() + xOffset, newPos.y() + yOffset);

        // Move the required point of the line to the center of the elipse
        QPointF p1 = isP1 ? newCenterPos : line->line().p1();
        QPointF p2 = isP1 ? line->line().p2() : newCenterPos;

        line->setLine(QLineF(p1, p2));
    }

private:
    QGraphicsLineItem *line;
    bool isP1;
};

int main(int argc, char *argv[])
{
    QApplication a(argc, argv);

    QGraphicsScene scene;

    CustomElipse *elipse1 = new CustomElipse(QRectF(30, 30, 15, 25));
    scene.addItem(elipse1);

    CustomElipse *elipse2 = new CustomElipse(QRectF(70, 70, 25, 15));
    scene.addItem(elipse2);

    QGraphicsLineItem *line = scene.addLine(QLineF(40, 40, 80, 80));

    elipse1->addLine(line, true);
    elipse2->addLine(line, false);

    QGraphicsView view(&scene);
    view.show();

    return a.exec();
}

上面的代碼繪制了兩個可移動的橢圓,並在它們之間畫了一條線。 線條位置會根據橢圓進行調整。

運行時,你會得到這樣的東西: 省略號和線的預覽

對於任何尋找 python 代碼的人來說,這里是 @AntonyG 代碼的翻譯:

import sys
from PyQt5 import QtWidgets, QtCore


class CustomItem(QtWidgets.QGraphicsEllipseItem):
    def __init__(self, *args, **kwargs):
        super().__init__(*args, **kwargs)

        self.setFlag(self.ItemIsMovable)
        self.setFlag(self.ItemSendsGeometryChanges)
        self.line = None
        self.isPoint = None

    def addLine(self, line, ispoint):
        self.line = line
        self.isPoint = ispoint

    def itemChange(self, change , value):

        if change == self.ItemPositionChange and self.scene():
            newPos = value

            self.moveLineToCenter(newPos)

        return super(CustomItem, self).itemChange(change, value)

    def moveLineToCenter(self, newPos):
        xOffset = self.rect().x() + self.rect().width()/2
        yOffset = self.rect().y() + self.rect().height()/2

        newCenterPos = QtCore.QPointF(newPos.x()+xOffset, newPos.y()+yOffset)

        p1 = newCenterPos if self.isPoint else self.line.line().p1()
        p2 = self.line.line().p2() if self.isPoint else newCenterPos

        self.line.setLine(QtCore.QLineF(p1, p2))


def main():
    app =QtWidgets.QApplication(sys.argv)
    scene = QtWidgets.QGraphicsScene()

    ellipse = CustomItem(QtCore.QRectF(30, 30, 15, 25))
    scene.addItem(ellipse)

    ellipse2 = CustomItem(QtCore.QRectF(70, 70, 25, 15))
    scene.addItem(ellipse2)

    line = scene.addLine(QtCore.QLineF(40, 40, 80, 80))

    ellipse.addLine(line, True)
    ellipse2.addLine(line, False)

    view = QtWidgets.QGraphicsView(scene)
    view.show()

    sys.exit(app.exec_())


if __name__ == '__main__':
    main()

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM