From 7a7042c3f5efb1739628201d38f709a6b7ae991b Mon Sep 17 00:00:00 2001 From: Kritkhanin Anantakul <65160144@go.buu.ac.th> Date: Tue, 18 Mar 2025 16:41:16 +0700 Subject: [PATCH] all gantt service store / connect fontend backend / fix calendar format/ fix some ui --- src/components/GanttChart/GanttChart.vue | 358 +++++++++++------------ 1 file changed, 172 insertions(+), 186 deletions(-) diff --git a/src/components/GanttChart/GanttChart.vue b/src/components/GanttChart/GanttChart.vue index 839de80..a732f38 100644 --- a/src/components/GanttChart/GanttChart.vue +++ b/src/components/GanttChart/GanttChart.vue @@ -4,9 +4,8 @@ <div class="gantt-header"> <!-- Calendar: เชื่อมกับ selectedDate --> <GanttCalendar v-model:modelValue="selectedDate" /> - <!-- ปุ่ม Make a queue พร้อม event listener --> - <MakequeueBtn @click="queueStore.makeQueue()" /> + <MakequeueBtn @click="openAddQueueDialog" /> </div> <!-- Gantt Chart UI --> @@ -21,31 +20,48 @@ </div> </div> - <!-- Rows: เครื่องจักรแต่ละตัว --> + <!-- Rows: แสดงเครื่องจักรจาก Machine Store --> <div class="rows"> - <div v-for="machine in machines" :key="machine.machineID" class="row" @dragover.prevent="onDragOver($event)" - @drop="onDrop($event, machine.machineID)"> + <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.machineID }} + {{ 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> + <div + v-for="hour in hours" + :key="'line-' + hour" + class="vertical-line" + :style="getLineStyle(hour)" + ></div> <!-- แสดง Queue ที่ตรงกับวันที่ (selectedDate), Page, และ Machine --> - <div v-for="item in filteredQueue(machine.machineID)" :key="item.queueID" class="order" - :class="{ 'faded': draggingQueue && draggingQueue.queueID === item.queueID }" - :style="getQueueStyle(item)"> - + <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)"> + <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> @@ -53,10 +69,9 @@ </div> </div> - <!-- Ghost Queue (ขณะลาก) --> + <!-- Ghost Queue (ขณะลาก/resize) --> <div v-if="ghostQueue" class="drag-ghost" :style="ghostStyle"> - {{ ghostQueue.orderID }} ({{ getTimeString(ghostQueue.startTime) }} - {{ getTimeString(ghostQueue.finishTime) - }}) + {{ ghostQueue.orderID }} ({{ getTimeString(ghostQueue.startTime) }} - {{ getTimeString(ghostQueue.finishTime) }}) </div> <v-divider :thickness="7"></v-divider> @@ -64,88 +79,81 @@ <!-- 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)"> + <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> - - <!-- ปุ่ม Add Queue อยู่ขวาสุด --> <div class="pagination-right"> <v-btn class="add-page-btn" @click="openAddQueueDialog" icon> <v-icon>mdi-plus</v-icon> </v-btn> </div> - - <!-- AddQueueDialog (สำหรับเพิ่ม/แก้ไข) --> - <AddQueueDialog :visible="showAddDialog" :queueItem="selectedQueueItem" @close="closeAddQueueDialog" - @add="handleAddQueue" @edit="handleEditQueue" /> + <AddQueueDialog + :visible="showAddDialog" + :queueItem="selectedQueueItem" + @close="closeAddQueueDialog" + @add="handleAddQueue" + @edit="handleEditQueue" + /> </div> - - <!-- Order Dialog (สำหรับดูรายละเอียดเพิ่มเติม) --> - <OrderDialog v-if="selectedQueueItem && !showAddDialog" :queueItem="selectedQueueItem" - :color="getQueueColor(selectedQueueItem.orderID)" @close="closeQueueDialog" @edit="openAddQueueDialogForEdit" - @delete="deleteQueueItem" /> + <OrderDialog + v-if="selectedQueueItem && !showAddDialog" + :queueItem="selectedQueueItem" + :color="getQueueColor(selectedQueueItem.orderID)" + @close="closeQueueDialog" + @edit="openAddQueueDialogForEdit" + @delete="deleteQueueItem" + /> </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 GanttCalendar from './GanttCalendar.vue'; import OrderDialog from './OrderDialog.vue'; import AddQueueDialog from './AddQueueDialog.vue'; import MakequeueBtn from './MakequeueBtn.vue'; import './ganttChart.css'; -import { useQueueStore } from '@/stores/queue'; -import type { QueueItem } from '@/types/QueueItem'; - -interface Machine { - machineID: string; - name: string; -} - - +// ใช้ store สำหรับ Queue และ Machine const queueStore = useQueueStore(); +const machineStore = useMachineStore(); -// NOTE: ถ้าต้องการโหลดข้อมูลทันทีที่เข้าหน้านี้ ให้ un-comment ด้านล่าง +// เมื่อเข้าหน้า ให้ดึงข้อมูลจาก backend onMounted(() => { - + machineStore.fetchMachines(); + queueStore.fetchQueues().then(() => { + console.log("📌 Queues Loaded:", queueStore.queues); + }); }); -// State +// State สำหรับ Gantt Chart const showAddDialog = ref(false); -const selectedDate = ref('1/1/2025'); +const selectedDate = ref('2025-01-01'); const startTime = ref('08:00'); const endTime = ref('17:00'); -const machines = ref<Machine[]>([ - { machineID: 'MC1', name: 'เครื่องเป่าขวด' }, - { machineID: 'MC2', name: 'เครื่องเป่าขวด2' }, - { machineID: 'MC3', name: 'เครื่องสวมฉลาก' }, - { machineID: 'MC4', name: 'เครื่องบรรจุ+แพ็ค' }, -]); - -// ดึง ref ของ QueueItem[] จาก Store -const Queues = computed(() => queueStore.Queues.value); - -// Drag/Drop & Resize -const draggingQueue = ref<QueueItem | null>(null); +const draggingQueue = ref<any | null>(null); const dragOffset = ref(0); -const resizingQueue = ref<QueueItem | null>(null); +const resizingQueue = ref<any | null>(null); const resizeDirection = ref<string | null>(null); -const ghostQueue = ref<QueueItem | null>(null); +const ghostQueue = ref<any | null>(null); const ghostStyle = reactive({ position: 'fixed', top: '0px', @@ -162,19 +170,42 @@ const ghostStyle = reactive({ 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, + pageNumber: q.page?.PageID || 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 = ref<number[]>([1, 2]); const currentPage = ref(1); const pageToShowDelete = ref<number | null>(null); -const selectedQueueItem = ref<QueueItem | null>(null); - +const selectedQueueItem = ref<any | null>(null); -watch(() => queueStore.Queues.value, (newVal) => { - console.log("🔄 Queues updated in GanttChart.vue:", newVal); -}, { deep: true }); +watch( + () => queueStore.queues, + (newVal) => { + console.log("🔄 Queues updated in GanttChart.vue:", newVal); + }, + { deep: true } +); - -// Computed const hours = computed(() => { const startHour = parseInt(startTime.value.split(':')[0]); const endHour = parseInt(endTime.value.split(':')[0]); @@ -182,78 +213,51 @@ const hours = computed(() => { }); // Methods -function handleUpdateQueue(newQueue: QueueItem[]) { - console.log("Received new queue:", newQueue); - // อัปเดตลงใน store - Queues.value = newQueue; -} - -function openQueueDialog(item: QueueItem) { +function openQueueDialog(item: any) { selectedQueueItem.value = { ...item }; } - function closeQueueDialog() { selectedQueueItem.value = null; } - function openAddQueueDialog() { selectedQueueItem.value = null; showAddDialog.value = true; } - -function deleteQueueItem(item: QueueItem) { - // ลบคิวใน store - Queues.value = Queues.value.filter(q => q.queueID !== item.queueID); +function deleteQueueItem(item: any) { + queueStore.removeQueue(item.QueueID); closeQueueDialog(); } - -function openAddQueueDialogForEdit(queueItem: QueueItem) { +function openAddQueueDialogForEdit(queueItem: any) { selectedQueueItem.value = { ...queueItem }; showAddDialog.value = true; } - function closeAddQueueDialog() { showAddDialog.value = false; selectedQueueItem.value = null; } - -function handleAddQueue(newQueueItem: Omit<QueueItem, 'queueID'>) { - const newQueueID = Queues.value.length + 1; - Queues.value.push({ queueID: newQueueID, ...newQueueItem }); +async function handleAddQueue(newQueueItem: any) { + await queueStore.addQueue(newQueueItem); showAddDialog.value = false; } - -function handleEditQueue(updatedQueueItem: QueueItem) { - const index = Queues.value.findIndex(q => q.queueID === updatedQueueItem.queueID); - if (index !== -1) { - Queues.value.splice(index, 1, { ...updatedQueueItem }); - } +async function handleEditQueue(updatedQueueItem: any) { + await queueStore.editQueue(updatedQueueItem.QueueID, updatedQueueItem); selectedQueueItem.value = null; showAddDialog.value = false; } -// Utility Functions 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: string) { - if (!queueStore.Queues?.values || typeof queueStore.Queues.values !== "function") { - console.warn("queueStore.Queues.values is not a function or is undefined"); - return []; - } - - return queueStore.Queues.filter(q => { +function filteredQueue(machineID: number) { + return formattedQueues.value.filter(q => { const queueDate = getDateString(q.startTime); - console.log(queueDate) return ( q.machineID === machineID && q.pageNumber === currentPage.value && @@ -262,59 +266,36 @@ function filteredQueue(machineID: string) { }); } - - - -const colorMap: Record<string, string> = {}; // Map เก็บสีของแต่ละ orderID - +const colorMap: Record<string, string> = {}; function getQueueColor(orderID: string | number): string { const key = String(orderID); - - // 🔴 ถ้าเป็น "ผลิตเผื่อ" → ใช้สีแดงพาสเทล - if (key === "ผลิตเผื่อ") { - return "#FFC1C1"; // Light Pastel Red - } - - // 💜 ถ้าเป็น "เปลี่ยนขนาด" → ใช้สีม่วงชมพูพาสเทล - if (key === "เปลี่ยนขนาด") { - return "#E1BEE7"; // Light Pastel Pink-Purple (Lavender) - } - - // ✅ ถ้ามีสีอยู่แล้ว ให้ใช้สีเดิม + 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: QueueItem) { +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 totalHours = hours.value.length; const leftPercent = ((startDecimal - timeToDecimal(startTime.value)) / totalHours) * 100; const widthPercent = ((endDecimal - startDecimal) / totalHours) * 100; - - const backgroundColor = getQueueColor(item.orderID); // ใช้ฟังก์ชันที่แก้ไขแล้ว - - console.log(`Queue ${item.queueID}: orderID=${item.orderID}, color=${backgroundColor}`); - + const orderID = item.orderID || "Unknown"; + const backgroundColor = getQueueColor(orderID); + console.log(`Queue ${item.QueueID}: orderID=${orderID}, color=${backgroundColor}`); return { left: `${leftPercent}%`, width: `${widthPercent}%`, - backgroundColor: backgroundColor, // ใช้สีแดงพาสเทลหรือสีธีมน้ำ + backgroundColor, color: '#333', borderRadius: '10px', textAlign: 'center', @@ -327,51 +308,39 @@ function getQueueStyle(item: QueueItem) { boxShadow: '0px 4px 6px rgba(0, 0, 0, 0.1)', }; } - - - - function getLineStyle(hour: number) { - const totalHours = hours.value.length; // นับจำนวนชั่วโมงที่แสดง (เช่น 10 ช่อง) - const leftPercent = (hour - timeToDecimal(startTime.value)) / totalHours * 100; - + const totalHours = hours.value.length; + const leftPercent = ((hour - timeToDecimal(startTime.value)) / totalHours) * 100; return { left: `${leftPercent}%` }; } - function timeToDecimal(timeStr: string): number { - const [hours, minutes] = timeStr.split(':').map(Number); - return hours + minutes / 60; + const [h, m] = timeStr.split(':').map(Number); + return h + m / 60; } - function decimalToTime(decimal: number): string { - const hours = Math.floor(decimal); - const minutes = Math.round((decimal - hours) * 60); - return (hours < 10 ? '0' + hours : hours) + ':' + (minutes < 10 ? '0' + minutes : minutes); + 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: QueueItem) { +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.orderID); - + 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; @@ -389,7 +358,6 @@ function onDragOverGlobal(event: DragEvent) { 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; @@ -407,11 +375,13 @@ function onDragOverGlobal(event: DragEvent) { 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; @@ -425,9 +395,10 @@ function onDragOverGlobal(event: DragEvent) { }); if (closestRow) { - const labelEl = closestRow.querySelector('.machine-label') as HTMLElement; - if (labelEl) { - ghostQueue.value!.machineID = labelEl.textContent?.trim() || ghostQueue.value!.machineID; + // ใช้ dataset เพื่อดึง machine id + const machineIDString = closestRow.dataset.machineId; + if (machineIDString) { + ghostQueue.value.machine = { MachineID: parseInt(machineIDString, 10) }; } ghostStyle.top = closestRow.getBoundingClientRect().top + 'px'; } @@ -442,38 +413,45 @@ function onDragEndGlobal() { draggingQueue.value = null; document.removeEventListener('dragend', onDragEndGlobal); } - -function onDragEnd(event: DragEvent, item: QueueItem) { +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; - item.machineID = ghostQueue.value.machineID; - document.removeEventListener('dragover', onDragOverGlobal); + // ใช้ ghostQueue.value.machine.MachineID แทนการอ่าน machineID จาก ghostQueue.value + const newMachineID = ghostQueue.value.machine?.MachineID || item.machineID; + await queueStore.updateQueue(item.QueueID, { + startTime: item.startTime, + finishTime: item.finishTime, + machine: { MachineID: newMachineID, name: '' } + }); ghostQueue.value = null; ghostStyle.display = 'none'; draggingQueue.value = null; } - -function onDrop(event: DragEvent, newMachine: string) { +async function onDrop(event: DragEvent, newMachineID: number) { event.preventDefault(); if (draggingQueue.value && ghostQueue.value) { draggingQueue.value.startTime = ghostQueue.value.startTime; draggingQueue.value.finishTime = ghostQueue.value.finishTime; - draggingQueue.value.machineID = newMachine; + draggingQueue.value.machine = { MachineID: newMachineID }; + await queueStore.updateQueue(draggingQueue.value.QueueID, { + startTime: draggingQueue.value.startTime, + finishTime: draggingQueue.value.finishTime, + machine: { MachineID: newMachineID, name: '' } + }); } ghostQueue.value = null; ghostStyle.display = 'none'; draggingQueue.value = null; } - -function onResizeStart(event: MouseEvent, item: QueueItem, direction: string) { +function onResizeStart(event: MouseEvent, item: any, direction: string) { event.preventDefault(); 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; @@ -481,10 +459,8 @@ function onResizing(event: MouseEvent) { 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)); @@ -492,25 +468,41 @@ function onResizing(event: MouseEvent) { 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); - resizingQueue.value.startTime = newStartStr; + // อัปเดต ghostQueue ให้แสดงผลแบบ real-time + ghostQueue.value.startTime = newStartStr; } else if (resizeDirection.value === 'right') { if (newTimeDecimal <= startDec) return; const newEndStr = datePart + ' ' + decimalToTime(newTimeDecimal); - resizingQueue.value.finishTime = newEndStr; + ghostQueue.value.finishTime = newEndStr; } + // อัปเดต ghostStyle ตาม ghostQueue ใหม่ + const start = getTimeString(ghostQueue.value.startTime); + const end = getTimeString(ghostQueue.value.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; + ghostStyle.left = `${leftPercent}%`; + ghostStyle.width = `${widthPercent}%`; } - -function onResizeEnd() { +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 function addPage() { if (pages.value.length < 10) { @@ -520,12 +512,10 @@ function addPage() { alert("Maximum of 10 pages allowed."); } } - function onPageRightClick(page: number, event: MouseEvent) { event.preventDefault(); pageToShowDelete.value = page; } - function deletePage(page: number) { const index = pages.value.indexOf(page); if (index !== -1) { @@ -536,19 +526,15 @@ function deletePage(page: number) { } pageToShowDelete.value = null; } - function onDocumentClick(event: MouseEvent) { const target = event.target as HTMLElement; if (!target.closest('.page-btn')) { pageToShowDelete.value = null; } } - -// Lifecycle Hooks onMounted(() => { document.addEventListener('click', onDocumentClick); }); - onBeforeUnmount(() => { document.removeEventListener('click', onDocumentClick); }); -- GitLab