QGIS API Documentation 3.43.0-Master (3ee7834ace6)
qgsmaptoolcapture.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgsmaptoolcapture.cpp - map tool for capturing points, lines, polygons
3 ---------------------
4 begin : January 2006
5 copyright : (C) 2006 by Martin Dobias
6 email : wonder.sk at gmail dot com
7 ***************************************************************************
8 * *
9 * This program is free software; you can redistribute it and/or modify *
10 * it under the terms of the GNU General Public License as published by *
11 * the Free Software Foundation; either version 2 of the License, or *
12 * (at your option) any later version. *
13 * *
14 ***************************************************************************/
15
16#include "qgsmaptoolcapture.h"
17#include "moc_qgsmaptoolcapture.cpp"
18#include "qgsexception.h"
19#include "qgsfeatureiterator.h"
21#include "qgslinestring.h"
22#include "qgslogger.h"
23#include "qgsmapcanvas.h"
24#include "qgsmapcanvastracer.h"
25#include "qgsmapmouseevent.h"
26#include "qgspolygon.h"
27#include "qgsrubberband.h"
28#include "qgssnapindicator.h"
29#include "qgsvectorlayer.h"
30#include "qgsvertexmarker.h"
32#include "qgsapplication.h"
33#include "qgsproject.h"
37#include "qgssnappingutils.h"
39
40#include <QAction>
41#include <QCursor>
42#include <QPixmap>
43#include <QStatusBar>
44#include <algorithm>
45#include <memory>
46
47
49 : QgsMapToolAdvancedDigitizing( canvas, cadDockWidget )
50 , mCaptureMode( mode )
51 , mCaptureModeFromLayer( mode == CaptureNone )
52{
53 mTempRubberBand.setParentOwner( canvas );
54
55 mSnapIndicator.reset( new QgsSnapIndicator( canvas ) );
56
58
59 connect( canvas, &QgsMapCanvas::currentLayerChanged, this, &QgsMapToolCapture::currentLayerChanged );
60
62 layerOptions.skipCrsValidation = true;
63 layerOptions.loadDefaultStyle = false;
64 mExtraSnapLayer = new QgsVectorLayer( QStringLiteral( "LineString?crs=" ), QStringLiteral( "extra snap" ), QStringLiteral( "memory" ), layerOptions );
65 mExtraSnapLayer->startEditing();
66 QgsFeature f;
67 mExtraSnapLayer->addFeature( f );
68 mExtraSnapFeatureId = f.id();
69
70 connect( QgsProject::instance(), &QgsProject::snappingConfigChanged, this, &QgsMapToolCapture::updateExtraSnapLayer );
71
72 currentLayerChanged( canvas->currentLayer() );
73}
74
76{
77 // during tear down we have to clean up mExtraSnapLayer first, before
78 // we call stop capturing. Otherwise stopCapturing tries to access members
79 // from the mapcanvas, which is likely already being destroyed and triggering
80 // the deletion of this object...
81 if ( mCanvas )
82 {
83 mCanvas->snappingUtils()->removeExtraSnapLayer( mExtraSnapLayer );
84 }
85 mExtraSnapLayer->deleteLater();
86 mExtraSnapLayer = nullptr;
87
89
90 if ( mValidator )
91 {
92 mValidator->deleteLater();
93 mValidator = nullptr;
94 }
95}
96
101
103{
104 switch ( technique )
105 {
107 return true;
111 return false;
112 }
114}
115
117{
118 if ( mTempRubberBand )
119 mTempRubberBand->show();
120
121 mCanvas->snappingUtils()->addExtraSnapLayer( mExtraSnapLayer );
123
124 if ( mCurrentCaptureTechnique == Qgis::CaptureTechnique::Shape && mCurrentShapeMapTool )
125 mCurrentShapeMapTool->activate( mCaptureMode, mCaptureLastPoint );
126}
127
129{
130 if ( mTempRubberBand )
131 mTempRubberBand->hide();
132
133 mSnapIndicator->setMatch( QgsPointLocator::Match() );
134
135 mCanvas->snappingUtils()->removeExtraSnapLayer( mExtraSnapLayer );
136
137 if ( mCurrentCaptureTechnique == Qgis::CaptureTechnique::Shape && mCurrentShapeMapTool )
138 mCurrentShapeMapTool->deactivate();
139
141}
142
143void QgsMapToolCapture::currentLayerChanged( QgsMapLayer *layer )
144{
145 if ( !mCaptureModeFromLayer )
146 return;
147
148 mCaptureMode = CaptureNone;
149
150 QgsVectorLayer *vlayer = qobject_cast<QgsVectorLayer *>( layer );
151 if ( !vlayer )
152 {
153 return;
154 }
155
156 if ( vlayer->isSpatial() )
157 {
159 }
160 else
161 {
162 setCursor( QCursor( Qt::ArrowCursor ) );
163 }
164
165 switch ( vlayer->geometryType() )
166 {
168 mCaptureMode = CapturePoint;
169 break;
171 mCaptureMode = CaptureLine;
172 break;
174 mCaptureMode = CapturePolygon;
175 break;
176 default:
177 mCaptureMode = CaptureNone;
178 break;
179 }
180
181 if ( mTempRubberBand )
182 mTempRubberBand->setRubberBandGeometryType( mCaptureMode == CapturePolygon ? Qgis::GeometryType::Polygon : Qgis::GeometryType::Line );
183
184 resetRubberBand();
186}
187
188
189bool QgsMapToolCapture::tracingEnabled()
190{
192 return tracer && ( !tracer->actionEnableTracing() || tracer->actionEnableTracing()->isChecked() )
193 && ( !tracer->actionEnableSnapping() || tracer->actionEnableSnapping()->isChecked() );
194}
195
196
197QgsPointXY QgsMapToolCapture::tracingStartPoint()
198{
199 // if we have starting point from previous trace, then preferably use that one
200 // (useful when tracing with offset)
201 if ( mTracingStartPoint != QgsPointXY() )
202 return mTracingStartPoint;
203
204 return mCaptureLastPoint;
205}
206
207
208bool QgsMapToolCapture::tracingMouseMove( QgsMapMouseEvent *e )
209{
210 if ( !e->isSnapped() )
211 return false;
212
213 QgsPointXY pt0 = tracingStartPoint();
214 if ( pt0 == QgsPointXY() )
215 return false;
216
218 if ( !tracer )
219 return false; // this should not happen!
220
222 QVector<QgsPointXY> points = tracer->findShortestPath( pt0, e->mapPoint(), &err );
223 if ( points.isEmpty() )
224 {
225 tracer->reportError( err, false );
226 return false;
227 }
228
229 mTempRubberBand->reset( mCaptureMode == CapturePolygon ? Qgis::GeometryType::Polygon : Qgis::GeometryType::Line, Qgis::WkbType::LineString, mCaptureFirstPoint );
230 mTempRubberBand->addPoint( mCaptureLastPoint );
231
232 // if there is offset, we need to fix the rubber bands to make sure they are aligned correctly.
233 // There are two cases we need to sort out:
234 // 1. the last point of mRubberBand may need to be moved off the traced curve to respect the offset
235 // 2. first point of mTempRubberBand may be needed to be moved to the beginning of the offset trace
236 const QgsPoint lastPoint = mCaptureLastPoint;
237 QgsPointXY lastPointXY( lastPoint );
238 if ( lastPointXY == pt0 && points[0] != lastPointXY )
239 {
240 if ( mRubberBand->numberOfVertices() != 0 )
241 {
242 // if rubber band had just one point, for some strange reason it contains the point twice
243 // we only want to move the last point if there are multiple points already
244 if ( mRubberBand->numberOfVertices() > 2 || ( mRubberBand->numberOfVertices() == 2 && *mRubberBand->getPoint( 0, 0 ) != *mRubberBand->getPoint( 0, 1 ) ) )
245 mRubberBand->movePoint( points[0] );
246 }
247
248 mTempRubberBand->movePoint( 0, QgsPoint( points[0] ) );
249 }
250
251 mTempRubberBand->movePoint( QgsPoint( points[0] ) );
252
253 // update temporary rubberband
254 for ( int i = 1; i < points.count(); ++i ) //points added in the rubber band are 2D but will not be added to the capture curve
255 mTempRubberBand->addPoint( QgsPoint( points.at( i ) ), i == points.count() - 1 );
256
257
258 mTempRubberBand->addPoint( QgsPoint( points[points.size() - 1] ) );
259
260 tracer->reportError( QgsTracer::ErrNone, false ); // clear messagebar if there was any error
261 return true;
262}
263
264
265bool QgsMapToolCapture::tracingAddVertex( const QgsPointXY &point )
266{
268 if ( !tracer )
269 return false; // this should not happen!
270
271 if ( mTempRubberBand->pointsCount() == 0 )
272 {
273 if ( !tracer->init() )
274 {
276 return false;
277 }
278
279 // only accept first point if it is snapped to the graph (to vertex or edge)
280 const bool res = tracer->isPointSnapped( point );
281 if ( res )
282 {
283 mTracingStartPoint = point;
284 }
285 return false;
286 }
287
288 QgsPointXY pt0 = tracingStartPoint();
289 if ( pt0 == QgsPointXY() )
290 return false;
291
293 const QVector<QgsPointXY> tracedPointsInMapCrs = tracer->findShortestPath( pt0, point, &err );
294 if ( tracedPointsInMapCrs.isEmpty() )
295 return false; // ignore the vertex - can't find path to the end point!
296
297 // transform points
298 QgsPointSequence layerPoints;
299 layerPoints.reserve( tracedPointsInMapCrs.size() );
300 QgsPointSequence mapPoints;
301 mapPoints.reserve( tracedPointsInMapCrs.size() );
302 for ( const QgsPointXY &tracedPointMapCrs : tracedPointsInMapCrs )
303 {
304 QgsPoint mapPoint( tracedPointMapCrs );
305
306 QgsPoint lp; // in layer coords
307 if ( nextPoint( mapPoint, lp ) != 0 )
308 return false;
309
310 // copy z and m from layer point back to mapPoint, as nextPoint() call will populate these based
311 // on the context of the trace
312 if ( lp.is3D() )
313 mapPoint.addZValue( lp.z() );
314 if ( lp.isMeasure() )
315 mapPoint.addMValue( lp.m() );
316
317 mapPoints << mapPoint;
318 layerPoints << lp;
319 }
320
321 // Move the last point of the captured curve to the first point on the trace string (necessary if there is offset)
322 const QgsVertexId lastVertexId( 0, 0, mCaptureCurve.numPoints() - 1 );
323 mCaptureCurve.moveVertex( lastVertexId, layerPoints.first() );
324 mSnappingMatches.removeLast();
325 mSnappingMatches.append( QgsPointLocator::Match() );
326
327 addCurve( new QgsLineString( mapPoints ) );
328
329 resetRubberBand();
330
331 // Curves de-approximation
333 {
334#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
335 int pointBefore = mCaptureCurve.numPoints();
336#endif
337
338 // If the tool and the layer support curves
339 QgsVectorLayer *vlayer = qobject_cast<QgsVectorLayer *>( layer() );
341 {
342 const QgsGeometry linear = QgsGeometry( mCaptureCurve.segmentize() );
343 const QgsGeometry curved = linear.convertToCurves(
346 );
348 {
349 mCaptureCurve.clear();
350 mCaptureCurve.addCurve( qgsgeometry_cast<const QgsCurve *>( curved.constGet() )->clone() );
351 }
352 else
353 {
354 mCaptureCurve = *qgsgeometry_cast<const QgsCompoundCurve *>( curved.constGet() );
355 }
356 }
357
358#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
359 // sync the snapping matches list
360 const int pointAfter = mCaptureCurve.numPoints();
361
362 for ( ; pointBefore < pointAfter; ++pointBefore )
363 mSnappingMatches.append( QgsPointLocator::Match() );
364
365 for ( ; pointBefore > pointAfter; --pointBefore )
366 mSnappingMatches.removeLast();
367#else
368 mSnappingMatches.resize( mCaptureCurve.numPoints() );
369#endif
370 }
371
372 tracer->reportError( QgsTracer::ErrNone, true ); // clear messagebar if there was any error
373
374 // adjust last captured point
375 const QgsPoint lastPt = mCaptureCurve.endPoint();
376 mCaptureLastPoint = toMapCoordinates( layer(), lastPt );
377
378 return true;
379}
380
381QgsMapToolCaptureRubberBand *QgsMapToolCapture::createCurveRubberBand() const
382{
383 QgsMapToolCaptureRubberBand *rb = new QgsMapToolCaptureRubberBand( mCanvas );
384 rb->setStrokeWidth( digitizingStrokeWidth() );
385 QColor color = digitizingStrokeColor();
386
388 color.setAlphaF( color.alphaF() * alphaScale );
389 rb->setLineStyle( Qt::DotLine );
390 rb->setStrokeColor( color );
391
392 const QColor fillColor = digitizingFillColor();
393 rb->setFillColor( fillColor );
394 rb->show();
395 return rb;
396}
397
398void QgsMapToolCapture::resetRubberBand()
399{
400 if ( !mRubberBand )
401 return;
402 QgsLineString *lineString = mCaptureCurve.curveToLine();
403
404 mRubberBand->reset( mCaptureMode == CapturePolygon ? Qgis::GeometryType::Polygon : Qgis::GeometryType::Line );
405 mRubberBand->addGeometry( QgsGeometry( lineString ), layer() );
406}
407
409{
410 return mRubberBand.release();
411}
412
420
428
430{
431 if ( mCurrentCaptureTechnique == technique )
432 return;
433
434 mStartNewCurve = true;
435
436 if ( mCurrentCaptureTechnique == Qgis::CaptureTechnique::Shape && mCurrentShapeMapTool )
437 {
438 mCurrentShapeMapTool->deactivate();
439 clean();
440 }
441
442 switch ( technique )
443 {
445 mLineDigitizingType = Qgis::WkbType::LineString;
446 break;
448 mLineDigitizingType = Qgis::WkbType::CircularString;
449 break;
451 mLineDigitizingType = Qgis::WkbType::LineString;
453 break;
455 mLineDigitizingType = Qgis::WkbType::LineString;
456 break;
457 }
458
459 if ( mTempRubberBand )
460 mTempRubberBand->setStringType( mLineDigitizingType );
461
462 mCurrentCaptureTechnique = technique;
463
464 if ( technique == Qgis::CaptureTechnique::Shape && mCurrentShapeMapTool && isActive() )
465 {
466 clean();
467 mCurrentShapeMapTool->activate( mCaptureMode, mCaptureLastPoint );
468 }
469}
470
472{
473 if ( mCurrentShapeMapTool )
474 {
475 if ( shapeMapToolMetadata && mCurrentShapeMapTool->id() == shapeMapToolMetadata->id() )
476 return;
477 if ( mCurrentCaptureTechnique == Qgis::CaptureTechnique::Shape )
478 mCurrentShapeMapTool->deactivate();
479 mCurrentShapeMapTool->deleteLater();
480 }
481
482 mCurrentShapeMapTool.reset( shapeMapToolMetadata ? shapeMapToolMetadata->factory( this ) : nullptr );
483
484 if ( mCurrentCaptureTechnique == Qgis::CaptureTechnique::Shape && isActive() )
485 {
486 clean();
487 if ( mCurrentShapeMapTool )
488 mCurrentShapeMapTool->activate( mCaptureMode, mCaptureLastPoint );
489 }
490}
491
493{
494 // If we are adding a record to a non-spatial layer, just return
495 if ( mCaptureModeFromLayer && ( !canvas()->currentLayer() || !canvas()->currentLayer()->isSpatial() ) )
496 return;
497
499
500 const QgsPointXY point = e->mapPoint();
501
502 mSnapIndicator->setMatch( e->mapPointMatch() );
503
504 if ( mCurrentCaptureTechnique == Qgis::CaptureTechnique::Shape )
505 {
506 if ( !mCurrentShapeMapTool )
507 {
508 emit messageEmitted( tr( "Select an option from the Shape Digitizing Toolbar in order to capture shapes" ), Qgis::MessageLevel::Warning );
509 }
510 else
511 {
512 if ( !mTempRubberBand )
513 {
514 mTempRubberBand.reset( createCurveRubberBand() );
515 mTempRubberBand->setStringType( mLineDigitizingType );
516 mTempRubberBand->setRubberBandGeometryType( mCaptureMode == CapturePolygon ? Qgis::GeometryType::Polygon : Qgis::GeometryType::Line );
517 }
518
519 mCurrentShapeMapTool->cadCanvasMoveEvent( e, mCaptureMode );
520 return;
521 }
522 }
523 else
524 {
525 const QgsPoint mapPoint = QgsPoint( point );
526
527 if ( mCaptureMode != CapturePoint && mTempRubberBand && mCapturing )
528 {
529 bool hasTrace = false;
530
531 if ( mCurrentCaptureTechnique == Qgis::CaptureTechnique::Streaming )
532 {
533 if ( !mCaptureCurve.isEmpty() )
534 {
535 const QgsPoint prevPoint = mCaptureCurve.curveAt( mCaptureCurve.nCurves() - 1 )->endPoint();
536 if ( QgsPointXY( toCanvasCoordinates( toMapCoordinates( layer(), prevPoint ) ) ).distance( toCanvasCoordinates( point ) ) < mStreamingToleranceInPixels )
537 return;
538 }
539
540 mAllowAddingStreamingPoints = true;
542 mAllowAddingStreamingPoints = false;
543 }
544 else if ( tracingEnabled() && mCaptureCurve.numPoints() != 0 )
545 {
546 // Store the intermediate point for circular string to retrieve after tracing mouse move if
547 // the digitizing type is circular and the temp rubber band is effectivly circular and if this point is existing
548 // Store an empty point if the digitizing type is linear ot the point is not existing (curve not complete)
549 if ( mLineDigitizingType == Qgis::WkbType::CircularString && mTempRubberBand->stringType() == Qgis::WkbType::CircularString && mTempRubberBand->curveIsComplete() )
550 mCircularItermediatePoint = mTempRubberBand->pointFromEnd( 1 );
551 else if ( mLineDigitizingType == Qgis::WkbType::LineString || !mTempRubberBand->curveIsComplete() )
552 mCircularItermediatePoint = QgsPoint();
553
554 hasTrace = tracingMouseMove( e );
555
556 if ( !hasTrace )
557 {
558 // Restore the temp rubber band
559 mTempRubberBand->reset( mCaptureMode == CapturePolygon ? Qgis::GeometryType::Polygon : Qgis::GeometryType::Line, mLineDigitizingType, mCaptureFirstPoint );
560 mTempRubberBand->addPoint( mCaptureLastPoint );
561 if ( !mCircularItermediatePoint.isEmpty() )
562 {
563 mTempRubberBand->movePoint( mCircularItermediatePoint );
564 mTempRubberBand->addPoint( mCircularItermediatePoint );
565 }
566 }
567 }
568
569 if ( mCurrentCaptureTechnique != Qgis::CaptureTechnique::Streaming && !hasTrace )
570 {
571 if ( mCaptureCurve.numPoints() > 0 )
572 {
573 const QgsPoint mapPt = mCaptureLastPoint;
574
575 if ( mTempRubberBand )
576 {
577 mTempRubberBand->movePoint( mapPoint );
578 mTempRubberBand->movePoint( 0, mapPt );
579 }
580
581 // fix existing rubber band after tracing - the last point may have been moved if using offset
582 if ( mRubberBand->numberOfVertices() )
583 mRubberBand->movePoint( mapPt );
584 }
585 else if ( mTempRubberBand )
586 mTempRubberBand->movePoint( mapPoint );
587 }
588 }
589 }
590} // mouseMoveEvent
591
592
593int QgsMapToolCapture::nextPoint( const QgsPoint &mapPoint, QgsPoint &layerPoint )
594{
595 if ( QgsVectorLayer *vlayer = qobject_cast<QgsVectorLayer *>( layer() ) )
596 {
597 try
598 {
599 QgsPointXY mapP( mapPoint.x(), mapPoint.y() ); //#spellok
600 const bool is3D = layerPoint.is3D();
601 const bool isMeasure = layerPoint.isMeasure();
602 mapP = toLayerCoordinates( vlayer, mapP ); //transform snapped point back to layer crs //#spellok
603 layerPoint = QgsPoint( layerPoint.wkbType(), mapP.x(), mapP.y(), layerPoint.z(), layerPoint.m() ); //#spellok
604 if ( QgsWkbTypes::hasZ( vlayer->wkbType() ) && !is3D )
606 if ( QgsWkbTypes::hasM( vlayer->wkbType() ) && !isMeasure )
608 }
609 catch ( QgsCsException & )
610 {
611 QgsDebugError( QStringLiteral( "transformation to layer coordinate failed" ) );
612 return 2;
613 }
614 }
615 else
616 {
617 layerPoint = QgsPoint( toLayerCoordinates( layer(), mapPoint ) );
618 }
619
620 return 0;
621}
622
623int QgsMapToolCapture::nextPoint( QPoint p, QgsPoint &layerPoint, QgsPoint &mapPoint )
624{
626 return nextPoint( mapPoint, layerPoint );
627}
628
630{
631 QgsVectorLayer *vlayer = qobject_cast<QgsVectorLayer *>( layer() );
632 QgsVectorLayer *sourceLayer = match.layer();
634 {
636 return 0;
637 }
638 else if ( !vlayer )
639 {
640 return 1;
641 }
642
643 if ( match.isValid() && sourceLayer )
644 {
645 if ( ( match.hasVertex() || match.hasLineEndpoint() ) )
646 {
647 if ( sourceLayer->crs() != vlayer->crs() )
648 {
649 layerPoint = match.interpolatedPoint();
650 return 1;
651 }
652 QgsFeature f;
653 QgsFeatureRequest request;
654 request.setFilterFid( match.featureId() );
655 const bool fetched = match.layer()->getFeatures( request ).nextFeature( f );
656 if ( fetched )
657 {
658 QgsVertexId vId;
659 if ( !f.geometry().vertexIdFromVertexNr( match.vertexIndex(), vId ) )
660 {
661 return 2;
662 }
663 layerPoint = f.geometry().constGet()->vertexAt( vId );
664 if ( QgsWkbTypes::hasZ( vlayer->wkbType() ) && !layerPoint.is3D() )
665 layerPoint.addZValue( defaultZValue() );
666 if ( QgsWkbTypes::hasM( vlayer->wkbType() ) && !layerPoint.isMeasure() )
667 layerPoint.addMValue( defaultMValue() );
668
669 // ZM support depends on the target layer
670 if ( !QgsWkbTypes::hasZ( vlayer->wkbType() ) )
671 {
672 layerPoint.dropZValue();
673 }
674
675 if ( !QgsWkbTypes::hasM( vlayer->wkbType() ) )
676 {
677 layerPoint.dropMValue();
678 }
679
680 return 0;
681 }
682 return 2;
683 }
684 else if ( QgsProject::instance()->topologicalEditing() && ( match.hasEdge() || match.hasMiddleSegment() ) )
685 {
686 layerPoint = toLayerCoordinates( vlayer, match.interpolatedPoint( mCanvas->mapSettings().destinationCrs() ) );
687 return 0;
688 }
689 }
690 return 2;
691}
692
694{
695 return addVertex( point, QgsPointLocator::Match() );
696}
697
699{
700 if ( mode() == CaptureNone )
701 {
702 QgsDebugError( QStringLiteral( "invalid capture mode" ) );
703 return 2;
704 }
705
706 if ( mCapturing && mCurrentCaptureTechnique == Qgis::CaptureTechnique::Streaming && !mAllowAddingStreamingPoints )
707 return 0;
708
709 QgsPoint layerPoint;
710 if ( layer() )
711 {
712 int res = fetchLayerPoint( match, layerPoint );
713 if ( res != 0 )
714 {
715 res = nextPoint( QgsPoint( point ), layerPoint );
716 if ( res != 0 )
717 {
718 return res;
719 }
720 }
721 }
722 else
723 {
724 layerPoint = QgsPoint( point );
725 }
726 const QgsPoint mapPoint = toMapCoordinates( layer(), layerPoint );
727
728 if ( mCaptureMode == CapturePoint )
729 {
730 mCaptureCurve.addVertex( layerPoint );
731 mSnappingMatches.append( match );
732 }
733 else
734 {
735 if ( mCaptureFirstPoint.isEmpty() )
736 {
737 mCaptureFirstPoint = mapPoint;
738 }
739
740 if ( !mRubberBand )
742
743 if ( !mTempRubberBand )
744 {
745 mTempRubberBand.reset( createCurveRubberBand() );
746 mTempRubberBand->setStringType( mLineDigitizingType );
747 mTempRubberBand->reset( mCaptureMode == CapturePolygon ? Qgis::GeometryType::Polygon : Qgis::GeometryType::Line, mLineDigitizingType, mapPoint );
748 }
749
750 bool traceCreated = false;
751 if ( tracingEnabled() )
752 {
753 traceCreated = tracingAddVertex( mapPoint );
754 }
755
756 // keep new tracing start point if we created a trace. This is useful when tracing with
757 // offset so that the user stays "snapped"
758 mTracingStartPoint = traceCreated ? point : QgsPointXY();
759
760 if ( !traceCreated )
761 {
762 // ordinary digitizing
763 mTempRubberBand->movePoint( mapPoint ); //move the last point of the temp rubberband before operating with it
764 if ( mTempRubberBand->curveIsComplete() ) //2 points for line and 3 points for circular
765 {
766 if ( QgsCurve *curve = mTempRubberBand->curve() )
767 {
768 addCurve( curve );
769 // add curve append only invalid match to mSnappingMatches,
770 // so we need to remove them and add the one from here if it is valid
771 if ( match.isValid() && mSnappingMatches.count() > 0 && !mSnappingMatches.last().isValid() )
772 {
773 mSnappingMatches.removeLast();
774 if ( mTempRubberBand->stringType() == Qgis::WkbType::CircularString )
775 {
776 // for circular string two points are added and match for intermediate point is stored
777 mSnappingMatches.removeLast();
778 mSnappingMatches.append( mCircularIntermediateMatch );
779 }
780 mSnappingMatches.append( match );
781 }
782 }
783 mCaptureLastPoint = mapPoint;
784 mTempRubberBand->reset( mCaptureMode == CapturePolygon ? Qgis::GeometryType::Polygon : Qgis::GeometryType::Line, mLineDigitizingType, mCaptureFirstPoint );
785 }
786 else if ( mTempRubberBand->pointsCount() == 0 )
787 {
788 mCaptureLastPoint = mapPoint;
789 mCaptureCurve.addVertex( layerPoint );
790 mSnappingMatches.append( match );
791 }
792 else
793 {
794 if ( mTempRubberBand->stringType() == Qgis::WkbType::CircularString )
795 {
796 mCircularIntermediateMatch = match;
797 }
798 }
799
800 mTempRubberBand->addPoint( mapPoint );
801 }
802 else
803 {
804 mTempRubberBand->reset( mCaptureMode == CapturePolygon ? Qgis::GeometryType::Polygon : Qgis::GeometryType::Line, mLineDigitizingType, mCaptureFirstPoint );
805 mTempRubberBand->addPoint( mCaptureLastPoint );
806 }
807 }
808
809 updateExtraSnapLayer();
810 validateGeometry();
811
812 return 0;
813}
814
816{
817 if ( !c )
818 {
819 return 1;
820 }
821
822 if ( !mRubberBand )
823 {
825 }
826
827 if ( mTempRubberBand )
828 {
829 mTempRubberBand->reset( mCaptureMode == CapturePolygon ? Qgis::GeometryType::Polygon : Qgis::GeometryType::Line, mLineDigitizingType, mCaptureFirstPoint );
830 const QgsPoint endPt = c->endPoint();
831 mTempRubberBand->addPoint( endPt ); //add last point of c
832 }
833
834 const int countBefore = mCaptureCurve.vertexCount();
835 //if there is only one point, this the first digitized point that are in the this first curve added --> remove the point
836 if ( mCaptureCurve.numPoints() == 1 )
837 mCaptureCurve.removeCurve( 0 );
838
839 // Transform back to layer CRS in case map CRS and layer CRS are different
840 const QgsCoordinateTransform ct = mCanvas->mapSettings().layerTransform( layer() );
841 if ( ct.isValid() && !ct.isShortCircuited() )
842 {
843 QgsLineString *segmented = c->curveToLine();
845 // Curve geometries will be converted to segments, so we explicitly set extentPrevious to false
846 // to be able to remove the whole curve in undo
847 mCaptureCurve.addCurve( segmented, false );
848 delete c;
849 }
850 else
851 {
852 // we set the extendPrevious option to true to avoid creating compound curves with many 2 vertex linestrings -- instead we prefer
853 // to extend linestring curves so that they continue the previous linestring wherever possible...
854 mCaptureCurve.addCurve( c, !mStartNewCurve );
855 }
856
857 mStartNewCurve = false;
858
859 const int countAfter = mCaptureCurve.vertexCount();
860 const int addedPoint = countAfter - countBefore;
861
862 updateExtraSnapLayer();
863
864 for ( int i = 0; i < addedPoint; ++i )
865 mSnappingMatches.append( QgsPointLocator::Match() );
866
867 resetRubberBand();
868
869 return 0;
870}
871
873{
874 mCaptureCurve.clear();
875 updateExtraSnapLayer();
876}
877
878QList<QgsPointLocator::Match> QgsMapToolCapture::snappingMatches() const
879{
880 return mSnappingMatches;
881}
882
883void QgsMapToolCapture::undo( bool isAutoRepeat )
884{
885 mTracingStartPoint = QgsPointXY();
886
887 if ( mTempRubberBand )
888 {
889 if ( size() <= 1 && mTempRubberBand->pointsCount() != 0 )
890 return;
891
892 if ( isAutoRepeat && mIgnoreSubsequentAutoRepeatUndo )
893 return;
894 mIgnoreSubsequentAutoRepeatUndo = false;
895
896 const QgsPoint lastPoint = mTempRubberBand->lastPoint();
897
898 if ( mTempRubberBand->stringType() == Qgis::WkbType::CircularString && mTempRubberBand->pointsCount() > 2 )
899 {
900 mTempRubberBand->removeLastPoint();
901 mTempRubberBand->movePoint( lastPoint );
902 return;
903 }
904
905 QgsVertexId vertexToRemove;
906 vertexToRemove.part = 0;
907 vertexToRemove.ring = 0;
908 vertexToRemove.vertex = size() - 1;
909
910 // If the geometry was reprojected, remove the entire last curve.
911 const QgsCoordinateTransform ct = mCanvas->mapSettings().layerTransform( layer() );
912 if ( ct.isValid() && !ct.isShortCircuited() )
913 {
914 mCaptureCurve.removeCurve( mCaptureCurve.nCurves() - 1 );
915 }
916 if ( mCaptureCurve.numPoints() == 2 && mCaptureCurve.nCurves() == 1 )
917 {
918 // store the first vertex to restore if after deleting the curve
919 // because when only two vertices, removing a point remove all the curve
920 const QgsPoint fp = mCaptureCurve.startPoint();
921 mCaptureCurve.deleteVertex( vertexToRemove );
922 mCaptureCurve.addVertex( fp );
923 }
924 else
925 {
926 const int curvesBefore = mCaptureCurve.nCurves();
927 const bool lastCurveIsLineString = qgsgeometry_cast<const QgsLineString *>( mCaptureCurve.curveAt( curvesBefore - 1 ) );
928
929 const int pointsCountBefore = mCaptureCurve.numPoints();
930 mCaptureCurve.deleteVertex( vertexToRemove );
931 int pointsCountAfter = mCaptureCurve.numPoints();
932 for ( ; pointsCountAfter < pointsCountBefore; pointsCountAfter++ )
933 if ( !mSnappingMatches.empty() )
934 mSnappingMatches.removeLast();
935
936 // if we have removed the last point in a linestring curve, then we "stick" here and ignore subsequent
937 // autorepeat undo actions until the user releases the undo key and holds it down again. This allows
938 // users to selectively remove portions of the geometry captured with the streaming mode by holding down
939 // the undo key, without risking accidental undo of non-streamed portions.
940 if ( mCaptureCurve.nCurves() < curvesBefore && lastCurveIsLineString )
941 mIgnoreSubsequentAutoRepeatUndo = true;
942 }
943
944 updateExtraSnapLayer();
945
946 resetRubberBand();
947
948 mTempRubberBand->reset( mCaptureMode == CapturePolygon ? Qgis::GeometryType::Polygon : Qgis::GeometryType::Line, mLineDigitizingType, mCaptureFirstPoint );
949
950 if ( mCaptureCurve.numPoints() > 0 )
951 {
952 const QgsPoint lastPt = mCaptureCurve.endPoint();
953 mCaptureLastPoint = toMapCoordinates( layer(), lastPt );
954 mTempRubberBand->addPoint( mCaptureLastPoint );
955 mTempRubberBand->movePoint( lastPoint );
956 }
957
959 validateGeometry();
960 }
961}
962
964{
965 if ( mCurrentCaptureTechnique == Qgis::CaptureTechnique::Shape && mCurrentShapeMapTool )
966 {
967 mCurrentShapeMapTool->keyPressEvent( e );
968 if ( e->isAccepted() )
969 return;
970 }
971
972 // this is backwards, but we can't change now without breaking api because
973 // forever QgsMapTools have had to explicitly mark events as ignored in order to
974 // indicate that they've consumed the event and that the default behavior should not
975 // be applied..!
976 // see QgsMapCanvas::keyPressEvent
977 e->accept();
978
979 if ( e->key() == Qt::Key_Backspace || e->key() == Qt::Key_Delete )
980 {
981 if ( mCurrentCaptureTechnique == Qgis::CaptureTechnique::Shape && mCurrentShapeMapTool )
982 {
983 if ( !e->isAutoRepeat() )
984 {
985 mCurrentShapeMapTool->undo();
986 }
987 }
988 else
989 {
990 undo( e->isAutoRepeat() );
991 }
992
993 // Override default shortcut management in MapCanvas
994 e->ignore();
995 }
996 else if ( e->key() == Qt::Key_Escape )
997 {
998 if ( mCurrentShapeMapTool )
999 mCurrentShapeMapTool->clean();
1000
1001 stopCapturing();
1002
1003 // Override default shortcut management in MapCanvas
1004 e->ignore();
1005 }
1006}
1007
1009{
1010 mCapturing = true;
1011}
1012
1014{
1015 return mCapturing;
1016}
1017
1019{
1020 mRubberBand.reset();
1021
1023
1024 qDeleteAll( mGeomErrorMarkers );
1025 mGeomErrorMarkers.clear();
1026 mGeomErrors.clear();
1027
1028 mCaptureFirstPoint = QgsPoint();
1029 mCaptureLastPoint = QgsPoint();
1030
1031 mTracingStartPoint = QgsPointXY();
1032
1033 mCapturing = false;
1034 mCaptureCurve.clear();
1035 updateExtraSnapLayer();
1036 mSnappingMatches.clear();
1037 if ( auto *lCurrentVectorLayer = currentVectorLayer() )
1038 lCurrentVectorLayer->triggerRepaint();
1039}
1040
1042{
1043 mTempRubberBand.reset();
1044}
1045
1047{
1048 stopCapturing();
1049 if ( mCurrentCaptureTechnique == Qgis::CaptureTechnique::Shape && mCurrentShapeMapTool )
1050 mCurrentShapeMapTool->clean();
1051
1052 clearCurve();
1053}
1054
1056{
1057 mCaptureCurve.close();
1058 updateExtraSnapLayer();
1059}
1060
1061void QgsMapToolCapture::validateGeometry()
1062{
1064 || !( capabilities() & ValidateGeometries ) )
1065 return;
1066
1067 if ( mValidator )
1068 {
1069 mValidator->deleteLater();
1070 mValidator = nullptr;
1071 }
1072
1073 mGeomErrors.clear();
1074 while ( !mGeomErrorMarkers.isEmpty() )
1075 {
1076 delete mGeomErrorMarkers.takeFirst();
1077 }
1078
1079 QgsGeometry geom;
1080
1081 switch ( mCaptureMode )
1082 {
1083 case CaptureNone:
1084 case CapturePoint:
1085 return;
1086 case CaptureLine:
1087 if ( size() < 2 )
1088 return;
1089 geom = QgsGeometry( mCaptureCurve.curveToLine() );
1090 break;
1091 case CapturePolygon:
1092 if ( size() < 3 )
1093 return;
1094 QgsLineString *exteriorRing = mCaptureCurve.curveToLine();
1095 exteriorRing->close();
1096 QgsPolygon *polygon = new QgsPolygon();
1097 polygon->setExteriorRing( exteriorRing );
1098 geom = QgsGeometry( polygon );
1099 break;
1100 }
1101
1102 if ( geom.isNull() )
1103 return;
1104
1108 mValidator = new QgsGeometryValidator( geom, nullptr, method );
1109 connect( mValidator, &QgsGeometryValidator::errorFound, this, &QgsMapToolCapture::addError );
1110 mValidator->start();
1111 QgsDebugMsgLevel( QStringLiteral( "Validation started" ), 4 );
1112}
1113
1114void QgsMapToolCapture::addError( const QgsGeometry::Error &e )
1115{
1116 mGeomErrors << e;
1117 QgsVectorLayer *vlayer = qobject_cast<QgsVectorLayer *>( layer() );
1118 if ( !vlayer )
1119 return;
1120
1121 if ( e.hasWhere() )
1122 {
1124 vm->setCenter( mCanvas->mapSettings().layerToMapCoordinates( vlayer, e.where() ) );
1126 vm->setPenWidth( 2 );
1127 vm->setToolTip( e.what() );
1128 vm->setColor( Qt::green );
1129 vm->setZValue( vm->zValue() + 1 );
1130 mGeomErrorMarkers << vm;
1131 }
1132}
1133
1135{
1136 return mCaptureCurve.numPoints();
1137}
1138
1139QVector<QgsPointXY> QgsMapToolCapture::points() const
1140{
1141 QVector<QgsPointXY> pointsXY;
1143
1144 return pointsXY;
1145}
1146
1148{
1149 QgsPointSequence pts;
1150 mCaptureCurve.points( pts );
1151 return pts;
1152}
1153
1154void QgsMapToolCapture::setPoints( const QVector<QgsPointXY> &pointList )
1155{
1156 QgsLineString *line = new QgsLineString( pointList );
1157 mCaptureCurve.clear();
1158 mCaptureCurve.addCurve( line );
1159 updateExtraSnapLayer();
1160 mSnappingMatches.clear();
1161 for ( int i = 0; i < line->length(); ++i )
1162 mSnappingMatches.append( QgsPointLocator::Match() );
1163 resetRubberBand();
1164}
1165
1167{
1168 QgsLineString *line = new QgsLineString( pointList );
1169 mCaptureCurve.clear();
1170 mCaptureCurve.addCurve( line );
1171 updateExtraSnapLayer();
1172 mSnappingMatches.clear();
1173 for ( int i = 0; i < line->length(); ++i )
1174 mSnappingMatches.append( QgsPointLocator::Match() );
1175 resetRubberBand();
1176}
1177
1179{
1180 QgsPoint newPoint( Qgis::WkbType::Point, point.x(), point.y() );
1181
1182 // get current layer
1183 QgsVectorLayer *vlayer = qobject_cast<QgsVectorLayer *>( layer() );
1184 if ( !vlayer )
1185 {
1186 return newPoint;
1187 }
1188
1189 // convert to the corresponding type for a full ZM support
1190 const Qgis::WkbType type = vlayer->wkbType();
1191 if ( QgsWkbTypes::hasZ( type ) && !QgsWkbTypes::hasM( type ) )
1192 {
1193 newPoint.convertTo( Qgis::WkbType::PointZ );
1194 }
1195 else if ( !QgsWkbTypes::hasZ( type ) && QgsWkbTypes::hasM( type ) )
1196 {
1197 newPoint.convertTo( Qgis::WkbType::PointM );
1198 }
1199 else if ( QgsWkbTypes::hasZ( type ) && QgsWkbTypes::hasM( type ) )
1200 {
1202 }
1203
1204 // set z value if necessary
1205 if ( QgsWkbTypes::hasZ( newPoint.wkbType() ) )
1206 {
1208 }
1209 // set m value if necessary
1210 if ( QgsWkbTypes::hasM( newPoint.wkbType() ) )
1211 {
1213 }
1214 return newPoint;
1215}
1216
1218{
1219 QgsPoint newPoint = mapPoint( e.mapPoint() );
1220
1221 // set z or m value from snapped point if necessary
1222 if ( QgsWkbTypes::hasZ( newPoint.wkbType() ) || QgsWkbTypes::hasM( newPoint.wkbType() ) )
1223 {
1224 // if snapped, z and m dimension are taken from the corresponding snapped
1225 // point.
1226 if ( e.isSnapped() )
1227 {
1228 const QgsPointLocator::Match match = e.mapPointMatch();
1229
1230 if ( match.layer() )
1231 {
1232 const QgsFeature ft = match.layer()->getFeature( match.featureId() );
1233 if ( QgsWkbTypes::hasZ( match.layer()->wkbType() ) )
1234 {
1235 newPoint.setZ( ft.geometry().vertexAt( match.vertexIndex() ).z() );
1236 }
1237 if ( QgsWkbTypes::hasM( match.layer()->wkbType() ) )
1238 {
1239 newPoint.setM( ft.geometry().vertexAt( match.vertexIndex() ).m() );
1240 }
1241 }
1242 }
1243 }
1244
1245 return newPoint;
1246}
1247
1248void QgsMapToolCapture::updateExtraSnapLayer()
1249{
1250 if ( !mExtraSnapLayer )
1251 return;
1252
1253 if ( canvas()->snappingUtils()->config().selfSnapping() && layer() && mCaptureCurve.numPoints() >= 2 )
1254 {
1255 // the current layer may have changed
1256 mExtraSnapLayer->setCrs( layer()->crs() );
1257 QgsGeometry geom = QgsGeometry( mCaptureCurve.clone() );
1258 // we close the curve to allow snapping on last segment
1259 if ( mCaptureMode == CapturePolygon && mCaptureCurve.numPoints() >= 3 )
1260 {
1261 qgsgeometry_cast<QgsCompoundCurve *>( geom.get() )->close();
1262 }
1263 mExtraSnapLayer->changeGeometry( mExtraSnapFeatureId, geom );
1264 }
1265 else
1266 {
1267 QgsGeometry geom;
1268 mExtraSnapLayer->changeGeometry( mExtraSnapFeatureId, geom );
1269 }
1270}
1271
1272
1274{
1275 // POINT CAPTURING
1276 if ( mode() == CapturePoint )
1277 {
1278 if ( e->button() != Qt::LeftButton )
1279 return;
1280
1281 QgsPoint savePoint; //point in layer coordinates
1282 bool isMatchPointZ = false;
1283 bool isMatchPointM = false;
1284 try
1285 {
1286 QgsPoint fetchPoint;
1287 int res = fetchLayerPoint( e->mapPointMatch(), fetchPoint );
1288 isMatchPointZ = QgsWkbTypes::hasZ( fetchPoint.wkbType() );
1289 isMatchPointM = QgsWkbTypes::hasM( fetchPoint.wkbType() );
1290
1291 if ( res == 0 )
1292 {
1294 if ( isMatchPointM && isMatchPointZ )
1295 {
1296 geomType = Qgis::WkbType::PointZM;
1297 }
1298 else if ( isMatchPointM )
1299 {
1300 geomType = Qgis::WkbType::PointM;
1301 }
1302 else if ( isMatchPointZ )
1303 {
1304 geomType = Qgis::WkbType::PointZ;
1305 }
1306 savePoint = QgsPoint( geomType, fetchPoint.x(), fetchPoint.y(), fetchPoint.z(), fetchPoint.m() );
1307 }
1308 else
1309 {
1310 QgsPointXY point = mCanvas->mapSettings().mapToLayerCoordinates( layer(), e->mapPoint() );
1311
1312 savePoint = QgsPoint( point.x(), point.y(), fetchPoint.z(), fetchPoint.m() );
1313 }
1314 }
1315 catch ( QgsCsException &cse )
1316 {
1317 Q_UNUSED( cse )
1318 emit messageEmitted( tr( "Cannot transform the point to the layer's coordinate system" ), Qgis::MessageLevel::Warning );
1319 return;
1320 }
1321
1322 QgsGeometry g( std::make_unique<QgsPoint>( savePoint ) );
1323
1324 // The snapping result needs to be added so it's available in the @snapping_results variable of default value etc. expression contexts
1325 addVertex( e->mapPoint(), e->mapPointMatch() );
1326
1327 geometryCaptured( g );
1328 pointCaptured( savePoint );
1329
1330 stopCapturing();
1331
1332 // we are done with digitizing for now so instruct advanced digitizing dock to reset its CAD points
1334 }
1335
1336 // LINE AND POLYGON CAPTURING
1337 else if ( mode() == CaptureLine || mode() == CapturePolygon )
1338 {
1339 bool digitizingFinished = false;
1340
1341 if ( mCurrentCaptureTechnique == Qgis::CaptureTechnique::Shape )
1342 {
1343 if ( !mCurrentShapeMapTool )
1344 {
1345 emit messageEmitted( tr( "Select an option from the Shape Digitizing Toolbar in order to capture shapes" ), Qgis::MessageLevel::Warning );
1346 return;
1347 }
1348 else
1349 {
1350 if ( !mTempRubberBand )
1351 {
1352 mTempRubberBand.reset( createCurveRubberBand() );
1353 mTempRubberBand->setStringType( mLineDigitizingType );
1354 mTempRubberBand->setRubberBandGeometryType( mCaptureMode == CapturePolygon ? Qgis::GeometryType::Polygon : Qgis::GeometryType::Line );
1355 }
1356
1357 digitizingFinished = mCurrentShapeMapTool->cadCanvasReleaseEvent( e, mCaptureMode );
1358 if ( digitizingFinished )
1359 mCurrentShapeMapTool->clean();
1360 }
1361 }
1362 else // i.e. not shape
1363 {
1364 //add point to list and to rubber band
1365 if ( e->button() == Qt::LeftButton )
1366 {
1367 const int error = addVertex( e->mapPoint(), e->mapPointMatch() );
1368 if ( error == 2 )
1369 {
1370 //problem with coordinate transformation
1371 emit messageEmitted( tr( "Cannot transform the point to the layers coordinate system" ), Qgis::MessageLevel::Warning );
1372 return;
1373 }
1374
1376 }
1377 else if ( e->button() == Qt::RightButton )
1378 {
1379 // End of string
1381
1382 //lines: bail out if there are not at least two vertices
1383 if ( mode() == CaptureLine && size() < 2 )
1384 {
1385 stopCapturing();
1386 return;
1387 }
1388
1389 //polygons: bail out if there are not at least two vertices
1390 if ( mode() == CapturePolygon && size() < 3 )
1391 {
1392 stopCapturing();
1393 return;
1394 }
1395
1396 if ( mode() == CapturePolygon || e->modifiers() == Qt::ShiftModifier )
1397 {
1398 closePolygon();
1399 }
1400
1401 digitizingFinished = true;
1402 }
1403 }
1404
1405 if ( digitizingFinished )
1406 {
1407 QgsGeometry g;
1408 std::unique_ptr<QgsCurve> curveToAdd( captureCurve()->clone() );
1409
1410 if ( mode() == CaptureLine )
1411 {
1412 g = QgsGeometry( curveToAdd->clone() );
1413 geometryCaptured( g );
1414 lineCaptured( curveToAdd.release() );
1415 }
1416 else
1417 {
1418 //does compoundcurve contain circular strings?
1419 //does provider support circular strings?
1420 if ( QgsVectorLayer *vlayer = qobject_cast<QgsVectorLayer *>( layer() ) )
1421 {
1422 const bool hasCurvedSegments = captureCurve()->hasCurvedSegments();
1423 const bool providerSupportsCurvedSegments = vlayer->dataProvider()->capabilities() & Qgis::VectorProviderCapability::CircularGeometries;
1424
1425 if ( hasCurvedSegments && providerSupportsCurvedSegments )
1426 {
1427 curveToAdd.reset( captureCurve()->clone() );
1428 }
1429 else
1430 {
1431 curveToAdd.reset( captureCurve()->curveToLine() );
1432 }
1433 }
1434 else
1435 {
1436 curveToAdd.reset( captureCurve()->clone() );
1437 }
1438 std::unique_ptr<QgsCurvePolygon> poly { new QgsCurvePolygon() };
1439 poly->setExteriorRing( curveToAdd.release() );
1440 g = QgsGeometry( poly->clone() );
1441 geometryCaptured( g );
1442 polygonCaptured( poly.get() );
1443 }
1444
1445 stopCapturing();
1446 }
1447 }
1448}
void setParentOwner(QObject *parent)
Sets the parent object.
void reset(T *p=nullptr)
Will reset the managed pointer to p.
void reset(T *p=nullptr)
Will reset the managed pointer to p.
T * release()
Clears the pointer and returns it.
The Qgis class provides global constants for use throughout the application.
Definition qgis.h:54
@ CircularGeometries
Supports circular geometry types (circularstring, compoundcurve, curvepolygon)
CaptureTechnique
Capture technique.
Definition qgis.h:376
@ Shape
Digitize shapes.
@ StraightSegments
Default capture mode - capture occurs with straight line segments.
@ CircularString
Capture in circular strings.
@ Streaming
Streaming points digitizing mode (points are automatically added as the mouse cursor moves).
GeometryValidationEngine
Available engines for validating geometries.
Definition qgis.h:2047
@ QgisInternal
Use internal QgsGeometryValidator method.
@ Geos
Use GEOS validation methods.
@ Warning
Warning message.
Definition qgis.h:156
@ Polygon
Polygons.
WkbType
The WKB type describes the number of dimensions a geometry has.
Definition qgis.h:256
@ CompoundCurve
CompoundCurve.
@ LineString
LineString.
@ PointM
PointM.
@ CircularString
CircularString.
@ PointZ
PointZ.
@ PointZM
PointZM.
@ Reverse
Reverse/inverse transform (from destination to source)
bool isMeasure() const
Returns true if the geometry contains m values.
bool is3D() const
Returns true if the geometry is 3D and contains a z-value.
virtual QgsPoint vertexAt(QgsVertexId id) const =0
Returns the point corresponding to a specified vertex id.
Qgis::WkbType wkbType() const
Returns the WKB type of the geometry.
The QgsAdvancedDigitizingDockWidget class is a dockable widget used to handle the CAD tools on top of...
bool cadEnabled() const
determines if CAD tools are enabled or if map tools behaves "nomally"
void switchZM()
Determines if Z or M will be enabled.
double getLineM() const
Convenient method to get the M value from the line edit wiget.
void clearPoints()
Removes all points from the CAD point list.
void removePreviousPoint()
Removes previous point in the CAD point list.
double getLineZ() const
Convenient method to get the Z value from the line edit wiget.
QgsPoint currentPointV2(bool *exists=nullptr) const
The last point.
QgsPoint currentPointLayerCoordinates(QgsMapLayer *layer) const
Returns the last CAD point, in a map layer's coordinates.
static QCursor getThemeCursor(Cursor cursor)
Helper to get a theme cursor.
@ CapturePoint
Select and capture a point or a feature.
QgsLineString * curveToLine(double tolerance=M_PI_2/90, SegmentationToleranceType toleranceType=MaximumAngle) const override
Returns a new line string geometry corresponding to a segmentized approximation of the curve.
void close()
Appends first point if not already closed.
bool isEmpty() const override
Returns true if the geometry is empty.
int nCurves() const
Returns the number of curves in the geometry.
void removeCurve(int i)
Removes a curve from the geometry.
void addCurve(QgsCurve *c, bool extendPrevious=false)
Adds a curve to the geometry (takes ownership).
bool moveVertex(QgsVertexId position, const QgsPoint &newPos) override
Moves a vertex within the geometry.
bool deleteVertex(QgsVertexId position) override
Deletes a vertex within the geometry.
QgsCompoundCurve * clone() const override
Clones the geometry by performing a deep copy.
const QgsCurve * curveAt(int i) const
Returns the curve at the specified index.
void points(QgsPointSequence &pts) const override
Returns a list of points within the curve.
void clear() override
Clears the geometry, ie reset it to a null geometry.
bool hasCurvedSegments() const override
Returns true if the geometry contains curved segments.
QgsPoint startPoint() const override
Returns the starting point of the curve.
QgsPoint endPoint() const override
Returns the end point of the curve.
int numPoints() const override
Returns the number of points in the curve.
void addVertex(const QgsPoint &pt)
Adds a vertex to the end of the geometry.
Class for doing transforms between two map coordinate systems.
bool isShortCircuited() const
Returns true if the transform short circuits because the source and destination are equivalent.
bool isValid() const
Returns true if the coordinate transform is valid, ie both the source and destination CRS have been s...
Custom exception class for Coordinate Reference System related exceptions.
Curve polygon geometry type.
virtual void setExteriorRing(QgsCurve *ring)
Sets the exterior ring of the polygon.
Abstract base class for curved geometry type.
Definition qgscurve.h:35
QgsCurve * segmentize(double tolerance=M_PI_2/90, SegmentationToleranceType toleranceType=MaximumAngle) const override
Returns a geometry without curves.
Definition qgscurve.cpp:175
int vertexCount(int part=0, int ring=0) const override
Returns the number of vertices of which this geometry is built.
Definition qgscurve.cpp:180
virtual QgsPoint endPoint() const =0
Returns the end point of the curve.
bool nextFeature(QgsFeature &f)
Fetch next feature and stores in f, returns true on success.
This class wraps a request for features to a vector layer (or directly its vector data provider).
QgsFeatureRequest & setFilterFid(QgsFeatureId fid)
Sets the feature ID that should be fetched.
The feature class encapsulates a single feature including its unique ID, geometry and a list of field...
Definition qgsfeature.h:58
QgsFeatureId id
Definition qgsfeature.h:66
QgsGeometry geometry
Definition qgsfeature.h:69
void errorFound(const QgsGeometry::Error &error)
Sent when an error has been found during the validation process.
A geometry error.
bool hasWhere() const
true if the location available from
QgsPointXY where() const
The coordinates at which the error is located and should be visualized.
QString what() const
A human readable error message containing details about the error.
A geometry is the spatial representation of a feature.
bool vertexIdFromVertexNr(int number, QgsVertexId &id) const
Calculates the vertex ID from a vertex number.
QgsPoint vertexAt(int atVertex) const
Returns coordinates of a vertex.
QgsAbstractGeometry * get()
Returns a modifiable (non-const) reference to the underlying abstract geometry primitive.
const QgsAbstractGeometry * constGet() const
Returns a non-modifiable (const) reference to the underlying abstract geometry primitive.
QgsGeometry convertToCurves(double distanceTolerance=1e-8, double angleTolerance=1e-8) const
Attempts to convert a non-curved geometry into a curved geometry type (e.g.
static void convertPointList(const QVector< QgsPointXY > &input, QgsPointSequence &output)
Upgrades a point list from QgsPointXY to QgsPoint.
Qgis::WkbType wkbType() const
Returns type of the geometry as a WKB type (point / linestring / polygon etc.)
Line string geometry type, with support for z-dimension and m-values.
double length() const override
Returns the planar, 2-dimensional length of the geometry.
void transform(const QgsCoordinateTransform &ct, Qgis::TransformDirection d=Qgis::TransformDirection::Forward, bool transformZ=false) override
Transforms the geometry using a coordinate transform.
QgsLineString * curveToLine(double tolerance=M_PI_2/90, SegmentationToleranceType toleranceType=MaximumAngle) const override
Returns a new line string geometry corresponding to a segmentized approximation of the curve.
void close()
Closes the line string by appending the first point to the end of the line, if it is not already clos...
Extension of QgsTracer that provides extra functionality:
QAction * actionEnableSnapping() const
Access to action that user may use to toggle snapping on/off.
void reportError(PathError err, bool addingVertex)
Report a path finding error to the user.
QAction * actionEnableTracing() const
Access to action that user may use to toggle tracing on/off. May be nullptr if no action was associat...
static QgsMapCanvasTracer * tracerForCanvas(QgsMapCanvas *canvas)
Retrieve instance of this class associated with given canvas (if any).
Map canvas is a class for displaying all GIS data types on a canvas.
void currentLayerChanged(QgsMapLayer *layer)
Emitted when the current layer is changed.
QgsMapLayer * currentLayer()
returns current layer (set by legend widget)
Base class for all map layer types.
Definition qgsmaplayer.h:76
QgsCoordinateReferenceSystem crs
Definition qgsmaplayer.h:83
void setCrs(const QgsCoordinateReferenceSystem &srs, bool emitSignal=true)
Sets layer's spatial reference system.
A QgsMapMouseEvent is the result of a user interaction with the mouse on a QgsMapCanvas.
bool isSnapped() const
Returns true if there is a snapped point cached.
QgsPointXY mapPoint() const
mapPoint returns the point in coordinates
QgsPointLocator::Match mapPointMatch() const
Returns the matching data from the most recently snapped point.
The QgsMapToolAdvancedDigitizing class is a QgsMapTool which gives event directly in map coordinates ...
virtual void cadCanvasMoveEvent(QgsMapMouseEvent *e)
Override this method when subclassing this class.
QgsAdvancedDigitizingDockWidget * mCadDockWidget
void deactivate() override
Unregisters this maptool from the cad dock widget.
virtual QgsMapLayer * layer() const
Returns the layer associated with the map tool.
QgsAdvancedDigitizingDockWidget * cadDockWidget() const
void activate() override
Registers this maptool with the cad dock widget.
void deactivate() override
Unregisters this maptool from the cad dock widget.
void stopCapturing()
Stop capturing.
int size()
Number of points digitized.
CaptureMode mode() const
The capture mode.
QgsMapToolCapture(QgsMapCanvas *canvas, QgsAdvancedDigitizingDockWidget *cadDockWidget, CaptureMode mode)
constructor
void undo(bool isAutoRepeat=false)
Removes the last vertex from mRubberBand and mCaptureList.
QFlags< Capability > Capabilities
QgsPoint mapPoint(const QgsMapMouseEvent &e) const
Creates a QgsPoint with ZM support if necessary (according to the WkbType of the current layer).
void keyPressEvent(QKeyEvent *e) override
Intercept key events like Esc or Del to delete the last point.
void activate() override
Registers this maptool with the cad dock widget.
CaptureMode
Different capture modes.
@ CapturePolygon
Capture polygons.
@ CaptureNone
Do not capture / determine mode from layer geometry type.
@ CapturePoint
Capture points.
@ CaptureLine
Capture lines.
Q_DECL_DEPRECATED void setCircularDigitizingEnabled(bool enable)
Enable the digitizing with curve.
void deleteTempRubberBand()
Clean a temporary rubberband.
void clean() override
convenient method to clean members
void closePolygon()
Close an open polygon.
int addCurve(QgsCurve *c)
Adds a whole curve (e.g. circularstring) to the captured geometry. Curve must be in map CRS.
int fetchLayerPoint(const QgsPointLocator::Match &match, QgsPoint &layerPoint)
Fetches the original point from the source layer if it has the same CRS as the current layer.
QgsPointSequence pointsZM() const
List of digitized points.
Q_DECL_DEPRECATED void setPoints(const QVector< QgsPointXY > &pointList)
Set the points on which to work.
const QgsCompoundCurve * captureCurve() const
Gets the capture curve.
QList< QgsPointLocator::Match > snappingMatches() const
Returns a list of matches for each point on the captureCurve.
Q_DECL_DEPRECATED QVector< QgsPointXY > points() const
List of digitized points.
bool isCapturing() const
Are we currently capturing?
virtual bool supportsTechnique(Qgis::CaptureTechnique technique) const
Returns true if the tool supports the specified capture technique.
void setCurrentShapeMapTool(const QgsMapToolShapeMetadata *shapeMapToolMetadata)
Sets the current shape tool.
int addVertex(const QgsPointXY &point)
Adds a point to the rubber band (in map coordinates) and to the capture list (in layer coordinates)
@ ValidateGeometries
Tool supports geometry validation.
@ SupportsCurves
Supports curved geometries input.
void setCurrentCaptureTechnique(Qgis::CaptureTechnique technique)
Sets the current capture if it is supported by the map tool.
virtual QgsMapToolCapture::Capabilities capabilities() const
Returns flags containing the supported capabilities.
void clearCurve()
Clear capture curve.
int nextPoint(const QgsPoint &mapPoint, QgsPoint &layerPoint)
Converts a map point to layer coordinates.
Q_DECL_DEPRECATED void setStreamDigitizingEnabled(bool enable)
Toggles the stream digitizing mode.
void cadCanvasMoveEvent(QgsMapMouseEvent *e) override
Override this method when subclassing this class.
void startCapturing()
Start capturing.
QgsRubberBand * takeRubberBand()
Returns the rubberBand currently owned by this map tool and transfers ownership to the caller.
void cadCanvasReleaseEvent(QgsMapMouseEvent *e) override
Override this method when subclassing this class.
QgsRubberBand * createRubberBand(Qgis::GeometryType geometryType=Qgis::GeometryType::Line, bool alternativeBand=false)
Creates a rubber band with the color/line width from the QGIS settings.
static double defaultMValue()
Returns default M value.
QgsVectorLayer * currentVectorLayer()
Returns the current vector layer of the map canvas or 0.
static QColor digitizingFillColor()
Returns fill color for rubber bands (from global settings)
static double defaultZValue()
Returns default Z value.
static QColor digitizingStrokeColor()
Returns stroke color for rubber bands (from global settings)
static int digitizingStrokeWidth()
Returns stroke width for rubber bands (from global settings)
virtual void deactivate()
Deactivates the map tool.
virtual void cadCanvasMoveEvent(QgsMapMouseEvent *e, QgsMapToolCapture::CaptureMode mode)=0
Called for a mouse move event.
QString id() const
Returns the id of the shape tool (equivalent to the one from the metadata)
virtual void undo()
Called to undo last action (last point added)
virtual void activate(QgsMapToolCapture::CaptureMode mode, const QgsPoint &lastCapturedMapPoint)
Activates the map tool with the last captured map point.
virtual void clean()
Called to clean the map tool (after canceling the operation or when the digitization has finished)
virtual bool cadCanvasReleaseEvent(QgsMapMouseEvent *e, QgsMapToolCapture::CaptureMode mode)=0
Called for a mouse release event Must return true if the digitization has ended and the geometry is c...
virtual void keyPressEvent(QKeyEvent *e)
Filters a key press event Ignores the event in default implementation.
QgsMapToolShapeMetadata is a base class for shape map tools metadata to be used in QgsMapToolShapeReg...
virtual QgsMapToolShapeAbstract * factory(QgsMapToolCapture *parentlTool) const =0
Creates the shape map tool for the given parentTool Caller takes ownership of the returned object.
virtual QString id() const =0
Unique ID for the shape map tool.
QgsPoint toLayerCoordinates(const QgsMapLayer *layer, const QgsPoint &point)
Transforms a point from map coordinates to layer coordinates.
QgsMapCanvas * canvas() const
returns pointer to the tool's map canvas
QgsPointXY toMapCoordinates(QPoint point)
Transforms a point from screen coordinates to map coordinates.
virtual void setCursor(const QCursor &cursor)
Sets a user defined cursor.
QPointer< QgsMapCanvas > mCanvas
The pointer to the map canvas.
Definition qgsmaptool.h:359
void messageEmitted(const QString &message, Qgis::MessageLevel level=Qgis::MessageLevel::Info)
Emitted when a message should be shown to the user in the application message bar.
QPoint toCanvasCoordinates(const QgsPointXY &point) const
Transforms a point from map coordinates to screen coordinates.
bool isActive() const
Returns if the current map tool active on the map canvas.
A class to represent a 2D point.
Definition qgspointxy.h:60
double y
Definition qgspointxy.h:64
double x
Definition qgspointxy.h:63
Point geometry type, with support for z-dimension and m-values.
Definition qgspoint.h:49
bool addMValue(double mValue=0) override
Adds a measure to the geometry, initialized to a preset value.
Definition qgspoint.cpp:569
bool dropMValue() override
Drops any measure values which exist in the geometry.
Definition qgspoint.cpp:610
bool addZValue(double zValue=0) override
Adds a z-dimension to the geometry, initialized to a preset value.
Definition qgspoint.cpp:558
double z
Definition qgspoint.h:54
double x
Definition qgspoint.h:52
void setM(double m)
Sets the point's m-value.
Definition qgspoint.h:365
bool convertTo(Qgis::WkbType type) override
Converts the geometry to a specified type.
Definition qgspoint.cpp:627
bool isEmpty() const override
Returns true if the geometry is empty.
Definition qgspoint.cpp:736
void setZ(double z)
Sets the point's z-coordinate.
Definition qgspoint.h:350
bool dropZValue() override
Drops any z-dimensions which exist in the geometry.
Definition qgspoint.cpp:599
double m
Definition qgspoint.h:55
double y
Definition qgspoint.h:53
Polygon geometry type.
Definition qgspolygon.h:33
void setExteriorRing(QgsCurve *ring) override
Sets the exterior ring of the polygon.
static QgsProject * instance()
Returns the QgsProject singleton instance.
void snappingConfigChanged(const QgsSnappingConfig &config)
Emitted whenever the configuration for snapping has changed.
A class for drawing transient features (e.g.
void movePoint(const QgsPointXY &p, int geometryIndex=0, int ringIndex=0)
Moves the rubber band point specified by index.
void reset(Qgis::GeometryType geometryType=Qgis::GeometryType::Line)
Clears all the geometries in this rubberband.
const QgsPointXY * getPoint(int i, int j=0, int ringIndex=0) const
Returns a vertex.
int numberOfVertices() const
Returns count of vertices in all lists of mPoint.
void addGeometry(const QgsGeometry &geometry, QgsMapLayer *layer, bool doUpdate=true)
Adds the geometry of an existing feature to a rubberband This is useful for multi feature highlightin...
T value(const QString &dynamicKeyPart=QString()) const
Returns settings value.
static const QgsSettingsEntryInteger * settingsDigitizingStreamTolerance
Settings entry digitizing stream tolerance.
static const QgsSettingsEntryDouble * settingsDigitizingLineColorAlphaScale
Settings entry digitizing line color alpha scale.
static const QgsSettingsEntryDouble * settingsDigitizingConvertToCurveAngleTolerance
Settings entry digitizing convert to curve angle tolerance.
static const QgsSettingsEntryDouble * settingsDigitizingConvertToCurveDistanceTolerance
Settings entry digitizing convert to curve distance tolerance.
static const QgsSettingsEntryInteger * settingsDigitizingValidateGeometries
Settings entry digitizing validate geometries.
static const QgsSettingsEntryBool * settingsDigitizingConvertToCurve
Settings entry digitizing convert to curve.
Class that shows snapping marker on map canvas for the current snapping match.
bool isPointSnapped(const QgsPointXY &pt)
Find out whether the point is snapped to a vertex or edge (i.e. it can be used for tracing start/stop...
QVector< QgsPointXY > findShortestPath(const QgsPointXY &p1, const QgsPointXY &p2, PathError *error=nullptr)
Given two points, find the shortest path and return points on the way.
PathError
Possible errors that may happen when calling findShortestPath()
Definition qgstracer.h:132
@ ErrNone
No error.
Definition qgstracer.h:133
@ ErrTooManyFeatures
Max feature count threshold was reached while reading features.
Definition qgstracer.h:134
bool init()
Build the internal data structures.
virtual Q_INVOKABLE Qgis::VectorProviderCapabilities capabilities() const
Returns flags containing the supported capabilities.
Represents a vector layer which manages a vector based data sets.
Q_INVOKABLE bool startEditing()
Makes the layer editable.
bool isSpatial() const FINAL
Returns true if this is a geometry layer and false in case of NoGeometry (table only) or UnknownGeome...
QgsFeatureIterator getFeatures(const QgsFeatureRequest &request=QgsFeatureRequest()) const FINAL
Queries the layer for features specified in request.
Q_INVOKABLE Qgis::WkbType wkbType() const FINAL
Returns the WKBType or WKBUnknown in case of error.
QgsVectorDataProvider * dataProvider() FINAL
Returns the layer's data provider, it may be nullptr.
bool addFeature(QgsFeature &feature, QgsFeatureSink::Flags flags=QgsFeatureSink::Flags()) FINAL
Adds a single feature to the sink.
Q_INVOKABLE Qgis::GeometryType geometryType() const
Returns point, line or polygon.
Q_INVOKABLE QgsFeature getFeature(QgsFeatureId fid) const
Queries the layer for the feature with the given id.
bool changeGeometry(QgsFeatureId fid, QgsGeometry &geometry, bool skipDefaultValue=false)
Changes a feature's geometry within the layer's edit buffer (but does not immediately commit the chan...
A class for marking vertices of features using e.g.
void setPenWidth(int width)
void setCenter(const QgsPointXY &point)
Sets the center point of the marker, in map coordinates.
void setIconType(int iconType)
void setColor(const QColor &color)
Sets the stroke color for the marker.
static bool hasZ(Qgis::WkbType type)
Tests whether a WKB type contains the z-dimension.
static bool hasM(Qgis::WkbType type)
Tests whether a WKB type contains m values.
static Qgis::WkbType flatType(Qgis::WkbType type)
Returns the flat type for a WKB type.
As part of the API refactoring and improvements which landed in the Processing API was substantially reworked from the x version This was done in order to allow much of the underlying Processing framework to be ported into c
#define BUILTIN_UNREACHABLE
Definition qgis.h:6873
QVector< QgsPoint > QgsPointSequence
#define QgsDebugMsgLevel(str, level)
Definition qgslogger.h:41
#define QgsDebugError(str)
Definition qgslogger.h:40
const QgsCoordinateReferenceSystem & crs
QgsFeatureId featureId() const
The id of the feature to which the snapped geometry belongs.
QgsVectorLayer * layer() const
The vector layer where the snap occurred.
QgsPoint interpolatedPoint(const QgsCoordinateReferenceSystem &destinationCrs=QgsCoordinateReferenceSystem()) const
Convenient method to return a point on an edge with linear interpolation of the Z value.
bool hasEdge() const
Returns true if the Match is an edge.
bool hasLineEndpoint() const
Returns true if the Match is a line endpoint (start or end vertex).
bool hasMiddleSegment() const
Returns true if the Match is the middle of a segment.
int vertexIndex() const
for vertex / edge match (first vertex of the edge)
bool hasVertex() const
Returns true if the Match is a vertex.
Setting options for loading vector layers.
bool skipCrsValidation
Controls whether the layer is allowed to have an invalid/unknown CRS.
bool loadDefaultStyle
Set to true if the default layer style should be loaded.
Utility class for identifying a unique vertex within a geometry.
Definition qgsvertexid.h:30
int vertex
Vertex number.
Definition qgsvertexid.h:94
int part
Part number.
Definition qgsvertexid.h:88
int ring
Ring number.
Definition qgsvertexid.h:91