Compare commits
	
		
			2 Commits
		
	
	
		
			3052aacb45
			...
			f3f8b7fe04
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | f3f8b7fe04 | ||
|  | 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 | ||||||
|  | |||||||
| @ -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 | ||||||
|  | |||||||
| @ -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 | ||||||
|  | |||||||
| @ -6,68 +6,88 @@ | |||||||
|       <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"> |         <div class="flex justify-end items-center mb-6"> | ||||||
|         <button @click="tambahSales" |             <button | ||||||
|  |             @click="tambahSales" | ||||||
|             v-if="isAdmin" |             v-if="isAdmin" | ||||||
|           class="px-4 py-2 bg-C text-D rounded-md hover:bg-C/80 transition duration-200 flex items-center gap-2"> |             class="px-4 py-2 bg-C text-D rounded-md hover:bg-C/80 transition duration-200 flex items-center gap-2" | ||||||
|  |             > | ||||||
|             <svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24"> |             <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" /> |               <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 4v16m8-8H4" /> | ||||||
|             </svg> |             </svg> | ||||||
|             Tambah Sales |             Tambah Sales | ||||||
|         </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-x-auto"> |         <div class="bg-white rounded-lg shadow-md overflow-x-auto"> | ||||||
|           <table class="w-full"> |           <table class="w-full"> | ||||||
|             <thead> |             <thead> | ||||||
|               <tr class="bg-C text-white border-D border"> |               <tr class="bg-C text-white border-D border"> | ||||||
|               <th class="px-6 py-4 text-center border-r border-D text-D"> |                 <th class="px-6 py-4 text-center border-r border-D text-D">No</th> | ||||||
|                 No |                 <th class="px-6 py-4 text-center border-r border-D text-D">Nama Sales</th> | ||||||
|               </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"> |                 <th class="px-6 py-4 text-center border-r border-D text-D">Alamat</th> | ||||||
|                 Nama Sales |                 <th v-if="isAdmin" class="px-6 py-4 text-center text-D">Aksi</th> | ||||||
|               </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> |               </tr> | ||||||
|             </thead> |             </thead> | ||||||
|             <tbody> |             <tbody> | ||||||
|             <tr v-for="(item, index) in sales" :key="item.id" |               <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="border-b border-C border-t-0 border-x hover:bg-gray-50 transition duration-150" | ||||||
|               :class="{ 'bg-gray-50': index % 2 === 1 }"> |                 :class="{ 'bg-gray-50': index % 2 === 1 }" | ||||||
|  |               > | ||||||
|                 <td class="px-6 py-4 border-r border-C text-center font-medium text-gray-900"> |                 <td class="px-6 py-4 border-r border-C text-center font-medium text-gray-900"> | ||||||
|                   {{ index + 1 }} |                   {{ index + 1 }} | ||||||
|                 </td> |                 </td> | ||||||
|               <td class="px-6 py-4 border-r border-C text-D"> |                 <td class="px-6 py-4 border-r border-C text-D">{{ item.nama }}</td> | ||||||
|                 {{ item.nama }} |                 <td class="px-6 py-4 border-r border-C text-gray-800">{{ item.no_hp }}</td> | ||||||
|               </td> |                 <td class="px-6 py-4 border-r border-C text-gray-800">{{ item.alamat }}</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"> |                 <td class="px-6 py-4 text-center" v-if="isAdmin"> | ||||||
|                   <div class="flex justify-center gap-2"> |                   <div class="flex justify-center gap-2"> | ||||||
|                   <button @click="ubahSales(item)" |                     <button | ||||||
|                     class="px-3 py-1 bg-yellow-500 text-white text-sm rounded hover:bg-yellow-600 transition duration-200"> |                       @click="ubahSales(item)" | ||||||
|  |                       class="px-3 py-1 bg-yellow-500 text-white text-sm rounded hover:bg-yellow-600 transition duration-200" | ||||||
|  |                     > | ||||||
|                       Ubah |                       Ubah | ||||||
|                     </button> |                     </button> | ||||||
|                   <button @click="hapusSales(item)" |                     <button | ||||||
|                     class="px-3 py-1 bg-red-500 text-white text-sm rounded hover:bg-red-600 transition duration-200"> |                       @click="hapusSales(item)" | ||||||
|  |                       class="px-3 py-1 bg-red-500 text-white text-sm rounded hover:bg-red-600 transition duration-200" | ||||||
|  |                     > | ||||||
|                       Hapus |                       Hapus | ||||||
|                     </button> |                     </button> | ||||||
|                   </div> |                   </div> | ||||||
| @ -78,9 +98,18 @@ | |||||||
|               <tr v-if="sales.length === 0 && !loading"> |               <tr v-if="sales.length === 0 && !loading"> | ||||||
|                 <td colspan="5" class="px-6 py-8 text-center text-gray-500"> |                 <td colspan="5" class="px-6 py-8 text-center text-gray-500"> | ||||||
|                   <div class="flex flex-col items-center"> |                   <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"> |                     <svg | ||||||
|                     <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" |                       class="w-12 h-12 text-gray-400 mb-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" /> |                       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> |                     </svg> | ||||||
|                     <p>Tidak ada data sales</p> |                     <p>Tidak ada data sales</p> | ||||||
|                   </div> |                   </div> | ||||||
| @ -118,6 +147,18 @@ const editingSales = ref(false); | |||||||
|   const confirmDeleteOpen = ref(false); |   const confirmDeleteOpen = ref(false); | ||||||
|   const salesToDelete = ref(null); |   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); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|   // Fetch data dari API |   // Fetch data dari API | ||||||
|   const fetchSales = async () => { |   const fetchSales = async () => { | ||||||
|     loading.value = true; |     loading.value = true; | ||||||
| @ -130,6 +171,7 @@ const fetchSales = async () => { | |||||||
|       sales.value = response.data; |       sales.value = response.data; | ||||||
|     } catch (error) { |     } catch (error) { | ||||||
|       console.error("Error fetching sales:", error); |       console.error("Error fetching sales:", error); | ||||||
|  |       showAlert("error", "Gagal memuat data sales."); | ||||||
|     } finally { |     } finally { | ||||||
|       loading.value = false; |       loading.value = false; | ||||||
|     } |     } | ||||||
| @ -162,8 +204,10 @@ const confirmDelete = async () => { | |||||||
|       }); |       }); | ||||||
|       fetchSales(); |       fetchSales(); | ||||||
|       confirmDeleteOpen.value = false; |       confirmDeleteOpen.value = false; | ||||||
|  |       showAlert("success", "Sales berhasil dihapus."); | ||||||
|     } catch (error) { |     } catch (error) { | ||||||
|       console.error("Error deleting sales:", error); |       console.error("Error deleting sales:", error); | ||||||
|  |       showAlert("error", "Gagal menghapus sales."); | ||||||
|     } |     } | ||||||
|   }; |   }; | ||||||
| 
 | 
 | ||||||
| @ -176,11 +220,13 @@ const closeDeleteModal = () => { | |||||||
|   const closeSales = () => { |   const closeSales = () => { | ||||||
|     creatingSales.value = false; |     creatingSales.value = false; | ||||||
|     fetchSales(); |     fetchSales(); | ||||||
|  |     showAlert("success", "Sales berhasil ditambahkan."); | ||||||
|   }; |   }; | ||||||
| 
 | 
 | ||||||
|   const closeEditSales = () => { |   const closeEditSales = () => { | ||||||
|     editingSales.value = false; |     editingSales.value = false; | ||||||
|     fetchSales(); |     fetchSales(); | ||||||
|  |     showAlert("success", "Sales berhasil diperbarui."); | ||||||
|   }; |   }; | ||||||
| 
 | 
 | ||||||
|   // Lifecycle |   // Lifecycle | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user