[update] alert respon messae
This commit is contained in:
		
							parent
							
								
									6f7a4df667
								
							
						
					
					
						commit
						e84a4bdadb
					
				| @ -48,7 +48,26 @@ | |||||||
|                     Tambah User |                     Tambah User | ||||||
|                 </button> |                 </button> | ||||||
|             </div> |             </div> | ||||||
| 
 |             <!-- 🔹 Alert Message --> | ||||||
|  |             <div class="px-6" v-if="alert"> | ||||||
|  |             <div | ||||||
|  |                 v-if="alert.error" | ||||||
|  |                 class="text-[#721c24] bg-[#f8d7da] border-l-4 border-[#dc3545] p-3 mb-5 rounded" | ||||||
|  |                 role="alert" | ||||||
|  |             > | ||||||
|  |                 <strong class="font-bold">Error! </strong> | ||||||
|  |                 <span class="block sm:inline">{{ alert.error }}</span> | ||||||
|  |             </div> | ||||||
|  |             <div | ||||||
|  |                 v-if="alert.success" | ||||||
|  |                 class="text-[#155724] bg-[#d4edda] border-l-4 border-[#28a745] p-3 mb-5 rounded" | ||||||
|  |                 role="alert" | ||||||
|  |             > | ||||||
|  |                 <strong class="font-bold">Success! </strong> | ||||||
|  |                 <span class="block sm:inline">{{ alert.success }}</span> | ||||||
|  |             </div> | ||||||
|  |             </div> | ||||||
|  |             <!-- 🔹 End Alert --> | ||||||
|             <!-- Table Section --> |             <!-- Table Section --> | ||||||
|             <div |             <div | ||||||
|                 class="bg-white rounded-lg shadow-md overflow-hidden" |                 class="bg-white rounded-lg shadow-md overflow-hidden" | ||||||
| @ -146,7 +165,16 @@ const detail = ref(null); | |||||||
| const editingAkun = ref(false); | const editingAkun = ref(false); | ||||||
| const confirmDeleteOpen = ref(false); | const confirmDeleteOpen = ref(false); | ||||||
| const akunToDelete = ref(null); | const akunToDelete = ref(null); | ||||||
|  | const alert = ref(null); | ||||||
|  | const timer = ref(null); | ||||||
| 
 | 
 | ||||||
|  | function showAlert(type, message) { | ||||||
|  |     alert.value = { [type]: message }; | ||||||
|  |     clearTimeout(timer.value); | ||||||
|  |     timer.value = setTimeout(() => { | ||||||
|  |       alert.value = null; | ||||||
|  |     }, 5000); | ||||||
|  |   } | ||||||
| // Fetch data dari API | // Fetch data dari API | ||||||
| const fetchAkun = async () => { | const fetchAkun = async () => { | ||||||
|     loading.value = true; |     loading.value = true; | ||||||
| @ -191,6 +219,7 @@ const confirmDelete = async () => { | |||||||
|         }); |         }); | ||||||
|         fetchAkun(); |         fetchAkun(); | ||||||
|         confirmDeleteOpen.value = false; |         confirmDeleteOpen.value = false; | ||||||
|  |         showAlert("success", "User berhasil dihapus."); | ||||||
|     } catch (error) { |     } catch (error) { | ||||||
|         console.error("Error deleting akun:", error); |         console.error("Error deleting akun:", error); | ||||||
|     } |     } | ||||||
| @ -205,11 +234,13 @@ const closeDeleteModal = () => { | |||||||
| const closeAkun = () => { | const closeAkun = () => { | ||||||
|     creatingAkun.value = false; |     creatingAkun.value = false; | ||||||
|     fetchAkun(); |     fetchAkun(); | ||||||
|  |     showAlert("success", "User berhasil ditambahkan."); | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| const closeEditAkun = () => { | const closeEditAkun = () => { | ||||||
|     editingAkun.value = false; |     editingAkun.value = false; | ||||||
|     fetchAkun(); |     fetchAkun(); | ||||||
|  |     showAlert("success", "User berhasil diubah."); | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| // Lifecycle | // Lifecycle | ||||||
|  | |||||||
| @ -288,7 +288,7 @@ const loadProduk = async () => { | |||||||
|     }); |     }); | ||||||
|     const produk = response.data; |     const produk = response.data; | ||||||
|     // console.log(produk); |     // console.log(produk); | ||||||
|      | 
 | ||||||
|     form.value = { |     form.value = { | ||||||
|         nama: produk.nama, |         nama: produk.nama, | ||||||
|         id_kategori: produk.id_kategori, |         id_kategori: produk.id_kategori, | ||||||
| @ -310,7 +310,7 @@ const loadFoto = async () => { | |||||||
|         // console.log(uploadedImages.value); |         // console.log(uploadedImages.value); | ||||||
|     } catch (e) { |     } catch (e) { | ||||||
|         console.error(e); |         console.error(e); | ||||||
|          | 
 | ||||||
|         uploadError.value = "Gagal memuat foto"; |         uploadError.value = "Gagal memuat foto"; | ||||||
|     } |     } | ||||||
| }; | }; | ||||||
| @ -415,7 +415,7 @@ const submitForm = async () => { | |||||||
|                 }, |                 }, | ||||||
|             } |             } | ||||||
|         ); |         ); | ||||||
|         router.push("/produk"); |         router.push("/produk?message=Produk berhasil diperbarui"); | ||||||
|     } catch (err) { |     } catch (err) { | ||||||
|         errorMessages.value = err.response?.data?.message || "Gagal menyimpan produk"; |         errorMessages.value = err.response?.data?.message || "Gagal menyimpan produk"; | ||||||
|         console.error(err); |         console.error(err); | ||||||
|  | |||||||
| @ -174,7 +174,7 @@ import CreateItemModal from "../components/CreateItemModal.vue"; | |||||||
| const router = useRouter(); | const router = useRouter(); | ||||||
| 
 | 
 | ||||||
| const form = ref({ | const form = ref({ | ||||||
|   nama: '', id_kategori: null, berat: 0, kadar: 0, harga_per_gram: 0, harga_jual: 0, |   nama: '', id_kategori: null, berat: null, kadar: null, harga_per_gram: null, harga_jual: null, | ||||||
| }); | }); | ||||||
| const category = ref([]); | const category = ref([]); | ||||||
| const showUploadMenu = ref(false); | const showUploadMenu = ref(false); | ||||||
|  | |||||||
| @ -16,7 +16,26 @@ | |||||||
| 					Tambah Kategori | 					Tambah Kategori | ||||||
| 				</button> | 				</button> | ||||||
| 			</div> | 			</div> | ||||||
| 
 |                     <!-- 🔹 Alert Message --> | ||||||
|  |             <div class="px-6" v-if="alert"> | ||||||
|  |             <div | ||||||
|  |                 v-if="alert.error" | ||||||
|  |                 class="text-[#721c24] bg-[#f8d7da] border-l-4 border-[#dc3545] p-3 mb-5 rounded" | ||||||
|  |                 role="alert" | ||||||
|  |             > | ||||||
|  |                 <strong class="font-bold">Error! </strong> | ||||||
|  |                 <span class="block sm:inline">{{ alert.error }}</span> | ||||||
|  |             </div> | ||||||
|  |             <div | ||||||
|  |                 v-if="alert.success" | ||||||
|  |                 class="text-[#155724] bg-[#d4edda] border-l-4 border-[#28a745] p-3 mb-5 rounded" | ||||||
|  |                 role="alert" | ||||||
|  |             > | ||||||
|  |                 <strong class="font-bold">Success! </strong> | ||||||
|  |                 <span class="block sm:inline">{{ alert.success }}</span> | ||||||
|  |             </div> | ||||||
|  |             </div> | ||||||
|  |             <!-- 🔹 End Alert --> | ||||||
| 			<!-- Table Section --> | 			<!-- Table Section --> | ||||||
| 			<div class="bg-white rounded-lg shadow-md overflow-hidden"> | 			<div class="bg-white rounded-lg shadow-md overflow-hidden"> | ||||||
| 				<table class="w-full"> | 				<table class="w-full"> | ||||||
| @ -96,9 +115,18 @@ const creatingKategori = ref(false); | |||||||
| const detail = ref(null); | const detail = ref(null); | ||||||
| const confirmDeleteOpen = ref(false); | const confirmDeleteOpen = ref(false); | ||||||
| const kategoriToDelete = ref(null); | const kategoriToDelete = ref(null); | ||||||
| 
 | const alert = ref(null); | ||||||
|  | const timer = ref(null); | ||||||
| const isAdmin = localStorage.getItem("role") === "owner"; | const isAdmin = localStorage.getItem("role") === "owner"; | ||||||
| 
 | 
 | ||||||
|  | function showAlert(type, message) { | ||||||
|  |     alert.value = { [type]: message }; | ||||||
|  |     clearTimeout(timer.value); | ||||||
|  |     timer.value = setTimeout(() => { | ||||||
|  |       alert.value = null; | ||||||
|  |     }, 5000); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
| // Fetch data kategori dari API | // Fetch data kategori dari API | ||||||
| const fetchKategoris = async () => { | const fetchKategoris = async () => { | ||||||
| 	loading.value = true; | 	loading.value = true; | ||||||
| @ -109,7 +137,6 @@ const fetchKategoris = async () => { | |||||||
| 			}, | 			}, | ||||||
| 		}); | 		}); | ||||||
| 		kategori.value = response.data; | 		kategori.value = response.data; | ||||||
| 		// console.log("Data kategori:", response.data); |  | ||||||
| 	} catch (error) { | 	} catch (error) { | ||||||
| 		console.error("Error fetching kategori:", error); | 		console.error("Error fetching kategori:", error); | ||||||
| 	} finally { | 	} finally { | ||||||
| @ -119,14 +146,19 @@ const fetchKategoris = async () => { | |||||||
| 
 | 
 | ||||||
| // Tambah kategori - open modal | // Tambah kategori - open modal | ||||||
| const tambahKategori = () => { | const tambahKategori = () => { | ||||||
| 	detail.value = null; // Reset detail untuk mode create | 	detail.value = null; | ||||||
| 	creatingKategori.value = true; | 	creatingKategori.value = true; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| // Close modal | // Close modal | ||||||
| const closeKategori = () => { | const closeKategori = () => { | ||||||
| 	creatingKategori.value = false; | 	creatingKategori.value = false; | ||||||
| 	fetchKategoris(); // Refresh data setelah modal ditutup | 	fetchKategoris(); | ||||||
|  |     if (detail.value==null) { | ||||||
|  | 
 | ||||||
|  |         showAlert("success", "Kategori berhasil ditambahkan"); | ||||||
|  |     } else | ||||||
|  |     showAlert("success", "Kategori berhasil diubah"); | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| // Ubah kategori | // Ubah kategori | ||||||
| @ -139,6 +171,7 @@ const ubahKategori = (item) => { | |||||||
| const hapusKategori = (item) => { | const hapusKategori = (item) => { | ||||||
| 	kategoriToDelete.value = item; | 	kategoriToDelete.value = item; | ||||||
| 	confirmDeleteOpen.value = true; | 	confirmDeleteOpen.value = true; | ||||||
|  |     showAlert("success", "Kategori berhasil dihapus"); | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| // 🔵 Ditambahkan: aksi konfirmasi hapus | // 🔵 Ditambahkan: aksi konfirmasi hapus | ||||||
|  | |||||||
| @ -1,11 +1,11 @@ | |||||||
| <template> | <template> | ||||||
|   <mainLayout> |   <mainLayout> | ||||||
|     <!-- Modal Buat Item --> |     <!-- Modal Buat Item --> | ||||||
|     <CreateItemModal  |     <CreateItemModal | ||||||
|   :isOpen="creatingItem"  |   :isOpen="creatingItem" | ||||||
|   :product="detail"  |   :product="detail" | ||||||
|   @close="closeItemModal"  |   @close="closeItemModal" | ||||||
|   @itemAdded="handleItemAdded"  |   @itemAdded="handleItemAdded" | ||||||
| /> | /> | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| @ -51,6 +51,27 @@ | |||||||
|         </div> |         </div> | ||||||
|       </div> |       </div> | ||||||
| 
 | 
 | ||||||
|  |     <!-- 🔹 Alert Message --> | ||||||
|  |     <div class="my-5" v-if="alert"> | ||||||
|  |       <div | ||||||
|  |         v-if="alert.error" | ||||||
|  |         class="text-[#721c24] bg-[#f8d7da] border-l-4 border-[#dc3545] p-3 mb-5 rounded" | ||||||
|  |         role="alert" | ||||||
|  |       > | ||||||
|  |         <strong class="font-bold">Error! </strong> | ||||||
|  |         <span class="block sm:inline">{{ alert.error }}</span> | ||||||
|  |       </div> | ||||||
|  |       <div | ||||||
|  |         v-if="alert.success" | ||||||
|  |         class="text-[#155724] bg-[#d4edda] border-l-4 border-[#28a745] p-3 mb-5 rounded" | ||||||
|  |         role="alert" | ||||||
|  |       > | ||||||
|  |         <strong class="font-bold">Success! </strong> | ||||||
|  |         <span class="block sm:inline">{{ alert.success }}</span> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |     <!-- 🔹 End Alert --> | ||||||
|  | 
 | ||||||
|       <!-- 🔵 Loading State (sama persis dengan kategori) --> |       <!-- 🔵 Loading State (sama persis dengan kategori) --> | ||||||
|       <div v-if="loading" class="flex justify-center items-center h-screen"> |       <div v-if="loading" class="flex justify-center items-center h-screen"> | ||||||
|         <div class="animate-spin rounded-full h-8 w-8 border-b-2 border-C"></div> |         <div class="animate-spin rounded-full h-8 w-8 border-b-2 border-C"></div> | ||||||
| @ -159,13 +180,21 @@ const searchQuery = ref(""); | |||||||
| const selectedCategory = ref(0); | const selectedCategory = ref(0); | ||||||
| const creatingItem = ref(false); | const creatingItem = ref(false); | ||||||
| const deleting = ref(false); | const deleting = ref(false); | ||||||
| 
 | const alert = ref(null); | ||||||
|  | const timer = ref(null); | ||||||
| const detail = ref({}); | const detail = ref({}); | ||||||
| const showOverlay = ref(false); | const showOverlay = ref(false); | ||||||
| const currentFotoIndex = ref(0); | const currentFotoIndex = ref(0); | ||||||
| 
 |  | ||||||
| const kategori = ref([]); | const kategori = ref([]); | ||||||
| const loading = ref(false); // 🔥 Loading persis kategori | const loading = ref(false); | ||||||
|  | 
 | ||||||
|  | function showAlert(type, message) { | ||||||
|  |     alert.value = { [type]: message }; | ||||||
|  |     clearTimeout(timer.value); | ||||||
|  |     timer.value = setTimeout(() => { | ||||||
|  |       alert.value = null; | ||||||
|  |     }, 5000); | ||||||
|  |   } | ||||||
| 
 | 
 | ||||||
| // Load kategori | // Load kategori | ||||||
| const loadKategori = async () => { | const loadKategori = async () => { | ||||||
| @ -221,6 +250,18 @@ const closeItemModal = () => { | |||||||
| onMounted(async () => { | onMounted(async () => { | ||||||
|   await loadKategori(); |   await loadKategori(); | ||||||
|   await loadProduk(); |   await loadProduk(); | ||||||
|  | 
 | ||||||
|  |   // 🔹 Cek apakah ada ?message= di URL | ||||||
|  |   const params = new URLSearchParams(window.location.search); | ||||||
|  |   const message = params.get("message"); | ||||||
|  | 
 | ||||||
|  |   if (message) { | ||||||
|  |     showAlert("success", message); | ||||||
|  | 
 | ||||||
|  |     // 🔹 Hapus query param biar tidak muncul lagi pas refresh | ||||||
|  |     const newUrl = window.location.pathname; | ||||||
|  |     window.history.replaceState({}, document.title, newUrl); | ||||||
|  |   } | ||||||
| }); | }); | ||||||
| 
 | 
 | ||||||
| // Filter produk | // Filter produk | ||||||
|  | |||||||
| @ -1,190 +1,236 @@ | |||||||
| <template> | <template> | ||||||
|   <mainLayout> |     <mainLayout> | ||||||
|     <!-- Modal Create/Edit Sales --> |       <!-- Modal Create/Edit Sales --> | ||||||
|     <CreateSales v-if="creatingSales" :isOpen="creatingSales" :sales="detail" @close="closeSales" /> |       <CreateSales v-if="creatingSales" :isOpen="creatingSales" :sales="detail" @close="closeSales" /> | ||||||
| 
 | 
 | ||||||
|     <EditSales v-if="editingSales" :isOpen="editingSales" :sales="detail" @close="closeEditSales" /> |       <EditSales v-if="editingSales" :isOpen="editingSales" :sales="detail" @close="closeEditSales" /> | ||||||
| 
 | 
 | ||||||
|     <!-- Modal Delete --> |       <!-- Modal Delete --> | ||||||
|     <ConfirmDeleteModal :isOpen="confirmDeleteOpen" title="Hapus Sales" |       <ConfirmDeleteModal | ||||||
|       message="Apakah Anda yakin ingin menghapus sales ini?" @confirm="confirmDelete" @cancel="closeDeleteModal" /> |         :isOpen="confirmDeleteOpen" | ||||||
|  |         title="Hapus Sales" | ||||||
|  |         message="Apakah Anda yakin ingin menghapus sales ini?" | ||||||
|  |         @confirm="confirmDelete" | ||||||
|  |         @cancel="closeDeleteModal" | ||||||
|  |       /> | ||||||
| 
 | 
 | ||||||
|     <div class="p-6 min-h-[75vh]"> |       <div class="p-6 min-h-[75vh]"> | ||||||
|       <p class="font-serif italic text-[25px] text-D">SALES</p> |         <p class="font-serif italic text-[25px] text-D">SALES</p> | ||||||
|       <div class="flex justify-end items-center mb-6"> | 
 | ||||||
|         <button @click="tambahSales" | 
 | ||||||
|           v-if="isAdmin" |         <div class="flex justify-end items-center mb-6"> | ||||||
|           class="px-4 py-2 bg-C text-D rounded-md hover:bg-C/80 transition duration-200 flex items-center gap-2"> |             <button | ||||||
|           <svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24"> |             @click="tambahSales" | ||||||
|             <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 4v16m8-8H4" /> |             v-if="isAdmin" | ||||||
|           </svg> |             class="px-4 py-2 bg-C text-D rounded-md hover:bg-C/80 transition duration-200 flex items-center gap-2" | ||||||
|           Tambah Sales |             > | ||||||
|  |             <svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24"> | ||||||
|  |               <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 4v16m8-8H4" /> | ||||||
|  |             </svg> | ||||||
|  |             Tambah Sales | ||||||
|         </button> |         </button> | ||||||
|  |     </div> | ||||||
|  |     <!-- 🔹 Alert Message --> | ||||||
|  |     <div class="px-6" v-if="alert"> | ||||||
|  |       <div | ||||||
|  |         v-if="alert.error" | ||||||
|  |         class="text-[#721c24] bg-[#f8d7da] border-l-4 border-[#dc3545] p-3 mb-5 rounded" | ||||||
|  |         role="alert" | ||||||
|  |       > | ||||||
|  |         <strong class="font-bold">Error! </strong> | ||||||
|  |         <span class="block sm:inline">{{ alert.error }}</span> | ||||||
|       </div> |       </div> | ||||||
| 
 |       <div | ||||||
|       <!-- Table Section --> |         v-if="alert.success" | ||||||
|       <div class="bg-white rounded-lg shadow-md  overflow-x-auto"> |         class="text-[#155724] bg-[#d4edda] border-l-4 border-[#28a745] p-3 mb-5 rounded" | ||||||
|         <table class="w-full "> |         role="alert" | ||||||
|           <thead> |       > | ||||||
|             <tr class="bg-C text-white border-D border"> |         <strong class="font-bold">Success! </strong> | ||||||
|               <th class="px-6 py-4 text-center border-r border-D text-D"> |         <span class="block sm:inline">{{ alert.success }}</span> | ||||||
|                 No |  | ||||||
|               </th> |  | ||||||
|               <th class="px-6 py-4 text-center border-r border-D text-D"> |  | ||||||
|                 Nama Sales |  | ||||||
|               </th> |  | ||||||
|               <th class="px-6 py-4 text-center border-r border-D text-D"> |  | ||||||
|                 No HP |  | ||||||
|               </th> |  | ||||||
|               <th class="px-6 py-4 text-center border-r border-D text-D"> |  | ||||||
|                 Alamat |  | ||||||
|               </th> |  | ||||||
|               <th v-if="isAdmin" class="px-6 py-4 text-center text-D"> |  | ||||||
|                 Aksi |  | ||||||
|               </th> |  | ||||||
|             </tr> |  | ||||||
|           </thead> |  | ||||||
|           <tbody> |  | ||||||
|             <tr v-for="(item, index) in sales" :key="item.id" |  | ||||||
|               class="border-b border-C border-t-0 border-x hover:bg-gray-50 transition duration-150" |  | ||||||
|               :class="{ 'bg-gray-50': index % 2 === 1 }"> |  | ||||||
|               <td class="px-6 py-4 border-r border-C text-center font-medium text-gray-900"> |  | ||||||
|                 {{ index + 1 }} |  | ||||||
|               </td> |  | ||||||
|               <td class="px-6 py-4 border-r border-C text-D"> |  | ||||||
|                 {{ item.nama }} |  | ||||||
|               </td> |  | ||||||
|               <td class="px-6 py-4 border-r border-C text-gray-800"> |  | ||||||
|                 {{ item.no_hp }} |  | ||||||
|               </td> |  | ||||||
|               <td class="px-6 py-4 border-r border-C text-gray-800"> |  | ||||||
|                 {{ item.alamat }} |  | ||||||
|               </td> |  | ||||||
|               <td class="px-6 py-4 text-center" v-if="isAdmin"> |  | ||||||
|                 <div class="flex justify-center gap-2"> |  | ||||||
|                   <button @click="ubahSales(item)" |  | ||||||
|                     class="px-3 py-1 bg-yellow-500 text-white text-sm rounded hover:bg-yellow-600 transition duration-200"> |  | ||||||
|                     Ubah |  | ||||||
|                   </button> |  | ||||||
|                   <button @click="hapusSales(item)" |  | ||||||
|                     class="px-3 py-1 bg-red-500 text-white text-sm rounded hover:bg-red-600 transition duration-200"> |  | ||||||
|                     Hapus |  | ||||||
|                   </button> |  | ||||||
|                 </div> |  | ||||||
|               </td> |  | ||||||
|             </tr> |  | ||||||
| 
 |  | ||||||
|             <!-- Empty State --> |  | ||||||
|             <tr v-if="sales.length === 0 && !loading"> |  | ||||||
|               <td colspan="5" class="px-6 py-8 text-center text-gray-500"> |  | ||||||
|                 <div class="flex flex-col items-center"> |  | ||||||
|                   <svg class="w-12 h-12 text-gray-400 mb-2" fill="none" stroke="currentColor" viewBox="0 0 24 24"> |  | ||||||
|                     <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" |  | ||||||
|                       d="M20 13V6a2 2 0 00-2-2H6a2 2 0 00-2 2v7m16 0v5a2 2 0 01-2 2H6a2 2 0 01-2 2v-5m16 0h-2M4 13h2" /> |  | ||||||
|                   </svg> |  | ||||||
|                   <p>Tidak ada data sales</p> |  | ||||||
|                 </div> |  | ||||||
|               </td> |  | ||||||
|             </tr> |  | ||||||
|           </tbody> |  | ||||||
|         </table> |  | ||||||
|       </div> |  | ||||||
| 
 |  | ||||||
|       <!-- Loading State --> |  | ||||||
|       <div v-if="loading" class="flex justify-center items-center py-8"> |  | ||||||
|         <div class="animate-spin rounded-full h-8 w-8 border-b-2 border-[#c6a77d]"></div> |  | ||||||
|         <span class="ml-2 text-gray-600">Memuat data...</span> |  | ||||||
|       </div> |       </div> | ||||||
|     </div> |     </div> | ||||||
|   </mainLayout> |     <!-- 🔹 End Alert --> | ||||||
| </template> |  | ||||||
| 
 | 
 | ||||||
| <script setup> |         <!-- Table Section --> | ||||||
| import { ref, onMounted } from "vue"; |         <div class="bg-white rounded-lg shadow-md overflow-x-auto"> | ||||||
| import axios from "axios"; |           <table class="w-full"> | ||||||
| import mainLayout from "../layouts/mainLayout.vue"; |             <thead> | ||||||
| import CreateSales from "../components/CreateSales.vue"; |               <tr class="bg-C text-white border-D border"> | ||||||
| import ConfirmDeleteModal from "../components/ConfirmDeleteModal.vue"; |                 <th class="px-6 py-4 text-center border-r border-D text-D">No</th> | ||||||
| import EditSales from "../components/EditSales.vue"; |                 <th class="px-6 py-4 text-center border-r border-D text-D">Nama Sales</th> | ||||||
|  |                 <th class="px-6 py-4 text-center border-r border-D text-D">No HP</th> | ||||||
|  |                 <th class="px-6 py-4 text-center border-r border-D text-D">Alamat</th> | ||||||
|  |                 <th v-if="isAdmin" class="px-6 py-4 text-center text-D">Aksi</th> | ||||||
|  |               </tr> | ||||||
|  |             </thead> | ||||||
|  |             <tbody> | ||||||
|  |               <tr | ||||||
|  |                 v-for="(item, index) in sales" | ||||||
|  |                 :key="item.id" | ||||||
|  |                 class="border-b border-C border-t-0 border-x hover:bg-gray-50 transition duration-150" | ||||||
|  |                 :class="{ 'bg-gray-50': index % 2 === 1 }" | ||||||
|  |               > | ||||||
|  |                 <td class="px-6 py-4 border-r border-C text-center font-medium text-gray-900"> | ||||||
|  |                   {{ index + 1 }} | ||||||
|  |                 </td> | ||||||
|  |                 <td class="px-6 py-4 border-r border-C text-D">{{ item.nama }}</td> | ||||||
|  |                 <td class="px-6 py-4 border-r border-C text-gray-800">{{ item.no_hp }}</td> | ||||||
|  |                 <td class="px-6 py-4 border-r border-C text-gray-800">{{ item.alamat }}</td> | ||||||
|  |                 <td class="px-6 py-4 text-center" v-if="isAdmin"> | ||||||
|  |                   <div class="flex justify-center gap-2"> | ||||||
|  |                     <button | ||||||
|  |                       @click="ubahSales(item)" | ||||||
|  |                       class="px-3 py-1 bg-yellow-500 text-white text-sm rounded hover:bg-yellow-600 transition duration-200" | ||||||
|  |                     > | ||||||
|  |                       Ubah | ||||||
|  |                     </button> | ||||||
|  |                     <button | ||||||
|  |                       @click="hapusSales(item)" | ||||||
|  |                       class="px-3 py-1 bg-red-500 text-white text-sm rounded hover:bg-red-600 transition duration-200" | ||||||
|  |                     > | ||||||
|  |                       Hapus | ||||||
|  |                     </button> | ||||||
|  |                   </div> | ||||||
|  |                 </td> | ||||||
|  |               </tr> | ||||||
| 
 | 
 | ||||||
| const isAdmin = localStorage.getItem("role") === "owner"; |               <!-- Empty State --> | ||||||
|  |               <tr v-if="sales.length === 0 && !loading"> | ||||||
|  |                 <td colspan="5" class="px-6 py-8 text-center text-gray-500"> | ||||||
|  |                   <div class="flex flex-col items-center"> | ||||||
|  |                     <svg | ||||||
|  |                       class="w-12 h-12 text-gray-400 mb-2" | ||||||
|  |                       fill="none" | ||||||
|  |                       stroke="currentColor" | ||||||
|  |                       viewBox="0 0 24 24" | ||||||
|  |                     > | ||||||
|  |                       <path | ||||||
|  |                         stroke-linecap="round" | ||||||
|  |                         stroke-linejoin="round" | ||||||
|  |                         stroke-width="2" | ||||||
|  |                         d="M20 13V6a2 2 0 00-2-2H6a2 2 0 00-2 2v7m16 0v5a2 2 0 01-2 2H6a2 2 0 01-2 2v-5m16 0h-2M4 13h2" | ||||||
|  |                       /> | ||||||
|  |                     </svg> | ||||||
|  |                     <p>Tidak ada data sales</p> | ||||||
|  |                   </div> | ||||||
|  |                 </td> | ||||||
|  |               </tr> | ||||||
|  |             </tbody> | ||||||
|  |           </table> | ||||||
|  |         </div> | ||||||
| 
 | 
 | ||||||
| // State |         <!-- Loading State --> | ||||||
| const sales = ref([]); |         <div v-if="loading" class="flex justify-center items-center py-8"> | ||||||
| const loading = ref(false); |           <div class="animate-spin rounded-full h-8 w-8 border-b-2 border-[#c6a77d]"></div> | ||||||
| const creatingSales = ref(false); |           <span class="ml-2 text-gray-600">Memuat data...</span> | ||||||
| const detail = ref(null); |         </div> | ||||||
| const editingSales = ref(false); |       </div> | ||||||
| const confirmDeleteOpen = ref(false); |     </mainLayout> | ||||||
| const salesToDelete = ref(null); |   </template> | ||||||
| 
 | 
 | ||||||
| // Fetch data dari API |   <script setup> | ||||||
| const fetchSales = async () => { |   import { ref, onMounted } from "vue"; | ||||||
|   loading.value = true; |   import axios from "axios"; | ||||||
|   try { |   import mainLayout from "../layouts/mainLayout.vue"; | ||||||
|     const response = await axios.get("/api/sales", { |   import CreateSales from "../components/CreateSales.vue"; | ||||||
|       headers: { |   import ConfirmDeleteModal from "../components/ConfirmDeleteModal.vue"; | ||||||
|         Authorization: `Bearer ${localStorage.getItem("token")}`, |   import EditSales from "../components/EditSales.vue"; | ||||||
|       }, | 
 | ||||||
|     }); |   const isAdmin = localStorage.getItem("role") === "owner"; | ||||||
|     sales.value = response.data; | 
 | ||||||
|   } catch (error) { |   // State | ||||||
|     console.error("Error fetching sales:", error); |   const sales = ref([]); | ||||||
|   } finally { |   const loading = ref(false); | ||||||
|     loading.value = false; |   const creatingSales = ref(false); | ||||||
|  |   const detail = ref(null); | ||||||
|  |   const editingSales = ref(false); | ||||||
|  |   const confirmDeleteOpen = ref(false); | ||||||
|  |   const salesToDelete = ref(null); | ||||||
|  | 
 | ||||||
|  |   // 🔹 Alert State | ||||||
|  |   const alert = ref(null); | ||||||
|  |   const timer = ref(null); | ||||||
|  | 
 | ||||||
|  |   function showAlert(type, message) { | ||||||
|  |     alert.value = { [type]: message }; | ||||||
|  |     clearTimeout(timer.value); | ||||||
|  |     timer.value = setTimeout(() => { | ||||||
|  |       alert.value = null; | ||||||
|  |     }, 5000); | ||||||
|   } |   } | ||||||
| }; |  | ||||||
| 
 | 
 | ||||||
| // Tambah |   // Fetch data dari API | ||||||
| const tambahSales = () => { |   const fetchSales = async () => { | ||||||
|   detail.value = null; |     loading.value = true; | ||||||
|   creatingSales.value = true; |     try { | ||||||
| }; |       const response = await axios.get("/api/sales", { | ||||||
|  |         headers: { | ||||||
|  |           Authorization: `Bearer ${localStorage.getItem("token")}`, | ||||||
|  |         }, | ||||||
|  |       }); | ||||||
|  |       sales.value = response.data; | ||||||
|  |     } catch (error) { | ||||||
|  |       console.error("Error fetching sales:", error); | ||||||
|  |       showAlert("error", "Gagal memuat data sales."); | ||||||
|  |     } finally { | ||||||
|  |       loading.value = false; | ||||||
|  |     } | ||||||
|  |   }; | ||||||
| 
 | 
 | ||||||
| // Ubah |   // Tambah | ||||||
| const ubahSales = (item) => { |   const tambahSales = () => { | ||||||
|   detail.value = item; |     detail.value = null; | ||||||
|   editingSales.value = true; |     creatingSales.value = true; | ||||||
| }; |   }; | ||||||
| 
 | 
 | ||||||
| // Hapus |   // Ubah | ||||||
| const hapusSales = (item) => { |   const ubahSales = (item) => { | ||||||
|   salesToDelete.value = item; |     detail.value = item; | ||||||
|   confirmDeleteOpen.value = true; |     editingSales.value = true; | ||||||
| }; |   }; | ||||||
| 
 | 
 | ||||||
| const confirmDelete = async () => { |   // Hapus | ||||||
|   try { |   const hapusSales = (item) => { | ||||||
|     await axios.delete(`/api/sales/${salesToDelete.value.id}`, { |     salesToDelete.value = item; | ||||||
|       headers: { |     confirmDeleteOpen.value = true; | ||||||
|         Authorization: `Bearer ${localStorage.getItem("token")}`, |   }; | ||||||
|       }, | 
 | ||||||
|     }); |   const confirmDelete = async () => { | ||||||
|     fetchSales(); |     try { | ||||||
|  |       await axios.delete(`/api/sales/${salesToDelete.value.id}`, { | ||||||
|  |         headers: { | ||||||
|  |           Authorization: `Bearer ${localStorage.getItem("token")}`, | ||||||
|  |         }, | ||||||
|  |       }); | ||||||
|  |       fetchSales(); | ||||||
|  |       confirmDeleteOpen.value = false; | ||||||
|  |       showAlert("success", "Sales berhasil dihapus."); | ||||||
|  |     } catch (error) { | ||||||
|  |       console.error("Error deleting sales:", error); | ||||||
|  |       showAlert("error", "Gagal menghapus sales."); | ||||||
|  |     } | ||||||
|  |   }; | ||||||
|  | 
 | ||||||
|  |   const closeDeleteModal = () => { | ||||||
|     confirmDeleteOpen.value = false; |     confirmDeleteOpen.value = false; | ||||||
|   } catch (error) { |     salesToDelete.value = null; | ||||||
|     console.error("Error deleting sales:", error); |   }; | ||||||
|   } |  | ||||||
| }; |  | ||||||
| 
 | 
 | ||||||
| const closeDeleteModal = () => { |   // Tutup modal Create/Edit | ||||||
|   confirmDeleteOpen.value = false; |   const closeSales = () => { | ||||||
|   salesToDelete.value = null; |     creatingSales.value = false; | ||||||
| }; |     fetchSales(); | ||||||
|  |     showAlert("success", "Sales berhasil ditambahkan."); | ||||||
|  |   }; | ||||||
| 
 | 
 | ||||||
| // Tutup modal Create/Edit |   const closeEditSales = () => { | ||||||
| const closeSales = () => { |     editingSales.value = false; | ||||||
|   creatingSales.value = false; |     fetchSales(); | ||||||
|   fetchSales(); |     showAlert("success", "Sales berhasil diperbarui."); | ||||||
| }; |   }; | ||||||
| 
 | 
 | ||||||
| const closeEditSales = () => { |   // Lifecycle | ||||||
|   editingSales.value = false; |   onMounted(() => { | ||||||
|   fetchSales(); |     fetchSales(); | ||||||
| }; |   }); | ||||||
| 
 |   </script> | ||||||
| // Lifecycle |  | ||||||
| onMounted(() => { |  | ||||||
|   fetchSales(); |  | ||||||
| }); |  | ||||||
| </script> |  | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user