Skip to content
2 changes: 1 addition & 1 deletion src/ROUTES.ts
Original file line number Diff line number Diff line change
Expand Up @@ -64,7 +64,7 @@ const ROUTES = {

// eslint-disable-next-line no-restricted-syntax -- Legacy route generation
WORKSPACES_LIST: {route: 'workspaces', getRoute: (backTo?: string) => getUrlWithBackToParam('workspaces', backTo)},

SEARCH_ROUTER: 'search-router',
Copy link
Member

Choose a reason for hiding this comment

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

Routing was not correct in this PR, which caused #80670

SEARCH_ROOT: {
route: 'search',
getRoute: ({query, rawQuery, name}: {query: SearchQueryString; rawQuery?: SearchQueryString; name?: string}) => {
Expand Down
1 change: 1 addition & 0 deletions src/SCREENS.ts
Original file line number Diff line number Diff line change
Expand Up @@ -243,6 +243,7 @@ const SCREENS = {
RIGHT_MODAL: {
SETTINGS: 'Settings',
TWO_FACTOR_AUTH: 'TwoFactorAuth',
SEARCH_ROUTER: 'Search_Router',
NEW_CHAT: 'NewChat',
DETAILS: 'Details',
PROFILE: 'Profile',
Expand Down
70 changes: 16 additions & 54 deletions src/components/Search/SearchRouter/SearchRouter.tsx
Original file line number Diff line number Diff line change
@@ -1,9 +1,7 @@
import {findFocusedRoute} from '@react-navigation/native';
import {deepEqual} from 'fast-equals';
import React, {useCallback, useEffect, useMemo, useRef, useState} from 'react';
import type {TextInputProps} from 'react-native';
import {InteractionManager, View} from 'react-native';
import type {OnyxCollection} from 'react-native-onyx';
import type {ValueOf} from 'type-fest';
import HeaderWithBackButton from '@components/HeaderWithBackButton';
import {usePersonalDetails} from '@components/OnyxListItemProvider';
Expand Down Expand Up @@ -38,57 +36,21 @@ import {getReportAction} from '@libs/ReportActionsUtils';
import {getReportOrDraftReport} from '@libs/ReportUtils';
import type {OptionData} from '@libs/ReportUtils';
import {getAutocompleteQueryWithComma, getQueryWithoutAutocompletedPart} from '@libs/SearchAutocompleteUtils';
import {getPolicyNameWithFallback, getQueryWithUpdatedValues, sanitizeSearchValue} from '@libs/SearchQueryUtils';
import {getQueryWithUpdatedValues, sanitizeSearchValue} from '@libs/SearchQueryUtils';
import StringUtils from '@libs/StringUtils';
import Navigation from '@navigation/Navigation';
import type {ReportsSplitNavigatorParamList} from '@navigation/types';
import variables from '@styles/variables';
import {navigateToAndOpenReport, searchInServer} from '@userActions/Report';
import {setSearchContext} from '@userActions/Search';
import CONST, {CONTINUATION_DETECTION_SEARCH_FILTER_KEYS} from '@src/CONST';
import ONYXKEYS from '@src/ONYXKEYS';
import ROUTES from '@src/ROUTES';
import SCREENS from '@src/SCREENS';
import type * as OnyxTypes from '@src/types/onyx';
import type Report from '@src/types/onyx/Report';
import isLoadingOnyxValue from '@src/types/utils/isLoadingOnyxValue';
import type {SubstitutionMap} from './getQueryWithSubstitutions';
import {getQueryWithSubstitutions} from './getQueryWithSubstitutions';
import {getUpdatedSubstitutionsMap} from './getUpdatedSubstitutionsMap';

function getContextualSearchAutocompleteKey(item: SearchQueryItem, policies: OnyxCollection<OnyxTypes.Policy>, reports?: OnyxCollection<OnyxTypes.Report>) {
if (item.roomType === CONST.SEARCH.DATA_TYPES.INVOICE) {
return `${CONST.SEARCH.SYNTAX_FILTER_KEYS.TO}:${item.searchQuery}`;
}
if (item.roomType === CONST.SEARCH.DATA_TYPES.CHAT) {
return `${CONST.SEARCH.SYNTAX_FILTER_KEYS.IN}:${item.searchQuery}`;
}
if (item.roomType === CONST.SEARCH.DATA_TYPES.EXPENSE) {
return `${CONST.SEARCH.SYNTAX_FILTER_KEYS.POLICY_ID}:${item.policyID ? getPolicyNameWithFallback(item.policyID, policies, reports) : ''}`;
}
}

function getContextualSearchQuery(item: SearchQueryItem, policies: OnyxCollection<OnyxTypes.Policy>, reports?: OnyxCollection<OnyxTypes.Report>) {
const baseQuery = `${CONST.SEARCH.SEARCH_USER_FRIENDLY_KEYS.TYPE}:${item.roomType}`;
let additionalQuery = '';

switch (item.roomType) {
case CONST.SEARCH.DATA_TYPES.EXPENSE:
additionalQuery += ` ${CONST.SEARCH.SEARCH_USER_FRIENDLY_KEYS.POLICY_ID}:${sanitizeSearchValue(item.policyID ? getPolicyNameWithFallback(item.policyID, policies, reports) : '')}`;
break;
case CONST.SEARCH.DATA_TYPES.INVOICE:
additionalQuery += ` ${CONST.SEARCH.SEARCH_USER_FRIENDLY_KEYS.POLICY_ID}:${item.policyID}`;
if (item.autocompleteID) {
additionalQuery += ` ${CONST.SEARCH.SEARCH_USER_FRIENDLY_KEYS.TO}:${sanitizeSearchValue(item.searchQuery ?? '')}`;
}
break;
case CONST.SEARCH.DATA_TYPES.CHAT:
default:
additionalQuery = ` ${CONST.SEARCH.SEARCH_USER_FRIENDLY_KEYS.IN}:${sanitizeSearchValue(item.searchQuery ?? '')}`;
break;
}
return baseQuery + additionalQuery;
}
import {getContextualReportData, getContextualSearchAutocompleteKey, getContextualSearchQuery} from './SearchRouterUtils';

type SearchRouterProps = {
onRouterClose: () => void;
Expand Down Expand Up @@ -127,18 +89,7 @@ function SearchRouter({onRouterClose, shouldHideInputCaret, isSearchRouterDispla
const [autocompleteSubstitutions, setAutocompleteSubstitutions] = useState<SubstitutionMap>({});
const textInputRef = useRef<AnimatedTextInputRef>(null);

const contextualReportID = useRootNavigationState((state) => {
// Safe handling when navigation is not yet initialized
if (!state) {
return undefined;
}

const focusedRoute = findFocusedRoute(state);
if (focusedRoute?.name === SCREENS.REPORT || focusedRoute?.name === SCREENS.RIGHT_MODAL.EXPENSE_REPORT) {
// We're guaranteed that the type of params is of SCREENS.REPORT
return (focusedRoute.params as ReportsSplitNavigatorParamList[typeof SCREENS.REPORT]).reportID;
}
});
const {contextualReportID, isSearchRouterScreen} = useRootNavigationState(getContextualReportData);

const getAdditionalSections: GetAdditionalSectionsCallback = useCallback(
({recentReports}) => {
Expand All @@ -151,7 +102,7 @@ function SearchRouter({onRouterClose, shouldHideInputCaret, isSearchRouterDispla
return undefined;
}

if (!isSearchRouterDisplayed) {
if (!isSearchRouterDisplayed && !isSearchRouterScreen) {
return undefined;
}
let reportForContextualSearch = recentReports.find((option) => option.reportID === contextualReportID);
Expand Down Expand Up @@ -215,7 +166,18 @@ function SearchRouter({onRouterClose, shouldHideInputCaret, isSearchRouterDispla
},
];
},
[contextualReportID, textInputValue, isSearchRouterDisplayed, translate, expensifyIcons.MagnifyingGlass, styles.activeComponentBG, reports, personalDetails, currentUserAccountID],
[
contextualReportID,
textInputValue,
isSearchRouterDisplayed,
isSearchRouterScreen,
translate,
expensifyIcons.MagnifyingGlass,
styles.activeComponentBG,
reports,
personalDetails,
currentUserAccountID,
],
);

const searchQueryItem = textInputValue
Expand Down
7 changes: 4 additions & 3 deletions src/components/Search/SearchRouter/SearchRouterContext.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ import CONST from '@src/CONST';
import NAVIGATORS from '@src/NAVIGATORS';
import SCREENS from '@src/SCREENS';
import type ChildrenProps from '@src/types/utils/ChildrenProps';
import {closeSearch, openSearch} from './toggleSearch';

type SearchRouterStateContextType = {
isSearchRouterDisplayed: boolean;
Expand Down Expand Up @@ -44,7 +45,6 @@ function SearchRouterContextProvider({children}: ChildrenProps) {
const [isSearchRouterDisplayed, setIsSearchRouterDisplayed] = useState(false);
const searchRouterDisplayedRef = useRef(false);
const searchPageInputRef = useRef<AnimatedTextInputRef | undefined>(undefined);

useEffect(() => {
if (!canListenPopState) {
return;
Expand Down Expand Up @@ -79,15 +79,16 @@ function SearchRouterContextProvider({children}: ChildrenProps) {
}
close(
() => {
setIsSearchRouterDisplayed(true);
openSearch(setIsSearchRouterDisplayed);
searchRouterDisplayedRef.current = true;
},
false,
true,
);
};

const closeSearchRouter = () => {
setIsSearchRouterDisplayed(false);
closeSearch(setIsSearchRouterDisplayed);
searchRouterDisplayedRef.current = false;
if (isBrowserWithHistory) {
const state = window.history.state as HistoryState | null;
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
/**
* On native devices SearchRouter is served from SearchRouterPage, on web from SearchRouterModal.
*/
function SearchRouterModal() {
return null;
}

export default SearchRouterModal;
Original file line number Diff line number Diff line change
Expand Up @@ -3,12 +3,12 @@ import {Dimensions} from 'react-native';
import FocusTrapForModal from '@components/FocusTrap/FocusTrapForModal';
import Modal from '@components/Modal';
import ScreenWrapperContainer from '@components/ScreenWrapper/ScreenWrapperContainer';
import SearchRouter from '@components/Search/SearchRouter/SearchRouter';
import {useSearchRouterActions, useSearchRouterState} from '@components/Search/SearchRouter/SearchRouterContext';
import useResponsiveLayout from '@hooks/useResponsiveLayout';
import useViewportOffsetTop from '@hooks/useViewportOffsetTop';
import {isMobileIOS} from '@libs/Browser';
import CONST from '@src/CONST';
import SearchRouter from './SearchRouter';
import {useSearchRouterActions, useSearchRouterState} from './SearchRouterContext';

const isMobileWebIOS = isMobileIOS();

Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,27 @@
import React from 'react';
import ScreenWrapper from '@components/ScreenWrapper';
import SearchRouter from '@components/Search/SearchRouter/SearchRouter';
import {useSearchRouterActions, useSearchRouterState} from '@components/Search/SearchRouter/SearchRouterContext';

function SearchRouterPage() {
const {closeSearchRouter} = useSearchRouterActions();
const {isSearchRouterDisplayed} = useSearchRouterState();

return (
<ScreenWrapper
testID="SearchRouterPage"
shouldEnableMaxHeight
enableEdgeToEdgeBottomSafeAreaPadding
includePaddingTop
includeSafeAreaPaddingBottom
>
<SearchRouter
onRouterClose={closeSearchRouter}
shouldHideInputCaret={false}
isSearchRouterDisplayed={isSearchRouterDisplayed}
/>
</ScreenWrapper>
);
}

export default SearchRouterPage;
11 changes: 11 additions & 0 deletions src/components/Search/SearchRouter/SearchRouterPage/index.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
import Navigation from '@libs/Navigation/Navigation';
import ROUTES from '@src/ROUTES';

/**
* On native devices SearchRouter is served from SearchRouterPage, on web from SearchRouterModal.
*/
function SearchRouterPage() {
return Navigation.navigate(ROUTES.HOME);
}

export default SearchRouterPage;
89 changes: 89 additions & 0 deletions src/components/Search/SearchRouter/SearchRouterUtils.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,89 @@
import {findFocusedRoute} from '@react-navigation/native';
import type {NavigationState} from '@react-navigation/routers';
import type {OnyxCollection} from 'react-native-onyx';
import type {SearchQueryItem} from '@components/SelectionListWithSections/Search/SearchQueryListItem';
import {getPolicyNameWithFallback, sanitizeSearchValue} from '@libs/SearchQueryUtils';
import type {ReportsSplitNavigatorParamList} from '@navigation/types';
import CONST from '@src/CONST';
import SCREENS from '@src/SCREENS';
import type * as OnyxTypes from '@src/types/onyx';

type ContextualReportData = {
contextualReportID: string | undefined;
isSearchRouterScreen: boolean;
};

/**
* Extracts contextual report data from the navigation state.
*
* This function determines:
* 1. Whether the current screen is the SearchRouter modal
* 2. The reportID of the contextual report (if any) that was focused before opening the SearchRouter
*
* When the SearchRouter is open, it looks at the previous route in the stack to find the underlying
* report context. Otherwise it looks @ current screen for report context.
* This allows the search to provide contextual suggestions based on the report
* the user was viewing when they opened the search.
*
* @param state - The root navigation state from useRootNavigationState hook
* @returns Object containing contextualReportID (the report ID if on a report screen) and isSearchRouterScreen (whether SearchRouter is focused)
*/
function getContextualReportData(state: NavigationState | undefined): ContextualReportData {
Copy link
Member

Choose a reason for hiding this comment

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

Can we use depend on findLastAccessedReport?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

I was thinking about it but as far as I understand how it works it may produce false positives / or make logic more complex.

  1. If we use just findLastAccessedReport - there is no guarantee that we came to search directly from a report -> false positives.
  2. If we use it in combination with current logic adding more complexity, in the point we know that we want to show the additional content we have report id in params already

based on that I chose current solution

// Safe handling when navigation is not yet initialized
if (!state) {
return {contextualReportID: undefined, isSearchRouterScreen: false};
}
let maybeReportRoute = findFocusedRoute(state);
const isSearchRouterScreen = maybeReportRoute?.name === SCREENS.RIGHT_MODAL.SEARCH_ROUTER;
if (isSearchRouterScreen) {
const stateWithoutLastRoute = {
...state,
routes: state.routes.slice(0, -1),
index: state.index !== 0 ? state.index - 1 : 0,
};
maybeReportRoute = findFocusedRoute(stateWithoutLastRoute);
}

if (maybeReportRoute?.name === SCREENS.REPORT || maybeReportRoute?.name === SCREENS.RIGHT_MODAL.EXPENSE_REPORT) {
// We're guaranteed that the type of params is of SCREENS.REPORT
return {contextualReportID: (maybeReportRoute?.params as ReportsSplitNavigatorParamList[typeof SCREENS.REPORT]).reportID, isSearchRouterScreen};
}
return {contextualReportID: undefined, isSearchRouterScreen};
}

function getContextualSearchAutocompleteKey(item: SearchQueryItem, policies: OnyxCollection<OnyxTypes.Policy>, reports?: OnyxCollection<OnyxTypes.Report>) {
if (item.roomType === CONST.SEARCH.DATA_TYPES.INVOICE) {
return `${CONST.SEARCH.SYNTAX_FILTER_KEYS.TO}:${item.searchQuery}`;
}
if (item.roomType === CONST.SEARCH.DATA_TYPES.CHAT) {
return `${CONST.SEARCH.SYNTAX_FILTER_KEYS.IN}:${item.searchQuery}`;
}
if (item.roomType === CONST.SEARCH.DATA_TYPES.EXPENSE) {
return `${CONST.SEARCH.SYNTAX_FILTER_KEYS.POLICY_ID}:${item.policyID ? getPolicyNameWithFallback(item.policyID, policies, reports) : ''}`;
}
}

function getContextualSearchQuery(item: SearchQueryItem, policies: OnyxCollection<OnyxTypes.Policy>, reports?: OnyxCollection<OnyxTypes.Report>) {
const baseQuery = `${CONST.SEARCH.SEARCH_USER_FRIENDLY_KEYS.TYPE}:${item.roomType}`;
let additionalQuery = '';

switch (item.roomType) {
case CONST.SEARCH.DATA_TYPES.EXPENSE:
additionalQuery += ` ${CONST.SEARCH.SEARCH_USER_FRIENDLY_KEYS.POLICY_ID}:${sanitizeSearchValue(item.policyID ? getPolicyNameWithFallback(item.policyID, policies, reports) : '')}`;
break;
case CONST.SEARCH.DATA_TYPES.INVOICE:
additionalQuery += ` ${CONST.SEARCH.SEARCH_USER_FRIENDLY_KEYS.POLICY_ID}:${item.policyID}`;
if (item.autocompleteID) {
additionalQuery += ` ${CONST.SEARCH.SEARCH_USER_FRIENDLY_KEYS.TO}:${sanitizeSearchValue(item.searchQuery ?? '')}`;
}
break;
case CONST.SEARCH.DATA_TYPES.CHAT:
default:
additionalQuery = ` ${CONST.SEARCH.SEARCH_USER_FRIENDLY_KEYS.IN}:${sanitizeSearchValue(item.searchQuery ?? '')}`;
break;
}
return baseQuery + additionalQuery;
}

export {getContextualReportData, getContextualSearchAutocompleteKey, getContextualSearchQuery};
export type {ContextualReportData};
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
import Navigation from '@libs/Navigation/Navigation';
import ROUTES from '@src/ROUTES';

function openSearch() {
return Navigation.navigate(ROUTES.SEARCH_ROUTER);
}

function closeSearch() {
return Navigation.dismissModal();
}

export {openSearch, closeSearch};
9 changes: 9 additions & 0 deletions src/components/Search/SearchRouter/toggleSearch/index.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
function openSearch(setSearchState: React.Dispatch<React.SetStateAction<boolean>>) {
return setSearchState(true);
}

function closeSearch(setSearchState: React.Dispatch<React.SetStateAction<boolean>>) {
return setSearchState(false);
}

export {openSearch, closeSearch};
Original file line number Diff line number Diff line change
Expand Up @@ -853,6 +853,10 @@ const TwoFactorAuthenticatorStackNavigator = createModalStackNavigator<EnablePay
[SCREENS.TWO_FACTOR_AUTH.SUCCESS]: () => require<ReactComponentModule>('../../../../pages/settings/Security/TwoFactorAuth/SuccessPage').default,
});

const SearchRouterModalStackNavigator = createModalStackNavigator({
[SCREENS.RIGHT_MODAL.SEARCH_ROUTER]: () => require<ReactComponentModule>('../../../../components/Search/SearchRouter/SearchRouterPage').default,
});

const EnablePaymentsStackNavigator = createModalStackNavigator<EnablePaymentsNavigatorParamList>({
[SCREENS.ENABLE_PAYMENTS_ROOT]: () => require<ReactComponentModule>('../../../../pages/EnablePayments/EnablePaymentsPage').default,
});
Expand Down Expand Up @@ -1068,4 +1072,5 @@ export {
WorkspaceDuplicateModalStackNavigator,
WorkspacesDomainModalStackNavigator,
MultifactorAuthenticationStackNavigator,
SearchRouterModalStackNavigator,
};
Original file line number Diff line number Diff line change
Expand Up @@ -185,6 +185,10 @@ function RightModalNavigator({navigation, route}: RightModalNavigatorProps) {
screenListeners={screenListeners}
id={NAVIGATORS.RIGHT_MODAL_NAVIGATOR}
>
<Stack.Screen
name={SCREENS.RIGHT_MODAL.SEARCH_ROUTER}
component={ModalStackNavigators.SearchRouterModalStackNavigator}
/>
<Stack.Screen
name={SCREENS.RIGHT_MODAL.SETTINGS}
component={ModalStackNavigators.SettingsModalStackNavigator}
Expand Down
4 changes: 4 additions & 0 deletions src/libs/Navigation/linkingConfig/config.ts
Original file line number Diff line number Diff line change
Expand Up @@ -155,6 +155,10 @@ const config: LinkingOptions<RootNavigatorParamList>['config'] = {
},
[NAVIGATORS.RIGHT_MODAL_NAVIGATOR]: {
screens: {
[SCREENS.RIGHT_MODAL.SEARCH_ROUTER]: {
path: ROUTES.SEARCH_ROUTER,
exact: true,
},
[SCREENS.RIGHT_MODAL.SETTINGS]: {
screens: {
[SCREENS.SETTINGS.PREFERENCES.PRIORITY_MODE]: {
Expand Down
1 change: 1 addition & 0 deletions src/libs/Navigation/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2283,6 +2283,7 @@ type WorkspacesDomainModalNavigatorParamList = {
type RightModalNavigatorParamList = {
[SCREENS.RIGHT_MODAL.SETTINGS]: NavigatorScreenParams<SettingsNavigatorParamList>;
[SCREENS.RIGHT_MODAL.TWO_FACTOR_AUTH]: NavigatorScreenParams<TwoFactorAuthNavigatorParamList>;
[SCREENS.RIGHT_MODAL.SEARCH_ROUTER]: undefined;
[SCREENS.RIGHT_MODAL.NEW_CHAT]: NavigatorScreenParams<NewChatNavigatorParamList>;
[SCREENS.RIGHT_MODAL.DETAILS]: NavigatorScreenParams<DetailsNavigatorParamList>;
[SCREENS.RIGHT_MODAL.PROFILE]: NavigatorScreenParams<ProfileNavigatorParamList>;
Expand Down
Loading
Loading