- 修复 folder-aciton.ts 文件名拼写错误为 folder-action.ts - 修复所有导入路径中的拼写错误 - 添加 repoGetPublicFolderById 和 actionGetPublicFolderById - 创建 ExploreDetailClient 详情页组件 - /explore/[id] 现在显示文件夹详情和链接到 /folders/[id] - 添加 exploreDetail 中英文翻译
144 lines
4.3 KiB
TypeScript
144 lines
4.3 KiB
TypeScript
"use client";
|
|
|
|
import {
|
|
ChevronRight,
|
|
Folder as Fd,
|
|
Heart,
|
|
} from "lucide-react";
|
|
import { useEffect, useState } from "react";
|
|
import { useRouter } from "next/navigation";
|
|
import { useTranslations } from "next-intl";
|
|
import { toast } from "sonner";
|
|
import { PageLayout } from "@/components/ui/PageLayout";
|
|
import { PageHeader } from "@/components/ui/PageHeader";
|
|
import { CardList } from "@/components/ui/CardList";
|
|
import { actionGetUserFavorites, actionToggleFavorite } from "@/modules/folder/folder-action";
|
|
|
|
type UserFavorite = {
|
|
id: number;
|
|
folderId: number;
|
|
folderName: string;
|
|
folderCreatedAt: Date;
|
|
folderTotalPairs: number;
|
|
folderOwnerId: string;
|
|
folderOwnerName: string | null;
|
|
folderOwnerUsername: string | null;
|
|
favoritedAt: Date;
|
|
};
|
|
|
|
interface FavoriteCardProps {
|
|
favorite: UserFavorite;
|
|
onRemoveFavorite: (folderId: number) => void;
|
|
}
|
|
|
|
const FavoriteCard = ({ favorite, onRemoveFavorite }: FavoriteCardProps) => {
|
|
const router = useRouter();
|
|
const t = useTranslations("favorites");
|
|
const [isRemoving, setIsRemoving] = useState(false);
|
|
|
|
const handleRemoveFavorite = async (e: React.MouseEvent) => {
|
|
e.stopPropagation();
|
|
if (isRemoving) return;
|
|
|
|
setIsRemoving(true);
|
|
const result = await actionToggleFavorite(favorite.folderId);
|
|
if (result.success) {
|
|
onRemoveFavorite(favorite.folderId);
|
|
} else {
|
|
toast.error(result.message);
|
|
}
|
|
setIsRemoving(false);
|
|
};
|
|
|
|
return (
|
|
<div
|
|
className="flex justify-between items-center group py-4 px-5 border-b border-gray-200 last:border-b-0 hover:bg-gray-50 cursor-pointer transition-colors"
|
|
onClick={() => {
|
|
router.push(`/explore/${favorite.folderId}`);
|
|
}}
|
|
>
|
|
<div className="flex items-center gap-4 flex-1">
|
|
<div className="shrink-0 text-primary-500">
|
|
<Fd size={24} />
|
|
</div>
|
|
|
|
<div className="flex-1 min-w-0">
|
|
<h3 className="font-semibold text-gray-900 truncate">{favorite.folderName}</h3>
|
|
<p className="text-sm text-gray-500 mt-0.5">
|
|
{t("folderInfo", {
|
|
userName: favorite.folderOwnerName ?? favorite.folderOwnerUsername ?? t("unknownUser"),
|
|
totalPairs: favorite.folderTotalPairs,
|
|
})}
|
|
</p>
|
|
</div>
|
|
</div>
|
|
|
|
<div className="flex items-center gap-2">
|
|
<Heart
|
|
size={18}
|
|
className="fill-red-500 text-red-500 cursor-pointer hover:scale-110 transition-transform"
|
|
onClick={handleRemoveFavorite}
|
|
/>
|
|
<ChevronRight size={20} className="text-gray-400" />
|
|
</div>
|
|
</div>
|
|
);
|
|
};
|
|
|
|
interface FavoritesClientProps {
|
|
userId: string;
|
|
}
|
|
|
|
export function FavoritesClient({ userId }: FavoritesClientProps) {
|
|
const t = useTranslations("favorites");
|
|
const [favorites, setFavorites] = useState<UserFavorite[]>([]);
|
|
const [loading, setLoading] = useState(true);
|
|
|
|
useEffect(() => {
|
|
loadFavorites();
|
|
}, [userId]);
|
|
|
|
const loadFavorites = async () => {
|
|
setLoading(true);
|
|
const result = await actionGetUserFavorites();
|
|
if (result.success && result.data) {
|
|
setFavorites(result.data);
|
|
}
|
|
setLoading(false);
|
|
};
|
|
|
|
const handleRemoveFavorite = (folderId: number) => {
|
|
setFavorites((prev) => prev.filter((f) => f.folderId !== folderId));
|
|
};
|
|
|
|
return (
|
|
<PageLayout>
|
|
<PageHeader title={t("title")} subtitle={t("subtitle")} />
|
|
|
|
<CardList>
|
|
{loading ? (
|
|
<div className="p-8 text-center">
|
|
<div className="w-8 h-8 border-2 border-gray-200 border-t-gray-400 rounded-full animate-spin mx-auto mb-3"></div>
|
|
<p className="text-sm text-gray-500">{t("loading")}</p>
|
|
</div>
|
|
) : favorites.length === 0 ? (
|
|
<div className="text-center py-12 text-gray-400">
|
|
<div className="w-16 h-16 mx-auto mb-3 rounded-full bg-gray-100 flex items-center justify-center">
|
|
<Heart size={24} className="text-gray-400" />
|
|
</div>
|
|
<p className="text-sm">{t("noFavorites")}</p>
|
|
</div>
|
|
) : (
|
|
favorites.map((favorite) => (
|
|
<FavoriteCard
|
|
key={favorite.id}
|
|
favorite={favorite}
|
|
onRemoveFavorite={handleRemoveFavorite}
|
|
/>
|
|
))
|
|
)}
|
|
</CardList>
|
|
</PageLayout>
|
|
);
|
|
}
|