feat: add search filters

This commit is contained in:
Kentai Radiquum 2024-11-21 18:56:54 +05:00
parent e985b65252
commit 4fb996353d
Signed by: Radiquum
GPG key ID: 858E8EE696525EED
2 changed files with 81 additions and 24 deletions

View file

@ -88,7 +88,7 @@ export const ReleaseInfoInfo = (props: {
return ( return (
<div key={index} className="inline"> <div key={index} className="inline">
{index > 0 && ", "} {index > 0 && ", "}
<ReleaseInfoSearchLink title={studio} searchBy={1} /> <ReleaseInfoSearchLink title={studio} searchBy={"studio"} />
</div> </div>
); );
})} })}
@ -98,14 +98,14 @@ export const ReleaseInfoInfo = (props: {
{props.author && ( {props.author && (
<> <>
{"Автор: "} {"Автор: "}
<ReleaseInfoSearchLink title={props.author} searchBy={3} /> <ReleaseInfoSearchLink title={props.author} searchBy={"author"} />
{props.director && ", "} {props.director && ", "}
</> </>
)} )}
{props.director && ( {props.director && (
<> <>
{"Режиссёр: "} {"Режиссёр: "}
<ReleaseInfoSearchLink title={props.director} searchBy={2} /> <ReleaseInfoSearchLink title={props.director} searchBy={"director"} />
</> </>
)} )}
</Table.Cell> </Table.Cell>
@ -120,7 +120,7 @@ export const ReleaseInfoInfo = (props: {
return ( return (
<div key={index} className="inline"> <div key={index} className="inline">
{index > 0 && ", "} {index > 0 && ", "}
<ReleaseInfoSearchLink title={genre} searchBy={4} /> <ReleaseInfoSearchLink title={genre} searchBy={"tag"} />
</div> </div>
); );
})} })}

View file

@ -27,23 +27,46 @@ const fetcher = async (url: string) => {
const ListsMapping = { const ListsMapping = {
watching: { watching: {
name: "Смотрю", name: "Смотрю",
id: 1 id: 1,
}, },
planned: { planned: {
name: "В планах", name: "В планах",
id: 2 id: 2,
}, },
watched: { watched: {
name: "Просмотрено", name: "Просмотрено",
id: 3 id: 3,
}, },
delayed: { delayed: {
name: "Отложено", name: "Отложено",
id: 4 id: 4,
}, },
abandoned: { abandoned: {
name: "Заброшено", name: "Заброшено",
id: 5 id: 5,
},
};
const TagMapping = {
name: {
name: "Названию",
id: 0,
},
studio: {
name: "Студии",
id: 1,
},
director: {
name: "Режиссёру",
id: 2,
},
author: {
name: "Автору",
id: 3,
},
tag: {
name: "Тегу",
id: 4,
}, },
}; };
@ -54,7 +77,7 @@ export function SearchPage() {
const [searchVal, setSearchVal] = useState(searchParams.get("q") || ""); const [searchVal, setSearchVal] = useState(searchParams.get("q") || "");
const [where, setWhere] = useState(searchParams.get("where") || "releases"); const [where, setWhere] = useState(searchParams.get("where") || "releases");
const [searchBy, setSearchBy] = useState( const [searchBy, setSearchBy] = useState(
searchParams.get("searchBy") || null searchParams.get("searchBy") || "name"
); );
const [list, setList] = useState(searchParams.get("list") || "watching"); const [list, setList] = useState(searchParams.get("list") || "watching");
const [filtersModalOpen, setFiltersModalOpen] = useState(false); const [filtersModalOpen, setFiltersModalOpen] = useState(false);
@ -79,14 +102,12 @@ export function SearchPage() {
url.searchParams.set("where", where); url.searchParams.set("where", where);
} }
if (searchBy) {
url.searchParams.set("searchBy", searchBy);
}
if (where == "list" && list && ListsMapping.hasOwnProperty(list)) { if (where == "list" && list && ListsMapping.hasOwnProperty(list)) {
url.searchParams.set("list", ListsMapping[list].id); url.searchParams.set("list", ListsMapping[list].id);
} }
url.searchParams.set("searchBy", TagMapping[searchBy].id);
if (query) { if (query) {
url.searchParams.set("q", query); url.searchParams.set("q", query);
return url.toString(); return url.toString();
@ -252,6 +273,8 @@ export function SearchPage() {
list={list} list={list}
setList={setList} setList={setList}
isAuth={userStore.isAuth} isAuth={userStore.isAuth}
searchBy={searchBy}
setSearchBy={setSearchBy}
/> />
</> </>
); );
@ -265,16 +288,19 @@ const FiltersModal = (props: {
list: string; list: string;
setList: any; setList: any;
isAuth: boolean; isAuth: boolean;
searchBy: string;
setSearchBy: any;
}) => { }) => {
const router = useRouter(); const router = useRouter();
const [where, setWhere] = useState(props.where); const [where, setWhere] = useState(props.where);
const [list, setList] = useState(props.list); const [list, setList] = useState(props.list);
const [searchBy, setSearchBy] = useState(props.searchBy);
function _cancel() { function _cancel() {
setWhere(props.where); setWhere(props.where);
setList(props.list); setList(props.list);
props.setIsOpen(false) setSearchBy(props.searchBy);
props.setIsOpen(false);
} }
function _accept() { function _accept() {
@ -292,6 +318,9 @@ const FiltersModal = (props: {
Params.delete("list"); Params.delete("list");
} }
Params.set("searchBy", searchBy);
props.setSearchBy(searchBy);
const url = new URL(`/search?${Params.toString()}`, window.location.origin); const url = new URL(`/search?${Params.toString()}`, window.location.origin);
router.push(url.toString()); router.push(url.toString());
} }
@ -326,27 +355,55 @@ const FiltersModal = (props: {
</Button.Group> </Button.Group>
</div> </div>
</div> </div>
{props.isAuth && where == "list" && ListsMapping.hasOwnProperty(list) ? ( {props.isAuth &&
where == "list" &&
ListsMapping.hasOwnProperty(list) ? (
<div className="my-4"> <div className="my-4">
<div className="flex items-center justify-between"> <div className="flex items-center justify-between">
<p className="font-bold dark:text-white">Список</p> <p className="font-bold dark:text-white">Список</p>
<Dropdown label={ListsMapping[list].name} color="blue"> <Dropdown label={ListsMapping[list].name} color="blue">
<Dropdown.Item onClick={() => setList("watching")}>Смотрю</Dropdown.Item> {Object.keys(ListsMapping).map((item) => {
<Dropdown.Item onClick={() => setList("planned")}>В планах</Dropdown.Item> return (
<Dropdown.Item onClick={() => setList("watched")}>Просмотрено</Dropdown.Item> <Dropdown.Item
<Dropdown.Item onClick={() => setList("delayed")}>Отложено</Dropdown.Item> onClick={() => setList(item)}
<Dropdown.Item onClick={() => setList("abandoned")}>Брошено</Dropdown.Item> key={`list--${item}`}
>
{ListsMapping[item].name}
</Dropdown.Item>
);
})}
</Dropdown> </Dropdown>
</div> </div>
</div> </div>
) : ( ) : (
"" ""
)} )}
<div className="my-4">
<div className="flex items-center justify-between">
<p className="font-bold dark:text-white">Искать по</p>
<Dropdown label={TagMapping[searchBy].name} color="blue">
{Object.keys(TagMapping).map((item) => {
return (
<Dropdown.Item
onClick={() => setSearchBy(item)}
key={`tag--${item}`}
>
{TagMapping[item].name}
</Dropdown.Item>
);
})}
</Dropdown>
</div>
</div>
</Modal.Body> </Modal.Body>
<Modal.Footer> <Modal.Footer>
<div className="flex justify-end w-full gap-2"> <div className="flex justify-end w-full gap-2">
<Button color="red" onClick={() => _cancel()}>Отменить</Button> <Button color="red" onClick={() => _cancel()}>
<Button color="blue" onClick={() => _accept()}>Применить</Button> Отменить
</Button>
<Button color="blue" onClick={() => _accept()}>
Применить
</Button>
</div> </div>
</Modal.Footer> </Modal.Footer>
</Modal> </Modal>