import algoliasearch from 'algoliasearch/lite' import { InstantSearch, SearchBox, SortBy, useCurrentRefinements, useInfiniteHits, useRange, useRefinementList, useSortBy, useToggleRefinement, } from 'react-instantsearch-hooks-web' import { Contract } from '../../common/contract' import { Sort, useInitialQueryAndSort, useUpdateQueryAndSort, } from '../hooks/use-sort-and-query-params' import { ContractsGrid } from './contract/contracts-list' import { Row } from './layout/row' import { useEffect, useRef, useState } from 'react' import { Spacer } from './layout/spacer' import { useRouter } from 'next/router' import { ENV } from 'common/envs/constants' import { CategorySelector } from './feed/category-selector' import { useUser } from 'web/hooks/use-user' const searchClient = algoliasearch( 'GJQPAYENIF', '75c28fc084a80e1129d427d470cf41a3' ) const indexPrefix = ENV === 'DEV' ? 'dev-' : '' const sortIndexes = [ { label: 'Newest', value: indexPrefix + 'contracts-newest' }, { label: 'Oldest', value: indexPrefix + 'contracts-oldest' }, { label: 'Most traded', value: indexPrefix + 'contracts-most-traded' }, { label: '24h volume', value: indexPrefix + 'contracts-24-hour-vol' }, { label: 'Close date', value: indexPrefix + 'contracts-close-date' }, { label: 'Resolve date', value: indexPrefix + 'contracts-resolve-date' }, ] type filter = 'open' | 'closed' | 'resolved' | 'all' export function ContractSearch(props: { querySortOptions?: { defaultSort: Sort defaultFilter?: filter shouldLoadFromStorage?: boolean } additionalFilter?: { creatorId?: string tag?: string category?: string } showCategorySelector: boolean }) { const { querySortOptions, additionalFilter, showCategorySelector } = props const user = useUser() const { initialSort } = useInitialQueryAndSort(querySortOptions) const sort = sortIndexes .map(({ value }) => value) .includes(`${indexPrefix}contracts-${initialSort ?? ''}`) ? initialSort : querySortOptions?.defaultSort const [filter, setFilter] = useState( querySortOptions?.defaultFilter ?? 'open' ) const [category, setCategory] = useState('all') if (!sort) return <> return (
{showCategorySelector && ( <> )}
) } export function ContractSearchInner(props: { querySortOptions?: { defaultSort: Sort shouldLoadFromStorage?: boolean } filter: filter additionalFilter: { creatorId?: string tag?: string category?: string } }) { const { querySortOptions, filter, additionalFilter } = props const { initialQuery } = useInitialQueryAndSort(querySortOptions) const { query, setQuery, setSort } = useUpdateQueryAndSort({ shouldLoadFromStorage: true, }) useEffect(() => { setQuery(initialQuery) }, [initialQuery]) const { currentRefinement: index } = useSortBy({ items: [], }) useEffect(() => { setQuery(query) }, [query]) const isFirstRender = useRef(true) useEffect(() => { if (isFirstRender.current) { isFirstRender.current = false return } const sort = index.split('contracts-')[1] as Sort if (sort) { setSort(sort) } }, [index]) const { creatorId, category, tag } = additionalFilter useFilterCreator(creatorId) useFilterTag(tag ?? (category === 'all' ? undefined : category)) useFilterClosed( filter === 'closed' ? true : filter === 'all' || filter === 'resolved' ? undefined : false ) useFilterResolved( filter === 'resolved' ? true : filter === 'all' ? undefined : false ) const { showMore, hits, isLastPage, results } = useInfiniteHits() const contracts = hits as any as Contract[] const router = useRouter() const hasLoaded = contracts.length > 0 || router.isReady if (!hasLoaded || !results) return <> return ( ) } const useFilterCreator = (creatorId: string | undefined) => { const { refine } = useRefinementList({ attribute: 'creatorId' }) useEffect(() => { if (creatorId) refine(creatorId) }, [creatorId, refine]) } const useFilterTag = (tag: string | undefined) => { const { items, refine: deleteRefinement } = useCurrentRefinements({ includedAttributes: ['lowercaseTags'], }) const { refine } = useRefinementList({ attribute: 'lowercaseTags' }) useEffect(() => { const refinements = items[0]?.refinements ?? [] if (tag) refine(tag.toLowerCase()) if (refinements[0]) deleteRefinement(refinements[0]) }, [tag]) } const useFilterClosed = (value: boolean | undefined) => { const [now] = useState(Date.now()) useRange({ attribute: 'closeTime', min: value === false ? now : undefined, max: value ? now : undefined, }) } const useFilterResolved = (value: boolean | undefined) => { // Note (James): I don't know why this works. const { refine: refineResolved } = useToggleRefinement({ attribute: value === undefined ? 'non-existant-field' : 'isResolved', on: true, off: value === undefined ? undefined : false, }) useEffect(() => { refineResolved({ isRefined: !value }) }, [value]) }