146 lines
4.6 KiB
TypeScript
146 lines
4.6 KiB
TypeScript
|
|
import { useState } from "react";
|
||
|
|
import { useTranslation } from "react-i18next";
|
||
|
|
import {
|
||
|
|
Card,
|
||
|
|
CardContent,
|
||
|
|
CardDescription,
|
||
|
|
CardFooter,
|
||
|
|
CardHeader,
|
||
|
|
CardTitle,
|
||
|
|
} from "@/components/ui/card";
|
||
|
|
import { Button } from "@/components/ui/button";
|
||
|
|
import { Badge } from "@/components/ui/badge";
|
||
|
|
import { ExternalLink, Download, Trash2, Loader2 } from "lucide-react";
|
||
|
|
import { settingsApi } from "@/lib/api";
|
||
|
|
import type { Skill } from "@/lib/api/skills";
|
||
|
|
|
||
|
|
interface SkillCardProps {
|
||
|
|
skill: Skill;
|
||
|
|
onInstall: (directory: string) => Promise<void>;
|
||
|
|
onUninstall: (directory: string) => Promise<void>;
|
||
|
|
}
|
||
|
|
|
||
|
|
export function SkillCard({ skill, onInstall, onUninstall }: SkillCardProps) {
|
||
|
|
const { t } = useTranslation();
|
||
|
|
const [loading, setLoading] = useState(false);
|
||
|
|
|
||
|
|
const handleInstall = async () => {
|
||
|
|
setLoading(true);
|
||
|
|
try {
|
||
|
|
await onInstall(skill.directory);
|
||
|
|
} finally {
|
||
|
|
setLoading(false);
|
||
|
|
}
|
||
|
|
};
|
||
|
|
|
||
|
|
const handleUninstall = async () => {
|
||
|
|
setLoading(true);
|
||
|
|
try {
|
||
|
|
await onUninstall(skill.directory);
|
||
|
|
} finally {
|
||
|
|
setLoading(false);
|
||
|
|
}
|
||
|
|
};
|
||
|
|
|
||
|
|
const handleOpenGithub = async () => {
|
||
|
|
if (skill.readmeUrl) {
|
||
|
|
try {
|
||
|
|
await settingsApi.openExternal(skill.readmeUrl);
|
||
|
|
} catch (error) {
|
||
|
|
console.error("Failed to open URL:", error);
|
||
|
|
}
|
||
|
|
}
|
||
|
|
};
|
||
|
|
|
||
|
|
const showDirectory =
|
||
|
|
Boolean(skill.directory) &&
|
||
|
|
skill.directory.trim().toLowerCase() !== skill.name.trim().toLowerCase();
|
||
|
|
|
||
|
|
return (
|
||
|
|
<Card className="flex flex-col h-full border-border-default bg-card transition-[border-color,box-shadow] duration-200 hover:border-border-hover hover:shadow-md">
|
||
|
|
<CardHeader className="pb-3">
|
||
|
|
<div className="flex items-start justify-between gap-2">
|
||
|
|
<div className="flex-1 min-w-0">
|
||
|
|
<CardTitle className="text-base font-semibold truncate">
|
||
|
|
{skill.name}
|
||
|
|
</CardTitle>
|
||
|
|
<div className="flex items-center gap-2 mt-1.5">
|
||
|
|
{showDirectory && (
|
||
|
|
<CardDescription className="text-xs truncate">
|
||
|
|
{skill.directory}
|
||
|
|
</CardDescription>
|
||
|
|
)}
|
||
|
|
{skill.repoOwner && skill.repoName && (
|
||
|
|
<Badge
|
||
|
|
variant="outline"
|
||
|
|
className="shrink-0 text-[10px] px-1.5 py-0 h-4 border-border-default"
|
||
|
|
>
|
||
|
|
{skill.repoOwner}/{skill.repoName}
|
||
|
|
</Badge>
|
||
|
|
)}
|
||
|
|
</div>
|
||
|
|
</div>
|
||
|
|
{skill.installed && (
|
||
|
|
<Badge
|
||
|
|
variant="default"
|
||
|
|
className="shrink-0 bg-green-600/90 hover:bg-green-600 dark:bg-green-700/90 dark:hover:bg-green-700 text-white border-0"
|
||
|
|
>
|
||
|
|
{t("skills.installed")}
|
||
|
|
</Badge>
|
||
|
|
)}
|
||
|
|
</div>
|
||
|
|
</CardHeader>
|
||
|
|
<CardContent className="flex-1 pt-0">
|
||
|
|
<p className="text-sm text-muted-foreground/90 line-clamp-4 leading-relaxed">
|
||
|
|
{skill.description || t("skills.noDescription")}
|
||
|
|
</p>
|
||
|
|
</CardContent>
|
||
|
|
<CardFooter className="flex gap-2 pt-3 border-t border-border-default">
|
||
|
|
{skill.readmeUrl && (
|
||
|
|
<Button
|
||
|
|
variant="ghost"
|
||
|
|
size="sm"
|
||
|
|
onClick={handleOpenGithub}
|
||
|
|
disabled={loading}
|
||
|
|
className="flex-1"
|
||
|
|
>
|
||
|
|
<ExternalLink className="h-3.5 w-3.5 mr-1.5" />
|
||
|
|
{t("skills.view")}
|
||
|
|
</Button>
|
||
|
|
)}
|
||
|
|
{skill.installed ? (
|
||
|
|
<Button
|
||
|
|
variant="outline"
|
||
|
|
size="sm"
|
||
|
|
onClick={handleUninstall}
|
||
|
|
disabled={loading}
|
||
|
|
className="flex-1 border-red-200 text-red-600 hover:bg-red-50 hover:text-red-700 dark:border-red-900/50 dark:text-red-400 dark:hover:bg-red-950/50 dark:hover:text-red-300"
|
||
|
|
>
|
||
|
|
{loading ? (
|
||
|
|
<Loader2 className="h-3.5 w-3.5 mr-1.5 animate-spin" />
|
||
|
|
) : (
|
||
|
|
<Trash2 className="h-3.5 w-3.5 mr-1.5" />
|
||
|
|
)}
|
||
|
|
{loading ? t("skills.uninstalling") : t("skills.uninstall")}
|
||
|
|
</Button>
|
||
|
|
) : (
|
||
|
|
<Button
|
||
|
|
variant="mcp"
|
||
|
|
size="sm"
|
||
|
|
onClick={handleInstall}
|
||
|
|
disabled={loading || !skill.repoOwner}
|
||
|
|
className="flex-1"
|
||
|
|
>
|
||
|
|
{loading ? (
|
||
|
|
<Loader2 className="h-3.5 w-3.5 mr-1.5 animate-spin" />
|
||
|
|
) : (
|
||
|
|
<Download className="h-3.5 w-3.5 mr-1.5" />
|
||
|
|
)}
|
||
|
|
{loading ? t("skills.installing") : t("skills.install")}
|
||
|
|
</Button>
|
||
|
|
)}
|
||
|
|
</CardFooter>
|
||
|
|
</Card>
|
||
|
|
);
|
||
|
|
}
|