Robustify atlas generation

This commit is contained in:
Hugo Mercier 2012-09-27 17:13:24 +02:00
parent 5bd9c3d512
commit faa9c3342a
6 changed files with 160 additions and 99 deletions

View File

@ -608,7 +608,7 @@ void QgsComposer::on_mActionExportAsPDF_triggered()
{ {
return; return;
} }
atlasMap->setAtlasFilenamePattern( "'output_'||$id" ); atlasMap->setAtlasFilenamePattern( "'output_'||$feature" );
} }
QSettings myQSettings; QSettings myQSettings;
@ -858,7 +858,7 @@ void QgsComposer::on_mActionExportAsImage_triggered()
{ {
return; return;
} }
atlasMap->setAtlasFilenamePattern( "'output_'||$id" ); atlasMap->setAtlasFilenamePattern( "'output_'||$feature" );
} }
QSettings myQSettings; QSettings myQSettings;
@ -1054,7 +1054,7 @@ void QgsComposer::on_mActionExportAsSVG_triggered()
{ {
return; return;
} }
atlasMap->setAtlasFilenamePattern( "'output_'||$id||'_'||$page" ); atlasMap->setAtlasFilenamePattern( "'output_'||$feature" );
} }
QSettings myQSettings; QSettings myQSettings;

View File

@ -89,7 +89,9 @@ QString QgsComposerLabel::displayText() const
{ {
QString displayText = mText; QString displayText = mText;
replaceDateText( displayText ); replaceDateText( displayText );
return QgsExpression::replaceExpressionText( displayText, mExpressionFeature, mExpressionLayer, &mSubstitutions ); QMap<QString, QVariant> subs = mSubstitutions;
subs[ "$page" ] = QVariant((int)mComposition->itemPageNumber( this ) + 1);
return QgsExpression::replaceExpressionText( displayText, mExpressionFeature, mExpressionLayer, &subs );
} }
void QgsComposerLabel::replaceDateText( QString& text ) const void QgsComposerLabel::replaceDateText( QString& text ) const

View File

@ -45,7 +45,7 @@ QgsComposerMap::QgsComposerMap( QgsComposition *composition, int x, int y, int w
mBottomGridAnnotationPosition( OutsideMapFrame ), mAnnotationFrameDistance( 1.0 ), mLeftGridAnnotationDirection( Horizontal ), mRightGridAnnotationDirection( Horizontal ), mBottomGridAnnotationPosition( OutsideMapFrame ), mAnnotationFrameDistance( 1.0 ), mLeftGridAnnotationDirection( Horizontal ), mRightGridAnnotationDirection( Horizontal ),
mTopGridAnnotationDirection( Horizontal ), mBottomGridAnnotationDirection( Horizontal ), mGridFrameStyle( NoGridFrame ), mGridFrameWidth( 2.0 ), mTopGridAnnotationDirection( Horizontal ), mBottomGridAnnotationDirection( Horizontal ), mGridFrameStyle( NoGridFrame ), mGridFrameWidth( 2.0 ),
mCrossLength( 3 ), mMapCanvas( 0 ), mDrawCanvasItems( true ), mCrossLength( 3 ), mMapCanvas( 0 ), mDrawCanvasItems( true ),
mAtlasMargin( 0.10 ), mAtlasFilenamePattern("'output_'||$id"), mAtlasCoverageLayer(0) mAtlasMargin( 0.10 ), mAtlasFilenamePattern("'output_'||$feature"), mAtlasCoverageLayer(0)
{ {
mComposition = composition; mComposition = composition;
mOverviewFrameMapSymbol = 0; mOverviewFrameMapSymbol = 0;
@ -87,7 +87,7 @@ QgsComposerMap::QgsComposerMap( QgsComposition *composition )
mBottomGridAnnotationPosition( OutsideMapFrame ), mAnnotationFrameDistance( 1.0 ), mLeftGridAnnotationDirection( Horizontal ), mRightGridAnnotationDirection( Horizontal ), mBottomGridAnnotationPosition( OutsideMapFrame ), mAnnotationFrameDistance( 1.0 ), mLeftGridAnnotationDirection( Horizontal ), mRightGridAnnotationDirection( Horizontal ),
mTopGridAnnotationDirection( Horizontal ), mBottomGridAnnotationDirection( Horizontal ), mGridFrameStyle( NoGridFrame ), mGridFrameWidth( 2.0 ), mCrossLength( 3 ), mTopGridAnnotationDirection( Horizontal ), mBottomGridAnnotationDirection( Horizontal ), mGridFrameStyle( NoGridFrame ), mGridFrameWidth( 2.0 ), mCrossLength( 3 ),
mMapCanvas( 0 ), mDrawCanvasItems( true ), mMapCanvas( 0 ), mDrawCanvasItems( true ),
mAtlasMargin( 0.10 ), mAtlasFilenamePattern("'output_'||$id"), mAtlasCoverageLayer(0) mAtlasMargin( 0.10 ), mAtlasFilenamePattern("'output_'||$feature"), mAtlasCoverageLayer(0)
{ {
mOverviewFrameMapSymbol = 0; mOverviewFrameMapSymbol = 0;
createDefaultOverviewFrameSymbol(); createDefaultOverviewFrameSymbol();
@ -633,6 +633,13 @@ void QgsComposerMap::syncAtlasCoverageLayer( QString lname )
} }
} }
void QgsComposerMap::setAtlasCoverageLayer( QgsVectorLayer* map )
{
mAtlasCoverageLayer = map;
emit atlasCoverageLayerChanged( map );
}
bool QgsComposerMap::writeXML( QDomElement& elem, QDomDocument & doc ) const bool QgsComposerMap::writeXML( QDomElement& elem, QDomDocument & doc ) const
{ {
if ( elem.isNull() ) if ( elem.isNull() )

View File

@ -330,7 +330,7 @@ class CORE_EXPORT QgsComposerMap : public QgsComposerItem
void setAtlasFilenamePattern( const QString& pattern ) { mAtlasFilenamePattern = pattern; } void setAtlasFilenamePattern( const QString& pattern ) { mAtlasFilenamePattern = pattern; }
QgsVectorLayer* atlasCoverageLayer() const { return mAtlasCoverageLayer; } QgsVectorLayer* atlasCoverageLayer() const { return mAtlasCoverageLayer; }
void setAtlasCoverageLayer( QgsVectorLayer* lmap ) { mAtlasCoverageLayer = lmap; } void setAtlasCoverageLayer( QgsVectorLayer* lmap );
bool atlasSingleFile() const { return mAtlasSingleFile; } bool atlasSingleFile() const { return mAtlasSingleFile; }
void setAtlasSingleFile( bool single ) { mAtlasSingleFile = single; } void setAtlasSingleFile( bool single ) { mAtlasSingleFile = single; }
@ -338,6 +338,8 @@ class CORE_EXPORT QgsComposerMap : public QgsComposerItem
signals: signals:
void extentChanged(); void extentChanged();
void atlasCoverageLayerChanged( QgsVectorLayer* );
public slots: public slots:
/**Called if map canvas has changed*/ /**Called if map canvas has changed*/

View File

@ -74,6 +74,9 @@ QgsAtlasRendering::QgsAtlasRendering( QgsComposition* composition )
void QgsAtlasRendering::begin( const QString& filenamePattern ) void QgsAtlasRendering::begin( const QString& filenamePattern )
{ {
if ( !impl->composition || !impl->composition->atlasMap() || !impl->composition->atlasMap()->atlasCoverageLayer() )
return;
impl->filenamePattern = filenamePattern; impl->filenamePattern = filenamePattern;
QgsVectorLayer* coverage = impl->composition->atlasMap()->atlasCoverageLayer(); QgsVectorLayer* coverage = impl->composition->atlasMap()->atlasCoverageLayer();
@ -103,12 +106,7 @@ void QgsAtlasRendering::begin( const QString& filenamePattern )
} }
// select all features with all attributes // select all features with all attributes
QgsAttributeList selectedAttributes; provider->select( provider->attributeIndexes() );
for ( QgsFieldMap::const_iterator fit = fieldmap.begin(); fit != fieldmap.end(); ++fit )
{
selectedAttributes.push_back( fit.key() );
}
provider->select( selectedAttributes );
// features must be stored in a list, since modifying the layer's extent rewinds nextFeature() // features must be stored in a list, since modifying the layer's extent rewinds nextFeature()
QgsFeature feature; QgsFeature feature;
@ -134,101 +132,104 @@ void QgsAtlasRendering::begin( const QString& filenamePattern )
// special columns for expressions // special columns for expressions
QgsExpression::setSpecialColumn( "$numfeatures", QVariant( (int)impl->nFeatures ) ); QgsExpression::setSpecialColumn( "$numfeatures", QVariant( (int)impl->nFeatures ) );
QgsExpression::setSpecialColumn( "$numpages", QVariant( (int)impl->composition->numPages() ) );
} }
void QgsAtlasRendering::prepareForFeature( size_t featureI ) void QgsAtlasRendering::prepareForFeature( size_t featureI )
{ {
QgsFeature* fit = &impl->features[featureI]; if ( !impl->composition || !impl->composition->atlasMap() || !impl->composition->atlasMap()->atlasCoverageLayer() )
return;
if ( impl->filenamePattern.size() > 0 ) QgsFeature* fit = &impl->features[featureI];
{
QgsExpression::setSpecialColumn( "$feature", QVariant( (int)featureI + 1 ) ); if ( impl->filenamePattern.size() > 0 )
QVariant filenameRes = impl->filenameExpr->evaluate( &*fit ); {
if ( impl->filenameExpr->hasEvalError() )
{
throw std::runtime_error( "Filename eval error: " + impl->filenameExpr->evalErrorString().toStdString() );
}
impl->currentFilename = filenameRes.toString();
}
//
// compute the new extent
// keep the original aspect ratio
// and apply a margin
// QgsGeometry::boundingBox is expressed in the geometry"s native CRS
// They have to be transformed to the MapRenderer's one
QgsRectangle geom_rect = impl->transform.transform( fit->geometry()->boundingBox() );
double xa1 = geom_rect.xMinimum();
double xa2 = geom_rect.xMaximum();
double ya1 = geom_rect.yMinimum();
double ya2 = geom_rect.yMaximum();
QgsRectangle new_extent = geom_rect;
// restore the original extent
// (successive calls to setNewExtent tend to deform the original rectangle)
impl->composition->atlasMap()->setNewExtent( impl->origExtent );
if ( impl->composition->atlasMap()->atlasFixedScale() )
{
// only translate, keep the original scale (i.e. width x height)
double geom_center_x = (xa1 + xa2) / 2.0;
double geom_center_y = (ya1 + ya2) / 2.0;
double xx = geom_center_x - impl->origExtent.width() / 2.0;
double yy = geom_center_y - impl->origExtent.height() / 2.0;
new_extent = QgsRectangle( xx,
yy,
xx + impl->origExtent.width(),
yy + impl->origExtent.height() );
}
else
{
// auto scale
double geom_ratio = geom_rect.width() / geom_rect.height();
double map_ratio = impl->origExtent.width() / impl->origExtent.height();
// geometry height is too big
if ( geom_ratio < map_ratio )
{
new_extent = QgsRectangle( (xa1 + xa2 + map_ratio * (ya1 - ya2)) / 2.0,
ya1,
xa1 + map_ratio * (ya2 - ya1),
ya2);
}
// geometry width is too big
else if ( geom_ratio > map_ratio )
{
new_extent = QgsRectangle( xa1,
(ya1 + ya2 + (xa1 - xa2) / map_ratio) / 2.0,
xa2,
ya1 + (xa2 - xa1) / map_ratio);
}
if ( impl->composition->atlasMap()->atlasMargin() > 0.0 )
{
new_extent.scale( 1 + impl->composition->atlasMap()->atlasMargin() );
}
}
// evaluate label expressions
QList<QgsComposerLabel*> labels;
impl->composition->composerItems( labels );
QgsExpression::setSpecialColumn( "$feature", QVariant( (int)featureI + 1 ) ); QgsExpression::setSpecialColumn( "$feature", QVariant( (int)featureI + 1 ) );
QVariant filenameRes = impl->filenameExpr->evaluate( &*fit );
for ( QList<QgsComposerLabel*>::iterator lit = labels.begin(); lit != labels.end(); ++lit ) if ( impl->filenameExpr->hasEvalError() )
{ {
// build a local substitution map throw std::runtime_error( "Filename eval error: " + impl->filenameExpr->evalErrorString().toStdString() );
QMap<QString, QVariant> pageMap;
pageMap.insert( "$page", QVariant( (int)impl->composition->itemPageNumber( *lit ) + 1 ) );
(*lit)->setExpressionContext( fit, impl->composition->atlasMap()->atlasCoverageLayer(), pageMap );
} }
impl->currentFilename = filenameRes.toString();
}
//
// compute the new extent
// keep the original aspect ratio
// and apply a margin
// QgsGeometry::boundingBox is expressed in the geometry"s native CRS
// We have to transform the grometry to the destination CRS and ask for the bounding box
// Note: we cannot directly take the transformation of the bounding box, since transformations are not linear
QgsGeometry tgeom( *fit->geometry() );
tgeom.transform( impl->transform );
QgsRectangle geom_rect = tgeom.boundingBox();
// set the new extent (and render) double xa1 = geom_rect.xMinimum();
impl->composition->atlasMap()->setNewExtent( new_extent ); double xa2 = geom_rect.xMaximum();
double ya1 = geom_rect.yMinimum();
double ya2 = geom_rect.yMaximum();
QgsRectangle new_extent = geom_rect;
// restore the original extent
// (successive calls to setNewExtent tend to deform the original rectangle)
impl->composition->atlasMap()->setNewExtent( impl->origExtent );
if ( impl->composition->atlasMap()->atlasFixedScale() )
{
// only translate, keep the original scale (i.e. width x height)
double geom_center_x = (xa1 + xa2) / 2.0;
double geom_center_y = (ya1 + ya2) / 2.0;
double xx = geom_center_x - impl->origExtent.width() / 2.0;
double yy = geom_center_y - impl->origExtent.height() / 2.0;
new_extent = QgsRectangle( xx,
yy,
xx + impl->origExtent.width(),
yy + impl->origExtent.height() );
}
else
{
// auto scale
double geom_ratio = geom_rect.width() / geom_rect.height();
double map_ratio = impl->origExtent.width() / impl->origExtent.height();
// geometry height is too big
if ( geom_ratio < map_ratio )
{
new_extent = QgsRectangle( (xa1 + xa2 + map_ratio * (ya1 - ya2)) / 2.0,
ya1,
xa1 + map_ratio * (ya2 - ya1),
ya2);
}
// geometry width is too big
else if ( geom_ratio > map_ratio )
{
new_extent = QgsRectangle( xa1,
(ya1 + ya2 + (xa1 - xa2) / map_ratio) / 2.0,
xa2,
ya1 + (xa2 - xa1) / map_ratio);
}
if ( impl->composition->atlasMap()->atlasMargin() > 0.0 )
{
new_extent.scale( 1 + impl->composition->atlasMap()->atlasMargin() );
}
}
// evaluate label expressions
QList<QgsComposerLabel*> labels;
impl->composition->composerItems( labels );
QgsExpression::setSpecialColumn( "$feature", QVariant( (int)featureI + 1 ) );
for ( QList<QgsComposerLabel*>::iterator lit = labels.begin(); lit != labels.end(); ++lit )
{
(*lit)->setExpressionContext( fit, impl->composition->atlasMap()->atlasCoverageLayer() );
}
// set the new extent (and render)
impl->composition->atlasMap()->setNewExtent( new_extent );
} }
size_t QgsAtlasRendering::numFeatures() const size_t QgsAtlasRendering::numFeatures() const
@ -243,6 +244,9 @@ const QString& QgsAtlasRendering::currentFilename() const
void QgsAtlasRendering::end() void QgsAtlasRendering::end()
{ {
if ( !impl->composition || !impl->composition->atlasMap() || !impl->composition->atlasMap()->atlasCoverageLayer() )
return;
// reset label expression contexts // reset label expression contexts
QList<QgsComposerLabel*> labels; QList<QgsComposerLabel*> labels;
impl->composition->composerItems( labels ); impl->composition->composerItems( labels );
@ -350,6 +354,10 @@ void QgsComposition::setNumPages( int pages )
mPages.removeLast(); mPages.removeLast();
} }
} }
// update the corresponding variable
QgsExpression::setSpecialColumn( "$numpages", QVariant((int)numPages()) );
emit nPagesChanged(); emit nPagesChanged();
} }
@ -1700,6 +1708,8 @@ void QgsComposition::addPaperItem()
addItem( paperItem ); addItem( paperItem );
paperItem->setZValue( 0 ); paperItem->setZValue( 0 );
mPages.push_back( paperItem ); mPages.push_back( paperItem );
QgsExpression::setSpecialColumn( "$numpages", QVariant((int)mPages.size()) );
} }
void QgsComposition::removePaperItems() void QgsComposition::removePaperItems()
@ -1709,6 +1719,7 @@ void QgsComposition::removePaperItems()
delete mPages.at( i ); delete mPages.at( i );
} }
mPages.clear(); mPages.clear();
QgsExpression::setSpecialColumn( "$numpages", QVariant((int)0) );
} }
void QgsComposition::deleteAndRemoveMultiFrames() void QgsComposition::deleteAndRemoveMultiFrames()
@ -1836,3 +1847,32 @@ void QgsComposition::renderPage( QPainter* p, int page )
mPlotStyle = savedPlotStyle; mPlotStyle = savedPlotStyle;
} }
void QgsComposition::setAtlasMap( QgsComposerMap* map )
{
mAtlasMap = map;
if ( map != 0 )
{
QObject::connect( map, SIGNAL( atlasCoverageLayerChanged( QgsVectorLayer* )), this, SLOT( onAtlasCoverageChanged( QgsVectorLayer* ) ) );
}
else
{
QObject::disconnect( map, SIGNAL( atlasCoverageLayerChanged( QgsVectorLayer* )), this, SLOT( onAtlasCoverageChanged( QgsVectorLayer* ) ) );
}
}
void QgsComposition::onAtlasCoverageChanged( QgsVectorLayer* )
{
// update variables
if ( mAtlasMap != 0 && mAtlasMap->atlasCoverageLayer() != 0 )
{
QgsVectorDataProvider* provider = mAtlasMap->atlasCoverageLayer()->dataProvider();
QgsExpression::setSpecialColumn( "$numfeatures", QVariant( (int)provider->featureCount() ) );
}
else
{
QgsExpression::setSpecialColumn( "$numfeatures", QVariant( (int)0 ) );
}
//
QgsExpression::setSpecialColumn( "$numpages", QVariant( (int)numPages() ) );
}

View File

@ -48,6 +48,7 @@ class QgsComposerShape;
class QgsComposerAttributeTable; class QgsComposerAttributeTable;
class QgsComposerMultiFrame; class QgsComposerMultiFrame;
class QgsComposerMultiFrameCommand; class QgsComposerMultiFrameCommand;
class QgsVectorLayer;
/** \ingroup MapComposer /** \ingroup MapComposer
* Class used to render an Atlas, iterating over geometry features. * Class used to render an Atlas, iterating over geometry features.
@ -59,12 +60,18 @@ class QgsAtlasRendering
public: public:
QgsAtlasRendering( QgsComposition* composition ); QgsAtlasRendering( QgsComposition* composition );
/** Begins the rendering. Sets an optional output filename pattern */
void begin( const QString& filenamePattern = "" ); void begin( const QString& filenamePattern = "" );
/** Ends the rendering. Restores original extent*/
void end(); void end();
/** Returns the number of features in the coverage layer */
size_t numFeatures() const; size_t numFeatures() const;
/** Prepare the atlas map for the given feature. Sets the extent and context variables */
void prepareForFeature( size_t i ); void prepareForFeature( size_t i );
/** Returns the current filename. Must be called after prepareForFeature( i ) */
const QString& currentFilename() const; const QString& currentFilename() const;
private: private:
@ -194,7 +201,7 @@ class CORE_EXPORT QgsComposition: public QGraphicsScene
QgsMapRenderer* mapRenderer() {return mMapRenderer;} QgsMapRenderer* mapRenderer() {return mMapRenderer;}
QgsComposerMap* atlasMap() { return mAtlasMap; } QgsComposerMap* atlasMap() { return mAtlasMap; }
void setAtlasMap( QgsComposerMap* map ) { mAtlasMap = map; } void setAtlasMap( QgsComposerMap* map );
QgsComposition::PlotStyle plotStyle() const {return mPlotStyle;} QgsComposition::PlotStyle plotStyle() const {return mPlotStyle;}
void setPlotStyle( QgsComposition::PlotStyle style ) {mPlotStyle = style;} void setPlotStyle( QgsComposition::PlotStyle style ) {mPlotStyle = style;}
@ -333,6 +340,9 @@ class CORE_EXPORT QgsComposition: public QGraphicsScene
/**Casts object to the proper subclass type and calls corresponding itemAdded signal*/ /**Casts object to the proper subclass type and calls corresponding itemAdded signal*/
void sendItemAddedSignal( QgsComposerItem* item ); void sendItemAddedSignal( QgsComposerItem* item );
private slots:
void onAtlasCoverageChanged( QgsVectorLayer* );
private: private:
/**Pointer to map renderer of QGIS main map*/ /**Pointer to map renderer of QGIS main map*/
QgsMapRenderer* mMapRenderer; QgsMapRenderer* mMapRenderer;