Organization_Management : Organization Hierarchy #443
| @ -16,6 +16,7 @@ import ManagePaymentMode from "./ManagePaymentMode"; | |||||||
| import ManageExpenseStatus from "./ManageExpenseStatus"; | import ManageExpenseStatus from "./ManageExpenseStatus"; | ||||||
| import ManageDocumentCategory from "./ManageDocumentCategory"; | import ManageDocumentCategory from "./ManageDocumentCategory"; | ||||||
| import ManageDocumentType from "./ManageDocumentType"; | import ManageDocumentType from "./ManageDocumentType"; | ||||||
|  | import ManageServices from "./Services/ManageServices"; | ||||||
| 
 | 
 | ||||||
| const MasterModal = ({ modaldata, closeModal }) => { | const MasterModal = ({ modaldata, closeModal }) => { | ||||||
|   if (!modaldata?.modalType || modaldata.modalType === "delete") { |   if (!modaldata?.modalType || modaldata.modalType === "delete") { | ||||||
| @ -60,6 +61,12 @@ const MasterModal = ({ modaldata, closeModal }) => { | |||||||
|     "Edit-Document Type": ( |     "Edit-Document Type": ( | ||||||
|       <ManageDocumentType data={item} onClose={closeModal} /> |       <ManageDocumentType data={item} onClose={closeModal} /> | ||||||
|     ), |     ), | ||||||
|  |     "Services": ( | ||||||
|  |       <ManageServices onClose={closeModal} /> | ||||||
|  |     ), | ||||||
|  |     "Edit-Services": ( | ||||||
|  |       <ManageServices data={item} onClose={closeModal} /> | ||||||
|  |     ), | ||||||
|   }; |   }; | ||||||
| 
 | 
 | ||||||
|   return modalComponents[modalType] || null; |   return modalComponents[modalType] || null; | ||||||
|  | |||||||
							
								
								
									
										107
									
								
								src/components/master/Services/ManageServices.jsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										107
									
								
								src/components/master/Services/ManageServices.jsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,107 @@ | |||||||
|  | import React, { useEffect } from "react"; | ||||||
|  | import Label from "../../common/Label"; | ||||||
|  | import { useForm } from "react-hook-form"; | ||||||
|  | import { useCreateService, useUpdateService } from "../../../hooks/masterHook/useMaster"; | ||||||
|  | import { z } from "zod"; | ||||||
|  | import { zodResolver } from "@hookform/resolvers/zod"; | ||||||
|  | 
 | ||||||
|  | const schema = z.object({ | ||||||
|  |   name: z.string().min(1, { message: "Service Name is required" }), | ||||||
|  |   description: z | ||||||
|  |     .string() | ||||||
|  |     .min(1, { message: "Description is required" }) | ||||||
|  |     .max(255, { message: "Description cannot exceed 255 characters" }), | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | const ManageServices = ({ data , onClose }) => { | ||||||
|  |   const { | ||||||
|  |     register, | ||||||
|  |     handleSubmit, | ||||||
|  |     reset, | ||||||
|  |     formState: { errors }, | ||||||
|  |   } = useForm({ | ||||||
|  |     resolver: zodResolver(schema), | ||||||
|  |     defaultValues: { name: "", description: "" }, | ||||||
|  |   }); | ||||||
|  | 
 | ||||||
|  |   const { mutate: CreateServices, isPending: Creating } = useCreateService(() => | ||||||
|  |     onClose?.() | ||||||
|  |   ); | ||||||
|  |   const { mutate: UpdateServices, isPending: Updating } = useUpdateService(() => | ||||||
|  |     onClose?.() | ||||||
|  |   ); | ||||||
|  | 
 | ||||||
|  |   const onSubmit = (payload) => { | ||||||
|  |     if (data && data.id) { | ||||||
|  |       UpdateServices({ id: data.id, payload: { ...payload, id: data.id } }); | ||||||
|  |     } else { | ||||||
|  |       CreateServices(payload); | ||||||
|  |     } | ||||||
|  |   }; | ||||||
|  | 
 | ||||||
|  |   useEffect(() => { | ||||||
|  |     if (data) { | ||||||
|  |       reset({ | ||||||
|  |         name: data.name ?? "", | ||||||
|  |         description: data.description ?? "", | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |   }, [data, reset]); | ||||||
|  | 
 | ||||||
|  |   return ( | ||||||
|  |     <form className="row g-2" onSubmit={handleSubmit(onSubmit)}> | ||||||
|  |       <div className="col-12 col-md-12 text-start"> | ||||||
|  |         <Label className="form-label" required> | ||||||
|  |           Service Name | ||||||
|  |         </Label> | ||||||
|  |         <input | ||||||
|  |           type="text" | ||||||
|  |           {...register("name")} | ||||||
|  |           className={`form-control ${errors.name ? "is-invalid" : ""}`} | ||||||
|  |         /> | ||||||
|  |         {errors.name && <p className="danger-text">{errors.name.message}</p>} | ||||||
|  |       </div> | ||||||
|  | 
 | ||||||
|  |       <div className="col-12 col-md-12 text-start"> | ||||||
|  |         <Label className="form-label" htmlFor="description" required> | ||||||
|  |           Description | ||||||
|  |         </Label> | ||||||
|  |         <textarea | ||||||
|  |           rows="3" | ||||||
|  |           {...register("description")} | ||||||
|  |           className={`form-control ${errors.description ? "is-invalid" : ""}`} | ||||||
|  |         ></textarea> | ||||||
|  |         {errors.description && ( | ||||||
|  |           <p className="danger-text">{errors.description.message}</p> | ||||||
|  |         )} | ||||||
|  |       </div> | ||||||
|  | 
 | ||||||
|  |       <div className="col-12 text-end"> | ||||||
|  |         <button | ||||||
|  |           type="reset" | ||||||
|  |           className="btn btn-sm btn-label-secondary me-3" | ||||||
|  |           data-bs-dismiss="modal" | ||||||
|  |           aria-label="Close" | ||||||
|  |           disabled={Creating || Updating} | ||||||
|  |         > | ||||||
|  |           Cancel | ||||||
|  |         </button> | ||||||
|  |         <button | ||||||
|  |           type="submit" | ||||||
|  |           className="btn btn-sm btn-primary" | ||||||
|  |           disabled={Creating || Updating} | ||||||
|  |         > | ||||||
|  |           {Creating | ||||||
|  |             ? "Please Wait..." | ||||||
|  |             : Updating | ||||||
|  |             ? "Please Wait..." | ||||||
|  |             : data | ||||||
|  |             ? "Update" | ||||||
|  |             : "Submit"} | ||||||
|  |         </button> | ||||||
|  |       </div> | ||||||
|  |     </form> | ||||||
|  |   ); | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | export default ManageServices; | ||||||
							
								
								
									
										9
									
								
								src/components/master/Services/ServicesSchema.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								src/components/master/Services/ServicesSchema.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,9 @@ | |||||||
|  | import { z } from "zod"; | ||||||
|  | 
 | ||||||
|  | const schema = z.object({ | ||||||
|  |   name: z.string().min(1, { message: "Service Name is required" }), | ||||||
|  |   description: z | ||||||
|  |     .string() | ||||||
|  |     .min(1, { message: "Description is required" }) | ||||||
|  |     .max(255, { message: "Description cannot exceed 255 characters" }), | ||||||
|  | }); | ||||||
| @ -248,6 +248,8 @@ const fetchMasterData = async (masterType) => { | |||||||
|       return (await MasterRespository.getJobRole()).data; |       return (await MasterRespository.getJobRole()).data; | ||||||
|     case "Activity": |     case "Activity": | ||||||
|       return (await MasterRespository.getActivites()).data; |       return (await MasterRespository.getActivites()).data; | ||||||
|  |       case "Services": | ||||||
|  |       return (await MasterRespository.getService()).data; | ||||||
|     case "Work Category": |     case "Work Category": | ||||||
|       return (await MasterRespository.getWorkCategory()).data; |       return (await MasterRespository.getWorkCategory()).data; | ||||||
|     case "Contact Category": |     case "Contact Category": | ||||||
| @ -684,6 +686,7 @@ export const useCreatePaymentMode = (onSuccessCallback)=>{ | |||||||
|     } |     } | ||||||
|   }) |   }) | ||||||
| } | } | ||||||
|  | 
 | ||||||
| export const useUpdatePaymentMode = (onSuccessCallback)=>{ | export const useUpdatePaymentMode = (onSuccessCallback)=>{ | ||||||
|     const queryClient = useQueryClient(); |     const queryClient = useQueryClient(); | ||||||
| 
 | 
 | ||||||
| @ -706,6 +709,81 @@ export const useUpdatePaymentMode = (onSuccessCallback)=>{ | |||||||
|   }) |   }) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
|  | // Services-------------------------------
 | ||||||
|  | 
 | ||||||
|  | // export const useCreateService = (onSuccessCallback) => {
 | ||||||
|  | //   const queryClient = useQueryClient();
 | ||||||
|  | 
 | ||||||
|  | //   return useMutation({
 | ||||||
|  | //     mutationFn: async (payload) => {
 | ||||||
|  | //       const resp = await MasterRespository.createService(payload);
 | ||||||
|  | //       return resp.data; // full API response
 | ||||||
|  | //     },
 | ||||||
|  | //     onSuccess: (data) => {
 | ||||||
|  | //       // Invalidate & refetch service list
 | ||||||
|  | //       queryClient.invalidateQueries({ queryKey: ["masterData", "Services"] });
 | ||||||
|  | 
 | ||||||
|  | //       showToast(data?.message || "Service added successfully", "success");
 | ||||||
|  | 
 | ||||||
|  | //       if (onSuccessCallback) onSuccessCallback(data?.data); // pass back new service object
 | ||||||
|  | //     },
 | ||||||
|  | //     onError: (error) => {
 | ||||||
|  | //       showToast(error.message || "Something went wrong", "error");
 | ||||||
|  | //     },
 | ||||||
|  | //   });
 | ||||||
|  | // };
 | ||||||
|  | 
 | ||||||
|  | export const useCreateService = (onSuccessCallback) => { | ||||||
|  |   const queryClient = useQueryClient(); | ||||||
|  | 
 | ||||||
|  |   return useMutation({ | ||||||
|  |     mutationFn: async (payload) => { | ||||||
|  |       debugger;  | ||||||
|  |       const resp = await MasterRespository.createService(payload); | ||||||
|  |       debugger;  | ||||||
|  |       return resp.data; | ||||||
|  |     }, | ||||||
|  |     onSuccess: (data) => { | ||||||
|  |       debugger;  | ||||||
|  |       queryClient.invalidateQueries({ queryKey: ["masterData", "Services"] }); | ||||||
|  | 
 | ||||||
|  |       showToast(data?.message || "Service added successfully", "success"); | ||||||
|  | 
 | ||||||
|  |       if (onSuccessCallback) onSuccessCallback(data?.data);  | ||||||
|  |     }, | ||||||
|  |     onError: (error) => { | ||||||
|  |       debugger;  | ||||||
|  |       showToast(error.message || "Something went wrong", "error"); | ||||||
|  |     }, | ||||||
|  |   }); | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | export const useUpdateService = (onSuccessCallback) => { | ||||||
|  |   const queryClient = useQueryClient(); | ||||||
|  | 
 | ||||||
|  |   return useMutation({ | ||||||
|  |     mutationFn: async ({ id, payload }) => { | ||||||
|  |       const response = await MasterRespository.updateService(id, payload); | ||||||
|  |       return response; // full response since it already has { success, message, data }
 | ||||||
|  |     }, | ||||||
|  |     onSuccess: (data, variables) => { | ||||||
|  |       queryClient.invalidateQueries({ | ||||||
|  |         queryKey: ["masterData", "Services"], | ||||||
|  |       }); | ||||||
|  | 
 | ||||||
|  |       showToast(data.message || "Service updated successfully.", "success"); | ||||||
|  | 
 | ||||||
|  |       if (onSuccessCallback) onSuccessCallback(data); | ||||||
|  |     }, | ||||||
|  |     onError: (error) => { | ||||||
|  |       showToast(error?.message || "Something went wrong", "error"); | ||||||
|  |     }, | ||||||
|  |   }); | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| // -------------------Expense Status----------------------------------
 | // -------------------Expense Status----------------------------------
 | ||||||
| export const useCreateExpenseStatus =(onSuccessCallback)=>{ | export const useCreateExpenseStatus =(onSuccessCallback)=>{ | ||||||
|       const queryClient = useQueryClient(); |       const queryClient = useQueryClient(); | ||||||
|  | |||||||
| @ -91,7 +91,7 @@ const MasterPage = () => { | |||||||
|       {modalConfig && ( |       {modalConfig && ( | ||||||
|         <GlobalModel |         <GlobalModel | ||||||
|           size={ |           size={ | ||||||
|             ["Application Role", "Edit-Application Role"].includes( |             ["Application Role", "Edit-Application Role", "Services"].includes( | ||||||
|               modalConfig.masterType |               modalConfig.masterType | ||||||
|             ) |             ) | ||||||
|               ? "lg" |               ? "lg" | ||||||
|  | |||||||
| @ -31,6 +31,13 @@ export const MasterRespository = { | |||||||
| 
 | 
 | ||||||
|   getActivites: () => api.get("api/master/activities"), |   getActivites: () => api.get("api/master/activities"), | ||||||
|   createActivity: (data) => api.post("api/master/activity", data), |   createActivity: (data) => api.post("api/master/activity", data), | ||||||
|  | 
 | ||||||
|  | //Services | ||||||
|  |   getService: () => api.get("api/master/service/list"), | ||||||
|  |   createService: (data) => api.post("api/master/service/create", data), | ||||||
|  |   updateService: (id, data) => api.put(`api/master/service/edit/${id}`, data), | ||||||
|  |   "Services": (id) => api.delete(`/api/master/service/delete/${id}`), | ||||||
|  | 
 | ||||||
|   updateActivity: (id, data) => |   updateActivity: (id, data) => | ||||||
|     api.post(`api/master/activity/edit/${id}`, data), |     api.post(`api/master/activity/edit/${id}`, data), | ||||||
|   getIndustries: () => api.get("api/master/industries"), |   getIndustries: () => api.get("api/master/industries"), | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user