Skip to content

Commit

Permalink
[composer] add svg export with layers
Browse files Browse the repository at this point in the history
  • Loading branch information
vmora authored and nyalldawson committed Apr 16, 2014
1 parent e6680a1 commit 0c19945
Show file tree
Hide file tree
Showing 10 changed files with 409 additions and 29 deletions.
10 changes: 10 additions & 0 deletions python/core/composer/qgscomposeritem.sip
Expand Up @@ -401,6 +401,16 @@ class QgsComposerItem : QObject, QGraphicsRectItem
@note there is not setter since one can't manually set the id*/
QString uuid() const;

/**Get the number of layers that this item exports
@returns 0 if this item is to be placed on the same layer as the previous item
@note this method was added in version 2.4 */
int numberExportLayers() const;

/**Set the layer to export
@param layerIdx can be set to -1 to export all layerr and must be less than numberExportLayers()
@note this method was added in version 2.4 */
void setCurrentExportLayer( int layerIdx = -1 );

public slots:
/**Sets the item rotation
* @deprecated Use setItemRotation( double rotation ) instead
Expand Down
5 changes: 5 additions & 0 deletions python/core/composer/qgscomposermap.sip
Expand Up @@ -407,6 +407,11 @@ class QgsComposerMap : QgsComposerItem
/** Returns whether updates to the composer map are enabled. */
bool updatesEnabled() const;

/**Get the number of layers that this item exports
@returns 0 if this item is to be placed on the same layer as the previous item
@note this method was added in version 2.4 */
int numberExportLayers() const;

signals:
void extentChanged();

Expand Down
5 changes: 5 additions & 0 deletions python/core/composer/qgscomposition.sip
Expand Up @@ -395,6 +395,11 @@ class QgsComposition : QGraphicsScene
/** Sets the current atlas mode of the composition. Returns false if the mode could not be changed. */
bool setAtlasMode( QgsComposition::AtlasMode mode );

/** Return pages in the correct order
@note composerItems(QList< QgsPaperItem* > &) may not return pages in the correct order
@note added in version 2.4*/
QList< QgsPaperItem* > pages();

public slots:
/**Casts object to the proper subclass type and calls corresponding itemAdded signal*/
void sendItemAddedSignal( QgsComposerItem* item );
Expand Down
208 changes: 188 additions & 20 deletions src/app/composer/qgscomposer.cpp
Expand Up @@ -55,6 +55,9 @@
#include "qgscursors.h"
#include "qgsmaplayeractionregistry.h"
#include "qgsgeometry.h"
#include "qgspaperitem.h"
#include "qgsmaplayerregistry.h"
#include "ui_qgssvgexportoptions.h"

#include <QCloseEvent>
#include <QCheckBox>
Expand Down Expand Up @@ -1748,6 +1751,37 @@ void QgsComposer::on_mActionExportAsSVG_triggered()
exportCompositionAsSVG( QgsComposer::Single );
}

// utility class that will hide all items until it's destroyed
struct QgsItemTempHider
{
explicit QgsItemTempHider( const QList<QGraphicsItem *> & items )
{
QList<QGraphicsItem *>::const_iterator it = items.begin();
for ( ; it != items.end(); ++it )
{
mItemVisibility[*it] = ( *it )->isVisible();
( *it )->hide();
}
}
void hideAll()
{
QgsItemVisibilityHash::iterator it = mItemVisibility.begin();
for ( ; it != mItemVisibility.end(); ++it ) it.key()->hide();
}
~QgsItemTempHider()
{
QgsItemVisibilityHash::iterator it = mItemVisibility.begin();
for ( ; it != mItemVisibility.end(); ++it )
{
it.key()->setVisible( it.value() );
}
}
private:
Q_DISABLE_COPY( QgsItemTempHider )
typedef QHash<QGraphicsItem*, bool> QgsItemVisibilityHash;
QgsItemVisibilityHash mItemVisibility;
};

void QgsComposer::exportCompositionAsSVG( QgsComposer::OutputMode mode )
{
if ( containsWMSLayer() )
Expand Down Expand Up @@ -1785,6 +1819,7 @@ void QgsComposer::exportCompositionAsSVG( QgsComposer::OutputMode mode )

QString outputFileName;
QString outputDir;
bool groupLayers = false;

if ( mode == QgsComposer::Single )
{
Expand All @@ -1800,14 +1835,25 @@ void QgsComposer::exportCompositionAsSVG( QgsComposer::OutputMode mode )
outputFileName = file.path();
}

// open file dialog
outputFileName = QFileDialog::getSaveFileName(
this,
tr( "Choose a file name to save the map as" ),
outputFileName,
tr( "SVG Format" ) + " (*.svg *.SVG)" );

if ( outputFileName.isEmpty() )
return;

// open otions dialog
{
QDialog dialog;
Ui::QgsSvgExportOptionsDialog options;
options.setupUi( &dialog );
dialog.exec();
groupLayers = options.chkMapLayersAsGroup->isChecked();
}

if ( !outputFileName.endsWith( ".svg", Qt::CaseInsensitive ) )
{
outputFileName += ".svg";
Expand All @@ -1834,10 +1880,12 @@ void QgsComposer::exportCompositionAsSVG( QgsComposer::OutputMode mode )
QSettings myQSettings;
QString lastUsedDir = myQSettings.value( "/UI/lastSaveAtlasAsSvgDir", "." ).toString();

// open file dialog
outputDir = QFileDialog::getExistingDirectory( this,
tr( "Directory where to save SVG files" ),
lastUsedDir,
QFileDialog::ShowDirsOnly );

if ( outputDir.isEmpty() )
{
return;
Expand All @@ -1852,6 +1900,16 @@ void QgsComposer::exportCompositionAsSVG( QgsComposer::OutputMode mode )
return;
}

// open otions dialog
{
QDialog dialog;
Ui::QgsSvgExportOptionsDialog options;
options.setupUi( &dialog );
dialog.exec();
groupLayers = options.chkMapLayersAsGroup->isChecked();
}


myQSettings.setValue( "/UI/lastSaveAtlasAsSvgDir", outputDir );
}

Expand Down Expand Up @@ -1907,32 +1965,142 @@ void QgsComposer::exportCompositionAsSVG( QgsComposer::OutputMode mode )
outputFileName = QDir( outputDir ).filePath( atlasMap->currentFilename() ) + ".svg";
}

for ( int i = 0; i < mComposition->numPages(); ++i )
if ( !groupLayers )
{
QSvgGenerator generator;
generator.setTitle( QgsProject::instance()->title() );
if ( i == 0 )
{
generator.setFileName( outputFileName );
}
else
for ( int i = 0; i < mComposition->numPages(); ++i )
{
QFileInfo fi( outputFileName );
generator.setFileName( fi.absolutePath() + "/" + fi.baseName() + "_" + QString::number( i + 1 ) + "." + fi.suffix() );
QSvgGenerator generator;
generator.setTitle( QgsProject::instance()->title() );
if ( i == 0 )
{
generator.setFileName( outputFileName );
}
else
{
QFileInfo fi( outputFileName );
generator.setFileName( fi.absolutePath() + "/" + fi.baseName() + "_" + QString::number( i + 1 ) + "." + fi.suffix() );
}

//width in pixel
int width = ( int )( mComposition->paperWidth() * mComposition->printResolution() / 25.4 );
//height in pixel
int height = ( int )( mComposition->paperHeight() * mComposition->printResolution() / 25.4 );
generator.setSize( QSize( width, height ) );
generator.setViewBox( QRect( 0, 0, width, height ) );
generator.setResolution( mComposition->printResolution() ); //because the rendering is done in mm, convert the dpi

QPainter p( &generator );

mComposition->renderPage( &p, i );
p.end();
}
}
else
{
//width and height in pixel
const int width = ( int )( mComposition->paperWidth() * mComposition->printResolution() / 25.4 );
const int height = ( int )( mComposition->paperHeight() * mComposition->printResolution() / 25.4 );
QList< QgsPaperItem* > paperItems( mComposition->pages() ) ;

//width in pixel
int width = ( int )( mComposition->paperWidth() * mComposition->printResolution() / 25.4 );
//height in pixel
int height = ( int )( mComposition->paperHeight() * mComposition->printResolution() / 25.4 );
generator.setSize( QSize( width, height ) );
generator.setViewBox( QRect( 0, 0, width, height ) );
generator.setResolution( mComposition->printResolution() ); //because the rendering is done in mm, convert the dpi
for ( int i = 0; i < mComposition->numPages(); ++i )
{
QDomDocument svg;
QDomNode svgDocRoot;
QgsPaperItem * paperItem = paperItems[i];
const QRectF paperRect = QRectF( paperItem->pos().x(),
paperItem->pos().y(),
paperItem->rect().width(),
paperItem->rect().height() );

QList<QGraphicsItem *> items = mComposition->items( paperRect,
Qt::IntersectsItemBoundingRect,
Qt::AscendingOrder );
if ( ! items.isEmpty()
&& dynamic_cast<QgsPaperGrid*>( items.last() )
&& !mComposition->gridVisible() ) items.pop_back();
QgsItemTempHider itemsHider( items );
int composerItemLayerIdx = 0;
QList<QGraphicsItem *>::const_iterator it = items.begin();
for ( unsigned svgLayerId = 1; it != items.end(); ++svgLayerId )
{
itemsHider.hideAll();
QgsComposerItem * composerItem = dynamic_cast<QgsComposerItem*>( *it );
QString layerName( "Layer " + QString::number( svgLayerId ) );
if ( composerItem && composerItem->numberExportLayers() )
{
composerItem->show();
composerItem->setCurrentExportLayer( composerItemLayerIdx );
++composerItemLayerIdx;
}
else
{
// show all items until the next item that renders on a separate layer
for ( ; it != items.end(); ++it )
{
composerItem = dynamic_cast<QgsComposerMap*>( *it );
if ( composerItem && composerItem->numberExportLayers() )
{
break;
}
else
{
( *it )->show();
}
}
}

QPainter p( &generator );
QBuffer svgBuffer;
{
QSvgGenerator generator;
generator.setTitle( QgsProject::instance()->title() );
generator.setOutputDevice( &svgBuffer );
generator.setSize( QSize( width, height ) );
generator.setViewBox( QRect( 0, 0, width, height ) );
generator.setResolution( mComposition->printResolution() ); //because the rendering is done in mm, convert the dpi

QPainter p( &generator );
mComposition->renderPage( &p, i );
}
// post-process svg output to create groups in a single svg file
// we create inkscape layers since it's nice and clean and free
// and fully svg compatible
{
svgBuffer.close();
svgBuffer.open( QIODevice::ReadOnly );
QDomDocument doc;
QString errorMsg;
int errorLine;
if ( ! doc.setContent( &svgBuffer, false, &errorMsg, &errorLine ) )
QMessageBox::warning( 0, tr( "Svg error" ), tr( "There was an error in svg ouput for svg layer " ) + layerName + tr( " on page " ) + QString::number( i + 1 ) + "(" + errorMsg + ")" );
if ( 1 == svgLayerId )
{
svg = QDomDocument( doc.doctype() );
svg.appendChild( svg.importNode( doc.firstChild(), false ) );
svgDocRoot = svg.importNode( doc.elementsByTagName( "svg" ).at( 0 ), false );
svgDocRoot.toElement().setAttribute( "xmlns:inkscape", "http://www.inkscape.org/namespaces/inkscape" );
svg.appendChild( svgDocRoot );
}
QDomNode mainGroup = svg.importNode( doc.elementsByTagName( "g" ).at( 0 ), true );
mainGroup.toElement().setAttribute( "id", layerName );
mainGroup.toElement().setAttribute( "inkscape:label", layerName );
mainGroup.toElement().setAttribute( "inkscape:groupmode", "layer" );
QDomNode defs = svg.importNode( doc.elementsByTagName( "defs" ).at( 0 ), true );
svgDocRoot.appendChild( defs );
svgDocRoot.appendChild( mainGroup );
}

mComposition->renderPage( &p, i );
p.end();
if ( composerItem && composerItem->numberExportLayers() && composerItem->numberExportLayers() == composerItemLayerIdx ) // restore and pass to next item
{
composerItem->setCurrentExportLayer();
composerItemLayerIdx = 0;
++it;
}
}
QFileInfo fi( outputFileName );
QFile out( i == 0 ? outputFileName : fi.absolutePath() + "/" + fi.baseName() + "_" + QString::number( i + 1 ) + "." + fi.suffix() );
out.open( QIODevice::WriteOnly | QIODevice::Text );
out.write( svg.toByteArray() );
}
}
featureI++;
}
Expand Down
2 changes: 2 additions & 0 deletions src/core/composer/qgscomposeritem.cpp
Expand Up @@ -64,6 +64,7 @@ QgsComposerItem::QgsComposerItem( QgsComposition* composition, bool manageZValue
, mEffectsEnabled( true )
, mTransparency( 0 )
, mLastUsedPositionMode( UpperLeft )
, mCurrentExportLayer( -1 )
, mId( "" )
, mUuid( QUuid::createUuid().toString() )
{
Expand All @@ -88,6 +89,7 @@ QgsComposerItem::QgsComposerItem( qreal x, qreal y, qreal width, qreal height, Q
, mEffectsEnabled( true )
, mTransparency( 0 )
, mLastUsedPositionMode( UpperLeft )
, mCurrentExportLayer( -1 )
, mId( "" )
, mUuid( QUuid::createUuid().toString() )
{
Expand Down
15 changes: 15 additions & 0 deletions src/core/composer/qgscomposeritem.h
Expand Up @@ -361,6 +361,16 @@ class CORE_EXPORT QgsComposerItem: public QObject, public QGraphicsRectItem
@note there is not setter since one can't manually set the id*/
QString uuid() const { return mUuid; }

/**Get the number of layers that this item exports
@returns 0 if this item is to be placed on the same layer as the previous item
@note this method was added in version 2.4 */
virtual int numberExportLayers() const { return 0; }

/**Set the layer to export
@param layerIdx can be set to -1 to export all layerr and must be less than numberExportLayers()
@note this method was added in version 2.4 */
virtual void setCurrentExportLayer( int layerIdx = -1 ) { mCurrentExportLayer = layerIdx; }

public slots:
/**Sets the item rotation
* @deprecated Use setItemRotation( double rotation ) instead
Expand Down Expand Up @@ -423,6 +433,11 @@ class CORE_EXPORT QgsComposerItem: public QObject, public QGraphicsRectItem
@note: this member was added in version 2.0*/
ItemPositionMode mLastUsedPositionMode;

/**The layer that needs to be exported
@note: if -1, all layers are to be exported
@note: this member was added in version 2.4*/
int mCurrentExportLayer;

/**Draw selection boxes around item*/
virtual void drawSelectionBoxes( QPainter* p );

Expand Down

0 comments on commit 0c19945

Please sign in to comment.