Read and update url params for query and sort
This commit is contained in:
parent
eeaf4bde14
commit
477fd70620
|
@ -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"
|
||||||
/>
|
/>
|
||||||
|
|
|
@ -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,
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue
Block a user