mirror of
https://github.com/qgis/QGIS.git
synced 2025-04-04 00:06:15 -04:00
[Geometry checker] Adapt for API changes
This commit is contained in:
parent
b52b2c51e4
commit
1642eb1601
src/plugins/geometry_checker
checks
qgsgeometryanglecheck.cppqgsgeometryareacheck.cppqgsgeometrycheck.cppqgsgeometrycheck.hqgsgeometrycontainedcheck.cppqgsgeometrycontainedcheck.hqgsgeometrydegeneratepolygoncheck.cppqgsgeometryduplicatecheck.cppqgsgeometryduplicatecheck.hqgsgeometryduplicatenodescheck.cppqgsgeometrygapcheck.cppqgsgeometryholecheck.cppqgsgeometrymultipartcheck.cppqgsgeometryoverlapcheck.cppqgsgeometryoverlapcheck.hqgsgeometrysegmentlengthcheck.cppqgsgeometryselfcontactcheck.cppqgsgeometryselfintersectioncheck.cppqgsgeometryselfintersectioncheck.hqgsgeometrytypecheck.cppqgsgeometrytypecheck.h
ui
utils
@ -51,11 +51,13 @@ void QgsGeometryAngleCheck::collectErrors( QList<QgsGeometryCheckError *> &error
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
double angle = std::acos( v21 * v23 ) / M_PI * 180.0;
|
double angle = qAcos( v21 * v23 ) / M_PI * 180.0;
|
||||||
if ( angle < mMinAngle )
|
if ( angle < mMinAngle )
|
||||||
{
|
{
|
||||||
QgsAbstractGeometry *part = QgsGeometryCheckerUtils::getGeomPart( geom, iPart )->clone();
|
QgsAbstractGeometry *part = QgsGeometryCheckerUtils::getGeomPart( geom, iPart )->clone();
|
||||||
errors.append( new QgsGeometryCheckError( this, layerFeature.layer().id(), layerFeature.feature().id(), part, p2, QgsVertexId( iPart, iRing, iVert ), angle ) );
|
part->transform( layerFeature.mapToLayerTransform(), QgsCoordinateTransform::ReverseTransform );
|
||||||
|
QgsPointXY pos = layerFeature.mapToLayerTransform().transform( p2, QgsCoordinateTransform::ReverseTransform );
|
||||||
|
errors.append( new QgsGeometryCheckError( this, layerFeature.layer().id(), layerFeature.feature().id(), part, pos, QgsVertexId( iPart, iRing, iVert ), angle ) );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -44,7 +44,10 @@ void QgsGeometryAreaCheck::collectErrors( QList<QgsGeometryCheckError *> &errors
|
|||||||
double value;
|
double value;
|
||||||
if ( checkThreshold( mapToLayerUnits, geom, value ) )
|
if ( checkThreshold( mapToLayerUnits, geom, value ) )
|
||||||
{
|
{
|
||||||
errors.append( new QgsGeometryCheckError( this, layerFeature.layer().id(), layerFeature.feature().id(), geom->clone(), geom->centroid(), QgsVertexId( 0 ), value / ( mapToLayerUnits * mapToLayerUnits ), QgsGeometryCheckError::ValueArea ) );
|
QgsAbstractGeometry *g = geom->clone();
|
||||||
|
g->transform( layerFeature.mapToLayerTransform(), QgsCoordinateTransform::ReverseTransform );
|
||||||
|
QgsPoint pos = g->centroid();
|
||||||
|
errors.append( new QgsGeometryCheckError( this, layerFeature.layer().id(), layerFeature.feature().id(), g, pos, QgsVertexId( 0 ), value / ( mapToLayerUnits * mapToLayerUnits ), QgsGeometryCheckError::ValueArea ) );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -13,7 +13,6 @@
|
|||||||
* *
|
* *
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
|
|
||||||
#include "qgscrscache.h"
|
|
||||||
#include "qgsgeometrycollection.h"
|
#include "qgsgeometrycollection.h"
|
||||||
#include "qgscurvepolygon.h"
|
#include "qgscurvepolygon.h"
|
||||||
#include "qgsgeometrycheck.h"
|
#include "qgsgeometrycheck.h"
|
||||||
@ -30,7 +29,7 @@ QgsGeometryCheckerContext::QgsGeometryCheckerContext( int _precision, const QStr
|
|||||||
|
|
||||||
QgsGeometryCheckError::QgsGeometryCheckError( const QgsGeometryCheck *check, const QString &layerId,
|
QgsGeometryCheckError::QgsGeometryCheckError( const QgsGeometryCheck *check, const QString &layerId,
|
||||||
QgsFeatureId featureId, QgsAbstractGeometry *geometry,
|
QgsFeatureId featureId, QgsAbstractGeometry *geometry,
|
||||||
const QgsPoint &errorLocation,
|
const QgsPointXY &errorLocation,
|
||||||
QgsVertexId vidx,
|
QgsVertexId vidx,
|
||||||
const QVariant &value, ValueType valueType )
|
const QVariant &value, ValueType valueType )
|
||||||
: mCheck( check )
|
: mCheck( check )
|
||||||
@ -47,9 +46,7 @@ QgsGeometryCheckError::QgsGeometryCheckError( const QgsGeometryCheck *check, con
|
|||||||
|
|
||||||
QgsRectangle QgsGeometryCheckError::affectedAreaBBox() const
|
QgsRectangle QgsGeometryCheckError::affectedAreaBBox() const
|
||||||
{
|
{
|
||||||
QString srcCrs = mCheck->getContext()->featurePools[ layerId() ]->getLayer()->crs().authid();
|
return mGeometry->boundingBox();
|
||||||
QgsCoordinateTransform t = QgsCoordinateTransformCache::instance()->transform( srcCrs, mCheck->getContext()->mapCrs );
|
|
||||||
return t.transformBoundingBox( mGeometry->boundingBox() );
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool QgsGeometryCheckError::handleChanges( const QgsGeometryCheck::Changes &changes )
|
bool QgsGeometryCheckError::handleChanges( const QgsGeometryCheck::Changes &changes )
|
||||||
|
@ -99,7 +99,7 @@ class QgsGeometryCheckError
|
|||||||
const QString &layerId,
|
const QString &layerId,
|
||||||
QgsFeatureId featureId,
|
QgsFeatureId featureId,
|
||||||
QgsAbstractGeometry *geometry,
|
QgsAbstractGeometry *geometry,
|
||||||
const QgsPoint &errorLocation,
|
const QgsPointXY &errorLocation,
|
||||||
QgsVertexId vidx = QgsVertexId(),
|
QgsVertexId vidx = QgsVertexId(),
|
||||||
const QVariant &value = QVariant(),
|
const QVariant &value = QVariant(),
|
||||||
ValueType valueType = ValueOther );
|
ValueType valueType = ValueOther );
|
||||||
@ -113,11 +113,13 @@ class QgsGeometryCheckError
|
|||||||
const QgsGeometryCheck *check() const { return mCheck; }
|
const QgsGeometryCheck *check() const { return mCheck; }
|
||||||
const QString &layerId() const { return mLayerId; }
|
const QString &layerId() const { return mLayerId; }
|
||||||
QgsFeatureId featureId() const { return mFeatureId; }
|
QgsFeatureId featureId() const { return mFeatureId; }
|
||||||
QgsAbstractGeometry *geometry() const { return mGeometry; }
|
// In map units
|
||||||
|
const QgsAbstractGeometry *geometry() const { return mGeometry; }
|
||||||
// In map units
|
// In map units
|
||||||
virtual QgsRectangle affectedAreaBBox() const;
|
virtual QgsRectangle affectedAreaBBox() const;
|
||||||
virtual QString description() const { return mCheck->errorDescription(); }
|
virtual QString description() const { return mCheck->errorDescription(); }
|
||||||
const QgsPoint &location() const { return mErrorLocation; }
|
// In map units
|
||||||
|
const QgsPointXY &location() const { return mErrorLocation; }
|
||||||
// Lengths, areas in map units
|
// Lengths, areas in map units
|
||||||
QVariant value() const { return mValue; }
|
QVariant value() const { return mValue; }
|
||||||
ValueType valueType() const { return mValueType; }
|
ValueType valueType() const { return mValueType; }
|
||||||
@ -165,7 +167,7 @@ class QgsGeometryCheckError
|
|||||||
QString mLayerId;
|
QString mLayerId;
|
||||||
QgsFeatureId mFeatureId;
|
QgsFeatureId mFeatureId;
|
||||||
QgsAbstractGeometry *mGeometry;
|
QgsAbstractGeometry *mGeometry;
|
||||||
QgsPoint mErrorLocation;
|
QgsPointXY mErrorLocation;
|
||||||
QgsVertexId mVidx;
|
QgsVertexId mVidx;
|
||||||
QVariant mValue;
|
QVariant mValue;
|
||||||
ValueType mValueType;
|
ValueType mValueType;
|
||||||
|
@ -13,7 +13,6 @@
|
|||||||
* *
|
* *
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
|
|
||||||
#include "qgscrscache.h"
|
|
||||||
#include "qgsgeometryengine.h"
|
#include "qgsgeometryengine.h"
|
||||||
#include "qgsgeometrycontainedcheck.h"
|
#include "qgsgeometrycontainedcheck.h"
|
||||||
#include "../utils/qgsfeaturepool.h"
|
#include "../utils/qgsfeaturepool.h"
|
||||||
@ -32,7 +31,9 @@ void QgsGeometryContainedCheck::collectErrors( QList<QgsGeometryCheckError *> &e
|
|||||||
QString errMsg;
|
QString errMsg;
|
||||||
if ( geomEngineA->within( *layerFeatureB.geometry(), &errMsg ) )
|
if ( geomEngineA->within( *layerFeatureB.geometry(), &errMsg ) )
|
||||||
{
|
{
|
||||||
errors.append( new QgsGeometryContainedCheckError( this, layerFeatureA.layer().id(), layerFeatureA.feature().id(), layerFeatureA.geometry()->clone(), layerFeatureA.geometry()->centroid(), qMakePair( layerFeatureB.layer().id(), layerFeatureB.feature().id() ) ) );
|
QgsAbstractGeometry *g = layerFeatureA.geometry()->clone();
|
||||||
|
QgsPoint pos = g->centroid();
|
||||||
|
errors.append( new QgsGeometryContainedCheckError( this, layerFeatureA.layer().id(), layerFeatureA.feature().id(), g, pos, qMakePair( layerFeatureB.layer().id(), layerFeatureB.feature().id() ) ) );
|
||||||
}
|
}
|
||||||
else if ( !errMsg.isEmpty() )
|
else if ( !errMsg.isEmpty() )
|
||||||
{
|
{
|
||||||
@ -56,16 +57,14 @@ void QgsGeometryContainedCheck::fixError( QgsGeometryCheckError *error, int meth
|
|||||||
error->setObsolete();
|
error->setObsolete();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
QgsCoordinateTransform crstA = QgsCoordinateTransformCache::instance()->transform( featurePoolA->getLayer()->crs().authid(), mContext->mapCrs );
|
|
||||||
QgsCoordinateTransform crstB = QgsCoordinateTransformCache::instance()->transform( featurePoolB->getLayer()->crs().authid(), mContext->mapCrs );
|
|
||||||
|
|
||||||
// Check if error still applies
|
// Check if error still applies
|
||||||
QgsAbstractGeometry *featureGeomA = featureA.geometry().geometry()->clone();
|
QgsAbstractGeometry *featureGeomA = featureA.geometry().geometry()->clone();
|
||||||
featureGeomA->transform( crstA );
|
featureGeomA->transform( featurePoolA->getMapToLayerTransform(), QgsCoordinateTransform::ReverseTransform );
|
||||||
QSharedPointer<QgsGeometryEngine> geomEngineA = QgsGeometryCheckerUtils::createGeomEngine( featureGeomA, mContext->tolerance );
|
QSharedPointer<QgsGeometryEngine> geomEngineA = QgsGeometryCheckerUtils::createGeomEngine( featureGeomA, mContext->tolerance );
|
||||||
|
|
||||||
QgsAbstractGeometry *featureGeomB = featureB.geometry().geometry()->clone();
|
QgsAbstractGeometry *featureGeomB = featureB.geometry().geometry()->clone();
|
||||||
featureGeomB->transform( crstB );
|
featureGeomB->transform( featurePoolB->getMapToLayerTransform(), QgsCoordinateTransform::ReverseTransform );
|
||||||
|
|
||||||
bool within = geomEngineA->within( *featureGeomB );
|
bool within = geomEngineA->within( *featureGeomB );
|
||||||
delete featureGeomA;
|
delete featureGeomA;
|
||||||
|
@ -25,7 +25,7 @@ class QgsGeometryContainedCheckError : public QgsGeometryCheckError
|
|||||||
const QString &layerId,
|
const QString &layerId,
|
||||||
QgsFeatureId featureId,
|
QgsFeatureId featureId,
|
||||||
QgsAbstractGeometry *geometry,
|
QgsAbstractGeometry *geometry,
|
||||||
const QgsPoint &errorLocation,
|
const QgsPointXY &errorLocation,
|
||||||
const QPair<QString, QgsFeatureId> &containingFeature
|
const QPair<QString, QgsFeatureId> &containingFeature
|
||||||
)
|
)
|
||||||
: QgsGeometryCheckError( check, layerId, featureId, geometry, errorLocation, QgsVertexId(), QString( "%1:%2" ).arg( containingFeature.first ).arg( containingFeature.second ), ValueOther )
|
: QgsGeometryCheckError( check, layerId, featureId, geometry, errorLocation, QgsVertexId(), QString( "%1:%2" ).arg( containingFeature.first ).arg( containingFeature.second ), ValueOther )
|
||||||
|
@ -29,7 +29,10 @@ void QgsGeometryDegeneratePolygonCheck::collectErrors( QList<QgsGeometryCheckErr
|
|||||||
{
|
{
|
||||||
if ( QgsGeometryCheckerUtils::polyLineSize( geom, iPart, iRing ) < 3 )
|
if ( QgsGeometryCheckerUtils::polyLineSize( geom, iPart, iRing ) < 3 )
|
||||||
{
|
{
|
||||||
errors.append( new QgsGeometryCheckError( this, layerFeature.layer().id(), layerFeature.feature().id(), geom->clone(), geom->vertexAt( QgsVertexId( iPart, iRing, 0 ) ), QgsVertexId( iPart, iRing ) ) );
|
QgsAbstractGeometry *g = geom->clone();
|
||||||
|
g->transform( layerFeature.mapToLayerTransform(), QgsCoordinateTransform::ReverseTransform );
|
||||||
|
QgsPointXY pos = layerFeature.mapToLayerTransform().transform( geom->vertexAt( QgsVertexId( iPart, iRing, 0 ) ), QgsCoordinateTransform::ReverseTransform );
|
||||||
|
errors.append( new QgsGeometryCheckError( this, layerFeature.layer().id(), layerFeature.feature().id(), g, pos, QgsVertexId( iPart, iRing ) ) );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -13,7 +13,6 @@
|
|||||||
* *
|
* *
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
|
|
||||||
#include "qgscrscache.h"
|
|
||||||
#include "qgsgeometryengine.h"
|
#include "qgsgeometryengine.h"
|
||||||
#include "qgsgeometryduplicatecheck.h"
|
#include "qgsgeometryduplicatecheck.h"
|
||||||
#include "qgsspatialindex.h"
|
#include "qgsspatialindex.h"
|
||||||
@ -56,7 +55,9 @@ void QgsGeometryDuplicateCheck::collectErrors( QList<QgsGeometryCheckError *> &e
|
|||||||
}
|
}
|
||||||
if ( !duplicates.isEmpty() )
|
if ( !duplicates.isEmpty() )
|
||||||
{
|
{
|
||||||
errors.append( new QgsGeometryDuplicateCheckError( this, layerFeatureA.layer().id(), layerFeatureA.feature().id(), layerFeatureA.geometry()->clone(), layerFeatureA.geometry()->centroid(), duplicates ) );
|
QgsAbstractGeometry *g = layerFeatureA.geometry()->clone();
|
||||||
|
QgsPoint pos = g->centroid();
|
||||||
|
errors.append( new QgsGeometryDuplicateCheckError( this, layerFeatureA.layer().id(), layerFeatureA.feature().id(), g, pos, duplicates ) );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -77,16 +78,14 @@ void QgsGeometryDuplicateCheck::fixError( QgsGeometryCheckError *error, int meth
|
|||||||
}
|
}
|
||||||
else if ( method == RemoveDuplicates )
|
else if ( method == RemoveDuplicates )
|
||||||
{
|
{
|
||||||
QgsCoordinateTransform crstA = QgsCoordinateTransformCache::instance()->transform( featurePoolA->getLayer()->crs().authid(), mContext->mapCrs );
|
|
||||||
QgsAbstractGeometry *featureGeomA = featureA.geometry().geometry()->clone();
|
QgsAbstractGeometry *featureGeomA = featureA.geometry().geometry()->clone();
|
||||||
featureGeomA->transform( crstA );
|
featureGeomA->transform( featurePoolA->getMapToLayerTransform(), QgsCoordinateTransform::ReverseTransform );
|
||||||
QSharedPointer<QgsGeometryEngine> geomEngine = QgsGeometryCheckerUtils::createGeomEngine( featureGeomA, mContext->tolerance );
|
QSharedPointer<QgsGeometryEngine> geomEngine = QgsGeometryCheckerUtils::createGeomEngine( featureGeomA, mContext->tolerance );
|
||||||
|
|
||||||
QgsGeometryDuplicateCheckError *duplicateError = static_cast<QgsGeometryDuplicateCheckError *>( error );
|
QgsGeometryDuplicateCheckError *duplicateError = static_cast<QgsGeometryDuplicateCheckError *>( error );
|
||||||
for ( const QString &layerIdB : duplicateError->duplicates().keys() )
|
for ( const QString &layerIdB : duplicateError->duplicates().keys() )
|
||||||
{
|
{
|
||||||
QgsFeaturePool *featurePoolB = mContext->featurePools[ layerIdB ];
|
QgsFeaturePool *featurePoolB = mContext->featurePools[ layerIdB ];
|
||||||
QgsCoordinateTransform crstB = QgsCoordinateTransformCache::instance()->transform( featurePoolB->getLayer()->crs().authid(), mContext->mapCrs );
|
|
||||||
for ( QgsFeatureId idB : duplicateError->duplicates()[layerIdB] )
|
for ( QgsFeatureId idB : duplicateError->duplicates()[layerIdB] )
|
||||||
{
|
{
|
||||||
QgsFeature featureB;
|
QgsFeature featureB;
|
||||||
@ -95,7 +94,7 @@ void QgsGeometryDuplicateCheck::fixError( QgsGeometryCheckError *error, int meth
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
QgsAbstractGeometry *featureGeomB = featureB.geometry().geometry()->clone();
|
QgsAbstractGeometry *featureGeomB = featureB.geometry().geometry()->clone();
|
||||||
featureGeomB->transform( crstB );
|
featureGeomB->transform( featurePoolB->getMapToLayerTransform(), QgsCoordinateTransform::ReverseTransform );
|
||||||
QgsAbstractGeometry *diffGeom = geomEngine->symDifference( *featureGeomB );
|
QgsAbstractGeometry *diffGeom = geomEngine->symDifference( *featureGeomB );
|
||||||
if ( diffGeom && diffGeom->area() < mContext->tolerance )
|
if ( diffGeom && diffGeom->area() < mContext->tolerance )
|
||||||
{
|
{
|
||||||
|
@ -25,7 +25,7 @@ class QgsGeometryDuplicateCheckError : public QgsGeometryCheckError
|
|||||||
const QString &layerId,
|
const QString &layerId,
|
||||||
QgsFeatureId featureId,
|
QgsFeatureId featureId,
|
||||||
QgsAbstractGeometry *geometry,
|
QgsAbstractGeometry *geometry,
|
||||||
const QgsPoint &errorLocation,
|
const QgsPointXY &errorLocation,
|
||||||
const QMap<QString, QList<QgsFeatureId>> &duplicates )
|
const QMap<QString, QList<QgsFeatureId>> &duplicates )
|
||||||
: QgsGeometryCheckError( check, layerId, featureId, geometry, errorLocation, QgsVertexId(), duplicatesString( duplicates ) )
|
: QgsGeometryCheckError( check, layerId, featureId, geometry, errorLocation, QgsVertexId(), duplicatesString( duplicates ) )
|
||||||
, mDuplicates( duplicates )
|
, mDuplicates( duplicates )
|
||||||
|
@ -33,11 +33,14 @@ void QgsGeometryDuplicateNodesCheck::collectErrors( QList<QgsGeometryCheckError
|
|||||||
continue;
|
continue;
|
||||||
for ( int iVert = nVerts - 1, jVert = 0; jVert < nVerts; iVert = jVert++ )
|
for ( int iVert = nVerts - 1, jVert = 0; jVert < nVerts; iVert = jVert++ )
|
||||||
{
|
{
|
||||||
QgsPointV2 pi = geom->vertexAt( QgsVertexId( iPart, iRing, iVert ) );
|
QgsPoint pi = geom->vertexAt( QgsVertexId( iPart, iRing, iVert ) );
|
||||||
QgsPointV2 pj = geom->vertexAt( QgsVertexId( iPart, iRing, jVert ) );
|
QgsPoint pj = geom->vertexAt( QgsVertexId( iPart, iRing, jVert ) );
|
||||||
if ( QgsGeometryUtils::sqrDistance2D( pi, pj ) < mContext->tolerance )
|
if ( QgsGeometryUtils::sqrDistance2D( pi, pj ) < mContext->tolerance )
|
||||||
{
|
{
|
||||||
errors.append( new QgsGeometryCheckError( this, layerFeature.layer().id(), layerFeature.feature().id(), geom->clone(), pj, QgsVertexId( iPart, iRing, jVert ) ) );
|
QgsAbstractGeometry *g = geom->clone();
|
||||||
|
g->transform( layerFeature.mapToLayerTransform(), QgsCoordinateTransform::ReverseTransform );
|
||||||
|
QgsPointXY pos = layerFeature.mapToLayerTransform().transform( pj, QgsCoordinateTransform::ReverseTransform );
|
||||||
|
errors.append( new QgsGeometryCheckError( this, layerFeature.layer().id(), layerFeature.feature().id(), g, pos, QgsVertexId( iPart, iRing, jVert ) ) );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -13,7 +13,6 @@
|
|||||||
* *
|
* *
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
|
|
||||||
#include "qgscrscache.h"
|
|
||||||
#include "qgsgeometryengine.h"
|
#include "qgsgeometryengine.h"
|
||||||
#include "qgsgeometrygapcheck.h"
|
#include "qgsgeometrygapcheck.h"
|
||||||
#include "qgsgeometrycollection.h"
|
#include "qgsgeometrycollection.h"
|
||||||
@ -100,12 +99,11 @@ void QgsGeometryGapCheck::collectErrors( QList<QgsGeometryCheckError *> &errors,
|
|||||||
for ( const QString &layerId : featureIds.keys() )
|
for ( const QString &layerId : featureIds.keys() )
|
||||||
{
|
{
|
||||||
QgsFeaturePool *featurePool = mContext->featurePools[ layerId ];
|
QgsFeaturePool *featurePool = mContext->featurePools[ layerId ];
|
||||||
QgsCoordinateTransform t = QgsCoordinateTransformCache::instance()->transform( mContext->mapCrs, featurePool->getLayer()->crs().authid() );
|
QgsRectangle gapAreaLayerBBox = featurePool->getMapToLayerTransform().transform( gapGeom->boundingBox() ); // Don't use gapAreaBBox since it is updated below
|
||||||
QgsRectangle gapAreaLayerBBox = t.transform( gapGeom->boundingBox() ); // Don't use gapAreaBBox since it is updated below
|
|
||||||
|
|
||||||
QgsFeatureIds intersectIds = featurePool->getIntersects( gapAreaLayerBBox );
|
QgsFeatureIds intersectIds = featurePool->getIntersects( gapAreaLayerBBox );
|
||||||
QgsAbstractGeometry *gapLayerGeom = gapGeom->clone();
|
QgsAbstractGeometry *gapLayerGeom = gapGeom->clone();
|
||||||
gapLayerGeom->transform( t );
|
gapLayerGeom->transform( featurePool->getMapToLayerTransform() );
|
||||||
|
|
||||||
for ( QgsFeatureId id : intersectIds )
|
for ( QgsFeatureId id : intersectIds )
|
||||||
{
|
{
|
||||||
@ -118,9 +116,9 @@ void QgsGeometryGapCheck::collectErrors( QList<QgsGeometryCheckError *> &errors,
|
|||||||
if ( QgsGeometryCheckerUtils::sharedEdgeLength( gapLayerGeom, featureGeom, mContext->reducedTolerance ) > 0 )
|
if ( QgsGeometryCheckerUtils::sharedEdgeLength( gapLayerGeom, featureGeom, mContext->reducedTolerance ) > 0 )
|
||||||
{
|
{
|
||||||
neighboringIds[layerId].insert( feature.id() );
|
neighboringIds[layerId].insert( feature.id() );
|
||||||
gapAreaLayerBBox.unionRect( featureGeom->boundingBox() );
|
gapAreaLayerBBox.combineExtentWith( featureGeom->boundingBox() );
|
||||||
}
|
}
|
||||||
gapAreaBBox.unionRect( t.transform( gapAreaLayerBBox, QgsCoordinateTransform::ReverseTransform ) );
|
gapAreaBBox.combineExtentWith( featurePool->getMapToLayerTransform().transform( gapAreaLayerBBox, QgsCoordinateTransform::ReverseTransform ) );
|
||||||
}
|
}
|
||||||
delete gapLayerGeom;
|
delete gapLayerGeom;
|
||||||
}
|
}
|
||||||
@ -170,15 +168,14 @@ bool QgsGeometryGapCheck::mergeWithNeighbor( QgsGeometryGapCheckError *err, Chan
|
|||||||
QgsFeature mergeFeature;
|
QgsFeature mergeFeature;
|
||||||
int mergePartIdx = -1;
|
int mergePartIdx = -1;
|
||||||
|
|
||||||
QgsAbstractGeometry *errGeometry = QgsGeometryCheckerUtils::getGeomPart( err->geometry(), 0 );
|
const QgsAbstractGeometry *errGeometry = QgsGeometryCheckerUtils::getGeomPart( err->geometry(), 0 );
|
||||||
|
|
||||||
// Search for touching neighboring geometries
|
// Search for touching neighboring geometries
|
||||||
for ( const QString &layerId : err->neighbors().keys() )
|
for ( const QString &layerId : err->neighbors().keys() )
|
||||||
{
|
{
|
||||||
QgsFeaturePool *featurePool = mContext->featurePools[ err->layerId() ];
|
QgsFeaturePool *featurePool = mContext->featurePools[ err->layerId() ];
|
||||||
QgsCoordinateTransform t = QgsCoordinateTransformCache::instance()->transform( mContext->mapCrs, featurePool->getLayer()->crs().authid() );
|
|
||||||
QgsAbstractGeometry *errLayerGeom = errGeometry->clone();
|
QgsAbstractGeometry *errLayerGeom = errGeometry->clone();
|
||||||
errLayerGeom->transform( t );
|
errLayerGeom->transform( featurePool->getMapToLayerTransform() );
|
||||||
|
|
||||||
for ( QgsFeatureId testId : err->neighbors()[layerId] )
|
for ( QgsFeatureId testId : err->neighbors()[layerId] )
|
||||||
{
|
{
|
||||||
@ -211,9 +208,8 @@ bool QgsGeometryGapCheck::mergeWithNeighbor( QgsGeometryGapCheckError *err, Chan
|
|||||||
|
|
||||||
// Merge geometries
|
// Merge geometries
|
||||||
QgsFeaturePool *featurePool = mContext->featurePools[ mergeLayerId ];
|
QgsFeaturePool *featurePool = mContext->featurePools[ mergeLayerId ];
|
||||||
QgsCoordinateTransform t = QgsCoordinateTransformCache::instance()->transform( mContext->mapCrs, featurePool->getLayer()->crs().authid() );
|
|
||||||
QgsAbstractGeometry *errLayerGeom = errGeometry->clone();
|
QgsAbstractGeometry *errLayerGeom = errGeometry->clone();
|
||||||
errLayerGeom->transform( t );
|
errLayerGeom->transform( featurePool->getMapToLayerTransform() );
|
||||||
QgsGeometry mergeFeatureGeom = mergeFeature.geometry();
|
QgsGeometry mergeFeatureGeom = mergeFeature.geometry();
|
||||||
QgsAbstractGeometry *mergeGeom = mergeFeatureGeom.geometry();
|
QgsAbstractGeometry *mergeGeom = mergeFeatureGeom.geometry();
|
||||||
QSharedPointer<QgsGeometryEngine> geomEngine = QgsGeometryCheckerUtils::createGeomEngine( errLayerGeom, mContext->tolerance );
|
QSharedPointer<QgsGeometryEngine> geomEngine = QgsGeometryCheckerUtils::createGeomEngine( errLayerGeom, mContext->tolerance );
|
||||||
|
@ -28,7 +28,10 @@ void QgsGeometryHoleCheck::collectErrors( QList<QgsGeometryCheckError *> &errors
|
|||||||
// Rings after the first one are interiors
|
// Rings after the first one are interiors
|
||||||
for ( int iRing = 1, nRings = geom->ringCount( iPart ); iRing < nRings; ++iRing )
|
for ( int iRing = 1, nRings = geom->ringCount( iPart ); iRing < nRings; ++iRing )
|
||||||
{
|
{
|
||||||
errors.append( new QgsGeometryCheckError( this, layerFeature.layer().id(), layerFeature.feature().id(), geom->clone(), QgsGeometryCheckerUtils::getGeomPart( geom, iPart )->centroid(), QgsVertexId( iPart, iRing ) ) );
|
QgsAbstractGeometry *g = geom->clone();
|
||||||
|
g->transform( layerFeature.mapToLayerTransform(), QgsCoordinateTransform::ReverseTransform );
|
||||||
|
QgsPointXY pos = layerFeature.mapToLayerTransform().transform( QgsGeometryCheckerUtils::getGeomPart( geom, iPart )->centroid(), QgsCoordinateTransform::ReverseTransform );
|
||||||
|
errors.append( new QgsGeometryCheckError( this, layerFeature.layer().id(), layerFeature.feature().id(), g, pos, QgsVertexId( iPart, iRing ) ) );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -26,7 +26,10 @@ void QgsGeometryMultipartCheck::collectErrors( QList<QgsGeometryCheckError *> &e
|
|||||||
QgsWkbTypes::Type type = geom->wkbType();
|
QgsWkbTypes::Type type = geom->wkbType();
|
||||||
if ( geom->partCount() == 1 && QgsWkbTypes::isMultiType( type ) )
|
if ( geom->partCount() == 1 && QgsWkbTypes::isMultiType( type ) )
|
||||||
{
|
{
|
||||||
errors.append( new QgsGeometryCheckError( this, layerFeature.layer().id(), layerFeature.feature().id(), geom->clone(), geom->centroid() ) );
|
QgsAbstractGeometry *g = geom->clone();
|
||||||
|
g->transform( layerFeature.mapToLayerTransform(), QgsCoordinateTransform::ReverseTransform );
|
||||||
|
QgsPoint pos = g->centroid();
|
||||||
|
errors.append( new QgsGeometryCheckError( this, layerFeature.layer().id(), layerFeature.feature().id(), g, pos ) );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -13,7 +13,6 @@
|
|||||||
* *
|
* *
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
|
|
||||||
#include "qgscrscache.h"
|
|
||||||
#include "qgsgeometryengine.h"
|
#include "qgsgeometryengine.h"
|
||||||
#include "qgsgeometryoverlapcheck.h"
|
#include "qgsgeometryoverlapcheck.h"
|
||||||
#include "../utils/qgsfeaturepool.h"
|
#include "../utils/qgsfeaturepool.h"
|
||||||
@ -81,16 +80,14 @@ void QgsGeometryOverlapCheck::fixError( QgsGeometryCheckError *error, int method
|
|||||||
error->setObsolete();
|
error->setObsolete();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
QgsCoordinateTransform crstA = QgsCoordinateTransformCache::instance()->transform( featurePoolA->getLayer()->crs().authid(), mContext->mapCrs );
|
|
||||||
QgsCoordinateTransform crstB = QgsCoordinateTransformCache::instance()->transform( featurePoolB->getLayer()->crs().authid(), mContext->mapCrs );
|
|
||||||
|
|
||||||
// Check if error still applies
|
// Check if error still applies
|
||||||
QgsAbstractGeometry *featureGeomA = featureA.geometry().geometry()->clone();
|
QgsAbstractGeometry *featureGeomA = featureA.geometry().geometry()->clone();
|
||||||
featureGeomA->transform( crstA );
|
featureGeomA->transform( featurePoolA->getMapToLayerTransform(), QgsCoordinateTransform::ReverseTransform );
|
||||||
QSharedPointer<QgsGeometryEngine> geomEngineA = QgsGeometryCheckerUtils::createGeomEngine( featureGeomA, mContext->reducedTolerance );
|
QSharedPointer<QgsGeometryEngine> geomEngineA = QgsGeometryCheckerUtils::createGeomEngine( featureGeomA, mContext->reducedTolerance );
|
||||||
|
|
||||||
QgsAbstractGeometry *featureGeomB = featureB.geometry().geometry()->clone();
|
QgsAbstractGeometry *featureGeomB = featureB.geometry().geometry()->clone();
|
||||||
featureGeomB->transform( crstB );
|
featureGeomB->transform( featurePoolB->getMapToLayerTransform(), QgsCoordinateTransform::ReverseTransform );
|
||||||
|
|
||||||
if ( !geomEngineA->overlaps( *featureGeomB ) )
|
if ( !geomEngineA->overlaps( *featureGeomB ) )
|
||||||
{
|
{
|
||||||
|
@ -25,7 +25,7 @@ class QgsGeometryOverlapCheckError : public QgsGeometryCheckError
|
|||||||
const QString &layerId,
|
const QString &layerId,
|
||||||
QgsFeatureId featureId,
|
QgsFeatureId featureId,
|
||||||
QgsAbstractGeometry *geometry,
|
QgsAbstractGeometry *geometry,
|
||||||
const QgsPoint &errorLocation,
|
const QgsPointXY &errorLocation,
|
||||||
const QVariant &value,
|
const QVariant &value,
|
||||||
const QPair<QString, QgsFeatureId> &overlappedFeature )
|
const QPair<QString, QgsFeatureId> &overlappedFeature )
|
||||||
: QgsGeometryCheckError( check, layerId, featureId, geometry, errorLocation, QgsVertexId(), value, ValueArea )
|
: QgsGeometryCheckError( check, layerId, featureId, geometry, errorLocation, QgsVertexId(), value, ValueArea )
|
||||||
|
@ -43,7 +43,10 @@ void QgsGeometrySegmentLengthCheck::collectErrors( QList<QgsGeometryCheckError *
|
|||||||
double dist = qSqrt( QgsGeometryUtils::sqrDistance2D( pi, pj ) );
|
double dist = qSqrt( QgsGeometryUtils::sqrDistance2D( pi, pj ) );
|
||||||
if ( dist < minLength )
|
if ( dist < minLength )
|
||||||
{
|
{
|
||||||
errors.append( new QgsGeometryCheckError( this, layerFeature.layer().id(), layerFeature.feature().id(), layerFeature.geometry()->clone(), QgsPoint( 0.5 * ( pi.x() + pj.x() ), 0.5 * ( pi.y() + pj.y() ) ), QgsVertexId( iPart, iRing, iVert ), dist / mapToLayerUnits, QgsGeometryCheckError::ValueLength ) );
|
QgsAbstractGeometry *g = layerFeature.geometry()->clone();
|
||||||
|
g->transform( layerFeature.mapToLayerTransform(), QgsCoordinateTransform::ReverseTransform );
|
||||||
|
QgsPointXY pos = layerFeature.mapToLayerTransform().transform( QgsPoint( 0.5 * ( pi.x() + pj.x() ), 0.5 * ( pi.y() + pj.y() ) ), QgsCoordinateTransform::ReverseTransform );
|
||||||
|
errors.append( new QgsGeometryCheckError( this, layerFeature.layer().id(), layerFeature.feature().id(), g, pos, QgsVertexId( iPart, iRing, iVert ), dist / mapToLayerUnits, QgsGeometryCheckError::ValueLength ) );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -26,7 +26,7 @@ void QgsGeometrySelfContactCheck::collectErrors( QList<QgsGeometryCheckError *>
|
|||||||
|
|
||||||
// Geometry ring without duplicate nodes
|
// Geometry ring without duplicate nodes
|
||||||
QVector<int> vtxMap;
|
QVector<int> vtxMap;
|
||||||
QVector<QgsPointV2> ring;
|
QVector<QgsPoint> ring;
|
||||||
vtxMap.append( 0 );
|
vtxMap.append( 0 );
|
||||||
ring.append( geom->vertexAt( QgsVertexId( iPart, iRing, 0 ) ) );
|
ring.append( geom->vertexAt( QgsVertexId( iPart, iRing, 0 ) ) );
|
||||||
for ( int i = 1; i < n; ++i )
|
for ( int i = 1; i < n; ++i )
|
||||||
@ -53,7 +53,7 @@ void QgsGeometrySelfContactCheck::collectErrors( QList<QgsGeometryCheckError *>
|
|||||||
// For each vertex, check whether it lies on a segment
|
// For each vertex, check whether it lies on a segment
|
||||||
for ( int iVert = 0, nVerts = n - isClosed; iVert < nVerts; ++iVert )
|
for ( int iVert = 0, nVerts = n - isClosed; iVert < nVerts; ++iVert )
|
||||||
{
|
{
|
||||||
const QgsPointV2 &p = ring[iVert];
|
const QgsPoint &p = ring[iVert];
|
||||||
for ( int i = 0, j = 1; j < n; i = j++ )
|
for ( int i = 0, j = 1; j < n; i = j++ )
|
||||||
{
|
{
|
||||||
if ( iVert == i || iVert == j || ( isClosed && iVert == 0 && j == n - 1 ) )
|
if ( iVert == i || iVert == j || ( isClosed && iVert == 0 && j == n - 1 ) )
|
||||||
@ -65,7 +65,10 @@ void QgsGeometrySelfContactCheck::collectErrors( QList<QgsGeometryCheckError *>
|
|||||||
QgsPoint q = QgsGeometryUtils::projPointOnSegment( p, si, sj );
|
QgsPoint q = QgsGeometryUtils::projPointOnSegment( p, si, sj );
|
||||||
if ( QgsGeometryUtils::sqrDistance2D( p, q ) < mContext->tolerance * mContext->tolerance )
|
if ( QgsGeometryUtils::sqrDistance2D( p, q ) < mContext->tolerance * mContext->tolerance )
|
||||||
{
|
{
|
||||||
errors.append( new QgsGeometryCheckError( this, layerFeature.layer().id(), layerFeature.feature().id(), geom->clone(), p, QgsVertexId( iPart, iRing, vtxMap[iVert] ) ) );
|
QgsAbstractGeometry *g = geom->clone();
|
||||||
|
g->transform( layerFeature.mapToLayerTransform(), QgsCoordinateTransform::ReverseTransform );
|
||||||
|
QgsPointXY pos = layerFeature.mapToLayerTransform().transform( p, QgsCoordinateTransform::ReverseTransform );
|
||||||
|
errors.append( new QgsGeometryCheckError( this, layerFeature.layer().id(), layerFeature.feature().id(), g, pos, QgsVertexId( iPart, iRing, vtxMap[iVert] ) ) );
|
||||||
break; // No need to report same contact on different segments multiple times
|
break; // No need to report same contact on different segments multiple times
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -73,7 +73,10 @@ void QgsGeometrySelfIntersectionCheck::collectErrors( QList<QgsGeometryCheckErro
|
|||||||
{
|
{
|
||||||
for ( const QgsGeometryUtils::SelfIntersection &inter : QgsGeometryUtils::getSelfIntersections( geom, iPart, iRing, mContext->tolerance ) )
|
for ( const QgsGeometryUtils::SelfIntersection &inter : QgsGeometryUtils::getSelfIntersections( geom, iPart, iRing, mContext->tolerance ) )
|
||||||
{
|
{
|
||||||
errors.append( new QgsGeometrySelfIntersectionCheckError( this, layerFeature.layer().id(), layerFeature.feature().id(), geom->clone(), inter.point, QgsVertexId( iPart, iRing ), inter ) );
|
QgsAbstractGeometry *g = geom->clone();
|
||||||
|
g->transform( layerFeature.mapToLayerTransform(), QgsCoordinateTransform::ReverseTransform );
|
||||||
|
QgsPointXY pos = layerFeature.mapToLayerTransform().transform( inter.point, QgsCoordinateTransform::ReverseTransform );
|
||||||
|
errors.append( new QgsGeometrySelfIntersectionCheckError( this, layerFeature.layer().id(), layerFeature.feature().id(), geom->clone(), pos, QgsVertexId( iPart, iRing ), inter ) );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -26,7 +26,7 @@ class QgsGeometrySelfIntersectionCheckError : public QgsGeometryCheckError
|
|||||||
const QString &layerId,
|
const QString &layerId,
|
||||||
QgsFeatureId featureId,
|
QgsFeatureId featureId,
|
||||||
QgsAbstractGeometry *geometry,
|
QgsAbstractGeometry *geometry,
|
||||||
const QgsPoint &errorLocation,
|
const QgsPointXY &errorLocation,
|
||||||
QgsVertexId vidx,
|
QgsVertexId vidx,
|
||||||
const QgsGeometryUtils::SelfIntersection &inter )
|
const QgsGeometryUtils::SelfIntersection &inter )
|
||||||
: QgsGeometryCheckError( check, layerId, featureId, geometry, errorLocation, vidx )
|
: QgsGeometryCheckError( check, layerId, featureId, geometry, errorLocation, vidx )
|
||||||
|
@ -33,7 +33,10 @@ void QgsGeometryTypeCheck::collectErrors( QList<QgsGeometryCheckError *> &errors
|
|||||||
QgsWkbTypes::Type type = QgsWkbTypes::flatType( geom->wkbType() );
|
QgsWkbTypes::Type type = QgsWkbTypes::flatType( geom->wkbType() );
|
||||||
if ( ( mAllowedTypes & ( 1 << type ) ) == 0 )
|
if ( ( mAllowedTypes & ( 1 << type ) ) == 0 )
|
||||||
{
|
{
|
||||||
errors.append( new QgsGeometryTypeCheckError( this, layerFeature.layer().id(), layerFeature.feature().id(), geom->clone(), geom->centroid(), type ) );
|
QgsAbstractGeometry *g = geom->clone();
|
||||||
|
g->transform( layerFeature.mapToLayerTransform(), QgsCoordinateTransform::ReverseTransform );
|
||||||
|
QgsPoint pos = g->centroid();
|
||||||
|
errors.append( new QgsGeometryTypeCheckError( this, layerFeature.layer().id(), layerFeature.feature().id(), g, pos, type ) );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -25,7 +25,7 @@ class QgsGeometryTypeCheckError : public QgsGeometryCheckError
|
|||||||
const QString &layerId,
|
const QString &layerId,
|
||||||
QgsFeatureId featureId,
|
QgsFeatureId featureId,
|
||||||
QgsAbstractGeometry *geometry,
|
QgsAbstractGeometry *geometry,
|
||||||
const QgsPoint &errorLocation,
|
const QgsPointXY &errorLocation,
|
||||||
QgsWkbTypes::Type flatType )
|
QgsWkbTypes::Type flatType )
|
||||||
: QgsGeometryCheckError( check, layerId, featureId, geometry, errorLocation )
|
: QgsGeometryCheckError( check, layerId, featureId, geometry, errorLocation )
|
||||||
{
|
{
|
||||||
|
@ -268,8 +268,7 @@ bool QgsGeometryCheckerResultTab::exportErrorsDo( const QString &file )
|
|||||||
f.setAttribute( fieldLayer, srcLayer->name() );
|
f.setAttribute( fieldLayer, srcLayer->name() );
|
||||||
f.setAttribute( fieldFeatureId, error->featureId() );
|
f.setAttribute( fieldFeatureId, error->featureId() );
|
||||||
f.setAttribute( fieldErrDesc, error->description() );
|
f.setAttribute( fieldErrDesc, error->description() );
|
||||||
QgsGeometry geom( error->location().clone() );
|
QgsGeometry geom( new QgsPoint( error->location() ) );
|
||||||
geom.transform( QgsCoordinateTransformCache::instance()->transform( srcLayer->crs().authid(), layer->crs().authid() ) );
|
|
||||||
f.setGeometry( geom );
|
f.setGeometry( geom );
|
||||||
layer->dataProvider()->addFeatures( QgsFeatureList() << f );
|
layer->dataProvider()->addFeatures( QgsFeatureList() << f );
|
||||||
}
|
}
|
||||||
@ -324,14 +323,13 @@ void QgsGeometryCheckerResultTab::highlightErrors( bool current )
|
|||||||
for ( QTableWidgetItem *item : items )
|
for ( QTableWidgetItem *item : items )
|
||||||
{
|
{
|
||||||
QgsGeometryCheckError *error = ui.tableWidgetErrors->item( item->row(), 0 )->data( Qt::UserRole ).value<QgsGeometryCheckError *>();
|
QgsGeometryCheckError *error = ui.tableWidgetErrors->item( item->row(), 0 )->data( Qt::UserRole ).value<QgsGeometryCheckError *>();
|
||||||
QgsVectorLayer *layer = !error->layerId().isEmpty() ? mChecker->getContext()->featurePools[error->layerId()]->getLayer() : nullptr;
|
|
||||||
|
|
||||||
QgsAbstractGeometry *geometry = error->geometry();
|
const QgsAbstractGeometry *geometry = error->geometry();
|
||||||
if ( ui.checkBoxHighlight->isChecked() && geometry )
|
if ( ui.checkBoxHighlight->isChecked() && geometry )
|
||||||
{
|
{
|
||||||
QgsRubberBand *featureRubberBand = new QgsRubberBand( mIface->mapCanvas() );
|
QgsRubberBand *featureRubberBand = new QgsRubberBand( mIface->mapCanvas() );
|
||||||
QgsGeometry geom( geometry->clone() );
|
QgsGeometry geom( geometry->clone() );
|
||||||
featureRubberBand->addGeometry( geom, layer );
|
featureRubberBand->addGeometry( geom, 0 );
|
||||||
featureRubberBand->setWidth( 5 );
|
featureRubberBand->setWidth( 5 );
|
||||||
featureRubberBand->setColor( Qt::yellow );
|
featureRubberBand->setColor( Qt::yellow );
|
||||||
mCurrentRubberBands.append( featureRubberBand );
|
mCurrentRubberBands.append( featureRubberBand );
|
||||||
@ -340,12 +338,11 @@ void QgsGeometryCheckerResultTab::highlightErrors( bool current )
|
|||||||
if ( ui.radioButtonError->isChecked() || current || error->status() == QgsGeometryCheckError::StatusFixed )
|
if ( ui.radioButtonError->isChecked() || current || error->status() == QgsGeometryCheckError::StatusFixed )
|
||||||
{
|
{
|
||||||
QgsRubberBand *pointRubberBand = new QgsRubberBand( mIface->mapCanvas(), QgsWkbTypes::PointGeometry );
|
QgsRubberBand *pointRubberBand = new QgsRubberBand( mIface->mapCanvas(), QgsWkbTypes::PointGeometry );
|
||||||
QgsPoint pos = mIface->mapCanvas()->mapSettings().layerToMapCoordinates( layer, QgsPointXY( error->location().x(), error->location().y() ) );
|
pointRubberBand->addPoint( error->location() );
|
||||||
pointRubberBand->addPoint( pos );
|
|
||||||
pointRubberBand->setWidth( 20 );
|
pointRubberBand->setWidth( 20 );
|
||||||
pointRubberBand->setColor( Qt::red );
|
pointRubberBand->setColor( Qt::red );
|
||||||
mCurrentRubberBands.append( pointRubberBand );
|
mCurrentRubberBands.append( pointRubberBand );
|
||||||
errorPositions.append( pos );
|
errorPositions.append( error->location() );
|
||||||
}
|
}
|
||||||
else if ( ui.radioButtonFeature->isChecked() )
|
else if ( ui.radioButtonFeature->isChecked() )
|
||||||
{
|
{
|
||||||
|
@ -24,6 +24,7 @@
|
|||||||
|
|
||||||
#include "qgsfeatureiterator.h"
|
#include "qgsfeatureiterator.h"
|
||||||
#include "qgisinterface.h"
|
#include "qgisinterface.h"
|
||||||
|
#include "qgscrscache.h"
|
||||||
#include "qgsproject.h"
|
#include "qgsproject.h"
|
||||||
#include "qgsvectorlayer.h"
|
#include "qgsvectorlayer.h"
|
||||||
#include "qgsmapcanvas.h"
|
#include "qgsmapcanvas.h"
|
||||||
@ -64,7 +65,7 @@ QgsGeometryCheckerSetupTab::QgsGeometryCheckerSetupTab( QgisInterface *iface, QW
|
|||||||
}
|
}
|
||||||
|
|
||||||
connect( mRunButton, &QAbstractButton::clicked, this, &QgsGeometryCheckerSetupTab::runChecks );
|
connect( mRunButton, &QAbstractButton::clicked, this, &QgsGeometryCheckerSetupTab::runChecks );
|
||||||
connect( ui.listWidgetInputLayers, &QListWidgetItem::itemChanged, this, &QgsGeometryCheckerSetupTab::validateInput() );
|
connect( ui.listWidgetInputLayers, &QListWidget::itemChanged, this, &QgsGeometryCheckerSetupTab::validateInput );
|
||||||
connect( QgsProject::instance(), &QgsProject::layersAdded, this, &QgsGeometryCheckerSetupTab::updateLayers );
|
connect( QgsProject::instance(), &QgsProject::layersAdded, this, &QgsGeometryCheckerSetupTab::updateLayers );
|
||||||
connect( QgsProject::instance(), static_cast<void ( QgsProject::* )( const QStringList & )>( &QgsProject::layersWillBeRemoved ), this, &QgsGeometryCheckerSetupTab::updateLayers );
|
connect( QgsProject::instance(), static_cast<void ( QgsProject::* )( const QStringList & )>( &QgsProject::layersWillBeRemoved ), this, &QgsGeometryCheckerSetupTab::updateLayers );
|
||||||
connect( ui.radioButtonOutputNew, &QAbstractButton::toggled, ui.frameOutput, &QWidget::setEnabled );
|
connect( ui.radioButtonOutputNew, &QAbstractButton::toggled, ui.frameOutput, &QWidget::setEnabled );
|
||||||
@ -397,7 +398,8 @@ void QgsGeometryCheckerSetupTab::runChecks()
|
|||||||
for ( QgsVectorLayer *layer : processLayers )
|
for ( QgsVectorLayer *layer : processLayers )
|
||||||
{
|
{
|
||||||
double mapToLayerUnits = 1. / mIface->mapCanvas()->mapSettings().layerToMapUnits( layer );
|
double mapToLayerUnits = 1. / mIface->mapCanvas()->mapSettings().layerToMapUnits( layer );
|
||||||
featurePools.insert( layer->id(), new QgsFeaturePool( layer, mapToLayerUnits, selectedOnly ) );
|
QgsCoordinateTransform mapToLayerTransform = QgsCoordinateTransformCache::instance()->transform( mIface->mapCanvas()->mapSettings().destinationCrs().authid(), layer->crs().authid() );
|
||||||
|
featurePools.insert( layer->id(), new QgsFeaturePool( layer, mapToLayerUnits, mapToLayerTransform, selectedOnly ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
QgsGeometryCheckerContext *context = new QgsGeometryCheckerContext( ui.spinBoxTolerance->value(), mIface->mapCanvas()->mapSettings().destinationCrs().authid(), featurePools );
|
QgsGeometryCheckerContext *context = new QgsGeometryCheckerContext( ui.spinBoxTolerance->value(), mIface->mapCanvas()->mapSettings().destinationCrs().authid(), featurePools );
|
||||||
|
@ -25,10 +25,11 @@
|
|||||||
#include <QMutexLocker>
|
#include <QMutexLocker>
|
||||||
#include <limits>
|
#include <limits>
|
||||||
|
|
||||||
QgsFeaturePool::QgsFeaturePool( QgsVectorLayer *layer, double mapToLayerUnits, bool selectedOnly )
|
QgsFeaturePool::QgsFeaturePool( QgsVectorLayer *layer, double mapToLayerUnits, const QgsCoordinateTransform &mapToLayerTransform, bool selectedOnly )
|
||||||
: mFeatureCache( CACHE_SIZE )
|
: mFeatureCache( CACHE_SIZE )
|
||||||
, mLayer( layer )
|
, mLayer( layer )
|
||||||
, mMapToLayerUnits( mapToLayerUnits )
|
, mMapToLayerUnits( mapToLayerUnits )
|
||||||
|
, mMapToLayerTransform( mapToLayerTransform )
|
||||||
, mSelectedOnly( selectedOnly )
|
, mSelectedOnly( selectedOnly )
|
||||||
{
|
{
|
||||||
if ( selectedOnly )
|
if ( selectedOnly )
|
||||||
|
@ -30,7 +30,7 @@ class QgsVectorLayer;
|
|||||||
class QgsFeaturePool
|
class QgsFeaturePool
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
QgsFeaturePool( QgsVectorLayer *layer, double mapToLayerUnits, bool selectedOnly = false );
|
QgsFeaturePool( QgsVectorLayer *layer, double mapToLayerUnits, const QgsCoordinateTransform &mapToLayerTransform, bool selectedOnly = false );
|
||||||
bool get( QgsFeatureId id, QgsFeature &feature );
|
bool get( QgsFeatureId id, QgsFeature &feature );
|
||||||
void addFeature( QgsFeature &feature );
|
void addFeature( QgsFeature &feature );
|
||||||
void updateFeature( QgsFeature &feature );
|
void updateFeature( QgsFeature &feature );
|
||||||
@ -38,7 +38,8 @@ class QgsFeaturePool
|
|||||||
QgsFeatureIds getIntersects( const QgsRectangle &rect ) const;
|
QgsFeatureIds getIntersects( const QgsRectangle &rect ) const;
|
||||||
QgsVectorLayer *getLayer() const { return mLayer; }
|
QgsVectorLayer *getLayer() const { return mLayer; }
|
||||||
const QgsFeatureIds &getFeatureIds() const { return mFeatureIds; }
|
const QgsFeatureIds &getFeatureIds() const { return mFeatureIds; }
|
||||||
double getMapToLayerUnits() const { return mMapToLayerUnits;}
|
double getMapToLayerUnits() const { return mMapToLayerUnits; }
|
||||||
|
const QgsCoordinateTransform &getMapToLayerTransform() const { return mMapToLayerTransform; }
|
||||||
bool getSelectedOnly() const { return mSelectedOnly; }
|
bool getSelectedOnly() const { return mSelectedOnly; }
|
||||||
void clearLayer() { mLayer = nullptr; }
|
void clearLayer() { mLayer = nullptr; }
|
||||||
|
|
||||||
@ -62,6 +63,7 @@ class QgsFeaturePool
|
|||||||
mutable QMutex mIndexMutex;
|
mutable QMutex mIndexMutex;
|
||||||
QgsSpatialIndex mIndex;
|
QgsSpatialIndex mIndex;
|
||||||
double mMapToLayerUnits;
|
double mMapToLayerUnits;
|
||||||
|
QgsCoordinateTransform mMapToLayerTransform;
|
||||||
bool mSelectedOnly;
|
bool mSelectedOnly;
|
||||||
|
|
||||||
bool getTouchingWithSharedEdge( QgsFeature &feature, QgsFeatureId &touchingId, const double & ( *comparator )( const double &, const double & ), double init );
|
bool getTouchingWithSharedEdge( QgsFeature &feature, QgsFeatureId &touchingId, const double & ( *comparator )( const double &, const double & ), double init );
|
||||||
|
@ -25,19 +25,19 @@
|
|||||||
|
|
||||||
namespace QgsGeometryCheckerUtils
|
namespace QgsGeometryCheckerUtils
|
||||||
{
|
{
|
||||||
LayerFeature::LayerFeature( const QgsVectorLayer *layer, const QgsFeature &feature, double mapToLayerUnits, const QString &targetCrs )
|
LayerFeature::LayerFeature( const QgsVectorLayer *layer, const QgsFeature &feature, double mapToLayerUnits, const QgsCoordinateTransform &mapToLayerTransform )
|
||||||
: mLayer( layer )
|
: mLayer( layer )
|
||||||
, mFeature( feature )
|
, mFeature( feature )
|
||||||
, mMapToLayerUnits( mapToLayerUnits )
|
, mMapToLayerUnits( mapToLayerUnits )
|
||||||
|
, mMapToLayerTransform( mapToLayerTransform )
|
||||||
, mClonedGeometry( false )
|
, mClonedGeometry( false )
|
||||||
{
|
{
|
||||||
mGeometry = feature.geometry().geometry();
|
mGeometry = feature.geometry().geometry();
|
||||||
if ( !targetCrs.isEmpty() && targetCrs != layer->crs().authid() )
|
if ( !mapToLayerTransform.isShortCircuited() )
|
||||||
{
|
{
|
||||||
mClonedGeometry = true;
|
mClonedGeometry = true;
|
||||||
mGeometry = mGeometry->clone();
|
mGeometry = mGeometry->clone();
|
||||||
QgsCoordinateTransform crst = QgsCoordinateTransformCache::instance()->transform( layer->crs().authid(), targetCrs );
|
mGeometry->transform( mapToLayerTransform, QgsCoordinateTransform::ReverseTransform );
|
||||||
mGeometry->transform( crst );
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
LayerFeature::~LayerFeature()
|
LayerFeature::~LayerFeature()
|
||||||
@ -55,7 +55,7 @@ namespace QgsGeometryCheckerUtils
|
|||||||
, mFeatureIt( featureIt )
|
, mFeatureIt( featureIt )
|
||||||
, mFeature( feature )
|
, mFeature( feature )
|
||||||
, mParent( parent )
|
, mParent( parent )
|
||||||
, mCurrentFeature( LayerFeature( parent->mFeaturePools[ * layerIt]->getLayer(), feature, parent->mFeaturePools[ * layerIt]->getMapToLayerUnits(), parent->mTargetCrs ) )
|
, mCurrentFeature( LayerFeature( parent->mFeaturePools[ * layerIt]->getLayer(), feature, parent->mFeaturePools[ * layerIt]->getMapToLayerUnits(), parent->mFeaturePools[ * layerIt]->getMapToLayerTransform() ) )
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -89,7 +89,7 @@ namespace QgsGeometryCheckerUtils
|
|||||||
mParent->mProgressCounter->fetchAndAddRelaxed( 1 );
|
mParent->mProgressCounter->fetchAndAddRelaxed( 1 );
|
||||||
if ( featurePool->get( *mFeatureIt, mFeature ) )
|
if ( featurePool->get( *mFeatureIt, mFeature ) )
|
||||||
{
|
{
|
||||||
mCurrentFeature = LayerFeature( mParent->mFeaturePools[*mLayerIt]->getLayer(), mFeature, mParent->mFeaturePools[*mLayerIt]->getMapToLayerUnits(), mParent->mTargetCrs );
|
mCurrentFeature = LayerFeature( mParent->mFeaturePools[*mLayerIt]->getLayer(), mFeature, mParent->mFeaturePools[*mLayerIt]->getMapToLayerUnits(), mParent->mFeaturePools[*mLayerIt]->getMapToLayerTransform() );
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -125,8 +125,7 @@ namespace QgsGeometryCheckerUtils
|
|||||||
const QgsFeaturePool *featurePool = featurePools[layerId];
|
const QgsFeaturePool *featurePool = featurePools[layerId];
|
||||||
if ( geometryTypes.contains( featurePool->getLayer()->geometryType() ) )
|
if ( geometryTypes.contains( featurePool->getLayer()->geometryType() ) )
|
||||||
{
|
{
|
||||||
QgsCoordinateTransform crst = QgsCoordinateTransformCache::instance()->transform( targetCrs, featurePool->getLayer()->crs().authid() );
|
mFeatureIds.insert( layerId, featurePool->getIntersects( featurePool->getMapToLayerTransform().transform( extent ) ) );
|
||||||
mFeatureIds.insert( layerId, featurePool->getIntersects( crst.transform( extent ) ) );
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
@ -17,7 +17,6 @@
|
|||||||
#ifndef QGS_GEOMETRYCHECKERUTILS_H
|
#ifndef QGS_GEOMETRYCHECKERUTILS_H
|
||||||
#define QGS_GEOMETRYCHECKERUTILS_H
|
#define QGS_GEOMETRYCHECKERUTILS_H
|
||||||
|
|
||||||
#include "qgscrscache.h"
|
|
||||||
#include "qgsfeature.h"
|
#include "qgsfeature.h"
|
||||||
#include "qgsvectorlayer.h"
|
#include "qgsvectorlayer.h"
|
||||||
#include "geometry/qgsabstractgeometry.h"
|
#include "geometry/qgsabstractgeometry.h"
|
||||||
@ -31,11 +30,12 @@ namespace QgsGeometryCheckerUtils
|
|||||||
class LayerFeature
|
class LayerFeature
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
LayerFeature( const QgsVectorLayer *layer, const QgsFeature &feature, double mapToLayerUnits, const QString &targetCrs );
|
LayerFeature( const QgsVectorLayer *layer, const QgsFeature &feature, double mapToLayerUnits, const QgsCoordinateTransform &mapToLayerTransform );
|
||||||
~LayerFeature();
|
~LayerFeature();
|
||||||
const QgsVectorLayer &layer() const { return *mLayer; }
|
const QgsVectorLayer &layer() const { return *mLayer; }
|
||||||
const QgsFeature &feature() const { return mFeature; }
|
const QgsFeature &feature() const { return mFeature; }
|
||||||
double mapToLayerUnits() const { return mMapToLayerUnits; }
|
double mapToLayerUnits() const { return mMapToLayerUnits; }
|
||||||
|
const QgsCoordinateTransform &mapToLayerTransform() const { return mMapToLayerTransform; }
|
||||||
const QgsAbstractGeometry *geometry() const { return mGeometry; }
|
const QgsAbstractGeometry *geometry() const { return mGeometry; }
|
||||||
QString id() const { return QString( "%1:%2" ).arg( mLayer->id() ).arg( mFeature.id() ); }
|
QString id() const { return QString( "%1:%2" ).arg( mLayer->id() ).arg( mFeature.id() ); }
|
||||||
|
|
||||||
@ -43,6 +43,7 @@ namespace QgsGeometryCheckerUtils
|
|||||||
const QgsVectorLayer *mLayer = nullptr;
|
const QgsVectorLayer *mLayer = nullptr;
|
||||||
QgsFeature mFeature;
|
QgsFeature mFeature;
|
||||||
double mMapToLayerUnits;
|
double mMapToLayerUnits;
|
||||||
|
QgsCoordinateTransform mMapToLayerTransform;
|
||||||
QgsAbstractGeometry *mGeometry = nullptr;
|
QgsAbstractGeometry *mGeometry = nullptr;
|
||||||
bool mClonedGeometry = false;
|
bool mClonedGeometry = false;
|
||||||
};
|
};
|
||||||
@ -132,7 +133,7 @@ namespace QgsGeometryCheckerUtils
|
|||||||
* \param tol The tolerance
|
* \param tol The tolerance
|
||||||
* \returns Whether the points are equal
|
* \returns Whether the points are equal
|
||||||
*/
|
*/
|
||||||
inline bool pointsFuzzyEqual( const QgsPoint &p1, const QgsPoint &p2, double tol )
|
inline bool pointsFuzzyEqual( const QgsPointXY &p1, const QgsPointXY &p2, double tol )
|
||||||
{
|
{
|
||||||
double dx = p1.x() - p2.x(), dy = p1.y() - p2.y();
|
double dx = p1.x() - p2.x(), dy = p1.y() - p2.y();
|
||||||
return ( dx * dx + dy * dy ) < tol * tol;
|
return ( dx * dx + dy * dy ) < tol * tol;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user