-
-
Notifications
You must be signed in to change notification settings - Fork 1.3k
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
base: master
Are you sure you want to change the base?
Conversation
✅ Deploy Preview for actualbudget ready!
To edit notification comments on pull requests, go to your Netlify site configuration. |
Bundle Stats — desktop-clientHey 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
Changeset
View detailed bundle breakdownAdded No assets were added Removed No assets were removed Bigger
Smaller
Unchanged
|
Bundle Stats — loot-coreHey 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
Changeset No files were changed View detailed bundle breakdownAdded No assets were added Removed No assets were removed Bigger No assets were bigger Smaller No assets were smaller Unchanged
|
WalkthroughThis 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
Suggested reviewers
Finishing Touches
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? 🪧 TipsChatThere are 3 ways to chat with CodeRabbit:
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)
Other keywords and placeholders
CodeRabbit Configuration File (
|
There was a problem hiding this 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 formatIn 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
⛔ 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
tolocator
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:
- Using
getByPlaceholder
for element selection is more reliable and readable.- 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:
- Makes the test more resilient to UI changes.
- Aligns with accessibility best practices.
- 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:
- The shift from
rootPage
tolocator
allows for more precise element selection.- The use of semantic selectors like
getByPlaceholder
andgetByRole
improves test robustness and aligns with accessibility best practices.- 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 patternThe
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() methodThe
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() methodThe
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 methodsThe action methods (
copyLastMonthBudget()
,setTo3MonthAverage()
,setTo6MonthAverage()
,setToYearlyAverage()
, andapplyBudgetTemplate()
) 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 testsThe 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 theopenBudgetMenu
method, indicating a refactoring of the budget menu functionality.packages/desktop-client/e2e/budget.mobile.test.js (1)
293-320
: Ensure correct usage ofsetBudgetAverageFn
in test casesWhen iterating over the array of
[numberOfMonths, setBudgetAverageFn]
, you're passingsetBudgetAverageFn
to thesetBudgetAverage
function:const averageSpent = await setBudgetAverage( budgetPage, categoryName, numberOfMonths, setBudgetAverageFn, );Given the earlier issue with naming conflicts in
setBudgetAverage
, ensure thatsetBudgetAverageFn
is correctly defined and passed after renaming the parameter in thesetBudgetAverage
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.
There was a problem hiding this 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 implementationThe
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 amountsThese 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
⛔ 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 utilitiesThe addition of
amountToCurrency
andcurrencyToAmount
imports is appropriate for the new budget average calculation functionality.
260-270
: LGTM: Updated test case for setting budgeted amountThe test case has been successfully updated to use the new
budgetMenuModal
object and itssetBudgetAmount
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 budgetThis new test case effectively verifies the 'copy last month's budget' functionality. It correctly:
- Navigates to the previous month
- Stores the previous month's budget
- Returns to the current month
- Applies the copy action
- 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.
There was a problem hiding this 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 ofgetByRole
andgetByTestId
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-implementededitNotes
method with good separation of concerns.The
editNotes
method is clear and follows good practices by creating a newEditNotesModal
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 duplicationThe new utility functions (
copyLastMonthBudget
,setTo3MonthAverage
,setTo6MonthAverage
, andsetToYearlyAverage
) 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 setToYearlyAverageThis 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 inuseEffect
dependency arrayThe
useEffect
hook depends onscrollY
,previousScrollY
,hide
, andopenDefault
, but onlyscrollY
is included in the dependency array. Omitting dependencies can lead to stale closures and unexpected behavior. Include all external variables used within theuseEffect
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
⛔ 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 theeditNotes
method, maintaining good modularity.
13-15
: LGTM: Concise and well-implementedclose
method.The
close
method is clear, concise, and follows good practices by using role-based element selection.
1-22
: Overall, excellent implementation of theCategoryMenuModal
class.This new file introduces a well-structured and clearly implemented
CategoryMenuModal
class for e2e testing. The code demonstrates good practices in terms of:
- Modularity and encapsulation
- Accessibility considerations (use of
getByRole
)- Testability (use of
getByTestId
)- Clear and descriptive naming conventions
The suggestions for improvement are minor and include:
- Adding JSDoc comments for better documentation
- Implementing error handling in the
editNotes
methodGreat 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
andCategoryMenuModal
are correctly formatted and align with the existing code style. These imports support the changes made to theopenBudgetMenu
andopenCategoryMenu
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
andCategoryMenuModal
. This is a positive step towards more maintainable code.However, since both
openBudgetMenu
andopenCategoryMenu
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 inopenBudgetMenu
.The changes to the
openBudgetMenu
method improve modularity by returning aBudgetMenuModal
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 inopenCategoryMenu
.The changes to the
openCategoryMenu
method improve modularity by returning aCategoryMenuModal
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 correctlyThe new import statement for
amountToCurrency
andcurrencyToAmount
functions from theloot-core/shared/util
module is appropriate and necessary for the new functionality introduced in this file.
33-59
: LGTM: Issues resolved insetBudgetAverage
functionThe
setBudgetAverage
function has been implemented correctly, addressing the previously identified issues:
- The infinite recursion problem has been resolved by using the
setBudgetAverageFn
parameter instead of calling the function recursively.- 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 interactionThe "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 budgetThe 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 testsThe 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 interactionsThe "applies budget template" test case has been successfully updated to use the new
CategoryMenuModal
andEditNotesModal
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 checkIn
goToSettingsPage()
, the navbar state is checked for both'default'
and'hidden'
before callingdragNavbarUp()
. 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 structuredThe imported modules and hooks, including the
type ComponentType
, are properly organized.
217-217
: EnsurenavbarState
is defined before usageWhen adding
data-navbar-state={navbarState}
, make sure thatnavbarState
is notundefined
. SincenavbarState
may initially beundefined
, 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 initializingnavbarState
.
181-183
: Verify the logic in the drag handlerIn the
useDrag
hook, the logic determining whether to callopenDefault
oropenFull
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 appliedThe addition of
userSelect: 'none'
prevents text selection on the navigation tabs, enhancing the user experience during interactions.
packages/desktop-client/src/components/mobile/MobileNavTabs.tsx
Outdated
Show resolved
Hide resolved
There was a problem hiding this 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 implementationThe
setBudgetAverage
function correctly calculates the average spending over a specified number of months. The issues mentioned in past review comments have been addressed:
- The potential infinite recursion has been resolved by correctly calling
setBudgetAverageFn
instead of recursively callingsetBudgetAverage
.- 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 budgetThis 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 sameThis 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 averagesThese 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 templateThis 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
📒 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 utilitiesThe addition of
amountToCurrency
andcurrencyToAmount
imports is appropriate for the new currency conversion operations in the file.
252-263
: LGTM: Updated test for budget amount changesThe 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 improvementThe 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:
- Refactor the budget utility functions to reduce code duplication.
- Improve the readability of the
setBudgetAverage
function by extracting helper methods.- 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!
There was a problem hiding this 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
andgetByTestId
suggests good accessibility practices, which is crucial for robust end-to-end tests.Consider adding JSDoc comments to describe the
page
andlocator
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 focusedclose
method.The
close
method is well-implemented, usingasync/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
📒 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
, andCategoryMenuModal
are correctly placed and follow the existing naming convention. These additions suggest a modular approach to handling different budget-related actions.
145-146
: LGTM: NewopenCategoryMenu
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 newCategoryMenuModal
instance. This addition enhances the functionality of theMobileBudgetPage
class.
202-202
: LGTM:openBalanceMenu
updated to useBalanceMenuModal
.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
, andCategoryMenuModal
) have been introduced and integrated into the existing methods.Key points:
- New imports added for modal classes.
openCategoryMenu
method added, following the existing pattern.openBudgetMenu
andopenBalanceMenu
methods updated to return modal instances.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:
- Verify that all tests using
MobileBudgetPage
have been updated to work with the new modal-based approach.- Check for any remaining references to the removed
setBudget
method in other parts of the codebase.- Update the documentation for
MobileBudgetPage
to reflect these changes, especially noting the new return types for methods likeopenBudgetMenu
andopenBalanceMenu
.
There was a problem hiding this 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 suggestionThe constructor is well-implemented, properly initializing instance variables with the provided
page
andlocator
parameters. Theheading
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 suggestionThe
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 averageThe 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 templateThis test case effectively covers the budget template feature:
- It enables the experimental 'Goal templates' feature.
- It creates a template by updating category notes.
- 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
📒 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 testingThe
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: NewopenCategoryMenu
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 useBalanceMenuModal
.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 thesetBudget
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 insetBudgetAverage
functionGreat job addressing the issues mentioned in the past review comments:
- The infinite recursion issue has been resolved by renaming the parameter to
setBudgetAverageFn
.- 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 budgetThis test case effectively verifies the
copyLastMonthBudget
functionality:
- It navigates to the previous month to get the last month's budget.
- It returns to the current month and applies the
copyLastMonthBudget
function.- 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 modalsThese test cases comprehensively cover the budget summary functionality:
- They test both Envelope and Tracking budget types separately.
- They verify that clicking the appropriate elements opens the correct budget summary modal.
- 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 changesThe changes in this file significantly enhance the budget management functionality and its testing:
- New utility functions have been added to support various budget operations.
- Comprehensive test cases cover new features like copying last month's budget, setting budget averages, and applying budget templates.
- Previous issues identified in past reviews have been addressed, improving code reliability.
- 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.
There was a problem hiding this 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:
- Add JSDoc comments to explain the purpose of the method and the
options
parameter.- 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 theView
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 withReact.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 ofdata-testid
and suggest test enhancements.The addition of
data-testid="account-list"
to theView
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
📒 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 awaitFor
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
andaccountListItems
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 theMobileAccountsPage
class!The changes made to this class enhance its flexibility and maintainability:
- The new element selection strategy using test IDs improves test resilience.
- The addition of the
waitFor
method helps handle asynchronous UI updates.- 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
, andaddTransactionButton
) enhance the organization of the class and improve its testability. The use ofgetByTestId
andgetByRole
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:
- Restructuring element access for better organization.
- Improving testability with the use of test-id and role-based selectors.
- Adding a
waitFor
method to increase test reliability.- 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 thedata-testid
change and verify existing tests.The change from
data-testid="account"
todata-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 olddata-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:
- Adding
data-testid
attributes to key elements, enhancing the testability of the components.- Introducing a new
TransactionEdit
component that uses React hooks for data fetching and wraps the existingTransactionEditUnconnected
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 declaringROW_HEIGHT
as a static class property
ROW_HEIGHT
is currently defined as an instance property. IfROW_HEIGHT
is intended to be a constant value shared across all instances ofMobileNavigation
, consider making it a static class property using thestatic
keyword.
44-45
: Usethis.navbar.getByRole
for link selection ingoToBudgetPage
In
goToBudgetPage()
, you're usingthis.page.getByRole
to select the 'Budget' link. To maintain consistency with other navigation methods, consider usingthis.navbar.getByRole
instead.
69-70
: Usethis.navbar.getByRole
for link selection ingoToAccountsPage
In
goToAccountsPage()
, you're usingthis.page.getByRole
to select the 'Accounts' link. It would be more consistent to usethis.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 correctlyThe 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 towaitFor(options)
. Please ensure that all references towaitForBudgetTable()
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 valueThe method
openBalanceMenu
now returns a new instance ofBalanceMenuModal
. 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 valueThe method
openEnvelopeBudgetSummary
now returns a new instance ofEnvelopeBudgetSummaryModal
. 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 2Additionally, 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 valueThe method
openCategoryMenu
now returns a new instance ofCategoryMenuModal
. 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 valueThe method
openTrackingBudgetSummary
now returns a new instance ofTrackingBudgetSummaryModal
. 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 2Also, 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 valueThe method
openBudgetMenu
now returns a new instance ofBudgetMenuModal
. 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 2Length 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 tsLength of output: 92
There was a problem hiding this 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
📒 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 MobileReportsPageThe import statement for
MobileReportsPage
is consistent with the naming convention of other imported classes and aligns with the newgoToReportsPage
method added to theMobileNavigation
class.
10-11
: LGTM: New properties for improved element selectionThe addition of
heading
andnavbar
properties usinggetByRole
is a good practice for accessibility and provides a centralized way to reference these elements throughout the class.
16-18
: LGTM: dragNavbarUp method implementationThe
dragNavbarUp
method provides a clear and concise way to drag the navbar to the heading position. The use of thedragTo
API is appropriate for this interaction.
27-29
: LGTM: hasNavbarState method implementationThe
hasNavbarState
method is well-implemented. It efficiently checks the navbar's state against multiple possible values using the spread operator andincludes
method. This approach provides flexibility and readability.
33-54
: LGTM: Improved goToBudgetPage methodThe updates to
goToBudgetPage
method enhance its functionality by:
- Handling different navbar states.
- Ensuring the navbar is visible before navigation.
- Returning the
budgetPage
instance consistently.These changes improve the robustness and reliability of the navigation process.
58-79
: LGTM: Consistent improvements in goToAccountsPage methodThe
goToAccountsPage
method has been updated consistently with thegoToBudgetPage
method. It now includes:
- Proper handling of navbar states.
- Ensuring navbar visibility before navigation.
- 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 methodThe
goToTransactionEntryPage
method has been improved consistently with other navigation methods. Notable enhancements include:
- Proper handling of navbar states.
- Use of
this.navbar
for link selection, which is more specific and consistent with the class structure.- Returning the
transactionEntryPage
instance.These changes contribute to a more robust and consistent navigation implementation across the class.
103-124
: LGTM: Well-implemented goToReportsPage methodThe new
goToReportsPage
method is a welcome addition to theMobileNavigation
class. It follows the same pattern as other navigation methods, ensuring:
- Proper handling of navbar states.
- Consistent navigation process.
- Return of the
reportsPage
instance.This implementation maintains the consistency and reliability of the navigation system.
127-148
: LGTM: Consistent implementation of goToSettingsPage methodThe new
goToSettingsPage
method is a solid addition to theMobileNavigation
class. It maintains consistency with other navigation methods by:
- Handling navbar states appropriately.
- Following the established navigation process.
- Returning the
settingsPage
instance.This implementation contributes to the overall consistency and reliability of the navigation system.
20-25
: Review dragNavbarDown implementationThe
dragNavbarDown
method usesthis.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 usingthis.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
andsetToYearlyAverage
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:
- Addition of utility functions for various budget operations.
- New test cases for copying last month's budget and setting budget averages.
- Implementation of budget template functionality with accessibility considerations.
- 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 withinbudget.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.jsLength 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.jsLength of output: 158
Script:
#!/bin/bash # Verify the presence of goToNextMonth function call rg 'await budgetPage\.goToNextMonth\(\)' packages/desktop-client/e2e/budget.mobile.test.jsLength of output: 228
There was a problem hiding this 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 addingaria-level
to the heading roleIn the
MobilePageHeader
component, you've correctly usedrole="heading"
. To further improve accessibility, consider adding anaria-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 typesIn 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 logicThe
Page
component contains logic for responsive design based onisNarrowWidth
. 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 suggestionThe
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 suggestionsThe
dragNavbarDown
method effectively simulates dragging the navbar down. It uses the bounding box of the navbar and theNAVBAR_ROWS
constant 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, similar to the suggestion fordragNavbarUp
.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 suggestionThe
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
📒 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 MobileReportsPageThe addition of the
MobileReportsPage
import is consistent with the existing import pattern and supports the newgoToReportsPage
method.
10-13
: LGTM: Enhanced constructor with key element selectorsThe 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 constantThe 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 methodThe
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 constantThe 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 constantThe 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 methodsThe 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 thegoToReportsPage
method is consistent with the other navigation methods.
1-126
: Overall assessment: Significant improvements to mobile navigationThe changes to the
MobileNavigation
class represent a substantial improvement in the mobile navigation functionality and code quality. Key improvements include:
- Introduction of new helper methods (
dragNavbarUp
,dragNavbarDown
,hasNavbarState
) for managing the navbar state.- Addition of constants (
NAVBAR_ROWS
,NAV_LINKS_HIDDEN_BY_DEFAULT
,ROUTES_BY_PAGE
) that centralize important configuration data.- Implementation of a generic
navigateToPage
method that centralizes navigation logic and reduces code duplication.- Refactoring of existing navigation methods to use the new
navigateToPage
method.- 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.
There was a problem hiding this 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-implementedopenFull
function with a suggestionThe
openFull
function is well-implemented, usinguseCallback
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-implementedopenDefault
andhide
functions with a suggestionThe
openDefault
andhide
functions are well-implemented, consistent withopenFull
, and useuseCallback
for optimization.Consider refactoring the common logic in
openFull
,openDefault
, andhide
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
, andhide
with the appropriate parameters.
217-217
: LGTM: Addeddata-navbar-state
attribute with a suggestionThe 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
📒 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 visibilityThe 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: UpdateduseEffect
and drag handling logicThe 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
2589558
to
5dcc22d
Compare
There was a problem hiding this 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:
- Using the
switchBudgetTypeButton
property improves consistency.- 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:
- Checking visibility before clicking prevents errors with hidden elements.
- 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 changesThe
setBudgetAverage
function is well-implemented, but consider the following improvements:
- Add a comment explaining the purpose of the
setBudgetAverageFn
parameter for clarity.- 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 clarityThe 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 assertionsThe test cases for setting budget to average are well-implemented. Consider the following improvements:
- Add a comment explaining the test structure:
// Parameterized test cases for 3, 6, and 12-month budget averages
- 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 precisionThis approach provides more precise comparisons and better handles potential rounding issues.
300-328
: Enhance test robustness and documentationThe test case for applying a budget template is well-implemented. Consider the following improvements:
- Add a comment explaining the purpose and format of the budget template:
// Test applying a budget template. Format: #template <amount>
- Add error handling for the experimental feature:
test.fail(({ browserName }) => browserName !== 'chromium', 'This test requires the experimental feature to be available');
- 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
⛔ 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 theexportDataButton
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 MobileReportsPageThe addition of the import statement for
MobileReportsPage
is correct and consistent with the existing import style. This import is likely used in the newgoToReportsPage
method, enhancing the navigation capabilities of the class.
10-13
: LGTM: Enhanced constructor with new propertiesThe 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 methodThe
dragNavbarUp
method is well-implemented. It effectively uses thepage.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 methodThe
hasNavbarState
method is well-implemented. It effectively checks the navbar's state based on itsdata-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 methodThe
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 (likehasNavbarState
,dragNavbarUp
, anddragNavbarDown
) 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 methodsThe navigation methods (
goToBudgetPage
,goToAccountsPage
,goToTransactionEntryPage
,goToReportsPage
) have been effectively refactored to use the newnavigateToPage
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 goToSettingsPageThe
goToSettingsPage
method has been refactored consistently with the other navigation methods. It now uses thenavigateToPage
method, passing the page name and a factory function for creating theSettingsPage
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: NewopenCategoryMenu
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 useBudgetMenuModal
.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 useBalanceMenuModal
.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 improvementThe 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:
- Refactor utility functions to reduce code duplication.
- Enhance test assertions for more precise comparisons.
- Improve error handling and documentation in test cases.
- 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 ofnavbarState
ensures consistent stateThe
navbarState
is now correctly initialized to'default'
, which prevents it from beingundefined
on the initial render. This ensures consistent behavior across renders.
packages/desktop-client/src/components/mobile/MobileNavTabs.tsx
Outdated
Show resolved
Hide resolved
There was a problem hiding this 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 suggestionThe 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 implementationThe
dragNavbarDown()
method is well-implemented, usingpage.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 suggestionThe
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
📒 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 maintainabilityThe 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 ofmainContentSelector
andnavbarSelector
improve maintainability and make the code more semantic.Also applies to: 10-13
32-47
: LGTM: Well-implemented dragNavbarUp methodThe
dragNavbarUp()
method is well-implemented, usingpage.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 methodThe
hasNavbarState()
method is well-implemented. It correctly handles the case where the navbar doesn't exist on the page and uses thedata-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 methodsThe refactoring of
goToBudgetPage()
,goToAccountsPage()
,goToTransactionEntryPage()
, andgoToReportsPage()
to use the newnavigateToPage()
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 goToSettingsPageThe refactoring of
goToSettingsPage()
to use the newnavigateToPage()
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 classThe changes made to the
MobileNavigation
class significantly improve its functionality, maintainability, and extensibility. Key improvements include:
- Centralized navigation logic in the
navigateToPage()
method, reducing code duplication.- Addition of methods for controlling navbar behavior (
dragNavbarUp()
,dragNavbarDown()
,hasNavbarState()
).- Use of static private fields for constants, improving encapsulation.
- Consistent refactoring of all navigation methods to use the new centralized logic.
- 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.
4c58904
to
9746c67
Compare
There was a problem hiding this 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 propertiesThe 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-implementeddragNavbarUp
methodThe
dragNavbarUp
method effectively simulates dragging the navbar into the main content area. The calculation of bounding boxes and use ofpage.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-implementeddragNavbarDown
methodThe
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-implementednavigateToPage
methodThe
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 managementThe new callback functions (
openFull
,openDefault
, andhide
) are well-implemented and provide clear management of the navbar's state and animations. The use ofuseCallback
for memoization is appropriate.One minor suggestion:
In the
openFull
function, thecanceled
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 navigationIn the
setBudgetAverage
function, you're correctly retrieving thespentButton
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 averageThese 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
⛔ 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 importsThe addition of the
MobileReportsPage
import and the new constructor properties (heading
,navbar
,mainContentSelector
, andnavbarSelector
) 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-implementedhasNavbarState
methodThe
hasNavbarState
method is well-implemented. It correctly handles cases where the navbar is not present and uses thedata-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 methodsThe updates to the navigation methods (
goToBudgetPage
,goToAccountsPage
,goToTransactionEntryPage
,goToReportsPage
) are well-implemented. The consistent use of the newnavigateToPage
method reduces code duplication and ensures uniform navigation behavior across different pages.
132-133
: LGTM: Consistent update togoToSettingsPage
The update to the
goToSettingsPage
method is well-implemented and consistent with the changes made to other navigation methods. It correctly uses the newnavigateToPage
method and theSettingsPage
class.packages/desktop-client/src/components/mobile/MobileNavTabs.tsx (5)
1-6
: LGTM: Necessary imports addedThe new imports (
useCallback
,useEffect
, anduseState
) are correctly added and are essential for the refactored component's functionality.
36-38
: LGTM: State initialization addressedThe
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 positionsThe introduction of
openFullY
andopenDefaultY
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 managementThe changes in the drag handler correctly use the new
openDefault
andopenFull
functions, aligning with the refactored navbar state management. The use of velocity inopenDefault
is a good touch for smooth animations.Note: As mentioned in a previous comment, consider reviewing the use of the
canceled
parameter in theopenFull
function, as it's passed here but not utilized in the function body.
217-217
: LGTM: Helpful data attribute for navbar stateThe 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 amountThe 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 budgetThis new test case effectively verifies the 'copy last month's budget' functionality. It covers the following key aspects:
- Navigating to the previous month
- Retrieving the budget amount from the previous month
- Returning to the current month
- Applying the 'copy last month's budget' action
- 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 goodThe
openCategoryMenu
method is correctly implemented and follows the existing patterns for opening modals.
86-87
: Ensure all references towaitForBudgetTable
are updatedThe method
waitForBudgetTable()
has been renamed towaitFor(options)
. Please verify that all references towaitForBudgetTable
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
openDefault(); | ||
} | ||
}, [scrollY]); | ||
}, [scrollY, hide, openDefault]); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
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.
openDefault(); | |
} | |
}, [scrollY]); | |
}, [scrollY, hide, openDefault]); | |
openDefault(); | |
} | |
}, [scrollY, previousScrollY, hide, openDefault]); |
const copyLastMonthBudget = async (budgetPage, categoryName) => { | ||
const budgetMenuModal = await budgetPage.openBudgetMenu(categoryName); | ||
await budgetMenuModal.copyLastMonthBudget(); | ||
await budgetMenuModal.close(); | ||
}; |
There was a problem hiding this comment.
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
).
Looks like the test is flaky:
Would you mind taking a peak at it to make it more stable? Apart from that LGTM |
88d5375
to
557a4f4
Compare
There was a problem hiding this 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 navigationThe 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 attributeThe 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
⛔ 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 andgetByRole()
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 MobileReportsPageThe addition of the import statement for
MobileReportsPage
is correct and consistent with the existing import style. This import is necessary for the newgoToReportsPage
method implemented later in the class.
10-13
: LGTM: Enhanced constructor with new propertiesThe additions to the constructor are well-implemented:
- Using
getByRole
forheading
andnavbar
improves accessibility and testability.- The
mainContentSelector
andnavbarSelector
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 configurationThe 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 methodThe
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 methodThe
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 logicThe
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 methodsThe 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 visibilityThe 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-implementedopenFull
callbackThe
openFull
callback is well-implemented, providing a smooth animation for fully opening the navbar. The use ofuseCallback
for optimization and the conditional wobbly effect add nice touches to the user experience.
123-133
: LGTM: Well-implementedopenDefault
callbackThe
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-implementedhide
callbackThe
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 logicThe updates to the drag handling logic align well with the new navbar state management. The use of velocity in
openDefault
and thecanceled
parameter inopenFull
should provide smoother and more engaging animations for the user.
158-160
:⚠️ Potential issueAdd missing dependency to useEffect
The change from
close()
toopenDefault()
aligns well with the new navbar state management. However, theuseEffect
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 addedThe 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 typeThe 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 interactionThe 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 interactionThe 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 amountThe 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 budgetThis 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 averagesThese 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 templateThis 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 modalsThese 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 suggestionsThe 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:
- Refactoring the utility functions to reduce code duplication.
- 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 fromopenEnvelopeBudgetSummaryMenu
toopenEnvelopeBudgetSummary
The method
openEnvelopeBudgetSummaryMenu
has been renamed toopenEnvelopeBudgetSummary
. 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 fromopenTrackingBudgetSummaryMenu
toopenTrackingBudgetSummary
The method
openTrackingBudgetSummaryMenu
has been renamed toopenTrackingBudgetSummary
. 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 toopenTrackingBudgetSummary
.🏁 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 codebaseThe method
waitForBudgetTable
has been renamed towaitFor
. Please ensure that all references towaitForBudgetTable
have been updated towaitFor
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 towaitFor
.🏁 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 2Length of output: 59
await expect(summary.getByText('Available Funds')).toBeVisible({ | ||
timeout: 10000, | ||
}); | ||
await expect(summary.getByText('Available Funds')).toBeVisible(); |
There was a problem hiding this comment.
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 multipletoBeVisible()
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:
- It may make the test faster if the "Available Funds" text typically appears quickly.
- 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:
- Monitoring the test results closely after this change, especially in CI environments.
- If the change is intentional and based on specific insights, consider adding a comment explaining the rationale.
- 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
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, | ||
}, | ||
}); |
There was a problem hiding this comment.
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.
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(); | ||
}; | ||
|
There was a problem hiding this comment.
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.
There was a problem hiding this 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 suggestionThe 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 suggestionThe
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 thedragNavbarUp
method.Consider adding a comment explaining the calculation of the
y
value in thetargetPosition
. 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 suggestionThe
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 ifdataNavbarState
is truthy before callingincludes
.
79-108
: LGTM: Well-implemented navigateToPage method with a suggestionThe
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
📒 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 selectionThe addition of
heading
andnavbar
properties usinggetByRole
is a good practice for accessibility and maintainability. ThemainContentSelector
andnavbarSelector
provide clear references to important page elements.
34-49
: LGTM: Well-implemented dragNavbarUp methodThe
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 methodThe
goToBudgetPage
method has been effectively refactored to use the newnavigateToPage
method. This simplifies the code and reduces duplication across navigation methods.
118-130
: LGTM: Consistent refactoring of navigation methodsThe
goToAccountsPage
andgoToTransactionEntryPage
methods have been refactored consistently withgoToBudgetPage
. This refactoring improves code consistency and maintainability across the class.
132-143
: LGTM: Consistent refactoring of remaining navigation methodsThe
goToReportsPage
andgoToSettingsPage
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 enhancementsThe changes to the
MobileNavigation
class significantly improve its structure, maintainability, and functionality. The introduction of thenavigateToPage
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!
738703d
to
75da4cc
Compare
There was a problem hiding this 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
: UseswaitForTimeout(1000)
andwaitForTimeout(500)
schedules-page.js
: UseswaitForTimeout(1000)
Consider:
- Adding state checks before clicks (e.g.,
await button.waitFor({ state: 'visible' })
)- Replacing fixed timeouts with proper element state waits
- 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 1Length 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:
- Add parameter validation in the constructor
- 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:
- Add retry logic for flaky tests:
test.describe.configure({ retries: 2 });
- 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); } });
- 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 parametersThe 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 renamingwaitFor
to a more descriptive method nameRenaming the method from
waitForBudgetTable
towaitFor
may reduce clarity, aswaitFor
is quite generic. Consider using a more descriptive name likewaitForBudgetTable
orwaitForPageContent
to better convey its purpose.
📜 Review details
Configuration used: CodeRabbit UI
Review profile: CHILL
⛔ 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
:
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:
- Restore the explicit timeout:
- await expect(summary.getByText('Available Funds')).toBeVisible();
+ await expect(summary.getByText('Available Funds')).toBeVisible({ timeout: 10000 });
- 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:
- Use the newly added
data-navbar-state
attribute to verify state transitions - Add delays or wait conditions after animations complete
- 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 thedata-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
async waitFor(options) { | ||
await this.transactionForm.waitFor(options); |
There was a problem hiding this comment.
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.
await this.addTransactionButton.click(); | ||
|
||
return new MobileAccountPage(this.page); | ||
} |
There was a problem hiding this comment.
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.
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(); | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
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.
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}`); | |
} | |
} |
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(); | ||
} |
There was a problem hiding this comment.
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:
- Adds consistent wait conditions
- Handles loading states
- Provides better error messages
- Reduces code duplication
Committable suggestion skipped: line range outside the PR's diff.
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, | ||
}, | ||
}); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Address test flakiness concerns
Given the reported flaky tests, the drag operations need additional stability measures:
- Add explicit waits for animations to complete
- Verify element positions after drag operations
- 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.
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 |
/update-vrt |
75da4cc
to
41e771d
Compare
There was a problem hiding this 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 animationsThe 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 auseLayoutEffect
: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 attributesWhile 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 orderThe
@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 constructorThe 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 handlingThe
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
⛔ 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
:
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
:
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
:
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}`);
}
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(); | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
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
async waitFor(options) { | ||
await this.settings.waitFor(options); |
There was a problem hiding this comment.
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.
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, | ||
}, | ||
}); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
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.
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}`); | |
} |
There was a problem hiding this 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 parametersWhile 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 navigateToPageWhile 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
📒 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
:
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
:
Add stability measures to dragNavbarUp
Given the reported test flakiness, the drag operation needs additional reliability measures:
- Add position verification after drag
- Implement retry mechanism
- 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.
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 |
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. |
Co-authored-by: coderabbitai[bot] <136622811+coderabbitai[bot]@users.noreply.github.com>
2d401c8
to
a4f45cc
Compare
There was a problem hiding this 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 issueEnhance 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 issueAdd 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:
- Add retry options to critical interactions
- Implement custom timeout configurations
- 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 testsThe 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
⛔ 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:
- Run these tests in isolation first to verify if stability improves
- Add logging to track timing of key operations
- 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 issueAdd retry mechanism to improve drag operation stability
Given the reported test flakiness issues, the drag operations should include:
- Retry logic for failed operations
- Position verification after drag
- 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 suggestionAdd 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 suggestionEnhance 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 suggestionAdd 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.
async waitFor(options) { | ||
await this.budgetTable.waitFor(options); |
There was a problem hiding this comment.
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.
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(() => {}); | |
} |
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'); |
There was a problem hiding this comment.
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.
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}`); | |
} | |
}); |
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; | ||
} |
There was a problem hiding this comment.
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.
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; | |
} |
No description provided.