Skip to content

Commit

Permalink
Add Last & Next zoom actions in layouts
Browse files Browse the repository at this point in the history
  • Loading branch information
YoannQDQ committed Nov 4, 2024
1 parent a4105a3 commit 8e059a5
Show file tree
Hide file tree
Showing 11 changed files with 191 additions and 4 deletions.
4 changes: 2 additions & 2 deletions python/PyQt6/gui/auto_additions/qgslayoutview.py
Original file line number Diff line number Diff line change
Expand Up @@ -5,8 +5,8 @@
QgsLayoutView.PasteModeCenter = QgsLayoutView.PasteMode.PasteModeCenter
QgsLayoutView.PasteModeInPlace = QgsLayoutView.PasteMode.PasteModeInPlace
try:
QgsLayoutView.__attribute_docs__ = {'layoutSet': 'Emitted when a ``layout`` is set for the view.\n\n.. seealso:: :py:func:`currentLayout`\n\n.. seealso:: :py:func:`setCurrentLayout`\n', 'toolSet': 'Emitted when the current ``tool`` is changed.\n\n.. seealso:: :py:func:`setTool`\n', 'zoomLevelChanged': 'Emitted whenever the zoom level of the view is changed.\n', 'cursorPosChanged': 'Emitted when the mouse cursor coordinates change within the view.\nThe ``layoutPoint`` argument indicates the cursor position within\nthe layout coordinate system.\n', 'pageChanged': 'Emitted when the page visible in the view is changed. This signal\nconsiders the page at the center of the view as the current visible\npage.\n\n.. seealso:: :py:func:`currentPage`\n', 'statusMessage': "Emitted when the view has a ``message`` for display in a parent window's\nstatus bar.\n\n.. seealso:: :py:func:`pushStatusMessage`\n", 'itemFocused': 'Emitted when an ``item`` is "focused" in the view, i.e. it becomes the active\nitem and should have its properties displayed in any designer windows.\n', 'willBeDeleted': 'Emitted in the destructor when the view is about to be deleted,\nbut is still in a perfectly valid state.\n'}
QgsLayoutView.__signal_arguments__ = {'layoutSet': ['layout: QgsLayout'], 'toolSet': ['tool: QgsLayoutViewTool'], 'cursorPosChanged': ['layoutPoint: QPointF'], 'pageChanged': ['page: int'], 'statusMessage': ['message: str'], 'itemFocused': ['item: QgsLayoutItem']}
QgsLayoutView.__attribute_docs__ = {'layoutSet': 'Emitted when a ``layout`` is set for the view.\n\n.. seealso:: :py:func:`currentLayout`\n\n.. seealso:: :py:func:`setCurrentLayout`\n', 'toolSet': 'Emitted when the current ``tool`` is changed.\n\n.. seealso:: :py:func:`setTool`\n', 'zoomLevelChanged': 'Emitted whenever the zoom level of the view is changed.\n', 'extentChanged': 'Emitted whenever the extent of the displayed scene has changed\n', 'cursorPosChanged': 'Emitted when the mouse cursor coordinates change within the view.\nThe ``layoutPoint`` argument indicates the cursor position within\nthe layout coordinate system.\n', 'pageChanged': 'Emitted when the page visible in the view is changed. This signal\nconsiders the page at the center of the view as the current visible\npage.\n\n.. seealso:: :py:func:`currentPage`\n', 'statusMessage': "Emitted when the view has a ``message`` for display in a parent window's\nstatus bar.\n\n.. seealso:: :py:func:`pushStatusMessage`\n", 'itemFocused': 'Emitted when an ``item`` is "focused" in the view, i.e. it becomes the active\nitem and should have its properties displayed in any designer windows.\n', 'willBeDeleted': 'Emitted in the destructor when the view is about to be deleted,\nbut is still in a perfectly valid state.\n', 'zoomLastStatusChanged': 'Emitted when zoom last status changed\n', 'zoomNextStatusChanged': 'Emitted when zoom next status changed\n'}
QgsLayoutView.__signal_arguments__ = {'layoutSet': ['layout: QgsLayout'], 'toolSet': ['tool: QgsLayoutViewTool'], 'cursorPosChanged': ['layoutPoint: QPointF'], 'pageChanged': ['page: int'], 'statusMessage': ['message: str'], 'itemFocused': ['item: QgsLayoutItem'], 'zoomLastStatusChanged': [': bool'], 'zoomNextStatusChanged': [': bool']}
QgsLayoutView.__group__ = ['layout']
except NameError:
pass
Expand Down
32 changes: 32 additions & 0 deletions python/PyQt6/gui/auto_generated/layout/qgslayoutview.sip.in
Original file line number Diff line number Diff line change
Expand Up @@ -334,6 +334,22 @@ Zooms to the actual size of the layout.
.. seealso:: :py:func:`zoomOut`
%End

void zoomLast();
%Docstring
Zooms to the previous zoom level

.. versionadded:: 3.42
%End

void zoomNext();
%Docstring
Zooms to the next zoom level

.. versionadded:: 3.42
%End



void emitZoomLevelChanged();


Expand Down Expand Up @@ -516,6 +532,11 @@ Emitted when the current ``tool`` is changed.
void zoomLevelChanged();
%Docstring
Emitted whenever the zoom level of the view is changed.
%End

void extentChanged();
%Docstring
Emitted whenever the extent of the displayed scene has changed
%End

void cursorPosChanged( QPointF layoutPoint );
Expand Down Expand Up @@ -554,6 +575,17 @@ Emitted in the destructor when the view is about to be deleted,
but is still in a perfectly valid state.
%End


void zoomLastStatusChanged( bool );
%Docstring
Emitted when zoom last status changed
%End

void zoomNextStatusChanged( bool );
%Docstring
Emitted when zoom next status changed
%End

protected:
virtual void mousePressEvent( QMouseEvent *event );

Expand Down
4 changes: 2 additions & 2 deletions python/gui/auto_additions/qgslayoutview.py
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
# The following has been generated automatically from src/gui/layout/qgslayoutview.h
try:
QgsLayoutView.__attribute_docs__ = {'layoutSet': 'Emitted when a ``layout`` is set for the view.\n\n.. seealso:: :py:func:`currentLayout`\n\n.. seealso:: :py:func:`setCurrentLayout`\n', 'toolSet': 'Emitted when the current ``tool`` is changed.\n\n.. seealso:: :py:func:`setTool`\n', 'zoomLevelChanged': 'Emitted whenever the zoom level of the view is changed.\n', 'cursorPosChanged': 'Emitted when the mouse cursor coordinates change within the view.\nThe ``layoutPoint`` argument indicates the cursor position within\nthe layout coordinate system.\n', 'pageChanged': 'Emitted when the page visible in the view is changed. This signal\nconsiders the page at the center of the view as the current visible\npage.\n\n.. seealso:: :py:func:`currentPage`\n', 'statusMessage': "Emitted when the view has a ``message`` for display in a parent window's\nstatus bar.\n\n.. seealso:: :py:func:`pushStatusMessage`\n", 'itemFocused': 'Emitted when an ``item`` is "focused" in the view, i.e. it becomes the active\nitem and should have its properties displayed in any designer windows.\n', 'willBeDeleted': 'Emitted in the destructor when the view is about to be deleted,\nbut is still in a perfectly valid state.\n'}
QgsLayoutView.__signal_arguments__ = {'layoutSet': ['layout: QgsLayout'], 'toolSet': ['tool: QgsLayoutViewTool'], 'cursorPosChanged': ['layoutPoint: QPointF'], 'pageChanged': ['page: int'], 'statusMessage': ['message: str'], 'itemFocused': ['item: QgsLayoutItem']}
QgsLayoutView.__attribute_docs__ = {'layoutSet': 'Emitted when a ``layout`` is set for the view.\n\n.. seealso:: :py:func:`currentLayout`\n\n.. seealso:: :py:func:`setCurrentLayout`\n', 'toolSet': 'Emitted when the current ``tool`` is changed.\n\n.. seealso:: :py:func:`setTool`\n', 'zoomLevelChanged': 'Emitted whenever the zoom level of the view is changed.\n', 'extentChanged': 'Emitted whenever the extent of the displayed scene has changed\n', 'cursorPosChanged': 'Emitted when the mouse cursor coordinates change within the view.\nThe ``layoutPoint`` argument indicates the cursor position within\nthe layout coordinate system.\n', 'pageChanged': 'Emitted when the page visible in the view is changed. This signal\nconsiders the page at the center of the view as the current visible\npage.\n\n.. seealso:: :py:func:`currentPage`\n', 'statusMessage': "Emitted when the view has a ``message`` for display in a parent window's\nstatus bar.\n\n.. seealso:: :py:func:`pushStatusMessage`\n", 'itemFocused': 'Emitted when an ``item`` is "focused" in the view, i.e. it becomes the active\nitem and should have its properties displayed in any designer windows.\n', 'willBeDeleted': 'Emitted in the destructor when the view is about to be deleted,\nbut is still in a perfectly valid state.\n', 'zoomLastStatusChanged': 'Emitted when zoom last status changed\n', 'zoomNextStatusChanged': 'Emitted when zoom next status changed\n'}
QgsLayoutView.__signal_arguments__ = {'layoutSet': ['layout: QgsLayout'], 'toolSet': ['tool: QgsLayoutViewTool'], 'cursorPosChanged': ['layoutPoint: QPointF'], 'pageChanged': ['page: int'], 'statusMessage': ['message: str'], 'itemFocused': ['item: QgsLayoutItem'], 'zoomLastStatusChanged': [': bool'], 'zoomNextStatusChanged': [': bool']}
QgsLayoutView.__group__ = ['layout']
except NameError:
pass
Expand Down
32 changes: 32 additions & 0 deletions python/gui/auto_generated/layout/qgslayoutview.sip.in
Original file line number Diff line number Diff line change
Expand Up @@ -334,6 +334,22 @@ Zooms to the actual size of the layout.
.. seealso:: :py:func:`zoomOut`
%End

void zoomLast();
%Docstring
Zooms to the previous zoom level

.. versionadded:: 3.42
%End

void zoomNext();
%Docstring
Zooms to the next zoom level

.. versionadded:: 3.42
%End



void emitZoomLevelChanged();


Expand Down Expand Up @@ -516,6 +532,11 @@ Emitted when the current ``tool`` is changed.
void zoomLevelChanged();
%Docstring
Emitted whenever the zoom level of the view is changed.
%End

void extentChanged();
%Docstring
Emitted whenever the extent of the displayed scene has changed
%End

void cursorPosChanged( QPointF layoutPoint );
Expand Down Expand Up @@ -554,6 +575,17 @@ Emitted in the destructor when the view is about to be deleted,
but is still in a perfectly valid state.
%End


void zoomLastStatusChanged( bool );
%Docstring
Emitted when zoom last status changed
%End

void zoomNextStatusChanged( bool );
%Docstring
Emitted when zoom next status changed
%End

protected:
virtual void mousePressEvent( QMouseEvent *event );

Expand Down
7 changes: 7 additions & 0 deletions src/app/layout/qgslayoutdesignerdialog.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -668,6 +668,11 @@ QgsLayoutDesignerDialog::QgsLayoutDesignerDialog( QWidget *parent, Qt::WindowFla
connect( mActionZoomActual, &QAction::triggered, mView, &QgsLayoutView::zoomActual );
connect( mActionZoomToWidth, &QAction::triggered, mView, &QgsLayoutView::zoomWidth );

connect( mActionZoomLast, &QAction::triggered, mView, &QgsLayoutView::zoomLast );
connect( mActionZoomNext, &QAction::triggered, mView, &QgsLayoutView::zoomNext );
connect( mView, &QgsLayoutView::zoomLastStatusChanged, mActionZoomLast, &QAction::setEnabled );
connect( mView, &QgsLayoutView::zoomNextStatusChanged, mActionZoomNext, &QAction::setEnabled );

connect( mActionSelectAll, &QAction::triggered, mView, &QgsLayoutView::selectAll );
connect( mActionDeselectAll, &QAction::triggered, mView, &QgsLayoutView::deselectAll );
connect( mActionInvertSelection, &QAction::triggered, mView, &QgsLayoutView::invertSelection );
Expand Down Expand Up @@ -4790,6 +4795,8 @@ void QgsLayoutDesignerDialog::toggleActions( bool layoutAvailable )
mActionZoomOut->setEnabled( layoutAvailable );
mActionZoomActual->setEnabled( layoutAvailable );
mActionZoomToWidth->setEnabled( layoutAvailable );
mActionZoomLast->setEnabled( layoutAvailable );
mActionZoomNext->setEnabled( layoutAvailable );
mActionAddPages->setEnabled( layoutAvailable );
mActionShowGrid->setEnabled( layoutAvailable );
mActionSnapGrid->setEnabled( layoutAvailable );
Expand Down
56 changes: 56 additions & 0 deletions src/gui/layout/qgslayoutview.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -64,7 +64,9 @@ QgsLayoutView::QgsLayoutView( QWidget *parent )
mPreviewEffect = new QgsPreviewEffect( this );
viewport()->setGraphicsEffect( mPreviewEffect );

connect( this, &QgsLayoutView::extentChanged, this, &QgsLayoutView::onExtentChanged );
connect( this, &QgsLayoutView::zoomLevelChanged, this, &QgsLayoutView::invalidateCachedRenders );
connect( this, &QgsLayoutView::zoomLevelChanged, this, &QgsLayoutView::extentChanged );

mScreenHelper = new QgsScreenHelper( this );
}
Expand Down Expand Up @@ -581,11 +583,63 @@ void QgsLayoutView::zoomActual()
setZoomLevel( 1.0 );
}

void QgsLayoutView::zoomLast()
{
mLastTransformIndex--;
centerOn( mLastTransform[mLastTransformIndex].first );
setTransform( mLastTransform[mLastTransformIndex].second );
invalidateCachedRenders();
// update controls' enabled state
emit zoomLastStatusChanged( mLastTransformIndex > 0 );
emit zoomNextStatusChanged( mLastTransformIndex < mLastTransform.size() - 1 );
}

void QgsLayoutView::zoomNext()
{
mLastTransformIndex++;
centerOn( mLastTransform[mLastTransformIndex].first );
setTransform( mLastTransform[mLastTransformIndex].second );
invalidateCachedRenders();
// update controls' enabled state
emit zoomLastStatusChanged( mLastTransformIndex > 0 );
emit zoomNextStatusChanged( mLastTransformIndex < mLastTransform.size() - 1 );
}

void QgsLayoutView::emitZoomLevelChanged()
{
emit zoomLevelChanged();
}

void QgsLayoutView::onExtentChanged()
{

//clear all transforms items after current index
for ( int i = mLastTransform.size() - 1; i > mLastTransformIndex; i-- )
{
mLastTransform.removeAt( i );
}

QPointF center = mapToScene( viewport()->rect().center() );

if ( mLastTransform.isEmpty() || mLastTransform.last().second != transform() || mLastTransform.last().first != center )
{
mLastTransform.append( {center, transform()} );
}

// adjust history to no more than 100
if ( mLastTransform.size() > 100 )
{
mLastTransform.removeAt( 0 );
}

// the last item is the current transform
mLastTransformIndex = mLastTransform.size() - 1;

// update controls' enabled state
emit zoomLastStatusChanged( mLastTransformIndex > 0 );
emit zoomNextStatusChanged( mLastTransformIndex < mLastTransform.size() - 1 );
}

void QgsLayoutView::selectAll()
{
if ( !currentLayout() )
Expand Down Expand Up @@ -1120,7 +1174,9 @@ void QgsLayoutView::keyReleaseEvent( QKeyEvent *event )
void QgsLayoutView::resizeEvent( QResizeEvent *event )
{
QGraphicsView::resizeEvent( event );
disconnect( this, &QgsLayoutView::zoomLevelChanged, this, &QgsLayoutView::extentChanged );
emit zoomLevelChanged();
connect( this, &QgsLayoutView::zoomLevelChanged, this, &QgsLayoutView::extentChanged );
viewChanged();
}

Expand Down
31 changes: 31 additions & 0 deletions src/gui/layout/qgslayoutview.h
Original file line number Diff line number Diff line change
Expand Up @@ -337,6 +337,20 @@ class GUI_EXPORT QgsLayoutView: public QGraphicsView
*/
void zoomActual();

/**
* Zooms to the previous zoom level
* \since QGIS 3.42
*/
void zoomLast();

/**
* Zooms to the next zoom level
* \since QGIS 3.42
*/
void zoomNext();



/**
* Emits the zoomLevelChanged() signal. This should be called after
* calling any of the QGraphicsView base class methods which alter
Expand Down Expand Up @@ -498,6 +512,11 @@ class GUI_EXPORT QgsLayoutView: public QGraphicsView
*/
void zoomLevelChanged();

/**
* Emitted whenever the extent of the displayed scene has changed
*/
void extentChanged();

/**
* Emitted when the mouse cursor coordinates change within the view.
* The \a layoutPoint argument indicates the cursor position within
Expand Down Expand Up @@ -532,6 +551,13 @@ class GUI_EXPORT QgsLayoutView: public QGraphicsView
*/
void willBeDeleted();


//! Emitted when zoom last status changed
void zoomLastStatusChanged( bool );

//! Emitted when zoom next status changed
void zoomNextStatusChanged( bool );

protected:
void mousePressEvent( QMouseEvent *event ) override;
void mouseReleaseEvent( QMouseEvent *event ) override;
Expand All @@ -548,6 +574,7 @@ class GUI_EXPORT QgsLayoutView: public QGraphicsView
private slots:

void invalidateCachedRenders();
void onExtentChanged();

private:

Expand Down Expand Up @@ -584,6 +611,10 @@ class GUI_EXPORT QgsLayoutView: public QGraphicsView
friend class QgsLayoutMouseHandles;

QGraphicsLineItem *createSnapLine() const;

//! recently used extent
QList <QPair<QPointF, QTransform>> mLastTransform;
int mLastTransformIndex = -1;
};


Expand Down
1 change: 1 addition & 0 deletions src/gui/layout/qgslayoutviewtoolpan.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -85,6 +85,7 @@ void QgsLayoutViewToolPan::layoutReleaseEvent( QgsLayoutViewMouseEvent *event )
}

mIsPanning = false;
emit view()->extentChanged();
view()->viewport()->setCursor( Qt::OpenHandCursor );
}

Expand Down
1 change: 1 addition & 0 deletions src/gui/layout/qgslayoutviewtooltemporarykeypan.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -38,6 +38,7 @@ void QgsLayoutViewToolTemporaryKeyPan::keyReleaseEvent( QKeyEvent *event )
if ( event->key() == Qt::Key_Space && !event->isAutoRepeat() )
{
view()->setTool( mPreviousViewTool );
emit view()->extentChanged();
}
}

Expand Down
1 change: 1 addition & 0 deletions src/gui/layout/qgslayoutviewtooltemporarymousepan.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -38,6 +38,7 @@ void QgsLayoutViewToolTemporaryMousePan::layoutReleaseEvent( QgsLayoutViewMouseE
if ( event->button() == Qt::MiddleButton )
{
view()->setTool( mPreviousViewTool );
emit view()->extentChanged();
}
}

Expand Down
26 changes: 26 additions & 0 deletions src/ui/layout/qgslayoutdesignerbase.ui
Original file line number Diff line number Diff line change
Expand Up @@ -116,6 +116,8 @@
<addaction name="mActionZoomOut"/>
<addaction name="mActionZoomActual"/>
<addaction name="mActionZoomAll"/>
<addaction name="mActionZoomLast"/>
<addaction name="mActionZoomNext"/>
<addaction name="mActionRefreshView"/>
</widget>
<widget class="QToolBar" name="mActionsToolbar">
Expand Down Expand Up @@ -1591,6 +1593,30 @@
<string>Page Properties…</string>
</property>
</action>
<action name="mActionZoomLast">
<property name="icon">
<iconset resource="../../../images/images.qrc">
<normaloff>:/images/themes/default/mActionZoomLast.svg</normaloff>:/images/themes/default/mActionZoomLast.svg</iconset>
</property>
<property name="text">
<string>Zoom Last</string>
</property>
<property name="shortcut">
<string>Alt+Left</string>
</property>
</action>
<action name="mActionZoomNext">
<property name="icon">
<iconset resource="../../../images/images.qrc">
<normaloff>:/images/themes/default/mActionZoomNext.svg</normaloff>:/images/themes/default/mActionZoomNext.svg</iconset>
</property>
<property name="text">
<string>Zoom Next</string>
</property>
<property name="shortcut">
<string>Alt+Right</string>
</property>
</action>
</widget>
<resources>
<include location="../../../images/images.qrc"/>
Expand Down

0 comments on commit 8e059a5

Please sign in to comment.