"use client"; import Grid from "@mui/material/Grid"; import Card from "@mui/material/Card"; import CardContent from "@mui/material/CardContent"; import Typography from "@mui/material/Typography"; import React, { SyntheticEvent, useCallback, useMemo, useState } from "react"; import { useTranslation } from "react-i18next"; import TextField from "@mui/material/TextField"; import FormControl from "@mui/material/FormControl"; import InputLabel from "@mui/material/InputLabel"; import Select, { SelectChangeEvent } from "@mui/material/Select"; import MenuItem from "@mui/material/MenuItem"; import CardActions from "@mui/material/CardActions"; import Button from "@mui/material/Button"; import RestartAlt from "@mui/icons-material/RestartAlt"; import Search from "@mui/icons-material/Search"; import dayjs from "dayjs"; import "dayjs/locale/zh-hk"; import { DatePicker } from "@mui/x-date-pickers/DatePicker"; import { LocalizationProvider } from "@mui/x-date-pickers/LocalizationProvider"; import { AdapterDayjs } from "@mui/x-date-pickers/AdapterDayjs"; import { Autocomplete, Box, Checkbox, Chip, ListSubheader, } from "@mui/material"; import MultiSelect from "@/components/SearchBox/MultiSelect"; import { intersectionWith } from "lodash"; interface BaseCriterion { label: string; label2?: string; paramName: T; paramName2?: T; // options?: T[] | string[]; filterObj?: T; handleSelectionChange?: (selectedOptions: T[]) => void; } interface OptionWithLabel { label: string; value: any; } interface TextCriterion extends BaseCriterion { type: "text"; } interface SelectCriterion extends BaseCriterion { type: "select"; options: string[]; } interface SelectWithLabelCriterion extends BaseCriterion { type: "select-labelled"; options: OptionWithLabel[]; } interface MultiSelectCriterion extends BaseCriterion { type: "multi-select"; options: T[]; selectedOptions: T[]; handleSelectionChange: (selectedOptions: T[]) => void; } interface AutocompleteOptions { value: string | number; label: string; group?: string; } interface AutocompleteCriterion extends BaseCriterion { type: "autocomplete"; options: AutocompleteOptions[]; multiple?: boolean; noOptionsText?: string; needAll?: boolean; } interface DateRangeCriterion extends BaseCriterion { type: "dateRange"; } interface DateCriterion extends BaseCriterion { type: "date"; } export type Criterion = | TextCriterion | SelectCriterion | SelectWithLabelCriterion | DateRangeCriterion | DateCriterion | MultiSelectCriterion | AutocompleteCriterion; interface Props { criteria: Criterion[]; // TODO: may need to check the type is "autocomplete" and "multiple" = true, then allow string[]. // TODO: may need to check the type is "dateRange", then add T and `${T}To` in the same time. // onSearch: (inputs: Record["type"] extends "dateRange" ? `${T}To` : never), string>) => void; onSearch: (inputs: Record) => void; onReset?: () => void; } function SearchBox({ criteria, onSearch, onReset, }: Props) { const { t } = useTranslation("common"); const defaultAll: AutocompleteOptions = { value: "All", label: t("All"), group: t("All"), }; const defaultInputs = useMemo( () => criteria.reduce>( (acc, c) => { let tempCriteria = { ...acc, [c.paramName]: c.type === "select" || c.type === "select-labelled" || (c.type === "autocomplete" && !Boolean(c.multiple)) ? "All" : c.type === "autocomplete" && Boolean(c.multiple) ? [defaultAll.value] : "", }; if (c.type === "dateRange") { tempCriteria = { ...tempCriteria, [c.paramName]: "", [`${c.paramName}To`]: "", }; } return tempCriteria; }, {} as Record, ), [criteria], ); const [inputs, setInputs] = useState(defaultInputs); const [isReset, setIsReset] = useState(false); const makeInputChangeHandler = useCallback( (paramName: T): React.ChangeEventHandler => { return (e) => { setInputs((i) => ({ ...i, [paramName]: e.target.value })); }; }, [], ); const makeSelectChangeHandler = useCallback((paramName: T) => { return (e: SelectChangeEvent) => { setInputs((i) => ({ ...i, [paramName]: e.target.value })); }; }, []); const makeAutocompleteChangeHandler = useCallback( (paramName: T, multiple: boolean) => { return ( e: SyntheticEvent, newValue: AutocompleteOptions | AutocompleteOptions[], ) => { if (multiple) { const multiNewValue = newValue as AutocompleteOptions[]; setInputs((i) => ({ ...i, [paramName]: multiNewValue.map(({ value }) => value), })); } else { const singleNewValue = newValue as AutocompleteOptions; setInputs((i) => ({ ...i, [paramName]: singleNewValue.value })); } }; }, [], ); const makeDateChangeHandler = useCallback((paramName: T) => { return (e: any) => { setInputs((i) => ({ ...i, [paramName]: dayjs(e).format("YYYY-MM-DD") })); }; }, []); const makeDateToChangeHandler = useCallback((paramName: T) => { return (e: any) => { setInputs((i) => ({ ...i, [paramName + "To"]: dayjs(e).format("YYYY-MM-DD"), })); }; }, []); const handleReset = () => { setInputs(defaultInputs); onReset?.(); setIsReset(!isReset); }; const handleSearch = () => { onSearch(inputs); }; return ( {t("Search Criteria")} {criteria.map((c) => { return ( {c.type === "text" && ( )} {/* eslint-disable-next-line @typescript-eslint/no-unused-vars */} {/* {c.type === "multi-select" && ( )} */} {c.type === "select" && ( {t(c.label)} )} {c.type === "select-labelled" && ( {t(c.label)} )} {c.type === "autocomplete" && ( option.group !== "All") .length > 0 && c.options.every((option) => option.group) ? (option) => option.group && option.group.trim() !== "" ? option.group : "Ungrouped" : undefined } multiple={Boolean(c.multiple)} noOptionsText={c.noOptionsText ?? t("No options")} disableClearable fullWidth value={ c.multiple ? intersectionWith( [defaultAll, ...c.options], inputs[c.paramName], (option, v) => { return option.value === (v ?? ""); }, ) : c.options.find( (option) => option.value === inputs[c.paramName], ) ?? defaultAll } onChange={makeAutocompleteChangeHandler( c.paramName, Boolean(c.multiple), )} getOptionLabel={(option) => option.label} options={[defaultAll, ...c.options]} disableCloseOnSelect={Boolean(c.multiple)} renderGroup={ c.options.every((option) => option.group) ? (params) => ( {params.group} {params.children} ) : undefined } renderTags={ c.multiple ? (value, getTagProps) => value.map((option, index) => { // eslint-disable-next-line @typescript-eslint/no-unused-vars const { key, ...chipProps } = getTagProps({ index, }); return ( ); }) : undefined } renderOption={( params: React.HTMLAttributes & { key?: React.Key; }, option, { selected }, ) => { // eslint-disable-next-line @typescript-eslint/no-unused-vars const { key, ...rest } = params; return ( {c.multiple && ( )} {option.label} ); }} renderInput={(params) => ( )} /> )} {c.type === "dateRange" && ( {"-"} )} {c.type === "date" && ( )} ); })} ); } export default SearchBox;