From 7bd76bbadc69862e0c423f931accd0333dfe91c4 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Richard=20Jedli=C4=8Dka?= Date: Wed, 6 Dec 2023 16:13:21 +0100 Subject: [PATCH] search autocomplete --- src/components/SearchInput.tsx | 95 ++++++++++++++++--- src/hooks/useAutocompleteSearchQuery.ts | 12 +++ .../runtimeMetadataAutocomplete.ts | 6 ++ src/repositories/runtimeMetadataRepository.ts | 8 +- src/screens/home.tsx | 3 +- src/services/runtimeMetadataService.ts | 40 +++++--- src/services/searchService.ts | 64 ++++++++++++- src/workers/runtimeSpecWorker.runtime.ts | 4 +- 8 files changed, 192 insertions(+), 40 deletions(-) create mode 100644 src/hooks/useAutocompleteSearchQuery.ts create mode 100644 src/model/runtime-metadata/runtimeMetadataAutocomplete.ts diff --git a/src/components/SearchInput.tsx b/src/components/SearchInput.tsx index f38e9a1e..fdea5de7 100644 --- a/src/components/SearchInput.tsx +++ b/src/components/SearchInput.tsx @@ -1,10 +1,11 @@ /** @jsxImportSource @emotion/react */ -import { FormHTMLAttributes, useCallback, useEffect, useState } from "react"; +import { FormHTMLAttributes, useCallback, useEffect, useMemo, useState } from "react"; import { useNavigate, useSearchParams } from "react-router-dom"; -import { Button, FormGroup, TextField } from "@mui/material"; +import { Autocomplete, Button, FormGroup, TextField, debounce } from "@mui/material"; import SearchIcon from "@mui/icons-material/Search"; import { css, Theme } from "@emotion/react"; +import { useAutocompleteSearchQuery } from "../hooks/useAutocompleteSearchQuery"; import { Network } from "../model/network"; import { getNetworks } from "../services/networksService"; @@ -17,7 +18,7 @@ const formGroupStyle = css` `; const networkSelectStyle = (theme: Theme) => css` - flex: 1 0 auto; + flex: 0 0 auto; border-top-right-radius: 0; border-bottom-right-radius: 0; @@ -53,6 +54,18 @@ const networkSelectStyle = (theme: Theme) => css` } `; +const inputStyle = css` + flex: 1 0 auto; + + .MuiOutlinedInput-root { + padding: 0 !important; + + .MuiAutocomplete-input { + padding: 12px 16px; + } + } +`; + const textFieldStyle = css` .MuiInputBase-root { border-radius: 0; @@ -70,6 +83,25 @@ const textFieldStyle = css` } `; +const autocompleteNameStyle = css` + flex: 1 1 auto; + overflow: hidden; + text-overflow: ellipsis; + padding-right: 16px; + font-size: 14px; +`; + +const autocompleteTypeStyle = css` + margin-left: auto; + flex: 0 0 auto; + font-size: 12px; + opacity: .75; + border: solid 1px gray; + border-radius: 8px; + padding: 0 4px; + background-color: rgba(0, 0, 0, .025); +`; + const buttonStyle = (theme: Theme) => css` border-radius: 8px; border-top-left-radius: 0px; @@ -99,6 +131,14 @@ const buttonStyle = (theme: Theme) => css` } `; +function storeNetworks(networks: Network[]) { + localStorage.setItem("networks", JSON.stringify(networks.map(it => it.name))); +} + +function loadNetworks() { + return getNetworks(JSON.parse(localStorage.getItem("networks") || "[]")); +} + export type SearchInputProps = FormHTMLAttributes & { persist?: boolean; defaultNetworks?: Network[]; @@ -109,13 +149,15 @@ function SearchInput(props: SearchInputProps) { const [qs] = useSearchParams(); + const navigate = useNavigate(); + const [networks, setNetworks] = useState(defaultNetworks || getNetworks(qs.getAll("network") || [])); const [query, setQuery] = useState(qs.get("query") || ""); + const [autocompleteQuery, _setAutocompleteQuery] = useState(query || ""); - const navigate = useNavigate(); + const setAutocompleteQuery = useMemo(() => debounce(_setAutocompleteQuery, 250), []); - const storeNetworks = (networks: Network[]) => localStorage.setItem("networks", JSON.stringify(networks.map(it => it.name))); - const loadNetworks = () => getNetworks(JSON.parse(localStorage.getItem("networks") || "[]")); + const autocompleteSuggestions = useAutocompleteSearchQuery(autocompleteQuery, networks); const handleNetworkSelect = useCallback((networks: Network[], isUserAction: boolean) => { if (isUserAction && persist) { @@ -126,6 +168,11 @@ function SearchInput(props: SearchInputProps) { setNetworks(networks); }, [persist]); + const handleQueryChange = useCallback((ev: any, value: string) => { + setQuery(value); + setAutocompleteQuery(value); + }, []); + const handleSubmit = useCallback((ev: any) => { ev.preventDefault(); @@ -166,13 +213,35 @@ function SearchInput(props: SearchInputProps) { value={networks} multiselect /> - setQuery(e.target.value)} - placeholder="Extrinsic hash / account address / block hash / block height / extrinsic name / event name" - value={query} + it} + inputValue={query} + onInputChange={handleQueryChange} + renderOption={(props, option) => ( +
  • +
    + {option.label.slice(0, option.highlight[0])} + {option.label.slice(option.highlight[0], option.highlight[1])} + {option.label.slice(option.highlight[1])} +
    +
    {option.type}
    +
  • + )} + renderInput={(params) => + + } />