update limit page
This commit is contained in:
@@ -14,13 +14,26 @@ import {
|
|||||||
} from "@/service/adminService";
|
} from "@/service/adminService";
|
||||||
import { useEffect, useState, useCallback } from "react";
|
import { useEffect, useState, useCallback } from "react";
|
||||||
import Pagination from "@/components/tables/Pagination";
|
import Pagination from "@/components/tables/Pagination";
|
||||||
|
import { LIMIT_ITEM_TABLE } from "../../../../../../constant";
|
||||||
|
|
||||||
export type SortColumn = "created_at" | "updated_at" | "display_name" | "email";
|
export type SortColumn = "created_at" | "updated_at" | "display_name" | "email";
|
||||||
|
|
||||||
export default function UserTable() { // UserTable
|
// Hàm helper format ngày giờ giống với Historian
|
||||||
|
const formatDateTimeToISO = (
|
||||||
|
dateStr: string,
|
||||||
|
timeStr: string,
|
||||||
|
isEndOfDay: boolean = false,
|
||||||
|
): string | undefined => {
|
||||||
|
if (!dateStr) return undefined;
|
||||||
|
|
||||||
|
const time = timeStr || (isEndOfDay ? "23:59" : "00:00");
|
||||||
|
|
||||||
|
return `${dateStr}T${time}:00.000000+07:00`;
|
||||||
|
};
|
||||||
|
|
||||||
|
export default function UserTable() {
|
||||||
const [page, setPage] = useState<number>(1);
|
const [page, setPage] = useState<number>(1);
|
||||||
const [limitInput, setLimitInput] = useState<string>("5");
|
const [limitInput, setLimitInput] = useState<string>(LIMIT_ITEM_TABLE.toString());
|
||||||
|
|
||||||
const [selectedRole, setSelectedRole] = useState<string>("");
|
const [selectedRole, setSelectedRole] = useState<string>("");
|
||||||
const [roles, setRoles] = useState<{ id: string; name: string }[]>([]);
|
const [roles, setRoles] = useState<{ id: string; name: string }[]>([]);
|
||||||
@@ -29,15 +42,26 @@ export default function UserTable() { // UserTable
|
|||||||
const [authProvider, setAuthProvider] = useState<string>("");
|
const [authProvider, setAuthProvider] = useState<string>("");
|
||||||
const [isDeleted, setIsDeleted] = useState<boolean | undefined>(undefined);
|
const [isDeleted, setIsDeleted] = useState<boolean | undefined>(undefined);
|
||||||
|
|
||||||
|
// --- THÊM STATE CHO NGÀY GIỜ ---
|
||||||
|
const [fromDate, setFromDate] = useState<string>("");
|
||||||
|
const [fromTime, setFromTime] = useState<string>("");
|
||||||
|
const [toDate, setToDate] = useState<string>("");
|
||||||
|
const [toTime, setToTime] = useState<string>("");
|
||||||
|
|
||||||
const [selectedUser, setSelectedUser] = useState<fullDataUser | null>(null);
|
const [selectedUser, setSelectedUser] = useState<fullDataUser | null>(null);
|
||||||
const [isModalOpen, setIsModalOpen] = useState(false);
|
const [isModalOpen, setIsModalOpen] = useState(false);
|
||||||
const [roleUser, setRoleUser] = useState<fullDataUser | null>(null);
|
const [roleUser, setRoleUser] = useState<fullDataUser | null>(null);
|
||||||
const [isRoleModalOpen, setIsRoleModalOpen] = useState(false);
|
const [isRoleModalOpen, setIsRoleModalOpen] = useState(false);
|
||||||
|
|
||||||
|
// Cập nhật params để chứa thêm ngày giờ
|
||||||
const [debouncedParams, setDebouncedParams] = useState({
|
const [debouncedParams, setDebouncedParams] = useState({
|
||||||
search: "",
|
search: "",
|
||||||
limit: 5,
|
limit: 5,
|
||||||
authProvider: "",
|
authProvider: "",
|
||||||
|
fromDate: "",
|
||||||
|
fromTime: "",
|
||||||
|
toDate: "",
|
||||||
|
toTime: "",
|
||||||
});
|
});
|
||||||
|
|
||||||
const [tableData, setTableData] = useState<responseUserTable | null>(null);
|
const [tableData, setTableData] = useState<responseUserTable | null>(null);
|
||||||
@@ -45,6 +69,32 @@ export default function UserTable() { // UserTable
|
|||||||
const [sortBy, setSortBy] = useState<SortColumn | undefined>(undefined);
|
const [sortBy, setSortBy] = useState<SortColumn | undefined>(undefined);
|
||||||
const [sortOrder, setSortOrder] = useState<"asc" | "desc">("asc");
|
const [sortOrder, setSortOrder] = useState<"asc" | "desc">("asc");
|
||||||
|
|
||||||
|
// Hàm Reset bộ lọc
|
||||||
|
const handleReset = () => {
|
||||||
|
setSearchTerm("");
|
||||||
|
setAuthProvider("");
|
||||||
|
setIsDeleted(undefined);
|
||||||
|
setSelectedRole("");
|
||||||
|
setLimitInput(LIMIT_ITEM_TABLE.toString());
|
||||||
|
|
||||||
|
setFromDate("");
|
||||||
|
setFromTime("");
|
||||||
|
setToDate("");
|
||||||
|
setToTime("");
|
||||||
|
|
||||||
|
setPage(1);
|
||||||
|
|
||||||
|
setDebouncedParams({
|
||||||
|
search: "",
|
||||||
|
limit: LIMIT_ITEM_TABLE,
|
||||||
|
authProvider: "",
|
||||||
|
fromDate: "",
|
||||||
|
fromTime: "",
|
||||||
|
toDate: "",
|
||||||
|
toTime: "",
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const fetchRoles = async () => {
|
const fetchRoles = async () => {
|
||||||
try {
|
try {
|
||||||
@@ -63,18 +113,31 @@ export default function UserTable() { // UserTable
|
|||||||
const handler = setTimeout(() => {
|
const handler = setTimeout(() => {
|
||||||
setDebouncedParams({
|
setDebouncedParams({
|
||||||
search: searchTerm,
|
search: searchTerm,
|
||||||
limit: parseInt(limitInput) || 5,
|
limit: parseInt(limitInput) || LIMIT_ITEM_TABLE,
|
||||||
authProvider: authProvider,
|
authProvider: authProvider,
|
||||||
|
fromDate,
|
||||||
|
fromTime,
|
||||||
|
toDate,
|
||||||
|
toTime,
|
||||||
});
|
});
|
||||||
setPage(1);
|
setPage(1);
|
||||||
}, 600);
|
}, 600);
|
||||||
return () => clearTimeout(handler);
|
return () => clearTimeout(handler);
|
||||||
}, [searchTerm, limitInput, authProvider]);
|
}, [
|
||||||
|
searchTerm,
|
||||||
|
limitInput,
|
||||||
|
authProvider,
|
||||||
|
fromDate,
|
||||||
|
fromTime,
|
||||||
|
toDate,
|
||||||
|
toTime,
|
||||||
|
]);
|
||||||
|
|
||||||
const fetchUsers = useCallback(async () => {
|
const fetchUsers = useCallback(async () => {
|
||||||
setLoading(true);
|
setLoading(true);
|
||||||
try {
|
try {
|
||||||
const payload: getUserDto = {
|
// Dùng any để tránh lỗi TS nếu interface getUserDto chưa update
|
||||||
|
const payload: any = {
|
||||||
page: page,
|
page: page,
|
||||||
limit: debouncedParams.limit,
|
limit: debouncedParams.limit,
|
||||||
search: debouncedParams.search || undefined,
|
search: debouncedParams.search || undefined,
|
||||||
@@ -85,7 +148,22 @@ export default function UserTable() { // UserTable
|
|||||||
role_ids: selectedRole ? [selectedRole] : undefined,
|
role_ids: selectedRole ? [selectedRole] : undefined,
|
||||||
};
|
};
|
||||||
|
|
||||||
const response = await apiGetListUser(payload);
|
// Thêm format ngày giờ vào payload
|
||||||
|
const createdFrom = formatDateTimeToISO(
|
||||||
|
debouncedParams.fromDate,
|
||||||
|
debouncedParams.fromTime,
|
||||||
|
false,
|
||||||
|
);
|
||||||
|
if (createdFrom) payload.created_from = createdFrom;
|
||||||
|
|
||||||
|
const createdTo = formatDateTimeToISO(
|
||||||
|
debouncedParams.toDate,
|
||||||
|
debouncedParams.toTime,
|
||||||
|
true,
|
||||||
|
);
|
||||||
|
if (createdTo) payload.created_to = createdTo;
|
||||||
|
|
||||||
|
const response = await apiGetListUser(payload as getUserDto);
|
||||||
if (response?.status) {
|
if (response?.status) {
|
||||||
setTableData(response);
|
setTableData(response);
|
||||||
}
|
}
|
||||||
@@ -179,8 +257,31 @@ export default function UserTable() { // UserTable
|
|||||||
<div>
|
<div>
|
||||||
<PageBreadcrumb pageTitle="Quản lý người dùng" />
|
<PageBreadcrumb pageTitle="Quản lý người dùng" />
|
||||||
<div className="space-y-6">
|
<div className="space-y-6">
|
||||||
<ComponentCard title="Bộ lọc tìm kiếm">
|
<ComponentCard
|
||||||
<div className="grid grid-cols-1 gap-4 mb-6 md:grid-cols-3 lg:grid-cols-4">
|
title="Bộ lọc tìm kiếm"
|
||||||
|
headerAction={
|
||||||
|
<button
|
||||||
|
onClick={handleReset}
|
||||||
|
className="flex items-center px-3 py-1.5 text-xs text-red-500 transition-colors border-red-100 dark:bg-red-500/10 dark:border-red-500/20 dark:hover:bg-red-500/20"
|
||||||
|
>
|
||||||
|
<svg
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
className="w-7 h-7"
|
||||||
|
fill="none"
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
stroke="currentColor"
|
||||||
|
>
|
||||||
|
<path
|
||||||
|
strokeLinecap="round"
|
||||||
|
strokeLinejoin="round"
|
||||||
|
strokeWidth={2.5}
|
||||||
|
d="M4 4v5h.582m15.356 2A8.001 8.001 0 004.582 9m0 0H9m11 11v-5h-.581m0 0a8.003 8.003 0 01-15.357-2m15.357 2H15"
|
||||||
|
/>
|
||||||
|
</svg>
|
||||||
|
</button>
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<div className="grid grid-cols-1 gap-4 mb-6 md:grid-cols-2 lg:grid-cols-3 xl:grid-cols-4">
|
||||||
<div>
|
<div>
|
||||||
<label className="block mb-2 text-sm font-medium">Search</label>
|
<label className="block mb-2 text-sm font-medium">Search</label>
|
||||||
<input
|
<input
|
||||||
@@ -188,7 +289,7 @@ export default function UserTable() { // UserTable
|
|||||||
placeholder="Name, email..."
|
placeholder="Name, email..."
|
||||||
value={searchTerm}
|
value={searchTerm}
|
||||||
onChange={(e) => setSearchTerm(e.target.value)}
|
onChange={(e) => setSearchTerm(e.target.value)}
|
||||||
className="w-full px-3 py-2 border rounded-lg dark:bg-gray-800 dark:border-gray-700"
|
className="w-full px-3 py-2 border rounded-lg dark:bg-gray-800 dark:border-gray-700 outline-none focus:border-brand-500"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -199,7 +300,7 @@ export default function UserTable() { // UserTable
|
|||||||
<select
|
<select
|
||||||
value={authProvider}
|
value={authProvider}
|
||||||
onChange={(e) => setAuthProvider(e.target.value)}
|
onChange={(e) => setAuthProvider(e.target.value)}
|
||||||
className="w-full px-3 py-2 border rounded-lg dark:bg-gray-800 dark:border-gray-700 cursor-pointer bg-white"
|
className="w-full px-3 py-2 border rounded-lg dark:bg-gray-800 dark:border-gray-700 cursor-pointer bg-white outline-none focus:border-brand-500"
|
||||||
>
|
>
|
||||||
<option value="">Tất cả</option>
|
<option value="">Tất cả</option>
|
||||||
<option value="google">Google</option>
|
<option value="google">Google</option>
|
||||||
@@ -218,7 +319,8 @@ export default function UserTable() { // UserTable
|
|||||||
: e.target.value === "true",
|
: e.target.value === "true",
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
className="w-full px-3 py-2 border rounded-lg dark:bg-gray-800 dark:border-gray-700"
|
value={isDeleted === undefined ? "" : isDeleted ? "true" : "false"}
|
||||||
|
className="w-full px-3 py-2 border rounded-lg dark:bg-gray-800 dark:border-gray-700 outline-none focus:border-brand-500 cursor-pointer"
|
||||||
>
|
>
|
||||||
<option value="">Tất cả</option>
|
<option value="">Tất cả</option>
|
||||||
<option value="false">Hoạt động</option>
|
<option value="false">Hoạt động</option>
|
||||||
@@ -226,6 +328,44 @@ export default function UserTable() { // UserTable
|
|||||||
</select>
|
</select>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
{/* Từ ngày */}
|
||||||
|
<div>
|
||||||
|
<label className="block mb-2 text-sm font-medium">Từ ngày</label>
|
||||||
|
<div className="flex gap-2">
|
||||||
|
<input
|
||||||
|
type="date"
|
||||||
|
value={fromDate}
|
||||||
|
onChange={(e) => setFromDate(e.target.value)}
|
||||||
|
className="w-full px-3 py-2 border rounded-lg dark:bg-gray-800 dark:border-gray-700 outline-none focus:border-brand-500"
|
||||||
|
/>
|
||||||
|
<input
|
||||||
|
type="time"
|
||||||
|
value={fromTime}
|
||||||
|
onChange={(e) => setFromTime(e.target.value)}
|
||||||
|
className="w-full px-3 py-2 border rounded-lg dark:bg-gray-800 dark:border-gray-700 outline-none focus:border-brand-500"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Đến ngày */}
|
||||||
|
<div>
|
||||||
|
<label className="block mb-2 text-sm font-medium">Đến ngày</label>
|
||||||
|
<div className="flex gap-2">
|
||||||
|
<input
|
||||||
|
type="date"
|
||||||
|
value={toDate}
|
||||||
|
onChange={(e) => setToDate(e.target.value)}
|
||||||
|
className="w-full px-3 py-2 border rounded-lg dark:bg-gray-800 dark:border-gray-700 outline-none focus:border-brand-500"
|
||||||
|
/>
|
||||||
|
<input
|
||||||
|
type="time"
|
||||||
|
value={toTime}
|
||||||
|
onChange={(e) => setToTime(e.target.value)}
|
||||||
|
className="w-full px-3 py-2 border rounded-lg dark:bg-gray-800 dark:border-gray-700 outline-none focus:border-brand-500"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
{/* Limit */}
|
{/* Limit */}
|
||||||
<div>
|
<div>
|
||||||
<label className="block mb-2 text-sm font-medium">
|
<label className="block mb-2 text-sm font-medium">
|
||||||
@@ -235,7 +375,7 @@ export default function UserTable() { // UserTable
|
|||||||
type="number"
|
type="number"
|
||||||
value={limitInput}
|
value={limitInput}
|
||||||
onChange={(e) => setLimitInput(e.target.value)}
|
onChange={(e) => setLimitInput(e.target.value)}
|
||||||
className="w-full px-3 py-2 border rounded-lg dark:bg-gray-800 dark:border-gray-700"
|
className="w-full px-3 py-2 border rounded-lg dark:bg-gray-800 dark:border-gray-700 outline-none focus:border-brand-500"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -244,7 +384,7 @@ export default function UserTable() { // UserTable
|
|||||||
<ComponentCard title="Danh sách người dùng">
|
<ComponentCard title="Danh sách người dùng">
|
||||||
<div className="relative min-h-[300px]">
|
<div className="relative min-h-[300px]">
|
||||||
{loading && (
|
{loading && (
|
||||||
<div className="absolute inset-0 z-10 flex items-center justify-center bg-white/50 dark:bg-gray-900/50">
|
<div className="absolute inset-0 z-10 flex items-center justify-center bg-white/50 dark:bg-gray-900/50 rounded-xl">
|
||||||
<div className="w-10 h-10 border-4 border-t-brand-500 rounded-full animate-spin"></div>
|
<div className="w-10 h-10 border-4 border-t-brand-500 rounded-full animate-spin"></div>
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
@@ -264,7 +404,8 @@ export default function UserTable() { // UserTable
|
|||||||
<div className="flex items-center justify-between mt-6">
|
<div className="flex items-center justify-between mt-6">
|
||||||
<p className="text-sm text-gray-500">
|
<p className="text-sm text-gray-500">
|
||||||
Hiển thị trang {pagination?.current_page || 1} /{" "}
|
Hiển thị trang {pagination?.current_page || 1} /{" "}
|
||||||
{pagination?.total_pages || 1} ({pagination?.total_records || 0} kết quả)
|
{pagination?.total_pages || 1} ({pagination?.total_records || 0}{" "}
|
||||||
|
kết quả)
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
{pagination && pagination.total_pages > 1 && (
|
{pagination && pagination.total_pages > 1 && (
|
||||||
|
|||||||
Reference in New Issue
Block a user