QGIS API Documentation 3.30.0-'s-Hertogenbosch (f186b8efe0)
qgslayoutguidecollection.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgslayoutguidecollection.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
18#include "qgslayout.h"
19#include "qgsreadwritecontext.h"
21#include "qgslayoutundostack.h"
22#include "qgsunittypes.h"
23
24#include <QGraphicsLineItem>
25
26
27//
28// QgsLayoutGuide
29//
30
31QgsLayoutGuide::QgsLayoutGuide( Qt::Orientation orientation, QgsLayoutMeasurement position, QgsLayoutItemPage *page )
32 : QObject( nullptr )
33 , mOrientation( orientation )
34 , mPosition( position )
35 , mPage( page )
36{}
37
39{
40 if ( mLayout && mLineItem )
41 {
42 mLayout->removeItem( mLineItem );
43 delete mLineItem;
44 }
45}
46
48{
49 return mPosition;
50}
51
53{
54 mPosition = position;
55 update();
56 emit positionChanged();
57}
58
60{
61 return mPage;
62}
63
65{
66 mPage = page;
67 update();
68}
69
71{
72 if ( !mLayout || !mLineItem )
73 return;
74
75 // first find matching page
76 if ( !mPage )
77 {
78 mLineItem->hide();
79 return;
80 }
81
82 double layoutPos = mLayout->convertToLayoutUnits( mPosition );
83 bool showGuide = mLayout->guides().visible();
84 switch ( mOrientation )
85 {
86 case Qt::Horizontal:
87 if ( layoutPos > mPage->rect().height() )
88 {
89 mLineItem->hide();
90 }
91 else
92 {
93 mLineItem->setLine( 0, layoutPos + mPage->y(), mPage->rect().width(), layoutPos + mPage->y() );
94 mLineItem->setVisible( showGuide );
95 }
96
97 break;
98
99 case Qt::Vertical:
100 if ( layoutPos > mPage->rect().width() )
101 {
102 mLineItem->hide();
103 }
104 else
105 {
106 mLineItem->setLine( layoutPos, mPage->y(), layoutPos, mPage->y() + mPage->rect().height() );
107 mLineItem->setVisible( showGuide );
108 }
109
110 break;
111 }
112}
113
114QGraphicsLineItem *QgsLayoutGuide::item()
115{
116 return mLineItem;
117}
118
120{
121 if ( !mLineItem )
122 return -999;
123
124 switch ( mOrientation )
125 {
126 case Qt::Horizontal:
127 return mLineItem->mapToScene( mLineItem->line().p1() ).y();
128
129 case Qt::Vertical:
130 return mLineItem->mapToScene( mLineItem->line().p1() ).x();
131 }
132 return -999; // avoid warning
133}
134
136{
137 if ( !mLayout )
138 return;
139
140 double p = 0;
141 switch ( mOrientation )
142 {
143 case Qt::Horizontal:
144 p = mPage->mapFromScene( QPointF( 0, position ) ).y();
145 break;
146
147 case Qt::Vertical:
148 p = mPage->mapFromScene( QPointF( position, 0 ) ).x();
149 break;
150 }
151 mPosition = mLayout->convertFromLayoutUnits( p, mPosition.units() );
152 update();
153 emit positionChanged();
154}
155
157{
158 return mLayout;
159}
160
162{
163 mLayout = layout;
164
165 if ( !mLineItem )
166 {
167 mLineItem = new QGraphicsLineItem();
168 mLineItem->hide();
169 mLineItem->setZValue( QgsLayout::ZGuide );
170 QPen linePen( Qt::DotLine );
171 linePen.setColor( Qt::red );
172 // use a pen width of 0, since this activates a cosmetic pen
173 // which doesn't scale with the layout and keeps a constant size
174 linePen.setWidthF( 0 );
175 mLineItem->setPen( linePen );
176 }
177
178 mLayout->addItem( mLineItem );
179 update();
180}
181
182Qt::Orientation QgsLayoutGuide::orientation() const
183{
184 return mOrientation;
185}
186
187
188
189//
190// QgsLayoutGuideCollection
191//
192
194 : QAbstractTableModel( layout )
195 , mLayout( layout )
196 , mPageCollection( pageCollection )
197{
198 QFont f;
199 mHeaderSize = QFontMetrics( f ).boundingRect( QStringLiteral( "XX" ) ).width();
200
201 connect( mPageCollection, &QgsLayoutPageCollection::pageAboutToBeRemoved, this, &QgsLayoutGuideCollection::pageAboutToBeRemoved );
202}
203
205{
206 qDeleteAll( mGuides );
207}
208
210{
211 return mLayout;
212}
213
214int QgsLayoutGuideCollection::rowCount( const QModelIndex & ) const
215{
216 return mGuides.count();
217}
218
219int QgsLayoutGuideCollection::columnCount( const QModelIndex &parent ) const
220{
221 if ( parent.isValid() )
222 return 0;
223
224 return 2;
225}
226
227QVariant QgsLayoutGuideCollection::data( const QModelIndex &index, int role ) const
228{
229 if ( !index.isValid() )
230 return QVariant();
231
232 if ( index.row() >= mGuides.count() || index.row() < 0 )
233 return QVariant();
234
235 QgsLayoutGuide *guide = mGuides.at( index.row() );
236 switch ( role )
237 {
238 case Qt::DisplayRole:
239 case Qt::EditRole:
240 {
241 if ( index.column() == 0 )
242 return guide->position().length();
243 else
245 }
246
247 case OrientationRole:
248 return guide->orientation();
249
250 case PositionRole:
251 return guide->position().length();
252
253 case UnitsRole:
254 return static_cast< int >( guide->position().units() );
255
256 case PageRole:
257 return mPageCollection->pageNumber( guide->page() );
258
260 return guide->layoutPosition();
261
262 default:
263 return QVariant();
264 }
265}
266
267bool QgsLayoutGuideCollection::setData( const QModelIndex &index, const QVariant &value, int role )
268{
269 if ( !index.isValid() )
270 return false;
271
272 if ( index.row() >= mGuides.count() || index.row() < 0 )
273 return false;
274
275 QgsLayoutGuide *guide = mGuides.at( index.row() );
276
277 switch ( role )
278 {
279 case Qt::EditRole:
280 {
281 bool ok = false;
282 double newPos = value.toDouble( &ok );
283 if ( !ok )
284 return false;
285
286 QgsLayoutMeasurement m = guide->position();
287 m.setLength( newPos );
288 mLayout->undoStack()->beginCommand( mPageCollection, tr( "Move Guide" ), Move + index.row() );
289 whileBlocking( guide )->setPosition( m );
290 guide->update();
291 mLayout->undoStack()->endCommand();
292 emit dataChanged( index, index, QVector<int>() << role );
293 return true;
294 }
295 case PositionRole:
296 {
297 bool ok = false;
298 double newPos = value.toDouble( &ok );
299 if ( !ok )
300 return false;
301
302 QgsLayoutMeasurement m = guide->position();
303 if ( qgsDoubleNear( m.length(), newPos ) )
304 return true;
305
306 m.setLength( newPos );
307 mLayout->undoStack()->beginCommand( mPageCollection, tr( "Move Guide" ), Move + index.row() );
308 whileBlocking( guide )->setPosition( m );
309 guide->update();
310 mLayout->undoStack()->endCommand();
311 emit dataChanged( index, index, QVector<int>() << role );
312 return true;
313 }
314
316 {
317 bool ok = false;
318 double newPos = value.toDouble( &ok );
319 if ( !ok )
320 return false;
321
322 mLayout->undoStack()->beginCommand( mPageCollection, tr( "Move Guide" ), Move + index.row() );
323 whileBlocking( guide )->setLayoutPosition( newPos );
324 mLayout->undoStack()->endCommand();
325 emit dataChanged( index, index, QVector<int>() << role );
326 return true;
327 }
328
329 case UnitsRole:
330 {
331 bool ok = false;
332 int units = value.toInt( &ok );
333 if ( !ok )
334 return false;
335
336 QgsLayoutMeasurement m = guide->position();
337 m.setUnits( static_cast< Qgis::LayoutUnit >( units ) );
338 mLayout->undoStack()->beginCommand( mPageCollection, tr( "Move Guide" ), Move + index.row() );
339 whileBlocking( guide )->setPosition( m );
340 guide->update();
341 mLayout->undoStack()->endCommand();
342 emit dataChanged( index, index, QVector<int>() << role );
343 return true;
344 }
345 }
346
347 return false;
348}
349
350Qt::ItemFlags QgsLayoutGuideCollection::flags( const QModelIndex &index ) const
351{
352 if ( !index.isValid() )
353 return Qt::ItemIsEnabled | Qt::ItemIsSelectable;
354 return Qt::ItemIsEnabled | Qt::ItemIsSelectable | Qt::ItemIsEditable;
355}
356
357QVariant QgsLayoutGuideCollection::headerData( int section, Qt::Orientation orientation, int role ) const
358{
359 if ( role == Qt::DisplayRole )
360 return QVariant();
361 else if ( role == Qt::SizeHintRole )
362 {
363 return QSize( mHeaderSize, mHeaderSize );
364 }
365 return QAbstractTableModel::headerData( section, orientation, role );
366}
367
368bool QgsLayoutGuideCollection::removeRows( int row, int count, const QModelIndex &parent )
369{
370 if ( parent.isValid() )
371 return false;
372
373 if ( !mBlockUndoCommands )
374 mLayout->undoStack()->beginCommand( mPageCollection, tr( "Remove Guide(s)" ), Remove + row );
375 beginRemoveRows( parent, row, row + count - 1 );
376 for ( int i = 0; i < count; ++ i )
377 {
378 delete mGuides.takeAt( row );
379 }
380 endRemoveRows();
381 if ( !mBlockUndoCommands )
382 mLayout->undoStack()->endCommand();
383 return true;
384}
385
387{
388 if ( guide->layout() != mLayout )
389 guide->setLayout( mLayout );
390
391 if ( !mBlockUndoCommands )
392 mLayout->undoStack()->beginCommand( mPageCollection, tr( "Create Guide" ) );
393 beginInsertRows( QModelIndex(), mGuides.count(), mGuides.count() );
394 mGuides.append( guide );
395 endInsertRows();
396 if ( !mBlockUndoCommands )
397 mLayout->undoStack()->endCommand();
398
399 QModelIndex index = createIndex( mGuides.length() - 1, 0 );
400 connect( guide, &QgsLayoutGuide::positionChanged, this, [ this, index ]
401 {
402 emit dataChanged( index, index );
403 } );
404}
405
407{
408 int row = mGuides.indexOf( guide );
409 if ( row < 0 )
410 return;
411
412 removeRow( row );
413}
414
416{
417 int row = mGuides.indexOf( guide );
418 if ( row < 0 )
419 return;
420
421 setData( index( row, 0 ), position, LayoutPositionRole );
422}
423
425{
426 mLayout->undoStack()->beginCommand( mPageCollection, tr( "Clear Guides" ) );
427 beginResetModel();
428 qDeleteAll( mGuides );
429 mGuides.clear();
430 endResetModel();
431 mLayout->undoStack()->endCommand();
432}
433
435{
436 mLayout->undoStack()->beginCommand( mPageCollection, tr( "Apply Guides" ) );
437 mBlockUndoCommands = true;
438 QgsLayoutItemPage *page = mPageCollection->page( sourcePage );
439 // remove other page's guides
440 const auto constMGuides = mGuides;
441 for ( QgsLayoutGuide *guide : constMGuides )
442 {
443 if ( guide->page() != page )
444 removeGuide( guide );
445 }
446
447 // remaining guides belong to source page - clone them to other pages
448 for ( QgsLayoutGuide *guide : std::as_const( mGuides ) )
449 {
450 for ( int p = 0; p < mPageCollection->pageCount(); ++p )
451 {
452 if ( p == sourcePage )
453 continue;
454
455 std::unique_ptr< QgsLayoutGuide> newGuide( new QgsLayoutGuide( guide->orientation(), guide->position(), mPageCollection->page( p ) ) );
456 newGuide->setLayout( mLayout );
457 if ( newGuide->item()->isVisible() )
458 {
459 // if invisible, new guide is outside of page bounds
460 addGuide( newGuide.release() );
461 }
462 }
463 }
464 mLayout->undoStack()->endCommand();
465 mBlockUndoCommands = false;
466}
467
469{
470 const auto constMGuides = mGuides;
471 for ( QgsLayoutGuide *guide : constMGuides )
472 {
473 guide->update();
474 }
475}
476
477QList<QgsLayoutGuide *> QgsLayoutGuideCollection::guides()
478{
479 return mGuides;
480}
481
482QList<QgsLayoutGuide *> QgsLayoutGuideCollection::guides( Qt::Orientation orientation, int page )
483{
484 QList<QgsLayoutGuide *> res;
485 const auto constMGuides = mGuides;
486 for ( QgsLayoutGuide *guide : constMGuides )
487 {
488 if ( guide->orientation() == orientation && guide->item()->isVisible() &&
489 ( page < 0 || mPageCollection->page( page ) == guide->page() ) )
490 res << guide;
491 }
492 return res;
493}
494
495QList<QgsLayoutGuide *> QgsLayoutGuideCollection::guidesOnPage( int page )
496{
497 QList<QgsLayoutGuide *> res;
498 const auto constMGuides = mGuides;
499 for ( QgsLayoutGuide *guide : constMGuides )
500 {
501 if ( mPageCollection->page( page ) == guide->page() )
502 res << guide;
503 }
504 return res;
505}
506
508{
509 return mGuidesVisible;
510}
511
513{
514 mLayout->undoStack()->beginCommand( mPageCollection, tr( "Change Guide Visibility" ) );
515 mGuidesVisible = visible;
516 mLayout->undoStack()->endCommand();
517 update();
518}
519
520void QgsLayoutGuideCollection::pageAboutToBeRemoved( int pageNumber )
521{
522 mBlockUndoCommands = true;
523 const auto constGuidesOnPage = guidesOnPage( pageNumber );
524 for ( QgsLayoutGuide *guide : constGuidesOnPage )
525 {
526 removeGuide( guide );
527 }
528 mBlockUndoCommands = false;
529}
530
531bool QgsLayoutGuideCollection::writeXml( QDomElement &parentElement, QDomDocument &document, const QgsReadWriteContext & ) const
532{
533 QDomElement element = document.createElement( QStringLiteral( "GuideCollection" ) );
534 element.setAttribute( QStringLiteral( "visible" ), mGuidesVisible );
535 const auto constMGuides = mGuides;
536 for ( QgsLayoutGuide *guide : constMGuides )
537 {
538 QDomElement guideElement = document.createElement( QStringLiteral( "Guide" ) );
539 guideElement.setAttribute( QStringLiteral( "orientation" ), guide->orientation() );
540 guideElement.setAttribute( QStringLiteral( "page" ), mPageCollection->pageNumber( guide->page() ) );
541 guideElement.setAttribute( QStringLiteral( "position" ), guide->position().length() );
542 guideElement.setAttribute( QStringLiteral( "units" ), QgsUnitTypes::encodeUnit( guide->position().units() ) );
543 element.appendChild( guideElement );
544 }
545
546 parentElement.appendChild( element );
547 return true;
548}
549
550bool QgsLayoutGuideCollection::readXml( const QDomElement &e, const QDomDocument &, const QgsReadWriteContext & )
551{
552 QDomElement element = e;
553 if ( element.nodeName() != QLatin1String( "GuideCollection" ) )
554 {
555 element = element.firstChildElement( QStringLiteral( "GuideCollection" ) );
556 }
557
558 if ( element.nodeName() != QLatin1String( "GuideCollection" ) )
559 {
560 return false;
561 }
562
563 mBlockUndoCommands = true;
564 beginResetModel();
565 qDeleteAll( mGuides );
566 mGuides.clear();
567
568 mGuidesVisible = element.attribute( QStringLiteral( "visible" ), QStringLiteral( "0" ) ) != QLatin1String( "0" );
569 QDomNodeList guideNodeList = element.elementsByTagName( QStringLiteral( "Guide" ) );
570 for ( int i = 0; i < guideNodeList.size(); ++i )
571 {
572 QDomElement element = guideNodeList.at( i ).toElement();
573 Qt::Orientation orientation = static_cast< Qt::Orientation >( element.attribute( QStringLiteral( "orientation" ), QStringLiteral( "1" ) ).toInt() );
574 double pos = element.attribute( QStringLiteral( "position" ), QStringLiteral( "0" ) ).toDouble();
575 Qgis::LayoutUnit unit = QgsUnitTypes::decodeLayoutUnit( element.attribute( QStringLiteral( "units" ) ) );
576 int page = element.attribute( QStringLiteral( "page" ), QStringLiteral( "0" ) ).toInt();
577 std::unique_ptr< QgsLayoutGuide > guide( new QgsLayoutGuide( orientation, QgsLayoutMeasurement( pos, unit ), mPageCollection->page( page ) ) );
578 guide->update();
579 addGuide( guide.release() );
580 }
581
582 endResetModel();
583 mBlockUndoCommands = false;
584 return true;
585}
586
587//
588// QgsLayoutGuideProxyModel
589//
590
591QgsLayoutGuideProxyModel::QgsLayoutGuideProxyModel( QObject *parent, Qt::Orientation orientation, int page )
592 : QSortFilterProxyModel( parent )
593 , mOrientation( orientation )
594 , mPage( page )
595{
596 setDynamicSortFilter( true );
597 sort( 0 );
598}
599
601{
602 mPage = page;
603 invalidateFilter();
604}
605
606bool QgsLayoutGuideProxyModel::filterAcceptsRow( int source_row, const QModelIndex &source_parent ) const
607{
608 QModelIndex index = sourceModel()->index( source_row, 0, source_parent );
609 Qt::Orientation orientation = static_cast< Qt::Orientation>( sourceModel()->data( index, QgsLayoutGuideCollection::OrientationRole ).toInt() );
610 if ( orientation != mOrientation )
611 return false;
612
613 int page = sourceModel()->data( index, QgsLayoutGuideCollection::PageRole ).toInt();
614 return page == mPage;
615}
616
617bool QgsLayoutGuideProxyModel::lessThan( const QModelIndex &left, const QModelIndex &right ) const
618{
619 double leftPos = sourceModel()->data( left, QgsLayoutGuideCollection::LayoutPositionRole ).toDouble();
620 double rightPos = sourceModel()->data( right, QgsLayoutGuideCollection::LayoutPositionRole ).toDouble();
621 return leftPos < rightPos;
622}
LayoutUnit
Layout measurement units.
Definition: qgis.h:3196
bool writeXml(QDomElement &parentElement, QDomDocument &document, const QgsReadWriteContext &context) const override
Stores the collection's state in a DOM element.
int columnCount(const QModelIndex &) const override
void addGuide(QgsLayoutGuide *guide)
Adds a guide to the collection.
bool setData(const QModelIndex &index, const QVariant &value, int role) override
QgsLayout * layout() override
Returns the layout the object belongs to.
QVariant headerData(int section, Qt::Orientation orientation, int role=Qt::DisplayRole) const override
QgsLayoutGuideCollection(QgsLayout *layout, QgsLayoutPageCollection *pageCollection)
Constructor for QgsLayoutGuideCollection belonging to the specified layout, and linked to the specifi...
bool readXml(const QDomElement &collectionElement, const QDomDocument &document, const QgsReadWriteContext &context) override
Sets the collection's state from a DOM element.
QVariant data(const QModelIndex &index, int role) const override
void applyGuidesToAllOtherPages(int sourcePage)
Resets all other pages' guides to match the guides from the specified sourcePage.
void removeGuide(QgsLayoutGuide *guide)
Removes the specified guide, and deletes it.
QList< QgsLayoutGuide * > guidesOnPage(int page)
Returns the list of guides contained on a matching page.
bool removeRows(int row, int count, const QModelIndex &parent=QModelIndex()) override
void setGuideLayoutPosition(QgsLayoutGuide *guide, double position)
Sets the absolute position (in layout coordinates) for guide within the layout.
QList< QgsLayoutGuide * > guides()
Returns a list of all guides contained in the collection.
int rowCount(const QModelIndex &) const override
void clear()
Removes all guides from the collection.
Qt::ItemFlags flags(const QModelIndex &index) const override
bool visible() const
Returns true if the guide lines should be drawn.
@ OrientationRole
Guide orientation role.
@ PositionRole
Guide position role.
@ LayoutPositionRole
Guide position in layout coordinates.
@ UnitsRole
Guide position units role.
void update()
Updates the position (and visibility) of all guide line items.
void setVisible(bool visible)
Sets whether the guide lines should be visible.
void setPage(int page)
Sets the current page for filtering matching guides.
QgsLayoutGuideProxyModel(QObject *parent, Qt::Orientation orientation, int page)
Constructor for QgsLayoutGuideProxyModel, filtered to guides of the specified orientation and page on...
bool lessThan(const QModelIndex &left, const QModelIndex &right) const override
bool filterAcceptsRow(int sourceRow, const QModelIndex &sourceParent) const override
Contains the configuration for a single snap guide used by a layout.
QgsLayoutMeasurement position() const
Returns the guide's position within the page.
QgsLayoutItemPage * page()
Returns the page the guide is contained within.
Qt::Orientation orientation() const
Returns the guide's orientation.
QgsLayout * layout() const
Returns the layout the guide belongs to.
void setLayout(QgsLayout *layout)
Sets the layout the guide belongs to.
void setLayoutPosition(double position)
Sets the guide's position in absolute layout units.
void setPage(QgsLayoutItemPage *page)
Sets the page the guide is contained within.
void setPosition(QgsLayoutMeasurement position)
Sets the guide's position within the page.
QgsLayoutGuide(Qt::Orientation orientation, QgsLayoutMeasurement position, QgsLayoutItemPage *page)
Constructor for a new guide with the specified orientation and initial position.
double layoutPosition() const
Returns the guide's position in absolute layout units.
void positionChanged()
Emitted when the guide's position is changed.
void update()
Updates the position of the guide's line item.
QGraphicsLineItem * item()
Returns the guide's line item.
Item representing the paper in a layout.
int page() const
Returns the page the item is currently on, with the first page returning 0.
This class provides a method of storing measurements for use in QGIS layouts using a variety of diffe...
void setLength(const double length)
Sets the length of the measurement.
Qgis::LayoutUnit units() const
Returns the units for the measurement.
void setUnits(const Qgis::LayoutUnit units)
Sets the units for the measurement.
double length() const
Returns the length of the measurement.
A manager for a collection of pages in a layout.
void pageAboutToBeRemoved(int pageNumber)
Emitted just before a page is removed from the collection.
int pageCount() const
Returns the number of pages in the collection.
QgsLayoutItemPage * page(int pageNumber)
Returns a specific page (by pageNumber) from the collection.
int pageNumber(QgsLayoutItemPage *page) const
Returns the page number for the specified page, or -1 if the page is not contained in the collection.
void endCommand()
Saves final state of an object and pushes the active command to the undo history.
void beginCommand(QgsLayoutUndoObjectInterface *object, const QString &commandText, int id=0)
Begins a new undo command for the specified object.
Base class for layouts, which can contain items such as maps, labels, scalebars, etc.
Definition: qgslayout.h:50
@ ZGuide
Z-value for page guides.
Definition: qgslayout.h:61
QgsLayoutUndoStack * undoStack()
Returns a pointer to the layout's undo stack, which manages undo/redo states for the layout and it's ...
Definition: qgslayout.cpp:692
The class is used as a container of context for various read/write operations on other objects.
static Q_INVOKABLE QString toAbbreviatedString(Qgis::DistanceUnit unit)
Returns a translated abbreviation representing a distance unit.
static Q_INVOKABLE Qgis::LayoutUnit decodeLayoutUnit(const QString &string, bool *ok=nullptr)
Decodes a layout unit from a string.
static Q_INVOKABLE QString encodeUnit(Qgis::DistanceUnit unit)
Encodes a distance unit to a string.
bool qgsDoubleNear(double a, double b, double epsilon=4 *std::numeric_limits< double >::epsilon())
Compare two doubles (but allow some difference)
Definition: qgis.h:3509
QgsSignalBlocker< Object > whileBlocking(Object *object)
Temporarily blocks signals from a QObject while calling a single method from the object.
Definition: qgis.h:3435