En una aplicación Python3 / QT5, estoy tratando de mostrar una imagen SVG construida originalmente como una cadena. Necesito manipular esta imagen SVG (por ejemplo, cambiar su color), por lo que tengo la cadena cambiando con el tiempo. Aquí hay un ejemplo de trabajo mínimo:

import sys
from PyQt5.QtWidgets import QApplication
from PyQt5.QtSvg import QSvgWidget

svg_str = """<?xml version="1.0" encoding="UTF-8" standalone="no"?>
<svg width="300" height="300" viewBox="0 0 300 300" id="smile" version="1.1">
    <path
        style="fill:#ffaaaa"
        d="M 150,0 A 150,150 0 0 0 0,150 150,150 0 0 0 150,300 150,150 0 0 0 
            300,150 150,150 0 0 0 150,0 Z M 72,65 A 21,29.5 0 0 1 93,94.33 
            21,29.5 0 0 1 72,124 21,29.5 0 0 1 51,94.33 21,29.5 0 0 1 72,65 Z 
            m 156,0 a 21,29.5 0 0 1 21,29.5 21,29.5 0 0 1 -21,29.5 21,29.5 0 0 1 
            -21,-29.5 21,29.5 0 0 1 21,-29.5 z m -158.75,89.5 161.5,0 c 0,44.67 
            -36.125,80.75 -80.75,80.75 -44.67,0 -80.75,-36.125 -80.75,-80.75 z"
    />
</svg>
"""
# ==========================================
with open('smile.svg', 'w') as f:
    f.write(svg_str)
# ==========================================
app = QApplication(sys.argv)
svgWidget = QSvgWidget('smile.svg')
svgWidget.setGeometry(100,100,300,300)
svgWidget.show()
sys.exit(app.exec_())

Mi problema es que no pude encontrar una forma de evitar la necesidad de guardar el archivo para crear una instancia de un objeto QSvgWidget desde la propia cadena. No quiero guardar archivos indiscriminadamente y no pude encontrar una manera de cargar información de xml directamente en un objeto QSvgWidget ...

Encontré una solución más cercana a mis deseos, y se ve así:

import sys
from PyQt5.QtWidgets import QApplication
from PyQt5.QtSvg import QSvgWidget, QSvgRenderer
from PyQt5.QtCore import QXmlStreamReader
from PyQt5.QtGui import QPainter

svg_str = """<?xml version="1.0" encoding="UTF-8" standalone="no"?>
<svg width="300" height="300" viewBox="0 0 300 300" id="smile" version="1.1">
    <path
        style="fill:#ffaaaa"
        d="M 150,0 A 150,150 0 0 0 0,150 150,150 0 0 0 150,300 150,150 0 0 0 
            300,150 150,150 0 0 0 150,0 Z M 72,65 A 21,29.5 0 0 1 93,94.33 
            21,29.5 0 0 1 72,124 21,29.5 0 0 1 51,94.33 21,29.5 0 0 1 72,65 Z 
            m 156,0 a 21,29.5 0 0 1 21,29.5 21,29.5 0 0 1 -21,29.5 21,29.5 0 0 1 
            -21,-29.5 21,29.5 0 0 1 21,-29.5 z m -158.75,89.5 161.5,0 c 0,44.67 
            -36.125,80.75 -80.75,80.75 -44.67,0 -80.75,-36.125 -80.75,-80.75 z"
    />
</svg>
"""
# ==========================================
class QSvgWidget_from_string(QSvgWidget):
    def __init__(self, strSVG):
        super().__init__()
        self.strSVG = strSVG
    def paintEvent(self, event):
        qp = QPainter()
        qp.begin(self)
        svg_render = QSvgRenderer(QXmlStreamReader(self.strSVG))
        qp.restore()
        svg_render.render(qp)
        qp.end()
# ==========================================
app = QApplication(sys.argv)
svgWidget = QSvgWidget_from_string(svg_str)
svgWidget.setGeometry(100,100,300,300)
svgWidget.show()
sys.exit(app.exec_())

Pero no estoy satisfecho porque necesito expandir la clase QSvgWidget solo para instanciarla desde una cadena xml. Mi pregunta es: ¿Hay alguna forma de hacer esto sin tener que recurrir a QPaint y paintEvent?

3
iperetta 16 oct. 2018 a las 17:19

2 respuestas

La mejor respuesta

Puede obtener el QSvgRenderer utilizado por QSvgWidget para representar el SVG. Permite cargar datos desde un archivo, matriz de bytes o una secuencia:

import sys
from PyQt5.QtWidgets import QApplication
from PyQt5.QtSvg import QSvgWidget, QSvgRenderer

svg_str = """<?xml version="1.0" encoding="UTF-8" standalone="no"?>
<svg width="300" height="300" viewBox="0 0 300 300" id="smile" version="1.1">
    <path
        style="fill:#ffaaaa"
        d="M 150,0 A 150,150 0 0 0 0,150 150,150 0 0 0 150,300 150,150 0 0 0 
            300,150 150,150 0 0 0 150,0 Z M 72,65 A 21,29.5 0 0 1 93,94.33 
            21,29.5 0 0 1 72,124 21,29.5 0 0 1 51,94.33 21,29.5 0 0 1 72,65 Z 
            m 156,0 a 21,29.5 0 0 1 21,29.5 21,29.5 0 0 1 -21,29.5 21,29.5 0 0 1 
            -21,-29.5 21,29.5 0 0 1 21,-29.5 z m -158.75,89.5 161.5,0 c 0,44.67 
            -36.125,80.75 -80.75,80.75 -44.67,0 -80.75,-36.125 -80.75,-80.75 z"
    />
</svg>
"""

svg_bytes = bytearray(svg_str, encoding='utf-8')

app = QApplication(sys.argv)
svgWidget = QSvgWidget()
svgWidget.renderer().load(svg_bytes)
svgWidget.setGeometry(100,100,300,300)
svgWidget.show()
sys.exit(app.exec_())

Más información aquí: http://doc.qt.io/qt-5/qsvgrenderer. html

3
Isma 16 oct. 2018 a las 14:40

Si no necesita QSvgWidget p. Ej. le gustaría obtener QImage (del cual puede tener QPixmap y QIcon) puede usar:

qimage = QtGui.QImage.fromData(svg_bytes)

Asegúrese de que svg esté en admitidosImageFormats.

2
user898678 7 abr. 2019 a las 06:17