feat(explore): 添加文件夹详情页面
- 修复 folder-aciton.ts 文件名拼写错误为 folder-action.ts - 修复所有导入路径中的拼写错误 - 添加 repoGetPublicFolderById 和 actionGetPublicFolderById - 创建 ExploreDetailClient 详情页组件 - /explore/[id] 现在显示文件夹详情和链接到 /folders/[id] - 添加 exploreDetail 中英文翻译
This commit is contained in:
@@ -297,6 +297,20 @@
|
|||||||
"sortByFavorites": "Sort by favorites",
|
"sortByFavorites": "Sort by favorites",
|
||||||
"sortByFavoritesActive": "Undo sort by favorites"
|
"sortByFavoritesActive": "Undo sort by favorites"
|
||||||
},
|
},
|
||||||
|
"exploreDetail": {
|
||||||
|
"title": "Folder Details",
|
||||||
|
"createdBy": "Created by: {name}",
|
||||||
|
"unknownUser": "Unknown User",
|
||||||
|
"totalPairs": "Total Pairs",
|
||||||
|
"favorites": "Favorites",
|
||||||
|
"createdAt": "Created At",
|
||||||
|
"viewContent": "View Content",
|
||||||
|
"favorite": "Favorite",
|
||||||
|
"unfavorite": "Unfavorite",
|
||||||
|
"favorited": "Favorited",
|
||||||
|
"unfavorited": "Unfavorited",
|
||||||
|
"pleaseLogin": "Please login first"
|
||||||
|
},
|
||||||
"favorites": {
|
"favorites": {
|
||||||
"title": "My Favorites",
|
"title": "My Favorites",
|
||||||
"subtitle": "Folders you've favorited",
|
"subtitle": "Folders you've favorited",
|
||||||
|
|||||||
@@ -297,6 +297,20 @@
|
|||||||
"sortByFavorites": "按收藏数排序",
|
"sortByFavorites": "按收藏数排序",
|
||||||
"sortByFavoritesActive": "取消按收藏数排序"
|
"sortByFavoritesActive": "取消按收藏数排序"
|
||||||
},
|
},
|
||||||
|
"exploreDetail": {
|
||||||
|
"title": "文件夹详情",
|
||||||
|
"createdBy": "创建者:{name}",
|
||||||
|
"unknownUser": "未知用户",
|
||||||
|
"totalPairs": "词对数量",
|
||||||
|
"favorites": "收藏数",
|
||||||
|
"createdAt": "创建时间",
|
||||||
|
"viewContent": "查看内容",
|
||||||
|
"favorite": "收藏",
|
||||||
|
"unfavorite": "取消收藏",
|
||||||
|
"favorited": "已收藏",
|
||||||
|
"unfavorited": "已取消收藏",
|
||||||
|
"pleaseLogin": "请先登录"
|
||||||
|
},
|
||||||
"favorites": {
|
"favorites": {
|
||||||
"title": "我的收藏",
|
"title": "我的收藏",
|
||||||
"subtitle": "收藏的公开文件夹",
|
"subtitle": "收藏的公开文件夹",
|
||||||
|
|||||||
@@ -11,7 +11,7 @@ import { Plus, RefreshCw } from "lucide-react";
|
|||||||
import { DictionaryEntry } from "./DictionaryEntry";
|
import { DictionaryEntry } from "./DictionaryEntry";
|
||||||
import { LanguageSelector } from "./LanguageSelector";
|
import { LanguageSelector } from "./LanguageSelector";
|
||||||
import { authClient } from "@/lib/auth-client";
|
import { authClient } from "@/lib/auth-client";
|
||||||
import { actionGetFoldersByUserId, actionCreatePair } from "@/modules/folder/folder-aciton";
|
import { actionGetFoldersByUserId, actionCreatePair } from "@/modules/folder/folder-action";
|
||||||
import { TSharedFolder } from "@/shared/folder-type";
|
import { TSharedFolder } from "@/shared/folder-type";
|
||||||
import { toast } from "sonner";
|
import { toast } from "sonner";
|
||||||
|
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import { DictionaryClient } from "./DictionaryClient";
|
import { DictionaryClient } from "./DictionaryClient";
|
||||||
import { auth } from "@/auth";
|
import { auth } from "@/auth";
|
||||||
import { headers } from "next/headers";
|
import { headers } from "next/headers";
|
||||||
import { actionGetFoldersByUserId } from "@/modules/folder/folder-aciton";
|
import { actionGetFoldersByUserId } from "@/modules/folder/folder-action";
|
||||||
import { TSharedFolder } from "@/shared/folder-type";
|
import { TSharedFolder } from "@/shared/folder-type";
|
||||||
|
|
||||||
export default async function DictionaryPage() {
|
export default async function DictionaryPage() {
|
||||||
|
|||||||
@@ -17,7 +17,7 @@ import {
|
|||||||
actionSearchPublicFolders,
|
actionSearchPublicFolders,
|
||||||
actionToggleFavorite,
|
actionToggleFavorite,
|
||||||
actionCheckFavorite,
|
actionCheckFavorite,
|
||||||
} from "@/modules/folder/folder-aciton";
|
} from "@/modules/folder/folder-action";
|
||||||
import { TPublicFolder } from "@/shared/folder-type";
|
import { TPublicFolder } from "@/shared/folder-type";
|
||||||
import { authClient } from "@/lib/auth-client";
|
import { authClient } from "@/lib/auth-client";
|
||||||
|
|
||||||
|
|||||||
146
src/app/(features)/explore/[id]/ExploreDetailClient.tsx
Normal file
146
src/app/(features)/explore/[id]/ExploreDetailClient.tsx
Normal file
@@ -0,0 +1,146 @@
|
|||||||
|
"use client";
|
||||||
|
|
||||||
|
import { Folder as Fd, Heart, ExternalLink, ArrowLeft } from "lucide-react";
|
||||||
|
import { CircleButton } from "@/design-system/base/button";
|
||||||
|
import { useEffect, useState } from "react";
|
||||||
|
import { useRouter } from "next/navigation";
|
||||||
|
import { useTranslations } from "next-intl";
|
||||||
|
import { toast } from "sonner";
|
||||||
|
import Link from "next/link";
|
||||||
|
import {
|
||||||
|
actionToggleFavorite,
|
||||||
|
actionCheckFavorite,
|
||||||
|
} from "@/modules/folder/folder-action";
|
||||||
|
import { ActionOutputPublicFolder } from "@/modules/folder/folder-action-dto";
|
||||||
|
import { authClient } from "@/lib/auth-client";
|
||||||
|
|
||||||
|
interface ExploreDetailClientProps {
|
||||||
|
folder: ActionOutputPublicFolder;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function ExploreDetailClient({ folder }: ExploreDetailClientProps) {
|
||||||
|
const router = useRouter();
|
||||||
|
const t = useTranslations("exploreDetail");
|
||||||
|
const [isFavorited, setIsFavorited] = useState(false);
|
||||||
|
const [favoriteCount, setFavoriteCount] = useState(folder.favoriteCount);
|
||||||
|
|
||||||
|
const { data: session } = authClient.useSession();
|
||||||
|
const currentUserId = session?.user?.id;
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (currentUserId) {
|
||||||
|
actionCheckFavorite(folder.id).then((result) => {
|
||||||
|
if (result.success && result.data) {
|
||||||
|
setIsFavorited(result.data.isFavorited);
|
||||||
|
setFavoriteCount(result.data.favoriteCount);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}, [folder.id, currentUserId]);
|
||||||
|
|
||||||
|
const handleToggleFavorite = async () => {
|
||||||
|
if (!currentUserId) {
|
||||||
|
toast.error(t("pleaseLogin"));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const result = await actionToggleFavorite(folder.id);
|
||||||
|
if (result.success && result.data) {
|
||||||
|
setIsFavorited(result.data.isFavorited);
|
||||||
|
setFavoriteCount(result.data.favoriteCount);
|
||||||
|
toast.success(
|
||||||
|
result.data.isFavorited ? t("favorited") : t("unfavorited")
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
toast.error(result.message);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const formatDate = (date: Date) => {
|
||||||
|
return new Intl.DateTimeFormat("zh-CN", {
|
||||||
|
year: "numeric",
|
||||||
|
month: "long",
|
||||||
|
day: "numeric",
|
||||||
|
}).format(new Date(date));
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="min-h-screen bg-gray-50">
|
||||||
|
<div className="max-w-3xl mx-auto px-4 py-6 sm:py-8">
|
||||||
|
<div className="flex items-center gap-3 mb-6">
|
||||||
|
<CircleButton onClick={() => router.push("/explore")}>
|
||||||
|
<ArrowLeft size={18} />
|
||||||
|
</CircleButton>
|
||||||
|
<h1 className="text-lg sm:text-xl font-semibold text-gray-900">
|
||||||
|
{t("title")}
|
||||||
|
</h1>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="bg-white border border-gray-200 rounded-xl p-5 sm:p-8 shadow-sm">
|
||||||
|
<div className="flex items-start justify-between mb-6">
|
||||||
|
<div className="flex items-center gap-4">
|
||||||
|
<div className="w-14 h-14 sm:w-16 sm:h-16 rounded-xl bg-primary-50 flex items-center justify-center text-primary-500">
|
||||||
|
<Fd size={28} className="sm:w-8 sm:h-8" />
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<h2 className="text-xl sm:text-2xl font-bold text-gray-900">
|
||||||
|
{folder.name}
|
||||||
|
</h2>
|
||||||
|
<p className="text-sm text-gray-500 mt-1">
|
||||||
|
{t("createdBy", {
|
||||||
|
name: folder.userName ?? folder.userUsername ?? t("unknownUser"),
|
||||||
|
})}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<CircleButton
|
||||||
|
onClick={handleToggleFavorite}
|
||||||
|
title={isFavorited ? t("unfavorite") : t("favorite")}
|
||||||
|
className="shrink-0"
|
||||||
|
>
|
||||||
|
<Heart
|
||||||
|
size={20}
|
||||||
|
className={isFavorited ? "fill-red-500 text-red-500" : ""}
|
||||||
|
/>
|
||||||
|
</CircleButton>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="grid grid-cols-3 gap-4 mb-6 py-4 border-y border-gray-100">
|
||||||
|
<div className="text-center">
|
||||||
|
<div className="text-2xl sm:text-3xl font-bold text-primary-600">
|
||||||
|
{folder.totalPairs}
|
||||||
|
</div>
|
||||||
|
<div className="text-xs sm:text-sm text-gray-500 mt-1">
|
||||||
|
{t("totalPairs")}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="text-center border-x border-gray-100">
|
||||||
|
<div className="text-2xl sm:text-3xl font-bold text-red-500 flex items-center justify-center gap-1">
|
||||||
|
<Heart size={18} className={isFavorited ? "fill-red-500" : ""} />
|
||||||
|
{favoriteCount}
|
||||||
|
</div>
|
||||||
|
<div className="text-xs sm:text-sm text-gray-500 mt-1">
|
||||||
|
{t("favorites")}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="text-center">
|
||||||
|
<div className="text-lg sm:text-xl font-semibold text-gray-700">
|
||||||
|
{formatDate(folder.createdAt)}
|
||||||
|
</div>
|
||||||
|
<div className="text-xs sm:text-sm text-gray-500 mt-1">
|
||||||
|
{t("createdAt")}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<Link
|
||||||
|
href={`/folders/${folder.id}`}
|
||||||
|
className="flex items-center justify-center gap-2 w-full py-3 px-4 bg-primary-500 hover:bg-primary-600 text-white rounded-lg font-medium transition-colors"
|
||||||
|
>
|
||||||
|
<ExternalLink size={18} />
|
||||||
|
{t("viewContent")}
|
||||||
|
</Link>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
import { redirect } from "next/navigation";
|
import { redirect } from "next/navigation";
|
||||||
import { InFolder } from "@/app/folders/[folder_id]/InFolder";
|
import { ExploreDetailClient } from "./ExploreDetailClient";
|
||||||
import { actionGetFolderVisibility } from "@/modules/folder/folder-aciton";
|
import { actionGetPublicFolderById } from "@/modules/folder/folder-action";
|
||||||
|
|
||||||
export default async function ExploreFolderPage({
|
export default async function ExploreFolderPage({
|
||||||
params,
|
params,
|
||||||
@@ -13,17 +13,11 @@ export default async function ExploreFolderPage({
|
|||||||
redirect("/explore");
|
redirect("/explore");
|
||||||
}
|
}
|
||||||
|
|
||||||
const folderInfo = (await actionGetFolderVisibility(Number(id))).data;
|
const result = await actionGetPublicFolderById(Number(id));
|
||||||
|
|
||||||
if (!folderInfo) {
|
if (!result.success || !result.data) {
|
||||||
redirect("/explore");
|
redirect("/explore");
|
||||||
}
|
}
|
||||||
|
|
||||||
const isPublic = folderInfo.visibility === "PUBLIC";
|
return <ExploreDetailClient folder={result.data} />;
|
||||||
|
|
||||||
if (!isPublic) {
|
|
||||||
redirect("/explore");
|
|
||||||
}
|
|
||||||
|
|
||||||
return <InFolder folderId={Number(id)} isReadOnly={true} />;
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { ExploreClient } from "./ExploreClient";
|
import { ExploreClient } from "./ExploreClient";
|
||||||
import { actionGetPublicFolders } from "@/modules/folder/folder-aciton";
|
import { actionGetPublicFolders } from "@/modules/folder/folder-action";
|
||||||
|
|
||||||
export default async function ExplorePage() {
|
export default async function ExplorePage() {
|
||||||
const publicFoldersResult = await actionGetPublicFolders();
|
const publicFoldersResult = await actionGetPublicFolders();
|
||||||
|
|||||||
@@ -12,7 +12,7 @@ import { toast } from "sonner";
|
|||||||
import { PageLayout } from "@/components/ui/PageLayout";
|
import { PageLayout } from "@/components/ui/PageLayout";
|
||||||
import { PageHeader } from "@/components/ui/PageHeader";
|
import { PageHeader } from "@/components/ui/PageHeader";
|
||||||
import { CardList } from "@/components/ui/CardList";
|
import { CardList } from "@/components/ui/CardList";
|
||||||
import { actionGetUserFavorites, actionToggleFavorite } from "@/modules/folder/folder-aciton";
|
import { actionGetUserFavorites, actionToggleFavorite } from "@/modules/folder/folder-action";
|
||||||
|
|
||||||
type UserFavorite = {
|
type UserFavorite = {
|
||||||
id: number;
|
id: number;
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ import { FolderSelector } from "./FolderSelector";
|
|||||||
import { Memorize } from "./Memorize";
|
import { Memorize } from "./Memorize";
|
||||||
import { auth } from "@/auth";
|
import { auth } from "@/auth";
|
||||||
import { headers } from "next/headers";
|
import { headers } from "next/headers";
|
||||||
import { actionGetFoldersWithTotalPairsByUserId, actionGetPairsByFolderId } from "@/modules/folder/folder-aciton";
|
import { actionGetFoldersWithTotalPairsByUserId, actionGetPairsByFolderId } from "@/modules/folder/folder-action";
|
||||||
|
|
||||||
export default async function MemorizePage({
|
export default async function MemorizePage({
|
||||||
searchParams,
|
searchParams,
|
||||||
|
|||||||
@@ -23,7 +23,7 @@ import {
|
|||||||
actionGetFoldersWithTotalPairsByUserId,
|
actionGetFoldersWithTotalPairsByUserId,
|
||||||
actionRenameFolderById,
|
actionRenameFolderById,
|
||||||
actionSetFolderVisibility,
|
actionSetFolderVisibility,
|
||||||
} from "@/modules/folder/folder-aciton";
|
} from "@/modules/folder/folder-action";
|
||||||
import { TSharedFolderWithTotalPairs } from "@/shared/folder-type";
|
import { TSharedFolderWithTotalPairs } from "@/shared/folder-type";
|
||||||
|
|
||||||
interface FolderCardProps {
|
interface FolderCardProps {
|
||||||
|
|||||||
@@ -9,7 +9,7 @@ import { useTranslations } from "next-intl";
|
|||||||
import { PageLayout } from "@/components/ui/PageLayout";
|
import { PageLayout } from "@/components/ui/PageLayout";
|
||||||
import { PrimaryButton, CircleButton, LinkButton } from "@/design-system/base/button";
|
import { PrimaryButton, CircleButton, LinkButton } from "@/design-system/base/button";
|
||||||
import { CardList } from "@/components/ui/CardList";
|
import { CardList } from "@/components/ui/CardList";
|
||||||
import { actionCreatePair, actionDeletePairById, actionGetPairsByFolderId } from "@/modules/folder/folder-aciton";
|
import { actionCreatePair, actionDeletePairById, actionGetPairsByFolderId } from "@/modules/folder/folder-action";
|
||||||
import { TSharedPair } from "@/shared/folder-type";
|
import { TSharedPair } from "@/shared/folder-type";
|
||||||
import { toast } from "sonner";
|
import { toast } from "sonner";
|
||||||
|
|
||||||
|
|||||||
@@ -4,7 +4,7 @@ import { CircleButton } from "@/design-system/base/button";
|
|||||||
import { UpdateTextPairModal } from "./UpdateTextPairModal";
|
import { UpdateTextPairModal } from "./UpdateTextPairModal";
|
||||||
import { useTranslations } from "next-intl";
|
import { useTranslations } from "next-intl";
|
||||||
import { TSharedPair } from "@/shared/folder-type";
|
import { TSharedPair } from "@/shared/folder-type";
|
||||||
import { actionUpdatePairById } from "@/modules/folder/folder-aciton";
|
import { actionUpdatePairById } from "@/modules/folder/folder-action";
|
||||||
import { ActionInputUpdatePairById } from "@/modules/folder/folder-action-dto";
|
import { ActionInputUpdatePairById } from "@/modules/folder/folder-action-dto";
|
||||||
import { toast } from "sonner";
|
import { toast } from "sonner";
|
||||||
|
|
||||||
|
|||||||
@@ -3,7 +3,7 @@ import { getTranslations } from "next-intl/server";
|
|||||||
import { InFolder } from "./InFolder";
|
import { InFolder } from "./InFolder";
|
||||||
import { auth } from "@/auth";
|
import { auth } from "@/auth";
|
||||||
import { headers } from "next/headers";
|
import { headers } from "next/headers";
|
||||||
import { actionGetFolderVisibility } from "@/modules/folder/folder-aciton";
|
import { actionGetFolderVisibility } from "@/modules/folder/folder-action";
|
||||||
|
|
||||||
export default async function FoldersPage({
|
export default async function FoldersPage({
|
||||||
params,
|
params,
|
||||||
|
|||||||
@@ -62,6 +62,12 @@ export type ActionOutputGetPublicFolders = {
|
|||||||
data?: ActionOutputPublicFolder[];
|
data?: ActionOutputPublicFolder[];
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export type ActionOutputGetPublicFolderById = {
|
||||||
|
message: string;
|
||||||
|
success: boolean;
|
||||||
|
data?: ActionOutputPublicFolder;
|
||||||
|
};
|
||||||
|
|
||||||
export type ActionOutputSetFolderVisibility = {
|
export type ActionOutputSetFolderVisibility = {
|
||||||
message: string;
|
message: string;
|
||||||
success: boolean;
|
success: boolean;
|
||||||
|
|||||||
@@ -11,6 +11,7 @@ import {
|
|||||||
ActionInputUpdatePairById,
|
ActionInputUpdatePairById,
|
||||||
ActionOutputGetFoldersWithTotalPairsByUserId,
|
ActionOutputGetFoldersWithTotalPairsByUserId,
|
||||||
ActionOutputGetPublicFolders,
|
ActionOutputGetPublicFolders,
|
||||||
|
ActionOutputGetPublicFolderById,
|
||||||
ActionOutputSetFolderVisibility,
|
ActionOutputSetFolderVisibility,
|
||||||
ActionOutputToggleFavorite,
|
ActionOutputToggleFavorite,
|
||||||
ActionOutputCheckFavorite,
|
ActionOutputCheckFavorite,
|
||||||
@@ -30,6 +31,7 @@ import {
|
|||||||
repoGetFoldersWithTotalPairsByUserId,
|
repoGetFoldersWithTotalPairsByUserId,
|
||||||
repoGetPairsByFolderId,
|
repoGetPairsByFolderId,
|
||||||
repoGetPublicFolders,
|
repoGetPublicFolders,
|
||||||
|
repoGetPublicFolderById,
|
||||||
repoGetUserIdByFolderId,
|
repoGetUserIdByFolderId,
|
||||||
repoRenameFolderById,
|
repoRenameFolderById,
|
||||||
repoSearchPublicFolders,
|
repoSearchPublicFolders,
|
||||||
@@ -383,6 +385,32 @@ export async function actionSearchPublicFolders(query: string): Promise<ActionOu
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export async function actionGetPublicFolderById(folderId: number): Promise<ActionOutputGetPublicFolderById> {
|
||||||
|
try {
|
||||||
|
const folder = await repoGetPublicFolderById(folderId);
|
||||||
|
if (!folder) {
|
||||||
|
return {
|
||||||
|
success: false,
|
||||||
|
message: 'Folder not found.',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
return {
|
||||||
|
success: true,
|
||||||
|
message: 'success',
|
||||||
|
data: {
|
||||||
|
...folder,
|
||||||
|
visibility: folder.visibility as "PRIVATE" | "PUBLIC",
|
||||||
|
},
|
||||||
|
};
|
||||||
|
} catch (e) {
|
||||||
|
log.error("Operation failed", { error: e });
|
||||||
|
return {
|
||||||
|
success: false,
|
||||||
|
message: 'Unknown error occured.',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
export async function actionToggleFavorite(
|
export async function actionToggleFavorite(
|
||||||
folderId: number,
|
folderId: number,
|
||||||
): Promise<ActionOutputToggleFavorite> {
|
): Promise<ActionOutputToggleFavorite> {
|
||||||
@@ -171,6 +171,32 @@ export async function repoGetFolderVisibility(
|
|||||||
return folder;
|
return folder;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export async function repoGetPublicFolderById(
|
||||||
|
folderId: number,
|
||||||
|
): Promise<RepoOutputPublicFolder | null> {
|
||||||
|
const folder = await prisma.folder.findUnique({
|
||||||
|
where: { id: folderId, visibility: Visibility.PUBLIC },
|
||||||
|
include: {
|
||||||
|
_count: { select: { pairs: true, favorites: true } },
|
||||||
|
user: { select: { name: true, username: true } },
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!folder) return null;
|
||||||
|
|
||||||
|
return {
|
||||||
|
id: folder.id,
|
||||||
|
name: folder.name,
|
||||||
|
visibility: folder.visibility,
|
||||||
|
createdAt: folder.createdAt,
|
||||||
|
userId: folder.userId,
|
||||||
|
userName: folder.user?.name ?? "Unknown",
|
||||||
|
userUsername: folder.user?.username ?? "unknown",
|
||||||
|
totalPairs: folder._count.pairs,
|
||||||
|
favoriteCount: folder._count.favorites,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
export async function repoGetPublicFolders(
|
export async function repoGetPublicFolders(
|
||||||
input: RepoInputGetPublicFolders = {},
|
input: RepoInputGetPublicFolders = {},
|
||||||
): Promise<RepoOutputPublicFolder[]> {
|
): Promise<RepoOutputPublicFolder[]> {
|
||||||
|
|||||||
Reference in New Issue
Block a user