spearated component for updated logs Expense and RequestedPayment

This commit is contained in:
pramod.mahajan 2025-11-04 22:54:10 +05:30
parent 69e60caf23
commit f688a7169e
6 changed files with 480 additions and 398 deletions

View File

@ -8,11 +8,11 @@ const ExpenseStatusLogs = ({ data }) => {
const [visibleCount, setVisibleCount] = useState(4); const [visibleCount, setVisibleCount] = useState(4);
const sortedLogs = useMemo(() => { const sortedLogs = useMemo(() => {
if (!data?.updateLogs) return []; if (!data?.expenseLogs) return [];
return [...data.updateLogs].sort( return [...data.expenseLogs].sort(
(a, b) => new Date(b.updatedAt) - new Date(a.updatedAt) (a, b) => new Date(b.updatedAt) - new Date(a.updatedAt)
); );
}, [data?.updateLogs]); }, [data?.expenseLogs]);
const logsToShow = useMemo( const logsToShow = useMemo(
() => sortedLogs.slice(0, visibleCount), () => sortedLogs.slice(0, visibleCount),
@ -22,8 +22,8 @@ const ExpenseStatusLogs = ({ data }) => {
const timelineData = useMemo(() => { const timelineData = useMemo(() => {
return logsToShow.map((log, index) => ({ return logsToShow.map((log, index) => ({
id: index + 1, id: index + 1,
title: log.nextStatus?.name || "Status Updated", title: log.action || "Status Updated",
description: log.nextStatus?.description || "", description: log.comment || "",
timeAgo: log.updatedAt, timeAgo: log.updatedAt,
color: getColorNameFromHex(log.nextStatus?.color) || "primary", color: getColorNameFromHex(log.nextStatus?.color) || "primary",
users: log.updatedBy users: log.updatedBy
@ -45,45 +45,7 @@ const ExpenseStatusLogs = ({ data }) => {
return ( return (
<div className="page-min-h overflow-auto"> <div className="page-min-h overflow-auto">
{/* <div className="row g-2">
{logsToShow.map((log) => (
<div key={log.id} className="col-12 d-flex align-items-start mb-1">
<Avatar
size="xs"
firstName={log.updatedBy.firstName}
lastName={log.updatedBy.lastName}
/>
<div className="flex-grow-1">
<div className="text-start">
<div className="flex">
<span>{`${log.updatedBy.firstName} ${log.updatedBy.lastName}`}</span>
<small className="text-secondary text-tiny ms-2">
<em>{log.action}</em>
</small>
<span className="text-tiny text-secondary d-block">
{formatUTCToLocalTime(log.updateAt, true)}
</span>
</div>
<div className="d-flex align-items-center text-muted small mt-1">
<span>{log.comment}</span>
</div>
</div>
</div>
</div>
))}
</div>
{sortedLogs.length > visibleCount && (
<div className="text-center my-1">
<button
className="btn btn-xs btn-outline-primary"
onClick={handleShowMore}
>
Show More
</button>
</div>
)} */}
<Timeline items={timelineData} /> <Timeline items={timelineData} />
</div> </div>

View File

@ -1,9 +1,10 @@
import React from "react"; import React from "react";
import { formatFileSize, getIconByFileType } from "../../utils/appUtils"; import { formatFileSize, getIconByFileType } from "../../utils/appUtils";
import Tooltip from "../common/Tooltip";
const Filelist = ({ files, removeFile, expenseToEdit }) => { const Filelist = ({ files, removeFile, expenseToEdit }) => {
return ( return (
<div className="d-block"> <div className="d-flex flex-wrap gap-2 mt-2">
{files {files
.filter((file) => { .filter((file) => {
if (expenseToEdit) { if (expenseToEdit) {
@ -12,7 +13,7 @@ const Filelist = ({ files, removeFile, expenseToEdit }) => {
return true; return true;
}) })
.map((file, idx) => ( .map((file, idx) => (
<div className="col-12 col-sm-6 col-md-4 col-lg-8 bg-white shadow-sm rounded p-2 m-2" key={idx}> <div className="col-12 col-sm-6 col-md-4 bg-white " key={idx}>
<div className="row align-items-center"> <div className="row align-items-center">
{/* File icon and info */} {/* File icon and info */}
<div className="col-10 d-flex align-items-center gap-2"> <div className="col-10 d-flex align-items-center gap-2">
@ -34,6 +35,7 @@ const Filelist = ({ files, removeFile, expenseToEdit }) => {
</div> </div>
<div className="col-2 text-end"> <div className="col-2 text-end">
<Tooltip text={"Remove file"}>
<i <i
className="bx bx-trash fs-4 cursor-pointer text-danger bx-sm " className="bx bx-trash fs-4 cursor-pointer text-danger bx-sm "
role="button" role="button"
@ -42,6 +44,8 @@ const Filelist = ({ files, removeFile, expenseToEdit }) => {
removeFile(expenseToEdit ? file.documentId : idx); removeFile(expenseToEdit ? file.documentId : idx);
}} }}
></i> ></i>
</Tooltip>
</div> </div>
</div> </div>
</div> </div>

View File

@ -9,7 +9,11 @@ import { useForm } from "react-hook-form";
import { zodResolver } from "@hookform/resolvers/zod"; import { zodResolver } from "@hookform/resolvers/zod";
import { defaultActionValues, ExpenseActionScheam } from "./ExpenseSchema"; import { defaultActionValues, ExpenseActionScheam } from "./ExpenseSchema";
import { useExpenseContext } from "../../pages/Expense/ExpensePage"; import { useExpenseContext } from "../../pages/Expense/ExpensePage";
import { getColorNameFromHex, getIconByFileType, localToUtc } from "../../utils/appUtils"; import {
getColorNameFromHex,
getIconByFileType,
localToUtc,
} from "../../utils/appUtils";
import { ExpenseDetailsSkeleton } from "./ExpenseSkeleton"; import { ExpenseDetailsSkeleton } from "./ExpenseSkeleton";
import { useHasUserPermission } from "../../hooks/useHasUserPermission"; import { useHasUserPermission } from "../../hooks/useHasUserPermission";
import { import {
@ -38,7 +42,8 @@ const ViewExpense = ({ ExpenseId }) => {
const IsReview = useHasUserPermission(REVIEW_EXPENSE); const IsReview = useHasUserPermission(REVIEW_EXPENSE);
const [imageLoaded, setImageLoaded] = useState({}); const [imageLoaded, setImageLoaded] = useState({});
const { setDocumentView } = useExpenseContext(); const { setDocumentView } = useExpenseContext();
const ActionSchema = ExpenseActionScheam(IsPaymentProcess,data?.createdAt) ?? z.object({}); const ActionSchema =
ExpenseActionScheam(IsPaymentProcess, data?.createdAt) ?? z.object({});
const navigate = useNavigate(); const navigate = useNavigate();
const { const {
register, register,
@ -91,7 +96,7 @@ const ViewExpense = ({ ExpenseId }) => {
const onSubmit = (formData) => { const onSubmit = (formData) => {
const Payload = { const Payload = {
...formData, ...formData,
reimburseDate:localToUtc(formData.reimburseDate), reimburseDate: localToUtc(formData.reimburseDate),
expenseId: ExpenseId, expenseId: ExpenseId,
comment: formData.comment, comment: formData.comment,
}; };
@ -106,12 +111,15 @@ 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-1">
<div className="col-12 mb-1"> <div className="col-12 mb-1">
<h5 className="fw-semibold m-0">Expense Details</h5> <h5 className="fw-semibold m-0">Expense Details</h5>
<hr />
</div> </div>
<div className="col-12 text-start fw-semibold my-2">{data?.expenseUId}</div> <div className="row mb-1">
<div className="col-12 col-md-7">
<div className="row">
<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">
@ -214,7 +222,9 @@ const ViewExpense = ({ ExpenseId }) => {
> >
Pre-Approved : Pre-Approved :
</label> </label>
<div className="text-muted">{data.preApproved ? "Yes" : "No"}</div> <div className="text-muted">
{data.preApproved ? "Yes" : "No"}
</div>
</div> </div>
</div> </div>
@ -297,10 +307,11 @@ const ViewExpense = ({ ExpenseId }) => {
<label className="fw-semibold form-label">Description : </label> <label className="fw-semibold form-label">Description : </label>
<div className="text-muted">{data?.description}</div> <div className="text-muted">{data?.description}</div>
</div> </div>
</div>
<div className="col-12 text-start"> <div className="col-12 text-start">
<label className="form-label me-2 mb-2 fw-semibold">Attachment :</label> <label className="form-label me-2 mb-2 fw-semibold">
Attachment :
</label>
<div className="d-flex flex-wrap gap-2"> <div className="d-flex flex-wrap gap-2">
{data?.documents?.map((doc) => { {data?.documents?.map((doc) => {
@ -363,8 +374,12 @@ const ViewExpense = ({ ExpenseId }) => {
<Avatar <Avatar
size="xs" size="xs"
classAvatar="m-0 me-1" classAvatar="m-0 me-1"
firstName={data?.expensesReimburse?.reimburseBy?.firstName} firstName={
lastName={data?.expensesReimburse?.reimburseBy?.lastName} data?.expensesReimburse?.reimburseBy?.firstName
}
lastName={
data?.expensesReimburse?.reimburseBy?.lastName
}
/> />
<span className="text-muted"> <span className="text-muted">
{`${data?.expensesReimburse?.reimburseBy?.firstName} ${data?.expensesReimburse?.reimburseBy?.lastName}`.trim()} {`${data?.expensesReimburse?.reimburseBy?.firstName} ${data?.expensesReimburse?.reimburseBy?.lastName}`.trim()}
@ -374,8 +389,6 @@ const ViewExpense = ({ ExpenseId }) => {
)} )}
</div> </div>
)} )}
<hr className="divider my-1 border-2 divider-primary my-2" />
{Array.isArray(data?.nextStatus) && data.nextStatus.length > 0 && ( {Array.isArray(data?.nextStatus) && data.nextStatus.length > 0 && (
<> <>
{IsPaymentProcess && nextStatusWithPermission?.length > 0 && ( {IsPaymentProcess && nextStatusWithPermission?.length > 0 && (
@ -418,10 +431,13 @@ const ViewExpense = ({ ExpenseId }) => {
</div> </div>
)} )}
<div className="col-12 mb-3 text-start"> <div className="col-12 mb-3 text-start">
{((nextStatusWithPermission.length > 0 && !IsRejectedExpense) || {((nextStatusWithPermission.length > 0 &&
!IsRejectedExpense) ||
(IsRejectedExpense && isCreatedBy)) && ( (IsRejectedExpense && isCreatedBy)) && (
<> <>
<Label className="form-label me-2 mb-0" required>Comment</Label> <Label className="form-label me-2 mb-0" required>
Comment
</Label>
<textarea <textarea
className="form-control form-control-sm" className="form-control form-control-sm"
{...register("comment")} {...register("comment")}
@ -460,8 +476,14 @@ const ViewExpense = ({ ExpenseId }) => {
</div> </div>
</> </>
)} )}
</div>
</div>
<div className="col-12 col-md-5">
<div className="row">
<ExpenseStatusLogs data={data} /> <ExpenseStatusLogs data={data} />
</div>
</div>
</div>
</form> </form>
); );
}; };

View File

@ -0,0 +1,93 @@
import { useState, useMemo } from "react";
import Avatar from "../common/Avatar";
import { formatUTCToLocalTime } from "../../utils/dateUtils";
import Timeline from "../common/TimeLine";
import moment from "moment";
import { getColorNameFromHex } from "../../utils/appUtils";
const PaymentStatusLogs = ({ data }) => {
const [visibleCount, setVisibleCount] = useState(4);
const sortedLogs = useMemo(() => {
if (!data?.updateLogs) return [];
return [...data.updateLogs].sort(
(a, b) => new Date(b.updatedAt) - new Date(a.updatedAt)
);
}, [data?.updateLogs]);
const logsToShow = useMemo(
() => sortedLogs.slice(0, visibleCount),
[sortedLogs, visibleCount]
);
const timelineData = useMemo(() => {
return logsToShow.map((log, index) => ({
id: index + 1,
title: log.nextStatus?.name || "Status Updated",
description: log.nextStatus?.description || "",
timeAgo: log.updatedAt,
color: getColorNameFromHex(log.nextStatus?.color) || "primary",
users: log.updatedBy
? [
{
firstName: log.updatedBy.firstName || "",
lastName: log?.updatedBy?.lastName || "",
role: log.updatedBy.jobRoleName || "",
avatar: log.updatedBy.photo,
},
]
: [],
}));
}, [logsToShow]);
const handleShowMore = () => {
setVisibleCount((prev) => prev + 4);
};
return (
<div className="page-min-h overflow-auto">
{/* <div className="row g-2">
{logsToShow.map((log) => (
<div key={log.id} className="col-12 d-flex align-items-start mb-1">
<Avatar
size="xs"
firstName={log.updatedBy.firstName}
lastName={log.updatedBy.lastName}
/>
<div className="flex-grow-1">
<div className="text-start">
<div className="flex">
<span>{`${log.updatedBy.firstName} ${log.updatedBy.lastName}`}</span>
<small className="text-secondary text-tiny ms-2">
<em>{log.action}</em>
</small>
<span className="text-tiny text-secondary d-block">
{formatUTCToLocalTime(log.updateAt, true)}
</span>
</div>
<div className="d-flex align-items-center text-muted small mt-1">
<span>{log.comment}</span>
</div>
</div>
</div>
</div>
))}
</div>
{sortedLogs.length > visibleCount && (
<div className="text-center my-1">
<button
className="btn btn-xs btn-outline-primary"
onClick={handleShowMore}
>
Show More
</button>
</div>
)} */}
<Timeline items={timelineData} />
</div>
);
};
export default PaymentStatusLogs;

View File

@ -38,6 +38,7 @@ import {
defaultPaymentRequestActionValues, defaultPaymentRequestActionValues,
PaymentRequestActionScheam, PaymentRequestActionScheam,
} from "./PaymentRequestSchema"; } from "./PaymentRequestSchema";
import PaymentStatusLogs from "./PaymentStatusLogs";
const ViewPaymentRequest = ({ requestId }) => { const ViewPaymentRequest = ({ requestId }) => {
const { data, isLoading, isError, error, isFetching } = const { data, isLoading, isError, error, isFetching } =
@ -474,7 +475,7 @@ const ViewPaymentRequest = ({ requestId }) => {
<i className="bx bx-time-five me-2 "></i>{" "} <i className="bx bx-time-five me-2 "></i>{" "}
<p className="fw-medium">TimeLine</p> <p className="fw-medium">TimeLine</p>
</div> </div>
<ExpenseStatusLogs data={data} /> <PaymentStatusLogs data={data} />
</div> </div>
</div> </div>
</form> </form>

View File

@ -196,7 +196,7 @@ const ExpensePage = () => {
{viewExpense.view && ( {viewExpense.view && (
<GlobalModel <GlobalModel
isOpen isOpen
size="lg" size="xl"
modalType="top" modalType="top"
closeModal={() => setViewExpense({ expenseId: null, view: false })} closeModal={() => setViewExpense({ expenseId: null, view: false })}
> >