26#include <proj_experimental.h>
45 const QString &error )> QgsCoordinateTransformPrivate::sCoordinateOperationCreationErrorHandler =
nullptr;
55QgsCoordinateTransformPrivate::QgsCoordinateTransformPrivate()
64 : mSourceCRS( source )
65 , mDestCRS( destination )
67 if ( mSourceCRS != mDestCRS )
68 calculateTransforms( context );
74 : mSourceCRS( source )
75 , mDestCRS( destination )
76 , mSourceDatumTransform( sourceDatumTransform )
77 , mDestinationDatumTransform( destDatumTransform )
81QgsCoordinateTransformPrivate::QgsCoordinateTransformPrivate(
const QgsCoordinateTransformPrivate &other )
82 : QSharedData( other )
83 , mAvailableOpCount( other.mAvailableOpCount )
84 , mIsValid( other.mIsValid )
85 , mShortCircuit( other.mShortCircuit )
86 , mGeographicToWebMercator( other.mGeographicToWebMercator )
87 , mHasVerticalComponent( other.mHasVerticalComponent )
88 , mSourceCRS( other.mSourceCRS )
89 , mDestCRS( other.mDestCRS )
90 , mSourceDatumTransform( other.mSourceDatumTransform )
91 , mDestinationDatumTransform( other.mDestinationDatumTransform )
92 , mProjCoordinateOperation( other.mProjCoordinateOperation )
93 , mShouldReverseCoordinateOperation( other.mShouldReverseCoordinateOperation )
94 , mAllowFallbackTransforms( other.mAllowFallbackTransforms )
95 , mSourceIsDynamic( other.mSourceIsDynamic )
96 , mDestIsDynamic( other.mDestIsDynamic )
97 , mSourceCoordinateEpoch( other.mSourceCoordinateEpoch )
98 , mDestCoordinateEpoch( other.mDestCoordinateEpoch )
99 , mDefaultTime( other.mDefaultTime )
100 , mIsReversed( other.mIsReversed )
103 , mProjFallbackProjections()
109QgsCoordinateTransformPrivate::~QgsCoordinateTransformPrivate()
116bool QgsCoordinateTransformPrivate::checkValidity()
118 if ( !mSourceCRS.isValid() || !mDestCRS.isValid() )
126void QgsCoordinateTransformPrivate::invalidate()
128 mShortCircuit =
true;
130 mAvailableOpCount = -1;
133bool QgsCoordinateTransformPrivate::initialize()
136 if ( !mSourceCRS.isValid() )
144 if ( !mDestCRS.isValid() )
148 mDestCRS = mSourceCRS;
155 if ( mSourceCRS == mDestCRS )
159 mShortCircuit =
true;
163 mGeographicToWebMercator =
164 mSourceCRS.isGeographic() &&
165 mDestCRS.authid() == QLatin1String(
"EPSG:3857" );
167 mHasVerticalComponent = mSourceCRS.hasVerticalAxis() && mDestCRS.hasVerticalAxis();
169 mSourceIsDynamic = mSourceCRS.isDynamic();
170 mSourceCoordinateEpoch = mSourceCRS.coordinateEpoch();
171 mDestIsDynamic = mDestCRS.isDynamic();
172 mDestCoordinateEpoch = mDestCRS.coordinateEpoch();
179 mDefaultTime = ( mSourceIsDynamic && !std::isnan( mSourceCoordinateEpoch ) && !mDestIsDynamic )
180 ? mSourceCoordinateEpoch
181 : ( mDestIsDynamic && !std::isnan( mDestCoordinateEpoch ) && !mSourceIsDynamic )
182 ? mDestCoordinateEpoch : std::numeric_limits< double >::quiet_NaN();
184 if ( mSourceIsDynamic && mDestIsDynamic && !
qgsNanCompatibleEquals( mSourceCoordinateEpoch, mDestCoordinateEpoch ) )
187 if ( sDynamicCrsToDynamicCrsWarningHandler )
189 sDynamicCrsToDynamicCrsWarningHandler( mSourceCRS, mDestCRS );
197 ProjData res = threadLocalProjData();
199#ifdef COORDINATE_TRANSFORM_VERBOSE
207#ifdef COORDINATE_TRANSFORM_VERBOSE
210 QgsDebugMsgLevel( QStringLiteral(
"------------------------------------------------------------" ), 2 );
211 QgsDebugMsgLevel( QStringLiteral(
"The OGR Coordinate transformation for this layer was set to" ), 2 );
212 QgsLogger::debug<QgsCoordinateReferenceSystem>(
"Input", mSourceCRS, __FILE__, __FUNCTION__, __LINE__ );
213 QgsLogger::debug<QgsCoordinateReferenceSystem>(
"Output", mDestCRS, __FILE__, __FUNCTION__, __LINE__ );
214 QgsDebugMsgLevel( QStringLiteral(
"------------------------------------------------------------" ), 2 );
218 QgsDebugError( QStringLiteral(
"The OGR Coordinate transformation FAILED TO INITIALIZE!" ) );
223 QgsDebugError( QStringLiteral(
"Coordinate transformation failed to initialize!" ) );
228 mShortCircuit =
false;
236 if ( mSourceCRS.isValid() && mDestCRS.isValid() )
244 mProjCoordinateOperation.clear();
245 mShouldReverseCoordinateOperation =
false;
246 mAllowFallbackTransforms =
false;
250static void proj_collecting_logger(
void *user_data,
int ,
const char *message )
252 QStringList *dest =
reinterpret_cast< QStringList *
>( user_data );
253 dest->append( QString( message ) );
256static void proj_logger(
void *,
int level,
const char *message )
261 if ( level == PJ_LOG_ERROR )
263 const QString messageString( message );
264 if ( messageString == QLatin1String(
"push: Invalid latitude" ) )
274 else if ( level == PJ_LOG_DEBUG )
280ProjData QgsCoordinateTransformPrivate::threadLocalProjData()
285 const QMap < uintptr_t, ProjData >::const_iterator it = mProjProjections.constFind(
reinterpret_cast< uintptr_t
>( context ) );
287 if ( it != mProjProjections.constEnd() )
289 ProjData res = it.value();
297 QStringList projErrors;
298 proj_log_func( context, &projErrors, proj_collecting_logger );
303 if ( !mProjCoordinateOperation.isEmpty() )
305 transform.reset( proj_create( context, mProjCoordinateOperation.toUtf8().constData() ) );
315 proj_context_is_network_enabled( context ) &&
316 !proj_coordoperation_is_instantiable( context, transform.get() ) )
319 if ( sMissingGridUsedByContextHandler )
322 desired.
proj = mProjCoordinateOperation;
325 sMissingGridUsedByContextHandler( mSourceCRS, mDestCRS, desired );
329 const QString err = QObject::tr(
"Could not use operation specified in project between %1 and %2. (Wanted to use: %3)." ).arg( mSourceCRS.authid(),
331 mProjCoordinateOperation );
339 mIsReversed = mShouldReverseCoordinateOperation;
343 QString nonAvailableError;
346 if ( !mSourceCRS.projObject() || ! mDestCRS.projObject() )
348 proj_log_func( context,
nullptr,
nullptr );
352 PJ_OPERATION_FACTORY_CONTEXT *operationContext = proj_create_operation_factory_context( context,
nullptr );
355 proj_operation_factory_context_set_grid_availability_use( context, operationContext, PROJ_GRID_AVAILABILITY_IGNORED );
358 proj_operation_factory_context_set_spatial_criterion( context, operationContext, PROJ_SPATIAL_CRITERION_PARTIAL_INTERSECTION );
360 if ( PJ_OBJ_LIST *ops = proj_create_operations( context, mSourceCRS.projObject(), mDestCRS.projObject(), operationContext ) )
362 mAvailableOpCount = proj_list_get_count( ops );
363 if ( mAvailableOpCount < 1 )
366 const int errNo = proj_context_errno( context );
367 if ( errNo && errNo != -61 )
369 nonAvailableError = QString( proj_errno_string( errNo ) );
373 nonAvailableError = QObject::tr(
"No coordinate operations are available between these two reference systems" );
376 else if ( mAvailableOpCount == 1 )
379 transform.reset( proj_list_get( context, ops, 0 ) );
382 if ( !proj_coordoperation_is_instantiable( context, transform.get() ) )
385 for (
int j = 0; j < proj_coordoperation_get_grid_used_count( context, transform.get() ); ++j )
387 const char *shortName =
nullptr;
388 const char *fullName =
nullptr;
389 const char *packageName =
nullptr;
390 const char *url =
nullptr;
391 int directDownload = 0;
394 proj_coordoperation_get_grid_used( context, transform.get(), j, &shortName, &fullName, &packageName, &url, &directDownload, &openLicense, &isAvailable );
398 if ( sMissingRequiredGridHandler )
401 gridDetails.
shortName = QString( shortName );
402 gridDetails.
fullName = QString( fullName );
404 gridDetails.
url = QString( url );
408 sMissingRequiredGridHandler( mSourceCRS, mDestCRS, gridDetails );
412 const QString err = QObject::tr(
"Cannot create transform between %1 and %2, missing required grid %3" ).arg( mSourceCRS.authid(),
425 transform.reset( proj_normalize_for_visualization( context, transform.get() ) );
428 const QString err = QObject::tr(
"Cannot normalize transform between %1 and %2" ).arg( mSourceCRS.authid(),
439 bool missingPreferred =
false;
440 bool stillLookingForPreferred =
true;
441 for (
int i = 0; i < mAvailableOpCount; ++ i )
443 transform.reset( proj_list_get( context, ops, i ) );
444 const bool isInstantiable = transform && proj_coordoperation_is_instantiable( context, transform.get() );
445 if ( stillLookingForPreferred && transform && !isInstantiable )
449 if ( !candidate.
proj.isEmpty() )
451 preferred = candidate;
452 missingPreferred =
true;
453 stillLookingForPreferred =
false;
456 if ( transform && isInstantiable )
464 if ( transform && missingPreferred )
468 if ( sMissingPreferredGridHandler )
470 sMissingPreferredGridHandler( mSourceCRS, mDestCRS, preferred, available );
474 const QString err = QObject::tr(
"Using non-preferred coordinate operation between %1 and %2. Using %3, preferred %4." ).arg( mSourceCRS.authid(),
484 transform.reset( proj_normalize_for_visualization( context, transform.get() ) );
487 const QString err = QObject::tr(
"Cannot normalize transform between %1 and %2" ).arg( mSourceCRS.authid(),
492 proj_list_destroy( ops );
494 proj_operation_factory_context_destroy( operationContext );
497 if ( !transform && nonAvailableError.isEmpty() )
499 const int errNo = proj_context_errno( context );
500 if ( errNo && errNo != -61 )
502 nonAvailableError = QString( proj_errno_string( errNo ) );
504 else if ( !projErrors.empty() )
506 nonAvailableError = projErrors.constLast();
509 if ( nonAvailableError.isEmpty() )
511 nonAvailableError = QObject::tr(
"No coordinate operations are available between these two reference systems" );
516 nonAvailableError = nonAvailableError.remove( QStringLiteral(
"internal_proj_create_operations: " ) );
520 if ( !nonAvailableError.isEmpty() )
522 if ( sCoordinateOperationCreationErrorHandler )
524 sCoordinateOperationCreationErrorHandler( mSourceCRS, mDestCRS, nonAvailableError );
528 const QString err = QObject::tr(
"Cannot create transform between %1 and %2: %3" ).arg( mSourceCRS.authid(),
536 proj_log_func( context,
nullptr, proj_logger );
544 ProjData res = transform.release();
545 mProjProjections.insert(
reinterpret_cast< uintptr_t
>( context ), res );
549ProjData QgsCoordinateTransformPrivate::threadLocalFallbackProjData()
554 const QMap < uintptr_t, ProjData >::const_iterator it = mProjFallbackProjections.constFind(
reinterpret_cast< uintptr_t
>( context ) );
556 if ( it != mProjFallbackProjections.constEnd() )
558 ProjData res = it.value();
567 transform.reset( proj_normalize_for_visualization(
QgsProjContext::get(), transform.get() ) );
569 ProjData res = transform.release();
570 mProjFallbackProjections.insert(
reinterpret_cast< uintptr_t
>( context ), res );
576 sMissingRequiredGridHandler = handler;
581 sMissingPreferredGridHandler = handler;
586 sCoordinateOperationCreationErrorHandler = handler;
591 sMissingGridUsedByContextHandler = handler;
596 sDynamicCrsToDynamicCrsWarningHandler = handler;
599void QgsCoordinateTransformPrivate::freeProj()
602 if ( mProjProjections.isEmpty() && mProjFallbackProjections.isEmpty() )
604 QMap < uintptr_t, ProjData >::const_iterator it = mProjProjections.constBegin();
611 PJ_CONTEXT *tmpContext = proj_context_create();
612 for ( ; it != mProjProjections.constEnd(); ++it )
614 proj_assign_context( it.value(), tmpContext );
615 proj_destroy( it.value() );
618 it = mProjFallbackProjections.constBegin();
619 for ( ; it != mProjFallbackProjections.constEnd(); ++it )
621 proj_assign_context( it.value(), tmpContext );
622 proj_destroy( it.value() );
625 proj_context_destroy( tmpContext );
626 mProjProjections.clear();
627 mProjFallbackProjections.clear();
630bool QgsCoordinateTransformPrivate::removeObjectsBelongingToCurrentThread(
void *pj_context )
634 QMap < uintptr_t, ProjData >::iterator it = mProjProjections.find(
reinterpret_cast< uintptr_t
>( pj_context ) );
635 if ( it != mProjProjections.end() )
637 proj_destroy( it.value() );
638 mProjProjections.erase( it );
641 it = mProjFallbackProjections.find(
reinterpret_cast< uintptr_t
>( pj_context ) );
642 if ( it != mProjFallbackProjections.end() )
644 proj_destroy( it.value() );
645 mProjFallbackProjections.erase( it );
648 return mProjProjections.isEmpty();
@ Critical
Critical/error message.
This class represents a coordinate reference system (CRS).
Contains information about the context in which a coordinate transform is executed.
bool allowFallbackTransform(const QgsCoordinateReferenceSystem &source, const QgsCoordinateReferenceSystem &destination) const
Returns true if approximate "ballpark" transforms may be used when transforming between a source and ...
QString calculateCoordinateOperation(const QgsCoordinateReferenceSystem &source, const QgsCoordinateReferenceSystem &destination) const
Returns the Proj coordinate operation string to use when transforming from the specified source CRS t...
bool mustReverseCoordinateOperation(const QgsCoordinateReferenceSystem &source, const QgsCoordinateReferenceSystem &destination) const
Returns true if the coordinate operation returned by calculateCoordinateOperation() for the source to...
static void logMessage(const QString &message, const QString &tag=QString(), Qgis::MessageLevel level=Qgis::MessageLevel::Warning, bool notifyUser=true)
Adds a message to the log instance (and creates it if necessary).
static PJ_CONTEXT * get()
Returns a thread local instance of a proj context, safe for use in the current thread.
static QList< QgsDatumTransform::GridDetails > gridsUsed(const QString &proj)
Returns a list of grids used by the given proj string.
std::unique_ptr< PJ, ProjPJDeleter > proj_pj_unique_ptr
Scoped Proj PJ object.
The QgsReadWriteLocker class is a convenience class that simplifies locking and unlocking QReadWriteL...
#define Q_NOWARN_DEPRECATED_POP
#define Q_NOWARN_DEPRECATED_PUSH
bool qgsNanCompatibleEquals(double a, double b)
Compare two doubles, treating nan values as equal.
struct projCtx_t PJ_CONTEXT
#define QgsDebugMsgLevel(str, level)
#define QgsDebugError(str)