QGIS API Documentation 3.99.0-Master (d270888f95f)
Loading...
Searching...
No Matches
qgslayoutitemshape.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgslayoutitemshape.cpp
3 -----------------------
4 begin : July 2017
5 copyright : (C) 2017 by Nyall Dawson
6 email : nyall dot dawson at gmail dot com
7 ***************************************************************************/
8/***************************************************************************
9 * *
10 * This program is free software; you can redistribute it and/or modify *
11 * it under the terms of the GNU General Public License as published by *
12 * the Free Software Foundation; either version 2 of the License, or *
13 * (at your option) any later version. *
14 * *
15 ***************************************************************************/
16
17#include "qgslayoutitemshape.h"
18
19#include "qgsfillsymbol.h"
20#include "qgslayout.h"
21#include "qgslayoutmodel.h"
23#include "qgslayoututils.h"
25#include "qgssymbollayerutils.h"
26
27#include <QPainter>
28#include <QString>
29
30#include "moc_qgslayoutitemshape.cpp"
31
32using namespace Qt::StringLiterals;
33
36 , mCornerRadius( 0 )
37{
38 setBackgroundEnabled( false );
39 setFrameEnabled( false );
40 QVariantMap properties;
41 properties.insert( u"color"_s, u"white"_s );
42 properties.insert( u"style"_s, u"solid"_s );
43 properties.insert( u"style_border"_s, u"solid"_s );
44 properties.insert( u"color_border"_s, u"black"_s );
45 properties.insert( u"width_border"_s, u"0.3"_s );
46 properties.insert( u"joinstyle"_s, u"miter"_s );
47 mShapeStyleSymbol = QgsFillSymbol::createSimple( properties );
48 refreshSymbol( false );
49
50 connect( this, &QgsLayoutItemShape::sizePositionChanged, this, [this]
51 {
52 updateBoundingRect();
53 update();
54 emit clipPathChanged();
55 } );
56}
57
59
64
69
71{
72 switch ( mShape )
73 {
74 case Ellipse:
75 return QgsApplication::getThemeIcon( u"/mLayoutItemShapeEllipse.svg"_s );
76 case Rectangle:
77 return QgsApplication::getThemeIcon( u"/mLayoutItemShapeRectangle.svg"_s );
78 case Triangle:
79 return QgsApplication::getThemeIcon( u"/mLayoutItemShapeTriangle.svg"_s );
80 }
81
82 return QIcon();
83}
84
86{
87 if ( !id().isEmpty() )
88 {
89 return id();
90 }
91
92 switch ( mShape )
93 {
94 case Ellipse:
95 return tr( "<Ellipse>" );
96 case Rectangle:
97 return tr( "<Rectangle>" );
98 case Triangle:
99 return tr( "<Triangle>" );
100 }
101
102 return tr( "<Shape>" );
103}
104
111
113{
114 if ( type == mShape )
115 {
116 return;
117 }
118
119 mShape = type;
120
121 if ( mLayout && id().isEmpty() )
122 {
123 //notify the model that the display name has changed
124 mLayout->itemsModel()->updateItemDisplayName( this );
125 }
126
127 emit clipPathChanged();
128}
129
130void QgsLayoutItemShape::refreshSymbol( bool redraw )
131{
132 if ( auto *lLayout = layout() )
133 {
134 const QgsRenderContext rc = QgsLayoutUtils::createRenderContextForLayout( lLayout, nullptr, lLayout->renderContext().dpi() );
135 mMaxSymbolBleed = ( 25.4 / lLayout->renderContext().dpi() ) * QgsSymbolLayerUtils::estimateMaxSymbolBleed( mShapeStyleSymbol.get(), rc );
136 }
137
138 updateBoundingRect();
139
140 if ( redraw )
141 update();
142
143 emit frameChanged();
144}
145
146void QgsLayoutItemShape::updateBoundingRect()
147{
148 QRectF rectangle = rect();
149 rectangle.adjust( -mMaxSymbolBleed, -mMaxSymbolBleed, mMaxSymbolBleed, mMaxSymbolBleed );
150 if ( rectangle != mCurrentRectangle )
151 {
152 prepareGeometryChange();
153 mCurrentRectangle = rectangle;
154 }
155}
156
158{
159 if ( !symbol )
160 return;
161
162 mShapeStyleSymbol.reset( symbol->clone() );
163 refreshSymbol( true );
164}
165
167{
168 mCornerRadius = radius;
169 emit clipPathChanged();
170}
171
173{
174 QPolygonF shapePolygon = mapToScene( calculatePolygon( 1.0 ) );
175 // ensure polygon is closed
176 if ( shapePolygon.at( 0 ) != shapePolygon.constLast() )
177 shapePolygon << shapePolygon.at( 0 );
178 return QgsGeometry::fromQPolygonF( shapePolygon );
179}
180
182{
183 return mCurrentRectangle;
184}
185
187{
188 return mMaxSymbolBleed;
189}
190
192{
193 if ( mShapeStyleSymbol )
194 {
195 QgsStyleSymbolEntity entity( mShapeStyleSymbol.get() );
196 if ( !visitor->visit( QgsStyleEntityVisitorInterface::StyleLeaf( &entity, uuid(), displayName() ) ) )
197 return false;
198 }
199
200 return true;
201}
202
204{
205 QgsRenderContext renderContext = context.renderContext();
206 // symbol clipping messes with geometry generators used in the symbol for this item, and has no
207 // valid use here. See https://github.com/qgis/QGIS/issues/58909
209
210 QPainter *painter = renderContext.painter();
211 painter->setPen( Qt::NoPen );
212 painter->setBrush( Qt::NoBrush );
213
214 const double scale = renderContext.convertToPainterUnits( 1, Qgis::RenderUnit::Millimeters );
215
216 const QVector<QPolygonF> rings; //empty list
217
218 symbol()->startRender( renderContext );
219 symbol()->renderPolygon( calculatePolygon( scale ), &rings, nullptr, renderContext );
220 symbol()->stopRender( renderContext );
221}
222
223QPolygonF QgsLayoutItemShape::calculatePolygon( double scale ) const
224{
225 QPolygonF shapePolygon;
226
227 //shapes with curves must be enlarged before conversion to QPolygonF, or
228 //the curves are approximated too much and appear jaggy
229 const QTransform t = QTransform::fromScale( 100, 100 );
230 //inverse transform used to scale created polygons back to expected size
231 const QTransform ti = t.inverted();
232
233 switch ( mShape )
234 {
235 case Ellipse:
236 {
237 //create an ellipse
238 QPainterPath ellipsePath;
239 ellipsePath.addEllipse( QRectF( 0, 0, rect().width() * scale, rect().height() * scale ) );
240 const QPolygonF ellipsePoly = ellipsePath.toFillPolygon( t );
241 shapePolygon = ti.map( ellipsePoly );
242 break;
243 }
244 case Rectangle:
245 {
246 //if corner radius set, then draw a rounded rectangle
247 if ( mCornerRadius.length() > 0 )
248 {
249 QPainterPath roundedRectPath;
250 const double radius = mLayout->convertToLayoutUnits( mCornerRadius ) * scale;
251 roundedRectPath.addRoundedRect( QRectF( 0, 0, rect().width() * scale, rect().height() * scale ), radius, radius );
252 const QPolygonF roundedPoly = roundedRectPath.toFillPolygon( t );
253 shapePolygon = ti.map( roundedPoly );
254 }
255 else
256 {
257 shapePolygon = QPolygonF( QRectF( 0, 0, rect().width() * scale, rect().height() * scale ) );
258 }
259 break;
260 }
261 case Triangle:
262 {
263 shapePolygon << QPointF( 0, rect().height() * scale );
264 shapePolygon << QPointF( rect().width() * scale, rect().height() * scale );
265 shapePolygon << QPointF( rect().width() / 2.0 * scale, 0 );
266 shapePolygon << QPointF( 0, rect().height() * scale );
267 break;
268 }
269 }
270 return shapePolygon;
271}
272
273bool QgsLayoutItemShape::writePropertiesToElement( QDomElement &element, QDomDocument &document, const QgsReadWriteContext &context ) const
274{
275 element.setAttribute( u"shapeType"_s, mShape );
276 element.setAttribute( u"cornerRadiusMeasure"_s, mCornerRadius.encodeMeasurement() );
277
278 const QDomElement shapeStyleElem = QgsSymbolLayerUtils::saveSymbol( QString(), mShapeStyleSymbol.get(), document, context );
279 element.appendChild( shapeStyleElem );
280
281 return true;
282}
283
284bool QgsLayoutItemShape::readPropertiesFromElement( const QDomElement &element, const QDomDocument &, const QgsReadWriteContext &context )
285{
286 mShape = static_cast< Shape >( element.attribute( u"shapeType"_s, u"0"_s ).toInt() );
287 if ( element.hasAttribute( u"cornerRadiusMeasure"_s ) )
288 mCornerRadius = QgsLayoutMeasurement::decodeMeasurement( element.attribute( u"cornerRadiusMeasure"_s, u"0"_s ) );
289 else
290 mCornerRadius = QgsLayoutMeasurement( element.attribute( u"cornerRadius"_s, u"0"_s ).toDouble() );
291
292 const QDomElement shapeStyleSymbolElem = element.firstChildElement( u"symbol"_s );
293 if ( !shapeStyleSymbolElem.isNull() )
294 {
295 mShapeStyleSymbol = QgsSymbolLayerUtils::loadSymbol<QgsFillSymbol>( shapeStyleSymbolElem, context );
296 refreshSymbol( false );
297 }
298
299 return true;
300}
@ Millimeters
Millimeters.
Definition qgis.h:5256
@ DisableSymbolClippingToExtent
Force symbol clipping to map extent to be disabled in all situations. This will result in slower rend...
Definition qgis.h:2828
static QIcon getThemeIcon(const QString &name, const QColor &fillColor=QColor(), const QColor &strokeColor=QColor())
Helper to get a theme icon.
A fill symbol type, for rendering Polygon and MultiPolygon geometries.
static std::unique_ptr< QgsFillSymbol > createSimple(const QVariantMap &properties)
Create a fill symbol with one symbol layer: SimpleFill with specified properties.
void renderPolygon(const QPolygonF &points, const QVector< QPolygonF > *rings, const QgsFeature *f, QgsRenderContext &context, int layer=-1, bool selected=false)
Renders the symbol using the given render context.
A geometry is the spatial representation of a feature.
static QgsGeometry fromQPolygonF(const QPolygonF &polygon)
Construct geometry from a QPolygonF.
Contains settings and helpers relating to a render of a QgsLayoutItem.
QgsRenderContext & renderContext()
Returns a reference to the context's render context.
QRectF boundingRect() const override
bool readPropertiesFromElement(const QDomElement &element, const QDomDocument &document, const QgsReadWriteContext &context) override
Sets item state from a DOM element.
static QgsLayoutItemShape * create(QgsLayout *layout)
Returns a new shape item for the specified layout.
double estimatedFrameBleed() const override
Returns the estimated amount the item's frame bleeds outside the item's actual rectangle.
QgsFillSymbol * symbol()
Returns the fill symbol used to draw the shape.
~QgsLayoutItemShape() override
void setShapeType(QgsLayoutItemShape::Shape type)
Sets the type of shape (e.g.
QString displayName() const override
Gets item display name.
QgsLayoutItemShape(QgsLayout *layout)
Constructor for QgsLayoutItemShape, with the specified parent layout.
QgsGeometry clipPath() const override
Returns the clipping path generated by this item, in layout coordinates.
bool writePropertiesToElement(QDomElement &element, QDomDocument &document, const QgsReadWriteContext &context) const override
Stores item state within an XML DOM element.
void setSymbol(QgsFillSymbol *symbol)
Sets the fill symbol used to draw the shape.
QIcon icon() const override
Returns the item's icon.
bool accept(QgsStyleEntityVisitorInterface *visitor) const override
Accepts the specified style entity visitor, causing it to visit all style entities associated with th...
int type() const override
@ Ellipse
Ellipse shape.
@ Rectangle
Rectangle shape.
@ Triangle
Triangle shape.
QgsLayoutItem::Flags itemFlags() const override
Returns the item's flags, which indicate how the item behaves.
void draw(QgsLayoutItemRenderContext &context) override
Draws the item's contents using the specified item render context.
void setCornerRadius(QgsLayoutMeasurement radius)
Sets the corner radius for rounded rectangle corners.
friend class QgsLayout
QgsLayoutItem(QgsLayout *layout, bool manageZValue=true)
Constructor for QgsLayoutItem, with the specified parent layout.
virtual void redraw()
Triggers a redraw (update) of the item.
virtual void setFrameEnabled(bool drawFrame)
Sets whether this item has a frame drawn around it or not.
@ FlagProvidesClipPath
Item can act as a clipping path provider (see clipPath()).
void sizePositionChanged()
Emitted when the item's size or position changes.
virtual QString uuid() const
Returns the item identification string.
QString id() const
Returns the item's ID name.
void frameChanged()
Emitted if the item's frame style changes.
virtual Flags itemFlags() const
Returns the item's flags, which indicate how the item behaves.
void clipPathChanged()
Emitted when the item's clipping path has changed.
QFlags< Flag > Flags
void setBackgroundEnabled(bool drawBackground)
Sets whether this item has a background drawn under it or not.
Provides a method of storing measurements for use in QGIS layouts using a variety of different measur...
static QgsLayoutMeasurement decodeMeasurement(const QString &string)
Decodes a measurement from a string.
const QgsLayout * layout() const
Returns the layout the object is attached to.
QPointer< QgsLayout > mLayout
static QgsRenderContext createRenderContextForLayout(QgsLayout *layout, QPainter *painter, double dpi=-1)
Creates a render context suitable for the specified layout and painter destination.
A container for the context for various read/write operations on objects.
Contains information about the context of a rendering operation.
double convertToPainterUnits(double size, Qgis::RenderUnit unit, const QgsMapUnitScale &scale=QgsMapUnitScale(), Qgis::RenderSubcomponentProperty property=Qgis::RenderSubcomponentProperty::Generic) const
Converts a size from the specified units to painter units (pixels).
QPainter * painter()
Returns the destination QPainter for the render operation.
void setFlag(Qgis::RenderContextFlag flag, bool on=true)
Enable or disable a particular flag (other flags are not affected).
An interface for classes which can visit style entity (e.g.
virtual bool visit(const QgsStyleEntityVisitorInterface::StyleLeaf &entity)
Called when the visitor will visit a style entity.
A symbol entity for QgsStyle databases.
Definition qgsstyle.h:1398
static std::unique_ptr< QgsSymbol > loadSymbol(const QDomElement &element, const QgsReadWriteContext &context)
Attempts to load a symbol from a DOM element.
static QDomElement saveSymbol(const QString &symbolName, const QgsSymbol *symbol, QDomDocument &doc, const QgsReadWriteContext &context)
Writes a symbol definition to XML.
static double estimateMaxSymbolBleed(QgsSymbol *symbol, const QgsRenderContext &context)
Returns the maximum estimated bleed for the symbol.
void stopRender(QgsRenderContext &context)
Ends the rendering process.
void startRender(QgsRenderContext &context, const QgsFields &fields=QgsFields())
Begins the rendering process for the symbol.
Contains information relating to the style entity currently being visited.