hrms-user/src/components/DialogDebug.vue

375 lines
12 KiB
Vue
Raw Normal View History

2026-01-28 13:27:14 +07:00
<script setup lang="ts">
import { computed, reactive, ref } from "vue";
import { useQuasar } from "quasar";
import axios from "axios";
import { storeToRefs } from "pinia";
import { useCounterMixin } from "@/stores/mixin";
import { usePositionKeycloakStore } from "@/stores/positionKeycloak";
import { useDataStore } from "@/stores/data";
import http from "@/plugins/http";
import config from "@/app.config";
import type { MenuMainList } from "@/modules/01_dashboard/interface/Main";
import DialogHeader from "@/components/DialogHeader.vue";
2026-03-25 11:05:40 +07:00
import FooterContact from "@/components/FooterContact.vue";
2026-01-28 13:27:14 +07:00
const $q = useQuasar();
const store = usePositionKeycloakStore();
const dataStore = useDataStore();
const { findOrgName } = store;
2026-01-30 16:03:09 +07:00
const { dataprofilePosition, menuList } = storeToRefs(dataStore);
2026-01-28 13:27:14 +07:00
const { dialogConfirm, showLoader, hideLoader, messageError, success } =
useCounterMixin();
const modal = defineModel<boolean>("modal", {
default: false,
});
interface OptionSelect {
label: string;
value: string;
}
const title = computed(() => "แจ้งปัญหาการใช้งานระบบ");
const orgName = computed(() => findOrgName(dataprofilePosition.value) || "");
const optionData = computed(() => {
2026-01-30 16:03:09 +07:00
return menuList.value.map((menu: MenuMainList) => ({
2026-01-28 13:27:14 +07:00
label: menu.title,
value: menu.title,
}));
});
const optionsMenu = ref(optionData.value);
const formData = reactive({
title: "",
description: "",
system: "user",
fileAttachments: [] as File[],
menu: "",
2026-02-04 13:02:18 +07:00
email: "",
phone: "",
2026-01-28 13:27:14 +07:00
});
/** ฟังก์ชันบันทึกข้อมูล */
function onSubmit() {
dialogConfirm($q, async () => {
try {
showLoader();
const payload = {
title: formData.title,
description: formData.description,
system: formData.system,
menu: formData.menu,
org: orgName.value,
2026-02-04 13:02:18 +07:00
email: formData.email,
phone: formData.phone,
2026-01-28 13:27:14 +07:00
};
const res = await http.post(config.API.orgIssues, payload);
const issueCode = res.data.result.codeIssue;
await uploadProfile(issueCode);
success($q, "บันทึกข้อมูลเรียบร้อย");
onClose();
} catch (error) {
messageError($q, error);
} finally {
hideLoader();
}
});
}
/**
* งกนเพมไฟล
* @param files ไฟลองการเพ
*/
async function onAddfile(files: any) {
files.forEach((file: any) => {
formData.fileAttachments.push(file);
});
}
/**
* งกนลบไฟล
* @param files ไฟลองการลบ
*/
async function onRemoveFile(files: any) {
files.forEach((file: any) => {
const index = formData.fileAttachments.findIndex(
(x: any) => x.__key == file.__key,
);
if (index > -1) {
formData.fileAttachments.splice(index, 1);
}
});
}
/**
* งกนสราง url ปโหลดไฟล
* @param code รห issue
*/
async function uploadProfile(code: string) {
if (formData.fileAttachments.length === 0) {
return;
}
try {
const fileName = formData.fileAttachments.map((file) => ({
fileName: file.name,
}));
const res = await http.post(
config.API.file("issueAttachments", formData.system, code),
{
replace: false,
fileList: fileName,
},
);
for (const file of formData.fileAttachments) {
const fileInfo = res.data[file.name];
if (fileInfo && fileInfo.uploadUrl) {
await uploadFileDoc(fileInfo.uploadUrl, file);
}
}
} catch (e) {
messageError($q, e);
}
}
/**
* งกนอปโหลดไฟลเอกสาร
* @param uploadUrl งกปโหลดไฟล
* @param file ไฟลองการอปโหลด
*/
async function uploadFileDoc(uploadUrl: string, file: any) {
try {
await axios.put(uploadUrl, file, {
headers: {
"Content-Type": file.type,
},
});
} catch (e) {
messageError($q, e);
}
}
/**
* งกนกรองขอมลใน select
* @param val าทกรอง
* @param update งกนอปเดตคาหลงกรอง
*/
function filterSelector(val: string, update: Function) {
update(() => {
if (!val) {
optionsMenu.value = optionData.value;
return;
}
optionsMenu.value = optionData.value.filter((item: OptionSelect) =>
item.label.toLowerCase().includes(val.toLowerCase()),
);
});
}
/** ฟังก์ชันปิด dialog และรีเซ็ตข้อมูล */
function onClose() {
modal.value = false;
formData.menu = "";
formData.title = "";
formData.description = "";
formData.fileAttachments = [];
2026-02-04 13:02:18 +07:00
formData.email = "";
formData.phone = "";
2026-01-28 13:27:14 +07:00
}
</script>
<template>
<q-dialog v-model="modal" persistent>
<q-card style="width: 700px; max-width: 80vw">
<q-form greedy @submit.prevent @validation-success="onSubmit">
<DialogHeader :tittle="title" :close="onClose" />
<q-separator />
<q-card-section>
<div class="row col q-col-gutter-md">
<div class="col-12">
<q-select
dense
outlined
label="ระบบ"
v-model="formData.menu"
:options="optionsMenu"
class="inputgreen"
:rules="[ (val: string) => !!val || 'กรุณาเลือกระบบ' ]"
hide-bottom-space
emit-value
map-options
use-input
@filter="(inputValue: string,
doneFn: Function) => filterSelector(inputValue, doneFn,
)"
/>
</div>
<div class="col-12">
<q-input
dense
outlined
label="หัวข้อปัญหา"
v-model="formData.title"
class="inputgreen"
:rules="[ (val: string) => !!val || 'กรุณากรอกหัวข้อปัญหา' ]"
hide-bottom-space
/>
</div>
<div class="col-12">
<q-input
dense
outlined
type="textarea"
label="รายละเอียดปัญหา"
v-model="formData.description"
class="inputgreen"
:rules="[ (val: string) => !!val || 'กรุณากรอกรายละเอียดปัญหา' ]"
hide-bottom-space
/>
</div>
<div class="col-12">
<q-uploader
color="gray"
type="file"
flat
ref="uploader"
class="full-width"
text-color="dark"
accept=".jpg,.png,.pdf,.csv,.doc"
bordered
label="[ไฟล์ jpg,png,pdf,csv,doc ขนาดไม่เกิน 5MB]"
multiple
max-file-size="5000000"
@added="onAddfile"
@removed="onRemoveFile"
>
<template v-slot:header="scope">
<div
class="row no-wrap items-center q-pa-sm q-gutter-xs text-white"
>
<q-btn
v-if="scope.queuedFiles.length > 0"
icon="clear_all"
@click="scope.removeQueuedFiles"
round
dense
flat
>
<q-tooltip>ลบทงหมด</q-tooltip>
</q-btn>
<q-btn
v-if="scope.uploadedFiles.length > 0"
icon="done_all"
@click="scope.removeUploadedFiles"
round
dense
flat
>
<q-tooltip>ลบไฟลปโหลด</q-tooltip>
</q-btn>
<q-spinner
v-if="scope.isUploading"
class="q-uploader__spinner"
/>
<div class="col">
<div class="q-uploader__title">
{{ "[ไฟล์ jpg,png,pdf,csv,doc ขนาดไม่เกิน 5MB]" }}
</div>
<div class="q-uploader__subtitle">
{{ scope.uploadSizeLabel }}
/
{{ scope.uploadProgressLabel }}
</div>
</div>
<q-btn
v-if="scope.canAddFiles"
type="a"
icon="add_box"
@click="scope.pickFiles"
round
dense
flat
>
<q-uploader-add-trigger />
<q-tooltip>เลอกไฟล</q-tooltip>
</q-btn>
<q-btn
v-if="scope.isUploading"
icon="clear"
@click="scope.abort"
round
dense
flat
>
<q-tooltip>ยกเลกการอปโหลด</q-tooltip>
</q-btn>
</div>
</template>
</q-uploader>
</div>
2026-02-04 13:02:18 +07:00
<div class="col-12">
<div class="text-caption text-grey-7 q-mb-sm">
แลระบบจะตดตอกลบผานทางอเมลทานระบ กรณาตรวจสอบอเมลของทานเปนระยะ
</div>
2026-02-04 13:02:18 +07:00
<div class="row col-12 q-col-gutter-sm">
<div class="col-xs-12 col-md-6 col-lg-6">
<q-input
dense
outlined
label="อีเมลติดต่อกลับ"
v-model="formData.email"
class="inputgreen"
hide-bottom-space
:rules="[
(val: string) => !!val || 'กรุณากรอกที่อยู่อีเมล',
(val: string) => {
const emailPattern = /^[^\s@]+@[^\s@]+\.[^\s@]+$/;
return emailPattern.test(val) || 'กรุณากรอกที่อยู่อีเมลในรูปแบบที่ถูกต้อง';
}
2026-02-04 13:02:18 +07:00
]"
/>
</div>
<div class="col-xs-12 col-md-6 col-lg-6">
<q-input
dense
outlined
label="เบอร์โทรติดต่อกลับ"
v-model="formData.phone"
class="inputgreen"
hide-bottom-space
/>
</div>
</div>
</div>
2026-01-28 13:27:14 +07:00
</div>
</q-card-section>
<q-separator />
2026-03-25 10:39:13 +07:00
<q-card-actions class="q-px-md items-center">
2026-03-25 11:05:40 +07:00
<FooterContact />
2026-03-25 10:39:13 +07:00
<q-space />
2026-01-28 13:27:14 +07:00
<q-btn
type="submit"
for="#submitForm"
class="q-px-md items-center"
color="public"
label="บันทึก"
>
<q-tooltip>นทกขอม</q-tooltip>
</q-btn>
</q-card-actions>
</q-form>
</q-card>
</q-dialog>
</template>
<style scoped></style>