Vielleicht werfen Sie auch einen Blick auf meine andere Antwort, da seitdem neue, aktuellere Tools entstanden sind.
Ich bin zu dem Code gekommen, der leider nicht voll funktionsfähig ist. Dies basiert auf der obigen Lösung und auf diesen anderen Fragen:
Wie exportiere ich eine Komposition programmgesteuert als Bild?
Wie kann ich die Einstellungen für das QgsPaperItem aus XML lesen?
Speichern von Map Canvas als PNG mit transparentem Hintergrund programmgesteuert mit QGIS?
Mein Code kann die .qpt-Datei aus einer .qgs-Datei extrahieren und einen Composer erfolgreich aus der Vorlage laden. Außerdem wird ein Komponist in eine PNG-Datei gedruckt und die im Komponisten gespeicherten Beschriftungen und Formen werden korrekt angezeigt.
Es können jedoch nicht alle Elemente geladen werden, die sich auf die tatsächliche Karte und Ebenen beziehen (Beschriftungen, die Ausdrücke aus Ebenen enthalten, werden ebenfalls nicht gezeichnet). Ich glaube, ich habe ein bisschen übersehen, wie das Projekt geladen und mit dem Komponisten verknüpft werden muss.
Einige Leute im Kommentar des Originalartikels von Tim Sutton erwähnten, dass sie unter Windows im selben Stadium feststeckten (es ist mein Fall). Das ist wirklich frustrierend, weil ich das Gefühl habe, dass die Antwort wirklich sehr nahe ist. Liebes Internet bitte helfen!
Auch dies sind meine ersten Versuche mit Python, also hoffe ich, dass du nett bist;)
#This python code aim to programmatically export the first composer stored in a qgs file using PyQgis API v 2.10
#Version 0.4 (non functional) WTFPL MarHoff 2015 - This code is mostly a "frankenstein" stub made with a lot of other snippets. Feel welcome to improve!
#Credits to gis.stackexchange community : drnextgis,ndawson,patdevelop,dakcarto,ahoi, underdark & Tim Sutton from kartoza
#More informations and feedback can be found at /gis/144792/
#This script assume your environement is setup for PyGis as a stand-alone script. Some nice hints for windows users : /gis//a/130102/17548
import sys
from PyQt4.QtCore import *
from PyQt4.QtXml import *
from qgis.core import *
from qgis.gui import *
gui_flag = True
app = QgsApplication(sys.argv, gui_flag)
# Make sure QGIS_PREFIX_PATH is set in your env if needed!
app.initQgis()
# Name of the .qgs file without extension
project_name = 'myproject'
#Important : The code is assuming that the .py file is in the same folder as the project
folderPath = QString(sys.path[0])+'/'
projectPath = QString(folderPath+project_name+'.qgs')
templatePath = QString(folderPath+project_name+'_firstcomposer.qpt')
imagePath = QString(folderPath+project_name+'.png')
#Getting project as Qfile and the first composer of the project as a QDomElement from the .qgs
projectAsFile = QFile(projectPath)
projectAsDocument = QDomDocument()
projectAsDocument.setContent(projectAsFile)
composerAsElement = projectAsDocument.elementsByTagName("Composer").at(0).toElement()
#This block store the composer into a template file
templateFile = QFile(templatePath)
templateFile.open(QIODevice.WriteOnly)
out = QTextStream(templateFile)
#I need next line cause UTF-8 is somewhat tricky in my setup, comment out if needed
out.setCodec("UTF-8")
param = QString
composerAsElement.save(out,2)
templateFile.close()
#And this block load back the composer into a QDomDocument
#Nb: This is ugly as hell, i guess there is a way to convert a QDomElement to a QDomDocument but every attemps failed on my side...
composerAsDocument = QDomDocument()
composerAsDocument.setContent(templateFile)
#Now that we got all we can open our project
canvas = QgsMapCanvas()
QgsProject.instance().read(QFileInfo(projectAsFile))
bridge = QgsLayerTreeMapCanvasBridge(
QgsProject.instance().layerTreeRoot(), canvas)
bridge.setCanvasLayers()
#Lets try load that composer template we just extracted
composition = QgsComposition(canvas.mapSettings())
composition.loadFromTemplate(composerAsDocument, {})
#And lets print in our .png
image = composition.printPageAsRaster(0)
image.save(imagePath,'png')
#Some cleanup maybe?
QgsProject.instance().clear()
QgsApplication.exitQgis()
Ich habe diese Zeilen aus dem vorherigen Code entfernt, da sie anscheinend überhaupt nichts zu tun hatten. Sie haben keinen Fehler hervorgebracht, aber es nicht besser gemacht.
# You must set the id in the template
map_item = composition.getComposerItemById('map')
map_item.setMapCanvas(canvas)
map_item.zoomToExtent(canvas.extent())
# You must set the id in the template
legend_item = composition.getComposerItemById('legend')
legend_item.updateLegend()
composition.refreshItems()
und diese werden ebenfalls entfernt, da sie bei Verwendung von printPageAsRaster () unnötig erschienen.
dpmm = dpi / 25.4
width = int(dpmm * composition.paperWidth())
height = int(dpmm * composition.paperHeight())
# create output image and initialize it
image = QImage(QSize(width, height), QImage.Format_ARGB32)
image.setDotsPerMeterX(dpmm * 1000)
image.setDotsPerMeterY(dpmm * 1000)
image.fill(0)
# render the composition
imagePainter = QPainter(image)
composition.renderPage(imagePainter, 0)
imagePainter.end()