Merge branch 'production' of https://git.abbauf.com/Magang-2025/Kasir into production
This commit is contained in:
		
						commit
						773cc1516f
					
				| @ -11,7 +11,7 @@ class FotoSementaraController extends Controller | |||||||
|     public function upload(Request $request) |     public function upload(Request $request) | ||||||
|     { |     { | ||||||
|         $request->validate([ |         $request->validate([ | ||||||
|             'id_produk' => 'required|exists:produk,id', |             'id_user' => 'required|exists:users,id', | ||||||
|             'foto'      => 'required|image|mimes:jpg,jpeg,png|max:2048', |             'foto'      => 'required|image|mimes:jpg,jpeg,png|max:2048', | ||||||
|         ]); |         ]); | ||||||
| 
 | 
 | ||||||
| @ -19,11 +19,11 @@ class FotoSementaraController extends Controller | |||||||
|         $url = asset('storage/' . $path); |         $url = asset('storage/' . $path); | ||||||
| 
 | 
 | ||||||
|         $foto = FotoSementara::create([ |         $foto = FotoSementara::create([ | ||||||
|             'id_produk' => $request->id_produk, |             'id_user' => $request->id_user, | ||||||
|             'url'       => $url, |             'url'       => $url, | ||||||
|         ]); |         ]); | ||||||
| 
 | 
 | ||||||
|         return response()->json(['message' => 'Foto berhasil disimpan'], 201); |         return response()->json($foto, 201); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public function hapus($id) |     public function hapus($id) | ||||||
| @ -45,6 +45,9 @@ class FotoSementaraController extends Controller | |||||||
|     public function getAll($user_id) |     public function getAll($user_id) | ||||||
|     { |     { | ||||||
|         $data = FotoSementara::where('id_user', $user_id); |         $data = FotoSementara::where('id_user', $user_id); | ||||||
|  |         if (!$data->exists()) { | ||||||
|  |             return response()->json(['message' => 'Tidak ada foto ditemukan'], 404); | ||||||
|  |         } | ||||||
|         return response()->json($data); |         return response()->json($data); | ||||||
|     } |     } | ||||||
|      |      | ||||||
|  | |||||||
| @ -13,7 +13,7 @@ class NampanController extends Controller | |||||||
|     public function index() |     public function index() | ||||||
|     { |     { | ||||||
|         return response()->json( |         return response()->json( | ||||||
|             Nampan::withCount('items')->get() |             Nampan::with('items.produk.foto')->withCount('items')->get() | ||||||
|         ); |         ); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| @ -43,7 +43,7 @@ class NampanController extends Controller | |||||||
|     public function show(int $id) |     public function show(int $id) | ||||||
|     { |     { | ||||||
|         return response()->json( |         return response()->json( | ||||||
|             Nampan::with('items')->find($id) |             Nampan::with('items.produk.foto')->find($id) | ||||||
|         ); |         ); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -13,14 +13,20 @@ class TransaksiController extends Controller | |||||||
|     // List semua transaksi
 |     // List semua transaksi
 | ||||||
|     public function index() |     public function index() | ||||||
|     { |     { | ||||||
|         $transaksi = Transaksi::with(['kasir', 'sales', 'items.item.produk'])->get(); |         $limit = request()->query('limit', null); | ||||||
|  |         $query = Transaksi::with(['kasir', 'sales', 'items.item.produk'])->latest(); | ||||||
|  |         if ($limit) { | ||||||
|  |             $query->limit((int)$limit); | ||||||
|  |         } | ||||||
|  |         $transaksi = $query->get(); | ||||||
|  |         $transaksi = Transaksi::with(['kasir', 'sales', 'items.item.produk'])->latest()->limit(100)->get(); | ||||||
|         return response()->json($transaksi); |         return response()->json($transaksi); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     // Detail transaksi by ID
 |     // Detail transaksi by ID
 | ||||||
|     public function show($id) |     public function show($id) | ||||||
|     { |     { | ||||||
|         $transaksi = Transaksi::with(['kasir', 'sales', 'items.item.produk'])->findOrFail($id); |         $transaksi = Transaksi::with(['kasir', 'sales', 'items.item.produk.foto'])->findOrFail($id); | ||||||
|         return response()->json($transaksi); |         return response()->json($transaksi); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -13,9 +13,18 @@ class Nampan extends Model | |||||||
|     protected $fillable = [ |     protected $fillable = [ | ||||||
|         'nama' |         'nama' | ||||||
|     ]; |     ]; | ||||||
|  |     protected $appends = ['berat_total']; | ||||||
|  | 
 | ||||||
| 
 | 
 | ||||||
|     public function items() |     public function items() | ||||||
|     { |     { | ||||||
|         return $this->hasMany(Item::class, 'id_nampan'); |         return $this->hasMany(Item::class, 'id_nampan'); | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|  |     public function getBeratTotalAttribute() | ||||||
|  |     { | ||||||
|  |         return $this->items() | ||||||
|  |             ->join('produks', 'items.id_produk', '=', 'produks.id') | ||||||
|  |             ->sum('produks.berat'); | ||||||
|  |     } | ||||||
| } | } | ||||||
|  | |||||||
							
								
								
									
										28
									
								
								resources/js/components/InputField.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								resources/js/components/InputField.vue
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,28 @@ | |||||||
|  | <template> | ||||||
|  |     <input | ||||||
|  |       :type="type" | ||||||
|  |       :value="modelValue" | ||||||
|  |       @input="$emit('update:modelValue', $event.target.value)" | ||||||
|  |       :placeholder="placeholder" | ||||||
|  |       class="mt-1 block w-full rounded-md shadow-sm sm:text-sm bg-A text-D border-B focus:border-C focus:ring focus:ring-D focus:ring-opacity-50 p-2" | ||||||
|  |     /> | ||||||
|  | </template> | ||||||
|  | 
 | ||||||
|  | <script setup> | ||||||
|  | const props = defineProps({ | ||||||
|  |   type: { | ||||||
|  |     type: String, | ||||||
|  |     default: 'text', | ||||||
|  |   }, | ||||||
|  |   modelValue: { | ||||||
|  |     type: [String, Number], | ||||||
|  |     default: '', | ||||||
|  |   }, | ||||||
|  |   placeholder: { | ||||||
|  |     type: String, | ||||||
|  |     default: '', | ||||||
|  |   }, | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | const emit = defineEmits(['update:modelValue']); | ||||||
|  | </script> | ||||||
							
								
								
									
										31
									
								
								resources/js/components/InputSelect.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								resources/js/components/InputSelect.vue
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,31 @@ | |||||||
|  | <template> | ||||||
|  |     <select | ||||||
|  |       :value="modelValue" | ||||||
|  |       @change="$emit('update:modelValue', $event.target.value)" | ||||||
|  |       class="mt-1 block w-full rounded-md shadow-sm sm:text-sm bg-A text-D border-B focus:border-C focus:ring focus:ring-D focus:ring-opacity-50 p-2" | ||||||
|  |     > | ||||||
|  |       <option value="" :disabled="!modelValue && placeholder" v-if="placeholder" class="hover:bg-C text-D">{{ placeholder }}</option> | ||||||
|  |       <option v-for="option in options" :key="option.value" :selected="option.selected" :value="option.value" class="hover:bg-C text-D"> | ||||||
|  |         {{ option.label }} | ||||||
|  |       </option> | ||||||
|  |     </select> | ||||||
|  | </template> | ||||||
|  | 
 | ||||||
|  | <script setup> | ||||||
|  | const props = defineProps({ | ||||||
|  |   modelValue: { | ||||||
|  |     type: [String, Number], | ||||||
|  |     default: '', | ||||||
|  |   }, | ||||||
|  |   placeholder: { | ||||||
|  |     type: String, | ||||||
|  |     default: '', | ||||||
|  |   }, | ||||||
|  |   options: { | ||||||
|  |     type: Array, | ||||||
|  |     required: true, | ||||||
|  |   }, | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | const emit = defineEmits(['update:modelValue']); | ||||||
|  | </script> | ||||||
| @ -1,65 +1,31 @@ | |||||||
| <template> | <template> | ||||||
|   <div> |   <div | ||||||
|     <!-- Card Produk --> |     class="relative border border-C rounded-md aspect-square flex items-center justify-center hover:shadow-md transition cursor-pointer overflow-hidden" | ||||||
|  |     @click="$emit('click', product.id)" | ||||||
|  |   > | ||||||
|  |     <!-- Foto Produk --> | ||||||
|  |     <img | ||||||
|  |       v-if="product.foto && product.foto.length > 0" | ||||||
|  |       :src="product.foto[0].url" | ||||||
|  |       :alt="product.nama" | ||||||
|  |       class="w-full h-full object-cover" | ||||||
|  |     /> | ||||||
|  |     <span v-else class="text-gray-400 text-sm">[tidak ada foto]</span> | ||||||
|  | 
 | ||||||
|  |     <!-- Nama Produk di bawah --> | ||||||
|     <div |     <div | ||||||
|       class="border border-C rounded-md aspect-square flex items-center justify-center hover:shadow-md transition cursor-pointer" |       class="absolute bottom-0 w-full bg-black/60 text-white text-center text-sm py-1" | ||||||
|       @click="showDetail = true" |  | ||||||
|     > |     > | ||||||
|       <span class="text-gray-700 font-medium text-center px-2"> |       {{ product.nama }} | ||||||
|         {{ product.nama }} |  | ||||||
|       </span> |  | ||||||
|     </div> |  | ||||||
| 
 |  | ||||||
|     <!-- Overlay Detail --> |  | ||||||
|     <div |  | ||||||
|       v-if="showDetail" |  | ||||||
|       class="fixed inset-0 bg-black bg-opacity-50 flex items-center justify-center z-50" |  | ||||||
|     > |  | ||||||
|       <div |  | ||||||
|         class="bg-white rounded-lg shadow-lg w-[90%] max-w-md p-6 relative" |  | ||||||
|       > |  | ||||||
|         <!-- Tombol Close --> |  | ||||||
|         <button |  | ||||||
|           class="absolute top-3 right-3 text-gray-500 hover:text-gray-800" |  | ||||||
|           @click="showDetail = false" |  | ||||||
|         > |  | ||||||
|           ✕ |  | ||||||
|         </button> |  | ||||||
| 
 |  | ||||||
|         <!-- Judul --> |  | ||||||
|         <h2 class="text-xl font-semibold text-D mb-4 text-center"> |  | ||||||
|           Detail Produk |  | ||||||
|         </h2> |  | ||||||
| 
 |  | ||||||
|         <!-- Data Produk --> |  | ||||||
|         <div class="space-y-2 text-gray-700"> |  | ||||||
|           <p><span class="font-semibold">Nama:</span> {{ product.nama }}</p> |  | ||||||
|           <p><span class="font-semibold">Kategori:</span> {{ product.kategori }}</p> |  | ||||||
|           <p><span class="font-semibold">Berat:</span> {{ product.berat }} gram</p> |  | ||||||
|           <p><span class="font-semibold">Kadar:</span> {{ product.kadar }}%</p> |  | ||||||
|           <p><span class="font-semibold">Harga/gram:</span> Rp {{ formatHarga(product.harga_per_gram) }}</p> |  | ||||||
|           <p><span class="font-semibold">Harga Jual:</span> Rp {{ formatHarga(product.harga_jual) }}</p> |  | ||||||
|           <p><span class="font-semibold">Stok:</span> {{ product.items_count }} pcs</p> |  | ||||||
|         </div> |  | ||||||
|       </div> |  | ||||||
|     </div> |     </div> | ||||||
|   </div> |   </div> | ||||||
| </template> | </template> | ||||||
| 
 | 
 | ||||||
| <script setup> | <script setup> | ||||||
| import { ref } from "vue"; | defineProps({ | ||||||
| 
 |  | ||||||
| const props = defineProps({ |  | ||||||
|   product: { |   product: { | ||||||
|     type: Object, |     type: Object, | ||||||
|     required: true, |     required: true, | ||||||
|   }, |   }, | ||||||
| }); | }); | ||||||
| 
 |  | ||||||
| const showDetail = ref(false); |  | ||||||
| 
 |  | ||||||
| // Format rupiah |  | ||||||
| function formatHarga(value) { |  | ||||||
|   return new Intl.NumberFormat("id-ID").format(value); |  | ||||||
| } |  | ||||||
| </script> | </script> | ||||||
|  | |||||||
							
								
								
									
										384
									
								
								resources/js/pages/InputProduk.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										384
									
								
								resources/js/pages/InputProduk.vue
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,384 @@ | |||||||
|  | <template> | ||||||
|  |   <mainLayout> | ||||||
|  |     <div class="p-6"> | ||||||
|  |       <p class="font-serif italic text-[25px] text-D">Produk Baru</p> | ||||||
|  |      | ||||||
|  |       <div class="flex flex-col md:flex-row mt-5 gap-6"> | ||||||
|  |         <!-- Form Section --> | ||||||
|  |         <div class="flex-1"> | ||||||
|  |           <div class="mb-3"> | ||||||
|  |             <label class="block text-D mb-1">Nama Produk</label> | ||||||
|  |             <InputField  | ||||||
|  |               v-model="form.nama" | ||||||
|  |               type="text"  | ||||||
|  |               placeholder="Masukkan nama produk"  | ||||||
|  |             /> | ||||||
|  |           </div> | ||||||
|  |            | ||||||
|  |           <div class="mb-3"> | ||||||
|  |             <label class="block text-D mb-1">Kategori</label> | ||||||
|  |             <InputSelect  | ||||||
|  |               v-model="form.kategori" | ||||||
|  |               :options="category"  | ||||||
|  |               placeholder="Pilih kategori"  | ||||||
|  |             /> | ||||||
|  |           </div> | ||||||
|  |            | ||||||
|  |           <div class="mb-3 flex flex-row w-full gap-3"> | ||||||
|  |             <div class="flex-1"> | ||||||
|  |               <label class="block text-D mb-1">Berat (g)</label> | ||||||
|  |               <InputField | ||||||
|  |                 v-model="form.berat" | ||||||
|  |                 type="number" | ||||||
|  |                 step="0.01" | ||||||
|  |                 placeholder="Masukkan berat" | ||||||
|  |                 @input="calculateHargaJual" | ||||||
|  |               /> | ||||||
|  |             </div> | ||||||
|  |             <div class="flex-1"> | ||||||
|  |               <label class="block text-D mb-1">Kadar (K)</label> | ||||||
|  |               <InputField  | ||||||
|  |                 v-model="form.kadar" | ||||||
|  |                 type="number"  | ||||||
|  |                 placeholder="Masukkan kadar"  | ||||||
|  |               /> | ||||||
|  |             </div> | ||||||
|  |           </div> | ||||||
|  |            | ||||||
|  |           <div class="mb-3 flex flex-row w-full gap-3"> | ||||||
|  |             <div class="flex-1"> | ||||||
|  |               <label class="block text-D mb-1">Harga per Gram</label> | ||||||
|  |               <InputField | ||||||
|  |                 v-model="form.harga_per_gram" | ||||||
|  |                 type="number" | ||||||
|  |                 step="0.01" | ||||||
|  |                 placeholder="Masukkan harga per gram" | ||||||
|  |                 @input="calculateHargaJual" | ||||||
|  |               /> | ||||||
|  |             </div> | ||||||
|  |             <div class="flex-1"> | ||||||
|  |               <label class="block text-D mb-1">Harga Jual</label> | ||||||
|  |               <InputField | ||||||
|  |                 v-model="form.harga_jual" | ||||||
|  |                 type="number" | ||||||
|  |                 step="0.01" | ||||||
|  |                 placeholder="Masukkan harga jual" | ||||||
|  |               /> | ||||||
|  |             </div> | ||||||
|  |           </div> | ||||||
|  | 
 | ||||||
|  |           <div class="mt-6"> | ||||||
|  |             <button  | ||||||
|  |               @click="submitForm" | ||||||
|  |               :disabled="loading || !isFormValid" | ||||||
|  |               class="bg-blue-600 text-white px-6 py-2 rounded-md hover:bg-blue-700 disabled:bg-gray-400 disabled:cursor-not-allowed" | ||||||
|  |             > | ||||||
|  |               {{ loading ? 'Menyimpan...' : 'Simpan Produk' }} | ||||||
|  |             </button> | ||||||
|  |           </div> | ||||||
|  |         </div> | ||||||
|  |          | ||||||
|  |         <!-- Image Upload Section --> | ||||||
|  |         <div class="flex-1"> | ||||||
|  |           <label class="block text-D mb-1">Foto</label> | ||||||
|  |            | ||||||
|  |           <!-- Image Grid --> | ||||||
|  |           <div class="grid grid-cols-3 gap-3"> | ||||||
|  |             <!-- Uploaded Images --> | ||||||
|  |             <div  | ||||||
|  |               v-for="(image, index) in uploadedImages"  | ||||||
|  |               :key="`img-${image.id}`" | ||||||
|  |               class="relative group aspect-square" | ||||||
|  |             > | ||||||
|  |               <div class="w-full h-full bg-gray-100 rounded-lg border-2 border-gray-200 overflow-hidden"> | ||||||
|  |                 <img  | ||||||
|  |                   :src="image.url"  | ||||||
|  |                   :alt="`Foto ${index + 1}`" | ||||||
|  |                   class="w-full h-full object-cover" | ||||||
|  |                 /> | ||||||
|  |                 <!-- Delete Button --> | ||||||
|  |                 <button  | ||||||
|  |                   @click="removeImage(image.id)" | ||||||
|  |                   :disabled="uploadLoading" | ||||||
|  |                   class="absolute -top-2 -right-2 bg-red-500 text-white rounded-full w-6 h-6 flex items-center justify-center text-sm font-bold shadow-lg hover:bg-red-600 transition-colors disabled:bg-gray-400" | ||||||
|  |                 > | ||||||
|  |                   × | ||||||
|  |                 </button> | ||||||
|  |               </div> | ||||||
|  |             </div> | ||||||
|  |              | ||||||
|  |             <!-- Upload Button --> | ||||||
|  |             <div  | ||||||
|  |               v-if="uploadedImages.length < 6" | ||||||
|  |               @drop="handleDrop" | ||||||
|  |               @dragover.prevent | ||||||
|  |               @dragenter.prevent="isDragging = true" | ||||||
|  |               @dragleave.prevent="isDragging = false" | ||||||
|  |               @click="triggerFileInput" | ||||||
|  |               class="aspect-square bg-gray-50 border-2 border-dashed border-gray-300 rounded-lg flex flex-col items-center justify-center cursor-pointer hover:border-blue-400 hover:bg-blue-50 transition-colors group" | ||||||
|  |               :class="{  | ||||||
|  |                 'border-blue-400 bg-blue-50': isDragging, | ||||||
|  |                 'cursor-not-allowed opacity-50': uploadLoading  | ||||||
|  |               }" | ||||||
|  |             > | ||||||
|  |               <div class="text-center"> | ||||||
|  |                 <!-- Upload Icon or Loading --> | ||||||
|  |                 <div v-if="!uploadLoading" class="w-12 h-12 bg-blue-600 rounded-lg flex items-center justify-center mx-auto mb-2 group-hover:bg-blue-700 transition-colors"> | ||||||
|  |                   <svg class="w-6 h-6 text-white" fill="none" stroke="currentColor" viewBox="0 0 24 24"> | ||||||
|  |                     <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 6v6m0 0v6m0-6h6m-6 0H6"></path> | ||||||
|  |                   </svg> | ||||||
|  |                 </div> | ||||||
|  |                 <div v-else class="w-12 h-12 bg-blue-600 rounded-lg flex items-center justify-center mx-auto mb-2"> | ||||||
|  |                   <svg class="animate-spin w-6 h-6 text-white" fill="none" viewBox="0 0 24 24"> | ||||||
|  |                     <circle class="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" stroke-width="4"></circle> | ||||||
|  |                     <path class="opacity-75" fill="currentColor" d="m4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z"></path> | ||||||
|  |                   </svg> | ||||||
|  |                 </div> | ||||||
|  |                 <p class="text-xs text-gray-600 font-medium" v-html="uploadLoading ? 'Uploading...' : 'Unggah<br/>Foto'"></p> | ||||||
|  |               </div> | ||||||
|  |             </div> | ||||||
|  |           </div> | ||||||
|  |            | ||||||
|  |           <!-- Hidden File Input --> | ||||||
|  |           <input | ||||||
|  |             ref="fileInput" | ||||||
|  |             type="file" | ||||||
|  |             multiple | ||||||
|  |             accept="image/jpeg,image/jpg,image/png" | ||||||
|  |             @change="handleFileSelect" | ||||||
|  |             class="hidden" | ||||||
|  |           /> | ||||||
|  |            | ||||||
|  |           <!-- Upload Info --> | ||||||
|  |           <p class="text-xs text-gray-500 mt-2">Format: JPG, JPEG, PNG (Max: 2MB per file, Max: 6 foto)</p> | ||||||
|  |            | ||||||
|  |           <!-- Error Message --> | ||||||
|  |           <div v-if="uploadError" class="mt-2 p-2 bg-red-50 border border-red-200 rounded text-sm text-red-600"> | ||||||
|  |             {{ uploadError }} | ||||||
|  |           </div> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |   </mainLayout> | ||||||
|  | </template> | ||||||
|  | 
 | ||||||
|  | <script setup> | ||||||
|  | import { ref, computed, onMounted, onUnmounted } from "vue"; | ||||||
|  | import axios from "axios"; | ||||||
|  | import mainLayout from "../layouts/mainLayout.vue"; | ||||||
|  | import InputField from "../components/InputField.vue"; | ||||||
|  | import InputSelect from "../components/InputSelect.vue"; | ||||||
|  | 
 | ||||||
|  | const form = ref({ | ||||||
|  |   nama: '', | ||||||
|  |   kategori: '', | ||||||
|  |   berat: 0, | ||||||
|  |   kadar: 0, | ||||||
|  |   harga_per_gram: 0, | ||||||
|  |   harga_jual: 0, | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | const category = ref([ | ||||||
|  |   { value: "cincin", label: "Cincin" }, | ||||||
|  |   { value: "gelang", label: "Gelang" }, | ||||||
|  |   { value: "kalung", label: "Kalung" }, | ||||||
|  |   { value: "anting", label: "Anting" }, | ||||||
|  | ]); | ||||||
|  | 
 | ||||||
|  | const loading = ref(false); | ||||||
|  | const uploadLoading = ref(false); | ||||||
|  | const uploadedImages = ref([]); | ||||||
|  | const isDragging = ref(false); | ||||||
|  | const uploadError = ref(''); | ||||||
|  | const fileInput = ref(null); | ||||||
|  | const userId = ref(1); // Sesuaikan dengan user yang login | ||||||
|  | 
 | ||||||
|  | const isFormValid = computed(() => { | ||||||
|  |   return form.value.nama &&  | ||||||
|  |          form.value.kategori &&  | ||||||
|  |          form.value.berat > 0 &&  | ||||||
|  |          form.value.kadar > 0 &&  | ||||||
|  |          form.value.harga_per_gram > 0 && | ||||||
|  |          form.value.harga_jual > 0; | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | const calculateHargaJual = () => { | ||||||
|  |   const berat = parseFloat(form.value.berat) || 0; | ||||||
|  |   const hargaPerGram = parseFloat(form.value.harga_per_gram) || 0; | ||||||
|  |   if (berat > 0 && hargaPerGram > 0) { | ||||||
|  |     form.value.harga_jual = berat * hargaPerGram; | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | const loadExistingPhotos = async () => { | ||||||
|  |   try { | ||||||
|  |     const response = await axios.get(`/api/foto/${userId.value}`); | ||||||
|  |     if (response.data && Array.isArray(response.data)) { | ||||||
|  |       uploadedImages.value = response.data; | ||||||
|  |     } | ||||||
|  |   } catch (error) { | ||||||
|  |     if (error.response?.status !== 404) { | ||||||
|  |       console.error('Error loading existing photos:', error); | ||||||
|  |     } | ||||||
|  |     // 404 is expected when no photos exist yet | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | const triggerFileInput = () => { | ||||||
|  |   if (!uploadLoading.value && uploadedImages.value.length < 6) { | ||||||
|  |     fileInput.value?.click(); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | const handleFileSelect = (event) => { | ||||||
|  |   const files = Array.from(event.target.files); | ||||||
|  |   uploadFiles(files); | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | const handleDrop = (event) => { | ||||||
|  |   event.preventDefault(); | ||||||
|  |   isDragging.value = false; | ||||||
|  |    | ||||||
|  |   if (uploadLoading.value || uploadedImages.value.length >= 6) return; | ||||||
|  |    | ||||||
|  |   const files = Array.from(event.dataTransfer.files); | ||||||
|  |   uploadFiles(files); | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | const uploadFiles = async (files) => { | ||||||
|  |   uploadError.value = ''; | ||||||
|  |    | ||||||
|  |   // Validate file count | ||||||
|  |   if (uploadedImages.value.length + files.length > 6) { | ||||||
|  |     uploadError.value = 'Maksimal 6 foto yang dapat diupload'; | ||||||
|  |     return; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   // Validate file types and sizes | ||||||
|  |   const validFiles = files.filter(file => { | ||||||
|  |     const isValidType = ['image/jpeg', 'image/jpg', 'image/png'].includes(file.type); | ||||||
|  |     const isValidSize = file.size <= 2 * 1024 * 1024; // 2MB | ||||||
|  |      | ||||||
|  |     if (!isValidType) { | ||||||
|  |       uploadError.value = 'Format file harus JPG, JPEG, atau PNG'; | ||||||
|  |       return false; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     if (!isValidSize) { | ||||||
|  |       uploadError.value = 'Ukuran file maksimal 2MB'; | ||||||
|  |       return false; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     return true; | ||||||
|  |   }); | ||||||
|  |    | ||||||
|  |   if (validFiles.length === 0) return; | ||||||
|  |    | ||||||
|  |   uploadLoading.value = true; | ||||||
|  |    | ||||||
|  |   try { | ||||||
|  |     for (const file of validFiles) { | ||||||
|  |       const formData = new FormData(); | ||||||
|  |       formData.append('foto', file); | ||||||
|  |       formData.append('id_user', userId.value); | ||||||
|  |        | ||||||
|  |       const response = await axios.post('/api/foto/upload', formData, { | ||||||
|  |         headers: { | ||||||
|  |           'Content-Type': 'multipart/form-data', | ||||||
|  |         }, | ||||||
|  |       }); | ||||||
|  |        | ||||||
|  |       uploadedImages.value.push(response.data); | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     // Clear file input | ||||||
|  |     if (fileInput.value) { | ||||||
|  |       fileInput.value.value = ''; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |   } catch (error) { | ||||||
|  |     console.error('Upload error:', error); | ||||||
|  |     uploadError.value = error.response?.data?.message || 'Gagal mengupload foto'; | ||||||
|  |   } finally { | ||||||
|  |     uploadLoading.value = false; | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | const removeImage = async (imageId) => { | ||||||
|  |   try { | ||||||
|  |     await axios.delete(`/api/foto/hapus/${imageId}`); | ||||||
|  |     uploadedImages.value = uploadedImages.value.filter(img => img.id !== imageId); | ||||||
|  |     uploadError.value = ''; | ||||||
|  |   } catch (error) { | ||||||
|  |     console.error('Delete error:', error); | ||||||
|  |     uploadError.value = 'Gagal menghapus foto'; | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | const submitForm = async () => { | ||||||
|  |   if (!isFormValid.value) { | ||||||
|  |     alert('Mohon lengkapi semua field yang diperlukan'); | ||||||
|  |     return; | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   loading.value = true; | ||||||
|  |    | ||||||
|  |   try { | ||||||
|  |     const response = await axios.post('/api/produk', { | ||||||
|  |       ...form.value, | ||||||
|  |       id_user: userId.value | ||||||
|  |     }); | ||||||
|  |      | ||||||
|  |     // Reset form | ||||||
|  |     form.value = { | ||||||
|  |       nama: '', | ||||||
|  |       kategori: '', | ||||||
|  |       berat: 0, | ||||||
|  |       kadar: 0, | ||||||
|  |       harga_per_gram: 0, | ||||||
|  |       harga_jual: 0, | ||||||
|  |     }; | ||||||
|  |      | ||||||
|  |     uploadedImages.value = []; | ||||||
|  |     uploadError.value = ''; | ||||||
|  |      | ||||||
|  |     if (fileInput.value) { | ||||||
|  |       fileInput.value.value = ''; | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     alert('Produk berhasil disimpan!'); | ||||||
|  |      | ||||||
|  |   } catch (error) { | ||||||
|  |     console.error('Submit error:', error); | ||||||
|  |      | ||||||
|  |     if (error.response?.data?.errors) { | ||||||
|  |       const errors = Object.values(error.response.data.errors).flat(); | ||||||
|  |       alert('Error: ' + errors.join(', ')); | ||||||
|  |     } else { | ||||||
|  |       alert('Gagal menyimpan produk: ' + (error.response?.data?.message || error.message)); | ||||||
|  |     } | ||||||
|  |   } finally { | ||||||
|  |     loading.value = false; | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | const resetPhotos = async () => { | ||||||
|  |   try { | ||||||
|  |     await axios.delete(`/api/foto/reset/${userId.value}`); | ||||||
|  |     uploadedImages.value = []; | ||||||
|  |   } catch (error) { | ||||||
|  |     console.error('Error resetting photos:', error); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | // Load existing photos on component mount | ||||||
|  | onMounted(() => { | ||||||
|  |   loadExistingPhotos(); | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | // Clean up photos if user leaves without saving | ||||||
|  | onUnmounted(() => { | ||||||
|  |   // Optional: You might want to clean up temporary photos here | ||||||
|  |   // resetPhotos(); | ||||||
|  | }); | ||||||
|  | </script> | ||||||
| @ -4,14 +4,27 @@ | |||||||
|       <!-- Judul --> |       <!-- Judul --> | ||||||
|       <p class="font-serif italic text-[25px] text-D">PRODUK</p> |       <p class="font-serif italic text-[25px] text-D">PRODUK</p> | ||||||
| 
 | 
 | ||||||
|       <!-- Search --> |       <!-- Filter --> | ||||||
|       <searchbar v-model:search="searchQuery" /> |       <div class="mt-3 flex flex-col md:flex-row md:items-center md:justify-between gap-3"> | ||||||
|  |         <!-- Dropdown Kategori --> | ||||||
|  |         <select | ||||||
|  |           v-model="selectedCategory" | ||||||
|  |           class="border border-gray-300 rounded-md px-3 py-2 bg-B focus:outline-none focus:ring-2 focus:ring-B w-full md:w-48" | ||||||
|  |         > | ||||||
|  |           <option value="semua">Semua</option> | ||||||
|  |           <option value="cincin">Cincin</option> | ||||||
|  |           <option value="gelang">Gelang</option> | ||||||
|  |           <option value="kalung">Kalung</option> | ||||||
|  |           <option value="anting">Anting</option> | ||||||
|  |         </select> | ||||||
|  | 
 | ||||||
|  |         <!-- Search --> | ||||||
|  |         <searchbar v-model:search="searchQuery" class="flex-1" /> | ||||||
|  |       </div> | ||||||
| 
 | 
 | ||||||
|       <!-- Tombol Tambah Produk --> |       <!-- Tombol Tambah Produk --> | ||||||
|       <div class="mt-3 flex justify-end"> |       <div class="mt-3 flex justify-end"> | ||||||
|         <button |         <button class="bg-C text-[#0a1a3c] px-4 py-2 rounded-md shadow hover:bg-C transition"> | ||||||
|           class="bg-B text-[#0a1a3c] px-4 py-2 rounded-md shadow hover:bg-C transition" |  | ||||||
|         > |  | ||||||
|           Tambah Produk |           Tambah Produk | ||||||
|         </button> |         </button> | ||||||
|       </div> |       </div> | ||||||
| @ -22,78 +35,90 @@ | |||||||
|           v-for="item in filteredProducts" |           v-for="item in filteredProducts" | ||||||
|           :key="item.id" |           :key="item.id" | ||||||
|           :product="item" |           :product="item" | ||||||
|           @showDetail="openOverlay" |           @click="openOverlay(item.id)" | ||||||
|         /> |         /> | ||||||
|       </div> |       </div> | ||||||
|     </div> |     </div> | ||||||
| 
 | 
 | ||||||
|     <!-- Overlay Detail Produk --> |     <!-- Overlay Detail Produk --> | ||||||
|     <div |     <!-- Overlay Detail Produk --> | ||||||
|       v-if="showOverlay" | <div | ||||||
|       class="fixed inset-0 bg-black bg-opacity-50 flex justify-center items-center z-50" |   v-if="showOverlay" | ||||||
|       @click.self="closeOverlay" |   class="fixed inset-0 bg-black/30 flex justify-center items-center z-50" | ||||||
|     > |   @click.self="closeOverlay" | ||||||
|       <div | > | ||||||
|         class="bg-white rounded-lg shadow-lg p-6 w-[400px] border-2 border-[#e6d3b3] relative" |   <div | ||||||
|       > |     class="bg-white rounded-lg shadow-lg p-6 w-[400px] border-2 border-[#e6d3b3] relative flex flex-col items-center" | ||||||
|         <!-- Tombol Close --> |     @mouseleave="closeOverlay" | ||||||
|         <button |   > | ||||||
|           @click="closeOverlay" |     <!-- Foto Produk dengan Slider --> | ||||||
|           class="absolute top-2 right-2 text-gray-500 hover:text-black" |     <div class="relative w-60 h-60 border border-[#e6d3b3] flex items-center justify-center mb-4 overflow-hidden rounded"> | ||||||
|         > |       <img | ||||||
|           ✕ |         v-if="detail.foto && detail.foto.length > 0" | ||||||
|         </button> |         :src="detail.foto[currentFotoIndex].url" | ||||||
|  |         :alt="detail.nama" | ||||||
|  |         class="w-full h-full object-contain" | ||||||
|  |       /> | ||||||
|  |       <span v-else class="text-gray-400 text-sm">[gambar]</span> | ||||||
| 
 | 
 | ||||||
|         <!-- Foto Produk --> |       <!-- Stok (pcs) pojok kiri atas --> | ||||||
|         <div class="border border-[#e6d3b3] p-2 mb-4 flex justify-center"> |       <div class="absolute top-1 left-1 bg-black/60 text-white text-xs px-2 py-1 rounded"> | ||||||
|           <img |         {{ detail.item_count }} pcs | ||||||
|             v-if="detail.gambar" |  | ||||||
|             :src="`http://127.0.0.1:8000/storage/${detail.gambar}`" |  | ||||||
|             :alt="detail.nama" |  | ||||||
|             class="w-40 h-40 object-contain" |  | ||||||
|           /> |  | ||||||
|           <span v-else class="text-gray-400 text-sm">[gambar]</span> |  | ||||||
|         </div> |  | ||||||
| 
 |  | ||||||
|         <!-- Stok --> |  | ||||||
|         <p class="text-sm mb-1">{{ detail.item_count }} pcs</p> |  | ||||||
| 
 |  | ||||||
|         <!-- Nama Produk --> |  | ||||||
|         <h2 class="text-xl font-semibold text-center mb-3"> |  | ||||||
|           {{ detail.nama }} |  | ||||||
|         </h2> |  | ||||||
| 
 |  | ||||||
|         <!-- Detail Harga & Info --> |  | ||||||
|         <div class="grid grid-cols-2 gap-2 text-sm mb-4"> |  | ||||||
|           <p>Harga Beli : Rp. {{ formatNumber(detail.harga_beli) }}</p> |  | ||||||
|           <p class="text-right">{{ detail.kadar }} K</p> |  | ||||||
|           <p>Harga Jual : Rp. {{ formatNumber(detail.harga_jual) }}</p> |  | ||||||
|           <p class="text-right">{{ detail.berat }} gram</p> |  | ||||||
|           <p class="col-span-2"> |  | ||||||
|             Harga/gram : Rp. {{ formatNumber(detail.harga_per_gram) }} |  | ||||||
|           </p> |  | ||||||
|         </div> |  | ||||||
| 
 |  | ||||||
|         <!-- Tombol Aksi --> |  | ||||||
|         <div class="flex justify-between"> |  | ||||||
|           <button |  | ||||||
|             class="bg-yellow-400 text-black px-4 py-2 rounded font-bold" |  | ||||||
|           > |  | ||||||
|             Ubah |  | ||||||
|           </button> |  | ||||||
|           <button |  | ||||||
|             class="bg-green-400 text-black px-4 py-2 rounded font-bold" |  | ||||||
|           > |  | ||||||
|             Tambah |  | ||||||
|           </button> |  | ||||||
|           <button |  | ||||||
|             class="bg-red-500 text-white px-4 py-2 rounded font-bold" |  | ||||||
|           > |  | ||||||
|             Hapus |  | ||||||
|           </button> |  | ||||||
|         </div> |  | ||||||
|       </div> |       </div> | ||||||
|  | 
 | ||||||
|  |       <!-- Nama Produk di bawah --> | ||||||
|  |       <div | ||||||
|  |         class="absolute bottom-0 w-full bg-black/70 text-white text-center text-sm py-1" | ||||||
|  |       > | ||||||
|  |         {{ detail.nama }} | ||||||
|  |       </div> | ||||||
|  | 
 | ||||||
|  |       <!-- Tombol Prev --> | ||||||
|  |       <button | ||||||
|  |         v-if="detail.foto && detail.foto.length > 1" | ||||||
|  |         @click.stop="prevFoto" | ||||||
|  |         class="absolute left-2 bg-white/70 hover:bg-white text-black px-2 py-1 rounded" | ||||||
|  |       > | ||||||
|  |         ‹ | ||||||
|  |       </button> | ||||||
|  |       <!-- Tombol Next --> | ||||||
|  |       <button | ||||||
|  |         v-if="detail.foto && detail.foto.length > 1" | ||||||
|  |         @click.stop="nextFoto" | ||||||
|  |         class="absolute right-2 bg-white/70 hover:bg-white text-black px-2 py-1 rounded" | ||||||
|  |       > | ||||||
|  |         › | ||||||
|  |       </button> | ||||||
|     </div> |     </div> | ||||||
|  | 
 | ||||||
|  |     <!-- Detail Harga & Info --> | ||||||
|  |     <div class="grid grid-cols-2 gap-2 text-sm mb-4 w-full"> | ||||||
|  |       <!-- harga beli dihapus --> | ||||||
|  |       <p>Harga Jual : Rp. {{ formatNumber(detail.harga_jual) }}</p> | ||||||
|  |       <p class="text-right">{{ detail.kadar }} K</p> | ||||||
|  |       <p class="col-span-2 text-center"> | ||||||
|  |         Berat : {{ detail.berat }} gram | ||||||
|  |       </p> | ||||||
|  |       <p class="col-span-2"> | ||||||
|  |         Harga/gram : Rp. {{ formatNumber(detail.harga_per_gram) }} | ||||||
|  |       </p> | ||||||
|  |     </div> | ||||||
|  | 
 | ||||||
|  |     <!-- Tombol Aksi --> | ||||||
|  |     <div class="flex justify-between w-full"> | ||||||
|  |       <button class="bg-yellow-400 text-black px-4 py-2 rounded font-bold"> | ||||||
|  |         Ubah | ||||||
|  |       </button> | ||||||
|  |       <button class="bg-green-400 text-black px-4 py-2 rounded font-bold"> | ||||||
|  |         Tambah | ||||||
|  |       </button> | ||||||
|  |       <button class="bg-red-500 text-white px-4 py-2 rounded font-bold"> | ||||||
|  |         Hapus | ||||||
|  |       </button> | ||||||
|  |     </div> | ||||||
|  |   </div> | ||||||
|  | </div> | ||||||
|  | 
 | ||||||
|   </mainLayout> |   </mainLayout> | ||||||
| </template> | </template> | ||||||
| 
 | 
 | ||||||
| @ -106,10 +131,12 @@ import searchbar from "../components/searchbar.vue"; | |||||||
| 
 | 
 | ||||||
| const products = ref([]); | const products = ref([]); | ||||||
| const searchQuery = ref(""); | const searchQuery = ref(""); | ||||||
|  | const selectedCategory = ref("semua"); | ||||||
| 
 | 
 | ||||||
| // overlay state | // overlay state | ||||||
| const showOverlay = ref(false); | const showOverlay = ref(false); | ||||||
| const detail = ref({}); | const detail = ref({}); | ||||||
|  | const currentFotoIndex = ref(0); | ||||||
| 
 | 
 | ||||||
| // Fetch data awal | // Fetch data awal | ||||||
| onMounted(async () => { | onMounted(async () => { | ||||||
| @ -121,29 +148,60 @@ onMounted(async () => { | |||||||
|   } |   } | ||||||
| }); | }); | ||||||
| 
 | 
 | ||||||
| // Filter | // Filter gabungan (kategori + search) | ||||||
| const filteredProducts = computed(() => { | const filteredProducts = computed(() => { | ||||||
|   if (!searchQuery.value) return products.value; |   let hasil = products.value; | ||||||
|   return products.value.filter((p) => | 
 | ||||||
|     p.nama.toLowerCase().includes(searchQuery.value.toLowerCase()) |   // filter kategori | ||||||
|   ); |   if (selectedCategory.value !== "semua") { | ||||||
|  |     hasil = hasil.filter( | ||||||
|  |       (p) => p.kategori.toLowerCase() === selectedCategory.value | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   // filter search | ||||||
|  |   if (searchQuery.value) { | ||||||
|  |     hasil = hasil.filter((p) => | ||||||
|  |       p.nama.toLowerCase().includes(searchQuery.value.toLowerCase()) | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   return hasil; | ||||||
| }); | }); | ||||||
| 
 | 
 | ||||||
| // Fungsi buka overlay | // buka overlay | ||||||
| async function openOverlay(id) { | async function openOverlay(id) { | ||||||
|   try { |   try { | ||||||
|     const res = await axios.get(`http://127.0.0.1:8000/api/produk/${id}`); |     const res = await axios.get(`http://127.0.0.1:8000/api/produk/${id}`); | ||||||
|     detail.value = res.data; |     detail.value = res.data; | ||||||
|  |     currentFotoIndex.value = 0; // reset ke foto pertama | ||||||
|     showOverlay.value = true; |     showOverlay.value = true; | ||||||
|   } catch (error) { |   } catch (error) { | ||||||
|     console.error("Gagal fetch detail produk:", error); |     console.error("Gagal fetch detail produk:", error); | ||||||
|   } |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // Fungsi tutup overlay | // tutup overlay | ||||||
| function closeOverlay() { | function closeOverlay() { | ||||||
|   showOverlay.value = false; |   showOverlay.value = false; | ||||||
|   detail.value = {}; |   detail.value = {}; | ||||||
|  |   currentFotoIndex.value = 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // foto navigation | ||||||
|  | function nextFoto() { | ||||||
|  |   if (detail.value.foto && detail.value.foto.length > 0) { | ||||||
|  |     currentFotoIndex.value = | ||||||
|  |       (currentFotoIndex.value + 1) % detail.value.foto.length; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | function prevFoto() { | ||||||
|  |   if (detail.value.foto && detail.value.foto.length > 0) { | ||||||
|  |     currentFotoIndex.value = | ||||||
|  |       (currentFotoIndex.value - 1 + detail.value.foto.length) % | ||||||
|  |       detail.value.foto.length; | ||||||
|  |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // Format angka | // Format angka | ||||||
|  | |||||||
| @ -3,6 +3,7 @@ import Home from '../pages/Home.vue' | |||||||
| import Produk from '../pages/Produk.vue' | import Produk from '../pages/Produk.vue' | ||||||
| import Brankas from '../pages/Brankas.vue' | import Brankas from '../pages/Brankas.vue' | ||||||
| import Tray from '../pages/Tray.vue' | import Tray from '../pages/Tray.vue' | ||||||
|  | import InputProduk from '../pages/InputProduk.vue' | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| const routes = [ | const routes = [ | ||||||
| @ -16,6 +17,11 @@ const routes = [ | |||||||
|     name: 'Produk', |     name: 'Produk', | ||||||
|     component: Produk |     component: Produk | ||||||
|   }, |   }, | ||||||
|  |   { | ||||||
|  |     path: '/produk/baru', | ||||||
|  |     name: 'Produk', | ||||||
|  |     component: InputProduk | ||||||
|  |   }, | ||||||
|   { |   { | ||||||
|     path: '/brankas', |     path: '/brankas', | ||||||
|     name: 'Brankas', |     name: 'Brankas', | ||||||
|  | |||||||
| @ -19,10 +19,12 @@ Route::prefix('api')->group(function () { | |||||||
|     Route::apiResource('transaksi', TransaksiController::class); |     Route::apiResource('transaksi', TransaksiController::class); | ||||||
|      |      | ||||||
|     Route::get('brankas', [ItemController::class, 'brankasItem']); |     Route::get('brankas', [ItemController::class, 'brankasItem']); | ||||||
|  |      | ||||||
|  |     // Foto Sementara
 | ||||||
|     Route::post('foto/upload', [FotoSementaraController::class, 'upload']); |     Route::post('foto/upload', [FotoSementaraController::class, 'upload']); | ||||||
|     Route::delete('foto/hapus/<id>', [FotoSementaraController::class, 'hapus']); |     Route::delete('foto/hapus/{id}', [FotoSementaraController::class, 'hapus']); | ||||||
|     Route::get('foto/<user_id>', [FotoSementaraController::class, 'getAll']); |     Route::get('foto/{user_id}', [FotoSementaraController::class, 'getAll']); | ||||||
|     Route::delete('foto/reset/<user_id>', [FotoSementaraController::class, 'reset']); |     Route::delete('foto/reset/{user_id}', [FotoSementaraController::class, 'reset']); | ||||||
| }); | }); | ||||||
| 
 | 
 | ||||||
| // Frontend SPA
 | // Frontend SPA
 | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user