This commit is contained in:
+102
-105
@@ -131,6 +131,7 @@ export default function ProjectsPage() {
|
|||||||
setFormData({ title: "", description: "", project_status: "PRIVATE" });
|
setFormData({ title: "", description: "", project_status: "PRIVATE" });
|
||||||
setImportSnapshot(null);
|
setImportSnapshot(null);
|
||||||
setImportSnapshotName(null);
|
setImportSnapshotName(null);
|
||||||
|
if (importJsonInputRef.current) importJsonInputRef.current.value = "";
|
||||||
fetchProjects();
|
fetchProjects();
|
||||||
router.push(`/editor/${projectId}`);
|
router.push(`/editor/${projectId}`);
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
@@ -199,15 +200,17 @@ export default function ProjectsPage() {
|
|||||||
return 0;
|
return 0;
|
||||||
});
|
});
|
||||||
|
|
||||||
// Helper format ngày
|
|
||||||
const formatDate = (dateString: string | null | undefined) => {
|
const formatDate = (dateString: string | null | undefined) => {
|
||||||
if (!dateString) return "-";
|
if (!dateString) return "-";
|
||||||
const date = new Date(dateString);
|
const date = new Date(dateString);
|
||||||
return `Updated on ${date.toLocaleDateString("vi-VN", {
|
if (isNaN(date.getTime())) return "-";
|
||||||
|
return date.toLocaleDateString("vi-VN", {
|
||||||
day: "2-digit",
|
day: "2-digit",
|
||||||
month: "short",
|
month: "short",
|
||||||
year: "numeric",
|
year: "numeric",
|
||||||
})}`;
|
hour: "2-digit",
|
||||||
|
minute: "2-digit",
|
||||||
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
const getStatusBadge = (status: string) => {
|
const getStatusBadge = (status: string) => {
|
||||||
@@ -228,17 +231,16 @@ export default function ProjectsPage() {
|
|||||||
return (
|
return (
|
||||||
<button
|
<button
|
||||||
onClick={() => handleSort(column)}
|
onClick={() => handleSort(column)}
|
||||||
className={`w-20 text-sm font-medium text-left hover:text-blue-500 transition-colors ${
|
className={`flex items-center gap-1 text-sm font-medium hover:text-blue-500 transition-colors ${
|
||||||
isActive ? "text-blue-600 dark:text-blue-400" : "text-gray-500 dark:text-gray-400"
|
isActive ? "text-blue-600 dark:text-blue-400" : "text-gray-500 dark:text-gray-400"
|
||||||
}`}
|
}`}
|
||||||
>
|
>
|
||||||
{label} {isActive && (sortOrder === "asc" ? "↑" : "↓")}
|
<span>{label}</span>
|
||||||
|
{isActive && <span>{sortOrder === "asc" ? "↑" : "↓"}</span>}
|
||||||
</button>
|
</button>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
console.log(projects);
|
|
||||||
|
|
||||||
const importLabel = useMemo(() => {
|
const importLabel = useMemo(() => {
|
||||||
if (!importSnapshotName) return "Chưa chọn JSON snapshot";
|
if (!importSnapshotName) return "Chưa chọn JSON snapshot";
|
||||||
return `JSON: ${importSnapshotName}`;
|
return `JSON: ${importSnapshotName}`;
|
||||||
@@ -266,137 +268,134 @@ export default function ProjectsPage() {
|
|||||||
|
|
||||||
{!isLoading && sortedProjects.length > 0 ? (
|
{!isLoading && sortedProjects.length > 0 ? (
|
||||||
<div className="max-w-full overflow-x-auto">
|
<div className="max-w-full overflow-x-auto">
|
||||||
<div className="overflow-hidden rounded-xl border border-gray-200 bg-white dark:border-gray-800 dark:bg-[#0d1117] min-w-[700px]">
|
<div className="overflow-hidden rounded-xl border border-gray-200 bg-white dark:border-gray-800 dark:bg-[#0d1117] min-w-[800px]">
|
||||||
<div className="flex items-center justify-between px-5 py-3 border-b border-gray-200 dark:border-gray-800 bg-gray-50/50 dark:bg-[#161b22]">
|
<div className="flex items-center px-5 py-3 border-b border-gray-200 dark:border-gray-800 bg-gray-50/50 dark:bg-[#161b22]">
|
||||||
<span className="text-sm font-semibold text-gray-700 dark:text-gray-300 w-40"></span>
|
<div className="flex-1 pr-4">
|
||||||
<div className="flex items-center gap-4 shrink-0">
|
<SortButton column="title" label="Tên dự án" />
|
||||||
<span className="text-sm text-gray-500 dark:text-gray-400 w-20">Sắp xếp:</span>
|
</div>
|
||||||
<SortButton column="title" label="Tên" />
|
<div className="w-48 px-4 text-sm font-medium text-gray-500 dark:text-gray-400">Trạng thái</div>
|
||||||
<SortButton column="created_at" label="Ngày tạo" />
|
<div className="w-48 px-4 text-sm font-medium text-gray-500 dark:text-gray-400">Thành viên</div>
|
||||||
|
<div className="w-32 px-4">
|
||||||
<SortButton column="updated_at" label="Cập nhật" />
|
<SortButton column="updated_at" label="Cập nhật" />
|
||||||
</div>
|
</div>
|
||||||
|
<div className="w-48 px-4 text-sm font-medium text-gray-500 dark:text-gray-400 text-right">Thao tác</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className="flex flex-col divide-y divide-gray-200 dark:divide-gray-800">
|
<div className="flex flex-col divide-y divide-gray-200 dark:divide-gray-800">
|
||||||
{sortedProjects.map((project: any) => (
|
{sortedProjects.map((project: any) => (
|
||||||
<div
|
<div
|
||||||
key={project.id}
|
key={project.id}
|
||||||
className="group flex flex-col p-5 md:flex-row md:items-center justify-between hover:bg-gray-50 dark:hover:bg-[#161b22] transition-colors"
|
className="group flex items-center p-5 hover:bg-gray-50 dark:hover:bg-[#161b22]/50 transition-colors"
|
||||||
>
|
>
|
||||||
<div className="flex-1 pr-4 max-w-full md:max-w-[75%]">
|
<div className="flex-1 pr-4 min-w-0">
|
||||||
<div
|
<div className="items-center gap-3 mb-1.5">
|
||||||
onClick={() => router.push(`/user/projects/${project.id}`)}
|
<h3
|
||||||
className="flex items-center gap-2 mb-2 cursor-pointer hover:underline"
|
onClick={() => router.push(`/user/projects/${project.id}`)}
|
||||||
>
|
className="font-semibold text-blue-600 dark:text-[#58a6ff] truncate cursor-pointer hover:underline"
|
||||||
<div className="w-6 h-6 shrink-0 flex items-center justify-center">
|
>
|
||||||
|
{project.title}
|
||||||
|
</h3>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
<div className="flex items-center gap-2 text-xs text-gray-500 dark:text-[#8b949e]">
|
||||||
|
<div className="flex items-center gap-1.5">
|
||||||
{project.user?.avatar_url ? (
|
{project.user?.avatar_url ? (
|
||||||
<div className="relative w-6 h-6 rounded-full overflow-hidden border border-gray-200 dark:border-gray-800">
|
<Image src={project.user.avatar_url} alt="avatar" width={16} height={16} className="rounded-full object-cover" />
|
||||||
<Image
|
|
||||||
src={project.user.avatar_url}
|
|
||||||
alt="avatar"
|
|
||||||
fill
|
|
||||||
className="object-cover rounded-full"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
) : (
|
) : (
|
||||||
<div className="w-6 h-6 rounded-full bg-gray-200 dark:bg-gray-700 flex items-center justify-center border border-gray-300 dark:border-gray-600">
|
<div className="w-4 h-4 rounded-full bg-gray-200 dark:bg-gray-700 flex items-center justify-center">
|
||||||
<span className="text-[10px] font-bold text-gray-500 dark:text-gray-300 leading-none">
|
<span className="text-[8px] font-bold text-gray-500 dark:text-gray-300">
|
||||||
{project.user?.display_name?.charAt(0)?.toUpperCase() || "U"}
|
{project.user?.display_name?.charAt(0)?.toUpperCase() || "U"}
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
<span className="truncate max-w-[150px]">{project.user?.display_name || "Unknown"}</span>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className="flex items-center max-w-[250px]">
|
|
||||||
<span className="text-[14px] font-medium text-gray-700 dark:text-gray-300 truncate">
|
|
||||||
{project.user?.display_name || "Unknown"}
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<span className="text-[14px] text-gray-400 dark:text-gray-600 shrink-0">/</span>
|
|
||||||
|
|
||||||
<h3 className="text-[14px] font-semibold text-blue-600 dark:text-[#58a6ff] truncate max-w-[300px]">
|
|
||||||
{project.title}
|
|
||||||
</h3>
|
|
||||||
|
|
||||||
<div className="shrink-0 w-20 flex justify-start">
|
|
||||||
{getStatusBadge(project.project_status)}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div className="flex flex-wrap items-center gap-4 text-xs text-gray-500 dark:text-[#8b949e] h-5">
|
|
||||||
<span>{formatDate(project.updated_at)}</span>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className="flex flex-wrap items-center justify-start md:justify-end mt-4 md:mt-0 gap-3 shrink-0">
|
<div className="w-48 px-4 shrink-0">
|
||||||
<Button
|
{getStatusBadge(project.project_status)}
|
||||||
size="sm"
|
</div>
|
||||||
variant="outline"
|
|
||||||
onClick={() => router.push(`/editor/${project.id}`)}
|
<div className="w-48 px-4 shrink-0">
|
||||||
>
|
|
||||||
Editor
|
|
||||||
</Button>
|
|
||||||
<Button
|
|
||||||
size="sm"
|
|
||||||
variant="outline"
|
|
||||||
disabled={isExportingProjectId === String(project.id)}
|
|
||||||
onClick={() => handleExportHeadSnapshot(project)}
|
|
||||||
// title="Export head commit snapshot_json"
|
|
||||||
>
|
|
||||||
ExportJSON
|
|
||||||
</Button>
|
|
||||||
<Button
|
|
||||||
size="sm"
|
|
||||||
variant="outline"
|
|
||||||
onClick={() => router.push(`/editor/${project.id}?only=wiki`)}
|
|
||||||
>
|
|
||||||
Editor only wiki
|
|
||||||
</Button>
|
|
||||||
|
|
||||||
<div className="flex -space-x-2 overflow-hidden">
|
<div className="flex -space-x-2 overflow-hidden">
|
||||||
{project.members && project.members.length > 0 ? (
|
{project.members && project.members.length > 0 ? (
|
||||||
<>
|
<>
|
||||||
{project.members.slice(0, 4).map((m: any, index: number) =>
|
{project.members.slice(0, 4).map((m: any, index: number) =>
|
||||||
m.avatar_url ? (
|
m.avatar_url ? (
|
||||||
<Image
|
<Image key={index} src={m.avatar_url} alt={m.display_name} width={32} height={32} title={m.display_name} className="inline-block w-8 h-8 rounded-full object-cover ring-2 ring-white dark:ring-[#0d1117]" />
|
||||||
key={index}
|
|
||||||
src={m.avatar_url}
|
|
||||||
alt={m.display_name}
|
|
||||||
width={32}
|
|
||||||
height={32}
|
|
||||||
title={m.display_name}
|
|
||||||
className="inline-block w-8 h-8 rounded-full object-cover ring-2 ring-white group-hover:ring-gray-50 dark:ring-[#0d1117] dark:group-hover:ring-[#161b22] transition-colors"
|
|
||||||
/>
|
|
||||||
) : (
|
) : (
|
||||||
<div
|
<div key={index} title={m.display_name} className="inline-flex items-center justify-center w-8 h-8 rounded-full bg-gray-200 dark:bg-gray-700 ring-2 ring-white dark:ring-[#0d1117]">
|
||||||
key={index}
|
<span className="text-xs font-medium text-gray-500 dark:text-gray-300">{m.display_name?.charAt(0)?.toUpperCase() || "U"}</span>
|
||||||
title={m.display_name}
|
|
||||||
className="inline-flex items-center justify-center w-8 h-8 rounded-full bg-gray-200 dark:bg-gray-700 ring-2 ring-white group-hover:ring-gray-50 dark:ring-[#0d1117] dark:group-hover:ring-[#161b22] transition-colors"
|
|
||||||
>
|
|
||||||
<span className="text-xs font-medium text-gray-500 dark:text-gray-300">
|
|
||||||
{m.display_name?.charAt(0)?.toUpperCase() || "U"}
|
|
||||||
</span>
|
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{project.members.length > 4 && (
|
{project.members.length > 4 && (
|
||||||
<div
|
<div title="Những người khác" className="inline-flex items-center justify-center w-8 h-8 rounded-full bg-gray-100 dark:bg-gray-800 ring-2 ring-white dark:ring-[#0d1117] z-10">
|
||||||
title="Những người khác"
|
<span className="text-xs font-medium text-gray-600 dark:text-gray-400">+{project.members.length - 4}</span>
|
||||||
className="inline-flex items-center justify-center w-8 h-8 rounded-full bg-gray-100 dark:bg-gray-800 ring-2 ring-white group-hover:ring-gray-50 dark:ring-[#0d1117] dark:group-hover:ring-[#161b22] transition-colors z-10"
|
|
||||||
>
|
|
||||||
<span className="text-xs font-medium text-gray-600 dark:text-gray-400">
|
|
||||||
+{project.members.length - 4}
|
|
||||||
</span>
|
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
</>
|
</>
|
||||||
) : (
|
) : (
|
||||||
<span className="text-sm text-gray-400 dark:text-gray-600 italic"></span>
|
<span className="text-xs text-gray-400 dark:text-gray-600 italic"></span>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div className="w-32 px-1 shrink-0 text-xs text-gray-500 dark:text-gray-400">
|
||||||
|
{formatDate(project.updated_at)}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="w-48 px-4 shrink-0 flex justify-end gap-2">
|
||||||
|
<div className="relative group/btn1 inline-flex">
|
||||||
|
<Button
|
||||||
|
size="sm"
|
||||||
|
variant="outline"
|
||||||
|
className="!p-0 w-9 h-9 flex items-center justify-center"
|
||||||
|
onClick={() => router.push(`/editor/${project.id}`)}
|
||||||
|
>
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" strokeWidth={1.5} stroke="currentColor" className="w-4 h-4">
|
||||||
|
<path strokeLinecap="round" strokeLinejoin="round" d="M16.862 4.487l1.687-1.688a1.875 1.875 0 112.652 2.652L10.582 16.07a4.5 4.5 0 01-1.897 1.13L6 18l.8-2.685a4.5 4.5 0 011.13-1.897l8.932-8.931zm0 0L19.5 7.125M18 14v4.75A2.25 2.25 0 0115.75 21H5.25A2.25 2.25 0 013 18.75V8.25A2.25 2.25 0 015.25 6H10" />
|
||||||
|
</svg>
|
||||||
|
</Button>
|
||||||
|
<span className="absolute -top-8 left-1/2 -translate-x-1/2 scale-0 rounded bg-gray-900 px-2 py-1 text-[11px] font-medium text-white opacity-0 transition-all group-hover/btn1:scale-100 group-hover/btn1:opacity-100 z-50 pointer-events-none whitespace-nowrap shadow-sm dark:bg-gray-700">
|
||||||
|
Editor
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="relative group/btn2 inline-flex">
|
||||||
|
<Button
|
||||||
|
size="sm"
|
||||||
|
variant="outline"
|
||||||
|
className="!p-0 w-9 h-9 flex items-center justify-center"
|
||||||
|
disabled={isExportingProjectId === String(project.id)}
|
||||||
|
onClick={() => handleExportHeadSnapshot(project)}
|
||||||
|
>
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" strokeWidth={1.5} stroke="currentColor" className="w-4 h-4">
|
||||||
|
<path strokeLinecap="round" strokeLinejoin="round" d="M19.5 14.25v-2.625a3.375 3.375 0 00-3.375-3.375h-1.5A1.125 1.125 0 0113.5 7.125v-1.5a3.375 3.375 0 00-3.375-3.375H8.25m.75 12l3 3m0 0l3-3m-3 3v-6m-1.5-9H5.625c-.621 0-1.125.504-1.125 1.125v17.25c0 .621.504 1.125 1.125 1.125h12.75c.621 0 1.125-.504 1.125-1.125V11.25a9 9 0 00-9-9z" />
|
||||||
|
</svg>
|
||||||
|
</Button>
|
||||||
|
<span className="absolute -top-8 left-1/2 -translate-x-1/2 scale-0 rounded bg-gray-900 px-2 py-1 text-[11px] font-medium text-white opacity-0 transition-all group-hover/btn2:scale-100 group-hover/btn2:opacity-100 z-50 pointer-events-none whitespace-nowrap shadow-sm dark:bg-gray-700">
|
||||||
|
Export JSON
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="relative group/btn3 inline-flex">
|
||||||
|
<Button
|
||||||
|
size="sm"
|
||||||
|
variant="outline"
|
||||||
|
className="!p-0 w-9 h-9 flex items-center justify-center"
|
||||||
|
onClick={() => router.push(`/editor/${project.id}?only=wiki`)}
|
||||||
|
>
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" strokeWidth={1.5} stroke="currentColor" className="w-4 h-4">
|
||||||
|
<path strokeLinecap="round" strokeLinejoin="round" d="M12 6.042A8.967 8.967 0 006 3.75c-1.052 0-2.062.18-3 .512v14.25A8.987 8.987 0 016 18c2.305 0 4.408.867 6 2.292m0-14.25a8.966 8.966 0 016-2.292c1.052 0 2.062.18 3 .512v14.25A8.987 8.987 0 0018 18a8.967 8.967 0 00-6 2.292m0-14.25v14.25" />
|
||||||
|
</svg>
|
||||||
|
</Button>
|
||||||
|
<span className="absolute -top-8 left-1/2 -translate-x-1/2 scale-0 rounded bg-gray-900 px-2 py-1 text-[11px] font-medium text-white opacity-0 transition-all group-hover/btn3:scale-100 group-hover/btn3:opacity-100 z-50 pointer-events-none whitespace-nowrap shadow-sm dark:bg-gray-700">
|
||||||
|
Wiki Editor
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
))}
|
))}
|
||||||
</div>
|
</div>
|
||||||
@@ -414,7 +413,6 @@ export default function ProjectsPage() {
|
|||||||
</ComponentCard>
|
</ComponentCard>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* Modal Tạo Dự án */}
|
|
||||||
<Modal isOpen={isOpen} onClose={closeModal} className="max-w-[500px] m-4">
|
<Modal isOpen={isOpen} onClose={closeModal} className="max-w-[500px] m-4">
|
||||||
<div className="p-6 bg-white rounded-3xl dark:bg-gray-900">
|
<div className="p-6 bg-white rounded-3xl dark:bg-gray-900">
|
||||||
<h3 className="mb-5 text-xl font-bold text-gray-800 dark:text-white/90">Tạo dự án mới</h3>
|
<h3 className="mb-5 text-xl font-bold text-gray-800 dark:text-white/90">Tạo dự án mới</h3>
|
||||||
@@ -484,7 +482,6 @@ export default function ProjectsPage() {
|
|||||||
disabled={isSubmitting}
|
disabled={isSubmitting}
|
||||||
className="bg-gray-900 hover:bg-gray-800 text-white"
|
className="bg-gray-900 hover:bg-gray-800 text-white"
|
||||||
onClick={handleCreateProjectWithJson}
|
onClick={handleCreateProjectWithJson}
|
||||||
// title="Tạo dự án và tạo commit đầu tiên từ JSON snapshot"
|
|
||||||
>
|
>
|
||||||
Tạo với JSON
|
Tạo với JSON
|
||||||
</Button>
|
</Button>
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ type ButtonProps = React.ButtonHTMLAttributes<HTMLButtonElement> & {
|
|||||||
variant?: "primary" | "outline"; // Button variant
|
variant?: "primary" | "outline"; // Button variant
|
||||||
startIcon?: ReactNode; // Icon before the text
|
startIcon?: ReactNode; // Icon before the text
|
||||||
endIcon?: ReactNode; // Icon after the text
|
endIcon?: ReactNode; // Icon after the text
|
||||||
|
title?: string; // Title text
|
||||||
};
|
};
|
||||||
|
|
||||||
const Button: React.FC<ButtonProps> = ({
|
const Button: React.FC<ButtonProps> = ({
|
||||||
@@ -16,6 +17,7 @@ const Button: React.FC<ButtonProps> = ({
|
|||||||
className = "",
|
className = "",
|
||||||
disabled = false,
|
disabled = false,
|
||||||
type = "button",
|
type = "button",
|
||||||
|
title,
|
||||||
...rest
|
...rest
|
||||||
}) => {
|
}) => {
|
||||||
// Size Classes
|
// Size Classes
|
||||||
@@ -39,6 +41,7 @@ const Button: React.FC<ButtonProps> = ({
|
|||||||
} ${variantClasses[variant]} ${
|
} ${variantClasses[variant]} ${
|
||||||
disabled ? "cursor-not-allowed opacity-50" : ""
|
disabled ? "cursor-not-allowed opacity-50" : ""
|
||||||
}`}
|
}`}
|
||||||
|
title={title}
|
||||||
disabled={disabled}
|
disabled={disabled}
|
||||||
type={type}
|
type={type}
|
||||||
{...rest}
|
{...rest}
|
||||||
|
|||||||
Reference in New Issue
Block a user