added expens group fied Name
This commit is contained in:
parent
997971629f
commit
b1c23aab4d
@ -20,4 +20,8 @@
|
|||||||
|
|
||||||
.text-md {
|
.text-md {
|
||||||
font-size: 2rem;
|
font-size: 2rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.text-md-b {
|
||||||
|
font-weight: normal;
|
||||||
}
|
}
|
@ -73,7 +73,7 @@ const { labels, series, total } = useMemo(() => {
|
|||||||
<>
|
<>
|
||||||
<div className="card-header d-flex justify-content-between align-items-center ">
|
<div className="card-header d-flex justify-content-between align-items-center ">
|
||||||
<div>
|
<div>
|
||||||
<h5 className="mb-1 fw-bold">Expense Breakdown</h5>
|
<h5 className="mb-1 card-title">Expense Breakdown</h5>
|
||||||
<p className="card-subtitle me-3">Detailed project expenses</p>
|
<p className="card-subtitle me-3">Detailed project expenses</p>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
@ -42,12 +42,14 @@ const ExpenseStatus = () => {
|
|||||||
<div className="card-body ">
|
<div className="card-body ">
|
||||||
<div className=" py-0 text-start mb-2">
|
<div className=" py-0 text-start mb-2">
|
||||||
<div className="d-flex justify-content-between align-items-center">
|
<div className="d-flex justify-content-between align-items-center">
|
||||||
<span className="fs-5"> Project Spendings:</span>{" "}
|
<div className="d-block">
|
||||||
|
<span className="fs-5 d-block">Project Spendings:</span>{" "}
|
||||||
|
<small className="d-block text-gary-80">{`(All Processed Payments)`}</small>
|
||||||
|
</div>
|
||||||
<span className="text-end text-royalblue text-md">
|
<span className="text-end text-royalblue text-md">
|
||||||
{formatCurrency(data?.totalAmount)}
|
{formatCurrency(data?.totalAmount)}
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
<small className=" text-gary-80">{`(All Processed Payments)`}</small>
|
|
||||||
</div>
|
</div>
|
||||||
<div className="report-list text-start">
|
<div className="report-list text-start">
|
||||||
{[
|
{[
|
||||||
@ -60,7 +62,7 @@ const ExpenseStatus = () => {
|
|||||||
status: EXPENSE_STATUS.payment_pending,
|
status: EXPENSE_STATUS.payment_pending,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: "Pending Approver",
|
title: "Pending Approve",
|
||||||
count: data?.processPending?.count,
|
count: data?.processPending?.count,
|
||||||
amount: data?.processPending?.totalAmount,
|
amount: data?.processPending?.totalAmount,
|
||||||
icon: "fa-solid fa-check",
|
icon: "fa-solid fa-check",
|
||||||
@ -71,7 +73,7 @@ const ExpenseStatus = () => {
|
|||||||
title: "Pending Reviewer",
|
title: "Pending Reviewer",
|
||||||
count: data?.reviewPending?.count,
|
count: data?.reviewPending?.count,
|
||||||
amount: data?.reviewPending?.totalAmount,
|
amount: data?.reviewPending?.totalAmount,
|
||||||
icon: "bx bx-file",
|
icon: "bx bx-search-alt-2",
|
||||||
iconColor: "text-secondary",
|
iconColor: "text-secondary",
|
||||||
status: EXPENSE_STATUS.review_pending,
|
status: EXPENSE_STATUS.review_pending,
|
||||||
},
|
},
|
||||||
@ -87,25 +89,25 @@ const ExpenseStatus = () => {
|
|||||||
<div
|
<div
|
||||||
key={idx}
|
key={idx}
|
||||||
className="report-list-item rounded-2 mb-4 bg-lighter px-2 py-3 cursor-pointer"
|
className="report-list-item rounded-2 mb-4 bg-lighter px-2 py-3 cursor-pointer"
|
||||||
onClick={()=>handleNavigate(item.status)}
|
onClick={() => handleNavigate(item?.status)}
|
||||||
>
|
>
|
||||||
<div className="d-flex align-items-center">
|
<div className="d-flex align-items-center">
|
||||||
<div className="report-list-icon shadow-xs me-4">
|
<div className="report-list-icon shadow-xs me-4">
|
||||||
<span className="d-inline-flex align-items-center justify-content-center rounded-circle border p-2">
|
<span className="d-inline-flex align-items-center justify-content-center rounded-circle border p-2">
|
||||||
<i className={`${item.icon} ${item.iconColor} bx-lg`}></i>
|
<i className={`${item?.icon} ${item?.iconColor} bx-lg`}></i>
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
<div className="d-flex justify-content-between align-items-center w-100 flex-wrap gap-2">
|
<div className="d-flex justify-content-between align-items-center w-100 flex-wrap gap-2">
|
||||||
<div className="d-flex flex-column gap-2">
|
<div className="d-flex flex-column gap-2">
|
||||||
<span className="fw-bold">{item.title}</span>
|
<span className="fw-bold">{item?.title}</span>
|
||||||
<small className="mb-0 text-primary">
|
{item?.amount ? <small className="mb-0 text-primary">
|
||||||
{formatCurrency(item.amount)}
|
{formatCurrency(item?.amount)}
|
||||||
</small>
|
</small> :""}
|
||||||
</div>
|
</div>
|
||||||
<div className="">
|
<div className="">
|
||||||
{" "}
|
{" "}
|
||||||
<small className="text-muted fs-semibold text-royalblue text-md">
|
<small className="text-muted fs-semibold text-royalblue text-md">
|
||||||
{item.count}{" "}
|
{item?.count}{" "}
|
||||||
</small>
|
</small>
|
||||||
<small className="text-muted fs-semibold text-royalblue text-md">
|
<small className="text-muted fs-semibold text-royalblue text-md">
|
||||||
<i className="bx bx-chevron-right"></i>
|
<i className="bx bx-chevron-right"></i>
|
||||||
|
@ -36,7 +36,7 @@ const ExpenseFilterPanel = ({ onApply, handleGroupBy }) => {
|
|||||||
].sort((a, b) => a.name.localeCompare(b.name));
|
].sort((a, b) => a.name.localeCompare(b.name));
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const [selectedGroup, setSelectedGroup] = useState(groupByList[0]);
|
const [selectedGroup, setSelectedGroup] = useState(groupByList[6]);
|
||||||
const [resetKey, setResetKey] = useState(0);
|
const [resetKey, setResetKey] = useState(0);
|
||||||
|
|
||||||
const dynamicDefaultFilter = useMemo(() => {
|
const dynamicDefaultFilter = useMemo(() => {
|
||||||
|
@ -10,7 +10,11 @@ import {
|
|||||||
EXPENSE_REJECTEDBY,
|
EXPENSE_REJECTEDBY,
|
||||||
ITEMS_PER_PAGE,
|
ITEMS_PER_PAGE,
|
||||||
} from "../../utils/constants";
|
} from "../../utils/constants";
|
||||||
import { formatCurrency, getColorNameFromHex, useDebounce } from "../../utils/appUtils";
|
import {
|
||||||
|
formatCurrency,
|
||||||
|
getColorNameFromHex,
|
||||||
|
useDebounce,
|
||||||
|
} from "../../utils/appUtils";
|
||||||
import { ExpenseTableSkeleton } from "./ExpenseSkeleton";
|
import { ExpenseTableSkeleton } from "./ExpenseSkeleton";
|
||||||
import ConfirmModal from "../common/ConfirmModal";
|
import ConfirmModal from "../common/ConfirmModal";
|
||||||
import { useHasUserPermission } from "../../hooks/useHasUserPermission";
|
import { useHasUserPermission } from "../../hooks/useHasUserPermission";
|
||||||
@ -59,40 +63,61 @@ const ExpenseList = ({ filters, groupBy = "transactionDate", searchText }) => {
|
|||||||
const groupByField = (items, field) => {
|
const groupByField = (items, field) => {
|
||||||
return items.reduce((acc, item) => {
|
return items.reduce((acc, item) => {
|
||||||
let key;
|
let key;
|
||||||
|
let displayField;
|
||||||
|
|
||||||
switch (field) {
|
switch (field) {
|
||||||
case "transactionDate":
|
case "transactionDate":
|
||||||
key = item?.transactionDate?.split("T")[0];
|
key = item?.transactionDate?.split("T")[0];
|
||||||
|
displayField = "Transaction Date";
|
||||||
break;
|
break;
|
||||||
case "status":
|
case "status":
|
||||||
key = item?.status?.displayName || "Unknown";
|
key = item?.status?.displayName || "Unknown";
|
||||||
|
displayField = "Status";
|
||||||
break;
|
break;
|
||||||
case "submittedBy":
|
case "submittedBy":
|
||||||
key = `${item?.createdBy?.firstName ?? ""} ${
|
key = `${item?.createdBy?.firstName ?? ""} ${
|
||||||
item.createdBy?.lastName ?? ""
|
item.createdBy?.lastName ?? ""
|
||||||
}`.trim();
|
}`.trim();
|
||||||
|
displayField = "Submitted By";
|
||||||
break;
|
break;
|
||||||
case "project":
|
case "project":
|
||||||
key = item?.project?.name || "Unknown Project";
|
key = item?.project?.name || "Unknown Project";
|
||||||
|
displayField = "Project";
|
||||||
break;
|
break;
|
||||||
case "paymentMode":
|
case "paymentMode":
|
||||||
key = item?.paymentMode?.name || "Unknown Mode";
|
key = item?.paymentMode?.name || "Unknown Mode";
|
||||||
|
displayField = "Payment Mode";
|
||||||
break;
|
break;
|
||||||
case "expensesType":
|
case "expensesType":
|
||||||
key = item?.expensesType?.name || "Unknown Type";
|
key = item?.expensesType?.name || "Unknown Type";
|
||||||
|
displayField = "Expense Type";
|
||||||
break;
|
break;
|
||||||
case "createdAt":
|
case "createdAt":
|
||||||
key = item?.createdAt?.split("T")[0] || "Unknown Type";
|
key = item?.createdAt?.split("T")[0] || "Unknown Date";
|
||||||
|
displayField = "Created Date";
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
key = "Others";
|
key = "Others";
|
||||||
|
displayField = "Others";
|
||||||
}
|
}
|
||||||
if (!acc[key]) acc[key] = [];
|
|
||||||
acc[key]?.push(item);
|
const groupKey = `${field}_${key}`; // unique key for object property
|
||||||
|
if (!acc[groupKey]) {
|
||||||
|
acc[groupKey] = { key, displayField, items: [] };
|
||||||
|
}
|
||||||
|
|
||||||
|
acc[groupKey].items.push(item);
|
||||||
return acc;
|
return acc;
|
||||||
}, {});
|
}, {});
|
||||||
};
|
};
|
||||||
|
|
||||||
const expenseColumns = [
|
const expenseColumns = [
|
||||||
|
{
|
||||||
|
key: "expenseUId",
|
||||||
|
label: "Expense Id",
|
||||||
|
getValue: (e) => e.expenseUId|| "N/A",
|
||||||
|
align: "text-start mx-2",
|
||||||
|
},
|
||||||
{
|
{
|
||||||
key: "expensesType",
|
key: "expensesType",
|
||||||
label: "Expense Type",
|
label: "Expense Type",
|
||||||
@ -138,11 +163,7 @@ const ExpenseList = ({ filters, groupBy = "transactionDate", searchText }) => {
|
|||||||
{
|
{
|
||||||
key: "amount",
|
key: "amount",
|
||||||
label: "Amount",
|
label: "Amount",
|
||||||
getValue: (e) => (
|
getValue: (e) => <>{formatCurrency(e?.amount)}</>,
|
||||||
<>
|
|
||||||
{formatCurrency(e?.amount)}
|
|
||||||
</>
|
|
||||||
),
|
|
||||||
isAlwaysVisible: true,
|
isAlwaysVisible: true,
|
||||||
align: "text-end",
|
align: "text-end",
|
||||||
},
|
},
|
||||||
@ -168,7 +189,11 @@ const ExpenseList = ({ filters, groupBy = "transactionDate", searchText }) => {
|
|||||||
const grouped = groupBy
|
const grouped = groupBy
|
||||||
? groupByField(data?.data ?? [], groupBy)
|
? groupByField(data?.data ?? [], groupBy)
|
||||||
: { All: data?.data ?? [] };
|
: { All: data?.data ?? [] };
|
||||||
const IsGroupedByDate = ["transactionDate", "createdAt"]?.includes(groupBy);
|
const IsGroupedByDate = [
|
||||||
|
{ key: "transactionDate", displayField: "Transaction Date" },
|
||||||
|
{ key: "createdAt", displayField: "created Date" },
|
||||||
|
]?.includes(groupBy);
|
||||||
|
|
||||||
const canEditExpense = (expense) => {
|
const canEditExpense = (expense) => {
|
||||||
return (
|
return (
|
||||||
(expense?.status?.id === EXPENSE_DRAFT ||
|
(expense?.status?.id === EXPENSE_DRAFT ||
|
||||||
@ -226,18 +251,24 @@ const ExpenseList = ({ filters, groupBy = "transactionDate", searchText }) => {
|
|||||||
</thead>
|
</thead>
|
||||||
<tbody>
|
<tbody>
|
||||||
{Object.keys(grouped).length > 0 ? (
|
{Object.keys(grouped).length > 0 ? (
|
||||||
Object.entries(grouped).map(([group, expenses]) => (
|
Object.values(grouped).map(({ key, displayField, items }) => (
|
||||||
<React.Fragment key={group}>
|
<React.Fragment key={key}>
|
||||||
<tr className="tr-group text-dark">
|
<tr className="tr-group text-dark">
|
||||||
<td colSpan={8} className="text-start">
|
<td colSpan={8} className="text-start">
|
||||||
<strong>
|
<div className="d-flex align-items-center">
|
||||||
{IsGroupedByDate
|
{" "}
|
||||||
? formatUTCToLocalTime(group)
|
<small className="fs-6 ">
|
||||||
: group}
|
{displayField} :{" "}
|
||||||
</strong>
|
</small>{" "}
|
||||||
|
<small className="fs-6 ms-3">
|
||||||
|
{IsGroupedByDate
|
||||||
|
? formatUTCToLocalTime(key)
|
||||||
|
: key}
|
||||||
|
</small>
|
||||||
|
</div>
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
{expenses.map((expense) => (
|
{items?.map((expense) => (
|
||||||
<tr key={expense.id}>
|
<tr key={expense.id}>
|
||||||
{expenseColumns.map(
|
{expenseColumns.map(
|
||||||
(col) =>
|
(col) =>
|
||||||
@ -274,7 +305,6 @@ const ExpenseList = ({ filters, groupBy = "transactionDate", searchText }) => {
|
|||||||
}
|
}
|
||||||
></i>
|
></i>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{canDetetExpense(expense) && (
|
{canDetetExpense(expense) && (
|
||||||
<i
|
<i
|
||||||
className="bx bx-trash text-danger cursor-pointer"
|
className="bx bx-trash text-danger cursor-pointer"
|
||||||
@ -293,9 +323,9 @@ const ExpenseList = ({ filters, groupBy = "transactionDate", searchText }) => {
|
|||||||
) : (
|
) : (
|
||||||
<tr>
|
<tr>
|
||||||
<td colSpan={8} className="text-center border-0 ">
|
<td colSpan={8} className="text-center border-0 ">
|
||||||
<div className="py-8">
|
<div className="py-8">
|
||||||
<p>No Expense Found</p>
|
<p>No Expense Found</p>
|
||||||
</div>
|
</div>
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
)}
|
)}
|
||||||
|
@ -106,11 +106,12 @@ const ViewExpense = ({ ExpenseId }) => {
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<form className="container px-3" onSubmit={handleSubmit(onSubmit)}>
|
<form className="container px-3" onSubmit={handleSubmit(onSubmit)}>
|
||||||
<div className="row mb-3">
|
<div className="row mb-1">
|
||||||
<div className="col-12 mb-3">
|
<div className="col-12 mb-1">
|
||||||
<h5 className="fw-semibold">Expense Details</h5>
|
<h5 className="fw-semibold m-0">Expense Details</h5>
|
||||||
<hr />
|
<hr />
|
||||||
</div>
|
</div>
|
||||||
|
<div className="col-12 text-start fw-semibold my-2">{data?.expenseUId}</div>
|
||||||
{/* Row 1 */}
|
{/* Row 1 */}
|
||||||
<div className="col-md-6 mb-3">
|
<div className="col-md-6 mb-3">
|
||||||
<div className="d-flex">
|
<div className="d-flex">
|
||||||
|
Loading…
x
Reference in New Issue
Block a user