mirror of
https://github.com/blakeblackshear/frigate.git
synced 2026-06-30 17:11:14 +03:00
Some checks failed
CI / AMD64 Build (push) Has been cancelled
CI / ARM Build (push) Has been cancelled
CI / Jetson Jetpack 6 (push) Has been cancelled
CI / AMD64 Extra Build (push) Has been cancelled
CI / ARM Extra Build (push) Has been cancelled
CI / Synaptics Build (push) Has been cancelled
CI / Assemble and push default build (push) Has been cancelled
* remove redundant per-view toasters in settings * add variants to standardize dialog footer button layouts * remove text-md this class name compiles to nothing in tailwind. we used to add it to prevent iOS from zooming when focusing on an input, but that is now solved via the viewport meta in index.html * make wizard footers consistent with dialog footers * consistent destructive button style remove text-white from individual buttons and add it to the variant
164 lines
4.8 KiB
TypeScript
164 lines
4.8 KiB
TypeScript
import React, { useCallback, useMemo, useRef, useState } from "react";
|
|
import { IconType } from "react-icons";
|
|
import * as LuIcons from "react-icons/lu";
|
|
import { Input } from "@/components/ui/input";
|
|
import {
|
|
Popover,
|
|
PopoverContent,
|
|
PopoverTrigger,
|
|
} from "@/components/ui/popover";
|
|
import { IoClose } from "react-icons/io5";
|
|
import Heading from "../ui/heading";
|
|
import { cn } from "@/lib/utils";
|
|
import { Button } from "../ui/button";
|
|
|
|
import { useTranslation } from "react-i18next";
|
|
|
|
export type IconName = keyof typeof LuIcons;
|
|
|
|
export type IconElement = {
|
|
name?: string;
|
|
Icon?: IconType;
|
|
};
|
|
|
|
type IconPickerProps = {
|
|
selectedIcon?: IconElement;
|
|
setSelectedIcon?: React.Dispatch<
|
|
React.SetStateAction<IconElement | undefined>
|
|
>;
|
|
};
|
|
|
|
export default function IconPicker({
|
|
selectedIcon,
|
|
setSelectedIcon,
|
|
}: IconPickerProps) {
|
|
const { t } = useTranslation(["components/icons"]);
|
|
const [open, setOpen] = useState(false);
|
|
const containerRef = useRef<HTMLDivElement>(null);
|
|
const [searchTerm, setSearchTerm] = useState("");
|
|
|
|
const iconSets = useMemo(() => [...Object.entries(LuIcons)], []);
|
|
|
|
const icons = useMemo(
|
|
() =>
|
|
iconSets.filter(
|
|
([name]) =>
|
|
name.toLowerCase().includes(searchTerm.toLowerCase()) ||
|
|
searchTerm === "",
|
|
),
|
|
[iconSets, searchTerm],
|
|
);
|
|
|
|
const handleIconSelect = useCallback(
|
|
({ name, Icon }: IconElement) => {
|
|
if (setSelectedIcon) {
|
|
setSelectedIcon({ name, Icon });
|
|
}
|
|
setSearchTerm("");
|
|
},
|
|
[setSelectedIcon],
|
|
);
|
|
|
|
return (
|
|
<div ref={containerRef}>
|
|
<Popover
|
|
open={open}
|
|
onOpenChange={(open) => {
|
|
setOpen(open);
|
|
}}
|
|
>
|
|
<PopoverTrigger asChild>
|
|
{!selectedIcon?.name || !selectedIcon?.Icon ? (
|
|
<Button
|
|
className="mt-2 w-full text-muted-foreground"
|
|
aria-label={t("iconPicker.selectIcon")}
|
|
>
|
|
{t("iconPicker.selectIcon")}
|
|
</Button>
|
|
) : (
|
|
<div className="hover:cursor-pointer">
|
|
<div className="my-3 flex w-full flex-row items-center justify-between gap-2">
|
|
<div className="flex flex-row items-center gap-2">
|
|
<selectedIcon.Icon size={15} />
|
|
<div className="text-sm">
|
|
{selectedIcon.name
|
|
.replace(/^Lu/, "")
|
|
.replace(/([A-Z])/g, " $1")}
|
|
</div>
|
|
</div>
|
|
|
|
<IoClose
|
|
className="mx-2 hover:cursor-pointer"
|
|
onClick={() => {
|
|
handleIconSelect({ name: undefined, Icon: undefined });
|
|
}}
|
|
/>
|
|
</div>
|
|
</div>
|
|
)}
|
|
</PopoverTrigger>
|
|
<PopoverContent
|
|
align="start"
|
|
side="top"
|
|
container={containerRef.current}
|
|
className="flex max-h-[50dvh] flex-col overflow-y-hidden md:max-h-[30dvh]"
|
|
>
|
|
<div className="mb-3 flex flex-row items-center justify-between">
|
|
<Heading as="h4">{t("iconPicker.selectIcon")}</Heading>
|
|
<span tabIndex={0} className="sr-only" />
|
|
<IoClose
|
|
size={15}
|
|
className="hover:cursor-pointer"
|
|
onClick={() => {
|
|
setOpen(false);
|
|
}}
|
|
/>
|
|
</div>
|
|
<Input
|
|
type="text"
|
|
placeholder={t("iconPicker.search.placeholder", {
|
|
ns: "components/icons",
|
|
})}
|
|
className="mb-3 md:text-sm"
|
|
value={searchTerm}
|
|
onChange={(e) => setSearchTerm(e.target.value)}
|
|
/>
|
|
<div className="scrollbar-container flex h-full flex-col overflow-y-auto">
|
|
<div className="grid grid-cols-6 gap-2 pr-1">
|
|
{icons.map(([name, Icon]) => (
|
|
<div
|
|
key={name}
|
|
className={cn(
|
|
"flex flex-row items-center justify-center rounded-lg p-1 hover:cursor-pointer",
|
|
selectedIcon?.name === name
|
|
? "bg-selected text-white"
|
|
: "hover:bg-secondary-foreground",
|
|
)}
|
|
>
|
|
<Icon
|
|
className="size-6"
|
|
onClick={() => {
|
|
handleIconSelect({ name, Icon });
|
|
setOpen(false);
|
|
}}
|
|
/>
|
|
</div>
|
|
))}
|
|
</div>
|
|
</div>
|
|
</PopoverContent>
|
|
</Popover>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
type IconRendererProps = {
|
|
icon: IconType;
|
|
size?: number;
|
|
className?: string;
|
|
};
|
|
|
|
export function IconRenderer({ icon, size, className }: IconRendererProps) {
|
|
return <>{React.createElement(icon, { size, className })}</>;
|
|
}
|