pyqtgraph中带有箭头的自定义线性区域元素

2024-05-14 23:35:32 发布

您现在位置:Python中文网/ 问答频道 /正文

我正在Pyqtgraph中制作一个数据可视化应用程序,并希望包括能够选择一个区域并进一步处理它的功能。我可以使用LinearRegionItem来实现这一点。但我还想显示的是一个带有标签的箭头,用于指示选择的长度。下面是我使用ArrowItem的尝试。问题是箭头的长度与预期不符

我假设箭头的长度是错误的,因为从一个场景缩放到另一个视图,或者反过来。我试图理解映射是没有意义的,因为mapToScene的输出不会改变,即使我更改了它的参数QPoint

我寻求你的帮助来指引我正确的方向

from PySide2 import QtCore, QtGui

import numpy as np
import pyqtgraph as pg

class LinearRegionItemWithArrow(pg.LinearRegionItem):

    def __init__(self, vb, *args, **kwds):
        pg.LinearRegionItem.__init__(self, *args, **kwds)
        self.vb = vb
        self.headLen = 20

        pos1, pos2 = self.getRegion()
        # pos1_vb = self.vb.mapToScene(QtCore.QPoint(pos2, 0)).x()
        # pos2_vb = self.vb.mapToScene(QtCore.QPoint(pos1, 0)).x()
        # print((pos1, pos2), (pos1_vb, pos2_vb))

        tailLen = pos2 - pos1 - self.headLen
        # tailLen_vb = pos2_vb - pos1_vb
        # print(tailLen/tailLen_vb)
        self.arrow1 = pg.ArrowItem(self, 
                                    angle=0, 
                                    headLen=self.headLen,
                                    tipAngle=45,
                                    pos= (pos1, 1), 
                                    tailLen = tailLen, #tailLen_vb if tailLen>2*self.headLen else None,
                                    tailWidth = 3, 
                                    pen=None, 
                                    brush=pg.mkBrush(255, 165, 0), 
                                    pxMode=True)
        
        self.sigRegionChanged.connect(self.updateArrows)

    def updateArrows(self):
        print(self.vb.mapFromScene(QtCore.QPoint(36, 0)))
        pos1, pos2 = self.getRegion()
        # pos1_vb = self.vb.mapToScene(QtCore.QPoint(pos2, 0)).x()
        # pos2_vb = self.vb.mapToScene(QtCore.QPoint(pos1, 0)).x()
        # print((pos1, pos2), (pos1_vb, pos2_vb))

        tailLen = pos2 - pos1 - self.headLen
        # tailLen_vb = pos2_vb - pos1_vb
        # print(tailLen/tailLen_vb)

        self.arrow1.setPos(pos1, 1)
        self.arrow1.setStyle(tailLen = tailLen)# if tailLen>2*self.headLen else None)

data = np.random.normal(size=1000)
win = pg.plot()
plot = win.plot(data, name='data', pen='r', title="Simplest possible plotting example")

lri = LinearRegionItemWithArrow(win.getViewBox(), values=[200, 500])
lri.setZValue(10)
win.addItem(lri)
print(win.mapFromScene(36 , 0))
lri.sigRegionChanged.connect(lambda: win.getViewBox().mapFromScene(QtCore.QPoint(36, 0))) #.mapSceneToView((1.0, 1.0))


## Start Qt event loop unless running in interactive mode or using pyside.
if __name__ == '__main__':
    import sys
    if sys.flags.interactive != 1 or not hasattr(QtCore, 'PYQT_VERSION'):
        pg.QtGui.QApplication.exec_()

Tags: importselfifwinpgvbprintpos1
1条回答
网友
1楼 · 发布于 2024-05-14 23:35:32

正如我在评论中提到的,我应该使用mapSceneToView 而不是mapFromScenemapSceneToView允许将点从绘图坐标系带到窗口(ViewBox)坐标系,这正是我所寻找的。下面是清理后的代码。希望这能帮助别人

from PySide2 import QtCore

import numpy as np
import pyqtgraph as pg


class LinearRegionItemWithArrow(pg.LinearRegionItem):

    def __init__(self, plotWindow, *args, **kwds):
        pg.LinearRegionItem.__init__(self, *args, **kwds)
        self.plotWindow = plotWindow
        self.headLen = 20

        self.arrow1 = pg.ArrowItem(self, 
                                    angle=0, 
                                    headLen=self.headLen,
                                    tipAngle=45,
                                    pos= (0, 0), 
                                    tailLen = 20, #tailLen_vb if tailLen>2*self.headLen else None,
                                    tailWidth = 3, 
                                    pen=None, 
                                    brush=pg.mkBrush(255, 165, 0), 
                                    pxMode=True)
        self.arrow2 = pg.ArrowItem(self, 
                                    angle=180, 
                                    headLen=self.headLen,
                                    tipAngle=45,
                                    pos= (0, 0), 
                                    tailLen = 20, #tailLen_vb if tailLen>2*self.headLen else None,
                                    tailWidth = 3, 
                                    pen=None, 
                                    brush=pg.mkBrush(255, 165, 0), 
                                    pxMode=True)
        self.updateArrows()
        self.sigRegionChanged.connect(self.updateArrows)
        self.plotWindow.sigRangeChanged.connect(self.updateArrows)  


        self.labels = [pg.TextItem(text='', anchor=(1, 0.5)), pg.TextItem(text='', anchor=(0.5, 0.5)), pg.TextItem(text='', anchor=(0, 0.5))]
        self.updateLabels()
        [self.plotWindow.addItem(l) for l in self.labels]
        self.sigRegionChanged.connect(self.updateLabels)
        self.plotWindow.sigRangeChanged.connect(self.updateLabels)  

    def updateLabels(self):
        pos1, pos2 = self.getRegion()
        for l, p in zip(self.labels, [pos1, 0.5*(pos1+pos2), pos2]):
            l.setPos(p, self.getYPosition())
            l.setText('%.2f'%p)
        self.labels[1].setText('%.2f'%(pos2 - pos1))

    def updateArrows(self):
        pos1, pos2 = self.getRegion()
        self.arrow1.setPos(pos1, self.getYPosition())
        self.arrow2.setPos(pos2, self.getYPosition())

    def getYPosition(self):
        return self.plotWindow.getViewBox().mapSceneToView(QtCore.QPoint(0, 0.1*self.plotWindow.range.height())).y()

data = np.random.normal(size=1000)
win = pg.plot()
plot = win.plot(data, name='data', pen='r', title="Simplest possible plotting example")

lri = LinearRegionItemWithArrow(win, values=[200, 500])
lri.setZValue(10)
win.addItem(lri)

## Start Qt event loop unless running in interactive mode or using pyside.
if __name__ == '__main__':
    import sys
    if sys.flags.interactive != 1 or not hasattr(QtCore, 'PYQT_VERSION'):
        pg.QtGui.QApplication.exec_()

相关问题 更多 >

    热门问题