Być może rzuć okiem na moją drugą odpowiedź , ponieważ od tego czasu pojawiły się nowe, bardziej aktualne narzędzia.
Doszedłem do kodu poniżej, który niestety nie jest w pełni funkcjonalny. Jest to oparte na powyższym rozwiązaniu i na innych pytaniach:
Jak programowo wyeksportować kompozycję jako obraz?
Jak mogę odczytać ustawienia QgsPaperItem z XML?
Zapisujesz płótno mapy w formacie PNG z przezroczystym tłem programowo z QGIS?
Mój kod jest w stanie wyodrębnić .qpt z pliku .qgs i pomyślnie załadować kompozytora z szablonu. Drukuje również kompozytora do pliku .png i wyświetla poprawnie etykiety i kształty zapisane w kompozytorze.
Jednak nie ładuje wszystkich elementów związanych z rzeczywistą mapą i warstwami (etykieta zawierająca wyrażenie z warstwy również nie jest rysowana). Myślę, że trochę mi brakowało, w jaki sposób projekt musi zostać załadowany i powiązany z kompozytorem.
Niektóre osoby w komentarzu do oryginalnego artykułu Tima Suttona wspomniały, że utknęły na tym samym etapie pod Windows (to mój przypadek). To jest naprawdę frustrujące, ponieważ czuję, że odpowiedź jest naprawdę bardzo blisko. Drogi Internecie, proszę o pomoc!
To także moje pierwsze próby Pythona, więc mam nadzieję, że będziesz miły;)
#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 : https://gis.stackexchange.com/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()
Usunąłem tezy z poprzedniego kodu, ponieważ wydawały się w ogóle nic nie robić. Nie pojawił się żaden błąd, ale nie zrobił nic lepszego.
# 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()
i te też są usuwane, ponieważ wydawały się niepotrzebne podczas używania printPageAsRaster ()
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()