Skip to content

Commit

Permalink
Merge pull request #5789 from nyalldawson/render_jobs
Browse files Browse the repository at this point in the history
[Rendering] Only render in preview jobs layers that are fast enough
  • Loading branch information
nyalldawson committed Dec 5, 2017
2 parents b26c14e + 38cca6c commit b0a36e9
Show file tree
Hide file tree
Showing 12 changed files with 128 additions and 12 deletions.
2 changes: 2 additions & 0 deletions python/core/qgis.sip
Original file line number Diff line number Diff line change
Expand Up @@ -241,6 +241,8 @@ const double DEFAULT_LINE_WIDTH;
const double DEFAULT_SEGMENT_EPSILON;




typedef unsigned long long qgssize;


Expand Down
1 change: 1 addition & 0 deletions python/core/qgsdataprovider.sip
Original file line number Diff line number Diff line change
Expand Up @@ -380,6 +380,7 @@ Current time stamp of data source
.. versionadded:: 3.0
%End


signals:

void fullExtentCalculated();
Expand Down
5 changes: 4 additions & 1 deletion python/core/qgsmaprendererjob.sip
Original file line number Diff line number Diff line change
Expand Up @@ -137,10 +137,12 @@ List of errors that happened during the rendering job - available when the rende

int renderingTime() const;
%Docstring
Find out how long it took to finish the job (in milliseconds)
Returns the total time it took to finish the job (in milliseconds).
.. seealso:: perLayerRenderingTime()
:rtype: int
%End


const QgsMapSettings &mapSettings() const;
%Docstring
Return map settings with which this job was started.
Expand Down Expand Up @@ -179,6 +181,7 @@ emitted when asynchronous rendering is finished (or canceled).




};


Expand Down
12 changes: 12 additions & 0 deletions src/core/qgis.h
Original file line number Diff line number Diff line change
Expand Up @@ -440,6 +440,18 @@ const double DEFAULT_LINE_WIDTH = 0.26;
//! Default snapping tolerance for segments
const double DEFAULT_SEGMENT_EPSILON = 1e-8;

///@cond PRIVATE
#ifndef SIP_RUN

//! Delay between the scheduling of 2 preview jobs
const int PREVIEW_JOB_DELAY_MS = 250;

//! Maximum rendering time for a layer of a preview job
const int MAXIMUM_LAYER_PREVIEW_TIME_MS = 250;
#endif

///@endcond

typedef QMap<QString, QString> QgsStringMap SIP_SKIP;

/**
Expand Down
4 changes: 4 additions & 0 deletions src/core/qgsdataprovider.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -42,3 +42,7 @@ void QgsDataProvider::setListening( bool isListening )
Q_UNUSED( isListening );
}

bool QgsDataProvider::renderInPreview( const PreviewContext &context )
{
return context.lastRenderingTimeMs <= context.maxRenderingTimeMs;
}
31 changes: 31 additions & 0 deletions src/core/qgsdataprovider.h
Original file line number Diff line number Diff line change
Expand Up @@ -461,6 +461,37 @@ class CORE_EXPORT QgsDataProvider : public QObject
*/
virtual void setListening( bool isListening );

#ifndef SIP_RUN

/**
* Stores settings related to the context in which a preview job runs.
* \note Not available in Python bindings
* \since QGIS 3.0
*/
struct PreviewContext
{
//! Previous rendering time for the layer, in ms
double lastRenderingTimeMs = -1;

//! Default maximum allowable render time, in ms
double maxRenderingTimeMs = MAXIMUM_LAYER_PREVIEW_TIME_MS;
};
#endif

/**
* Returns whether the layer must be rendered in preview jobs.
*
* The \a context argument gives useful information which can be used
* to determine whether the layer should be rendered or not.
*
* The base implementation returns true if lastRenderingTimeMs <= maxRenderingTimeMs.
*
* \since QGIS 3.0
*
* \note not available in Python bindings
*/
virtual bool renderInPreview( const QgsDataProvider::PreviewContext &context ); // SIP_SKIP

signals:

/**
Expand Down
15 changes: 14 additions & 1 deletion src/core/qgsmaprendererjob.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -64,6 +64,17 @@ void QgsMapRendererJob::setCache( QgsMapRendererCache *cache )
mCache = cache;
}

QHash<QgsMapLayer *, int> QgsMapRendererJob::perLayerRenderingTime() const
{
QHash<QgsMapLayer *, int> result;
for ( auto it = mPerLayerRenderingTime.constBegin(); it != mPerLayerRenderingTime.constEnd(); ++it )
{
if ( it.key() )
result.insert( it.key(), it.value() );
}
return result;
}

const QgsMapSettings &QgsMapRendererJob::mapSettings() const
{
return mSettings;
Expand Down Expand Up @@ -412,8 +423,10 @@ void QgsMapRendererJob::cleanupJobs( LayerRenderJobs &jobs )
delete job.renderer;
job.renderer = nullptr;
}
}

if ( job.layer )
mPerLayerRenderingTime.insert( job.layer, job.renderingTime );
}

jobs.clear();
}
Expand Down
15 changes: 14 additions & 1 deletion src/core/qgsmaprendererjob.h
Original file line number Diff line number Diff line change
Expand Up @@ -199,9 +199,19 @@ class CORE_EXPORT QgsMapRendererJob : public QObject
*/
void setCache( QgsMapRendererCache *cache );

//! Find out how long it took to finish the job (in milliseconds)
/**
* Returns the total time it took to finish the job (in milliseconds).
* \see perLayerRenderingTime()
*/
int renderingTime() const { return mRenderingTime; }

/**
* Returns the render time (in ms) per layer.
* \note Not available in Python bindings.
* \since QGIS 3.0
*/
QHash< QgsMapLayer *, int > perLayerRenderingTime() const SIP_SKIP;

/**
* Return map settings with which this job was started.
* \returns A QgsMapSettings instance with render settings
Expand Down Expand Up @@ -239,6 +249,9 @@ class CORE_EXPORT QgsMapRendererJob : public QObject

int mRenderingTime = 0;

//! Render time (in ms) per layer, by layer ID
QHash< QgsWeakMapLayerPointer, int > mPerLayerRenderingTime;

/**
* Prepares the cache for storing the result of labeling. Returns false if
* the render cannot use cached labels and should not cache the result.
Expand Down
41 changes: 34 additions & 7 deletions src/gui/qgsmapcanvas.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -66,7 +66,6 @@ email : sherman at mrcc.com
#include "qgsmapthemecollection.h"
#include <cmath>


/**
* \ingroup gui
* Deprecated to be deleted, stuff from here should be moved elsewhere.
Expand Down Expand Up @@ -612,6 +611,13 @@ void QgsMapCanvas::rendererJobFinished()
p.end();

mMap->setContent( img, imageRect( img, mSettings ) );

mLastLayerRenderTime.clear();
const auto times = mJob->perLayerRenderingTime();
for ( auto it = times.constBegin(); it != times.constEnd(); ++it )
{
mLastLayerRenderTime.insert( it.key()->id(), it.value() );
}
if ( mUsePreviewJobs )
startPreviewJobs();
}
Expand All @@ -633,6 +639,13 @@ void QgsMapCanvas::previewJobFinished()
{
mMap->addPreviewImage( job->renderedImage(), job->mapSettings().extent() );
mPreviewJobs.removeAll( job );

int number = job->property( "number" ).toInt();
if ( number < 8 )
{
startPreviewJob( number + 1 );
}

delete job;
}
}
Expand Down Expand Up @@ -2269,15 +2282,29 @@ void QgsMapCanvas::startPreviewJob( int number )
jobSettings.setFlag( QgsMapSettings::DrawLabeling, false );
jobSettings.setFlag( QgsMapSettings::RenderPreviewJob, true );

// truncate preview layers to fast layers
const QList<QgsMapLayer *> layers = jobSettings.layers();
QList< QgsMapLayer * > previewLayers;
QgsDataProvider::PreviewContext context;
context.maxRenderingTimeMs = MAXIMUM_LAYER_PREVIEW_TIME_MS;
for ( QgsMapLayer *layer : layers )
{
context.lastRenderingTimeMs = mLastLayerRenderTime.value( layer->id(), 0 );
if ( !layer->dataProvider()->renderInPreview( context ) )
{
QgsDebugMsgLevel( QString( "Layer %1 not rendered because it does not match the renderInPreview criterion %2" ).arg( layer->id() ).arg( mLastLayerRenderTime.value( layer->id() ) ), 3 );
continue;
}

previewLayers << layer;
}
jobSettings.setLayers( previewLayers );

QgsMapRendererQImageJob *job = new QgsMapRendererSequentialJob( jobSettings );
job->setProperty( "number", number );
mPreviewJobs.append( job );
connect( job, &QgsMapRendererJob::finished, this, &QgsMapCanvas::previewJobFinished );
job->start();

if ( number < 8 )
{
schedulePreviewJob( number + 1 );
}
}

void QgsMapCanvas::stopPreviewJobs()
Expand All @@ -2299,7 +2326,7 @@ void QgsMapCanvas::stopPreviewJobs()
void QgsMapCanvas::schedulePreviewJob( int number )
{
mPreviewTimer.setSingleShot( true );
mPreviewTimer.setInterval( 250 );
mPreviewTimer.setInterval( PREVIEW_JOB_DELAY_MS );
disconnect( mPreviewTimerConnection );
mPreviewTimerConnection = connect( &mPreviewTimer, &QTimer::timeout, this, [ = ]()
{
Expand Down
2 changes: 2 additions & 0 deletions src/gui/qgsmapcanvas.h
Original file line number Diff line number Diff line change
Expand Up @@ -1000,6 +1000,8 @@ class GUI_EXPORT QgsMapCanvas : public QGraphicsView

bool mUsePreviewJobs = false;

QHash< QString, int > mLastLayerRenderTime;

/**
* Force a resize of the map canvas item
* \since QGIS 2.16
Expand Down
11 changes: 9 additions & 2 deletions src/providers/wms/qgswmsprovider.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -3199,18 +3199,25 @@ QString QgsWmsProvider::lastErrorFormat()
QString QgsWmsProvider::name() const
{
return WMS_KEY;
} // QgsWmsProvider::name()
}


QString QgsWmsProvider::description() const
{
return WMS_DESCRIPTION;
} // QgsWmsProvider::description()
}

void QgsWmsProvider::reloadData()
{
}

bool QgsWmsProvider::renderInPreview( const QgsDataProvider::PreviewContext &context )
{
if ( mSettings.mTiled || mSettings.mXyz )
return true;

return QgsRasterDataProvider::renderInPreview( context );
}

QVector<QgsWmsSupportedFormat> QgsWmsProvider::supportedFormats()
{
Expand Down
1 change: 1 addition & 0 deletions src/providers/wms/qgswmsprovider.h
Original file line number Diff line number Diff line change
Expand Up @@ -208,6 +208,7 @@ class QgsWmsProvider : public QgsRasterDataProvider
QString name() const override;
QString description() const override;
virtual void reloadData() override;
bool renderInPreview( const QgsDataProvider::PreviewContext &context ) override;

static QVector<QgsWmsSupportedFormat> supportedFormats();

Expand Down

0 comments on commit b0a36e9

Please sign in to comment.