Files
clash-verge-rev-lite/src/components/profile/group-item.tsx
2025-07-14 05:23:32 +03:00

127 lines
3.6 KiB
TypeScript

import { useEffect, useState } from "react";
import { useSortable } from "@dnd-kit/sortable";
import { CSS } from "@dnd-kit/utilities";
import { downloadIconCache } from "@/services/cmds";
import { convertFileSrc } from "@tauri-apps/api/core";
import { cn } from "@root/lib/utils";
// Новые импорты
import { Button } from "@/components/ui/button";
import { Badge } from "@/components/ui/badge";
import { GripVertical, Trash2, Undo2 } from "lucide-react";
interface Props {
type: "prepend" | "original" | "delete" | "append";
group: IProxyGroupConfig;
onDelete: () => void;
}
// Определяем стили для каждого типа элемента
const typeStyles = {
original: "bg-secondary/50",
delete: "bg-destructive/20 text-muted-foreground line-through",
prepend: "bg-green-500/20",
append: "bg-green-500/20",
};
export const GroupItem = (props: Props) => {
const { type, group, onDelete } = props;
// Drag-and-drop будет работать только для 'prepend' и 'append' типов
const isSortable = type === "prepend" || type === "append";
const {
attributes,
listeners,
setNodeRef,
transform,
transition,
isDragging,
} = useSortable({ id: group.name, disabled: !isSortable });
const [iconCachePath, setIconCachePath] = useState("");
const getFileName = (url: string) => url.substring(url.lastIndexOf("/") + 1);
async function initIconCachePath() {
if (group.icon && group.icon.trim().startsWith("http")) {
const fileName =
group.name.replaceAll(" ", "") + "-" + getFileName(group.icon);
const iconPath = await downloadIconCache(group.icon, fileName);
setIconCachePath(convertFileSrc(iconPath));
}
}
useEffect(() => {
initIconCachePath();
}, [group.icon, group.name]);
const style = {
transform: CSS.Transform.toString(transform),
transition,
zIndex: isDragging ? 100 : undefined,
};
return (
<div
ref={setNodeRef}
style={style}
// Применяем стили в зависимости от типа
className={cn(
"flex items-center p-2 mb-1 rounded-lg transition-shadow",
typeStyles[type],
isDragging && "shadow-lg",
)}
>
{/* Ручка для перетаскивания */}
<div
{...attributes}
{...listeners}
className={cn(
"p-1 text-muted-foreground rounded-sm",
isSortable ? "cursor-move hover:bg-accent" : "cursor-default",
)}
>
<GripVertical className="h-5 w-5" />
</div>
{/* Иконка группы */}
{group.icon && (
<img
src={
group.icon.startsWith("data")
? group.icon
: group.icon.startsWith("<svg")
? `data:image/svg+xml;base64,${btoa(group.icon ?? "")}`
: iconCachePath || group.icon
}
className="w-8 h-8 mx-2 rounded-md"
alt={group.name}
/>
)}
{/* Название и тип группы */}
<div className="flex-1 min-w-0">
<p className="text-sm font-semibold truncate">{group.name}</p>
<div className="flex items-center text-xs text-muted-foreground mt-1">
<Badge variant="outline">{group.type}</Badge>
</div>
</div>
{/* Кнопка действия */}
<Button
variant="ghost"
size="icon"
className="h-8 w-8"
onClick={onDelete}
>
{type === "delete" ? (
<Undo2 className="h-4 w-4" />
) : (
<Trash2 className="h-4 w-4 text-destructive" />
)}
</Button>
</div>
);
};