Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

23033: Parts are displayed at incorrect zoom level when the Navigator is open #25886

Open
wants to merge 1 commit into
base: master
Choose a base branch
from

Conversation

krasko78
Copy link
Contributor

Resolves: #23033

KEY POINTS:
Overview:

  1. When the Navigator is open, it creates a NotationNavigator instance which inherits from AbstractNotationPaintView.

  2. For the actual notation an instance of NotationPaintView is created which also inherits from AbstractNotationPaintView.

  3. Both of these listen for current notation changes in AbstractNotationPaintView::onNotationSetup() (line 622) and call onCurrentNotationChanged() (line 623).

  4. AbstractNotationPaintView::onCurrentNotationChanged() calls onLoadNotation (line 226).

  5. onLoadNotation is supposed to call m_inputController->initZoom(); (line 232) where the zoom on the notation should be set (if not previously set) following these calls:

  • NotationViewInputController::initZoom()
  • NotationViewInputController::setZoom / doZoomToPageWidth / doZoomToWholePage etc.
  • AbstractNotationPaintView::setScaling
  • AbstractNotationPaintView::scale
  • AbstractNotationPaintView::onMatrixChanged for the NotationNavigator instance or NotationPaintView::onMatrixChanged for the NotationPaintView instance (an override).

The problem is in that:

  1. The zoom is actually set in NotationPaintView::onMatrixChanged when it calls notation()->viewState()->setMatrix (line 62).

  2. The above steps execute first for the NotationNavigator instance since it registers to listen for onCurrentNotationChanged notification before the NotationPaintView instance registers itself.

  3. When the steps execute for the NotationNavigator instance, the zoom won't be set but NotationViewInputController::initZoom() will set isMatrixInited on the notation's viewstate to true (line 171).

  4. When the steps execute the second time for the NotationPaintView instance, onLoadNotation does not call m_inputController->initZoom() (line 232) because the if condition returns false due to isMatrixInited() being true for the notation.

  5. The zoom of the notation remains unset and defaults to 0.

When the Navigator is not visible, the NotationNavigator instance is not created and does not interfere.

Further observations:
When the Navigator is active the two AbstractNotationPaintView descendants share the same notation. As a result the code in AbstractNotationPaintView::onLoadNotation that operates on the notation executes twice, e.g. event listeners are bound twice and the handlers are executed twice. Therefore those pieces of code should probably be moved elsewhere (not as part of this issue though).

  • I signed the CLA
  • The title of the PR describes the problem it addresses
  • Each commit's message describes its purpose and effects, and references the issue it resolves
  • If changes are extensive, there is a sequence of easily reviewable commits
  • The code in the PR follows the coding rules
  • There are no unnecessary changes
  • The code compiles and runs on my machine, preferably after each commit individually
  • I created a unit test or vtest to verify the changes I made (if applicable)

@cbjeukendrup
Copy link
Contributor

Great investigation!
I'm not sure whether this is the safest fix though, because it relies on the fact that the onMatrixChanged from initZoom is the very first onMatrixChanged, and afaik there is no strong guarantee that that's the case.
Instead, maybe we should make different initZoom variants for different AbstractNotationPaintView classes. This would of course be easiest if it just were a method of AbstractNotationPaintView rather than NotationViewInputController. Or, you could create a virtual method initMatrix (or initZoomAndPosition) in AbstractNotationPaintView, and only in NotationPaintView let it call m_inputController->initZoom(); and m_inputController->initCanvasPos();.

@krasko78 krasko78 force-pushed the 23033-PartsDisplayedZoomedWhenTheNavigatorIsActive branch from 0510207 to f00043e Compare January 13, 2025 01:26
@krasko78
Copy link
Contributor Author

Something like this?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Parts are displayed at very large zoom level when the Navigator is activated
2 participants