diff --git a/src/api.ts b/src/api.ts index 06f3f93786..be4b919a1d 100644 --- a/src/api.ts +++ b/src/api.ts @@ -11,7 +11,6 @@ import { makeAudioTimestampsUrl, makeChapterAudioDataUrl, makeAvailableRecitersUrl, - makeSearchResultsUrl, makeTranslationsInfoUrl, makeTranslationsUrl, makeVersesUrl, @@ -31,10 +30,9 @@ import { } from '@/utils/apiPaths'; import generateSignature from '@/utils/auth/signature'; import { isStaticBuild } from '@/utils/build'; -import { SearchRequest, AdvancedCopyRequest, PagesLookUpRequest } from 'types/ApiRequests'; +import { AdvancedCopyRequest, PagesLookUpRequest } from 'types/ApiRequests'; import { TranslationsResponse, - SearchResponse, AdvancedCopyRawResultResponse, LanguagesResponse, RecitersResponse, @@ -244,15 +242,6 @@ export const getAdvancedCopyRawResult = async ( params: AdvancedCopyRequest, ): Promise => fetcher(makeAdvancedCopyUrl(params)); -/** - * Get the search results of a query. - * - * @param {SearchRequest} params - * @returns {Promise} - */ -export const getSearchResults = async (params: SearchRequest): Promise => - fetcher(makeSearchResultsUrl(params)); - /** * Get the search results of a query. * diff --git a/src/components/CommandBar/CommandBarBody/index.tsx b/src/components/CommandBar/CommandBarBody/index.tsx index 9b9fbfca89..f06cf31bdf 100644 --- a/src/components/CommandBar/CommandBarBody/index.tsx +++ b/src/components/CommandBar/CommandBarBody/index.tsx @@ -17,15 +17,15 @@ import VoiceSearchBodyContainer from '@/components/TarteelVoiceSearch/BodyContai import TarteelVoiceSearchTrigger from '@/components/TarteelVoiceSearch/Trigger'; import useDebounce from '@/hooks/useDebounce'; import IconSearch from '@/icons/search.svg'; -import { selectRecentNavigations } from '@/redux/slices/CommandBar/state'; +import { selectInitialSearchQuery, selectRecentNavigations } from '@/redux/slices/CommandBar/state'; import { selectIsCommandBarVoiceFlowStarted } from '@/redux/slices/voiceSearch'; -import { SearchMode } from '@/types/Search/SearchRequestParams'; import SearchQuerySource from '@/types/SearchQuerySource'; import { makeNewSearchResultsUrl } from '@/utils/apiPaths'; import { areArraysEqual } from '@/utils/array'; import { logButtonClick, logTextSearchQuery } from '@/utils/eventLogger'; +import { getQuickSearchQuery } from '@/utils/search'; import { SearchResponse } from 'types/ApiResponses'; -import { SearchNavigationType } from 'types/SearchNavigationResult'; +import { SearchNavigationType } from 'types/Search/SearchNavigationResult'; const NAVIGATE_TO = [ { @@ -66,7 +66,8 @@ const CommandBarBody: React.FC = () => { const { t } = useTranslation('common'); const recentNavigations = useSelector(selectRecentNavigations, areArraysEqual); const isVoiceSearchFlowStarted = useSelector(selectIsCommandBarVoiceFlowStarted, shallowEqual); - const [searchQuery, setSearchQuery] = useState(null); + const initialSearchQuery = useSelector(selectInitialSearchQuery, shallowEqual); + const [searchQuery, setSearchQuery] = useState(initialSearchQuery || null); // Debounce search query to avoid having to call the API on every type. The API will be called once the user stops typing. const debouncedSearchQuery = useDebounce(searchQuery, DEBOUNCING_PERIOD_MS); @@ -115,12 +116,7 @@ const CommandBarBody: React.FC = () => { ); const quickSearchFetcher = useCallback(() => { - return getNewSearchResults({ - mode: SearchMode.Quick, - query: searchQuery, - getText: 1, - highlight: 1, - }); + return getNewSearchResults(getQuickSearchQuery(searchQuery)); }, [searchQuery]); /** @@ -191,6 +187,7 @@ const CommandBarBody: React.FC = () => { inputMode="text" // eslint-disable-next-line jsx-a11y/no-autofocus autoFocus + value={searchQuery} /> )} @@ -210,14 +207,7 @@ const CommandBarBody: React.FC = () => { ) : ( import('@/components/Search/SearchBodyContainer'), { @@ -63,12 +62,7 @@ const SearchDrawer: React.FC = () => { addToSearchHistory(dispatch, debouncedSearchQuery, SearchQuerySource.SearchDrawer); setIsSearching(true); logTextSearchQuery(debouncedSearchQuery, SearchQuerySource.SearchDrawer); - getNewSearchResults({ - mode: SearchMode.Quick, - query: debouncedSearchQuery, - getText: 1, - highlight: 1, - }) + getNewSearchResults(getQuickSearchQuery(debouncedSearchQuery)) .then((response) => { setSearchResult({ ...response, @@ -150,6 +144,7 @@ const SearchDrawer: React.FC = () => { searchResult={searchResult} isSearching={isSearching} hasError={hasError} + shouldSuggestFullSearchWhenNoResults /> )} diff --git a/src/components/Search/NavigationItem/index.tsx b/src/components/Search/NavigationItem/index.tsx index f81dd37f0b..53cfffa337 100644 --- a/src/components/Search/NavigationItem/index.tsx +++ b/src/components/Search/NavigationItem/index.tsx @@ -13,7 +13,7 @@ import { logButtonClick } from '@/utils/eventLogger'; import { toLocalizedNumber, toLocalizedVerseKey } from '@/utils/locale'; import { resolveUrlBySearchNavigationType } from '@/utils/navigation'; import { getSearchNavigationResult } from '@/utils/search'; -import { SearchNavigationResult, SearchNavigationType } from 'types/SearchNavigationResult'; +import { SearchNavigationResult, SearchNavigationType } from 'types/Search/SearchNavigationResult'; interface Props { navigation: SearchNavigationResult; @@ -24,7 +24,7 @@ interface Props { const NavigationItem: React.FC = ({ navigation, isSearchDrawer, - service = SearchService.QDC, + service = SearchService.KALIMAT, }) => { const { t, lang } = useTranslation('common'); const chaptersData = useContext(DataContext); diff --git a/src/components/Search/NoResults/index.tsx b/src/components/Search/NoResults/index.tsx index 418fdbcf70..e160fe9631 100644 --- a/src/components/Search/NoResults/index.tsx +++ b/src/components/Search/NoResults/index.tsx @@ -4,14 +4,37 @@ import useTranslation from 'next-translate/useTranslation'; import styles from './NoResults.module.scss'; +import Link, { LinkVariant } from '@/dls/Link/Link'; import IconSearch from '@/icons/search.svg'; +import { logButtonClick } from '@/utils/eventLogger'; +import { getSearchQueryNavigationUrl } from '@/utils/navigation'; interface Props { searchQuery: string; + shouldSuggestFullSearchWhenNoResults?: boolean; } -const NoResults: React.FC = ({ searchQuery }) => { +const NoResults: React.FC = ({ + searchQuery, + shouldSuggestFullSearchWhenNoResults = false, +}) => { const { t } = useTranslation('common'); + if (shouldSuggestFullSearchWhenNoResults) { + return ( + { + logButtonClick('no_results_advanced_search_link'); + }} + variant={LinkVariant.Blend} + > + {t('search-for', { + searchQuery, + })} + + ); + } return ( <>
diff --git a/src/components/Search/SearchBodyContainer.tsx b/src/components/Search/SearchBodyContainer.tsx index de71b89026..3d6047b254 100644 --- a/src/components/Search/SearchBodyContainer.tsx +++ b/src/components/Search/SearchBodyContainer.tsx @@ -22,6 +22,7 @@ interface Props { currentPage?: number; pageSize?: number; onPageChange?: (page: number) => void; + shouldSuggestFullSearchWhenNoResults?: boolean; } const SearchBodyContainer: React.FC = ({ @@ -35,6 +36,7 @@ const SearchBodyContainer: React.FC = ({ currentPage, pageSize, onPageChange, + shouldSuggestFullSearchWhenNoResults = false, }) => { const { t } = useTranslation('common'); const isEmptyResponse = @@ -61,7 +63,10 @@ const SearchBodyContainer: React.FC = ({ {!hasError && searchResult && ( <> {isEmptyResponse ? ( - + ) : ( = ({ result, source, service = SearchService.QDC }) => { +const SearchResultItem: React.FC = ({ result, source, service = SearchService.KALIMAT }) => { const { lang } = useTranslation('quran-reader'); - const localizedVerseKey = useMemo( - () => toLocalizedVerseKey(result.verseKey, lang), - [lang, result.verseKey], - ); + const url = resolveUrlBySearchNavigationType(result.resultType, result.key, true); - const chaptersData = useGetChaptersData(lang); + const getKalimatResultSuffix = () => { + if (result.resultType === SearchNavigationType.SURAH) { + return `(${toLocalizedNumber(Number(result.key), lang)})`; + } - if (!chaptersData) return null; + if (result.resultType === SearchNavigationType.AYAH) { + return `(${toLocalizedVerseKey(result.key as string, lang)})`; + } + + return undefined; + }; - const chapterNumber = getChapterNumberFromKey(result.verseKey); - const chapterData = getChapterData(chaptersData, chapterNumber.toString()); + const suffix = getKalimatResultSuffix(); const onResultItemClicked = () => { logButtonClick(`search_result_item`, { @@ -48,35 +48,20 @@ const SearchResultItem: React.FC = ({ result, source, service = SearchSer return (
- - {chapterData.transliteratedName} {localizedVerseKey} - -
-
- {result.words.map((word, index) => { - return ( - - ); - })} -
+
+ {result.resultType === SearchNavigationType.AYAH ? ( + + ) : ( + + )}
- {result.translations?.map((translation) => ( -
-
- {/* eslint-disable-next-line i18next/no-literal-string */} -

- {translation.resourceName}

-
- ))}
); diff --git a/src/components/Search/SearchResults/SearchResults.module.scss b/src/components/Search/SearchResults/SearchResults.module.scss index 7075c92f5a..1aec0724e7 100644 --- a/src/components/Search/SearchResults/SearchResults.module.scss +++ b/src/components/Search/SearchResults/SearchResults.module.scss @@ -21,4 +21,8 @@ } .navigationItemContainer { margin-inline-end: var(--spacing-small); + em { + font-weight: var(--font-weight-semibold); + text-decoration: underline; + } } diff --git a/src/components/Search/SearchResults/index.tsx b/src/components/Search/SearchResults/index.tsx index 88650d699d..3016fd8c6f 100644 --- a/src/components/Search/SearchResults/index.tsx +++ b/src/components/Search/SearchResults/index.tsx @@ -57,7 +57,7 @@ const SearchResults: React.FC = ({ <> {searchResult.result.verses.map((result) => ( = ({ searchResult, isCommandBar }) => { return ( <> {data.verses.map((verse) => ( - = ({ + result, + source, + service = SearchService.Tarteel, +}) => { + const { lang } = useTranslation('quran-reader'); + const localizedVerseKey = useMemo( + () => toLocalizedVerseKey(result.verseKey, lang), + [lang, result.verseKey], + ); + + const chaptersData = useGetChaptersData(lang); + + if (!chaptersData) return null; + + const chapterNumber = getChapterNumberFromKey(result.verseKey); + const chapterData = getChapterData(chaptersData, chapterNumber.toString()); + + const onResultItemClicked = () => { + logButtonClick(`search_result_item`, { + service, + source, + }); + }; + + return ( +
+
+ + {chapterData.transliteratedName} {localizedVerseKey} + +
+
+ {result.words.map((word, index) => { + return ( + + ); + })} +
+
+ {result.translations?.map((translation) => ( +
+
+ {/* eslint-disable-next-line i18next/no-literal-string */} +

- {translation.resourceName}

+
+ ))} +
+
+ ); +}; +export default TarteelSearchResultItem; diff --git a/src/pages/search.tsx b/src/pages/search.tsx index 8b9ac7172e..40a22eda36 100644 --- a/src/pages/search.tsx +++ b/src/pages/search.tsx @@ -9,32 +9,23 @@ import { useDispatch } from 'react-redux'; import styles from './search.module.scss'; -import { getAvailableLanguages, getAvailableTranslations } from '@/api'; +import { getAvailableLanguages } from '@/api'; import NextSeoWrapper from '@/components/NextSeoWrapper'; -import TranslationsFilter from '@/components/Search/Filters/TranslationsFilter'; import SearchBodyContainer from '@/components/Search/SearchBodyContainer'; -import Button, { ButtonSize, ButtonVariant } from '@/dls/Button/Button'; -import ContentModal, { ContentModalSize } from '@/dls/ContentModal/ContentModal'; import Input, { InputVariant } from '@/dls/Forms/Input'; import useAddQueryParamsToUrl from '@/hooks/useAddQueryParamsToUrl'; import useDebounce from '@/hooks/useDebounce'; import useFocus from '@/hooks/useFocusElement'; -import FilterIcon from '@/icons/filter.svg'; import SearchIcon from '@/icons/search.svg'; +import { setInitialSearchQuery, setIsOpen } from '@/redux/slices/CommandBar/state'; import SearchQuerySource from '@/types/SearchQuerySource'; import { getAllChaptersData } from '@/utils/chapter'; -import { logButtonClick, logEvent, logValueChange } from '@/utils/eventLogger'; -import filterTranslations from '@/utils/filter-translations'; -import { getLanguageAlternates, toLocalizedNumber } from '@/utils/locale'; +import { logButtonClick, logEvent } from '@/utils/eventLogger'; +import { getLanguageAlternates } from '@/utils/locale'; import { getCanonicalUrl } from '@/utils/navigation'; -import { - addToSearchHistory, - getDefaultTranslationIdsByLang, - searchGetResults, -} from '@/utils/search'; +import { addToSearchHistory, searchGetResults } from '@/utils/search'; import { SearchResponse } from 'types/ApiResponses'; import AvailableLanguage from 'types/AvailableLanguage'; -import AvailableTranslation from 'types/AvailableTranslation'; import ChaptersData from 'types/ChaptersData'; const PAGE_SIZE = 10; @@ -42,22 +33,16 @@ const DEBOUNCING_PERIOD_MS = 1000; type SearchProps = { languages: AvailableLanguage[]; - translations: AvailableTranslation[]; chaptersData: ChaptersData; }; -const Search: NextPage = ({ translations }): JSX.Element => { +const Search: NextPage = (): JSX.Element => { const { t, lang } = useTranslation('common'); const router = useRouter(); const [searchQuery, setSearchQuery] = useState(''); const [focusInput, searchInputRef]: [() => void, RefObject] = useFocus(); const [currentPage, setCurrentPage] = useState(1); const [selectedLanguages, setSelectedLanguages] = useState(''); - const [selectedTranslations, setSelectedTranslations] = useState(() => { - return getDefaultTranslationIdsByLang(translations, lang) as string; - }); - const [translationSearchQuery, setTranslationSearchQuery] = useState(''); - const [isContentModalOpen, setIsContentModalOpen] = useState(false); const [isSearching, setIsSearching] = useState(false); const [hasError, setHasError] = useState(false); const [searchResult, setSearchResult] = useState(null); @@ -70,9 +55,8 @@ const Search: NextPage = ({ translations }): JSX.Element => { page: currentPage, languages: selectedLanguages, q: debouncedSearchQuery, - translations: selectedTranslations, }), - [currentPage, debouncedSearchQuery, selectedLanguages, selectedTranslations], + [currentPage, debouncedSearchQuery, selectedLanguages], ); useAddQueryParamsToUrl('/search', queryParams); @@ -82,15 +66,10 @@ const Search: NextPage = ({ translations }): JSX.Element => { // in the query object. @see https://nextjs.org/docs/routing/dynamic-routes#caveats useEffect(() => { // we don't want to focus the main search input when the translation filter modal is open. - if (router.isReady && !isContentModalOpen) { + if (router.isReady) { focusInput(); } - }, [focusInput, router, isContentModalOpen]); - - // handle when language changes - useEffect(() => { - setSelectedTranslations(getDefaultTranslationIdsByLang(translations, lang) as string); - }, [lang, translations]); + }, [focusInput, router]); useEffect(() => { if (router.query.q || router.query.query) { @@ -107,16 +86,7 @@ const Search: NextPage = ({ translations }): JSX.Element => { if (router.query.languages) { setSelectedLanguages(router.query.languages as string); } - if (router.query.translations) { - setSelectedTranslations(router.query.translations as string); - } - }, [ - router.query.q, - router.query.query, - router.query.page, - router.query.languages, - router.query.translations, - ]); + }, [router.query.q, router.query.query, router.query.page, router.query.languages]); /** * Handle when the search query is changed. @@ -125,7 +95,8 @@ const Search: NextPage = ({ translations }): JSX.Element => { * @returns {void} */ const onSearchQueryChange = (newSearchQuery: string): void => { - setSearchQuery(newSearchQuery || ''); + dispatch({ type: setIsOpen.type, payload: true }); + dispatch({ type: setInitialSearchQuery.type, payload: newSearchQuery }); }; const onClearClicked = () => { @@ -138,25 +109,20 @@ const Search: NextPage = ({ translations }): JSX.Element => { * * @param {string} query * @param {number} page - * @param {string} translation * @param {string} language */ - const getResults = useCallback( - (query: string, page: number, translation: string, language: string) => { - searchGetResults( - SearchQuerySource.SearchPage, - query, - page, - PAGE_SIZE, - setIsSearching, - setHasError, - setSearchResult, - language, - translation, - ); - }, - [], - ); + const getResults = useCallback((query: string, page: number, language: string) => { + searchGetResults( + SearchQuerySource.SearchPage, + query, + page, + PAGE_SIZE, + setIsSearching, + setHasError, + setSearchResult, + language, + ); + }, []); // a ref to know whether this is the initial search request made when the user loads the page or not const isInitialSearch = useRef(true); @@ -176,7 +142,6 @@ const Search: NextPage = ({ translations }): JSX.Element => { debouncedSearchQuery, // if it is the initial search request, use the page number in the url, otherwise, reset it isInitialSearch.current ? currentPage : 1, - selectedTranslations, selectedLanguages, ); @@ -188,97 +153,20 @@ const Search: NextPage = ({ translations }): JSX.Element => { // we don't want to run this effect when currentPage is changed // because we are already handeling this in onPageChange // eslint-disable-next-line react-hooks/exhaustive-deps - }, [debouncedSearchQuery, getResults, selectedLanguages, selectedTranslations]); + }, [debouncedSearchQuery, getResults, selectedLanguages]); const onPageChange = (page: number) => { logEvent('search_page_number_change', { page }); setCurrentPage(page); - getResults(debouncedSearchQuery, page, selectedTranslations, selectedLanguages); + getResults(debouncedSearchQuery, page, selectedLanguages); }; - const onTranslationChange = useCallback((translationIds: string[]) => { - // convert the array into a string - setSelectedTranslations((prevTranslationIds) => { - // filter out the empty strings - const newTranslationIds = translationIds.filter((id) => id !== '').join(','); - logValueChange('search_page_selected_translations', prevTranslationIds, newTranslationIds); - return newTranslationIds; - }); - // reset the current page since most probable the results will change. - setCurrentPage(1); - }, []); - const onSearchKeywordClicked = useCallback((keyword: string) => { setSearchQuery(keyword); }, []); const navigationUrl = '/search'; - const formattedSelectedTranslations = useMemo(() => { - if (!selectedTranslations) return t('search:default-translations'); - - let selectedValueString; - - const selectedTranslationsArray = selectedTranslations.split(','); - - const firstSelectedTranslation = translations.find( - (translation) => translation.id.toString() === selectedTranslationsArray[0], - ); - - if (!firstSelectedTranslation) return t('search:all-translations'); - - if (selectedTranslationsArray.length === 1) { - selectedValueString = firstSelectedTranslation.name; - } - if (selectedTranslationsArray.length === 2) { - selectedValueString = t('settings.value-and-other', { - value: firstSelectedTranslation?.name, - othersCount: toLocalizedNumber(selectedTranslationsArray.length - 1, lang), - }); - } - if (selectedTranslationsArray.length > 2) { - selectedValueString = t('settings.value-and-others', { - value: firstSelectedTranslation?.name, - othersCount: toLocalizedNumber(selectedTranslationsArray.length - 1, lang), - }); - } - - return selectedValueString; - }, [lang, selectedTranslations, t, translations]); - - const filteredTranslations = translationSearchQuery - ? filterTranslations(translations, translationSearchQuery) - : translations; - - const onResetButtonClicked = () => { - logButtonClick('search_page_reset_button'); - const defaultLangTranslationIds = getDefaultTranslationIdsByLang( - translations, - lang, - false, - ) as string[]; - onTranslationChange(defaultLangTranslationIds); - }; - - const onTranslationSearchQueryChange = (newTranslationSearchQuery: string) => { - logValueChange( - 'search_page_translation_search_query', - translationSearchQuery, - newTranslationSearchQuery, - ); - setTranslationSearchQuery(newTranslationSearchQuery); - }; - - const onTranslationSearchClearClicked = () => { - logButtonClick('search_page_translation_search_clear'); - setTranslationSearchQuery(''); - }; - - const onTranslationsFiltersClicked = () => { - logButtonClick('search_page_translation_filter'); - setIsContentModalOpen(true); - }; - return ( <> = ({ translations }): JSX.Element => { fixedWidth={false} variant={InputVariant.Main} /> - -
- } - onChange={onTranslationSearchQueryChange} - onClearClicked={onTranslationSearchClearClicked} - clearable - value={translationSearchQuery} - placeholder={t('settings.search-translations')} - fixedWidth={false} - variant={InputVariant.Main} - /> -
- -
- } - isOpen={isContentModalOpen} - onClose={() => setIsContentModalOpen(false)} - > - - -
- -
- {/* eslint-disable-next-line i18next/no-literal-string */} - {t('search:searching-translations')}: - {formattedSelectedTranslations} -
-
@@ -385,32 +221,22 @@ const Search: NextPage = ({ translations }): JSX.Element => { export const getStaticProps: GetStaticProps = async ({ locale }) => { try { - const [availableLanguagesResponse, availableTranslationsResponse] = await Promise.all([ - getAvailableLanguages(locale), - getAvailableTranslations(locale), - ]); + const availableLanguagesResponse = await getAvailableLanguages(locale); - let translations = []; let languages = []; if (availableLanguagesResponse.status !== 500) { const { languages: responseLanguages } = availableLanguagesResponse; languages = responseLanguages; } - if (availableTranslationsResponse.status !== 500) { - const { translations: responseTranslations } = availableTranslationsResponse; - translations = responseTranslations; - } const chaptersData = await getAllChaptersData(locale); return { props: { chaptersData, languages, - translations, }, }; } catch (e) { - console.log(e); return { props: { hasError: true, diff --git a/src/redux/slices/CommandBar/persistConfig.ts b/src/redux/slices/CommandBar/persistConfig.ts index d9b070ed23..a86f006323 100644 --- a/src/redux/slices/CommandBar/persistConfig.ts +++ b/src/redux/slices/CommandBar/persistConfig.ts @@ -6,7 +6,7 @@ const commandBarPersistConfig = { key: SliceName.COMMAND_BAR, storage, version: 1, - blacklist: ['isOpen'], + blacklist: ['isOpen', 'initialSearchQuery'], }; export default commandBarPersistConfig; diff --git a/src/redux/slices/CommandBar/state.ts b/src/redux/slices/CommandBar/state.ts index 659967cd5c..e38751d576 100644 --- a/src/redux/slices/CommandBar/state.ts +++ b/src/redux/slices/CommandBar/state.ts @@ -2,16 +2,17 @@ import { createSlice, PayloadAction } from '@reduxjs/toolkit'; import { RootState } from '@/redux/RootState'; import SliceName from '@/redux/types/SliceName'; -import { SearchNavigationResult } from 'types/SearchNavigationResult'; +import { SearchNavigationResult } from 'types/Search/SearchNavigationResult'; export type CommandBar = { isOpen: boolean; recentNavigations: SearchNavigationResult[]; + initialSearchQuery: string; }; const MAXIMUM_RECENT_NAVIGATIONS = 5; -const initialState: CommandBar = { isOpen: false, recentNavigations: [] }; +const initialState: CommandBar = { isOpen: false, recentNavigations: [], initialSearchQuery: '' }; export const commandBarSlice = createSlice({ name: SliceName.COMMAND_BAR, @@ -24,6 +25,7 @@ export const commandBarSlice = createSlice({ toggleIsOpen: (state: CommandBar) => ({ ...state, isOpen: !state.isOpen, + initialSearchQuery: '', // we reset the initial search query when the command bar is toggled }), addRecentNavigation: (state: CommandBar, action: PayloadAction) => { let newRecentNavigations = [...state.recentNavigations]; @@ -53,13 +55,22 @@ export const commandBarSlice = createSlice({ recentNavigations: newRecentNavigations, }; }, + setInitialSearchQuery: (state: CommandBar, action: PayloadAction) => ({ + ...state, + initialSearchQuery: action.payload, + }), }, }); -export const { setIsOpen, toggleIsOpen, addRecentNavigation, removeRecentNavigation } = - commandBarSlice.actions; +export const { + setIsOpen, + toggleIsOpen, + addRecentNavigation, + removeRecentNavigation, + setInitialSearchQuery, +} = commandBarSlice.actions; export const selectCommandBarIsOpen = (state: RootState) => state.commandBar.isOpen; export const selectRecentNavigations = (state: RootState) => state.commandBar.recentNavigations; - +export const selectInitialSearchQuery = (state: RootState) => state.commandBar.initialSearchQuery; export default commandBarSlice.reducer; diff --git a/src/utils/apiPaths.ts b/src/utils/apiPaths.ts index 39f9ffa5f0..b1703e1ecb 100644 --- a/src/utils/apiPaths.ts +++ b/src/utils/apiPaths.ts @@ -10,7 +10,7 @@ import { getTranslationsInitialState, } from '@/redux/defaultSettings/util'; import { SearchRequestParams, SearchMode } from '@/types/Search/SearchRequestParams'; -import { AdvancedCopyRequest, PagesLookUpRequest, SearchRequest } from 'types/ApiRequests'; +import { AdvancedCopyRequest, PagesLookUpRequest } from 'types/ApiRequests'; import { MushafLines, QuranFont } from 'types/QuranReader'; export const DEFAULT_VERSES_PARAMS = { @@ -146,14 +146,6 @@ export const makeTranslationsInfoUrl = (locale: string, translations: number[]): export const makeAdvancedCopyUrl = (params: AdvancedCopyRequest): string => makeUrl('/verses/advanced_copy', params as Record); -/** - * Compose the url for search API. - * - * @param {SearchRequest} params the request params. - * @returns {string} - */ -export const makeSearchResultsUrl = (params: SearchRequest): string => makeUrl('/search', params); - export const makeNewSearchApiUrl = (params: Record) => { const baseUrl = process.env.NEXT_PUBLIC_SEARCH_BASE_URL; diff --git a/src/utils/navigation.ts b/src/utils/navigation.ts index a2b5d3056d..32807effe0 100644 --- a/src/utils/navigation.ts +++ b/src/utils/navigation.ts @@ -8,7 +8,7 @@ import { getVerseAndChapterNumbersFromKey, getVerseNumberRangeFromKey } from './ import QueryParam from '@/types/QueryParam'; import { QuranReaderFlow } from '@/types/QuranReader'; -import { SearchNavigationType } from 'types/SearchNavigationResult'; +import { SearchNavigationType } from 'types/Search/SearchNavigationResult'; /** * Get the href link to a verse. diff --git a/src/utils/search.ts b/src/utils/search.ts index 3dc8784256..ceaed36148 100644 --- a/src/utils/search.ts +++ b/src/utils/search.ts @@ -8,18 +8,18 @@ import { AnyAction } from 'redux'; import { logEmptySearchResults, logSearchResults, logTextSearchQuery } from './eventLogger'; -import { getNewSearchResults, getSearchResults } from '@/api'; +import { getNewSearchResults } from '@/api'; import { addSearchHistoryRecord } from '@/redux/slices/Search/search'; import { SearchResponse } from '@/types/ApiResponses'; import AvailableTranslation from '@/types/AvailableTranslation'; import ChaptersData from '@/types/ChaptersData'; -import { SearchMode } from '@/types/Search/SearchRequestParams'; +import { SearchMode, SearchRequestParams } from '@/types/Search/SearchRequestParams'; import SearchService from '@/types/Search/SearchService'; import SearchQuerySource from '@/types/SearchQuerySource'; import { getChapterData } from '@/utils/chapter'; import { toLocalizedNumber } from '@/utils/locale'; import { getVerseAndChapterNumbersFromKey, getVerseNumberRangeFromKey } from '@/utils/verse'; -import { SearchNavigationResult, SearchNavigationType } from 'types/SearchNavigationResult'; +import { SearchNavigationResult, SearchNavigationType } from 'types/Search/SearchNavigationResult'; export const LOCALE_TO_TRANSLATION_LANGUAGE = { en: 'english', @@ -177,8 +177,7 @@ export const getSearchNavigationResult = ( }; /** - * Call BE to fetch the search results using the passed filters - * and if there are no results call Kalimat API. + * Call Kalimat API to fetch the search results using the passed filters. * * @param {SearchQuerySource} source * @param {string} query @@ -188,7 +187,6 @@ export const getSearchNavigationResult = ( * @param {(arg: boolean) => void} setHasError * @param {(data: SearchResponse) => void} setSearchResult * @param {string} languages - * @param {string} translations */ export const searchGetResults = ( source: SearchQuerySource, @@ -199,65 +197,37 @@ export const searchGetResults = ( setHasError: (arg: boolean) => void, setSearchResult: (data: SearchResponse) => void, languages?: string, - translations?: string, ) => { setIsSearching(true); logTextSearchQuery(query, source); - getSearchResults({ + getNewSearchResults({ + mode: SearchMode.Advanced, query, - ...(languages && { filterLanguages: languages }), // languages will be included only when there is a selected language size: pageSize, + filterLanguages: languages, page, - ...(translations && { filterTranslations: translations }), // translations will be included only when there is a selected translation + exactMatchesOnly: 0, + getText: 1, + highlight: 1, }) - .then(async (response) => { - if (response.status === 500) { - setHasError(true); - } else { - setSearchResult({ ...response, service: SearchService.QDC }); - const noQdcResults = - response.pagination.totalRecords === 0 && !response.result.navigation.length; - // if there is no navigations nor verses in the response - if (noQdcResults) { - logEmptySearchResults({ - query, - source, - service: SearchService.QDC, - }); - - const kalimatResponse = await getNewSearchResults({ - mode: SearchMode.Advanced, - query, - size: pageSize, - filterLanguages: languages, - page, - exactMatchesOnly: 0, - // translations will be included only when there is a selected translation - ...(translations && { - filterTranslations: translations, - translationFields: 'resource_name', - }), - }); + .then(async (kalimatResponse) => { + setSearchResult({ + ...kalimatResponse, + service: SearchService.KALIMAT, + }); - setSearchResult({ - ...kalimatResponse, - service: SearchService.KALIMAT, - }); - - if (kalimatResponse.pagination.totalRecords === 0) { - logEmptySearchResults({ - query, - source, - service: SearchService.KALIMAT, - }); - } else { - logSearchResults({ - query, - source, - service: SearchService.KALIMAT, - }); - } - } + if (kalimatResponse.pagination.totalRecords === 0) { + logEmptySearchResults({ + query, + source, + service: SearchService.KALIMAT, + }); + } else { + logSearchResults({ + query, + source, + service: SearchService.KALIMAT, + }); } }) .catch(() => { @@ -283,3 +253,18 @@ export const addToSearchHistory = ( dispatch({ type: addSearchHistoryRecord.type, payload: debouncedSearchQuery }); logTextSearchQuery(debouncedSearchQuery, source); }; + +/** + * Get the quick search query. + * + * @param {string} query + * @returns {SearchRequestParams} + */ +export const getQuickSearchQuery = (query: string): SearchRequestParams => { + return { + mode: SearchMode.Quick, + query, + getText: 1, + highlight: 1, + }; +}; diff --git a/types/ApiRequests.ts b/types/ApiRequests.ts index cb44fb7f94..e7da06d773 100644 --- a/types/ApiRequests.ts +++ b/types/ApiRequests.ts @@ -1,11 +1,3 @@ -export type SearchRequest = { - query: string; - filterLanguages?: string; - filterTranslations?: string; - size?: number; - page?: number; -}; - export type AdvancedCopyRequest = { from: string; to: string; diff --git a/types/ApiResponses.ts b/types/ApiResponses.ts index cb55c05c9c..4f1f46e505 100644 --- a/types/ApiResponses.ts +++ b/types/ApiResponses.ts @@ -9,8 +9,9 @@ import LookupRange from './LookupRange'; import LookupRecord from './LookupRecord'; import MetaData from './MetaData'; import Reciter from './Reciter'; +import { SearchNavigationResult } from './Search/SearchNavigationResult'; import SearchService from './Search/SearchService'; -import { SearchNavigationResult } from './SearchNavigationResult'; +import SearchVerseItem from './Search/SearchVerseItem'; import TafsirInfo from './TafsirInfo'; import Verse from './Verse'; @@ -19,7 +20,7 @@ export interface BaseResponse { error?: string; } -interface Pagination { +export interface Pagination { perPage: number; currentPage: number; nextPage: number | null; @@ -84,7 +85,7 @@ export interface SearchResponse extends BaseResponse { pagination: Pagination; result?: { navigation: SearchNavigationResult[]; - verses: Verse[]; + verses: SearchVerseItem[]; }; } diff --git a/types/SearchNavigationResult.ts b/types/Search/SearchNavigationResult.ts similarity index 100% rename from types/SearchNavigationResult.ts rename to types/Search/SearchNavigationResult.ts diff --git a/types/Search/SearchRequestParams.ts b/types/Search/SearchRequestParams.ts index 167515b826..1d2daa92b9 100644 --- a/types/Search/SearchRequestParams.ts +++ b/types/Search/SearchRequestParams.ts @@ -20,7 +20,6 @@ export type SearchRequestParams = { size?: number; page?: number; getText?: SearchBoolean; - filterTranslations?: string; filterLanguages?: string; fields?: string; translationFields?: string; diff --git a/types/Search/SearchResponse.ts b/types/Search/SearchResponse.ts index 7bb72ad810..8ce399109b 100644 --- a/types/Search/SearchResponse.ts +++ b/types/Search/SearchResponse.ts @@ -1,6 +1,6 @@ -import { BaseResponse } from '../ApiResponses'; -import { SearchNavigationResult } from '../SearchNavigationResult'; +import { BaseResponse, Pagination } from '../ApiResponses'; +import { SearchNavigationResult } from './SearchNavigationResult'; import SearchVerseItem from './SearchVerseItem'; interface SearchResponse extends BaseResponse { @@ -8,13 +8,7 @@ interface SearchResponse extends BaseResponse { navigation: SearchNavigationResult[]; verses: SearchVerseItem[]; }; - pagination: { - perPage: number; - currentPage: number; - nextPage: number | null; - totalRecords: number; - totalPages: number; - }; + pagination: Pagination; } export default SearchResponse; diff --git a/types/Search/SearchVerseItem.ts b/types/Search/SearchVerseItem.ts index 1d85a42a3b..98cded5ea3 100644 --- a/types/Search/SearchVerseItem.ts +++ b/types/Search/SearchVerseItem.ts @@ -1,11 +1,4 @@ -import Verse from '../Verse'; -import Word from '../Word'; +import { SearchNavigationResult } from './SearchNavigationResult'; -type SearchVerseItem = Verse & { - words: Word[]; -} & { - kalimatData: { - matches?: string; - }; -}; +type SearchVerseItem = SearchNavigationResult; export default SearchVerseItem;