541 lines
17 KiB
TypeScript
541 lines
17 KiB
TypeScript
"use client";
|
|
import {
|
|
BannerIcon,
|
|
CopyIcon,
|
|
CreateIconIon,
|
|
DeleteIcon,
|
|
DotsYIcon,
|
|
EyeIconMdi,
|
|
SearchIcon,
|
|
} from "@/components/icons";
|
|
import { close, error, loading, success, successToast } from "@/config/swal";
|
|
import { Article } from "@/types/globals";
|
|
import { convertDateFormat, formatDate } from "@/utils/global";
|
|
import Link from "next/link";
|
|
import { Key, useCallback, useEffect, useState } from "react";
|
|
import Swal from "sweetalert2";
|
|
import withReactContent from "sweetalert2-react-content";
|
|
import Cookies from "js-cookie";
|
|
import {
|
|
deleteArticle,
|
|
getArticleByCategory,
|
|
getArticlePagination,
|
|
updateIsBannerArticle,
|
|
} from "@/service/article";
|
|
import {
|
|
DropdownMenu,
|
|
DropdownMenuContent,
|
|
DropdownMenuItem,
|
|
DropdownMenuTrigger,
|
|
} from "../ui/dropdown-menu";
|
|
import { Button } from "../ui/button";
|
|
import { Input } from "../ui/input";
|
|
import {
|
|
Select,
|
|
SelectTrigger,
|
|
SelectValue,
|
|
SelectContent,
|
|
SelectItem,
|
|
} from "@/components/ui/select";
|
|
import {
|
|
Table,
|
|
TableHeader,
|
|
TableBody,
|
|
TableRow,
|
|
TableHead,
|
|
TableCell,
|
|
} from "@/components/ui/table";
|
|
import CustomPagination from "../layout/custom-pagination";
|
|
import DatePicker from "react-datepicker";
|
|
|
|
const columns = [
|
|
{ name: "No", uid: "no" },
|
|
{ name: "Judul", uid: "title" },
|
|
{ name: "Banner", uid: "isBanner" },
|
|
{ name: "Kategori", uid: "category" },
|
|
{ name: "Tanggal Unggah", uid: "createdAt" },
|
|
{ name: "Kreator", uid: "customCreatorName" },
|
|
{ name: "Status", uid: "isPublish" },
|
|
{ name: "Aksi", uid: "actions" },
|
|
];
|
|
const columnsOtherRole = [
|
|
{ name: "No", uid: "no" },
|
|
{ name: "Judul", uid: "title" },
|
|
{ name: "Source", uid: "source" },
|
|
{ name: "Kategori", uid: "category" },
|
|
{ name: "Tanggal Unggah", uid: "createdAt" },
|
|
{ name: "Kreator", uid: "customCreatorName" },
|
|
{ name: "Status", uid: "publishStatus" },
|
|
{ name: "Aksi", uid: "actions" },
|
|
];
|
|
|
|
// interface Category {
|
|
// id: number;
|
|
// title: string;
|
|
// }
|
|
|
|
export default function ArticleTable() {
|
|
const MySwal = withReactContent(Swal);
|
|
const username = Cookies.get("username");
|
|
const userId = Cookies.get("uie");
|
|
|
|
const [page, setPage] = useState(1);
|
|
const [totalPage, setTotalPage] = useState(1);
|
|
const [article, setArticle] = useState<any[]>([]);
|
|
const [showData, setShowData] = useState("10");
|
|
const [search, setSearch] = useState("");
|
|
const [categories, setCategories] = useState<any>([]);
|
|
const [selectedCategories, setSelectedCategories] = useState<any>("");
|
|
const [selectedCategoryId, setSelectedCategoryId] = useState<any>("");
|
|
const [selectedSource, setSelectedSource] = useState<any>("");
|
|
const [selectedStatus, setSelectedStatus] = useState<string>("");
|
|
const [dateRange, setDateRange] = useState<any>({
|
|
startDate: null,
|
|
endDate: null,
|
|
});
|
|
|
|
useEffect(() => {
|
|
initState();
|
|
getCategories();
|
|
}, []);
|
|
|
|
async function getCategories() {
|
|
const res = await getArticleByCategory();
|
|
const data = res?.data?.data;
|
|
setCategories(data);
|
|
console.log("category", data);
|
|
}
|
|
|
|
useEffect(() => {
|
|
initState();
|
|
}, [
|
|
page,
|
|
showData,
|
|
search,
|
|
selectedCategoryId,
|
|
selectedSource,
|
|
dateRange,
|
|
selectedStatus,
|
|
]);
|
|
|
|
async function initState() {
|
|
loading();
|
|
const req = {
|
|
limit: showData,
|
|
page: page,
|
|
search: search,
|
|
category: selectedCategoryId || "",
|
|
source: selectedSource || "",
|
|
isPublish:
|
|
selectedStatus !== "" ? selectedStatus === "publish" : undefined,
|
|
startDate: formatDate(dateRange.startDate),
|
|
endDate: formatDate(dateRange.endDate),
|
|
sort: "desc",
|
|
sortBy: "created_at",
|
|
};
|
|
|
|
const res = await getArticlePagination(req);
|
|
|
|
let data = res.data?.data || [];
|
|
|
|
await getTableNumber(parseInt(showData), data);
|
|
setTotalPage(res?.data?.meta?.totalPage);
|
|
close();
|
|
}
|
|
// panggil ulang setiap state berubah
|
|
useEffect(() => {
|
|
initState();
|
|
}, [page, showData, search, selectedCategories]);
|
|
|
|
const getTableNumber = async (limit: number, data: Article[]) => {
|
|
if (data) {
|
|
const startIndex = limit * (page - 1);
|
|
let iterate = 0;
|
|
const newData = data.map((value: any) => {
|
|
iterate++;
|
|
value.no = startIndex + iterate;
|
|
return value;
|
|
});
|
|
setArticle(newData);
|
|
} else {
|
|
setArticle([]);
|
|
}
|
|
};
|
|
|
|
async function doDelete(id: any) {
|
|
// loading();
|
|
const resDelete = await deleteArticle(id);
|
|
|
|
if (resDelete?.error) {
|
|
error(resDelete.message);
|
|
return false;
|
|
}
|
|
close();
|
|
success("Berhasil Hapus");
|
|
initState();
|
|
}
|
|
|
|
const handleDelete = (id: any) => {
|
|
MySwal.fire({
|
|
title: "Hapus Data",
|
|
icon: "warning",
|
|
showCancelButton: true,
|
|
cancelButtonColor: "#3085d6",
|
|
confirmButtonColor: "#d33",
|
|
confirmButtonText: "Hapus",
|
|
}).then((result) => {
|
|
if (result.isConfirmed) {
|
|
doDelete(id);
|
|
}
|
|
});
|
|
};
|
|
|
|
const handleBanner = async (id: number, status: boolean) => {
|
|
const res = await updateIsBannerArticle(id, status);
|
|
if (res?.error) {
|
|
error(res?.message);
|
|
return false;
|
|
}
|
|
initState();
|
|
};
|
|
|
|
const copyUrlArticle = async (slug: any) => {
|
|
const url =
|
|
`${window.location.protocol}//${window.location.host}` +
|
|
"/details/" +
|
|
`${slug}`;
|
|
try {
|
|
await navigator.clipboard.writeText(url);
|
|
successToast("Success", "Article Copy to Clipboard");
|
|
setTimeout(() => {}, 1500);
|
|
} catch (err) {
|
|
("Failed to copy!");
|
|
}
|
|
};
|
|
|
|
const renderCell = useCallback(
|
|
(article: any, columnKey: Key) => {
|
|
const cellValue = article[columnKey as keyof any];
|
|
|
|
switch (columnKey) {
|
|
case "customCreatorName":
|
|
return (
|
|
<p>
|
|
{article.customCreatorName &&
|
|
article.customCreatorName.trim() !== ""
|
|
? article.customCreatorName
|
|
: article.createdByName}
|
|
</p>
|
|
);
|
|
case "publishStatus":
|
|
return (
|
|
// <Chip
|
|
// className="capitalize "
|
|
// color={statusColorMap[article.status]}
|
|
// size="lg"
|
|
// variant="flat"
|
|
// >
|
|
// <div className="flex flex-row items-center gap-2 justify-center">
|
|
// {article.status}
|
|
// </div>
|
|
// </Chip>
|
|
<p>{article.publishStatus}</p>
|
|
);
|
|
case "isBanner":
|
|
return <p>{article.isBanner ? "Ya" : "Tidak"}</p>;
|
|
case "createdAt":
|
|
return <p>{convertDateFormat(article.createdAt)}</p>;
|
|
case "category":
|
|
return (
|
|
<p>
|
|
{article?.categories?.map((list: any) => list.title).join(", ") +
|
|
" "}
|
|
</p>
|
|
);
|
|
|
|
case "actions":
|
|
return (
|
|
<div className="relative flex items-center gap-2">
|
|
<DropdownMenu>
|
|
<DropdownMenuTrigger asChild>
|
|
<Button variant="ghost" size="icon">
|
|
<DotsYIcon className="h-5 w-5 text-muted-foreground" />
|
|
</Button>
|
|
</DropdownMenuTrigger>
|
|
<DropdownMenuContent className="w-56">
|
|
<DropdownMenuItem
|
|
onClick={() => copyUrlArticle(article.slug)}
|
|
>
|
|
<CopyIcon className="mr-2 h-4 w-4" />
|
|
Copy Url Article
|
|
</DropdownMenuItem>
|
|
|
|
<DropdownMenuItem asChild>
|
|
<Link
|
|
href={`/admin/article/detail/${article.id}`}
|
|
className="flex items-center"
|
|
>
|
|
<EyeIconMdi className="mr-2 h-4 w-4" />
|
|
Detail
|
|
</Link>
|
|
</DropdownMenuItem>
|
|
|
|
<DropdownMenuItem asChild>
|
|
<Link
|
|
href={`/admin/article/edit/${article.id}`}
|
|
className="flex items-center"
|
|
>
|
|
<CreateIconIon className="mr-2 h-4 w-4" />
|
|
Edit
|
|
</Link>
|
|
</DropdownMenuItem>
|
|
|
|
{username === "admin-mabes" && (
|
|
<DropdownMenuItem
|
|
onClick={() =>
|
|
handleBanner(article.id, !article.isBanner)
|
|
}
|
|
>
|
|
<BannerIcon className="mr-2 h-4 w-4" />
|
|
{article.isBanner
|
|
? "Hapus dari Banner"
|
|
: "Jadikan Banner"}
|
|
</DropdownMenuItem>
|
|
)}
|
|
|
|
<DropdownMenuItem onClick={() => handleDelete(article.id)}>
|
|
<DeleteIcon className="mr-2 h-4 w-4 text-red-500" />
|
|
Delete
|
|
</DropdownMenuItem>
|
|
</DropdownMenuContent>
|
|
</DropdownMenu>
|
|
</div>
|
|
);
|
|
|
|
default:
|
|
return cellValue;
|
|
}
|
|
},
|
|
[article, page],
|
|
);
|
|
|
|
let typingTimer: NodeJS.Timeout;
|
|
const doneTypingInterval = 1500;
|
|
|
|
const handleKeyUp = () => {
|
|
clearTimeout(typingTimer);
|
|
typingTimer = setTimeout(doneTyping, doneTypingInterval);
|
|
};
|
|
|
|
const handleKeyDown = () => {
|
|
clearTimeout(typingTimer);
|
|
};
|
|
|
|
async function doneTyping() {
|
|
setPage(1);
|
|
initState();
|
|
}
|
|
|
|
return (
|
|
<>
|
|
<div className="py-3">
|
|
<div className="flex flex-col items-start rounded-2xl gap-3">
|
|
<div className="flex flex-col md:flex-row gap-3 w-full">
|
|
<div className="flex flex-col gap-1 w-full lg:w-1/3">
|
|
<p className="font-semibold text-sm">Pencarian</p>
|
|
<div className="relative">
|
|
<SearchIcon className="absolute left-3 top-1/2 -translate-y-1/2 text-muted-foreground h-4 w-4 pointer-events-none" />
|
|
<Input
|
|
type="text"
|
|
placeholder="Cari..."
|
|
className="pl-9 text-sm bg-muted"
|
|
onChange={(e) => setSearch(e.target.value)}
|
|
onKeyUp={handleKeyUp}
|
|
onKeyDown={handleKeyDown}
|
|
/>
|
|
</div>
|
|
</div>
|
|
<div className="flex flex-col gap-1 w-full lg:w-[72px]">
|
|
<p className="font-semibold text-sm">Data</p>
|
|
<Select
|
|
value={showData}
|
|
onValueChange={(value) => setShowData(value)}
|
|
>
|
|
<SelectTrigger className="w-full text-sm border">
|
|
<SelectValue placeholder="Select" />
|
|
</SelectTrigger>
|
|
<SelectContent>
|
|
<SelectItem value="5">5</SelectItem>
|
|
<SelectItem value="10">10</SelectItem>
|
|
<SelectItem value="25">25</SelectItem>
|
|
<SelectItem value="50">50</SelectItem>
|
|
</SelectContent>
|
|
</Select>
|
|
</div>
|
|
<div className="flex flex-col gap-1 w-full lg:w-[230px]">
|
|
<p className="font-semibold text-sm">Kategori</p>
|
|
<Select
|
|
value={selectedCategoryId}
|
|
onValueChange={(value) => setSelectedCategoryId(value)} // simpan ID
|
|
>
|
|
<SelectTrigger className="w-full text-sm border">
|
|
<SelectValue placeholder="Kategori" />
|
|
</SelectTrigger>
|
|
<SelectContent>
|
|
{categories
|
|
?.filter((category: any) => category.title != null)
|
|
.map((category: any) => (
|
|
<SelectItem
|
|
key={category.id}
|
|
value={category.id.toString()} // kirim ID, bukan title
|
|
>
|
|
{category.title}
|
|
</SelectItem>
|
|
))}
|
|
</SelectContent>
|
|
</Select>
|
|
</div>
|
|
<div className="flex flex-col gap-1 w-full lg:w-[150px]">
|
|
<p className="font-semibold text-sm">Source</p>
|
|
<Select
|
|
value={selectedSource}
|
|
onValueChange={(value) => setSelectedSource(value)}
|
|
>
|
|
<SelectTrigger className="w-full text-sm border">
|
|
<SelectValue placeholder="Pilih Source" />
|
|
</SelectTrigger>
|
|
<SelectContent>
|
|
<SelectItem value="internal">INTERNAL</SelectItem>
|
|
<SelectItem value="external">EXTERNAL</SelectItem>
|
|
</SelectContent>
|
|
</Select>
|
|
</div>
|
|
<div className="flex flex-col gap-1 w-full lg:w-[150px]">
|
|
<p className="font-semibold text-sm">Status</p>
|
|
<Select
|
|
value={selectedStatus}
|
|
onValueChange={(value) => setSelectedStatus(value)}
|
|
>
|
|
<SelectTrigger className="w-full text-sm border">
|
|
<SelectValue placeholder="Pilih Status" />
|
|
</SelectTrigger>
|
|
<SelectContent>
|
|
<SelectItem value="publish">PUBLISH</SelectItem>
|
|
<SelectItem value="draft">DRAFT</SelectItem>
|
|
</SelectContent>
|
|
</Select>
|
|
</div>
|
|
<div className="flex flex-col gap-1 w-full lg:w-[240px]">
|
|
<p className="font-semibold text-sm">Tanggal</p>
|
|
<DatePicker
|
|
selectsRange
|
|
startDate={dateRange.startDate}
|
|
endDate={dateRange.endDate}
|
|
onChange={(update: [Date | null, Date | null]) => {
|
|
setDateRange({
|
|
startDate: update[0],
|
|
endDate: update[1],
|
|
});
|
|
}}
|
|
isClearable
|
|
dateFormat="dd/MM/yyyy"
|
|
className="z-50 w-full text-sm bg-transparent border border-gray-200 px-2 py-[6px] rounded-xl h-[40px] text-gray-600 dark:text-gray-300"
|
|
placeholderText="Pilih rentang tanggal"
|
|
/>
|
|
</div>
|
|
</div>
|
|
<div className="w-full overflow-x-hidden">
|
|
<div className="w-full overflow-x-auto">
|
|
<Table className="min-w-[1000px] w-full table-auto border text-sm">
|
|
<TableHeader>
|
|
<TableRow>
|
|
{(username === "admin-mabes"
|
|
? columns
|
|
: columnsOtherRole
|
|
).map((column) => (
|
|
<TableHead
|
|
key={column.uid}
|
|
className={`bg-white dark:bg-black text-black dark:text-white
|
|
text-sm font-semibold border-b px-3 py-3
|
|
${
|
|
column.uid === "no"
|
|
? "min-w-[60px] text-center"
|
|
: column.uid === "title"
|
|
? "min-w-[280px]"
|
|
: column.uid === "actions"
|
|
? "min-w-[100px] text-center"
|
|
: "min-w-[160px]"
|
|
}
|
|
`}
|
|
>
|
|
{column.name}
|
|
</TableHead>
|
|
))}
|
|
</TableRow>
|
|
</TableHeader>
|
|
<TableBody>
|
|
{article.length > 0 ? (
|
|
article.map((item: any) => (
|
|
<TableRow key={item.id}>
|
|
{(username === "admin-mabes"
|
|
? columns
|
|
: columnsOtherRole
|
|
).map((column) => (
|
|
<TableCell
|
|
key={column.uid}
|
|
className={`text-black dark:text-white text-sm px-3 py-3 align-top
|
|
${
|
|
column.uid === "no"
|
|
? "min-w-[60px] text-center font-medium"
|
|
: column.uid === "title"
|
|
? "min-w-[280px] whitespace-normal break-words leading-snug"
|
|
: column.uid === "actions"
|
|
? "min-w-[100px] text-center"
|
|
: "min-w-[160px] whitespace-normal break-words"
|
|
}
|
|
`}
|
|
>
|
|
{renderCell(item, column.uid)}
|
|
</TableCell>
|
|
))}
|
|
</TableRow>
|
|
))
|
|
) : (
|
|
<TableRow>
|
|
<TableCell
|
|
colSpan={columns.length}
|
|
className="text-center py-4"
|
|
>
|
|
No data to display.
|
|
</TableCell>
|
|
</TableRow>
|
|
)}
|
|
</TableBody>
|
|
</Table>
|
|
</div>
|
|
</div>
|
|
<div className="my-2 w-full flex justify-center">
|
|
{/* <Pagination
|
|
isCompact
|
|
showControls
|
|
showShadow
|
|
color="primary"
|
|
classNames={{
|
|
base: "bg-transparent",
|
|
wrapper: "bg-transparent",
|
|
}}
|
|
page={page}
|
|
total={totalPage}
|
|
onChange={(page) => setPage(page)}
|
|
/> */}
|
|
<CustomPagination
|
|
totalPage={totalPage}
|
|
onPageChange={(data) => setPage(data)}
|
|
/>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</>
|
|
);
|
|
}
|