From a917e0644a85e5593624dc9a052b5155c9dcc93e Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Thu, 16 Nov 2023 09:32:54 -0500 Subject: [PATCH 001/117] adds violations functions to report utils and implements --- .../ReportActionItem/ReportPreview.js | 2 +- src/libs/ReportUtils.js | 78 +++++++++++++++++++ src/libs/SidebarUtils.ts | 2 +- 3 files changed, 80 insertions(+), 2 deletions(-) diff --git a/src/components/ReportActionItem/ReportPreview.js b/src/components/ReportActionItem/ReportPreview.js index 45fe7d42e299..38fc7b008095 100644 --- a/src/components/ReportActionItem/ReportPreview.js +++ b/src/components/ReportActionItem/ReportPreview.js @@ -125,7 +125,7 @@ function ReportPreview(props) { const hasReceipts = transactionsWithReceipts.length > 0; const hasOnlyDistanceRequests = ReportUtils.hasOnlyDistanceRequestTransactions(props.iouReportID); const isScanning = hasReceipts && ReportUtils.areAllRequestsBeingSmartScanned(props.iouReportID, props.action); - const hasErrors = hasReceipts && ReportUtils.hasMissingSmartscanFields(props.iouReportID); + const hasErrors = (hasReceipts && ReportUtils.hasMissingSmartscanFields(props.iouReportID)) || ReportUtils.reportHasViolations(props.iouReportID); const lastThreeTransactionsWithReceipts = transactionsWithReceipts.slice(-3); const lastThreeReceipts = _.map(lastThreeTransactionsWithReceipts, (transaction) => ReceiptUtils.getThumbnailAndImageURIs(transaction)); const hasNonReimbursableTransactions = ReportUtils.hasNonReimbursableTransactions(props.iouReportID); diff --git a/src/libs/ReportUtils.js b/src/libs/ReportUtils.js index 973ed1f0dfd1..12808ef747a5 100644 --- a/src/libs/ReportUtils.js +++ b/src/libs/ReportUtils.js @@ -12,6 +12,7 @@ import CONST from '@src/CONST'; import ONYXKEYS from '@src/ONYXKEYS'; import ROUTES from '@src/ROUTES'; import * as IOU from './actions/IOU'; +import * as CollectionUtils from './CollectionUtils'; import * as CurrencyUtils from './CurrencyUtils'; import DateUtils from './DateUtils'; import isReportMessageAttachment from './isReportMessageAttachment'; @@ -81,6 +82,19 @@ Onyx.connect({ callback: (val) => (loginList = val), }); +const transactionViolations = {}; +Onyx.connect({ + key: ONYXKEYS.COLLECTION.TRANSACTION_VIOLATIONS, + callback: (violations, key) => { + if (!key || !violations) { + return; + } + + const transactionID = CollectionUtils.extractCollectionItemID(key); + transactionViolations[transactionID] = violations; + }, +}); + function getChatType(report) { return report ? report.chatType : ''; } @@ -3294,6 +3308,63 @@ function shouldHideReport(report, currentReportId) { return parentReport.reportID !== report.reportID && !isChildReportHasComment; } +/** + * @param {String} transactionID + * @returns {Boolean} + */ + +function transactionHasViolation(transactionID) { + const violations = lodashGet(transactionViolations, transactionID, []); + return _.some(violations, (violation) => violation.type === 'violation'); +} + +/** + * + * @param {Object} report + * @returns {Boolean} + */ + +function transactionThreadHasViolations(report) { + if (!Permissions.canUseViolations()) { + return false; + } + // eslint-disable-next-line es/no-nullish-coalescing-operators + if (!report.parentReportActionID ?? 0) { + return false; + } + + const reportActions = ReportActionsUtils.getAllReportActions(report.reportID); + + const parentReportAction = lodashGet(reportActions, `${report.parentReportID}.${report.parentReportActionID}`); + if (!parentReportAction) { + return false; + } + // eslint-disable-next-line es/no-nullish-coalescing-operators + const transactionID = parentReportAction.originalMessage.IOUTransactionID ?? 0; + if (!transactionID) { + return false; + } + // eslint-disable-next-line es/no-nullish-coalescing-operators + const reportID = parentReportAction.originalMessage.IOUReportID ?? 0; + if (!reportID) { + return false; + } + if (!isCurrentUserSubmitter(reportID)) { + return false; + } + return transactionHasViolation(transactionID); +} + +/** + * @param {String} reportID + * @returns {Boolean} + */ + +function reportHasViolations(reportID) { + const transactions = TransactionUtils.getAllReportTransactions(reportID); + return _.some(transactions, (transaction) => transactionHasViolation(transaction.transactionID)); +} + /** * Takes several pieces of data from Onyx and evaluates if a report should be shown in the option list (either when searching * for reports or the reports shown in the LHN). @@ -3369,6 +3440,11 @@ function shouldReportBeInOptionList(report, currentReportId, isInGSDMode, betas, return true; } + // Always show IOU reports with violations + if (isExpenseRequest(report) && transactionThreadHasViolations(report)) { + return true; + } + // All unread chats (even archived ones) in GSD mode will be shown. This is because GSD mode is specifically for focusing the user on the most relevant chats, primarily, the unread ones if (isInGSDMode) { return isUnread(report); @@ -4358,4 +4434,6 @@ export { getReimbursementQueuedActionMessage, getPersonalDetailsForAccountID, getRoom, + transactionThreadHasViolations, + reportHasViolations, }; diff --git a/src/libs/SidebarUtils.ts b/src/libs/SidebarUtils.ts index 58c4a124335d..2940a6286a32 100644 --- a/src/libs/SidebarUtils.ts +++ b/src/libs/SidebarUtils.ts @@ -361,7 +361,7 @@ function getOptionData( result.shouldShowSubscript = ReportUtils.shouldReportShowSubscript(report); result.pendingAction = report.pendingFields ? report.pendingFields.addWorkspaceRoom || report.pendingFields.createChat : null; result.allReportErrors = OptionsListUtils.getAllReportErrors(report, reportActions) as OnyxCommon.Errors; - result.brickRoadIndicator = Object.keys(result.allReportErrors ?? {}).length !== 0 ? CONST.BRICK_ROAD_INDICATOR_STATUS.ERROR : ''; + result.brickRoadIndicator = Object.keys(result.allReportErrors ?? {}).length !== 0 || ReportUtils.transactionThreadHasViolations(report) ? CONST.BRICK_ROAD_INDICATOR_STATUS.ERROR : ''; result.ownerAccountID = report.ownerAccountID; result.managerID = report.managerID; result.reportID = report.reportID; From 3756ef9c53ee0ba0fb0a93d8a4f8a8c5f89750b7 Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Thu, 16 Nov 2023 11:03:03 -0500 Subject: [PATCH 002/117] Small fix in ReportUtils --- src/libs/ReportUtils.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/libs/ReportUtils.js b/src/libs/ReportUtils.js index 12808ef747a5..6fe0ff8ad7ea 100644 --- a/src/libs/ReportUtils.js +++ b/src/libs/ReportUtils.js @@ -3329,7 +3329,7 @@ function transactionThreadHasViolations(report) { return false; } // eslint-disable-next-line es/no-nullish-coalescing-operators - if (!report.parentReportActionID ?? 0) { + if (!report.parentReportActionID) { return false; } From 3cc2e0f6bcc8e9ffbf16127831ea4fc8faebae0d Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Thu, 16 Nov 2023 15:15:20 -0500 Subject: [PATCH 003/117] Fix Onyx Connection for reportActions --- src/libs/ReportUtils.js | 19 +++++++++++++++---- 1 file changed, 15 insertions(+), 4 deletions(-) diff --git a/src/libs/ReportUtils.js b/src/libs/ReportUtils.js index 6fe0ff8ad7ea..e90f2eb955d4 100644 --- a/src/libs/ReportUtils.js +++ b/src/libs/ReportUtils.js @@ -95,6 +95,19 @@ Onyx.connect({ }, }); +const reportActions = {}; +Onyx.connect({ + key: ONYXKEYS.COLLECTION.REPORT_ACTIONS, + callback: (actions, key) => { + if (!key || !actions) { + return; + } + + const reportID = CollectionUtils.extractCollectionItemID(key); + reportActions[reportID] = actions; + }, +}); + function getChatType(report) { return report ? report.chatType : ''; } @@ -3303,8 +3316,8 @@ function canAccessReport(report, policies, betas, allReportActions) { */ function shouldHideReport(report, currentReportId) { const parentReport = getParentReport(getReport(currentReportId)); - const reportActions = ReportActionsUtils.getAllReportActions(report.reportID); - const isChildReportHasComment = _.some(reportActions, (reportAction) => (reportAction.childVisibleActionCount || 0) > 0); + const allReportActions = ReportActionsUtils.getAllReportActions(report.reportID); + const isChildReportHasComment = _.some(allReportActions, (reportAction) => (reportAction.childVisibleActionCount || 0) > 0); return parentReport.reportID !== report.reportID && !isChildReportHasComment; } @@ -3333,8 +3346,6 @@ function transactionThreadHasViolations(report) { return false; } - const reportActions = ReportActionsUtils.getAllReportActions(report.reportID); - const parentReportAction = lodashGet(reportActions, `${report.parentReportID}.${report.parentReportActionID}`); if (!parentReportAction) { return false; From a98b3a12290754acc7610c3e84083a9ac9f515a6 Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Tue, 21 Nov 2023 12:17:38 -0500 Subject: [PATCH 004/117] Fix Permissions import --- src/libs/ReportUtils.js | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/src/libs/ReportUtils.js b/src/libs/ReportUtils.js index cc7e841de251..78f636a7ad74 100644 --- a/src/libs/ReportUtils.js +++ b/src/libs/ReportUtils.js @@ -8,6 +8,8 @@ import Onyx from 'react-native-onyx'; import _ from 'underscore'; import * as Expensicons from '@components/Icon/Expensicons'; import * as defaultWorkspaceAvatars from '@components/Icon/WorkspaceDefaultAvatars'; +// eslint-disable-next-line @dword-design/import-alias/prefer-alias +import Permissions from '@libs/Permissions'; import CONST from '@src/CONST'; import ONYXKEYS from '@src/ONYXKEYS'; import ROUTES from '@src/ROUTES'; @@ -20,7 +22,6 @@ import * as Localize from './Localize'; import linkingConfig from './Navigation/linkingConfig'; import Navigation from './Navigation/Navigation'; import * as NumberUtils from './NumberUtils'; -import Permissions from './Permissions'; import * as PolicyUtils from './PolicyUtils'; import * as ReportActionsUtils from './ReportActionsUtils'; import * as TransactionUtils from './TransactionUtils'; From f7dc7311cba335f50fc44fa36616a9e082f121c1 Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Wed, 22 Nov 2023 09:50:25 -0500 Subject: [PATCH 005/117] Correctly pass down betas --- src/components/LHNOptionsList/LHNOptionsList.js | 11 ++++++++++- src/components/LHNOptionsList/OptionRowLHNData.js | 2 +- src/libs/ReportUtils.js | 10 +++++----- src/libs/SidebarUtils.ts | 4 +++- 4 files changed, 19 insertions(+), 8 deletions(-) diff --git a/src/components/LHNOptionsList/LHNOptionsList.js b/src/components/LHNOptionsList/LHNOptionsList.js index 0d300c5e2179..605ad761240a 100644 --- a/src/components/LHNOptionsList/LHNOptionsList.js +++ b/src/components/LHNOptionsList/LHNOptionsList.js @@ -64,6 +64,9 @@ const propTypes = { transactions: PropTypes.objectOf(transactionPropTypes), /** List of draft comments */ draftComments: PropTypes.objectOf(PropTypes.string), + + /** The list of betas the user has access to */ + betas: PropTypes.arrayOf(PropTypes.string), ...withCurrentReportIDPropTypes, }; @@ -77,6 +80,7 @@ const defaultProps = { personalDetails: {}, transactions: {}, draftComments: {}, + betas: {}, ...withCurrentReportIDDefaultProps, }; @@ -97,6 +101,7 @@ function LHNOptionsList({ transactions, draftComments, currentReportID, + betas, }) { const styles = useThemeStyles(); /** @@ -134,10 +139,11 @@ function LHNOptionsList({ onSelectRow={onSelectRow} preferredLocale={preferredLocale} comment={itemComment} + betas={betas} /> ); }, - [currentReportID, draftComments, onSelectRow, optionMode, personalDetails, policy, preferredLocale, reportActions, reports, shouldDisableFocusOptions, transactions], + [currentReportID, draftComments, onSelectRow, optionMode, personalDetails, policy, preferredLocale, reportActions, reports, shouldDisableFocusOptions, transactions, betas], ); return ( @@ -186,5 +192,8 @@ export default compose( draftComments: { key: ONYXKEYS.COLLECTION.REPORT_DRAFT_COMMENT, }, + betas: { + key: ONYXKEYS.BETAS, + }, }), )(LHNOptionsList); diff --git a/src/components/LHNOptionsList/OptionRowLHNData.js b/src/components/LHNOptionsList/OptionRowLHNData.js index e11bfc3cab98..f3b0e78b0df0 100644 --- a/src/components/LHNOptionsList/OptionRowLHNData.js +++ b/src/components/LHNOptionsList/OptionRowLHNData.js @@ -87,7 +87,7 @@ function OptionRowLHNData({ const optionItem = useMemo(() => { // Note: ideally we'd have this as a dependent selector in onyx! - const item = SidebarUtils.getOptionData(fullReport, reportActions, personalDetails, preferredLocale, policy, parentReportAction); + const item = SidebarUtils.getOptionData(fullReport, reportActions, personalDetails, preferredLocale, policy, parentReportAction, propsToForward.betas); if (deepEqual(item, optionItemRef.current)) { return optionItemRef.current; } diff --git a/src/libs/ReportUtils.js b/src/libs/ReportUtils.js index 78f636a7ad74..bac1b45440c3 100644 --- a/src/libs/ReportUtils.js +++ b/src/libs/ReportUtils.js @@ -8,8 +8,6 @@ import Onyx from 'react-native-onyx'; import _ from 'underscore'; import * as Expensicons from '@components/Icon/Expensicons'; import * as defaultWorkspaceAvatars from '@components/Icon/WorkspaceDefaultAvatars'; -// eslint-disable-next-line @dword-design/import-alias/prefer-alias -import Permissions from '@libs/Permissions'; import CONST from '@src/CONST'; import ONYXKEYS from '@src/ONYXKEYS'; import ROUTES from '@src/ROUTES'; @@ -22,6 +20,7 @@ import * as Localize from './Localize'; import linkingConfig from './Navigation/linkingConfig'; import Navigation from './Navigation/Navigation'; import * as NumberUtils from './NumberUtils'; +import Permissions from './Permissions'; import * as PolicyUtils from './PolicyUtils'; import * as ReportActionsUtils from './ReportActionsUtils'; import * as TransactionUtils from './TransactionUtils'; @@ -3361,14 +3360,15 @@ function transactionHasViolation(transactionID) { /** * * @param {Object} report + * @param {Array | null} betas * @returns {Boolean} */ -function transactionThreadHasViolations(report) { - if (!Permissions.canUseViolations()) { +function transactionThreadHasViolations(report, betas) { + if (!Permissions.canUseViolations(betas)) { return false; } - // eslint-disable-next-line es/no-nullish-coalescing-operators + if (!report.parentReportActionID) { return false; } diff --git a/src/libs/SidebarUtils.ts b/src/libs/SidebarUtils.ts index 2940a6286a32..510ca2347a14 100644 --- a/src/libs/SidebarUtils.ts +++ b/src/libs/SidebarUtils.ts @@ -299,6 +299,7 @@ function getOptionData( preferredLocale: ValueOf, policy: Policy, parentReportAction: ReportAction, + betas: Beta[], ): OptionData | undefined { // When a user signs out, Onyx is cleared. Due to the lazy rendering with a virtual list, it's possible for // this method to be called after the Onyx data has been cleared out. In that case, it's fine to do @@ -361,7 +362,8 @@ function getOptionData( result.shouldShowSubscript = ReportUtils.shouldReportShowSubscript(report); result.pendingAction = report.pendingFields ? report.pendingFields.addWorkspaceRoom || report.pendingFields.createChat : null; result.allReportErrors = OptionsListUtils.getAllReportErrors(report, reportActions) as OnyxCommon.Errors; - result.brickRoadIndicator = Object.keys(result.allReportErrors ?? {}).length !== 0 || ReportUtils.transactionThreadHasViolations(report) ? CONST.BRICK_ROAD_INDICATOR_STATUS.ERROR : ''; + result.brickRoadIndicator = + Object.keys(result.allReportErrors ?? {}).length !== 0 || ReportUtils.transactionThreadHasViolations(report, betas) ? CONST.BRICK_ROAD_INDICATOR_STATUS.ERROR : ''; result.ownerAccountID = report.ownerAccountID; result.managerID = report.managerID; result.reportID = report.reportID; From 93ea5b7de9f4b9ef3409e1250549590ae1eabc29 Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Wed, 22 Nov 2023 13:38:37 -0500 Subject: [PATCH 006/117] Update tests --- src/libs/__mocks__/Permissions.ts | 1 + tests/unit/SidebarFilterTest.js | 11 ++++++++--- 2 files changed, 9 insertions(+), 3 deletions(-) diff --git a/src/libs/__mocks__/Permissions.ts b/src/libs/__mocks__/Permissions.ts index e95d13f52803..23939d037f9a 100644 --- a/src/libs/__mocks__/Permissions.ts +++ b/src/libs/__mocks__/Permissions.ts @@ -13,4 +13,5 @@ export default { canUseDefaultRooms: (betas: Beta[]) => betas.includes(CONST.BETAS.DEFAULT_ROOMS), canUsePolicyRooms: (betas: Beta[]) => betas.includes(CONST.BETAS.POLICY_ROOMS), canUseCustomStatus: (betas: Beta[]) => betas.includes(CONST.BETAS.CUSTOM_STATUS), + canUseViolations: (betas: Beta[]) => betas.includes(CONST.BETAS.VIOLATIONS), }; diff --git a/tests/unit/SidebarFilterTest.js b/tests/unit/SidebarFilterTest.js index 23a958e3aa9d..f4a4a64a3907 100644 --- a/tests/unit/SidebarFilterTest.js +++ b/tests/unit/SidebarFilterTest.js @@ -111,6 +111,7 @@ describe('Sidebar', () => { // When Onyx is updated to contain that report .then(() => Onyx.multiSet({ + [ONYXKEYS.BETAS]: [CONST.BETAS.VIOLATIONS], [`${ONYXKEYS.COLLECTION.REPORT}${report.reportID}`]: report, [ONYXKEYS.PERSONAL_DETAILS_LIST]: LHNTestUtils.fakePersonalDetails, [ONYXKEYS.IS_LOADING_REPORT_DATA]: false, @@ -141,7 +142,7 @@ describe('Sidebar', () => { // When Onyx is updated to contain that data and the sidebar re-renders .then(() => Onyx.multiSet({ - [ONYXKEYS.BETAS]: [], + [ONYXKEYS.BETAS]: [CONST.BETAS.VIOLATIONS], [ONYXKEYS.PERSONAL_DETAILS_LIST]: LHNTestUtils.fakePersonalDetails, [ONYXKEYS.IS_LOADING_REPORT_DATA]: false, [`${ONYXKEYS.COLLECTION.REPORT}${report.reportID}`]: report, @@ -194,7 +195,7 @@ describe('Sidebar', () => { // When Onyx is updated to contain that data and the sidebar re-renders .then(() => Onyx.multiSet({ - [ONYXKEYS.BETAS]: [], + [ONYXKEYS.BETAS]: [CONST.BETAS.VIOLATIONS], [ONYXKEYS.PERSONAL_DETAILS_LIST]: LHNTestUtils.fakePersonalDetails, [ONYXKEYS.IS_LOADING_REPORT_DATA]: false, [`${ONYXKEYS.COLLECTION.REPORT}${report1.reportID}`]: report1, @@ -246,7 +247,7 @@ describe('Sidebar', () => { // When Onyx is updated to contain that data and the sidebar re-renders .then(() => Onyx.multiSet({ - [ONYXKEYS.BETAS]: [], + [ONYXKEYS.BETAS]: [CONST.BETAS.VIOLATIONS], [ONYXKEYS.PERSONAL_DETAILS_LIST]: LHNTestUtils.fakePersonalDetails, [ONYXKEYS.IS_LOADING_REPORT_DATA]: false, [`${ONYXKEYS.COLLECTION.REPORT}${report.reportID}`]: report, @@ -379,6 +380,7 @@ describe('Sidebar', () => { // When Onyx is updated to contain that data and the sidebar re-renders .then(() => Onyx.multiSet({ + [ONYXKEYS.BETAS]: [CONST.BETAS.VIOLATIONS], [ONYXKEYS.NVP_PRIORITY_MODE]: CONST.PRIORITY_MODE.GSD, [ONYXKEYS.PERSONAL_DETAILS_LIST]: LHNTestUtils.fakePersonalDetails, [ONYXKEYS.IS_LOADING_REPORT_DATA]: false, @@ -445,11 +447,14 @@ describe('Sidebar', () => { }; LHNTestUtils.getDefaultRenderedSidebarLinks(draftReport.reportID); + const betas = [CONST.BETAS.VIOLATIONS]; + return ( waitForBatchedUpdates() // When Onyx is updated to contain that data and the sidebar re-renders .then(() => Onyx.multiSet({ + [ONYXKEYS.BETAS]: betas, [ONYXKEYS.NVP_PRIORITY_MODE]: CONST.PRIORITY_MODE.GSD, [ONYXKEYS.PERSONAL_DETAILS_LIST]: LHNTestUtils.fakePersonalDetails, [ONYXKEYS.IS_LOADING_REPORT_DATA]: false, From 1f9d6458b700bb00e5e1fb68fcda8171ee2b7643 Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Mon, 27 Nov 2023 13:19:05 -0500 Subject: [PATCH 007/117] Fixing some beta props --- src/components/LHNOptionsList/OptionRowLHNData.js | 5 +++-- src/libs/ReportUtils.js | 4 ++-- tests/unit/SidebarFilterTest.js | 4 +--- 3 files changed, 6 insertions(+), 7 deletions(-) diff --git a/src/components/LHNOptionsList/OptionRowLHNData.js b/src/components/LHNOptionsList/OptionRowLHNData.js index f3b0e78b0df0..765c4998adce 100644 --- a/src/components/LHNOptionsList/OptionRowLHNData.js +++ b/src/components/LHNOptionsList/OptionRowLHNData.js @@ -73,6 +73,7 @@ function OptionRowLHNData({ receiptTransactions, parentReportAction, transaction, + betas, ...propsToForward }) { const reportID = propsToForward.reportID; @@ -87,7 +88,7 @@ function OptionRowLHNData({ const optionItem = useMemo(() => { // Note: ideally we'd have this as a dependent selector in onyx! - const item = SidebarUtils.getOptionData(fullReport, reportActions, personalDetails, preferredLocale, policy, parentReportAction, propsToForward.betas); + const item = SidebarUtils.getOptionData(fullReport, reportActions, personalDetails, preferredLocale, policy, parentReportAction, betas); if (deepEqual(item, optionItemRef.current)) { return optionItemRef.current; } @@ -96,7 +97,7 @@ function OptionRowLHNData({ // Listen parentReportAction to update title of thread report when parentReportAction changed // Listen to transaction to update title of transaction report when transaction changed // eslint-disable-next-line react-hooks/exhaustive-deps - }, [fullReport, linkedTransaction, reportActions, personalDetails, preferredLocale, policy, parentReportAction, transaction]); + }, [fullReport, linkedTransaction, reportActions, personalDetails, preferredLocale, policy, parentReportAction, transaction, betas]); useEffect(() => { if (!optionItem || optionItem.hasDraftComment || !comment || comment.length <= 0 || isFocused) { diff --git a/src/libs/ReportUtils.js b/src/libs/ReportUtils.js index 32588d9d6f97..e9362bc8afcc 100644 --- a/src/libs/ReportUtils.js +++ b/src/libs/ReportUtils.js @@ -8,6 +8,8 @@ import Onyx from 'react-native-onyx'; import _ from 'underscore'; import * as Expensicons from '@components/Icon/Expensicons'; import * as defaultWorkspaceAvatars from '@components/Icon/WorkspaceDefaultAvatars'; +// eslint-disable-next-line @dword-design/import-alias/prefer-alias +import Permissions from '@libs/Permissions'; import CONST from '@src/CONST'; import ONYXKEYS from '@src/ONYXKEYS'; import ROUTES from '@src/ROUTES'; @@ -20,7 +22,6 @@ import * as Localize from './Localize'; import linkingConfig from './Navigation/linkingConfig'; import Navigation from './Navigation/Navigation'; import * as NumberUtils from './NumberUtils'; -import Permissions from './Permissions'; import * as PolicyUtils from './PolicyUtils'; import * as ReportActionsUtils from './ReportActionsUtils'; import * as TransactionUtils from './TransactionUtils'; @@ -3399,7 +3400,6 @@ function transactionThreadHasViolations(report, betas) { if (!Permissions.canUseViolations(betas)) { return false; } - if (!report.parentReportActionID) { return false; } diff --git a/tests/unit/SidebarFilterTest.js b/tests/unit/SidebarFilterTest.js index f4a4a64a3907..f54b2a6f5c95 100644 --- a/tests/unit/SidebarFilterTest.js +++ b/tests/unit/SidebarFilterTest.js @@ -447,14 +447,12 @@ describe('Sidebar', () => { }; LHNTestUtils.getDefaultRenderedSidebarLinks(draftReport.reportID); - const betas = [CONST.BETAS.VIOLATIONS]; - return ( waitForBatchedUpdates() // When Onyx is updated to contain that data and the sidebar re-renders .then(() => Onyx.multiSet({ - [ONYXKEYS.BETAS]: betas, + [ONYXKEYS.BETAS]: [CONST.BETAS.VIOLATIONS], [ONYXKEYS.NVP_PRIORITY_MODE]: CONST.PRIORITY_MODE.GSD, [ONYXKEYS.PERSONAL_DETAILS_LIST]: LHNTestUtils.fakePersonalDetails, [ONYXKEYS.IS_LOADING_REPORT_DATA]: false, From 6d47be552a2266c9c57c7768716790a7675e095a Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Mon, 27 Nov 2023 16:34:41 -0500 Subject: [PATCH 008/117] Started TS conversion --- src/ONYXKEYS.ts | 3 +++ src/libs/ReportUtils.ts | 7 ++--- src/types/onyx/TransactionViolation.ts | 36 ++++++++++++++++++++++++++ 3 files changed, 43 insertions(+), 3 deletions(-) create mode 100644 src/types/onyx/TransactionViolation.ts diff --git a/src/ONYXKEYS.ts b/src/ONYXKEYS.ts index 5576eb64736d..cf98433da76d 100755 --- a/src/ONYXKEYS.ts +++ b/src/ONYXKEYS.ts @@ -266,6 +266,9 @@ const ONYXKEYS = { SECURITY_GROUP: 'securityGroup_', TRANSACTION: 'transactions_', + // Transaction Violations + TRANSACTION_VIOLATIONS: 'transactionViolations_', + // Holds temporary transactions used during the creation and edit flow TRANSACTION_DRAFT: 'transactionsDraft_', SPLIT_TRANSACTION_DRAFT: 'splitTransactionDraft_', diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index 8beeff76c5da..7cd21e19ab0f 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -21,6 +21,7 @@ import {Errors, Icon, PendingAction} from '@src/types/onyx/OnyxCommon'; import {ChangeLog, IOUMessage, OriginalMessageActionName} from '@src/types/onyx/OriginalMessage'; import {Message, ReportActions} from '@src/types/onyx/ReportAction'; import {Receipt, WaypointCollection} from '@src/types/onyx/Transaction'; +import {TransactionViolation, TransactionViolations} from '@src/types/onyx/TransactionViolation'; import DeepValueOf from '@src/types/utils/DeepValueOf'; import {EmptyObject, isEmptyObject, isNotEmptyObject} from '@src/types/utils/EmptyObject'; import * as CollectionUtils from './CollectionUtils'; @@ -402,10 +403,10 @@ Onyx.connect({ callback: (value) => (loginList = value), }); -const transactionViolations = {}; +const transactionViolations: OnyxCollection = {}; Onyx.connect({ key: ONYXKEYS.COLLECTION.TRANSACTION_VIOLATIONS, - callback: (violations, key) => { + callback: (violations: OnyxCollection, key) => { if (!key || !violations) { return; } @@ -415,7 +416,7 @@ Onyx.connect({ }, }); -const reportActions = {}; +const reportActions: OnyxCollection = {}; Onyx.connect({ key: ONYXKEYS.COLLECTION.REPORT_ACTIONS, callback: (actions, key) => { diff --git a/src/types/onyx/TransactionViolation.ts b/src/types/onyx/TransactionViolation.ts new file mode 100644 index 000000000000..fa7f212aa794 --- /dev/null +++ b/src/types/onyx/TransactionViolation.ts @@ -0,0 +1,36 @@ +/** + * @module TransactionViolation + * @description Transaction Violation + */ + +/** + * Names of the various Transaction Violation types + */ +type ViolationName = + | 'perDayLimit' + | 'maxAge' + | 'overLimit' + | 'overLimitAttendee' + | 'overCategoryLimit' + | 'receiptRequired' + | 'missingCategory' + | 'categoryOutOfPolicy' + | 'missingTag' + | 'tagOutOfPolicy' + | 'missingComment' + | 'taxRequired' + | 'taxOutOfPolicy' + | 'billableExpense'; + +type ViolationType = string; + +type TransactionViolation = { + type: ViolationType; + name: ViolationName; + userMessage: string; + data?: Record; +}; + +type TransactionViolations = Record; + +export type {TransactionViolation, TransactionViolations, ViolationName, ViolationType}; From 58498036860ed27e1e2848dc6213ade02d5d7777 Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Tue, 28 Nov 2023 21:35:03 -0500 Subject: [PATCH 009/117] Fixed up the Onyx Types for Violations & I think I got the functions working --- src/ONYXKEYS.ts | 1 + src/libs/ReportUtils.ts | 72 ++++++++++++-------------- src/types/onyx/OriginalMessage.ts | 2 +- src/types/onyx/TransactionViolation.ts | 57 +++++++++++++------- src/types/onyx/index.ts | 2 + 5 files changed, 75 insertions(+), 59 deletions(-) diff --git a/src/ONYXKEYS.ts b/src/ONYXKEYS.ts index cf98433da76d..62067a864ed5 100755 --- a/src/ONYXKEYS.ts +++ b/src/ONYXKEYS.ts @@ -451,6 +451,7 @@ type OnyxValues = { [ONYXKEYS.COLLECTION.TRANSACTION]: OnyxTypes.Transaction; [ONYXKEYS.COLLECTION.POLICY_RECENTLY_USED_TAGS]: OnyxTypes.RecentlyUsedTags; [ONYXKEYS.COLLECTION.SELECTED_TAB]: string; + [ONYXKEYS.COLLECTION.TRANSACTION_VIOLATIONS]: OnyxTypes.TransactionViolation[]; // Forms [ONYXKEYS.FORMS.ADD_DEBIT_CARD_FORM]: OnyxTypes.AddDebitCardForm; diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index 7cd21e19ab0f..af57a2969b5b 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -16,12 +16,11 @@ import CONST from '@src/CONST'; import {TranslationPaths} from '@src/languages/types'; import ONYXKEYS from '@src/ONYXKEYS'; import ROUTES from '@src/ROUTES'; -import {Beta, Login, PersonalDetails, Policy, PolicyTags, Report, ReportAction, Transaction} from '@src/types/onyx'; +import {Beta, Login, PersonalDetails, Policy, PolicyTags, Report, ReportAction, Transaction, TransactionViolation} from '@src/types/onyx'; import {Errors, Icon, PendingAction} from '@src/types/onyx/OnyxCommon'; -import {ChangeLog, IOUMessage, OriginalMessageActionName} from '@src/types/onyx/OriginalMessage'; +import OriginalMessage, {ChangeLog, IOUMessage, OriginalMessageActionName, OriginalMessageIOU} from '@src/types/onyx/OriginalMessage'; import {Message, ReportActions} from '@src/types/onyx/ReportAction'; import {Receipt, WaypointCollection} from '@src/types/onyx/Transaction'; -import {TransactionViolation, TransactionViolations} from '@src/types/onyx/TransactionViolation'; import DeepValueOf from '@src/types/utils/DeepValueOf'; import {EmptyObject, isEmptyObject, isNotEmptyObject} from '@src/types/utils/EmptyObject'; import * as CollectionUtils from './CollectionUtils'; @@ -403,11 +402,16 @@ Onyx.connect({ callback: (value) => (loginList = value), }); -const transactionViolations: OnyxCollection = {}; +const transactionViolations: OnyxCollection = {}; Onyx.connect({ key: ONYXKEYS.COLLECTION.TRANSACTION_VIOLATIONS, - callback: (violations: OnyxCollection, key) => { - if (!key || !violations) { + callback: (violations, key) => { + if (!key) { + return; + } + + if (!violations) { + delete transactionViolations[key]; return; } @@ -3396,43 +3400,36 @@ function shouldHideReport(report: OnyxEntry, currentReportId: string): b return parentReport?.reportID !== report?.reportID && !isChildReportHasComment; } -/** - * @param {String} transactionID - * @returns {Boolean} - */ - -function transactionHasViolation(transactionID) { - const violations = lodashGet(transactionViolations, transactionID, []); - return _.some(violations, (violation) => violation.type === 'violation'); +function transactionHasViolation(transactionID: string): boolean { + const violations = transactionViolations ? transactionViolations?.[transactionID] : []; + if (!violations) { + return false; + } + return violations.some((violation: TransactionViolation) => violation.type === 'violation'); } -/** - * - * @param {Object} report - * @param {Array | null} betas - * @returns {Boolean} - */ +function isOriginalMessageIOU(message: OriginalMessage): message is OriginalMessageIOU { + return (message as OriginalMessageIOU).actionName === CONST.REPORT.ACTIONS.TYPE.IOU; +} -function transactionThreadHasViolations(report, betas) { - if (!Permissions.canUseViolations(betas)) { +function transactionThreadHasViolations(report: Report, betas: Beta[]): boolean { + if (!Permissions.canUseViolations(betas) || !reportActions) { return false; } if (!report.parentReportActionID) { return false; } - - const parentReportAction = lodashGet(reportActions, `${report.parentReportID}.${report.parentReportActionID}`); + const parentReportAction = reportActions[`${report.parentReportID}`]?.[`${report.parentReportActionID}`]; if (!parentReportAction) { return false; } - // eslint-disable-next-line es/no-nullish-coalescing-operators - const transactionID = parentReportAction.originalMessage.IOUTransactionID ?? 0; - if (!transactionID) { - return false; - } - // eslint-disable-next-line es/no-nullish-coalescing-operators - const reportID = parentReportAction.originalMessage.IOUReportID ?? 0; - if (!reportID) { + const transactionID = isOriginalMessageIOU(parentReportAction?.originalMessage as OriginalMessage) + ? (parentReportAction.originalMessage as OriginalMessageIOU).originalMessage.IOUTransactionID + : ''; + const reportID = isOriginalMessageIOU(parentReportAction?.originalMessage as OriginalMessage) + ? (parentReportAction.originalMessage as OriginalMessageIOU).originalMessage.IOUReportID + : ''; + if (!transactionID || !reportID) { return false; } if (!isCurrentUserSubmitter(reportID)) { @@ -3441,14 +3438,9 @@ function transactionThreadHasViolations(report, betas) { return transactionHasViolation(transactionID); } -/** - * @param {String} reportID - * @returns {Boolean} - */ - -function reportHasViolations(reportID) { +function reportHasViolations(reportID: string): boolean { const transactions = TransactionUtils.getAllReportTransactions(reportID); - return _.some(transactions, (transaction) => transactionHasViolation(transaction.transactionID)); + return transactions.some((transaction) => transactionHasViolation(transaction.transactionID)); } /** @@ -3524,7 +3516,7 @@ function shouldReportBeInOptionList( } // Always show IOU reports with violations - if (isExpenseRequest(report) && transactionThreadHasViolations(report)) { + if (isExpenseRequest(report) && transactionThreadHasViolations(report, betas)) { return true; } diff --git a/src/types/onyx/OriginalMessage.ts b/src/types/onyx/OriginalMessage.ts index c5d9c27d34a1..6e12a7d286e7 100644 --- a/src/types/onyx/OriginalMessage.ts +++ b/src/types/onyx/OriginalMessage.ts @@ -223,4 +223,4 @@ type OriginalMessage = | OriginalMessageMoved; export default OriginalMessage; -export type {ChronosOOOEvent, Decision, Reaction, ActionName, IOUMessage, Closed, OriginalMessageActionName, ChangeLog}; +export type {ChronosOOOEvent, Decision, Reaction, ActionName, IOUMessage, Closed, OriginalMessageActionName, OriginalMessageIOU, ChangeLog}; diff --git a/src/types/onyx/TransactionViolation.ts b/src/types/onyx/TransactionViolation.ts index fa7f212aa794..02db3921c573 100644 --- a/src/types/onyx/TransactionViolation.ts +++ b/src/types/onyx/TransactionViolation.ts @@ -2,25 +2,36 @@ * @module TransactionViolation * @description Transaction Violation */ +import PropTypes from 'prop-types'; /** - * Names of the various Transaction Violation types + * Names of the various Transaction Violation types. + * Defined as an array so it can be used in `PropTypes.oneOf` */ -type ViolationName = - | 'perDayLimit' - | 'maxAge' - | 'overLimit' - | 'overLimitAttendee' - | 'overCategoryLimit' - | 'receiptRequired' - | 'missingCategory' - | 'categoryOutOfPolicy' - | 'missingTag' - | 'tagOutOfPolicy' - | 'missingComment' - | 'taxRequired' - | 'taxOutOfPolicy' - | 'billableExpense'; +const violationNames = [ + 'perDayLimit', + 'maxAge', + 'overLimit', + 'overLimitAttendee', + 'overCategoryLimit', + 'receiptRequired', + 'missingCategory', + 'categoryOutOfPolicy', + 'missingTag', + 'tagOutOfPolicy', + 'missingComment', + 'taxRequired', + 'taxOutOfPolicy', + 'billableExpense', +] as const; + +/** + * Names of the various Transaction Violation types. + * + * The list is first defined as an array so it can be used in `PropTypes.oneOf`, and + * converted to a union type here for use in typescript. + */ +type ViolationName = (typeof violationNames)[number]; type ViolationType = string; @@ -31,6 +42,16 @@ type TransactionViolation = { data?: Record; }; -type TransactionViolations = Record; +const transactionViolationPropType = PropTypes.shape({ + type: PropTypes.string.isRequired, + name: PropTypes.oneOf(violationNames).isRequired, + userMessage: PropTypes.string.isRequired, + data: PropTypes.objectOf(PropTypes.string), +}); + +const transactionViolationsPropTypes = PropTypes.arrayOf(transactionViolationPropType); + +export default TransactionViolation; +export {transactionViolationPropType, transactionViolationsPropTypes}; -export type {TransactionViolation, TransactionViolations, ViolationName, ViolationType}; +export type {ViolationName, ViolationType}; diff --git a/src/types/onyx/index.ts b/src/types/onyx/index.ts index e7b9c7661c79..ad1e1c17d6ca 100644 --- a/src/types/onyx/index.ts +++ b/src/types/onyx/index.ts @@ -42,6 +42,7 @@ import SecurityGroup from './SecurityGroup'; import Session from './Session'; import Task from './Task'; import Transaction from './Transaction'; +import TransactionViolation from './TransactionViolation'; import User from './User'; import UserLocation from './UserLocation'; import UserWallet from './UserWallet'; @@ -103,6 +104,7 @@ export type { Session, Task, Transaction, + TransactionViolation, User, UserWallet, WalletAdditionalDetails, From f7fa4bf321fd2851d37e5c35a26c5a6149b637bf Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Wed, 29 Nov 2023 09:30:35 -0500 Subject: [PATCH 010/117] Finally got the tests up and working correctly --- src/components/LHNOptionsList/LHNOptionsList.js | 3 ++- tests/unit/SidebarFilterTest.js | 6 ------ 2 files changed, 2 insertions(+), 7 deletions(-) diff --git a/src/components/LHNOptionsList/LHNOptionsList.js b/src/components/LHNOptionsList/LHNOptionsList.js index 605ad761240a..4a2d13d716c4 100644 --- a/src/components/LHNOptionsList/LHNOptionsList.js +++ b/src/components/LHNOptionsList/LHNOptionsList.js @@ -80,7 +80,7 @@ const defaultProps = { personalDetails: {}, transactions: {}, draftComments: {}, - betas: {}, + betas: [], ...withCurrentReportIDDefaultProps, }; @@ -194,6 +194,7 @@ export default compose( }, betas: { key: ONYXKEYS.BETAS, + initialValue: [], }, }), )(LHNOptionsList); diff --git a/tests/unit/SidebarFilterTest.js b/tests/unit/SidebarFilterTest.js index f54b2a6f5c95..362990e113ad 100644 --- a/tests/unit/SidebarFilterTest.js +++ b/tests/unit/SidebarFilterTest.js @@ -111,7 +111,6 @@ describe('Sidebar', () => { // When Onyx is updated to contain that report .then(() => Onyx.multiSet({ - [ONYXKEYS.BETAS]: [CONST.BETAS.VIOLATIONS], [`${ONYXKEYS.COLLECTION.REPORT}${report.reportID}`]: report, [ONYXKEYS.PERSONAL_DETAILS_LIST]: LHNTestUtils.fakePersonalDetails, [ONYXKEYS.IS_LOADING_REPORT_DATA]: false, @@ -142,7 +141,6 @@ describe('Sidebar', () => { // When Onyx is updated to contain that data and the sidebar re-renders .then(() => Onyx.multiSet({ - [ONYXKEYS.BETAS]: [CONST.BETAS.VIOLATIONS], [ONYXKEYS.PERSONAL_DETAILS_LIST]: LHNTestUtils.fakePersonalDetails, [ONYXKEYS.IS_LOADING_REPORT_DATA]: false, [`${ONYXKEYS.COLLECTION.REPORT}${report.reportID}`]: report, @@ -195,7 +193,6 @@ describe('Sidebar', () => { // When Onyx is updated to contain that data and the sidebar re-renders .then(() => Onyx.multiSet({ - [ONYXKEYS.BETAS]: [CONST.BETAS.VIOLATIONS], [ONYXKEYS.PERSONAL_DETAILS_LIST]: LHNTestUtils.fakePersonalDetails, [ONYXKEYS.IS_LOADING_REPORT_DATA]: false, [`${ONYXKEYS.COLLECTION.REPORT}${report1.reportID}`]: report1, @@ -247,7 +244,6 @@ describe('Sidebar', () => { // When Onyx is updated to contain that data and the sidebar re-renders .then(() => Onyx.multiSet({ - [ONYXKEYS.BETAS]: [CONST.BETAS.VIOLATIONS], [ONYXKEYS.PERSONAL_DETAILS_LIST]: LHNTestUtils.fakePersonalDetails, [ONYXKEYS.IS_LOADING_REPORT_DATA]: false, [`${ONYXKEYS.COLLECTION.REPORT}${report.reportID}`]: report, @@ -380,7 +376,6 @@ describe('Sidebar', () => { // When Onyx is updated to contain that data and the sidebar re-renders .then(() => Onyx.multiSet({ - [ONYXKEYS.BETAS]: [CONST.BETAS.VIOLATIONS], [ONYXKEYS.NVP_PRIORITY_MODE]: CONST.PRIORITY_MODE.GSD, [ONYXKEYS.PERSONAL_DETAILS_LIST]: LHNTestUtils.fakePersonalDetails, [ONYXKEYS.IS_LOADING_REPORT_DATA]: false, @@ -452,7 +447,6 @@ describe('Sidebar', () => { // When Onyx is updated to contain that data and the sidebar re-renders .then(() => Onyx.multiSet({ - [ONYXKEYS.BETAS]: [CONST.BETAS.VIOLATIONS], [ONYXKEYS.NVP_PRIORITY_MODE]: CONST.PRIORITY_MODE.GSD, [ONYXKEYS.PERSONAL_DETAILS_LIST]: LHNTestUtils.fakePersonalDetails, [ONYXKEYS.IS_LOADING_REPORT_DATA]: false, From 02fa71460eaddce5f1c10d543c4702005da7963d Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Wed, 29 Nov 2023 10:31:23 -0500 Subject: [PATCH 011/117] Fixed types and property checks for violations --- src/libs/ReportUtils.ts | 17 ++++++----------- 1 file changed, 6 insertions(+), 11 deletions(-) diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index af57a2969b5b..d06a8baa2dae 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -3401,17 +3401,13 @@ function shouldHideReport(report: OnyxEntry, currentReportId: string): b } function transactionHasViolation(transactionID: string): boolean { - const violations = transactionViolations ? transactionViolations?.[transactionID] : []; + const violations = transactionViolations ? transactionViolations[transactionID]?.values : []; if (!violations) { return false; } return violations.some((violation: TransactionViolation) => violation.type === 'violation'); } -function isOriginalMessageIOU(message: OriginalMessage): message is OriginalMessageIOU { - return (message as OriginalMessageIOU).actionName === CONST.REPORT.ACTIONS.TYPE.IOU; -} - function transactionThreadHasViolations(report: Report, betas: Beta[]): boolean { if (!Permissions.canUseViolations(betas) || !reportActions) { return false; @@ -3423,12 +3419,11 @@ function transactionThreadHasViolations(report: Report, betas: Beta[]): boolean if (!parentReportAction) { return false; } - const transactionID = isOriginalMessageIOU(parentReportAction?.originalMessage as OriginalMessage) - ? (parentReportAction.originalMessage as OriginalMessageIOU).originalMessage.IOUTransactionID - : ''; - const reportID = isOriginalMessageIOU(parentReportAction?.originalMessage as OriginalMessage) - ? (parentReportAction.originalMessage as OriginalMessageIOU).originalMessage.IOUReportID - : ''; + if (parentReportAction.actionName !== CONST.REPORT.ACTIONS.TYPE.IOU) { + return false; + } + const transactionID = parentReportAction?.originalMessage?.IOUTransactionID; + const reportID = parentReportAction?.originalMessage?.IOUReportID; if (!transactionID || !reportID) { return false; } From 63ab54424aa8be7377a1f6e8c8ad0fc093659a5a Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Thu, 30 Nov 2023 10:36:15 -0500 Subject: [PATCH 012/117] Added some doc comments and also fixed the violation type --- src/ONYXKEYS.ts | 2 +- src/libs/ReportUtils.ts | 24 ++++++-- src/types/onyx/TransactionViolation.ts | 84 ++++++++++++-------------- src/types/onyx/index.ts | 3 +- 4 files changed, 61 insertions(+), 52 deletions(-) diff --git a/src/ONYXKEYS.ts b/src/ONYXKEYS.ts index 62067a864ed5..fc6d2927a1b1 100755 --- a/src/ONYXKEYS.ts +++ b/src/ONYXKEYS.ts @@ -451,7 +451,7 @@ type OnyxValues = { [ONYXKEYS.COLLECTION.TRANSACTION]: OnyxTypes.Transaction; [ONYXKEYS.COLLECTION.POLICY_RECENTLY_USED_TAGS]: OnyxTypes.RecentlyUsedTags; [ONYXKEYS.COLLECTION.SELECTED_TAB]: string; - [ONYXKEYS.COLLECTION.TRANSACTION_VIOLATIONS]: OnyxTypes.TransactionViolation[]; + [ONYXKEYS.COLLECTION.TRANSACTION_VIOLATIONS]: OnyxTypes.TransactionViolations; // Forms [ONYXKEYS.FORMS.ADD_DEBIT_CARD_FORM]: OnyxTypes.AddDebitCardForm; diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index d06a8baa2dae..e250d8ad83ff 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -16,9 +16,9 @@ import CONST from '@src/CONST'; import {TranslationPaths} from '@src/languages/types'; import ONYXKEYS from '@src/ONYXKEYS'; import ROUTES from '@src/ROUTES'; -import {Beta, Login, PersonalDetails, Policy, PolicyTags, Report, ReportAction, Transaction, TransactionViolation} from '@src/types/onyx'; +import {Beta, Login, PersonalDetails, Policy, PolicyTags, Report, ReportAction, Transaction, TransactionViolation, TransactionViolations} from '@src/types/onyx'; import {Errors, Icon, PendingAction} from '@src/types/onyx/OnyxCommon'; -import OriginalMessage, {ChangeLog, IOUMessage, OriginalMessageActionName, OriginalMessageIOU} from '@src/types/onyx/OriginalMessage'; +import {ChangeLog, IOUMessage, OriginalMessageActionName} from '@src/types/onyx/OriginalMessage'; import {Message, ReportActions} from '@src/types/onyx/ReportAction'; import {Receipt, WaypointCollection} from '@src/types/onyx/Transaction'; import DeepValueOf from '@src/types/utils/DeepValueOf'; @@ -402,7 +402,7 @@ Onyx.connect({ callback: (value) => (loginList = value), }); -const transactionViolations: OnyxCollection = {}; +const transactionViolations: OnyxCollection = {}; Onyx.connect({ key: ONYXKEYS.COLLECTION.TRANSACTION_VIOLATIONS, callback: (violations, key) => { @@ -3400,14 +3400,24 @@ function shouldHideReport(report: OnyxEntry, currentReportId: string): b return parentReport?.reportID !== report?.reportID && !isChildReportHasComment; } +/** + * + * Check if there are any violations belonging to the transaction in the transactionsViolations Onyx object + * then checks that the violation is of the proper type + */ function transactionHasViolation(transactionID: string): boolean { - const violations = transactionViolations ? transactionViolations[transactionID]?.values : []; + const violations = transactionViolations ? transactionViolations[transactionID]?.value : []; if (!violations) { return false; } return violations.some((violation: TransactionViolation) => violation.type === 'violation'); } +/** + * Checks to see if a report's parentAction is a money request that contains a violation + * This only pertains to report's that a user submitted and if it is open or processing + */ + function transactionThreadHasViolations(report: Report, betas: Beta[]): boolean { if (!Permissions.canUseViolations(betas) || !reportActions) { return false; @@ -3430,9 +3440,15 @@ function transactionThreadHasViolations(report: Report, betas: Beta[]): boolean if (!isCurrentUserSubmitter(reportID)) { return false; } + if (report?.stateNum !== CONST.REPORT.STATE_NUM.OPEN && report?.stateNum !== CONST.REPORT.STATE_NUM.PROCESSING) { + return false; + } return transactionHasViolation(transactionID); } +/** + * Checks to see if a report contains a violation + */ function reportHasViolations(reportID: string): boolean { const transactions = TransactionUtils.getAllReportTransactions(reportID); return transactions.some((transaction) => transactionHasViolation(transaction.transactionID)); diff --git a/src/types/onyx/TransactionViolation.ts b/src/types/onyx/TransactionViolation.ts index 02db3921c573..1bb1f179f601 100644 --- a/src/types/onyx/TransactionViolation.ts +++ b/src/types/onyx/TransactionViolation.ts @@ -1,57 +1,49 @@ /** - * @module TransactionViolation - * @description Transaction Violation + * Names of transaction violations */ -import PropTypes from 'prop-types'; -/** - * Names of the various Transaction Violation types. - * Defined as an array so it can be used in `PropTypes.oneOf` - */ -const violationNames = [ - 'perDayLimit', - 'maxAge', - 'overLimit', - 'overLimitAttendee', - 'overCategoryLimit', - 'receiptRequired', - 'missingCategory', - 'categoryOutOfPolicy', - 'missingTag', - 'tagOutOfPolicy', - 'missingComment', - 'taxRequired', - 'taxOutOfPolicy', - 'billableExpense', -] as const; - -/** - * Names of the various Transaction Violation types. - * - * The list is first defined as an array so it can be used in `PropTypes.oneOf`, and - * converted to a union type here for use in typescript. - */ -type ViolationName = (typeof violationNames)[number]; - -type ViolationType = string; +type ViolationName = + | 'allTagLevelsRequired' + | 'autoReportedRejectedExpense' + | 'billableExpense' + | 'cashExpenseWithNoReceipt' + | 'categoryOutOfPolicy' + | 'conversionSurcharge' + | 'customUnitOutOfPolicy' + | 'duplicatedTransaction' + | 'fieldRequired' + | 'futureDate' + | 'invoiceMarkup' + | 'maxAge' + | 'missingCategory' + | 'missingComment' + | 'missingTag' + | 'modifiedAmount' + | 'modifiedDate' + | 'nonExpensiworksExpense' + | 'overAutoApprovalLimit' + | 'overCategoryLimit' + | 'overLimit' + | 'overLimitAttendee' + | 'perDayLimit' + | 'receiptNotSmartScanned' + | 'receiptRequired' + | 'rter' + | 'smartscanFailed' + | 'someTagLevelsRequired' + | 'tagOutOfPolicy' + | 'taxAmountChanged' + | 'taxOutOfPolicy' + | 'taxRateChanged' + | 'taxRequired'; type TransactionViolation = { - type: ViolationType; + type: string; name: ViolationName; userMessage: string; data?: Record; }; -const transactionViolationPropType = PropTypes.shape({ - type: PropTypes.string.isRequired, - name: PropTypes.oneOf(violationNames).isRequired, - userMessage: PropTypes.string.isRequired, - data: PropTypes.objectOf(PropTypes.string), -}); - -const transactionViolationsPropTypes = PropTypes.arrayOf(transactionViolationPropType); - -export default TransactionViolation; -export {transactionViolationPropType, transactionViolationsPropTypes}; +type TransactionViolations = Record; -export type {ViolationName, ViolationType}; +export type {TransactionViolation, TransactionViolations, ViolationName}; diff --git a/src/types/onyx/index.ts b/src/types/onyx/index.ts index ad1e1c17d6ca..9e81f2025255 100644 --- a/src/types/onyx/index.ts +++ b/src/types/onyx/index.ts @@ -42,7 +42,7 @@ import SecurityGroup from './SecurityGroup'; import Session from './Session'; import Task from './Task'; import Transaction from './Transaction'; -import TransactionViolation from './TransactionViolation'; +import {TransactionViolation, TransactionViolations} from './TransactionViolation'; import User from './User'; import UserLocation from './UserLocation'; import UserWallet from './UserWallet'; @@ -105,6 +105,7 @@ export type { Task, Transaction, TransactionViolation, + TransactionViolations, User, UserWallet, WalletAdditionalDetails, From c69c1df69a7f79a306af37339e07b248dda663ad Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Thu, 30 Nov 2023 11:13:28 -0500 Subject: [PATCH 013/117] Fixed perf test --- tests/perf-test/SidebarUtils.perf-test.ts | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/tests/perf-test/SidebarUtils.perf-test.ts b/tests/perf-test/SidebarUtils.perf-test.ts index 7f9957232cfb..fb3cac24053a 100644 --- a/tests/perf-test/SidebarUtils.perf-test.ts +++ b/tests/perf-test/SidebarUtils.perf-test.ts @@ -3,7 +3,7 @@ import {measureFunction} from 'reassure'; import SidebarUtils from '@libs/SidebarUtils'; import CONST from '@src/CONST'; import ONYXKEYS from '@src/ONYXKEYS'; -import {PersonalDetails} from '@src/types/onyx'; +import {Beta, PersonalDetails} from '@src/types/onyx'; import Policy from '@src/types/onyx/Policy'; import Report from '@src/types/onyx/Report'; import ReportAction, {ReportActions} from '@src/types/onyx/ReportAction'; @@ -51,13 +51,14 @@ test('getOptionData on 5k reports', async () => { const preferredLocale = 'en'; const policy = createRandomPolicy(1); const parentReportAction = createRandomReportAction(1); + const betas: Beta[] = []; Onyx.multiSet({ ...mockedResponseMap, }); await waitForBatchedUpdates(); - await measureFunction(() => SidebarUtils.getOptionData(report, reportActions, personalDetails, preferredLocale, policy, parentReportAction), {runs}); + await measureFunction(() => SidebarUtils.getOptionData(report, reportActions, personalDetails, preferredLocale, policy, parentReportAction, betas), {runs}); }); test('getOrderedReportIDs on 5k reports', async () => { From 5e122efec4221d246d3c9ba87f34bd666454aa07 Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Thu, 30 Nov 2023 13:07:34 -0500 Subject: [PATCH 014/117] Removed unnecessary export --- src/types/onyx/OriginalMessage.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/types/onyx/OriginalMessage.ts b/src/types/onyx/OriginalMessage.ts index 6c86a6b1be69..0dc532ebeded 100644 --- a/src/types/onyx/OriginalMessage.ts +++ b/src/types/onyx/OriginalMessage.ts @@ -225,4 +225,4 @@ type OriginalMessage = | OriginalMessageMoved; export default OriginalMessage; -export type {ChronosOOOEvent, Decision, Reaction, ActionName, IOUMessage, Closed, OriginalMessageActionName, OriginalMessageIOU, ChangeLog}; +export type {ChronosOOOEvent, Decision, Reaction, ActionName, IOUMessage, Closed, OriginalMessageActionName, ChangeLog}; From e53902e3f60cd1f191e85ef85d8c7175d2fc54c0 Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Thu, 30 Nov 2023 13:09:29 -0500 Subject: [PATCH 015/117] Restore betas in test --- tests/unit/SidebarFilterTest.js | 3 +++ 1 file changed, 3 insertions(+) diff --git a/tests/unit/SidebarFilterTest.js b/tests/unit/SidebarFilterTest.js index 362990e113ad..23a958e3aa9d 100644 --- a/tests/unit/SidebarFilterTest.js +++ b/tests/unit/SidebarFilterTest.js @@ -141,6 +141,7 @@ describe('Sidebar', () => { // When Onyx is updated to contain that data and the sidebar re-renders .then(() => Onyx.multiSet({ + [ONYXKEYS.BETAS]: [], [ONYXKEYS.PERSONAL_DETAILS_LIST]: LHNTestUtils.fakePersonalDetails, [ONYXKEYS.IS_LOADING_REPORT_DATA]: false, [`${ONYXKEYS.COLLECTION.REPORT}${report.reportID}`]: report, @@ -193,6 +194,7 @@ describe('Sidebar', () => { // When Onyx is updated to contain that data and the sidebar re-renders .then(() => Onyx.multiSet({ + [ONYXKEYS.BETAS]: [], [ONYXKEYS.PERSONAL_DETAILS_LIST]: LHNTestUtils.fakePersonalDetails, [ONYXKEYS.IS_LOADING_REPORT_DATA]: false, [`${ONYXKEYS.COLLECTION.REPORT}${report1.reportID}`]: report1, @@ -244,6 +246,7 @@ describe('Sidebar', () => { // When Onyx is updated to contain that data and the sidebar re-renders .then(() => Onyx.multiSet({ + [ONYXKEYS.BETAS]: [], [ONYXKEYS.PERSONAL_DETAILS_LIST]: LHNTestUtils.fakePersonalDetails, [ONYXKEYS.IS_LOADING_REPORT_DATA]: false, [`${ONYXKEYS.COLLECTION.REPORT}${report.reportID}`]: report, From c49b03f103b401ddddf3b04b9912f152f84d7426 Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Thu, 30 Nov 2023 13:25:11 -0500 Subject: [PATCH 016/117] Switch to destructuring --- src/libs/ReportUtils.ts | 9 ++++----- 1 file changed, 4 insertions(+), 5 deletions(-) diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index 1412cfa73067..809a47195a36 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -3448,18 +3448,17 @@ function transactionThreadHasViolations(report: Report, betas: Beta[]): boolean if (parentReportAction.actionName !== CONST.REPORT.ACTIONS.TYPE.IOU) { return false; } - const transactionID = parentReportAction?.originalMessage?.IOUTransactionID; - const reportID = parentReportAction?.originalMessage?.IOUReportID; - if (!transactionID || !reportID) { + const {IOUTransactionID, IOUReportID} = parentReportAction?.originalMessage; + if (!IOUTransactionID || !IOUReportID) { return false; } - if (!isCurrentUserSubmitter(reportID)) { + if (!isCurrentUserSubmitter(IOUReportID)) { return false; } if (report?.stateNum !== CONST.REPORT.STATE_NUM.OPEN && report?.stateNum !== CONST.REPORT.STATE_NUM.PROCESSING) { return false; } - return transactionHasViolation(transactionID); + return transactionHasViolation(IOUTransactionID); } /** From 375eabbea4231cc09d418bc9e1b797a5b069295d Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Tue, 5 Dec 2023 10:47:27 -0500 Subject: [PATCH 017/117] Update src/libs/ReportUtils.ts Co-authored-by: Carlos Alvarez --- src/libs/ReportUtils.ts | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index 1412cfa73067..ad5ed0e46b20 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -3417,8 +3417,7 @@ function shouldHideReport(report: OnyxEntry, currentReportId: string): b } /** - * - * Check if there are any violations belonging to the transaction in the transactionsViolations Onyx object + * Checks if there are any violations belonging to the transaction in the transactionsViolations Onyx object * then checks that the violation is of the proper type */ function transactionHasViolation(transactionID: string): boolean { From b6c24db81675e801039333344d093f7d7e12adc2 Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Tue, 5 Dec 2023 10:47:37 -0500 Subject: [PATCH 018/117] Update src/libs/ReportUtils.ts Co-authored-by: Carlos Alvarez --- src/libs/ReportUtils.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index ad5ed0e46b20..0b6b95442cf4 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -3430,7 +3430,7 @@ function transactionHasViolation(transactionID: string): boolean { /** * Checks to see if a report's parentAction is a money request that contains a violation - * This only pertains to report's that a user submitted and if it is open or processing + * This only applies to report submitter and for reports in the open and processing states */ function transactionThreadHasViolations(report: Report, betas: Beta[]): boolean { From ccd3eb33afc33ae4d8e98293eb75f043ca502ac4 Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Tue, 5 Dec 2023 15:47:39 -0500 Subject: [PATCH 019/117] remove unneeded changes --- src/libs/ReportUtils.ts | 10 ++-------- 1 file changed, 2 insertions(+), 8 deletions(-) diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index b46cdfeabd34..c8ecb8576812 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -11,8 +11,6 @@ import {SvgProps} from 'react-native-svg'; import {ValueOf} from 'type-fest'; import * as Expensicons from '@components/Icon/Expensicons'; import * as defaultWorkspaceAvatars from '@components/Icon/WorkspaceDefaultAvatars'; -// eslint-disable-next-line @dword-design/import-alias/prefer-alias -import Permissions from '@libs/Permissions'; import CONST from '@src/CONST'; import {ParentNavigationSummaryParams, TranslationPaths} from '@src/languages/types'; import ONYXKEYS from '@src/ONYXKEYS'; @@ -33,6 +31,7 @@ import * as Localize from './Localize'; import linkingConfig from './Navigation/linkingConfig'; import Navigation from './Navigation/Navigation'; import * as NumberUtils from './NumberUtils'; +import Permissions from './Permissions'; import * as PolicyUtils from './PolicyUtils'; import * as ReportActionsUtils from './ReportActionsUtils'; import {LastVisibleMessage} from './ReportActionsUtils'; @@ -411,12 +410,7 @@ const transactionViolations: OnyxCollection = {}; Onyx.connect({ key: ONYXKEYS.COLLECTION.TRANSACTION_VIOLATIONS, callback: (violations, key) => { - if (!key) { - return; - } - - if (!violations) { - delete transactionViolations[key]; + if (!key || !violations) { return; } From 1d2cfa2caa60b595bc6d54a97a78595d6edeb225 Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Tue, 5 Dec 2023 16:13:20 -0500 Subject: [PATCH 020/117] Remove duplicate TRANSACTION_VIOLATIONS constant --- src/ONYXKEYS.ts | 1 - 1 file changed, 1 deletion(-) diff --git a/src/ONYXKEYS.ts b/src/ONYXKEYS.ts index 82b6838cc75a..fc6d2927a1b1 100755 --- a/src/ONYXKEYS.ts +++ b/src/ONYXKEYS.ts @@ -265,7 +265,6 @@ const ONYXKEYS = { REPORT_USER_IS_LEAVING_ROOM: 'reportUserIsLeavingRoom_', SECURITY_GROUP: 'securityGroup_', TRANSACTION: 'transactions_', - TRANSACTION_VIOLATIONS: 'transactionViolations_', // Transaction Violations TRANSACTION_VIOLATIONS: 'transactionViolations_', From bcd1ee862bf896246fb5165ec4ffe0c6fb6dbb45 Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Wed, 6 Dec 2023 10:49:28 -0500 Subject: [PATCH 021/117] Update src/ONYXKEYS.ts Co-authored-by: Carlos Alvarez --- src/ONYXKEYS.ts | 2 -- 1 file changed, 2 deletions(-) diff --git a/src/ONYXKEYS.ts b/src/ONYXKEYS.ts index fc6d2927a1b1..5f85b29cb281 100755 --- a/src/ONYXKEYS.ts +++ b/src/ONYXKEYS.ts @@ -265,8 +265,6 @@ const ONYXKEYS = { REPORT_USER_IS_LEAVING_ROOM: 'reportUserIsLeavingRoom_', SECURITY_GROUP: 'securityGroup_', TRANSACTION: 'transactions_', - - // Transaction Violations TRANSACTION_VIOLATIONS: 'transactionViolations_', // Holds temporary transactions used during the creation and edit flow From cae1e0d75c77a8bcb3f0a5c261e92ed231940614 Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Thu, 7 Dec 2023 15:13:59 -0500 Subject: [PATCH 022/117] Working on removing the onyx connect and using withOnyx --- .../LHNOptionsList/LHNOptionsList.js | 47 +++++++++--- .../LHNOptionsList/OptionRowLHNData.js | 7 +- .../ReportActionItem/ReportPreview.js | 24 ++++++- src/libs/OptionsListUtils.js | 4 +- src/libs/ReportUtils.ts | 71 ++++++++++--------- src/libs/SidebarUtils.ts | 12 ++-- 6 files changed, 112 insertions(+), 53 deletions(-) diff --git a/src/components/LHNOptionsList/LHNOptionsList.js b/src/components/LHNOptionsList/LHNOptionsList.js index 54830f5319f0..ab8101bddf45 100644 --- a/src/components/LHNOptionsList/LHNOptionsList.js +++ b/src/components/LHNOptionsList/LHNOptionsList.js @@ -8,6 +8,7 @@ import _ from 'underscore'; import participantPropTypes from '@components/participantPropTypes'; import transactionPropTypes from '@components/transactionPropTypes'; import withCurrentReportID, {withCurrentReportIDDefaultProps, withCurrentReportIDPropTypes} from '@components/withCurrentReportID'; +import usePermissions from '@hooks/usePermissions'; import compose from '@libs/compose'; import * as OptionsListUtils from '@libs/OptionsListUtils'; import * as ReportUtils from '@libs/ReportUtils'; @@ -66,8 +67,21 @@ const propTypes = { /** List of draft comments */ draftComments: PropTypes.objectOf(PropTypes.string), - /** The list of betas the user has access to */ - betas: PropTypes.arrayOf(PropTypes.string), + /** The list of transaction violations */ + transactionViolations: PropTypes.shape({ + violations: PropTypes.arrayOf( + PropTypes.shape({ + /** The transaction ID */ + transactionID: PropTypes.number, + + /** The transaction violation type */ + type: PropTypes.string, + + /** The transaction violation message */ + message: PropTypes.string, + }), + ), + }), ...withCurrentReportIDPropTypes, }; @@ -81,7 +95,7 @@ const defaultProps = { personalDetails: {}, transactions: {}, draftComments: {}, - betas: [], + transactionViolations: {}, ...withCurrentReportIDDefaultProps, }; @@ -102,9 +116,10 @@ function LHNOptionsList({ transactions, draftComments, currentReportID, - betas, + transactionViolations, }) { const styles = useThemeStyles(); + const {canUseViolations} = usePermissions(); /** * Function which renders a row in the list * @@ -142,11 +157,26 @@ function LHNOptionsList({ onSelectRow={onSelectRow} preferredLocale={preferredLocale} comment={itemComment} - betas={betas} + transactionViolations={transactionViolations} + canUseViolations={canUseViolations} /> ); }, - [currentReportID, draftComments, onSelectRow, optionMode, personalDetails, policy, preferredLocale, reportActions, reports, shouldDisableFocusOptions, transactions, betas], + [ + currentReportID, + draftComments, + onSelectRow, + optionMode, + personalDetails, + policy, + preferredLocale, + reportActions, + reports, + shouldDisableFocusOptions, + transactions, + transactionViolations, + canUseViolations, + ], ); return ( @@ -195,9 +225,8 @@ export default compose( draftComments: { key: ONYXKEYS.COLLECTION.REPORT_DRAFT_COMMENT, }, - betas: { - key: ONYXKEYS.BETAS, - initialValue: [], + transactionViolations: { + key: ONYXKEYS.COLLECTION.TRANSACTION_VIOLATION, }, }), )(LHNOptionsList); diff --git a/src/components/LHNOptionsList/OptionRowLHNData.js b/src/components/LHNOptionsList/OptionRowLHNData.js index 765c4998adce..975299bb8800 100644 --- a/src/components/LHNOptionsList/OptionRowLHNData.js +++ b/src/components/LHNOptionsList/OptionRowLHNData.js @@ -73,7 +73,8 @@ function OptionRowLHNData({ receiptTransactions, parentReportAction, transaction, - betas, + transactionViolations, + canUseViolations, ...propsToForward }) { const reportID = propsToForward.reportID; @@ -88,7 +89,7 @@ function OptionRowLHNData({ const optionItem = useMemo(() => { // Note: ideally we'd have this as a dependent selector in onyx! - const item = SidebarUtils.getOptionData(fullReport, reportActions, personalDetails, preferredLocale, policy, parentReportAction, betas); + const item = SidebarUtils.getOptionData(fullReport, reportActions, personalDetails, preferredLocale, policy, parentReportAction, transactionViolations, canUseViolations); if (deepEqual(item, optionItemRef.current)) { return optionItemRef.current; } @@ -97,7 +98,7 @@ function OptionRowLHNData({ // Listen parentReportAction to update title of thread report when parentReportAction changed // Listen to transaction to update title of transaction report when transaction changed // eslint-disable-next-line react-hooks/exhaustive-deps - }, [fullReport, linkedTransaction, reportActions, personalDetails, preferredLocale, policy, parentReportAction, transaction, betas]); + }, [fullReport, linkedTransaction, reportActions, personalDetails, preferredLocale, policy, parentReportAction, transaction, transactionViolations, canUseViolations]); useEffect(() => { if (!optionItem || optionItem.hasDraftComment || !comment || comment.length <= 0 || isFocused) { diff --git a/src/components/ReportActionItem/ReportPreview.js b/src/components/ReportActionItem/ReportPreview.js index 7f2b50534a2d..d6fdf099cbaa 100644 --- a/src/components/ReportActionItem/ReportPreview.js +++ b/src/components/ReportActionItem/ReportPreview.js @@ -91,6 +91,22 @@ const propTypes = { /** Whether a message is a whisper */ isWhisper: PropTypes.bool, + /** All of the transaction violations */ + transactionViolations: PropTypes.shape({ + violations: PropTypes.arrayOf( + PropTypes.shape({ + /** The transaction ID */ + transactionID: PropTypes.number, + + /** The transaction violation type */ + type: PropTypes.string, + + /** The transaction violation message */ + message: PropTypes.string, + }), + ), + }), + ...withLocalizePropTypes, }; @@ -104,6 +120,9 @@ const defaultProps = { accountID: null, }, isWhisper: false, + transactionViolations: { + violations: [], + }, }; function ReportPreview(props) { @@ -127,7 +146,7 @@ function ReportPreview(props) { const hasReceipts = transactionsWithReceipts.length > 0; const hasOnlyDistanceRequests = ReportUtils.hasOnlyDistanceRequestTransactions(props.iouReportID); const isScanning = hasReceipts && ReportUtils.areAllRequestsBeingSmartScanned(props.iouReportID, props.action); - const hasErrors = (hasReceipts && ReportUtils.hasMissingSmartscanFields(props.iouReportID)) || ReportUtils.reportHasViolations(props.iouReportID); + const hasErrors = (hasReceipts && ReportUtils.hasMissingSmartscanFields(props.iouReportID)) || ReportUtils.reportHasViolations(props.iouReportID, props.transactionViolations); const lastThreeTransactionsWithReceipts = transactionsWithReceipts.slice(-3); const lastThreeReceipts = _.map(lastThreeTransactionsWithReceipts, (transaction) => ReceiptUtils.getThumbnailAndImageURIs(transaction)); const hasNonReimbursableTransactions = ReportUtils.hasNonReimbursableTransactions(props.iouReportID); @@ -300,5 +319,8 @@ export default compose( session: { key: ONYXKEYS.SESSION, }, + transactionViolations: { + key: ONYXKEYS.COLLECTION.TRANSACTION_VIOLATIONS, + }, }), )(ReportPreview); diff --git a/src/libs/OptionsListUtils.js b/src/libs/OptionsListUtils.js index 4678ce395f76..43fdb14850a5 100644 --- a/src/libs/OptionsListUtils.js +++ b/src/libs/OptionsListUtils.js @@ -1127,7 +1127,9 @@ function getOptions( const searchValue = parsedPhoneNumber.possible ? parsedPhoneNumber.number.e164 : searchInputValue.toLowerCase(); // Filter out all the reports that shouldn't be displayed - const filteredReports = _.filter(reports, (report) => ReportUtils.shouldReportBeInOptionList(report, Navigation.getTopmostReportId(), false, betas, policies)); + const filteredReports = _.filter(reports, (report) => + ReportUtils.shouldReportBeInOptionList(report, Navigation.getTopmostReportId(), false, betas, policies, reportActions, false, transactionViolations), + ); // Sorting the reports works like this: // - Order everything by the last message timestamp (descending) diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index c8ecb8576812..6b143b26418a 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -22,7 +22,7 @@ import {Message, ReportActions} from '@src/types/onyx/ReportAction'; import {Receipt, WaypointCollection} from '@src/types/onyx/Transaction'; import DeepValueOf from '@src/types/utils/DeepValueOf'; import {EmptyObject, isEmptyObject, isNotEmptyObject} from '@src/types/utils/EmptyObject'; -import * as CollectionUtils from './CollectionUtils'; +// import * as CollectionUtils from './CollectionUtils'; import * as CurrencyUtils from './CurrencyUtils'; import DateUtils from './DateUtils'; import isReportMessageAttachment from './isReportMessageAttachment'; @@ -406,31 +406,31 @@ Onyx.connect({ callback: (value) => (loginList = value), }); -const transactionViolations: OnyxCollection = {}; -Onyx.connect({ - key: ONYXKEYS.COLLECTION.TRANSACTION_VIOLATIONS, - callback: (violations, key) => { - if (!key || !violations) { - return; - } - - const transactionID = CollectionUtils.extractCollectionItemID(key); - transactionViolations[transactionID] = violations; - }, -}); - -const reportActions: OnyxCollection = {}; -Onyx.connect({ - key: ONYXKEYS.COLLECTION.REPORT_ACTIONS, - callback: (actions, key) => { - if (!key || !actions) { - return; - } - - const reportID = CollectionUtils.extractCollectionItemID(key); - reportActions[reportID] = actions; - }, -}); +// const transactionViolations: OnyxCollection = {}; +// Onyx.connect({ +// key: ONYXKEYS.COLLECTION.TRANSACTION_VIOLATIONS, +// callback: (violations, key) => { +// if (!key || !violations) { +// return; +// } + +// const transactionID = CollectionUtils.extractCollectionItemID(key); +// transactionViolations[transactionID] = violations; +// }, +// }); + +// const reportActions: OnyxCollection = {}; +// Onyx.connect({ +// key: ONYXKEYS.COLLECTION.REPORT_ACTIONS, +// callback: (actions, key) => { +// if (!key || !actions) { +// return; +// } + +// const reportID = CollectionUtils.extractCollectionItemID(key); +// reportActions[reportID] = actions; +// }, +// }); let allPolicyTags: Record = {}; @@ -3435,8 +3435,8 @@ function shouldHideReport(report: OnyxEntry, currentReportId: string): b * Checks if there are any violations belonging to the transaction in the transactionsViolations Onyx object * then checks that the violation is of the proper type */ -function transactionHasViolation(transactionID: string): boolean { - const violations = transactionViolations ? transactionViolations[transactionID]?.value : []; +function transactionHasViolation(transactionID: string, transactionViolations?: TransactionViolations): boolean { + const violations = transactionViolations ? transactionViolations[transactionID] : []; if (!violations) { return false; } @@ -3448,14 +3448,14 @@ function transactionHasViolation(transactionID: string): boolean { * This only applies to report submitter and for reports in the open and processing states */ -function transactionThreadHasViolations(report: Report, betas: Beta[]): boolean { - if (!Permissions.canUseViolations(betas) || !reportActions) { +function transactionThreadHasViolations(report: Report, canUseViolations: boolean, transactionViolations?: TransactionViolations, reportActions?: ReportActions | null): boolean { + if (!canUseViolations || !reportActions) { return false; } if (!report.parentReportActionID) { return false; } - const parentReportAction = reportActions[`${report.parentReportID}`]?.[`${report.parentReportActionID}`]; + const parentReportAction = reportActions[`${report.parentReportActionID}`]; if (!parentReportAction) { return false; } @@ -3472,15 +3472,15 @@ function transactionThreadHasViolations(report: Report, betas: Beta[]): boolean if (report?.stateNum !== CONST.REPORT.STATE_NUM.OPEN && report?.stateNum !== CONST.REPORT.STATE_NUM.PROCESSING) { return false; } - return transactionHasViolation(IOUTransactionID); + return transactionHasViolation(IOUTransactionID, transactionViolations); } /** * Checks to see if a report contains a violation */ -function reportHasViolations(reportID: string): boolean { +function reportHasViolations(reportID: string, transactionViolations: TransactionViolations): boolean { const transactions = TransactionUtils.getAllReportTransactions(reportID); - return transactions.some((transaction) => transactionHasViolation(transaction.transactionID)); + return transactions.some((transaction) => transactionHasViolation(transaction.transactionID, transactionViolations)); } /** @@ -3498,6 +3498,7 @@ function shouldReportBeInOptionList( policies: OnyxCollection, allReportActions?: OnyxCollection, excludeEmptyChats = false, + transactionViolations?: TransactionViolations, ) { const isInDefaultMode = !isInGSDMode; // Exclude reports that have no data because there wouldn't be anything to show in the option item. @@ -3558,7 +3559,7 @@ function shouldReportBeInOptionList( } // Always show IOU reports with violations - if (isExpenseRequest(report) && transactionThreadHasViolations(report, betas)) { + if (isExpenseRequest(report) && transactionThreadHasViolations(report, betas.includes(CONST.BETAS.VIOLATIONS), transactionViolations, allReportActions?.[report.reportID])) { return true; } diff --git a/src/libs/SidebarUtils.ts b/src/libs/SidebarUtils.ts index 0c99aca64148..dce959a73872 100644 --- a/src/libs/SidebarUtils.ts +++ b/src/libs/SidebarUtils.ts @@ -4,7 +4,7 @@ import Onyx, {OnyxCollection} from 'react-native-onyx'; import {ValueOf} from 'type-fest'; import CONST from '@src/CONST'; import ONYXKEYS from '@src/ONYXKEYS'; -import {PersonalDetails} from '@src/types/onyx'; +import {PersonalDetails, TransactionViolations} from '@src/types/onyx'; import Beta from '@src/types/onyx/Beta'; import * as OnyxCommon from '@src/types/onyx/OnyxCommon'; import Policy from '@src/types/onyx/Policy'; @@ -118,6 +118,7 @@ function getOrderedReportIDs( policies: Record, priorityMode: ValueOf, allReportActions: OnyxCollection, + transactionViolations: TransactionViolations, ): string[] { // Generate a unique cache key based on the function arguments const cachedReportsKey = JSON.stringify( @@ -150,7 +151,7 @@ function getOrderedReportIDs( const allReportsDictValues = Object.values(allReports); // Filter out all the reports that shouldn't be displayed const reportsToDisplay = allReportsDictValues.filter((report) => - ReportUtils.shouldReportBeInOptionList(report, currentReportId ?? '', isInGSDMode, betas, policies, allReportActions, true), + ReportUtils.shouldReportBeInOptionList(report, currentReportId ?? '', isInGSDMode, betas, policies, allReportActions, true, transactionViolations), ); if (reportsToDisplay.length === 0) { @@ -237,7 +238,8 @@ function getOptionData( preferredLocale: ValueOf, policy: Policy, parentReportAction: ReportAction, - betas: Beta[], + transactionViolations: TransactionViolations, + canUseViolations: boolean, ): ReportUtils.OptionData | undefined { // When a user signs out, Onyx is cleared. Due to the lazy rendering with a virtual list, it's possible for // this method to be called after the Onyx data has been cleared out. In that case, it's fine to do @@ -291,7 +293,9 @@ function getOptionData( result.pendingAction = report.pendingFields ? report.pendingFields.addWorkspaceRoom || report.pendingFields.createChat : null; result.allReportErrors = OptionsListUtils.getAllReportErrors(report, reportActions) as OnyxCommon.Errors; result.brickRoadIndicator = - Object.keys(result.allReportErrors ?? {}).length !== 0 || ReportUtils.transactionThreadHasViolations(report, betas) ? CONST.BRICK_ROAD_INDICATOR_STATUS.ERROR : ''; + Object.keys(result.allReportErrors ?? {}).length !== 0 || ReportUtils.transactionThreadHasViolations(report, canUseViolations, transactionViolations, reportActions) + ? CONST.BRICK_ROAD_INDICATOR_STATUS.ERROR + : ''; result.ownerAccountID = report.ownerAccountID; result.managerID = report.managerID; result.reportID = report.reportID; From 90b7138e07a082816778408d16e99d645e029481 Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Mon, 11 Dec 2023 16:13:00 -0500 Subject: [PATCH 023/117] Starting so swap in the transactionViolations --- src/pages/home/report/ReportActionItem.js | 1 + 1 file changed, 1 insertion(+) diff --git a/src/pages/home/report/ReportActionItem.js b/src/pages/home/report/ReportActionItem.js index 92bb370155c9..f743d75cb590 100644 --- a/src/pages/home/report/ReportActionItem.js +++ b/src/pages/home/report/ReportActionItem.js @@ -342,6 +342,7 @@ function ReportActionItem(props) { contextMenuAnchor={popoverAnchorRef} checkIfContextMenuActive={toggleContextMenuFromActiveReportAction} isWhisper={isWhisper} + transactionViolations={props.transactionViolations} /> ); } else if ( From 35de23f38d48365b681a8d7aef348c7c3cc135b5 Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Tue, 12 Dec 2023 09:35:19 -0500 Subject: [PATCH 024/117] Fixed tests and created new hook mock --- src/components/LHNOptionsList/LHNOptionsList.js | 2 +- src/hooks/__mocks__/usePermissions.ts | 11 +++++++++++ src/libs/OptionsListUtils.js | 1 + tests/perf-test/SidebarUtils.perf-test.ts | 9 +++++---- tests/unit/SidebarFilterTest.js | 1 + tests/unit/SidebarOrderTest.js | 1 + tests/unit/SidebarTest.js | 1 + 7 files changed, 21 insertions(+), 5 deletions(-) create mode 100644 src/hooks/__mocks__/usePermissions.ts diff --git a/src/components/LHNOptionsList/LHNOptionsList.js b/src/components/LHNOptionsList/LHNOptionsList.js index ab8101bddf45..d2dc958d811f 100644 --- a/src/components/LHNOptionsList/LHNOptionsList.js +++ b/src/components/LHNOptionsList/LHNOptionsList.js @@ -226,7 +226,7 @@ export default compose( key: ONYXKEYS.COLLECTION.REPORT_DRAFT_COMMENT, }, transactionViolations: { - key: ONYXKEYS.COLLECTION.TRANSACTION_VIOLATION, + key: ONYXKEYS.COLLECTION.TRANSACTION_VIOLATIONS, }, }), )(LHNOptionsList); diff --git a/src/hooks/__mocks__/usePermissions.ts b/src/hooks/__mocks__/usePermissions.ts new file mode 100644 index 000000000000..4af004095712 --- /dev/null +++ b/src/hooks/__mocks__/usePermissions.ts @@ -0,0 +1,11 @@ +import UsePermissions from '@hooks/usePermissions'; + +/** + * + * @returns {UsePermissions} A mock of the usePermissions hook. + */ + +const usePermissions = (): typeof UsePermissions => () => ({ + canUseViolations: true, +}); +export default usePermissions; diff --git a/src/libs/OptionsListUtils.js b/src/libs/OptionsListUtils.js index 43fdb14850a5..b431505fa9d6 100644 --- a/src/libs/OptionsListUtils.js +++ b/src/libs/OptionsListUtils.js @@ -1081,6 +1081,7 @@ function getOptions( recentlyUsedTags = [], canInviteUser = true, includeSelectedOptions = false, + transactionViolations = {}, }, ) { if (includeCategories) { diff --git a/tests/perf-test/SidebarUtils.perf-test.ts b/tests/perf-test/SidebarUtils.perf-test.ts index fb3cac24053a..4424a66d3936 100644 --- a/tests/perf-test/SidebarUtils.perf-test.ts +++ b/tests/perf-test/SidebarUtils.perf-test.ts @@ -3,7 +3,7 @@ import {measureFunction} from 'reassure'; import SidebarUtils from '@libs/SidebarUtils'; import CONST from '@src/CONST'; import ONYXKEYS from '@src/ONYXKEYS'; -import {Beta, PersonalDetails} from '@src/types/onyx'; +import {PersonalDetails, TransactionViolations} from '@src/types/onyx'; import Policy from '@src/types/onyx/Policy'; import Report from '@src/types/onyx/Report'; import ReportAction, {ReportActions} from '@src/types/onyx/ReportAction'; @@ -51,20 +51,21 @@ test('getOptionData on 5k reports', async () => { const preferredLocale = 'en'; const policy = createRandomPolicy(1); const parentReportAction = createRandomReportAction(1); - const betas: Beta[] = []; + const transactionViolations = {} as TransactionViolations; Onyx.multiSet({ ...mockedResponseMap, }); await waitForBatchedUpdates(); - await measureFunction(() => SidebarUtils.getOptionData(report, reportActions, personalDetails, preferredLocale, policy, parentReportAction, betas), {runs}); + await measureFunction(() => SidebarUtils.getOptionData(report, reportActions, personalDetails, preferredLocale, policy, parentReportAction, transactionViolations, false), {runs}); }); test('getOrderedReportIDs on 5k reports', async () => { const currentReportId = '1'; const allReports = getMockedReports(); const betas = [CONST.BETAS.DEFAULT_ROOMS, CONST.BETAS.POLICY_ROOMS]; + const transactionViolations = {} as TransactionViolations; const policies = createCollection( (item) => `${ONYXKEYS.COLLECTION.POLICY}${item.id}`, @@ -94,5 +95,5 @@ test('getOrderedReportIDs on 5k reports', async () => { }); await waitForBatchedUpdates(); - await measureFunction(() => SidebarUtils.getOrderedReportIDs(currentReportId, allReports, betas, policies, CONST.PRIORITY_MODE.DEFAULT, allReportActions), {runs}); + await measureFunction(() => SidebarUtils.getOrderedReportIDs(currentReportId, allReports, betas, policies, CONST.PRIORITY_MODE.DEFAULT, allReportActions, transactionViolations), {runs}); }); diff --git a/tests/unit/SidebarFilterTest.js b/tests/unit/SidebarFilterTest.js index 23a958e3aa9d..4ecddfb8fd9e 100644 --- a/tests/unit/SidebarFilterTest.js +++ b/tests/unit/SidebarFilterTest.js @@ -10,6 +10,7 @@ import wrapOnyxWithWaitForBatchedUpdates from '../utils/wrapOnyxWithWaitForBatch // Be sure to include the mocked permissions library or else the beta tests won't work jest.mock('../../src/libs/Permissions'); +jest.mock('../../src/hooks/usePermissions.ts'); const ONYXKEYS = { PERSONAL_DETAILS_LIST: 'personalDetailsList', diff --git a/tests/unit/SidebarOrderTest.js b/tests/unit/SidebarOrderTest.js index 6eef3e40bf1c..d9ab881a442e 100644 --- a/tests/unit/SidebarOrderTest.js +++ b/tests/unit/SidebarOrderTest.js @@ -11,6 +11,7 @@ import wrapOnyxWithWaitForBatchedUpdates from '../utils/wrapOnyxWithWaitForBatch // Be sure to include the mocked Permissions and Expensicons libraries or else the beta tests won't work jest.mock('../../src/libs/Permissions'); +jest.mock('../../src/hooks/usePermissions.ts'); jest.mock('../../src/components/Icon/Expensicons'); const ONYXKEYS = { diff --git a/tests/unit/SidebarTest.js b/tests/unit/SidebarTest.js index 4bd0795aa3b9..70c7a33a8dc6 100644 --- a/tests/unit/SidebarTest.js +++ b/tests/unit/SidebarTest.js @@ -9,6 +9,7 @@ import wrapOnyxWithWaitForBatchedUpdates from '../utils/wrapOnyxWithWaitForBatch // Be sure to include the mocked Permissions and Expensicons libraries or else the beta tests won't work jest.mock('../../src/libs/Permissions'); +jest.mock('../../src/hooks/usePermissions.ts'); jest.mock('../../src/components/Icon/Expensicons'); const ONYXKEYS = { From bbfd1fa60a61af863c91e23472c17e26056f3151 Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Tue, 12 Dec 2023 11:36:13 -0500 Subject: [PATCH 025/117] Switched to just passing down parentReportAction --- src/libs/ReportUtils.ts | 17 +++++++---------- src/libs/SidebarUtils.ts | 2 +- 2 files changed, 8 insertions(+), 11 deletions(-) diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index 6b143b26418a..ecb7041771f6 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -3448,15 +3448,9 @@ function transactionHasViolation(transactionID: string, transactionViolations?: * This only applies to report submitter and for reports in the open and processing states */ -function transactionThreadHasViolations(report: Report, canUseViolations: boolean, transactionViolations?: TransactionViolations, reportActions?: ReportActions | null): boolean { - if (!canUseViolations || !reportActions) { - return false; - } - if (!report.parentReportActionID) { - return false; - } - const parentReportAction = reportActions[`${report.parentReportActionID}`]; - if (!parentReportAction) { +function transactionThreadHasViolations(report: Report, canUseViolations: boolean, transactionViolations?: TransactionViolations, parentReportAction?: ReportAction | null): boolean { + console.log('Parent Report Action : ', parentReportAction); + if (!canUseViolations || !parentReportAction) { return false; } if (parentReportAction.actionName !== CONST.REPORT.ACTIONS.TYPE.IOU) { @@ -3559,7 +3553,10 @@ function shouldReportBeInOptionList( } // Always show IOU reports with violations - if (isExpenseRequest(report) && transactionThreadHasViolations(report, betas.includes(CONST.BETAS.VIOLATIONS), transactionViolations, allReportActions?.[report.reportID])) { + if ( + isExpenseRequest(report) && + transactionThreadHasViolations(report, betas.includes(CONST.BETAS.VIOLATIONS), transactionViolations, allReportActions?.[`${report.parentReportActionID}`]) + ) { return true; } diff --git a/src/libs/SidebarUtils.ts b/src/libs/SidebarUtils.ts index dce959a73872..15c6e4a4c235 100644 --- a/src/libs/SidebarUtils.ts +++ b/src/libs/SidebarUtils.ts @@ -293,7 +293,7 @@ function getOptionData( result.pendingAction = report.pendingFields ? report.pendingFields.addWorkspaceRoom || report.pendingFields.createChat : null; result.allReportErrors = OptionsListUtils.getAllReportErrors(report, reportActions) as OnyxCommon.Errors; result.brickRoadIndicator = - Object.keys(result.allReportErrors ?? {}).length !== 0 || ReportUtils.transactionThreadHasViolations(report, canUseViolations, transactionViolations, reportActions) + Object.keys(result.allReportErrors ?? {}).length !== 0 || ReportUtils.transactionThreadHasViolations(report, canUseViolations, transactionViolations, parentReportAction) ? CONST.BRICK_ROAD_INDICATOR_STATUS.ERROR : ''; result.ownerAccountID = report.ownerAccountID; From 2c52edf9bfe2777b3b6a0be3762a044259b622d8 Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Tue, 12 Dec 2023 15:53:04 -0500 Subject: [PATCH 026/117] Move the report actions selector to the SidebarUtils file --- src/libs/ReportUtils.ts | 12 +++-- src/libs/SidebarUtils.ts | 52 ++++++++++++++++++++-- src/pages/home/sidebar/SidebarLinksData.js | 48 +++++++++++--------- tests/perf-test/SidebarUtils.perf-test.ts | 4 +- 4 files changed, 85 insertions(+), 31 deletions(-) diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index 3d105cc548fa..30a805d1b9c8 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -3504,7 +3504,7 @@ function shouldHideReport(report: OnyxEntry, currentReportId: string): b * then checks that the violation is of the proper type */ function transactionHasViolation(transactionID: string, transactionViolations?: TransactionViolations): boolean { - const violations = transactionViolations ? transactionViolations[transactionID] : []; + const violations: TransactionViolation[] = transactionViolations ? transactionViolations[transactionID] || [] : []; if (!violations) { return false; } @@ -3517,8 +3517,7 @@ function transactionHasViolation(transactionID: string, transactionViolations?: */ function transactionThreadHasViolations(report: Report, canUseViolations: boolean, transactionViolations?: TransactionViolations, parentReportAction?: ReportAction | null): boolean { - console.log('Parent Report Action : ', parentReportAction); - if (!canUseViolations || !parentReportAction) { + if (!parentReportAction) { return false; } if (parentReportAction.actionName !== CONST.REPORT.ACTIONS.TYPE.IOU) { @@ -3623,7 +3622,12 @@ function shouldReportBeInOptionList( // Always show IOU reports with violations if ( isExpenseRequest(report) && - transactionThreadHasViolations(report, betas.includes(CONST.BETAS.VIOLATIONS), transactionViolations, allReportActions?.[`${report.parentReportActionID}`]) + transactionThreadHasViolations( + report, + betas.includes(CONST.BETAS.VIOLATIONS), + transactionViolations, + allReportActions?.[`${report.parentReportID}`]?.[`${report.parentReportActionID}`], + ) ) { return true; } diff --git a/src/libs/SidebarUtils.ts b/src/libs/SidebarUtils.ts index 2a09758b26d5..8db53e46882b 100644 --- a/src/libs/SidebarUtils.ts +++ b/src/libs/SidebarUtils.ts @@ -8,7 +8,7 @@ import {PersonalDetails, TransactionViolations} from '@src/types/onyx'; import Beta from '@src/types/onyx/Beta'; import * as OnyxCommon from '@src/types/onyx/OnyxCommon'; import Policy from '@src/types/onyx/Policy'; -import Report from '@src/types/onyx/Report'; +import Report, {Message} from '@src/types/onyx/Report'; import ReportAction, {ReportActions} from '@src/types/onyx/ReportAction'; import * as CollectionUtils from './CollectionUtils'; import * as LocalePhoneNumber from './LocalePhoneNumber'; @@ -108,6 +108,51 @@ function setWithLimit(map: Map, key: TKey, value: TV // Variable to verify if ONYX actions are loaded let hasInitialReportActions = false; +// Originally we were passing down the reportActions as the type OnyxCollection but +// it was used as a Record in the cachedReportsKey. This was causing a type error +// for the ReportUtils.shouldReportBeInOptionList function. To fix this, we are now passing down the reportActions +// as a Record and then converting it to a Record in the +// reportActionsSelector function. +// This function was originally in the parent component as an Onyx selector, but it was moved here to prevent +// to simplify the logic and prevent the need to pass down the reportActions as an array. + +type MappedReportAction = { + id: string; + errors: OnyxCommon.Errors; + message: Message[]; +}; + +type MappedReportActions = Record; + +const reportActionsSelector = (reportActions?: MappedReportActions): Record | undefined => { + if (!reportActions) { + return undefined; + } + + return Object.values(reportActions).reduce((acc, reportAction) => { + const modifiedAction: MappedReportAction = { + id: reportAction.id, + errors: reportAction.errors || [], + message: [ + { + moderationDecision: { + decision: reportAction.message?.[0]?.moderationDecision?.decision, + }, + }, + ], + }; + + // If the key already exists, append to the array, otherwise create a new array + if (acc[reportAction.id]) { + acc[reportAction.id].push(modifiedAction); + } else { + acc[reportAction.id] = [modifiedAction]; + } + + return acc; + }, {} as Record); +}; + /** * @returns An array of reportIDs sorted in the proper order */ @@ -117,13 +162,14 @@ function getOrderedReportIDs( betas: Beta[], policies: Record, priorityMode: ValueOf, - allReportActions: OnyxCollection, + allReportActions: OnyxCollection, transactionViolations: TransactionViolations, ): string[] { + const mappedReportActions = reportActionsSelector(allReportActions as unknown as MappedReportActions); // Generate a unique cache key based on the function arguments const cachedReportsKey = JSON.stringify( // eslint-disable-next-line @typescript-eslint/prefer-nullish-coalescing - [currentReportId, allReports, betas, policies, priorityMode, allReportActions?.[`${ONYXKEYS.COLLECTION.REPORT_ACTIONS}${currentReportId}`]?.length || 1], + [currentReportId, allReports, betas, policies, priorityMode, mappedReportActions?.[`${ONYXKEYS.COLLECTION.REPORT_ACTIONS}${currentReportId}`]?.length || 1], (key, value: unknown) => { /** * Exclude 'participantAccountIDs', 'participants' and 'lastMessageText' not to overwhelm a cached key value with huge data, diff --git a/src/pages/home/sidebar/SidebarLinksData.js b/src/pages/home/sidebar/SidebarLinksData.js index 97f119bbd2e8..856af9fd5223 100644 --- a/src/pages/home/sidebar/SidebarLinksData.js +++ b/src/pages/home/sidebar/SidebarLinksData.js @@ -1,5 +1,4 @@ import {deepEqual} from 'fast-equals'; -import lodashGet from 'lodash/get'; import PropTypes from 'prop-types'; import React, {useCallback, useMemo, useRef} from 'react'; import {View} from 'react-native'; @@ -55,6 +54,22 @@ const propTypes = { /** The policies which the user has access to */ // eslint-disable-next-line react/forbid-prop-types policies: PropTypes.object, + + /** All of the transaction violations */ + transactionViolations: PropTypes.shape({ + violations: PropTypes.arrayOf( + PropTypes.shape({ + /** The transaction ID */ + transactionID: PropTypes.number, + + /** The transaction violation type */ + type: PropTypes.string, + + /** The transaction violation message */ + message: PropTypes.string, + }), + ), + }), }; const defaultProps = { @@ -64,16 +79,17 @@ const defaultProps = { priorityMode: CONST.PRIORITY_MODE.DEFAULT, betas: [], policies: {}, + transactionViolations: {}, }; -function SidebarLinksData({isFocused, allReportActions, betas, chatReports, currentReportID, insets, isLoadingApp, onLinkClick, policies, priorityMode, network}) { +function SidebarLinksData({isFocused, allReportActions, betas, chatReports, currentReportID, insets, isLoadingApp, onLinkClick, policies, priorityMode, network, transactionViolations}) { const styles = useThemeStyles(); const {translate} = useLocalize(); const reportIDsRef = useRef(null); const isLoading = isLoadingApp; const optionListItems = useMemo(() => { - const reportIDs = SidebarUtils.getOrderedReportIDs(null, chatReports, betas, policies, priorityMode, allReportActions); + const reportIDs = SidebarUtils.getOrderedReportIDs(null, chatReports, betas, policies, priorityMode, allReportActions, transactionViolations); if (deepEqual(reportIDsRef.current, reportIDs)) { return reportIDsRef.current; @@ -85,7 +101,7 @@ function SidebarLinksData({isFocused, allReportActions, betas, chatReports, curr reportIDsRef.current = reportIDs; } return reportIDsRef.current || []; - }, [allReportActions, betas, chatReports, policies, priorityMode, isLoading, network.isOffline]); + }, [allReportActions, betas, chatReports, policies, priorityMode, isLoading, network.isOffline, transactionViolations]); // We need to make sure the current report is in the list of reports, but we do not want // to have to re-generate the list every time the currentReportID changes. To do that @@ -94,10 +110,10 @@ function SidebarLinksData({isFocused, allReportActions, betas, chatReports, curr // case we re-generate the list a 2nd time with the current report included. const optionListItemsWithCurrentReport = useMemo(() => { if (currentReportID && !_.contains(optionListItems, currentReportID)) { - return SidebarUtils.getOrderedReportIDs(currentReportID, chatReports, betas, policies, priorityMode, allReportActions); + return SidebarUtils.getOrderedReportIDs(currentReportID, chatReports, betas, policies, priorityMode, allReportActions, transactionViolations); } return optionListItems; - }, [currentReportID, optionListItems, chatReports, betas, policies, priorityMode, allReportActions]); + }, [currentReportID, optionListItems, chatReports, betas, policies, priorityMode, allReportActions, transactionViolations]); const currentReportIDRef = useRef(currentReportID); currentReportIDRef.current = currentReportID; @@ -170,21 +186,6 @@ const chatReportSelector = (report) => parentReportID: report.parentReportID, }; -/** - * @param {Object} [reportActions] - * @returns {Object|undefined} - */ -const reportActionsSelector = (reportActions) => - reportActions && - _.map(reportActions, (reportAction) => ({ - errors: lodashGet(reportAction, 'errors', []), - message: [ - { - moderationDecision: {decision: lodashGet(reportAction, 'message[0].moderationDecision.decision')}, - }, - ], - })); - /** * @param {Object} [policy] * @returns {Object|undefined} @@ -219,7 +220,6 @@ export default compose( }, allReportActions: { key: ONYXKEYS.COLLECTION.REPORT_ACTIONS, - selector: reportActionsSelector, initialValue: {}, }, policies: { @@ -227,5 +227,9 @@ export default compose( selector: policySelector, initialValue: {}, }, + transactionViolations: { + key: ONYXKEYS.TRANSACTION_VIOLATIONS, + initialValue: {}, + }, }), )(SidebarLinksData); diff --git a/tests/perf-test/SidebarUtils.perf-test.ts b/tests/perf-test/SidebarUtils.perf-test.ts index 11e90b83ebc5..b627d54edc5d 100644 --- a/tests/perf-test/SidebarUtils.perf-test.ts +++ b/tests/perf-test/SidebarUtils.perf-test.ts @@ -6,7 +6,7 @@ import ONYXKEYS from '@src/ONYXKEYS'; import {PersonalDetails, TransactionViolations} from '@src/types/onyx'; import Policy from '@src/types/onyx/Policy'; import Report from '@src/types/onyx/Report'; -import ReportAction from '@src/types/onyx/ReportAction'; +import ReportAction, {ReportActions} from '@src/types/onyx/ReportAction'; import createCollection from '../utils/collections/createCollection'; import createPersonalDetails from '../utils/collections/personalDetails'; import createRandomPolicy from '../utils/collections/policies'; @@ -88,7 +88,7 @@ test('[SidebarUtils] getOrderedReportIDs on 5k reports', async () => { }, ], ]), - ) as unknown as OnyxCollection; + ) as unknown as OnyxCollection; Onyx.multiSet({ ...mockedResponseMap, From 73971f074a22e97379e36d6faba327a4089264a4 Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Tue, 12 Dec 2023 20:08:31 -0500 Subject: [PATCH 027/117] Correct type import --- src/libs/SidebarUtils.ts | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/libs/SidebarUtils.ts b/src/libs/SidebarUtils.ts index 8db53e46882b..5001725dfbb5 100644 --- a/src/libs/SidebarUtils.ts +++ b/src/libs/SidebarUtils.ts @@ -8,8 +8,8 @@ import {PersonalDetails, TransactionViolations} from '@src/types/onyx'; import Beta from '@src/types/onyx/Beta'; import * as OnyxCommon from '@src/types/onyx/OnyxCommon'; import Policy from '@src/types/onyx/Policy'; -import Report, {Message} from '@src/types/onyx/Report'; -import ReportAction, {ReportActions} from '@src/types/onyx/ReportAction'; +import Report from '@src/types/onyx/Report'; +import ReportAction, {Message, ReportActions} from '@src/types/onyx/ReportAction'; import * as CollectionUtils from './CollectionUtils'; import * as LocalePhoneNumber from './LocalePhoneNumber'; import * as Localize from './Localize'; From 1234b8890d0220ab59d16ca585e5df1a1091cbba Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Tue, 12 Dec 2023 20:19:38 -0500 Subject: [PATCH 028/117] Another type correction --- src/libs/SidebarUtils.ts | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/src/libs/SidebarUtils.ts b/src/libs/SidebarUtils.ts index 5001725dfbb5..acf58424c003 100644 --- a/src/libs/SidebarUtils.ts +++ b/src/libs/SidebarUtils.ts @@ -135,9 +135,9 @@ const reportActionsSelector = (reportActions?: MappedReportActions): Record Date: Tue, 12 Dec 2023 20:54:54 -0500 Subject: [PATCH 029/117] Final fix for the test of the report actions --- src/libs/SidebarUtils.ts | 51 ++++++++++++++++++++++------------------ 1 file changed, 28 insertions(+), 23 deletions(-) diff --git a/src/libs/SidebarUtils.ts b/src/libs/SidebarUtils.ts index acf58424c003..8db1ca430626 100644 --- a/src/libs/SidebarUtils.ts +++ b/src/libs/SidebarUtils.ts @@ -117,40 +117,45 @@ let hasInitialReportActions = false; // to simplify the logic and prevent the need to pass down the reportActions as an array. type MappedReportAction = { - id: string; + reportActionID: string; errors: OnyxCommon.Errors; message: Message[]; }; type MappedReportActions = Record; -const reportActionsSelector = (reportActions?: MappedReportActions): Record | undefined => { - if (!reportActions) { +const reportActionsSelector = (reportActionsCollection?: Record): Record | undefined => { + if (!reportActionsCollection) { return undefined; } - return Object.values(reportActions).reduce((acc, reportAction) => { - const modifiedAction: MappedReportAction = { - id: reportAction.id, - errors: reportAction.errors || [], - message: [ - { - moderationDecision: reportAction.message?.[0]?.moderationDecision, - type: reportAction.message?.[0]?.type, - text: reportAction.message?.[0]?.text, - }, - ], - }; - - // If the key already exists, append to the array, otherwise create a new array - if (acc[reportAction.id]) { - acc[reportAction.id].push(modifiedAction); - } else { - acc[reportAction.id] = [modifiedAction]; + return Object.values(reportActionsCollection).reduce>((acc, reportActions) => { + if (!reportActions) { + return acc; } + Object.entries(reportActions).forEach(([reportActionID, reportAction]) => { + const modifiedAction: MappedReportAction = { + reportActionID, + errors: reportAction.errors || [], + message: [ + { + moderationDecision: reportAction.message?.[0]?.moderationDecision, + type: reportAction.message?.[0]?.type, + text: reportAction.message?.[0]?.text, + }, + ], + }; + + // If the key already exists, append to the array, otherwise create a new array + if (acc[reportActionID]) { + acc[reportActionID].push(modifiedAction); + } else { + acc[reportActionID] = [modifiedAction]; + } + }); return acc; - }, {} as Record); + }, {}); }; /** @@ -165,7 +170,7 @@ function getOrderedReportIDs( allReportActions: OnyxCollection, transactionViolations: TransactionViolations, ): string[] { - const mappedReportActions = reportActionsSelector(allReportActions as unknown as MappedReportActions); + const mappedReportActions = allReportActions ? reportActionsSelector(allReportActions as unknown as Record) : {}; // Generate a unique cache key based on the function arguments const cachedReportsKey = JSON.stringify( // eslint-disable-next-line @typescript-eslint/prefer-nullish-coalescing From 4366d642a53131a693f15963aa6804caf0062526 Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Tue, 12 Dec 2023 22:15:21 -0500 Subject: [PATCH 030/117] Update Comments --- src/libs/ReportUtils.ts | 26 -------------------------- src/libs/SidebarUtils.ts | 2 +- 2 files changed, 1 insertion(+), 27 deletions(-) diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index cca13ecbc1c2..cc933e62b244 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -416,32 +416,6 @@ Onyx.connect({ callback: (value) => (loginList = value), }); -// const transactionViolations: OnyxCollection = {}; -// Onyx.connect({ -// key: ONYXKEYS.COLLECTION.TRANSACTION_VIOLATIONS, -// callback: (violations, key) => { -// if (!key || !violations) { -// return; -// } - -// const transactionID = CollectionUtils.extractCollectionItemID(key); -// transactionViolations[transactionID] = violations; -// }, -// }); - -// const reportActions: OnyxCollection = {}; -// Onyx.connect({ -// key: ONYXKEYS.COLLECTION.REPORT_ACTIONS, -// callback: (actions, key) => { -// if (!key || !actions) { -// return; -// } - -// const reportID = CollectionUtils.extractCollectionItemID(key); -// reportActions[reportID] = actions; -// }, -// }); - let allPolicyTags: Record = {}; Onyx.connect({ diff --git a/src/libs/SidebarUtils.ts b/src/libs/SidebarUtils.ts index 8db1ca430626..d550d899e078 100644 --- a/src/libs/SidebarUtils.ts +++ b/src/libs/SidebarUtils.ts @@ -113,7 +113,7 @@ let hasInitialReportActions = false; // for the ReportUtils.shouldReportBeInOptionList function. To fix this, we are now passing down the reportActions // as a Record and then converting it to a Record in the // reportActionsSelector function. -// This function was originally in the parent component as an Onyx selector, but it was moved here to prevent +// This function was originally in the parent component as an Onyx selector, but it was moved here // to simplify the logic and prevent the need to pass down the reportActions as an array. type MappedReportAction = { From 57168006f2a429f21c2f016f0ec51054d98a3971 Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Tue, 12 Dec 2023 22:29:03 -0500 Subject: [PATCH 031/117] A few fixes --- src/components/LHNOptionsList/LHNOptionsList.js | 9 +++++++++ src/components/ReportActionItem/ReportPreview.js | 9 +++++++++ src/pages/home/sidebar/SidebarLinksData.js | 9 +++++++++ tests/perf-test/ReportUtils.perf-test.ts | 2 +- tests/perf-test/SidebarLinks.perf-test.js | 1 + 5 files changed, 29 insertions(+), 1 deletion(-) diff --git a/src/components/LHNOptionsList/LHNOptionsList.js b/src/components/LHNOptionsList/LHNOptionsList.js index d2dc958d811f..6d02c26d1753 100644 --- a/src/components/LHNOptionsList/LHNOptionsList.js +++ b/src/components/LHNOptionsList/LHNOptionsList.js @@ -79,6 +79,15 @@ const propTypes = { /** The transaction violation message */ message: PropTypes.string, + + /** The transaction violation data */ + data: PropTypes.shape({ + /** The transaction violation data field */ + field: PropTypes.string, + + /** The transaction violation data value */ + value: PropTypes.string, + }), }), ), }), diff --git a/src/components/ReportActionItem/ReportPreview.js b/src/components/ReportActionItem/ReportPreview.js index ab6fa5173a16..e88100ff7567 100644 --- a/src/components/ReportActionItem/ReportPreview.js +++ b/src/components/ReportActionItem/ReportPreview.js @@ -112,6 +112,15 @@ const propTypes = { /** The transaction violation message */ message: PropTypes.string, + + /** The transaction violation data */ + data: PropTypes.shape({ + /** The transaction violation data field */ + field: PropTypes.string, + + /** The transaction violation data value */ + value: PropTypes.string, + }), }), ), }), diff --git a/src/pages/home/sidebar/SidebarLinksData.js b/src/pages/home/sidebar/SidebarLinksData.js index 856af9fd5223..a5788a91794a 100644 --- a/src/pages/home/sidebar/SidebarLinksData.js +++ b/src/pages/home/sidebar/SidebarLinksData.js @@ -67,6 +67,15 @@ const propTypes = { /** The transaction violation message */ message: PropTypes.string, + + /** The transaction violation data */ + data: PropTypes.shape({ + /** The transaction violation data field */ + field: PropTypes.string, + + /** The transaction violation data value */ + value: PropTypes.string, + }), }), ), }), diff --git a/tests/perf-test/ReportUtils.perf-test.ts b/tests/perf-test/ReportUtils.perf-test.ts index ab6ee72a0082..03c79095b6a7 100644 --- a/tests/perf-test/ReportUtils.perf-test.ts +++ b/tests/perf-test/ReportUtils.perf-test.ts @@ -183,7 +183,7 @@ test('[ReportUtils] shouldReportBeInOptionList on 1k participant', async () => { const policies = getMockedPolicies(); await waitForBatchedUpdates(); - await measureFunction(() => ReportUtils.shouldReportBeInOptionList(report, currentReportId, isInGSDMode, betas, policies), {runs}); + await measureFunction(() => ReportUtils.shouldReportBeInOptionList(report, currentReportId, isInGSDMode, betas, policies, {}), {runs}); }); test('[ReportUtils] getWorkspaceIcon on 5k policies', async () => { diff --git a/tests/perf-test/SidebarLinks.perf-test.js b/tests/perf-test/SidebarLinks.perf-test.js index c6e6c024c597..0b05a14fa067 100644 --- a/tests/perf-test/SidebarLinks.perf-test.js +++ b/tests/perf-test/SidebarLinks.perf-test.js @@ -10,6 +10,7 @@ import waitForBatchedUpdates from '../utils/waitForBatchedUpdates'; import wrapOnyxWithWaitForBatchedUpdates from '../utils/wrapOnyxWithWaitForBatchedUpdates'; jest.mock('../../src/libs/Permissions'); +jest.mock('../../src/hooks/usePermissions.ts'); jest.mock('../../src/libs/Navigation/Navigation'); jest.mock('../../src/components/Icon/Expensicons'); From adf19333272d0158099c63d05ae584eccc70ec97 Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Wed, 13 Dec 2023 09:15:04 -0500 Subject: [PATCH 032/117] Fixed some prop types and also added a beta guard --- .../ReportActionItem/ReportPreview.js | 6 +++- src/libs/ReportUtils.ts | 32 ++++++++++--------- src/libs/SidebarUtils.ts | 2 +- src/pages/home/sidebar/SidebarLinksData.js | 18 +++-------- 4 files changed, 27 insertions(+), 31 deletions(-) diff --git a/src/components/ReportActionItem/ReportPreview.js b/src/components/ReportActionItem/ReportPreview.js index e88100ff7567..005f244cb6b1 100644 --- a/src/components/ReportActionItem/ReportPreview.js +++ b/src/components/ReportActionItem/ReportPreview.js @@ -19,6 +19,7 @@ import ControlSelection from '@libs/ControlSelection'; import * as CurrencyUtils from '@libs/CurrencyUtils'; import * as DeviceCapabilities from '@libs/DeviceCapabilities'; import Navigation from '@libs/Navigation/Navigation'; +import {usePermissions} from '@libs/Permissions'; import * as ReceiptUtils from '@libs/ReceiptUtils'; import * as ReportActionUtils from '@libs/ReportActionsUtils'; import * as ReportUtils from '@libs/ReportUtils'; @@ -149,6 +150,8 @@ function ReportPreview(props) { const styles = useThemeStyles(); const {translate} = useLocalize(); + const {canUseViolations} = usePermissions(); + const managerID = props.iouReport.managerID || 0; const isCurrentUserManager = managerID === lodashGet(props.session, 'accountID'); const {totalDisplaySpend, reimbursableSpend} = ReportUtils.getMoneyRequestSpendBreakdown(props.iouReport); @@ -167,7 +170,8 @@ function ReportPreview(props) { const hasReceipts = transactionsWithReceipts.length > 0; const hasOnlyDistanceRequests = ReportUtils.hasOnlyDistanceRequestTransactions(props.iouReportID); const isScanning = hasReceipts && ReportUtils.areAllRequestsBeingSmartScanned(props.iouReportID, props.action); - const hasErrors = (hasReceipts && ReportUtils.hasMissingSmartscanFields(props.iouReportID)) || ReportUtils.reportHasViolations(props.iouReportID, props.transactionViolations); + const hasErrors = + (hasReceipts && ReportUtils.hasMissingSmartscanFields(props.iouReportID)) || (canUseViolations && ReportUtils.reportHasViolations(props.iouReportID, props.transactionViolations)); const lastThreeTransactionsWithReceipts = transactionsWithReceipts.slice(-3); const lastThreeReceipts = _.map(lastThreeTransactionsWithReceipts, (transaction) => ReceiptUtils.getThumbnailAndImageURIs(transaction)); const hasNonReimbursableTransactions = ReportUtils.hasNonReimbursableTransactions(props.iouReportID); diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index cc933e62b244..08e2cd44dc67 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -3468,8 +3468,8 @@ function canAccessReport(report: OnyxEntry, policies: OnyxCollection, currentReportId: string): boolean { const currentReport = getReport(currentReportId); const parentReport = getParentReport(isNotEmptyObject(currentReport) ? currentReport : null); - const allReportActions = ReportActionsUtils.getAllReportActions(report?.reportID ?? ''); - const isChildReportHasComment = Object.values(allReportActions ?? {})?.some((reportAction) => (reportAction?.childVisibleActionCount ?? 0) > 0); + const reportActions = ReportActionsUtils.getAllReportActions(report?.reportID ?? ''); + const isChildReportHasComment = Object.values(reportActions ?? {})?.some((reportAction) => (reportAction?.childVisibleActionCount ?? 0) > 0); return parentReport?.reportID !== report?.reportID && !isChildReportHasComment; } @@ -3490,14 +3490,24 @@ function transactionHasViolation(transactionID: string, transactionViolations?: * This only applies to report submitter and for reports in the open and processing states */ -function transactionThreadHasViolations(report: Report, canUseViolations: boolean, transactionViolations?: TransactionViolations, parentReportAction?: ReportAction | null): boolean { - if (!parentReportAction) { +function transactionThreadHasViolations( + report: Report, + canUseViolations: boolean, + transactionViolations?: TransactionViolations, + reportActions?: OnyxCollection | null, + parentReportAction?: ReportAction | null, +): boolean { + if (!canUseViolations || !reportActions) { + return false; + } + const resolvedParentReportAction = parentReportAction ?? reportActions?.[`${report.parentReportID}`]?.[`${report.parentReportActionID}`]; + if (!resolvedParentReportAction) { return false; } - if (parentReportAction.actionName !== CONST.REPORT.ACTIONS.TYPE.IOU) { + if (resolvedParentReportAction.actionName !== CONST.REPORT.ACTIONS.TYPE.IOU) { return false; } - const {IOUTransactionID, IOUReportID} = parentReportAction?.originalMessage; + const {IOUTransactionID, IOUReportID} = resolvedParentReportAction?.originalMessage; if (!IOUTransactionID || !IOUReportID) { return false; } @@ -3594,15 +3604,7 @@ function shouldReportBeInOptionList( } // Always show IOU reports with violations - if ( - isExpenseRequest(report) && - transactionThreadHasViolations( - report, - betas.includes(CONST.BETAS.VIOLATIONS), - transactionViolations, - allReportActions?.[`${report.parentReportID}`]?.[`${report.parentReportActionID}`], - ) - ) { + if (isExpenseRequest(report) && transactionThreadHasViolations(report, betas.includes(CONST.BETAS.VIOLATIONS), transactionViolations, allReportActions)) { return true; } diff --git a/src/libs/SidebarUtils.ts b/src/libs/SidebarUtils.ts index d550d899e078..3968f2f95547 100644 --- a/src/libs/SidebarUtils.ts +++ b/src/libs/SidebarUtils.ts @@ -343,7 +343,7 @@ function getOptionData( result.pendingAction = report.pendingFields ? report.pendingFields.addWorkspaceRoom || report.pendingFields.createChat : undefined; result.allReportErrors = OptionsListUtils.getAllReportErrors(report, reportActions) as OnyxCommon.Errors; result.brickRoadIndicator = - Object.keys(result.allReportErrors ?? {}).length !== 0 || ReportUtils.transactionThreadHasViolations(report, canUseViolations, transactionViolations, parentReportAction) + Object.keys(result.allReportErrors ?? {}).length !== 0 || ReportUtils.transactionThreadHasViolations(report, canUseViolations, transactionViolations, null, parentReportAction) ? CONST.BRICK_ROAD_INDICATOR_STATUS.ERROR : ''; result.ownerAccountID = report.ownerAccountID; diff --git a/src/pages/home/sidebar/SidebarLinksData.js b/src/pages/home/sidebar/SidebarLinksData.js index a5788a91794a..cdf3140f174f 100644 --- a/src/pages/home/sidebar/SidebarLinksData.js +++ b/src/pages/home/sidebar/SidebarLinksData.js @@ -11,6 +11,7 @@ import withNavigationFocus from '@components/withNavigationFocus'; import useLocalize from '@hooks/useLocalize'; import compose from '@libs/compose'; import SidebarUtils from '@libs/SidebarUtils'; +import reportActionPropTypes from '@pages/home/report/reportActionPropTypes'; import reportPropTypes from '@pages/reportPropTypes'; import useThemeStyles from '@styles/useThemeStyles'; import CONST from '@src/CONST'; @@ -25,20 +26,9 @@ const propTypes = { chatReports: PropTypes.objectOf(reportPropTypes), /** All report actions for all reports */ - allReportActions: PropTypes.objectOf( - PropTypes.arrayOf( - PropTypes.shape({ - error: PropTypes.string, - message: PropTypes.arrayOf( - PropTypes.shape({ - moderationDecision: PropTypes.shape({ - decision: PropTypes.string, - }), - }), - ), - }), - ), - ), + + /** Object of report actions for this report */ + allReportActions: PropTypes.objectOf(PropTypes.shape(reportActionPropTypes)), /** Whether the reports are loading. When false it means they are ready to be used. */ isLoadingApp: PropTypes.bool, From da8f3769f3775ffa443f0880823de155f91d1dd9 Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Wed, 13 Dec 2023 09:55:59 -0500 Subject: [PATCH 033/117] Added usePermissions mock to ReportPreview --- tests/perf-test/ReportScreen.perf-test.js | 1 + 1 file changed, 1 insertion(+) diff --git a/tests/perf-test/ReportScreen.perf-test.js b/tests/perf-test/ReportScreen.perf-test.js index 96514112cd05..8a5b35cb2f3a 100644 --- a/tests/perf-test/ReportScreen.perf-test.js +++ b/tests/perf-test/ReportScreen.perf-test.js @@ -56,6 +56,7 @@ jest.mock('../../src/hooks/useEnvironment', () => jest.mock('../../src/libs/Permissions', () => ({ canUseLinkPreviews: jest.fn(() => true), })); +jest.mock('../../src/hooks/usePermissions.ts'); jest.mock('../../src/libs/Navigation/Navigation'); From cda1aeeac983b13514baf4221c6157a0f8361ed5 Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Wed, 13 Dec 2023 12:22:50 -0500 Subject: [PATCH 034/117] Trying this out --- src/libs/SidebarUtils.ts | 63 +++++++++------------------------------- 1 file changed, 13 insertions(+), 50 deletions(-) diff --git a/src/libs/SidebarUtils.ts b/src/libs/SidebarUtils.ts index 3968f2f95547..5e41b48eab5b 100644 --- a/src/libs/SidebarUtils.ts +++ b/src/libs/SidebarUtils.ts @@ -108,55 +108,12 @@ function setWithLimit(map: Map, key: TKey, value: TV // Variable to verify if ONYX actions are loaded let hasInitialReportActions = false; -// Originally we were passing down the reportActions as the type OnyxCollection but -// it was used as a Record in the cachedReportsKey. This was causing a type error -// for the ReportUtils.shouldReportBeInOptionList function. To fix this, we are now passing down the reportActions -// as a Record and then converting it to a Record in the -// reportActionsSelector function. -// This function was originally in the parent component as an Onyx selector, but it was moved here -// to simplify the logic and prevent the need to pass down the reportActions as an array. - -type MappedReportAction = { - reportActionID: string; - errors: OnyxCommon.Errors; - message: Message[]; -}; - -type MappedReportActions = Record; - -const reportActionsSelector = (reportActionsCollection?: Record): Record | undefined => { - if (!reportActionsCollection) { - return undefined; +type ReportActionsCount = Record< + string, + { + reportActionsCount: number; } - - return Object.values(reportActionsCollection).reduce>((acc, reportActions) => { - if (!reportActions) { - return acc; - } - Object.entries(reportActions).forEach(([reportActionID, reportAction]) => { - const modifiedAction: MappedReportAction = { - reportActionID, - errors: reportAction.errors || [], - message: [ - { - moderationDecision: reportAction.message?.[0]?.moderationDecision, - type: reportAction.message?.[0]?.type, - text: reportAction.message?.[0]?.text, - }, - ], - }; - - // If the key already exists, append to the array, otherwise create a new array - if (acc[reportActionID]) { - acc[reportActionID].push(modifiedAction); - } else { - acc[reportActionID] = [modifiedAction]; - } - }); - - return acc; - }, {}); -}; +>; /** * @returns An array of reportIDs sorted in the proper order @@ -170,11 +127,17 @@ function getOrderedReportIDs( allReportActions: OnyxCollection, transactionViolations: TransactionViolations, ): string[] { - const mappedReportActions = allReportActions ? reportActionsSelector(allReportActions as unknown as Record) : {}; + const reportActionsCount: ReportActionsCount | null = + allReportActions && + Object.keys(allReportActions).reduce((acc, reportID) => { + acc[reportID] = {reportActionsCount: Object.keys(allReportActions[reportID] as Record).length}; + return acc; + }, {}); + // Generate a unique cache key based on the function arguments const cachedReportsKey = JSON.stringify( // eslint-disable-next-line @typescript-eslint/prefer-nullish-coalescing - [currentReportId, allReports, betas, policies, priorityMode, mappedReportActions?.[`${ONYXKEYS.COLLECTION.REPORT_ACTIONS}${currentReportId}`]?.length || 1], + [currentReportId, allReports, betas, policies, priorityMode, reportActionsCount?.[`${ONYXKEYS.COLLECTION.REPORT_ACTIONS}${currentReportId}`]?.reportActionsCount || 1], (key, value: unknown) => { /** * Exclude 'participantAccountIDs', 'participants' and 'lastMessageText' not to overwhelm a cached key value with huge data, From cd6586a74297aab87b969e4a834566d3bd03e088 Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Wed, 13 Dec 2023 12:22:50 -0500 Subject: [PATCH 035/117] Trying this out --- src/libs/SidebarUtils.ts | 63 +++++------------------ tests/perf-test/ReportScreen.perf-test.js | 1 + 2 files changed, 14 insertions(+), 50 deletions(-) diff --git a/src/libs/SidebarUtils.ts b/src/libs/SidebarUtils.ts index 3968f2f95547..5e41b48eab5b 100644 --- a/src/libs/SidebarUtils.ts +++ b/src/libs/SidebarUtils.ts @@ -108,55 +108,12 @@ function setWithLimit(map: Map, key: TKey, value: TV // Variable to verify if ONYX actions are loaded let hasInitialReportActions = false; -// Originally we were passing down the reportActions as the type OnyxCollection but -// it was used as a Record in the cachedReportsKey. This was causing a type error -// for the ReportUtils.shouldReportBeInOptionList function. To fix this, we are now passing down the reportActions -// as a Record and then converting it to a Record in the -// reportActionsSelector function. -// This function was originally in the parent component as an Onyx selector, but it was moved here -// to simplify the logic and prevent the need to pass down the reportActions as an array. - -type MappedReportAction = { - reportActionID: string; - errors: OnyxCommon.Errors; - message: Message[]; -}; - -type MappedReportActions = Record; - -const reportActionsSelector = (reportActionsCollection?: Record): Record | undefined => { - if (!reportActionsCollection) { - return undefined; +type ReportActionsCount = Record< + string, + { + reportActionsCount: number; } - - return Object.values(reportActionsCollection).reduce>((acc, reportActions) => { - if (!reportActions) { - return acc; - } - Object.entries(reportActions).forEach(([reportActionID, reportAction]) => { - const modifiedAction: MappedReportAction = { - reportActionID, - errors: reportAction.errors || [], - message: [ - { - moderationDecision: reportAction.message?.[0]?.moderationDecision, - type: reportAction.message?.[0]?.type, - text: reportAction.message?.[0]?.text, - }, - ], - }; - - // If the key already exists, append to the array, otherwise create a new array - if (acc[reportActionID]) { - acc[reportActionID].push(modifiedAction); - } else { - acc[reportActionID] = [modifiedAction]; - } - }); - - return acc; - }, {}); -}; +>; /** * @returns An array of reportIDs sorted in the proper order @@ -170,11 +127,17 @@ function getOrderedReportIDs( allReportActions: OnyxCollection, transactionViolations: TransactionViolations, ): string[] { - const mappedReportActions = allReportActions ? reportActionsSelector(allReportActions as unknown as Record) : {}; + const reportActionsCount: ReportActionsCount | null = + allReportActions && + Object.keys(allReportActions).reduce((acc, reportID) => { + acc[reportID] = {reportActionsCount: Object.keys(allReportActions[reportID] as Record).length}; + return acc; + }, {}); + // Generate a unique cache key based on the function arguments const cachedReportsKey = JSON.stringify( // eslint-disable-next-line @typescript-eslint/prefer-nullish-coalescing - [currentReportId, allReports, betas, policies, priorityMode, mappedReportActions?.[`${ONYXKEYS.COLLECTION.REPORT_ACTIONS}${currentReportId}`]?.length || 1], + [currentReportId, allReports, betas, policies, priorityMode, reportActionsCount?.[`${ONYXKEYS.COLLECTION.REPORT_ACTIONS}${currentReportId}`]?.reportActionsCount || 1], (key, value: unknown) => { /** * Exclude 'participantAccountIDs', 'participants' and 'lastMessageText' not to overwhelm a cached key value with huge data, diff --git a/tests/perf-test/ReportScreen.perf-test.js b/tests/perf-test/ReportScreen.perf-test.js index 8a5b35cb2f3a..830c23a613d7 100644 --- a/tests/perf-test/ReportScreen.perf-test.js +++ b/tests/perf-test/ReportScreen.perf-test.js @@ -55,6 +55,7 @@ jest.mock('../../src/hooks/useEnvironment', () => jest.mock('../../src/libs/Permissions', () => ({ canUseLinkPreviews: jest.fn(() => true), + canUseViolations: jest.fn(() => false), })); jest.mock('../../src/hooks/usePermissions.ts'); From 6478b9402536dfd73688d681ca02228300df4e74 Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Wed, 13 Dec 2023 13:59:01 -0500 Subject: [PATCH 036/117] fix for perf test --- src/components/ReportActionItem/ReportPreview.js | 2 +- tests/perf-test/ReportScreen.perf-test.js | 1 - 2 files changed, 1 insertion(+), 2 deletions(-) diff --git a/src/components/ReportActionItem/ReportPreview.js b/src/components/ReportActionItem/ReportPreview.js index 005f244cb6b1..be1973751a91 100644 --- a/src/components/ReportActionItem/ReportPreview.js +++ b/src/components/ReportActionItem/ReportPreview.js @@ -14,12 +14,12 @@ import {showContextMenuForReport} from '@components/ShowContextMenuContext'; import Text from '@components/Text'; import withLocalize, {withLocalizePropTypes} from '@components/withLocalize'; import useLocalize from '@hooks/useLocalize'; +import {usePermissions} from '@hooks/usePermissions'; import compose from '@libs/compose'; import ControlSelection from '@libs/ControlSelection'; import * as CurrencyUtils from '@libs/CurrencyUtils'; import * as DeviceCapabilities from '@libs/DeviceCapabilities'; import Navigation from '@libs/Navigation/Navigation'; -import {usePermissions} from '@libs/Permissions'; import * as ReceiptUtils from '@libs/ReceiptUtils'; import * as ReportActionUtils from '@libs/ReportActionsUtils'; import * as ReportUtils from '@libs/ReportUtils'; diff --git a/tests/perf-test/ReportScreen.perf-test.js b/tests/perf-test/ReportScreen.perf-test.js index 830c23a613d7..8a5b35cb2f3a 100644 --- a/tests/perf-test/ReportScreen.perf-test.js +++ b/tests/perf-test/ReportScreen.perf-test.js @@ -55,7 +55,6 @@ jest.mock('../../src/hooks/useEnvironment', () => jest.mock('../../src/libs/Permissions', () => ({ canUseLinkPreviews: jest.fn(() => true), - canUseViolations: jest.fn(() => false), })); jest.mock('../../src/hooks/usePermissions.ts'); From 1736c2107ca865dc107effcd9ea1118bd4648075 Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Wed, 13 Dec 2023 14:03:17 -0500 Subject: [PATCH 037/117] Remove unused type --- src/libs/SidebarUtils.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/libs/SidebarUtils.ts b/src/libs/SidebarUtils.ts index 5e41b48eab5b..2faf6c4b4170 100644 --- a/src/libs/SidebarUtils.ts +++ b/src/libs/SidebarUtils.ts @@ -9,7 +9,7 @@ import Beta from '@src/types/onyx/Beta'; import * as OnyxCommon from '@src/types/onyx/OnyxCommon'; import Policy from '@src/types/onyx/Policy'; import Report from '@src/types/onyx/Report'; -import ReportAction, {Message, ReportActions} from '@src/types/onyx/ReportAction'; +import ReportAction, {ReportActions} from '@src/types/onyx/ReportAction'; import * as CollectionUtils from './CollectionUtils'; import * as LocalePhoneNumber from './LocalePhoneNumber'; import * as Localize from './Localize'; From 16a0a9adf7e614b731a92c6fc775067c49aa1e70 Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Wed, 13 Dec 2023 14:52:33 -0500 Subject: [PATCH 038/117] Fixing some issues with tests --- src/libs/SidebarUtils.ts | 6 ++++-- tests/perf-test/ReportScreen.perf-test.js | 6 +++++- 2 files changed, 9 insertions(+), 3 deletions(-) diff --git a/src/libs/SidebarUtils.ts b/src/libs/SidebarUtils.ts index 2faf6c4b4170..049652fbb254 100644 --- a/src/libs/SidebarUtils.ts +++ b/src/libs/SidebarUtils.ts @@ -129,8 +129,10 @@ function getOrderedReportIDs( ): string[] { const reportActionsCount: ReportActionsCount | null = allReportActions && - Object.keys(allReportActions).reduce((acc, reportID) => { - acc[reportID] = {reportActionsCount: Object.keys(allReportActions[reportID] as Record).length}; + Object.entries(allReportActions).reduce((acc, [reportID, reportActions]) => { + if (reportActions) { + acc[reportID] = {reportActionsCount: Object.keys(reportActions as Record).length}; + } return acc; }, {}); diff --git a/tests/perf-test/ReportScreen.perf-test.js b/tests/perf-test/ReportScreen.perf-test.js index 8a5b35cb2f3a..f5badad99ad3 100644 --- a/tests/perf-test/ReportScreen.perf-test.js +++ b/tests/perf-test/ReportScreen.perf-test.js @@ -56,7 +56,11 @@ jest.mock('../../src/hooks/useEnvironment', () => jest.mock('../../src/libs/Permissions', () => ({ canUseLinkPreviews: jest.fn(() => true), })); -jest.mock('../../src/hooks/usePermissions.ts'); +jest.mock('../../src/hooks/usePermissions.ts', () => + jest.fn(() => ({ + canUseViolations: true, + })), +); jest.mock('../../src/libs/Navigation/Navigation'); From feb359689af7fbd0791e792bcd2d7f113ffbf005 Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Wed, 13 Dec 2023 15:22:42 -0500 Subject: [PATCH 039/117] Was accidentally importing named instead of default --- src/components/ReportActionItem/ReportPreview.js | 2 +- tests/perf-test/ReportScreen.perf-test.js | 6 +----- 2 files changed, 2 insertions(+), 6 deletions(-) diff --git a/src/components/ReportActionItem/ReportPreview.js b/src/components/ReportActionItem/ReportPreview.js index be1973751a91..2a5f3e07de16 100644 --- a/src/components/ReportActionItem/ReportPreview.js +++ b/src/components/ReportActionItem/ReportPreview.js @@ -14,7 +14,7 @@ import {showContextMenuForReport} from '@components/ShowContextMenuContext'; import Text from '@components/Text'; import withLocalize, {withLocalizePropTypes} from '@components/withLocalize'; import useLocalize from '@hooks/useLocalize'; -import {usePermissions} from '@hooks/usePermissions'; +import usePermissions from '@hooks/usePermissions'; import compose from '@libs/compose'; import ControlSelection from '@libs/ControlSelection'; import * as CurrencyUtils from '@libs/CurrencyUtils'; diff --git a/tests/perf-test/ReportScreen.perf-test.js b/tests/perf-test/ReportScreen.perf-test.js index f5badad99ad3..8a5b35cb2f3a 100644 --- a/tests/perf-test/ReportScreen.perf-test.js +++ b/tests/perf-test/ReportScreen.perf-test.js @@ -56,11 +56,7 @@ jest.mock('../../src/hooks/useEnvironment', () => jest.mock('../../src/libs/Permissions', () => ({ canUseLinkPreviews: jest.fn(() => true), })); -jest.mock('../../src/hooks/usePermissions.ts', () => - jest.fn(() => ({ - canUseViolations: true, - })), -); +jest.mock('../../src/hooks/usePermissions.ts'); jest.mock('../../src/libs/Navigation/Navigation'); From 69b767b8c998af1e1a099dfa487be1b29e4e5bba Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Wed, 13 Dec 2023 21:08:45 -0500 Subject: [PATCH 040/117] Update src/libs/ReportUtils.ts Co-authored-by: Carlos Alvarez --- src/libs/ReportUtils.ts | 4 ---- 1 file changed, 4 deletions(-) diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index 08e2cd44dc67..3df5e24a4a1d 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -3473,10 +3473,6 @@ function shouldHideReport(report: OnyxEntry, currentReportId: string): b return parentReport?.reportID !== report?.reportID && !isChildReportHasComment; } -/** - * Checks if there are any violations belonging to the transaction in the transactionsViolations Onyx object - * then checks that the violation is of the proper type - */ function transactionHasViolation(transactionID: string, transactionViolations?: TransactionViolations): boolean { const violations: TransactionViolation[] = transactionViolations ? transactionViolations[transactionID] || [] : []; if (!violations) { From 9535a2bbf7585a1ed334700a057d85d1d04545fb Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Wed, 13 Dec 2023 21:09:00 -0500 Subject: [PATCH 041/117] Update src/components/ReportActionItem/ReportPreview.js Co-authored-by: Carlos Alvarez --- src/components/ReportActionItem/ReportPreview.js | 2 -- 1 file changed, 2 deletions(-) diff --git a/src/components/ReportActionItem/ReportPreview.js b/src/components/ReportActionItem/ReportPreview.js index 053fce6cfcd1..df012ca2f6cd 100644 --- a/src/components/ReportActionItem/ReportPreview.js +++ b/src/components/ReportActionItem/ReportPreview.js @@ -149,9 +149,7 @@ function ReportPreview(props) { const theme = useTheme(); const styles = useThemeStyles(); const {translate} = useLocalize(); - const {canUseViolations} = usePermissions(); - const managerID = props.iouReport.managerID || 0; const isCurrentUserManager = managerID === lodashGet(props.session, 'accountID'); const {totalDisplaySpend, reimbursableSpend} = ReportUtils.getMoneyRequestSpendBreakdown(props.iouReport); From 99d3c0d8e7d941e44ddf1715e99b4b9a406c6c09 Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Wed, 13 Dec 2023 21:09:10 -0500 Subject: [PATCH 042/117] Update src/hooks/__mocks__/usePermissions.ts Co-authored-by: Carlos Alvarez --- src/hooks/__mocks__/usePermissions.ts | 2 -- 1 file changed, 2 deletions(-) diff --git a/src/hooks/__mocks__/usePermissions.ts b/src/hooks/__mocks__/usePermissions.ts index 4af004095712..89a871e3c991 100644 --- a/src/hooks/__mocks__/usePermissions.ts +++ b/src/hooks/__mocks__/usePermissions.ts @@ -1,10 +1,8 @@ import UsePermissions from '@hooks/usePermissions'; /** - * * @returns {UsePermissions} A mock of the usePermissions hook. */ - const usePermissions = (): typeof UsePermissions => () => ({ canUseViolations: true, }); From 07209109f074649a167b12a9c82b1399cc31f2f2 Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Wed, 13 Dec 2023 21:38:57 -0500 Subject: [PATCH 043/117] Remove unused import --- src/libs/ReportUtils.ts | 1 - 1 file changed, 1 deletion(-) diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index 3df5e24a4a1d..3ac9db07057b 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -37,7 +37,6 @@ import {Message, ReportActionBase, ReportActions} from '@src/types/onyx/ReportAc import {Receipt, WaypointCollection} from '@src/types/onyx/Transaction'; import DeepValueOf from '@src/types/utils/DeepValueOf'; import {EmptyObject, isEmptyObject, isNotEmptyObject} from '@src/types/utils/EmptyObject'; -// import * as CollectionUtils from './CollectionUtils'; import * as CurrencyUtils from './CurrencyUtils'; import DateUtils from './DateUtils'; import isReportMessageAttachment from './isReportMessageAttachment'; From 7faf50bcbef2a3ae13f4c8a11f2088d272104e0e Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Wed, 13 Dec 2023 21:48:14 -0500 Subject: [PATCH 044/117] Fix transactionHasViolation function and ReportActionsCount type --- src/libs/ReportUtils.ts | 6 +++--- src/libs/SidebarUtils.ts | 11 +++-------- 2 files changed, 6 insertions(+), 11 deletions(-) diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index 3ac9db07057b..1a90a0d8aae0 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -3472,8 +3472,8 @@ function shouldHideReport(report: OnyxEntry, currentReportId: string): b return parentReport?.reportID !== report?.reportID && !isChildReportHasComment; } -function transactionHasViolation(transactionID: string, transactionViolations?: TransactionViolations): boolean { - const violations: TransactionViolation[] = transactionViolations ? transactionViolations[transactionID] || [] : []; +function transactionHasViolation(transactionID: string, transactionViolations: TransactionViolations): boolean { + const violations: TransactionViolation[] = transactionViolations[transactionID]; if (!violations) { return false; } @@ -3488,7 +3488,7 @@ function transactionHasViolation(transactionID: string, transactionViolations?: function transactionThreadHasViolations( report: Report, canUseViolations: boolean, - transactionViolations?: TransactionViolations, + transactionViolations: TransactionViolations, reportActions?: OnyxCollection | null, parentReportAction?: ReportAction | null, ): boolean { diff --git a/src/libs/SidebarUtils.ts b/src/libs/SidebarUtils.ts index 049652fbb254..78b9a3eae23a 100644 --- a/src/libs/SidebarUtils.ts +++ b/src/libs/SidebarUtils.ts @@ -108,12 +108,7 @@ function setWithLimit(map: Map, key: TKey, value: TV // Variable to verify if ONYX actions are loaded let hasInitialReportActions = false; -type ReportActionsCount = Record< - string, - { - reportActionsCount: number; - } ->; +type ReportActionsCount = Record; /** * @returns An array of reportIDs sorted in the proper order @@ -131,7 +126,7 @@ function getOrderedReportIDs( allReportActions && Object.entries(allReportActions).reduce((acc, [reportID, reportActions]) => { if (reportActions) { - acc[reportID] = {reportActionsCount: Object.keys(reportActions as Record).length}; + acc[reportID] = Object.keys(reportActions as Record).length; } return acc; }, {}); @@ -139,7 +134,7 @@ function getOrderedReportIDs( // Generate a unique cache key based on the function arguments const cachedReportsKey = JSON.stringify( // eslint-disable-next-line @typescript-eslint/prefer-nullish-coalescing - [currentReportId, allReports, betas, policies, priorityMode, reportActionsCount?.[`${ONYXKEYS.COLLECTION.REPORT_ACTIONS}${currentReportId}`]?.reportActionsCount || 1], + [currentReportId, allReports, betas, policies, priorityMode, reportActionsCount?.[`${ONYXKEYS.COLLECTION.REPORT_ACTIONS}${currentReportId}`] || 1], (key, value: unknown) => { /** * Exclude 'participantAccountIDs', 'participants' and 'lastMessageText' not to overwhelm a cached key value with huge data, From bf401afca2acaade3fff7c9b34ac7622dcc732b2 Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Wed, 13 Dec 2023 21:49:35 -0500 Subject: [PATCH 045/117] Remove unnecessary code in shouldHideReport function --- src/libs/ReportUtils.ts | 3 --- 1 file changed, 3 deletions(-) diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index 1a90a0d8aae0..daacc3fc1f41 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -3474,9 +3474,6 @@ function shouldHideReport(report: OnyxEntry, currentReportId: string): b function transactionHasViolation(transactionID: string, transactionViolations: TransactionViolations): boolean { const violations: TransactionViolation[] = transactionViolations[transactionID]; - if (!violations) { - return false; - } return violations.some((violation: TransactionViolation) => violation.type === 'violation'); } From f0a360b40f066e0a996b4027ba84cbf9a44e08ee Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Wed, 13 Dec 2023 21:50:22 -0500 Subject: [PATCH 046/117] Remove unnecessary comment in ReportUtils.ts --- src/libs/ReportUtils.ts | 1 - 1 file changed, 1 deletion(-) diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index daacc3fc1f41..16ff2da4fc24 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -3479,7 +3479,6 @@ function transactionHasViolation(transactionID: string, transactionViolations: T /** * Checks to see if a report's parentAction is a money request that contains a violation - * This only applies to report submitter and for reports in the open and processing states */ function transactionThreadHasViolations( From 4088e8f5651d54e260659ea396caa8c7d9b1c85a Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Wed, 13 Dec 2023 22:02:43 -0500 Subject: [PATCH 047/117] Fix betas check to be outside of the function --- src/libs/ReportUtils.ts | 5 ++--- src/libs/SidebarUtils.ts | 2 +- 2 files changed, 3 insertions(+), 4 deletions(-) diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index 16ff2da4fc24..ea33ee3123b2 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -3483,12 +3483,11 @@ function transactionHasViolation(transactionID: string, transactionViolations: T function transactionThreadHasViolations( report: Report, - canUseViolations: boolean, transactionViolations: TransactionViolations, reportActions?: OnyxCollection | null, parentReportAction?: ReportAction | null, ): boolean { - if (!canUseViolations || !reportActions) { + if (!reportActions) { return false; } const resolvedParentReportAction = parentReportAction ?? reportActions?.[`${report.parentReportID}`]?.[`${report.parentReportActionID}`]; @@ -3595,7 +3594,7 @@ function shouldReportBeInOptionList( } // Always show IOU reports with violations - if (isExpenseRequest(report) && transactionThreadHasViolations(report, betas.includes(CONST.BETAS.VIOLATIONS), transactionViolations, allReportActions)) { + if (isExpenseRequest(report) && betas.includes(CONST.BETAS.VIOLATIONS) && transactionThreadHasViolations(report, transactionViolations, allReportActions)) { return true; } diff --git a/src/libs/SidebarUtils.ts b/src/libs/SidebarUtils.ts index 78b9a3eae23a..ab7a822a01ea 100644 --- a/src/libs/SidebarUtils.ts +++ b/src/libs/SidebarUtils.ts @@ -303,7 +303,7 @@ function getOptionData( result.pendingAction = report.pendingFields ? report.pendingFields.addWorkspaceRoom || report.pendingFields.createChat : undefined; result.allReportErrors = OptionsListUtils.getAllReportErrors(report, reportActions) as OnyxCommon.Errors; result.brickRoadIndicator = - Object.keys(result.allReportErrors ?? {}).length !== 0 || ReportUtils.transactionThreadHasViolations(report, canUseViolations, transactionViolations, null, parentReportAction) + Object.keys(result.allReportErrors ?? {}).length !== 0 || (canUseViolations && ReportUtils.transactionThreadHasViolations(report, transactionViolations, null, parentReportAction)) ? CONST.BRICK_ROAD_INDICATOR_STATUS.ERROR : ''; result.ownerAccountID = report.ownerAccountID; From 766d37e1068b98c46290925ae5febf126338d177 Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Wed, 13 Dec 2023 22:10:53 -0500 Subject: [PATCH 048/117] Add comment about using usePermissions hook mock for beta tests --- tests/unit/SidebarFilterTest.js | 2 +- tests/unit/SidebarOrderTest.js | 2 +- tests/unit/SidebarTest.js | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/tests/unit/SidebarFilterTest.js b/tests/unit/SidebarFilterTest.js index 326ae6f76be2..8c16ada8154e 100644 --- a/tests/unit/SidebarFilterTest.js +++ b/tests/unit/SidebarFilterTest.js @@ -8,7 +8,7 @@ import * as LHNTestUtils from '../utils/LHNTestUtils'; import waitForBatchedUpdates from '../utils/waitForBatchedUpdates'; import wrapOnyxWithWaitForBatchedUpdates from '../utils/wrapOnyxWithWaitForBatchedUpdates'; -// Be sure to include the mocked permissions library or else the beta tests won't work +// Be sure to include the mocked permissions library as well as the usePermissions hook or else the beta tests won't work jest.mock('../../src/libs/Permissions'); jest.mock('../../src/hooks/usePermissions.ts'); diff --git a/tests/unit/SidebarOrderTest.js b/tests/unit/SidebarOrderTest.js index 55dcb7dd6258..b27fe9721e04 100644 --- a/tests/unit/SidebarOrderTest.js +++ b/tests/unit/SidebarOrderTest.js @@ -9,7 +9,7 @@ import * as LHNTestUtils from '../utils/LHNTestUtils'; import waitForBatchedUpdates from '../utils/waitForBatchedUpdates'; import wrapOnyxWithWaitForBatchedUpdates from '../utils/wrapOnyxWithWaitForBatchedUpdates'; -// Be sure to include the mocked Permissions and Expensicons libraries or else the beta tests won't work +// Be sure to include the mocked Permissions and Expensicons libraries as well as the usePermissions hook or else the beta tests won't work jest.mock('../../src/libs/Permissions'); jest.mock('../../src/hooks/usePermissions.ts'); jest.mock('../../src/components/Icon/Expensicons'); diff --git a/tests/unit/SidebarTest.js b/tests/unit/SidebarTest.js index 94ac50a6207e..78de26a4edef 100644 --- a/tests/unit/SidebarTest.js +++ b/tests/unit/SidebarTest.js @@ -7,7 +7,7 @@ import * as LHNTestUtils from '../utils/LHNTestUtils'; import waitForBatchedUpdates from '../utils/waitForBatchedUpdates'; import wrapOnyxWithWaitForBatchedUpdates from '../utils/wrapOnyxWithWaitForBatchedUpdates'; -// Be sure to include the mocked Permissions and Expensicons libraries or else the beta tests won't work +// Be sure to include the mocked Permissions and Expensicons libraries as well as the usePermissions hook or else the beta tests won't work jest.mock('../../src/libs/Permissions'); jest.mock('../../src/hooks/usePermissions.ts'); jest.mock('../../src/components/Icon/Expensicons'); From 561a4d4a2db715765a9f3815a15c5c63676f1513 Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Wed, 13 Dec 2023 22:21:42 -0500 Subject: [PATCH 049/117] Add default value for transactionViolations --- src/libs/ReportUtils.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index ea33ee3123b2..46ae84718b14 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -3594,7 +3594,7 @@ function shouldReportBeInOptionList( } // Always show IOU reports with violations - if (isExpenseRequest(report) && betas.includes(CONST.BETAS.VIOLATIONS) && transactionThreadHasViolations(report, transactionViolations, allReportActions)) { + if (isExpenseRequest(report) && betas.includes(CONST.BETAS.VIOLATIONS) && transactionThreadHasViolations(report, transactionViolations || {}, allReportActions)) { return true; } From abad008fe26bfb90e5ebf0fd193241e1a72c6771 Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Wed, 13 Dec 2023 22:25:41 -0500 Subject: [PATCH 050/117] Make linter happy --- src/hooks/__mocks__/usePermissions.ts | 6 ++---- src/libs/ReportUtils.ts | 2 +- 2 files changed, 3 insertions(+), 5 deletions(-) diff --git a/src/hooks/__mocks__/usePermissions.ts b/src/hooks/__mocks__/usePermissions.ts index 89a871e3c991..a12b1f52d8ab 100644 --- a/src/hooks/__mocks__/usePermissions.ts +++ b/src/hooks/__mocks__/usePermissions.ts @@ -1,9 +1,7 @@ -import UsePermissions from '@hooks/usePermissions'; - /** - * @returns {UsePermissions} A mock of the usePermissions hook. + * @returns A mock of the usePermissions hook. */ -const usePermissions = (): typeof UsePermissions => () => ({ +const usePermissions = () => ({ canUseViolations: true, }); export default usePermissions; diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index 46ae84718b14..5e6e208d7632 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -3594,7 +3594,7 @@ function shouldReportBeInOptionList( } // Always show IOU reports with violations - if (isExpenseRequest(report) && betas.includes(CONST.BETAS.VIOLATIONS) && transactionThreadHasViolations(report, transactionViolations || {}, allReportActions)) { + if (isExpenseRequest(report) && betas.includes(CONST.BETAS.VIOLATIONS) && transactionThreadHasViolations(report, transactionViolations ?? {}, allReportActions)) { return true; } From 563ca7b62e094c805460198becbb16d8c7f5b9a9 Mon Sep 17 00:00:00 2001 From: Daniel Edwards Date: Wed, 13 Dec 2023 22:40:30 -0500 Subject: [PATCH 051/117] Refactor getOrderedReportIDs function to improve readability --- src/libs/SidebarUtils.ts | 20 +++++++++----------- 1 file changed, 9 insertions(+), 11 deletions(-) diff --git a/src/libs/SidebarUtils.ts b/src/libs/SidebarUtils.ts index ab7a822a01ea..3f6b58020b27 100644 --- a/src/libs/SidebarUtils.ts +++ b/src/libs/SidebarUtils.ts @@ -108,8 +108,6 @@ function setWithLimit(map: Map, key: TKey, value: TV // Variable to verify if ONYX actions are loaded let hasInitialReportActions = false; -type ReportActionsCount = Record; - /** * @returns An array of reportIDs sorted in the proper order */ @@ -122,19 +120,19 @@ function getOrderedReportIDs( allReportActions: OnyxCollection, transactionViolations: TransactionViolations, ): string[] { - const reportActionsCount: ReportActionsCount | null = - allReportActions && - Object.entries(allReportActions).reduce((acc, [reportID, reportActions]) => { - if (reportActions) { - acc[reportID] = Object.keys(reportActions as Record).length; - } - return acc; - }, {}); + const countActions = (actions: ReportActions | null): number => (actions ? Object.keys(actions).length : 0); + + const getReportActionCount = (reportIDKey: string): number | null => { + if (!allReportActions?.[reportIDKey]) { + return null; + } + return countActions(allReportActions[reportIDKey]); + }; // Generate a unique cache key based on the function arguments const cachedReportsKey = JSON.stringify( // eslint-disable-next-line @typescript-eslint/prefer-nullish-coalescing - [currentReportId, allReports, betas, policies, priorityMode, reportActionsCount?.[`${ONYXKEYS.COLLECTION.REPORT_ACTIONS}${currentReportId}`] || 1], + [currentReportId, allReports, betas, policies, priorityMode, getReportActionCount(`${ONYXKEYS.COLLECTION.REPORT_ACTIONS}${currentReportId}`) || 1], (key, value: unknown) => { /** * Exclude 'participantAccountIDs', 'participants' and 'lastMessageText' not to overwhelm a cached key value with huge data, From 2e5c39bdba1e5fa7212ab63c0b49c4bdaea3bdf8 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Thu, 14 Dec 2023 13:30:06 -0500 Subject: [PATCH 052/117] feat(Violations): implement types from money-request-preview branch --- src/CONST.ts | 38 ++++++++++++- src/libs/{ => Violations}/ViolationsUtils.ts | 0 src/libs/Violations/propTypes.ts | 32 +++++++++++ src/types/onyx/PolicyCategory.ts | 4 +- src/types/onyx/PolicyTag.ts | 3 +- src/types/onyx/TransactionViolation.ts | 60 ++++++++------------ src/types/onyx/index.ts | 4 +- tests/unit/ViolationUtilsTest.js | 2 +- 8 files changed, 99 insertions(+), 44 deletions(-) rename src/libs/{ => Violations}/ViolationsUtils.ts (100%) create mode 100644 src/libs/Violations/propTypes.ts diff --git a/src/CONST.ts b/src/CONST.ts index c0e3d64b5eee..4830b1f1dead 100755 --- a/src/CONST.ts +++ b/src/CONST.ts @@ -3018,12 +3018,48 @@ const CONST = { }, /** - * Constants for maxToRenderPerBatch parameter that is used for FlatList or SectionList. This controls the amount of items rendered per batch, which is the next chunk of items rendered on every scroll. + * Constants for maxToRenderPerBatch parameter that is used for FlatList or SectionList. This controls the amount + * of items rendered per batch, which is the next chunk of items rendered on every scroll. */ MAX_TO_RENDER_PER_BATCH: { DEFAULT: 5, CAROUSEL: 3, }, + VIOLATIONS: { + ALL_TAG_LEVELS_REQUIRED: 'allTagLevelsRequired', + AUTO_REPORTED_REJECTED_EXPENSE: 'autoReportedRejectedExpense', + BILLABLE_EXPENSE: 'billableExpense', + CASH_EXPENSE_WITH_NO_RECEIPT: 'cashExpenseWithNoReceipt', + CATEGORY_OUT_OF_POLICY: 'categoryOutOfPolicy', + CONVERSION_SURCHARGE: 'conversionSurcharge', + CUSTOM_UNIT_OUT_OF_POLICY: 'customUnitOutOfPolicy', + DUPLICATED_TRANSACTION: 'duplicatedTransaction', + FIELD_REQUIRED: 'fieldRequired', + FUTURE_DATE: 'futureDate', + INVOICE_MARKUP: 'invoiceMarkup', + MAX_AGE: 'maxAge', + MISSING_CATEGORY: 'missingCategory', + MISSING_COMMENT: 'missingComment', + MISSING_TAG: 'missingTag', + MODIFIED_AMOUNT: 'modifiedAmount', + MODIFIED_DATE: 'modifiedDate', + NON_EXPENSIWORKS_EXPENSE: 'nonExpensiworksExpense', + OVER_AUTO_APPROVAL_LIMIT: 'overAutoApprovalLimit', + OVER_CATEGORY_LIMIT: 'overCategoryLimit', + OVER_LIMIT: 'overLimit', + OVER_LIMIT_ATTENDEE: 'overLimitAttendee', + PER_DAY_LIMIT: 'perDayLimit', + RECEIPT_NOT_SMART_SCANNED: 'receiptNotSmartScanned', + RECEIPT_REQUIRED: 'receiptRequired', + RTER: 'rter', + SMARTSCAN_FAILED: 'smartscanFailed', + SOME_TAG_LEVELS_REQUIRED: 'someTagLevelsRequired', + TAG_OUT_OF_POLICY: 'tagOutOfPolicy', + TAX_AMOUNT_CHANGED: 'taxAmountChanged', + TAX_OUT_OF_POLICY: 'taxOutOfPolicy', + TAX_RATE_CHANGED: 'taxRateChanged', + TAX_REQUIRED: 'taxRequired', + }, } as const; export default CONST; diff --git a/src/libs/ViolationsUtils.ts b/src/libs/Violations/ViolationsUtils.ts similarity index 100% rename from src/libs/ViolationsUtils.ts rename to src/libs/Violations/ViolationsUtils.ts diff --git a/src/libs/Violations/propTypes.ts b/src/libs/Violations/propTypes.ts new file mode 100644 index 000000000000..e32d9ddd6a2c --- /dev/null +++ b/src/libs/Violations/propTypes.ts @@ -0,0 +1,32 @@ +import PropTypes from 'prop-types'; +import CONST from '@src/CONST'; + +const violationNames = Object.values(CONST.VIOLATIONS); + +const transactionViolationPropType = PropTypes.shape({ + type: PropTypes.string.isRequired, + name: PropTypes.oneOf(violationNames).isRequired, + data: PropTypes.shape({ + rejectedBy: PropTypes.string, + rejectReason: PropTypes.string, + amount: PropTypes.string, + surcharge: PropTypes.number, + invoiceMarkup: PropTypes.number, + maxAge: PropTypes.number, + tagName: PropTypes.string, + formattedLimitAmount: PropTypes.string, + categoryLimit: PropTypes.string, + limit: PropTypes.string, + category: PropTypes.string, + brokenBankConnection: PropTypes.bool, + isAdmin: PropTypes.bool, + email: PropTypes.string, + isTransactionOlderThan7Days: PropTypes.bool, + member: PropTypes.string, + taxName: PropTypes.string, + }), +}); + +const transactionViolationsPropType = PropTypes.arrayOf(transactionViolationPropType); + +export {transactionViolationsPropType, transactionViolationPropType}; diff --git a/src/types/onyx/PolicyCategory.ts b/src/types/onyx/PolicyCategory.ts index b6dfb7bbab9a..03d5877bc5b5 100644 --- a/src/types/onyx/PolicyCategory.ts +++ b/src/types/onyx/PolicyCategory.ts @@ -20,5 +20,5 @@ type PolicyCategory = { }; type PolicyCategories = Record; -export default PolicyCategory; -export type {PolicyCategories}; + +export type {PolicyCategory, PolicyCategories}; diff --git a/src/types/onyx/PolicyTag.ts b/src/types/onyx/PolicyTag.ts index 7807dcc00433..58a21dcf4df5 100644 --- a/src/types/onyx/PolicyTag.ts +++ b/src/types/onyx/PolicyTag.ts @@ -12,5 +12,4 @@ type PolicyTag = { type PolicyTags = Record; -export default PolicyTag; -export type {PolicyTags}; +export type {PolicyTag, PolicyTags}; diff --git a/src/types/onyx/TransactionViolation.ts b/src/types/onyx/TransactionViolation.ts index bd8f3059caea..1d02de12a52a 100644 --- a/src/types/onyx/TransactionViolation.ts +++ b/src/types/onyx/TransactionViolation.ts @@ -1,46 +1,34 @@ +import CONST from '@src/CONST'; + /** - * Names of transaction violations + * Names of violations. + * Derived from `CONST.VIOLATIONS` to maintain a single source of truth. */ -type ViolationName = - | 'allTagLevelsRequired' - | 'autoReportedRejectedExpense' - | 'billableExpense' - | 'cashExpenseWithNoReceipt' - | 'categoryOutOfPolicy' - | 'conversionSurcharge' - | 'customUnitOutOfPolicy' - | 'duplicatedTransaction' - | 'fieldRequired' - | 'futureDate' - | 'invoiceMarkup' - | 'maxAge' - | 'missingCategory' - | 'missingComment' - | 'missingTag' - | 'modifiedAmount' - | 'modifiedDate' - | 'nonExpensiworksExpense' - | 'overAutoApprovalLimit' - | 'overCategoryLimit' - | 'overLimit' - | 'overLimitAttendee' - | 'perDayLimit' - | 'receiptNotSmartScanned' - | 'receiptRequired' - | 'rter' - | 'smartscanFailed' - | 'someTagLevelsRequired' - | 'tagOutOfPolicy' - | 'taxAmountChanged' - | 'taxOutOfPolicy' - | 'taxRateChanged' - | 'taxRequired'; +type ViolationName = (typeof CONST.VIOLATIONS)[keyof typeof CONST.VIOLATIONS]; type TransactionViolation = { type: string; name: ViolationName; userMessage: string; - data?: Record; + data?: { + rejectedBy?: string; + rejectReason?: string; + amount?: string; + surcharge?: number; + invoiceMarkup?: number; + maxAge?: number; + tagName?: string; + formattedLimitAmount?: string; + categoryLimit?: string; + limit?: string; + category?: string; + brokenBankConnection?: boolean; + isAdmin?: boolean; + email?: string; + isTransactionOlderThan7Days?: boolean; + member?: string; + taxName?: string; + }; }; type TransactionViolations = Record; diff --git a/src/types/onyx/index.ts b/src/types/onyx/index.ts index b52a9c919c39..23e2eb1f3bcd 100644 --- a/src/types/onyx/index.ts +++ b/src/types/onyx/index.ts @@ -23,9 +23,9 @@ import PersonalBankAccount from './PersonalBankAccount'; import PersonalDetails, {PersonalDetailsList} from './PersonalDetails'; import PlaidData from './PlaidData'; import Policy from './Policy'; -import PolicyCategory, {PolicyCategories} from './PolicyCategory'; +import {PolicyCategories, PolicyCategory} from './PolicyCategory'; import PolicyMember, {PolicyMembers} from './PolicyMember'; -import PolicyTag, {PolicyTags} from './PolicyTag'; +import {PolicyTag, PolicyTags} from './PolicyTag'; import PrivatePersonalDetails from './PrivatePersonalDetails'; import RecentlyUsedCategories from './RecentlyUsedCategories'; import RecentlyUsedTags from './RecentlyUsedTags'; diff --git a/tests/unit/ViolationUtilsTest.js b/tests/unit/ViolationUtilsTest.js index cc84c547da2e..f0b53443831e 100644 --- a/tests/unit/ViolationUtilsTest.js +++ b/tests/unit/ViolationUtilsTest.js @@ -1,6 +1,6 @@ import {beforeEach} from '@jest/globals'; import Onyx from 'react-native-onyx'; -import ViolationsUtils from '@libs/ViolationsUtils'; +import ViolationsUtils from '@libs/Violations/ViolationsUtils'; import ONYXKEYS from '@src/ONYXKEYS'; const categoryOutOfPolicyViolation = { From 30a77c92d3d05272f193ba856df513e53189f99c Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Thu, 14 Dec 2023 13:31:20 -0500 Subject: [PATCH 053/117] feat(Violations): replace transactionViolationsProp with imported prop from libs/Violations --- .../LHNOptionsList/LHNOptionsList.js | 25 ++----------------- 1 file changed, 2 insertions(+), 23 deletions(-) diff --git a/src/components/LHNOptionsList/LHNOptionsList.js b/src/components/LHNOptionsList/LHNOptionsList.js index 87bff1667c3f..ab6da0afa1c8 100644 --- a/src/components/LHNOptionsList/LHNOptionsList.js +++ b/src/components/LHNOptionsList/LHNOptionsList.js @@ -13,6 +13,7 @@ import useThemeStyles from '@hooks/useThemeStyles'; import compose from '@libs/compose'; import * as OptionsListUtils from '@libs/OptionsListUtils'; import * as ReportUtils from '@libs/ReportUtils'; +import {transactionViolationsPropType} from '@libs/Violations/propTypes'; import reportActionPropTypes from '@pages/home/report/reportActionPropTypes'; import reportPropTypes from '@pages/reportPropTypes'; import stylePropTypes from '@styles/stylePropTypes'; @@ -68,29 +69,7 @@ const propTypes = { draftComments: PropTypes.objectOf(PropTypes.string), /** The list of transaction violations */ - transactionViolations: PropTypes.shape({ - violations: PropTypes.arrayOf( - PropTypes.shape({ - /** The transaction ID */ - transactionID: PropTypes.number, - - /** The transaction violation type */ - type: PropTypes.string, - - /** The transaction violation message */ - message: PropTypes.string, - - /** The transaction violation data */ - data: PropTypes.shape({ - /** The transaction violation data field */ - field: PropTypes.string, - - /** The transaction violation data value */ - value: PropTypes.string, - }), - }), - ), - }), + transactionViolations: transactionViolationsPropType, ...withCurrentReportIDPropTypes, }; From 2e3d3e751274fee7da86998745b5432963d9735a Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Thu, 14 Dec 2023 14:19:37 -0500 Subject: [PATCH 054/117] feat(Violations): make reportActions mandatory and remove parentReportAction --- src/libs/ReportUtils.ts | 10 ++-------- 1 file changed, 2 insertions(+), 8 deletions(-) diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index 96bd16f03e9e..164feac59b8c 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -3485,17 +3485,11 @@ function transactionHasViolation(transactionID: string, transactionViolations: T /** * Checks to see if a report's parentAction is a money request that contains a violation */ - -function transactionThreadHasViolations( - report: Report, - transactionViolations: TransactionViolations, - reportActions?: OnyxCollection | null, - parentReportAction?: ReportAction | null, -): boolean { +function transactionThreadHasViolations(report: Report, transactionViolations: TransactionViolations, reportActions: OnyxCollection | null = {}): boolean { if (!reportActions) { return false; } - const resolvedParentReportAction = parentReportAction ?? reportActions?.[`${report.parentReportID}`]?.[`${report.parentReportActionID}`]; + const resolvedParentReportAction = reportActions[`${report.parentReportID}`]?.[`${report.parentReportActionID}`]; if (!resolvedParentReportAction) { return false; } From 9503748199f93349acd0a3c70dfd059345b89666 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Thu, 14 Dec 2023 14:19:48 -0500 Subject: [PATCH 055/117] feat(Violations): remove whitespace --- src/libs/ReportUtils.ts | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index 164feac59b8c..357404e8283c 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -3483,7 +3483,7 @@ function transactionHasViolation(transactionID: string, transactionViolations: T } /** - * Checks to see if a report's parentAction is a money request that contains a violation + * Checks to see if a report's parentAction is a money request that contains a violation */ function transactionThreadHasViolations(report: Report, transactionViolations: TransactionViolations, reportActions: OnyxCollection | null = {}): boolean { if (!reportActions) { @@ -3510,7 +3510,7 @@ function transactionThreadHasViolations(report: Report, transactionViolations: T } /** - * Checks to see if a report contains a violation + * Checks to see if a report contains a violation */ function reportHasViolations(reportID: string, transactionViolations: TransactionViolations): boolean { const transactions = TransactionUtils.getAllReportTransactions(reportID); From 2871cd833d1c5d5c2c8c906e446f8b50e2702d7c Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Thu, 14 Dec 2023 15:10:21 -0500 Subject: [PATCH 056/117] feat(Violations): simplify --- src/libs/SidebarUtils.ts | 37 ++++++++++++------------------------- 1 file changed, 12 insertions(+), 25 deletions(-) diff --git a/src/libs/SidebarUtils.ts b/src/libs/SidebarUtils.ts index 3f6b58020b27..6a5089d4de55 100644 --- a/src/libs/SidebarUtils.ts +++ b/src/libs/SidebarUtils.ts @@ -120,30 +120,17 @@ function getOrderedReportIDs( allReportActions: OnyxCollection, transactionViolations: TransactionViolations, ): string[] { - const countActions = (actions: ReportActions | null): number => (actions ? Object.keys(actions).length : 0); - - const getReportActionCount = (reportIDKey: string): number | null => { - if (!allReportActions?.[reportIDKey]) { - return null; - } - return countActions(allReportActions[reportIDKey]); - }; + const reportIDKey = `${ONYXKEYS.COLLECTION.REPORT_ACTIONS}${currentReportId}`; + const reportActionCount = allReportActions?.[reportIDKey]?.length ?? 1; // Generate a unique cache key based on the function arguments const cachedReportsKey = JSON.stringify( - // eslint-disable-next-line @typescript-eslint/prefer-nullish-coalescing - [currentReportId, allReports, betas, policies, priorityMode, getReportActionCount(`${ONYXKEYS.COLLECTION.REPORT_ACTIONS}${currentReportId}`) || 1], - (key, value: unknown) => { - /** - * Exclude 'participantAccountIDs', 'participants' and 'lastMessageText' not to overwhelm a cached key value with huge data, - * which we don't need to store in a cacheKey - */ - if (key === 'participantAccountIDs' || key === 'participants' || key === 'lastMessageText') { - return undefined; - } - - return value; - }, + [currentReportId, allReports, betas, policies, priorityMode, reportActionCount], + /** + * Exclude 'participantAccountIDs', 'participants' and 'lastMessageText' keys from the stringified + * objects to keep the size of the resulting key manageable. + */ + (key, value: unknown) => (['participantAccountIDs', 'participants', 'lastMessageText'].includes(key) ? undefined : value), ); // Check if the result is already in the cache @@ -286,8 +273,11 @@ function getOptionData( isWaitingOnBankAccount: false, isAllowedToComment: true, }; + const participantPersonalDetailList: PersonalDetails[] = Object.values(OptionsListUtils.getPersonalDetailsForAccountIDs(report.participantAccountIDs ?? [], personalDetails)); const personalDetail = participantPersonalDetailList[0] ?? {}; + const hasErrors = Object.keys(result.allReportErrors ?? {}).length !== 0; + const hasViolations = canUseViolations && ReportUtils.transactionThreadHasViolations(report, transactionViolations, null); result.isThread = ReportUtils.isChatThread(report); result.isChatRoom = ReportUtils.isChatRoom(report); @@ -300,10 +290,7 @@ function getOptionData( result.shouldShowSubscript = ReportUtils.shouldReportShowSubscript(report); result.pendingAction = report.pendingFields ? report.pendingFields.addWorkspaceRoom || report.pendingFields.createChat : undefined; result.allReportErrors = OptionsListUtils.getAllReportErrors(report, reportActions) as OnyxCommon.Errors; - result.brickRoadIndicator = - Object.keys(result.allReportErrors ?? {}).length !== 0 || (canUseViolations && ReportUtils.transactionThreadHasViolations(report, transactionViolations, null, parentReportAction)) - ? CONST.BRICK_ROAD_INDICATOR_STATUS.ERROR - : ''; + result.brickRoadIndicator = hasErrors || hasViolations ? CONST.BRICK_ROAD_INDICATOR_STATUS.ERROR : ''; result.ownerAccountID = report.ownerAccountID; result.managerID = report.managerID; result.reportID = report.reportID; From 5dd7d01072bfacc86b1d898c599b0d204dd6ec8f Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Thu, 14 Dec 2023 15:10:33 -0500 Subject: [PATCH 057/117] feat(Violations): update comment --- tests/unit/SidebarFilterTest.js | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/tests/unit/SidebarFilterTest.js b/tests/unit/SidebarFilterTest.js index 8c16ada8154e..7ae49e2eab9e 100644 --- a/tests/unit/SidebarFilterTest.js +++ b/tests/unit/SidebarFilterTest.js @@ -8,7 +8,8 @@ import * as LHNTestUtils from '../utils/LHNTestUtils'; import waitForBatchedUpdates from '../utils/waitForBatchedUpdates'; import wrapOnyxWithWaitForBatchedUpdates from '../utils/wrapOnyxWithWaitForBatchedUpdates'; -// Be sure to include the mocked permissions library as well as the usePermissions hook or else the beta tests won't work +// Be sure to include the mocked permissions library, as some components that are rendered +// during the test depend on its methods. jest.mock('../../src/libs/Permissions'); jest.mock('../../src/hooks/usePermissions.ts'); From d169a435d8048fcbbeece60fb74ba08cc6ea54e2 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Thu, 14 Dec 2023 15:17:06 -0500 Subject: [PATCH 058/117] feat(Violations): simplify feat(Violations): simplify feat(Violations): simplify feat(Violations): simplify feat(Violations): simplify feat(Violations): simplify feat(Violations): simplify feat(Violations): simplify --- src/libs/ReportUtils.ts | 65 +++++++++++------------------------------ 1 file changed, 17 insertions(+), 48 deletions(-) diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index 357404e8283c..d4fd2cc80960 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -483,9 +483,7 @@ function getPolicyName(report: OnyxEntry | undefined | EmptyObject, retu // Public rooms send back the policy name with the reportSummary, // since they can also be accessed by people who aren't in the workspace // eslint-disable-next-line @typescript-eslint/prefer-nullish-coalescing - const policyName = finalPolicy?.name || report?.policyName || report?.oldPolicyName || noPolicyFound; - - return policyName; + return finalPolicy?.name || report?.policyName || report?.oldPolicyName || noPolicyFound; } /** @@ -536,11 +534,7 @@ function isCanceledTaskReport(report: OnyxEntry | EmptyObject = {}, pare return true; } - if (isNotEmptyObject(report) && report?.isDeletedParentAction) { - return true; - } - - return false; + return Boolean(isNotEmptyObject(report) && report?.isDeletedParentAction); } /** @@ -1176,8 +1170,7 @@ function getReportRecipientAccountIDs(report: OnyxEntry, currentLoginAcc } const reportParticipants = finalParticipantAccountIDs?.filter((accountID) => accountID !== currentLoginAccountID) ?? []; - const participantsWithoutExpensifyAccountIDs = reportParticipants.filter((participant) => !CONST.EXPENSIFY_ACCOUNT_IDS.includes(participant ?? 0)); - return participantsWithoutExpensifyAccountIDs; + return reportParticipants.filter((participant) => !CONST.EXPENSIFY_ACCOUNT_IDS.includes(participant ?? 0)); } /** @@ -1281,13 +1274,12 @@ function getWorkspaceIcon(report: OnyxEntry, policy: OnyxEntry = ? allPolicies?.[`${ONYXKEYS.COLLECTION.POLICY}${report?.policyID}`]?.avatar : getDefaultWorkspaceAvatar(workspaceName); - const workspaceIcon: Icon = { + return { source: policyExpenseChatAvatarSource ?? '', type: CONST.ICON_TYPE_WORKSPACE, name: workspaceName, id: -1, - }; - return workspaceIcon; + } as Icon; } /** @@ -1625,12 +1617,8 @@ function requiresAttentionFromCurrentUser(optionOrReport: OnyxEntry | Op return true; } - // Has a child report that is awaiting action (e.g. approve, pay, add bank account) from current user - if (optionOrReport.hasOutstandingChildRequest) { - return true; - } - - return false; + // has a child report that is awaiting action (e.g. approve, pay, add bank account) from current user + return Boolean(optionOrReport.hasOutstandingChildRequest); } /** @@ -3460,11 +3448,7 @@ function canAccessReport(report: OnyxEntry, policies: OnyxCollection { const sortedNewParticipantList = newParticipantList.sort(); @@ -3719,7 +3704,8 @@ function shouldShowFlagComment(reportAction: OnyxEntry, report: On } /** - * @param sortedAndFilteredReportActions - reportActions for the report, sorted newest to oldest, and filtered for only those that should be visible + * @param sortedAndFilteredReportActions - reportActions for the report, sorted newest to oldest, and filtered for only + * those that should be visible */ function getNewMarkerReportActionID(report: OnyxEntry, sortedAndFilteredReportActions: ReportAction[]): string { if (!isUnread(report)) { @@ -3988,19 +3974,7 @@ function shouldReportShowSubscript(report: OnyxEntry): boolean { return true; } - if (isExpenseRequest(report)) { - return true; - } - - if (isWorkspaceTaskReport(report)) { - return true; - } - - if (isWorkspaceThread(report)) { - return true; - } - - return false; + return isExpenseRequest(report) || isWorkspaceTaskReport(report) || isWorkspaceThread(report); } /** @@ -4236,8 +4210,8 @@ function getParticipantsIDs(report: OnyxEntry): number[] { // Build participants list for IOU/expense reports if (isMoneyRequestReport(report)) { const onlyTruthyValues = [report.managerID, report.ownerAccountID, ...participants].filter(Boolean) as number[]; - const onlyUnique = [...new Set([...onlyTruthyValues])]; - return onlyUnique; + // return only unique values + return [...new Set([...onlyTruthyValues])]; } return participants; } @@ -4327,8 +4301,7 @@ function shouldUseFullTitleToDisplay(report: OnyxEntry): boolean { } function getRoom(type: ValueOf, policyID: string): OnyxEntry | undefined { - const room = Object.values(allReports ?? {}).find((report) => report?.policyID === policyID && report?.chatType === type && !isThread(report)); - return room; + return Object.values(allReports ?? {}).find((report) => report?.policyID === policyID && report?.chatType === type && !isThread(report)); } /** @@ -4364,11 +4337,7 @@ function shouldAutoFocusOnKeyPress(event: KeyboardEvent): boolean { return false; } - if (event.code === 'Space') { - return false; - } - - return true; + return event.code !== 'Space'; } /** From aa7822e75831f25cc1cd9d6ac0c9365edfcba90a Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Thu, 14 Dec 2023 16:05:32 -0500 Subject: [PATCH 059/117] feat(Violations): fix type of transactionViolations propType --- src/components/LHNOptionsList/LHNOptionsList.js | 3 ++- src/components/LHNOptionsList/OptionRowLHNData.js | 4 ++++ src/libs/Violations/propTypes.ts | 2 +- 3 files changed, 7 insertions(+), 2 deletions(-) diff --git a/src/components/LHNOptionsList/LHNOptionsList.js b/src/components/LHNOptionsList/LHNOptionsList.js index ab6da0afa1c8..8b2384c4dec8 100644 --- a/src/components/LHNOptionsList/LHNOptionsList.js +++ b/src/components/LHNOptionsList/LHNOptionsList.js @@ -65,6 +65,7 @@ const propTypes = { /** The transaction from the parent report action */ transactions: PropTypes.objectOf(transactionPropTypes), + /** List of draft comments */ draftComments: PropTypes.objectOf(PropTypes.string), @@ -214,7 +215,7 @@ export default compose( key: ONYXKEYS.COLLECTION.REPORT_DRAFT_COMMENT, }, transactionViolations: { - key: ONYXKEYS.COLLECTION.TRANSACTION_VIOLATIONS, + key: ({currentReportId}) => `${ONYXKEYS.COLLECTION.TRANSACTION_VIOLATIONS}${currentReportId}`, }, }), )(LHNOptionsList); diff --git a/src/components/LHNOptionsList/OptionRowLHNData.js b/src/components/LHNOptionsList/OptionRowLHNData.js index 975299bb8800..82ec30cafe77 100644 --- a/src/components/LHNOptionsList/OptionRowLHNData.js +++ b/src/components/LHNOptionsList/OptionRowLHNData.js @@ -7,6 +7,7 @@ import transactionPropTypes from '@components/transactionPropTypes'; import * as ReportActionsUtils from '@libs/ReportActionsUtils'; import SidebarUtils from '@libs/SidebarUtils'; import * as TransactionUtils from '@libs/TransactionUtils'; +import {transactionViolationsPropType} from '@libs/Violations/propTypes'; import reportActionPropTypes from '@pages/home/report/reportActionPropTypes'; import * as Report from '@userActions/Report'; import CONST from '@src/CONST'; @@ -42,6 +43,9 @@ const propTypes = { /** The transaction from the parent report action */ transaction: transactionPropTypes, + /** Any violations associated with the report */ + transactionViolations: transactionViolationsPropType, + ...basePropTypes, }; diff --git a/src/libs/Violations/propTypes.ts b/src/libs/Violations/propTypes.ts index e32d9ddd6a2c..f0b84cf7d3b1 100644 --- a/src/libs/Violations/propTypes.ts +++ b/src/libs/Violations/propTypes.ts @@ -27,6 +27,6 @@ const transactionViolationPropType = PropTypes.shape({ }), }); -const transactionViolationsPropType = PropTypes.arrayOf(transactionViolationPropType); +const transactionViolationsPropType = PropTypes.objectOf(transactionViolationPropType); export {transactionViolationsPropType, transactionViolationPropType}; From d86616def9715f7511617cb97c40daf722109897 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Fri, 15 Dec 2023 10:37:35 -0500 Subject: [PATCH 060/117] feat(Violations): revert simplifications --- src/libs/ReportUtils.ts | 59 ++++++++++++++++++++++++++++++++--------- 1 file changed, 46 insertions(+), 13 deletions(-) diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index d4fd2cc80960..177718c2c36c 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -483,7 +483,9 @@ function getPolicyName(report: OnyxEntry | undefined | EmptyObject, retu // Public rooms send back the policy name with the reportSummary, // since they can also be accessed by people who aren't in the workspace // eslint-disable-next-line @typescript-eslint/prefer-nullish-coalescing - return finalPolicy?.name || report?.policyName || report?.oldPolicyName || noPolicyFound; + const policyName = finalPolicy?.name || report?.policyName || report?.oldPolicyName || noPolicyFound; + + return policyName; } /** @@ -534,7 +536,11 @@ function isCanceledTaskReport(report: OnyxEntry | EmptyObject = {}, pare return true; } - return Boolean(isNotEmptyObject(report) && report?.isDeletedParentAction); + if (isNotEmptyObject(report) && report?.isDeletedParentAction) { + return true; + } + + return false; } /** @@ -1170,7 +1176,8 @@ function getReportRecipientAccountIDs(report: OnyxEntry, currentLoginAcc } const reportParticipants = finalParticipantAccountIDs?.filter((accountID) => accountID !== currentLoginAccountID) ?? []; - return reportParticipants.filter((participant) => !CONST.EXPENSIFY_ACCOUNT_IDS.includes(participant ?? 0)); + const participantsWithoutExpensifyAccountIDs = reportParticipants.filter((participant) => !CONST.EXPENSIFY_ACCOUNT_IDS.includes(participant ?? 0)); + return participantsWithoutExpensifyAccountIDs; } /** @@ -1274,12 +1281,13 @@ function getWorkspaceIcon(report: OnyxEntry, policy: OnyxEntry = ? allPolicies?.[`${ONYXKEYS.COLLECTION.POLICY}${report?.policyID}`]?.avatar : getDefaultWorkspaceAvatar(workspaceName); - return { + const workspaceIcon: Icon = { source: policyExpenseChatAvatarSource ?? '', type: CONST.ICON_TYPE_WORKSPACE, name: workspaceName, id: -1, - } as Icon; + }; + return workspaceIcon; } /** @@ -1617,8 +1625,12 @@ function requiresAttentionFromCurrentUser(optionOrReport: OnyxEntry | Op return true; } - // has a child report that is awaiting action (e.g. approve, pay, add bank account) from current user - return Boolean(optionOrReport.hasOutstandingChildRequest); + // Has a child report that is awaiting action (e.g. approve, pay, add bank account) from current user + if (optionOrReport.hasOutstandingChildRequest) { + return true; + } + + return false; } /** @@ -3448,7 +3460,11 @@ function canAccessReport(report: OnyxEntry, policies: OnyxCollection): boolean { return true; } - return isExpenseRequest(report) || isWorkspaceTaskReport(report) || isWorkspaceThread(report); + if (isExpenseRequest(report)) { + return true; + } + + if (isWorkspaceTaskReport(report)) { + return true; + } + + if (isWorkspaceThread(report)) { + return true; + } + + return false; } /** @@ -4210,8 +4238,8 @@ function getParticipantsIDs(report: OnyxEntry): number[] { // Build participants list for IOU/expense reports if (isMoneyRequestReport(report)) { const onlyTruthyValues = [report.managerID, report.ownerAccountID, ...participants].filter(Boolean) as number[]; - // return only unique values - return [...new Set([...onlyTruthyValues])]; + const onlyUnique = [...new Set([...onlyTruthyValues])]; + return onlyUnique; } return participants; } @@ -4301,7 +4329,8 @@ function shouldUseFullTitleToDisplay(report: OnyxEntry): boolean { } function getRoom(type: ValueOf, policyID: string): OnyxEntry | undefined { - return Object.values(allReports ?? {}).find((report) => report?.policyID === policyID && report?.chatType === type && !isThread(report)); + const room = Object.values(allReports ?? {}).find((report) => report?.policyID === policyID && report?.chatType === type && !isThread(report)); + return room; } /** @@ -4337,7 +4366,11 @@ function shouldAutoFocusOnKeyPress(event: KeyboardEvent): boolean { return false; } - return event.code !== 'Space'; + if (event.code === 'Space') { + return false; + } + + return true; } /** From ccb12ccdc8e7883de6a14fe76b5ce50f667bbe52 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Fri, 15 Dec 2023 10:48:28 -0500 Subject: [PATCH 061/117] feat(Violations): add second withOnyx --- src/components/LHNOptionsList/LHNOptionsList.js | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/src/components/LHNOptionsList/LHNOptionsList.js b/src/components/LHNOptionsList/LHNOptionsList.js index 8b2384c4dec8..2999cd9ebda5 100644 --- a/src/components/LHNOptionsList/LHNOptionsList.js +++ b/src/components/LHNOptionsList/LHNOptionsList.js @@ -1,5 +1,6 @@ import {FlashList} from '@shopify/flash-list'; import lodashGet from 'lodash/get'; +import lodashMap from 'lodash/map'; import PropTypes from 'prop-types'; import React, {useCallback} from 'react'; import {View} from 'react-native'; @@ -214,8 +215,10 @@ export default compose( draftComments: { key: ONYXKEYS.COLLECTION.REPORT_DRAFT_COMMENT, }, + }), + withOnyx({ transactionViolations: { - key: ({currentReportId}) => `${ONYXKEYS.COLLECTION.TRANSACTION_VIOLATIONS}${currentReportId}`, + key: ({transactions}) => lodashMap(transactions, (t) => `${ONYXKEYS.COLLECTION.TRANSACTION_VIOLATIONS}${t.transactionID}`), }, }), )(LHNOptionsList); From e3d4ae85c83554a7e493806c2acbd7fbe8ee0aff Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Fri, 15 Dec 2023 16:29:42 -0500 Subject: [PATCH 062/117] feat(Violations): get transactions directly --- src/components/LHNOptionsList/LHNOptionsList.js | 5 +---- 1 file changed, 1 insertion(+), 4 deletions(-) diff --git a/src/components/LHNOptionsList/LHNOptionsList.js b/src/components/LHNOptionsList/LHNOptionsList.js index 2999cd9ebda5..1dfb70f65847 100644 --- a/src/components/LHNOptionsList/LHNOptionsList.js +++ b/src/components/LHNOptionsList/LHNOptionsList.js @@ -1,6 +1,5 @@ import {FlashList} from '@shopify/flash-list'; import lodashGet from 'lodash/get'; -import lodashMap from 'lodash/map'; import PropTypes from 'prop-types'; import React, {useCallback} from 'react'; import {View} from 'react-native'; @@ -215,10 +214,8 @@ export default compose( draftComments: { key: ONYXKEYS.COLLECTION.REPORT_DRAFT_COMMENT, }, - }), - withOnyx({ transactionViolations: { - key: ({transactions}) => lodashMap(transactions, (t) => `${ONYXKEYS.COLLECTION.TRANSACTION_VIOLATIONS}${t.transactionID}`), + key: ONYXKEYS.COLLECTION.TRANSACTION_VIOLATIONS, }, }), )(LHNOptionsList); From d4678532df22c35f90852c51dc4289cb15fdfa70 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Mon, 18 Dec 2023 20:47:38 -0500 Subject: [PATCH 063/117] feat(Violations): fix propsTypes --- src/libs/ReportUtils.ts | 6 ++++-- src/libs/Violations/propTypes.ts | 2 +- src/pages/home/ReportScreen.js | 4 ++-- 3 files changed, 7 insertions(+), 5 deletions(-) diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index 0eda7f18a5ac..61e8845c95d0 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -3477,9 +3477,11 @@ function shouldHideReport(report: OnyxEntry, currentReportId: string): b return parentReport?.reportID !== report?.reportID && !isChildReportHasComment; } +/** + * Checks if any violations for the provided transaction are of type 'violation' + */ function transactionHasViolation(transactionID: string, transactionViolations: TransactionViolations): boolean { - const violations: TransactionViolation[] = transactionViolations[transactionID]; - return violations.some((violation: TransactionViolation) => violation.type === 'violation'); + return transactionViolations[transactionID]?.some((violation: TransactionViolation) => violation.type === 'violation'); } /** diff --git a/src/libs/Violations/propTypes.ts b/src/libs/Violations/propTypes.ts index f0b84cf7d3b1..4b5e84405cba 100644 --- a/src/libs/Violations/propTypes.ts +++ b/src/libs/Violations/propTypes.ts @@ -27,6 +27,6 @@ const transactionViolationPropType = PropTypes.shape({ }), }); -const transactionViolationsPropType = PropTypes.objectOf(transactionViolationPropType); +const transactionViolationsPropType = PropTypes.objectOf(PropTypes.arrayOf(transactionViolationPropType)); export {transactionViolationsPropType, transactionViolationPropType}; diff --git a/src/pages/home/ReportScreen.js b/src/pages/home/ReportScreen.js index 161b8aa8889d..c3154d600672 100644 --- a/src/pages/home/ReportScreen.js +++ b/src/pages/home/ReportScreen.js @@ -66,7 +66,7 @@ const propTypes = { reportMetadata: reportMetadataPropTypes, /** Array of report actions for this report */ - reportActions: PropTypes.arrayOf(PropTypes.shape(reportActionPropTypes)), + reportActions: PropTypes.objectOf(PropTypes.shape(reportActionPropTypes)), /** Whether the composer is full size */ isComposerFullSize: PropTypes.bool, @@ -103,7 +103,7 @@ const propTypes = { const defaultProps = { isSidebarLoaded: false, - reportActions: [], + reportActions: {}, report: {}, reportMetadata: { isLoadingInitialReportActions: true, From fdab084465de771457755a5ed50965d25deef131 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Tue, 19 Dec 2023 11:10:58 -0500 Subject: [PATCH 064/117] feat(Violations): rename --- src/components/ReportActionItem/ReportPreview.js | 2 +- src/libs/ReportUtils.ts | 4 ++-- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/src/components/ReportActionItem/ReportPreview.js b/src/components/ReportActionItem/ReportPreview.js index df012ca2f6cd..ead4244e14ec 100644 --- a/src/components/ReportActionItem/ReportPreview.js +++ b/src/components/ReportActionItem/ReportPreview.js @@ -169,7 +169,7 @@ function ReportPreview(props) { const hasOnlyDistanceRequests = ReportUtils.hasOnlyDistanceRequestTransactions(props.iouReportID); const isScanning = hasReceipts && ReportUtils.areAllRequestsBeingSmartScanned(props.iouReportID, props.action); const hasErrors = - (hasReceipts && ReportUtils.hasMissingSmartscanFields(props.iouReportID)) || (canUseViolations && ReportUtils.reportHasViolations(props.iouReportID, props.transactionViolations)); + (hasReceipts && ReportUtils.hasMissingSmartscanFields(props.iouReportID)) || (canUseViolations && ReportUtils.hasViolations(props.iouReportID, props.transactionViolations)); const lastThreeTransactionsWithReceipts = transactionsWithReceipts.slice(-3); const lastThreeReceipts = _.map(lastThreeTransactionsWithReceipts, (transaction) => ReceiptUtils.getThumbnailAndImageURIs(transaction)); const hasNonReimbursableTransactions = ReportUtils.hasNonReimbursableTransactions(props.iouReportID); diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index 61e8845c95d0..a36a6cee813a 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -3514,7 +3514,7 @@ function transactionThreadHasViolations(report: Report, transactionViolations: T /** * Checks to see if a report contains a violation */ -function reportHasViolations(reportID: string, transactionViolations: TransactionViolations): boolean { +function hasViolations(reportID: string, transactionViolations: TransactionViolations): boolean { const transactions = TransactionUtils.getAllReportTransactions(reportID); return transactions.some((transaction) => transactionHasViolation(transaction.transactionID, transactionViolations)); } @@ -4557,7 +4557,7 @@ export { getPersonalDetailsForAccountID, getRoom, transactionThreadHasViolations, - reportHasViolations, + hasViolations, shouldDisableWelcomeMessage, navigateToPrivateNotes, canEditWriteCapability, From bf37bb04afeac9a3178b001b4938fac554858af8 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Tue, 19 Dec 2023 11:12:56 -0500 Subject: [PATCH 065/117] feat(Violations): rename --- src/libs/ReportUtils.ts | 6 +++--- src/libs/SidebarUtils.ts | 2 +- 2 files changed, 4 insertions(+), 4 deletions(-) diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index a36a6cee813a..8d1137531c85 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -3487,7 +3487,7 @@ function transactionHasViolation(transactionID: string, transactionViolations: T /** * Checks to see if a report's parentAction is a money request that contains a violation */ -function transactionThreadHasViolations(report: Report, transactionViolations: TransactionViolations, reportActions: OnyxCollection | null = {}): boolean { +function doesTransactionThreadHaveViolations(report: Report, transactionViolations: TransactionViolations, reportActions: OnyxCollection | null = {}): boolean { if (!reportActions) { return false; } @@ -3595,7 +3595,7 @@ function shouldReportBeInOptionList( } // Always show IOU reports with violations - if (isExpenseRequest(report) && betas.includes(CONST.BETAS.VIOLATIONS) && transactionThreadHasViolations(report, transactionViolations ?? {}, allReportActions)) { + if (isExpenseRequest(report) && betas.includes(CONST.BETAS.VIOLATIONS) && doesTransactionThreadHaveViolations(report, transactionViolations ?? {}, allReportActions)) { return true; } @@ -4556,7 +4556,7 @@ export { getReimbursementDeQueuedActionMessage, getPersonalDetailsForAccountID, getRoom, - transactionThreadHasViolations, + doesTransactionThreadHaveViolations, hasViolations, shouldDisableWelcomeMessage, navigateToPrivateNotes, diff --git a/src/libs/SidebarUtils.ts b/src/libs/SidebarUtils.ts index 69c50e566639..ffd45011759c 100644 --- a/src/libs/SidebarUtils.ts +++ b/src/libs/SidebarUtils.ts @@ -277,7 +277,7 @@ function getOptionData( const participantPersonalDetailList: PersonalDetails[] = Object.values(OptionsListUtils.getPersonalDetailsForAccountIDs(report.participantAccountIDs ?? [], personalDetails)); const personalDetail = participantPersonalDetailList[0] ?? {}; const hasErrors = Object.keys(result.allReportErrors ?? {}).length !== 0; - const hasViolations = canUseViolations && ReportUtils.transactionThreadHasViolations(report, transactionViolations, null); + const hasViolations = canUseViolations && ReportUtils.doesTransactionThreadHaveViolations(report, transactionViolations, null); result.isThread = ReportUtils.isChatThread(report); result.isChatRoom = ReportUtils.isChatRoom(report); From db5fa36b0d1944c97b26b7efc1d8a4904c8d4d44 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Tue, 19 Dec 2023 11:22:36 -0500 Subject: [PATCH 066/117] feat(Violations): move ReportUtils.transactionHasViolation to TransactionUtils.hasViolation --- src/libs/ReportUtils.ts | 72 +++++++++++++++--------------------- src/libs/TransactionUtils.ts | 14 ++++++- 2 files changed, 42 insertions(+), 44 deletions(-) diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index 8d1137531c85..487780e201b0 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -16,20 +16,7 @@ import CONST from '@src/CONST'; import {ParentNavigationSummaryParams, TranslationPaths} from '@src/languages/types'; import ONYXKEYS from '@src/ONYXKEYS'; import ROUTES from '@src/ROUTES'; -import { - Beta, - Login, - PersonalDetails, - PersonalDetailsList, - Policy, - PolicyTags, - Report, - ReportAction, - Session, - Transaction, - TransactionViolation, - TransactionViolations, -} from '@src/types/onyx'; +import {Beta, Login, PersonalDetails, PersonalDetailsList, Policy, PolicyTags, Report, ReportAction, Session, Transaction, TransactionViolations} from '@src/types/onyx'; import {Errors, Icon, PendingAction} from '@src/types/onyx/OnyxCommon'; import {IOUMessage, OriginalMessageActionName, OriginalMessageCreated} from '@src/types/onyx/OriginalMessage'; import {NotificationPreference} from '@src/types/onyx/Report'; @@ -50,6 +37,7 @@ import * as PolicyUtils from './PolicyUtils'; import * as ReportActionsUtils from './ReportActionsUtils'; import {LastVisibleMessage} from './ReportActionsUtils'; import * as TransactionUtils from './TransactionUtils'; +import {hasViolation} from './TransactionUtils'; import * as Url from './Url'; import * as UserUtils from './UserUtils'; @@ -528,8 +516,8 @@ function isTaskReport(report: OnyxEntry): boolean { * Checks if a task has been cancelled * When a task is deleted, the parentReportAction is updated to have a isDeletedParentAction deleted flag * This is because when you delete a task, we still allow you to chat on the report itself - * There's another situation where you don't have access to the parentReportAction (because it was created in a chat you don't have access to) - * In this case, we have added the key to the report itself + * There's another situation where you don't have access to the parentReportAction (because it was created in a chat + * you don't have access to) In this case, we have added the key to the report itself */ function isCanceledTaskReport(report: OnyxEntry | EmptyObject = {}, parentReportAction: OnyxEntry | EmptyObject = {}): boolean { if (isNotEmptyObject(parentReportAction) && (parentReportAction?.message?.[0]?.isDeletedParentAction ?? false)) { @@ -1410,8 +1398,8 @@ function getIcons( } /** - * Gets the personal details for a login by looking in the ONYXKEYS.PERSONAL_DETAILS_LIST Onyx key (stored in the local variable, allPersonalDetails). If it doesn't exist in Onyx, - * then a default object is constructed. + * Gets the personal details for a login by looking in the ONYXKEYS.PERSONAL_DETAILS_LIST Onyx key (stored in the local + * variable, allPersonalDetails). If it doesn't exist in Onyx, then a default object is constructed. */ function getPersonalDetailsForAccountID(accountID: number): Partial { if (!accountID) { @@ -1561,8 +1549,8 @@ function getReimbursementDeQueuedActionMessage(report: OnyxEntry): strin * Returns the last visible message for a given report after considering the given optimistic actions * * @param reportID - the report for which last visible message has to be fetched - * @param [actionsToMerge] - the optimistic merge actions that needs to be considered while fetching last visible message - + * @param [actionsToMerge] - the optimistic merge actions that needs to be considered while fetching last visible + * message */ function getLastVisibleMessage(reportID: string | undefined, actionsToMerge: ReportActions = {}): LastVisibleMessage { const report = getReport(reportID); @@ -2422,8 +2410,9 @@ function goBackToDetailsPage(report: OnyxEntry) { /** * Generate a random reportID up to 53 bits aka 9,007,199,254,740,991 (Number.MAX_SAFE_INTEGER). - * There were approximately 98,000,000 reports with sequential IDs generated before we started using this approach, those make up roughly one billionth of the space for these numbers, - * so we live with the 1 in a billion chance of a collision with an older ID until we can switch to 64-bit IDs. + * There were approximately 98,000,000 reports with sequential IDs generated before we started using this approach, + * those make up roughly one billionth of the space for these numbers, so we live with the 1 in a billion chance of a + * collision with an older ID until we can switch to 64-bit IDs. * * In a test of 500M reports (28 years of reports at our current max rate) we got 20-40 collisions meaning that * this is more than random enough for our needs. @@ -2437,8 +2426,9 @@ function hasReportNameError(report: OnyxEntry): boolean { } /** - * For comments shorter than or equal to 10k chars, convert the comment from MD into HTML because that's how it is stored in the database - * For longer comments, skip parsing, but still escape the text, and display plaintext for performance reasons. It takes over 40s to parse a 100k long string!! + * For comments shorter than or equal to 10k chars, convert the comment from MD into HTML because that's how it is + * stored in the database For longer comments, skip parsing, but still escape the text, and display plaintext for + * performance reasons. It takes over 40s to parse a 100k long string!! */ function getParsedComment(text: string): string { const parser = new ExpensiMark(); @@ -2742,11 +2732,13 @@ function getIOUReportActionMessage(iouReportID: string, type: string, total: num * @param participants - An array with participants details. * @param [transactionID] - Not required if the IOUReportAction type is 'pay' * @param [paymentType] - Only required if the IOUReportAction type is 'pay'. Can be oneOf(elsewhere, Expensify). - * @param [iouReportID] - Only required if the IOUReportActions type is oneOf(decline, cancel, pay). Generates a randomID as default. + * @param [iouReportID] - Only required if the IOUReportActions type is oneOf(decline, cancel, pay). Generates a + * randomID as default. * @param [isSettlingUp] - Whether we are settling up an IOU. * @param [isSendMoneyFlow] - Whether this is send money flow * @param [receipt] - * @param [isOwnPolicyExpenseChat] - Whether this is an expense report create from the current user's policy expense chat + * @param [isOwnPolicyExpenseChat] - Whether this is an expense report create from the current user's policy expense + * chat */ function buildOptimisticIOUReportAction( @@ -3467,7 +3459,8 @@ function canAccessReport(report: OnyxEntry, policies: OnyxCollection, currentReportId: string): boolean { const currentReport = getReport(currentReportId); @@ -3477,13 +3470,6 @@ function shouldHideReport(report: OnyxEntry, currentReportId: string): b return parentReport?.reportID !== report?.reportID && !isChildReportHasComment; } -/** - * Checks if any violations for the provided transaction are of type 'violation' - */ -function transactionHasViolation(transactionID: string, transactionViolations: TransactionViolations): boolean { - return transactionViolations[transactionID]?.some((violation: TransactionViolation) => violation.type === 'violation'); -} - /** * Checks to see if a report's parentAction is a money request that contains a violation */ @@ -3508,7 +3494,7 @@ function doesTransactionThreadHaveViolations(report: Report, transactionViolatio if (report?.stateNum !== CONST.REPORT.STATE_NUM.OPEN && report?.stateNum !== CONST.REPORT.STATE_NUM.PROCESSING) { return false; } - return transactionHasViolation(IOUTransactionID, transactionViolations); + return hasViolation(IOUTransactionID, transactionViolations); } /** @@ -3516,15 +3502,15 @@ function doesTransactionThreadHaveViolations(report: Report, transactionViolatio */ function hasViolations(reportID: string, transactionViolations: TransactionViolations): boolean { const transactions = TransactionUtils.getAllReportTransactions(reportID); - return transactions.some((transaction) => transactionHasViolation(transaction.transactionID, transactionViolations)); + return transactions.some((transaction) => hasViolation(transaction.transactionID, transactionViolations)); } /** - * Takes several pieces of data from Onyx and evaluates if a report should be shown in the option list (either when searching - * for reports or the reports shown in the LHN). + * Takes several pieces of data from Onyx and evaluates if a report should be shown in the option list (either when + * searching for reports or the reports shown in the LHN). * - * This logic is very specific and the order of the logic is very important. It should fail quickly in most cases and also - * filter out the majority of reports before filtering out very specific minority of reports. + * This logic is very specific and the order of the logic is very important. It should fail quickly in most cases and + * also filter out the majority of reports before filtering out very specific minority of reports. */ function shouldReportBeInOptionList( report: OnyxEntry, @@ -3828,9 +3814,11 @@ function hasIOUWaitingOnCurrentUserBankAccount(chatReport: OnyxEntry): b /** * Users can request money: - * - in policy expense chats only if they are in a role of a member in the chat (in other words, if it's their policy expense chat) + * - in policy expense chats only if they are in a role of a member in the chat (in other words, if it's their policy + * expense chat) * - in an open or submitted expense report tied to a policy expense chat the user owns - * - employee can request money in submitted expense report only if the policy has Instant Submit settings turned on + * - employee can request money in submitted expense report only if the policy has Instant Submit settings turned + * on * - in an IOU report, which is not settled yet * - in a 1:1 DM chat */ diff --git a/src/libs/TransactionUtils.ts b/src/libs/TransactionUtils.ts index 232c30658838..9fea1fd98603 100644 --- a/src/libs/TransactionUtils.ts +++ b/src/libs/TransactionUtils.ts @@ -3,7 +3,7 @@ import Onyx, {OnyxCollection, OnyxEntry} from 'react-native-onyx'; import {ValueOf} from 'type-fest'; import CONST from '@src/CONST'; import ONYXKEYS from '@src/ONYXKEYS'; -import {RecentWaypoint, ReportAction, Transaction} from '@src/types/onyx'; +import {RecentWaypoint, ReportAction, Transaction, TransactionViolation, TransactionViolations} from '@src/types/onyx'; import {Comment, Receipt, Waypoint, WaypointCollection} from '@src/types/onyx/Transaction'; import {EmptyObject} from '@src/types/utils/EmptyObject'; import {isCorporateCard, isExpensifyCard} from './CardUtils'; @@ -71,7 +71,9 @@ function isManualRequest(transaction: Transaction): boolean { * Optimistically generate a transaction. * * @param amount – in cents - * @param [existingTransactionID] When creating a distance request, an empty transaction has already been created with a transactionID. In that case, the transaction here needs to have it's transactionID match what was already generated. + * @param [existingTransactionID] When creating a distance request, an empty transaction has already been created with + * a transactionID. In that case, the transaction here needs to have it's transactionID match what was already + * generated. */ function buildOptimisticTransaction( amount: number, @@ -513,6 +515,13 @@ function getRecentTransactions(transactions: Record, size = 2): .slice(0, size); } +/** + * Checks if any violations for the provided transaction are of type 'violation' + */ +function hasViolation(transactionID: string, transactionViolations: TransactionViolations): boolean { + return transactionViolations[transactionID]?.some((violation: TransactionViolation) => violation.type === 'violation'); +} + export { buildOptimisticTransaction, getUpdatedTransaction, @@ -555,4 +564,5 @@ export { getWaypointIndex, waypointHasValidAddress, getRecentTransactions, + hasViolation, }; From ba5e94041b1f1230ed7665c7c762b841248fd00e Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Tue, 19 Dec 2023 11:22:45 -0500 Subject: [PATCH 067/117] feat(Violations): restore selector --- src/pages/home/sidebar/SidebarLinksData.js | 16 ++++++++++++++++ 1 file changed, 16 insertions(+) diff --git a/src/pages/home/sidebar/SidebarLinksData.js b/src/pages/home/sidebar/SidebarLinksData.js index ac43f29a852f..f51700e3c0cd 100644 --- a/src/pages/home/sidebar/SidebarLinksData.js +++ b/src/pages/home/sidebar/SidebarLinksData.js @@ -185,6 +185,21 @@ const chatReportSelector = (report) => parentReportID: report.parentReportID, }; +/** + * @param {Object} [reportActions] + * @returns {Object|undefined} + */ +const reportActionsSelector = (reportActions) => + reportActions && + _.map(reportActions, (reportAction) => ({ + errors: _.get(reportAction, 'errors', []), + message: [ + { + moderationDecision: {decision: _.get(reportAction, 'message[0].moderationDecision.decision')}, + }, + ], + })); + /** * @param {Object} [policy] * @returns {Object|undefined} @@ -219,6 +234,7 @@ export default compose( }, allReportActions: { key: ONYXKEYS.COLLECTION.REPORT_ACTIONS, + selector: reportActionsSelector, initialValue: {}, }, policies: { From e873fa4f046927ab67d6d04645f2e83201de9847 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Tue, 19 Dec 2023 11:27:52 -0500 Subject: [PATCH 068/117] feat(Violations): remove null guard --- src/libs/ReportUtils.ts | 9 +++------ 1 file changed, 3 insertions(+), 6 deletions(-) diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index 487780e201b0..db0502ff6582 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -3473,11 +3473,8 @@ function shouldHideReport(report: OnyxEntry, currentReportId: string): b /** * Checks to see if a report's parentAction is a money request that contains a violation */ -function doesTransactionThreadHaveViolations(report: Report, transactionViolations: TransactionViolations, reportActions: OnyxCollection | null = {}): boolean { - if (!reportActions) { - return false; - } - const resolvedParentReportAction = reportActions[`${report.parentReportID}`]?.[`${report.parentReportActionID}`]; +function doesTransactionThreadHaveViolations(report: Report, transactionViolations: TransactionViolations, reportActions: OnyxCollection): boolean { + const resolvedParentReportAction = reportActions?.[`${report.parentReportID}`]?.[`${report.parentReportActionID}`]; if (!resolvedParentReportAction) { return false; } @@ -3581,7 +3578,7 @@ function shouldReportBeInOptionList( } // Always show IOU reports with violations - if (isExpenseRequest(report) && betas.includes(CONST.BETAS.VIOLATIONS) && doesTransactionThreadHaveViolations(report, transactionViolations ?? {}, allReportActions)) { + if (isExpenseRequest(report) && betas.includes(CONST.BETAS.VIOLATIONS) && doesTransactionThreadHaveViolations(report, transactionViolations ?? {}, allReportActions ?? {})) { return true; } From 68f933ad8e6cfe96b44932b6268a71dba7e4ccea Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Tue, 19 Dec 2023 13:43:14 -0500 Subject: [PATCH 069/117] feat(Violations): revert line width changes --- src/libs/ReportUtils.ts | 41 +++++++++++++++++------------------------ 1 file changed, 17 insertions(+), 24 deletions(-) diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index db0502ff6582..0ebc27ce0fc6 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -1398,8 +1398,8 @@ function getIcons( } /** - * Gets the personal details for a login by looking in the ONYXKEYS.PERSONAL_DETAILS_LIST Onyx key (stored in the local - * variable, allPersonalDetails). If it doesn't exist in Onyx, then a default object is constructed. + * Gets the personal details for a login by looking in the ONYXKEYS.PERSONAL_DETAILS_LIST Onyx key (stored in the local variable, allPersonalDetails). If it doesn't exist in Onyx, + * then a default object is constructed. */ function getPersonalDetailsForAccountID(accountID: number): Partial { if (!accountID) { @@ -1549,8 +1549,8 @@ function getReimbursementDeQueuedActionMessage(report: OnyxEntry): strin * Returns the last visible message for a given report after considering the given optimistic actions * * @param reportID - the report for which last visible message has to be fetched - * @param [actionsToMerge] - the optimistic merge actions that needs to be considered while fetching last visible - * message + * @param [actionsToMerge] - the optimistic merge actions that needs to be considered while fetching last visible message + */ function getLastVisibleMessage(reportID: string | undefined, actionsToMerge: ReportActions = {}): LastVisibleMessage { const report = getReport(reportID); @@ -2410,9 +2410,8 @@ function goBackToDetailsPage(report: OnyxEntry) { /** * Generate a random reportID up to 53 bits aka 9,007,199,254,740,991 (Number.MAX_SAFE_INTEGER). - * There were approximately 98,000,000 reports with sequential IDs generated before we started using this approach, - * those make up roughly one billionth of the space for these numbers, so we live with the 1 in a billion chance of a - * collision with an older ID until we can switch to 64-bit IDs. + * There were approximately 98,000,000 reports with sequential IDs generated before we started using this approach, those make up roughly one billionth of the space for these numbers, + * so we live with the 1 in a billion chance of a collision with an older ID until we can switch to 64-bit IDs. * * In a test of 500M reports (28 years of reports at our current max rate) we got 20-40 collisions meaning that * this is more than random enough for our needs. @@ -2426,9 +2425,8 @@ function hasReportNameError(report: OnyxEntry): boolean { } /** - * For comments shorter than or equal to 10k chars, convert the comment from MD into HTML because that's how it is - * stored in the database For longer comments, skip parsing, but still escape the text, and display plaintext for - * performance reasons. It takes over 40s to parse a 100k long string!! + * For comments shorter than or equal to 10k chars, convert the comment from MD into HTML because that's how it is stored in the database + * For longer comments, skip parsing, but still escape the text, and display plaintext for performance reasons. It takes over 40s to parse a 100k long string!! */ function getParsedComment(text: string): string { const parser = new ExpensiMark(); @@ -2732,13 +2730,11 @@ function getIOUReportActionMessage(iouReportID: string, type: string, total: num * @param participants - An array with participants details. * @param [transactionID] - Not required if the IOUReportAction type is 'pay' * @param [paymentType] - Only required if the IOUReportAction type is 'pay'. Can be oneOf(elsewhere, Expensify). - * @param [iouReportID] - Only required if the IOUReportActions type is oneOf(decline, cancel, pay). Generates a - * randomID as default. + * @param [iouReportID] - Only required if the IOUReportActions type is oneOf(decline, cancel, pay). Generates a randomID as default. * @param [isSettlingUp] - Whether we are settling up an IOU. * @param [isSendMoneyFlow] - Whether this is send money flow * @param [receipt] - * @param [isOwnPolicyExpenseChat] - Whether this is an expense report create from the current user's policy expense - * chat + * @param [isOwnPolicyExpenseChat] - Whether this is an expense report create from the current user's policy expense chat */ function buildOptimisticIOUReportAction( @@ -3459,8 +3455,7 @@ function canAccessReport(report: OnyxEntry, policies: OnyxCollection, currentReportId: string): boolean { const currentReport = getReport(currentReportId); @@ -3503,11 +3498,11 @@ function hasViolations(reportID: string, transactionViolations: TransactionViola } /** - * Takes several pieces of data from Onyx and evaluates if a report should be shown in the option list (either when - * searching for reports or the reports shown in the LHN). + * Takes several pieces of data from Onyx and evaluates if a report should be shown in the option list (either when searching + * for reports or the reports shown in the LHN). * - * This logic is very specific and the order of the logic is very important. It should fail quickly in most cases and - * also filter out the majority of reports before filtering out very specific minority of reports. + * This logic is very specific and the order of the logic is very important. It should fail quickly in most cases and also + * filter out the majority of reports before filtering out very specific minority of reports. */ function shouldReportBeInOptionList( report: OnyxEntry, @@ -3811,11 +3806,9 @@ function hasIOUWaitingOnCurrentUserBankAccount(chatReport: OnyxEntry): b /** * Users can request money: - * - in policy expense chats only if they are in a role of a member in the chat (in other words, if it's their policy - * expense chat) + * - in policy expense chats only if they are in a role of a member in the chat (in other words, if it's their policy expense chat) * - in an open or submitted expense report tied to a policy expense chat the user owns - * - employee can request money in submitted expense report only if the policy has Instant Submit settings turned - * on + * - employee can request money in submitted expense report only if the policy has Instant Submit settings turned on * - in an IOU report, which is not settled yet * - in a 1:1 DM chat */ From b41a3de127aef6cb29041f42a1b44f61f12a8595 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Tue, 19 Dec 2023 13:53:48 -0500 Subject: [PATCH 070/117] feat(Violations): change signature of shouldReportBeInOptionList --- src/libs/OptionsListUtils.js | 2 +- src/libs/ReportUtils.ts | 2 +- src/libs/SidebarUtils.ts | 2 +- tests/perf-test/ReportUtils.perf-test.ts | 2 +- 4 files changed, 4 insertions(+), 4 deletions(-) diff --git a/src/libs/OptionsListUtils.js b/src/libs/OptionsListUtils.js index 536518dcad7b..263636e5a0ca 100644 --- a/src/libs/OptionsListUtils.js +++ b/src/libs/OptionsListUtils.js @@ -1138,7 +1138,7 @@ function getOptions( // Filter out all the reports that shouldn't be displayed const filteredReports = _.filter(reports, (report) => - ReportUtils.shouldReportBeInOptionList(report, Navigation.getTopmostReportId(), false, betas, policies, reportActions, false, transactionViolations), + ReportUtils.shouldReportBeInOptionList(report, Navigation.getTopmostReportId(), false, betas, policies, false, reportActions, transactionViolations), ); // Sorting the reports works like this: diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index 0ebc27ce0fc6..9ed3efaf4d57 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -3510,8 +3510,8 @@ function shouldReportBeInOptionList( isInGSDMode: boolean, betas: Beta[], policies: OnyxCollection, - allReportActions?: OnyxCollection, excludeEmptyChats = false, + allReportActions?: OnyxCollection, transactionViolations?: TransactionViolations, ) { const isInDefaultMode = !isInGSDMode; diff --git a/src/libs/SidebarUtils.ts b/src/libs/SidebarUtils.ts index ffd45011759c..363e3dc82ba2 100644 --- a/src/libs/SidebarUtils.ts +++ b/src/libs/SidebarUtils.ts @@ -147,7 +147,7 @@ function getOrderedReportIDs( const allReportsDictValues = Object.values(allReports); // Filter out all the reports that shouldn't be displayed const reportsToDisplay = allReportsDictValues.filter((report) => - ReportUtils.shouldReportBeInOptionList(report, currentReportId ?? '', isInGSDMode, betas, policies, allReportActions, true, transactionViolations), + ReportUtils.shouldReportBeInOptionList(report, currentReportId ?? '', isInGSDMode, betas, policies, true, allReportActions, transactionViolations), ); if (reportsToDisplay.length === 0) { diff --git a/tests/perf-test/ReportUtils.perf-test.ts b/tests/perf-test/ReportUtils.perf-test.ts index 03c79095b6a7..92cf83372331 100644 --- a/tests/perf-test/ReportUtils.perf-test.ts +++ b/tests/perf-test/ReportUtils.perf-test.ts @@ -183,7 +183,7 @@ test('[ReportUtils] shouldReportBeInOptionList on 1k participant', async () => { const policies = getMockedPolicies(); await waitForBatchedUpdates(); - await measureFunction(() => ReportUtils.shouldReportBeInOptionList(report, currentReportId, isInGSDMode, betas, policies, {}), {runs}); + await measureFunction(() => ReportUtils.shouldReportBeInOptionList(report, currentReportId, isInGSDMode, betas, policies, false, {}), {runs}); }); test('[ReportUtils] getWorkspaceIcon on 5k policies', async () => { From 2492bf9368be364b02f814b0bf41651522b71aa8 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Tue, 19 Dec 2023 14:01:58 -0500 Subject: [PATCH 071/117] feat(Violations): rename resolvedParentReportAction --- src/libs/ReportUtils.ts | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index 9ed3efaf4d57..fa64be4ed67f 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -3469,14 +3469,14 @@ function shouldHideReport(report: OnyxEntry, currentReportId: string): b * Checks to see if a report's parentAction is a money request that contains a violation */ function doesTransactionThreadHaveViolations(report: Report, transactionViolations: TransactionViolations, reportActions: OnyxCollection): boolean { - const resolvedParentReportAction = reportActions?.[`${report.parentReportID}`]?.[`${report.parentReportActionID}`]; - if (!resolvedParentReportAction) { + const parentReportAction = reportActions?.[`${report.parentReportID}`]?.[`${report.parentReportActionID}`]; + if (!parentReportAction) { return false; } - if (resolvedParentReportAction.actionName !== CONST.REPORT.ACTIONS.TYPE.IOU) { + if (parentReportAction.actionName !== CONST.REPORT.ACTIONS.TYPE.IOU) { return false; } - const {IOUTransactionID, IOUReportID} = resolvedParentReportAction?.originalMessage; + const {IOUTransactionID, IOUReportID} = parentReportAction?.originalMessage; if (!IOUTransactionID || !IOUReportID) { return false; } From ea6bf2ebf526469f81dbb6a2c845186b3572017f Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Thu, 21 Dec 2023 13:55:56 -0500 Subject: [PATCH 072/117] feat(Violations): add whitespace --- src/components/LHNOptionsList/LHNOptionsList.js | 1 + 1 file changed, 1 insertion(+) diff --git a/src/components/LHNOptionsList/LHNOptionsList.js b/src/components/LHNOptionsList/LHNOptionsList.js index 1dfb70f65847..52ca3a9d06bc 100644 --- a/src/components/LHNOptionsList/LHNOptionsList.js +++ b/src/components/LHNOptionsList/LHNOptionsList.js @@ -71,6 +71,7 @@ const propTypes = { /** The list of transaction violations */ transactionViolations: transactionViolationsPropType, + ...withCurrentReportIDPropTypes, }; From 4d4c476b7660395b9feae1eca8c9d456a1b30512 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Thu, 21 Dec 2023 13:59:16 -0500 Subject: [PATCH 073/117] feat(Violations): use imported propType --- .../ReportActionItem/ReportPreview.js | 25 ++----------------- 1 file changed, 2 insertions(+), 23 deletions(-) diff --git a/src/components/ReportActionItem/ReportPreview.js b/src/components/ReportActionItem/ReportPreview.js index ead4244e14ec..53d88857c90c 100644 --- a/src/components/ReportActionItem/ReportPreview.js +++ b/src/components/ReportActionItem/ReportPreview.js @@ -26,6 +26,7 @@ import * as ReceiptUtils from '@libs/ReceiptUtils'; import * as ReportActionUtils from '@libs/ReportActionsUtils'; import * as ReportUtils from '@libs/ReportUtils'; import * as TransactionUtils from '@libs/TransactionUtils'; +import {transactionViolationsPropType} from '@libs/Violations/propTypes'; import reportActionPropTypes from '@pages/home/report/reportActionPropTypes'; import reportPropTypes from '@pages/reportPropTypes'; import * as IOU from '@userActions/IOU'; @@ -102,29 +103,7 @@ const propTypes = { isWhisper: PropTypes.bool, /** All of the transaction violations */ - transactionViolations: PropTypes.shape({ - violations: PropTypes.arrayOf( - PropTypes.shape({ - /** The transaction ID */ - transactionID: PropTypes.number, - - /** The transaction violation type */ - type: PropTypes.string, - - /** The transaction violation message */ - message: PropTypes.string, - - /** The transaction violation data */ - data: PropTypes.shape({ - /** The transaction violation data field */ - field: PropTypes.string, - - /** The transaction violation data value */ - value: PropTypes.string, - }), - }), - ), - }), + transactionViolations: transactionViolationsPropType, ...withLocalizePropTypes, }; From 04ed5951f5fc4ed436f40acf05c89d9ccb8badf5 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Fri, 22 Dec 2023 12:03:06 -0500 Subject: [PATCH 074/117] feat(Violations): revert commment wrapping --- src/libs/ReportUtils.ts | 10 ++++------ 1 file changed, 4 insertions(+), 6 deletions(-) diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index 1d9002823829..8cde45cfd4ea 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -496,8 +496,8 @@ function isTaskReport(report: OnyxEntry): boolean { * Checks if a task has been cancelled * When a task is deleted, the parentReportAction is updated to have a isDeletedParentAction deleted flag * This is because when you delete a task, we still allow you to chat on the report itself - * There's another situation where you don't have access to the parentReportAction (because it was created in a chat - * you don't have access to) In this case, we have added the key to the report itself + * There's another situation where you don't have access to the parentReportAction (because it was created in a chat you don't have access to) + * In this case, we have added the key to the report itself */ function isCanceledTaskReport(report: OnyxEntry | EmptyObject = {}, parentReportAction: OnyxEntry | EmptyObject = {}): boolean { if (isNotEmptyObject(parentReportAction) && (parentReportAction?.message?.[0]?.isDeletedParentAction ?? false)) { @@ -3442,8 +3442,7 @@ function shouldReportBeInOptionList( } /** - * Attempts to find a report in onyx with the provided list of participants. Does not include threads, task, money - * request, room, and policy expense chat. + * Attempts to find a report in onyx with the provided list of participants. Does not include threads, task, money request, room, and policy expense chat. */ function getChatByParticipants(newParticipantList: number[]): OnyxEntry { const sortedNewParticipantList = newParticipantList.sort(); @@ -3546,8 +3545,7 @@ function shouldShowFlagComment(reportAction: OnyxEntry, report: On } /** - * @param sortedAndFilteredReportActions - reportActions for the report, sorted newest to oldest, and filtered for only - * those that should be visible + * @param sortedAndFilteredReportActions - reportActions for the report, sorted newest to oldest, and filtered for only those that should be visible */ function getNewMarkerReportActionID(report: OnyxEntry, sortedAndFilteredReportActions: ReportAction[]): string { if (!isUnread(report)) { From aa1425802658a9ac11fcc908253f163bb42f89e4 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Fri, 22 Dec 2023 12:05:27 -0500 Subject: [PATCH 075/117] feat(Violations): cast return value to type --- src/libs/TransactionUtils.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/libs/TransactionUtils.ts b/src/libs/TransactionUtils.ts index 42c086f44e85..0182ec777cb2 100644 --- a/src/libs/TransactionUtils.ts +++ b/src/libs/TransactionUtils.ts @@ -514,7 +514,7 @@ function getRecentTransactions(transactions: Record, size = 2): * Checks if any violations for the provided transaction are of type 'violation' */ function hasViolation(transactionID: string, transactionViolations: TransactionViolations): boolean { - return transactionViolations[transactionID]?.some((violation: TransactionViolation) => violation.type === 'violation'); + return Boolean(transactionViolations[transactionID]?.some((violation: TransactionViolation) => violation.type === 'violation')); } export { From 0214a1252e61e563c423feabdaea48d77e529267 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Fri, 22 Dec 2023 12:43:16 -0500 Subject: [PATCH 076/117] feat(Violations): use correct key --- src/libs/TransactionUtils.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/libs/TransactionUtils.ts b/src/libs/TransactionUtils.ts index 0182ec777cb2..29d2c133d630 100644 --- a/src/libs/TransactionUtils.ts +++ b/src/libs/TransactionUtils.ts @@ -514,7 +514,7 @@ function getRecentTransactions(transactions: Record, size = 2): * Checks if any violations for the provided transaction are of type 'violation' */ function hasViolation(transactionID: string, transactionViolations: TransactionViolations): boolean { - return Boolean(transactionViolations[transactionID]?.some((violation: TransactionViolation) => violation.type === 'violation')); + return Boolean(transactionViolations[ONYXKEYS.COLLECTION.TRANSACTION_VIOLATIONS + transactionID]?.some((violation: TransactionViolation) => violation.type === 'violation')); } export { From cd4e987c5f0ecd84c1796ada9ae48a236c8408f2 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Tue, 2 Jan 2024 14:34:17 -0500 Subject: [PATCH 077/117] feat(Violations): fix comment wrapping --- src/CONST.ts | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/CONST.ts b/src/CONST.ts index b36bb47cbcf9..0dd29935947f 100755 --- a/src/CONST.ts +++ b/src/CONST.ts @@ -3044,8 +3044,8 @@ const CONST = { }, /** - * Constants for maxToRenderPerBatch parameter that is used for FlatList or SectionList. This controls the amount - * of items rendered per batch, which is the next chunk of items rendered on every scroll. + * Constants for maxToRenderPerBatch parameter that is used for FlatList or SectionList. This controls the amount of items rendered per batch, which is the next chunk of items + * rendered on every scroll. */ MAX_TO_RENDER_PER_BATCH: { DEFAULT: 5, From 93d08a0501de93f28c8b6cdf4e96412574e1e2bc Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Wed, 3 Jan 2024 12:22:32 -0500 Subject: [PATCH 078/117] feat(Violations): fix proptypes of allReportActions --- src/pages/home/sidebar/SidebarLinksData.js | 5 ++--- 1 file changed, 2 insertions(+), 3 deletions(-) diff --git a/src/pages/home/sidebar/SidebarLinksData.js b/src/pages/home/sidebar/SidebarLinksData.js index f51700e3c0cd..90cd587a3be1 100644 --- a/src/pages/home/sidebar/SidebarLinksData.js +++ b/src/pages/home/sidebar/SidebarLinksData.js @@ -28,7 +28,7 @@ const propTypes = { /** All report actions for all reports */ /** Object of report actions for this report */ - allReportActions: PropTypes.objectOf(PropTypes.shape(reportActionPropTypes)), + allReportActions: PropTypes.objectOf(PropTypes.arrayOf(PropTypes.shape(reportActionPropTypes))), /** Whether the reports are loading. When false it means they are ready to be used. */ isLoadingApp: PropTypes.bool, @@ -190,8 +190,7 @@ const chatReportSelector = (report) => * @returns {Object|undefined} */ const reportActionsSelector = (reportActions) => - reportActions && - _.map(reportActions, (reportAction) => ({ + _.map(reportActions || [], (reportAction) => ({ errors: _.get(reportAction, 'errors', []), message: [ { From b3b367dd70fd94239885b1fcf0b3bb924e8d179e Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Wed, 3 Jan 2024 12:30:16 -0500 Subject: [PATCH 079/117] feat(Violations): revert change to reportActionsSelector --- src/pages/home/sidebar/SidebarLinksData.js | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/src/pages/home/sidebar/SidebarLinksData.js b/src/pages/home/sidebar/SidebarLinksData.js index 90cd587a3be1..32d6f799a3aa 100644 --- a/src/pages/home/sidebar/SidebarLinksData.js +++ b/src/pages/home/sidebar/SidebarLinksData.js @@ -190,7 +190,8 @@ const chatReportSelector = (report) => * @returns {Object|undefined} */ const reportActionsSelector = (reportActions) => - _.map(reportActions || [], (reportAction) => ({ + reportActions && + _.map(reportActions, (reportAction) => ({ errors: _.get(reportAction, 'errors', []), message: [ { From 00ce2cf82254a1e773efa1fb38e0fbd6172c0662 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Wed, 3 Jan 2024 12:38:36 -0500 Subject: [PATCH 080/117] feat(Violations): restore proptypes --- src/components/ReportActionItem/ReportPreview.js | 2 +- src/pages/home/sidebar/SidebarLinksData.js | 16 ++++++++++++++-- 2 files changed, 15 insertions(+), 3 deletions(-) diff --git a/src/components/ReportActionItem/ReportPreview.js b/src/components/ReportActionItem/ReportPreview.js index 5daa20572a22..36d76078f2b4 100644 --- a/src/components/ReportActionItem/ReportPreview.js +++ b/src/components/ReportActionItem/ReportPreview.js @@ -163,7 +163,7 @@ function ReportPreview(props) { const numberOfScanningReceipts = _.filter(transactionsWithReceipts, (transaction) => TransactionUtils.isReceiptBeingScanned(transaction)).length; const hasReceipts = transactionsWithReceipts.length > 0; const isScanning = hasReceipts && areAllRequestsBeingSmartScanned; - const hasErrors = hasReceipts && hasMissingSmartscanFields || (canUseViolations && ReportUtils.hasViolations(props.iouReportID, props.transactionViolations)); + const hasErrors = (hasReceipts && hasMissingSmartscanFields) || (canUseViolations && ReportUtils.hasViolations(props.iouReportID, props.transactionViolations)); const lastThreeTransactionsWithReceipts = transactionsWithReceipts.slice(-3); const lastThreeReceipts = _.map(lastThreeTransactionsWithReceipts, (transaction) => ReceiptUtils.getThumbnailAndImageURIs(transaction)); let formattedMerchant = numberOfRequests === 1 && hasReceipts ? TransactionUtils.getMerchant(transactionsWithReceipts[0]) : null; diff --git a/src/pages/home/sidebar/SidebarLinksData.js b/src/pages/home/sidebar/SidebarLinksData.js index 32d6f799a3aa..57239799f91b 100644 --- a/src/pages/home/sidebar/SidebarLinksData.js +++ b/src/pages/home/sidebar/SidebarLinksData.js @@ -12,7 +12,6 @@ import useLocalize from '@hooks/useLocalize'; import useThemeStyles from '@hooks/useThemeStyles'; import compose from '@libs/compose'; import SidebarUtils from '@libs/SidebarUtils'; -import reportActionPropTypes from '@pages/home/report/reportActionPropTypes'; import reportPropTypes from '@pages/reportPropTypes'; import CONST from '@src/CONST'; import ONYXKEYS from '@src/ONYXKEYS'; @@ -28,7 +27,20 @@ const propTypes = { /** All report actions for all reports */ /** Object of report actions for this report */ - allReportActions: PropTypes.objectOf(PropTypes.arrayOf(PropTypes.shape(reportActionPropTypes))), + allReportActions: PropTypes.objectOf( + PropTypes.arrayOf( + PropTypes.shape({ + error: PropTypes.string, + message: PropTypes.arrayOf( + PropTypes.shape({ + moderationDecision: PropTypes.shape({ + decision: PropTypes.string, + }), + }), + ), + }), + ), + ), /** Whether the reports are loading. When false it means they are ready to be used. */ isLoadingApp: PropTypes.bool, From a5ecebe6b8f76ee22d8c1a4ba62a072aa652b55a Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Wed, 3 Jan 2024 13:13:23 -0500 Subject: [PATCH 081/117] feat(Violations): unwrap comment --- src/libs/TransactionUtils.ts | 5 ++--- 1 file changed, 2 insertions(+), 3 deletions(-) diff --git a/src/libs/TransactionUtils.ts b/src/libs/TransactionUtils.ts index 18b0c2c36ccb..5cb962b27cdc 100644 --- a/src/libs/TransactionUtils.ts +++ b/src/libs/TransactionUtils.ts @@ -79,9 +79,8 @@ function isManualRequest(transaction: Transaction): boolean { * Optimistically generate a transaction. * * @param amount – in cents - * @param [existingTransactionID] When creating a distance request, an empty transaction has already been created with - * a transactionID. In that case, the transaction here needs to have it's transactionID match what was already - * generated. + * @param [existingTransactionID] When creating a distance request, an empty transaction has already been created with a transactionID. In that case, the transaction here needs to have + * it's transactionID match what was already generated. */ function buildOptimisticTransaction( amount: number, From 93a300b0d3cffb39a73c522f03522a04f0690672 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Thu, 4 Jan 2024 11:39:47 -0500 Subject: [PATCH 082/117] fix types broken by merge --- src/types/onyx/TransactionViolation.ts | 1 + src/types/onyx/index.ts | 6 ++---- tests/perf-test/SidebarUtils.perf-test.ts | 3 ++- 3 files changed, 5 insertions(+), 5 deletions(-) diff --git a/src/types/onyx/TransactionViolation.ts b/src/types/onyx/TransactionViolation.ts index 1d02de12a52a..73714e78fa27 100644 --- a/src/types/onyx/TransactionViolation.ts +++ b/src/types/onyx/TransactionViolation.ts @@ -1,3 +1,4 @@ +// eslint-disable-next-line @typescript-eslint/consistent-type-imports import CONST from '@src/CONST'; /** diff --git a/src/types/onyx/index.ts b/src/types/onyx/index.ts index 3404885a3c1c..9fd52bd0628e 100644 --- a/src/types/onyx/index.ts +++ b/src/types/onyx/index.ts @@ -27,13 +27,11 @@ import type {PersonalDetailsList} from './PersonalDetails'; import type PersonalDetails from './PersonalDetails'; import type PlaidData from './PlaidData'; import type Policy from './Policy'; -import type {PolicyCategories} from './PolicyCategory'; -import type PolicyCategory from './PolicyCategory'; +import type {PolicyCategories, PolicyCategory} from './PolicyCategory'; import type {PolicyMembers} from './PolicyMember'; import type PolicyMember from './PolicyMember'; import type PolicyReportField from './PolicyReportField'; -import type {PolicyTags} from './PolicyTag'; -import type PolicyTag from './PolicyTag'; +import type {PolicyTag, PolicyTags} from './PolicyTag'; import type PrivatePersonalDetails from './PrivatePersonalDetails'; import type RecentlyUsedCategories from './RecentlyUsedCategories'; import type RecentlyUsedReportFields from './RecentlyUsedReportFields'; diff --git a/tests/perf-test/SidebarUtils.perf-test.ts b/tests/perf-test/SidebarUtils.perf-test.ts index 953d1a858fcc..1842cee851b3 100644 --- a/tests/perf-test/SidebarUtils.perf-test.ts +++ b/tests/perf-test/SidebarUtils.perf-test.ts @@ -7,7 +7,8 @@ import ONYXKEYS from '@src/ONYXKEYS'; import type {PersonalDetails, TransactionViolations} from '@src/types/onyx'; import type Policy from '@src/types/onyx/Policy'; import type Report from '@src/types/onyx/Report'; -import type ReportAction, {ReportActions} from '@src/types/onyx/ReportAction'; +import type {ReportActions} from '@src/types/onyx/ReportAction'; +import type ReportAction from '@src/types/onyx/ReportAction'; import createCollection from '../utils/collections/createCollection'; import createPersonalDetails from '../utils/collections/personalDetails'; import createRandomPolicy from '../utils/collections/policies'; From 289d00f6ab4054680883bba3e94f19a8c8845701 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Thu, 4 Jan 2024 12:08:09 -0500 Subject: [PATCH 083/117] feat(Violations): remove comment --- src/types/onyx/TransactionViolation.ts | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/src/types/onyx/TransactionViolation.ts b/src/types/onyx/TransactionViolation.ts index 73714e78fa27..253e2e8e2a29 100644 --- a/src/types/onyx/TransactionViolation.ts +++ b/src/types/onyx/TransactionViolation.ts @@ -1,5 +1,4 @@ -// eslint-disable-next-line @typescript-eslint/consistent-type-imports -import CONST from '@src/CONST'; +import type CONST from '@src/CONST'; /** * Names of violations. From c2667a8172408c1a25fdecceaef2a57344da5ae8 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Thu, 4 Jan 2024 13:18:31 -0500 Subject: [PATCH 084/117] feat(Violations): remove duplicate violations --- src/CONST.ts | 36 ------------------------------------ 1 file changed, 36 deletions(-) diff --git a/src/CONST.ts b/src/CONST.ts index 36dc4d6649d4..f08f7334535c 100755 --- a/src/CONST.ts +++ b/src/CONST.ts @@ -3099,42 +3099,6 @@ const CONST = { TAX_REQUIRED: 'taxRequired', }, - VIOLATIONS: { - ALL_TAG_LEVELS_REQUIRED: 'allTagLevelsRequired', - AUTO_REPORTED_REJECTED_EXPENSE: 'autoReportedRejectedExpense', - BILLABLE_EXPENSE: 'billableExpense', - CASH_EXPENSE_WITH_NO_RECEIPT: 'cashExpenseWithNoReceipt', - CATEGORY_OUT_OF_POLICY: 'categoryOutOfPolicy', - CONVERSION_SURCHARGE: 'conversionSurcharge', - CUSTOM_UNIT_OUT_OF_POLICY: 'customUnitOutOfPolicy', - DUPLICATED_TRANSACTION: 'duplicatedTransaction', - FIELD_REQUIRED: 'fieldRequired', - FUTURE_DATE: 'futureDate', - INVOICE_MARKUP: 'invoiceMarkup', - MAX_AGE: 'maxAge', - MISSING_CATEGORY: 'missingCategory', - MISSING_COMMENT: 'missingComment', - MISSING_TAG: 'missingTag', - MODIFIED_AMOUNT: 'modifiedAmount', - MODIFIED_DATE: 'modifiedDate', - NON_EXPENSIWORKS_EXPENSE: 'nonExpensiworksExpense', - OVER_AUTO_APPROVAL_LIMIT: 'overAutoApprovalLimit', - OVER_CATEGORY_LIMIT: 'overCategoryLimit', - OVER_LIMIT: 'overLimit', - OVER_LIMIT_ATTENDEE: 'overLimitAttendee', - PER_DAY_LIMIT: 'perDayLimit', - RECEIPT_NOT_SMART_SCANNED: 'receiptNotSmartScanned', - RECEIPT_REQUIRED: 'receiptRequired', - RTER: 'rter', - SMARTSCAN_FAILED: 'smartscanFailed', - SOME_TAG_LEVELS_REQUIRED: 'someTagLevelsRequired', - TAG_OUT_OF_POLICY: 'tagOutOfPolicy', - TAX_AMOUNT_CHANGED: 'taxAmountChanged', - TAX_OUT_OF_POLICY: 'taxOutOfPolicy', - TAX_RATE_CHANGED: 'taxRateChanged', - TAX_REQUIRED: 'taxRequired', - }, - /** Context menu types */ CONTEXT_MENU_TYPES: { LINK: 'LINK', From d1a6ee1e2145422b95950ba6edc1d502369c9ccc Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Thu, 4 Jan 2024 13:20:02 -0500 Subject: [PATCH 085/117] feat(Violations): fix import paths --- src/components/ViolationMessages.tsx | 2 +- src/libs/Violations/ViolationsUtils.ts | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/components/ViolationMessages.tsx b/src/components/ViolationMessages.tsx index 8eb555184596..f350bd3810b9 100644 --- a/src/components/ViolationMessages.tsx +++ b/src/components/ViolationMessages.tsx @@ -2,7 +2,7 @@ import React, {useMemo} from 'react'; import {View} from 'react-native'; import useLocalize from '@hooks/useLocalize'; import useThemeStyles from '@hooks/useThemeStyles'; -import ViolationsUtils from '@libs/ViolationsUtils'; +import ViolationsUtils from '@libs/Violations/ViolationsUtils'; import type {TransactionViolation} from '@src/types/onyx'; import Text from './Text'; diff --git a/src/libs/Violations/ViolationsUtils.ts b/src/libs/Violations/ViolationsUtils.ts index 2637686e726b..c0558d7487cf 100644 --- a/src/libs/Violations/ViolationsUtils.ts +++ b/src/libs/Violations/ViolationsUtils.ts @@ -1,9 +1,9 @@ import reject from 'lodash/reject'; import Onyx from 'react-native-onyx'; +import type {Phrase, PhraseParameters} from '@libs/Localize'; import type {TranslationPaths} from '@src/languages/types'; import ONYXKEYS from '@src/ONYXKEYS'; import type {PolicyCategories, PolicyTags, Transaction, TransactionViolation} from '@src/types/onyx'; -import type {Phrase, PhraseParameters} from './Localize'; const ViolationsUtils = { /** From 6cabad0960f310d4155365ff73d672c9457759c4 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Thu, 4 Jan 2024 14:57:16 -0500 Subject: [PATCH 086/117] feat(Violations): use reportActions --- src/libs/ReportUtils.ts | 45 ++++++++++++++++++++++---- src/libs/SidebarUtils.ts | 4 +-- src/libs/TransactionUtils.ts | 6 ++-- src/types/onyx/TransactionViolation.ts | 4 +-- 4 files changed, 45 insertions(+), 14 deletions(-) diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index 500877d3a425..40e5dfca9fbe 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -14,7 +14,20 @@ import CONST from '@src/CONST'; import type {ParentNavigationSummaryParams, TranslationPaths} from '@src/languages/types'; import ONYXKEYS from '@src/ONYXKEYS'; import ROUTES from '@src/ROUTES'; -import type {Beta, Login, PersonalDetails, PersonalDetailsList, Policy, Report, ReportAction, ReportMetadata, Session, Transaction, TransactionViolations} from '@src/types/onyx'; +import type { + Beta, + Login, + PersonalDetails, + PersonalDetailsList, + Policy, + Report, + ReportAction, + ReportMetadata, + Session, + Transaction, + TransactionViolation, + TransactionViolations, +} from '@src/types/onyx'; import type {Errors, Icon, PendingAction} from '@src/types/onyx/OnyxCommon'; import type {IOUMessage, OriginalMessageActionName, OriginalMessageCreated} from '@src/types/onyx/OriginalMessage'; import type {Status} from '@src/types/onyx/PersonalDetails'; @@ -25,6 +38,7 @@ import type DeepValueOf from '@src/types/utils/DeepValueOf'; import type {EmptyObject} from '@src/types/utils/EmptyObject'; import {isEmptyObject, isNotEmptyObject} from '@src/types/utils/EmptyObject'; import type IconAsset from '@src/types/utils/IconAsset'; +import * as CollectionUtils from './CollectionUtils'; import * as CurrencyUtils from './CurrencyUtils'; import DateUtils from './DateUtils'; import isReportMessageAttachment from './isReportMessageAttachment'; @@ -431,6 +445,19 @@ Onyx.connect({ }, }); +const reportActionsByReport: OnyxCollection = {}; +Onyx.connect({ + key: ONYXKEYS.COLLECTION.REPORT_ACTIONS, + callback: (actions, key) => { + if (!key || !actions) { + return; + } + + const reportID = CollectionUtils.extractCollectionItemID(key); + reportActionsByReport[reportID] = actions; + }, +}); + function getChatType(report: OnyxEntry): ValueOf | undefined { return report?.chatType; } @@ -3384,8 +3411,14 @@ function shouldHideReport(report: OnyxEntry, currentReportId: string): b /** * Checks to see if a report's parentAction is a money request that contains a violation */ -function doesTransactionThreadHaveViolations(report: Report, transactionViolations: TransactionViolations, reportActions: OnyxCollection): boolean { - const parentReportAction = reportActions?.[`${report.parentReportID}`]?.[`${report.parentReportActionID}`]; +function doesTransactionThreadHaveViolations(report: Report, transactionViolations: OnyxCollection | undefined): boolean { + const {parentReportActionID, parentReportID} = report; + + if (!parentReportID || !parentReportActionID) { + return false; + } + const parentReportAction = reportActionsByReport?.[parentReportID]?.[parentReportActionID]; + if (!parentReportAction) { return false; } @@ -3427,8 +3460,8 @@ function shouldReportBeInOptionList( betas: Beta[], policies: OnyxCollection, excludeEmptyChats = false, - allReportActions?: OnyxCollection, - transactionViolations?: TransactionViolations, + allReportActions?: OnyxCollection, + transactionViolations?: OnyxCollection, ) { const isInDefaultMode = !isInGSDMode; // Exclude reports that have no data because there wouldn't be anything to show in the option item. @@ -3489,7 +3522,7 @@ function shouldReportBeInOptionList( } // Always show IOU reports with violations - if (isExpenseRequest(report) && betas.includes(CONST.BETAS.VIOLATIONS) && doesTransactionThreadHaveViolations(report, transactionViolations ?? {}, allReportActions ?? {})) { + if (isExpenseRequest(report) && betas.includes(CONST.BETAS.VIOLATIONS) && doesTransactionThreadHaveViolations(report, transactionViolations)) { return true; } diff --git a/src/libs/SidebarUtils.ts b/src/libs/SidebarUtils.ts index d87d2f48bb38..3172a32fc805 100644 --- a/src/libs/SidebarUtils.ts +++ b/src/libs/SidebarUtils.ts @@ -119,7 +119,7 @@ function getOrderedReportIDs( betas: Beta[], policies: Record, priorityMode: ValueOf, - allReportActions: OnyxCollection, + allReportActions: OnyxCollection, transactionViolations: TransactionViolations, ): string[] { const reportIDKey = `${ONYXKEYS.COLLECTION.REPORT_ACTIONS}${currentReportId}`; @@ -283,7 +283,7 @@ function getOptionData( const participantPersonalDetailList: PersonalDetails[] = Object.values(OptionsListUtils.getPersonalDetailsForAccountIDs(report.participantAccountIDs ?? [], personalDetails)); const personalDetail = participantPersonalDetailList[0] ?? {}; const hasErrors = Object.keys(result.allReportErrors ?? {}).length !== 0; - const hasViolations = canUseViolations && ReportUtils.doesTransactionThreadHaveViolations(report, transactionViolations, null); + const hasViolations = canUseViolations && ReportUtils.doesTransactionThreadHaveViolations(report, transactionViolations); result.isThread = ReportUtils.isChatThread(report); result.isChatRoom = ReportUtils.isChatRoom(report); diff --git a/src/libs/TransactionUtils.ts b/src/libs/TransactionUtils.ts index 42e448896b97..0c7214d686f7 100644 --- a/src/libs/TransactionUtils.ts +++ b/src/libs/TransactionUtils.ts @@ -4,7 +4,7 @@ import Onyx from 'react-native-onyx'; import type {ValueOf} from 'type-fest'; import CONST from '@src/CONST'; import ONYXKEYS from '@src/ONYXKEYS'; -import type {RecentWaypoint, Report, ReportAction, Transaction, TransactionViolation, TransactionViolations} from '@src/types/onyx'; +import type {RecentWaypoint, Report, ReportAction, Transaction, TransactionViolation} from '@src/types/onyx'; import type {PolicyTaxRates} from '@src/types/onyx/PolicyTaxRates'; import type PolicyTaxRate from '@src/types/onyx/PolicyTaxRates'; import type {Comment, Receipt, Waypoint, WaypointCollection} from '@src/types/onyx/Transaction'; @@ -524,8 +524,8 @@ function getRecentTransactions(transactions: Record, size = 2): /** * Checks if any violations for the provided transaction are of type 'violation' */ -function hasViolation(transactionID: string, transactionViolations: TransactionViolations): boolean { - return Boolean(transactionViolations[ONYXKEYS.COLLECTION.TRANSACTION_VIOLATIONS + transactionID]?.some((violation: TransactionViolation) => violation.type === 'violation')); +function hasViolation(transactionID: string, transactionViolations: OnyxCollection | undefined): boolean { + return Boolean(transactionViolations?.[ONYXKEYS.COLLECTION.TRANSACTION_VIOLATIONS + transactionID]?.some((violation: TransactionViolation) => violation.type === 'violation')); } /** diff --git a/src/types/onyx/TransactionViolation.ts b/src/types/onyx/TransactionViolation.ts index 253e2e8e2a29..dd7a9ef65746 100644 --- a/src/types/onyx/TransactionViolation.ts +++ b/src/types/onyx/TransactionViolation.ts @@ -31,6 +31,4 @@ type TransactionViolation = { }; }; -type TransactionViolations = Record; - -export type {TransactionViolation, TransactionViolations, ViolationName}; +export type {TransactionViolation, ViolationName}; From c8ebe6f529defef6e0210da1acb1542162a0ec64 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Fri, 5 Jan 2024 16:04:55 -0500 Subject: [PATCH 087/117] feat(Violations): remove transaction violations --- src/types/onyx/index.ts | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/src/types/onyx/index.ts b/src/types/onyx/index.ts index 9fd52bd0628e..7bd9c321be5e 100644 --- a/src/types/onyx/index.ts +++ b/src/types/onyx/index.ts @@ -55,7 +55,7 @@ import type SecurityGroup from './SecurityGroup'; import type Session from './Session'; import type Task from './Task'; import type Transaction from './Transaction'; -import type {TransactionViolation, TransactionViolations, ViolationName} from './TransactionViolation'; +import type {TransactionViolation, ViolationName} from './TransactionViolation'; import type User from './User'; import type UserLocation from './UserLocation'; import type UserWallet from './UserWallet'; @@ -125,7 +125,6 @@ export type { Task, Transaction, TransactionViolation, - TransactionViolations, User, UserLocation, UserWallet, From ec7dc4f4779a0fdbd927f45877c7f719ee915218 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Fri, 5 Jan 2024 16:08:16 -0500 Subject: [PATCH 088/117] feat(Violations): remove transaction violations --- src/libs/SidebarUtils.ts | 29 +++++++++++++---------------- 1 file changed, 13 insertions(+), 16 deletions(-) diff --git a/src/libs/SidebarUtils.ts b/src/libs/SidebarUtils.ts index 1c356d0f56d2..6054eb664e4c 100644 --- a/src/libs/SidebarUtils.ts +++ b/src/libs/SidebarUtils.ts @@ -5,7 +5,7 @@ import Onyx from 'react-native-onyx'; import type {ValueOf} from 'type-fest'; import CONST from '@src/CONST'; import ONYXKEYS from '@src/ONYXKEYS'; -import type {PersonalDetails, TransactionViolations} from '@src/types/onyx'; +import type {PersonalDetails, TransactionViolation} from '@src/types/onyx'; import type Beta from '@src/types/onyx/Beta'; import type * as OnyxCommon from '@src/types/onyx/OnyxCommon'; import type Policy from '@src/types/onyx/Policy'; @@ -120,26 +120,23 @@ function getOrderedReportIDs( policies: Record, priorityMode: ValueOf, allReportActions: OnyxCollection, - transactionViolations: TransactionViolations, + transactionViolations: OnyxCollection, ): string[] { const reportIDKey = `${ONYXKEYS.COLLECTION.REPORT_ACTIONS}${currentReportId}`; const reportActionCount = allReportActions?.[reportIDKey]?.length ?? 1; // Generate a unique cache key based on the function arguments - const cachedReportsKey = JSON.stringify( - [currentReportId, allReports, betas, policies, priorityMode, reportActionCount], - (key, value: unknown) => { - /** - * Exclude some properties not to overwhelm a cached key value with huge data, - * which we don't need to store in a cacheKey - */ - if (key === 'participantAccountIDs' || key === 'participants' || key === 'lastMessageText' || key === 'visibleChatMemberAccountIDs') { - return undefined; - } + const cachedReportsKey = JSON.stringify([currentReportId, allReports, betas, policies, priorityMode, reportActionCount], (key, value: unknown) => { + /** + * Exclude some properties not to overwhelm a cached key value with huge data, + * which we don't need to store in a cacheKey + */ + if (key === 'participantAccountIDs' || key === 'participants' || key === 'lastMessageText' || key === 'visibleChatMemberAccountIDs') { + return undefined; + } - return value; - }, - ); + return value; + }); // Check if the result is already in the cache const cachedIDs = reportIDsCache.get(cachedReportsKey); @@ -245,7 +242,7 @@ function getOptionData( preferredLocale: ValueOf, policy: Policy, parentReportAction: ReportAction, - transactionViolations: TransactionViolations, + transactionViolations: OnyxCollection, canUseViolations: boolean, ): ReportUtils.OptionData | undefined { // When a user signs out, Onyx is cleared. Due to the lazy rendering with a virtual list, it's possible for From aacb8941f3f4ed6c1b9416e372d905963daff9a8 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Fri, 5 Jan 2024 16:17:14 -0500 Subject: [PATCH 089/117] feat(Violations): lint --- src/libs/ReportUtils.ts | 17 ++--------------- 1 file changed, 2 insertions(+), 15 deletions(-) diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index 502b8e21081b..d8d6315f1e10 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -14,20 +14,7 @@ import CONST from '@src/CONST'; import type {ParentNavigationSummaryParams, TranslationPaths} from '@src/languages/types'; import ONYXKEYS from '@src/ONYXKEYS'; import ROUTES from '@src/ROUTES'; -import type { - Beta, - Login, - PersonalDetails, - PersonalDetailsList, - Policy, - Report, - ReportAction, - ReportMetadata, - Session, - Transaction, - TransactionViolation, - TransactionViolations, -} from '@src/types/onyx'; +import type {Beta, Login, PersonalDetails, PersonalDetailsList, Policy, Report, ReportAction, ReportMetadata, Session, Transaction, TransactionViolation} from '@src/types/onyx'; import type {Errors, Icon, PendingAction} from '@src/types/onyx/OnyxCommon'; import type {IOUMessage, OriginalMessageActionName, OriginalMessageCreated} from '@src/types/onyx/OriginalMessage'; import type {Status} from '@src/types/onyx/PersonalDetails'; @@ -3490,7 +3477,7 @@ function doesTransactionThreadHaveViolations(report: Report, transactionViolatio /** * Checks to see if a report contains a violation */ -function hasViolations(reportID: string, transactionViolations: TransactionViolations): boolean { +function hasViolations(reportID: string, transactionViolations: OnyxCollection): boolean { const transactions = TransactionUtils.getAllReportTransactions(reportID); return transactions.some((transaction) => hasViolation(transaction.transactionID, transactionViolations)); } From 9e576414fc05e728a4470e6aa0d2b70391c062c3 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Mon, 8 Jan 2024 13:56:45 -0500 Subject: [PATCH 090/117] feat(Violations): create params object for getOptionData --- .../LHNOptionsList/OptionRowLHNData.js | 11 ++++++- src/libs/ReportUtils.ts | 1 - src/libs/SidebarUtils.ts | 33 ++++++++++++------- tests/perf-test/SidebarUtils.perf-test.ts | 22 ++++++++++--- 4 files changed, 49 insertions(+), 18 deletions(-) diff --git a/src/components/LHNOptionsList/OptionRowLHNData.js b/src/components/LHNOptionsList/OptionRowLHNData.js index 82ec30cafe77..4f811b55a1a1 100644 --- a/src/components/LHNOptionsList/OptionRowLHNData.js +++ b/src/components/LHNOptionsList/OptionRowLHNData.js @@ -93,7 +93,16 @@ function OptionRowLHNData({ const optionItem = useMemo(() => { // Note: ideally we'd have this as a dependent selector in onyx! - const item = SidebarUtils.getOptionData(fullReport, reportActions, personalDetails, preferredLocale, policy, parentReportAction, transactionViolations, canUseViolations); + const item = SidebarUtils.getOptionData({ + report: fullReport, + reportActions, + personalDetails, + preferredLocale, + policy, + parentReportAction, + transactionViolations, + canUseViolations, + }); if (deepEqual(item, optionItemRef.current)) { return optionItemRef.current; } diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index d8d6315f1e10..caa5dbb46cd3 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -3496,7 +3496,6 @@ function shouldReportBeInOptionList( betas: Beta[], policies: OnyxCollection, excludeEmptyChats = false, - allReportActions?: OnyxCollection, transactionViolations?: OnyxCollection, ) { const isInDefaultMode = !isInGSDMode; diff --git a/src/libs/SidebarUtils.ts b/src/libs/SidebarUtils.ts index 6054eb664e4c..853d54628e5b 100644 --- a/src/libs/SidebarUtils.ts +++ b/src/libs/SidebarUtils.ts @@ -119,7 +119,7 @@ function getOrderedReportIDs( betas: Beta[], policies: Record, priorityMode: ValueOf, - allReportActions: OnyxCollection, + allReportActions: OnyxCollection, transactionViolations: OnyxCollection, ): string[] { const reportIDKey = `${ONYXKEYS.COLLECTION.REPORT_ACTIONS}${currentReportId}`; @@ -152,7 +152,7 @@ function getOrderedReportIDs( const allReportsDictValues = Object.values(allReports); // Filter out all the reports that shouldn't be displayed const reportsToDisplay = allReportsDictValues.filter((report) => - ReportUtils.shouldReportBeInOptionList(report, currentReportId ?? '', isInGSDMode, betas, policies, true, allReportActions, transactionViolations), + ReportUtils.shouldReportBeInOptionList(report, currentReportId ?? '', isInGSDMode, betas, policies, true, transactionViolations), ); if (reportsToDisplay.length === 0) { @@ -235,16 +235,25 @@ type ActorDetails = { /** * Gets all the data necessary for rendering an OptionRowLHN component */ -function getOptionData( - report: Report, - reportActions: Record, - personalDetails: Record, - preferredLocale: ValueOf, - policy: Policy, - parentReportAction: ReportAction, - transactionViolations: OnyxCollection, - canUseViolations: boolean, -): ReportUtils.OptionData | undefined { +function getOptionData({ + report, + reportActions, + personalDetails, + preferredLocale, + policy, + parentReportAction, + transactionViolations, + canUseViolations, +}: { + report: Report; + reportActions: Record; + personalDetails: Record; + preferredLocale: ValueOf; + policy: Policy; + parentReportAction: ReportAction; + transactionViolations: OnyxCollection; + canUseViolations: boolean; +}): ReportUtils.OptionData | undefined { // When a user signs out, Onyx is cleared. Due to the lazy rendering with a virtual list, it's possible for // this method to be called after the Onyx data has been cleared out. In that case, it's fine to do // a null check here and return early. diff --git a/tests/perf-test/SidebarUtils.perf-test.ts b/tests/perf-test/SidebarUtils.perf-test.ts index 1842cee851b3..fca4bac0e522 100644 --- a/tests/perf-test/SidebarUtils.perf-test.ts +++ b/tests/perf-test/SidebarUtils.perf-test.ts @@ -4,7 +4,7 @@ import {measureFunction} from 'reassure'; import SidebarUtils from '@libs/SidebarUtils'; import CONST from '@src/CONST'; import ONYXKEYS from '@src/ONYXKEYS'; -import type {PersonalDetails, TransactionViolations} from '@src/types/onyx'; +import type {PersonalDetails, TransactionViolation} from '@src/types/onyx'; import type Policy from '@src/types/onyx/Policy'; import type Report from '@src/types/onyx/Report'; import type {ReportActions} from '@src/types/onyx/ReportAction'; @@ -53,21 +53,35 @@ test('[SidebarUtils] getOptionData on 5k reports', async () => { const preferredLocale = 'en'; const policy = createRandomPolicy(1); const parentReportAction = createRandomReportAction(1); - const transactionViolations = {} as TransactionViolations; + const transactionViolations = {} as OnyxCollection; Onyx.multiSet({ ...mockedResponseMap, }); await waitForBatchedUpdates(); - await measureFunction(() => SidebarUtils.getOptionData(report, reportActions, personalDetails, preferredLocale, policy, parentReportAction, transactionViolations, false), {runs}); + + await measureFunction( + () => + SidebarUtils.getOptionData({ + report, + reportActions, + personalDetails, + preferredLocale, + policy, + parentReportAction, + transactionViolations, + canUseViolations: false, + }), + {runs}, + ); }); test('[SidebarUtils] getOrderedReportIDs on 5k reports', async () => { const currentReportId = '1'; const allReports = getMockedReports(); const betas = [CONST.BETAS.DEFAULT_ROOMS, CONST.BETAS.POLICY_ROOMS]; - const transactionViolations = {} as TransactionViolations; + const transactionViolations = {} as OnyxCollection; const policies = createCollection( (item) => `${ONYXKEYS.COLLECTION.POLICY}${item.id}`, From 79ac93a435fb1e3a664567858eaaa39ecf23b9b9 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Mon, 8 Jan 2024 14:04:03 -0500 Subject: [PATCH 091/117] feat(Violations): create params object for shouldReportBeInOptionsList --- src/libs/OptionsListUtils.js | 8 +++++++- src/libs/ReportUtils.ts | 24 ++++++++++++++++-------- src/libs/SidebarUtils.ts | 2 +- src/libs/UnreadIndicatorUpdater/index.ts | 9 ++++++++- tests/perf-test/ReportUtils.perf-test.ts | 2 +- 5 files changed, 33 insertions(+), 12 deletions(-) diff --git a/src/libs/OptionsListUtils.js b/src/libs/OptionsListUtils.js index f4062d0a5a3f..c9bbcb8dd26e 100644 --- a/src/libs/OptionsListUtils.js +++ b/src/libs/OptionsListUtils.js @@ -1319,7 +1319,13 @@ function getOptions( // Filter out all the reports that shouldn't be displayed const filteredReports = _.filter(reports, (report) => - ReportUtils.shouldReportBeInOptionList(report, Navigation.getTopmostReportId(), false, betas, policies, false, reportActions, transactionViolations), + ReportUtils.shouldReportBeInOptionList({ + report, + currentReportId: Navigation.getTopmostReportId(), + betas, + policies, + transactionViolations, + }), ); // Sorting the reports works like this: diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index caa5dbb46cd3..c7d01de18337 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -3489,15 +3489,23 @@ function hasViolations(reportID: string, transactionViolations: OnyxCollection, - currentReportId: string, - isInGSDMode: boolean, - betas: Beta[], - policies: OnyxCollection, +function shouldReportBeInOptionList({ + report, + currentReportId, + isInGSDMode = false, + betas, + policies, excludeEmptyChats = false, - transactionViolations?: OnyxCollection, -) { + transactionViolations, +}: { + report: OnyxEntry; + currentReportId: string; + isInGSDMode?: boolean; + betas: Beta[]; + policies: OnyxCollection; + excludeEmptyChats?: boolean; + transactionViolations?: OnyxCollection; +}) { const isInDefaultMode = !isInGSDMode; // Exclude reports that have no data because there wouldn't be anything to show in the option item. // This can happen if data is currently loading from the server or a report is in various stages of being created. diff --git a/src/libs/SidebarUtils.ts b/src/libs/SidebarUtils.ts index 853d54628e5b..a657c05125b6 100644 --- a/src/libs/SidebarUtils.ts +++ b/src/libs/SidebarUtils.ts @@ -152,7 +152,7 @@ function getOrderedReportIDs( const allReportsDictValues = Object.values(allReports); // Filter out all the reports that shouldn't be displayed const reportsToDisplay = allReportsDictValues.filter((report) => - ReportUtils.shouldReportBeInOptionList(report, currentReportId ?? '', isInGSDMode, betas, policies, true, transactionViolations), + ReportUtils.shouldReportBeInOptionList({report, currentReportId: currentReportId ?? '', isInGSDMode, betas, policies, excludeEmptyChats: true, transactionViolations}), ); if (reportsToDisplay.length === 0) { diff --git a/src/libs/UnreadIndicatorUpdater/index.ts b/src/libs/UnreadIndicatorUpdater/index.ts index 4126f2af4e83..42584cc9d877 100644 --- a/src/libs/UnreadIndicatorUpdater/index.ts +++ b/src/libs/UnreadIndicatorUpdater/index.ts @@ -13,7 +13,14 @@ const triggerUnreadUpdate = () => { // We want to keep notification count consistent with what can be accessed from the LHN list const unreadReports = Object.values(allReports ?? {}).filter( - (report) => ReportUtils.isUnread(report) && ReportUtils.shouldReportBeInOptionList(report, currentReportID ?? '', false, [], {}), + (report) => + ReportUtils.isUnread(report) && + ReportUtils.shouldReportBeInOptionList({ + report, + currentReportId: currentReportID ?? '', + betas: [], + policies: {}, + }), ); updateUnread(unreadReports.length); }; diff --git a/tests/perf-test/ReportUtils.perf-test.ts b/tests/perf-test/ReportUtils.perf-test.ts index 81579c1be277..b29095207303 100644 --- a/tests/perf-test/ReportUtils.perf-test.ts +++ b/tests/perf-test/ReportUtils.perf-test.ts @@ -159,7 +159,7 @@ test('[ReportUtils] shouldReportBeInOptionList on 1k participant', async () => { const policies = getMockedPolicies(); await waitForBatchedUpdates(); - await measureFunction(() => ReportUtils.shouldReportBeInOptionList(report, currentReportId, isInGSDMode, betas, policies, false, {}), {runs}); + await measureFunction(() => ReportUtils.shouldReportBeInOptionList({report, currentReportId, isInGSDMode, betas, policies, transactionViolations: {}}), {runs}); }); test('[ReportUtils] getWorkspaceIcon on 5k policies', async () => { From 13fe960651b6f827177b7c17314678e65f11aeae Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Mon, 8 Jan 2024 14:37:58 -0500 Subject: [PATCH 092/117] feat(Violations): invert control and pass doesTransactionThreadHaveViolations into calling functions as a param --- .../LHNOptionsList/OptionRowLHNData.js | 6 +++-- src/libs/OptionsListUtils.js | 7 +++++- src/libs/ReportUtils.ts | 13 ++++------- src/libs/SidebarUtils.ts | 22 +++++++++++------- src/libs/UnreadIndicatorUpdater/index.ts | 23 +++++++++++-------- tests/perf-test/ReportUtils.perf-test.ts | 2 +- tests/perf-test/SidebarUtils.perf-test.ts | 4 +--- 7 files changed, 44 insertions(+), 33 deletions(-) diff --git a/src/components/LHNOptionsList/OptionRowLHNData.js b/src/components/LHNOptionsList/OptionRowLHNData.js index 4f811b55a1a1..f2d6785753a9 100644 --- a/src/components/LHNOptionsList/OptionRowLHNData.js +++ b/src/components/LHNOptionsList/OptionRowLHNData.js @@ -5,6 +5,7 @@ import _ from 'underscore'; import participantPropTypes from '@components/participantPropTypes'; import transactionPropTypes from '@components/transactionPropTypes'; import * as ReportActionsUtils from '@libs/ReportActionsUtils'; +import * as ReportUtils from '@libs/ReportUtils'; import SidebarUtils from '@libs/SidebarUtils'; import * as TransactionUtils from '@libs/TransactionUtils'; import {transactionViolationsPropType} from '@libs/Violations/propTypes'; @@ -91,6 +92,8 @@ function OptionRowLHNData({ // eslint-disable-next-line react-hooks/exhaustive-deps }, [fullReport.reportID, receiptTransactions, reportActions]); + const hasViolations = canUseViolations && ReportUtils.doesTransactionThreadHaveViolations({report: fullReport, transactionViolations}); + const optionItem = useMemo(() => { // Note: ideally we'd have this as a dependent selector in onyx! const item = SidebarUtils.getOptionData({ @@ -100,8 +103,7 @@ function OptionRowLHNData({ preferredLocale, policy, parentReportAction, - transactionViolations, - canUseViolations, + hasViolations, }); if (deepEqual(item, optionItemRef.current)) { return optionItemRef.current; diff --git a/src/libs/OptionsListUtils.js b/src/libs/OptionsListUtils.js index c9bbcb8dd26e..b95994f06d63 100644 --- a/src/libs/OptionsListUtils.js +++ b/src/libs/OptionsListUtils.js @@ -1324,7 +1324,12 @@ function getOptions( currentReportId: Navigation.getTopmostReportId(), betas, policies, - transactionViolations, + doesReportTransactionThreadHaveViolations: + betas.included(CONST.BETAS.VIOLATIONS) && + ReportUtils.doesTransactionThreadHaveViolations({ + report, + transactionViolations, + }), }), ); diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index c7d01de18337..7784c867ea28 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -3447,7 +3447,7 @@ function shouldHideReport(report: OnyxEntry, currentReportId: string): b /** * Checks to see if a report's parentAction is a money request that contains a violation */ -function doesTransactionThreadHaveViolations(report: Report, transactionViolations: OnyxCollection | undefined): boolean { +function doesTransactionThreadHaveViolations({report, transactionViolations}: {report: Report; transactionViolations?: OnyxCollection}): boolean { const {parentReportActionID, parentReportID} = report; if (!parentReportID || !parentReportActionID) { @@ -3455,10 +3455,7 @@ function doesTransactionThreadHaveViolations(report: Report, transactionViolatio } const parentReportAction = reportActionsByReport?.[parentReportID]?.[parentReportActionID]; - if (!parentReportAction) { - return false; - } - if (parentReportAction.actionName !== CONST.REPORT.ACTIONS.TYPE.IOU) { + if (!parentReportAction || parentReportAction.actionName !== CONST.REPORT.ACTIONS.TYPE.IOU) { return false; } const {IOUTransactionID, IOUReportID} = parentReportAction?.originalMessage; @@ -3496,7 +3493,7 @@ function shouldReportBeInOptionList({ betas, policies, excludeEmptyChats = false, - transactionViolations, + doesReportTransactionThreadHaveViolations, }: { report: OnyxEntry; currentReportId: string; @@ -3504,7 +3501,7 @@ function shouldReportBeInOptionList({ betas: Beta[]; policies: OnyxCollection; excludeEmptyChats?: boolean; - transactionViolations?: OnyxCollection; + doesReportTransactionThreadHaveViolations: boolean; }) { const isInDefaultMode = !isInGSDMode; // Exclude reports that have no data because there wouldn't be anything to show in the option item. @@ -3565,7 +3562,7 @@ function shouldReportBeInOptionList({ } // Always show IOU reports with violations - if (isExpenseRequest(report) && betas.includes(CONST.BETAS.VIOLATIONS) && doesTransactionThreadHaveViolations(report, transactionViolations)) { + if (isExpenseRequest(report) && doesReportTransactionThreadHaveViolations) { return true; } diff --git a/src/libs/SidebarUtils.ts b/src/libs/SidebarUtils.ts index a657c05125b6..92f6d530c134 100644 --- a/src/libs/SidebarUtils.ts +++ b/src/libs/SidebarUtils.ts @@ -151,9 +151,18 @@ function getOrderedReportIDs( const isInDefaultMode = !isInGSDMode; const allReportsDictValues = Object.values(allReports); // Filter out all the reports that shouldn't be displayed - const reportsToDisplay = allReportsDictValues.filter((report) => - ReportUtils.shouldReportBeInOptionList({report, currentReportId: currentReportId ?? '', isInGSDMode, betas, policies, excludeEmptyChats: true, transactionViolations}), - ); + const reportsToDisplay = allReportsDictValues.filter((report) => { + const doesReportTransactionThreadHaveViolations = betas.includes(CONST.BETAS.VIOLATIONS) && ReportUtils.doesTransactionThreadHaveViolations({report, transactionViolations}); + return ReportUtils.shouldReportBeInOptionList({ + report, + currentReportId: currentReportId ?? '', + isInGSDMode, + betas, + policies, + excludeEmptyChats: true, + doesReportTransactionThreadHaveViolations, + }); + }); if (reportsToDisplay.length === 0) { // Display Concierge chat report when there is no report to be displayed @@ -242,8 +251,7 @@ function getOptionData({ preferredLocale, policy, parentReportAction, - transactionViolations, - canUseViolations, + hasViolations, }: { report: Report; reportActions: Record; @@ -251,8 +259,7 @@ function getOptionData({ preferredLocale: ValueOf; policy: Policy; parentReportAction: ReportAction; - transactionViolations: OnyxCollection; - canUseViolations: boolean; + hasViolations: boolean; }): ReportUtils.OptionData | undefined { // When a user signs out, Onyx is cleared. Due to the lazy rendering with a virtual list, it's possible for // this method to be called after the Onyx data has been cleared out. In that case, it's fine to do @@ -295,7 +302,6 @@ function getOptionData({ const participantPersonalDetailList: PersonalDetails[] = Object.values(OptionsListUtils.getPersonalDetailsForAccountIDs(report.participantAccountIDs ?? [], personalDetails)); const personalDetail = participantPersonalDetailList[0] ?? {}; const hasErrors = Object.keys(result.allReportErrors ?? {}).length !== 0; - const hasViolations = canUseViolations && ReportUtils.doesTransactionThreadHaveViolations(report, transactionViolations); result.isThread = ReportUtils.isChatThread(report); result.isChatRoom = ReportUtils.isChatRoom(report); diff --git a/src/libs/UnreadIndicatorUpdater/index.ts b/src/libs/UnreadIndicatorUpdater/index.ts index 42584cc9d877..5be9f0bbb268 100644 --- a/src/libs/UnreadIndicatorUpdater/index.ts +++ b/src/libs/UnreadIndicatorUpdater/index.ts @@ -12,16 +12,19 @@ const triggerUnreadUpdate = () => { const currentReportID = navigationRef.isReady() ? Navigation.getTopmostReportId() : ''; // We want to keep notification count consistent with what can be accessed from the LHN list - const unreadReports = Object.values(allReports ?? {}).filter( - (report) => - ReportUtils.isUnread(report) && - ReportUtils.shouldReportBeInOptionList({ - report, - currentReportId: currentReportID ?? '', - betas: [], - policies: {}, - }), - ); + const unreadReports = Object.values(allReports ?? {}).filter((report) => { + if (!ReportUtils.isUnread(report)) { + return false; + } + + return ReportUtils.shouldReportBeInOptionList({ + report, + currentReportId: currentReportID ?? '', + betas: [], + policies: {}, + doesReportTransactionThreadHaveViolations: false, + }); + }); updateUnread(unreadReports.length); }; diff --git a/tests/perf-test/ReportUtils.perf-test.ts b/tests/perf-test/ReportUtils.perf-test.ts index b29095207303..14cf45545890 100644 --- a/tests/perf-test/ReportUtils.perf-test.ts +++ b/tests/perf-test/ReportUtils.perf-test.ts @@ -159,7 +159,7 @@ test('[ReportUtils] shouldReportBeInOptionList on 1k participant', async () => { const policies = getMockedPolicies(); await waitForBatchedUpdates(); - await measureFunction(() => ReportUtils.shouldReportBeInOptionList({report, currentReportId, isInGSDMode, betas, policies, transactionViolations: {}}), {runs}); + await measureFunction(() => ReportUtils.shouldReportBeInOptionList({report, currentReportId, isInGSDMode, betas, policies, doesReportTransactionThreadHaveViolations: false}), {runs}); }); test('[ReportUtils] getWorkspaceIcon on 5k policies', async () => { diff --git a/tests/perf-test/SidebarUtils.perf-test.ts b/tests/perf-test/SidebarUtils.perf-test.ts index fca4bac0e522..a57cffba1023 100644 --- a/tests/perf-test/SidebarUtils.perf-test.ts +++ b/tests/perf-test/SidebarUtils.perf-test.ts @@ -53,7 +53,6 @@ test('[SidebarUtils] getOptionData on 5k reports', async () => { const preferredLocale = 'en'; const policy = createRandomPolicy(1); const parentReportAction = createRandomReportAction(1); - const transactionViolations = {} as OnyxCollection; Onyx.multiSet({ ...mockedResponseMap, @@ -70,8 +69,7 @@ test('[SidebarUtils] getOptionData on 5k reports', async () => { preferredLocale, policy, parentReportAction, - transactionViolations, - canUseViolations: false, + hasViolations: false, }), {runs}, ); From 66c181e463b9ebebeff98525ae656460e1fc44e8 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Mon, 8 Jan 2024 14:58:39 -0500 Subject: [PATCH 093/117] feat(Violations): fix TransactionViolations export --- src/types/onyx/TransactionViolation.ts | 3 +++ src/types/onyx/index.ts | 2 ++ 2 files changed, 5 insertions(+) diff --git a/src/types/onyx/TransactionViolation.ts b/src/types/onyx/TransactionViolation.ts index dd7a9ef65746..20603adc7cfa 100644 --- a/src/types/onyx/TransactionViolation.ts +++ b/src/types/onyx/TransactionViolation.ts @@ -31,4 +31,7 @@ type TransactionViolation = { }; }; +type TransactionViolations = Record; + export type {TransactionViolation, ViolationName}; +export default TransactionViolations; diff --git a/src/types/onyx/index.ts b/src/types/onyx/index.ts index 7bd9c321be5e..d1a98b3ee5ec 100644 --- a/src/types/onyx/index.ts +++ b/src/types/onyx/index.ts @@ -56,6 +56,7 @@ import type Session from './Session'; import type Task from './Task'; import type Transaction from './Transaction'; import type {TransactionViolation, ViolationName} from './TransactionViolation'; +import type TransactionViolations from './TransactionViolation'; import type User from './User'; import type UserLocation from './UserLocation'; import type UserWallet from './UserWallet'; @@ -125,6 +126,7 @@ export type { Task, Transaction, TransactionViolation, + TransactionViolations, User, UserLocation, UserWallet, From 2f75e741c4b1482f02c1046c3b76dcf175e740b5 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Mon, 8 Jan 2024 15:37:20 -0500 Subject: [PATCH 094/117] feat(Violations): take parentReportAction as param an update callers --- .../LHNOptionsList/OptionRowLHNData.js | 2 +- src/libs/OptionsListUtils.js | 11 +++++++---- src/libs/ReportUtils.ts | 19 ++++++++++--------- src/libs/SidebarUtils.ts | 5 ++++- src/pages/NewChatPage.js | 5 ++++- 5 files changed, 26 insertions(+), 16 deletions(-) diff --git a/src/components/LHNOptionsList/OptionRowLHNData.js b/src/components/LHNOptionsList/OptionRowLHNData.js index f2d6785753a9..5134eed63bc1 100644 --- a/src/components/LHNOptionsList/OptionRowLHNData.js +++ b/src/components/LHNOptionsList/OptionRowLHNData.js @@ -92,7 +92,7 @@ function OptionRowLHNData({ // eslint-disable-next-line react-hooks/exhaustive-deps }, [fullReport.reportID, receiptTransactions, reportActions]); - const hasViolations = canUseViolations && ReportUtils.doesTransactionThreadHaveViolations({report: fullReport, transactionViolations}); + const hasViolations = canUseViolations && ReportUtils.doesTransactionThreadHaveViolations({report: fullReport, transactionViolations, parentReportAction}); const optionItem = useMemo(() => { // Note: ideally we'd have this as a dependent selector in onyx! diff --git a/src/libs/OptionsListUtils.js b/src/libs/OptionsListUtils.js index b95994f06d63..034dc401ae92 100644 --- a/src/libs/OptionsListUtils.js +++ b/src/libs/OptionsListUtils.js @@ -1318,8 +1318,10 @@ function getOptions( const searchValue = parsedPhoneNumber.possible ? parsedPhoneNumber.number.e164 : searchInputValue.toLowerCase(); // Filter out all the reports that shouldn't be displayed - const filteredReports = _.filter(reports, (report) => - ReportUtils.shouldReportBeInOptionList({ + const filteredReports = _.filter(reports, (report) => { + const parentReportAction = !report.parentReportID || !report.parentReportActionID ? {} : lodashGet(allReportActions, [report.parentReportID, report.parentReportActionID], {}); + + return ReportUtils.shouldReportBeInOptionList({ report, currentReportId: Navigation.getTopmostReportId(), betas, @@ -1329,9 +1331,10 @@ function getOptions( ReportUtils.doesTransactionThreadHaveViolations({ report, transactionViolations, + parentReportAction, }), - }), - ); + }); + }); // Sorting the reports works like this: // - Order everything by the last message timestamp (descending) diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index a9664700a51b..0ab76ef68842 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -3447,15 +3447,16 @@ function shouldHideReport(report: OnyxEntry, currentReportId: string): b /** * Checks to see if a report's parentAction is a money request that contains a violation */ -function doesTransactionThreadHaveViolations({report, transactionViolations}: {report: Report; transactionViolations?: OnyxCollection}): boolean { - const {parentReportActionID, parentReportID} = report; - - if (!parentReportID || !parentReportActionID) { - return false; - } - const parentReportAction = reportActionsByReport?.[parentReportID]?.[parentReportActionID]; - - if (!parentReportAction || parentReportAction.actionName !== CONST.REPORT.ACTIONS.TYPE.IOU) { +function doesTransactionThreadHaveViolations({ + report, + transactionViolations, + parentReportAction, +}: { + report: Report; + transactionViolations?: OnyxCollection; + parentReportAction: ReportAction; +}): boolean { + if (parentReportAction.actionName !== CONST.REPORT.ACTIONS.TYPE.IOU) { return false; } const {IOUTransactionID, IOUReportID} = parentReportAction?.originalMessage; diff --git a/src/libs/SidebarUtils.ts b/src/libs/SidebarUtils.ts index 92f6d530c134..87d77b6f6053 100644 --- a/src/libs/SidebarUtils.ts +++ b/src/libs/SidebarUtils.ts @@ -1,5 +1,6 @@ /* eslint-disable rulesdir/prefer-underscore-method */ import Str from 'expensify-common/lib/str'; +import lodashGet from 'lodash/get'; import type {OnyxCollection} from 'react-native-onyx'; import Onyx from 'react-native-onyx'; import type {ValueOf} from 'type-fest'; @@ -152,7 +153,9 @@ function getOrderedReportIDs( const allReportsDictValues = Object.values(allReports); // Filter out all the reports that shouldn't be displayed const reportsToDisplay = allReportsDictValues.filter((report) => { - const doesReportTransactionThreadHaveViolations = betas.includes(CONST.BETAS.VIOLATIONS) && ReportUtils.doesTransactionThreadHaveViolations({report, transactionViolations}); + const parentReportAction = !report.parentReportID || !report.parentReportActionID ? {} : lodashGet(allReportActions, [report.parentReportID, report.parentReportActionID], {}); + const doesReportTransactionThreadHaveViolations = + betas.includes(CONST.BETAS.VIOLATIONS) && ReportUtils.doesTransactionThreadHaveViolations({report, transactionViolations, parentReportAction}); return ReportUtils.shouldReportBeInOptionList({ report, currentReportId: currentReportId ?? '', diff --git a/src/pages/NewChatPage.js b/src/pages/NewChatPage.js index 9ef3e50e8d26..049ca8b4ef84 100755 --- a/src/pages/NewChatPage.js +++ b/src/pages/NewChatPage.js @@ -21,6 +21,7 @@ import variables from '@styles/variables'; import * as Report from '@userActions/Report'; import CONST from '@src/CONST'; import ONYXKEYS from '@src/ONYXKEYS'; +import reportActionPropTypes from './home/report/reportActionPropTypes'; import personalDetailsPropType from './personalDetailsPropType'; import reportPropTypes from './reportPropTypes'; @@ -40,6 +41,7 @@ const propTypes = { /** Whether we are searching for reports in the server */ isSearchingForReports: PropTypes.bool, + reportActions: PropTypes.objectOf(PropTypes.shape(reportActionPropTypes)), }; const defaultProps = { @@ -47,11 +49,12 @@ const defaultProps = { personalDetails: {}, reports: {}, isSearchingForReports: false, + reportActions: {}, }; const excludedGroupEmails = _.without(CONST.EXPENSIFY_EMAILS, CONST.EMAIL.CONCIERGE); -function NewChatPage({betas, isGroupChat, personalDetails, reports, translate, isSearchingForReports}) { +function NewChatPage({betas, isGroupChat, personalDetails, reports, translate, isSearchingForReports, reportActions}) { const styles = useThemeStyles(); const [searchTerm, setSearchTerm] = useState(''); const [filteredRecentReports, setFilteredRecentReports] = useState([]); From e110f14cbf6e25d420c674e071a3a9f00e18dd7e Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Mon, 8 Jan 2024 17:37:26 -0500 Subject: [PATCH 095/117] feat(Violations): fix import --- src/libs/actions/IOU.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/libs/actions/IOU.js b/src/libs/actions/IOU.js index f2584cb8accd..6d20c3badac5 100644 --- a/src/libs/actions/IOU.js +++ b/src/libs/actions/IOU.js @@ -22,7 +22,7 @@ import * as ReportActionsUtils from '@libs/ReportActionsUtils'; import * as ReportUtils from '@libs/ReportUtils'; import * as TransactionUtils from '@libs/TransactionUtils'; import * as UserUtils from '@libs/UserUtils'; -import ViolationsUtils from '@libs/ViolationsUtils'; +import ViolationsUtils from '@libs/Violations/ViolationsUtils'; import CONST from '@src/CONST'; import ONYXKEYS from '@src/ONYXKEYS'; import ROUTES from '@src/ROUTES'; From 00a8bec8b1f5196f3a91cf9521838947442ac07c Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Mon, 8 Jan 2024 17:56:05 -0500 Subject: [PATCH 096/117] feat(Violations): pass proper parentReportAction in to doesTransactionThreadHaveViolations --- src/libs/OptionsListUtils.js | 8 ++++++-- src/libs/ReportUtils.ts | 4 ++-- src/libs/SidebarUtils.ts | 11 ++++++---- src/pages/home/sidebar/SidebarLinksData.js | 24 ++++++++++++++-------- 4 files changed, 31 insertions(+), 16 deletions(-) diff --git a/src/libs/OptionsListUtils.js b/src/libs/OptionsListUtils.js index 034dc401ae92..da21f9d308ef 100644 --- a/src/libs/OptionsListUtils.js +++ b/src/libs/OptionsListUtils.js @@ -1319,7 +1319,11 @@ function getOptions( // Filter out all the reports that shouldn't be displayed const filteredReports = _.filter(reports, (report) => { - const parentReportAction = !report.parentReportID || !report.parentReportActionID ? {} : lodashGet(allReportActions, [report.parentReportID, report.parentReportActionID], {}); + const {parentReportID, parentReportActionID} = report ?? {}; + if (!parentReportID || !parentReportActionID || !allReportActions) { + return false; + } + const parentReportAction = lodashGet(allReportActions, [parentReportID, parentReportActionID], {}); return ReportUtils.shouldReportBeInOptionList({ report, @@ -1327,7 +1331,7 @@ function getOptions( betas, policies, doesReportTransactionThreadHaveViolations: - betas.included(CONST.BETAS.VIOLATIONS) && + betas.includes(CONST.BETAS.VIOLATIONS) && ReportUtils.doesTransactionThreadHaveViolations({ report, transactionViolations, diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index 0ab76ef68842..bb70914e30be 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -3459,14 +3459,14 @@ function doesTransactionThreadHaveViolations({ if (parentReportAction.actionName !== CONST.REPORT.ACTIONS.TYPE.IOU) { return false; } - const {IOUTransactionID, IOUReportID} = parentReportAction?.originalMessage; + const {IOUTransactionID, IOUReportID} = parentReportAction.originalMessage ?? {}; if (!IOUTransactionID || !IOUReportID) { return false; } if (!isCurrentUserSubmitter(IOUReportID)) { return false; } - if (report?.stateNum !== CONST.REPORT.STATE_NUM.OPEN && report?.stateNum !== CONST.REPORT.STATE_NUM.PROCESSING) { + if (report.stateNum !== CONST.REPORT.STATE_NUM.OPEN && report.stateNum !== CONST.REPORT.STATE_NUM.PROCESSING) { return false; } return hasViolation(IOUTransactionID, transactionViolations); diff --git a/src/libs/SidebarUtils.ts b/src/libs/SidebarUtils.ts index 87d77b6f6053..a25ae0790df8 100644 --- a/src/libs/SidebarUtils.ts +++ b/src/libs/SidebarUtils.ts @@ -120,7 +120,7 @@ function getOrderedReportIDs( betas: Beta[], policies: Record, priorityMode: ValueOf, - allReportActions: OnyxCollection, + allReportActions: OnyxCollection, transactionViolations: OnyxCollection, ): string[] { const reportIDKey = `${ONYXKEYS.COLLECTION.REPORT_ACTIONS}${currentReportId}`; @@ -153,9 +153,12 @@ function getOrderedReportIDs( const allReportsDictValues = Object.values(allReports); // Filter out all the reports that shouldn't be displayed const reportsToDisplay = allReportsDictValues.filter((report) => { - const parentReportAction = !report.parentReportID || !report.parentReportActionID ? {} : lodashGet(allReportActions, [report.parentReportID, report.parentReportActionID], {}); - const doesReportTransactionThreadHaveViolations = - betas.includes(CONST.BETAS.VIOLATIONS) && ReportUtils.doesTransactionThreadHaveViolations({report, transactionViolations, parentReportAction}); + const parentReportActionsKey = `${ONYXKEYS.COLLECTION.REPORT_ACTIONS}${report?.parentReportID}`; + const parentReportActions = allReportActions?.[parentReportActionsKey]; + const parentReportAction = parentReportActions?.find((action) => action && report && action?.reportActionID === report?.parentReportActionID); + const doesReportTransactionThreadHaveViolations = Boolean( + betas.includes(CONST.BETAS.VIOLATIONS) && parentReportAction && ReportUtils.doesTransactionThreadHaveViolations({report, transactionViolations, parentReportAction}), + ); return ReportUtils.shouldReportBeInOptionList({ report, currentReportId: currentReportId ?? '', diff --git a/src/pages/home/sidebar/SidebarLinksData.js b/src/pages/home/sidebar/SidebarLinksData.js index ead72e841bde..b14097eedc93 100644 --- a/src/pages/home/sidebar/SidebarLinksData.js +++ b/src/pages/home/sidebar/SidebarLinksData.js @@ -204,14 +204,22 @@ const chatReportSelector = (report) => */ const reportActionsSelector = (reportActions) => reportActions && - _.map(reportActions, (reportAction) => ({ - errors: _.get(reportAction, 'errors', []), - message: [ - { - moderationDecision: {decision: _.get(reportAction, 'message[0].moderationDecision.decision')}, - }, - ], - })); + _.map(reportActions, (reportAction) => { + const {reportActionID, parentReportActionID, actionName, errors = []} = reportAction; + const decision = _.get(reportAction, 'message[0].moderationDecision.decision'); + + return { + reportActionID, + parentReportActionID, + actionName, + errors, + message: [ + { + moderationDecision: {decision}, + }, + ], + }; + }); /** * @param {Object} [policy] From 696865011a610a626985f87c8e04bde5984dc3cf Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Mon, 8 Jan 2024 18:01:01 -0500 Subject: [PATCH 097/117] feat(Violations): remove unneeded imports and lint --- src/libs/OptionsListUtils.js | 2 +- src/libs/SidebarUtils.ts | 1 - src/pages/NewChatPage.js | 5 +---- 3 files changed, 2 insertions(+), 6 deletions(-) diff --git a/src/libs/OptionsListUtils.js b/src/libs/OptionsListUtils.js index a7502100d30b..8dbb4bdba135 100644 --- a/src/libs/OptionsListUtils.js +++ b/src/libs/OptionsListUtils.js @@ -1319,7 +1319,7 @@ function getOptions( // Filter out all the reports that shouldn't be displayed const filteredReports = _.filter(reports, (report) => { - const {parentReportID, parentReportActionID} = report ?? {}; + const {parentReportID, parentReportActionID} = report || {}; if (!parentReportID || !parentReportActionID || !allReportActions) { return false; } diff --git a/src/libs/SidebarUtils.ts b/src/libs/SidebarUtils.ts index a25ae0790df8..605f866a82e0 100644 --- a/src/libs/SidebarUtils.ts +++ b/src/libs/SidebarUtils.ts @@ -1,6 +1,5 @@ /* eslint-disable rulesdir/prefer-underscore-method */ import Str from 'expensify-common/lib/str'; -import lodashGet from 'lodash/get'; import type {OnyxCollection} from 'react-native-onyx'; import Onyx from 'react-native-onyx'; import type {ValueOf} from 'type-fest'; diff --git a/src/pages/NewChatPage.js b/src/pages/NewChatPage.js index 049ca8b4ef84..9ef3e50e8d26 100755 --- a/src/pages/NewChatPage.js +++ b/src/pages/NewChatPage.js @@ -21,7 +21,6 @@ import variables from '@styles/variables'; import * as Report from '@userActions/Report'; import CONST from '@src/CONST'; import ONYXKEYS from '@src/ONYXKEYS'; -import reportActionPropTypes from './home/report/reportActionPropTypes'; import personalDetailsPropType from './personalDetailsPropType'; import reportPropTypes from './reportPropTypes'; @@ -41,7 +40,6 @@ const propTypes = { /** Whether we are searching for reports in the server */ isSearchingForReports: PropTypes.bool, - reportActions: PropTypes.objectOf(PropTypes.shape(reportActionPropTypes)), }; const defaultProps = { @@ -49,12 +47,11 @@ const defaultProps = { personalDetails: {}, reports: {}, isSearchingForReports: false, - reportActions: {}, }; const excludedGroupEmails = _.without(CONST.EXPENSIFY_EMAILS, CONST.EMAIL.CONCIERGE); -function NewChatPage({betas, isGroupChat, personalDetails, reports, translate, isSearchingForReports, reportActions}) { +function NewChatPage({betas, isGroupChat, personalDetails, reports, translate, isSearchingForReports}) { const styles = useThemeStyles(); const [searchTerm, setSearchTerm] = useState(''); const [filteredRecentReports, setFilteredRecentReports] = useState([]); From ad15ca6f4a29927f647369fe84cbba7b5155c117 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Mon, 8 Jan 2024 18:13:35 -0500 Subject: [PATCH 098/117] fix type --- tests/perf-test/SidebarUtils.perf-test.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/perf-test/SidebarUtils.perf-test.ts b/tests/perf-test/SidebarUtils.perf-test.ts index 2108e65f432e..9148a703d312 100644 --- a/tests/perf-test/SidebarUtils.perf-test.ts +++ b/tests/perf-test/SidebarUtils.perf-test.ts @@ -102,7 +102,7 @@ test('[SidebarUtils] getOrderedReportIDs on 5k reports', async () => { }, ], ]), - ) as unknown as OnyxCollection; + ) as unknown as OnyxCollection; Onyx.multiSet({ ...mockedResponseMap, From 85fa56d4462d7cb1208c615346e0b4744407b762 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Mon, 8 Jan 2024 18:18:41 -0500 Subject: [PATCH 099/117] feat(Violations): remove unneeded import --- tests/perf-test/SidebarUtils.perf-test.ts | 1 - 1 file changed, 1 deletion(-) diff --git a/tests/perf-test/SidebarUtils.perf-test.ts b/tests/perf-test/SidebarUtils.perf-test.ts index 9148a703d312..b51c9e04a187 100644 --- a/tests/perf-test/SidebarUtils.perf-test.ts +++ b/tests/perf-test/SidebarUtils.perf-test.ts @@ -7,7 +7,6 @@ import ONYXKEYS from '@src/ONYXKEYS'; import type {PersonalDetails, TransactionViolation} from '@src/types/onyx'; import type Policy from '@src/types/onyx/Policy'; import type Report from '@src/types/onyx/Report'; -import type {ReportActions} from '@src/types/onyx/ReportAction'; import type ReportAction from '@src/types/onyx/ReportAction'; import createCollection from '../utils/collections/createCollection'; import createPersonalDetails from '../utils/collections/personalDetails'; From 2452d31dfbf0ed3262638dd2620a490209603dd9 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Tue, 9 Jan 2024 11:32:17 -0500 Subject: [PATCH 100/117] fix filtering --- src/libs/OptionsListUtils.js | 7 ++----- 1 file changed, 2 insertions(+), 5 deletions(-) diff --git a/src/libs/OptionsListUtils.js b/src/libs/OptionsListUtils.js index 8dbb4bdba135..79d7ce210124 100644 --- a/src/libs/OptionsListUtils.js +++ b/src/libs/OptionsListUtils.js @@ -1320,10 +1320,7 @@ function getOptions( // Filter out all the reports that shouldn't be displayed const filteredReports = _.filter(reports, (report) => { const {parentReportID, parentReportActionID} = report || {}; - if (!parentReportID || !parentReportActionID || !allReportActions) { - return false; - } - const parentReportAction = lodashGet(allReportActions, [parentReportID, parentReportActionID], {}); + const canGetParentReport = parentReportID && parentReportActionID && allReportActions; return ReportUtils.shouldReportBeInOptionList({ report, @@ -1335,7 +1332,7 @@ function getOptions( ReportUtils.doesTransactionThreadHaveViolations({ report, transactionViolations, - parentReportAction, + parentReportAction: canGetParentReport ? lodashGet(allReportActions, [parentReportID, parentReportActionID], {}) : {}, }), }); }); From 192899c5d8efa9f50b0b62e2039651a50236d6c7 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Tue, 9 Jan 2024 17:16:43 -0500 Subject: [PATCH 101/117] feat(Violations): remove optional param --- src/libs/ReportUtils.ts | 2 +- src/libs/TransactionUtils.ts | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index db016c1983ed..224db77e393c 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -3454,7 +3454,7 @@ function doesTransactionThreadHaveViolations({ parentReportAction, }: { report: Report; - transactionViolations?: OnyxCollection; + transactionViolations: OnyxCollection; parentReportAction: ReportAction; }): boolean { if (parentReportAction.actionName !== CONST.REPORT.ACTIONS.TYPE.IOU) { diff --git a/src/libs/TransactionUtils.ts b/src/libs/TransactionUtils.ts index 0c7214d686f7..e81fb7969f79 100644 --- a/src/libs/TransactionUtils.ts +++ b/src/libs/TransactionUtils.ts @@ -524,7 +524,7 @@ function getRecentTransactions(transactions: Record, size = 2): /** * Checks if any violations for the provided transaction are of type 'violation' */ -function hasViolation(transactionID: string, transactionViolations: OnyxCollection | undefined): boolean { +function hasViolation(transactionID: string, transactionViolations: OnyxCollection): boolean { return Boolean(transactionViolations?.[ONYXKEYS.COLLECTION.TRANSACTION_VIOLATIONS + transactionID]?.some((violation: TransactionViolation) => violation.type === 'violation')); } From 0c41aecc239e59f98fcc9f8d62f393c4755e6668 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Tue, 9 Jan 2024 17:41:38 -0500 Subject: [PATCH 102/117] feat(Violations): remove check for undefined report --- src/libs/OptionsListUtils.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/libs/OptionsListUtils.js b/src/libs/OptionsListUtils.js index 79d7ce210124..96d23e72ebe3 100644 --- a/src/libs/OptionsListUtils.js +++ b/src/libs/OptionsListUtils.js @@ -1319,7 +1319,7 @@ function getOptions( // Filter out all the reports that shouldn't be displayed const filteredReports = _.filter(reports, (report) => { - const {parentReportID, parentReportActionID} = report || {}; + const {parentReportID, parentReportActionID} = report; const canGetParentReport = parentReportID && parentReportActionID && allReportActions; return ReportUtils.shouldReportBeInOptionList({ From 891c3d24d4dffbf75fbefabee6423e2609f1119b Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Tue, 9 Jan 2024 17:55:29 -0500 Subject: [PATCH 103/117] feat(Violations): PR review comments --- src/components/LHNOptionsList/OptionRowLHNData.js | 2 +- src/libs/OptionsListUtils.js | 13 +++++-------- src/libs/ReportUtils.ts | 10 +--------- 3 files changed, 7 insertions(+), 18 deletions(-) diff --git a/src/components/LHNOptionsList/OptionRowLHNData.js b/src/components/LHNOptionsList/OptionRowLHNData.js index 5134eed63bc1..bb473ede8701 100644 --- a/src/components/LHNOptionsList/OptionRowLHNData.js +++ b/src/components/LHNOptionsList/OptionRowLHNData.js @@ -92,7 +92,7 @@ function OptionRowLHNData({ // eslint-disable-next-line react-hooks/exhaustive-deps }, [fullReport.reportID, receiptTransactions, reportActions]); - const hasViolations = canUseViolations && ReportUtils.doesTransactionThreadHaveViolations({report: fullReport, transactionViolations, parentReportAction}); + const hasViolations = canUseViolations && ReportUtils.doesTransactionThreadHaveViolations(fullReport, transactionViolations, parentReportAction); const optionItem = useMemo(() => { // Note: ideally we'd have this as a dependent selector in onyx! diff --git a/src/libs/OptionsListUtils.js b/src/libs/OptionsListUtils.js index 96d23e72ebe3..60084511fe0d 100644 --- a/src/libs/OptionsListUtils.js +++ b/src/libs/OptionsListUtils.js @@ -1319,21 +1319,18 @@ function getOptions( // Filter out all the reports that shouldn't be displayed const filteredReports = _.filter(reports, (report) => { - const {parentReportID, parentReportActionID} = report; + const {parentReportID, parentReportActionID} = report || {}; const canGetParentReport = parentReportID && parentReportActionID && allReportActions; + const parentReportAction = canGetParentReport ? lodashGet(allReportActions, [parentReportID, parentReportActionID], {}) : {}; + const doesReportTransactionThreadHaveViolations = + betas.includes(CONST.BETAS.VIOLATIONS) && ReportUtils.doesTransactionThreadHaveViolations(report, transactionViolations, parentReportAction); return ReportUtils.shouldReportBeInOptionList({ report, currentReportId: Navigation.getTopmostReportId(), betas, policies, - doesReportTransactionThreadHaveViolations: - betas.includes(CONST.BETAS.VIOLATIONS) && - ReportUtils.doesTransactionThreadHaveViolations({ - report, - transactionViolations, - parentReportAction: canGetParentReport ? lodashGet(allReportActions, [parentReportID, parentReportActionID], {}) : {}, - }), + doesReportTransactionThreadHaveViolations, }); }); diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index 224db77e393c..4feb7c3f2c34 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -3448,15 +3448,7 @@ function shouldHideReport(report: OnyxEntry, currentReportId: string): b /** * Checks to see if a report's parentAction is a money request that contains a violation */ -function doesTransactionThreadHaveViolations({ - report, - transactionViolations, - parentReportAction, -}: { - report: Report; - transactionViolations: OnyxCollection; - parentReportAction: ReportAction; -}): boolean { +function doesTransactionThreadHaveViolations(report: Report, transactionViolations: OnyxCollection, parentReportAction: ReportAction): boolean { if (parentReportAction.actionName !== CONST.REPORT.ACTIONS.TYPE.IOU) { return false; } From 09d08e23a0a803dbf35bf827bf7ea7933a78952b Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Wed, 10 Jan 2024 11:07:59 -0500 Subject: [PATCH 104/117] feat(Violations): rename param --- src/libs/OptionsListUtils.js | 5 ++--- src/libs/ReportUtils.ts | 6 +++--- src/libs/SidebarUtils.ts | 7 +++---- src/libs/UnreadIndicatorUpdater/index.ts | 2 +- tests/perf-test/ReportUtils.perf-test.ts | 2 +- 5 files changed, 10 insertions(+), 12 deletions(-) diff --git a/src/libs/OptionsListUtils.js b/src/libs/OptionsListUtils.js index 60084511fe0d..a3a21ff078db 100644 --- a/src/libs/OptionsListUtils.js +++ b/src/libs/OptionsListUtils.js @@ -1322,15 +1322,14 @@ function getOptions( const {parentReportID, parentReportActionID} = report || {}; const canGetParentReport = parentReportID && parentReportActionID && allReportActions; const parentReportAction = canGetParentReport ? lodashGet(allReportActions, [parentReportID, parentReportActionID], {}) : {}; - const doesReportTransactionThreadHaveViolations = - betas.includes(CONST.BETAS.VIOLATIONS) && ReportUtils.doesTransactionThreadHaveViolations(report, transactionViolations, parentReportAction); + const doesReportHaveViolations = betas.includes(CONST.BETAS.VIOLATIONS) && ReportUtils.doesTransactionThreadHaveViolations(report, transactionViolations, parentReportAction); return ReportUtils.shouldReportBeInOptionList({ report, currentReportId: Navigation.getTopmostReportId(), betas, policies, - doesReportTransactionThreadHaveViolations, + doesReportHaveViolations, }); }); diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index 4feb7c3f2c34..a1a397e134a0 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -3487,7 +3487,7 @@ function shouldReportBeInOptionList({ betas, policies, excludeEmptyChats = false, - doesReportTransactionThreadHaveViolations, + doesReportHaveViolations, }: { report: OnyxEntry; currentReportId: string; @@ -3495,7 +3495,7 @@ function shouldReportBeInOptionList({ betas: Beta[]; policies: OnyxCollection; excludeEmptyChats?: boolean; - doesReportTransactionThreadHaveViolations: boolean; + doesReportHaveViolations: boolean; }) { const isInDefaultMode = !isInGSDMode; // Exclude reports that have no data because there wouldn't be anything to show in the option item. @@ -3556,7 +3556,7 @@ function shouldReportBeInOptionList({ } // Always show IOU reports with violations - if (isExpenseRequest(report) && doesReportTransactionThreadHaveViolations) { + if (isExpenseRequest(report) && doesReportHaveViolations) { return true; } diff --git a/src/libs/SidebarUtils.ts b/src/libs/SidebarUtils.ts index 605f866a82e0..04a4068e15f4 100644 --- a/src/libs/SidebarUtils.ts +++ b/src/libs/SidebarUtils.ts @@ -155,9 +155,8 @@ function getOrderedReportIDs( const parentReportActionsKey = `${ONYXKEYS.COLLECTION.REPORT_ACTIONS}${report?.parentReportID}`; const parentReportActions = allReportActions?.[parentReportActionsKey]; const parentReportAction = parentReportActions?.find((action) => action && report && action?.reportActionID === report?.parentReportActionID); - const doesReportTransactionThreadHaveViolations = Boolean( - betas.includes(CONST.BETAS.VIOLATIONS) && parentReportAction && ReportUtils.doesTransactionThreadHaveViolations({report, transactionViolations, parentReportAction}), - ); + const doesReportTransactionThreadHaveViolations = + betas.includes(CONST.BETAS.VIOLATIONS) && !!parentReportAction && ReportUtils.doesTransactionThreadHaveViolations(report, transactionViolations, parentReportAction); return ReportUtils.shouldReportBeInOptionList({ report, currentReportId: currentReportId ?? '', @@ -165,7 +164,7 @@ function getOrderedReportIDs( betas, policies, excludeEmptyChats: true, - doesReportTransactionThreadHaveViolations, + doesReportHaveViolations: doesReportTransactionThreadHaveViolations, }); }); diff --git a/src/libs/UnreadIndicatorUpdater/index.ts b/src/libs/UnreadIndicatorUpdater/index.ts index 5be9f0bbb268..973774eed3e8 100644 --- a/src/libs/UnreadIndicatorUpdater/index.ts +++ b/src/libs/UnreadIndicatorUpdater/index.ts @@ -22,7 +22,7 @@ const triggerUnreadUpdate = () => { currentReportId: currentReportID ?? '', betas: [], policies: {}, - doesReportTransactionThreadHaveViolations: false, + doesReportHaveViolations: false, }); }); updateUnread(unreadReports.length); diff --git a/tests/perf-test/ReportUtils.perf-test.ts b/tests/perf-test/ReportUtils.perf-test.ts index 14cf45545890..b5f9de3393af 100644 --- a/tests/perf-test/ReportUtils.perf-test.ts +++ b/tests/perf-test/ReportUtils.perf-test.ts @@ -159,7 +159,7 @@ test('[ReportUtils] shouldReportBeInOptionList on 1k participant', async () => { const policies = getMockedPolicies(); await waitForBatchedUpdates(); - await measureFunction(() => ReportUtils.shouldReportBeInOptionList({report, currentReportId, isInGSDMode, betas, policies, doesReportTransactionThreadHaveViolations: false}), {runs}); + await measureFunction(() => ReportUtils.shouldReportBeInOptionList({report, currentReportId, isInGSDMode, betas, policies, doesReportHaveViolations: false}), {runs}); }); test('[ReportUtils] getWorkspaceIcon on 5k policies', async () => { From ae525eb17f2c543af840aec5115006200a9bed6c Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Wed, 10 Jan 2024 11:08:51 -0500 Subject: [PATCH 105/117] feat(Violations): use !! instead of Boolean to allow type narrowing --- src/libs/OptionsListUtils.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/libs/OptionsListUtils.js b/src/libs/OptionsListUtils.js index a3a21ff078db..cabe3d352584 100644 --- a/src/libs/OptionsListUtils.js +++ b/src/libs/OptionsListUtils.js @@ -1319,7 +1319,7 @@ function getOptions( // Filter out all the reports that shouldn't be displayed const filteredReports = _.filter(reports, (report) => { - const {parentReportID, parentReportActionID} = report || {}; + const {parentReportID, parentReportActionID} = report; const canGetParentReport = parentReportID && parentReportActionID && allReportActions; const parentReportAction = canGetParentReport ? lodashGet(allReportActions, [parentReportID, parentReportActionID], {}) : {}; const doesReportHaveViolations = betas.includes(CONST.BETAS.VIOLATIONS) && ReportUtils.doesTransactionThreadHaveViolations(report, transactionViolations, parentReportAction); From 454732148fe1ef2e16ea0b70a7f1abee5a6081be Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Wed, 10 Jan 2024 11:19:38 -0500 Subject: [PATCH 106/117] feat(Violations): make all params required --- src/libs/OptionsListUtils.js | 2 ++ src/libs/ReportUtils.ts | 8 ++++---- src/libs/UnreadIndicatorUpdater/index.ts | 2 ++ tests/perf-test/ReportUtils.perf-test.ts | 4 +++- 4 files changed, 11 insertions(+), 5 deletions(-) diff --git a/src/libs/OptionsListUtils.js b/src/libs/OptionsListUtils.js index cabe3d352584..adb6d86f9f1b 100644 --- a/src/libs/OptionsListUtils.js +++ b/src/libs/OptionsListUtils.js @@ -1330,6 +1330,8 @@ function getOptions( betas, policies, doesReportHaveViolations, + isInGSDMode: false, + excludeEmptyChats: false, }); }); diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index a1a397e134a0..c855e9ae1bf2 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -3483,18 +3483,18 @@ function hasViolations(reportID: string, transactionViolations: OnyxCollection; currentReportId: string; - isInGSDMode?: boolean; + isInGSDMode: boolean; betas: Beta[]; policies: OnyxCollection; - excludeEmptyChats?: boolean; + excludeEmptyChats: boolean; doesReportHaveViolations: boolean; }) { const isInDefaultMode = !isInGSDMode; diff --git a/src/libs/UnreadIndicatorUpdater/index.ts b/src/libs/UnreadIndicatorUpdater/index.ts index 973774eed3e8..11472ce3e385 100644 --- a/src/libs/UnreadIndicatorUpdater/index.ts +++ b/src/libs/UnreadIndicatorUpdater/index.ts @@ -23,6 +23,8 @@ const triggerUnreadUpdate = () => { betas: [], policies: {}, doesReportHaveViolations: false, + isInGSDMode: false, + excludeEmptyChats: false, }); }); updateUnread(unreadReports.length); diff --git a/tests/perf-test/ReportUtils.perf-test.ts b/tests/perf-test/ReportUtils.perf-test.ts index b5f9de3393af..a9a92602de12 100644 --- a/tests/perf-test/ReportUtils.perf-test.ts +++ b/tests/perf-test/ReportUtils.perf-test.ts @@ -159,7 +159,9 @@ test('[ReportUtils] shouldReportBeInOptionList on 1k participant', async () => { const policies = getMockedPolicies(); await waitForBatchedUpdates(); - await measureFunction(() => ReportUtils.shouldReportBeInOptionList({report, currentReportId, isInGSDMode, betas, policies, doesReportHaveViolations: false}), {runs}); + await measureFunction(() => ReportUtils.shouldReportBeInOptionList({report, currentReportId, isInGSDMode, betas, policies, doesReportHaveViolations: false, excludeEmptyChats: false}), { + runs, + }); }); test('[ReportUtils] getWorkspaceIcon on 5k policies', async () => { From 3b66eeb691cdd8f37aac761579c2583481b96989 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Wed, 10 Jan 2024 11:19:50 -0500 Subject: [PATCH 107/117] feat(Violations): fix comment --- src/components/LHNOptionsList/OptionRowLHNData.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/components/LHNOptionsList/OptionRowLHNData.js b/src/components/LHNOptionsList/OptionRowLHNData.js index bb473ede8701..8bdf065a94fd 100644 --- a/src/components/LHNOptionsList/OptionRowLHNData.js +++ b/src/components/LHNOptionsList/OptionRowLHNData.js @@ -44,7 +44,7 @@ const propTypes = { /** The transaction from the parent report action */ transaction: transactionPropTypes, - /** Any violations associated with the report */ + /** Any violations associated with the transaction */ transactionViolations: transactionViolationsPropType, ...basePropTypes, From a5833d08bde7e5e3f663a18db3d6ecf072f15802 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Wed, 10 Jan 2024 12:35:41 -0500 Subject: [PATCH 108/117] feat(Violations): use shorthand --- src/libs/SidebarUtils.ts | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/libs/SidebarUtils.ts b/src/libs/SidebarUtils.ts index 04a4068e15f4..0ee16e4a0e96 100644 --- a/src/libs/SidebarUtils.ts +++ b/src/libs/SidebarUtils.ts @@ -155,7 +155,7 @@ function getOrderedReportIDs( const parentReportActionsKey = `${ONYXKEYS.COLLECTION.REPORT_ACTIONS}${report?.parentReportID}`; const parentReportActions = allReportActions?.[parentReportActionsKey]; const parentReportAction = parentReportActions?.find((action) => action && report && action?.reportActionID === report?.parentReportActionID); - const doesReportTransactionThreadHaveViolations = + const doesReportHaveViolations = betas.includes(CONST.BETAS.VIOLATIONS) && !!parentReportAction && ReportUtils.doesTransactionThreadHaveViolations(report, transactionViolations, parentReportAction); return ReportUtils.shouldReportBeInOptionList({ report, @@ -164,7 +164,7 @@ function getOrderedReportIDs( betas, policies, excludeEmptyChats: true, - doesReportHaveViolations: doesReportTransactionThreadHaveViolations, + doesReportHaveViolations, }); }); From 587a44c142b6434cb3890ce9c9160d3e64c6ee32 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Fri, 12 Jan 2024 13:42:08 -0500 Subject: [PATCH 109/117] feat(Violations): use lodash --- src/pages/home/sidebar/SidebarLinksData.js | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/src/pages/home/sidebar/SidebarLinksData.js b/src/pages/home/sidebar/SidebarLinksData.js index b14097eedc93..b8d45e147796 100644 --- a/src/pages/home/sidebar/SidebarLinksData.js +++ b/src/pages/home/sidebar/SidebarLinksData.js @@ -1,4 +1,6 @@ import {deepEqual} from 'fast-equals'; +import lodashGet from 'lodash/get'; +import lodashMap from 'lodash/map'; import PropTypes from 'prop-types'; import React, {useCallback, useMemo, useRef} from 'react'; import {View} from 'react-native'; @@ -204,9 +206,9 @@ const chatReportSelector = (report) => */ const reportActionsSelector = (reportActions) => reportActions && - _.map(reportActions, (reportAction) => { + lodashMap(reportActions, (reportAction) => { const {reportActionID, parentReportActionID, actionName, errors = []} = reportAction; - const decision = _.get(reportAction, 'message[0].moderationDecision.decision'); + const decision = lodashGet(reportAction, 'message[0].moderationDecision.decision'); return { reportActionID, From c732dddc4aa049f7b746724314a9cc47aae9e2f6 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Fri, 12 Jan 2024 13:57:36 -0500 Subject: [PATCH 110/117] feat(Violations): ensure reportActionCount defaults to 1 --- src/libs/SidebarUtils.ts | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/src/libs/SidebarUtils.ts b/src/libs/SidebarUtils.ts index 0ee16e4a0e96..c4f8124fb252 100644 --- a/src/libs/SidebarUtils.ts +++ b/src/libs/SidebarUtils.ts @@ -122,8 +122,9 @@ function getOrderedReportIDs( allReportActions: OnyxCollection, transactionViolations: OnyxCollection, ): string[] { - const reportIDKey = `${ONYXKEYS.COLLECTION.REPORT_ACTIONS}${currentReportId}`; - const reportActionCount = allReportActions?.[reportIDKey]?.length ?? 1; + const currentReportActions = allReportActions?.[`${ONYXKEYS.COLLECTION.REPORT_ACTIONS}${currentReportId}`]; + let reportActionCount = currentReportActions?.length ?? 0; + reportActionCount = Math.max(reportActionCount, 1); // Generate a unique cache key based on the function arguments const cachedReportsKey = JSON.stringify([currentReportId, allReports, betas, policies, priorityMode, reportActionCount], (key, value: unknown) => { From 3eaf3d46191cac9953137bcfe7c62db8a3399fbd Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Fri, 12 Jan 2024 13:58:09 -0500 Subject: [PATCH 111/117] feat(Violations): refactor for readability --- src/libs/SidebarUtils.ts | 16 +++++----------- 1 file changed, 5 insertions(+), 11 deletions(-) diff --git a/src/libs/SidebarUtils.ts b/src/libs/SidebarUtils.ts index c4f8124fb252..9d3a99982341 100644 --- a/src/libs/SidebarUtils.ts +++ b/src/libs/SidebarUtils.ts @@ -127,17 +127,11 @@ function getOrderedReportIDs( reportActionCount = Math.max(reportActionCount, 1); // Generate a unique cache key based on the function arguments - const cachedReportsKey = JSON.stringify([currentReportId, allReports, betas, policies, priorityMode, reportActionCount], (key, value: unknown) => { - /** - * Exclude some properties not to overwhelm a cached key value with huge data, - * which we don't need to store in a cacheKey - */ - if (key === 'participantAccountIDs' || key === 'participants' || key === 'lastMessageText' || key === 'visibleChatMemberAccountIDs') { - return undefined; - } - - return value; - }); + const cachedReportsKey = JSON.stringify( + [currentReportId, allReports, betas, policies, priorityMode, reportActionCount], + // Exclude some properties not to overwhelm a cached key value with huge data, which we don't need to store in a cacheKey + (key, value: unknown) => (['participantAccountIDs', 'participants', 'lastMessageText', 'visibleChatMemberAccountIDs'].includes(key) ? undefined : value), + ); // Check if the result is already in the cache const cachedIDs = reportIDsCache.get(cachedReportsKey); From 27e83158f8b53e8794b73f1a07d632a0a82c138c Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Fri, 12 Jan 2024 14:04:42 -0500 Subject: [PATCH 112/117] feat(Violations): restore guard --- src/libs/OptionsListUtils.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/libs/OptionsListUtils.js b/src/libs/OptionsListUtils.js index be9492100b5c..46ce9457915f 100644 --- a/src/libs/OptionsListUtils.js +++ b/src/libs/OptionsListUtils.js @@ -1360,7 +1360,7 @@ function getOptions( // Filter out all the reports that shouldn't be displayed const filteredReports = _.filter(reports, (report) => { - const {parentReportID, parentReportActionID} = report; + const {parentReportID, parentReportActionID} = report || {}; const canGetParentReport = parentReportID && parentReportActionID && allReportActions; const parentReportAction = canGetParentReport ? lodashGet(allReportActions, [parentReportID, parentReportActionID], {}) : {}; const doesReportHaveViolations = betas.includes(CONST.BETAS.VIOLATIONS) && ReportUtils.doesTransactionThreadHaveViolations(report, transactionViolations, parentReportAction); From 8e0250cb4f264bd954f85d579d644a0789a341d6 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Mon, 15 Jan 2024 12:22:06 -0500 Subject: [PATCH 113/117] feat(Violations): change STATE_NUM to reflect changes from 54e7f76d9f5ede5acb255813a67e1c3f3f857fde and a5d07180966c77bdd0d2e5d3e358d8a6911161c0 --- src/libs/ReportUtils.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index 8425bfee70f6..fe54084e7ed1 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -3493,7 +3493,7 @@ function doesTransactionThreadHaveViolations(report: Report, transactionViolatio if (!isCurrentUserSubmitter(IOUReportID)) { return false; } - if (report.stateNum !== CONST.REPORT.STATE_NUM.OPEN && report.stateNum !== CONST.REPORT.STATE_NUM.PROCESSING) { + if (report.stateNum !== CONST.REPORT.STATE_NUM.OPEN && report.stateNum !== CONST.REPORT.STATE_NUM.SUBMITTED) { return false; } return hasViolation(IOUTransactionID, transactionViolations); From 00fefc1c7a0148b47e24b0e70cd55725125cdf63 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Mon, 15 Jan 2024 12:30:30 -0500 Subject: [PATCH 114/117] prettier --- tests/perf-test/ReportUtils.perf-test.ts | 9 ++++--- tests/perf-test/SidebarUtils.perf-test.ts | 32 ++++++++++++----------- 2 files changed, 23 insertions(+), 18 deletions(-) diff --git a/tests/perf-test/ReportUtils.perf-test.ts b/tests/perf-test/ReportUtils.perf-test.ts index 4a007a816430..953fc88a99cf 100644 --- a/tests/perf-test/ReportUtils.perf-test.ts +++ b/tests/perf-test/ReportUtils.perf-test.ts @@ -142,9 +142,12 @@ describe('ReportUtils', () => { const policies = getMockedPolicies(); await waitForBatchedUpdates(); - await measureFunction(() => ReportUtils.shouldReportBeInOptionList({report, currentReportId, isInGSDMode, betas, policies, doesReportHaveViolations: false, excludeEmptyChats: false}), { - runs, - }); + await measureFunction( + () => ReportUtils.shouldReportBeInOptionList({report, currentReportId, isInGSDMode, betas, policies, doesReportHaveViolations: false, excludeEmptyChats: false}), + { + runs, + }, + ); }); test('[ReportUtils] getWorkspaceIcon on 5k policies', async () => { diff --git a/tests/perf-test/SidebarUtils.perf-test.ts b/tests/perf-test/SidebarUtils.perf-test.ts index 7874ce4876d1..7b2fd873f3de 100644 --- a/tests/perf-test/SidebarUtils.perf-test.ts +++ b/tests/perf-test/SidebarUtils.perf-test.ts @@ -59,26 +59,26 @@ describe('SidebarUtils', () => { await waitForBatchedUpdates(); - await measureFunction( - () => - SidebarUtils.getOptionData({ - report, - reportActions, - personalDetails, - preferredLocale, - policy, - parentReportAction, - hasViolations: false, - }), - {runs}, - ); + await measureFunction( + () => + SidebarUtils.getOptionData({ + report, + reportActions, + personalDetails, + preferredLocale, + policy, + parentReportAction, + hasViolations: false, + }), + {runs}, + ); }); test('[SidebarUtils] getOrderedReportIDs on 5k reports', async () => { const currentReportId = '1'; const allReports = getMockedReports(); const betas = [CONST.BETAS.DEFAULT_ROOMS]; - const transactionViolations = {} as OnyxCollection; + const transactionViolations = {} as OnyxCollection; const policies = createCollection( (item) => `${ONYXKEYS.COLLECTION.POLICY}${item.id}`, @@ -104,6 +104,8 @@ describe('SidebarUtils', () => { ) as unknown as OnyxCollection; await waitForBatchedUpdates(); - await measureFunction(() => SidebarUtils.getOrderedReportIDs(currentReportId, allReports, betas, policies, CONST.PRIORITY_MODE.DEFAULT, allReportActions, transactionViolations), {runs}); + await measureFunction(() => SidebarUtils.getOrderedReportIDs(currentReportId, allReports, betas, policies, CONST.PRIORITY_MODE.DEFAULT, allReportActions, transactionViolations), { + runs, + }); }); }); From e3882e7609316901f68d9a8771835ae90c069d9d Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Mon, 15 Jan 2024 15:47:32 -0500 Subject: [PATCH 115/117] Update src/libs/ReportUtils.ts Co-authored-by: Carlos Alvarez --- src/libs/ReportUtils.ts | 15 ++++++++++++++- 1 file changed, 14 insertions(+), 1 deletion(-) diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index 6129ea86038a..d37f7b31c758 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -14,7 +14,20 @@ import CONST from '@src/CONST'; import type {ParentNavigationSummaryParams, TranslationPaths} from '@src/languages/types'; import ONYXKEYS from '@src/ONYXKEYS'; import ROUTES from '@src/ROUTES'; -import type {Beta, Login, PersonalDetails, PersonalDetailsList, Policy, PolicyReportField, Report, ReportAction, ReportMetadata, Session, Transaction, TransactionViolation} from '@src/types/onyx'; +import type { + Beta, + Login, + PersonalDetails, + PersonalDetailsList, + Policy, + PolicyReportField, + Report, + ReportAction, + ReportMetadata, + Session, + Transaction, + TransactionViolation, +} from '@src/types/onyx'; import type {Errors, Icon, PendingAction} from '@src/types/onyx/OnyxCommon'; import type {IOUMessage, OriginalMessageActionName, OriginalMessageCreated, ReimbursementDeQueuedMessage} from '@src/types/onyx/OriginalMessage'; import type {Status} from '@src/types/onyx/PersonalDetails'; From 64ab957f3cb7fe5250fb0017ff12fb8e27eb6544 Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Tue, 16 Jan 2024 15:42:41 -0500 Subject: [PATCH 116/117] feat(Violations): clarify import from TransactionUtils --- src/libs/ReportUtils.ts | 7 +++---- 1 file changed, 3 insertions(+), 4 deletions(-) diff --git a/src/libs/ReportUtils.ts b/src/libs/ReportUtils.ts index d37f7b31c758..af51f123ac84 100644 --- a/src/libs/ReportUtils.ts +++ b/src/libs/ReportUtils.ts @@ -50,10 +50,9 @@ import * as NumberUtils from './NumberUtils'; import Permissions from './Permissions'; import * as PersonalDetailsUtils from './PersonalDetailsUtils'; import * as PolicyUtils from './PolicyUtils'; -import * as ReportActionsUtils from './ReportActionsUtils'; import type {LastVisibleMessage} from './ReportActionsUtils'; +import * as ReportActionsUtils from './ReportActionsUtils'; import * as TransactionUtils from './TransactionUtils'; -import {hasViolation} from './TransactionUtils'; import * as Url from './Url'; import * as UserUtils from './UserUtils'; @@ -3509,7 +3508,7 @@ function doesTransactionThreadHaveViolations(report: Report, transactionViolatio if (report.stateNum !== CONST.REPORT.STATE_NUM.OPEN && report.stateNum !== CONST.REPORT.STATE_NUM.SUBMITTED) { return false; } - return hasViolation(IOUTransactionID, transactionViolations); + return TransactionUtils.hasViolation(IOUTransactionID, transactionViolations); } /** @@ -3517,7 +3516,7 @@ function doesTransactionThreadHaveViolations(report: Report, transactionViolatio */ function hasViolations(reportID: string, transactionViolations: OnyxCollection): boolean { const transactions = TransactionUtils.getAllReportTransactions(reportID); - return transactions.some((transaction) => hasViolation(transaction.transactionID, transactionViolations)); + return transactions.some((transaction) => TransactionUtils.hasViolation(transaction.transactionID, transactionViolations)); } /** From 7305a48a7cb88067074e8094643557ae8c9a1ccf Mon Sep 17 00:00:00 2001 From: Trevor Coleman Date: Thu, 18 Jan 2024 11:37:12 -0500 Subject: [PATCH 117/117] feat(Violations): fix const/let --- src/libs/SidebarUtils.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/libs/SidebarUtils.ts b/src/libs/SidebarUtils.ts index 8b29a68bfe03..445d9dc30dd8 100644 --- a/src/libs/SidebarUtils.ts +++ b/src/libs/SidebarUtils.ts @@ -148,7 +148,7 @@ function getOrderedReportIDs( const isInDefaultMode = !isInGSDMode; const allReportsDictValues = Object.values(allReports); // Filter out all the reports that shouldn't be displayed - const reportsToDisplay = allReportsDictValues.filter((report) => { + let reportsToDisplay = allReportsDictValues.filter((report) => { const parentReportActionsKey = `${ONYXKEYS.COLLECTION.REPORT_ACTIONS}${report?.parentReportID}`; const parentReportActions = allReportActions?.[parentReportActionsKey]; const parentReportAction = parentReportActions?.find((action) => action && report && action?.reportActionID === report?.parentReportActionID);