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

[e2e] Mobile budget menu modal VRT #3583

Open
wants to merge 17 commits into
base: master
Choose a base branch
from

Conversation

joel-jeremy
Copy link
Contributor

No description provided.

@actual-github-bot actual-github-bot bot changed the title Mobile budget menu modal VRT [WIP] Mobile budget menu modal VRT Oct 6, 2024
Copy link

netlify bot commented Oct 6, 2024

Deploy Preview for actualbudget ready!

Name Link
🔨 Latest commit a4f45cc
🔍 Latest deploy log https://app.netlify.com/sites/actualbudget/deploys/678164119ec3460008958e5e
😎 Deploy Preview https://deploy-preview-3583.demo.actualbudget.org
📱 Preview on mobile
Toggle QR Code...

QR Code

Use your smartphone camera to open QR code link.

To edit notification comments on pull requests, go to your Netlify site configuration.

Copy link
Contributor

github-actions bot commented Oct 6, 2024

Bundle Stats — desktop-client

Hey there, this message comes from a GitHub action that helps you and reviewers to understand how these changes affect the size of this project's bundle.

As this PR is updated, I'll keep you updated on how the bundle size is impacted.

Total

Files count Total bundle size % Changed
18 5.93 MB → 5.92 MB (-18.42 kB) -0.30%
Changeset
File Δ Size
node_modules/clsx/dist/clsx.js 🆕 +509 B 0 B → 509 B
node_modules/clsx/dist/clsx.js?commonjs-module 🆕 +27 B 0 B → 27 B
src/components/mobile/MobileNavTabs.tsx 📈 +227 B (+4.40%) 5.04 kB → 5.26 kB
src/components/Page.tsx 📈 +22 B (+0.77%) 2.78 kB → 2.8 kB
src/components/settings/index.tsx 📈 +33 B (+0.45%) 7.21 kB → 7.24 kB
src/components/reports/Overview.tsx 📈 +41 B (+0.27%) 14.79 kB → 14.83 kB
src/components/Notifications.tsx 📈 +19 B (+0.27%) 6.97 kB → 6.99 kB
src/components/mobile/transactions/TransactionEdit.jsx 📈 +87 B (+0.25%) 33.6 kB → 33.68 kB
src/components/mobile/accounts/Accounts.tsx 📈 +10 B (+0.12%) 8.33 kB → 8.34 kB
locale/en.json 📈 +2 B (+0.00%) 93.64 kB → 93.64 kB
node_modules/react-grid-layout/build/ReactGridLayout.js 📉 -1 B (-0.00%) 24.96 kB → 24.96 kB
node_modules/react-grid-layout/build/GridItem.js 📉 -1 B (-0.00%) 21.49 kB → 21.49 kB
node_modules/react-grid-layout/build/components/WidthProvider.js 📉 -1 B (-0.02%) 5.22 kB → 5.22 kB
locale/de.json 📉 -194 B (-3.97%) 4.78 kB → 4.59 kB
locale/uk.json 📉 -15.6 kB (-11.45%) 136.3 kB → 120.7 kB
locale/fr.json 📉 -2.88 kB (-18.62%) 15.49 kB → 12.6 kB
locale/pl.json 📉 -281 B (-94.30%) 298 B → 17 B
node_modules/clsx/dist/clsx.mjs 🔥 -368 B (-100%) 368 B → 0 B
node_modules/clsx/dist/clsx.mjs?commonjs-proxy 🔥 -64 B (-100%) 64 B → 0 B
View detailed bundle breakdown

Added

No assets were added

Removed

No assets were removed

Bigger

Asset File Size % Changed
static/js/index.js 3.74 MB → 3.74 MB (+556 B) +0.01%
static/js/narrow.js 84.3 kB → 84.31 kB (+10 B) +0.01%
static/js/en.js 93.64 kB → 93.64 kB (+2 B) +0.00%

Smaller

Asset File Size % Changed
static/js/uk.js 136.3 kB → 120.7 kB (-15.6 kB) -11.45%
static/js/fr.js 15.49 kB → 12.6 kB (-2.88 kB) -18.62%
static/js/pl.js 298 B → 17 B (-281 B) -94.30%
static/js/de.js 4.78 kB → 4.59 kB (-194 B) -3.97%
static/js/ReportRouter.js 1.58 MB → 1.58 MB (-26 B) -0.00%

Unchanged

Asset File Size % Changed
static/js/es.js 3.45 kB 0%
static/js/ta.js 17 B 0%
static/js/indexeddb-main-thread-worker-e59fee74.js 13.5 kB 0%
static/js/useAccountPreviewTransactions.js 1.63 kB 0%
static/js/workbox-window.prod.es5.js 5.69 kB 0%
static/js/AppliedFilters.js 10.24 kB 0%
static/js/resize-observer.js 18.37 kB 0%
static/js/zh-Hant.js 10.13 kB 0%
static/js/BackgroundImage.js 122.29 kB 0%
static/js/wide.js 104.78 kB 0%

Copy link
Contributor

github-actions bot commented Oct 6, 2024

Bundle Stats — loot-core

Hey there, this message comes from a GitHub action that helps you and reviewers to understand how these changes affect the size of this project's bundle.

As this PR is updated, I'll keep you updated on how the bundle size is impacted.

Total

Files count Total bundle size % Changed
1 1.33 MB 0%

Changeset

No files were changed

View detailed bundle breakdown

Added

No assets were added

Removed

No assets were removed

Bigger

No assets were bigger

Smaller

No assets were smaller

Unchanged

Asset File Size % Changed
kcab.worker.js 1.33 MB 0%

Copy link
Contributor

coderabbitai bot commented Oct 6, 2024

Walkthrough

This pull request introduces comprehensive enhancements to the mobile testing framework and page models for budget management, navigation, and transaction interactions. The changes primarily focus on improving end-to-end testing capabilities by adding new modal classes, refactoring existing page models, and introducing more robust interaction methods. Key modifications include creating modal classes for budget, balance, category, and notes interactions, updating navigation methods, and adding more granular test identifiers across various components. The changes aim to provide more structured and flexible ways of interacting with the mobile application's user interface during testing.

Possibly related PRs

Suggested labels

e2e, mobile, testing, refactoring

Suggested reviewers

  • MikesGlitch
  • JohnDoe (if available)
  • SarahSmith (if available)

Finishing Touches

  • 📝 Generate Docstrings (Beta)

Thank you for using CodeRabbit. We offer it for free to the OSS community and would appreciate your support in helping us grow. If you find it useful, would you consider giving us a shout-out on your favorite social media?

❤️ Share
🪧 Tips

Chat

There are 3 ways to chat with CodeRabbit:

  • Review comments: Directly reply to a review comment made by CodeRabbit. Example:
    • I pushed a fix in commit <commit_id>, please review it.
    • Generate unit testing code for this file.
    • Open a follow-up GitHub issue for this discussion.
  • Files and specific lines of code (under the "Files changed" tab): Tag @coderabbitai in a new review comment at the desired location with your query. Examples:
    • @coderabbitai generate unit testing code for this file.
    • @coderabbitai modularize this function.
  • PR comments: Tag @coderabbitai in a new PR comment to ask questions about the PR branch. For the best results, please provide a very specific query, as very limited context is provided in this mode. Examples:
    • @coderabbitai gather interesting stats about this repository and render them as a table. Additionally, render a pie chart showing the language distribution in the codebase.
    • @coderabbitai read src/utils.ts and generate unit testing code.
    • @coderabbitai read the files in the src/scheduler package and generate a class diagram using mermaid and a README in the markdown format.
    • @coderabbitai help me debug CodeRabbit configuration file.

Note: Be mindful of the bot's finite context window. It's strongly recommended to break down tasks such as reading entire modules into smaller chunks. For a focused discussion, use review comments to chat about specific files and their changes, instead of using the PR comments.

CodeRabbit Commands (Invoked using PR comments)

  • @coderabbitai pause to pause the reviews on a PR.
  • @coderabbitai resume to resume the paused reviews.
  • @coderabbitai review to trigger an incremental review. This is useful when automatic reviews are disabled for the repository.
  • @coderabbitai full review to do a full review from scratch and review all the files again.
  • @coderabbitai summary to regenerate the summary of the PR.
  • @coderabbitai generate docstrings to generate docstrings for this PR. (Beta)
  • @coderabbitai resolve resolve all the CodeRabbit review comments.
  • @coderabbitai configuration to show the current CodeRabbit configuration for the repository.
  • @coderabbitai help to get help.

Other keywords and placeholders

  • Add @coderabbitai ignore anywhere in the PR description to prevent this PR from being reviewed.
  • Add @coderabbitai summary to generate the high-level summary at a specific location in the PR description.
  • Add @coderabbitai anywhere in the PR title to generate the title automatically.

CodeRabbit Configuration File (.coderabbit.yaml)

  • You can programmatically configure CodeRabbit by adding a .coderabbit.yaml file to the root of your repository.
  • Please see the configuration documentation for more information.
  • If your editor has YAML language server enabled, you can add the path at the top of this file to enable auto-completion and validation: # yaml-language-server: $schema=https://coderabbit.ai/integrations/schema.v2.json

Documentation and Community

  • Visit our Documentation for detailed information on how to use CodeRabbit.
  • Join our Discord Community to get help, request features, and share feedback.
  • Follow us on X/Twitter for updates and announcements.

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

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

Actionable comments posted: 5

🧹 Outside diff range and nitpick comments (1)
packages/desktop-client/e2e/budget.mobile.test.js (1)

261-271: Clarify the budget amount input format

In the test case, you're setting the budget amount using the string '12300':

await budgetMenuModal.setBudgetAmount('12300');

And then expecting the budgeted button to display '123.00':

await expect(budgetedButton).toHaveText('123.00');

This might cause confusion regarding the input format for setBudgetAmount. If the amount is expected in cents, consider adding a comment to clarify this. Alternatively, input the amount as a decimal string to make the code clearer:

- await budgetMenuModal.setBudgetAmount('12300');
+ await budgetMenuModal.setBudgetAmount('123.00');

This makes the intention explicit and improves code readability.

📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Files that changed from the base of the PR and between dc5d117 and 269e34c.

⛔ Files ignored due to path filters (24)
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-12-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-12-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-12-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-3-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-3-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-3-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-6-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-6-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-6-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-updates-the-budgeted-amount-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-updates-the-budgeted-amount-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-updates-the-budgeted-amount-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-12-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-12-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-12-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-3-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-3-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-3-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-6-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-6-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-6-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-updates-the-budgeted-amount-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-updates-the-budgeted-amount-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-updates-the-budgeted-amount-3-chromium-linux.png is excluded by !**/*.png
📒 Files selected for processing (6)
  • packages/desktop-client/e2e/budget.mobile.test.js (2 hunks)
  • packages/desktop-client/e2e/page-models/account-page.js (1 hunks)
  • packages/desktop-client/e2e/page-models/close-account-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-budget-menu-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-budget-page.js (2 hunks)
  • packages/desktop-client/src/components/mobile/transactions/FocusableAmountInput.tsx (1 hunks)
✅ Files skipped from review due to trivial changes (1)
  • packages/desktop-client/e2e/page-models/account-page.js
🧰 Additional context used
📓 Learnings (1)
packages/desktop-client/e2e/page-models/close-account-modal.js (1)
Learnt from: UnderKoen
PR: actualbudget/actual#3499
File: packages/desktop-client/e2e/page-models/close-account-modal.js:16-18
Timestamp: 2024-09-24T20:20:59.061Z
Learning: In the `CloseAccountModal` class, methods are implemented without additional error handling, return values, or additional comments to maintain consistency with existing codebase practices.
🔇 Additional comments (11)
packages/desktop-client/e2e/page-models/close-account-modal.js (4)

2-4: LGTM! Constructor update improves element interaction.

The change from rootPage to locator in the constructor signature and assignment is a good improvement. This update likely allows for more precise and flexible element selection, which is beneficial for robust end-to-end testing.


8-9: Great improvement in element selection and user simulation!

The changes in this method enhance the robustness of the test:

  1. Using getByPlaceholder for element selection is more reliable and readable.
  2. Adding the Enter key press simulates real user behavior more accurately.

These updates will likely result in more stable and realistic tests.


13-13: Excellent use of accessibility-aware selector!

The change to use getByRole with a specific button name is a significant improvement. This approach:

  1. Makes the test more resilient to UI changes.
  2. Aligns with accessibility best practices.
  3. Improves the test's readability and maintainability.

This update demonstrates a commitment to both robust testing and accessible UI design.


1-15: Overall excellent improvements to the CloseAccountModal class!

The changes made to this class consistently enhance the quality and reliability of the end-to-end tests:

  1. The shift from rootPage to locator allows for more precise element selection.
  2. The use of semantic selectors like getByPlaceholder and getByRole improves test robustness and aligns with accessibility best practices.
  3. The addition of keyboard interactions in selectTransferAccount more accurately simulates user behavior.

These updates maintain the class's simplicity while significantly improving its effectiveness in testing. Great work on modernizing and strengthening these tests!

packages/desktop-client/e2e/page-models/mobile-budget-menu-modal.js (4)

1-23: LGTM: Well-structured class following Page Object Model pattern

The BudgetMenuModal class is well-structured and follows the Page Object Model pattern, which is excellent for e2e testing. The constructor properly initializes the page and locator, and the UI elements are well-defined using appropriate locator strategies.


25-27: LGTM: Concise and accessible close() method

The close() method is well-implemented. It uses a role-based selector to find the close button, which is good for accessibility and test resilience.


29-33: LGTM: Comprehensive setBudgetAmount() method

The setBudgetAmount() method is well-implemented. It properly handles type conversion, triggers blur events, and follows up with closing the modal, which represents a complete interaction flow.


35-53: LGTM: Consistent and clear action methods

The action methods (copyLastMonthBudget(), setTo3MonthAverage(), setTo6MonthAverage(), setToYearlyAverage(), and applyBudgetTemplate()) are consistently implemented. They are concise, use pre-defined locators, and have clear, descriptive names.

packages/desktop-client/src/components/mobile/transactions/FocusableAmountInput.tsx (1)

153-153: Approve change and update related tests

The change from "amount-fake-input" to "amount-input-text" improves the clarity of the test identifier. This new identifier more accurately describes the purpose of the Text component, which displays the amount either as editing text or formatted currency.

Please ensure that any existing tests using the previous test identifier "amount-fake-input" are updated to use "amount-input-text". Run the following script to check for any occurrences of the old test identifier:

If any occurrences are found, update them to use the new test identifier.

packages/desktop-client/e2e/page-models/mobile-budget-page.js (1)

2-2: LGTM: New import for BudgetMenuModal.

The new import statement for BudgetMenuModal is correctly placed and follows the existing import style in the file. This import is likely related to the changes in the openBudgetMenu method, indicating a refactoring of the budget menu functionality.

packages/desktop-client/e2e/budget.mobile.test.js (1)

293-320: Ensure correct usage of setBudgetAverageFn in test cases

When iterating over the array of [numberOfMonths, setBudgetAverageFn], you're passing setBudgetAverageFn to the setBudgetAverage function:

const averageSpent = await setBudgetAverage(
  budgetPage,
  categoryName,
  numberOfMonths,
  setBudgetAverageFn,
);

Given the earlier issue with naming conflicts in setBudgetAverage, ensure that setBudgetAverageFn is correctly defined and passed after renaming the parameter in the setBudgetAverage function.

Double-check that setBudgetAverageFn corresponds to the appropriate function (setTo3MonthAverage, setTo6MonthAverage, setToYearlyAverage) and that these functions are correctly refactored if you implement the earlier refactoring suggestion.

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

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

Actionable comments posted: 1

🧹 Outside diff range and nitpick comments (2)
packages/desktop-client/e2e/budget.mobile.test.js (2)

33-59: LGTM: setBudgetAverage function implementation

The setBudgetAverage function correctly calculates the average spending over a specified number of months. It properly navigates through previous months, retrieves spent amounts, and returns to the current month.

One potential improvement to consider:

  • The function modifies the current month during its execution. While it does return to the original month, this approach might lead to unexpected behavior if there's an error during execution. Consider using a separate function to retrieve historical data without changing the current month.

292-318: LGTM: New test cases for setting budget to average amounts

These new test cases effectively verify the functionality of setting budgets to 3, 6, and 12 month averages. The use of a loop to generate multiple similar test cases is efficient and reduces code duplication. Each test correctly uses the setBudgetAverage function to calculate the expected average and verify the result.

One point to consider:

  • The use of Math.abs() in the expectation suggests that negative averages are being handled. It might be worth adding a comment or a separate test case to explicitly cover the scenario of negative averages, ensuring that this behavior is intentional and correctly implemented.
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Files that changed from the base of the PR and between 269e34c and ffa6635.

⛔ Files ignored due to path filters (1)
  • upcoming-release-notes/3583.md is excluded by !**/*.md
📒 Files selected for processing (1)
  • packages/desktop-client/e2e/budget.mobile.test.js (2 hunks)
🧰 Additional context used
🔇 Additional comments (3)
packages/desktop-client/e2e/budget.mobile.test.js (3)

3-3: LGTM: New import statement for currency conversion utilities

The addition of amountToCurrency and currencyToAmount imports is appropriate for the new budget average calculation functionality.


260-270: LGTM: Updated test case for setting budgeted amount

The test case has been successfully updated to use the new budgetMenuModal object and its setBudgetAmount method. This change aligns with the new budget menu modal functionality and maintains the correct verification of the budgeted amount update.


272-290: LGTM: New test case for copying last month's budget

This new test case effectively verifies the 'copy last month's budget' functionality. It correctly:

  1. Navigates to the previous month
  2. Stores the previous month's budget
  3. Returns to the current month
  4. Applies the copy action
  5. Verifies that the current month's budget matches the previous month's budget

The use of the copyLastMonthBudget utility function is appropriate and aligns with the new functionality introduced earlier in the file.

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

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

Actionable comments posted: 5

🧹 Outside diff range and nitpick comments (8)
packages/desktop-client/e2e/page-models/mobile-edit-notes-modal.js (3)

2-9: Constructor looks good, consider adding input validation.

The constructor is well-structured and uses roles for locators, which is a good practice. However, consider adding type checking or validation for the input parameters to improve robustness.

You could add simple checks like:

if (!page || !locator) {
  throw new Error('Page and locator are required parameters');
}

11-13: close method looks good, consider adding error handling.

The close method is concise and uses roles for element selection, which is good. However, consider adding error handling to make the method more robust.

You could improve error handling like this:

async close() {
  try {
    await this.heading.getByRole('button', { name: 'Close' }).click();
  } catch (error) {
    console.error('Failed to close the modal:', error);
    throw new Error('Failed to close the modal');
  }
}

15-18: updateNotes method is functional, consider adding input validation and error handling.

The updateNotes method performs the expected actions for updating notes. However, it could benefit from input validation and error handling to improve robustness.

Consider improving the method like this:

async updateNotes(notes) {
  if (typeof notes !== 'string') {
    throw new TypeError('Notes must be a string');
  }
  try {
    await this.textArea.fill(notes);
    await this.saveNotesButton.click();
  } catch (error) {
    console.error('Failed to update notes:', error);
    throw new Error('Failed to update notes');
  }
}
packages/desktop-client/e2e/page-models/mobile-category-menu-modal.js (2)

3-11: LGTM: Well-structured class with clear property definitions.

The CategoryMenuModal class is well-designed with a clear constructor and descriptive property names. The use of getByRole and getByTestId demonstrates good practices for accessibility and testability.

Consider adding JSDoc comments to describe the purpose of the class and the parameters of the constructor. This would enhance code readability and maintainability.


17-21: LGTM: Well-implemented editNotes method with good separation of concerns.

The editNotes method is clear and follows good practices by creating a new EditNotesModal instance for further interactions.

Consider adding error handling to catch potential issues when clicking the button or creating the new modal. For example:

async editNotes() {
  try {
    await this.editNotesButton.click();
    return new EditNotesModal(this.page, this.page.getByRole('dialog'));
  } catch (error) {
    console.error('Error opening edit notes modal:', error);
    throw error;
  }
}
packages/desktop-client/e2e/page-models/settings-page.js (1)

22-24: Approve changes with a minor suggestion for improvement.

The added conditional checks for element visibility and state before interaction are excellent improvements. They enhance the robustness of the enableExperimentalFeature method by preventing potential errors when elements are not visible or already in the desired state.

To further improve code readability and maintainability, consider extracting these checks into separate helper methods.

Here's a suggested refactoring:

async clickIfVisible(element) {
  if (await element.isVisible()) {
    await element.click();
  }
}

async checkIfUnchecked(checkbox) {
  if (!(await checkbox.isChecked())) {
    await checkbox.click();
  }
}

async enableExperimentalFeature(featureName) {
  await this.clickIfVisible(this.page.getByTestId('advanced-settings'));
  await this.clickIfVisible(this.page.getByTestId('experimental-settings'));
  await this.checkIfUnchecked(this.page.getByRole('checkbox', { name: featureName }));
}

This refactoring improves code reusability and makes the enableExperimentalFeature method more concise and easier to read.

Also applies to: 29-31, 36-38

packages/desktop-client/e2e/budget.mobile.test.js (1)

9-31: LGTM with suggestion: Consider refactoring to reduce duplication

The new utility functions (copyLastMonthBudget, setTo3MonthAverage, setTo6MonthAverage, and setToYearlyAverage) are correctly implemented and serve their intended purposes. However, as previously suggested, there's an opportunity to reduce code duplication by refactoring these functions into a single higher-order function.

Consider implementing a higher-order function to handle these actions:

const performBudgetMenuAction = async (budgetPage, categoryName, action) => {
  const budgetMenuModal = await budgetPage.openBudgetMenu(categoryName);
  await action(budgetMenuModal);
  await budgetMenuModal.close();
};

const copyLastMonthBudget = (budgetPage, categoryName) =>
  performBudgetMenuAction(budgetPage, categoryName, menu => menu.copyLastMonthBudget());

const setTo3MonthAverage = (budgetPage, categoryName) =>
  performBudgetMenuAction(budgetPage, categoryName, menu => menu.setTo3MonthAverage());

// ... similar for setTo6MonthAverage and setToYearlyAverage

This refactoring will make the code more maintainable and easier to extend in the future.

packages/desktop-client/src/components/mobile/MobileNavTabs.tsx (1)

Line range hint 150-160: Include all dependencies in useEffect dependency array

The useEffect hook depends on scrollY, previousScrollY, hide, and openDefault, but only scrollY is included in the dependency array. Omitting dependencies can lead to stale closures and unexpected behavior. Include all external variables used within the useEffect to ensure it runs correctly when any of them change.

Apply this diff to include missing dependencies:

 useEffect(() => {
   if (
     scrollY &&
     previousScrollY &&
     scrollY > previousScrollY &&
     previousScrollY !== 0
   ) {
     hide();
   } else {
     openDefault();
   }
- }, [scrollY]);
+ }, [scrollY, previousScrollY, hide, openDefault]);
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Files that changed from the base of the PR and between ffa6635 and bc45c35.

⛔ Files ignored due to path filters (16)
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-applies-budget-template-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-applies-budget-template-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-applies-budget-template-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-copies-last-month-s-budget-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-copies-last-month-s-budget-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-copies-last-month-s-budget-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-applies-budget-template-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-applies-budget-template-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-applies-budget-template-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-copies-last-month-s-budget-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-copies-last-month-s-budget-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-copies-last-month-s-budget-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/settings.mobile.test.js-snapshots/Mobile-Settings-checks-that-settings-page-can-be-opened-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/settings.mobile.test.js-snapshots/Mobile-Settings-checks-that-settings-page-can-be-opened-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/settings.mobile.test.js-snapshots/Mobile-Settings-checks-that-settings-page-can-be-opened-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/settings.mobile.test.js-snapshots/Mobile-Settings-checks-that-settings-page-can-be-opened-4-chromium-linux.png is excluded by !**/*.png
📒 Files selected for processing (7)
  • packages/desktop-client/e2e/budget.mobile.test.js (2 hunks)
  • packages/desktop-client/e2e/page-models/mobile-budget-page.js (3 hunks)
  • packages/desktop-client/e2e/page-models/mobile-category-menu-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-edit-notes-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-navigation.js (1 hunks)
  • packages/desktop-client/e2e/page-models/settings-page.js (1 hunks)
  • packages/desktop-client/src/components/mobile/MobileNavTabs.tsx (7 hunks)
🧰 Additional context used
🔇 Additional comments (19)
packages/desktop-client/e2e/page-models/mobile-edit-notes-modal.js (1)

1-19: LGTM: Well-structured class following Page Object Model pattern.

The EditNotesModal class is well-structured and follows the Page Object Model pattern, which is a best practice for e2e testing. The class and method names are descriptive and follow appropriate naming conventions.

packages/desktop-client/e2e/page-models/mobile-category-menu-modal.js (3)

1-1: LGTM: Import statement is appropriate.

The import of EditNotesModal is correctly used in the editNotes method, maintaining good modularity.


13-15: LGTM: Concise and well-implemented close method.

The close method is clear, concise, and follows good practices by using role-based element selection.


1-22: Overall, excellent implementation of the CategoryMenuModal class.

This new file introduces a well-structured and clearly implemented CategoryMenuModal class for e2e testing. The code demonstrates good practices in terms of:

  1. Modularity and encapsulation
  2. Accessibility considerations (use of getByRole)
  3. Testability (use of getByTestId)
  4. Clear and descriptive naming conventions

The suggestions for improvement are minor and include:

  1. Adding JSDoc comments for better documentation
  2. Implementing error handling in the editNotes method

Great job on this implementation!

packages/desktop-client/e2e/page-models/mobile-budget-page.js (4)

2-3: LGTM: New imports added correctly.

The new import statements for BudgetMenuModal and CategoryMenuModal are correctly formatted and align with the existing code style. These imports support the changes made to the openBudgetMenu and openCategoryMenu methods.


Line range hint 1-300: Overall: Good improvements to modularity, verify impacts on calling code.

The changes in this file improve modularity and separation of concerns by introducing BudgetMenuModal and CategoryMenuModal. This is a positive step towards more maintainable code.

However, since both openBudgetMenu and openCategoryMenu now return values instead of being void methods, it's crucial to ensure that all calling code is updated accordingly. This includes both the application code and the test suite.

Please run the verification scripts provided in the previous comments to identify all occurrences where these methods are called, and update them as necessary to handle the returned modal instances.


184-184: LGTM: Improved modularity in openBudgetMenu.

The changes to the openBudgetMenu method improve modularity by returning a BudgetMenuModal instance. This encapsulation of budget menu interactions is a good practice.

As noted in a previous review comment, ensure that all existing tests using this method have been updated to work with the new return value (BudgetMenuModal instance instead of void).

To verify the impact of this change, please run the following script:

#!/bin/bash
# Search for usages of openBudgetMenu in test files
echo "Searching for openBudgetMenu usages in test files:"
rg "openBudgetMenu" --type js --type ts -g "*test*" -g "*spec*"

144-145: LGTM: Improved modularity in openCategoryMenu.

The changes to the openCategoryMenu method improve modularity by returning a CategoryMenuModal instance. This encapsulation of category menu interactions is a good practice.

However, since the method now returns a value, ensure that any code calling this method is updated to handle the returned CategoryMenuModal instance correctly.

To verify the impact of this change, please run the following script:

packages/desktop-client/e2e/budget.mobile.test.js (6)

3-3: LGTM: Import statement added correctly

The new import statement for amountToCurrency and currencyToAmount functions from the loot-core/shared/util module is appropriate and necessary for the new functionality introduced in this file.


33-59: LGTM: Issues resolved in setBudgetAverage function

The setBudgetAverage function has been implemented correctly, addressing the previously identified issues:

  1. The infinite recursion problem has been resolved by using the setBudgetAverageFn parameter instead of calling the function recursively.
  2. The spent amounts are now correctly retrieved after navigating to each month, ensuring accurate data collection.

The function effectively calculates the average spent amount over the specified number of months and returns to the current month after the calculation.


260-272: LGTM: Test case updated to use new modal interaction

The "updates the budgeted amount" test case has been successfully updated to use the new BudgetMenuModal class. The changes align with the new modal-based interaction model and correctly verify the expected behavior of setting and displaying the budgeted amount.


274-292: LGTM: New test case for copying last month's budget

The new test case "copies last month's budget" effectively verifies the functionality of copying the budget from the previous month. It correctly uses the copyLastMonthBudget utility function and ensures that the budgeted amount matches the previous month's value after the operation.


294-320: LGTM: Efficient implementation of average budget tests

The new test cases for setting the budget to 3, 6, and 12-month averages are well-implemented. The use of a loop to generate tests for different month ranges is an efficient approach that reduces code duplication. Each test case correctly uses the setBudgetAverage function along with the corresponding utility function to verify the expected behavior.


322-347: LGTM: Test case updated to use new modal interactions

The "applies budget template" test case has been successfully updated to use the new CategoryMenuModal and EditNotesModal classes. The changes align with the new modal-based interaction model and correctly verify the expected behavior of applying a budget template. The test effectively covers the process of setting up a template and verifying its application.

packages/desktop-client/e2e/page-models/mobile-navigation.js (1)

104-106: Verify the inclusion of 'default' state in navbar state check

In goToSettingsPage(), the navbar state is checked for both 'default' and 'hidden' before calling dragNavbarUp(). In other navigation methods, only 'hidden' is checked. Please verify if including 'default' is necessary for this method.

packages/desktop-client/src/components/mobile/MobileNavTabs.tsx (4)

1-6: Imports are correctly structured

The imported modules and hooks, including the type ComponentType, are properly organized.


217-217: Ensure navbarState is defined before usage

When adding data-navbar-state={navbarState}, make sure that navbarState is not undefined. Since navbarState may initially be undefined, this could lead to an attribute with an undefined value, which might cause unexpected behavior.

Ensure that navbarState is initialized properly. Refer to the previous comment about initializing navbarState.


181-183: Verify the logic in the drag handler

In the useDrag hook, the logic determining whether to call openDefault or openFull after a drag action depends on the vertical offset and velocity. Verify that the thresholds and conditions correctly reflect the desired user experience, ensuring smooth and intuitive interactions.

Consider testing different drag scenarios to confirm that the navbar opens and closes as expected.


270-270: userSelect: 'none' is correctly applied

The addition of userSelect: 'none' prevents text selection on the navigation tabs, enhancing the user experience during interactions.

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

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

Actionable comments posted: 1

🧹 Outside diff range and nitpick comments (4)
packages/desktop-client/e2e/budget.mobile.test.js (4)

33-59: LGTM: setBudgetAverage function implementation

The setBudgetAverage function correctly calculates the average spending over a specified number of months. The issues mentioned in past review comments have been addressed:

  1. The potential infinite recursion has been resolved by correctly calling setBudgetAverageFn instead of recursively calling setBudgetAverage.
  2. The retrieval of spent amounts is now correctly placed inside the loop, ensuring accurate data for each month.

To further improve readability, consider extracting the month navigation logic into separate helper functions:

const navigateToPreviousMonths = async (budgetPage, numberOfMonths) => {
  for (let i = 0; i < numberOfMonths; i++) {
    await budgetPage.goToPreviousMonth();
  }
};

const navigateToCurrentMonth = async (budgetPage, numberOfMonths) => {
  for (let i = 0; i < numberOfMonths; i++) {
    await budgetPage.goToNextMonth();
  }
};

Then use these helper functions in setBudgetAverage:

async function setBudgetAverage(
  budgetPage,
  categoryName,
  numberOfMonths,
  setBudgetAverageFn,
) {
  await navigateToPreviousMonths(budgetPage, numberOfMonths);
  
  // ... (calculate average spent)

  await navigateToCurrentMonth(budgetPage, numberOfMonths);

  await setBudgetAverageFn(budgetPage, categoryName, numberOfMonths);

  return averageSpent;
}

This change would make the main function more concise and easier to understand.


266-284: LGTM: New test for copying last month's budget

This new test effectively verifies the functionality of copying the previous month's budget using the copyLastMonthBudget utility function. It correctly navigates between months and checks that the budget amount is copied accurately.

To improve the test's robustness, consider adding an initial step to set a specific budget amount in the previous month before copying. This would ensure that the test is not dependent on pre-existing data:

// Set a specific budget amount in the previous month
await budgetPage.goToPreviousMonth();
const budgetMenuModal = await budgetPage.openBudgetMenu(categoryName);
await budgetMenuModal.setBudgetAmount('15000'); // Set to 150.00
await budgetMenuModal.close();

// Rest of the test remains the same

This addition would make the test more deterministic and less likely to fail due to changes in test data.


286-312: LGTM: New tests for setting budget to month averages

These new tests effectively verify the functionality of setting the budget to 3, 6, and 12 month averages using the setBudgetAverage utility function. The use of a loop to generate tests for different month ranges is an efficient approach.

To improve the tests' clarity, consider adding a descriptive comment before the loop and using a more descriptive name for the test cases:

// Test budget averaging for 3, 6, and 12 month periods
[
  [3, setTo3MonthAverage, 'quarterly'],
  [6, setTo6MonthAverage, 'semi-annual'],
  [12, setToYearlyAverage, 'annual'],
].forEach(([numberOfMonths, setBudgetAverageFn, periodName]) => {
  test(`set budget to ${periodName} average (${numberOfMonths} months)`, async () => {
    // ... (rest of the test implementation)
  });
});

This change would make the purpose of each test case more immediately clear to readers.


314-339: LGTM: New test for applying budget template

This new test effectively verifies the functionality of applying a budget template. It correctly sets up the environment by enabling the experimental feature, creates a template, and verifies its application.

To improve the test's clarity and maintainability, consider extracting the template setup into a separate helper function:

const setupBudgetTemplate = async (budgetPage, categoryName, amount) => {
  const categoryMenuModal = await budgetPage.openCategoryMenu(categoryName);
  const editNotesModal = await categoryMenuModal.editNotes();
  await editNotesModal.updateNotes(`#template ${amount}`);
  await editNotesModal.close();
};

test(`applies budget template`, async () => {
  const settingsPage = await navigation.goToSettingsPage();
  await settingsPage.enableExperimentalFeature('Goal templates');

  const budgetPage = await navigation.goToBudgetPage();
  await budgetPage.waitForBudgetTable();

  const categoryName = await budgetPage.getCategoryNameForRow(1);
  const amountToTemplate = 123;

  await setupBudgetTemplate(budgetPage, categoryName, amountToTemplate);

  // ... (rest of the test implementation)
});

This refactoring would make the test more readable and easier to maintain, especially if you need to set up templates in other tests in the future.

📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Files that changed from the base of the PR and between bc45c35 and 4ad5208.

📒 Files selected for processing (1)
  • packages/desktop-client/e2e/budget.mobile.test.js (4 hunks)
🧰 Additional context used
🔇 Additional comments (3)
packages/desktop-client/e2e/budget.mobile.test.js (3)

3-3: LGTM: New import for currency conversion utilities

The addition of amountToCurrency and currencyToAmount imports is appropriate for the new currency conversion operations in the file.


252-263: LGTM: Updated test for budget amount changes

The test has been successfully updated to use the new budgetMenuModal approach for setting the budget amount. This change aligns with the overall shift towards using modals for budget operations, maintaining consistency across the codebase.


Line range hint 1-424: Overall assessment: Good additions with room for improvement

The changes in this file significantly enhance the testing coverage for budget-related features in the mobile application. The new utility functions and tests are well-implemented and cover important use cases. However, there are opportunities for improvement:

  1. Refactor the budget utility functions to reduce code duplication.
  2. Improve the readability of the setBudgetAverage function by extracting helper methods.
  3. Enhance the robustness and clarity of the new tests by adding setup steps and using more descriptive names.

These suggested improvements will make the code more maintainable and the tests more reliable. Overall, good job on expanding the test suite!

@joel-jeremy joel-jeremy changed the title [WIP] Mobile budget menu modal VRT Mobile budget menu modal VRT Oct 7, 2024
Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

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

Actionable comments posted: 1

🧹 Outside diff range and nitpick comments (3)
packages/desktop-client/e2e/page-models/mobile-balance-menu-modal.js (3)

2-20: LGTM: Well-structured constructor with comprehensive UI element locators.

The constructor is well-designed, initializing necessary instance variables and UI element locators. The use of getByRole and getByTestId suggests good accessibility practices, which is crucial for robust end-to-end tests.

Consider adding JSDoc comments to describe the page and locator parameters for better code documentation. For example:

/**
 * @param {Page} page - The Playwright Page object
 * @param {Locator} locator - The base locator for this modal
 */
constructor(page, locator) {
  // ... existing code ...
}

6-19: LGTM: Comprehensive and well-defined UI element properties.

The class properties cover all necessary UI elements for interacting with the modal. The consistent use of semantic roles and test IDs follows best practices for robust and maintainable end-to-end tests.

Consider grouping related buttons into an object for better organization and easier maintenance. For example:

this.buttons = {
  transferToAnotherCategory: locator.getByRole('button', { name: 'Transfer to another category' }),
  coverOverspending: locator.getByRole('button', { name: 'Cover overspending' }),
  rolloverOverspending: locator.getByRole('button', { name: 'Rollover overspending' }),
  removeOverspendingRollover: locator.getByRole('button', { name: 'Remove overspending rollover' })
};

This structure could make it easier to add or modify buttons in the future and provide a cleaner API for using these buttons in tests.


22-24: LGTM: Simple and focused close method.

The close method is well-implemented, using async/await correctly for the asynchronous click operation. Its single responsibility is clear and appropriate.

To improve robustness, consider adding error handling and a fallback mechanism. For example:

async close() {
  try {
    await this.heading.getByRole('button', { name: 'Close' }).click();
  } catch (error) {
    console.warn('Failed to find Close button in heading, trying modal-wide search');
    await this.locator.getByRole('button', { name: 'Close' }).click();
  }
}

This approach would first try to find the Close button within the heading, and if that fails, it would search for it within the entire modal. This could help prevent test failures if the button's location changes in the future.

📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Files that changed from the base of the PR and between 4ad5208 and 17c887a.

📒 Files selected for processing (3)
  • packages/desktop-client/e2e/budget.mobile.test.js (5 hunks)
  • packages/desktop-client/e2e/page-models/mobile-balance-menu-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-budget-page.js (4 hunks)
🚧 Files skipped from review as they are similar to previous changes (1)
  • packages/desktop-client/e2e/budget.mobile.test.js
🧰 Additional context used
🔇 Additional comments (5)
packages/desktop-client/e2e/page-models/mobile-balance-menu-modal.js (1)

1-1: LGTM: Class structure and export.

The BalanceMenuModal class is well-named and appropriately exported as a named export. This follows good practices for modular code organization in JavaScript/TypeScript projects.

packages/desktop-client/e2e/page-models/mobile-budget-page.js (4)

2-4: LGTM: New modal imports added.

The new imports for BalanceMenuModal, BudgetMenuModal, and CategoryMenuModal are correctly placed and follow the existing naming convention. These additions suggest a modular approach to handling different budget-related actions.


145-146: LGTM: New openCategoryMenu method added.

The new openCategoryMenu method is well-implemented. It follows the existing pattern of other methods in the class, uses the private #getButtonForCategory method to locate the button, and returns a new CategoryMenuModal instance. This addition enhances the functionality of the MobileBudgetPage class.


202-202: LGTM: openBalanceMenu updated to use BalanceMenuModal.

The modification to return a new BalanceMenuModal instance is consistent with the modal-based approach seen in other methods. The existing error handling for when the balance button is not visible has been retained, which is good for maintaining robustness.


Line range hint 1-324: Summary: Consistent implementation of modal-based approach for budget actions.

The changes in this file consistently implement a modal-based approach for various budget-related actions, improving modularity and consistency. New modal classes (BalanceMenuModal, BudgetMenuModal, and CategoryMenuModal) have been introduced and integrated into the existing methods.

Key points:

  1. New imports added for modal classes.
  2. openCategoryMenu method added, following the existing pattern.
  3. openBudgetMenu and openBalanceMenu methods updated to return modal instances.
  4. setBudget method removed (as per AI summary, not visible in the diff).

These changes align well with the PR objectives of enhancing the mobile testing framework for budget management. However, the removal of the setBudget method may have implications for other parts of the codebase or existing tests.

To ensure a smooth transition, please:

  1. Verify that all tests using MobileBudgetPage have been updated to work with the new modal-based approach.
  2. Check for any remaining references to the removed setBudget method in other parts of the codebase.
  3. Update the documentation for MobileBudgetPage to reflect these changes, especially noting the new return types for methods like openBudgetMenu and openBalanceMenu.

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

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

Actionable comments posted: 2

🧹 Outside diff range and nitpick comments (8)
packages/desktop-client/e2e/page-models/mobile-envelope-budget-summary-modal.js (2)

1-7: LGTM! Consider adding JSDoc comments.

The class declaration and constructor are well-structured and follow good practices. The use of role-based selectors (getByRole('heading')) is excellent for maintaining test stability.

Consider adding JSDoc comments to describe the purpose of the class and the expected types of the constructor parameters. This would improve code documentation and maintainability.


1-12: Great implementation! Consider expanding functionality.

The EnvelopeBudgetSummaryModal class is well-structured, focused, and follows good testing practices. It provides a solid foundation for interacting with the envelope budget summary modal in your e2e tests.

As the testing suite evolves, consider expanding this class to include additional methods for interacting with other elements within the modal, such as retrieving budget information or interacting with form elements if present. This would further enhance the reusability and maintainability of your test code.

packages/desktop-client/e2e/page-models/mobile-tracking-budget-summary-modal.js (2)

2-7: LGTM: Constructor implementation with a suggestion

The constructor is well-implemented, properly initializing instance variables with the provided page and locator parameters. The heading property initialization is a good optimization for frequently accessed elements.

Consider adding JSDoc comments to describe the parameters and their types for better code documentation:

/**
 * @param {import('@playwright/test').Page} page - The Playwright Page object
 * @param {import('@playwright/test').Locator} locator - The Locator for this modal
 */
constructor(page, locator) {
  // ... existing code ...
}

9-11: LGTM: Well-implemented close method with a suggestion

The close method is well-implemented, using asynchronous operations and chained locators to find and click the close button. The method name clearly describes its action.

Consider adding error handling to make the method more robust:

async close() {
  try {
    await this.heading.getByRole('button', { name: 'Close' }).click();
  } catch (error) {
    console.error('Failed to close the TrackingBudgetSummaryModal:', error);
    throw error; // Re-throw the error for the test to catch
  }
}

This will provide more informative error messages if the close action fails during testing.

packages/desktop-client/e2e/page-models/mobile-budget-page.js (2)

276-283: LGTM: openEnvelopeBudgetSummary updated to use modal approach.

The renaming and modification to return an EnvelopeBudgetSummaryModal instance align well with the overall refactoring. This change is consistent with the modal-based approach implemented in other methods.

Consider using object property shorthand for improved readability:

 return new EnvelopeBudgetSummaryModal(
-  this.page,
+  this.page,
   this.page.getByRole('dialog'),
 );

308-315: LGTM: openTrackingBudgetSummary updated to use modal approach.

The renaming and modification to return a TrackingBudgetSummaryModal instance are consistent with the overall refactoring and the modal-based approach implemented in other methods.

Consider using object property shorthand for improved readability:

 return new TrackingBudgetSummaryModal(
-  this.page,
+  this.page,
   this.page.getByRole('dialog'),
 );
packages/desktop-client/e2e/budget.mobile.test.js (2)

286-312: Efficient test cases for setting budget to average

The approach of using a loop to test multiple average periods (3, 6, and 12 months) is efficient and reduces code duplication. These tests effectively cover the functionality of setting budgets to different average periods.

To improve clarity, consider adding a comment explaining the structure of the test data:

// Test data structure: [number of months, corresponding setBudgetAverage function]
[
  [3, setTo3MonthAverage],
  [6, setTo6MonthAverage],
  [12, setToYearlyAverage],
].forEach(([numberOfMonths, setBudgetAverageFn]) => {
  // ... (rest of the code remains the same)
});

This addition would make the purpose of each array element more immediately clear to other developers.


314-340: Comprehensive test for applying budget template

This test case effectively covers the budget template feature:

  1. It enables the experimental 'Goal templates' feature.
  2. It creates a template by updating category notes.
  3. It applies the template and verifies the budgeted amount.

To improve robustness, consider adding a cleanup step to disable the experimental feature after the test:

test.afterEach(async () => {
  const settingsPage = await navigation.goToSettingsPage();
  await settingsPage.disableExperimentalFeature('Goal templates');
});

This ensures that the experimental feature doesn't affect other tests.

📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Files that changed from the base of the PR and between 17c887a and 4b7565f.

📒 Files selected for processing (4)
  • packages/desktop-client/e2e/budget.mobile.test.js (7 hunks)
  • packages/desktop-client/e2e/page-models/mobile-budget-page.js (6 hunks)
  • packages/desktop-client/e2e/page-models/mobile-envelope-budget-summary-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-tracking-budget-summary-modal.js (1 hunks)
🧰 Additional context used
🔇 Additional comments (10)
packages/desktop-client/e2e/page-models/mobile-envelope-budget-summary-modal.js (1)

9-11: LGTM! Well-implemented close method.

The close method is concise, uses async/await correctly, and employs accessibility-friendly selectors. The implementation is robust and should reliably close the modal.

packages/desktop-client/e2e/page-models/mobile-tracking-budget-summary-modal.js (1)

1-12: LGTM: Well-structured class for e2e testing

The TrackingBudgetSummaryModal class is well-structured and follows the Page Object Model pattern, which is excellent for maintainability and readability in e2e tests. The class name clearly indicates its purpose, and the exported structure allows for easy reuse in other test files.

packages/desktop-client/e2e/page-models/mobile-budget-page.js (4)

2-6: LGTM: New modal imports align with updated class methods.

The addition of these import statements is consistent with the changes made in the class methods, indicating a shift towards modal-based interactions for budget-related actions.


147-148: LGTM: New openCategoryMenu method aligns with modal-based approach.

The new method correctly clicks on the category button before returning a CategoryMenuModal instance, which is consistent with the shift towards modal-based interactions for budget management.


204-204: LGTM: openBalanceMenu updated to use BalanceMenuModal.

The modification to return a new BalanceMenuModal instance is consistent with the modal-based approach implemented in other methods. This change aligns well with the overall refactoring of the class.


Line range hint 1-316: Overall: Consistent implementation of modal-based approach for budget interactions.

The changes in this file demonstrate a systematic shift towards using modal-based interactions for budget-related actions. This refactoring improves the overall structure and consistency of the MobileBudgetPage class. The removal of the setBudget method and the introduction of various modal classes suggest a significant change in how budget operations are handled.

While these changes enhance the modularity of the code, they may require updates in other parts of the codebase that previously relied on the removed setBudget method. Ensure that all affected components are updated to use the new modal-based approach consistently.

To ensure a smooth transition, please run comprehensive tests covering all budget-related functionalities, paying special attention to areas that previously used the setBudget method.

packages/desktop-client/e2e/budget.mobile.test.js (4)

33-59: Improvements implemented in setBudgetAverage function

Great job addressing the issues mentioned in the past review comments:

  1. The infinite recursion issue has been resolved by renaming the parameter to setBudgetAverageFn.
  2. The spentButton is now correctly retrieved inside the loop, ensuring accurate data for each month.

These changes improve the function's reliability and correctness.


266-284: Well-structured test case for copying last month's budget

This test case effectively verifies the copyLastMonthBudget functionality:

  1. It navigates to the previous month to get the last month's budget.
  2. It returns to the current month and applies the copyLastMonthBudget function.
  3. It verifies that the current month's budget matches the previous month's budget.

The test is comprehensive and well-structured.


Line range hint 371-397: Effective tests for budget summary modals

These test cases comprehensively cover the budget summary functionality:

  1. They test both Envelope and Tracking budget types separately.
  2. They verify that clicking the appropriate elements opens the correct budget summary modal.
  3. They check that the modal headings are correct.

The conditional tests based on budget type ensure that the appropriate functionality is tested for each type. This approach is well-structured and thorough.


Line range hint 1-397: Overall assessment of budget.mobile.test.js changes

The changes in this file significantly enhance the budget management functionality and its testing:

  1. New utility functions have been added to support various budget operations.
  2. Comprehensive test cases cover new features like copying last month's budget, setting budget averages, and applying budget templates.
  3. Previous issues identified in past reviews have been addressed, improving code reliability.
  4. The tests are well-structured and cover both Envelope and Tracking budget types.

While there are some opportunities for minor improvements (as noted in previous comments), the overall quality of the changes is high. The new functionality is well-tested and should provide a solid foundation for mobile budget management features.

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

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

Actionable comments posted: 2

🧹 Outside diff range and nitpick comments (11)
packages/desktop-client/e2e/page-models/mobile-reports-page.js (2)

2-6: Constructor looks good, consider adding error handling.

The constructor correctly initializes the necessary properties and uses a test ID selector for the overview element, which is a good practice for test stability.

Consider adding error handling in case the overview element is not found. Here's a suggested improvement:

 constructor(page) {
   this.page = page;

-  this.overview = page.getByTestId('reports-overview');
+  this.overview = page.getByTestId('reports-overview');
+  if (!this.overview) {
+    throw new Error("Failed to locate the reports overview element");
+  }
 }

This change will make the test fail early if the crucial overview element is missing, making debugging easier.


8-10: waitFor method is good, but could use documentation and error handling.

The waitFor method provides a useful way to synchronize tests with the page state. The use of async/await is correct.

Consider the following improvements:

  1. Add JSDoc comments to explain the purpose of the method and the options parameter.
  2. Implement error handling.

Here's a suggested improvement:

+/**
+ * Waits for the reports overview element to be ready.
+ * @param {Object} options - Options to pass to the underlying Playwright `waitFor` method.
+ * @throws {Error} If waiting for the overview element times out.
+ */
 async waitFor(options) {
-  await this.overview.waitFor(options);
+  try {
+    await this.overview.waitFor(options);
+  } catch (error) {
+    throw new Error(`Timed out waiting for the reports overview: ${error.message}`);
+  }
 }

These changes will improve the method's documentation and make it more robust by providing meaningful error messages.

packages/desktop-client/e2e/page-models/mobile-accounts-page.js (2)

11-13: LGTM! Good addition for handling asynchronous UI updates.

The waitFor method is a valuable addition that will help improve the reliability of tests by ensuring the account list is ready before interacting with it.

Consider adding a default timeout to the options parameter to make the method more robust:

- async waitFor(options) {
+ async waitFor(options = { timeout: 5000 }) {
    await this.accountList.waitFor(options);
  }

31-31: LGTM! Consistent update to use the new element selection strategy.

The change to use this.accountListItems is consistent with the updates made elsewhere in the class.

Consider adding error handling to improve robustness:

  async openNthAccount(idx) {
-   await this.accountListItems.nth(idx).click();
+   const accountItem = this.accountListItems.nth(idx);
+   if (await accountItem.count() === 0) {
+     throw new Error(`Account at index ${idx} not found`);
+   }
+   await accountItem.click();

    return new MobileAccountPage(this.page);
  }
packages/desktop-client/e2e/page-models/mobile-transaction-entry-page.js (2)

15-16: Good addition of waitFor method to improve test reliability.

The new waitFor method enhances the reliability of tests by ensuring the transaction form is ready before interactions. This is a good practice in E2E testing.

Consider adding a default timeout to the waitFor method for consistency across tests:

async waitFor(options = { timeout: 5000 }) {
  await this.transactionForm.waitFor(options);
}

26-26: Consistent update to createTransaction method.

The change to use addTransactionButton aligns well with the restructured element access introduced in the constructor.

Consider adding error handling to improve robustness:

async createTransaction() {
  try {
    await this.addTransactionButton.click();
    return new MobileAccountPage(this.page);
  } catch (error) {
    console.error('Failed to create transaction:', error);
    throw error;
  }
}
packages/desktop-client/src/components/settings/index.tsx (3)

150-150: LGTM! Consider extending test coverage.

The addition of the data-testid="settings" attribute to the View component is a good practice. It enhances the testability of the Settings component by providing a reliable selector for automated tests.

To further improve test coverage, consider adding more specific data-testid attributes to child components within the Settings page. This would allow for more granular testing of individual settings sections.


Line range hint 76-83: Consider addressing commented-out code.

There's a section of commented-out code related to displaying additional IDs. It's generally a good practice to remove commented-out code or add a clear TODO comment explaining why it's kept.

If these IDs are no longer needed, consider removing the commented code. If they might be used in the future, add a clear TODO comment explaining the plan for this code.


Line range hint 93-93: Consider using React.memo for performance optimization.

The Settings component renders multiple child components and uses several hooks. To potentially improve performance, especially if this component re-renders frequently, consider wrapping it with React.memo.

Here's how you could apply React.memo:

export const Settings = React.memo(function Settings() {
  // ... existing component code ...
});

This optimization would prevent unnecessary re-renders if the component's props (in this case, there are none) haven't changed.

packages/desktop-client/src/components/mobile/accounts/Accounts.jsx (1)

187-187: Approve the addition of data-testid and suggest test enhancements.

The addition of data-testid="account-list" to the View component enhances the testability of the account list container. This change allows for more precise targeting in tests, potentially improving test reliability and readability.

Consider updating or adding tests that utilize this new data-testid to improve test coverage and specificity. For example, you could add tests to verify the structure and content of the account list container.

packages/desktop-client/src/components/mobile/transactions/TransactionEdit.jsx (1)

Line range hint 1062-1079: Good addition: New TransactionEdit component with hooks.

The new TransactionEdit component is a good addition that separates concerns and uses React hooks for data fetching. This improves code organization and reusability.

Consider adding a prop-types definition for this component to improve type checking and documentation.

You could add prop-types like this:

import PropTypes from 'prop-types';

// ... component code ...

TransactionEdit.propTypes = {
  // Add any props that might be passed to this component
};
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Files that changed from the base of the PR and between 4b7565f and c1c69d2.

📒 Files selected for processing (11)
  • packages/desktop-client/e2e/budget.mobile.test.js (4 hunks)
  • packages/desktop-client/e2e/page-models/mobile-accounts-page.js (2 hunks)
  • packages/desktop-client/e2e/page-models/mobile-budget-page.js (7 hunks)
  • packages/desktop-client/e2e/page-models/mobile-navigation.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-reports-page.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-transaction-entry-page.js (2 hunks)
  • packages/desktop-client/e2e/page-models/settings-page.js (1 hunks)
  • packages/desktop-client/src/components/mobile/accounts/Accounts.jsx (2 hunks)
  • packages/desktop-client/src/components/mobile/transactions/TransactionEdit.jsx (2 hunks)
  • packages/desktop-client/src/components/reports/Overview.tsx (1 hunks)
  • packages/desktop-client/src/components/settings/index.tsx (1 hunks)
✅ Files skipped from review due to trivial changes (1)
  • packages/desktop-client/src/components/reports/Overview.tsx
🚧 Files skipped from review as they are similar to previous changes (2)
  • packages/desktop-client/e2e/budget.mobile.test.js
  • packages/desktop-client/e2e/page-models/settings-page.js
🧰 Additional context used
🔇 Additional comments (20)
packages/desktop-client/e2e/page-models/mobile-reports-page.js (1)

1-11: LGTM: Well-structured Page Object Model class.

The MobileReportsPage class follows the Page Object Model pattern, which is a best practice for organizing end-to-end tests. The class name accurately describes its purpose, and the structure with a constructor and a waitFor method provides a good foundation for interacting with the mobile reports page in tests.

packages/desktop-client/e2e/page-models/mobile-accounts-page.js (3)

7-8: LGTM! Improved element selection strategy.

The new accountList and accountListItems properties provide a more structured and flexible approach to selecting account elements. Using test IDs for element selection is a good practice in E2E testing, as it makes the tests more resilient to UI changes.


19-19: LGTM! Consistent use of the new element selection strategy.

The update to use this.accountListItems aligns well with the changes made in the constructor, maintaining consistency throughout the class.


Line range hint 1-35: Overall, great improvements to the MobileAccountsPage class!

The changes made to this class enhance its flexibility and maintainability:

  1. The new element selection strategy using test IDs improves test resilience.
  2. The addition of the waitFor method helps handle asynchronous UI updates.
  3. Consistent updates across methods maintain the class's coherence.

These improvements will likely lead to more robust and reliable E2E tests for the mobile accounts functionality.

packages/desktop-client/e2e/page-models/mobile-transaction-entry-page.js (2)

7-12: Improved structure and testability of the MobileTransactionEntryPage class.

The new properties (transactionForm, amountField, footer, and addTransactionButton) enhance the organization of the class and improve its testability. The use of getByTestId and getByRole methods aligns with best practices for robust test selectors.


Line range hint 1-29: Overall improvements to MobileTransactionEntryPage class structure and testability.

The changes in this file significantly enhance the MobileTransactionEntryPage class by:

  1. Restructuring element access for better organization.
  2. Improving testability with the use of test-id and role-based selectors.
  3. Adding a waitFor method to increase test reliability.
  4. Updating the createTransaction method for consistency.

These modifications align well with best practices in E2E testing and should lead to more robust and maintainable tests.

packages/desktop-client/src/components/mobile/accounts/Accounts.jsx (1)

74-74: Approve the data-testid change and verify existing tests.

The change from data-testid="account" to data-testid="account-list-item" improves the specificity and clarity of the test identifier. This modification better represents the component's role as an item within the account list.

Please ensure that all existing tests referencing this component are updated to use the new data-testid value. Run the following script to check for any occurrences of the old data-testid:

✅ Verification successful

Verification successful: No tests reference the old data-testid="account" value.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Search for occurrences of the old data-testid in test files

# Test: Look for "data-testid="account"" in test files
rg --type test 'data-testid="account"'

Length of output: 69


Script:

#!/bin/bash
# Description: Search for occurrences of the old data-testid in test files with common test file extensions

# Search in files ending with .test.js, .spec.js, .test.jsx, or .spec.jsx
rg 'data-testid="account"' --glob '*.test.js' --glob '*.spec.js' --glob '*.test.jsx' --glob '*.spec.jsx'

Length of output: 106

packages/desktop-client/src/components/mobile/transactions/TransactionEdit.jsx (3)

191-191: Minor enhancement: Added data-testid attribute.

A data-testid attribute has been added to the footer view. This is a good practice for improving testability of the component.


745-748: Improved testability: Added data-testid to transaction form.

The addition of a data-testid attribute to the main transaction form view enhances the component's testability. This is consistent with best practices for creating easily identifiable elements in automated tests.


Line range hint 1-1079: Overall: Improved testability and code organization.

The changes in this file focus on two main areas:

  1. Adding data-testid attributes to key elements, enhancing the testability of the components.
  2. Introducing a new TransactionEdit component that uses React hooks for data fetching and wraps the existing TransactionEditUnconnected component.

These changes align with best practices in React development, improving both the testability and organization of the code. The additions are consistent and do not introduce any apparent issues or significant logic changes.

packages/desktop-client/e2e/page-models/mobile-navigation.js (3)

14-14: Consider declaring ROW_HEIGHT as a static class property

ROW_HEIGHT is currently defined as an instance property. If ROW_HEIGHT is intended to be a constant value shared across all instances of MobileNavigation, consider making it a static class property using the static keyword.


44-45: Use this.navbar.getByRole for link selection in goToBudgetPage

In goToBudgetPage(), you're using this.page.getByRole to select the 'Budget' link. To maintain consistency with other navigation methods, consider using this.navbar.getByRole instead.


69-70: Use this.navbar.getByRole for link selection in goToAccountsPage

In goToAccountsPage(), you're using this.page.getByRole to select the 'Accounts' link. It would be more consistent to use this.navbar.getByRole, as done in other methods.

packages/desktop-client/e2e/page-models/mobile-budget-page.js (7)

2-6: LGTM: New modal classes imported correctly

The imports for the new modal classes are appropriately added and correspond with their usage within the file.


86-87: Verify updates to method calls for 'waitFor'

The method waitForBudgetTable() has been renamed to waitFor(options). Please ensure that all references to waitForBudgetTable() have been updated to use the new method signature to prevent any broken references.

Run the following script to find any lingering references to the old method:

#!/bin/bash
# Description: Search for references to 'waitForBudgetTable'
# Expected Result: No occurrences of 'waitForBudgetTable' should be found

rg "waitForBudgetTable" --type js --type ts

204-204: Ensure callers of 'openBalanceMenu' handle the new return value

The method openBalanceMenu now returns a new instance of BalanceMenuModal. Please verify that all callers of this method are updated to handle the returned modal appropriately.

Run the following script to find usages of openBalanceMenu and check for handling of the return value:

#!/bin/bash
# Description: Find usages of 'openBalanceMenu' and review usage
# Expected Result: Callers should handle the returned 'BalanceMenuModal' instance

rg "openBalanceMenu\(" --type js --type ts -A 2

276-283: Ensure callers of 'openEnvelopeBudgetSummary' handle the new return value

The method openEnvelopeBudgetSummary now returns a new instance of EnvelopeBudgetSummaryModal. Please verify that all callers of this method are updated to handle the returned modal appropriately.

Run the following script to find usages of openEnvelopeBudgetSummary and check for handling of the return value:

#!/bin/bash
# Description: Find usages of 'openEnvelopeBudgetSummary' and review usage
# Expected Result: Callers should handle the returned 'EnvelopeBudgetSummaryModal' instance

rg "openEnvelopeBudgetSummary\(" --type js --type ts -A 2

Additionally, ensure all references to the old method name openEnvelopeBudgetSummaryMenu have been updated:

#!/bin/bash
# Description: Search for references to 'openEnvelopeBudgetSummaryMenu'
# Expected Result: No occurrences of 'openEnvelopeBudgetSummaryMenu' should be found

rg "openEnvelopeBudgetSummaryMenu\(" --type js --type ts

147-148: Ensure callers of 'openCategoryMenu' handle the new return value

The method openCategoryMenu now returns a new instance of CategoryMenuModal. Please verify that all callers of this method are updated to handle the returned modal appropriately.

Run the following script to find usages of openCategoryMenu and check for handling of the return value:

#!/bin/bash
# Description: Find usages of 'openCategoryMenu' and review usage
# Expected Result: Callers should handle the returned 'CategoryMenuModal' instance

rg "openCategoryMenu\(" --type js --type ts -A 2

308-315: Ensure callers of 'openTrackingBudgetSummary' handle the new return value

The method openTrackingBudgetSummary now returns a new instance of TrackingBudgetSummaryModal. Please verify that all callers of this method are updated to handle the returned modal appropriately.

Run the following script to find usages of openTrackingBudgetSummary and check for handling of the return value:

#!/bin/bash
# Description: Find usages of 'openTrackingBudgetSummary' and review usage
# Expected Result: Callers should handle the returned 'TrackingBudgetSummaryModal' instance

rg "openTrackingBudgetSummary\(" --type js --type ts -A 2

Also, ensure all references to the old method name openTrackingBudgetSummaryMenu have been updated:

#!/bin/bash
# Description: Search for references to 'openTrackingBudgetSummaryMenu'
# Expected Result: No occurrences of 'openTrackingBudgetSummaryMenu' should be found

rg "openTrackingBudgetSummaryMenu\(" --type js --type ts

187-187: Ensure callers of 'openBudgetMenu' handle the new return value

The method openBudgetMenu now returns a new instance of BudgetMenuModal. Please verify that all callers of this method are updated to handle the returned modal appropriately.

Run the following script to find usages of openBudgetMenu and check for handling of the return value:

✅ Verification successful

No callers of openBudgetMenu found. No action required.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Find usages of 'openBudgetMenu' and review usage
# Expected Result: Callers should handle the returned 'BudgetMenuModal' instance

rg "openBudgetMenu\(" --type js --type ts -A 2

Length of output: 682


Script:

#!/bin/bash
# Description: Find all calls to 'openBudgetMenu' and check if the return value is handled
# Expected Result: All calls to 'openBudgetMenu' should handle the returned 'BudgetMenuModal' instance

ast-grep --lang javascript --pattern "call_expression_expression: call_expression: identifier: \"openBudgetMenu\"" --actions 'print @file ":" @line ":" @column " - " @text'

Length of output: 375


Script:

#!/bin/bash
# Description: Find all calls to 'openBudgetMenu' and check if the return value is handled
# Expected Result: All calls to 'openBudgetMenu' should handle the returned 'BudgetMenuModal' instance

ast-grep --lang javascript --pattern 'call_expression: identifier: "openBudgetMenu"'

Length of output: 86


Script:

#!/bin/bash
# Description: Find all calls to 'openBudgetMenu' and check if the return value is handled
# Expected Result: All calls to 'openBudgetMenu' should handle the returned 'BudgetMenuModal' instance

rg -n -A2 "(const|let|let|await)\s+\w+\s*=\s*await\s+openBudgetMenu\(" --type js --type ts

Length of output: 92

@joel-jeremy joel-jeremy changed the title Mobile budget menu modal VRT [e2e] Mobile budget menu modal VRT Oct 8, 2024
Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

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

Actionable comments posted: 1

🧹 Outside diff range and nitpick comments (3)
packages/desktop-client/e2e/budget.mobile.test.js (3)

273-298: LGTM: Test cases for setting budget to average are well-implemented.

The parameterized approach for testing 3, 6, and 12-month averages is efficient and reduces code duplication.

Consider adding a brief comment explaining the structure of these tests for better readability:

// Test cases for setting budget to 3, 6, and 12-month averages
[
  [3, setTo3MonthAverage],
  [6, setTo6MonthAverage],
  [12, setToYearlyAverage],
].forEach(([numberOfMonths, setBudgetAverageFn]) => {
  // ... (rest of the code)
});

300-328: LGTM: Test case for applying budget template is well-implemented with improved accessibility.

The test case thoroughly verifies the budget template functionality, including setup and application.

Great job on improving accessibility by adding the role="alert" attribute to the notification element. This enhances the user experience for screen reader users.


218-220: LGTM: Existing test cases updated consistently with new modal objects.

The updates to existing test cases improve code consistency and align with the new utility functions and modal implementations.

For better consistency, consider using the same naming convention for all modal variables. For example:

const categoryMenuModal = await budgetPage.openCategoryMenu(categoryName);
const budgetMenuModal = await budgetPage.openBudgetMenu(categoryName);
const balanceMenuModal = await budgetPage.openBalanceMenu(categoryName);
const budgetSummaryModal = await budgetPage.openEnvelopeBudgetSummary(); // or openTrackingBudgetSummary()

Also applies to: 230-232, 240-252, 350-352, 360-365, 374-379

📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Files that changed from the base of the PR and between c1c69d2 and 4f9f6f3.

📒 Files selected for processing (4)
  • packages/desktop-client/e2e/budget.mobile.test.js (4 hunks)
  • packages/desktop-client/e2e/page-models/mobile-navigation.js (1 hunks)
  • packages/desktop-client/e2e/page-models/settings-page.js (1 hunks)
  • packages/desktop-client/src/components/Notifications.tsx (1 hunks)
✅ Files skipped from review due to trivial changes (1)
  • packages/desktop-client/src/components/Notifications.tsx
🚧 Files skipped from review as they are similar to previous changes (1)
  • packages/desktop-client/e2e/page-models/settings-page.js
🧰 Additional context used
🔇 Additional comments (16)
packages/desktop-client/e2e/page-models/mobile-navigation.js (10)

3-3: LGTM: Import statement for MobileReportsPage

The import statement for MobileReportsPage is consistent with the naming convention of other imported classes and aligns with the new goToReportsPage method added to the MobileNavigation class.


10-11: LGTM: New properties for improved element selection

The addition of heading and navbar properties using getByRole is a good practice for accessibility and provides a centralized way to reference these elements throughout the class.


16-18: LGTM: dragNavbarUp method implementation

The dragNavbarUp method provides a clear and concise way to drag the navbar to the heading position. The use of the dragTo API is appropriate for this interaction.


27-29: LGTM: hasNavbarState method implementation

The hasNavbarState method is well-implemented. It efficiently checks the navbar's state against multiple possible values using the spread operator and includes method. This approach provides flexibility and readability.


33-54: LGTM: Improved goToBudgetPage method

The updates to goToBudgetPage method enhance its functionality by:

  1. Handling different navbar states.
  2. Ensuring the navbar is visible before navigation.
  3. Returning the budgetPage instance consistently.

These changes improve the robustness and reliability of the navigation process.


58-79: LGTM: Consistent improvements in goToAccountsPage method

The goToAccountsPage method has been updated consistently with the goToBudgetPage method. It now includes:

  1. Proper handling of navbar states.
  2. Ensuring navbar visibility before navigation.
  3. Returning the accountsPage instance.

These changes maintain consistency across navigation methods and improve the overall reliability of the navigation process.


83-100: LGTM: Enhanced goToTransactionEntryPage method

The goToTransactionEntryPage method has been improved consistently with other navigation methods. Notable enhancements include:

  1. Proper handling of navbar states.
  2. Use of this.navbar for link selection, which is more specific and consistent with the class structure.
  3. Returning the transactionEntryPage instance.

These changes contribute to a more robust and consistent navigation implementation across the class.


103-124: LGTM: Well-implemented goToReportsPage method

The new goToReportsPage method is a welcome addition to the MobileNavigation class. It follows the same pattern as other navigation methods, ensuring:

  1. Proper handling of navbar states.
  2. Consistent navigation process.
  3. Return of the reportsPage instance.

This implementation maintains the consistency and reliability of the navigation system.


127-148: LGTM: Consistent implementation of goToSettingsPage method

The new goToSettingsPage method is a solid addition to the MobileNavigation class. It maintains consistency with other navigation methods by:

  1. Handling navbar states appropriately.
  2. Following the established navigation process.
  3. Returning the settingsPage instance.

This implementation contributes to the overall consistency and reliability of the navigation system.


20-25: Review dragNavbarDown implementation

The dragNavbarDown method uses this.navbar as both the source and target of the drag operation. This might not produce the intended effect. Consider adjusting the implementation to ensure the navbar is dragged down correctly.

To verify the correct implementation, you can run the following script:

Based on the results, you may need to adjust the dragNavbarDown method. Consider using this.page.mouse for a more precise drag operation or adjusting the target element.

packages/desktop-client/e2e/budget.mobile.test.js (6)

9-13: LGTM: Utility function implementation is correct.

The copyLastMonthBudget function correctly implements the budget menu interaction pattern.


15-19: LGTM: Utility function implementation is correct.

The setTo3MonthAverage function correctly implements the budget menu interaction pattern.


21-31: LGTM: Utility functions implementation is correct.

The setTo6MonthAverage and setToYearlyAverage functions correctly implement the budget menu interaction pattern.


254-271: LGTM: Test case for copying last month's budget is well-implemented.

The test case correctly verifies the functionality of the copyLastMonthBudget function by comparing the budgeted amount with the previous month's value.


Line range hint 1-383: Overall, excellent improvements to the budget management test suite.

The changes in this file significantly enhance the budget management functionality and test coverage. New utility functions and test cases are well-implemented, and existing tests have been updated consistently. The code demonstrates good practices in test structure and maintainability.

Key improvements:

  1. Addition of utility functions for various budget operations.
  2. New test cases for copying last month's budget and setting budget averages.
  3. Implementation of budget template functionality with accessibility considerations.
  4. Consistent updates to existing test cases using new modal objects.

Great job on these enhancements!


33-59: LGTM: Budget average calculation implemented correctly.

The setBudgetAverage function correctly calculates the average spent amount and applies it to the budget. It addresses the past review comments about infinite recursion and potential stale elements.

To ensure the function correctly returns to the current month, we can run the following verification:

This script will help confirm that the function properly returns to the current month after calculating the average.

✅ Verification successful

Action Required: Correct the shell script to use the proper file type for ripgrep.

Please run the following script to verify that the function correctly navigates back to the current month:


Action Required: Verify the presence of goToNextMonth function call within the test file.

Please run the following script to check if the goToNextMonth function is being called within budget.mobile.test.js:


LGTM: Budget average calculation implemented correctly.

The setBudgetAverage function correctly calculates the average spent amount and ensures navigation back to the current month after computation. It addresses past review comments regarding infinite recursion and potential stale elements.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Verify that the function navigates back to the current month

# Search for the navigation logic
rg --type javascript 'for \(let i = 0; i < numberOfMonths; i\+\+\) \{.*?await budgetPage\.goToNextMonth\(\);.*?\}' packages/desktop-client/e2e/budget.mobile.test.js

Length of output: 201


Script:

#!/bin/bash
# Verify that the function navigates back to the current month

# Search for the navigation logic
rg --type js 'for \(let i = 0; i < numberOfMonths; i\+\+\) \{.*?await budgetPage\.goToNextMonth\(\);.*?\}' packages/desktop-client/e2e/budget.mobile.test.js

Length of output: 158


Script:

#!/bin/bash
# Verify the presence of goToNextMonth function call

rg 'await budgetPage\.goToNextMonth\(\)' packages/desktop-client/e2e/budget.mobile.test.js

Length of output: 228

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

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

Actionable comments posted: 0

🧹 Outside diff range and nitpick comments (6)
packages/desktop-client/src/components/Page.tsx (3)

Line range hint 67-67: Consider adding aria-level to the heading role

In the MobilePageHeader component, you've correctly used role="heading". To further improve accessibility, consider adding an aria-level attribute to specify the heading level. For example:

<View
  role="heading"
  aria-level={1}
  // ... other props
>
  {title}
</View>

This helps screen reader users understand the document structure better.


Line range hint 106-106: Enhance type safety with explicit prop types

In the Page component, consider using more specific TypeScript types for some props. For example:

type PageProps = {
  header: ReactNode;
  style?: CSSProperties;
  padding?: number;
  children: ReactNode;
  footer?: ReactNode;
  isNarrowWidth?: boolean; // Add this if it's a prop
};

This improves type safety and makes the component's API more clear.


Line range hint 108-150: Consider extracting responsive logic

The Page component contains logic for responsive design based on isNarrowWidth. To improve maintainability and reusability, consider extracting this logic into a custom hook or a higher-order component. This separation of concerns could make the component easier to test and maintain.

For example, you could create a usePageLayout hook:

function usePageLayout(header: ReactNode, isNarrowWidth: boolean) {
  const headerToRender =
    typeof header === 'string'
      ? isNarrowWidth
        ? <MobilePageHeader title={header} />
        : <PageHeader title={header} />
      : header;

  const style = {
    ...(isNarrowWidth ? {} : styles.page),
    flex: 1,
    backgroundColor: isNarrowWidth
      ? theme.mobilePageBackground
      : theme.pageBackground,
  };

  return { headerToRender, style };
}

Then use it in your Page component:

export function Page({ header, style, padding, children, footer }: PageProps) {
  const { isNarrowWidth } = useResponsive();
  const { headerToRender, pageStyle } = usePageLayout(header, isNarrowWidth);
  // ... rest of the component
}

This refactoring would make the Page component more focused on composition and less on logic.

packages/desktop-client/e2e/page-models/mobile-navigation.js (3)

18-26: LGTM: Added dragNavbarUp method with a suggestion

The dragNavbarUp method effectively simulates dragging the navbar up. It uses the bounding box of the main content area to calculate the drag distance, which is a good approach.

Consider extracting the magic number 1 used in the x-coordinates to a named constant for better readability:

const DRAG_X_OFFSET = 1;
// ...
sourcePosition: { x: DRAG_X_OFFSET, y: 0 },
targetPosition: { x: DRAG_X_OFFSET, y: boundingBox.height / 2 },

29-41: LGTM: Added dragNavbarDown method with suggestions

The dragNavbarDown method effectively simulates dragging the navbar down. It uses the bounding box of the navbar and the NAVBAR_ROWS constant to calculate the drag distance, which is a good approach.

  1. Consider extracting the magic number 1 used in the x-coordinates to a named constant for better readability, similar to the suggestion for dragNavbarUp.

  2. The method uses the same element (this.navbarSelector) as both source and target for the drag operation. While this might work as intended, it's worth verifying that this produces the desired behavior in all scenarios.

const DRAG_X_OFFSET = 1;
// ...
sourcePosition: { x: DRAG_X_OFFSET, y: 0 },
targetPosition: {
  x: DRAG_X_OFFSET,
  y: boundingBox.height / this.NAVBAR_ROWS,
},

70-97: LGTM: Added navigateToPage method with a suggestion

The navigateToPage method is well-implemented and effectively centralizes the navigation logic for all pages. It handles various scenarios and uses the constants and helper methods defined earlier in the class.

Consider adding a timeout to the waitFor calls to handle potential loading issues:

await pageInstance.waitFor({ timeout: 5000 });

This ensures that the test doesn't hang indefinitely if there's an issue with page loading.

📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Files that changed from the base of the PR and between 4f9f6f3 and 689c979.

📒 Files selected for processing (2)
  • packages/desktop-client/e2e/page-models/mobile-navigation.js (1 hunks)
  • packages/desktop-client/src/components/Page.tsx (1 hunks)
🧰 Additional context used
🔇 Additional comments (9)
packages/desktop-client/src/components/Page.tsx (1)

143-143: Excellent accessibility enhancement!

Adding the role="main" attribute to the <View> component that wraps the main content is a great improvement. This change enhances the semantic structure of the page, making it more accessible to users of assistive technologies. It aligns well with Web Content Accessibility Guidelines (WCAG) and helps screen readers identify the primary content area of the page.

packages/desktop-client/e2e/page-models/mobile-navigation.js (8)

3-3: LGTM: New import for MobileReportsPage

The addition of the MobileReportsPage import is consistent with the existing import pattern and supports the new goToReportsPage method.


10-13: LGTM: Enhanced constructor with key element selectors

The additions to the constructor provide convenient access to important page elements and selectors. The use of roles for element selection is a good practice for accessibility.


16-16: LGTM: Added NAVBAR_ROWS constant

The addition of the NAVBAR_ROWS constant is a good practice. It makes the code more maintainable by centralizing the definition of the number of navbar rows.


44-51: LGTM: Added hasNavbarState method

The hasNavbarState method is well-implemented. It efficiently checks for the presence of the navbar and its state. The use of the rest parameter syntax to accept multiple states provides flexibility in usage.


54-60: LGTM: Added NAV_LINKS_HIDDEN_BY_DEFAULT constant

The addition of the NAV_LINKS_HIDDEN_BY_DEFAULT constant is a good practice. It centralizes the definition of which navigation links are hidden by default, making the code more maintainable and easier to update.


62-68: LGTM: Added ROUTES_BY_PAGE constant

The addition of the ROUTES_BY_PAGE constant is a good practice. It centralizes the mapping between page names and their routes, making the code more maintainable and easier to update if routes change in the future.


100-126: LGTM: Updated navigation methods

The refactoring of the navigation methods to use the new navigateToPage method is a significant improvement. It reduces code duplication and centralizes the navigation logic, making the code more maintainable. The addition of the goToReportsPage method is consistent with the other navigation methods.


1-126: Overall assessment: Significant improvements to mobile navigation

The changes to the MobileNavigation class represent a substantial improvement in the mobile navigation functionality and code quality. Key improvements include:

  1. Introduction of new helper methods (dragNavbarUp, dragNavbarDown, hasNavbarState) for managing the navbar state.
  2. Addition of constants (NAVBAR_ROWS, NAV_LINKS_HIDDEN_BY_DEFAULT, ROUTES_BY_PAGE) that centralize important configuration data.
  3. Implementation of a generic navigateToPage method that centralizes navigation logic and reduces code duplication.
  4. Refactoring of existing navigation methods to use the new navigateToPage method.
  5. Addition of a new goToReportsPage method, consistent with other navigation methods.

These changes enhance the maintainability, readability, and flexibility of the mobile navigation code. The use of constants and helper methods makes the code more robust and easier to update in the future.

Great job on these improvements! The code is now more organized and easier to maintain.

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

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

Actionable comments posted: 0

🧹 Outside diff range and nitpick comments (3)
packages/desktop-client/src/components/mobile/MobileNavTabs.tsx (3)

109-121: LGTM: Well-implemented openFull function with a suggestion

The openFull function is well-implemented, using useCallback for optimization and providing different spring configurations for a better user experience.

Consider adding a comment explaining the purpose of the canceled parameter and how it affects the animation. This would improve code readability for future maintainers.


123-145: LGTM: Well-implemented openDefault and hide functions with a suggestion

The openDefault and hide functions are well-implemented, consistent with openFull, and use useCallback for optimization.

Consider refactoring the common logic in openFull, openDefault, and hide into a single function to reduce code duplication. For example:

const animateNavbar = useCallback(
  (state: 'open' | 'default' | 'hidden', y: number, velocity = 0, config = config.stiff) => {
    setNavbarState(state);
    api.start({
      y,
      immediate: false,
      config: { ...config, velocity },
    });
  },
  [api]
);

Then you can call this function from openFull, openDefault, and hide with the appropriate parameters.


217-217: LGTM: Added data-navbar-state attribute with a suggestion

The addition of the data-navbar-state attribute is a good practice. It improves the component's testability, accessibility, and debuggability by exposing the current state of the navbar.

Consider adding appropriate ARIA attributes to further enhance accessibility. For example, you could add aria-expanded={navbarState !== 'hidden'} to indicate when the navbar is expanded or collapsed.

📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Files that changed from the base of the PR and between 689c979 and 2589558.

📒 Files selected for processing (1)
  • packages/desktop-client/src/components/mobile/MobileNavTabs.tsx (7 hunks)
🧰 Additional context used
🔇 Additional comments (2)
packages/desktop-client/src/components/mobile/MobileNavTabs.tsx (2)

36-38: LGTM: Improved state management for navbar visibility

The introduction of navbarState with clear, descriptive values ('default', 'open', 'hidden') enhances the control over the navbar's visibility. Setting the initial state to 'default' resolves the previous issue of potential undefined state.


158-158: LGTM: Updated useEffect and drag handling logic

The changes in the useEffect hook and drag handling logic align well with the new state management and callback functions. The behavior remains consistent while providing more granular control over the navbar's state and animations.

Also applies to: 181-181, 183-183

@joel-jeremy joel-jeremy force-pushed the mobile-budget-menu-modal-vrt branch from 2589558 to 5dcc22d Compare October 8, 2024 18:15
Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

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

Actionable comments posted: 6

🧹 Outside diff range and nitpick comments (9)
packages/desktop-client/e2e/page-models/settings-page.js (3)

4-16: LGTM! Consider adding JSDoc comments for better documentation.

The addition of these properties in the constructor improves code readability and maintainability. Using data-testid attributes and ARIA roles for element selection is a good practice for E2E testing.

Consider adding JSDoc comments for each property to provide more context about their purpose and usage. For example:

/**
 * @type {Locator}
 * @description Locator for the export data button
 */
this.exportDataButton = this.settings.getByRole('button', {
  name: 'Export data',
});

29-34: LGTM! Consider adding error handling for unexpected button text.

The changes to the useBudgetType method are good improvements:

  1. Using the switchBudgetTypeButton property improves consistency.
  2. Checking the button text before clicking prevents unnecessary actions.

Consider adding error handling for cases where the button text doesn't match either expected value. For example:

const buttonText = await this.switchBudgetTypeButton.textContent();
if (buttonText.toLowerCase().includes(budgetType.toLowerCase())) {
  await this.switchBudgetTypeButton.click();
} else if (!buttonText.toLowerCase().includes('envelope') && !buttonText.toLowerCase().includes('tracking')) {
  throw new Error(`Unexpected budget type button text: ${buttonText}`);
}

38-51: LGTM! Consider adding a timeout for visibility checks.

The changes to the enableExperimentalFeature method significantly improve its robustness:

  1. Checking visibility before clicking prevents errors with hidden elements.
  2. Verifying the checkbox state before clicking prevents unnecessary toggling.

Consider adding a timeout to the visibility checks to prevent potential hangs if an element never becomes visible. For example:

const timeout = 5000; // 5 seconds
if (await this.advancedSettingsButton.isVisible({ timeout })) {
  await this.advancedSettingsButton.click();
}

if (await this.experimentalSettingsButton.isVisible({ timeout })) {
  await this.experimentalSettingsButton.click();
}
packages/desktop-client/e2e/page-models/mobile-budget-page.js (1)

86-87: Consider adding a comment to clarify the method's purpose.

The waitFor method has been made more generic, which is good for flexibility. However, it might be helpful to add a comment explaining its specific use in the context of the budget table.

packages/desktop-client/e2e/budget.mobile.test.js (4)

33-59: Consider adding comments and handling state changes

The setBudgetAverage function is well-implemented, but consider the following improvements:

  1. Add a comment explaining the purpose of the setBudgetAverageFn parameter for clarity.
  2. Consider wrapping the month navigation in a try-finally block to ensure the budget page always returns to the original month, even if an error occurs:
const originalMonth = await budgetPage.getSelectedMonth();
try {
  // Existing logic for calculating average
} finally {
  // Navigate back to the original month
  while (await budgetPage.getSelectedMonth() !== originalMonth) {
    await budgetPage.goToNextMonth();
  }
}

This ensures that the test state is preserved regardless of the function's outcome.


254-271: Enhance assertion clarity

The test case for copying last month's budget is well-implemented. To improve clarity, consider using a more explicit assertion:

const expectedBudget = currencyToAmount(lastMonthBudget);
const actualBudget = currencyToAmount(await budgetedButton.textContent());
expect(actualBudget).toBe(expectedBudget);

This approach converts the currency strings to numbers for comparison, making the assertion more robust against formatting differences.


273-298: Enhance test case clarity and assertions

The test cases for setting budget to average are well-implemented. Consider the following improvements:

  1. Add a comment explaining the test structure:
// Parameterized test cases for 3, 6, and 12-month budget averages
  1. Enhance the assertion for better clarity:
const expectedBudget = Math.abs(averageSpent);
const actualBudget = currencyToAmount(await budgetedButton.textContent());
expect(actualBudget).toBeCloseTo(expectedBudget, 2); // Compare with 2 decimal places precision

This approach provides more precise comparisons and better handles potential rounding issues.


300-328: Enhance test robustness and documentation

The test case for applying a budget template is well-implemented. Consider the following improvements:

  1. Add a comment explaining the purpose and format of the budget template:
// Test applying a budget template. Format: #template <amount>
  1. Add error handling for the experimental feature:
test.fail(({ browserName }) => browserName !== 'chromium', 'This test requires the experimental feature to be available');
  1. Verify that the experimental feature was successfully enabled:
await expect(settingsPage.isExperimentalFeatureEnabled('Goal templates')).resolves.toBe(true);

These changes will improve the test's robustness and clarity.

packages/desktop-client/src/components/mobile/MobileNavTabs.tsx (1)

112-112: Typo in comment: 'passed' should be 'past'

In the comment, replace "passed" with "past" for grammatical correctness.

Apply this diff to fix the typo:

-      // when cancel is true, it means that the user passed the upwards threshold
+      // when cancel is true, it means that the user past the upwards threshold
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Files that changed from the base of the PR and between 2589558 and 5dcc22d.

⛔ Files ignored due to path filters (41)
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-applies-budget-template-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-applies-budget-template-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-applies-budget-template-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-copies-last-month-s-budget-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-copies-last-month-s-budget-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-copies-last-month-s-budget-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-12-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-12-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-12-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-3-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-3-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-3-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-6-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-6-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-6-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-updates-the-budgeted-amount-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-updates-the-budgeted-amount-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-updates-the-budgeted-amount-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-applies-budget-template-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-applies-budget-template-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-applies-budget-template-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-copies-last-month-s-budget-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-copies-last-month-s-budget-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-copies-last-month-s-budget-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-12-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-12-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-12-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-3-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-3-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-3-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-6-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-6-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-6-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-updates-the-budgeted-amount-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-updates-the-budgeted-amount-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-updates-the-budgeted-amount-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/settings.mobile.test.js-snapshots/Mobile-Settings-checks-that-settings-page-can-be-opened-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/settings.mobile.test.js-snapshots/Mobile-Settings-checks-that-settings-page-can-be-opened-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/settings.mobile.test.js-snapshots/Mobile-Settings-checks-that-settings-page-can-be-opened-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/settings.mobile.test.js-snapshots/Mobile-Settings-checks-that-settings-page-can-be-opened-4-chromium-linux.png is excluded by !**/*.png
  • upcoming-release-notes/3583.md is excluded by !**/*.md
📒 Files selected for processing (23)
  • packages/desktop-client/e2e/budget.mobile.test.js (4 hunks)
  • packages/desktop-client/e2e/page-models/account-page.js (1 hunks)
  • packages/desktop-client/e2e/page-models/close-account-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-accounts-page.js (2 hunks)
  • packages/desktop-client/e2e/page-models/mobile-balance-menu-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-budget-menu-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-budget-page.js (7 hunks)
  • packages/desktop-client/e2e/page-models/mobile-category-menu-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-edit-notes-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-envelope-budget-summary-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-navigation.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-reports-page.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-tracking-budget-summary-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-transaction-entry-page.js (2 hunks)
  • packages/desktop-client/e2e/page-models/settings-page.js (1 hunks)
  • packages/desktop-client/src/components/Notifications.tsx (1 hunks)
  • packages/desktop-client/src/components/Page.tsx (1 hunks)
  • packages/desktop-client/src/components/mobile/MobileNavTabs.tsx (7 hunks)
  • packages/desktop-client/src/components/mobile/accounts/Accounts.jsx (2 hunks)
  • packages/desktop-client/src/components/mobile/transactions/FocusableAmountInput.tsx (1 hunks)
  • packages/desktop-client/src/components/mobile/transactions/TransactionEdit.jsx (2 hunks)
  • packages/desktop-client/src/components/reports/Overview.tsx (1 hunks)
  • packages/desktop-client/src/components/settings/index.tsx (1 hunks)
🚧 Files skipped from review as they are similar to previous changes (18)
  • packages/desktop-client/e2e/page-models/account-page.js
  • packages/desktop-client/e2e/page-models/close-account-modal.js
  • packages/desktop-client/e2e/page-models/mobile-accounts-page.js
  • packages/desktop-client/e2e/page-models/mobile-balance-menu-modal.js
  • packages/desktop-client/e2e/page-models/mobile-budget-menu-modal.js
  • packages/desktop-client/e2e/page-models/mobile-category-menu-modal.js
  • packages/desktop-client/e2e/page-models/mobile-edit-notes-modal.js
  • packages/desktop-client/e2e/page-models/mobile-envelope-budget-summary-modal.js
  • packages/desktop-client/e2e/page-models/mobile-reports-page.js
  • packages/desktop-client/e2e/page-models/mobile-tracking-budget-summary-modal.js
  • packages/desktop-client/e2e/page-models/mobile-transaction-entry-page.js
  • packages/desktop-client/src/components/Notifications.tsx
  • packages/desktop-client/src/components/Page.tsx
  • packages/desktop-client/src/components/mobile/accounts/Accounts.jsx
  • packages/desktop-client/src/components/mobile/transactions/FocusableAmountInput.tsx
  • packages/desktop-client/src/components/mobile/transactions/TransactionEdit.jsx
  • packages/desktop-client/src/components/reports/Overview.tsx
  • packages/desktop-client/src/components/settings/index.tsx
🧰 Additional context used
🔇 Additional comments (18)
packages/desktop-client/e2e/page-models/settings-page.js (2)

18-20: LGTM! Good addition for managing asynchronous behavior.

The waitFor method is a useful addition that provides a way to wait for the settings element to be ready. It's correctly implemented as an asynchronous method, which is appropriate for Playwright operations.


23-23: LGTM! Good refactoring for consistency.

The exportData method has been simplified to use the exportDataButton property defined in the constructor. This change improves code consistency and maintainability.

packages/desktop-client/e2e/page-models/mobile-navigation.js (7)

3-3: LGTM: New import for MobileReportsPage

The addition of the import statement for MobileReportsPage is correct and consistent with the existing import style. This import is likely used in the new goToReportsPage method, enhancing the navigation capabilities of the class.


10-13: LGTM: Enhanced constructor with new properties

The additions to the constructor are well-structured and use role-based selectors, which is good for accessibility and testing. These new properties (heading, navbar, mainContentSelector, navbarSelector) are likely used in the new navigation methods, improving the overall functionality of the class.


18-26: LGTM: Well-implemented dragNavbarUp method

The dragNavbarUp method is well-implemented. It effectively uses the page.dragAndDrop method to simulate dragging the navbar up, and calculates the target position based on the main content's bounding box. The use of class properties (mainContentSelector, navbarSelector) makes the code clean and maintainable.


44-51: LGTM: Well-implemented hasNavbarState method

The hasNavbarState method is well-implemented. It effectively checks the navbar's state based on its data-navbar-state attribute and handles the case where the navbar might not exist on the page. The use of the rest parameter syntax (...states) allows for flexible checking of multiple states, which is a good practice.


70-97: LGTM: Well-implemented navigateToPage method

The navigateToPage method is well-implemented and effectively centralizes the logic for navigating to different pages. It handles checking the current URL, managing the navbar state, and performing the navigation in a clear and organized manner. The use of class properties and methods (like hasNavbarState, dragNavbarUp, and dragNavbarDown) makes the code clean and maintainable.

The method also takes into account the NAV_LINKS_HIDDEN_BY_DEFAULT to determine the appropriate navbar states, which is a good practice for handling different types of navigation links.


100-123: LGTM: Well-refactored navigation methods

The navigation methods (goToBudgetPage, goToAccountsPage, goToTransactionEntryPage, goToReportsPage) have been effectively refactored to use the new navigateToPage method. This refactoring simplifies the methods, reduces code duplication, and improves maintainability by using a consistent pattern across all navigation methods.

The use of arrow functions for creating page instances is a good practice, as it ensures that the page instance is created only when needed. This approach is both efficient and clean.


125-127: LGTM: Consistent refactoring of goToSettingsPage

The goToSettingsPage method has been refactored consistently with the other navigation methods. It now uses the navigateToPage method, passing the page name and a factory function for creating the SettingsPage instance. This change maintains consistency across all navigation methods and improves the overall structure of the class.

packages/desktop-client/e2e/page-models/mobile-budget-page.js (7)

2-6: LGTM: New modal imports enhance modularity.

The addition of these modal imports indicates a shift towards a more modular architecture, which should improve code organization and maintainability.


147-148: LGTM: New openCategoryMenu method enhances functionality.

This new method follows the established pattern and provides a clean interface for opening the category menu modal.


187-187: LGTM: openBudgetMenu updated to use BudgetMenuModal.

This change aligns with the modal-based approach seen in other methods, providing a consistent interface for budget-related actions.


204-204: LGTM: openBalanceMenu updated to use BalanceMenuModal.

This change is consistent with the updates to other methods and enhances the functionality by providing access to the balance menu modal.


276-283: LGTM: openEnvelopeBudgetSummary updated for consistency.

The method has been renamed for clarity and now returns an EnvelopeBudgetSummaryModal instance, consistent with other modal-based methods.


308-315: LGTM: openTrackingBudgetSummary updated for consistency.

The method has been renamed for clarity and now returns a TrackingBudgetSummaryModal instance, consistent with other modal-based methods.


Line range hint 1-316: Overall: Excellent refactoring to a modal-based approach.

The changes in this file consistently implement a modal-based approach for various budget-related actions. This refactoring improves modularity, consistency, and should enhance maintainability of the codebase. The new structure allows for better separation of concerns between the main page and specific modal interactions.

Great job on this refactoring effort!

packages/desktop-client/e2e/budget.mobile.test.js (1)

Line range hint 1-383: Overall assessment: Good additions with room for improvement

The changes introduce valuable new functionality for budget management and comprehensive test coverage. The new utility functions and test cases are well-implemented and provide good coverage of the new features. However, there are opportunities for improvement:

  1. Refactor utility functions to reduce code duplication.
  2. Enhance test assertions for more precise comparisons.
  3. Improve error handling and documentation in test cases.
  4. Consider adding more inline comments to explain complex logic or test structures.

These improvements will enhance the maintainability and robustness of the codebase. Great work on expanding the functionality and test coverage!

packages/desktop-client/src/components/mobile/MobileNavTabs.tsx (1)

36-38: Initialization of navbarState ensures consistent state

The navbarState is now correctly initialized to 'default', which prevents it from being undefined on the initial render. This ensures consistent behavior across renders.

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

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

Actionable comments posted: 0

🧹 Outside diff range and nitpick comments (3)
packages/desktop-client/e2e/page-models/mobile-navigation.js (3)

16-30: LGTM: Well-structured constants with a minor suggestion

The addition of static private properties (#NAVBAR_ROWS, #NAV_LINKS_HIDDEN_BY_DEFAULT, and #ROUTES_BY_PAGE) is a good practice for encapsulating constants within the class. The use of the # prefix for private fields is a nice touch, leveraging modern JavaScript features.

Consider adding JSDoc comments for these static properties to provide more context about their purpose and usage. For example:

/**
 * The number of rows in the navbar.
 * @type {number}
 */
static #NAVBAR_ROWS = 3;

50-62: LGTM with a suggestion: dragNavbarDown implementation

The dragNavbarDown() method is well-implemented, using page.dragAndDrop() to simulate dragging the navbar down. The calculation of the target position based on the navbar's height and #NAVBAR_ROWS is correct.

Consider modifying the drag target to be a point just below the navbar instead of using the navbar itself as both source and target. This might produce a more reliable drag effect across different testing environments:

- await this.page.dragAndDrop(this.navbarSelector, this.navbarSelector, {
+ await this.page.dragAndDrop(this.navbarSelector, this.mainContentSelector, {
    sourcePosition: { x: 1, y: 0 },
    targetPosition: {
      x: 1,
-     y: boundingBox.height / MobileNavigation.#NAVBAR_ROWS,
+     y: boundingBox.height + 1, // Just below the navbar
    },
  });

75-104: LGTM: Well-structured navigateToPage method with a minor suggestion

The navigateToPage() method is well-implemented, centralizing the navigation logic for different pages. It correctly checks the current page, handles navbar states, and uses the static properties defined earlier. The use of a factory function for creating page instances is a good practice.

Consider adding error handling for cases where the page name doesn't exist in #ROUTES_BY_PAGE. For example:

if (!MobileNavigation.#ROUTES_BY_PAGE[pageName]) {
  throw new Error(`Invalid page name: ${pageName}`);
}

This would help catch potential typos or invalid page names early in the navigation process.

📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Files that changed from the base of the PR and between 5dcc22d and a7928ac.

📒 Files selected for processing (1)
  • packages/desktop-client/e2e/page-models/mobile-navigation.js (1 hunks)
🧰 Additional context used
🔇 Additional comments (6)
packages/desktop-client/e2e/page-models/mobile-navigation.js (6)

3-3: LGTM: Improved organization and maintainability

The addition of the MobileReportsPage import and the new properties in the constructor enhance the organization of the code. The use of role-based selectors (heading, navbar) and the introduction of mainContentSelector and navbarSelector improve maintainability and make the code more semantic.

Also applies to: 10-13


32-47: LGTM: Well-implemented dragNavbarUp method

The dragNavbarUp() method is well-implemented, using page.dragAndDrop() to simulate user interaction. The calculation of drag positions based on bounding boxes is correct and should accurately simulate dragging the navbar up into the main content area.


65-72: LGTM: Well-implemented hasNavbarState method

The hasNavbarState() method is well-implemented. It correctly handles the case where the navbar doesn't exist on the page and uses the data-navbar-state attribute to determine the navbar's state. The use of rest parameters (...states) provides flexibility in checking multiple states.


107-130: LGTM: Excellent refactoring of navigation methods

The refactoring of goToBudgetPage(), goToAccountsPage(), goToTransactionEntryPage(), and goToReportsPage() to use the new navigateToPage() method is excellent. This change significantly reduces code duplication and improves maintainability. The consistent use of arrow functions for creating page instances is clean and concise.


132-133: LGTM: Consistent refactoring of goToSettingsPage

The refactoring of goToSettingsPage() to use the new navigateToPage() method is consistent with the changes made to other navigation methods. This change maintains the uniformity of the navigation logic across all pages.


1-134: Excellent refactoring and enhancement of MobileNavigation class

The changes made to the MobileNavigation class significantly improve its functionality, maintainability, and extensibility. Key improvements include:

  1. Centralized navigation logic in the navigateToPage() method, reducing code duplication.
  2. Addition of methods for controlling navbar behavior (dragNavbarUp(), dragNavbarDown(), hasNavbarState()).
  3. Use of static private fields for constants, improving encapsulation.
  4. Consistent refactoring of all navigation methods to use the new centralized logic.
  5. Improved organization of selectors and properties in the constructor.

These changes make the code more robust and easier to maintain. The use of modern JavaScript features like private fields and arrow functions is commendable. This refactoring will make it easier to add new pages or modify navigation behavior in the future.

Great job on this refactoring! The code is now more organized, efficient, and easier to understand.

@joel-jeremy joel-jeremy force-pushed the mobile-budget-menu-modal-vrt branch from 4c58904 to 9746c67 Compare October 14, 2024 17:25
Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

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

Actionable comments posted: 3

🧹 Outside diff range and nitpick comments (7)
packages/desktop-client/e2e/page-models/mobile-navigation.js (4)

16-30: LGTM: Well-structured static properties

The addition of static properties #NAVBAR_ROWS, #NAV_LINKS_HIDDEN_BY_DEFAULT, and #ROUTES_BY_PAGE is well-implemented. The use of private fields (#) ensures encapsulation.

Consider adding JSDoc comments to describe the purpose of each static property for better code documentation.


32-47: LGTM: Well-implemented dragNavbarUp method

The dragNavbarUp method effectively simulates dragging the navbar into the main content area. The calculation of bounding boxes and use of page.dragAndDrop are appropriate.

Consider adding error handling for cases where the main content or navbar elements might not be found:

async dragNavbarUp() {
  const mainContent = this.page.locator(this.mainContentSelector);
  const navbar = this.page.locator(this.navbarSelector);

  await Promise.all([mainContent.waitFor(), navbar.waitFor()]);

  const mainContentBoundingBox = await mainContent.boundingBox();
  const navbarBoundingBox = await navbar.boundingBox();

  if (!mainContentBoundingBox || !navbarBoundingBox) {
    throw new Error('Failed to get bounding boxes for main content or navbar');
  }

  // ... rest of the method
}

50-62: LGTM: Well-implemented dragNavbarDown method

The dragNavbarDown method effectively simulates dragging the navbar down. The use of #NAVBAR_ROWS for calculating the drag distance is appropriate.

Consider adding error handling similar to the suggestion for dragNavbarUp:

async dragNavbarDown() {
  const navbar = this.page.locator(this.navbarSelector);
  await navbar.waitFor();

  const boundingBox = await navbar.boundingBox();
  if (!boundingBox) {
    throw new Error('Failed to get bounding box for navbar');
  }

  // ... rest of the method
}

75-104: LGTM: Well-implemented navigateToPage method

The navigateToPage method effectively centralizes navigation logic, reducing code duplication and ensuring consistent behavior across different pages. The handling of navbar states and use of static properties is well-implemented.

Consider adding a timeout to the waitFor calls to handle potential navigation issues:

const timeout = 5000; // 5 seconds
await this.navbar.waitFor({ timeout });
// ...
await pageInstance.waitFor({ timeout });

This ensures that the navigation doesn't hang indefinitely if there's an unexpected issue.

packages/desktop-client/src/components/mobile/MobileNavTabs.tsx (1)

109-145: LGTM: Well-structured callback functions for navbar management

The new callback functions (openFull, openDefault, and hide) are well-implemented and provide clear management of the navbar's state and animations. The use of useCallback for memoization is appropriate.

One minor suggestion:

In the openFull function, the canceled property is destructured but not used. Consider removing it if it's not needed:

-const openFull = useCallback(
-  ({ canceled }: { canceled: boolean }) => {
+const openFull = useCallback(
+  () => {
   setNavbarState('open');
-  // when cancel is true, it means that the user passed the upwards threshold
-  // so we change the spring config to create a nice wobbly effect
   api.start({
     y: openFullY,
     immediate: false,
-    config: canceled ? config.wobbly : config.stiff,
+    config: config.stiff,
   });
 },
 [api, openFullY],
);

If the canceled property is indeed needed for the wobbly effect, please add a comment explaining its purpose and how it's determined.

packages/desktop-client/e2e/budget.mobile.test.js (2)

33-59: Ensure correct retrieval of spent amounts after month navigation

In the setBudgetAverage function, you're correctly retrieving the spentButton after navigating to the previous month within the loop. This ensures that you're accessing the correct element corresponding to the current month in each iteration. Good job on implementing this correctly!

However, to improve readability and make the intent clearer, consider adding a comment explaining why you're retrieving the spentButton inside the loop. For example:

for (let i = 0; i < numberOfMonths; i++) {
  await budgetPage.goToPreviousMonth();
  // Retrieve the spent button after navigation to ensure we're getting the correct element for the current month
  const spentButton = await budgetPage.getButtonForSpent(categoryName);
  const spent = await spentButton.textContent();
  totalSpent += currencyToAmount(spent);
}

This comment will help future maintainers understand the reasoning behind this implementation.


273-298: LGTM: New test cases for setting budget to average

These new test cases effectively verify the functionality of setting the budget to 3, 6, and 12 month averages. The use of a parameterized approach is excellent for reducing code duplication and ensuring consistent testing across different average periods.

Suggestion for improvement:
Consider adding a descriptive comment before the test cases to explain the purpose of the array and how it's used in the parameterized tests. This will enhance readability for other developers. For example:

// Test cases for setting budget to average over different periods
// Each entry is [number of months, corresponding setter function]
[
  [3, setTo3MonthAverage],
  [6, setTo6MonthAverage],
  [12, setToYearlyAverage],
].forEach(([numberOfMonths, setBudgetAverageFn]) => {
  // ... (rest of the code remains the same)
});
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Files that changed from the base of the PR and between 4c58904 and 9746c67.

⛔ Files ignored due to path filters (38)
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-applies-budget-template-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-applies-budget-template-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-applies-budget-template-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-copies-last-month-s-budget-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-copies-last-month-s-budget-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-copies-last-month-s-budget-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-12-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-12-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-12-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-3-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-3-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-3-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-6-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-6-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-6-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-applies-budget-template-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-applies-budget-template-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-applies-budget-template-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-copies-last-month-s-budget-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-copies-last-month-s-budget-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-copies-last-month-s-budget-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-12-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-12-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-12-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-3-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-3-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-3-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-6-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-6-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-6-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-updates-the-budgeted-amount-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-updates-the-budgeted-amount-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-updates-the-budgeted-amount-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/settings.mobile.test.js-snapshots/Mobile-Settings-checks-that-settings-page-can-be-opened-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/settings.mobile.test.js-snapshots/Mobile-Settings-checks-that-settings-page-can-be-opened-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/settings.mobile.test.js-snapshots/Mobile-Settings-checks-that-settings-page-can-be-opened-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/settings.mobile.test.js-snapshots/Mobile-Settings-checks-that-settings-page-can-be-opened-4-chromium-linux.png is excluded by !**/*.png
  • upcoming-release-notes/3583.md is excluded by !**/*.md
📒 Files selected for processing (23)
  • packages/desktop-client/e2e/budget.mobile.test.js (4 hunks)
  • packages/desktop-client/e2e/page-models/account-page.js (1 hunks)
  • packages/desktop-client/e2e/page-models/close-account-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-accounts-page.js (2 hunks)
  • packages/desktop-client/e2e/page-models/mobile-balance-menu-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-budget-menu-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-budget-page.js (7 hunks)
  • packages/desktop-client/e2e/page-models/mobile-category-menu-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-edit-notes-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-envelope-budget-summary-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-navigation.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-reports-page.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-tracking-budget-summary-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-transaction-entry-page.js (2 hunks)
  • packages/desktop-client/e2e/page-models/settings-page.js (1 hunks)
  • packages/desktop-client/src/components/Notifications.tsx (1 hunks)
  • packages/desktop-client/src/components/Page.tsx (1 hunks)
  • packages/desktop-client/src/components/mobile/MobileNavTabs.tsx (7 hunks)
  • packages/desktop-client/src/components/mobile/accounts/Accounts.jsx (2 hunks)
  • packages/desktop-client/src/components/mobile/transactions/FocusableAmountInput.tsx (1 hunks)
  • packages/desktop-client/src/components/mobile/transactions/TransactionEdit.jsx (2 hunks)
  • packages/desktop-client/src/components/reports/Overview.tsx (1 hunks)
  • packages/desktop-client/src/components/settings/index.tsx (1 hunks)
🚧 Files skipped from review as they are similar to previous changes (19)
  • packages/desktop-client/e2e/page-models/account-page.js
  • packages/desktop-client/e2e/page-models/close-account-modal.js
  • packages/desktop-client/e2e/page-models/mobile-accounts-page.js
  • packages/desktop-client/e2e/page-models/mobile-balance-menu-modal.js
  • packages/desktop-client/e2e/page-models/mobile-budget-menu-modal.js
  • packages/desktop-client/e2e/page-models/mobile-category-menu-modal.js
  • packages/desktop-client/e2e/page-models/mobile-edit-notes-modal.js
  • packages/desktop-client/e2e/page-models/mobile-envelope-budget-summary-modal.js
  • packages/desktop-client/e2e/page-models/mobile-reports-page.js
  • packages/desktop-client/e2e/page-models/mobile-tracking-budget-summary-modal.js
  • packages/desktop-client/e2e/page-models/mobile-transaction-entry-page.js
  • packages/desktop-client/e2e/page-models/settings-page.js
  • packages/desktop-client/src/components/Notifications.tsx
  • packages/desktop-client/src/components/Page.tsx
  • packages/desktop-client/src/components/mobile/accounts/Accounts.jsx
  • packages/desktop-client/src/components/mobile/transactions/FocusableAmountInput.tsx
  • packages/desktop-client/src/components/mobile/transactions/TransactionEdit.jsx
  • packages/desktop-client/src/components/reports/Overview.tsx
  • packages/desktop-client/src/components/settings/index.tsx
🧰 Additional context used
🔇 Additional comments (13)
packages/desktop-client/e2e/page-models/mobile-navigation.js (4)

3-3: LGTM: Enhanced constructor and imports

The addition of the MobileReportsPage import and the new constructor properties (heading, navbar, mainContentSelector, and navbarSelector) are well-implemented. The use of role-based selectors is a good practice for accessibility and testing.

Also applies to: 10-13


65-72: LGTM: Well-implemented hasNavbarState method

The hasNavbarState method is well-implemented. It correctly handles cases where the navbar is not present and uses the data-navbar-state attribute to determine the state. The flexibility to check for multiple states is a good design choice.


107-130: LGTM: Consistent updates to navigation methods

The updates to the navigation methods (goToBudgetPage, goToAccountsPage, goToTransactionEntryPage, goToReportsPage) are well-implemented. The consistent use of the new navigateToPage method reduces code duplication and ensures uniform navigation behavior across different pages.


132-133: LGTM: Consistent update to goToSettingsPage

The update to the goToSettingsPage method is well-implemented and consistent with the changes made to other navigation methods. It correctly uses the new navigateToPage method and the SettingsPage class.

packages/desktop-client/src/components/mobile/MobileNavTabs.tsx (5)

1-6: LGTM: Necessary imports added

The new imports (useCallback, useEffect, and useState) are correctly added and are essential for the refactored component's functionality.


36-38: LGTM: State initialization addressed

The navbarState is now properly initialized with a default value, addressing the concern raised in a previous review. The state values are well-defined, providing clear representation of the navbar's possible states.


103-104: LGTM: Clear constants for navbar positions

The introduction of openFullY and openDefaultY constants improves code readability and maintainability. These values clearly represent different positions of the navbar.


181-183: LGTM: Updated drag handler aligns with new state management

The changes in the drag handler correctly use the new openDefault and openFull functions, aligning with the refactored navbar state management. The use of velocity in openDefault is a good touch for smooth animations.

Note: As mentioned in a previous comment, consider reviewing the use of the canceled parameter in the openFull function, as it's passed here but not utilized in the function body.


217-217: LGTM: Helpful data attribute for navbar state

The addition of the data-navbar-state attribute is a good practice. It provides a clear indicator of the navbar's current state, which can be useful for testing and potentially for CSS styling based on the navbar state.

packages/desktop-client/e2e/budget.mobile.test.js (2)

240-252: LGTM: Updated test case for setting budget amount

The changes to this test case are consistent with the new implementation of budget management using modals. The test now correctly opens the budget menu modal, sets the budget amount, and verifies the updated value. This ensures that the new modal-based interaction is properly tested.


254-271: LGTM: New test case for copying last month's budget

This new test case effectively verifies the 'copy last month's budget' functionality. It covers the following key aspects:

  1. Navigating to the previous month
  2. Retrieving the budget amount from the previous month
  3. Returning to the current month
  4. Applying the 'copy last month's budget' action
  5. Verifying that the current month's budget matches the previous month's budget

The test is well-structured and provides good coverage for this feature.

packages/desktop-client/e2e/page-models/mobile-budget-page.js (2)

147-148: openCategoryMenu method implementation looks good

The openCategoryMenu method is correctly implemented and follows the existing patterns for opening modals.


86-87: Ensure all references to waitForBudgetTable are updated

The method waitForBudgetTable() has been renamed to waitFor(options). Please verify that all references to waitForBudgetTable in the codebase have been updated to prevent any broken functionality.

Run the following script to check for any remaining references:

✅ Verification successful

All references to waitForBudgetTable have been successfully updated.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Find any references to waitForBudgetTable in the codebase.

# Test: Search for occurrences of 'waitForBudgetTable'. Expect: No matches.
rg --type js --type ts 'waitForBudgetTable'

Length of output: 43

Comment on lines 158 to 161
openDefault();
}
}, [scrollY]);
}, [scrollY, hide, openDefault]);
Copy link
Contributor

Choose a reason for hiding this comment

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

⚠️ Potential issue

Add missing dependency to useEffect

The change from close() to openDefault() aligns well with the new navbar state management. However, the useEffect dependency array is still incomplete.

Add previousScrollY to the dependency array to ensure the effect runs correctly when this value changes:

useEffect(() => {
  if (
    scrollY &&
    previousScrollY &&
    scrollY > previousScrollY &&
    previousScrollY !== 0
  ) {
    hide();
  } else {
    openDefault();
  }
-}, [scrollY, hide, openDefault]);
+}, [scrollY, previousScrollY, hide, openDefault]);
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
openDefault();
}
}, [scrollY]);
}, [scrollY, hide, openDefault]);
openDefault();
}
}, [scrollY, previousScrollY, hide, openDefault]);

Comment on lines +9 to +13
const copyLastMonthBudget = async (budgetPage, categoryName) => {
const budgetMenuModal = await budgetPage.openBudgetMenu(categoryName);
await budgetMenuModal.copyLastMonthBudget();
await budgetMenuModal.close();
};
Copy link
Contributor

Choose a reason for hiding this comment

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

🛠️ Refactor suggestion

Consider refactoring to reduce code duplication

The copyLastMonthBudget function follows a pattern that's repeated in other utility functions. To improve maintainability and reduce duplication, consider implementing a higher-order function as suggested in the past review comments:

const performBudgetMenuAction = async (budgetPage, categoryName, action) => {
  const budgetMenuModal = await budgetPage.openBudgetMenu(categoryName);
  await action(budgetMenuModal);
  await budgetMenuModal.close();
};

const copyLastMonthBudget = (budgetPage, categoryName) =>
  performBudgetMenuAction(budgetPage, categoryName, menu => menu.copyLastMonthBudget());

This refactoring can be applied to all similar functions (setTo3MonthAverage, setTo6MonthAverage, setToYearlyAverage).

@MatissJanis
Copy link
Member

Looks like the test is flaky:

Would you mind taking a peak at it to make it more stable? Apart from that LGTM

@joel-jeremy joel-jeremy force-pushed the mobile-budget-menu-modal-vrt branch from 88d5375 to 557a4f4 Compare October 17, 2024 16:09
Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

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

Actionable comments posted: 4

🧹 Outside diff range and nitpick comments (5)
packages/desktop-client/e2e/page-models/settings-page.js (3)

18-20: Good addition of the waitFor method.

This method enhances test stability by providing a consistent way to wait for the settings element. To further improve its flexibility, consider adding a default timeout value:

async waitFor(options = { timeout: 5000 }) {
  await this.settings.waitFor(options);
}

This allows users to override the default timeout if needed.


29-34: Good improvements to the useBudgetType method.

The use of the pre-defined switchBudgetTypeButton property and the text check before clicking are excellent improvements. They enhance consistency and prevent unnecessary actions.

To make the method more robust, consider using a case-insensitive comparison:

if (buttonText.toLowerCase().includes(budgetType.toLowerCase())) {
  await this.switchBudgetTypeButton.click();
}

This change would make the method more resilient to potential variations in button text capitalization.


38-51: Good enhancements to the enableExperimentalFeature method.

The added visibility checks and the conditional click on the checkbox improve the method's robustness and efficiency. These changes will help prevent errors and unnecessary actions during test execution.

To further improve error handling, consider adding timeouts and error messages:

async enableExperimentalFeature(featureName, timeout = 5000) {
  try {
    await this.advancedSettingsButton.waitFor({ state: 'visible', timeout });
    await this.advancedSettingsButton.click();
    
    await this.experimentalSettingsButton.waitFor({ state: 'visible', timeout });
    await this.experimentalSettingsButton.click();
    
    const featureCheckbox = this.page.getByRole('checkbox', { name: featureName });
    await featureCheckbox.waitFor({ state: 'visible', timeout });
    
    if (!(await featureCheckbox.isChecked())) {
      await featureCheckbox.click();
    }
  } catch (error) {
    throw new Error(`Failed to enable experimental feature "${featureName}": ${error.message}`);
  }
}

This implementation adds timeouts to the visibility checks and wraps the entire method in a try-catch block for better error reporting.

packages/desktop-client/e2e/page-models/mobile-navigation.js (1)

1-141: Overall assessment: Excellent improvements to mobile navigation

The changes made to the MobileNavigation class significantly enhance its functionality and maintainability:

  • The addition of static constants improves configuration management.
  • New methods for navbar manipulation (dragNavbarUp, dragNavbarDown, hasNavbarState) provide fine-grained control over navigation behavior.
  • The centralized navigateToPage method and refactored navigation methods reduce code duplication and improve consistency.
  • The use of role-based selectors enhances accessibility and testability.

These improvements will likely lead to more robust and maintainable mobile navigation testing.

Consider documenting the expected navbar states and their implications in a comment at the class level. This would provide valuable context for future maintainers and help ensure consistent usage of the navigation methods.

packages/desktop-client/src/components/mobile/MobileNavTabs.tsx (1)

217-217: LGTM: Added data-navbar-state attribute

The addition of the data-navbar-state attribute is a good improvement. It provides a clear indication of the navbar's current state in the DOM, which can be useful for styling or testing purposes.

Consider using this attribute in your CSS or test scripts to ensure the navbar is in the correct state during different interactions.

📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Files that changed from the base of the PR and between 9746c67 and 557a4f4.

⛔ Files ignored due to path filters (41)
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-applies-budget-template-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-applies-budget-template-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-applies-budget-template-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-copies-last-month-s-budget-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-copies-last-month-s-budget-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-copies-last-month-s-budget-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-12-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-12-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-12-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-3-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-3-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-3-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-6-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-6-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-6-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-updates-the-budgeted-amount-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-updates-the-budgeted-amount-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-updates-the-budgeted-amount-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-applies-budget-template-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-applies-budget-template-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-applies-budget-template-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-copies-last-month-s-budget-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-copies-last-month-s-budget-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-copies-last-month-s-budget-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-12-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-12-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-12-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-3-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-3-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-3-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-6-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-6-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-6-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-updates-the-budgeted-amount-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-updates-the-budgeted-amount-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-updates-the-budgeted-amount-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/settings.mobile.test.js-snapshots/Mobile-Settings-checks-that-settings-page-can-be-opened-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/settings.mobile.test.js-snapshots/Mobile-Settings-checks-that-settings-page-can-be-opened-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/settings.mobile.test.js-snapshots/Mobile-Settings-checks-that-settings-page-can-be-opened-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/settings.mobile.test.js-snapshots/Mobile-Settings-checks-that-settings-page-can-be-opened-4-chromium-linux.png is excluded by !**/*.png
  • upcoming-release-notes/3583.md is excluded by !**/*.md
📒 Files selected for processing (24)
  • packages/desktop-client/e2e/budget.mobile.test.js (4 hunks)
  • packages/desktop-client/e2e/budget.test.js (1 hunks)
  • packages/desktop-client/e2e/page-models/account-page.js (1 hunks)
  • packages/desktop-client/e2e/page-models/close-account-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-accounts-page.js (2 hunks)
  • packages/desktop-client/e2e/page-models/mobile-balance-menu-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-budget-menu-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-budget-page.js (7 hunks)
  • packages/desktop-client/e2e/page-models/mobile-category-menu-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-edit-notes-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-envelope-budget-summary-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-navigation.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-reports-page.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-tracking-budget-summary-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-transaction-entry-page.js (2 hunks)
  • packages/desktop-client/e2e/page-models/settings-page.js (1 hunks)
  • packages/desktop-client/src/components/Notifications.tsx (1 hunks)
  • packages/desktop-client/src/components/Page.tsx (1 hunks)
  • packages/desktop-client/src/components/mobile/MobileNavTabs.tsx (7 hunks)
  • packages/desktop-client/src/components/mobile/accounts/Accounts.jsx (2 hunks)
  • packages/desktop-client/src/components/mobile/transactions/FocusableAmountInput.tsx (1 hunks)
  • packages/desktop-client/src/components/mobile/transactions/TransactionEdit.jsx (2 hunks)
  • packages/desktop-client/src/components/reports/Overview.tsx (1 hunks)
  • packages/desktop-client/src/components/settings/index.tsx (1 hunks)
🚧 Files skipped from review as they are similar to previous changes (18)
  • packages/desktop-client/e2e/page-models/account-page.js
  • packages/desktop-client/e2e/page-models/close-account-modal.js
  • packages/desktop-client/e2e/page-models/mobile-accounts-page.js
  • packages/desktop-client/e2e/page-models/mobile-balance-menu-modal.js
  • packages/desktop-client/e2e/page-models/mobile-budget-menu-modal.js
  • packages/desktop-client/e2e/page-models/mobile-category-menu-modal.js
  • packages/desktop-client/e2e/page-models/mobile-edit-notes-modal.js
  • packages/desktop-client/e2e/page-models/mobile-envelope-budget-summary-modal.js
  • packages/desktop-client/e2e/page-models/mobile-reports-page.js
  • packages/desktop-client/e2e/page-models/mobile-tracking-budget-summary-modal.js
  • packages/desktop-client/e2e/page-models/mobile-transaction-entry-page.js
  • packages/desktop-client/src/components/Notifications.tsx
  • packages/desktop-client/src/components/Page.tsx
  • packages/desktop-client/src/components/mobile/accounts/Accounts.jsx
  • packages/desktop-client/src/components/mobile/transactions/FocusableAmountInput.tsx
  • packages/desktop-client/src/components/mobile/transactions/TransactionEdit.jsx
  • packages/desktop-client/src/components/reports/Overview.tsx
  • packages/desktop-client/src/components/settings/index.tsx
🧰 Additional context used
🔇 Additional comments (28)
packages/desktop-client/e2e/page-models/settings-page.js (2)

4-15: Excellent improvements to the constructor!

The addition of these properties enhances code readability, maintainability, and test stability. Using data-testid attributes and getByRole() methods also improves accessibility testing. This refactoring will make future updates easier and more consistent.


23-23: Great simplification of the exportData method.

Using the pre-defined exportDataButton property improves consistency and reduces duplication. This change makes the code more maintainable and less prone to errors if selectors need to be updated in the future.

packages/desktop-client/e2e/page-models/mobile-navigation.js (7)

3-3: LGTM: New import for MobileReportsPage

The addition of the import statement for MobileReportsPage is correct and consistent with the existing import style. This import is necessary for the new goToReportsPage method implemented later in the class.


10-13: LGTM: Enhanced constructor with new properties

The additions to the constructor are well-implemented:

  • Using getByRole for heading and navbar improves accessibility and testability.
  • The mainContentSelector and navbarSelector are consistently defined using role attributes.

These new properties lay the groundwork for the enhanced navigation functionality implemented in the class.


16-30: LGTM: Well-defined static constants for navigation configuration

The addition of static private constants is a good practice:

  • #NAVBAR_ROWS defines the number of rows in the navbar.
  • #NAV_LINKS_HIDDEN_BY_DEFAULT lists the navigation links that are initially hidden.
  • #ROUTES_BY_PAGE maps page names to their respective routes.

These constants improve code maintainability and readability by centralizing configuration values.


32-47: LGTM: Well-implemented dragNavbarUp method

The dragNavbarUp method is well-implemented:

  • It correctly calculates the drag distance using bounding boxes.
  • The use of page.dragAndDrop accurately simulates user interaction.
  • The navbar is positioned correctly at the bottom of the main content area.

This method enhances the navigation functionality by allowing the navbar to be revealed when hidden.


65-72: LGTM: Well-implemented hasNavbarState method

The hasNavbarState method is well-designed:

  • It first checks for the existence of the navbar, preventing potential errors.
  • The use of the 'data-navbar-state' attribute is a good way to track the navbar's state.
  • The method's flexibility in checking multiple states enhances its utility.

This method provides a robust way to determine the navbar's current state, which is crucial for the navigation logic.


75-104: LGTM: Excellent implementation of centralized navigation logic

The navigateToPage method is very well-implemented:

  • It efficiently checks if the user is already on the requested page, avoiding unnecessary navigation.
  • The method handles the navbar state appropriately, revealing it when needed.
  • Waiting for the new page to load and adjusting the navbar state afterwards ensures a smooth navigation experience.
  • The use of a pageModelFactory provides flexibility in instantiating page objects.

This centralized navigation method significantly improves code maintainability and reduces duplication in individual navigation methods.


107-139: LGTM: Excellent refactoring of navigation methods

The refactoring of all navigation methods (goToBudgetPage, goToAccountsPage, goToTransactionEntryPage, goToReportsPage, goToSettingsPage) is excellent:

  • Each method now uses the centralized navigateToPage function, significantly reducing code duplication.
  • The consistent pattern across all navigation methods improves readability and maintainability.
  • The use of arrow functions for page model instantiation is concise and appropriate.

This refactoring greatly improves the overall structure and maintainability of the navigation logic in the MobileNavigation class.

packages/desktop-client/src/components/mobile/MobileNavTabs.tsx (6)

36-38: LGTM: Improved state management for navbar visibility

The introduction of navbarState with a defined initial state of 'default' is a good improvement. This addresses the previous concern about potential undefined state and provides more granular control over the navbar's visibility.


109-121: LGTM: Well-implemented openFull callback

The openFull callback is well-implemented, providing a smooth animation for fully opening the navbar. The use of useCallback for optimization and the conditional wobbly effect add nice touches to the user experience.


123-133: LGTM: Well-implemented openDefault callback

The openDefault callback is well-implemented, providing a smooth animation for partially opening the navbar. The inclusion of a velocity parameter allows for more natural animations based on user interactions.


135-145: LGTM: Well-implemented hide callback

The hide callback is well-implemented, providing a smooth animation for hiding the navbar. The inclusion of a velocity parameter allows for more natural animations based on user interactions.


181-183: LGTM: Improved drag handling logic

The updates to the drag handling logic align well with the new navbar state management. The use of velocity in openDefault and the canceled parameter in openFull should provide smoother and more engaging animations for the user.


158-160: ⚠️ Potential issue

Add missing dependency to useEffect

The change from close() to openDefault() aligns well with the new navbar state management. However, the useEffect dependency array is still incomplete.

Add previousScrollY to the dependency array to ensure the effect runs correctly when this value changes:

useEffect(() => {
  if (
    scrollY &&
    previousScrollY &&
    scrollY > previousScrollY &&
    previousScrollY !== 0
  ) {
    hide();
  } else {
    openDefault();
  }
-}, [scrollY, hide, openDefault]);
+}, [scrollY, previousScrollY, hide, openDefault]);
packages/desktop-client/e2e/budget.mobile.test.js (10)

3-7: LGTM: Appropriate imports added

The new imports for currency conversion and month utilities are relevant to the added functionality and correctly placed.


93-94: LGTM: Proper test setup for budget type

The addition of navigation to the settings page and setting the budget type ensures that the tests are run with the correct configuration.


218-220: LGTM: Improved category menu modal interaction

The updates to opening the category menu modal and checking its heading enhance the readability and maintainability of the test.


230-232: LGTM: Improved budget menu modal interaction

The updates to opening the budget menu modal and checking its heading enhance the readability and maintainability of the test.


240-252: LGTM: Well-structured test for updating budgeted amount

The test for updating the budgeted amount is well-structured and correctly uses the new budget menu modal interaction. It properly sets the budget amount and verifies the result.


254-271: LGTM: Comprehensive test for copying last month's budget

This new test thoroughly verifies the functionality of copying the previous month's budget. It correctly navigates between months, uses the new copyLastMonthBudget utility function, and checks the expected outcome.


273-298: LGTM: Comprehensive tests for setting budget to averages

These new tests efficiently cover setting the budget to 3, 6, and 12 month averages. The use of a loop to generate tests for different periods is a good practice, reducing code duplication. The tests correctly use the setBudgetAverage function and verify the expected outcomes.


300-327: LGTM: Thorough test for applying budget template

This new test comprehensively covers the functionality of applying a budget template. It correctly enables the experimental 'Goal templates' feature, sets up a template, applies it, and verifies the result. The test is well-structured and includes all necessary steps to validate the feature.


350-379: LGTM: Well-structured tests for balance and budget summary modals

These updated tests for opening balance menu modal and budget summary modals are well-structured and correctly differentiate between Envelope and Tracking budget types. They use the new modal interaction methods appropriately and verify the expected outcomes.


Line range hint 1-382: Overall assessment: High-quality changes with minor improvement suggestions

The changes in this file significantly enhance the test coverage for mobile budget functionality. New utility functions and tests have been added to cover various budget management scenarios. The code is generally well-structured and maintainable.

Two minor suggestions for improvement were made:

  1. Refactoring the utility functions to reduce code duplication.
  2. Improving the retrieval of spent amounts in the setBudgetAverage function.

These suggestions aim to further enhance code maintainability and reliability. Overall, the changes are of high quality and ready for merging after addressing the minor suggestions.

packages/desktop-client/e2e/page-models/mobile-budget-page.js (3)

276-283: Verify method name change from openEnvelopeBudgetSummaryMenu to openEnvelopeBudgetSummary

The method openEnvelopeBudgetSummaryMenu has been renamed to openEnvelopeBudgetSummary. Ensure that all places where this method is called have been updated accordingly to avoid broken links or method not found errors.

Run the following script to find any remaining references to openEnvelopeBudgetSummaryMenu:

✅ Verification successful

All references to openEnvelopeBudgetSummaryMenu have been successfully updated.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Find any remaining references to 'openEnvelopeBudgetSummaryMenu'
rg --type js --fixed-strings 'openEnvelopeBudgetSummaryMenu'

Length of output: 60


308-315: Verify method name change from openTrackingBudgetSummaryMenu to openTrackingBudgetSummary

The method openTrackingBudgetSummaryMenu has been renamed to openTrackingBudgetSummary. Please verify that all existing calls to this method have been updated to reflect the new name to prevent any runtime errors.

Run the following script to check for any remaining references to openTrackingBudgetSummaryMenu:

✅ Verification successful

Method Name Change Verified

All references to openTrackingBudgetSummaryMenu have been successfully updated to openTrackingBudgetSummary.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Find any remaining references to 'openTrackingBudgetSummaryMenu'
rg --type js --fixed-strings 'openTrackingBudgetSummaryMenu'

Length of output: 60


Script:

#!/bin/bash
# Description: Find any remaining references to 'openTrackingBudgetSummaryMenu'
rg --type js --fixed-strings 'openTrackingBudgetSummaryMenu'

Length of output: 60


86-87: Verify method renaming for consistency across the codebase

The method waitForBudgetTable has been renamed to waitFor. Please ensure that all references to waitForBudgetTable have been updated to waitFor to prevent any potential issues due to unresolved references.

Run the following script to check for any remaining references to waitForBudgetTable:

✅ Verification successful

Method renaming verified: All references to waitForBudgetTable have been updated to waitFor.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Find any remaining references to 'waitForBudgetTable'
rg --type js --fixed-strings 'waitForBudgetTable'

Length of output: 49


Script:

#!/bin/bash
# Description: Verify any remaining references to 'waitForBudgetTable' with context
rg --type js --fixed-strings 'waitForBudgetTable' -A 2 -B 2

Length of output: 59

await expect(summary.getByText('Available Funds')).toBeVisible({
timeout: 10000,
});
await expect(summary.getByText('Available Funds')).toBeVisible();
Copy link
Contributor

Choose a reason for hiding this comment

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

💡 Codebase verification

Inconsistent Timeout Configuration Across Test Files

The removal of the timeout: 10000 parameter from the toBeVisible() expectation in packages/desktop-client/e2e/budget.test.js is inconsistent with other test files that utilize custom timeouts:

  • packages/desktop-client/e2e/onboarding.test.js contains multiple toBeVisible() calls with custom timeout values.

This inconsistency could lead to unexpected test behaviors and reduced stability. It's recommended to standardize timeout configurations across all end-to-end tests or ensure that each timeout adjustment is appropriately justified and documented.

🔗 Analysis chain

Removal of explicit timeout: Potential impact on test stability

The removal of the timeout: 10000 parameter from the toBeVisible() expectation could affect the test's behavior:

  1. It may make the test faster if the "Available Funds" text typically appears quickly.
  2. However, it could potentially make the test more brittle if the element sometimes takes longer to appear than the default timeout.

Given the concerns about flaky tests mentioned in the PR comments, this change warrants careful consideration.

Could you please clarify the reasoning behind removing this timeout? Was it causing issues, or is it part of a broader strategy to standardize timeouts?

To ensure this change doesn't negatively impact test stability, I recommend:

  1. Monitoring the test results closely after this change, especially in CI environments.
  2. If the change is intentional and based on specific insights, consider adding a comment explaining the rationale.
  3. If you haven't already, consider implementing retry logic for flaky tests to improve overall stability.

This will help us understand if this is an isolated change or part of a broader refactoring effort.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Check for other custom timeouts in the test files
echo "Checking for other custom timeouts in test files:"
rg --type js 'toBeVisible\(\s*\{.*timeout:' packages/desktop-client/e2e

Length of output: 546

Comment on lines 50 to 65
async dragNavbarDown() {
const boundingBox = await this.page
.locator(this.navbarSelector)
.boundingBox();

return new MobileAccountsPage(this.page);
await this.page.dragAndDrop(this.navbarSelector, this.navbarSelector, {
sourcePosition: { x: 1, y: 0 },
targetPosition: {
x: 1,
// Only scroll until bottom of screen i.e. bottom of first navbar row.
y: boundingBox.height / MobileNavigation.#NAVBAR_ROWS,
},
});
Copy link
Contributor

Choose a reason for hiding this comment

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

🛠️ Refactor suggestion

LGTM with suggestion: Well-implemented dragNavbarDown method

The dragNavbarDown method is well-implemented overall:

  • It correctly calculates the drag distance based on the navbar's height and #NAVBAR_ROWS.
  • The use of page.dragAndDrop accurately simulates user interaction.

However, there's a potential improvement:

Consider adjusting the drag target to be a point just below the navbar instead of using the navbar itself as both source and target. This might produce a more reliable drag effect across different environments.

- await this.page.dragAndDrop(this.navbarSelector, this.navbarSelector, {
+ await this.page.dragAndDrop(this.navbarSelector, this.mainContentSelector, {
    sourcePosition: { x: 1, y: 0 },
    targetPosition: {
      x: 1,
-     y: boundingBox.height / MobileNavigation.#NAVBAR_ROWS,
+     y: boundingBox.height + 1, // Just below the navbar
    },
  });

This change ensures that the drag operation moves the navbar to a position just below its current location.

Committable suggestion was skipped due to low confidence.

Comment on lines +9 to +32
const copyLastMonthBudget = async (budgetPage, categoryName) => {
const budgetMenuModal = await budgetPage.openBudgetMenu(categoryName);
await budgetMenuModal.copyLastMonthBudget();
await budgetMenuModal.close();
};

const setTo3MonthAverage = async (budgetPage, categoryName) => {
const budgetMenuModal = await budgetPage.openBudgetMenu(categoryName);
await budgetMenuModal.setTo3MonthAverage();
await budgetMenuModal.close();
};

const setTo6MonthAverage = async (budgetPage, categoryName) => {
const budgetMenuModal = await budgetPage.openBudgetMenu(categoryName);
await budgetMenuModal.setTo6MonthAverage();
await budgetMenuModal.close();
};

const setToYearlyAverage = async (budgetPage, categoryName) => {
const budgetMenuModal = await budgetPage.openBudgetMenu(categoryName);
await budgetMenuModal.setToYearlyAverage();
await budgetMenuModal.close();
};

Copy link
Contributor

Choose a reason for hiding this comment

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

🛠️ Refactor suggestion

Consider refactoring to reduce code duplication

The utility functions copyLastMonthBudget, setTo3MonthAverage, setTo6MonthAverage, and setToYearlyAverage share identical structures, leading to code duplication. To improve maintainability and reduce duplication, consider implementing a higher-order function:

const performBudgetMenuAction = async (budgetPage, categoryName, action) => {
  const budgetMenuModal = await budgetPage.openBudgetMenu(categoryName);
  await action(budgetMenuModal);
  await budgetMenuModal.close();
};

const copyLastMonthBudget = (budgetPage, categoryName) =>
  performBudgetMenuAction(budgetPage, categoryName, menu => menu.copyLastMonthBudget());

const setTo3MonthAverage = (budgetPage, categoryName) =>
  performBudgetMenuAction(budgetPage, categoryName, menu => menu.setTo3MonthAverage());

// Similar for setTo6MonthAverage and setToYearlyAverage

This refactoring will make the code more maintainable and easier to extend in the future.

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

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

Actionable comments posted: 0

🧹 Outside diff range and nitpick comments (4)
packages/desktop-client/e2e/page-models/mobile-navigation.js (4)

18-32: LGTM: Good use of static constants with a minor suggestion

The addition of static constants improves maintainability and readability. The use of private fields (#) is a good practice for encapsulation.

Consider adding JSDoc comments for these constants to provide more context about their usage, especially for #NAVBAR_ROWS and #NAV_LINKS_HIDDEN_BY_DEFAULT.


52-64: LGTM: Well-implemented dragNavbarDown method with a suggestion

The dragNavbarDown method is well-implemented:

  • It correctly calculates the drag distance based on the navbar's height and #NAVBAR_ROWS.
  • The use of page.dragAndDrop is consistent with the dragNavbarUp method.

Consider adding a comment explaining the calculation of the y value in the targetPosition. While the current comment is helpful, a more detailed explanation of why dividing by #NAVBAR_ROWS achieves the desired effect would improve maintainability.


67-76: LGTM: Well-implemented hasNavbarState method with a suggestion

The hasNavbarState method is well-implemented:

  • It handles the case when the navbar is not present.
  • The use of data-navbar-state attribute is a good practice for state management.
  • The expect assertion ensures the attribute exists.

Consider adding error handling for the case where the data-navbar-state attribute exists but is empty. This could be done by checking if dataNavbarState is truthy before calling includes.


79-108: LGTM: Well-implemented navigateToPage method with a suggestion

The navigateToPage method is well-implemented:

  • It checks the current URL to avoid unnecessary navigation.
  • The handling of navbar visibility during navigation is comprehensive.
  • The use of a factory function for creating page instances is a good practice for flexibility.

Consider adding a timeout or retry mechanism for the link.click() operation, as network conditions or animations might sometimes delay the navigation. This would improve the robustness of the navigation process.

📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Files that changed from the base of the PR and between 557a4f4 and 738703d.

📒 Files selected for processing (1)
  • packages/desktop-client/e2e/page-models/mobile-navigation.js (1 hunks)
🧰 Additional context used
🔇 Additional comments (6)
packages/desktop-client/e2e/page-models/mobile-navigation.js (6)

12-15: LGTM: Good use of roles for element selection

The addition of heading and navbar properties using getByRole is a good practice for accessibility and maintainability. The mainContentSelector and navbarSelector provide clear references to important page elements.


34-49: LGTM: Well-implemented dragNavbarUp method

The dragNavbarUp method is well-implemented:

  • It correctly calculates drag positions based on bounding boxes.
  • The use of page.dragAndDrop accurately simulates user interaction.

111-116: LGTM: Well-refactored goToBudgetPage method

The goToBudgetPage method has been effectively refactored to use the new navigateToPage method. This simplifies the code and reduces duplication across navigation methods.


118-130: LGTM: Consistent refactoring of navigation methods

The goToAccountsPage and goToTransactionEntryPage methods have been refactored consistently with goToBudgetPage. This refactoring improves code consistency and maintainability across the class.


132-143: LGTM: Consistent refactoring of remaining navigation methods

The goToReportsPage and goToSettingsPage methods have been refactored consistently with the other navigation methods. This completes the refactoring of all navigation methods in the class, resulting in a more maintainable and consistent codebase.


1-143: Overall: Well-implemented refactoring and enhancements

The changes to the MobileNavigation class significantly improve its structure, maintainability, and functionality. The introduction of the navigateToPage method and the consistent refactoring of all navigation methods reduce code duplication and enhance readability. The new utility methods for navbar manipulation are well-implemented and provide robust functionality for mobile navigation testing.

While some minor suggestions were made for further improvements, the overall quality of the changes is high. Great job on this refactoring effort!

@joel-jeremy joel-jeremy force-pushed the mobile-budget-menu-modal-vrt branch from 738703d to 75da4cc Compare November 4, 2024 17:24
Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

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

Actionable comments posted: 5

🧹 Outside diff range and nitpick comments (7)
packages/desktop-client/e2e/page-models/close-account-modal.js (1)

1-15: Consider adding explicit waiting mechanisms for better test stability.

Given the reported test flakiness in GitHub Actions, consider implementing explicit waiting mechanisms before interactions:

  • Wait for element to be stable before filling account name
  • Wait for dropdown to appear after typing
  • Wait for button to be clickable before clicking

Example implementation:

async selectTransferAccount(accountName) {
  const input = this.locator.getByPlaceholder('Select account...');
  await input.waitFor({ state: 'stable' });
  await input.fill(accountName);
  await this.page.keyboard.press('Enter');
}
packages/desktop-client/e2e/page-models/mobile-transaction-entry-page.js (2)

7-12: Consider using consistent selector strategy for better test stability.

While the parent-child relationship between elements is well structured, mixing TestID and role selectors could contribute to test flakiness. Consider using TestID consistently for all selectors, especially for the addTransactionButton.

    this.transactionForm = page.getByTestId('transaction-form');
    this.amountField = this.transactionForm.getByTestId('amount-input');
    this.footer = page.getByTestId('transaction-form-footer');
-   this.addTransactionButton = this.footer.getByRole('button', {
-     name: 'Add transaction',
-   });
+   this.addTransactionButton = this.footer.getByTestId('add-transaction-button');

Wait patterns need improvement across page models

Several page models have direct button clicks without proper wait strategies, which could lead to flaky tests. Key findings:

  • Most mobile page models have basic waitFor implementations but lack granular waits before interactions
  • Multiple direct .click() calls without waiting for button states (enabled/visible)
  • Some pages use arbitrary timeouts (waitForTimeout) which is an anti-pattern:
    • mobile-budget-page.js: Uses waitForTimeout(1000) and waitForTimeout(500)
    • schedules-page.js: Uses waitForTimeout(1000)

Consider:

  1. Adding state checks before clicks (e.g., await button.waitFor({ state: 'visible' }))
  2. Replacing fixed timeouts with proper element state waits
  3. Implementing consistent wait patterns across all page models
🔗 Analysis chain

Line range hint 1-29: Verify wait patterns across other page models.

Given the test stability issues, let's check if other page models could benefit from similar wait strategy improvements.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Find similar patterns in other page models that might need wait improvements

# Look for page model classes that might be missing proper wait strategies
ast-grep --pattern 'class $_Page {
  constructor(page) {
    this.page = page;
    $$$
  }
  $$$
}'

# Look for direct button clicks without waits
ast-grep --pattern 'async $_($$_) {
  await this.$_button.click();
  $$$
}'

Length of output: 179


Script:

#!/bin/bash
# Let's try a different approach to find page models and their patterns

# First, find all page model files
fd -e js "page.js$" packages/desktop-client/e2e/page-models/

# Then examine their content for click patterns
echo "=== Click patterns ==="
rg "button.*click\(\)" packages/desktop-client/e2e/page-models/ -A 2

# Look for wait patterns
echo -e "\n=== Wait patterns ==="
rg "waitFor|waitUntil" packages/desktop-client/e2e/page-models/ -B 1 -A 1

Length of output: 15769

packages/desktop-client/e2e/page-models/mobile-budget-menu-modal.js (1)

1-23: Add parameter validation and visibility check method.

The class structure is well-organized, but could benefit from additional robustness:

  1. Add parameter validation in the constructor
  2. Implement a visibility check method as suggested in the past review
 export class BudgetMenuModal {
   constructor(page, locator) {
+    if (!page || !locator) {
+      throw new Error('Page and locator are required parameters');
+    }
     this.page = page;
     this.locator = locator;
     // ... rest of the constructor
   }

+  async isVisible() {
+    return await this.heading.isVisible();
+  }
packages/desktop-client/e2e/budget.test.js (1)

Line range hint 1-70: Consider implementing additional test stability measures.

To address the reported flakiness, consider implementing the following improvements:

  1. Add retry logic for flaky tests:
test.describe.configure({ retries: 2 });
  1. Implement custom test fixtures for common setup patterns:
// In test-utils.js
export const withBudgetPage = test.extend({
  budgetPage: async ({ page }, use) => {
    const configPage = new ConfigurationPage(page);
    await page.goto('/');
    const budgetPage = await configPage.createTestFile();
    await page.mouse.move(0, 0);
    await use(budgetPage);
  }
});
  1. Add explicit wait strategies before screenshot comparisons:
const waitForBudgetStability = async (page) => {
  await page.waitForLoadState('networkidle');
  // Add other stability checks as needed
};

These changes would help improve test reliability while maintaining the existing functionality.

packages/desktop-client/src/components/mobile/MobileNavTabs.tsx (1)

110-146: Consider adding type definition for openFull parameters

The navigation state management functions are well-implemented. However, the openFull function's parameter type could be more explicit.

Consider this improvement:

-  const openFull = useCallback(
-    ({ canceled }: { canceled: boolean }) => {
+  type OpenFullParams = {
+    canceled: boolean;
+  };
+  const openFull = useCallback(
+    ({ canceled }: OpenFullParams) => {
packages/desktop-client/e2e/page-models/mobile-budget-page.js (1)

86-87: Consider renaming waitFor to a more descriptive method name

Renaming the method from waitForBudgetTable to waitFor may reduce clarity, as waitFor is quite generic. Consider using a more descriptive name like waitForBudgetTable or waitForPageContent to better convey its purpose.

📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Reviewing files that changed from the base of the PR and between 738703d and 75da4cc.

⛔ Files ignored due to path filters (40)
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-applies-budget-template-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-applies-budget-template-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-applies-budget-template-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-copies-last-month-s-budget-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-copies-last-month-s-budget-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-copies-last-month-s-budget-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-12-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-12-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-12-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-3-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-3-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-3-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-6-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-6-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-6-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-updates-the-budgeted-amount-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-updates-the-budgeted-amount-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-updates-the-budgeted-amount-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-applies-budget-template-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-applies-budget-template-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-applies-budget-template-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-copies-last-month-s-budget-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-copies-last-month-s-budget-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-copies-last-month-s-budget-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-12-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-12-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-12-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-3-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-3-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-3-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-6-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-6-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-6-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-updates-the-budgeted-amount-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-updates-the-budgeted-amount-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-updates-the-budgeted-amount-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/settings.mobile.test.js-snapshots/Mobile-Settings-checks-that-settings-page-can-be-opened-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/settings.mobile.test.js-snapshots/Mobile-Settings-checks-that-settings-page-can-be-opened-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/settings.mobile.test.js-snapshots/Mobile-Settings-checks-that-settings-page-can-be-opened-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/settings.mobile.test.js-snapshots/Mobile-Settings-checks-that-settings-page-can-be-opened-4-chromium-linux.png is excluded by !**/*.png
📒 Files selected for processing (23)
  • packages/desktop-client/e2e/budget.mobile.test.js (4 hunks)
  • packages/desktop-client/e2e/budget.test.js (1 hunks)
  • packages/desktop-client/e2e/page-models/account-page.js (1 hunks)
  • packages/desktop-client/e2e/page-models/close-account-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-accounts-page.js (2 hunks)
  • packages/desktop-client/e2e/page-models/mobile-balance-menu-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-budget-menu-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-budget-page.js (7 hunks)
  • packages/desktop-client/e2e/page-models/mobile-category-menu-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-edit-notes-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-envelope-budget-summary-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-navigation.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-reports-page.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-tracking-budget-summary-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-transaction-entry-page.js (2 hunks)
  • packages/desktop-client/e2e/page-models/settings-page.js (1 hunks)
  • packages/desktop-client/src/components/Notifications.tsx (1 hunks)
  • packages/desktop-client/src/components/Page.tsx (1 hunks)
  • packages/desktop-client/src/components/mobile/MobileNavTabs.tsx (7 hunks)
  • packages/desktop-client/src/components/mobile/accounts/Accounts.jsx (2 hunks)
  • packages/desktop-client/src/components/mobile/transactions/FocusableAmountInput.tsx (1 hunks)
  • packages/desktop-client/src/components/mobile/transactions/TransactionEdit.jsx (2 hunks)
  • packages/desktop-client/src/components/reports/Overview.tsx (1 hunks)
✅ Files skipped from review due to trivial changes (1)
  • packages/desktop-client/e2e/page-models/account-page.js
🚧 Files skipped from review as they are similar to previous changes (15)
  • packages/desktop-client/e2e/budget.mobile.test.js
  • packages/desktop-client/e2e/page-models/mobile-accounts-page.js
  • packages/desktop-client/e2e/page-models/mobile-balance-menu-modal.js
  • packages/desktop-client/e2e/page-models/mobile-category-menu-modal.js
  • packages/desktop-client/e2e/page-models/mobile-edit-notes-modal.js
  • packages/desktop-client/e2e/page-models/mobile-envelope-budget-summary-modal.js
  • packages/desktop-client/e2e/page-models/mobile-reports-page.js
  • packages/desktop-client/e2e/page-models/mobile-tracking-budget-summary-modal.js
  • packages/desktop-client/e2e/page-models/settings-page.js
  • packages/desktop-client/src/components/Notifications.tsx
  • packages/desktop-client/src/components/Page.tsx
  • packages/desktop-client/src/components/mobile/accounts/Accounts.jsx
  • packages/desktop-client/src/components/mobile/transactions/FocusableAmountInput.tsx
  • packages/desktop-client/src/components/mobile/transactions/TransactionEdit.jsx
  • packages/desktop-client/src/components/reports/Overview.tsx
🧰 Additional context used
📓 Learnings (1)
packages/desktop-client/e2e/page-models/close-account-modal.js (2)
Learnt from: UnderKoen
PR: actualbudget/actual#3499
File: packages/desktop-client/e2e/page-models/close-account-modal.js:16-18
Timestamp: 2024-09-24T20:20:59.061Z
Learning: In the `CloseAccountModal` class, methods are implemented without additional error handling, return values, or additional comments to maintain consistency with existing codebase practices.
Learnt from: UnderKoen
PR: actualbudget/actual#3499
File: packages/desktop-client/e2e/page-models/close-account-modal.js:16-18
Timestamp: 2024-10-08T15:46:15.739Z
Learning: In the `CloseAccountModal` class, methods are implemented without additional error handling, return values, or additional comments to maintain consistency with existing codebase practices.
🔇 Additional comments (20)
packages/desktop-client/e2e/page-models/close-account-modal.js (3)

2-4: LGTM! Constructor refactor improves test reliability.

The switch from rootPage to locator parameter aligns with Playwright's best practices for element selection.


13-13: LGTM! Button interaction follows best practices.

Using getByRole with button name is a reliable approach for button interactions.


8-9: Verify the stability of account selection.

The implementation looks good, but given the reported flaky tests, let's verify the stability of this interaction.

Let's check for similar patterns in other stable tests:

✅ Verification successful

The search results show that the implementation pattern of using keyboard.press('Enter') is consistently used across multiple test files in the codebase, including budget-page.js, schedules-page.js, rules-page.js, and mobile-transaction-entry-page.js. However, let's verify if there are any other patterns for account selection that might be more stable.


Based on the search results, I can now generate the final response:

The account selection implementation is consistent with stable patterns

The current implementation using getByPlaceholder('Select account...').fill() followed by keyboard.press('Enter') is the only pattern used for account selection in the e2e tests, and it's consistent with keyboard interaction patterns used across multiple stable test files (budget-page.js, schedules-page.js, rules-page.js, mobile-transaction-entry-page.js). The implementation is reliable as it:

  • Uses Playwright's recommended locator strategy
  • Follows the same keyboard interaction pattern used in other stable parts of the test suite
  • Has proper element targeting before interaction
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Look for similar account selection patterns in other test files
# to compare implementation approaches

# Search for similar account selection patterns
rg -A 2 "getByPlaceholder\('Select account...'\)" "packages/desktop-client/e2e"

# Search for keyboard.press usage in test files
rg -A 2 "keyboard\.press\('Enter'\)" "packages/desktop-client/e2e"

Length of output: 2196


Script:

#!/bin/bash
# Search for other account selection patterns that might be using different approaches
rg -A 3 "Select account" "packages/desktop-client/e2e"

# Look for any wait or expect statements around account selection
rg -A 2 "waitFor.*account|expect.*account" "packages/desktop-client/e2e"

Length of output: 8766

packages/desktop-client/e2e/budget.test.js (1)

30-30: ⚠️ Potential issue

Consider maintaining explicit timeout for improved test stability.

Given the reported test flakiness and the importance of this visibility check for the budget summary, removing the explicit timeout might impact test reliability. This element's visibility is crucial for subsequent visual regression tests.

Recommendations:

  1. Restore the explicit timeout:
-    await expect(summary.getByText('Available Funds')).toBeVisible();
+    await expect(summary.getByText('Available Funds')).toBeVisible({ timeout: 10000 });
  1. Consider implementing a global timeout configuration:
// In a test configuration file
export const TEST_TIMEOUTS = {
  elementVisibility: 10000,
  // ... other timeout constants
};

This would help maintain consistency across all test files and make timeout management more maintainable.

packages/desktop-client/e2e/page-models/mobile-navigation.js (7)

1-15: LGTM! Well-structured initialization with accessibility-first approach

The class initialization is well-implemented with proper role-based selectors, making the tests more resilient and accessibility-friendly.


18-32: LGTM! Well-designed static configuration using private fields

The use of private static fields (#) for configuration is a good practice as it:

  • Prevents external modification of these constants
  • Makes the code more maintainable by centralizing navigation-related constants

67-76: LGTM! Robust navbar state checking

The hasNavbarState method is well-implemented with proper error handling and state verification.


79-108: LGTM! Well-designed navigation helper with proper state management

The navigateToPage method is well-implemented with:

  • Proper URL checking to avoid unnecessary navigation
  • Smart handling of hidden navigation links
  • Appropriate navbar state management

111-143: LGTM! Consistent page navigation implementation

The page navigation methods are well-implemented using the factory pattern and the common navigation helper, following the DRY principle.


52-64: 🛠️ Refactor suggestion

Improve reliability of dragNavbarDown

Using the same element as both source and target in dragAndDrop could be unreliable. Consider using a point calculation relative to the viewport instead.

-    await this.page.dragAndDrop(this.navbarSelector, this.navbarSelector, {
+    const viewport = await this.page.viewportSize();
+    await this.page.dragAndDrop(this.navbarSelector, this.mainContentSelector, {
       sourcePosition: { x: 1, y: 0 },
       targetPosition: {
         x: 1,
-        y: boundingBox.height / MobileNavigation.#NAVBAR_ROWS,
+        y: Math.min(viewport.height, boundingBox.height / MobileNavigation.#NAVBAR_ROWS),
       },
     });

Likely invalid or redundant comment.


34-49: Consider adding retry logic for drag operations

The drag calculations based on bounding boxes could be fragile in E2E tests. Consider adding retry logic with position verification to ensure the drag operation succeeds.

packages/desktop-client/src/components/mobile/MobileNavTabs.tsx (4)

4-6: LGTM: Clean state management setup

The state management setup with TypeScript types and initial state looks good. The proper initialization of navbarState to 'default' prevents undefined state issues.

Also applies to: 37-39


159-161: Previous dependency array fix wasn't fully applied

A past review comment already identified the missing dependencies in the useEffect hook. The fix needs to be fully applied.


218-218: Good UX improvements

The addition of data-navbar-state attribute and userSelect: 'none' are good improvements:

  • The data attribute helps with testing and state tracking
  • Preventing text selection during drag gestures enhances the mobile experience

Also applies to: 271-271


37-39: Consider improving test stability

Given the reported flaky tests in GitHub Actions, here are suggestions to improve test reliability:

  1. Use the newly added data-navbar-state attribute to verify state transitions
  2. Add delays or wait conditions after animations complete
  3. Consider mocking animations in tests to reduce timing dependencies

Also applies to: 218-218

✅ Verification successful

Let me gather more information about the test coverage and animation handling in the codebase.


Based on the search results, I can now provide a final response since we have enough context about the testing patterns in the codebase.

Consider using data-navbar-state for test stability

The data-navbar-state attribute you've added is aligned with the project's existing testing patterns. To improve test reliability:

  • Use getByTestId with the data-navbar-state attribute to assert navigation states
  • Follow the established pattern seen in other components (e.g., PayeeAutocomplete.test.tsx) for selector-based testing
  • Avoid timing-based assertions since there's no evidence of animation mocking utilities in the codebase
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Look for existing test files and patterns
# that might need updating with the new state attribute

# Find test files related to this component
fd -e test.tsx -e test.ts | grep -i "mobilenav"

# Look for existing test patterns that might need updating
rg -l "getBy(Role|TestId)" $(fd -e test.tsx -e test.ts)

Length of output: 2798


Script:

#!/bin/bash
# Look for existing animation-related test utilities and patterns
rg -i "animation|transition" $(fd -e test.tsx -e test.ts)

# Search for existing navbar/navigation test patterns
rg -i "nav.*test" -g "*.{tsx,ts}"

# Look for any existing gesture or animation mocking utilities
rg -i "mock.*animation|mock.*gesture" -g "*.{tsx,ts,js}"

# Check if there are any existing data-testid patterns for navigation
rg "data-(testid|navbar)" -g "*.{tsx,ts,test.tsx,test.ts}"

Length of output: 8450

packages/desktop-client/e2e/page-models/mobile-budget-page.js (5)

308-315: Verify that callers of openTrackingBudgetSummary handle the returned TrackingBudgetSummaryModal instance

Since openTrackingBudgetSummary now returns a TrackingBudgetSummaryModal, please verify that all invoking code handles the returned instance appropriately.

Run the following script to find usages of openTrackingBudgetSummary and check if the returned instance is utilized:

#!/bin/bash
# Description: Verify that all calls to `openTrackingBudgetSummary` handle the returned value.

# Search for calls to `openTrackingBudgetSummary` across the codebase
rg 'await\s+(\w+\s*=\s*)?.*\.openTrackingBudgetSummary\(' -A1

187-187: Verify that callers of openBudgetMenu handle the returned BudgetMenuModal instance

The method openBudgetMenu now returns a BudgetMenuModal instance. Ensure that all code invoking this method properly handles the returned value.

Run the following script to find usages of openBudgetMenu and check if the returned instance is utilized:

#!/bin/bash
# Description: Verify that all calls to `openBudgetMenu` handle the returned value.

# Search for calls to `openBudgetMenu` across the codebase
rg 'await\s+(\w+\s*=\s*)?.*\.openBudgetMenu\(' -A1

204-204: Verify that callers of openBalanceMenu handle the returned BalanceMenuModal instance

Since openBalanceMenu now returns a BalanceMenuModal, please ensure that all calling code is updated to handle the returned instance appropriately.

Run the following script to find usages of openBalanceMenu and check if the returned instance is utilized:

#!/bin/bash
# Description: Verify that all calls to `openBalanceMenu` handle the returned value.

# Search for calls to `openBalanceMenu` across the codebase
rg 'await\s+(\w+\s*=\s*)?.*\.openBalanceMenu\(' -A1

276-283: Verify that callers of openEnvelopeBudgetSummary handle the returned EnvelopeBudgetSummaryModal instance

Now that openEnvelopeBudgetSummary returns an EnvelopeBudgetSummaryModal, ensure that all calling code properly handles this returned value.

Run the following script to find usages of openEnvelopeBudgetSummary and check if the returned instance is utilized:

#!/bin/bash
# Description: Verify that all calls to `openEnvelopeBudgetSummary` handle the returned value.

# Search for calls to `openEnvelopeBudgetSummary` across the codebase
rg 'await\s+(\w+\s*=\s*)?.*\.openEnvelopeBudgetSummary\(' -A1

147-148: Verify that callers of openCategoryMenu handle the returned CategoryMenuModal instance

Since openCategoryMenu now returns a CategoryMenuModal instance, ensure that all calling code has been updated to handle this return value appropriately.

Run the following script to find usages of openCategoryMenu and check if the returned instance is utilized:

✅ Verification successful

The search pattern in the executed script seems incorrect as it's searching for openTrackingBudgetSummary instead of openCategoryMenu. Let me search with the correct pattern.


All callers properly handle the returned CategoryMenuModal instance

The verification shows that there are two calls to openCategoryMenu in budget.mobile.test.js, and both correctly store the returned value in categoryMenuModal. In one case, the returned modal is further used to call editNotes(), demonstrating proper handling of the returned instance.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Verify that all calls to `openCategoryMenu` handle the returned value.

# Search for calls to `openCategoryMenu` across the codebase
rg 'await\s+(\w+\s*=\s*)?.*\.openCategoryMenu\(' -A1

Length of output: 220


Script:

#!/bin/bash
# Search for calls to openCategoryMenu across the codebase
rg 'await\s+(\w+\s*=\s*)?.*\.openCategoryMenu\(' -A1

Length of output: 486

Comment on lines +15 to +16
async waitFor(options) {
await this.transactionForm.waitFor(options);
Copy link
Contributor

Choose a reason for hiding this comment

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

🛠️ Refactor suggestion

Enhance wait strategy to improve test stability.

Given the reported test flakiness, consider implementing a more comprehensive wait strategy that ensures the form is not just present but also interactive.

  async waitFor(options) {
-   await this.transactionForm.waitFor(options);
+   await this.transactionForm.waitFor({ state: 'visible', ...options });
+   await this.amountField.waitFor({ state: 'visible' });
+   await this.addTransactionButton.waitFor({ state: 'enabled' });
  }

Committable suggestion skipped: line range outside the PR's diff.

Comment on lines +26 to 29
await this.addTransactionButton.click();

return new MobileAccountPage(this.page);
}
Copy link
Contributor

Choose a reason for hiding this comment

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

🛠️ Refactor suggestion

Add navigation wait to prevent race conditions.

To improve test stability, consider adding an explicit wait for navigation after clicking the add button.

  async createTransaction() {
    await this.addTransactionButton.click();
+   await this.page.waitForNavigation({ waitUntil: 'networkidle' });
    return new MobileAccountPage(this.page);
  }

Committable suggestion skipped: line range outside the PR's diff.

Comment on lines +25 to +33
async close() {
await this.heading.getByRole('button', { name: 'Close' }).click();
}

async setBudgetAmount(newAmount) {
await this.budgetAmountInput.fill(String(newAmount));
await this.budgetAmountInput.blur();
await this.close();
}
Copy link
Contributor

Choose a reason for hiding this comment

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

⚠️ Potential issue

Add wait conditions and error handling to improve test stability.

Given the reported flaky tests, enhance the modal interaction methods with proper wait conditions and error handling:

   async close() {
+    try {
+      await this.heading.waitFor({ state: 'visible' });
       await this.heading.getByRole('button', { name: 'Close' }).click();
+      await this.heading.waitFor({ state: 'hidden' });
+    } catch (error) {
+      throw new Error(`Failed to close modal: ${error.message}`);
+    }
   }

   async setBudgetAmount(newAmount) {
+    try {
+      await this.budgetAmountInput.waitFor({ state: 'visible' });
       await this.budgetAmountInput.fill(String(newAmount));
       await this.budgetAmountInput.blur();
+      // Verify the input value
+      const actualValue = await this.budgetAmountInput.inputValue();
+      if (actualValue !== String(newAmount)) {
+        throw new Error(`Budget amount not set correctly. Expected: ${newAmount}, Got: ${actualValue}`);
+      }
       await this.close();
+    } catch (error) {
+      throw new Error(`Failed to set budget amount: ${error.message}`);
+    }
   }
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
async close() {
await this.heading.getByRole('button', { name: 'Close' }).click();
}
async setBudgetAmount(newAmount) {
await this.budgetAmountInput.fill(String(newAmount));
await this.budgetAmountInput.blur();
await this.close();
}
async close() {
try {
await this.heading.waitFor({ state: 'visible' });
await this.heading.getByRole('button', { name: 'Close' }).click();
await this.heading.waitFor({ state: 'hidden' });
} catch (error) {
throw new Error(`Failed to close modal: ${error.message}`);
}
}
async setBudgetAmount(newAmount) {
try {
await this.budgetAmountInput.waitFor({ state: 'visible' });
await this.budgetAmountInput.fill(String(newAmount));
await this.budgetAmountInput.blur();
// Verify the input value
const actualValue = await this.budgetAmountInput.inputValue();
if (actualValue !== String(newAmount)) {
throw new Error(`Budget amount not set correctly. Expected: ${newAmount}, Got: ${actualValue}`);
}
await this.close();
} catch (error) {
throw new Error(`Failed to set budget amount: ${error.message}`);
}
}

Comment on lines +35 to +53
async copyLastMonthBudget() {
await this.copyLastMonthBudgetButton.click();
}

async setTo3MonthAverage() {
await this.setTo3MonthAverageButton.click();
}

async setTo6MonthAverage() {
await this.setTo6MonthAverageButton.click();
}

async setToYearlyAverage() {
await this.setToYearlyAverageButton.click();
}

async applyBudgetTemplate() {
await this.applyBudgetTemplateButton.click();
}
Copy link
Contributor

Choose a reason for hiding this comment

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

🛠️ Refactor suggestion

Implement reusable action handler to improve stability.

The budget action methods follow the same pattern and could benefit from a shared implementation that includes proper wait conditions and error handling:

+  async performAction(button, actionName) {
+    try {
+      await button.waitFor({ state: 'visible' });
+      await button.click();
+      // Wait for any loading indicators to disappear
+      await this.page.waitForSelector('[data-testid="loading-indicator"]', 
+        { state: 'hidden', timeout: 5000 }).catch(() => {});
+    } catch (error) {
+      throw new Error(`Failed to ${actionName}: ${error.message}`);
+    }
+  }

   async copyLastMonthBudget() {
-    await this.copyLastMonthBudgetButton.click();
+    await this.performAction(
+      this.copyLastMonthBudgetButton,
+      'copy last month budget'
+    );
   }

   async setTo3MonthAverage() {
-    await this.setTo3MonthAverageButton.click();
+    await this.performAction(
+      this.setTo3MonthAverageButton,
+      'set to 3 month average'
+    );
   }

   // Apply similar changes to other methods

This refactor:

  1. Adds consistent wait conditions
  2. Handles loading states
  3. Provides better error messages
  4. Reduces code duplication

Committable suggestion skipped: line range outside the PR's diff.

Comment on lines 34 to 65
async dragNavbarUp() {
const mainContentBoundingBox = await this.page
.locator(this.mainContentSelector)
.boundingBox();

const navbarBoundingBox = await this.page
.locator(this.navbarSelector)
.boundingBox();

return new MobileBudgetPage(this.page);
await this.page.dragAndDrop(this.navbarSelector, this.mainContentSelector, {
sourcePosition: { x: 1, y: 0 },
targetPosition: {
x: 1,
y: mainContentBoundingBox.height - navbarBoundingBox.height,
},
});
}

async goToAccountsPage() {
const link = this.page.getByRole('link', { name: 'Accounts' });
await link.click();
async dragNavbarDown() {
const boundingBox = await this.page
.locator(this.navbarSelector)
.boundingBox();

return new MobileAccountsPage(this.page);
await this.page.dragAndDrop(this.navbarSelector, this.navbarSelector, {
sourcePosition: { x: 1, y: 0 },
targetPosition: {
x: 1,
// Only scroll until bottom of screen i.e. bottom of first navbar row.
y: boundingBox.height / MobileNavigation.#NAVBAR_ROWS,
},
});
Copy link
Contributor

Choose a reason for hiding this comment

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

⚠️ Potential issue

Address test flakiness concerns

Given the reported flaky tests, the drag operations need additional stability measures:

  1. Add explicit waits for animations to complete
  2. Verify element positions after drag operations
  3. Implement retry logic for failed drag operations
 async dragNavbarUp() {
+    const maxRetries = 3;
+    for (let i = 0; i < maxRetries; i++) {
+      try {
         const mainContentBoundingBox = await this.page
           .locator(this.mainContentSelector)
           .boundingBox();
         
         const navbarBoundingBox = await this.page
           .locator(this.navbarSelector)
           .boundingBox();
 
         await this.page.dragAndDrop(this.navbarSelector, this.mainContentSelector, {
           sourcePosition: { x: 1, y: 0 },
           targetPosition: {
             x: 1,
             y: mainContentBoundingBox.height - navbarBoundingBox.height,
           },
         });
 
+        // Verify the drag was successful
+        await this.page.waitForTimeout(100); // Wait for any animations
+        const newPosition = await this.page
+          .locator(this.navbarSelector)
+          .boundingBox();
+        
+        if (Math.abs(newPosition.y - (mainContentBoundingBox.height - navbarBoundingBox.height)) < 5) {
+          return; // Drag successful
+        }
+      } catch (error) {
+        if (i === maxRetries - 1) throw error;
+        await this.page.waitForTimeout(100); // Wait before retry
+      }
+    }
 }

Committable suggestion skipped: line range outside the PR's diff.

@matt-fidd
Copy link
Contributor

Finally got a chance to go through this and do a code review, no problems to comment on (other than the flaky test that's been mentioned)

Happy to approve once the conflicts are resolved and the flaky tests are made stable

@matt-fidd
Copy link
Contributor

/update-vrt

@joel-jeremy joel-jeremy force-pushed the mobile-budget-menu-modal-vrt branch from 75da4cc to 41e771d Compare December 10, 2024 19:52
Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

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

Actionable comments posted: 3

🧹 Outside diff range and nitpick comments (5)
packages/desktop-client/src/components/mobile/MobileNavTabs.tsx (2)

Line range hint 52-87: Consider synchronizing state updates with animations

The state updates and animations are currently separate operations. In React 18's concurrent rendering, this could potentially lead to visual inconsistencies if the state update and animation get out of sync.

Consider batching the updates using React's flushSync or moving the animation into a useLayoutEffect:

 const openFull = useCallback(
   ({ canceled }: { canceled?: boolean }) => {
+    // Ensure state and animation update together
+    flushSync(() => {
       setNavbarState('open');
+    });
     api.start({
       y: OPEN_FULL_Y,
       immediate: false,
       config: canceled ? config.wobbly : config.stiff,
     });
   },
-  [api, OPEN_FULL_Y],
+  [api, OPEN_FULL_Y]
 );

Don't forget to import flushSync:

+ import { flushSync } from 'react-dom';

211-211: Consider enhancing accessibility with ARIA attributes

While the data-navbar-state attribute is good for testing, consider adding ARIA attributes to improve accessibility.

 data-navbar-state={navbarState}
+aria-expanded={navbarState === 'open'}
+aria-hidden={navbarState === 'hidden'}
packages/desktop-client/e2e/page-models/mobile-navigation.js (3)

1-2: Fix import order

The @playwright/test import should occur before the local imports to maintain consistency.

Apply this diff:

+import { expect } from '@playwright/test';
+
 import { MobileAccountPage } from './mobile-account-page';
-import { expect } from '@playwright/test';
🧰 Tools
🪛 GitHub Check: lint

[warning] 1-1:
There should be at least one empty line between import groups


[warning] 2-2:
@playwright/test import should occur before import of ./mobile-account-page


13-16: Consider adding null checks and error handling in constructor

The constructor initializes critical selectors and elements. Consider adding null checks and error handling to ensure robustness.

Apply this diff:

   constructor(page) {
+    if (!page) {
+      throw new Error('Page object is required');
+    }
     this.page = page;
     this.heading = page.getByRole('heading');
     this.navbar = page.getByRole('navigation');
     this.mainContentSelector = '[role=main]';
     this.navbarSelector = '[role=navigation]';
+
+    // Validate critical elements
+    if (!this.navbar) {
+      throw new Error('Navigation element not found');
+    }
   }

80-110: LGTM! Well-structured navigation with suggestion for timeout handling

The navigateToPage method effectively centralizes navigation logic. Consider adding timeout handling for better reliability.

Apply this diff to add timeout handling:

   async navigateToPage(pageName, pageModelFactory) {
+    const timeout = 30000; // 30 seconds
     const pageInstance = pageModelFactory();

     if (this.page.url().endsWith(MobileNavigation.#ROUTES_BY_PAGE[pageName])) {
       return pageInstance;
     }

-    await this.navbar.waitFor();
+    await this.navbar.waitFor({ timeout });

     const navbarStates = MobileNavigation.#NAV_LINKS_HIDDEN_BY_DEFAULT.includes(
       pageName,
     )
       ? ['default', 'hidden']
       : ['hidden'];

     if (await this.hasNavbarState(...navbarStates)) {
       await this.dragNavbarUp();
     }

     const link = this.navbar.getByRole('link', { name: pageName });
-    await link.click();
+    await Promise.race([
+      link.click(),
+      new Promise((_, reject) =>
+        setTimeout(() => reject(new Error(`Navigation to ${pageName} timed out`)), timeout)
+      ),
+    ]);

-    await pageInstance.waitFor();
+    await pageInstance.waitFor({ timeout });

     if (await this.hasNavbarState('open')) {
       await this.dragNavbarDown();
     }

     return pageInstance;
   }
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Reviewing files that changed from the base of the PR and between 75da4cc and 41e771d.

⛔ Files ignored due to path filters (36)
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-applies-budget-template-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-applies-budget-template-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-applies-budget-template-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-copies-last-month-s-budget-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-copies-last-month-s-budget-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-copies-last-month-s-budget-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-12-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-12-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-12-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-3-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-3-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-3-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-6-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-6-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-6-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-updates-the-budgeted-amount-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-updates-the-budgeted-amount-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-updates-the-budgeted-amount-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-applies-budget-template-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-applies-budget-template-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-applies-budget-template-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-copies-last-month-s-budget-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-copies-last-month-s-budget-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-copies-last-month-s-budget-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-12-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-12-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-12-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-3-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-3-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-3-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-6-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-6-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-6-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-updates-the-budgeted-amount-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-updates-the-budgeted-amount-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-updates-the-budgeted-amount-3-chromium-linux.png is excluded by !**/*.png
📒 Files selected for processing (24)
  • packages/desktop-client/e2e/budget.mobile.test.js (4 hunks)
  • packages/desktop-client/e2e/budget.test.js (1 hunks)
  • packages/desktop-client/e2e/page-models/account-page.js (1 hunks)
  • packages/desktop-client/e2e/page-models/close-account-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-accounts-page.js (3 hunks)
  • packages/desktop-client/e2e/page-models/mobile-balance-menu-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-budget-menu-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-budget-page.js (7 hunks)
  • packages/desktop-client/e2e/page-models/mobile-category-menu-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-edit-notes-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-envelope-budget-summary-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-navigation.js (2 hunks)
  • packages/desktop-client/e2e/page-models/mobile-reports-page.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-tracking-budget-summary-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-transaction-entry-page.js (2 hunks)
  • packages/desktop-client/e2e/page-models/settings-page.js (1 hunks)
  • packages/desktop-client/src/components/Notifications.tsx (1 hunks)
  • packages/desktop-client/src/components/Page.tsx (1 hunks)
  • packages/desktop-client/src/components/mobile/MobileNavTabs.tsx (7 hunks)
  • packages/desktop-client/src/components/mobile/accounts/Accounts.tsx (1 hunks)
  • packages/desktop-client/src/components/mobile/transactions/FocusableAmountInput.tsx (1 hunks)
  • packages/desktop-client/src/components/mobile/transactions/TransactionEdit.jsx (2 hunks)
  • packages/desktop-client/src/components/reports/Overview.tsx (1 hunks)
  • packages/desktop-client/src/components/settings/index.tsx (1 hunks)
✅ Files skipped from review due to trivial changes (1)
  • packages/desktop-client/src/components/mobile/accounts/Accounts.tsx
🚧 Files skipped from review as they are similar to previous changes (16)
  • packages/desktop-client/src/components/Notifications.tsx
  • packages/desktop-client/src/components/settings/index.tsx
  • packages/desktop-client/e2e/page-models/mobile-reports-page.js
  • packages/desktop-client/e2e/page-models/mobile-envelope-budget-summary-modal.js
  • packages/desktop-client/e2e/page-models/mobile-tracking-budget-summary-modal.js
  • packages/desktop-client/e2e/budget.test.js
  • packages/desktop-client/e2e/page-models/mobile-balance-menu-modal.js
  • packages/desktop-client/e2e/page-models/mobile-edit-notes-modal.js
  • packages/desktop-client/e2e/page-models/mobile-category-menu-modal.js
  • packages/desktop-client/e2e/page-models/account-page.js
  • packages/desktop-client/src/components/mobile/transactions/FocusableAmountInput.tsx
  • packages/desktop-client/e2e/page-models/close-account-modal.js
  • packages/desktop-client/e2e/page-models/mobile-accounts-page.js
  • packages/desktop-client/src/components/reports/Overview.tsx
  • packages/desktop-client/src/components/Page.tsx
  • packages/desktop-client/src/components/mobile/transactions/TransactionEdit.jsx
👮 Files not reviewed due to content moderation or server errors (2)
  • packages/desktop-client/e2e/page-models/mobile-budget-page.js
  • packages/desktop-client/e2e/budget.mobile.test.js
🧰 Additional context used
📓 Learnings (1)
packages/desktop-client/e2e/budget.mobile.test.js (1)
Learnt from: matt-fidd
PR: actualbudget/actual#3583
File: packages/desktop-client/e2e/budget.mobile.test.js:33-59
Timestamp: 2024-11-12T12:03:19.523Z
Learning: In the `setBudgetAverage` function in `budget.mobile.test.js`, the `spentButton` retrieval is correctly placed inside the loop.
🪛 GitHub Check: lint
packages/desktop-client/src/components/mobile/MobileNavTabs.tsx

[warning] 40-40:
Replace ⏎····'default'·|·'open'·|·'hidden' with 'default'·|·'open'·|·'hidden'>(


[warning] 42-42:
Replace >('default' with ··'default',⏎··

packages/desktop-client/e2e/page-models/mobile-navigation.js

[warning] 2-2:
@playwright/test import should occur before import of ./mobile-account-page

🔇 Additional comments (11)
packages/desktop-client/src/components/mobile/MobileNavTabs.tsx (3)

40-42: LGTM! Well-structured state management

The navbar state implementation with TypeScript union types provides good type safety and clear state management.

🧰 Tools
🪛 GitHub Check: lint

[warning] 40-40:
Replace ⏎····'default'·|·'open'·|·'hidden' with 'default'·|·'open'·|·'hidden'>(


[warning] 42-42:
Replace >('default' with ··'default',⏎··


265-265: LGTM! Good mobile UX enhancement

Adding userSelect: 'none' improves the mobile experience by preventing accidental text selection during swipe interactions.


Line range hint 1-265: Verify animation performance on low-end devices

The component uses complex animations and gesture interactions. Consider monitoring performance metrics to ensure smooth operation on low-end mobile devices.

packages/desktop-client/e2e/page-models/mobile-transaction-entry-page.js (2)

7-12: LGTM! Well-structured element locators.

The hierarchical organization of elements using data-testid attributes and proper role selectors is a good practice for reliable test automation.


15-16: ⚠️ Potential issue

Enhance wait strategy to improve test stability.

Given the reported test flakiness, consider implementing a more robust wait strategy that ensures all form elements are ready for interaction.

  async waitFor(options) {
-   await this.transactionForm.waitFor(options);
+   await this.transactionForm.waitFor({ state: 'visible', ...options });
+   await this.amountField.waitFor({ state: 'visible' });
+   await this.addTransactionButton.waitFor({ state: 'enabled' });
+   // Wait for any loading indicators to disappear
+   await this.page.waitForSelector('[data-testid="loading-indicator"]', 
+     { state: 'hidden', timeout: 5000 }).catch(() => {});
  }
packages/desktop-client/e2e/page-models/settings-page.js (1)

4-15: LGTM! Well-organized element locators.

Good use of data-testid attributes and role selectors for reliable element selection.

packages/desktop-client/e2e/page-models/mobile-budget-menu-modal.js (2)

1-23: LGTM! Well-structured modal class with clear element organization.

The element locators are well-organized and use appropriate selectors for reliable element identification.


25-53: ⚠️ Potential issue

Implement shared action handler with proper error handling.

To improve test stability and reduce code duplication, implement a shared action handler for modal interactions.

+ async performModalAction(actionName, element, options = {}) {
+   try {
+     await element.waitFor({ state: 'visible' });
+     if (options.fill) {
+       await element.fill(String(options.fill));
+       await element.blur();
+     } else {
+       await element.click();
+     }
+     // Wait for any loading indicators to disappear
+     await this.page.waitForSelector('[data-testid="loading-indicator"]', 
+       { state: 'hidden', timeout: 5000 }).catch(() => {});
+   } catch (error) {
+     throw new Error(`Failed to ${actionName}: ${error.message}`);
+   }
+ }

  async setBudgetAmount(newAmount) {
-   await this.budgetAmountInput.fill(String(newAmount));
-   await this.budgetAmountInput.blur();
-   await this.close();
+   await this.performModalAction('set budget amount', this.budgetAmountInput, { fill: newAmount });
+   await this.close();
  }

  async copyLastMonthBudget() {
-   await this.copyLastMonthBudgetButton.click();
+   await this.performModalAction('copy last month budget', this.copyLastMonthBudgetButton);
  }

  // Apply similar changes to other action methods
packages/desktop-client/e2e/page-models/mobile-navigation.js (3)

19-33: LGTM! Well-structured static configuration

The static private properties are well-organized and properly encapsulated:

  • #NAVBAR_ROWS defines the navigation structure
  • #NAV_LINKS_HIDDEN_BY_DEFAULT clearly specifies hidden navigation items
  • #ROUTES_BY_PAGE maintains a clean mapping of routes

Line range hint 112-151: LGTM! Well-refactored navigation methods

The navigation methods are consistently implemented using the centralized navigateToPage approach, which reduces code duplication and improves maintainability.


53-66: ⚠️ Potential issue

Improve dragNavbarDown reliability and add position verification

Similar to dragNavbarUp, add retry mechanism and position verification to ensure reliable drag operations.

Apply this diff:

   async dragNavbarDown() {
+    const maxRetries = 3;
+    let lastError;
+
+    for (let attempt = 1; attempt <= maxRetries; attempt++) {
+      try {
         const boundingBox = await this.page
           .locator(this.navbarSelector)
           .boundingBox();

         await this.page.dragAndDrop(this.navbarSelector, this.navbarSelector, {
           sourcePosition: { x: 1, y: 0 },
           targetPosition: {
             x: 1,
             y: boundingBox.height / MobileNavigation.#NAVBAR_ROWS,
           },
         });

+        // Verify position after drag
+        await this.page.waitForTimeout(100);
+        const finalPosition = await this.page
+          .locator(this.navbarSelector)
+          .boundingBox();
+
+        if (Math.abs(finalPosition.y - (boundingBox.height / MobileNavigation.#NAVBAR_ROWS)) < 5) {
+          return;
+        }
+      } catch (error) {
+        lastError = error;
+        if (attempt < maxRetries) {
+          await this.page.waitForTimeout(100 * attempt);
+        }
+      }
+    }
+    throw new Error(`Drag operation failed after ${maxRetries} attempts: ${lastError?.message}`);
   }

Comment on lines +27 to +34
await this.enableExperimentalFeature('Budget mode toggle');

await this.switchBudgetTypeButton.waitFor();

await switchBudgetTypeButton.click();
const buttonText = await this.switchBudgetTypeButton.textContent();
if (buttonText.includes(budgetType.toLowerCase())) {
await this.switchBudgetTypeButton.click();
}
Copy link
Contributor

Choose a reason for hiding this comment

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

⚠️ Potential issue

Add error handling to improve test stability.

The methods handling experimental features and budget type changes should include proper error handling.

  async useBudgetType(budgetType) {
+   try {
      await this.enableExperimentalFeature('Budget mode toggle');
      await this.switchBudgetTypeButton.waitFor();
      const buttonText = await this.switchBudgetTypeButton.textContent();
      if (buttonText.includes(budgetType.toLowerCase())) {
        await this.switchBudgetTypeButton.click();
+       // Wait for the change to take effect
+       await this.page.waitForSelector('[data-testid="loading-indicator"]', 
+         { state: 'hidden', timeout: 5000 }).catch(() => {});
      }
+   } catch (error) {
+     throw new Error(`Failed to switch budget type: ${error.message}`);
+   }
  }

  async enableExperimentalFeature(featureName) {
+   try {
      if (await this.advancedSettingsButton.isVisible()) {
        await this.advancedSettingsButton.click();
      }

      if (await this.experimentalSettingsButton.isVisible()) {
        await this.experimentalSettingsButton.click();
      }

      const featureCheckbox = this.page.getByRole('checkbox', {
        name: featureName,
      });
+     await featureCheckbox.waitFor({ state: 'visible' });
      if (!(await featureCheckbox.isChecked())) {
        await featureCheckbox.click();
+       // Verify the checkbox state
+       if (!(await featureCheckbox.isChecked())) {
+         throw new Error('Feature checkbox did not toggle as expected');
+       }
      }
+   } catch (error) {
+     throw new Error(`Failed to enable experimental feature: ${error.message}`);
+   }
  }

Also applies to: 37-51

Comment on lines +18 to +19
async waitFor(options) {
await this.settings.waitFor(options);
Copy link
Contributor

Choose a reason for hiding this comment

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

🛠️ Refactor suggestion

Enhance wait strategy for settings page.

Consider implementing a more comprehensive wait strategy to ensure all settings elements are ready.

  async waitFor(options) {
-   await this.settings.waitFor(options);
+   await this.settings.waitFor({ state: 'visible', ...options });
+   // Wait for any loading indicators to disappear
+   await this.page.waitForSelector('[data-testid="loading-indicator"]', 
+     { state: 'hidden', timeout: 5000 }).catch(() => {});
  }

Committable suggestion skipped: line range outside the PR's diff.

Comment on lines +35 to +50
async dragNavbarUp() {
const mainContentBoundingBox = await this.page
.locator(this.mainContentSelector)
.boundingBox();

return new MobileBudgetPage(this.page);
const navbarBoundingBox = await this.page
.locator(this.navbarSelector)
.boundingBox();

await this.page.dragAndDrop(this.navbarSelector, this.mainContentSelector, {
sourcePosition: { x: 1, y: 0 },
targetPosition: {
x: 1,
y: mainContentBoundingBox.height - navbarBoundingBox.height,
},
});
Copy link
Contributor

Choose a reason for hiding this comment

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

⚠️ Potential issue

Add retry mechanism to handle flaky drag operations

Given the reported test flakiness, the drag operations should include retry logic and position verification.

Apply this diff to improve reliability:

   async dragNavbarUp() {
+    const maxRetries = 3;
+    let lastError;
+
+    for (let attempt = 1; attempt <= maxRetries; attempt++) {
+      try {
         const mainContentBoundingBox = await this.page
           .locator(this.mainContentSelector)
           .boundingBox();

         const navbarBoundingBox = await this.page
           .locator(this.navbarSelector)
           .boundingBox();

         await this.page.dragAndDrop(this.navbarSelector, this.mainContentSelector, {
           sourcePosition: { x: 1, y: 0 },
           targetPosition: {
             x: 1,
             y: mainContentBoundingBox.height - navbarBoundingBox.height,
           },
         });

+        // Verify position after drag
+        await this.page.waitForTimeout(100); // Wait for animations
+        const finalPosition = await this.page
+          .locator(this.navbarSelector)
+          .boundingBox();
+
+        if (Math.abs(finalPosition.y - (mainContentBoundingBox.height - navbarBoundingBox.height)) < 5) {
+          return; // Successful drag
+        }
+      } catch (error) {
+        lastError = error;
+        if (attempt < maxRetries) {
+          await this.page.waitForTimeout(100 * attempt); // Exponential backoff
+        }
+      }
+    }
+    throw new Error(`Drag operation failed after ${maxRetries} attempts: ${lastError?.message}`);
   }
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
async dragNavbarUp() {
const mainContentBoundingBox = await this.page
.locator(this.mainContentSelector)
.boundingBox();
return new MobileBudgetPage(this.page);
const navbarBoundingBox = await this.page
.locator(this.navbarSelector)
.boundingBox();
await this.page.dragAndDrop(this.navbarSelector, this.mainContentSelector, {
sourcePosition: { x: 1, y: 0 },
targetPosition: {
x: 1,
y: mainContentBoundingBox.height - navbarBoundingBox.height,
},
});
async dragNavbarUp() {
const maxRetries = 3;
let lastError;
for (let attempt = 1; attempt <= maxRetries; attempt++) {
try {
const mainContentBoundingBox = await this.page
.locator(this.mainContentSelector)
.boundingBox();
const navbarBoundingBox = await this.page
.locator(this.navbarSelector)
.boundingBox();
await this.page.dragAndDrop(this.navbarSelector, this.mainContentSelector, {
sourcePosition: { x: 1, y: 0 },
targetPosition: {
x: 1,
y: mainContentBoundingBox.height - navbarBoundingBox.height,
},
});
// Verify position after drag
await this.page.waitForTimeout(100); // Wait for animations
const finalPosition = await this.page
.locator(this.navbarSelector)
.boundingBox();
if (Math.abs(finalPosition.y - (mainContentBoundingBox.height - navbarBoundingBox.height)) < 5) {
return; // Successful drag
}
} catch (error) {
lastError = error;
if (attempt < maxRetries) {
await this.page.waitForTimeout(100 * attempt); // Exponential backoff
}
}
}
throw new Error(`Drag operation failed after ${maxRetries} attempts: ${lastError?.message}`);
}

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

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

Actionable comments posted: 0

🧹 Outside diff range and nitpick comments (2)
packages/desktop-client/src/components/mobile/MobileNavTabs.tsx (1)

Line range hint 52-87: Consider improving type safety for openFull parameters

While the callback implementations look good overall, we can improve type safety for the openFull function parameters.

Consider this improvement:

-const openFull = useCallback(
-  ({ canceled }: { canceled?: boolean }) => {
+type OpenFullParams = {
+  canceled?: boolean;
+};
+const openFull = useCallback(
+  ({ canceled }: OpenFullParams) => {

This makes the parameter type more maintainable and reusable if needed elsewhere.

packages/desktop-client/e2e/page-models/mobile-navigation.js (1)

80-110: Enhance error handling in navigateToPage

While the navigation logic is well-structured, it could benefit from better error handling and timeout management.

Apply this diff:

 async navigateToPage(pageName, pageModelFactory) {
+  try {
     const pageInstance = pageModelFactory();

     if (this.page.url().endsWith(MobileNavigation.#ROUTES_BY_PAGE[pageName])) {
       return pageInstance;
     }

-    await this.navbar.waitFor();
+    await this.navbar.waitFor({ timeout: 5000 });

     const navbarStates = MobileNavigation.#NAV_LINKS_HIDDEN_BY_DEFAULT.includes(
       pageName,
     )
       ? ['default', 'hidden']
       : ['hidden'];

     if (await this.hasNavbarState(...navbarStates)) {
       await this.dragNavbarUp();
     }

     const link = this.navbar.getByRole('link', { name: pageName });
-    await link.click();
+    await link.click({ timeout: 5000 });

-    await pageInstance.waitFor();
+    await pageInstance.waitFor({ timeout: 5000 });

     if (await this.hasNavbarState('open')) {
       await this.dragNavbarDown();
     }

     return pageInstance;
+  } catch (error) {
+    throw new Error(`Failed to navigate to ${pageName}: ${error.message}`);
+  }
 }
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Reviewing files that changed from the base of the PR and between 41e771d and 2d401c8.

📒 Files selected for processing (2)
  • packages/desktop-client/e2e/page-models/mobile-navigation.js (2 hunks)
  • packages/desktop-client/src/components/mobile/MobileNavTabs.tsx (7 hunks)
🔇 Additional comments (10)
packages/desktop-client/src/components/mobile/MobileNavTabs.tsx (4)

5-6: LGTM: Well-implemented state management

The navbar state implementation with TypeScript types and proper initial value looks good. This addresses the previous concerns about undefined state.

Also applies to: 40-42


211-211: LGTM: Good addition of data attribute for testing

The data-navbar-state attribute is a good addition that will help with state verification and testing.


265-265: LGTM: Good UX improvement

Adding userSelect: 'none' is a good improvement that prevents text selection during drag interactions on mobile.


Line range hint 40-87: Verify animation performance on lower-end devices

The spring animations combined with scroll and drag interactions might impact performance on lower-end devices. Consider testing on various device tiers to ensure smooth interactions.

packages/desktop-client/e2e/page-models/mobile-navigation.js (6)

1-16: LGTM! Well-structured initialization with accessibility-first selectors

The constructor properly initializes role-based selectors that follow accessibility best practices, making the tests more resilient to UI changes.


19-33: LGTM! Well-structured static configuration

Good use of private static fields to encapsulate configuration data. The constants are logically organized and protected from external modification.


68-78: LGTM! Well-implemented state checking

The hasNavbarState method effectively handles edge cases and uses proper assertions.


Line range hint 112-151: LGTM! Well-refactored navigation methods

The navigation methods effectively utilize the centralized navigateToPage helper, reducing code duplication and maintaining consistency across all page transitions.


53-66: ⚠️ Potential issue

Improve dragNavbarDown reliability

Using the same element as both source and target in dragAndDrop might be unreliable. Consider using a point below the navbar as the target.

Apply this diff:

 async dragNavbarDown() {
   const boundingBox = await this.page
     .locator(this.navbarSelector)
     .boundingBox();

-  await this.page.dragAndDrop(this.navbarSelector, this.navbarSelector, {
+  await this.page.dragAndDrop(this.navbarSelector, this.mainContentSelector, {
     sourcePosition: { x: 1, y: 0 },
     targetPosition: {
       x: 1,
-      y: boundingBox.height / MobileNavigation.#NAVBAR_ROWS,
+      y: boundingBox.height + 10, // Point below navbar
     },
   });
+
+  // Verify final position
+  await this.page.waitForTimeout(100); // Wait for animations
+  const finalPosition = await this.page
+    .locator(this.navbarSelector)
+    .boundingBox();
+  
+  if (Math.abs(finalPosition.y - boundingBox.height) > 5) {
+    throw new Error('Navbar did not reach expected position after drag');
+  }
 }

Likely invalid or redundant comment.


35-50: ⚠️ Potential issue

Add stability measures to dragNavbarUp

Given the reported test flakiness, the drag operation needs additional reliability measures:

  1. Add position verification after drag
  2. Implement retry mechanism
  3. Add explicit waits for animations

Apply this diff:

 async dragNavbarUp() {
+  const maxRetries = 3;
+  let lastError;
+
+  for (let attempt = 1; attempt <= maxRetries; attempt++) {
+    try {
       const mainContentBoundingBox = await this.page
         .locator(this.mainContentSelector)
         .boundingBox();

       const navbarBoundingBox = await this.page
         .locator(this.navbarSelector)
         .boundingBox();

       await this.page.dragAndDrop(this.navbarSelector, this.mainContentSelector, {
         sourcePosition: { x: 1, y: 0 },
         targetPosition: {
           x: 1,
           y: mainContentBoundingBox.height - navbarBoundingBox.height,
         },
       });

+      // Verify position after drag
+      await this.page.waitForTimeout(100); // Wait for animations
+      const finalPosition = await this.page
+        .locator(this.navbarSelector)
+        .boundingBox();
+
+      const expectedY = mainContentBoundingBox.height - navbarBoundingBox.height;
+      if (Math.abs(finalPosition.y - expectedY) < 5) {
+        return; // Successful drag
+      }
+    } catch (error) {
+      lastError = error;
+      if (attempt < maxRetries) {
+        await this.page.waitForTimeout(100 * attempt); // Exponential backoff
+      }
+    }
+  }
+  throw new Error(`Drag operation failed after ${maxRetries} attempts: ${lastError?.message}`);
 }

Likely invalid or redundant comment.

@matt-fidd
Copy link
Contributor

It doesn't seem too happy and it's definitely getting stuck somewhere, run time is up from circa 5 minutes to 35

https://github.com/actualbudget/actual/actions/runs/12263780733/job/34220238485

Copy link
Contributor

This PR is stale because it has been open 30 days with no activity. Remove stale label or comment or this will be closed in 5 days.

@joel-jeremy joel-jeremy force-pushed the mobile-budget-menu-modal-vrt branch from 2d401c8 to a4f45cc Compare January 10, 2025 18:16
Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

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

Actionable comments posted: 3

♻️ Duplicate comments (2)
packages/desktop-client/e2e/page-models/mobile-transaction-entry-page.js (2)

15-16: ⚠️ Potential issue

Enhance wait strategy to improve test stability.

Given the reported test flakiness in the PR objectives, the current wait strategy might not be sufficient.

Consider implementing a more comprehensive wait strategy that ensures the form is not just present but also interactive:

  async waitFor(options) {
-   await this.transactionForm.waitFor(options);
+   await this.transactionForm.waitFor({ state: 'visible', ...options });
+   await this.amountField.waitFor({ state: 'visible' });
+   await this.addTransactionButton.waitFor({ state: 'enabled' });
  }

26-26: ⚠️ Potential issue

Add navigation wait to prevent test timeouts.

The missing navigation wait could be contributing to the reported increase in test run times (from 5 to 35 minutes).

Add an explicit wait for navigation after clicking the add button:

  async createTransaction() {
    await this.addTransactionButton.click();
+   await this.page.waitForNavigation({ waitUntil: 'networkidle' });
    return new MobileAccountPage(this.page);
  }
🧹 Nitpick comments (8)
packages/desktop-client/e2e/page-models/mobile-accounts-page.js (3)

8-8: LGTM! Consider adding JSDoc for the new property.

The change follows good E2E testing practices by using test IDs and proper element scoping.

Add JSDoc to document the purpose and structure of accountListItems:

+  /** @type {Locator} List of account items within the account list */
   this.accountListItems = this.accountList.getByTestId('account-list-item');

31-31: Consider adding stability measures to prevent flaky tests.

Given the reported test stability issues in the PR, consider adding explicit waits or retry mechanisms:

-    await this.accountListItems.nth(idx).click();
+    const accountItem = this.accountListItems.nth(idx);
+    await accountItem.waitFor({ state: 'visible' });
+    await accountItem.click();

Line range hint 1-35: Consider global stability improvements for the page model.

Given the test stability issues mentioned in the PR, consider these improvements:

  1. Add retry options to critical interactions
  2. Implement custom timeout configurations
  3. Add error handling for common failure scenarios

Example implementation:

class MobileAccountsPage {
  constructor(page) {
    this.page = page;
    this.defaultTimeout = 30000; // Configurable timeout
    this.retryOptions = { timeout: this.defaultTimeout, retries: 3 };
    
    this.accountList = this.page.getByLabel('Account list');
    this.accountListItems = this.accountList.getByTestId('account-list-item');
  }

  async retry(action, options = this.retryOptions) {
    let lastError;
    for (let i = 0; i < options.retries; i++) {
      try {
        return await action();
      } catch (error) {
        lastError = error;
        await this.page.waitForTimeout(1000); // Add delay between retries
      }
    }
    throw lastError;
  }
}
packages/desktop-client/e2e/page-models/mobile-transaction-entry-page.js (1)

7-13: LGTM! Well-structured element organization.

The use of test IDs and ARIA roles for element selection improves test reliability. The hierarchical organization (form → footer → button) is clean and maintainable.

Consider adding timeout options to element locators for better control over test timing:

-    this.transactionForm = page.getByTestId('transaction-form');
+    this.transactionForm = page.getByTestId('transaction-form', { timeout: 10000 });
packages/desktop-client/e2e/page-models/mobile-navigation.js (1)

68-110: Add debug logging to help troubleshoot flaky tests

The navigation logic is well-structured, but adding debug logs would help troubleshoot the reported test stability issues.

Apply this diff:

  async navigateToPage(pageName, pageModelFactory) {
+   console.log(`Navigating to ${pageName} page...`);
    const pageInstance = pageModelFactory();

    if (this.page.url().endsWith(MobileNavigation.#ROUTES_BY_PAGE[pageName])) {
+     console.log(`Already on ${pageName} page`);
      return pageInstance;
    }

    await this.navbar.waitFor();
    const navbarStates = MobileNavigation.#NAV_LINKS_HIDDEN_BY_DEFAULT.includes(
      pageName,
    )
      ? ['default', 'hidden']
      : ['hidden'];

+   console.log(`Current navbar state: ${await this.navbar.getAttribute('data-navbar-state')}`);
    if (await this.hasNavbarState(...navbarStates)) {
+     console.log('Dragging navbar up...');
      await this.dragNavbarUp();
    }

    const link = this.navbar.getByRole('link', { name: pageName });
    await link.click();
+   console.log(`Clicked ${pageName} link`);

    await pageInstance.waitFor();

    if (await this.hasNavbarState('open')) {
+     console.log('Dragging navbar down...');
      await this.dragNavbarDown();
    }

+   console.log(`Navigation to ${pageName} complete`);
    return pageInstance;
  }
packages/desktop-client/e2e/page-models/mobile-budget-page.js (1)

147-148: Refactor modal instantiation to reduce code duplication.

The instantiation of modal classes with similar parameters is repeated across multiple methods.

+ #createModal(ModalClass) {
+   return new ModalClass(this.page, this.page.getByRole('dialog'));
+ }

  async openCategoryMenu(categoryName) {
    const categoryButton = await this.#getButtonForCategory(categoryName);
    await categoryButton.click();
-   return new CategoryMenuModal(this.page, this.page.getByRole('dialog'));
+   return this.#createModal(CategoryMenuModal);
  }

  async openBudgetMenu(categoryName) {
    const budgetedButton = await this.getButtonForBudgeted(categoryName);
    await budgetedButton.click();
-   return new BudgetMenuModal(this.page, this.page.getByRole('dialog'));
+   return this.#createModal(BudgetMenuModal);
  }

  async openBalanceMenu(categoryName) {
    const balanceButton = this.budgetTable.getByRole('button', {
      name: `Open balance menu for ${categoryName} category`,
    });
    if (await balanceButton.isVisible()) {
      await balanceButton.click();
-     return new BalanceMenuModal(this.page, this.page.getByRole('dialog'));
+     return this.#createModal(BalanceMenuModal);
    } else {
      throw new Error(
        `Balance button for category ${categoryName} not found or not visible`,
      );
    }
  }

  async openEnvelopeBudgetSummary() {
    const budgetSummaryButton = await this.#getButtonForEnvelopeBudgetSummary();
    await budgetSummaryButton.click();
-   return new EnvelopeBudgetSummaryModal(
-     this.page,
-     this.page.getByRole('dialog'),
-   );
+   return this.#createModal(EnvelopeBudgetSummaryModal);
  }

  async openTrackingBudgetSummary() {
    const budgetSummaryButton = await this.#getButtonForTrackingBudgetSummary();
    await budgetSummaryButton.click();
-   return new TrackingBudgetSummaryModal(
-     this.page,
-     this.page.getByRole('dialog'),
-   );
+   return this.#createModal(TrackingBudgetSummaryModal);
  }

Also applies to: 187-187, 204-204, 280-283, 312-315

packages/desktop-client/e2e/budget.mobile.test.js (2)

9-31: Refactor to reduce code duplication in budget utility functions.

The utility functions share identical structures.

+ const performBudgetMenuAction = async (budgetPage, categoryName, action) => {
+   const budgetMenuModal = await budgetPage.openBudgetMenu(categoryName);
+   await action(budgetMenuModal);
+   await budgetMenuModal.close();
+ };

- const copyLastMonthBudget = async (budgetPage, categoryName) => {
-   const budgetMenuModal = await budgetPage.openBudgetMenu(categoryName);
-   await budgetMenuModal.copyLastMonthBudget();
-   await budgetMenuModal.close();
- };
+ const copyLastMonthBudget = (budgetPage, categoryName) =>
+   performBudgetMenuAction(budgetPage, categoryName, menu => menu.copyLastMonthBudget());

- const setTo3MonthAverage = async (budgetPage, categoryName) => {
-   const budgetMenuModal = await budgetPage.openBudgetMenu(categoryName);
-   await budgetMenuModal.setTo3MonthAverage();
-   await budgetMenuModal.close();
- };
+ const setTo3MonthAverage = (budgetPage, categoryName) =>
+   performBudgetMenuAction(budgetPage, categoryName, menu => menu.setTo3MonthAverage());

- const setTo6MonthAverage = async (budgetPage, categoryName) => {
-   const budgetMenuModal = await budgetPage.openBudgetMenu(categoryName);
-   await budgetMenuModal.setTo6MonthAverage();
-   await budgetMenuModal.close();
- };
+ const setTo6MonthAverage = (budgetPage, categoryName) =>
+   performBudgetMenuAction(budgetPage, categoryName, menu => menu.setTo6MonthAverage());

- const setToYearlyAverage = async (budgetPage, categoryName) => {
-   const budgetMenuModal = await budgetPage.openBudgetMenu(categoryName);
-   await budgetMenuModal.setToYearlyAverage();
-   await budgetMenuModal.close();
- };
+ const setToYearlyAverage = (budgetPage, categoryName) =>
+   performBudgetMenuAction(budgetPage, categoryName, menu => menu.setToYearlyAverage());

254-298: Add test data isolation.

The tests rely on the same category row (index 3) which could lead to test interference.

+ const TEST_CATEGORIES = {
+   COPY_LAST_MONTH: 3,
+   THREE_MONTH_AVERAGE: 4,
+   SIX_MONTH_AVERAGE: 5,
+   YEARLY_AVERAGE: 6,
+ };

  test(`copies last month's budget`, async () => {
    const budgetPage = await navigation.goToBudgetPage();
-   const categoryName = await budgetPage.getCategoryNameForRow(3);
+   const categoryName = await budgetPage.getCategoryNameForRow(TEST_CATEGORIES.COPY_LAST_MONTH);
    // ... rest of the test
  });

  [
-   [3, setTo3MonthAverage],
-   [6, setTo6MonthAverage],
-   [12, setToYearlyAverage],
+   [3, setTo3MonthAverage, TEST_CATEGORIES.THREE_MONTH_AVERAGE],
+   [6, setTo6MonthAverage, TEST_CATEGORIES.SIX_MONTH_AVERAGE],
+   [12, setToYearlyAverage, TEST_CATEGORIES.YEARLY_AVERAGE],
  ].forEach(([numberOfMonths, setBudgetAverageFn, categoryIndex]) => {
    test(`set budget to ${numberOfMonths} month average`, async () => {
      const budgetPage = await navigation.goToBudgetPage();
-     const categoryName = await budgetPage.getCategoryNameForRow(3);
+     const categoryName = await budgetPage.getCategoryNameForRow(categoryIndex);
      // ... rest of the test
    });
  });
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro

📥 Commits

Reviewing files that changed from the base of the PR and between 2d401c8 and a4f45cc.

⛔ Files ignored due to path filters (33)
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-applies-budget-template-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-applies-budget-template-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-applies-budget-template-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-copies-last-month-s-budget-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-copies-last-month-s-budget-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-copies-last-month-s-budget-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-12-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-12-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-12-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-3-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-3-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-3-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-6-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-6-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-set-budget-to-6-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-updates-the-budgeted-amount-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-updates-the-budgeted-amount-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Envelope-updates-the-budgeted-amount-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-applies-budget-template-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-applies-budget-template-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-applies-budget-template-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-copies-last-month-s-budget-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-copies-last-month-s-budget-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-copies-last-month-s-budget-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-12-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-12-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-12-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-3-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-3-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-3-month-average-3-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-6-month-average-1-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-6-month-average-2-chromium-linux.png is excluded by !**/*.png
  • packages/desktop-client/e2e/budget.mobile.test.js-snapshots/Mobile-Budget-Tracking-set-budget-to-6-month-average-3-chromium-linux.png is excluded by !**/*.png
📒 Files selected for processing (21)
  • packages/desktop-client/e2e/budget.mobile.test.js (4 hunks)
  • packages/desktop-client/e2e/page-models/account-page.js (1 hunks)
  • packages/desktop-client/e2e/page-models/close-account-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-accounts-page.js (3 hunks)
  • packages/desktop-client/e2e/page-models/mobile-balance-menu-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-budget-menu-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-budget-page.js (7 hunks)
  • packages/desktop-client/e2e/page-models/mobile-category-menu-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-edit-notes-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-envelope-budget-summary-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-navigation.js (2 hunks)
  • packages/desktop-client/e2e/page-models/mobile-reports-page.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-tracking-budget-summary-modal.js (1 hunks)
  • packages/desktop-client/e2e/page-models/mobile-transaction-entry-page.js (2 hunks)
  • packages/desktop-client/e2e/page-models/settings-page.js (1 hunks)
  • packages/desktop-client/src/components/Notifications.tsx (1 hunks)
  • packages/desktop-client/src/components/Page.tsx (1 hunks)
  • packages/desktop-client/src/components/mobile/MobileNavTabs.tsx (7 hunks)
  • packages/desktop-client/src/components/mobile/accounts/Accounts.tsx (1 hunks)
  • packages/desktop-client/src/components/mobile/transactions/FocusableAmountInput.tsx (1 hunks)
  • packages/desktop-client/src/components/mobile/transactions/TransactionEdit.jsx (2 hunks)
🚧 Files skipped from review as they are similar to previous changes (14)
  • packages/desktop-client/src/components/Page.tsx
  • packages/desktop-client/e2e/page-models/mobile-envelope-budget-summary-modal.js
  • packages/desktop-client/e2e/page-models/mobile-balance-menu-modal.js
  • packages/desktop-client/src/components/Notifications.tsx
  • packages/desktop-client/src/components/mobile/accounts/Accounts.tsx
  • packages/desktop-client/e2e/page-models/account-page.js
  • packages/desktop-client/e2e/page-models/mobile-tracking-budget-summary-modal.js
  • packages/desktop-client/e2e/page-models/close-account-modal.js
  • packages/desktop-client/e2e/page-models/mobile-reports-page.js
  • packages/desktop-client/e2e/page-models/mobile-edit-notes-modal.js
  • packages/desktop-client/src/components/mobile/transactions/TransactionEdit.jsx
  • packages/desktop-client/e2e/page-models/mobile-category-menu-modal.js
  • packages/desktop-client/src/components/mobile/transactions/FocusableAmountInput.tsx
  • packages/desktop-client/src/components/mobile/MobileNavTabs.tsx
🧰 Additional context used
📓 Learnings (1)
packages/desktop-client/e2e/budget.mobile.test.js (1)
Learnt from: matt-fidd
PR: actualbudget/actual#3583
File: packages/desktop-client/e2e/budget.mobile.test.js:33-59
Timestamp: 2024-11-12T12:03:19.523Z
Learning: In the `setBudgetAverage` function in `budget.mobile.test.js`, the `spentButton` retrieval is correctly placed inside the loop.
⏰ Context from checks skipped due to timeout of 90000ms (5)
  • GitHub Check: Functional
  • GitHub Check: Visual regression
  • GitHub Check: build (macos-latest)
  • GitHub Check: build (windows-latest)
  • GitHub Check: build (ubuntu-latest)
🔇 Additional comments (12)
packages/desktop-client/e2e/page-models/mobile-accounts-page.js (1)

19-19: LGTM! Clean refactor maintaining functionality.

The change consistently uses the new accountListItems property while preserving the method's interface.

packages/desktop-client/e2e/page-models/mobile-transaction-entry-page.js (1)

Line range hint 1-29: Consider comprehensive test stability improvements.

Given the reported test flakiness and increased run times:

  1. Run these tests in isolation first to verify if stability improves
  2. Add logging to track timing of key operations
  3. Consider implementing retry mechanisms for flaky operations

Run the following script to analyze test execution patterns:

packages/desktop-client/e2e/page-models/mobile-navigation.js (4)

1-16: Well-structured initialization with accessibility-first selectors!

Good use of role-based selectors for heading and navigation elements, which aligns with accessibility best practices and makes the tests more maintainable.


19-33: Well-organized static constants with proper encapsulation!

Good use of private static fields (#) for constants. The organization of navigation-related constants improves maintainability and readability.


Line range hint 112-151: Clean and consistent navigation methods!

Great refactoring to use the centralized navigateToPage method. This reduces code duplication and maintains consistency across all navigation operations.


35-66: ⚠️ Potential issue

Add retry mechanism to improve drag operation stability

Given the reported test flakiness issues, the drag operations should include:

  1. Retry logic for failed operations
  2. Position verification after drag
  3. Explicit waits for animations

Apply this diff to improve reliability:

  async dragNavbarUp() {
+   const maxRetries = 3;
+   let lastError;
+
+   for (let attempt = 1; attempt <= maxRetries; attempt++) {
+     try {
        const mainContentBoundingBox = await this.page
          .locator(this.mainContentSelector)
          .boundingBox();

        const navbarBoundingBox = await this.page
          .locator(this.navbarSelector)
          .boundingBox();

        await this.page.dragAndDrop(this.navbarSelector, this.mainContentSelector, {
          sourcePosition: { x: 1, y: 0 },
          targetPosition: {
            x: 1,
            y: mainContentBoundingBox.height - navbarBoundingBox.height,
          },
        });

+       // Verify position after drag
+       await this.page.waitForTimeout(100); // Wait for animations
+       const finalPosition = await this.page
+         .locator(this.navbarSelector)
+         .boundingBox();
+
+       const expectedY = mainContentBoundingBox.height - navbarBoundingBox.height;
+       if (Math.abs(finalPosition.y - expectedY) < 5) {
+         return; // Successful drag
+       }
+     } catch (error) {
+       lastError = error;
+       if (attempt < maxRetries) {
+         await this.page.waitForTimeout(100 * attempt); // Exponential backoff
+         continue;
+       }
+       throw new Error(`Drag operation failed after ${maxRetries} attempts: ${lastError?.message}`);
+     }
+   }
+   throw new Error(`Drag operation failed: navbar did not reach expected position`);
  }

Apply similar changes to dragNavbarDown method.

Likely invalid or redundant comment.

packages/desktop-client/e2e/page-models/mobile-budget-menu-modal.js (2)

1-23: LGTM! Well-structured class initialization.

The class is well-organized with clear property initialization in the constructor. The use of test IDs and role-based selectors follows good testing practices.


25-53: 🛠️ Refactor suggestion

Add wait conditions and error handling to improve test stability.

Given the reported flaky tests in the PR objectives, enhance the modal interaction methods with proper wait conditions and error handling.

  async close() {
+   try {
+     await this.heading.waitFor({ state: 'visible' });
      await this.heading.getByRole('button', { name: 'Close' }).click();
+     await this.heading.waitFor({ state: 'hidden', timeout: 5000 });
+   } catch (error) {
+     throw new Error(`Failed to close modal: ${error.message}`);
+   }
  }

  async setBudgetAmount(newAmount) {
+   try {
+     await this.budgetAmountInput.waitFor({ state: 'visible' });
      await this.budgetAmountInput.fill(String(newAmount));
      await this.budgetAmountInput.blur();
+     // Verify the input value
+     const actualValue = await this.budgetAmountInput.inputValue();
+     if (actualValue !== String(newAmount)) {
+       throw new Error(`Budget amount not set correctly. Expected: ${newAmount}, Got: ${actualValue}`);
+     }
      await this.close();
+   } catch (error) {
+     throw new Error(`Failed to set budget amount: ${error.message}`);
+   }
  }

+ async #performAction(button, actionName) {
+   try {
+     await button.waitFor({ state: 'visible' });
+     await button.click();
+     // Wait for any loading indicators to disappear
+     await this.page.waitForSelector('[data-testid="loading-indicator"]', 
+       { state: 'hidden', timeout: 5000 }).catch(() => {});
+   } catch (error) {
+     throw new Error(`Failed to ${actionName}: ${error.message}`);
+   }
+ }

  async copyLastMonthBudget() {
-   await this.copyLastMonthBudgetButton.click();
+   await this.#performAction(this.copyLastMonthBudgetButton, 'copy last month budget');
  }

  async setTo3MonthAverage() {
-   await this.setTo3MonthAverageButton.click();
+   await this.#performAction(this.setTo3MonthAverageButton, 'set to 3 month average');
  }

  async setTo6MonthAverage() {
-   await this.setTo6MonthAverageButton.click();
+   await this.#performAction(this.setTo6MonthAverageButton, 'set to 6 month average');
  }

  async setToYearlyAverage() {
-   await this.setToYearlyAverageButton.click();
+   await this.#performAction(this.setToYearlyAverageButton, 'set to yearly average');
  }

  async applyBudgetTemplate() {
-   await this.applyBudgetTemplateButton.click();
+   await this.#performAction(this.applyBudgetTemplateButton, 'apply budget template');
  }

Likely invalid or redundant comment.

packages/desktop-client/e2e/page-models/settings-page.js (3)

4-15: LGTM! Clear property initialization.

The properties are well-organized and use appropriate test IDs and role-based selectors.


18-19: 🛠️ Refactor suggestion

Enhance wait strategy for settings page.

Add comprehensive wait conditions to improve test stability.

  async waitFor(options) {
-   await this.settings.waitFor(options);
+   await this.settings.waitFor({ state: 'visible', ...options });
+   // Wait for any loading indicators to disappear
+   await this.page.waitForSelector('[data-testid="loading-indicator"]', 
+     { state: 'hidden', timeout: 5000 }).catch(() => {});
  }

Likely invalid or redundant comment.


27-51: 🛠️ Refactor suggestion

Add error handling to improve test stability.

Enhance the methods with proper error handling and verification.

  async useBudgetType(budgetType) {
+   try {
      await this.enableExperimentalFeature('Budget mode toggle');
      await this.switchBudgetTypeButton.waitFor();
      const buttonText = await this.switchBudgetTypeButton.textContent();
      if (buttonText.includes(budgetType.toLowerCase())) {
        await this.switchBudgetTypeButton.click();
+       // Wait for the change to take effect
+       await this.page.waitForSelector('[data-testid="loading-indicator"]', 
+         { state: 'hidden', timeout: 5000 }).catch(() => {});
      }
+   } catch (error) {
+     throw new Error(`Failed to switch budget type: ${error.message}`);
+   }
  }

  async enableExperimentalFeature(featureName) {
+   try {
      if (await this.advancedSettingsButton.isVisible()) {
        await this.advancedSettingsButton.click();
      }

      if (await this.experimentalSettingsButton.isVisible()) {
        await this.experimentalSettingsButton.click();
      }

      const featureCheckbox = this.page.getByRole('checkbox', {
        name: featureName,
      });
+     await featureCheckbox.waitFor({ state: 'visible' });
      if (!(await featureCheckbox.isChecked())) {
        await featureCheckbox.click();
+       // Verify the checkbox state
+       if (!(await featureCheckbox.isChecked())) {
+         throw new Error('Feature checkbox did not toggle as expected');
+       }
      }
+   } catch (error) {
+     throw new Error(`Failed to enable experimental feature: ${error.message}`);
+   }
  }

Likely invalid or redundant comment.

packages/desktop-client/e2e/page-models/mobile-budget-page.js (1)

2-6: LGTM! Well-organized imports.

The imports are clearly structured and follow a logical order.

Comment on lines +86 to +87
async waitFor(options) {
await this.budgetTable.waitFor(options);
Copy link
Contributor

Choose a reason for hiding this comment

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

🛠️ Refactor suggestion

Add wait conditions to improve test stability.

Enhance the wait method with proper wait conditions.

  async waitFor(options) {
-   await this.budgetTable.waitFor(options);
+   await this.budgetTable.waitFor({ state: 'visible', ...options });
+   // Wait for any loading indicators to disappear
+   await this.page.waitForSelector('[data-testid="loading-indicator"]', 
+     { state: 'hidden', timeout: 5000 }).catch(() => {});
  }
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
async waitFor(options) {
await this.budgetTable.waitFor(options);
async waitFor(options) {
await this.budgetTable.waitFor({ state: 'visible', ...options });
// Wait for any loading indicators to disappear
await this.page.waitForSelector('[data-testid="loading-indicator"]',
{ state: 'hidden', timeout: 5000 }).catch(() => {});
}

Comment on lines +300 to +326
test(`applies budget template`, async () => {
const settingsPage = await navigation.goToSettingsPage();
await settingsPage.enableExperimentalFeature('Goal templates');

const budgetPage = await navigation.goToBudgetPage();

const categoryName = await budgetPage.getCategoryNameForRow(1);

const amountToTemplate = 123;

const categoryMenuModal = await budgetPage.openCategoryMenu(categoryName);
const editNotesModal = await categoryMenuModal.editNotes();
const templateNotes = `#template ${amountToTemplate}`;
await editNotesModal.updateNotes(templateNotes);
await editNotesModal.close();

const budgetedButton =
await budgetPage.getButtonForBudgeted(categoryName);

const budgetMenuModal = await budgetPage.openBudgetMenu(categoryName);
await budgetMenuModal.applyBudgetTemplate();
await budgetMenuModal.close();

await expect(budgetedButton).toHaveText(
amountToCurrency(amountToTemplate),
);
const notification = page.getByRole('alert');
Copy link
Contributor

Choose a reason for hiding this comment

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

🛠️ Refactor suggestion

Add error handling for experimental feature.

The test enables an experimental feature without proper error handling.

  test(`applies budget template`, async () => {
+   try {
      const settingsPage = await navigation.goToSettingsPage();
      await settingsPage.enableExperimentalFeature('Goal templates');

      const budgetPage = await navigation.goToBudgetPage();
      const categoryName = await budgetPage.getCategoryNameForRow(1);
      const amountToTemplate = 123;

      const categoryMenuModal = await budgetPage.openCategoryMenu(categoryName);
      const editNotesModal = await categoryMenuModal.editNotes();
      const templateNotes = `#template ${amountToTemplate}`;
      await editNotesModal.updateNotes(templateNotes);
      await editNotesModal.close();

      const budgetedButton = await budgetPage.getButtonForBudgeted(categoryName);
      const budgetMenuModal = await budgetPage.openBudgetMenu(categoryName);
      await budgetMenuModal.applyBudgetTemplate();
      await budgetMenuModal.close();

      await expect(budgetedButton).toHaveText(amountToCurrency(amountToTemplate));
      const notification = page.getByRole('alert');
      await expect(notification).toContainText(templateNotes);
      await expect(page).toMatchThemeScreenshots();
+   } catch (error) {
+     throw new Error(`Failed to apply budget template: ${error.message}`);
+   }
  });
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
test(`applies budget template`, async () => {
const settingsPage = await navigation.goToSettingsPage();
await settingsPage.enableExperimentalFeature('Goal templates');
const budgetPage = await navigation.goToBudgetPage();
const categoryName = await budgetPage.getCategoryNameForRow(1);
const amountToTemplate = 123;
const categoryMenuModal = await budgetPage.openCategoryMenu(categoryName);
const editNotesModal = await categoryMenuModal.editNotes();
const templateNotes = `#template ${amountToTemplate}`;
await editNotesModal.updateNotes(templateNotes);
await editNotesModal.close();
const budgetedButton =
await budgetPage.getButtonForBudgeted(categoryName);
const budgetMenuModal = await budgetPage.openBudgetMenu(categoryName);
await budgetMenuModal.applyBudgetTemplate();
await budgetMenuModal.close();
await expect(budgetedButton).toHaveText(
amountToCurrency(amountToTemplate),
);
const notification = page.getByRole('alert');
test(`applies budget template`, async () => {
try {
const settingsPage = await navigation.goToSettingsPage();
await settingsPage.enableExperimentalFeature('Goal templates');
const budgetPage = await navigation.goToBudgetPage();
const categoryName = await budgetPage.getCategoryNameForRow(1);
const amountToTemplate = 123;
const categoryMenuModal = await budgetPage.openCategoryMenu(categoryName);
const editNotesModal = await categoryMenuModal.editNotes();
const templateNotes = `#template ${amountToTemplate}`;
await editNotesModal.updateNotes(templateNotes);
await editNotesModal.close();
const budgetedButton =
await budgetPage.getButtonForBudgeted(categoryName);
const budgetMenuModal = await budgetPage.openBudgetMenu(categoryName);
await budgetMenuModal.applyBudgetTemplate();
await budgetMenuModal.close();
await expect(budgetedButton).toHaveText(
amountToCurrency(amountToTemplate),
);
const notification = page.getByRole('alert');
await expect(notification).toContainText(templateNotes);
await expect(page).toMatchThemeScreenshots();
} catch (error) {
throw new Error(`Failed to apply budget template: ${error.message}`);
}
});

Comment on lines +33 to +59
async function setBudgetAverage(
budgetPage,
categoryName,
numberOfMonths,
setBudgetAverageFn,
) {
let totalSpent = 0;

for (let i = 0; i < numberOfMonths; i++) {
await budgetPage.goToPreviousMonth();
const spentButton = await budgetPage.getButtonForSpent(categoryName);
const spent = await spentButton.textContent();
totalSpent += currencyToAmount(spent);
}

// Calculate average amount
const averageSpent = totalSpent / numberOfMonths;

// Go back to the current month
for (let i = 0; i < numberOfMonths; i++) {
await budgetPage.goToNextMonth();
}

await setBudgetAverageFn(budgetPage, categoryName, numberOfMonths);

return averageSpent;
}
Copy link
Contributor

Choose a reason for hiding this comment

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

🛠️ Refactor suggestion

Add retry mechanism to improve test stability.

Given the reported flaky tests, enhance the setBudgetAverage function with a retry mechanism.

+ const MAX_RETRIES = 3;
+ const RETRY_DELAY = 1000;

+ async function retry(fn, errorMessage) {
+   let lastError;
+   for (let attempt = 0; attempt < MAX_RETRIES; attempt++) {
+     try {
+       return await fn();
+     } catch (error) {
+       lastError = error;
+       await new Promise(resolve => setTimeout(resolve, RETRY_DELAY));
+     }
+   }
+   throw new Error(`${errorMessage}: ${lastError.message}`);
+ }

  async function setBudgetAverage(
    budgetPage,
    categoryName,
    numberOfMonths,
    setBudgetAverageFn,
  ) {
    let totalSpent = 0;

    for (let i = 0; i < numberOfMonths; i++) {
-     await budgetPage.goToPreviousMonth();
-     const spentButton = await budgetPage.getButtonForSpent(categoryName);
-     const spent = await spentButton.textContent();
-     totalSpent += currencyToAmount(spent);
+     await retry(
+       async () => {
+         await budgetPage.goToPreviousMonth();
+         const spentButton = await budgetPage.getButtonForSpent(categoryName);
+         const spent = await spentButton.textContent();
+         totalSpent += currencyToAmount(spent);
+       },
+       `Failed to get spent amount for month ${i + 1}`,
+     );
    }

    const averageSpent = totalSpent / numberOfMonths;

    for (let i = 0; i < numberOfMonths; i++) {
-     await budgetPage.goToNextMonth();
+     await retry(
+       () => budgetPage.goToNextMonth(),
+       `Failed to navigate to next month ${i + 1}`,
+     );
    }

-   await setBudgetAverageFn(budgetPage, categoryName, numberOfMonths);
+   await retry(
+     () => setBudgetAverageFn(budgetPage, categoryName, numberOfMonths),
+     'Failed to set budget average',
+   );

    return averageSpent;
  }
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
async function setBudgetAverage(
budgetPage,
categoryName,
numberOfMonths,
setBudgetAverageFn,
) {
let totalSpent = 0;
for (let i = 0; i < numberOfMonths; i++) {
await budgetPage.goToPreviousMonth();
const spentButton = await budgetPage.getButtonForSpent(categoryName);
const spent = await spentButton.textContent();
totalSpent += currencyToAmount(spent);
}
// Calculate average amount
const averageSpent = totalSpent / numberOfMonths;
// Go back to the current month
for (let i = 0; i < numberOfMonths; i++) {
await budgetPage.goToNextMonth();
}
await setBudgetAverageFn(budgetPage, categoryName, numberOfMonths);
return averageSpent;
}
const MAX_RETRIES = 3;
const RETRY_DELAY = 1000;
async function retry(fn, errorMessage) {
let lastError;
for (let attempt = 0; attempt < MAX_RETRIES; attempt++) {
try {
return await fn();
} catch (error) {
lastError = error;
await new Promise(resolve => setTimeout(resolve, RETRY_DELAY));
}
}
throw new Error(`${errorMessage}: ${lastError.message}`);
}
async function setBudgetAverage(
budgetPage,
categoryName,
numberOfMonths,
setBudgetAverageFn,
) {
let totalSpent = 0;
for (let i = 0; i < numberOfMonths; i++) {
await retry(
async () => {
await budgetPage.goToPreviousMonth();
const spentButton = await budgetPage.getButtonForSpent(categoryName);
const spent = await spentButton.textContent();
totalSpent += currencyToAmount(spent);
},
`Failed to get spent amount for month ${i + 1}`,
);
}
const averageSpent = totalSpent / numberOfMonths;
for (let i = 0; i < numberOfMonths; i++) {
await retry(
() => budgetPage.goToNextMonth(),
`Failed to navigate to next month ${i + 1}`,
);
}
await retry(
() => setBudgetAverageFn(budgetPage, categoryName, numberOfMonths),
'Failed to set budget average',
);
return averageSpent;
}

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

Successfully merging this pull request may close these issues.

4 participants