<template> <div class="gantt-container"> <div class="gantt-header"> <!-- Calendar: เชื่อมกับ selectedDate --> <GanttCalendar /> <!-- ปุ่มจัดเรียงและเพิ่มระยะห่าง --> <div class="gantt-buttons"> <MakequeueBtn @click="openAddQueueDialog" /> <SettingBtn class="mr-4" /> </div> </div> <!-- Gantt Chart UI --> <div class="gantt-chart"> <!-- Header: Time Scale --> <div class="header"> <div class="machine-label"></div> <div class="time-scale"> <div v-for="hour in hours" :key="hour" class="time-cell"> {{ formatHour(hour) }} </div> </div> </div> <!-- Rows: แสดงเครื่องจักรจาก Machine Store --> <div class="rows"> <div v-for="machine in machineStore.machines" :key="machine.MachineID" class="row" :data-machine-id="machine.MachineID" @dragover.prevent="onDragOver($event)" @drop="onDrop($event, machine.MachineID)" > <div class="machine-label"> {{ machine.name }} </div> <div class="row-timeline"> <!-- เส้นแนวตั้ง (Grid Lines) --> <div v-for="hour in hours" :key="'line-' + hour" class="vertical-line" :style="getLineStyle(hour)" ></div> <!-- แสดง Queue ที่ตรงกับวันที่ (selectedDate), Page (pagenum), และ Machine --> <div v-for="item in filteredQueue(machine.MachineID)" :key="item.QueueID" class="order" :class="{ faded: draggingQueue && draggingQueue.QueueID === item.QueueID }" :style="getQueueStyle(item)" > <!-- Handle สำหรับ Resize ด้านซ้าย --> <div class="resize-handle left" @mousedown="onResizeStart($event, item, 'left')"></div> <!-- ส่วนกลางของ Order ใช้สำหรับลาก และเปิด Dialog เมื่อคลิก --> <div class="order-content" draggable="true" @dragstart="onDragStart($event, item)" @dragend="onDragEnd($event, item)" @click.stop="openQueueDialog(item)" > {{ item.orderID }} ({{ getTimeString(item.startTime) }} - {{ getTimeString(item.finishTime) }}) </div> <!-- Handle สำหรับ Resize ด้านขวา --> <div class="resize-handle right" @mousedown="onResizeStart($event, item, 'right')"></div> </div> </div> </div> </div> <!-- Ghost Queue (ขณะลาก/resize) --> <div v-if="ghostQueue" class="drag-ghost" :style="ghostStyle"> {{ ghostQueue.orderID }} ({{ getTimeString(ghostQueue.startTime) }} - {{ getTimeString(ghostQueue.finishTime) }}) </div> <v-divider :thickness="7"></v-divider> <!-- Pagination --> <div class="pagination-container"> <div class="pagination"> <button v-for="p in pages" :key="p" :class="['page-btn', { active: p === currentPage }]" @click="currentPage = p" @contextmenu.prevent="onPageRightClick(p, $event)" > {{ p }} <button v-if="pageToShowDelete === p" class="delete-btn" @click.stop="deletePage(p)" > Delete </button> </button> <button class="page-btn add-page" @click="addPage" :disabled="pages.length >= 10"> + </button> </div> <div class="pagination-right"> <v-btn class="add-page-btn" @click="openAddQueueDialog" icon> <v-icon>mdi-plus</v-icon> </v-btn> </div> <AddQueueDialog :visible="showAddDialog" :queueItem="selectedQueueItem" :currentPage="currentPage" @close="closeAddQueueDialog" /> </div> <OrderDialog v-if="selectedQueueItem && !showAddDialog" :queueItem="selectedQueueItem" :color="getQueueColor(selectedQueueItem.orderID)" @close="closeQueueDialog" @edit="openAddQueueDialogForEdit" @delete="handleDelete" /> </div> </div> </template> <script lang="ts" setup> import { ref, reactive, computed, onMounted, onBeforeUnmount, watch } from 'vue'; import { useQueueStore } from '@/stores/queue'; import { useMachineStore } from '@/stores/machine'; import { usePageStore } from '@/stores/page'; import { usePageContextStore } from '@/stores/pageContext'; import { useDateStore } from '@/stores/dateStore'; import { storeToRefs } from 'pinia'; import GanttCalendar from './GanttCalendar.vue'; import OrderDialog from './OrderDialog.vue'; import AddQueueDialog from './AddQueueDialog.vue'; import MakequeueBtn from './MakequeueBtn.vue'; import SettingBtn from './SettingBtn.vue'; import './ganttChart.css'; // Stores const queueStore = useQueueStore(); const machineStore = useMachineStore(); const pageStore = usePageStore(); const dateStore = useDateStore(); const { currentDate: selectedDate } = storeToRefs(dateStore); const pageContext = usePageContextStore(); // ใช้ currentPage เป็น pagenum จาก pageContext const currentPage = computed({ get: () => pageContext.currentPage, set: (val) => (pageContext.currentPage = val), }); // เมื่อเข้าหน้า ให้ดึงข้อมูลจาก backend onMounted(() => { machineStore.fetchMachines(); queueStore.fetchQueues(); pageStore.fetchPages(); }); // State สำหรับ Gantt Chart const showAddDialog = ref(false); const startTime = ref('06:00'); const endTime = ref('20:00'); const draggingQueue = ref<any | null>(null); const dragOffset = ref(0); const resizingQueue = ref<any | null>(null); const resizeDirection = ref<string | null>(null); const ghostQueue = ref<any | null>(null); const ghostStyle = reactive({ position: 'fixed', top: '0px', left: '0px', display: 'none', width: 'auto', height: '40px', lineHeight: '40px', padding: '0 10px', borderRadius: '10px', pointerEvents: 'none', backgroundColor: '#4caf50', color: '#fff', zIndex: 9999, }); // แปลงข้อมูล Queue จาก backend ให้เข้ากับรูปแบบที่ Gantt Chart ใช้ const formattedQueues = computed(() => { return queueStore.queues.map(q => ({ QueueID: q.QueueID, orderID: q.order?.OrderID || "Unknown", machineID: q.machine?.MachineID || 0, machineName: q.machine?.name || "Unknown Machine", startTime: q.startTime ? convertToLocalTime(q.startTime) : "Unknown", finishTime: q.finishTime ? convertToLocalTime(q.finishTime) : "Unknown", status: q.status || "Unknown", bottleSize: q.bottleSize || "Unknown", producedQuantity: q.producedQuantity || 0, // เปลี่ยนจาก PageID เป็น pagenum pageNumber: q.page?.pagenum || 0, })); }); function convertToLocalTime(utcString: string): string { const date = new Date(utcString); date.setHours(date.getHours() + 7); return date.toISOString().slice(0, 16).replace("T", " "); } // Pagination & Dialog const pages = computed(() => pageStore.pages.map(p => p.pagenum)); const pageToShowDelete = ref<number | null>(null); const selectedQueueItem = ref<any | null>(null); watch( () => queueStore.queues, (newVal) => { console.log("🔄 Queues updated in GanttChart.vue:", newVal); }, { deep: true } ); const hours = computed(() => { const startHour = parseInt(startTime.value.split(':')[0]); const endHour = parseInt(endTime.value.split(':')[0]); return Array.from({ length: endHour - startHour + 1 }, (_, i) => startHour + i); }); // Methods function openQueueDialog(item: any) { selectedQueueItem.value = { ...item }; } function closeQueueDialog() { selectedQueueItem.value = null; } function openAddQueueDialog() { selectedQueueItem.value = null; showAddDialog.value = true; } function openAddQueueDialogForEdit(queueItem: any) { selectedQueueItem.value = { ...queueItem }; showAddDialog.value = true; } function handleDelete(queueItem: { QueueID: number; }) { if (!queueItem) return; queueStore .deleteQueue(queueItem.QueueID) .then(() => { console.log(`✅ Queue ID ${queueItem.QueueID} ถูกลบสำเร็จ`); closeAddQueueDialog(); }) .catch((error) => console.error(`❌ Error deleting queue:`, error)); } function closeAddQueueDialog() { showAddDialog.value = false; selectedQueueItem.value = null; } function formatHour(hour: number): string { return (hour < 10 ? '0' + hour : hour) + ':00'; } function getDateString(dateTimeStr: string): string { return dateTimeStr.split(' ')[0]; } function getTimeString(dateTimeStr: string): string { return dateTimeStr.split(' ')[1]; } function filteredQueue(machineID: number) { return formattedQueues.value.filter(q => { if (!q || !q.startTime || !q.finishTime) return false; const queueDate = getDateString(q.startTime); return ( q.machineID === machineID && q.pageNumber === currentPage.value && queueDate === selectedDate.value ); }); } const colorMap: Record<string, string> = {}; function getQueueColor(orderID: string | number): string { const key = String(orderID); if (key === "ผลิตเผื่อ") return "#FFC1C1"; if (key === "เปลี่ยนขนาด") return "#E1BEE7"; if (colorMap[key]) return colorMap[key]; const newColor = generateWaterPastel(); colorMap[key] = newColor; return newColor; } function generateWaterPastel(): string { const waterHues = [180, 190, 200, 210, 220, 230, 240]; const hue = waterHues[Math.floor(Math.random() * waterHues.length)]; return `hsl(${hue}, 60%, 80%)`; } function getQueueStyle(item: any) { const start = getTimeString(item.startTime); const end = getTimeString(item.finishTime); const startDecimal = timeToDecimal(start); const endDecimal = timeToDecimal(end); const totalHours = hours.value.length; const leftPercent = ((startDecimal - timeToDecimal(startTime.value)) / totalHours) * 100; const widthPercent = ((endDecimal - startDecimal) / totalHours) * 100; const orderID = item.orderID || "Unknown"; const backgroundColor = getQueueColor(orderID); console.log(`Queue ${item.QueueID}: orderID=${orderID}, color=${backgroundColor}`); return { left: `${leftPercent}%`, width: `${widthPercent}%`, backgroundColor, color: '#333', borderRadius: '10px', textAlign: 'center', display: 'flex', alignItems: 'center', justifyContent: 'center', padding: '5px', fontSize: '14px', fontWeight: 'bold', boxShadow: '0px 4px 6px rgba(0, 0, 0, 0.1)', }; } function getLineStyle(hour: number) { const totalHours = hours.value.length; const leftPercent = ((hour - timeToDecimal(startTime.value)) / totalHours) * 100; return { left: `${leftPercent}%` }; } function timeToDecimal(timeStr: string): number { const [h, m] = timeStr.split(':').map(Number); return h + m / 60; } function decimalToTime(decimal: number): string { const hrs = Math.floor(decimal); const minutes = Math.round((decimal - hrs) * 60); return (hrs < 10 ? '0' + hrs : hrs) + ':' + (minutes < 10 ? '0' + minutes : minutes); } // Drag/Drop & Resize Functions function onDragStart(event: DragEvent, item: any) { draggingQueue.value = item; const target = event.target as HTMLElement; const rect = target.getBoundingClientRect(); dragOffset.value = rect.width / 2; const emptyImg = new Image(); emptyImg.src = ''; event.dataTransfer?.setDragImage(emptyImg, 0, 0); ghostQueue.value = { ...item }; ghostStyle.display = 'block'; ghostStyle.backgroundColor = getQueueColor(item.order?.OrderID || "Unknown"); document.addEventListener('dragover', onDragOverGlobal); document.addEventListener('dragend', onDragEndGlobal); } function onDragOver(event: DragEvent) { event.preventDefault(); } function onDragOverGlobal(event: DragEvent) { event.preventDefault(); const rowTimeline = document.querySelector('.row-timeline') as HTMLElement; if (!rowTimeline) return; const timelineRect = rowTimeline.getBoundingClientRect(); const timelineWidth = timelineRect.width; const offsetX = event.clientX - timelineRect.left; let ratio = offsetX / timelineWidth; ratio = Math.min(Math.max(ratio, 0), 1); const timelineStartDec = timeToDecimal(startTime.value); const timelineEndDec = timeToDecimal(endTime.value); if (!draggingQueue.value) return; const startDecimal = timeToDecimal(getTimeString(draggingQueue.value.startTime)); const endDecimal = timeToDecimal(getTimeString(draggingQueue.value.finishTime)); const duration = endDecimal - startDecimal; let newStartDecimal = timelineStartDec + ratio * (timelineEndDec - timelineStartDec); newStartDecimal = Math.round(newStartDecimal * 2) / 2; let newEndDecimal = newStartDecimal + duration; if (newEndDecimal > timelineEndDec) { newEndDecimal = timelineEndDec; newStartDecimal = newEndDecimal - duration; } const datePart = getDateString(draggingQueue.value.startTime); const newStartStr = datePart + ' ' + decimalToTime(newStartDecimal); const newEndStr = datePart + ' ' + decimalToTime(newEndDecimal); if (ghostQueue.value) { ghostQueue.value.startTime = newStartStr; ghostQueue.value.finishTime = newEndStr; } // คำนวณตำแหน่ง ghost element const snappedRatioStart = (newStartDecimal - timelineStartDec) / (timelineEndDec - timelineStartDec); const snappedLeft = timelineRect.left + snappedRatioStart * timelineWidth; const snappedRatioEnd = (newEndDecimal - timelineStartDec) / (timelineEndDec - timelineStartDec); const snappedWidth = (snappedRatioEnd - snappedRatioStart) * timelineWidth; // หาจุด row ที่ใกล้เคียงที่สุดและดึง machine id จาก attribute const rows = document.querySelectorAll('.row'); let closestRow: HTMLElement | null = null; let minDistance = Infinity; rows.forEach((row) => { const rect = (row as HTMLElement).getBoundingClientRect(); const distance = Math.abs(event.clientY - rect.top); if (distance < minDistance) { minDistance = distance; closestRow = row as HTMLElement; } }); if (closestRow) { const machineIDString = (closestRow as HTMLElement).dataset.machineId; if (machineIDString) { ghostQueue.value.machineID = parseInt(machineIDString, 10); } ghostStyle.top = (closestRow as HTMLElement).getBoundingClientRect().top + 'px'; } ghostStyle.left = snappedLeft + 'px'; ghostStyle.width = snappedWidth + 'px'; } function onDragEndGlobal() { ghostQueue.value = null; ghostStyle.display = 'none'; draggingQueue.value = null; document.removeEventListener('dragend', onDragEndGlobal); } async function onDragEnd(event: DragEvent, item: any) { if (!ghostQueue.value) return; console.log(`🔄 Updating Queue ${item.QueueID} after drag...`); item.startTime = ghostQueue.value.startTime; item.finishTime = ghostQueue.value.finishTime; const newMachineID = ghostQueue.value.machine?.MachineID || item.machineID; await queueStore.updateQueue(item.QueueID, { startTime: item.startTime, finishTime: item.finishTime, MachineID: newMachineID, }); ghostQueue.value = null; ghostStyle.display = 'none'; draggingQueue.value = null; } async function onDrop(event: DragEvent, newMachineID: number) { event.preventDefault(); if (!draggingQueue.value || !ghostQueue.value) return; console.log(`🔄 Dropped Queue ${draggingQueue.value.QueueID} to Machine ${newMachineID}`); draggingQueue.value.startTime = ghostQueue.value.startTime; draggingQueue.value.finishTime = ghostQueue.value.finishTime; draggingQueue.value.machineID = newMachineID; try { await queueStore.updateQueue(draggingQueue.value.QueueID, { startTime: draggingQueue.value.startTime, finishTime: draggingQueue.value.finishTime, MachineID: newMachineID, }); console.log(`✅ Queue ${draggingQueue.value.QueueID} updated to Machine ${newMachineID}`); } catch (error) { console.error(`❌ Error updating queue:`, error); } ghostQueue.value = null; ghostStyle.display = 'none'; draggingQueue.value = null; } function onResizeStart(event: MouseEvent, item: any, direction: string) { event.preventDefault(); ghostQueue.value = { ...item }; resizingQueue.value = item; resizeDirection.value = direction; document.addEventListener('mousemove', onResizing); document.addEventListener('mouseup', onResizeEnd); } function onResizing(event: MouseEvent) { if (!resizingQueue.value) return; const rowTimeline = document.querySelector('.row-timeline') as HTMLElement; if (!rowTimeline) return; const timelineRect = rowTimeline.getBoundingClientRect(); const timelineWidth = timelineRect.width; const offsetX = event.clientX - timelineRect.left; let ratio = offsetX / timelineWidth; ratio = Math.min(Math.max(ratio, 0), 1); const timelineStartDec = timeToDecimal(startTime.value); const timelineEndDec = timeToDecimal(endTime.value); const startDec = timeToDecimal(getTimeString(resizingQueue.value.startTime)); const endDec = timeToDecimal(getTimeString(resizingQueue.value.finishTime)); let newTimeDecimal = timelineStartDec + ratio * (timelineEndDec - timelineStartDec); newTimeDecimal = Math.round(newTimeDecimal * 2) / 2; const datePart = getDateString(resizingQueue.value.startTime); if (resizeDirection.value === 'left') { if (newTimeDecimal >= endDec) return; const newStartStr = datePart + ' ' + decimalToTime(newTimeDecimal); ghostQueue.value.startTime = newStartStr; } else if (resizeDirection.value === 'right') { if (newTimeDecimal <= startDec) return; const newEndStr = datePart + ' ' + decimalToTime(newTimeDecimal); ghostQueue.value.finishTime = newEndStr; } } async function onResizeEnd() { if (!resizingQueue.value) return; console.log(`🔄 Updating Queue ${resizingQueue.value.QueueID} after resize...`); await queueStore.updateQueue(resizingQueue.value.QueueID, { startTime: ghostQueue.value.startTime, finishTime: ghostQueue.value.finishTime, }); resizingQueue.value = null; resizeDirection.value = null; ghostQueue.value = null; ghostStyle.display = 'none'; document.removeEventListener('mousemove', onResizing); document.removeEventListener('mouseup', onResizeEnd); } // Pagination Functions async function addPage() { if (pageStore.pages.length < 10) { await pageStore.addPage(); await pageStore.fetchPages(); // 👈 โหลดหน้าทั้งหมดใหม่ } else { alert('Maximum of 10 pages allowed.'); } } function onPageRightClick(page: number, event: MouseEvent) { event.preventDefault(); pageToShowDelete.value = page; } async function deletePage(pageNum: number) { await pageStore.removePage(pageNum); await pageStore.fetchPages(); // 👈 โหลดหน้าทั้งหมดใหม่ if (currentPage.value === pageNum) { const remaining = pageStore.pages.map(p => p.pagenum); currentPage.value = remaining.length > 0 ? remaining[0] : 1; } pageToShowDelete.value = null; } function onDocumentClick(event: MouseEvent) { const target = event.target as HTMLElement; if (!target.closest('.page-btn')) { pageToShowDelete.value = null; } } onMounted(() => { document.addEventListener('click', onDocumentClick); }); onBeforeUnmount(() => { document.removeEventListener('click', onDocumentClick); }); </script>