32#include "moc_qgsrubberband.cpp"
37 , mGeometryType( geometryType )
39 reset( geometryType );
40 QColor color( Qt::lightGray );
65 if ( mBrush.color() == color )
68 mBrush.setColor( color );
73 mPen.setColor( color );
78 mSecondaryPen.setColor( color );
83 mPen.setWidthF(
width );
94 mSvgRenderer = std::make_unique<QSvgRenderer>( path );
95 mSvgOffset = drawOffset;
105 mPen.setStyle( penStyle );
110 mBrush.setStyle( brushStyle );
116 mGeometryType = geometryType;
123 if ( geometryIndex < 0 )
125 geometryIndex = mPoints.size() - 1;
128 if ( geometryIndex < 0 || geometryIndex > mPoints.size() )
133 if ( geometryIndex == mPoints.size() )
144 if ( mPoints.at( geometryIndex ).isEmpty() )
147 ringIndex = mPoints.at( geometryIndex ).size() - 1;
150 if ( ringIndex > mPoints.at( geometryIndex ).size() )
153 if ( ringIndex == mPoints.at( geometryIndex ).size() )
157 mPoints[geometryIndex][ringIndex].append( p );
160 if ( mPoints.at( geometryIndex ).at( ringIndex ).size() == 2 && mPoints.at( geometryIndex ).at( ringIndex ).at( 0 ) == mPoints.at( geometryIndex ).at( ringIndex ).at( 1 ) )
162 mPoints[geometryIndex][ringIndex].last() = p;
166 mPoints[geometryIndex][ringIndex].append( p );
180 if ( geometryIndex < 0 || ringIndex < 0 || mPoints.size() <= geometryIndex || mPoints.at( geometryIndex ).size() <= ringIndex || mPoints.at( geometryIndex ).at( ringIndex ).isEmpty() )
185 if ( mPoints.at( geometryIndex ).at( ringIndex ).constFirst() != mPoints.at( geometryIndex ).at( ringIndex ).constLast() )
187 mPoints[geometryIndex][ringIndex].append( mPoints.at( geometryIndex ).at( ringIndex ).constFirst() );
201 if ( geometryIndex < 0 || ringIndex < 0 || mPoints.size() <= geometryIndex || mPoints.at( geometryIndex ).size() <= ringIndex || mPoints.at( geometryIndex ).at( ringIndex ).size() <= index || mPoints.at( geometryIndex ).at( ringIndex ).size() < -index || mPoints.at( geometryIndex ).at( ringIndex ).isEmpty() )
209 index = mPoints.at( geometryIndex ).at( ringIndex ).size() + index;
211 mPoints[geometryIndex][ringIndex].removeAt( index );
222 removePoint( -1, doUpdate, geometryIndex, ringIndex );
227 if ( geometryIndex < 0 || ringIndex < 0 || mPoints.size() <= geometryIndex || mPoints.at( geometryIndex ).size() <= ringIndex || mPoints.at( geometryIndex ).at( ringIndex ).isEmpty() )
232 mPoints[geometryIndex][ringIndex].last() = p;
240 if ( geometryIndex < 0 || ringIndex < 0 || index < 0 || mPoints.size() <= geometryIndex || mPoints.at( geometryIndex ).size() <= ringIndex || mPoints.at( geometryIndex ).at( ringIndex ).size() <= index )
245 mPoints[geometryIndex][ringIndex][index] = p;
255 reset( mGeometryType );
267 reset( mGeometryType );
304 int idx = mPoints.size();
316 QgsDebugError( QStringLiteral(
"Could not transform rubber band geometry to map CRS" ) );
349 if ( line.isEmpty() )
368 addPoint( pt,
false, idx, ringIdx );
378 if ( poly.isEmpty() )
386 addPoint( pt,
false, idx, ringIdx );
428 reset( other->mGeometryType );
429 mPoints = other->mPoints;
436 if ( mPoints.isEmpty() )
439 QVector<QVector<QPolygonF>> shapes;
440 shapes.reserve( mPoints.size() );
441 for (
const QgsPolygonXY &poly : std::as_const( mPoints ) )
443 QVector<QPolygonF> rings;
444 rings.reserve( poly.size() );
447 QVector<QPointF> pts;
448 pts.reserve( line.size() );
452 if ( pts.isEmpty() || std::abs( pts.last().x() - cur.x() ) > 1 || std::abs( pts.last().y() - cur.y() ) > 1 )
457 shapes.append( rings );
465 lineSymbol->startRender( context );
466 for (
const QVector<QPolygonF> &shape : std::as_const( shapes ) )
468 for (
const QPolygonF &ring : shape )
470 lineSymbol->renderPolyline( ring,
nullptr, context );
473 lineSymbol->stopRender( context );
480 fillSymbol->startRender( context );
481 for (
const QVector<QPolygonF> &shape : std::as_const( shapes ) )
483 for (
const QPolygonF &ring : shape )
485 fillSymbol->renderPolygon( ring,
nullptr,
nullptr, context );
488 fillSymbol->stopRender( context );
492 int iterations = mSecondaryPen.color().isValid() ? 2 : 1;
493 for (
int i = 0; i < iterations; ++i )
495 if ( i == 0 && iterations > 1 )
499 p->setBrush( Qt::NoBrush );
500 p->setPen( mSecondaryPen );
505 p->setBrush( mBrush );
509 for (
const QVector<QPolygonF> &shape : std::as_const( shapes ) )
519 if ( rings.size() == 1 )
526 for (
const QPolygonF &poly : rings )
528 path.addPolygon( poly );
536 switch ( mGeometryType )
540 p->drawPolygon( pts );
546 const auto constPts = pts;
547 for ( QPointF pt : constPts )
552 qreal s = ( mIconSize - 1 ) / 2.0;
560 p->drawLine( QLineF( x - s, y, x + s, y ) );
561 p->drawLine( QLineF( x, y - s, x, y + s ) );
565 p->drawLine( QLineF( x - s, y - s, x + s, y + s ) );
566 p->drawLine( QLineF( x - s, y + s, x + s, y - s ) );
570 p->drawLine( QLineF( x - s, y - s, x + s, y - s ) );
571 p->drawLine( QLineF( x + s, y - s, x + s, y + s ) );
572 p->drawLine( QLineF( x + s, y + s, x - s, y + s ) );
573 p->drawLine( QLineF( x - s, y + s, x - s, y - s ) );
577 p->drawRect( QRectF(
static_cast<int>( x - s ),
static_cast<int>( y - s ), mIconSize, mIconSize ) );
581 p->drawEllipse( QRectF(
static_cast<int>( x - s ),
static_cast<int>( y - s ), mIconSize, mIconSize ) );
594 p->drawPolygon( pts, 4 );
596 p->drawPolyline( pts, 4 );
602 QRectF viewBox = mSvgRenderer->viewBoxF();
603 QRectF r( mSvgOffset.x(), mSvgOffset.y(), viewBox.width(), viewBox.height() );
606 mSvgRenderer->render( p, r );
617 p->drawPolyline( pts );
625 if ( mPoints.isEmpty() )
635 double iconSize = ( mIconSize + 1 ) / 2.;
638 QRectF viewBox = mSvgRenderer->viewBoxF();
639 iconSize = std::max( std::fabs( mSvgOffset.x() ) + .5 * viewBox.width(), std::fabs( mSvgOffset.y() ) + .5 * viewBox.height() );
643 qreal w = ( ( mIconSize - 1 ) / 2 + mPen.widthF() );
646 for (
const QgsPolygonXY &poly : std::as_const( mPoints ) )
648 for (
const QgsPointXY &point : poly.at( 0 ) )
650 QgsPointXY p( point.x() + mTranslationOffsetX, point.y() + mTranslationOffsetY );
669 return mSymbol.get();
689 mTranslationOffsetX = dx;
690 mTranslationOffsetY = dy;
696 return mPoints.size();
701 if ( geometryIndex < 0 || geometryIndex >= mPoints.size() || mPoints.at( geometryIndex ).isEmpty() )
703 return mPoints.at( geometryIndex ).at( 0 ).size();
709 for (
const QgsPolygonXY &poly : std::as_const( mPoints ) )
713 count += ring.size();
721 if ( i < 0 || ringIndex < 0 || j < 0 || mPoints.size() <= i || mPoints.at( i ).size() <= ringIndex || mPoints.at( i ).at( ringIndex ).size() <= j )
724 return &mPoints[i][ringIndex][j];
731 switch ( mGeometryType )
743 for (
const QgsPolygonXY &poly : std::as_const( mPoints ) )
745 if ( poly.isEmpty() )
747 multiPoint.append( poly.at( 0 ) );
756 if ( !mPoints.isEmpty() )
758 if ( mPoints.size() > 1 )
761 for (
const QgsPolygonXY &poly : std::as_const( mPoints ) )
763 if ( poly.isEmpty() )
765 multiPolyline.append( poly.at( 0 ) );
771 if ( !mPoints.at( 0 ).isEmpty() )
GeometryType
The geometry types are used to group Qgis::WkbType in a coarse way.
@ Antialiasing
Use antialiasing while drawing.
WkbType
The WKB type describes the number of dimensions a geometry has.
Represents a coordinate reference system (CRS).
bool isValid() const
Returns whether this CRS is correctly initialized and usable.
Custom exception class for Coordinate Reference System related exceptions.
A fill symbol type, for rendering Polygon and MultiPolygon geometries.
A geometry is the spatial representation of a feature.
QgsMultiPolygonXY asMultiPolygon() const
Returns the contents of the geometry as a multi-polygon.
static QgsGeometry fromMultiPolylineXY(const QgsMultiPolylineXY &multiline)
Creates a new geometry from a QgsMultiPolylineXY object.
Qgis::GeometryOperationResult transform(const QgsCoordinateTransform &ct, Qgis::TransformDirection direction=Qgis::TransformDirection::Forward, bool transformZ=false)
Transforms this geometry as described by the coordinate transform ct.
static QgsGeometry fromPolylineXY(const QgsPolylineXY &polyline)
Creates a new LineString geometry from a list of QgsPointXY points.
QgsMultiPointXY asMultiPoint() const
Returns the contents of the geometry as a multi-point.
QgsPolygonXY asPolygon() const
Returns the contents of the geometry as a polygon.
static QgsGeometry fromMultiPointXY(const QgsMultiPointXY &multipoint)
Creates a new geometry from a QgsMultiPointXY object.
QgsPolylineXY asPolyline() const
Returns the contents of the geometry as a polyline.
QgsPointXY asPoint() const
Returns the contents of the geometry as a 2-dimensional point.
QgsMultiPolylineXY asMultiPolyline() const
Returns the contents of the geometry as a multi-linestring.
static QgsGeometry fromMultiPolygonXY(const QgsMultiPolygonXY &multipoly)
Creates a new geometry from a QgsMultiPolygonXY.
bool isEmpty() const
Returns true if the geometry is empty (eg a linestring with no vertices, or a collection with no geom...
Qgis::WkbType wkbType() const
Returns type of the geometry as a WKB type (point / linestring / polygon etc.).
A line symbol type, for rendering LineString and MultiLineString geometries.
An abstract class for items that can be placed on the map canvas.
QgsRectangle rect() const
returns canvas item rectangle in map units
QPointF toCanvasCoordinates(const QgsPointXY &point) const
transformation from map coordinates to screen coordinates
QgsMapCanvas * mMapCanvas
pointer to map canvas
void setRect(const QgsRectangle &r, bool resetRotation=true)
sets canvas item rectangle in map units
QgsMapCanvasItem(QgsMapCanvas *mapCanvas)
protected constructor: cannot be constructed directly
Map canvas is a class for displaying all GIS data types on a canvas.
Base class for all map layer types.
Contains configuration for rendering maps.
QgsCoordinateReferenceSystem destinationCrs() const
Returns the destination coordinate reference system for the map render.
Perform transforms between map coordinates and device coordinates.
double mapUnitsPerPixel() const
Returns the current map units per pixel.
QgsPointXY toMapCoordinates(int x, int y) const
Transforms device coordinates to map (world) coordinates.
QgsPointXY transform(const QgsPointXY &p) const
Transforms a point p from map (world) coordinates to device coordinates.
static QgsProject * instance()
Returns the QgsProject singleton instance.
A rectangle specified with double values.
void combineExtentWith(const QgsRectangle &rect)
Expands the rectangle so that it covers both the original rectangle and the given rectangle.
Contains information about the context of a rendering operation.
void setFlag(Qgis::RenderContextFlag flag, bool on=true)
Enable or disable a particular flag (other flags are not affected).
static QgsRenderContext fromQPainter(QPainter *painter)
Creates a default render context given a pixel based QPainter destination.
void setIconSize(double iconSize)
Sets the size of the point icons.
void closePoints(bool doUpdate=true, int geometryIndex=0, int ringIndex=0)
Ensures that a polygon geometry is closed and that the last vertex equals the first vertex.
~QgsRubberBand() override
void setSymbol(QgsSymbol *symbol)
Sets the symbol used for rendering the rubberband.
void removeLastPoint(int geometryIndex=0, bool doUpdate=true, int ringIndex=0)
Removes the last point.
void movePoint(const QgsPointXY &p, int geometryIndex=0, int ringIndex=0)
Moves the rubber band point specified by index.
QgsRubberBand(QgsMapCanvas *mapCanvas, Qgis::GeometryType geometryType=Qgis::GeometryType::Line)
Creates a new RubberBand.
QgsGeometry asGeometry() const
Returns the rubberband as a Geometry.
int size() const
Returns number of geometries.
void paint(QPainter *p) override
Paints the rubber band in response to an update event.
IconType icon() const
Returns the current icon type to highlight point geometries.
int partSize(int geometryIndex) const
Returns number of vertices in feature part.
void setSvgIcon(const QString &path, QPoint drawOffset)
Set the path to the svg file to use to draw points.
void setWidth(double width)
Sets the width of the line.
void reset(Qgis::GeometryType geometryType=Qgis::GeometryType::Line)
Clears all the geometries in this rubberband.
void setSecondaryStrokeColor(const QColor &color)
Sets a secondary stroke color for the rubberband which will be drawn under the main stroke color.
@ ICON_X
A cross is used to highlight points (x).
@ ICON_FULL_DIAMOND
A diamond is used to highlight points (◆).
@ ICON_FULL_BOX
A full box is used to highlight points (■).
@ ICON_NONE
No icon is used.
@ ICON_CROSS
A cross is used to highlight points (+).
@ ICON_SVG
An svg image is used to highlight points.
@ ICON_CIRCLE
A circle is used to highlight points (○).
@ ICON_DIAMOND
A diamond is used to highlight points (◇).
@ ICON_BOX
A box is used to highlight points (□).
const QgsPointXY * getPoint(int i, int j=0, int ringIndex=0) const
Returns a vertex.
void setColor(const QColor &color)
Sets the color for the rubberband.
void setToGeometry(const QgsGeometry &geom, QgsVectorLayer *layer)
Sets this rubber band to geom.
void setStrokeColor(const QColor &color)
Sets the stroke color for the rubberband.
void setLineStyle(Qt::PenStyle penStyle)
Sets the style of the line.
void updateRect()
Recalculates needed rectangle.
QgsSymbol * symbol() const
Returns the symbol used for rendering the rubberband, if set.
void setToCanvasRectangle(QRect rect)
Sets this rubber band to a map canvas rectangle.
void setIcon(IconType icon)
Sets the icon type to highlight point geometries.
void setBrushStyle(Qt::BrushStyle brushStyle)
Sets the style of the brush.
void updatePosition() override
called on changed extent or resize event to update position of the item
void copyPointsFrom(const QgsRubberBand *other)
Copies the points from another rubber band.
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...
void removePoint(int index=0, bool doUpdate=true, int geometryIndex=0, int ringIndex=0)
Removes a vertex from the rubberband and (optionally) updates canvas.
void setTranslationOffset(double dx, double dy)
Adds translation to original coordinates (all in map coordinates).
void setFillColor(const QColor &color)
Sets the fill color for the rubberband.
void drawShape(QPainter *p, const QVector< QPointF > &pts)
Draws shape of the rubber band.
void addPoint(const QgsPointXY &p, bool doUpdate=true, int geometryIndex=0, int ringIndex=0)
Adds a vertex to the rubberband and update canvas.
Scoped object for saving and restoring a QPainter object's state.
Abstract base class for all rendered symbols.
Represents a vector layer which manages a vector based dataset.
static Qgis::GeometryType geometryType(Qgis::WkbType type)
Returns the geometry type for a WKB type, e.g., both MultiPolygon and CurvePolygon would have a Polyg...
static Q_INVOKABLE bool isMultiType(Qgis::WkbType type)
Returns true if the WKB type is a multi type.
int scaleIconSize(int standardSize)
Scales an icon size to compensate for display pixel density, making the icon size hi-dpi friendly,...
QVector< QgsPolylineXY > QgsPolygonXY
Polygon: first item of the list is outer ring, inner rings (if any) start from second item.
QVector< QgsPolylineXY > QgsMultiPolylineXY
A collection of QgsPolylines that share a common collection of attributes.
QVector< QgsPointXY > QgsMultiPointXY
A collection of QgsPoints that share a common collection of attributes.
QVector< QgsPointXY > QgsPolylineXY
Polyline as represented as a vector of two-dimensional points.
QVector< QgsPolygonXY > QgsMultiPolygonXY
A collection of QgsPolygons that share a common collection of attributes.
#define QgsDebugError(str)