Read and update url params for query and sort

This commit is contained in:
James Grugett 2022-07-28 23:54:18 -07:00
parent eeaf4bde14
commit 477fd70620
2 changed files with 60 additions and 35 deletions

View File

@ -4,7 +4,7 @@ import algoliasearch from 'algoliasearch/lite'
import { Contract } from 'common/contract' import { Contract } from 'common/contract'
import { import {
Sort, Sort,
useInitialQueryAndSort, useQueryAndSortParams,
} from '../hooks/use-sort-and-query-params' } from '../hooks/use-sort-and-query-params'
import { ContractsGrid } from './contract/contracts-list' import { ContractsGrid } from './contract/contracts-list'
import { Row } from './layout/row' import { Row } from './layout/row'
@ -91,14 +91,12 @@ export function ContractSearch(props: {
memberPillGroups.length > 0 ? memberPillGroups : defaultPillGroups memberPillGroups.length > 0 ? memberPillGroups : defaultPillGroups
const follows = useFollows(user?.id) const follows = useFollows(user?.id)
const { initialSort: savedSort } = useInitialQueryAndSort(querySortOptions)
const initialSort = const { shouldLoadFromStorage, defaultSort } = querySortOptions ?? {}
savedSort && sortOptions.map(({ value }) => value).includes(savedSort) const { query, setQuery, sort, setSort } = useQueryAndSortParams({
? savedSort defaultSort,
: querySortOptions?.defaultSort ?? DEFAULT_SORT shouldLoadFromStorage,
})
const [sort, setSort] = useState(initialSort)
const [filter, setFilter] = useState<filter>( const [filter, setFilter] = useState<filter>(
querySortOptions?.defaultFilter ?? 'open' querySortOptions?.defaultFilter ?? 'open'
@ -156,7 +154,6 @@ export function ContractSearch(props: {
const indexName = `${indexPrefix}contracts-${sort}` const indexName = `${indexPrefix}contracts-${sort}`
const index = useMemo(() => searchClient.initIndex(indexName), [indexName]) const index = useMemo(() => searchClient.initIndex(indexName), [indexName])
const [query, setQuery] = useState('')
const [page, setPage] = useState(0) const [page, setPage] = useState(0)
const [numPages, setNumPages] = useState(1) const [numPages, setNumPages] = useState(1)
const [hitsByPage, setHitsByPage] = useState<{ [page: string]: Contract[] }>( const [hitsByPage, setHitsByPage] = useState<{ [page: string]: Contract[] }>(
@ -172,6 +169,7 @@ export function ContractSearch(props: {
facetFilters, facetFilters,
numericFilters, numericFilters,
}) })
let wasMostRecentQuery = true
index index
.search(query, { .search(query, {
facetFilters, facetFilters,
@ -180,6 +178,8 @@ export function ContractSearch(props: {
hitsPerPage: 20, hitsPerPage: 20,
}) })
.then((results) => { .then((results) => {
if (!wasMostRecentQuery) return
if (page === 0) { if (page === 0) {
setHitsByPage({ setHitsByPage({
[0]: results.hits as any as Contract[], [0]: results.hits as any as Contract[],
@ -191,8 +191,11 @@ export function ContractSearch(props: {
})) }))
} }
setNumPages(results.nbPages) setNumPages(results.nbPages)
console.log(results.page, '/', results.nbPages, results.hits) console.log(results.page + 1, '/', results.nbPages, results.hits)
}) })
return () => {
wasMostRecentQuery = false
}
// Note numeric filters are unique based on current time, so can't compare // Note numeric filters are unique based on current time, so can't compare
// them by value. // them by value.
}, [query, page, index, JSON.stringify(facetFilters), filter]) }, [query, page, index, JSON.stringify(facetFilters), filter])
@ -215,9 +218,16 @@ export function ContractSearch(props: {
const showTime = const showTime =
sort === 'close-date' || sort === 'resolve-date' ? sort : undefined sort === 'close-date' || sort === 'resolve-date' ? sort : undefined
const selectFilter = (filter: filter) => { const updateQuery = (newQuery: string) => {
setFilter(filter) setQuery(newQuery)
trackCallback('select search filter', { filter }) setPage(0)
}
const selectFilter = (newFilter: filter) => {
if (newFilter === filter) return
setFilter(newFilter)
setPage(0)
trackCallback('select search filter', { filter: newFilter })
} }
const selectSort = (newSort: Sort) => { const selectSort = (newSort: Sort) => {
@ -243,7 +253,7 @@ export function ContractSearch(props: {
<input <input
type="text" type="text"
value={query} value={query}
onChange={(e) => setQuery(e.target.value)} onChange={(e) => updateQuery(e.target.value)}
placeholder={showPlaceHolder ? `Search ${filter} markets` : ''} placeholder={showPlaceHolder ? `Search ${filter} markets` : ''}
className="input input-bordered w-full" className="input input-bordered w-full"
/> />

View File

@ -1,8 +1,6 @@
import { defaults, debounce } from 'lodash' import { defaults, debounce } from 'lodash'
import { useRouter } from 'next/router' import { useRouter } from 'next/router'
import { useEffect, useMemo, useState } from 'react' import { useEffect, useMemo, useState } from 'react'
import { useSearchBox } from 'react-instantsearch-hooks-web'
import { track } from 'web/lib/service/analytics'
import { DEFAULT_SORT } from 'web/components/contract-search' import { DEFAULT_SORT } from 'web/components/contract-search'
const MARKETS_SORT = 'markets_sort' const MARKETS_SORT = 'markets_sort'
@ -74,25 +72,32 @@ export function useInitialQueryAndSort(options?: {
} }
} }
export function useUpdateQueryAndSort(props: { export function useQueryAndSortParams(options?: {
shouldLoadFromStorage: boolean defaultSort?: Sort
shouldLoadFromStorage?: boolean
}) { }) {
const { shouldLoadFromStorage } = props const { defaultSort = DEFAULT_SORT, shouldLoadFromStorage = true } =
options ?? {}
const router = useRouter() const router = useRouter()
const { s: sort, q: query } = router.query as {
q?: string
s?: Sort
}
const setSort = (sort: Sort | undefined) => { const setSort = (sort: Sort | undefined) => {
if (sort !== router.query.s) {
router.query.s = sort router.query.s = sort
router.replace({ query: { ...router.query, s: sort } }, undefined, { router.push(router, undefined, { shallow: true })
shallow: true,
})
if (shouldLoadFromStorage) { if (shouldLoadFromStorage) {
localStorage.setItem(MARKETS_SORT, sort || '') localStorage.setItem(MARKETS_SORT, sort || '')
} }
} }
}
const { query, refine } = useSearchBox() const [queryState, setQueryState] = useState(query)
useEffect(() => {
setQueryState(query)
}, [query])
// Debounce router query update. // Debounce router query update.
const pushQuery = useMemo( const pushQuery = useMemo(
@ -103,22 +108,32 @@ export function useUpdateQueryAndSort(props: {
} else { } else {
delete router.query.q delete router.query.q
} }
router.replace({ query: router.query }, undefined, { router.replace(router, undefined, { shallow: true })
shallow: true, }, 100),
})
track('search', { query })
}, 500),
[router] [router]
) )
const setQuery = (query: string | undefined) => { const setQuery = (query: string | undefined) => {
refine(query ?? '') setQueryState(query)
pushQuery(query) pushQuery(query)
} }
useEffect(() => {
// If there's no sort option, then set the one from localstorage
if (router.isReady && !sort && shouldLoadFromStorage) {
const localSort = localStorage.getItem(MARKETS_SORT) as Sort
if (localSort) {
router.query.s = localSort
// Use replace to not break navigating back.
router.replace(router, undefined, { shallow: true })
}
}
})
return { return {
sort: sort ?? defaultSort,
query: queryState ?? '',
setSort, setSort,
setQuery, setQuery,
query,
} }
} }