Ошибка PyQTGraph ImageExporter, связанная с размером изображения

Я использую pyqtgraph для создания графиков в окне PyQT. Я хочу экспортировать PNG-изображение графиков.

У меня есть эта ошибка, когда я пытаюсь экспортировать свои графики:

ImageExporter.py", строка 70, в экспорте bg = np.empty((self.params['width'], self.params['height'], 4), >dtype=np.ubyte) TypeError: 'float' объект не может быть интерпретирован как целое число

Я заметил, что self.params['width'] и self.params['height'] являются числами с плавающей запятой. Но np.empty не может создать массив, используя размеры с плавающей запятой.

Даже если я установлю ширину и высоту вручную, используя:

exporter.parameters()['width'] = self.raw_DataPlot.width()
exporter.parameters()['height'] = self.raw_DataPlot.height()

Результаты плавают.

Я заметил, что если я изменю строку 70 ImageExporter.py на:

bg = np.empty((int(self.params['width']), int(self.params['height']), 4),dtype=np.ubyte)

Экспорт работает нормально.

Можно ли решить эту проблему и обновить библиотеку? Или есть обходной путь, который не заставляет меня менять саму библиотеку pyqtgraph.

Спасибо


person Dr ALOUI    schedule 10.07.2018    source источник
comment
Сообщить об ошибке.   -  person eyllanesc    schedule 11.07.2018
comment
Я сообщил о проблеме на доске проблем github. Я надеялся на обходной путь, ожидая обновления.   -  person Dr ALOUI    schedule 11.07.2018
comment
то, что вы указываете в своем вопросе, является обходным путем :)   -  person eyllanesc    schedule 11.07.2018
comment
Это не элегантный обходной путь, поскольку я внес изменения в саму библиотеку. Когда я отправляю свой код кому-то другому, он у него не работает.   -  person Dr ALOUI    schedule 23.07.2018
comment
это не говорит о том, что он элегантный, но он единственный, который есть, загрузите свое изменение на github, pypi и т. д., а затем вы указываете, что они устанавливают его оттуда.   -  person eyllanesc    schedule 23.07.2018


Ответы (1)


Чтобы решить эту проблему, я создал новый ImageExporter, который интегрировал в свой проект. Я назвал этот экспортер PQG_ImageExporter, чтобы мы не запутались.

Я сообщил об этой проблеме на github. Я надеюсь, что они исправят это в ближайшее время. Когда проблема будет решена, я смогу вернуться к классическому ImageExporter:

from pyqtgraph.exporters import Exporter
from pyqtgraph.parametertree import Parameter
from pyqtgraph.Qt import QtGui, QtCore, QtSvg, USE_PYSIDE
from pyqtgraph import functions as fn
import numpy as np
import pyqtgraph as pg

__all__ = ['PQG_ImageExporter']


class PQG_ImageExporter(Exporter):
    Name = "Image File (PNG, TIF, JPG, ...)"
    allowCopy = True

    def __init__(self, item):
        Exporter.__init__(self, item)
        tr = self.getTargetRect()
        if isinstance(item, QtGui.QGraphicsItem):
            scene = item.scene()
        else:
            scene = item
        # scene.views()[0].backgroundBrush()
        bgbrush = pg.mkBrush('w')
        bg = bgbrush.color()
        if bgbrush.style() == QtCore.Qt.NoBrush:
            bg.setAlpha(0)

        self.params = Parameter(name='params', type='group', children=[
            {'name': 'width', 'type': 'int',
                'value': tr.width(), 'limits': (0, None)},
            {'name': 'height', 'type': 'int',
                'value': tr.height(), 'limits': (0, None)},
            {'name': 'antialias', 'type': 'bool', 'value': True},
            {'name': 'background', 'type': 'color', 'value': bg},
        ])
        self.params.param('width').sigValueChanged.connect(self.widthChanged)
        self.params.param('height').sigValueChanged.connect(self.heightChanged)

    def widthChanged(self):
        sr = self.getSourceRect()
        ar = float(sr.height()) / sr.width()
        self.params.param('height').setValue(
            self.params['width'] * ar, blockSignal=self.heightChanged)

    def heightChanged(self):
        sr = self.getSourceRect()
        ar = float(sr.width()) / sr.height()
        self.params.param('width').setValue(
            self.params['height'] * ar, blockSignal=self.widthChanged)

    def parameters(self):
        return self.params

    def export(self, fileName=None, toBytes=False, copy=False):
        if fileName is None and not toBytes and not copy:
            if USE_PYSIDE:
                filter = ["*."+str(f)
                          for f in QtGui.QImageWriter.supportedImageFormats()]
            else:
                filter = ["*."+bytes(f).decode('utf-8')
                          for f in QtGui.QImageWriter.supportedImageFormats()]
            preferred = ['*.png', '*.tif', '*.jpg']
            for p in preferred[::-1]:
                if p in filter:
                    filter.remove(p)
                    filter.insert(0, p)
            self.fileSaveDialog(filter=filter)
            return

        targetRect = QtCore.QRect(
            0, 0, self.params['width'], self.params['height'])
        sourceRect = self.getSourceRect()

        #self.png = QtGui.QImage(targetRect.size(), QtGui.QImage.Format_ARGB32)
        # self.png.fill(pyqtgraph.mkColor(self.params['background']))
        w, h = self.params['width'], self.params['height']
        if w == 0 or h == 0:
            raise Exception(
                "Cannot export image with size=0 (requested export size is %dx%d)" % (w, h))
        bg = np.empty((int(self.params['width']), int(
            self.params['height']), 4), dtype=np.ubyte)
        color = self.params['background']
        bg[:, :, 0] = color.blue()
        bg[:, :, 1] = color.green()
        bg[:, :, 2] = color.red()
        bg[:, :, 3] = color.alpha()
        self.png = fn.makeQImage(bg, alpha=True)

        # set resolution of image:
        origTargetRect = self.getTargetRect()
        resolutionScale = targetRect.width() / origTargetRect.width()
        #self.png.setDotsPerMeterX(self.png.dotsPerMeterX() * resolutionScale)
        #self.png.setDotsPerMeterY(self.png.dotsPerMeterY() * resolutionScale)

        painter = QtGui.QPainter(self.png)
        #dtr = painter.deviceTransform()
        try:
            self.setExportMode(True, {
                               'antialias': self.params['antialias'], 'background': self.params['background'], 'painter': painter, 'resolutionScale': resolutionScale})
            painter.setRenderHint(
                QtGui.QPainter.Antialiasing, self.params['antialias'])
            self.getScene().render(painter, QtCore.QRectF(
                targetRect), QtCore.QRectF(sourceRect))
        finally:
            self.setExportMode(False)
        painter.end()

        if copy:
            QtGui.QApplication.clipboard().setImage(self.png)
        elif toBytes:
            return self.png
        else:
            self.png.save(fileName)


PQG_ImageExporter.register()
person Dr ALOUI    schedule 24.07.2018