hrms-mgt/src/modules/09_leave/components/01_RoundTime/DialogForm.vue

493 lines
16 KiB
Vue
Raw Normal View History

<script setup lang="ts">
import { ref, reactive, watch } from "vue";
2024-09-18 17:26:53 +07:00
import { useQuasar } from "quasar";
import http from "@/plugins/http";
import config from "@/app.config";
2024-09-18 17:26:53 +07:00
import { useCounterMixin } from "@/stores/mixin";
2023-10-31 12:09:24 +07:00
import type {
FormData,
2023-10-31 12:09:24 +07:00
MyObjectRoundRef,
} from "@/modules/09_leave/interface/response/round";
import DialogHeader from "@/components/DialogHeader.vue";
2023-12-15 09:58:02 +07:00
2023-10-31 12:09:24 +07:00
const $q = useQuasar();
2023-12-15 09:58:02 +07:00
2023-10-31 12:09:24 +07:00
const mixin = useCounterMixin();
const { dialogConfirm, success, messageError, showLoader, hideLoader } = mixin;
const isRead = defineModel<boolean>("isRead", { required: true });
/** propsData จาก RoundMain */
const props = defineProps({
modal: {
type: Boolean,
require: true,
},
editCheck: {
type: String,
require: true,
},
fetchData: {
type: Function,
require: true,
},
closeDialog: {
type: Function,
require: true,
},
detailData: {
type: Object,
require: false,
},
});
2023-10-31 12:09:24 +07:00
/** Ref INPUT เวลา*/
2023-10-31 12:09:24 +07:00
const amRef = ref<Object | null>(null);
const amOutRef = ref<Object | null>(null);
const pmRef = ref<Object | null>(null);
const pmOutRef = ref<Object | null>(null);
const objectRound: MyObjectRoundRef = {
startTimeMorning: amRef,
endTimeMorning: amOutRef,
startTimeAfternoon: pmRef,
endTimeAfternoon: pmOutRef,
};
/** Form ข้อมูล */
const formData = reactive<FormData>({
startTimeMorning: "",
endTimeMorning: "",
startTimeAfternoon: "",
endTimeAfternoon: "",
description: "",
2023-10-31 12:09:24 +07:00
isDefault: false,
isActive: false,
});
/**FunctionCheckbox แก้ไขสถานะ */
function checkDefault() {
if (formData.isDefault === true) {
formData.isActive = true;
}
}
/** Function validateForm*/
2023-10-31 12:09:24 +07:00
function validateForm() {
const hasError = [];
for (const key in objectRound) {
if (Object.prototype.hasOwnProperty.call(objectRound, key)) {
const property = objectRound[key];
if (property.value && typeof property.value.validate === "function") {
const isValid = property.value.validate();
hasError.push(isValid);
}
}
}
if (hasError.every((result) => result === true)) {
onSubmit();
}
}
/** Function ยืนยันการบันทึกข้อมูล*/
2023-10-31 12:09:24 +07:00
function onSubmit() {
dialogConfirm(
$q,
async () => {
const dataId = props.detailData?.id;
props.editCheck === "add" ? postRoundDuty() : putRoundDuty(dataId);
2023-10-31 12:09:24 +07:00
},
"ยืนยันการบันทึกข้อมูล",
"ต้องการยืนยันการบันทึกข้อมูลนี้หรือไม่ ?"
);
}
/** Function สร้างรอบการปฏิบัติงาน*/
async function postRoundDuty() {
showLoader();
await http
.post(config.API.roundDutytime(), formData)
.then(async () => {
await props.fetchData?.();
await props.closeDialog?.();
success($q, "บันทึกข้อมูล");
})
.catch((err) => {
messageError($q, err);
})
.finally(() => {
hideLoader();
});
}
/** Functiom แก้ไข้รอบการปฏิบัติงาน*/
async function putRoundDuty(id: string) {
showLoader();
const data = {
description: formData.description,
isDefault: formData.isDefault,
isActive: formData.isActive,
};
await http
.put(config.API.roundDutytimeByid(id), data)
.then(async () => {
await props.fetchData?.();
await props.closeDialog?.();
success($q, "บันทึกข้อมูล");
})
.catch((err) => {
messageError($q, err);
})
.finally(() => {
hideLoader();
});
}
//** Function ปิด Popup */
function close() {
props.closeDialog?.();
isRead.value = false;
}
/***/
watch(
() => props.modal,
2023-12-15 09:58:02 +07:00
() => {
2023-10-31 12:09:24 +07:00
if (props.editCheck === "add") {
formData.startTimeMorning = "";
formData.endTimeMorning = "";
formData.startTimeAfternoon = "";
formData.endTimeAfternoon = "";
formData.description = "";
formData.isDefault = false;
formData.isActive = false;
} else if (props.editCheck === "edit") {
2023-10-31 12:09:24 +07:00
if (props.detailData) {
formData.startTimeMorning = props.detailData.startTimeMorning;
formData.endTimeMorning = props.detailData.endTimeMorning;
formData.startTimeAfternoon = props.detailData.startTimeAfternoon;
formData.endTimeAfternoon = props.detailData.endTimeAfternoon;
formData.description = props.detailData.description;
formData.isDefault = props.detailData.isDefault;
formData.isActive = props.detailData.isActive;
}
}
}
);
</script>
<template>
<q-dialog v-model="props.modal" persistent>
<q-card style="min-width: 350px" class="bg-grey-11">
2023-10-31 12:09:24 +07:00
<form @submit.prevent="validateForm">
<DialogHeader
:tittle="
props.editCheck === 'add'
? 'เพิ่มรอบการปฏิบัติงาน'
: props.detailData
? isRead
? `รายละเอียดรอบ ${props.detailData.round}`
: `แก้ไขรอบ ${props.detailData.round}`
2023-10-31 12:09:24 +07:00
: ''
"
:close="close"
/>
<q-separator color="grey-4" />
2023-12-12 12:07:35 +07:00
<q-card-section style="max-height: 80vh" class="scroll q-pa-none">
2023-10-31 12:09:24 +07:00
<div
class="col q-ma-md q-pa-sm bg-white border_custom text-weight-medium"
>
<p style="color: #06884d; font-size: 16px">ครงเช</p>
<div class="row justify-between q-my-sm items-start">
<p class="q-ma-none mt">เวลาเขางาน</p>
<q-input
ref="amRef"
class="inputgreen"
:readonly="props.editCheck === 'edit' || isRead"
2023-10-31 12:09:24 +07:00
:outlined="props.editCheck === 'add'"
dense
lazy-rules
borderless
v-model="formData.startTimeMorning"
2023-10-31 12:09:24 +07:00
:rules="[
2024-09-25 16:44:07 +07:00
(val:string) => !!val || 'กรุณากรอกเวลาเข้างาน',
(val:string) => {
if (val && formData.endTimeMorning) {
if (val > formData.endTimeMorning) {
return 'ต้องน้อยกว่าเวลาออกงาน';
2023-10-31 12:09:24 +07:00
}
if (
val >= formData.startTimeAfternoon &&
val <= formData.endTimeAfternoon
) {
return 'ช่วงเวลาทับซ้อนกับช่วงบ่าย';
2023-10-31 12:09:24 +07:00
}
if (val === formData.endTimeAfternoon) {
return 'เวลาเข้างานช่วงเช้าต้องไม่ซ้ำกับออกงานเช้า';
2023-10-31 12:09:24 +07:00
}
}
return true;
},
]"
hide-bottom-space
type="time"
style="width: 140px"
/>
</div>
<q-separator inset />
<div class="row items-start q-my-sm justify-between">
<p class="q-ma-none mt">เวลาออกงาน</p>
<q-input
ref="amOutRef"
class="inputgreen"
:readonly="props.editCheck === 'edit' || isRead"
2023-10-31 12:09:24 +07:00
:outlined="props.editCheck === 'add'"
dense
v-model="formData.endTimeMorning"
2023-10-31 12:09:24 +07:00
lazy-rules
borderless
:rules="[
2024-09-25 16:44:07 +07:00
(val:string) => !!val || 'กรุณากรอกเวลาออกงาน',
(val:string) => {
if (val && formData.startTimeMorning) {
if (val < formData.startTimeMorning) {
return 'ต้องมากกว่าเวลาเข้างาน';
2023-10-31 12:09:24 +07:00
}
if (
val >= formData.startTimeAfternoon &&
val <= formData.endTimeAfternoon
) {
return 'ช่วงเวลาทับซ้อนกับช่วงบ่าย';
2023-10-31 12:09:24 +07:00
}
if (val === formData.startTimeMorning) {
return 'เวลาออกงานช่วงเช้าต้องไม่ซ้ำกับเข้างานเช้า';
2023-10-31 12:09:24 +07:00
}
}
return true;
},
]"
hide-bottom-space
type="time"
style="width: 140px"
/>
</div>
</div>
<div
class="col q-ma-md q-pa-sm bg-white border_custom text-weight-medium"
>
<p style="color: #06884d; font-size: 16px">ครงบาย</p>
<div class="row justify-between q-my-sm items-start">
<p class="q-ma-none mt">เวลาเขางาน</p>
<q-input
ref="pmRef"
:rules="[
2024-09-25 16:44:07 +07:00
(val:string) => !!val || 'กรุณากรอกเวลาเข้างาน',
(val:string) => {
if (val && formData.endTimeAfternoon) {
if (val > formData.endTimeAfternoon) {
return 'ต้องน้อยกว่าเวลาออกงาน';
2023-10-31 12:09:24 +07:00
}
if (
val >= formData.startTimeMorning &&
val <= formData.endTimeMorning
) {
return 'ช่วงเวลาทับซ้อนกับช่วงเช้า';
2023-10-31 12:09:24 +07:00
}
if (val === formData.endTimeAfternoon) {
return 'เวลาเข้างานช่วงบ่ายต้องไม่ซ้ำกับออกงานช่วงบ่าย';
2023-10-31 12:09:24 +07:00
}
}
return true;
},
]"
class="inputgreen"
:readonly="props.editCheck === 'edit' || isRead"
2023-10-31 12:09:24 +07:00
:outlined="props.editCheck === 'add'"
dense
lazy-rules
borderless
v-model="formData.startTimeAfternoon"
2023-10-31 12:09:24 +07:00
hide-bottom-space
type="time"
style="width: 140px"
/>
</div>
<q-separator inset />
<div class="row items-start q-my-sm justify-between">
<p class="q-ma-none mt">เวลาออกงาน</p>
<q-input
ref="pmOutRef"
:rules="[
2024-09-25 16:44:07 +07:00
(val:string) => !!val || 'กรุณากรอกเวลาออกงาน',
(val:string) => {
if (val && formData.startTimeAfternoon) {
if (val < formData.startTimeAfternoon) {
return 'ต้องมากกว่าเวลาเข้างาน';
2023-10-31 12:09:24 +07:00
}
if (
val >= formData.startTimeMorning &&
val <= formData.endTimeMorning
) {
return 'ช่วงเวลาทับซ้อนกับช่วงเช้า';
2023-10-31 12:09:24 +07:00
}
if (val === formData.startTimeAfternoon) {
return 'เวลาออกงานช่วงบ่ายต้องไม่ซ้ำกับเข้างานช่วงบ่าย';
2023-10-31 12:09:24 +07:00
}
}
return true;
},
]"
class="inputgreen"
:readonly="props.editCheck === 'edit' || isRead"
2023-10-31 12:09:24 +07:00
:outlined="props.editCheck === 'add'"
dense
v-model="formData.endTimeAfternoon"
2023-10-31 12:09:24 +07:00
lazy-rules
borderless
hide-bottom-space
type="time"
style="width: 140px"
/>
</div>
</div>
<q-input
2023-10-31 12:09:24 +07:00
class="col-12 bg-white q-ma-md"
outlined
stack-label
:readonly="isRead"
v-model="formData.description"
2023-10-31 12:09:24 +07:00
label="คำอธิบาย"
hide-bottom-space
2023-10-31 12:09:24 +07:00
type="textarea"
></q-input>
<div
class="col q-ma-md q-pa-sm bg-white border_custom text-weight-medium"
>
<div class="row items-center q-my-sm justify-between">
<p class="q-ma-none">เวลา Default</p>
2024-09-03 11:28:01 +07:00
<label
:class="`toggle-control ${isRead ? 'no-pointer-events' : ''}`"
>
<input
:readonly="isRead"
type="checkbox"
v-model="formData.isDefault"
@change="checkDefault()"
/>
2023-10-31 12:09:24 +07:00
<span class="control"></span>
</label>
</div>
</div>
<div
class="col q-ma-md q-pa-sm bg-white border_custom text-weight-medium"
>
2023-10-31 12:09:24 +07:00
<div class="row items-center q-my-sm justify-between">
<p class="q-ma-none">สถานะการใชงาน</p>
2024-09-03 11:28:01 +07:00
<label
:class="`toggle-control ${isRead ? 'no-pointer-events' : ''}`"
>
2023-12-21 09:33:00 +07:00
<input
type="checkbox"
v-model="formData.isActive"
:disabled="formData.isDefault"
/>
2023-10-31 12:09:24 +07:00
<span class="control"></span>
</label>
</div>
</div>
</q-card-section>
2024-09-03 11:28:01 +07:00
<q-separator color="grey-4" v-if="!isRead" />
<div class="q-pa-sm" v-if="!isRead">
2023-10-31 12:09:24 +07:00
<div class="row justify-end">
<q-btn
id="onSubmit"
type="submit"
2023-12-12 12:07:35 +07:00
dense
unelevated
label="บันทึก"
color="public"
class="q-px-md"
2023-10-31 12:09:24 +07:00
>
2023-12-12 12:07:35 +07:00
<q-tooltip>นท</q-tooltip>
</q-btn>
2023-10-31 12:09:24 +07:00
</div>
</div>
2023-10-31 12:09:24 +07:00
</form>
</q-card>
</q-dialog>
</template>
<style scoped lang="scss">
2024-09-25 17:13:44 +07:00
@use "sass:math";
2023-10-31 12:09:24 +07:00
.mt {
margin-top: 10px;
}
.border_custom {
border-radius: 6px !important;
border: 1px solid #e1e1e1;
}
$toggle-background-color-on: #06884d;
$toggle-background-color-off: darkgray;
$toggle-control-color: white;
$toggle-width: 40px;
$toggle-height: 25px;
$toggle-gutter: 3px;
$toggle-radius: 50%;
$toggle-control-speed: 0.15s;
$toggle-control-ease: ease-in;
// These are our computed variables
// change at your own risk.
2024-09-25 17:13:44 +07:00
$toggle-radius: math.div($toggle-height, 2);
$toggle-control-size: $toggle-height - ($toggle-gutter * 2);
.toggle-control {
display: block;
position: relative;
padding-left: $toggle-width;
margin-bottom: 12px;
cursor: pointer;
font-size: 22px;
user-select: none;
input {
position: absolute;
opacity: 0;
cursor: pointer;
height: 0;
width: 0;
}
input:checked ~ .control {
background-color: $toggle-background-color-on;
&:after {
left: $toggle-width - $toggle-control-size - $toggle-gutter;
}
}
.control {
position: absolute;
top: -7px;
left: -15px;
height: $toggle-height;
width: $toggle-width;
border-radius: $toggle-radius;
background-color: $toggle-background-color-off;
transition: background-color $toggle-control-speed $toggle-control-ease;
&:after {
content: "";
position: absolute;
left: $toggle-gutter;
top: $toggle-gutter;
width: $toggle-control-size;
height: $toggle-control-size;
border-radius: $toggle-radius;
background: $toggle-control-color;
transition: left $toggle-control-speed $toggle-control-ease;
}
}
}
</style>