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

Introduce Signal Based Navigation Model to have Self-Contained Pages #427

Merged
merged 5 commits into from
Oct 28, 2024

Conversation

jarolrod
Copy link
Member

Here we introduce a Signal Based navigation model, so that we can have self-contained pages. Self-contained pages include everything they need where they are declared, and do not have more than one definition. This makes pages modular, easier to reason about, but also easier to maintain by eliminating dependencies on hard-coded IDs, direct calls to parent views, and interlinked properties between pages.

When we want to reason about PageA.qml, we only have to look in there and the child components it uses, and not have to use a search function to ensure we have the entire scope of PageA.qml.

To Achieve this, this PR implements the following main ideas:

  1. Ensure that pages are not hardcoding and reaching into an outside view.
  2. Ensure we don't alias down any components into pages, and instead have pages define all their components, but react to state events
  3. Ensure we don't have interlinked properties between pages
  4. Move navigation and page linking logic from the pages themselves and into the declaration of the view itself.

Additionally, while here we structure the CreateWallet flow into a Wizard.

Follow-Ups:

Consider moving onboarding and walletName into more appropriate backend objects.

Build Artifacts

Pages exist within a view, in this page <-> view dependency, somewhere
inside of there, at least for views that are 'flows' or 'wizards,' there
needs to be logic on:

    a) when to move to a new page
    b) what page to move to.

We are currently handling this navigation logic primarily inside the
definition of a page itself. Inside of the page, we assume a hard-coded
ID for a parent view and call navigation function directly to the view
from within the page itself telling it to:

    a) to pop or push
    b) the id of the page we want to push.

This is bad because:

    1. It creates a dependency between page declarations that does not
       need to exist. As in, PageB.qml doesn't really exist unless
       PageA.qml includes code to go to it.

    2. This prevents us from being able to 'statically' define and
       clearly contain pages. Clearly containing pages makes it so that
       when we want to reason about PageA.qml, we only have to look
       inside of PageA.qml and the components that are used within that
       file; we don't have to go on a chase for interlinked dependencies
       on id's or properties that are hidden or at least not easy to
       find in other pages.

    3. We should never assume outside id's from within a page, as these
       can change, and when they change, the page that assumes said
       outside id is now broken, and debugging can be annoying.

All of this is unnecessary.

In this commit, we address the issue of the logic for navigation being
contained within the page definitions. This proposes a new navigation
model based on signals and first applies it to the onboarding flow.
Pages within our onboarding flow now:

    1. Emit a signal when it's time to go to a new page
    2. The parent View itself handles the signals and contains the logic
       for navigation.

This removes the dependence on having a hardcoded outside ID within a
a page, and is one step to moving us to pages with no dependency on
other pages.
Here, pages are clearly contained, but the logic of moving to
SettingsDeveloper from within SettingsAbout and SettingsProxy from
within SettingsConnection is still contained here within the definitions
of SettingsAbout and SettingsConnection. While all pages are
independently still usable, we do want to iron out this one kink in a
follow-up.
Makes our CreateWallet flow into a wizard, as that is what it should be.
The CreateWalletWizard is introduced as a StackView with the associated
pages implementing signal based navigation.

Previously, there was an interlinked dependency between CreateName and
CreatePassword in that CreateName would pass a string for walletName
to CreatePassword. This dependency is removed by having CreateName set
the string in a property contained in the CreateWalletWizard StackView,
and CreatePassword pulling in this property to satisfy its requirement
for walletName. This is a temporary workaround so that we can still
have clearly contained pages here. A follow-up should move this into a
more appropriate backend object
@jarolrod jarolrod mentioned this pull request Oct 25, 2024
Copy link
Contributor

@MarnixCroes MarnixCroes left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

cACK
I went through all the pages, including onboarding and it work ok.

related lint failure should be fixable

src/qml/pages/settings/SettingsConnection.qml Outdated Show resolved Hide resolved
onNext: aboutSwipe.incrementCurrentIndex()
}

states : [
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

nit

Suggested change
states : [
states: [

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

we should set a style guide actually

src/qml/pages/settings/SettingsConnection.qml Outdated Show resolved Hide resolved
src/qml/pages/settings/SettingsStorage.qml Outdated Show resolved Hide resolved
@johnny9
Copy link
Contributor

johnny9 commented Oct 26, 2024

Concept ACK

No qml errors when going through onboard. Looks like there are just some whitespace issues to fix to make the lint job happy.

In containing pages, everything a page needs should be defined where the
page is declared. Allowing to alias nav elements into a page breaks
the idea of pages being statically defined. The action of aliasing
down nav elements means that the real declaration of a page is not in
its 'filename.qml', but instead where the page is used, meaning a page
can be defined an infinite amount of times in an infinite amount of
ways; we don't want this.

The only reason that we were aliasing down nav elements is that we have
reusable pages for settings and what elements the nav bar will contain
is different depending on whether we are onboarding or not. So, we
should treat that as the navbar for the page has two states, so we
should have logic for these states instead of defining the page in
different locations in different ways.

This state logic is handled by setting a property of onboarding or not,
a follow-up can potentially move this into an appropriate backend object
such as AppMode.
Renaming for consistency with other pages that 'back' signal and handles
the 'back' signal. No reason to have some pages call this differently.
@jarolrod
Copy link
Member Author

Updated from from 59aeba2 to 3f94240 compare

Changes: address style feedback and fix white-space lint

Copy link
Contributor

@johnny9 johnny9 left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

ACK 3f94240

Copy link
Contributor

@MarnixCroes MarnixCroes left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

ACK 3f94240

@hebasto hebasto merged commit c117acc into bitcoin-core:main Oct 28, 2024
9 checks passed
Copy link
Contributor

@pablomartin4btc pablomartin4btc left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

post-merge tACK 3f94240

we should set a style guide actually

As discussed offline, we should do that + some specific dev notes for this repo.

hebasto added a commit that referenced this pull request Nov 24, 2024
c52566d qml: Swap existing SwipeViews for PageStack (jarolrod)
f0e5951 qml: Swap existing StackViews for PageStack (jarolrod)
7ba2f62 qml: declare OnboardingWizard file, reintroduce as PageStack (jarolrod)
d49b888 qml: Introduce PageStack control (jarolrod)

Pull request description:

  After fixing our sins in #427, migrating to A StackView based GUI is quite simple.

  Here we use a custom StackView control, PageStack, that has a property `vertical` used to declare if we want vertical or horizontal animations, and additionally implements the [custom animation](#422) we want.

  Closes #422
  Closes #219

  [![Build Artifacts](https://img.shields.io/badge/Build%20Artifacts-green
  )]()

ACKs for top commit:
  pablomartin4btc:
    re-ACK c52566d
  MarnixCroes:
    tACK c52566d
  D33r-Gee:
    tACK  [c52566d](c52566d) Works as expected on WSL Ubuntu 22.04
  johnny9:
    ACK c52566d

Tree-SHA512: f9a025944db24a46e1f78f4ad8a9b9aca4e1f3ff4dd5927eebfa2a7fb28ed390a17da79e23c6248c3c0e82b361ff1b2dedbf4df9df2a1d0677b05bacb7763bcb
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
enhancement New feature or request
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants