QGIS API Documentation 3.99.0-Master (26c88405ac0)
Loading...
Searching...
No Matches
qgslayoutlabelwidget.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgslayoutlabelwidget.cpp
3 ------------------------
4 begin : October 2017
5 copyright : (C) 2017 by Nyall Dawson
6 email : nyall dot dawson at gmail dot com
7 ***************************************************************************/
8
9/***************************************************************************
10 * *
11 * This program is free software; you can redistribute it and/or modify *
12 * it under the terms of the GNU General Public License as published by *
13 * the Free Software Foundation; either version 2 of the License, or *
14 * (at your option) any later version. *
15 * *
16 ***************************************************************************/
17
19
21#include "qgsexpressionfinder.h"
22#include "qgslayout.h"
23#include "qgslayoutitemlabel.h"
24#include "qgslayoutitemmap.h"
26#include "qgsprojoperation.h"
27#include "qgsvectorlayer.h"
28
29#include <QAction>
30#include <QColorDialog>
31#include <QFontDialog>
32#include <QMenu>
33#include <QWidget>
34
35#include "moc_qgslayoutlabelwidget.cpp"
36
38 : QgsLayoutItemBaseWidget( nullptr, label )
39 , mLabel( label )
40{
41 Q_ASSERT( mLabel );
42
43 setupUi( this );
44 connect( mHtmlCheckBox, &QCheckBox::stateChanged, this, &QgsLayoutLabelWidget::mHtmlCheckBox_stateChanged );
45 connect( mTextEdit, &QPlainTextEdit::textChanged, this, &QgsLayoutLabelWidget::mTextEdit_textChanged );
46 connect( mInsertExpressionButton, &QPushButton::clicked, this, &QgsLayoutLabelWidget::mInsertExpressionButton_clicked );
47 connect( mMarginXDoubleSpinBox, static_cast<void ( QDoubleSpinBox::* )( double )>( &QDoubleSpinBox::valueChanged ), this, &QgsLayoutLabelWidget::mMarginXDoubleSpinBox_valueChanged );
48 connect( mMarginYDoubleSpinBox, static_cast<void ( QDoubleSpinBox::* )( double )>( &QDoubleSpinBox::valueChanged ), this, &QgsLayoutLabelWidget::mMarginYDoubleSpinBox_valueChanged );
49
50 mHAlignmentComboBox->setAvailableAlignments( Qt::AlignLeft | Qt::AlignHCenter | Qt::AlignRight | Qt::AlignJustify );
51 mVAlignmentComboBox->setAvailableAlignments( Qt::AlignTop | Qt::AlignVCenter | Qt::AlignBottom );
52
53 connect( mHAlignmentComboBox, &QgsAlignmentComboBox::changed, this, &QgsLayoutLabelWidget::horizontalAlignmentChanged );
54 connect( mVAlignmentComboBox, &QgsAlignmentComboBox::changed, this, &QgsLayoutLabelWidget::verticalAlignmentChanged );
55
56 setPanelTitle( tr( "Label Properties" ) );
57
58 mFontButton->setMode( QgsFontButton::ModeTextRenderer );
59 mFontButton->setDialogTitle( tr( "Label Font" ) );
60 mFontButton->registerExpressionContextGenerator( this );
61
62 //add widget for general composer item properties
63 mItemPropertiesWidget = new QgsLayoutItemPropertiesWidget( this, label );
64 mainLayout->addWidget( mItemPropertiesWidget );
65
66 mMarginXDoubleSpinBox->setClearValue( 0.0 );
67 mMarginYDoubleSpinBox->setClearValue( 0.0 );
68
69 setGuiElementValues();
70 connect( mLabel, &QgsLayoutObject::changed, this, &QgsLayoutLabelWidget::setGuiElementValues );
71
72 connect( mFontButton, &QgsFontButton::changed, this, &QgsLayoutLabelWidget::fontChanged );
73
74 mDynamicTextMenu = new QMenu( this );
75 mDynamicTextButton->setMenu( mDynamicTextMenu );
76
77 connect( mDynamicTextMenu, &QMenu::aboutToShow, this, [this] {
78 mDynamicTextMenu->clear();
79 if ( mLabel->layout() )
80 {
81 // we need to rebuild this on each show, as the content varies depending on other available items...
82 buildInsertDynamicTextMenu( mLabel->layout(), mDynamicTextMenu, [this]( const QString &expression ) {
83 mLabel->beginCommand( tr( "Insert dynamic text" ) );
84 mTextEdit->insertPlainText( "[%" + expression.trimmed() + "%]" );
85 mLabel->endCommand();
86 } );
87 }
88 } );
89
90 QMenu *expressionMenu = new QMenu( this );
91 QAction *convertToStaticAction = new QAction( tr( "Convert to Static Text" ), this );
92 expressionMenu->addAction( convertToStaticAction );
93 connect( convertToStaticAction, &QAction::triggered, mLabel, &QgsLayoutItemLabel::convertToStaticText );
94 mInsertExpressionButton->setMenu( expressionMenu );
95
96 mFontButton->setLayer( coverageLayer() );
97 if ( mLabel->layout() )
98 {
99 connect( &mLabel->layout()->reportContext(), &QgsLayoutReportContext::layerChanged, mFontButton, &QgsFontButton::setLayer );
100 }
101}
102
104{
105 if ( mItemPropertiesWidget )
106 mItemPropertiesWidget->setMasterLayout( masterLayout );
107}
108
110{
111 return mLabel->createExpressionContext();
112}
113
114void QgsLayoutLabelWidget::buildInsertDynamicTextMenu( QgsLayout *layout, QMenu *menu, const std::function<void( const QString & )> &callback )
115{
116 Q_ASSERT( layout );
117 auto addExpression = [&callback]( QMenu *menu, const QString &name, const QString &expression ) {
118 QAction *action = new QAction( name, menu );
119 connect( action, &QAction::triggered, action, [callback, expression] {
120 callback( expression );
121 } );
122 menu->addAction( action );
123 };
124
125 QMenu *dateMenu = new QMenu( tr( "Current Date" ), menu );
126 for ( const std::pair<QString, QString> &expression :
127 {
128 std::make_pair( tr( "ISO Format (%1)" ).arg( QDateTime::currentDateTime().toString( QStringLiteral( "yyyy-MM-dd" ) ) ), QStringLiteral( "format_date(now(), 'yyyy-MM-dd')" ) ),
129 std::make_pair( tr( "Day/Month/Year (%1)" ).arg( QDateTime::currentDateTime().toString( QStringLiteral( "dd/MM/yyyy" ) ) ), QStringLiteral( "format_date(now(), 'dd/MM/yyyy')" ) ),
130 std::make_pair( tr( "Month/Day/Year (%1)" ).arg( QDateTime::currentDateTime().toString( QStringLiteral( "MM/dd/yyyy" ) ) ), QStringLiteral( "format_date(now(), 'MM/dd/yyyy')" ) ),
131 } )
132 {
133 addExpression( dateMenu, expression.first, expression.second );
134 }
135 menu->addMenu( dateMenu );
136
137 QMenu *mapsMenu = new QMenu( tr( "Map Properties" ), menu );
138 QList<QgsLayoutItemMap *> maps;
139 layout->layoutItems( maps );
140 for ( QgsLayoutItemMap *map : std::as_const( maps ) )
141 {
142 // these expressions require the map to have a non-empty ID set
143 if ( map->id().isEmpty() )
144 continue;
145
146 QMenu *mapMenu = new QMenu( map->displayName(), mapsMenu );
147 for ( const std::pair<QString, QString> &expression :
148 {
149 std::make_pair( tr( "Scale (%1)" ).arg( map->scale() ), QStringLiteral( "format_number(item_variables('%1')['map_scale'], places:=6, omit_group_separators:=true, trim_trailing_zeroes:=true)" ).arg( map->id() ) ),
150 std::make_pair( tr( "Rotation (%1)" ).arg( map->rotation() ), QStringLiteral( "item_variables('%1')['map_rotation']" ).arg( map->id() ) ),
151 } )
152 {
153 addExpression( mapMenu, expression.first, expression.second );
154 }
155 mapMenu->addSeparator();
156 for ( const std::pair<QString, QString> &expression :
157 {
158 std::make_pair( tr( "CRS Identifier (%1)" ).arg( map->crs().authid() ), QStringLiteral( "item_variables('%1')['map_crs']" ).arg( map->id() ) ),
159 std::make_pair( tr( "CRS Name (%1)" ).arg( map->crs().description() ), QStringLiteral( "item_variables('%1')['map_crs_description']" ).arg( map->id() ) ),
160 std::make_pair( tr( "Ellipsoid Name (%1)" ).arg( map->crs().ellipsoidAcronym() ), QStringLiteral( "item_variables('%1')['map_crs_ellipsoid']" ).arg( map->id() ) ),
161 std::make_pair( tr( "Units (%1)" ).arg( QgsUnitTypes::toString( map->crs().mapUnits() ) ), QStringLiteral( "item_variables('%1')['map_units']" ).arg( map->id() ) ),
162 std::make_pair( tr( "Projection (%1)" ).arg( map->crs().operation().description() ), QStringLiteral( "item_variables('%1')['map_crs_projection']" ).arg( map->id() ) ),
163 } )
164 {
165 addExpression( mapMenu, expression.first, expression.second );
166 }
167 mapMenu->addSeparator();
168
169 const QgsRectangle mapExtent = map->extent();
170 const QgsPointXY center = mapExtent.center();
171 for ( const std::pair<QString, QString> &expression :
172 {
173 std::make_pair( tr( "Center (X) (%1)" ).arg( center.x() ), QStringLiteral( "x(item_variables('%1')['map_extent_center'])" ).arg( map->id() ) ),
174 std::make_pair( tr( "Center (Y) (%1)" ).arg( center.y() ), QStringLiteral( "y(item_variables('%1')['map_extent_center'])" ).arg( map->id() ) ),
175 std::make_pair( tr( "X Minimum (%1)" ).arg( mapExtent.xMinimum() ), QStringLiteral( "x_min(item_variables('%1')['map_extent'])" ).arg( map->id() ) ),
176 std::make_pair( tr( "Y Minimum (%1)" ).arg( mapExtent.yMinimum() ), QStringLiteral( "y_min(item_variables('%1')['map_extent'])" ).arg( map->id() ) ),
177 std::make_pair( tr( "X Maximum (%1)" ).arg( mapExtent.xMaximum() ), QStringLiteral( "x_max(item_variables('%1')['map_extent'])" ).arg( map->id() ) ),
178 std::make_pair( tr( "Y Maximum (%1)" ).arg( mapExtent.yMaximum() ), QStringLiteral( "y_max(item_variables('%1')['map_extent'])" ).arg( map->id() ) ),
179 } )
180 {
181 addExpression( mapMenu, expression.first, expression.second );
182 }
183 mapMenu->addSeparator();
184 for ( const std::pair<QString, QString> &expression :
185 {
186 std::make_pair( tr( "Layer Credits" ), QStringLiteral( "array_to_string(map_credits('%1'))" ).arg( map->id() ) ),
187 } )
188 {
189 addExpression( mapMenu, expression.first, expression.second );
190 }
191 mapsMenu->addMenu( mapMenu );
192 }
193 menu->addMenu( mapsMenu );
194 menu->addSeparator();
195
196 if ( layout->reportContext().layer() )
197 {
198 const QgsFields fields = layout->reportContext().layer()->fields();
199
200 QMenu *fieldsMenu = new QMenu( tr( "Field" ), menu );
201 for ( const QgsField &field : fields )
202 {
203 addExpression( fieldsMenu, field.displayName(), QStringLiteral( "\"%1\"" ).arg( field.name() ) );
204 }
205
206 menu->addMenu( fieldsMenu );
207 menu->addSeparator();
208 }
209
210 for ( const std::pair<QString, QString> &expression :
211 {
212 std::make_pair( tr( "Layout Name" ), QStringLiteral( "@layout_name" ) ),
213 std::make_pair( tr( "Layout Page Number" ), QStringLiteral( "@layout_page" ) ),
214 std::make_pair( tr( "Layout Page Count" ), QStringLiteral( "@layout_numpages" ) ),
215 std::make_pair( tr( "Layer Credits" ), QStringLiteral( "array_to_string(map_credits())" ) )
216 } )
217 {
218 addExpression( menu, expression.first, expression.second );
219 }
220 menu->addSeparator();
221 for ( const std::pair<QString, QString> &expression :
222 {
223 std::make_pair( tr( "Project Author" ), QStringLiteral( "@project_author" ) ),
224 std::make_pair( tr( "Project Title" ), QStringLiteral( "@project_title" ) ),
225 std::make_pair( tr( "Project Path" ), QStringLiteral( "@project_path" ) )
226 } )
227 {
228 addExpression( menu, expression.first, expression.second );
229 }
230 menu->addSeparator();
231 for ( const std::pair<QString, QString> &expression :
232 {
233 std::make_pair( tr( "Current User Name" ), QStringLiteral( "@user_full_name" ) ),
234 std::make_pair( tr( "Current User Account" ), QStringLiteral( "@user_account_name" ) )
235 } )
236 {
237 addExpression( menu, expression.first, expression.second );
238 }
239}
240
242{
244 return false;
245
246 if ( mLabel )
247 {
248 disconnect( mLabel, &QgsLayoutObject::changed, this, &QgsLayoutLabelWidget::setGuiElementValues );
249 }
250
251 mLabel = qobject_cast<QgsLayoutItemLabel *>( item );
252 mItemPropertiesWidget->setItem( mLabel );
253
254 if ( mLabel )
255 {
256 connect( mLabel, &QgsLayoutObject::changed, this, &QgsLayoutLabelWidget::setGuiElementValues );
257 }
258
259 setGuiElementValues();
260
261 return true;
262}
263
264void QgsLayoutLabelWidget::mHtmlCheckBox_stateChanged( int state )
265{
266 if ( mLabel )
267 {
268 mVerticalAlignmentLabel->setDisabled( state );
269 mVAlignmentComboBox->setDisabled( state );
270
271 mLabel->beginCommand( tr( "Change Label Mode" ) );
272 mLabel->blockSignals( true );
274 mLabel->setText( mTextEdit->toPlainText() );
275 mLabel->update();
276 mLabel->blockSignals( false );
277 mLabel->endCommand();
278 }
279}
280
281void QgsLayoutLabelWidget::mTextEdit_textChanged()
282{
283 if ( mLabel )
284 {
285 mLabel->beginCommand( tr( "Change Label Text" ), QgsLayoutItem::UndoLabelText );
286 mLabel->blockSignals( true );
287 mLabel->setText( mTextEdit->toPlainText() );
288 mLabel->update();
289 mLabel->blockSignals( false );
290 mLabel->endCommand();
291 }
292}
293
294void QgsLayoutLabelWidget::fontChanged()
295{
296 if ( mLabel )
297 {
298 mLabel->beginCommand( tr( "Change Label Font" ), QgsLayoutItem::UndoLabelFont );
299 mLabel->setTextFormat( mFontButton->textFormat() );
300 mLabel->update();
301 mLabel->endCommand();
302 }
303}
304
305void QgsLayoutLabelWidget::mMarginXDoubleSpinBox_valueChanged( double d )
306{
307 if ( mLabel )
308 {
309 mLabel->beginCommand( tr( "Change Label Margin" ), QgsLayoutItem::UndoLabelMargin );
310 mLabel->setMarginX( d );
311 mLabel->update();
312 mLabel->endCommand();
313 }
314}
315
316void QgsLayoutLabelWidget::mMarginYDoubleSpinBox_valueChanged( double d )
317{
318 if ( mLabel )
319 {
320 mLabel->beginCommand( tr( "Change Label Margin" ), QgsLayoutItem::UndoLabelMargin );
321 mLabel->setMarginY( d );
322 mLabel->update();
323 mLabel->endCommand();
324 }
325}
326
327void QgsLayoutLabelWidget::mInsertExpressionButton_clicked()
328{
329 if ( !mLabel )
330 {
331 return;
332 }
333
334 QString expression = QgsExpressionFinder::findAndSelectActiveExpression( mTextEdit );
335
336 // use the atlas coverage layer, if any
337 QgsVectorLayer *layer = coverageLayer();
338
339 QgsExpressionContext context = mLabel->createExpressionContext();
340 QgsExpressionBuilderDialog exprDlg( layer, expression, this, QStringLiteral( "generic" ), context );
341
342 exprDlg.setWindowTitle( tr( "Insert Expression" ) );
343 if ( exprDlg.exec() == QDialog::Accepted )
344 {
345 expression = exprDlg.expressionText();
346 if ( !expression.isEmpty() )
347 {
348 mLabel->beginCommand( tr( "Insert expression" ) );
349 mTextEdit->insertPlainText( "[%" + expression.trimmed() + "%]" );
350 mLabel->endCommand();
351 }
352 }
353}
354
355void QgsLayoutLabelWidget::horizontalAlignmentChanged()
356{
357 if ( mLabel )
358 {
359 mLabel->beginCommand( tr( "Change Label Alignment" ) );
360 mLabel->setHAlign( static_cast<Qt::AlignmentFlag>( static_cast<int>( mHAlignmentComboBox->currentAlignment() ) ) );
361 mLabel->update();
362 mLabel->endCommand();
363 }
364}
365
366void QgsLayoutLabelWidget::verticalAlignmentChanged()
367{
368 if ( mLabel )
369 {
370 mLabel->beginCommand( tr( "Change Label Alignment" ) );
371 mLabel->setVAlign( static_cast<Qt::AlignmentFlag>( static_cast<int>( mVAlignmentComboBox->currentAlignment() ) ) );
372 mLabel->update();
373 mLabel->endCommand();
374 }
375}
376
377void QgsLayoutLabelWidget::setGuiElementValues()
378{
379 blockAllSignals( true );
380 mTextEdit->setPlainText( mLabel->text() );
381 mTextEdit->moveCursor( QTextCursor::End, QTextCursor::MoveAnchor );
382 mMarginXDoubleSpinBox->setValue( mLabel->marginX() );
383 mMarginYDoubleSpinBox->setValue( mLabel->marginY() );
384 mHtmlCheckBox->setChecked( mLabel->mode() == QgsLayoutItemLabel::ModeHtml );
385
386 mHAlignmentComboBox->setCurrentAlignment( mLabel->hAlign() );
387 mVAlignmentComboBox->setCurrentAlignment( mLabel->vAlign() );
388
389 mFontButton->setTextFormat( mLabel->textFormat() );
390 mVerticalAlignmentLabel->setDisabled( mLabel->mode() == QgsLayoutItemLabel::ModeHtml );
391 mVAlignmentComboBox->setDisabled( mLabel->mode() == QgsLayoutItemLabel::ModeHtml );
392
393 blockAllSignals( false );
394}
395
396void QgsLayoutLabelWidget::blockAllSignals( bool block )
397{
398 mTextEdit->blockSignals( block );
399 mHtmlCheckBox->blockSignals( block );
400 mMarginXDoubleSpinBox->blockSignals( block );
401 mMarginYDoubleSpinBox->blockSignals( block );
402 mHAlignmentComboBox->blockSignals( block );
403 mVAlignmentComboBox->blockSignals( block );
404 mFontButton->blockSignals( block );
405}
void changed()
Emitted when the alignment is changed.
Expression contexts are used to encapsulate the parameters around which a QgsExpression should be eva...
static QString findAndSelectActiveExpression(QgsCodeEditor *editor, const QString &pattern=QString())
Find the expression under the cursor in the given editor and select it.
Encapsulate a field in an attribute table or data source.
Definition qgsfield.h:54
Container of fields for a vector layer.
Definition qgsfields.h:46
@ ModeTextRenderer
Configure font settings for use with QgsTextRenderer.
void setLayer(QgsVectorLayer *layer)
Sets a layer to associate with the widget.
void changed()
Emitted when the widget's text format settings are changed.
QgsVectorLayer * coverageLayer() const
Returns the current layout context coverage layer (if set).
QgsLayoutItemBaseWidget(QWidget *parent SIP_TRANSFERTHIS, QgsLayoutObject *layoutObject)
Constructor for QgsLayoutItemBaseWidget, linked with the specified layoutObject.
A layout item subclass for text labels.
void convertToStaticText()
Converts the label's text() to a static string, by evaluating any expressions included in the text an...
@ ModeHtml
Label displays rendered HTML content.
@ ModeFont
Label displays text rendered using a single font.
Layout graphical items for displaying a map.
A widget for controlling the common properties of layout items (e.g.
Base class for graphical items within a QgsLayout.
@ UndoLabelMargin
Label margin.
@ UndoLabelFont
Label font.
@ UndoLabelText
Label text.
int type() const override
Returns a unique graphics item type identifier.
void setMasterLayout(QgsMasterLayoutInterface *masterLayout) override
Sets the master layout associated with the item.
QgsExpressionContext createExpressionContext() const override
This method needs to be reimplemented in all classes which implement this interface and return an exp...
bool setNewItem(QgsLayoutItem *item) override
Attempts to update the widget to show the properties for the specified item.
static void buildInsertDynamicTextMenu(QgsLayout *layout, QMenu *menu, const std::function< void(const QString &expression)> &callback)
Populates the specified menu with actions reflecting dynamic text expressions applicable for a layout...
QgsLayoutLabelWidget(QgsLayoutItemLabel *label)
constructor
void changed()
Emitted when the object's properties change.
void layerChanged(QgsVectorLayer *layer)
Emitted when the context's layer is changed.
QgsVectorLayer * layer() const
Returns the vector layer associated with the layout's context.
Base class for layouts, which can contain items such as maps, labels, scalebars, etc.
Definition qgslayout.h:50
void layoutItems(QList< T * > &itemList) const
Returns a list of layout items of a specific type.
Definition qgslayout.h:121
QgsLayoutReportContext & reportContext()
Returns a reference to the layout's report context, which stores information relating to the current ...
Interface for master layout type objects, such as print layouts and reports.
void setPanelTitle(const QString &panelTitle)
Set the title of the panel when shown in the interface.
Represents a 2D point.
Definition qgspointxy.h:60
double y
Definition qgspointxy.h:64
double x
Definition qgspointxy.h:63
A rectangle specified with double values.
double xMinimum
double yMinimum
double xMaximum
double yMaximum
QgsPointXY center
static Q_INVOKABLE QString toString(Qgis::DistanceUnit unit)
Returns a translated string representing a distance unit.