0

我很难将一个项目的坐标中的点转换为另一个项目的坐标,就像这样

from PySide import QtGui, QtCore
import sys

class Editor(QtGui.QMainWindow):
    def __init__(self, parent=None):
        super(Editor, self).__init__(parent)

        scene = QtGui.QGraphicsScene()

        line0 = QtGui.QGraphicsLineItem(  10 , 210 ,  10 , 300 )
        line1 = QtGui.QGraphicsLineItem( 100 , 210 , 100 , 300 )

        scene.addItem( line0 )
        scene.addItem( line1 )

        view = QtGui.QGraphicsView()
        view.setScene( scene )

        self.setGeometry( 250 , 250 , 600 , 600 )
        self.setCentralWidget(view)
        self.show()

        print line1.mapToItem( line0 , QtCore.QPoint( 0 , 0 ) )  # QPoint( 0 , 0 ) in line0's coordinates -> line1's coordinates
        print line1.mapToScene( QtCore.QPointF( 0 , 0 ) )        # QPoint( 0 , 0 ) in line0's coordinates -> screen coordinates


if __name__=="__main__":
    app=QtGui.QApplication(sys.argv)
    myapp = Editor()
    sys.exit(app.exec_())

结果似乎表明转换失败

PySide.QtCore.QPointF(0.000000, 0.000000)
PySide.QtCore.QPointF(0.000000, 0.000000)
4

1 回答 1

1

我认为您假设项目的坐标系从其 topLeft boundingRect 开始,但事实并非如此。此外,另一个概念是您传递给 QGraphicsLineItem 构造函数的坐标是相对于该项目的,而不是相对于场景的坐标。

p1(x1, y1)
    ╲
     ╲
      ╲
       ╲
        ╲
         ╲
        p2(x2, y2)
The coordinates (x1, y1) and (x2, y2) are relative to the QGraphicsLineItem

因此,如果您想获得两条线之间相对于 line1 的位置差异,您必须映射与 QGraphicsLineItem 关联的 QLineF 的 p1() 的值:

print(line1.mapToItem(line0 , line0.line().p1()) -   line1.line().p1())
      └----p1 that belongs to line0 ----------┘    └--p1 that belongs--┘    
              with respect to line1            to line1 with respect to line1

输出:

PySide.QtCore.QPointF(-90.000000, 0.000000)

解释:

图形视图框架处理 3 种类型的坐标系:

  • 关于QGraphicsView的viewport的坐标,即坐标依赖于view
  • 相对于场景的坐标系,所有视图都具有相同的坐标系。
  • 每个项目的坐标系

可以用图像或视频记录系统进行类比。第一个坐标系指的是相对于相机将看到的内容,这取决于类似于 QGraphicsView 的每个相机。第二个坐标系是相对于现实世界的,它不依赖于相机。第三个坐标系是相对于场景中的一个元素,例如一个演员

项目坐标系的点(0, 0)与项目在场景中的位置相匹配。在您的情况下,您的项目相对于场景的位置是 (0, 0)(使用 进行检查print(line0.pos())。

从视觉上理解以下代码是相同的,但在概念上却不同。

class Editor(QtGui.QMainWindow):
    def __init__(self, parent=None):
        super(Editor, self).__init__(parent)
        scene = QtGui.QGraphicsScene()
        line0 = QtGui.QGraphicsLineItem(0, 0, 0, 90)
        line0.setPos(10, 210)
        line1 = QtGui.QGraphicsLineItem(0, 0, 0, 90)
        line1.setPos(100, 210)
        scene.addItem( line0 )
        scene.addItem( line1 )
        view = QtGui.QGraphicsView()
        view.setScene( scene )
        self.setGeometry( 250 , 250 , 600 , 600 )
        self.setCentralWidget(view)
        self.show()

于 2019-04-15T18:19:49.277 回答