30 #include <QDomElement>
31 #include <QApplication>
33 #include <QStringList>
36 #if PROJ_VERSION_MAJOR>=6
51 QReadWriteLock QgsCoordinateTransform::sCacheLock;
53 bool QgsCoordinateTransform::sDisableCache =
false;
57 const QString &desiredOperation )> QgsCoordinateTransform::sFallbackOperationOccurredHandler =
nullptr;
61 d =
new QgsCoordinateTransformPrivate();
67 d =
new QgsCoordinateTransformPrivate( source, destination, mContext );
72 if ( !d->checkValidity() )
76 #if PROJ_VERSION_MAJOR>=6
77 if ( !setFromCache( d->mSourceCRS, d->mDestCRS, d->mProjCoordinateOperation, d->mAllowFallbackTransforms ) )
79 if ( !setFromCache( d->mSourceCRS, d->mDestCRS, d->mSourceDatumTransform, d->mDestinationDatumTransform ) )
91 d =
new QgsCoordinateTransformPrivate( source, destination, mContext );
97 if ( !d->checkValidity() )
101 #if PROJ_VERSION_MAJOR>=6
102 if ( !setFromCache( d->mSourceCRS, d->mDestCRS, d->mProjCoordinateOperation, d->mAllowFallbackTransforms ) )
104 if ( !setFromCache( d->mSourceCRS, d->mDestCRS, d->mSourceDatumTransform, d->mDestinationDatumTransform ) )
115 d =
new QgsCoordinateTransformPrivate( source, destination, sourceDatumTransform, destinationDatumTransform );
120 if ( !d->checkValidity() )
124 #if PROJ_VERSION_MAJOR>=6
125 if ( !setFromCache( d->mSourceCRS, d->mDestCRS, d->mProjCoordinateOperation, d->mAllowFallbackTransforms ) )
127 if ( !setFromCache( d->mSourceCRS, d->mDestCRS, d->mSourceDatumTransform, d->mDestinationDatumTransform ) )
137 : mContext( o.mContext )
139 , mHasContext( o.mHasContext )
150 mHasContext = o.mHasContext;
152 mContext = o.mContext;
153 mLastError = QString();
163 if ( !d->checkValidity() )
166 d->calculateTransforms( mContext );
168 #if PROJ_VERSION_MAJOR>=6
169 if ( !setFromCache( d->mSourceCRS, d->mDestCRS, d->mProjCoordinateOperation, d->mAllowFallbackTransforms ) )
171 if ( !setFromCache( d->mSourceCRS, d->mDestCRS, d->mSourceDatumTransform, d->mDestinationDatumTransform ) )
183 if ( !d->checkValidity() )
186 d->calculateTransforms( mContext );
188 #if PROJ_VERSION_MAJOR>=6
189 if ( !setFromCache( d->mSourceCRS, d->mDestCRS, d->mProjCoordinateOperation, d->mAllowFallbackTransforms ) )
191 if ( !setFromCache( d->mSourceCRS, d->mDestCRS, d->mSourceDatumTransform, d->mDestinationDatumTransform ) )
207 if ( !d->checkValidity() )
210 d->calculateTransforms( mContext );
212 #if PROJ_VERSION_MAJOR>=6
213 if ( !setFromCache( d->mSourceCRS, d->mDestCRS, d->mProjCoordinateOperation, d->mAllowFallbackTransforms ) )
215 if ( !setFromCache( d->mSourceCRS, d->mDestCRS, d->mSourceDatumTransform, d->mDestinationDatumTransform ) )
231 return d->mSourceCRS;
241 if ( !d->mIsValid || d->mShortCircuit )
245 double x = point.
x();
246 double y = point.
y();
279 if ( !d->mIsValid || d->mShortCircuit )
303 #ifdef COORDINATE_TRANSFORM_VERBOSE
304 QgsDebugMsg( QStringLiteral(
"Rect projection..." ) );
316 if ( !d->mIsValid || d->mShortCircuit )
337 double xd =
static_cast< double >( x ), yd =
static_cast< double >( y );
346 if ( !d->mIsValid || d->mShortCircuit )
372 if ( !d->mIsValid || d->mShortCircuit )
378 int nVertices = poly.size();
380 QVector<double> x( nVertices );
381 QVector<double> y( nVertices );
382 QVector<double> z( nVertices );
383 double *destX = x.data();
384 double *destY = y.data();
385 double *destZ = z.data();
387 const QPointF *polyData = poly.constData();
388 for (
int i = 0; i < nVertices; ++i )
390 *destX++ = polyData->x();
391 *destY++ = polyData->y();
407 QPointF *destPoint = poly.data();
408 const double *srcX = x.constData();
409 const double *srcY = y.constData();
410 for (
int i = 0; i < nVertices; ++i )
412 destPoint->rx() = *srcX++;
413 destPoint->ry() = *srcY++;
418 if ( !err.isEmpty() )
423 QVector<double> &x, QVector<double> &y, QVector<double> &z,
427 if ( !d->mIsValid || d->mShortCircuit )
430 Q_ASSERT( x.size() == y.size() );
451 QVector<float> &x, QVector<float> &y, QVector<float> &z,
454 if ( !d->mIsValid || d->mShortCircuit )
457 Q_ASSERT( x.size() == y.size() );
467 int vectorSize = x.size();
468 QVector<double> xd( x.size() );
469 QVector<double> yd( y.size() );
470 QVector<double> zd( z.size() );
472 double *destX = xd.data();
473 double *destY = yd.data();
474 double *destZ = zd.data();
476 const float *srcX = x.constData();
477 const float *srcY = y.constData();
478 const float *srcZ = z.constData();
480 for (
int i = 0; i < vectorSize; ++i )
482 *destX++ =
static_cast< double >( *srcX++ );
483 *destY++ =
static_cast< double >( *srcY++ );
484 *destZ++ =
static_cast< double >( *srcZ++ );
490 float *destFX = x.data();
491 float *destFY = y.data();
492 float *destFZ = z.data();
493 const double *srcXD = xd.constData();
494 const double *srcYD = yd.constData();
495 const double *srcZD = zd.constData();
496 for (
int i = 0; i < vectorSize; ++i )
498 *destFX++ =
static_cast< float >( *srcXD++ );
499 *destFY++ =
static_cast< float >( *srcYD++ );
500 *destFZ++ =
static_cast< float >( *srcZD++ );
518 if ( !d->mIsValid || d->mShortCircuit )
531 const int nPoints = 1000;
532 double d = std::sqrt( ( rect.
width() * rect.
height() ) / std::pow( std::sqrt(
static_cast< double >( nPoints ) ) - 1, 2.0 ) );
533 int nXPoints = std::min(
static_cast< int >( std::ceil( rect.
width() / d ) ) + 1, 1000 );
534 int nYPoints = std::min(
static_cast< int >( std::ceil( rect.
height() / d ) ) + 1, 1000 );
541 QVector<double> x( nXPoints * nYPoints );
542 QVector<double> y( nXPoints * nYPoints );
543 QVector<double> z( nXPoints * nYPoints );
545 QgsDebugMsgLevel( QStringLiteral(
"Entering transformBoundingBox..." ), 4 );
549 double dx = rect.
width() /
static_cast< double >( nXPoints - 1 );
550 double dy = rect.
height() /
static_cast< double >( nYPoints - 1 );
554 for (
int i = 0; i < nYPoints ; i++ )
560 for (
int j = 0; j < nXPoints; j++ )
562 x[( i * nXPoints ) + j] = pointX;
563 y[( i * nXPoints ) + j] = pointY;
565 z[( i * nXPoints ) + j] = 0.0;
576 transformCoords( nXPoints * nYPoints, x.data(), y.data(), z.data(), direction );
587 for (
int i = 0; i < nXPoints * nYPoints; i++ )
589 if ( !std::isfinite( x[i] ) || !std::isfinite( y[i] ) )
594 if ( handle180Crossover )
608 throw QgsCsException( QObject::tr(
"Could not transform bounding box to target CRS" ) );
611 if ( handle180Crossover )
632 if ( !d->mIsValid || d->mShortCircuit )
635 if ( !d->mSourceCRS.isValid() )
638 "The coordinates can not be reprojected. The CRS is: %1" )
639 .arg( d->mSourceCRS.toProj() ), QObject::tr(
"CRS" ) );
642 if ( !d->mDestCRS.isValid() )
645 "The coordinates can not be reprojected. The CRS is: %1" ).arg( d->mDestCRS.toProj() ), QObject::tr(
"CRS" ) );
649 std::vector< int > zNanPositions;
650 for (
int i = 0; i < numPoints; i++ )
652 if ( std::isnan( z[i] ) )
654 zNanPositions.push_back( i );
659 #if PROJ_VERSION_MAJOR>=6
660 std::vector< double > xprev( numPoints );
661 memcpy( xprev.data(), x,
sizeof(
double ) * numPoints );
662 std::vector< double > yprev( numPoints );
663 memcpy( yprev.data(), y,
sizeof(
double ) * numPoints );
664 std::vector< double > zprev( numPoints );
665 memcpy( zprev.data(), z,
sizeof(
double ) * numPoints );
668 #ifdef COORDINATE_TRANSFORM_VERBOSE
671 QgsDebugMsg( QStringLiteral(
"[[[[[[ Number of points to transform: %1 ]]]]]]" ).arg( numPoints ) );
676 QgsDebugMsgLevel( QStringLiteral(
"No QgsCoordinateTransformContext context set for transform" ), 4 );
683 ProjData projData = d->threadLocalProjData();
686 #if PROJ_VERSION_MAJOR>=6
687 proj_errno_reset( projData );
688 proj_trans_generic( projData, ( direction ==
ForwardTransform && !d->mIsReversed ) || ( direction ==
ReverseTransform && d->mIsReversed ) ? PJ_FWD : PJ_INV,
689 x,
sizeof(
double ), numPoints,
690 y,
sizeof(
double ), numPoints,
691 z,
sizeof(
double ), numPoints,
692 nullptr,
sizeof(
double ), 0 );
702 if ( numPoints == 1 )
704 projResult = proj_errno( projData );
705 actualRes = projResult;
709 actualRes = proj_errno( projData );
711 if ( actualRes == 0 )
715 if ( std::any_of( x, x + numPoints, [](
double v ) {
return std::isinf( v ); } )
716 || std::any_of( y, y + numPoints, [](
double v ) {
return std::isinf( v ); } )
717 || std::any_of( z, z + numPoints, [](
double v ) {
return std::isinf( v ); } ) )
723 bool sourceIsLatLong =
false;
724 bool destIsLatLong =
false;
726 projPJ sourceProj = projData.first;
727 projPJ destProj = projData.second;
728 sourceIsLatLong = pj_is_latlong( sourceProj );
729 destIsLatLong = pj_is_latlong( destProj );
734 for (
int i = 0; i < numPoints; ++i )
742 #if PROJ_VERSION_MAJOR<6
745 projResult = pj_transform( destProj, sourceProj, numPoints, 0, x, y, z );
749 Q_ASSERT( sourceProj );
750 Q_ASSERT( destProj );
751 projResult = pj_transform( sourceProj, destProj, numPoints, 0, x, y, z );
755 #if PROJ_VERSION_MAJOR>=6
757 mFallbackOperationOccurred =
false;
759 && ( d->mAvailableOpCount > 1 || d->mAvailableOpCount == -1 )
760 && ( d->mAllowFallbackTransforms || mBallparkTransformsAreAppropriate ) )
763 if ( PJ *
transform = d->threadLocalFallbackProjData() )
768 xprev.data(),
sizeof(
double ), numPoints,
769 yprev.data(),
sizeof(
double ), numPoints,
770 zprev.data(),
sizeof(
double ), numPoints,
771 nullptr,
sizeof(
double ), 0 );
780 if ( numPoints == 1 )
785 projResult = std::isinf( xprev[0] ) || std::isinf( yprev[0] ) || std::isinf( zprev[0] ) ? 1 : 0;
788 if ( projResult == 0 )
790 memcpy( x, xprev.data(),
sizeof(
double ) * numPoints );
791 memcpy( y, yprev.data(),
sizeof(
double ) * numPoints );
792 memcpy( z, zprev.data(),
sizeof(
double ) * numPoints );
793 mFallbackOperationOccurred =
true;
796 if ( !mBallparkTransformsAreAppropriate && !mDisableFallbackHandler && sFallbackOperationOccurredHandler )
798 sFallbackOperationOccurredHandler( d->mSourceCRS, d->mDestCRS, d->mProjCoordinateOperation );
800 const QString warning = QStringLiteral(
"A fallback coordinate operation was used between %1 and %2" ).arg( d->mSourceCRS.authid(),
801 d->mDestCRS.authid() );
802 qWarning(
"%s", warning.toLatin1().constData() );
809 for (
const int &pos : zNanPositions )
811 z[pos] = std::numeric_limits<double>::quiet_NaN();
814 if ( projResult != 0 )
819 for (
int i = 0; i < numPoints; ++i )
823 points += QStringLiteral(
"(%1, %2)\n" ).arg( x[i], 0,
'f' ).arg( y[i], 0,
'f' );
827 #if PROJ_VERSION_MAJOR>=6
828 points += QStringLiteral(
"(%1, %2)\n" ).arg( x[i], 0,
'f' ).arg( y[i], 0,
'f' );
830 points += QStringLiteral(
"(%1, %2)\n" ).arg( x[i] * RAD_TO_DEG, 0,
'f' ).arg( y[i] * RAD_TO_DEG, 0,
'f' );
835 QString dir = ( direction ==
ForwardTransform ) ? QObject::tr(
"forward transform" ) : QObject::tr(
"inverse transform" );
837 #if PROJ_VERSION_MAJOR>=6
838 QString msg = QObject::tr(
"%1 of\n"
843 projResult < 0 ? QString::fromUtf8( proj_errno_string( projResult ) ) : QObject::tr(
"Fallback transform failed" ) );
845 char *srcdef = pj_get_def( sourceProj, 0 );
846 char *dstdef = pj_get_def( destProj, 0 );
848 QString msg = QObject::tr(
"%1 of\n"
855 QString::fromUtf8( pj_strerrno( projResult ) ) );
862 if ( msg != mLastError )
864 QgsDebugMsg(
"Projection failed emitting invalid transform signal: " + msg );
872 #if PROJ_VERSION_MAJOR<6
878 for (
int i = 0; i < numPoints; ++i )
885 #ifdef COORDINATE_TRANSFORM_VERBOSE
886 QgsDebugMsg( QStringLiteral(
"[[[[[[ Projected %1, %2 to %3, %4 ]]]]]]" )
887 .arg( xorg, 0,
'g', 15 ).arg( yorg, 0,
'g', 15 )
888 .arg( *x, 0,
'g', 15 ).arg( *y, 0,
'g', 15 ) );
899 return !d->mIsValid || d->mShortCircuit;
904 return d->mProjCoordinateOperation;
909 #if PROJ_VERSION_MAJOR>=6
910 ProjData projData = d->threadLocalProjData();
911 return QgsDatumTransform::transformDetailsFromPj( projData );
920 d->mProjCoordinateOperation = operation;
921 d->mShouldReverseCoordinateOperation =
false;
927 d->mAllowFallbackTransforms = allowed;
932 return d->mAllowFallbackTransforms;
937 mBallparkTransformsAreAppropriate = appropriate;
942 mDisableFallbackHandler = disabled;
947 return mFallbackOperationOccurred;
954 proj = QApplication::applicationDirPath()
955 +
"/share/proj/" + QString( name );
959 return proj.toUtf8();
962 #if PROJ_VERSION_MAJOR>=6
968 const QString sourceKey = src.
authid().isEmpty() ?
970 const QString destKey = dest.
authid().isEmpty() ?
973 if ( sourceKey.isEmpty() || destKey.isEmpty() )
980 const QList< QgsCoordinateTransform > values = sTransforms.values( qMakePair( sourceKey, destKey ) );
981 for (
auto valIt = values.constBegin(); valIt != values.constEnd(); ++valIt )
983 if ( ( *valIt ).coordinateOperation() == coordinateOperationProj && ( *valIt ).allowFallbackTransforms() == allowFallback )
988 bool hasContext = mHasContext;
995 mHasContext = hasContext;
1009 const QString sourceKey = src.
authid().isEmpty() ?
1011 const QString destKey = dest.
authid().isEmpty() ?
1014 if ( sourceKey.isEmpty() || destKey.isEmpty() )
1018 if ( sDisableCache )
1021 const QList< QgsCoordinateTransform > values = sTransforms.values( qMakePair( src.
authid(), dest.
authid() ) );
1022 for (
auto valIt = values.constBegin(); valIt != values.constEnd(); ++valIt )
1025 if ( ( *valIt ).sourceDatumTransformId() == srcDatumTransform &&
1026 ( *valIt ).destinationDatumTransformId() == destDatumTransform )
1031 bool hasContext = mHasContext;
1038 mHasContext = hasContext;
1049 void QgsCoordinateTransform::addToCache()
1051 if ( !d->mSourceCRS.isValid() || !d->mDestCRS.isValid() )
1054 const QString sourceKey = d->mSourceCRS.authid().isEmpty() ?
1056 const QString destKey = d->mDestCRS.authid().isEmpty() ?
1059 if ( sourceKey.isEmpty() || destKey.isEmpty() )
1063 if ( sDisableCache )
1066 sTransforms.insert( qMakePair( sourceKey, destKey ), *
this );
1072 return d->mSourceDatumTransform;
1080 d->mSourceDatumTransform = dt;
1087 return d->mDestinationDatumTransform;
1095 d->mDestinationDatumTransform = dt;
1102 if ( sDisableCache )
1107 sDisableCache =
true;
1110 sTransforms.clear();
1113 #if PROJ_VERSION_MAJOR>=6
1114 void QgsCoordinateTransform::removeFromCacheObjectsBelongingToCurrentThread(
void *pj_context )
1120 if ( sDisableCache )
1125 if ( sDisableCache )
1128 for (
auto it = sTransforms.begin(); it != sTransforms.end(); )
1130 auto &v = it.value();
1131 if ( v.d->removeObjectsBelongingToCurrentThread( pj_context ) )
1132 it = sTransforms.erase( it );
1143 double distSourceUnits = std::sqrt( source1.
sqrDist( source2 ) );
1146 double distDestUnits = std::sqrt( dest1.
sqrDist( dest2 ) );
1147 return distDestUnits / distSourceUnits;
1152 QgsCoordinateTransformPrivate::setCustomMissingRequiredGridHandler( handler );
1157 QgsCoordinateTransformPrivate::setCustomMissingPreferredGridHandler( handler );
1162 QgsCoordinateTransformPrivate::setCustomCoordinateOperationCreationErrorHandler( handler );
1167 QgsCoordinateTransformPrivate::setCustomMissingGridUsedByContextHandler( handler );
1172 sFallbackOperationOccurredHandler = handler;