Попытка оптимизировать графику pyqtgraph

Я пытаюсь рисовать многоугольники без использования pg.PolyLineROI() в pyqtgraph. Моя цель — иметь возможность использовать большие наборы данных вместо «данных» в коде, показанном ниже. Моя проблема с PolyLineROI() заключается в том, что мне не нужны дескрипторы или события, поэтому загрузка больших объемов данных занимает много времени и ресурсов, которые просто тратятся на ненужную функциональность.

Я пытался использовать QPainter и QPen, но мне не удалось получить ничего работающего, поэтому я застрял. Любые идеи?

ОТРЕДАКТИРОВАНО Код, пытающийся включить пример из segFaultCoder

from PyQt4 import QtCore, QtGui
import pyqtgraph as pg
import sys

class plotwindow(QtGui.QMainWindow):
        def setupUi(self, MainWindow):

            self.centralwidget = QtGui.QWidget(MainWindow)
            MainWindow.resize(1900, 1000)

            self.qt = pg.GraphicsView(MainWindow)
            self.qt.setGeometry(QtCore.QRect(0,0, 1900, 1000))
            self.qt2 = pg.GraphicsLayout()
            self.qt.setCentralItem(self.qt2)
            self.qt.show()
            self.layout = self.qt2.addLayout()
            self.qt3 = self.layout.addViewBox()


            self.plot()

        def plot(self): #This is looped for multiple data sets
            data = [[6,6],[6,0],[0,6],[0,0]] #changes based on data import
            self.picture = QtGui.QPicture()
            p = QtGui.QPainter(self.picture)
            p.setPen(pg.mkPen('w'))
            self.points = []
            for item in data:
                point = QtCore.QPoint(item[0], item[1])
                self.points.append(point)
            p.drawPolygon(*self.points)
            p.end()
            self.qt3.addItem(p)

    def paint(self, p, *args):
        p.drawPicture(0, 0, self.picture)

    def boundingRect(self):
        return QtCore.QRectF(self.picture.boundingRect())

if __name__ == "__main__":
    app = QtGui.QApplication(sys.argv)
    MainWindow = QtGui.QMainWindow()
    ui = plotwindow()
    ui.setupUi(MainWindow)
    MainWindow.show()
    sys.exit(app.exec_()) 

person user3047917    schedule 28.10.2016    source источник


Ответы (1)


Я думаю, что вы были на правильном пути с QPainter, просто взглянув на их папку с примерами, вы увидите customGraphicsItem.py. Я просто скопировал это и заменил их переменную «данные» вашими точками, а затем переписал метод generatePicture(), используя QPainter.drawPolygon(). Надеюсь, все мои отступы верны, я не могу разобраться с форматированием сообщений на SO.

import pyqtgraph as pg
from pyqtgraph import QtCore, QtGui

# Create a subclass of GraphicsObject.
# The only required methods are paint() and boundingRect() 
# (see QGraphicsItem documentation)
class customPolyItem(pg.GraphicsObject):
    def __init__(self, data):
        pg.GraphicsObject.__init__(self)
        self.data = data  
        self.points = []
        self.generatePicture()


    def generatePicture(self):
        # pre-computing a QPicture object allows paint() to run much more quickly, 
        # rather than re-drawing the shapes every time.
        self.picture = QtGui.QPicture()
        p = QtGui.QPainter(self.picture)
        p.setPen(pg.mkPen('w'))
        for item in self.data:
            point = QtCore.QPoint(item[0],item[1])
            self.points.append(point)
        p.drawPolygon(*self.points)
        p.end()

    def paint(self, p, *args):
        p.drawPicture(0, 0, self.picture)

    def boundingRect(self):
        # boundingRect _must_ indicate the entire area that will be drawn on
        # or else we will get artifacts and possibly crashing.
        # (in this case, QPicture does all the work of computing the bouning rect for us)
        return QtCore.QRectF(self.picture.boundingRect())

data = [[6,6],[6,0],[0,6],[0,0]]

item = customPolyItem(data)
plt = pg.plot()
plt.addItem(item)
plt.setWindowTitle('pyqtgraph example: customPolyItem')

## 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'):
        QtGui.QApplication.instance().exec_()

Изменить: что касается вашего нового кода, замените метод построения на этот (и удалите краску и ограничивающий прямоугольник):

    def plot(self): #This is looped for multiple data sets
        data = [[6,6],[6,0],[0,6],[0,0]] #changes based on data import
        newCPI = customPolyItem(data) # create new instance with changed data
        self.qt3.addItem(newCPI) # add the new instance to your viewbox

Вам нужно будет импортировать класс customPolyItem, если он находится в отдельном файле, или просто скопировать и вставить все объявление класса под текущим классом окна графика.

person segFaultCoder    schedule 29.10.2016
comment
Я застрял, пытаясь внедрить ваш код в свой цикл. Я отредактировал свой код выше, чтобы представить, где я сейчас нахожусь. Я получаю ту же ошибку с моим отредактированным кодом выше, что и при вызове вашего класса в моем коде, поэтому я не уверен, что делать дальше. - person user3047917; 31.10.2016
comment
просто посмотрел поближе на ваш код. Обратите внимание в комментариях выше моего примера, customPolyItem, что он является подклассом GraphicsObject, и, как упоминается там, 'paint' и 'boundingrect' являются переопределением методов 'QGraphicsObject -> QGraphicsItem'. В вашем коде вы не можете теперь использовать эти два метода в подклассе QMainWindow. Для каждого нового набора данных вам необходимо создать экземпляр customPolyItem и соответствующим образом добавить его к своим графикам/макетам, и в этом сценарии customPolyItem по-прежнему владеет методами рисования и ограничения. - person segFaultCoder; 02.11.2016
comment
Смотрите мой отредактированный ответ и обязательно примите ответ, если он работает :) - person segFaultCoder; 02.11.2016
comment
Я попробовал именно то, что у вас есть, и при построении больших наборов данных он вырезает большую часть данных и просто отображает несколько квадратов, когда они помещаются в отдельный файл и вызываются в моем коде. Я установил данные в набор данных, содержащий 30 точек, и он нарисовал один прямоугольник. - person user3047917; 02.11.2016
comment
поскольку я не совсем понимаю, чего вы пытаетесь достичь, я не могу решить эту проблему. если вы можете опубликовать код, который не работает, и объяснить, что вы хотели бы, чтобы он делал, я был бы рад попытаться помочь. Я, конечно, не могу проверить граничные условия для вас без более подробной информации. Вы заставили его работать с 4-точечным примером, который вы опубликовали? если да, то в какой момент он ломается? - person segFaultCoder; 03.11.2016
comment
Ага, спасибо за помощь. В итоге я использовал GraphItem() для создания объектов, и пока я строю всю структуру моего массива до вызова addItem(), он имеет очень хорошую производительность. - person user3047917; 10.11.2016