Organization_Management : Organization Hierarchy #443
| @ -16,6 +16,7 @@ import ManagePaymentMode from "./ManagePaymentMode"; | ||||
| import ManageExpenseStatus from "./ManageExpenseStatus"; | ||||
| import ManageDocumentCategory from "./ManageDocumentCategory"; | ||||
| import ManageDocumentType from "./ManageDocumentType"; | ||||
| import ManageServices from "./Services/ManageServices"; | ||||
| 
 | ||||
| const MasterModal = ({ modaldata, closeModal }) => { | ||||
|   if (!modaldata?.modalType || modaldata.modalType === "delete") { | ||||
| @ -60,6 +61,12 @@ const MasterModal = ({ modaldata, closeModal }) => { | ||||
|     "Edit-Document Type": ( | ||||
|       <ManageDocumentType data={item} onClose={closeModal} /> | ||||
|     ), | ||||
|     "Services": ( | ||||
|       <ManageServices onClose={closeModal} /> | ||||
|     ), | ||||
|     "Edit-Services": ( | ||||
|       <ManageServices data={item} onClose={closeModal} /> | ||||
|     ), | ||||
|   }; | ||||
| 
 | ||||
|   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; | ||||
|     case "Activity": | ||||
|       return (await MasterRespository.getActivites()).data; | ||||
|       case "Services": | ||||
|       return (await MasterRespository.getService()).data; | ||||
|     case "Work Category": | ||||
|       return (await MasterRespository.getWorkCategory()).data; | ||||
|     case "Contact Category": | ||||
| @ -684,6 +686,7 @@ export const useCreatePaymentMode = (onSuccessCallback)=>{ | ||||
|     } | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| export const useUpdatePaymentMode = (onSuccessCallback)=>{ | ||||
|     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----------------------------------
 | ||||
| export const useCreateExpenseStatus =(onSuccessCallback)=>{ | ||||
|       const queryClient = useQueryClient(); | ||||
|  | ||||
| @ -91,7 +91,7 @@ const MasterPage = () => { | ||||
|       {modalConfig && ( | ||||
|         <GlobalModel | ||||
|           size={ | ||||
|             ["Application Role", "Edit-Application Role"].includes( | ||||
|             ["Application Role", "Edit-Application Role", "Services"].includes( | ||||
|               modalConfig.masterType | ||||
|             ) | ||||
|               ? "lg" | ||||
|  | ||||
| @ -31,6 +31,13 @@ export const MasterRespository = { | ||||
| 
 | ||||
|   getActivites: () => api.get("api/master/activities"), | ||||
|   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) => | ||||
|     api.post(`api/master/activity/edit/${id}`, data), | ||||
|   getIndustries: () => api.get("api/master/industries"), | ||||
|  | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user