Added a Work Category dropdown in task create/update forms and a new column to display the category in Project Infrastructure and Daily Task Planning views.

This commit is contained in:
ashutosh.nehete 2025-05-12 13:11:29 +05:30
parent 634dc12e5b
commit e8ac267d46
5 changed files with 203 additions and 45 deletions

View File

@ -3,7 +3,7 @@ import { useForm } from "react-hook-form";
import { zodResolver } from "@hookform/resolvers/zod"; import { zodResolver } from "@hookform/resolvers/zod";
import { z } from "zod"; import { z } from "zod";
import { useActivitiesMaster } from "../../../hooks/masterHook/useMaster"; import { useActivitiesMaster, useWorkCategoriesMaster } from "../../../hooks/masterHook/useMaster";
import { useProjectDetails } from "../../../hooks/useProjects"; import { useProjectDetails } from "../../../hooks/useProjects";
import { useDispatch, useSelector } from "react-redux"; import { useDispatch, useSelector } from "react-redux";
import ProjectRepository from "../../../repositories/ProjectRepository"; import ProjectRepository from "../../../repositories/ProjectRepository";
@ -18,6 +18,7 @@ import showToast from "../../../services/toastService";
const taskSchema = z const taskSchema = z
.object({ .object({
activityID: z.string().min(1, "Activity is required"), activityID: z.string().min(1, "Activity is required"),
workCategoryId: z.string().min(1, "Work Category is required"),
plannedWork: z.number().min(1, "Planned Work must be greater than 0"), plannedWork: z.number().min(1, "Planned Work must be greater than 0"),
completedWork: z.number().min(0, "Completed Work must be greater than 0"), completedWork: z.number().min(0, "Completed Work must be greater than 0"),
}) })
@ -43,17 +44,21 @@ const EditActivityModal = ({
); );
const defaultModel = { const defaultModel = {
activityID: 0, activityID: 0,
workCategoryId: 0,
plannedWork: 0, plannedWork: 0,
completedWork: 0, completedWork: 0,
}; };
const {projects_Details, refetch} = useProjectDetails( selectedProject ); const { projects_Details, refetch } = useProjectDetails(selectedProject);
const [ActivityUnit,setActivityUnit]= useState("") const [ActivityUnit, setActivityUnit] = useState("");
const { activities, loading, error } = useActivitiesMaster(); const { activities, loading, error } = useActivitiesMaster();
const { categories, categoryLoading, categoryError } =
useWorkCategoriesMaster();
const [formData, setFormData] = useState(defaultModel); const [formData, setFormData] = useState(defaultModel);
const [selectedActivity, setSelectedActivity] = useState(null); const [selectedActivity, setSelectedActivity] = useState(null);
const [isSubmitting, setIsSubmitting] = useState(false); const [isSubmitting, setIsSubmitting] = useState(false);
const [activityData, setActivityData] = useState([]); const [activityData, setActivityData] = useState([]);
const [categoryData, setCategoryData] = useState([]);
const dispatch = useDispatch(); const dispatch = useDispatch();
const { const {
@ -70,12 +75,19 @@ const EditActivityModal = ({
}); });
const handleActivityChange = (e) => { const handleActivityChange = (e) => {
const selectedId = Number(e.target.value); const selectedId = String(e.target.value);
const selected = activityData.find( ( a ) => a.id === selectedId ); const selected = activityData.find((a) => a.id === selectedId);
setSelectedActivity(selected || null); setSelectedActivity(selected || null);
setValue("activityID", selectedId); setValue("activityID", selectedId);
}; };
const handleCategoryChange = (e) => {
const selectedId = String(e.target.value);
const category = categoryData.find((b) => b.id === selectedId);
setSelectedCategory(category || null);
setValue("workCategoryId", selectedId);
};
const onSubmitForm = async ( data ) => const onSubmitForm = async ( data ) =>
{ {
setIsSubmitting(true) setIsSubmitting(true)
@ -167,6 +179,8 @@ const EditActivityModal = ({
useEffect(() => { useEffect(() => {
reset({ reset({
activityID: workItem?.workItem?.activityId || workItem?.activityId || 0, activityID: workItem?.workItem?.activityId || workItem?.activityId || 0,
workCategoryId:
workItem?.workItem?.workCategoryId || workItem?.workCategoryId || 0,
plannedWork: plannedWork:
workItem?.workItem?.plannedWork || workItem?.plannedWork || 0, workItem?.workItem?.plannedWork || workItem?.plannedWork || 0,
completedWork: completedWork:
@ -177,14 +191,13 @@ const EditActivityModal = ({
const ISselectedActivity = watch("activityID"); const ISselectedActivity = watch("activityID");
useEffect(() => { useEffect(() => {
if( ISselectedActivity ){ if (ISselectedActivity) {
const selected = activities.find((a) => a.id === ISselectedActivity); const selected = activities.find((a) => a.id === ISselectedActivity);
setSelectedActivity( selected || null ); setSelectedActivity(selected || null);
setActivityUnit(selected?.unitOfMeasurement) setActivityUnit(selected?.unitOfMeasurement);
} }
}, [ ISselectedActivity,activities] ); }, [ISselectedActivity, activities]);
return ( return (
<div className="modal-dialog modal-lg modal-simple modal-edit-user"> <div className="modal-dialog modal-lg modal-simple modal-edit-user">
<div className="modal-content"> <div className="modal-content">
@ -279,6 +292,45 @@ const EditActivityModal = ({
)} )}
</div> </div>
{/* Select Category */}
<div className="col-12 col-md-12">
<label className="form-label" htmlFor="activityID">
Select Work Category
</label>
<select
id="workCategoryId"
className="form-select form-select-sm"
{...register("workCategoryId")}
>
{loading ? (
<option value="">Loading...</option>
) : (
<option disabled>Select Category</option>
)}
{categories &&
categories.length > 0 &&
categories
.slice()
.sort((a, b) =>
(a.name || "").localeCompare(
b.name || ""
)
)
.map((category) => (
<option key={category.id} value={category.id}>
{category.name}
</option>
))}
{!loading && categories.length === 0 && (
<option disabled>No categories available</option>
)}
</select>
{errors.workCategoryId && (
<p className="danger-text">{errors.workCategoryId.message}</p>
)}
</div>
{/* Planned Work */} {/* Planned Work */}
{/* {ISselectedActivity && ( */} {/* {ISselectedActivity && ( */}
<div className="col-5 col-md-5"> <div className="col-5 col-md-5">

View File

@ -2,23 +2,28 @@ import React, { useState, useEffect } from "react";
import { useForm } from "react-hook-form"; import { useForm } from "react-hook-form";
import { zodResolver } from "@hookform/resolvers/zod"; import { zodResolver } from "@hookform/resolvers/zod";
import { z } from "zod"; import { z } from "zod";
import {useActivitiesMaster} from "../../../hooks/masterHook/useMaster"; import {
useActivitiesMaster,
useWorkCategoriesMaster,
} from "../../../hooks/masterHook/useMaster";
const taskSchema = z.object({ const taskSchema = z.object({
buildingID: z.string().min(1, "Building is required"), buildingID: z.string().min(1, "Building is required"),
floorId: z.string().min(1, "Floor is required"), floorId: z.string().min(1, "Floor is required"),
workAreaId: z.string().min(1, "Work Area is required"), workAreaId: z.string().min(1, "Work Area is required"),
activityID: z.string().min(1, "Activity is required"), activityID: z.string().min(1, "Activity is required"),
workCategoryId: z.string().min(1, "Work Category is required"),
plannedWork: z.number().min(1, "Planned Work must be greater than 0"), plannedWork: z.number().min(1, "Planned Work must be greater than 0"),
completedWork: z.number().min(0, "Completed Work must be greater than 0"), completedWork: z.number().min(0, "Completed Work must be greater than 0"),
}); });
const defaultModel = { const defaultModel = {
id: null, id: null,
buildingID:"0", buildingID: "0",
floorId: "0", floorId: "0",
workAreaId: "0", workAreaId: "0",
activityID: null, activityID: null,
workCategoryId: "",
plannedWork: 0, plannedWork: 0,
completedWork: 0, completedWork: 0,
}; };
@ -30,15 +35,18 @@ const TaskModel = ({
onClearComplete, onClearComplete,
onClose, onClose,
}) => { }) => {
const [formData, setFormData] = useState(defaultModel); const [formData, setFormData] = useState(defaultModel);
const [selectedBuilding, setSelectedBuilding] = useState(null); const [selectedBuilding, setSelectedBuilding] = useState(null);
const [selectedFloor, setSelectedFloor] = useState(null); const [selectedFloor, setSelectedFloor] = useState(null);
const [selectedWorkArea, setSelectedWorkArea] = useState(null); const [selectedWorkArea, setSelectedWorkArea] = useState(null);
const [selectedActivity, setSelectedActivity] = useState(null); const [selectedActivity, setSelectedActivity] = useState(null);
const [selectedCategory, setSelectedCategory] = useState(null);
const [isSubmitting, setIsSubmitting] = useState(false); const [isSubmitting, setIsSubmitting] = useState(false);
const [activityData, setActivityData] = useState([]); const [activityData, setActivityData] = useState([]);
const {activities, loading, error} = useActivitiesMaster(); const [categoryData, setCategoryData] = useState([]);
const { activities, loading, error } = useActivitiesMaster();
const { categories, categoryLoading, categoryError } =
useWorkCategoriesMaster();
const { const {
register, register,
@ -79,6 +87,7 @@ const TaskModel = ({
floorId: value, floorId: value,
workAreaId: 0, workAreaId: 0,
activityID: 0, activityID: 0,
workCategoryId: categoryData?.[0]?.id?.toString() ?? "",
})); }));
}; };
@ -104,13 +113,24 @@ const TaskModel = ({
})); }));
}; };
const onSubmitForm = async ( data ) => const handleCategoryChange = (e) => {
{ const { value } = e.target;
const category = categoryData.find((b) => b.id === String(value));
setSelectedCategory(category);
reset((prev) => ({
...prev,
workCategoryId: String(value),
}));
};
const onSubmitForm = async (data) => {
console.log(data);
setIsSubmitting(true); setIsSubmitting(true);
await onSubmit(data); await onSubmit(data);
setValue("plannedWork", 0); setValue("plannedWork", 0);
setValue( "completedWork", 0 ); setValue("completedWork", 0);
setValue("activityID",0) setValue("activityID", 0);
setValue("workCategoryId", categoryData?.[0]?.id?.toString() ?? "");
setIsSubmitting(false); setIsSubmitting(false);
}; };
@ -120,15 +140,32 @@ const TaskModel = ({
setSelectedFloor(null); setSelectedFloor(null);
setSelectedWorkArea(null); setSelectedWorkArea(null);
setSelectedActivity(null); setSelectedActivity(null);
setSelectedCategory(categoryData?.[0]?.id?.toString() ?? "");
reset(defaultModel); reset(defaultModel);
}; };
useEffect(() => { useEffect(() => {
if (!loading && Array.isArray(activities) && activities.length > 0) { if (!loading && Array.isArray(activities) && activities.length > 0) {
setActivityData(activities); setActivityData(activities);
} }
}, [activities, loading]); }, [activities, loading]);
useEffect(() => {
if (
!categoryLoading &&
Array.isArray(categories) &&
categories.length > 0
) {
const newCategories = categories?.slice()?.sort((a, b) => {
const nameA = a?.name || "";
const nameB = b?.name || "";
return nameA.localeCompare(nameB);
});
setCategoryData(newCategories);
setSelectedCategory(newCategories[0])
}
}, [categories, categoryLoading]);
return ( return (
<div className="modal-dialog modal-lg modal-simple modal-edit-user"> <div className="modal-dialog modal-lg modal-simple modal-edit-user">
<div className="modal-content"> <div className="modal-content">
@ -247,9 +284,7 @@ const TaskModel = ({
{selectedWorkArea && ( {selectedWorkArea && (
<div className="col-12 col-md-12"> <div className="col-12 col-md-12">
<label className="form-label" > <label className="form-label">Select Activity</label>
Select Activity
</label>
<select <select
id="activityID" id="activityID"
className="form-select form-select-sm" className="form-select form-select-sm"
@ -257,28 +292,25 @@ const TaskModel = ({
onChange={handleActivityChange} onChange={handleActivityChange}
> >
<option value="0">Select Activity</option> <option value="0">Select Activity</option>
{activityData && activityData.length > 0 && ( {activityData &&
activityData.length > 0 &&
activityData activityData
?.slice() ?.slice()
?.sort( ( a, b ) => ?.sort((a, b) => {
{
const nameA = a?.activityName || ""; const nameA = a?.activityName || "";
const nameB = b?.activityName || ""; const nameB = b?.activityName || "";
return nameA.localeCompare( nameB ); return nameA.localeCompare(nameB);
} ) })
?.map( ( activity ) => ( ?.map((activity) => (
<option key={activity.id} value={activity.id}> <option key={activity.id} value={activity.id}>
{ {activity.activityName ||
activity.activityName || `Unnamed (id: ${activity.id})`}
`Unnamed (id: ${ activity.id })`}
</option> </option>
) ) ))}
) } {!loading && activities.length === 0 && (
{(!loading && activities.length === 0 )&& (
<option disabled>No activities available</option> <option disabled>No activities available</option>
)} )}
{loading && ( <option disabled>Loading...</option>)} {loading && <option disabled>Loading...</option>}
</select> </select>
{errors.activityID && ( {errors.activityID && (
@ -287,7 +319,38 @@ const TaskModel = ({
</div> </div>
)} )}
{selectedActivity && ( {selectedWorkArea && (
<div className="col-12 col-md-12">
<label className="form-label">Select Work Category</label>
<select
id="workCategoryId"
className="form-select form-select-sm"
{...register("workCategoryId")}
onChange={handleCategoryChange}
>
{categoryData &&
categoryData.length > 0 &&
categoryData
?.map((category) => (
<option key={category.id} value={category.id}>
{category.name || `Unnamed (id: ${category.id})`}
</option>
))}
{!categoryLoading && categories.length === 0 && (
<option disabled>No activities available</option>
)}
{categoryLoading && <option disabled>Loading...</option>}
</select>
{errors.workCategoryId && (
<p className="danger-text">
{errors.workCategoryId.message}
</p>
)}
</div>
)}
{selectedActivity && selectedCategory && (
<div className="col-5 col-md-5"> <div className="col-5 col-md-5">
<label className="form-label" htmlFor="plannedWork"> <label className="form-label" htmlFor="plannedWork">
{formData.id !== "0" ? "Modify " : "Enter "} Planned Work {formData.id !== "0" ? "Modify " : "Enter "} Planned Work
@ -304,7 +367,7 @@ const TaskModel = ({
</div> </div>
)} )}
{selectedActivity && ( {selectedActivity && selectedCategory && (
<div className="col-5 col-md-5"> <div className="col-5 col-md-5">
<label className="form-label" htmlFor="completedWork"> <label className="form-label" htmlFor="completedWork">
{formData.id !== "0" ? "Modify " : "Enter "} Completed Work {formData.id !== "0" ? "Modify " : "Enter "} Completed Work
@ -324,7 +387,7 @@ const TaskModel = ({
)} )}
{/* Unit */} {/* Unit */}
{selectedActivity && ( {selectedActivity && selectedCategory && (
<div className="col-2 col-md-2"> <div className="col-2 col-md-2">
<label className="form-label" htmlFor="unit"> <label className="form-label" htmlFor="unit">
Unit Unit
@ -340,9 +403,7 @@ const TaskModel = ({
<div className="col-12 text-center"> <div className="col-12 text-center">
<button type="submit" className="btn btn-sm btn-primary me-3"> <button type="submit" className="btn btn-sm btn-primary me-3">
{isSubmitting {isSubmitting ? "Please Wait.." : "Add Task"}
? "Please Wait.."
: "Add Task"}
</button> </button>
<button <button
type="button" type="button"

View File

@ -121,6 +121,9 @@ const WorkArea = ({ workArea, floor, forBuilding }) => {
<th className="infra-activity-table-header d-sm-none d-sm-table-cell"> <th className="infra-activity-table-header d-sm-none d-sm-table-cell">
Status Status
</th> </th>
<th className="infra-activity-table-header-first">
Category
</th>
{/* for greather than mobile view ************* */} {/* for greather than mobile view ************* */}
<th className="infra-activity-table-header d-none d-md-table-cell"> <th className="infra-activity-table-header d-none d-md-table-cell">
Planned Planned

View File

@ -160,6 +160,15 @@ const WorkItem = ( {
? NewWorkItem?.workItem?.plannedWork || workItem?.plannedWork ? NewWorkItem?.workItem?.plannedWork || workItem?.plannedWork
: "NA"} : "NA"}
</td> </td>
<td className="text-start table-cell-small">
<span className="fw-light">
{hasWorkItem
? NewWorkItem?.workItem?.workCategoryMaster?.name ||
workItem.workCategoryMaster?.name || "NA"
: "NA"}
</span>
</td>
{/* for greather than mobile view ************* */} {/* for greather than mobile view ************* */}
<td className="text-center d-none d-md-table-cell"> <td className="text-center d-none d-md-table-cell">
{hasWorkItem {hasWorkItem

View File

@ -116,4 +116,37 @@ export const useActivitiesMaster = () =>
}, [] ) }, [] )
return {activities,loading,error} return {activities,loading,error}
} }
export const useWorkCategoriesMaster = () =>
{
const [ categories, setCategories ] = useState( [] )
const [ categoryLoading, setloading ] = useState( false );
const [ categoryError, setError ] = useState( "" )
const fetchCategories =async () => {
const cacheddata = getCachedData("Work Category");
if (!cacheddata) {
setloading(true);
try {
const response = await MasterRespository.getWorkCategory();
setCategories(response.data);
cacheData("Work Category", response.data);
} catch (err) {
setError(err);
console.log(err);
} finally {
setloading(false);
}
} else {
setCategories(cacheddata);
}
}
useEffect( () =>
{
fetchCategories()
}, [] )
return {categories,categoryLoading,categoryError}
}