mirror of
https://github.com/ershisan99/flashcards-example-project.git
synced 2025-12-16 12:33:18 +00:00
feat: add debounce to search
This commit is contained in:
@@ -37,6 +37,7 @@
|
||||
"react-toastify": "^9.1.3",
|
||||
"remeda": "^1.33.0",
|
||||
"storybook-addon-react-router-v6": "^2.0.10",
|
||||
"usehooks-ts": "^3.1.0",
|
||||
"zod": "^3.22.4"
|
||||
},
|
||||
"devDependencies": {
|
||||
|
||||
5099
pnpm-lock.yaml
generated
5099
pnpm-lock.yaml
generated
File diff suppressed because it is too large
Load Diff
@@ -1,3 +1,5 @@
|
||||
import { useCallback } from 'react'
|
||||
|
||||
import { isNil } from 'remeda'
|
||||
|
||||
export function useQueryParam<T extends boolean | number | string>(
|
||||
@@ -9,14 +11,17 @@ export function useQueryParam<T extends boolean | number | string>(
|
||||
const paramValue = searchParams.get(param)
|
||||
const convertedValue = getConvertedValue<T>(paramValue, defaultValue)
|
||||
|
||||
const setParamValue = (value: T | null): void => {
|
||||
const setParamValue = useCallback(
|
||||
(value: T | null): void => {
|
||||
if (isNil(value) || value === '') {
|
||||
searchParams.delete(param)
|
||||
} else {
|
||||
searchParams.set(param, String(value))
|
||||
}
|
||||
setSearchParams(searchParams)
|
||||
}
|
||||
},
|
||||
[searchParams, setSearchParams]
|
||||
)
|
||||
|
||||
return [convertedValue, setParamValue]
|
||||
}
|
||||
|
||||
@@ -13,6 +13,7 @@ import {
|
||||
useGetDecksQuery,
|
||||
useUpdateDeckMutation,
|
||||
} from '@/services/decks'
|
||||
import { useDebounceCallback } from 'usehooks-ts'
|
||||
|
||||
import s from './decks-page.module.scss'
|
||||
|
||||
@@ -21,7 +22,6 @@ export const DecksPage = () => {
|
||||
const [showCreateModal, setShowCreateModal] = useState(false)
|
||||
const [deckToDeleteId, setDeckToDeleteId] = useState<null | string>(null)
|
||||
const [deckToEditId, setDeckToEditId] = useState<null | string>(null)
|
||||
|
||||
const showEditModal = !!deckToEditId
|
||||
|
||||
const {
|
||||
@@ -40,6 +40,7 @@ export const DecksPage = () => {
|
||||
setSort,
|
||||
sort,
|
||||
} = useDeckSearchParams()
|
||||
const [searchInputValue, setSearchInputValue] = useState<null | string>(search)
|
||||
|
||||
const currentUserId = me?.id
|
||||
const authorId = currentTab === 'my' ? currentUserId : undefined
|
||||
@@ -54,6 +55,7 @@ export const DecksPage = () => {
|
||||
const resetFilters = () => {
|
||||
setCurrentPage(null)
|
||||
setSearch(null)
|
||||
setSearchInputValue(null)
|
||||
setMinCards(null)
|
||||
setMaxCards(null)
|
||||
setRangeValue([0, decks?.maxCardsCount ?? null])
|
||||
@@ -72,9 +74,10 @@ export const DecksPage = () => {
|
||||
|
||||
const openCreateModal = () => setShowCreateModal(true)
|
||||
|
||||
const updateSearch = useDebounceCallback(setSearch, 500)
|
||||
const handleSearch = (search: null | string) => {
|
||||
setCurrentPage(null)
|
||||
setSearch(search)
|
||||
setSearchInputValue(search)
|
||||
}
|
||||
const handleSliderCommitted = (value: number[]) => {
|
||||
setCurrentPage(null)
|
||||
@@ -132,10 +135,13 @@ export const DecksPage = () => {
|
||||
</div>
|
||||
<div className={s.filters}>
|
||||
<TextField
|
||||
onValueChange={handleSearch}
|
||||
onValueChange={value => {
|
||||
updateSearch(value)
|
||||
handleSearch(value)
|
||||
}}
|
||||
placeholder={'Search'}
|
||||
search
|
||||
value={search ?? ''}
|
||||
value={searchInputValue ?? ''}
|
||||
/>
|
||||
<Tabs asChild onValueChange={handleTabChange} value={currentTab ?? undefined}>
|
||||
<TabsList>
|
||||
|
||||
Reference in New Issue
Block a user