Compare commits

..

No commits in common. "dev" and "v1.1.41" have entirely different histories.
dev ... v1.1.41

145 changed files with 790 additions and 3078 deletions

View file

@ -30,13 +30,11 @@
"axios": "^1.6.7", "axios": "^1.6.7",
"bma-org-chart": "^0.0.8", "bma-org-chart": "^0.0.8",
"esri-loader": "^3.7.0", "esri-loader": "^3.7.0",
"exceljs": "^4.4.0",
"html-to-image": "^1.11.13", "html-to-image": "^1.11.13",
"keycloak-js": "^20.0.2", "keycloak-js": "^20.0.2",
"moment": "^2.29.4", "moment": "^2.29.4",
"pdf-lib": "^1.17.1", "pdf-lib": "^1.17.1",
"pinia": "^2.0.29", "pinia": "^2.0.29",
"pinia-plugin-persistedstate": "^3.2.3",
"quasar": "^2.11.1", "quasar": "^2.11.1",
"socket.io-client": "^4.7.4", "socket.io-client": "^4.7.4",
"structure-chart": "^0.0.9", "structure-chart": "^0.0.9",

View file

@ -192,9 +192,6 @@ export default {
`${orgProfile}/keycloak/permissionProfile/${rootId}`, `${orgProfile}/keycloak/permissionProfile/${rootId}`,
profileidPosition: (type: string) => profileidPosition: (type: string) =>
`${orgProfile}${type}/profileid/position`, `${orgProfile}${type}/profileid/position`,
uploadProfile: (type: string, id: string) =>
`${organization}/upload/${type}-profileSalaryTemp/${id}`,
sortOrderByDate: `${orgProfile}/salaryTemp/sort-order`,
workflowCommanderOperate: `${workflow}/commander/operate`, workflowCommanderOperate: `${workflow}/commander/operate`,
workflowCommanderSign: `${workflow}/commander/sign`, workflowCommanderSign: `${workflow}/commander/sign`,

View file

@ -96,8 +96,4 @@ export default {
applicationFormPDF: (candidateId: string) => applicationFormPDF: (candidateId: string) =>
`${env.API_URI}/placement/candidate/pdf/${candidateId}`, `${env.API_URI}/placement/candidate/pdf/${candidateId}`,
downloadCandidateExam: (id: string) =>
`${periodExam}download/candidate-exam/${id}`,
downloadPassExam: (id: string) => `${periodExam}download/pass-exam/${id}`,
}; };

View file

@ -23,7 +23,6 @@ export default {
uploadCandidates: (id: string) => `${recruit}candidate/${id}`, uploadCandidates: (id: string) => `${recruit}candidate/${id}`,
uploadResult: (id: string) => `${recruit}result/${id}`, uploadResult: (id: string) => `${recruit}result/${id}`,
getImportHistory: (id: string) => `${recruit}history/${id}`, getImportHistory: (id: string) => `${recruit}history/${id}`,
getImportStatus: (jobId: string) => `${recruit}import/status/${jobId}`,
//upload //upload
periodRecruitDoc: (examId: string) => `${recruit}doc/${examId}`, periodRecruitDoc: (examId: string) => `${recruit}doc/${examId}`,

View file

@ -8,7 +8,6 @@ import config from "@/app.config";
import type { PropType } from "vue"; import type { PropType } from "vue";
import type { FormProfile } from "@/interface/main"; import type { FormProfile } from "@/interface/main";
import type { DataProfile } from "@/modules/05_placement/interface/index/Main"; import type { DataProfile } from "@/modules/05_placement/interface/index/Main";
import avatarMain from "@/assets/avatar_user.jpg";
/** importComponents*/ /** importComponents*/
import PopupPersonal from "@/components/Dialogs/PopupPersonalNew.vue"; import PopupPersonal from "@/components/Dialogs/PopupPersonalNew.vue";
@ -44,9 +43,9 @@ async function fetchDataProfile(data: DataProfile) {
profile.avatar = data?.avatar ? data.avatar : ""; profile.avatar = data?.avatar ? data.avatar : "";
} }
profile.id = data.profileId; profile.id = data.profileId;
profile.fullName = `${data.rank ? data.rank : data.prefix ?? ""}${ profile.fullName = `${data.prefix ?? ""}${data.firstName ?? ""} ${
data.firstName ?? "" data.lastName ?? ""
} ${data.lastName ?? ""} `; } `;
if (data["posTypeNameOld"] !== undefined) { if (data["posTypeNameOld"] !== undefined) {
profile.position = profile.position =
@ -92,11 +91,8 @@ function fetchProfile(id: string, name: string) {
if (profile.avatar === "") { if (profile.avatar === "") {
http http
.get(config.API.fileByFile("ทะเบียนประวัติ", "โปรไฟล์", id, `${name}`)) .get(config.API.fileByFile("ทะเบียนประวัติ", "โปรไฟล์", id, `${name}`))
.then((res) => { .then(async (res) => {
profile.avatar = res.data.downloadUrl; profile.avatar = res.data.downloadUrl;
})
.catch(() => {
profile.avatar = avatarMain;
}); });
} }
} }

View file

@ -9,8 +9,6 @@
round round
dense dense
@click="close" @click="close"
@keydown.enter.prevent
@keydown.space.prevent
style="color: #ff8080; background-color: #ffdede" style="color: #ff8080; background-color: #ffdede"
/> />
</q-toolbar> </q-toolbar>

View file

@ -332,7 +332,6 @@ watch(
outlined outlined
dense dense
label="คำค้น" label="คำค้น"
@keydown.enter.prevent="searchInput()"
> >
<template v-slot:after> <template v-slot:after>
<q-btn <q-btn

View file

@ -354,11 +354,6 @@ function onClose() {
<div class="col-12"> <div class="col-12">
<div class="row col-12 q-col-gutter-sm"> <div class="row col-12 q-col-gutter-sm">
<div class="col-12">
<div class="text-caption text-grey-8">
แลระบบจะตดตอกลบผานทางอเมลทานระบ กรณาตรวจสอบอเมลของทานเปนระยะ
</div>
</div>
<div class="col-xs-12 col-md-6 col-lg-6"> <div class="col-xs-12 col-md-6 col-lg-6">
<q-input <q-input
dense dense
@ -368,11 +363,10 @@ function onClose() {
class="inputgreen" class="inputgreen"
hide-bottom-space hide-bottom-space
:rules="[ :rules="[
(val: string) => !!val || 'กรุณากรอกที่อยู่อีเมล', () =>
(val: string) => { !!formData.email ||
const emailPattern = /^[^\s@]+@[^\s@]+\.[^\s@]+$/; !!formData.phone ||
return emailPattern.test(val) || 'กรุณากรอกที่อยู่อีเมลในรูปแบบที่ถูกต้อง'; 'กรุณากรอกอีเมลหรือเบอร์โทรติดต่อกลับ',
}
]" ]"
/> />
</div> </div>
@ -384,6 +378,12 @@ function onClose() {
v-model="formData.phone" v-model="formData.phone"
class="inputgreen" class="inputgreen"
hide-bottom-space hide-bottom-space
:rules="[
() =>
!!formData.email ||
!!formData.phone ||
'กรุณากรอกอีเมลหรือเบอร์โทรติดต่อกลับ',
]"
/> />
</div> </div>
</div> </div>

View file

@ -7,7 +7,6 @@ import http from "@/plugins/http";
import config from "@/app.config"; import config from "@/app.config";
import { useCounterMixin } from "@/stores/mixin"; import { useCounterMixin } from "@/stores/mixin";
import { useStructureTree } from "@/stores/structureTree"; import { useStructureTree } from "@/stores/structureTree";
import { formatPosmasterNo } from "@/utils/function";
import type { QTableProps } from "quasar"; import type { QTableProps } from "quasar";
import type { import type {
@ -25,14 +24,14 @@ import DialogHeader from "@/components/DialogHeader.vue";
const props = defineProps({ const props = defineProps({
dataRows: { dataRows: {
type: Object, type: Object,
default: () => ({}), require: true,
}, },
onSubmit: Function, onSubmit: Function,
}); });
const $q = useQuasar(); const $q = useQuasar();
const route = useRoute(); const route = useRoute();
const storeTree = useStructureTree(); const storeTree = useStructureTree();
const { fetchStructureTree } = storeTree; const { fetchStructureTree } = useStructureTree();
const mixin = useCounterMixin(); const mixin = useCounterMixin();
const { const {
dialogConfirm, dialogConfirm,
@ -48,10 +47,10 @@ const {
/** props*/ /** props*/
const modal = defineModel<boolean>("modal", { required: true }); const modal = defineModel<boolean>("modal", { required: true });
const title = defineModel<string>("title", { required: true }); const title = defineModel<string>("title", { required: true });
const type = defineModel<string | null>("type", { required: true }); const type = defineModel<any>("type", { required: true });
const posType = defineModel<string | null>("posType", { required: true }); const posType = defineModel<any>("posType", { required: true });
const posLevel = defineModel<string | null>("posLevel", { required: true }); const posLevel = defineModel<any>("posLevel", { required: true });
const position = defineModel<string | null>("position", { required: true }); const position = defineModel<any>("position", { required: true });
// const routeName = ref<string>(route?.name); // const routeName = ref<string>(route?.name);
const orgRevisionId = ref<string>(""); const orgRevisionId = ref<string>("");
@ -65,7 +64,7 @@ const itemTaps = ref<string[]>();
const filters = ref<string>(""); const filters = ref<string>("");
const positionId = ref<string>(""); const positionId = ref<string>("");
const selectedPos = ref<any[]>([]); const selectedPos = ref<any[]>([]);
const selectId = ref<string>(""); const seletcId = ref<string>("");
const datePos = ref<Date>(new Date()); const datePos = ref<Date>(new Date());
const rowsPosition = ref<Positions[]>([]); const rowsPosition = ref<Positions[]>([]);
const positionData = ref<any[]>([]); const positionData = ref<any[]>([]);
@ -80,6 +79,7 @@ const formActive = reactive<FormActive>({
}); });
/** node */ /** node */
const nodes = ref<Array<OrgTree>>([]); const nodes = ref<Array<OrgTree>>([]);
const lazy = ref(nodes);
const expanded = ref<string[]>([]); const expanded = ref<string[]>([]);
const nodeLevel = ref<number>(0); const nodeLevel = ref<number>(0);
const nodeId = ref<string>(""); // id Tree const nodeId = ref<string>(""); // id Tree
@ -150,7 +150,7 @@ const columns = ref<QTableProps["columns"]>([
style: "font-size: 14px", style: "font-size: 14px",
}, },
]); ]);
const columnsPosition = ref<QTableProps["columns"]>([ const columnsPostition = ref<QTableProps["columns"]>([
{ {
name: "no", name: "no",
align: "left", align: "left",
@ -181,7 +181,7 @@ const columnsPosition = ref<QTableProps["columns"]>([
{ {
name: "posTypeName", name: "posTypeName",
align: "left", align: "left",
label: "ประเภทตำหน่ง", label: "ประเภทตำเเหน่ง",
sortable: true, sortable: true,
field: "posTypeName", field: "posTypeName",
headerStyle: "font-size: 14px", headerStyle: "font-size: 14px",
@ -244,16 +244,10 @@ function close() {
expanded.value = []; expanded.value = [];
nodeLevel.value = 0; nodeLevel.value = 0;
nodeId.value = ""; nodeId.value = "";
positionData.value = [];
positionNo.value = [];
rowData.value = [];
} }
async function getDataTable(id: string, level: number = 0) { async function getDataTable(id: string, level: number = 0) {
showLoader(); showLoader();
positionData.value = [];
positionNo.value = [];
rowData.value = [];
const body = { const body = {
node: level, node: level,
nodeId: id, nodeId: id,
@ -263,12 +257,11 @@ async function getDataTable(id: string, level: number = 0) {
posType: posType.value ? posType.value : "", posType: posType.value ? posType.value : "",
isAll: isAll.value, isAll: isAll.value,
isBlank: isBlank.value, isBlank: isBlank.value,
profileId: type.value === "MOVE" ? props.dataRows.profileId : undefined,
}; };
await http await http
.post(config.API.orgPosPlacement, body) .post(config.API.orgPosPlacement, body)
.then(async (res) => { .then((res) => {
const dataMain: PositionMain[] = []; const dataMain: PositionMain[] = [];
posMasterMain.value = res.data.result.data; posMasterMain.value = res.data.result.data;
@ -284,12 +277,12 @@ async function getDataTable(id: string, level: number = 0) {
const listPosNo: DataPositionNo[] = dataMain.map((e: PositionMain) => ({ const listPosNo: DataPositionNo[] = dataMain.map((e: PositionMain) => ({
id: e.id, id: e.id,
isPosition: e.isPosition, isPosition: e.isPosition,
posMasterNo: formatPosmasterNo( posMasterNo:
e.orgShortname, e.orgShortname +
e.posMasterNoPrefix, (e.posMasterNoPrefix != null ? e.posMasterNoPrefix : "") +
e.posMasterNo.toString(), " " +
e.posMasterNoSuffix e.posMasterNo +
), (e.posMasterNoSuffix != null ? e.posMasterNoSuffix : ""),
positionName: e.positionName, positionName: e.positionName,
posTypeName: e.posTypeName, posTypeName: e.posTypeName,
posLevelName: e.posLevelName, posLevelName: e.posLevelName,
@ -306,6 +299,7 @@ async function getDataTable(id: string, level: number = 0) {
positionNo.value = listPosNo; positionNo.value = listPosNo;
rowData.value = listPosNo; rowData.value = listPosNo;
// positionData.value = listPosNo;
if (props.dataRows?.posmasterId) { if (props.dataRows?.posmasterId) {
const newUse = positionUse.value.filter( const newUse = positionUse.value.filter(
@ -319,13 +313,14 @@ async function getDataTable(id: string, level: number = 0) {
(e: any) => !positionUse.value.includes(e.id) (e: any) => !positionUse.value.includes(e.id)
); );
} }
await onClickSelectPos(positionId.value);
}) })
.catch((err) => { .catch((err) => {
messageError($q, err); messageError($q, err);
}) })
.finally(() => { .finally(() => {
hideLoader(); setTimeout(() => {
hideLoader();
}, 1000);
}); });
} }
@ -336,11 +331,11 @@ async function getDataTable(id: string, level: number = 0) {
function updateSelected(data: DataTree) { function updateSelected(data: DataTree) {
if (props?.dataRows?.nodeId === data.orgTreeId) { if (props?.dataRows?.nodeId === data.orgTreeId) {
positionId.value = props?.dataRows?.posmasterId; positionId.value = props?.dataRows?.posmasterId;
selectId.value = props?.dataRows?.positionId; seletcId.value = props?.dataRows?.positionId;
datePos.value = props?.dataRows?.reportingDate; datePos.value = props?.dataRows?.reportingDate;
} else { } else {
positionId.value = ""; positionId.value = "";
selectId.value = ""; seletcId.value = "";
selectedPos.value = []; selectedPos.value = [];
datePos.value = new Date(); datePos.value = new Date();
} }
@ -387,9 +382,9 @@ async function onClickSelectPos(id: string) {
// //
if (position) { if (position) {
rowsPosition.value = position.positions; rowsPosition.value = position.positions;
if (selectId.value) { if (seletcId.value) {
selectedPos.value = rowsPosition.value.filter( selectedPos.value = rowsPosition.value.filter(
(e) => e.id === selectId.value (e) => e.id === seletcId.value
); );
} }
} }
@ -408,19 +403,18 @@ async function fetchPosFind(level: number, id: string) {
}; };
await http await http
.post(config.API.orgPosFind, body) .post(config.API.orgPosFind, body)
.then(async (res) => { .then((res) => {
const data = res.data.result; const data = res.data.result;
expanded.value = data; expanded.value = data;
nodeId.value = id; nodeId.value = id;
positionId.value = props?.dataRows?.posmasterId; positionId.value = props?.dataRows?.posmasterId;
selectId.value = props?.dataRows?.positionId; seletcId.value = props?.dataRows?.positionId;
datePos.value = props?.dataRows?.reportingDate; datePos.value = props?.dataRows?.reportingDate;
await getDataTable(nodeId.value, level);
getDataTable(nodeId.value, level);
}) })
.catch((e) => { .catch((e) => {
messageError($q, e); messageError($q, e);
})
.finally(() => {
hideLoader(); hideLoader();
}); });
} }
@ -433,6 +427,11 @@ watch(
if (props?.dataRows?.node !== null && props?.dataRows?.nodeId !== null) { if (props?.dataRows?.node !== null && props?.dataRows?.nodeId !== null) {
await fetchPosFind(props?.dataRows?.node, props?.dataRows?.nodeId); await fetchPosFind(props?.dataRows?.node, props?.dataRows?.nodeId);
if (positionId.value) {
setTimeout(async () => {
await onClickSelectPos(positionId.value);
}, 200);
}
} else { } else {
expanded.value = []; expanded.value = [];
} }
@ -464,15 +463,19 @@ function fetchPositionUes() {
watch( watch(
() => isAll.value, () => isAll.value,
() => { (value, oldVal) => {
getDataTable(nodeId.value, nodeLevel.value); if (value !== oldVal) {
getDataTable(nodeId.value, nodeLevel.value);
}
} }
); );
watch( watch(
() => isBlank.value, () => isBlank.value,
() => { (value, oldVal) => {
getDataTable(nodeId.value, nodeLevel.value); if (value !== oldVal) {
getDataTable(nodeId.value, nodeLevel.value);
}
} }
); );
@ -488,43 +491,35 @@ function onSubmit() {
const dataPosMaster = posMasterMain.value?.find( const dataPosMaster = posMasterMain.value?.find(
(e: any) => e.id === positionId.value (e: any) => e.id === positionId.value
); );
if (!dataPosMaster) {
dialogMessageNotify($q, "ไม่พบข้อมูลตำแหน่ง");
return;
}
if (selectedPos.value.length === 0) { if (selectedPos.value.length === 0) {
dialogMessageNotify($q, "กรุณาเลือกตำแหน่ง"); dialogMessageNotify($q, "กรุณาเลือกตำแหน่ง");
return; } else {
dialogConfirm($q, async () => {
const body = {
personalId: props?.dataRows?.id,
node: dataPosMaster.node,
nodeId: dataPosMaster.nodeId,
orgRevisionId: formActive.activeId,
positionId: selectedPos.value[0].id,
posMasterNo: dataPosMaster.posMasterNo, //()
positionName: selectedPos.value[0].positionName, //
positionField: selectedPos.value[0].positionField, //
posTypeId: selectedPos.value[0].posTypeId, //
posTypeName: selectedPos.value[0].posTypeName, //
posLevelId: selectedPos.value[0].posLevelId, //
posLevelName: selectedPos.value[0].posLevelName, //
posExecutiveName: selectedPos.value[0].posExecutiveName, //
reportingDate: convertDateToAPI(datePos.value),
posmasterId: dataPosMaster.id,
typeCommand: type.value,
positionExecutiveField: selectedPos.value[0].positionExecutiveField, //
positionArea: selectedPos.value[0].positionArea, ///
};
await props.onSubmit?.(body);
close();
});
} }
dialogConfirm($q, async () => {
const body = {
personalId: props?.dataRows?.id,
node: dataPosMaster.node,
nodeId: dataPosMaster.nodeId,
orgRevisionId: formActive.activeId,
positionId: selectedPos.value[0].id,
posMasterNo: dataPosMaster.posMasterNo, //()
positionName: selectedPos.value[0].positionName, //
positionField: selectedPos.value[0].positionField, //
posTypeId: selectedPos.value[0].posTypeId, //
posTypeName: selectedPos.value[0].posTypeName, //
posLevelId: selectedPos.value[0].posLevelId, //
posLevelName: selectedPos.value[0].posLevelName, //
posExecutiveName: selectedPos.value[0].posExecutiveName, //
posExecutiveId: selectedPos.value[0].posExecutiveId, //
reportingDate: convertDateToAPI(datePos.value),
posmasterId: dataPosMaster.id,
typeCommand: type.value,
positionExecutiveField: selectedPos.value[0].positionExecutiveField, //
positionArea: selectedPos.value[0].positionArea, ///
};
await props.onSubmit?.(body);
close();
});
} }
function onSearch() { function onSearch() {
@ -539,7 +534,6 @@ onMounted(async () => {
await fetchTree(); await fetchTree();
}); });
</script> </script>
<template> <template>
<q-dialog v-model="modal" persistent full-width> <q-dialog v-model="modal" persistent full-width>
<q-card class="no-scroll"> <q-card class="no-scroll">
@ -571,7 +565,7 @@ onMounted(async () => {
<q-tree <q-tree
class="q-pa-sm q-gutter-sm" class="q-pa-sm q-gutter-sm"
dense dense
:nodes="nodes" :nodes="lazy"
node-key="orgTreeId" node-key="orgTreeId"
label-key="labelName" label-key="labelName"
:filter="filter" :filter="filter"
@ -837,7 +831,7 @@ onMounted(async () => {
</q-toolbar> </q-toolbar>
<d-table <d-table
ref="table" ref="table"
:columns="columnsPosition" :columns="columnsPostition"
:rows="rowsPosition" :rows="rowsPosition"
row-key="id" row-key="id"
flat flat
@ -935,7 +929,6 @@ onMounted(async () => {
</q-card> </q-card>
</q-dialog> </q-dialog>
</template> </template>
<style scoped> <style scoped>
.my-list-link { .my-list-link {
color: rgb(118, 168, 222); color: rgb(118, 168, 222);

View file

@ -7,7 +7,6 @@ import http from "@/plugins/http";
import config from "@/app.config"; import config from "@/app.config";
import { useCounterMixin } from "@/stores/mixin"; import { useCounterMixin } from "@/stores/mixin";
import { useStructureTree } from "@/stores/structureTree"; import { useStructureTree } from "@/stores/structureTree";
import { formatPosmasterNo } from "@/utils/function";
import type { QTableProps } from "quasar"; import type { QTableProps } from "quasar";
import type { import type {
@ -233,8 +232,7 @@ function close() {
async function getDataTable(id: string, level: number = 0) { async function getDataTable(id: string, level: number = 0) {
showLoader(); showLoader();
positionNo.value = [];
rowsData.value = [];
const body = { const body = {
node: level, node: level,
nodeId: id, nodeId: id,
@ -264,12 +262,11 @@ async function getDataTable(id: string, level: number = 0) {
const listPosNo: DataPositionNo[] = dataMain.map((e: PositionMain) => ({ const listPosNo: DataPositionNo[] = dataMain.map((e: PositionMain) => ({
id: e.id, id: e.id,
isPosition: e.isPosition, isPosition: e.isPosition,
posMasterNo: formatPosmasterNo( posMasterNo:
e.orgShortname, e.orgShortname +
e.posMasterNoPrefix, (e.posMasterNoPrefix != null ? e.posMasterNoPrefix : "") +
e.posMasterNo.toString(), e.posMasterNo +
e.posMasterNoSuffix (e.posMasterNoSuffix != null ? e.posMasterNoSuffix : ""),
),
positionName: e.positionName, positionName: e.positionName,
posTypeName: e.posTypeName, posTypeName: e.posTypeName,
posLevelName: e.posLevelName, posLevelName: e.posLevelName,
@ -305,7 +302,9 @@ async function getDataTable(id: string, level: number = 0) {
messageError($q, err); messageError($q, err);
}) })
.finally(() => { .finally(() => {
hideLoader(); setTimeout(() => {
hideLoader();
}, 1000);
}); });
} }

View file

@ -14,7 +14,6 @@ import type {
GovermentEmpTemp, GovermentEmpTemp,
} from "@/components/information/interface/response/Government"; } from "@/components/information/interface/response/Government";
import type { Avatar } from "@/components/information/interface/response/avatar"; import type { Avatar } from "@/components/information/interface/response/avatar";
import avatarMain from "@/assets/avatar_user.jpg";
/** importStore*/ /** importStore*/
import { useCounterMixin } from "@/stores/mixin"; import { useCounterMixin } from "@/stores/mixin";
@ -168,9 +167,9 @@ async function fetchInformation(id: string) {
avatar.position = data.position ? data.position : "-"; avatar.position = data.position ? data.position : "-";
// Function fetchProfile // Function fetchProfile
if (data.avatarName) { if (data.avatarName) {
fetchProfile(data.id as string, data.avatarName); await fetchProfile(data.id as string, data.avatarName);
} else { } else {
avatar.avatar = avatarMain; avatar.avatar = "";
} }
if (props.id) { if (props.id) {
@ -261,14 +260,11 @@ async function fetchProfileGovTemp(id: string) {
* @param id profileID * @param id profileID
* @param avatarName อไฟล * @param avatarName อไฟล
*/ */
function fetchProfile(id: string, avatarName: string) { async function fetchProfile(id: string, avatarName: string) {
http await http
.get(config.API.fileByFile("ทะเบียนประวัติ", "โปรไฟล์", id, avatarName)) .get(config.API.fileByFile("ทะเบียนประวัติ", "โปรไฟล์", id, avatarName))
.then((res) => { .then(async (res) => {
avatar.avatar = res.data.downloadUrl; avatar.avatar = await res.data.downloadUrl;
})
.catch(() => {
avatar.avatar = avatarMain;
}); });
} }

View file

@ -5,6 +5,7 @@ import { useQuasar } from "quasar";
import { useCounterMixin } from "@/stores/mixin"; import { useCounterMixin } from "@/stores/mixin";
import http from "@/plugins/http"; import http from "@/plugins/http";
import config from "@/app.config"; import config from "@/app.config";
import { getColumnLabel } from "@/utils/function";
import type { QTableProps } from "quasar"; import type { QTableProps } from "quasar";
@ -186,7 +187,6 @@ watch(modal, (val) => {
dense dense
label="คำค้น" label="คำค้น"
@clear="search = ''" @clear="search = ''"
@keydown.enter.prevent="onSearchData()"
/> />
</div> </div>
<q-checkbox <q-checkbox
@ -253,7 +253,7 @@ watch(modal, (val) => {
<q-th auto-width /> <q-th auto-width />
<q-th v-for="col in props.cols" :key="col.name" :props="props"> <q-th v-for="col in props.cols" :key="col.name" :props="props">
<span class="text-weight-medium"> <span class="text-weight-medium">
{{ col.label }} {{ getColumnLabel(col, isAct) }}
</span> </span>
</q-th> </q-th>
</q-tr> </q-tr>

View file

@ -13,7 +13,6 @@ import th from "quasar/lang/th";
import "@vuepic/vue-datepicker/dist/main.css"; import "@vuepic/vue-datepicker/dist/main.css";
import http from "./plugins/http"; import http from "./plugins/http";
import { createPinia } from "pinia"; import { createPinia } from "pinia";
import piniaPluginPersistedstate from "pinia-plugin-persistedstate";
// import './assets/main.css' // import './assets/main.css'
@ -21,7 +20,6 @@ import filters from "./plugins/filters";
const app = createApp(App); const app = createApp(App);
const pinia = createPinia(); const pinia = createPinia();
pinia.use(piniaPluginPersistedstate);
// เพิ่ม Global Filters ลงใน App // เพิ่ม Global Filters ลงใน App
app.config.globalProperties.$filters = filters; app.config.globalProperties.$filters = filters;

View file

@ -213,9 +213,7 @@ function formatHistoryOwnerData(data: HistoryPos[]) {
return data.map((item) => ({ return data.map((item) => ({
...item, ...item,
fullname: item.firstName fullname: item.firstName
? `${item.prefix || ""}${item.firstName || ""} ${ ? `${item.prefix}${item.firstName} ${item.lastName}`.trim()
item.lastName || ""
}`.trim()
: "ว่าง", : "ว่าง",
})); }));
} }
@ -277,7 +275,7 @@ watch(
(store.typeOrganizational === "draft" || (store.typeOrganizational === "draft" ||
store.typeOrganizational === "old") store.typeOrganizational === "old")
? "2 (ปัจจุบัน)" ? "2 (ปัจจุบัน)"
: props.rowIndex + 1 : props.rowIndex + 1
}} }}
</div> </div>

View file

@ -48,7 +48,7 @@ const modal = defineModel<boolean>("modal", { required: true });
let reqMaster = defineModel<FilterMaster>("reqMaster", { required: true }); let reqMaster = defineModel<FilterMaster>("reqMaster", { required: true });
const totalPage = defineModel<number>("totalPage", { required: true }); const totalPage = defineModel<number>("totalPage", { required: true });
const nodeTree = defineModel<OrgTree[]>("nodeTree", { required: true }); const nodeTree = defineModel<OrgTree[]>("nodeTree", { required: true });
const columns = defineModel<QTableProps['columns']>("columns", {}); const columns = defineModel<QTableProps[]>("columns", {});
const rows = defineModel<PosMaster2[]>("rows", { required: true }); const rows = defineModel<PosMaster2[]>("rows", { required: true });
const props = defineProps({ const props = defineProps({
fetchDataTree: { fetchDataTree: {

View file

@ -9,7 +9,6 @@ import { checkPermission } from "@/utils/permissions";
import { updateCurrentPage } from "@/utils/function"; import { updateCurrentPage } from "@/utils/function";
import { useOrganizational } from "@/modules/02_organization/store/organizational"; import { useOrganizational } from "@/modules/02_organization/store/organizational";
import { useCounterMixin } from "@/stores/mixin"; import { useCounterMixin } from "@/stores/mixin";
import { formatPosmasterNo } from "@/utils/function";
/** importType*/ /** importType*/
import type { QTableProps } from "quasar"; import type { QTableProps } from "quasar";
@ -206,14 +205,6 @@ const columns = ref<QTableProps["columns"]>([
sortable: false, sortable: false,
field: "posMasterNo", field: "posMasterNo",
headerStyle: "font-size: 14px", headerStyle: "font-size: 14px",
format(val, row) {
return formatPosmasterNo(
row.orgShortname,
row.posMasterNoPrefix,
row.posMasterNoMain,
row.posMasterNoSuffix
);
},
style: "font-size: 14px", style: "font-size: 14px",
}, },
{ {
@ -1013,6 +1004,7 @@ watch(
table-class="text-grey-9" table-class="text-grey-9"
row-key="id" row-key="id"
dense dense
hide-bottom
bordered bordered
separator="vertical" separator="vertical"
class="custom-header-table-expand" class="custom-header-table-expand"
@ -1133,7 +1125,7 @@ watch(
<DialogMovePos <DialogMovePos
v-model:modal="modalDialogMMove" v-model:modal="modalDialogMMove"
v-model:nodeTree="nodeTree" v-model:nodeTree="nodeTree"
v-model:columns="columns" v-model:columns="columns as QTableProps[]"
v-model:rows="posMaster" v-model:rows="posMaster"
v-model:totalPage="totalPage" v-model:totalPage="totalPage"
v-model:reqMaster="reqMaster" v-model:reqMaster="reqMaster"

View file

@ -108,11 +108,6 @@ export const useOrganizational = defineStore("organizationalStore", () => {
posLevelName: e.isSit ? e.profilePoslevel : e.posLevelName, posLevelName: e.isSit ? e.profilePoslevel : e.posLevelName,
posExecutiveName: e.posExecutiveName, posExecutiveName: e.posExecutiveName,
isSit: e.isSit, isSit: e.isSit,
orgShortname: e.orgShortname,
posMasterNoPrefix: e.posMasterNoPrefix,
posMasterNoSuffix: e.posMasterNoSuffix,
posMasterNoMain: e.posMasterNo,
})); }));
return newPosMaster || []; return newPosMaster || [];

View file

@ -365,7 +365,7 @@ async function downloadFileDashboard() {
async function clickPassExam() { async function clickPassExam() {
showLoader(); showLoader();
await http await http
.get(config.API.downloadPassExam(examId.value)) .get(config.API.exportExamPassExamList(examId.value))
.then(async (res) => { .then(async (res) => {
const data = res.data.result; const data = res.data.result;
data.reportName = `Candidate_Dashboard_${dateToISO(new Date())}`; data.reportName = `Candidate_Dashboard_${dateToISO(new Date())}`;
@ -382,7 +382,7 @@ async function clickPassExam() {
async function clickCandidateList() { async function clickCandidateList() {
showLoader(); showLoader();
await http await http
.get(config.API.downloadCandidateExam(examId.value)) .get(config.API.exportExamCandidateList(examId.value))
.then(async (res) => { .then(async (res) => {
const data = res.data.result; const data = res.data.result;
data.reportName = `Candidate_Dashboard_${dateToISO(new Date())}`; data.reportName = `Candidate_Dashboard_${dateToISO(new Date())}`;
@ -397,6 +397,8 @@ async function clickCandidateList() {
} }
async function onCheckShowExaminfo() { async function onCheckShowExaminfo() {
console.log(props.isShowExaminfo);
dialogMessage( dialogMessage(
$q, $q,
`ยืนยันการ${props.isShowExaminfo ? "ปิด" : "เปิด"}`, `ยืนยันการ${props.isShowExaminfo ? "ปิด" : "เปิด"}`,

View file

@ -45,7 +45,6 @@ interface RequestPeriodCompete {
order: number; order: number;
year: number; year: number;
announcementDate: string | null; announcementDate: string | null;
rootDnaId: string;
} }
interface RequestPeriodDisable { interface RequestPeriodDisable {
@ -64,7 +63,6 @@ interface RequestPeriodDisable {
round: number; round: number;
year: number; year: number;
announcementDate: string | null; announcementDate: string | null;
rootDnaId: string;
} }
interface RequestPosition { interface RequestPosition {

View file

@ -232,7 +232,7 @@ export default [
component: PeriodDisable, component: PeriodDisable,
meta: { meta: {
Auth: true, Auth: true,
Key: "SYS_EXAM_SELECT_PERIOD_DIS", Key: "SYS_EXAM_SELECT_LISTNAME",
Role: "STAFF", Role: "STAFF",
}, },
}, },
@ -242,7 +242,7 @@ export default [
component: PeriodDisableAdd, component: PeriodDisableAdd,
meta: { meta: {
Auth: true, Auth: true,
Key: "SYS_EXAM_SELECT_PERIOD_DIS", Key: "SYS_EXAM_SELECT_LISTNAME",
Role: "STAFF", Role: "STAFF",
}, },
}, },
@ -252,7 +252,7 @@ export default [
component: PeriodDisableAdd, component: PeriodDisableAdd,
meta: { meta: {
Auth: true, Auth: true,
Key: "SYS_EXAM_SELECT_PERIOD_DIS", Key: "SYS_EXAM_SELECT_LISTNAME",
Role: "STAFF", Role: "STAFF",
}, },
}, },
@ -262,7 +262,7 @@ export default [
component: MainDisableDetail, component: MainDisableDetail,
meta: { meta: {
Auth: true, Auth: true,
Key: "SYS_EXAM_SELECT_PERIOD_DIS", Key: "SYS_EXAM_SELECT_LISTNAME",
Role: "STAFF", Role: "STAFF",
}, },
}, },
@ -272,7 +272,7 @@ export default [
component: MainDisableEx, component: MainDisableEx,
meta: { meta: {
Auth: true, Auth: true,
Key: "SYS_EXAM_SELECT_PERIOD_DIS", Key: "SYS_EXAM_SELECT_LISTNAME",
Role: "STAFF", Role: "STAFF",
}, },
}, },

View file

@ -1,7 +1,7 @@
<!-- page:ดการรอบการสอบ สรรหา --> <!-- page:ดการรอบการสอบ สรรหา -->
<script setup lang="ts"> <script setup lang="ts">
import type { QTableProps } from "quasar"; import type { QTableProps } from "quasar";
import { onMounted, ref, watch } from "vue"; import { onMounted, ref } from "vue";
import { useRouter } from "vue-router"; import { useRouter } from "vue-router";
import { useQuasar } from "quasar"; import { useQuasar } from "quasar";
import genReport from "@/plugins/genreport"; import genReport from "@/plugins/genreport";
@ -11,9 +11,6 @@ import config from "@/app.config";
import { checkPermission } from "@/utils/permissions"; import { checkPermission } from "@/utils/permissions";
import { useCounterMixin } from "@/stores/mixin"; import { useCounterMixin } from "@/stores/mixin";
import { calculateFiscalYear } from "@/utils/function"; import { calculateFiscalYear } from "@/utils/function";
import { useUploadProgressStore } from "@/stores/uploadProgress";
import { useSocketStore } from "@/stores/socket";
import { storeToRefs } from "pinia";
import type { Pagination } from "@/modules/03_recruiting/interface/index/Main"; import type { Pagination } from "@/modules/03_recruiting/interface/index/Main";
import type { import type {
@ -36,14 +33,9 @@ const {
messageError, messageError,
onSearchDataTable, onSearchDataTable,
dialogRemove, dialogRemove,
dialogMessage,
} = mixin; } = mixin;
const router = useRouter(); const router = useRouter();
const uploadProgress = useUploadProgressStore();
const socketStore = useSocketStore();
const { notificationCounter } = storeToRefs(socketStore);
const name = ref<string>(""); const name = ref<string>("");
const year = ref<number>(calculateFiscalYear(new Date()) + 543); const year = ref<number>(calculateFiscalYear(new Date()) + 543);
const order = ref<number>(1); const order = ref<number>(1);
@ -57,11 +49,6 @@ const modalScore = ref<boolean>(false);
const modalCandidate = ref<boolean>(false); const modalCandidate = ref<boolean>(false);
const modalResult = ref<boolean>(false); const modalResult = ref<boolean>(false);
const selected_row_id = ref<string>(""); const selected_row_id = ref<string>("");
const jobStatus = ref<{
candidate?: "running" | "completed" | "failed";
score?: "running" | "completed" | "failed";
result?: "running" | "completed" | "failed";
}>({});
const rowsHistory = ref<ResponseHistoryObject[]>([]); //select data history const rowsHistory = ref<ResponseHistoryObject[]>([]); //select data history
const tittleHistory = ref<string>("ประวัติการนำเข้าข้อมูล"); // const tittleHistory = ref<string>("ประวัติการนำเข้าข้อมูล"); //
const filterHistory = ref<string>(""); //search data table history const filterHistory = ref<string>(""); //search data table history
@ -75,54 +62,6 @@ const textTittle = ref<string>("");
const textTittleScore = ref<string>(""); const textTittleScore = ref<string>("");
const textTittleCandidate = ref<string>(""); const textTittleCandidate = ref<string>("");
const textTittleResult = ref<string>(""); const textTittleResult = ref<string>("");
// Dialog message constants
const UPLOAD_RUNNING_DIALOG = {
title: "ไม่สามารถอัปโหลดไฟล์ได้",
message: "อยู่ระหว่างนำเข้าข้อมูล ระบบจะแจ้งผลให้ทราบทันทีเมื่อเสร็จสิ้น",
icon: "mdi-progress-alert",
btnLabel: "ตกลง",
color: "primary",
};
const UPLOAD_SUCCESS_DIALOG = {
title: "อัปโหลดไฟล์สำเร็จ",
message:
"ระบบกำลังนำเข้าข้อมูลคุณสามารถใช้งานเมนูอื่นในระหว่างนี้ได้ โดยระบบจะแจ้งผลการดำเนินการให้ทราบทันทีเมื่อเสร็จสิ้น",
icon: "check",
btnLabel: "ตกลง",
color: "primary",
};
// Function to show upload running dialog
function showUploadRunningDialog() {
dialogMessage(
$q,
UPLOAD_RUNNING_DIALOG.title,
UPLOAD_RUNNING_DIALOG.message,
UPLOAD_RUNNING_DIALOG.icon,
UPLOAD_RUNNING_DIALOG.btnLabel,
UPLOAD_RUNNING_DIALOG.color,
undefined,
undefined,
true
);
}
// Function to show upload success dialog
function showUploadSuccessDialog() {
dialogMessage(
$q,
UPLOAD_SUCCESS_DIALOG.title,
UPLOAD_SUCCESS_DIALOG.message,
UPLOAD_SUCCESS_DIALOG.icon,
UPLOAD_SUCCESS_DIALOG.btnLabel,
UPLOAD_SUCCESS_DIALOG.color,
undefined,
undefined,
true
);
}
const rows = ref<ResponseRecruitPeriod[]>([]); const rows = ref<ResponseRecruitPeriod[]>([]);
const rowsData = ref<ResponseRecruitPeriod[]>([]); const rowsData = ref<ResponseRecruitPeriod[]>([]);
const initialPagination = ref<Pagination>({ const initialPagination = ref<Pagination>({
@ -396,15 +335,9 @@ function clickDetail(id: string) {
* @param id รอบสอบเเขงข * @param id รอบสอบเเขงข
*/ */
async function clickUpload(id: string) { async function clickUpload(id: string) {
modalCandidate.value = true;
textTittleCandidate.value = "นำเข้าผู้สมัครสอบแข่งขัน";
selected_row_id.value = id; selected_row_id.value = id;
const isRunning = await checkJobStatus(id, "candidate");
if (isRunning) {
showUploadRunningDialog();
} else {
modalCandidate.value = true;
textTittleCandidate.value = "นำเข้าผู้สมัครสอบแข่งขัน";
}
} }
/** /**
@ -412,15 +345,9 @@ async function clickUpload(id: string) {
* @param id รอบสอบเเขงข * @param id รอบสอบเเขงข
*/ */
async function clickEdit(id: string) { async function clickEdit(id: string) {
modalScore.value = true;
textTittleScore.value = "นำเข้าบัญชีรวมคะแนน";
selected_row_id.value = id; selected_row_id.value = id;
const isRunning = await checkJobStatus(id, "score");
if (isRunning) {
showUploadRunningDialog();
} else {
modalScore.value = true;
textTittleScore.value = "นำเข้าบัญชีรวมคะแนน";
}
} }
/** /**
@ -428,57 +355,9 @@ async function clickEdit(id: string) {
* @param id รอบสอบเเขงข * @param id รอบสอบเเขงข
*/ */
async function clickResult(id: string) { async function clickResult(id: string) {
modalResult.value = true;
textTittleResult.value = "นำเข้าไฟล์ผลการสอบ";
selected_row_id.value = id; selected_row_id.value = id;
const isRunning = await checkJobStatus(id, "result");
if (isRunning) {
showUploadRunningDialog();
} else {
modalResult.value = true;
textTittleResult.value = "นำเข้าไฟล์ผลการสอบ";
}
}
/**
* ตรวจสอบสถานะการนำเขาขอม
* @param id รอบสอบเเขงข
* @param type ประเภทไฟลองการตรวจสอบ
* @return true ากำล running, false าไม job หร job เสรจแล
*/
async function checkJobStatus(
id: string,
type: "candidate" | "score" | "result"
): Promise<boolean> {
const uploads = uploadProgress.pendingUploads.filter(
(u) => u.periodId === id && u.type === type
);
let hasRunningJob = false;
for (const upload of uploads) {
try {
const res = await http.get(config.API.getImportStatus(upload.jobId));
const status = res.data.result.status.toLowerCase(); // 'running', 'completed', 'failed'
if (status === "completed" || status === "failed") {
status === "completed" && fetchData();
uploadProgress.removeUpload(upload.jobId);
} else if (status === "running") {
jobStatus.value[type] = "running";
hasRunningJob = true;
}
} catch (e) {
// if error, remove from pending
uploadProgress.removeUpload(upload.jobId);
}
}
// if no running jobs, clear status
if (!uploadProgress.isUploading(id, type)) {
jobStatus.value[type] = undefined;
}
return hasRunningJob;
} }
/** /**
@ -584,13 +463,11 @@ async function checkSaveCandidate() {
await http await http
.post(config.API.uploadCandidates(selected_row_id.value), fd) .post(config.API.uploadCandidates(selected_row_id.value), fd)
.then((res) => { .then((res) => {
const jobId = res.data.result.jobId; success($q, "นำเข้าข้อมูลผู้สมัครสอบสำเร็จ");
uploadProgress.addUpload(jobId, selected_row_id.value, "candidate");
success($q, UPLOAD_SUCCESS_DIALOG.message);
modalCandidate.value = false; modalCandidate.value = false;
files_candidate.value = null; files_candidate.value = null;
selected_row_id.value = ""; selected_row_id.value = "";
fetchData();
}) })
.catch((e) => { .catch((e) => {
messageError($q, e); messageError($q, e);
@ -600,7 +477,7 @@ async function checkSaveCandidate() {
}); });
} }
/** บันทึด คะเนน */ /** บันทึด คะเนน */
async function checkSaveScore() { async function checkSaveScore() {
const fd = new FormData(); const fd = new FormData();
fd.append("attachment", files_score.value[0]); fd.append("attachment", files_score.value[0]);
@ -608,12 +485,11 @@ async function checkSaveScore() {
await http await http
.post(config.API.saveScores(selected_row_id.value), fd) .post(config.API.saveScores(selected_row_id.value), fd)
.then((res) => { .then((res) => {
const jobId = res.data.result.jobId; success($q, "นำเข้าข้อมูลผลคะแนนสอบสำเร็จ");
uploadProgress.addUpload(jobId, selected_row_id.value, "score");
success($q, UPLOAD_SUCCESS_DIALOG.message);
modalScore.value = false; modalScore.value = false;
files_score.value = null; files_score.value = null;
selected_row_id.value = ""; selected_row_id.value = "";
fetchData();
}) })
.catch((e) => { .catch((e) => {
messageError($q, e); messageError($q, e);
@ -631,12 +507,11 @@ async function checkSaveResult() {
await http await http
.post(config.API.uploadResult(selected_row_id.value), fd) .post(config.API.uploadResult(selected_row_id.value), fd)
.then((res) => { .then((res) => {
const jobId = res.data.result.jobId; success($q, "นำเข้าข้อมูลผลการสอบแข่งขันฯ (บัญชีรายชื่อ)");
uploadProgress.addUpload(jobId, selected_row_id.value, "result");
success($q, UPLOAD_SUCCESS_DIALOG.message);
modalResult.value = false; modalResult.value = false;
files_result.value = null; files_result.value = null;
selected_row_id.value = ""; selected_row_id.value = "";
fetchData();
}) })
.catch((e) => { .catch((e) => {
messageError($q, e); messageError($q, e);
@ -657,9 +532,6 @@ async function checkSave() {
await http await http
.post(config.API.saveCandidates, fd) .post(config.API.saveCandidates, fd)
.then((res) => { .then((res) => {
const jobId = res.data.result.jobId;
uploadProgress.addUpload(jobId, selected_row_id.value, "period");
success($q, "นำเข้าข้อมูลผู้สมัครสอบแข่งขันสำเร็จ"); success($q, "นำเข้าข้อมูลผู้สมัครสอบแข่งขันสำเร็จ");
modalAdd.value = false; modalAdd.value = false;
fetchData(); fetchData();
@ -685,18 +557,12 @@ onMounted(async () => {
hideLoader(); hideLoader();
await fetchData(); await fetchData();
}); });
/** Watch notification counter on socket */
watch(notificationCounter, () => {
fetchData();
});
</script> </script>
<template> <template>
<div class="toptitle text-dark col-12 row items-center"> <div class="toptitle text-dark col-12 row items-center">
ดการรอบสอบแขงข ดการรอบสอบแขงข
</div> </div>
<q-card flat bordered class="col-12 q-mt-sm q-pt-sm q-pa-md"> <q-card flat bordered class="col-12 q-mt-sm q-pt-sm q-pa-md">
<div> <div>
<Table <Table
@ -803,16 +669,16 @@ watch(notificationCounter, () => {
</div> </div>
<div v-else-if="col.name == 'examCount'" class="table_ellipsis2"> <div v-else-if="col.name == 'examCount'" class="table_ellipsis2">
<q-btn <q-btn
v-if="
(col.value == null || col.value == '0') &&
checkPermission($route)?.attrIsUpdate
"
flat flat
dense dense
size="12px" size="12px"
color="green" color="green"
round round
@click.stop.prevent="clickUpload(props.row.id)" @click.stop.prevent="clickUpload(props.row.id)"
v-if="
(col.value == null || col.value == '0') &&
checkPermission($route)?.attrIsUpdate
"
> >
<q-icon name="mdi-file-excel-outline" size="20px" /> <q-icon name="mdi-file-excel-outline" size="20px" />
<q-tooltip>นำเขาไฟลสมครสอบ</q-tooltip> <q-tooltip>นำเขาไฟลสมครสอบ</q-tooltip>
@ -849,9 +715,6 @@ watch(notificationCounter, () => {
</div> </div>
<div v-else-if="col.name == 'scoreCount'" class="table_ellipsis2"> <div v-else-if="col.name == 'scoreCount'" class="table_ellipsis2">
<q-btn <q-btn
v-if="
col.value == null && checkPermission($route)?.attrIsUpdate
"
:disable="props.row.examCount == 0" :disable="props.row.examCount == 0"
flat flat
dense dense
@ -859,6 +722,9 @@ watch(notificationCounter, () => {
round round
color="green" color="green"
@click.stop.prevent="clickEdit(props.row.id)" @click.stop.prevent="clickEdit(props.row.id)"
v-if="
col.value == null && checkPermission($route)?.attrIsUpdate
"
> >
<q-icon name="mdi-file-excel-outline" size="20px" /> <q-icon name="mdi-file-excel-outline" size="20px" />
<!-- นำเขาไฟลผลคะแนนสอบ --> <!-- นำเขาไฟลผลคะแนนสอบ -->
@ -885,11 +751,6 @@ watch(notificationCounter, () => {
<div v-else-if="col.name == 'result'" class="table_ellipsis2"> <div v-else-if="col.name == 'result'" class="table_ellipsis2">
<q-btn <q-btn
v-if="
(props.row.score == null ||
props.row.score.resultCount == 0) &&
checkPermission($route)?.attrIsUpdate
"
:disable="props.row.score == null" :disable="props.row.score == null"
flat flat
dense dense
@ -897,6 +758,11 @@ watch(notificationCounter, () => {
color="green" color="green"
round round
@click.stop.prevent="clickResult(props.row.id)" @click.stop.prevent="clickResult(props.row.id)"
v-if="
(props.row.score == null ||
props.row.score.resultCount == 0) &&
checkPermission($route)?.attrIsUpdate
"
> >
<q-icon name="mdi-file-excel-outline" size="20px" /> <q-icon name="mdi-file-excel-outline" size="20px" />
<q-tooltip>นำเขาไฟลผลการสอบ (ญชรายช)</q-tooltip> <q-tooltip>นำเขาไฟลผลการสอบ (ญชรายช)</q-tooltip>
@ -1055,12 +921,12 @@ watch(notificationCounter, () => {
<q-dialog v-model="modalCandidate" persistent> <q-dialog v-model="modalCandidate" persistent>
<q-card style="width: 600px"> <q-card style="width: 600px">
<DialogHeadTemplate
:title="textTittleCandidate"
:close="clickCloseCandidate"
title-type="ข้อมูลผู้สมัครสอบ"
/>
<q-form ref="myFormScore"> <q-form ref="myFormScore">
<DialogHeadTemplate
:title="textTittleCandidate"
:close="clickCloseCandidate"
title-type="ข้อมูลผู้สมัครสอบ"
/>
<q-separator /> <q-separator />
<q-card-section> <q-card-section>
<div class="col-12 row items-center q-col-gutter-sm"> <div class="col-12 row items-center q-col-gutter-sm">
@ -1099,12 +965,12 @@ watch(notificationCounter, () => {
<q-dialog v-model="modalScore" persistent> <q-dialog v-model="modalScore" persistent>
<q-card style="width: 600px"> <q-card style="width: 600px">
<DialogHeadTemplate
:title="textTittleScore"
:close="clickCloseScore"
title-type="บัญชีรวมคะแนน"
/>
<q-form ref="myFormScore"> <q-form ref="myFormScore">
<DialogHeadTemplate
:title="textTittleScore"
:close="clickCloseScore"
title-type="บัญชีรวมคะแนน"
/>
<q-separator /> <q-separator />
<q-card-section> <q-card-section>
<div class="col-12 row items-center q-col-gutter-sm"> <div class="col-12 row items-center q-col-gutter-sm">
@ -1143,12 +1009,12 @@ watch(notificationCounter, () => {
<q-dialog v-model="modalResult" persistent> <q-dialog v-model="modalResult" persistent>
<q-card style="width: 600px"> <q-card style="width: 600px">
<DialogHeadTemplate
:title="textTittleResult"
:close="clickCloseResult"
title-type="ผลการสอบ (บัญชีรายชื่อ)"
/>
<q-form ref="myFormScore"> <q-form ref="myFormScore">
<DialogHeadTemplate
:title="textTittleResult"
:close="clickCloseResult"
title-type="ผลการสอบ (บัญชีรายชื่อ)"
/>
<q-separator /> <q-separator />
<q-card-section> <q-card-section>
<div class="col-12 row items-center q-col-gutter-sm"> <div class="col-12 row items-center q-col-gutter-sm">

View file

@ -2,13 +2,11 @@
import { onMounted, ref, watch, computed } from "vue"; import { onMounted, ref, watch, computed } from "vue";
import { useQuasar, QForm } from "quasar"; import { useQuasar, QForm } from "quasar";
import { useRouter, useRoute } from "vue-router"; import { useRouter, useRoute } from "vue-router";
import { storeToRefs } from "pinia";
import http from "@/plugins/http"; import http from "@/plugins/http";
import config from "@/app.config"; import config from "@/app.config";
import { useCounterMixin } from "@/stores/mixin"; import { useCounterMixin } from "@/stores/mixin";
import { calculateFiscalYear } from "@/utils/function"; import { calculateFiscalYear } from "@/utils/function";
import { usePositionKeycloakStore } from "@/stores/positionKeycloak";
import type { RequestPeriodCompete } from "@/modules/03_recruiting/interface/request/Period"; import type { RequestPeriodCompete } from "@/modules/03_recruiting/interface/request/Period";
import type { import type {
@ -18,7 +16,6 @@ import type {
const $q = useQuasar(); // show dialog const $q = useQuasar(); // show dialog
const mixin = useCounterMixin(); const mixin = useCounterMixin();
const { dataPositionKeycloak } = storeToRefs(usePositionKeycloakStore());
const router = useRouter(); const router = useRouter();
const route = useRoute(); const route = useRoute();
const { const {
@ -256,7 +253,6 @@ function sendData() {
year: yearly.value, year: yearly.value,
announcementDate: announcementDate:
dateAnnounce.value !== null ? convertDateToAPI(dateAnnounce.value) : null, dateAnnounce.value !== null ? convertDateToAPI(dateAnnounce.value) : null,
rootDnaId: !edit.value ? dataPositionKeycloak.value.rootDnaId : undefined,
}; };
return valueData; return valueData;
} }

View file

@ -309,8 +309,8 @@ async function fetchData(actionType?: string) {
if (data.length > 0) { if (data.length > 0) {
data.map((r: ResponseRecruitPeriod) => { data.map((r: ResponseRecruitPeriod) => {
if (r.score != null) { if (r.score != null) {
r.scoreCount = r.score.scoreCount ? r.score.scoreCount : 0; r.scoreCount = r.score.scoreCount;
r.scoreImportDate = r.score.importDate ? r.score.importDate : "-"; r.scoreImportDate = r.score.importDate;
} }
result.push(r); result.push(r);
}); });
@ -715,11 +715,7 @@ onMounted(async () => {
<q-tooltip>นำเขาไฟลผลคะแนนสอบ</q-tooltip> <q-tooltip>นำเขาไฟลผลคะแนนสอบ</q-tooltip>
</q-btn> </q-btn>
<div v-else> <div v-else>
{{ {{ props.row.score.scoreCount.toLocaleString() }}
props.row.score
? props.row.score.scoreCount.toLocaleString()
: "-"
}}
<q-btn <q-btn
v-if="checkPermission($route)?.attrIsUpdate" v-if="checkPermission($route)?.attrIsUpdate"
dense dense
@ -754,11 +750,7 @@ onMounted(async () => {
<q-tooltip>นำเขาไฟลผลการสอบ (ญชรายช)</q-tooltip> <q-tooltip>นำเขาไฟลผลการสอบ (ญชรายช)</q-tooltip>
</q-btn> </q-btn>
<div v-else> <div v-else>
{{ {{ props.row.score.resultCount.toLocaleString() }}
props.row.score
? props.row.score.resultCount.toLocaleString()
: "-"
}}
<q-btn <q-btn
v-if="checkPermission($route)?.attrIsUpdate" v-if="checkPermission($route)?.attrIsUpdate"
dense dense

View file

@ -3,13 +3,11 @@
import { onMounted, ref, watch } from "vue"; import { onMounted, ref, watch } from "vue";
import { useQuasar, QForm } from "quasar"; import { useQuasar, QForm } from "quasar";
import { useRouter, useRoute } from "vue-router"; import { useRouter, useRoute } from "vue-router";
import { storeToRefs } from "pinia";
import http from "@/plugins/http"; import http from "@/plugins/http";
import config from "@/app.config"; import config from "@/app.config";
import { useCounterMixin } from "@/stores/mixin"; import { useCounterMixin } from "@/stores/mixin";
import { calculateFiscalYear } from "@/utils/function"; import { calculateFiscalYear } from "@/utils/function";
import { usePositionKeycloakStore } from "@/stores/positionKeycloak";
import type { RequestPeriodDisable } from "@/modules/03_recruiting/interface/request/Period"; import type { RequestPeriodDisable } from "@/modules/03_recruiting/interface/request/Period";
import type { import type {
@ -19,7 +17,6 @@ import type {
const $q = useQuasar(); // show dialog const $q = useQuasar(); // show dialog
const mixin = useCounterMixin(); const mixin = useCounterMixin();
const { dataPositionKeycloak } = storeToRefs(usePositionKeycloakStore());
const router = useRouter(); const router = useRouter();
const route = useRoute(); const route = useRoute();
const { const {
@ -162,7 +159,6 @@ function sendData() {
year: yearly.value, year: yearly.value,
announcementDate: announcementDate:
dateAnnounce.value !== null ? convertDateToAPI(dateAnnounce.value) : null, dateAnnounce.value !== null ? convertDateToAPI(dateAnnounce.value) : null,
rootDnaId: !edit.value ? dataPositionKeycloak.value.rootDnaId : undefined,
}; };
return valueData; return valueData;
} }

View file

@ -212,17 +212,17 @@ const columnsPosition = ref<QTableProps["columns"]>([
a.localeCompare(b, undefined, { numeric: true, sensitivity: "base" }), a.localeCompare(b, undefined, { numeric: true, sensitivity: "base" }),
}, },
// { {
// name: "type", name: "type",
// align: "left", align: "left",
// label: "", label: "ประเภทแบบฟอร์ม",
// sortable: true, sortable: true,
// field: "type", field: "type",
// headerStyle: "font-size: 14px", headerStyle: "font-size: 14px",
// style: "font-size: 14px", style: "font-size: 14px",
// sort: (a: string, b: string) => sort: (a: string, b: string) =>
// a.localeCompare(b, undefined, { numeric: true, sensitivity: "base" }), a.localeCompare(b, undefined, { numeric: true, sensitivity: "base" }),
// }, },
]); ]);
const shouldShowPaymentFields = computed(() => { const shouldShowPaymentFields = computed(() => {
@ -1476,7 +1476,7 @@ onMounted(async () => {
</div> </div>
</q-td> </q-td>
<!-- <q-td key="type" :props="props"> <q-td key="type" :props="props">
<selector <selector
class="" class=""
outlined outlined
@ -1490,7 +1490,7 @@ onMounted(async () => {
lazy-rules lazy-rules
:rules="[(val:any) => !!val || `${'กรุณาเลือกประเภทแบบฟอร์ม'}`]" :rules="[(val:any) => !!val || `${'กรุณาเลือกประเภทแบบฟอร์ม'}`]"
></selector> ></selector>
</q-td> --> </q-td>
</q-tr> </q-tr>
</template> </template>
</ProfileTable> </ProfileTable>

View file

@ -260,6 +260,7 @@ async function fetchData(loading: boolean = true) {
total.value = data.total; total.value = data.total;
maxPage.value = await Math.ceil(data.total / pageSize.value); maxPage.value = await Math.ceil(data.total / pageSize.value);
maxPage.value = maxPage.value < 1 ? 1 : maxPage.value; maxPage.value = maxPage.value < 1 ? 1 : maxPage.value;
rows.value = []; rows.value = [];
data.data.map((r: any) => { data.data.map((r: any) => {
rows.value.push({ rows.value.push({
@ -401,7 +402,6 @@ onMounted(async () => {
await fetchDataCom(); await fetchDataCom();
}); });
</script> </script>
<template> <template>
<div class="toptitle text-dark col-12 row items-center"> <div class="toptitle text-dark col-12 row items-center">
<q-btn <q-btn
@ -474,9 +474,9 @@ onMounted(async () => {
</div> </div>
</q-card> </q-card>
</q-slide-transition> </q-slide-transition>
<q-card flat bordered class="col-12 q-pt-sm"> <q-card flat bordered class="col-12 q-pt-sm">
<TableCandidate <TableCandidate
style="max-height: 80vh"
:rows="rows" :rows="rows"
:columns="columns" :columns="columns"
:filter="filter" :filter="filter"
@ -603,7 +603,6 @@ onMounted(async () => {
</template> </template>
</TableCandidate> </TableCandidate>
</q-card> </q-card>
<q-dialog v-model="modal" persistent> <q-dialog v-model="modal" persistent>
<q-card style="min-width: 600px"> <q-card style="min-width: 600px">
<q-card-section class="row items-center q-py-sm"> <q-card-section class="row items-center q-py-sm">

View file

@ -50,10 +50,18 @@ const formMain = reactive<FormMain>({
workDate: null, // workDate: null, //
reasonSameDate: "", reasonSameDate: "",
retireDate: null, // retireDate: null, //
ageAll: "", // ageAll: {
year: 0,
month: 0,
day: 0,
}, //
absent: 0, // absent: 0, //
age: 0, // age: 0, //
govAgeBkk: "", govAgeBkk:{
year: 0,
month: 0,
day: 0,
}
}); });
const modalEdit = ref<boolean>(false); // popup const modalEdit = ref<boolean>(false); // popup
@ -157,22 +165,10 @@ async function getData() {
formMain.reasonSameDate = data.reasonSameDate; formMain.reasonSameDate = data.reasonSameDate;
formMain.retireDate = data.dateLeave; formMain.retireDate = data.dateLeave;
formMain.dateRetireLaw = data.dateRetireLaw; formMain.dateRetireLaw = data.dateRetireLaw;
formMain.ageAll = data.govAge formMain.ageAll = data.govAge;
? (
(data.govAge.year > 0 ? `${data.govAge.year} ปี ` : "") +
(data.govAge.month > 0 ? `${data.govAge.month} เดือน ` : "") +
(data.govAge.day > 0 ? `${data.govAge.day} วัน` : "")
).trim() || "-"
: "-";
formMain.absent = data.govAgeAbsent; formMain.absent = data.govAgeAbsent;
formMain.age = data.govAgePlus; formMain.age = data.govAgePlus;
formMain.govAgeBkk = data.govAgeBkk formMain.govAgeBkk = data.govAgeBkk;
? (
(data.govAgeBkk.year > 0 ? `${data.govAgeBkk.year} ปี ` : "") +
(data.govAgeBkk.month > 0 ? `${data.govAgeBkk.month} เดือน ` : "") +
(data.govAgeBkk.day > 0 ? `${data.govAgeBkk.day} วัน` : "")
).trim() || "-"
: "-";
}) })
.catch((e) => { .catch((e) => {
messageError($q, e); messageError($q, e);
@ -459,19 +455,25 @@ onMounted(() => {
<span class="text-grey-6 text-weight-medium">อายราชการ</span> <span class="text-grey-6 text-weight-medium">อายราชการ</span>
</div> </div>
<div class="col-12 col-sm-12 col-md-7"> <div class="col-12 col-sm-12 col-md-7">
<span>{{ formMain.ageAll }}</span> <span>{{
formMain.ageAll
? `${formMain.ageAll.year} ปี ${formMain.ageAll.month} เดือน ${formMain.ageAll.day} วัน`
: "-"
}}</span>
</div> </div>
</div> </div>
</div> </div>
<div class="col-12 col-sm-6 col-md-6"> <div class="col-12 col-sm-6 col-md-6">
<div class="row"> <div class="row">
<div class="col-12 col-sm-12 col-md-5"> <div class="col-12 col-sm-12 col-md-5">
<span class="text-grey-6 text-weight-medium" <span class="text-grey-6 text-weight-medium">อายราชการ (กทม.)</span>
>อายราชการ (กทม.)</span
>
</div> </div>
<div class="col-12 col-sm-12 col-md-7"> <div class="col-12 col-sm-12 col-md-7">
<span>{{ formMain.govAgeBkk }}</span> <span>{{
formMain.govAgeBkk
? `${formMain.govAgeBkk.year} ปี ${formMain.govAgeBkk.month} เดือน ${formMain.govAgeBkk.day} วัน`
: "-"
}}</span>
</div> </div>
</div> </div>
</div> </div>

View file

@ -16,11 +16,9 @@ import type {
DisciplineOps, DisciplineOps,
DataOption, DataOption,
} from "@/modules/04_registryPerson/interface/index/discipline"; } from "@/modules/04_registryPerson/interface/index/discipline";
import type { Discipline } from "@/modules/04_registryPerson/interface/response/Discipline";
import DialogHeader from "@/components/DialogHeader.vue"; import DialogHeader from "@/components/DialogHeader.vue";
import DialogHistory from "@/modules/04_registryPerson/components/detail/DialogHistory.vue"; import DialogHistory from "@/modules/04_registryPerson/components/detail/DialogHistory.vue";
import DialogPreviewCommand from "@/modules/18_command/components/DialogPreviewCommand.vue";
const route = useRoute(); const route = useRoute();
const $q = useQuasar(); const $q = useQuasar();
@ -49,9 +47,6 @@ const empType = ref<string>(pathRegistryEmp(route.name?.toString() ?? ""));
const isLeave = defineModel<boolean>("isLeave", { const isLeave = defineModel<boolean>("isLeave", {
required: true, required: true,
}); });
const citizenId = defineModel<string>("citizenId", {
required: true,
});
const baseColumns = ref<QTableColumn[]>([ const baseColumns = ref<QTableColumn[]>([
{ {
@ -157,8 +152,8 @@ const baseVisibleColumns = ref<string[]>([
"lastUpdatedAt", "lastUpdatedAt",
]); ]);
const rows = ref<Discipline[]>([]); // const rows = ref<RequestItemsObject[]>([]); //
const rowsMain = ref<Discipline[]>([]); // const rowsMain = ref<RequestItemsObject[]>([]); //
const mode = ref<string>("table"); // Table card const mode = ref<string>("table"); // Table card
const filterKeyword = ref<string>(""); // const filterKeyword = ref<string>(""); //
@ -249,10 +244,6 @@ const fileUpload = ref<File | null>(null);
const fileData = ref<ResFileData | null>(null); const fileData = ref<ResFileData | null>(null);
const isUpload = ref<boolean>(false); const isUpload = ref<boolean>(false);
const modalCommand = ref<boolean>(false);
const command = ref<string>("");
const commandId = ref<string>("");
/** /**
* function นหา คำใน option * function นหา คำใน option
* @param val คำคนหา * @param val คำคนหา
@ -314,7 +305,7 @@ async function fetchData(id: string) {
* กดเลอกขอมลทจะแกไข * กดเลอกขอมลทจะแกไข
* @param props props ใน row เลอก * @param props props ใน row เลอก
*/ */
async function openDialogEdit(props: Discipline) { async function openDialogEdit(props: RequestItemsObject) {
showLoader(); showLoader();
try { try {
modal.value = true; modal.value = true;
@ -487,12 +478,6 @@ function handleDelete(id: string) {
}); });
} }
function openViewCommand(data: Discipline) {
command.value = data.refCommandNo;
commandId.value = data.refCommandId;
modalCommand.value = true;
}
/** ทำงานเมื่อ Components ถูกเรียกใช้งาน*/ /** ทำงานเมื่อ Components ถูกเรียกใช้งาน*/
onMounted(() => { onMounted(() => {
fetchData(profileId.value); fetchData(profileId.value);
@ -636,19 +621,7 @@ onMounted(() => {
:key="col.id" :key="col.id"
:class="props.row.isEntry ? 'text-grey' : ''" :class="props.row.isEntry ? 'text-grey' : ''"
> >
<div <div>
v-if="col.name == 'refCommandNo' && props.row.refCommandId"
@click="col.value ? openViewCommand(props.row) : null"
:class="
col.value
? 'table_ellipsis text-blue cursor-pointer'
: 'table_ellipsis'
"
>
{{ col.value ? col.value : "-" }}
<q-tooltip v-if="col.value">ดูคำสั่ง</q-tooltip>
</div>
<div v-else>
{{ col.value ? col.value : "-" }} {{ col.value ? col.value : "-" }}
</div> </div>
</q-td> </q-td>
@ -1002,13 +975,6 @@ onMounted(() => {
:visible-columns="visibleColumnsHistory" :visible-columns="visibleColumnsHistory"
:fetch-data="fetchDataHistory" :fetch-data="fetchDataHistory"
/> />
<DialogPreviewCommand
v-model:modal="modalCommand"
v-model:command="command"
v-model:command-id="commandId"
:citizen-id="citizenId"
/>
</template> </template>
<style scoped></style> <style scoped></style>

View file

@ -866,8 +866,7 @@ onMounted(() => {
outlined outlined
dense dense
:model-value="date2Thai(formData.dateEnd)" :model-value="date2Thai(formData.dateEnd)"
clearable :rules="[(val:string) => !!val || `${'กรุณาเลือก วันที่สิ้นสุด'}`]"
@clear="formData.dateEnd = null"
hide-bottom-space hide-bottom-space
:label="`${'วันที่สิ้นสุด'}`" :label="`${'วันที่สิ้นสุด'}`"
> >

View file

@ -58,10 +58,7 @@ const storeRegistry = useRegistryNewDataStore();
<Info :is-leave="storeRegistry.isLeave" /> <Info :is-leave="storeRegistry.isLeave" />
</q-tab-panel> </q-tab-panel>
<q-tab-panel name="2"> <q-tab-panel name="2">
<Discipline <Discipline :is-leave="storeRegistry.isLeave" />
:is-leave="storeRegistry.isLeave"
:citizen-id="storeRegistry.citizenId"
/>
</q-tab-panel> </q-tab-panel>
<q-tab-panel name="3"> <q-tab-panel name="3">
<Leave :is-leave="storeRegistry.isLeave" /> <Leave :is-leave="storeRegistry.isLeave" />

View file

@ -408,8 +408,7 @@ function calculateMinDate() {
function prefixRankRule() { function prefixRankRule() {
return [ return [
() => () => !!formData.rank || !!formData.prefix || "กรุณาเลือกคำนำหน้าชื่อ หรือยศ",
!!formData.rank || !!formData.prefix || "กรุณาเลือกคำนำหน้าชื่อ หรือยศ",
]; ];
} }
@ -589,7 +588,6 @@ onMounted(() => {
<div class="col-xs-6 col-sm-6 col-md-6"> <div class="col-xs-6 col-sm-6 col-md-6">
<q-select <q-select
:readonly="checkPermission($route)?.attrOwnership !== 'OWNER'"
dense dense
outlined outlined
use-input use-input
@ -604,6 +602,7 @@ onMounted(() => {
option-value="name" option-value="name"
v-model="formData.prefix" v-model="formData.prefix"
clearable clearable
class="inputgreen"
:options="store.Ops.prefixOps" :options="store.Ops.prefixOps"
:label="dataLabel.prefix" :label="dataLabel.prefix"
:rules="prefixRankRule()" :rules="prefixRankRule()"
@ -641,24 +640,24 @@ onMounted(() => {
</div> </div>
<div class="col-xs-6 col-sm-6 col-md-6"> <div class="col-xs-6 col-sm-6 col-md-6">
<q-input <q-input
:readonly="checkPermission($route)?.attrOwnership !== 'OWNER'"
dense dense
outlined outlined
lazy-rules lazy-rules
hide-bottom-space hide-bottom-space
v-model="formData.firstName" v-model="formData.firstName"
class="inputgreen"
:label="dataLabel.firstName" :label="dataLabel.firstName"
:rules="[(val: string) => !!val || `${'กรุณากรอก ชื่อ'}`]" :rules="[(val: string) => !!val || `${'กรุณากรอก ชื่อ'}`]"
/> />
</div> </div>
<div class="col-xs-6 col-sm-6 col-md-6"> <div class="col-xs-6 col-sm-6 col-md-6">
<q-input <q-input
:readonly="checkPermission($route)?.attrOwnership !== 'OWNER'"
dense dense
outlined outlined
lazy-rules lazy-rules
hide-bottom-space hide-bottom-space
v-model="formData.lastName" v-model="formData.lastName"
class="inputgreen"
:label="dataLabel.lastName" :label="dataLabel.lastName"
:rules="[(val: string) => !!val || `${'กรุณากรอก นามสกุล'}`]" :rules="[(val: string) => !!val || `${'กรุณากรอก นามสกุล'}`]"
/> />
@ -722,7 +721,6 @@ onMounted(() => {
</div> </div>
<div class="col-xs-6 col-sm-6 col-md-6"> <div class="col-xs-6 col-sm-6 col-md-6">
<q-select <q-select
:readonly="checkPermission($route)?.attrOwnership !== 'OWNER'"
dense dense
outlined outlined
use-input use-input
@ -737,6 +735,7 @@ onMounted(() => {
option-label="name" option-label="name"
option-value="name" option-value="name"
v-model="formData.gender" v-model="formData.gender"
class="inputgreen"
:options="store.Ops.genderOps" :options="store.Ops.genderOps"
:label="dataLabel.gender" :label="dataLabel.gender"
@filter="(inputValue: string, @filter="(inputValue: string,

View file

@ -59,7 +59,6 @@ const changeNameData = reactive<FormChangeName>({
profileId: profileId.value, profileId: profileId.value,
prefixId: "", // id prefixId: "", // id
prefix: "", // prefix: "", //
rank: "", //
firstName: "", // firstName: "", //
lastName: "", // lastName: "", //
status: "", // status: "", //
@ -70,13 +69,11 @@ const selection = ref<string[]>([]); // ตัวเลือก
const prefixChange = ref<string | null | undefined>(""); // const prefixChange = ref<string | null | undefined>(""); //
const firstNameChange = ref<string | null | undefined>(""); // const firstNameChange = ref<string | null | undefined>(""); //
const lastNameChange = ref<string | null | undefined>(""); // const lastNameChange = ref<string | null | undefined>(""); //
const rankChange = ref<string | null | undefined>(""); //
const profileInfo = ref<DataProfile>(); // Profile const profileInfo = ref<DataProfile>(); // Profile
const statusOption = ref<string[]>([ const statusOption = ref<string[]>([
"เปลี่ยนคำนำหน้าชื่อ", "เปลี่ยนคำนำหน้าชื่อ",
"เปลี่ยนยศ",
"เปลี่ยนชื่อ", "เปลี่ยนชื่อ",
"เปลี่ยนนามสกุล", "เปลี่ยนนามสกุล",
"เปลี่ยนชื่อ-นามสกุล", "เปลี่ยนชื่อ-นามสกุล",
@ -87,7 +84,6 @@ const statusOption = ref<string[]>([
const statusOptionFilter = ref<string[]>([ const statusOptionFilter = ref<string[]>([
"เปลี่ยนคำนำหน้าชื่อ", "เปลี่ยนคำนำหน้าชื่อ",
"เปลี่ยนชื่อ", "เปลี่ยนชื่อ",
"เปลี่ยนยศ",
"เปลี่ยนนามสกุล", "เปลี่ยนนามสกุล",
"เปลี่ยนชื่อ-นามสกุล", "เปลี่ยนชื่อ-นามสกุล",
"เปลี่ยนคำนำหน้าชื่อ และชื่อ", "เปลี่ยนคำนำหน้าชื่อ และชื่อ",
@ -111,17 +107,6 @@ const columns = ref<QTableProps["columns"]>([
sort: (a: string, b: string) => sort: (a: string, b: string) =>
a.localeCompare(b, undefined, { numeric: true, sensitivity: "base" }), a.localeCompare(b, undefined, { numeric: true, sensitivity: "base" }),
}, },
{
name: "rank",
align: "left",
label: "ยศ",
sortable: true,
field: "rank",
headerStyle: "font-size: 14px",
style: "font-size: 14px",
sort: (a: string, b: string) =>
a.localeCompare(b, undefined, { numeric: true, sensitivity: "base" }),
},
{ {
name: "firstName", name: "firstName",
align: "left", align: "left",
@ -171,7 +156,6 @@ const columns = ref<QTableProps["columns"]>([
]); ]);
const visibleColumns = ref<string[]>([ const visibleColumns = ref<string[]>([
"prefix", "prefix",
"rank",
"firstName", "firstName",
"lastName", "lastName",
"lastUpdateFullName", "lastUpdateFullName",
@ -231,7 +215,6 @@ function onSubmit() {
firstName: changeNameData.firstName, firstName: changeNameData.firstName,
lastName: changeNameData.lastName, lastName: changeNameData.lastName,
status: changeNameData.status, status: changeNameData.status,
rank: changeNameData.rank,
documentId: changeNameData.documentId, documentId: changeNameData.documentId,
}) })
.then(async (res) => { .then(async (res) => {
@ -363,14 +346,6 @@ function filterSelector(val: string, update: Function, refData: string) {
); );
}); });
break; break;
case "rankOps":
update(() => {
store.Ops.rankOps = store.OpsFilter.rankOps.filter(
(v: DataOption) => v.name.indexOf(val) > -1
);
});
break;
default: default:
break; break;
} }
@ -415,14 +390,12 @@ watch(
changeNameData.prefix, changeNameData.prefix,
changeNameData.firstName, changeNameData.firstName,
changeNameData.lastName, changeNameData.lastName,
changeNameData.rank,
], ],
() => { () => {
submitDisable.value = submitDisable.value =
changeNameData.prefix === prefixChange.value && changeNameData.prefix === prefixChange.value &&
changeNameData.firstName === firstNameChange.value && changeNameData.firstName === firstNameChange.value &&
changeNameData.lastName === lastNameChange.value && changeNameData.lastName === lastNameChange.value;
changeNameData.rank === rankChange.value;
} }
); );
@ -438,9 +411,6 @@ watch(
if (!selection.value.includes("lastname")) { if (!selection.value.includes("lastname")) {
changeNameData.lastName = lastNameChange.value; changeNameData.lastName = lastNameChange.value;
} }
if (!selection.value.includes("rank")) {
changeNameData.rank = rankChange.value;
}
} }
); );
/** ทำงานเมื่อ Components ถูกเรียกใช้งาน*/ /** ทำงานเมื่อ Components ถูกเรียกใช้งาน*/
@ -475,14 +445,12 @@ onMounted(async () => {
icon="add" icon="add"
@click=" @click="
() => { () => {
changeNameData.prefix = profileInfo?.prefix; changeNameData.prefix = profileInfo?.prefixMain;
changeNameData.firstName = profileInfo?.firstName; changeNameData.firstName = profileInfo?.firstName;
changeNameData.lastName = profileInfo?.lastName; changeNameData.lastName = profileInfo?.lastName;
changeNameData.rank = profileInfo?.rank;
prefixChange = changeNameData.prefix; prefixChange = changeNameData.prefix;
firstNameChange = changeNameData.firstName; firstNameChange = changeNameData.firstName;
lastNameChange = changeNameData.lastName; lastNameChange = changeNameData.lastName;
rankChange = changeNameData.rank;
changeNameData.status = ''; changeNameData.status = '';
dialogStatus = 'create'; dialogStatus = 'create';
submitDisable = true; submitDisable = true;
@ -595,17 +563,12 @@ onMounted(async () => {
<div class="row q-mb-sm"> <div class="row q-mb-sm">
<div class="col-12"> <div class="col-12">
<div class="row"> <div class="row">
<div class="q-col-gutter-sm"> <div class="q-gutter-sm">
<q-checkbox <q-checkbox
v-model="selection" v-model="selection"
val="prefix" val="prefix"
label="เปลี่ยนคำนำหน้าชื่อ" label="เปลี่ยนคำนำหน้าชื่อ"
/> />
<q-checkbox
v-model="selection"
val="rank"
label="เปลี่ยนยศ"
/>
<q-checkbox <q-checkbox
v-model="selection" v-model="selection"
val="firstname" val="firstname"
@ -646,31 +609,6 @@ onMounted(async () => {
) " ) "
/> />
</div> </div>
<div class="col">
<q-select
:readonly="!selection.includes('rank')"
v-model="changeNameData.rank"
:options="store.Ops.rankOps"
label="ยศ"
dense
:class="!selection.includes('rank') ? '' : 'inputgreen'"
outlined
use-input
hide-selected
fill-input
lazy-rules
emit-value
map-options
:rules="[(val:string) => !!val || `${'กรุณาเลือกยศ'}`]"
hide-bottom-space
input-debounce="0"
option-label="name"
option-value="name"
@filter="(inputValue:string,
doneFn:Function) => filterSelector(inputValue, doneFn,'rankOps'
) "
/>
</div>
<div class="col"> <div class="col">
<q-input <q-input
:readonly="!selection.includes('firstname')" :readonly="!selection.includes('firstname')"

View file

@ -13,8 +13,8 @@ interface FormMain {
workDate: any; workDate: any;
reasonSameDate: string; reasonSameDate: string;
retireDate: any; retireDate: any;
ageAll: GovAgeForm | string; ageAll: GovAgeForm;
govAgeBkk: GovAgeForm | string; govAgeBkk: GovAgeForm;
absent: number; absent: number;
age: number; age: number;
[key: string]: any; [key: string]: any;

View file

@ -66,7 +66,6 @@ interface FormChangeName {
profileId: string; profileId: string;
prefixId: string | null | undefined; prefixId: string | null | undefined;
prefix: string | null | undefined; prefix: string | null | undefined;
rank: string | null | undefined;
firstName: string | null | undefined; firstName: string | null | undefined;
lastName: string | null | undefined; lastName: string | null | undefined;
status: string | null | undefined; status: string | null | undefined;

View file

@ -1,23 +0,0 @@
interface Discipline {
createdAt: Date | null;
createdFullName: string;
createdUserId: string;
date: Date | null;
detail: string;
id: string;
isDeleted: false;
isEntry: false;
isUpload: true;
lastUpdateFullName: string;
lastUpdateUserId: string;
lastUpdatedAt: Date | null;
level: string;
profileEmployeeId: string;
profileId: string;
refCommandDate: Date | null;
refCommandNo: string;
refCommandId: string;
unStigma: string;
}
export type { Discipline };

View file

@ -70,7 +70,6 @@ interface DataPosition {
status: string; status: string;
posNumCodeSitAbb: string; posNumCodeSitAbb: string;
posNumCodeSit: string; posNumCodeSit: string;
positionExecutiveField: string;
} }
export type { DataSalaryPos, DataPosition }; export type { DataSalaryPos, DataPosition };

View file

@ -25,7 +25,8 @@ export const useRegistryNewDataStore = defineStore("registryNew", () => {
{ {
id: "posNo", id: "posNo",
name: name:
route.name === "registryNew" || route.name === "registryRetireOfficer" route.name === "registryNew" ||
route.name === "registryRetireOfficer"
? "เลขที่ตำแหน่ง" ? "เลขที่ตำแหน่ง"
: "ตำแหน่งเลขที่", : "ตำแหน่งเลขที่",
}, },
@ -129,8 +130,6 @@ export const useRegistryNewDataStore = defineStore("registryNew", () => {
return "ให้ออกจากราชการ"; return "ให้ออกจากราชการ";
case "DISCIPLINE_TEMP_SUSPEND": case "DISCIPLINE_TEMP_SUSPEND":
return "ให้ออกจากราชการไว้ก่อน"; return "ให้ออกจากราชการไว้ก่อน";
case "RETIRE_MILITARY":
return "ลาออกเพื่อไปรับราชการทหาร";
default: default:
return ""; return "";
} }

View file

@ -1,177 +0,0 @@
import * as XLSX from "xlsx";
export interface ExcelPreviewData {
fileName: string;
fileSize: number;
sheetNames: string[];
headers: string[];
rows: Array<Record<string, any>>;
totalRows: number;
previewRows: number;
}
export interface ParseOptions {
maxPreviewRows?: number;
sheetIndex?: number;
}
const DEFAULT_CONFIG = {
MAX_FILE_SIZE: 10 * 1024 * 1024, // 10MB
MAX_PREVIEW_ROWS: 0, // 0 = ไม่จำกัด แสดงทั้งหมด
ALLOWED_EXTENSIONS: [".xlsx", ".xls"],
};
function formatFileSize(bytes: number): string {
if (bytes === 0) return "0 Bytes";
const k = 1024;
const sizes = ["Bytes", "KB", "MB", "GB"];
const i = Math.floor(Math.log(bytes) / Math.log(k));
return Math.round((bytes / Math.pow(k, i)) * 100) / 100 + " " + sizes[i];
}
function getFileExtension(fileName: string): string {
return fileName.substring(fileName.lastIndexOf(".")).toLowerCase();
}
function validateFile(file: File): { message: string } | null {
const extension = getFileExtension(file.name);
if (!DEFAULT_CONFIG.ALLOWED_EXTENSIONS.includes(extension)) {
return {
message: `กรุณาเลือกไฟล์ Excel เท่านั้น (${DEFAULT_CONFIG.ALLOWED_EXTENSIONS.join(", ")})`,
};
}
if (file.size > DEFAULT_CONFIG.MAX_FILE_SIZE) {
const fileSizeMB = (file.size / (1024 * 1024)).toFixed(2);
const maxSizeMB = (DEFAULT_CONFIG.MAX_FILE_SIZE / (1024 * 1024)).toFixed(2);
return {
message: `ขนาดไฟล์เกิน ${maxSizeMB}MB (ไฟล์ของคุณ: ${fileSizeMB}MB)`,
};
}
return null;
}
export async function parseExcelFile(
file: File,
options: ParseOptions = {}
): Promise<ExcelPreviewData> {
const maxPreviewRows = options.maxPreviewRows ?? DEFAULT_CONFIG.MAX_PREVIEW_ROWS;
const sheetIndex = options.sheetIndex ?? 0;
// Validate file
const validationError = validateFile(file);
if (validationError) {
throw new Error(validationError.message);
}
return new Promise((resolve, reject) => {
const reader = new FileReader();
reader.onload = (e) => {
try {
const data = e.target?.result;
if (!data) {
reject(new Error("ไม่สามารถอ่านไฟล์ได้"));
return;
}
const workbook = XLSX.read(data, { type: "binary" });
if (workbook.SheetNames.length === 0) {
reject(new Error("ไฟล์ไม่มีข้อมูล"));
return;
}
const sheetName = workbook.SheetNames[sheetIndex];
const worksheet = workbook.Sheets[sheetName];
// ใช้ sheet_to_json กับ raw: true เพื่ออ่าน raw values ก่อน
const jsonData = XLSX.utils.sheet_to_json(worksheet, {
header: 1,
defval: "",
raw: true, // อ่าน raw values
}) as any[][];
// หลังจากได้ raw data แล้ว ต้องไปอ่านค่าจาก formula ที่ cells โดยตรง
// เพราะบาง formula อาจจะยังไม่ถูกคำนวณ
const range = XLSX.utils.decode_range(worksheet["!ref"] || "A1");
// อ่านค่าจาก cells ที่มี formula (column สุดท้าย)
for (let row = range.s.r; row <= range.e.r; row++) {
for (let col = range.s.c; col <= range.e.c; col++) {
const cellAddress = XLSX.utils.encode_cell({ r: row, c: col });
const cell = worksheet[cellAddress];
// ถ้ามี formula ให้ใช้ผลลัพธ์ที่คำนวณแล้ว
if (cell && cell.f && !cell.v) {
// ลองใช้ cell.w (formatted value) ถ้ามี
if (cell.w) {
// แปลง array index ให้ถูกต้อง (row + 1 เพราะมี header row)
const arrayRowIndex = row - range.s.r;
if (jsonData[arrayRowIndex] && jsonData[arrayRowIndex][col] !== undefined) {
jsonData[arrayRowIndex][col] = cell.w;
}
}
}
}
}
if (jsonData.length === 0) {
reject(new Error("ไฟล์ไม่มีข้อมูล"));
return;
}
// Extract headers from first row (ภาษาไทย)
const headers = jsonData[0].map((h: any) => String(h ?? ""));
// Extract data rows (skip header row)
const dataRows = jsonData.slice(1).filter((row) => row.some((cell) => cell !== ""));
if (dataRows.length === 0) {
reject(new Error("ไฟล์ไม่มีข้อมูล"));
return;
}
// Convert to array of objects - ใช้ header จาก Excel เป็น field names โดยตรง
// ถ้า maxPreviewRows = 0 จะแสดงทั้งหมด มิฉะนั้นจะแสดงตามจำนวนที่กำหนด
const rowsToProcess = maxPreviewRows === 0 ? dataRows : dataRows.slice(0, maxPreviewRows);
const rows = rowsToProcess.map((row, rowIndex) => {
const obj: Record<string, any> = {
id: `row_${rowIndex}`,
};
// ใช้ header เป็น field names โดยตรง
headers.forEach((header, index) => {
const cellValue = row[index] ?? "";
// ใช้ค่าจาก Excel โดยตรง ไม่แปลงค่า
obj[header] = cellValue;
});
return obj;
});
resolve({
fileName: file.name,
fileSize: file.size,
sheetNames: workbook.SheetNames,
headers,
rows,
totalRows: dataRows.length,
previewRows: rows.length,
});
} catch (error) {
reject(new Error("ไม่สามารถอ่านไฟล์ Excel ได้ กรุณาตรวจสอบไฟล์อีกครั้ง"));
}
};
reader.onerror = () => {
reject(new Error("ไม่สามารถอ่านไฟล์ได้"));
};
reader.readAsBinaryString(file);
});
}
export { formatFileSize };

View file

@ -1,157 +0,0 @@
import ExcelJS from "exceljs";
import { useEditPosDataStore } from "@/modules/04_registryPerson/stores/Edit";
import type { DataPosition } from "@/modules/04_registryPerson/interface/response/Edit";
const store = useEditPosDataStore();
// ฟังก์ชันแปลงวันที่จาก ISO format เป็นรูปแบบ dd/mm/yyyy (เช่น 18/05/2564)
function formatDateToDDMMYYYY(dateString: string | null | Date): string {
if (!dateString) return "";
const date = new Date(dateString);
if (isNaN(date.getTime())) return "";
const day = String(date.getDate()).padStart(2, "0");
const month = String(date.getMonth() + 1).padStart(2, "0");
const year = date.getFullYear() + 543; // แปลงเป็นปีพุทธศักราช
return `${day}/${month}/${year}`;
}
export async function exportToExcelPosition(data: DataPosition[]) {
const workbook = new ExcelJS.Workbook();
const worksheet = workbook.addWorksheet("รายการประวัติตำแหน่งเงินเดือน");
// --- ส่วนที่ 1: สร้าง Master Data Sheet สำหรับอ้างอิง ID ---
// เราจะซ่อนแผ่นงานนี้ไว้ (hidden) เพื่อใช้ทำ Dropdown และ VLOOKUP
const masterSheet = workbook.addWorksheet("MasterData", { state: "hidden" });
const masterData = store.commandCodeData; // [{id: 1, name: "ย้าย"}, ...]
masterData.forEach((item, index) => {
masterSheet.getCell(`A${index + 1}`).value = item.name;
masterSheet.getCell(`B${index + 1}`).value = item.id;
});
// --- ส่วนที่ 2: กำหนด Columns ---
worksheet.columns = [
{ header: "ลำดับ", key: "no", width: 8 },
{ header: "วันที่คำสั่งมีผล", key: "commandDateAffect", width: 18 },
{ header: "ตำแหน่งในสายงาน", key: "positionName", width: 25 },
{ header: "ตำแหน่งประเภท", key: "positionType", width: 18 },
{ header: "ระดับ", key: "positionLevel", width: 12 },
{ header: "ระดับซี", key: "positionCee", width: 12 },
{ header: "สายงาน", key: "positionLine", width: 20 },
{ header: "ด้าน/สาขา", key: "positionPathSide", width: 15 },
{ header: "ตำแหน่งทางการบริหาร", key: "positionExecutive", width: 20 },
{ header: "ด้านทางการบริหาร", key: "positionExecutiveField", width: 20 },
{ header: "เงินเดือน", key: "amount", width: 15 },
{ header: "เงินค่าตอบแทนรายเดือน", key: "mouthSalaryAmount", width: 15 },
{ header: "เงินประจำตำแหน่ง", key: "positionSalaryAmount", width: 15 },
{ header: "เงินค่าตอบแทนพิเศษ", key: "amountSpecial", width: 15 },
{ header: "หน่วยงาน", key: "organization", width: 30 },
{ header: "ส่วนราชการระดับ 1", key: "orgChild1", width: 20 },
{ header: "ส่วนราชการระดับ 2", key: "orgChild2", width: 20 },
{ header: "ส่วนราชการระดับ 3", key: "orgChild3", width: 20 },
{ header: "ส่วนราชการระดับ 4", key: "orgChild4", width: 20 },
{ header: "ตัวย่อเลขที่ตำแหน่ง", key: "posNoAbb", width: 15 },
{ header: "เลขที่ตำแหน่ง", key: "posNo", width: 15 },
{ header: "หน่วยงานที่ออกคำสั่ง", key: "posNumCodeSit", width: 20 },
{
header: "ตัวย่อหน่วยงานที่ออกคำสั่ง",
key: "posNumCodeSitAbb",
width: 15,
},
{ header: "เลขที่คำสั่ง", key: "commandNo", width: 15 },
{ header: "ปีเลขที่คำสั่ง", key: "commandYear", width: 12 },
{ header: "วันที่ลงนาม", key: "commandDateSign", width: 18 },
{ header: "ประเภทคำสั่ง", key: "commandCodeName", width: 25 }, // AA
{ header: "หมายเหตุ", key: "remark", width: 20 },
{ header: "commandId", key: "commandId", width: 20 }, // AC (ลำดับที่ 29)
{ header: "commandCode", key: "commandCode", width: 20 }, // AD (ลำดับที่ 30)
];
// 3. Map ข้อมูล
const newData = data.map((e, index) => ({
no: index + 1,
commandDateAffect: e.commandDateAffect
? formatDateToDDMMYYYY(e.commandDateAffect)
: "",
positionName: e.positionName,
positionType: e.positionType,
positionLevel: e.positionLevel,
positionCee: e.positionCee,
positionLine: e.positionLine || "",
positionPathSide: e.positionPathSide || "",
positionExecutive: e.positionExecutive,
positionExecutiveField: e.positionExecutiveField || "",
amount: e.amount || 0,
mouthSalaryAmount: e.mouthSalaryAmount || 0,
positionSalaryAmount: e.positionSalaryAmount || 0,
amountSpecial: e.amountSpecial || 0,
organization: e.orgRoot,
orgChild1: e.orgChild1,
orgChild2: e.orgChild2,
orgChild3: e.orgChild3,
orgChild4: e.orgChild4,
posNoAbb: e.posNoAbb,
posNo: e.posNo,
posNumCodeSit: e.posNumCodeSit,
posNumCodeSitAbb: e.posNumCodeSitAbb,
commandNo: e.commandNo,
commandYear: e.commandYear ? Number(e.commandYear) + 543 : "",
commandDateSign: e.commandDateSign
? formatDateToDDMMYYYY(e.commandDateSign)
: "",
commandCodeName: store.convertCommandCodeName(e.commandCode),
remark: e.remark,
commandId: e.commandId || "",
commandCode: e.commandCode || "", // ใส่ค่าเริ่มต้นไว้
}));
worksheet.addRows(newData);
// 4. ตกแต่งสี, Dropdown และ สูตร VLOOKUP
newData.forEach((_, index) => {
const rowIndex = index + 2;
// --- ตั้งค่า Format สำหรับเซลล์วันที่ ---
// เซลล์ B (commandDateAffect) และ Z (commandDateSign)
const dateAffectCell = worksheet.getCell(`B${rowIndex}`);
const dateSignCell = worksheet.getCell(`Z${rowIndex}`);
[dateAffectCell, dateSignCell].forEach((cell) => {
cell.numFmt = "dd/mm/yyyy";
});
// --- ทำ Dropdown คอลัมน์ AA ---
// อ้างอิงรายการจาก MasterData Sheet จะทำให้ Excel ทำงานได้เสถียรกว่า (กรณีรายการเยอะ)
worksheet.getCell(`AA${rowIndex}`).dataValidation = {
type: "list",
allowBlank: true,
formulae: [`MasterData!$A$1:$A$${masterData.length}`],
};
// --- ผูกสูตรให้ commandCode (AD) เปลี่ยนตามการเลือกใน AA ---
// AA คือประเภทคำสั่ง, AD คือ commandCode
worksheet.getCell(`AD${rowIndex}`).value = {
formula: `=IFERROR(VLOOKUP(AA${rowIndex}, MasterData!$A$1:$B$${masterData.length}, 2, FALSE), "")`,
};
});
// 5. สไตล์ Header
worksheet.getRow(1).font = { bold: true };
worksheet.getRow(1).alignment = { vertical: "middle", horizontal: "center" };
// 6. เขียนไฟล์
const buffer = await workbook.xlsx.writeBuffer();
const blob = new Blob([buffer], {
type: "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet",
});
const url = window.URL.createObjectURL(blob);
const a = document.createElement("a");
a.href = url;
a.download = "รายการประวัติตำแหน่งเงินเดือน.xlsx";
a.click();
window.URL.revokeObjectURL(url);
}

View file

@ -323,10 +323,10 @@ async function uploadFileURL(uploadUrl: string, file: any) {
* งกนดงขอมลรปโปรไฟล * งกนดงขอมลรปโปรไฟล
* @param id โปรไฟล * @param id โปรไฟล
*/ */
function fetchProfile(id: string) { async function fetchProfile(id: string) {
http await http
.get(config.API.fileByFile("ทะเบียนประวัติ", "โปรไฟล์", id, fileName.value)) .get(config.API.fileByFile("ทะเบียนประวัติ", "โปรไฟล์", id, fileName.value))
.then((res) => { .then(async (res) => {
profilePicture.value = res.data.downloadUrl; profilePicture.value = res.data.downloadUrl;
}) })
.catch(() => { .catch(() => {
@ -403,7 +403,7 @@ async function fetchDataPersonal() {
fileName.value = res.data.result.avatarName; fileName.value = res.data.result.avatarName;
if (formDetail.value?.avatarName) { if (formDetail.value?.avatarName) {
fetchProfile(profileId.value); await fetchProfile(profileId.value);
} else { } else {
profilePicture.value = avatar; profilePicture.value = avatar;
} }
@ -857,8 +857,7 @@ onMounted(async () => {
])) || ])) ||
leaveType === 'DISCIPLINE_TEMP_SUSPEND' || leaveType === 'DISCIPLINE_TEMP_SUSPEND' ||
leaveType === 'RETIRE_RESIGN' || leaveType === 'RETIRE_RESIGN' ||
leaveType === 'DISCIPLINE_SUSPEND' || leaveType === 'DISCIPLINE_SUSPEND'
leaveType === 'RETIRE_MILITARY'
" "
size="md" size="md"
rounded rounded

View file

@ -1,270 +0,0 @@
<script setup lang="ts">
import { ref, computed, watch } from "vue";
import { useQuasar } from "quasar";
import moment from "moment";
import {
parseExcelFile,
formatFileSize,
} from "@/modules/04_registryPerson/utils/excelParser";
import { useCounterMixin } from "@/stores/mixin";
const $q = useQuasar();
const { dialogConfirm } = useCounterMixin();
interface Props {
modal: boolean;
file: File | null;
}
interface Emits {
(e: "update:modal", value: boolean): void;
(e: "confirm", file: File): void;
(e: "cancel"): void;
}
const props = defineProps<Props>();
const emit = defineEmits<Emits>();
const isLoading = ref(false);
const isUploading = ref(false);
const previewData = ref<Record<string, any>[]>([]);
const excelHeaders = ref<string[]>([]); // Headers Excel
const fileName = computed(() => props.file?.name ?? "");
const fileSize = computed(() => props.file?.size ?? 0);
const totalRows = computed(() => previewData.value.length);
const previewRows = computed(() => previewData.value.length);
const pagination = ref({
page: 1,
rowsPerPage: 50,
});
// ( 543 Excel .. )
function formatDateForDisplay(value: any): string {
if (!value) return "-";
// string format dd/mm/yyyy ( Excel ..)
if (typeof value === "string") {
const parts = value.split("/");
if (parts.length === 3) {
const day = parts[0];
const monthNum = parseInt(parts[1], 10);
const year = parts[2];
//
const thaiMonths = [
"ม.ค.",
"ก.พ.",
"มี.ค.",
"เม.ย.",
"พ.ค.",
"มิ.ย.",
"ก.ค.",
"ส.ค.",
"ก.ย.",
"ต.ค.",
"พ.ย.",
"ธ.ค.",
];
const thaiMonth = thaiMonths[monthNum - 1] || parts[1];
return `${day} ${thaiMonth} ${year}`;
}
return value;
}
// Date object 543 ( Excel serial date .. ..)
const dateMoment = moment(value);
const day = dateMoment.format("DD");
const month = dateMoment.format("MMM");
const year = +dateMoment.format("YYYY"); // 543
return `${day} ${month} ${year}`;
}
// Headers
const DATE_HEADERS = ["วันที่คำสั่งมีผล", "วันที่ลงนาม"];
const SALARY_HEADERS = [
"เงินเดือน",
"ค่าจ้าง",
"เงินค่าตอบแทนรายเดือน",
"เงินประจำตำแหน่ง",
"เงินค่าตอบแทนพิเศษ",
];
// columns Excel headers
const tableColumns = computed(() => {
if (excelHeaders.value.length === 0) return [];
return excelHeaders.value.map((header, index) => {
const isDateColumn = DATE_HEADERS.includes(header);
const isSalaryColumn = SALARY_HEADERS.includes(header);
return {
name: `col_${index}`,
label: header,
field: header,
align: "left" as const,
sortable: false,
style: "white-space: nowrap;",
headerStyle: "font-size: 13px; font-weight: bold;",
format: (val: any) => {
// -
if (val == null || val === "") return "-";
// column
if (isDateColumn) {
// string format dd/mm/yyyy ( Excel)
if (
typeof val === "string" &&
val.match(/^\d{1,2}\/\d{1,2}\/\d{4}$/)
) {
return formatDateForDisplay(val);
}
// number (Excel serial date) Date object
if (typeof val === "number") {
return formatDateForDisplay((val - 25569) * 86400 * 1000);
}
}
// column format comma separator ( label)
if (isSalaryColumn) {
const numericValue = Number(val);
if (typeof numericValue === "number" && !isNaN(numericValue)) {
return numericValue.toLocaleString("en-US");
}
}
return val;
},
};
});
});
async function loadPreview() {
if (!props.file) return;
isLoading.value = true;
try {
const result = await parseExcelFile(props.file, {
maxPreviewRows: 0, // 0 =
sheetIndex: 0, // sheet
});
previewData.value = result.rows;
excelHeaders.value = result.headers; // headers Excel
} catch (error) {
console.error("Error parsing Excel file:", error);
previewData.value = [];
excelHeaders.value = [];
} finally {
isLoading.value = false;
}
}
function onConfirm() {
dialogConfirm($q, () => {
if (props.file) {
isUploading.value = true;
emit("confirm", props.file);
}
});
}
function onClose() {
emit("cancel");
emit("update:modal", false);
}
function onModalUpdate(value: boolean) {
emit("update:modal", value);
}
watch(
() => props.modal,
(newValue) => {
if (newValue && props.file) {
isUploading.value = false;
loadPreview();
}
}
);
</script>
<template>
<q-dialog :model-value="modal" @update:model-value="onModalUpdate" persistent>
<q-card style="min-width: 80vw; max-width: 95vw">
<q-card-section class="row items-center q-pb-none">
<div class="text-h6">วอยางขอมลไฟล Excel</div>
<q-space />
<q-btn icon="close" flat round dense v-close-popup @click="onClose" />
</q-card-section>
<q-separator />
<!-- File Info Section -->
<q-card-section>
<div class="row q-col-gutter-md">
<div class="col-12">
<div class="text-subtitle2 text-grey-7">อมลไฟล</div>
</div>
<div class="col-md-4 col-sm-6 col-12">
<div class="flex items-center">
<q-icon
name="insert_drive_file"
class="q-mr-sm"
color="primary"
/>
<span class="text-body2">อไฟล: {{ fileName }}</span>
</div>
</div>
<div class="col-md-4 col-sm-6 col-12">
<div class="flex items-center">
<q-icon name="folder" class="q-mr-sm" color="primary" />
<span class="text-body2"
>ขนาด: {{ formatFileSize(fileSize) }}</span
>
</div>
</div>
<div class="col-md-4 col-sm-12 col-12">
<div class="flex items-center">
<q-icon name="table_chart" class="q-mr-sm" color="primary" />
<span class="text-body2">จำนวนแถว: {{ totalRows }} แถว</span>
</div>
</div>
</div>
</q-card-section>
<q-separator />
<!-- Preview Table -->
<q-card-section style="max-height: 70vh" class="scroll">
<div class="text-subtitle2 q-mb-md">
รายการตำแหน ({{ previewRows }} รายการ)
</div>
<d-table
:columns="tableColumns"
:rows="previewData"
:paging="true"
:rows-per-page-options="[20, 50, 100, 0]"
v-model:pagination="pagination"
:loading="isLoading"
row-key="id"
flat
bordered
dense
/>
</q-card-section>
<!-- Actions -->
<q-card-actions align="right" class="q-pa-md">
<q-btn
type="submit"
color="public"
label="ยืนยันการอัปโหลด"
@click="onConfirm"
:loading="isUploading"
/>
</q-card-actions>
</q-card>
</q-dialog>
</template>

View file

@ -706,7 +706,6 @@ function classInput(val: boolean) {
hide-bottom-space hide-bottom-space
autocomplete="on" autocomplete="on"
name="organization" name="organization"
:label="`${'หน่วยงาน'}`"
/> />
</div> </div>

View file

@ -1,13 +1,13 @@
<script setup lang="ts"> <script setup lang="ts">
import { ref, onMounted, computed, reactive } from "vue"; import { ref, onMounted, computed, reactive } from "vue";
import { useQuasar } from "quasar"; import { useQuasar } from "quasar";
import * as XLSX from "xlsx";
import http from "@/plugins/http"; import http from "@/plugins/http";
import config from "@/app.config"; import config from "@/app.config";
import { useRoute } from "vue-router"; import { useRoute } from "vue-router";
import { useCounterMixin } from "@/stores/mixin"; import { useCounterMixin } from "@/stores/mixin";
import { useEditPosDataStore } from "@/modules/04_registryPerson/stores/Edit"; import { useEditPosDataStore } from "@/modules/04_registryPerson/stores/Edit";
import { exportToExcelPosition } from "@/modules/04_registryPerson/utils/exportPosition";
import type { QTableColumn } from "quasar"; import type { QTableColumn } from "quasar";
import type { FormDataSalary } from "@/modules/04_registryPerson/interface/index/Edit"; import type { FormDataSalary } from "@/modules/04_registryPerson/interface/index/Edit";
@ -20,7 +20,6 @@ import type {
import DialogForm from "@/modules/04_registryPerson/views/edit/components/DialogForm.vue"; import DialogForm from "@/modules/04_registryPerson/views/edit/components/DialogForm.vue";
import DialogSort from "@/modules/04_registryPerson/views/edit/components/DialogSort.vue"; import DialogSort from "@/modules/04_registryPerson/views/edit/components/DialogSort.vue";
import DialogExcelPreview from "@/modules/04_registryPerson/views/edit/components/DialogExcelPreview.vue";
import CurruncyInput from "@/components/CurruncyInput.vue"; import CurruncyInput from "@/components/CurruncyInput.vue";
const $q = useQuasar(); const $q = useQuasar();
@ -53,11 +52,6 @@ const amountRef = ref<any>(null);
const amountSpecialRef = ref<any>(null); const amountSpecialRef = ref<any>(null);
const currencyPopupRef = ref<any>(null); const currencyPopupRef = ref<any>(null);
// Excel Preview
const excelPreviewModal = ref<boolean>(false);
const selectedExcelFile = ref<File | null>(null);
const isParsingExcel = ref<boolean>(false);
//Table //Table
const isLoad = ref<boolean>(true); const isLoad = ref<boolean>(true);
const rowIndex = ref<number>(0); const rowIndex = ref<number>(0);
@ -436,7 +430,66 @@ function classColorRow(isDelete: boolean, isEdit: boolean, isEntry: boolean) {
/** ฟังก์ชันดาวน์โหลดไฟล Excel */ /** ฟังก์ชันดาวน์โหลดไฟล Excel */
function exportToExcel() { function exportToExcel() {
exportToExcelPosition(rows.value); const newData = rows.value.map((e: DataPosition) => {
return {
commandDateAffect: date2Thai(e.commandDateAffect),
positionName: e.positionName,
positionType: e.positionType,
positionLevel: e.positionLevel
? e.positionLevel
: e.positionCee
? e.positionCee
: "",
positionExecutive: e.positionExecutive,
amount: e.amount,
mouthSalaryAmount: e.mouthSalaryAmount,
positionSalaryAmount: e.positionSalaryAmount,
organization: findOrgName({
root: e.orgRoot,
child1: e.orgChild1,
child2: e.orgChild2,
child3: e.orgChild3,
child4: e.orgChild4,
}),
posNo:
e.posNoAbb && e.posNo
? `${e.posNoAbb} ${e.posNo}`
: e.posNo
? e.posNo
: "",
posNumCodeSit:
e.posNumCodeSitAbb && e.posNumCodeSit
? `${e.posNumCodeSit} (${e.posNumCodeSitAbb})`
: e.posNumCodeSit
? e.posNumCodeSit
: "",
commandNo:
e.commandNo && e.commandYear
? `${e.commandNo}/${Number(e.commandYear) + 543}`
: "",
commandDateSign: date2Thai(e.commandDateSign),
commandCode: store.convertCommandCodeName(e.commandCode),
remark: e.remark,
};
});
const headers = columns.value.map((item: any) => item.label) || []; //
const worksheet = XLSX.utils.json_to_sheet(newData, {
header: visibleColumns.value,
});
// ( A1, B1, C1 )
XLSX.utils.sheet_add_aoa(worksheet, [headers], { origin: "A1" });
// Create a new workbook and append the worksheet
const workbook = XLSX.utils.book_new();
XLSX.utils.book_append_sheet(
workbook,
worksheet,
`รายการประวัติตำแหน่งเงินเดือน`
);
XLSX.writeFile(workbook, "รายการประวัติตำแหน่งเงินเดือน.xlsx");
} }
const commandCodeOptions = ref<DataOption[]>(store.commandCodeData); // const commandCodeOptions = ref<DataOption[]>(store.commandCodeData); //
@ -794,106 +847,6 @@ async function validateAndSave(
} }
} }
/**
* งกนอปโหลดไฟล Excel
* เป preview modal อนอปโหลด
*/
function handUploadFile() {
const input = document.createElement("input");
input.type = "file";
input.accept = ".xlsx,.xls";
input.onchange = async (e: Event) => {
const file = (e.target as HTMLInputElement).files?.[0];
if (!file) return;
//
const validExtensions = [".xlsx", ".xls"];
const fileExtension = file.name
.substring(file.name.lastIndexOf("."))
.toLowerCase();
if (!validExtensions.includes(fileExtension)) {
messageError($q, {
response: {
data: {
title: "กรุณาเลือกไฟล์ Excel เท่านั้น (.xlsx, .xls)",
},
},
});
return;
}
// preview modal
selectedExcelFile.value = file;
excelPreviewModal.value = true;
};
input.click();
}
/**
* งกนยนยนการอปโหลดไฟล
* งไฟลไปย API เม user นยนจาก preview modal
*/
async function onConfirmUpload(file: File) {
try {
isParsingExcel.value = true;
showLoader();
const type = empType.value === "officer" ? "office" : "employee";
const formData = new FormData();
formData.append("file", file);
await http.post(config.API.uploadProfile(type, profileId.value), formData, {
headers: {
"Content-Type": "multipart/form-data",
},
});
success($q, "อัปโหลดไฟล์สำเร็จ");
await fetchData();
excelPreviewModal.value = false;
} catch (error) {
messageError($q, error);
} finally {
hideLoader();
isParsingExcel.value = false;
}
}
/**
* งกนยกเลกการอปโหลด
* preview modal และลางค
*/
function onCancelUpload() {
selectedExcelFile.value = null;
excelPreviewModal.value = false;
}
/** ฟังก์ชันจัดเรียงข้อมูลตามวันที่*/
function handleSortByDate() {
dialogConfirm(
$q,
async () => {
try {
showLoader();
await http.put(config.API.sortOrderByDate, {
type: empType.value.toLocaleUpperCase(),
profileId: profileId.value,
});
await fetchData();
} catch (error) {
messageError($q, error);
} finally {
hideLoader();
}
},
"ยืนยันการจัดลำดับ",
"ต้องการยืนยันการจัดลำดับข้อมูลตามวันที่คำสั่งมีผลใช่หรือไม่?"
);
}
onMounted(async () => { onMounted(async () => {
await Promise.all([fetchData(), fetchType()]); await Promise.all([fetchData(), fetchType()]);
}); });
@ -929,39 +882,10 @@ onMounted(async () => {
icon="mdi-sort" icon="mdi-sort"
@click="modalSort = true" @click="modalSort = true"
> >
<q-tooltip>ดลำดบขอม</q-tooltip> <q-tooltip>ดลำดบขอม</q-tooltip></q-btn
</q-btn>
<q-btn
v-if="
tabs === 'PENDING' && statusCheckEdit == 'PENDING' && isConfirmEdit
"
class="q-ml-sm"
round
flat
dense
color="indigo-5"
icon="mdi-calendar-export"
@click="handleSortByDate()"
> >
<q-tooltip>ดลำดบตามวนทคำสงมผล</q-tooltip>
</q-btn>
<q-space /> <q-space />
<div> <div>
<q-btn
v-if="
tabs === 'PENDING' &&
statusCheckEdit == 'PENDING' &&
isConfirmEdit
"
flat
round
color="blue"
icon="upload"
@click="handUploadFile()"
>
<q-tooltip>ปโหลดไฟลแกไข</q-tooltip>
</q-btn>
<q-btn <q-btn
flat flat
round round
@ -969,9 +893,7 @@ onMounted(async () => {
icon="download" icon="download"
:disable="rows.length == 0" :disable="rows.length == 0"
@click="exportToExcel()" @click="exportToExcel()"
> />
<q-tooltip>ดาวนโหลดไฟล</q-tooltip>
</q-btn>
</div> </div>
<q-input <q-input
@ -1802,13 +1724,6 @@ onMounted(async () => {
:fetch-data="fetchData" :fetch-data="fetchData"
:columns="columns" :columns="columns"
/> />
<DialogExcelPreview
v-model:modal="excelPreviewModal"
:file="selectedExcelFile"
@confirm="onConfirmUpload"
@cancel="onCancelUpload"
/>
</template> </template>
<style scoped></style> <style scoped></style>

View file

@ -54,7 +54,6 @@ const reason = ref<string>(""); //หมายเหตุ
const status = ref<string>(""); const status = ref<string>("");
const fullName = ref<string>(""); const fullName = ref<string>("");
const mianData = ref<MainData>(); const mianData = ref<MainData>();
const typeCommand = ref<string>(""); //
/** fetch รายละเอียดการปรับระดับชั้นงานลูกจ้าง*/ /** fetch รายละเอียดการปรับระดับชั้นงานลูกจ้าง*/
async function fecthappointmentByid() { async function fecthappointmentByid() {
@ -80,7 +79,6 @@ async function fecthappointmentByid() {
salary.value = data.salary ?? 0; salary.value = data.salary ?? 0;
reason.value = data.reason; reason.value = data.reason;
date.value = data.positionDate; date.value = data.positionDate;
typeCommand.value = data.typeCommand;
}) })
.catch((e) => { .catch((e) => {
messageError($q, e); messageError($q, e);
@ -325,10 +323,7 @@ onMounted(async () => {
</div> </div>
<div class="col-12"><q-separator /></div> <div class="col-12"><q-separator /></div>
<div <div class="col-xs-6 col-sm-6 row items-center">
class="col-xs-6 col-sm-6 row items-center"
v-if="typeCommand !== 'MOVE'"
>
<div class="col-12"> <div class="col-12">
<datepicker <datepicker
:readonly="!edit" :readonly="!edit"

View file

@ -45,7 +45,6 @@ const date = ref<Date | null>(null); //ดำรงตำแหน่งใน
const reason = ref<string>(""); // const reason = ref<string>(""); //
const status = ref<string>(""); const status = ref<string>("");
const fullName = ref<string>(""); const fullName = ref<string>("");
const typeCommand = ref<string>(""); //
/** fetch รายละเอียดการแต่งตั้ง-เลื่อน-ย้าย*/ /** fetch รายละเอียดการแต่งตั้ง-เลื่อน-ย้าย*/
async function fecthappointmentByid() { async function fecthappointmentByid() {
@ -62,7 +61,6 @@ async function fecthappointmentByid() {
}`; }`;
status.value = data.status; status.value = data.status;
typeCommand.value = data.typeCommand;
educationOld.value = data.educationOld ?? "-"; educationOld.value = data.educationOld ?? "-";
organizationPositionOld.value = data.organizationPositionOld; organizationPositionOld.value = data.organizationPositionOld;
positionTypeOld.value = data.positionTypeOld; positionTypeOld.value = data.positionTypeOld;
@ -315,10 +313,7 @@ onMounted(() => {
</div> </div>
<div class="col-12"><q-separator /></div> <div class="col-12"><q-separator /></div>
<div <div class="col-xs-6 col-sm-6 row items-center">
class="col-xs-6 col-sm-6 row items-center"
v-if="typeCommand !== 'MOVE'"
>
<div class="col-12"> <div class="col-12">
<datepicker <datepicker
:readonly="!edit" :readonly="!edit"

View file

@ -316,10 +316,9 @@ onMounted(() => {
:model-value=" :model-value="
dateEnd !== null ? date2Thai(dateEnd) : null dateEnd !== null ? date2Thai(dateEnd) : null
" "
:rules="edit ? [(val:string) => !!val || `${'กรุณาเลือกตั้งแต่วัน'}`]:[]"
hide-bottom-space hide-bottom-space
:label="`${'ถึงวันที่'}`" :label="`${'ถึงวันที่'}`"
clearable
@clear="dateEnd = null"
> >
<template v-slot:prepend> <template v-slot:prepend>
<q-icon <q-icon

View file

@ -2,7 +2,7 @@
import { ref, onMounted, watch } from "vue"; import { ref, onMounted, watch } from "vue";
/** importType*/ /** importType*/
import { type QTableProps } from "quasar"; import type { QTableProps } from "quasar";
import type { import type {
Positions, Positions,
FormPosType, FormPosType,
@ -22,14 +22,14 @@ const store = useSelectOrgStore();
/** props*/ /** props*/
const selected = defineModel("selectedPos", { required: true }); // const selected = defineModel("selectedPos", { required: true }); //
const positionId = defineModel<string>("positionId", { required: true }); //id const positionId = defineModel<string>("positionId", { required: true }); //id
const selectId = defineModel<string>("selectId", { required: true }); // const seletcId = defineModel<string>("seletcId", { required: true }); //
const date = defineModel<Date | null>("datePos", { required: true }); // const date = defineModel<Date>("datePos", { required: true }); //
const positionData = defineModel<any[]>("position", { required: true }); // const positionData = defineModel<any[]>("position", { required: true }); //
const isAll = defineModel<boolean>("isAll", { required: true }); // const isAll = defineModel<boolean>("isAll", { required: true }); //
const isBlank = defineModel<boolean>("isBlank", { required: true }); // const isBlank = defineModel<boolean>("isBlank", { required: true }); //
const isPosition = defineModel<string>("isPosition", { required: true }); const isPosition = defineModel<string>("isPosition", { required: true });
// const isPositionOld = defineModel<boolean>("isPositionOld", { required: true }); // // const isPositionOld = defineModel<boolean>("isPositionOld", { required: true }); //
const posType = defineModel<FormPosType | null>("posType", { required: true }); // const posType = defineModel<FormPosType>("posType", { required: true }); //
const posLevel = defineModel<string>("posLevel", { required: true }); // const posLevel = defineModel<string>("posLevel", { required: true }); //
const optionPosType = defineModel<FormPosType[]>("optionPosType", { const optionPosType = defineModel<FormPosType[]>("optionPosType", {
required: true, required: true,
@ -47,7 +47,6 @@ const props = defineProps({
onPosType: Function, onPosType: Function,
nodeId: String, nodeId: String,
nodeLevel: Number, nodeLevel: Number,
isLoadPosition: Boolean,
}); });
//Table //Table
@ -214,9 +213,9 @@ async function onClickSelectPos(id: string) {
// //
if (position) { if (position) {
rowsPosition.value = position.positions; rowsPosition.value = position.positions;
if (selectId.value) { if (seletcId.value) {
selected.value = rowsPosition.value.filter( selected.value = rowsPosition.value.filter(
(e) => e.id === selectId.value (e) => e.id === seletcId.value
); );
} }
} }
@ -249,20 +248,15 @@ watch(positionData, (newVal, oldVal) => {
} }
}); });
watch(
() => props.isLoadPosition,
(newVal) => {
if (newVal && positionId.value) {
onClickSelectPos(positionId.value);
}
}
);
/** /**
* ทำงานเม Components กเรยกใชงาน * ทำงานเม Components กเรยกใชงาน
*/ */
onMounted(async () => { onMounted(async () => {
if (!positionId.value) { if (positionId.value) {
setTimeout(async () => {
await onClickSelectPos(positionId.value);
}, 1000);
} else {
positionRows.value = positionData.value; positionRows.value = positionData.value;
positionRowsData.value = positionData.value; positionRowsData.value = positionData.value;
} }

View file

@ -242,7 +242,7 @@ watch(
v.code === "C-PM-04" || v.code === "C-PM-04" ||
v.code === "C-PM-06")) || v.code === "C-PM-06")) ||
((status?.examTypeName === "สอบคัดเลือก" || ((status?.examTypeName === "สอบคัดเลือก" ||
status?.examTypeName === "คัดเลือกอื่นๆ") && status?.examTypeName === "คัดเลือกคนพิการ") &&
(v.code === "C-PM-02" || (v.code === "C-PM-02" ||
v.code === "C-PM-03" || v.code === "C-PM-03" ||
v.code === "C-PM-04" || v.code === "C-PM-04" ||

View file

@ -1,5 +1,5 @@
<script setup lang="ts"> <script setup lang="ts">
import { onMounted, ref, watch } from "vue"; import { onMounted, ref, watch, reactive } from "vue";
import { useQuasar } from "quasar"; import { useQuasar } from "quasar";
import { useRoute } from "vue-router"; import { useRoute } from "vue-router";
@ -11,7 +11,6 @@ import { useStructureTree } from "@/stores/structureTree";
/** importType*/ /** importType*/
import type { import type {
DataList,
PositionMaim, PositionMaim,
PositionNo, PositionNo,
Positions, Positions,
@ -42,14 +41,17 @@ const {
/**props*/ /**props*/
const modal = defineModel<boolean>("modal", { required: true }); const modal = defineModel<boolean>("modal", { required: true });
const props = defineProps({
interface Props { dataRow: {
dataRow?: DataList; type: Object,
fetchTable?: () => Promise<void>; require: true,
fetchStatCard?: () => Promise<void>; },
} fetchTable: {
type: Function,
const props = defineProps<Props>(); require: true,
},
fetchStatCard: { type: Function, require: true },
});
/** Tree*/ /** Tree*/
const nodeId = ref<string>(""); const nodeId = ref<string>("");
@ -63,16 +65,15 @@ const expanded = ref<string[]>([]);
const positionUse = ref<string[]>([]); const positionUse = ref<string[]>([]);
const positionNo = ref<DataPositionNo[]>([]); const positionNo = ref<DataPositionNo[]>([]);
const positionId = ref<string>(""); const positionId = ref<string>("");
const selectId = ref<string>(""); const seletcId = ref<string>("");
const posType = ref<FormPosType | null>(null); const posType = ref<FormPosType | null>(null);
const posLevel = ref<string>(""); const posLevel = ref<string>("");
const selectedPos = ref<Positions[]>([]); const selectedPos = ref<any[]>([]);
const datePos = ref<Date | null>(new Date()); const datePos = ref<Date>(new Date());
const posMasterMain = ref<PositionMaim[]>([]); const posMasterMain = ref<any[]>([]);
const orgRevisionId = ref<string>(""); const orgRevisionId = ref<string>("");
const optionPosType = ref<FormPosType[]>([]); const optionPosType = ref<FormPosType[]>([]);
const optionPosLevel = ref<FormPosLevel[]>([]); const optionPosLevel = ref<FormPosLevel[]>([]);
const isLoadPosition = ref<boolean>(false);
/** function เรียกข้อมูลโครงสร้าง แบบปัจุบันและ แบบร่าง*/ /** function เรียกข้อมูลโครงสร้าง แบบปัจุบันและ แบบร่าง*/
async function fetchStructure() { async function fetchStructure() {
@ -90,14 +91,12 @@ async function fetchStructure() {
*/ */
function updateSelected(data: TreeMain) { function updateSelected(data: TreeMain) {
if (props?.dataRow?.nodeId === data.orgTreeId) { if (props?.dataRow?.nodeId === data.orgTreeId) {
positionId.value = props?.dataRow?.posmasterId ?? ""; positionId.value = props?.dataRow?.posmasterId;
selectId.value = props?.dataRow?.positionId ?? ""; seletcId.value = props?.dataRow?.positionId;
datePos.value = props?.dataRow?.reportingDate datePos.value = props?.dataRow?.reportingDate;
? new Date(props.dataRow.reportingDate)
: new Date();
} else { } else {
positionId.value = ""; positionId.value = "";
selectId.value = ""; seletcId.value = "";
selectedPos.value = []; selectedPos.value = [];
datePos.value = new Date(); datePos.value = new Date();
} }
@ -118,8 +117,6 @@ const isPosition = ref<string>("exam");
// const isPositionOld = ref<boolean>(false); // const isPositionOld = ref<boolean>(false);
async function fetchDataTable(id: string, level: number = 0) { async function fetchDataTable(id: string, level: number = 0) {
showLoader(); showLoader();
isLoadPosition.value = false;
positionNo.value = [];
const body = { const body = {
node: level, node: level,
nodeId: id, nodeId: id,
@ -153,7 +150,7 @@ async function fetchDataTable(id: string, level: number = 0) {
if (p.length !== 0) { if (p.length !== 0) {
const a = p.find((el: Positions) => el.positionIsSelected === true); const a = p.find((el: Positions) => el.positionIsSelected === true);
const { id, ...rest } = a ? a : p[0]; const { id, ...rest } = a ? a : p[0];
const data: PositionMaim = { ...e, ...rest } as PositionMaim; const data: any = { ...e, ...rest };
dataMain.push(data); dataMain.push(data);
} }
}); });
@ -165,22 +162,21 @@ async function fetchDataTable(id: string, level: number = 0) {
(e) => e !== props.dataRow?.posmasterId (e) => e !== props.dataRow?.posmasterId
); );
positionNo.value = posMain.filter( positionNo.value = posMain.filter((e: any) => !newUse.includes(e.id));
(e: DataPositionNo) => !newUse.includes(e.id)
);
} else { } else {
positionNo.value = posMain.filter( positionNo.value = posMain.filter(
(e: DataPositionNo) => !positionUse.value.includes(e.id) (e: any) => !positionUse.value.includes(e.id)
); );
} }
isLoadPosition.value = true;
}) })
.catch((err) => { .catch((err) => {
messageError($q, err); messageError($q, err);
hideLoader();
}) })
.finally(() => { .finally(() => {
hideLoader(); setTimeout(() => {
hideLoader();
}, 1000);
}); });
} }
@ -202,11 +198,9 @@ async function fetchPosFind(level: number, id: string) {
expanded.value = data; expanded.value = data;
nodeId.value = id; nodeId.value = id;
positionId.value = props?.dataRow?.posmasterId ?? ""; positionId.value = props?.dataRow?.posmasterId;
selectId.value = props?.dataRow?.positionId ?? ""; seletcId.value = props?.dataRow?.positionId;
datePos.value = props?.dataRow?.reportingDate datePos.value = props?.dataRow?.reportingDate;
? new Date(props.dataRow.reportingDate)
: new Date();
fetchDataTable(nodeId.value, level); fetchDataTable(nodeId.value, level);
}) })
@ -218,14 +212,12 @@ async function fetchPosFind(level: number, id: string) {
/** function บันทึกข้อมูลตำแหน่ง*/ /** function บันทึกข้อมูลตำแหน่ง*/
async function onClickSubmit() { async function onClickSubmit() {
const dataPosMaster = posMasterMain.value?.find( const dataPosMaster = await posMasterMain.value?.find(
(e) => e.id === positionId.value (e: any) => e.id === positionId.value
); );
if (selectedPos.value.length === 0) { if (selectedPos.value.length === 0) {
dialogMessageNotify($q, "กรุณาเลือกตำแหน่ง"); dialogMessageNotify($q, "กรุณาเลือกตำแหน่ง");
} else if (!dataPosMaster) {
dialogMessageNotify($q, "ไม่พบข้อมูลตำแหน่ง");
} else { } else {
dialogConfirm($q, async () => { dialogConfirm($q, async () => {
showLoader(); showLoader();
@ -243,7 +235,6 @@ async function onClickSubmit() {
posLevelId: selectedPos.value[0].posLevelId, // posLevelId: selectedPos.value[0].posLevelId, //
posLevelName: selectedPos.value[0].posLevelName, // posLevelName: selectedPos.value[0].posLevelName, //
posExecutiveName: selectedPos.value[0].posExecutiveName, posExecutiveName: selectedPos.value[0].posExecutiveName,
posExecutiveId: selectedPos.value[0].posExecutiveId,
reportingDate: convertDateToAPI(datePos.value), reportingDate: convertDateToAPI(datePos.value),
posmasterId: dataPosMaster.id, posmasterId: dataPosMaster.id,
positionExecutiveField: selectedPos.value[0].positionExecutiveField, // positionExecutiveField: selectedPos.value[0].positionExecutiveField, //
@ -281,9 +272,6 @@ function clearData() {
posLevel.value = ""; posLevel.value = "";
isPosition.value = "exam"; isPosition.value = "exam";
filterTree.value = ""; filterTree.value = "";
datePos.value = new Date();
isLoadPosition.value = false;
positionNo.value = [];
} }
/** callback function เมื่อมีการเปิด popup*/ /** callback function เมื่อมีการเปิด popup*/
@ -292,11 +280,9 @@ watch(
async () => { async () => {
if (modal.value) { if (modal.value) {
await fetchPositionUes(); await fetchPositionUes();
if (props?.dataRow?.node !== null && props?.dataRow?.nodeId !== null) { if (props?.dataRow?.node !== null && props?.dataRow?.nodeId !== null) {
await fetchPosFind( await fetchPosFind(props?.dataRow?.node, props?.dataRow?.nodeId);
props?.dataRow?.node ?? 0,
props?.dataRow?.nodeId ?? ""
);
} else { } else {
expanded.value = []; expanded.value = [];
} }
@ -337,7 +323,8 @@ async function getOrgPosType() {
}) })
.catch((e) => { .catch((e) => {
messageError($q, e); messageError($q, e);
}); })
.finally(() => {});
} }
function onPosType() { function onPosType() {
@ -349,15 +336,28 @@ function onPosType() {
} }
watch( watch(
[isAll, isBlank, () => isPosition.value], () => isAll.value,
([newAll, newBlank, newPos], [oldAll, oldBlank, oldPos]) => { (value, oldVal) => {
const shouldFetch = newAll !== oldAll || newBlank !== oldBlank; if (value !== oldVal) {
const isSelectMode = newPos === "select" && oldPos !== "select";
if (shouldFetch || isSelectMode) {
fetchDataTable(nodeId.value, nodeLevel.value); fetchDataTable(nodeId.value, nodeLevel.value);
} }
if (isSelectMode) { }
);
watch(
() => isBlank.value,
(value, oldVal) => {
if (value !== oldVal) {
fetchDataTable(nodeId.value, nodeLevel.value);
}
}
);
watch(
() => isPosition.value === "select",
(value, oldVal) => {
if (value !== oldVal) {
fetchDataTable(nodeId.value, nodeLevel.value);
getOrgPosType(); getOrgPosType();
} }
} }
@ -458,15 +458,15 @@ onMounted(() => {
:name="item" :name="item"
> >
<CardPosition <CardPosition
v-model:position="positionNo" v-model:position="positionNo as []"
v-model:selectedPos="selectedPos" v-model:selectedPos="selectedPos"
v-model:datePos="datePos" v-model:datePos="datePos"
v-model:positionId="positionId" v-model:positionId="positionId"
v-model:selectId="selectId" v-model:seletcId="seletcId"
v-model:is-all="isAll" v-model:is-all="isAll"
v-model:is-blank="isBlank" v-model:is-blank="isBlank"
v-model:is-position="isPosition" v-model:is-position="isPosition"
v-model:pos-type="posType" v-model:pos-type="posType as FormPosType"
v-model:pos-level="posLevel" v-model:pos-level="posLevel"
v-model:option-pos-type="optionPosType" v-model:option-pos-type="optionPosType"
v-model:option-pos-level="optionPosLevel" v-model:option-pos-level="optionPosLevel"
@ -474,8 +474,7 @@ onMounted(() => {
:on-pos-type="onPosType" :on-pos-type="onPosType"
:node-id="nodeId" :node-id="nodeId"
:node-level="nodeLevel" :node-level="nodeLevel"
:bma-officer="props.dataRow?.bmaOfficer ?? ''" :bma-officer="props.dataRow?.bmaOfficer"
:is-load-position="isLoadPosition"
/> />
</q-tab-panel> </q-tab-panel>
</q-tab-panels> </q-tab-panels>

View file

@ -1,5 +1,5 @@
<script setup lang="ts"> <script setup lang="ts">
import { ref, onMounted, watch, reactive, computed, type PropType } from "vue"; import { ref, onMounted, watch, reactive, computed } from "vue";
import { useQuasar, QForm } from "quasar"; import { useQuasar, QForm } from "quasar";
import http from "@/plugins/http"; import http from "@/plugins/http";
@ -10,7 +10,6 @@ import { checkPermission } from "@/utils/permissions";
import { useCounterMixin } from "@/stores/mixin"; import { useCounterMixin } from "@/stores/mixin";
import { usePlacementDataStore } from "@/modules/05_placement/store"; import { usePlacementDataStore } from "@/modules/05_placement/store";
import { useMenuDataStore } from "@/stores/menuList"; import { useMenuDataStore } from "@/stores/menuList";
import { validateFileSize } from "@/utils/function";
import avatar from "@/assets/avatar_user.jpg"; import avatar from "@/assets/avatar_user.jpg";
import type { PartialTableName } from "@/modules/05_placement/interface/request/placement"; import type { PartialTableName } from "@/modules/05_placement/interface/request/placement";
@ -48,7 +47,7 @@ const {
/** รับค่ามาจากหน้าหลัก */ /** รับค่ามาจากหน้าหลัก */
const props = defineProps({ const props = defineProps({
statCard: { statCard: {
type: Function as PropType<() => Promise<void>>, type: Function,
default: () => console.log("getStat"), default: () => console.log("getStat"),
}, },
}); });
@ -1057,8 +1056,7 @@ onMounted(async () => {
checkPermission($route)?.attrIsUpdate && checkPermission($route)?.attrIsUpdate &&
props.row.isDraft && props.row.isDraft &&
props.row.statusId === 'PREPARE-CONTAIN' && props.row.statusId === 'PREPARE-CONTAIN' &&
(DataStore.isOfficer || (DataStore.isOfficer || checkPermission($route)?.attrOwnership == 'OWNER')
checkPermission($route)?.attrOwnership == 'OWNER')
" "
clickable clickable
v-close-popup v-close-popup
@ -1545,10 +1543,7 @@ onMounted(async () => {
:label="`${'เลือกไฟล์เอกสารหลักฐาน'}`" :label="`${'เลือกไฟล์เอกสารหลักฐาน'}`"
outlined outlined
use-chips use-chips
:rules="[ :rules="[(val:string) => !!val || 'กรุณาเลือกไฟล์เอกสารหลักฐาน']"
(val) => !!val || 'กรุณาเลือกไฟล์เอกสารหลักฐาน',
(val) => validateFileSize(val),
]"
multiple multiple
@update:model-value="clickEditRow" @update:model-value="clickEditRow"
class="q-py-sm" class="q-py-sm"

View file

@ -20,7 +20,7 @@ const dataMapToSend = computed(() => {
return selected.value.map((i: any) => ({ return selected.value.map((i: any) => ({
id: i.id, id: i.id,
profileId: i.profileId, profileId: i.profileId,
prefix: i.rank ? i.rank : i.prefix, prefix: i.prefix,
firstName: i.firstName, firstName: i.firstName,
lastName: i.lastName, lastName: i.lastName,
citizenId: i.citizenId, citizenId: i.citizenId,
@ -89,9 +89,7 @@ const columns2 = ref<QTableProps["columns"]>([
sort: (a: string, b: string) => sort: (a: string, b: string) =>
a.localeCompare(b, undefined, { numeric: true, sensitivity: "base" }), a.localeCompare(b, undefined, { numeric: true, sensitivity: "base" }),
format(val, row) { format(val, row) {
return `${row.rank ? row.rank : row.prefix ?? ""}${row.firstName ?? ""} ${ return `${row.prefix ?? ""}${row.firstName ?? ""} ${row.lastName ?? ""}`;
row.lastName ?? ""
}`;
}, },
}, },
{ {
@ -112,11 +110,9 @@ const columns2 = ref<QTableProps["columns"]>([
headerStyle: "font-size: 14px", headerStyle: "font-size: 14px",
style: "font-size: 14px", style: "font-size: 14px",
format(val, row) { format(val, row) {
return row.positionTypeOld && row.positionTypeOl !== "-" return row.positionTypeOld
? `${row.positionTypeOld}${ ? `${row.positionTypeOld}${
row.positionLevelOld && row.positionLevelOld !== "-" row.positionLevelOld ? `(${row.positionLevelOld})` : ""
? `(${row.positionLevelOld})`
: ""
}` }`
: ""; : "";
}, },

View file

@ -48,7 +48,6 @@ const informaData = ref<FormAddPerson>({
employeeType: null, employeeType: null,
employeeClass: null, employeeClass: null,
profileType: null, profileType: null,
rank: "",
}); });
// //
@ -61,7 +60,6 @@ const Ops = ref<InformationOps>({
religionOps: [], religionOps: [],
employeeClassOps: [], employeeClassOps: [],
employeeTypeOps: [], employeeTypeOps: [],
rankOps: [],
}); });
// //
const OpsFilter = ref<InformationOps>({ const OpsFilter = ref<InformationOps>({
@ -73,7 +71,6 @@ const OpsFilter = ref<InformationOps>({
religionOps: [], religionOps: [],
employeeClassOps: [], employeeClassOps: [],
employeeTypeOps: [], employeeTypeOps: [],
rankOps: [],
}); });
// profile // profile
@ -138,16 +135,6 @@ async function fetchPerson() {
}); });
Ops.value.religionOps = optionreligions; Ops.value.religionOps = optionreligions;
OpsFilter.value.religionOps = optionreligions; OpsFilter.value.religionOps = optionreligions;
let rank: DataOption[] = [];
data.rank.map((r: DataOptioninfo) => {
rank.push({
id: r.id.toString(),
name: r.name.toString(),
});
});
Ops.value.rankOps = rank;
OpsFilter.value.rankOps = rank;
}) })
.catch((e) => { .catch((e) => {
messageError($q, e); messageError($q, e);
@ -217,13 +204,7 @@ function filterSelector(val: string, update: Function, refData: string) {
); );
}); });
break; break;
case "rankOps":
update(() => {
Ops.value.rankOps = OpsFilter.value.rankOps.filter(
(v: DataOption) => v.name.toLowerCase().indexOf(newVal) > -1
);
});
break;
default: default:
break; break;
} }
@ -246,7 +227,7 @@ function onSubmit() {
if (fileData.value != null) formData.append("File", fileData.value); // if (fileData.value != null) formData.append("File", fileData.value); //
if (informaData.value.citizenId != undefined) if (informaData.value.citizenId != undefined)
formData.append("citizenId", informaData.value.citizenId); formData.append("citizenId", informaData.value.citizenId);
if (informaData.value.prefix != undefined && informaData.value.prefix != "") if (informaData.value.prefix != undefined)
formData.append("prefix", informaData.value.prefix); formData.append("prefix", informaData.value.prefix);
if (informaData.value.firstName != undefined) if (informaData.value.firstName != undefined)
formData.append("firstName", informaData.value.firstName); formData.append("firstName", informaData.value.firstName);
@ -275,8 +256,6 @@ function onSubmit() {
formData.append("employeeType", informaData.value.employeeType); formData.append("employeeType", informaData.value.employeeType);
if (informaData.value.employeeClass != undefined) if (informaData.value.employeeClass != undefined)
formData.append("employeeClass", informaData.value.employeeClass); formData.append("employeeClass", informaData.value.employeeClass);
if (informaData.value.rank != undefined && informaData.value.rank != "")
formData.append("rank", informaData.value.rank);
dialogConfirm($q, async () => { dialogConfirm($q, async () => {
showLoader(); showLoader();
@ -310,15 +289,6 @@ function updateBirthDate(v: Date) {
age.value = calculateAge(v); age.value = calculateAge(v);
} }
function prefixRankRule() {
return [
() =>
!!informaData.value.rank ||
!!informaData.value.prefix ||
"กรุณาเลือกคำนำหน้าชื่อ หรือยศ",
];
}
/** /**
* ทำงานเมอมการเรยกใช Components * ทำงานเมอมการเรยกใช Components
*/ */
@ -413,7 +383,7 @@ onMounted(async () => {
</div> </div>
<q-card-section class="q-px-lg"> <q-card-section class="q-px-lg">
<div class="row q-col-gutter-sm"> <div class="row q-col-gutter-sm">
<div class="col-2"> <div class="col-3">
<q-input <q-input
bg-color="white" bg-color="white"
outlined outlined
@ -434,14 +404,13 @@ onMounted(async () => {
mask="#############" mask="#############"
/> />
</div> </div>
<div class="col-2"> <div class="col-3">
<q-select <q-select
bg-color="white" bg-color="white"
v-model="informaData.prefix" v-model="informaData.prefix"
label="คำนำหน้าชื่อ" label="คำนำหน้าชื่อ"
outlined outlined
dense dense
clearable
lazy-rules lazy-rules
class="inputgreen" class="inputgreen"
:options="Ops.prefixOps" :options="Ops.prefixOps"
@ -449,8 +418,11 @@ onMounted(async () => {
option-value="name" option-value="name"
map-options map-options
hide-bottom-space hide-bottom-space
:rules="prefixRankRule()" :rules="[
reactive-rules (val:string) => {
return val.length > 0 || 'กรุณาเลือกคำนำหน้าชื่อ';
},
]"
emit-value emit-value
use-input use-input
hide-selected hide-selected
@ -460,32 +432,6 @@ onMounted(async () => {
)" )"
/> />
</div> </div>
<div class="col-2">
<q-select
bg-color="white"
v-model="informaData.rank"
label="ยศ"
outlined
dense
lazy-rules
clearable
class="inputgreen"
:options="Ops.rankOps"
option-label="name"
option-value="name"
map-options
hide-bottom-space
:rules="prefixRankRule()"
reactive-rules
emit-value
use-input
hide-selected
fill-input
@filter="(inputValue:string,
doneFn: Function) => filterSelector(inputValue, doneFn, 'rankOps'
)"
/>
</div>
<div class="col-3"> <div class="col-3">
<q-input <q-input
bg-color="white" bg-color="white"

View file

@ -113,7 +113,6 @@ const Ops = ref<InformationOps>({
{ id: "gov", name: "งบประมาณเงินอุดหนุนรัฐบาล" }, { id: "gov", name: "งบประมาณเงินอุดหนุนรัฐบาล" },
{ id: "bkk", name: "งบประมาณกรุงเทพมหานคร" }, { id: "bkk", name: "งบประมาณกรุงเทพมหานคร" },
], ],
rankOps: [],
}); });
const OpsFilter = ref<InformationOps>({ const OpsFilter = ref<InformationOps>({
prefixOps: [], prefixOps: [],
@ -130,7 +129,6 @@ const OpsFilter = ref<InformationOps>({
{ id: "gov", name: "งบประมาณเงินอุดหนุนรัฐบาล" }, { id: "gov", name: "งบประมาณเงินอุดหนุนรัฐบาล" },
{ id: "bkk", name: "งบประมาณกรุงเทพมหานคร" }, { id: "bkk", name: "งบประมาณกรุงเทพมหานคร" },
], ],
rankOps: [],
}); });
/** ฟังก์ชันดึงข้อมูลรายการข้อมูลเกี่ยวกับบุคคล (dropdown list)*/ /** ฟังก์ชันดึงข้อมูลรายการข้อมูลเกี่ยวกับบุคคล (dropdown list)*/
@ -188,16 +186,6 @@ async function fetchPerson() {
}); });
Ops.value.religionOps = optionreligions; Ops.value.religionOps = optionreligions;
OpsFilter.value.religionOps = optionreligions; OpsFilter.value.religionOps = optionreligions;
let rank: DataOption[] = [];
data.rank.map((r: DataOptioninfo) => {
rank.push({
id: r.id.toString(),
name: r.name.toString(),
});
});
Ops.value.rankOps = rank;
OpsFilter.value.rankOps = rank;
}) })
.catch((e) => { .catch((e) => {
messageError($q, e); messageError($q, e);
@ -227,9 +215,9 @@ async function getData() {
} }
rows.value = list; rows.value = list;
profileId.value = data.profileId; profileId.value = data.profileId;
title.value.fullname = `${data.rank ? data.rank : data.prefix ?? ""}${ title.value.fullname = `${data.prefix ?? ""}${data.firstName ?? ""} ${
data.firstName ?? "" data.lastName ?? ""
} ${data.lastName ?? ""}`; }`;
title.value.organizationPositionOld = data.organizationPositionOld ?? "-"; title.value.organizationPositionOld = data.organizationPositionOld ?? "-";
title.value.positionLevelOld = data.positionLevelOld ?? "-"; title.value.positionLevelOld = data.positionLevelOld ?? "-";
title.value.positionTypeOld = data.positionTypeOld ?? "-"; title.value.positionTypeOld = data.positionTypeOld ?? "-";
@ -242,7 +230,6 @@ async function getData() {
(data.prefix == "00000000-0000-0000-0000-000000000000" (data.prefix == "00000000-0000-0000-0000-000000000000"
? null ? null
: data.prefix) ?? "", : data.prefix) ?? "",
rank: data.rank ?? "",
firstname: data.firstName ?? "", firstname: data.firstName ?? "",
lastname: data.lastName ?? "", lastname: data.lastName ?? "",
birthDate: birthDate:
@ -308,7 +295,6 @@ async function fetchData(data: any) {
(data.prefix == "00000000-0000-0000-0000-000000000000" (data.prefix == "00000000-0000-0000-0000-000000000000"
? null ? null
: data.prefix) ?? "", : data.prefix) ?? "",
rank: data.rank ?? "",
firstname: data.firstName ?? "", firstname: data.firstName ?? "",
lastname: data.lastName ?? "", lastname: data.lastName ?? "",
birthDate: data.dateOfBirth !== null ? new Date(data.dateOfBirth) : null, birthDate: data.dateOfBirth !== null ? new Date(data.dateOfBirth) : null,
@ -450,14 +436,6 @@ function filterSelector(val: string, update: Function, refData: string) {
}); });
break; break;
case "rankOps":
update(() => {
Ops.value.rankOps = OpsFilter.value.rankOps.filter(
(v: DataOption) => v.name.toLowerCase().indexOf(newVal) > -1
);
});
break;
default: default:
break; break;
} }
@ -492,7 +470,6 @@ function saveData() {
positionNumberOld: posNo.value, positionNumberOld: posNo.value,
amount: 0, amount: 0,
amountOld: salary.value, amountOld: salary.value,
rank: informaData.value.rank,
}; };
showLoader(); showLoader();
http http
@ -540,15 +517,6 @@ function updateBirthDate(v: Date) {
informaData.value.age = calculateAge(v); informaData.value.age = calculateAge(v);
} }
function prefixRankRule() {
return [
() =>
!!informaData.value.rank ||
!!informaData.value.prefixId ||
"กรุณาเลือกคำนำหน้าชื่อ หรือยศ",
];
}
/** ทำงานเมื่อมีการเรียกใช้ Components*/ /** ทำงานเมื่อมีการเรียกใช้ Components*/
onMounted(async () => { onMounted(async () => {
await fetchPerson(); await fetchPerson();
@ -630,7 +598,7 @@ onMounted(async () => {
<div class="col-xs-12"> <div class="col-xs-12">
<div class="text-weight-bold text-grey">อมลสวนต</div> <div class="text-weight-bold text-grey">อมลสวนต</div>
</div> </div>
<div class="col-xs-6 col-sm-2 col-md-2"> <div class="col-xs-6 col-sm-3 col-md-3">
<q-input <q-input
:class="getClass(edit)" :class="getClass(edit)"
hide-bottom-space hide-bottom-space
@ -646,7 +614,7 @@ onMounted(async () => {
mask="#############" mask="#############"
/> />
</div> </div>
<div class="col-xs-6 col-sm-2 col-md-2"> <div class="col-xs-6 col-sm-3 col-md-3">
<selector <selector
:hide-dropdown-icon="!edit" :hide-dropdown-icon="!edit"
hide-bottom-space hide-bottom-space
@ -666,42 +634,10 @@ onMounted(async () => {
option-value="name" option-value="name"
:label="`${'คำนำหน้าชื่อ'}`" :label="`${'คำนำหน้าชื่อ'}`"
use-input use-input
clearable
input-debounce="0" input-debounce="0"
:rules="prefixRankRule()"
reactive-rules
@filter="(inputValue:string, doneFn:Function) => filterSelector(inputValue, doneFn,'prefixOps' ) " @filter="(inputValue:string, doneFn:Function) => filterSelector(inputValue, doneFn,'prefixOps' ) "
/> />
</div> </div>
<div class="col-xs-6 col-sm-2 col-md-2">
<selector
:hide-dropdown-icon="!edit"
hide-bottom-space
:class="getClass(edit)"
:readonly="!edit"
:borderless="!edit"
:outlined="edit"
dense
lazy-rules
v-model="informaData.rank"
emit-value
map-options
hide-selected
fill-input
:rules="prefixRankRule()"
option-label="name"
:options="Ops.rankOps"
option-value="name"
:label="`${'ยศ'}`"
reactive-rules
use-input
input-debounce="0"
@filter="(inputValue:string, doneFn:Function) => filterSelector(inputValue, doneFn,'rankOps' ) "
clearable
/>
</div>
<div class="col-xs-6 col-sm-3 col-md-3"> <div class="col-xs-6 col-sm-3 col-md-3">
<q-input <q-input
:class="getClass(edit)" :class="getClass(edit)"

View file

@ -201,7 +201,6 @@ watch(modal, (val) => {
hide-bottom-space hide-bottom-space
dense dense
label="คำค้น" label="คำค้น"
@keydown.enter.prevent="onSearchData()"
/> />
</div> </div>
<q-checkbox <q-checkbox

View file

@ -7,7 +7,6 @@ interface InformationOps {
religionOps: DataOption[]; religionOps: DataOption[];
employeeClassOps: DataOption[]; employeeClassOps: DataOption[];
employeeTypeOps: DataOption[]; employeeTypeOps: DataOption[];
rankOps: DataOption[];
} }
//ข้อมูลส่วนตัว //ข้อมูลส่วนตัว
@ -64,7 +63,6 @@ interface FormAddPerson {
employeeType: string | null; employeeType: string | null;
employeeClass: string | null; employeeClass: string | null;
profileType: string | null; profileType: string | null;
rank?: string;
} }
const defaultInformation: Information = { const defaultInformation: Information = {
@ -85,7 +83,6 @@ const defaultInformation: Information = {
employeeType: null, employeeType: null,
employeeClass: null, employeeClass: null,
profileType: null, profileType: null,
rank: null,
}; };
export { defaultInformation }; export { defaultInformation };

View file

@ -152,7 +152,6 @@ interface FormDataAppoint {
reportingDate: string; reportingDate: string;
posmasterId: string; posmasterId: string;
posExecutiveName?: string; posExecutiveName?: string;
posExecutiveId?: string;
typeCommand: string; typeCommand: string;
positionExecutiveField?: string; positionExecutiveField?: string;
positionArea?: string; positionArea?: string;

View file

@ -1,24 +1,17 @@
interface DataList { interface DataList {
avatar: string; avatar: string;
bmaOfficer: string; bmaOfficer: string;
bmaOfficerCheck?: string;
deferment: boolean; deferment: boolean;
draft: string; draft: string;
examNumber: number; examNumber: number;
fullName: string; fullName: string;
idCard: string; idCard: string;
name: string; name: string;
node: number | null;
nodeId: string | null;
orgName: string | null; orgName: string | null;
organizationName: string; organizationName: string;
organizationShortName: string | null; organizationShortName: string | null;
personalId: string; personalId: string;
posLevelCandidateId: string | null;
posmasterId: string | null;
posTypeCandidateId: string | null;
positionCandidate: string; positionCandidate: string;
positionId: string | null;
positionNumber: string | null; positionNumber: string | null;
positionPath: string | null; positionPath: string | null;
profilePhoto: string; profilePhoto: string;

View file

@ -10,25 +10,25 @@ import type {
} from "@/modules/05_placement/interface/request/Main"; } from "@/modules/05_placement/interface/request/Main";
import type { FormOrderPlacementMainData } from "@/modules/05_placement/interface/request/Main"; import type { FormOrderPlacementMainData } from "@/modules/05_placement/interface/request/Main";
interface profile {
main: { columns: String[] };
education: { columns: String[] };
certicate: { columns: String[] };
train: { columns: String[] };
insignia: { columns: String[] };
coined: { columns: String[] };
assessment: { columns: String[] };
salary: { columns: String[] };
discipline: { columns: String[] };
leave: { columns: String[] };
talent: { columns: String[] };
work: { columns: String[] };
record: { columns: String[] };
other: { columns: String[] };
document: { columns: String[] };
}
export const useProfileDataStore = defineStore("profilePlacenent", () => { export const useProfileDataStore = defineStore("profilePlacenent", () => {
interface profile {
main: { columns: String[] };
education: { columns: String[] };
certicate: { columns: String[] };
train: { columns: String[] };
insignia: { columns: String[] };
coined: { columns: String[] };
assessment: { columns: String[] };
salary: { columns: String[] };
discipline: { columns: String[] };
leave: { columns: String[] };
talent: { columns: String[] };
work: { columns: String[] };
record: { columns: String[] };
other: { columns: String[] };
document: { columns: String[] };
}
const birthDate = ref<Date>(new Date()); const birthDate = ref<Date>(new Date());
const retireText = ref<string | null>(null); const retireText = ref<string | null>(null);
const changeRetireText = (val: string | null) => { const changeRetireText = (val: string | null) => {
@ -87,15 +87,15 @@ export const useProfileDataStore = defineStore("profilePlacenent", () => {
changeRetireText, changeRetireText,
}; };
}); });
interface placement {
mappingPosition: { columns: String[] };
}
export const usePlacementDataStore = defineStore("placement", () => { export const usePlacementDataStore = defineStore("placement", () => {
const mixin = useCounterMixin(); //เรียกฟังก์ชันกลาง const mixin = useCounterMixin(); //เรียกฟังก์ชันกลาง
const tabsMain = ref<string>("probation"); const tabsMain = ref<string>("probation");
const isOfficer = ref<boolean | null>(null); const isOfficer = ref<boolean | null>(null);
const isStaff = ref<boolean | null>(null); const isStaff = ref<boolean | null>(null);
const { hideLoader } = mixin; const { hideLoader } = mixin;
interface placement {
mappingPosition: { columns: String[] };
}
const placementData = ref<placement>({ const placementData = ref<placement>({
mappingPosition: { columns: [] }, mappingPosition: { columns: [] },
}); });
@ -218,10 +218,10 @@ export const usePlacementDataStore = defineStore("placement", () => {
isStaff, isStaff,
}; };
}); });
interface placementOrder {
mappingPosition: { columns: String[] };
}
export const useOrderPlacementDataStore = defineStore("placementOrder", () => { export const useOrderPlacementDataStore = defineStore("placementOrder", () => {
interface placementOrder {
mappingPosition: { columns: String[] };
}
const placementOrderData = ref<placementOrder>({ const placementOrderData = ref<placementOrder>({
mappingPosition: { columns: [] }, mappingPosition: { columns: [] },
}); });
@ -369,8 +369,6 @@ export const useTransferDataStore = defineStore("transferDataStore", () => {
]); ]);
const statusOp = ref<DataOptions[]>(statusMainOp.value); const statusOp = ref<DataOptions[]>(statusMainOp.value);
const statusDelete = ["REPORT", "WAITING", "DONE"];
const statusText = (val: string) => { const statusText = (val: string) => {
switch (val) { switch (val) {
case "WAITTING": case "WAITTING":
@ -440,6 +438,5 @@ export const useTransferDataStore = defineStore("transferDataStore", () => {
statusOp, statusOp,
statusMainOp, statusMainOp,
filterOption, filterOption,
statusDelete,
}; };
}); });

View file

@ -1,5 +1,5 @@
import { defineStore } from "pinia"; import { defineStore } from "pinia";
import { formatPosmasterNo } from "@/utils/function"; import { ref } from "vue";
/** importType*/ /** importType*/
import type { PositionMaim } from "@/modules/05_placement/interface/response/SelectOrg"; import type { PositionMaim } from "@/modules/05_placement/interface/response/SelectOrg";
import type { DataPositionNo } from "@/modules/05_placement/interface/index/SelectOrg"; import type { DataPositionNo } from "@/modules/05_placement/interface/index/SelectOrg";
@ -12,13 +12,11 @@ export const useSelectOrgStore = defineStore("selectorg", () => {
const listPosNo: DataPositionNo[] = data.map((e: PositionMaim) => ({ const listPosNo: DataPositionNo[] = data.map((e: PositionMaim) => ({
id: e.id, id: e.id,
isPosition: e.isPosition, isPosition: e.isPosition,
posMasterNo: formatPosmasterNo( posMasterNo:
e.orgShortname, e.orgShortname +
e.posMasterNoPrefix, (e.posMasterNoPrefix != null ? e.posMasterNoPrefix : " ") +
e.posMasterNo.toString(), e.posMasterNo +
e.posMasterNoSuffix (e.posMasterNoSuffix != null ? e.posMasterNoSuffix : " "),
),
positionName: e.positionName, positionName: e.positionName,
posTypeName: e.posTypeName, posTypeName: e.posTypeName,
posLevelName: e.posLevelName, posLevelName: e.posLevelName,

View file

@ -2,7 +2,7 @@
import { ref, onMounted, computed } from "vue"; import { ref, onMounted, computed } from "vue";
import { useQuasar } from "quasar"; import { useQuasar } from "quasar";
import { useRouter, useRoute } from "vue-router"; import { useRouter } from "vue-router";
import { import {
checkPermission, checkPermission,
checkPermissionList, checkPermissionList,
@ -20,19 +20,11 @@ import DialogOrders from "@/modules/05_placement/components/Transfer/DialogOrder
const $q = useQuasar(); const $q = useQuasar();
const router = useRouter(); const router = useRouter();
const route = useRoute();
const mixin = useCounterMixin(); const mixin = useCounterMixin();
const store = useTransferDataStore(); const store = useTransferDataStore();
const { statusText, filterOption } = useTransferDataStore(); const { statusText, filterOption } = useTransferDataStore();
const { const { date2Thai, messageError, showLoader, hideLoader, onSearchDataTable } =
date2Thai, mixin;
messageError,
showLoader,
hideLoader,
onSearchDataTable,
dialogRemove,
success,
} = mixin;
const modal = ref<boolean>(false); // const modal = ref<boolean>(false); //
const dataTransfer = ref<ResponseData[]>([]); // const dataTransfer = ref<ResponseData[]>([]); //
@ -149,15 +141,6 @@ const visibleColumns = ref<string[]>([
"createdAt", "createdAt",
]); ]);
const isPermissionDelete = computed(() => {
return (status: string) => {
return (
checkPermission(route)?.attrOwnership === "OWNER" &&
!store.statusDelete.includes(status)
);
};
});
/** ฟังก์ชันดึงข้อมูรายการขอโอน*/ /** ฟังก์ชันดึงข้อมูรายการขอโอน*/
async function fetchData() { async function fetchData() {
showLoader(); showLoader();
@ -211,21 +194,6 @@ function onSearch() {
); );
} }
function handleDelete(id: string) {
dialogRemove($q, async () => {
try {
showLoader();
await http.delete(config.API.transfer + `/admin/${id}`);
await fetchData();
success($q, "ลบข้อมูลสำเร็จ");
} catch (error) {
messageError($q, error);
} finally {
hideLoader();
}
});
}
/** /**
* ทำงานเม Components กเรยกใชงาน * ทำงานเม Components กเรยกใชงาน
* จะเรยกใช fetchData เพอดงขอมลรายการขอโอน * จะเรยกใช fetchData เพอดงขอมลรายการขอโอน
@ -369,18 +337,6 @@ onMounted(async () => {
> >
<q-tooltip>รายละเอยด</q-tooltip> <q-tooltip>รายละเอยด</q-tooltip>
</q-btn> </q-btn>
<q-btn
v-if="isPermissionDelete(props.row.status)"
flat
round
dense
icon="mdi-delete"
color="red"
@click.prevent="handleDelete(props.row.id)"
>
<q-tooltip>ลบขอม</q-tooltip>
</q-btn>
</q-td> </q-td>
<q-td v-for="col in props.cols" :key="col.id"> <q-td v-for="col in props.cols" :key="col.id">
<div v-if="col.name === 'no'"> <div v-if="col.name === 'no'">

View file

@ -110,9 +110,7 @@ const columns = ref<QTableProps["columns"]>([
sort: (a: string, b: string) => sort: (a: string, b: string) =>
a.localeCompare(b, undefined, { numeric: true, sensitivity: "base" }), a.localeCompare(b, undefined, { numeric: true, sensitivity: "base" }),
format(val, row) { format(val, row) {
return ` ${row.rank ? row.rank : row.prefix ?? ""}${ return `${row.prefix ?? ""}${row.firstName ?? ""} ${row.lastName ?? ""}`;
row.firstName ?? ""
} ${row.lastName ?? ""}`;
}, },
}, },
{ {
@ -133,11 +131,9 @@ const columns = ref<QTableProps["columns"]>([
headerStyle: "font-size: 14px", headerStyle: "font-size: 14px",
style: "font-size: 14px", style: "font-size: 14px",
format(val, row) { format(val, row) {
return row.positionTypeOld && row.positionTypeOld !== "-" return row.positionTypeOld
? `${row.positionTypeOld}${ ? `${row.positionTypeOld}${
row.positionLevelOld && row.positionLevelOld !== "-" row.positionLevelOld ? `(${row.positionLevelOld})` : ""
? `(${row.positionLevelOld})`
: ""
}` }`
: ""; : "";
}, },
@ -313,7 +309,7 @@ function openDelete(id: string) {
dialogRemove($q, async () => { dialogRemove($q, async () => {
showLoader(); showLoader();
await http await http
.delete(config.API.receiveData() + `/admin/${id}`) .delete(config.API.receiveDataId(id))
.then(async () => { .then(async () => {
await fecthlistRecevice(); await fecthlistRecevice();
await success($q, "ลบข้อมูลสำเร็จ"); await success($q, "ลบข้อมูลสำเร็จ");
@ -382,7 +378,6 @@ async function onSave(data: FormDataAppoint) {
typeCommand: data.typeCommand, typeCommand: data.typeCommand,
positionExecutiveField: data.positionExecutiveField, positionExecutiveField: data.positionExecutiveField,
positionArea: data.positionArea, positionArea: data.positionArea,
posExecutiveId: data.posExecutiveId,
}; };
showLoader(); showLoader();
@ -606,10 +601,8 @@ onMounted(async () => {
<q-item <q-item
v-if=" v-if="
checkPermission($route)?.attrOwnership === checkPermission($route)?.attrIsDelete &&
'OWNER' &&
props.row.status !== 'REPORT' && props.row.status !== 'REPORT' &&
props.row.status !== 'WAITING' &&
props.row.status !== 'DONE' props.row.status !== 'DONE'
" "
clickable clickable

View file

@ -77,7 +77,7 @@ const columns = ref<QTableProps["columns"]>([
headerStyle: "font-size: 14px", headerStyle: "font-size: 14px",
style: "font-size: 14px", style: "font-size: 14px",
format(val, row) { format(val, row) {
return row.positionNumberOld ?? "-"; return findPosMasterNoOld(row);
}, },
}, },
{ {
@ -236,7 +236,8 @@ function openModalOrder() {
item.status == "APPROVE") && item.status == "APPROVE") &&
item.organizationPositionOld && item.organizationPositionOld &&
item.organization && item.organization &&
item.dateStart item.dateStart &&
item.dateEnd
); );
rows2.value = row; rows2.value = row;
rows2Data.value = row; rows2Data.value = row;

View file

@ -381,7 +381,7 @@ async function fecthlistappointment() {
e.positionTypeOld && e.positionTypeOld &&
e.positionLevelOld && e.positionLevelOld &&
e.positionNumberOld && e.positionNumberOld &&
(e.typeCommand === "MOVE" || e.positionDate) e.positionDate
); );
rows2.value = listData; rows2.value = listData;
rows2Data.value = listData; rows2Data.value = listData;
@ -457,7 +457,6 @@ async function onSave(data: FormDataAppoint) {
typeCommand: data.typeCommand, typeCommand: data.typeCommand,
positionExecutiveField: data.positionExecutiveField, positionExecutiveField: data.positionExecutiveField,
positionArea: data.positionArea, positionArea: data.positionArea,
posExecutiveId: data.posExecutiveId,
}; };
showLoader(); showLoader();

View file

@ -252,7 +252,7 @@ async function fecthlistappointment() {
e.positionTypeOld && e.positionTypeOld &&
e.positionLevelOld && e.positionLevelOld &&
e.positionNumberOld && e.positionNumberOld &&
(e.positionDate || e.typeCommand === "MOVE") e.positionDate
); );
rows2.value = listData; rows2.value = listData;
rows2Data.value = listData; rows2Data.value = listData;

View file

@ -286,7 +286,6 @@ async function onSaveSelectOrg(data: any) {
typeCommand: data.typeCommand, typeCommand: data.typeCommand,
positionExecutiveField: data.positionExecutiveField, positionExecutiveField: data.positionExecutiveField,
positionArea: data.positionArea, positionArea: data.positionArea,
posExecutiveId: data.posExecutiveId,
}; };
showLoader(); showLoader();

View file

@ -370,32 +370,18 @@ async function uploadFile(event: any, date: any) {
*/ */
async function downloadAttachment(type: string, id: string) { async function downloadAttachment(type: string, id: string) {
showLoader(); showLoader();
try { await http
const response = await http.get( .get(config.API.reportRetireList(type, id))
config.API.retirementReport + `/${type}/${id}`, .then(async (res) => {
{ const data = res.data.result;
headers: { await genReport(data, `รายชื่อผู้เกษียณอายุราชการ`, type);
accept: })
type === "pdf" .catch(async (e) => {
? "application/pdf" messageError($q, JSON.parse(await e.response.data.text()));
: "application/vnd.openxmlformats-officedocument.wordprocessingml.document", })
"content-Type": "application/json", .finally(() => {
}, hideLoader();
responseType: "blob", });
}
);
const blob = response.data;
const url = URL.createObjectURL(blob);
const link = document.createElement("a");
link.href = url;
link.download = `รายชื่อผู้เกษียณอายุราชการ.${type}`;
link.click();
URL.revokeObjectURL(url);
} catch (e) {
messageError($q, e);
} finally {
hideLoader();
}
} }
// //

View file

@ -5,12 +5,10 @@ import { useQuasar } from "quasar";
import { useCounterMixin } from "@/stores/mixin"; import { useCounterMixin } from "@/stores/mixin";
import { useRetirementDataStore } from "@/modules/06_retirement/store/Main"; import { useRetirementDataStore } from "@/modules/06_retirement/store/Main";
import { useRoleWorkflowDataStore } from "@/stores/roleWorkflow"; import { useRoleWorkflowDataStore } from "@/stores/roleWorkflow";
import { useCommandMainStore } from "@/modules/18_command/store/Main";
import type { PropType } from "vue"; import type { PropType } from "vue";
import type { QTableProps } from "quasar"; import type { QTableProps } from "quasar";
import type { ResponseItems } from "@/modules/06_retirement/interface/response/Main"; import type { ResponseItems } from "@/modules/06_retirement/interface/response/Main";
import type { ListCommand } from "@/modules/18_command/interface/index/Main";
import DialogHeader from "@/components/DialogHeader.vue"; import DialogHeader from "@/components/DialogHeader.vue";
import DialogCreateCommand from "@/modules/18_command/components/DialogCreateCommand.vue"; import DialogCreateCommand from "@/modules/18_command/components/DialogCreateCommand.vue";
@ -18,7 +16,6 @@ import DialogCreateCommand from "@/modules/18_command/components/DialogCreateCom
/** use */ /** use */
const $q = useQuasar(); const $q = useQuasar();
const stroe = useRetirementDataStore(); const stroe = useRetirementDataStore();
const storeCommand = useCommandMainStore();
const { fetchDataCheckIsoffice } = useRoleWorkflowDataStore(); const { fetchDataCheckIsoffice } = useRoleWorkflowDataStore();
const { statusText } = stroe; const { statusText } = stroe;
const selected = ref<ResponseItems[]>([]); const selected = ref<ResponseItems[]>([]);
@ -50,10 +47,6 @@ const props = defineProps({
cancel: { type: Boolean, default: false }, cancel: { type: Boolean, default: false },
}); });
const commandType = ref<string>("");
const commandMainOp = ref<ListCommand[]>([]); //
const commandOp = ref<ListCommand[]>([]); //
//Table //Table
const filterKeyword = defineModel<string>("filterKeyword", { required: true }); const filterKeyword = defineModel<string>("filterKeyword", { required: true });
const rowsData = ref<ResponseItems[]>([]); const rowsData = ref<ResponseItems[]>([]);
@ -176,11 +169,6 @@ const visibleColumns = ref<string[]>([
]); ]);
const modalCommand = ref<boolean>(false); const modalCommand = ref<boolean>(false);
const isDisable = computed(() => {
return props.mainTabs === "1"
? selected.value.length === 0 || commandType.value === ""
: selected.value.length === 0;
});
/** popup ยืนยันส่งัว */ /** popup ยืนยันส่งัว */
function saveOrder() { function saveOrder() {
@ -222,19 +210,6 @@ async function fetchCheckOfficer() {
} }
} }
/**
* ลเตอร คำส
* @param val าจาก Input
* @param update Funtion quasar
*/
function filterSelector(val: string, update: Function) {
update(() => {
commandOp.value = commandMainOp.value.filter(
(v: ListCommand) => v.name.indexOf(val) > -1
);
});
}
watch( watch(
() => props.modal, () => props.modal,
async (val) => { async (val) => {
@ -256,11 +231,6 @@ watch(
rowsData.value = data ? data : []; rowsData.value = data ? data : [];
rowsDataMain.value = data ? data : []; rowsDataMain.value = data ? data : [];
commandType.value = "";
const dataCommandTypes = await storeCommand.getCommandTypes();
commandMainOp.value = dataCommandTypes.filter(
(e: ListCommand) => e.code === "C-PM-17" || e.code === "C-PM-48"
);
} else { } else {
rowsData.value = []; rowsData.value = [];
rowsDataMain.value = []; rowsDataMain.value = [];
@ -268,7 +238,6 @@ watch(
} }
); );
</script> </script>
<template> <template>
<q-dialog v-model="props.modal" persistent> <q-dialog v-model="props.modal" persistent>
<q-card style="min-width: 85%"> <q-card style="min-width: 85%">
@ -280,33 +249,6 @@ watch(
<q-card-section> <q-card-section>
<div class="row col-12 q-col-gutter-sm"> <div class="row col-12 q-col-gutter-sm">
<div class="col-12 row"> <div class="col-12 row">
<q-select
v-if="props.mainTabs === '1'"
v-model="commandType"
dense
outlined
label="ประเภทคำสั่ง"
:options="commandOp"
option-label="name"
option-value="code"
emit-value
map-options
use-input
hide-selected
fill-input
@update:model-value="onSearch"
@filter="(inputValue:string,
doneFn:Function) => filterSelector(inputValue, doneFn
) "
>
<template v-slot:no-option>
<q-item>
<q-item-section class="text-grey">
ไมอม
</q-item-section>
</q-item>
</template>
</q-select>
<q-space /> <q-space />
<q-input <q-input
borderless borderless
@ -390,7 +332,7 @@ watch(
<q-btn <q-btn
label="ส่งไปออกคำสั่ง" label="ส่งไปออกคำสั่ง"
@click="saveOrder" @click="saveOrder"
:disable="isDisable" :disable="selected.length === 0"
color="public" color="public"
/> />
</q-card-actions> </q-card-actions>
@ -399,7 +341,7 @@ watch(
<DialogCreateCommand <DialogCreateCommand
v-model:modal="modalCommand" v-model:modal="modalCommand"
:command-type-code="props.mainTabs === '1' ? commandType : 'C-PM-41'" :command-type-code="props.mainTabs === '1' ? 'C-PM-17' : 'C-PM-41'"
:persons="dataMapToSend" :persons="dataMapToSend"
:fetch-data="props.fecthList" :fetch-data="props.fecthList"
/> />

View file

@ -47,7 +47,6 @@ const {
success, success,
dialogConfirm, dialogConfirm,
dialogRemove, dialogRemove,
convertDateToAPI,
} = mixin; } = mixin;
/** ตัวแปร */ /** ตัวแปร */
@ -478,8 +477,9 @@ function onSubmitConditions() {
function onSubmitAttached() { function onSubmitAttached() {
dialogConfirm($q, async () => { dialogConfirm($q, async () => {
const formData = new FormData(); const formData = new FormData();
const send: string = convertDateToAPI(date.value) ?? ""; const send = date.value !== null ? new Date(date.value).toUTCString() : "";
const activeDate: string = convertDateToAPI(dateLeave.value) ?? ""; const activeDate =
dateLeave.value !== null ? new Date(dateLeave.value).toUTCString() : "";
formData.append("Location", location.value); formData.append("Location", location.value);
formData.append("SendDate", send); formData.append("SendDate", send);
formData.append("ActiveDate", activeDate); formData.append("ActiveDate", activeDate);
@ -771,13 +771,10 @@ onMounted(async () => {
<q-space /> <q-space />
<div <div
v-if=" v-if="
(!checkRoutePermisson && !checkRoutePermisson &&
isStaff && isStaff &&
status === 'WAITTING' && status === 'WAITTING' &&
checkPermission($route)?.attrIsUpdate) || checkPermission($route)?.attrIsUpdate
(!checkRoutePermisson &&
((isStaff && group === '1.1' && approveStep === 'st1') ||
(isOfficer && group !== '1.1' && approveStep === 'st3')))
" "
> >
<q-btn <q-btn

View file

@ -34,7 +34,6 @@ const {
hideLoader, hideLoader,
success, success,
dialogConfirm, dialogConfirm,
convertDateToAPI,
} = useCounterMixin(); } = useCounterMixin();
const checkRoutePermisson = ref<boolean>(route.name == "resignDetailreject"); const checkRoutePermisson = ref<boolean>(route.name == "resignDetailreject");
@ -314,8 +313,9 @@ async function clickCancel() {
function onSubmitAttached() { function onSubmitAttached() {
dialogConfirm($q, () => { dialogConfirm($q, () => {
const formData = new FormData(); const formData = new FormData();
const send: string = convertDateToAPI(date.value) ?? ""; const send = date.value !== null ? new Date(date.value).toUTCString() : "";
const activeDate: string = convertDateToAPI(dateLeave.value) ?? ""; const activeDate =
dateLeave.value !== null ? new Date(dateLeave.value).toUTCString() : "";
formData.append("Location", location.value); formData.append("Location", location.value);
formData.append("SendDate", send); formData.append("SendDate", send);
formData.append("ActiveDate", activeDate); formData.append("ActiveDate", activeDate);
@ -439,7 +439,7 @@ onMounted(async () => {
class="q-mr-sm" class="q-mr-sm"
@click="router.push('/retirement/resign')" @click="router.push('/retirement/resign')"
/> />
รายละเอยดการยกเลกลาออก รายละเอยดการยกเลกลาออก
{{ {{
dataDetail.prefix + dataDetail.firstName + " " + dataDetail.lastName dataDetail.prefix + dataDetail.firstName + " " + dataDetail.lastName
}} }}

View file

@ -33,8 +33,7 @@ const {
showLoader, showLoader,
hideLoader, hideLoader,
onSearchDataTable, onSearchDataTable,
dialogRemove, findOrgName,
success,
} = mixin; } = mixin;
/** Table */ /** Table */
@ -258,21 +257,6 @@ function onSearch() {
); );
} }
function handleDelete(id: string) {
dialogRemove($q, async () => {
try {
showLoader();
await http.delete(config.API.listResign() + `/admin/${id}`);
await fecthlist();
success($q, "ลบข้อมูลสำเร็จ");
} catch (error) {
messageError($q, error);
} finally {
hideLoader();
}
});
}
/**Hook */ /**Hook */
onMounted(async () => { onMounted(async () => {
status.value = stroeResign.formQurey.status; status.value = stroeResign.formQurey.status;
@ -324,11 +308,7 @@ onMounted(async () => {
color="primary" color="primary"
icon="mdi-account-arrow-right" icon="mdi-account-arrow-right"
> >
<q-tooltip>{{ <q-tooltip>{{ `ส่งไปออกคำสั่ง${stroeResign.mainTabs == '2'?"ยกเลิกการ":''}ลาออก` }}</q-tooltip>
`ส่งไปออกคำสั่ง${
stroeResign.mainTabs == "2" ? "ยกเลิกการ" : ""
}ลาออก`
}}</q-tooltip>
</q-btn> </q-btn>
</div> </div>
<q-space /> <q-space />
@ -413,26 +393,6 @@ onMounted(async () => {
> >
<q-tooltip>แกไขขอม</q-tooltip> <q-tooltip>แกไขขอม</q-tooltip>
</q-btn> </q-btn>
<q-btn
v-if="
checkPermission($route)?.attrOwnership === 'OWNER' &&
props.row.status !== 'REPORT' &&
props.row.status !== 'WAITING' &&
props.row.status !== 'DONE' &&
props.row.status !== 'CANCELING' &&
props.row.status !== 'CANCEL' &&
stroeResign.mainTabs === '1'
"
flat
dense
round
color="red"
icon="delete"
@click.prevent="handleDelete(props.row.id)"
>
<q-tooltip>ลบขอม</q-tooltip>
</q-btn>
</q-td> </q-td>
<q-td v-for="col in props.cols" :key="col.id"> <q-td v-for="col in props.cols" :key="col.id">
<div <div

View file

@ -801,11 +801,10 @@ onMounted(async () => {
<q-space /> <q-space />
<div <div
v-if=" v-if="
(!checkRoutePermisson && !checkRoutePermisson &&
isStaff && isStaff &&
status === 'WAITTING' && status === 'WAITTING' &&
checkPermission($route)?.attrIsUpdate) || checkPermission($route)?.attrIsUpdate
(!checkRoutePermisson && isStaff && approveStep === 'st1')
" "
> >
<q-btn <q-btn

View file

@ -38,7 +38,6 @@ const {
hideLoader, hideLoader,
success, success,
dialogConfirm, dialogConfirm,
convertDateToAPI,
} = mixin; } = mixin;
/** ตัวแปร */ /** ตัวแปร */
@ -302,9 +301,9 @@ async function clickCancel() {
function onSubmitAttached() { function onSubmitAttached() {
dialogConfirm($q, () => { dialogConfirm($q, () => {
const formData = new FormData(); const formData = new FormData();
const send: string = convertDateToAPI(date.value) ?? ""; const send = date.value !== null ? new Date(date.value).toUTCString() : "";
const activeDate: string = convertDateToAPI(dateLeave.value) ?? ""; const activeDate =
dateLeave.value !== null ? new Date(dateLeave.value).toUTCString() : "";
formData.append("Location", location.value); formData.append("Location", location.value);
formData.append("SendDate", send); formData.append("SendDate", send);
formData.append("ActiveDate", activeDate); formData.append("ActiveDate", activeDate);

View file

@ -27,15 +27,8 @@ const stroeResign = useDataStore();
const { statusText } = stroe; const { statusText } = stroe;
const router = useRouter(); const router = useRouter();
const mixin = useCounterMixin(); const mixin = useCounterMixin();
const { const { messageError, date2Thai, showLoader, hideLoader, onSearchDataTable } =
messageError, mixin;
date2Thai,
showLoader,
hideLoader,
onSearchDataTable,
dialogRemove,
success,
} = mixin;
/** Table */ /** Table */
const rows = ref<ResponseItems[]>([]); const rows = ref<ResponseItems[]>([]);
@ -136,7 +129,7 @@ const columns = ref<QTableProps["columns"]>([
sortable: true, sortable: true,
field: "status", field: "status",
format(val, row) { format(val, row) {
return stroeResign.mainTabsEMP === "1" return stroeResign.mainTabs === "1"
? statusText(row.status) ? statusText(row.status)
: statusText(row.status, "อนุญาต"); : statusText(row.status, "อนุญาต");
}, },
@ -259,21 +252,6 @@ function onSearch() {
); );
} }
function handleDelete(id: string) {
dialogRemove($q, async () => {
try {
showLoader();
await http.delete(config.API.listResignEMP() + `/admin/${id}`);
await fecthlist();
success($q, "ลบข้อมูลสำเร็จ");
} catch (error) {
messageError($q, error);
} finally {
hideLoader();
}
});
}
/**Hook */ /**Hook */
onMounted(async () => { onMounted(async () => {
statusEMP.value = stroeResign.formQureyEMP.status; statusEMP.value = stroeResign.formQureyEMP.status;
@ -413,26 +391,6 @@ onMounted(async () => {
> >
<q-tooltip>แกไขขอม</q-tooltip> <q-tooltip>แกไขขอม</q-tooltip>
</q-btn> </q-btn>
<q-btn
v-if="
checkPermission($route)?.attrOwnership === 'OWNER' &&
props.row.status !== 'REPORT' &&
props.row.status !== 'WAITING' &&
props.row.status !== 'DONE' &&
props.row.status !== 'CANCELING' &&
props.row.status !== 'CANCEL' &&
stroeResign.mainTabsEMP === '1'
"
flat
dense
round
color="red"
icon="delete"
@click.prevent="handleDelete(props.row.id)"
>
<q-tooltip>ลบขอม</q-tooltip>
</q-btn>
</q-td> </q-td>
<q-td v-for="col in props.cols" :key="col.id"> <q-td v-for="col in props.cols" :key="col.id">
<div <div

View file

@ -4,6 +4,7 @@ import { useQuasar, type QTableProps } from "quasar";
import http from "@/plugins/http"; import http from "@/plugins/http";
import config from "@/app.config"; import config from "@/app.config";
import { getColumnLabel } from "@/utils/function";
import { useCounterMixin } from "@/stores/mixin"; import { useCounterMixin } from "@/stores/mixin";
import DialogHeader from "@/components/DialogHeader.vue"; import DialogHeader from "@/components/DialogHeader.vue";
@ -228,7 +229,6 @@ watch(
label="ค้นหา" label="ค้นหา"
v-model="keyword" v-model="keyword"
style="width: 300px" style="width: 300px"
@keydown.enter.prevent="onSearchData"
> >
<template v-slot:append> <template v-slot:append>
<q-icon name="search"></q-icon> <q-icon name="search"></q-icon>
@ -308,7 +308,9 @@ watch(
:key="col.name" :key="col.name"
:props="props" :props="props"
> >
<span class="text-weight-medium">{{ col.label }}</span> <span class="text-weight-medium">{{
getColumnLabel(col, isAct)
}}</span>
</q-th> </q-th>
</q-tr> </q-tr>
</template> </template>

View file

@ -5,7 +5,6 @@ import { useQuasar } from "quasar";
import { useRoute } from "vue-router"; import { useRoute } from "vue-router";
import { useCounterMixin } from "@/stores/mixin"; import { useCounterMixin } from "@/stores/mixin";
import { useStructureTree } from "@/stores/structureTree"; import { useStructureTree } from "@/stores/structureTree";
import { formatPosmasterNo } from "@/utils/function";
import http from "@/plugins/http"; import http from "@/plugins/http";
import config from "@/app.config"; import config from "@/app.config";
@ -75,12 +74,7 @@ const columns = ref<QTableProps["columns"]>([
sortable: true, sortable: true,
field: "posMasterNo", field: "posMasterNo",
format(val, row) { format(val, row) {
return formatPosmasterNo( return `${row.orgShortname} ${row.posMasterNo}`;
row.orgShortname,
row.posMasterNoPrefix,
row.posMasterNo.toString(),
row.posMasterNoSuffix
);
}, },
headerStyle: "font-size: 14px", headerStyle: "font-size: 14px",
style: "font-size: 14px", style: "font-size: 14px",

View file

@ -267,13 +267,7 @@ watch(
<div class="q-pa-md q-gutter-md"> <div class="q-pa-md q-gutter-md">
<div class="row"> <div class="row">
<div class="col text-grey-8">เวลาเขางาน</div> <div class="col text-grey-8">เวลาเขางาน</div>
<div class="col"> <div class="col">{{ formData.checkInTime }}</div>
{{
formData.checkInDate
? `${formData.checkInDate} ${formData.checkInTime} น.`
: "-"
}}
</div>
</div> </div>
<div class="row"> <div class="row">
<div class="col text-grey-8">สถานททำงาน</div> <div class="col text-grey-8">สถานททำงาน</div>
@ -343,23 +337,17 @@ watch(
<div class="q-pa-md q-gutter-md"> <div class="q-pa-md q-gutter-md">
<div class="row"> <div class="row">
<div class="col text-grey-8">เวลาออกงาน</div> <div class="col text-grey-8">เวลาออกงาน</div>
<div class="col"> <div class="col">{{ formData.checkOutTime }}</div>
{{
formData.checkOutDate
? `${formData.checkOutDate} ${formData.checkOutTime} น.`
: "-"
}}
</div>
</div> </div>
<div class="row"> <div class="row">
<div class="col text-grey-8">สถานททำงาน</div> <div class="col text-grey-8">สถานททำงาน</div>
<div class="col" v-if="formData.isLocationCheckOut"> <div class="col" v-if="formData.isLocationCheckOut">
{{ formData.checkOutLat ? "ในสถานที่" : "-" }} ในสถานท
</div> </div>
<div class="col" v-else> <div class="col" v-else>
{{ {{
formData.checkOutLocationName formData.checkInLocationName
? `นอกสถานที่ (${formData.checkOutLocationName})` ? `นอกสถานที่ (${formData.checkInLocationName})`
: "" : ""
}} }}
</div> </div>

View file

@ -74,12 +74,6 @@ const columns = ref<QTableProps["columns"]>([
field: "checkInTime", field: "checkInTime",
headerStyle: "font-size: 14px", headerStyle: "font-size: 14px",
style: "font-size: 14px", style: "font-size: 14px",
format(val, row) {
if (row.checkInDate && val) {
return `${row.checkInDate} ${val} น.`;
}
return "-";
},
}, },
{ {
name: "checkInLocation", name: "checkInLocation",
@ -107,12 +101,6 @@ const columns = ref<QTableProps["columns"]>([
field: "checkOutTime", field: "checkOutTime",
headerStyle: "font-size: 14px", headerStyle: "font-size: 14px",
style: "font-size: 14px", style: "font-size: 14px",
format(val, row) {
if (row.checkOutDate && val) {
return `${row.checkOutDate} ${val} น.`;
}
return "-";
},
}, },
{ {
name: "checkOutLocation", name: "checkOutLocation",
@ -172,19 +160,19 @@ async function fetchListTimeRecord() {
id: e.id, id: e.id,
fullName: e.fullName, fullName: e.fullName,
profileType: e.profileType, profileType: e.profileType,
checkInDate: date2Thai(e.checkInDate), checkInDate: e.checkInDate ? date2Thai(e.checkInDate) : "-",
checkInTime: e.checkInTime, checkInTime: e.checkInTime,
checkInLocation: e.checkInLocation, checkInLocation: e.checkInLocation,
checkInLat: e.checkInLat, checkInLat: e.checkInLat ? e.checkInLat : "",
checkInLon: e.checkInLon, checkInLon: e.checkInLon ? e.checkInLon : "",
checkInStatus: e.checkInStatus checkInStatus: e.checkInStatus
? workStore.convertSatatus(e.checkInStatus) ? workStore.convertSatatus(e.checkInStatus)
: "-", : "-",
checkOutDate: date2Thai(e.checkOutDate), checkOutDate: e.checkOutDate ? date2Thai(e.checkOutDate) : "-",
checkOutLocation: e.checkOutLocation, checkOutLocation: e.checkOutLocation ? e.checkOutLocation : "-",
checkOutTime: e.checkOutTime, checkOutTime: e.checkOutTime ? e.checkOutTime : "-",
checkOutLat: e.checkOutLat, checkOutLat: e.checkOutLat ? e.checkOutLat : "",
checkOutLon: e.checkOutLon, checkOutLon: e.checkOutLat ? e.checkOutLon : "",
checkOutStatus: e.checkOutStatus checkOutStatus: e.checkOutStatus
? workStore.convertSatatus(e.checkOutStatus) ? workStore.convertSatatus(e.checkOutStatus)
: "-", : "-",

View file

@ -70,12 +70,6 @@ const columns = ref<QTableProps["columns"]>([
field: "checkInTime", field: "checkInTime",
headerStyle: "font-size: 14px", headerStyle: "font-size: 14px",
style: "font-size: 14px", style: "font-size: 14px",
format(val, row) {
if (row.checkInDate && val) {
return `${row.checkInDate} ${val} น.`;
}
return "-";
},
}, },
{ {
name: "checkInLocation", name: "checkInLocation",
@ -94,12 +88,6 @@ const columns = ref<QTableProps["columns"]>([
field: "checkOutTime", field: "checkOutTime",
headerStyle: "font-size: 14px", headerStyle: "font-size: 14px",
style: "font-size: 14px", style: "font-size: 14px",
format(val, row) {
if (row.checkOutDate && val) {
return `${row.checkOutDate} ${val} น.`;
}
return "-";
},
}, },
{ {
name: "checkOutLocation", name: "checkOutLocation",
@ -145,15 +133,15 @@ async function fetchListLogRecord() {
profileType: e.profileType, profileType: e.profileType,
fullName: e.fullName, fullName: e.fullName,
checkInDate: e.checkInDate && date2Thai(e.checkInDate), checkInDate: e.checkInDate && date2Thai(e.checkInDate),
checkInTime: e.checkInTime, checkInTime: e.checkInTime ? e.checkInTime : "-",
checkInLocation: e.checkInLocation, checkInLocation: e.checkInLocation ? e.checkInLocation : "-",
checkInLat: e.checkInLat, checkInLat: e.checkInLat ? e.checkInLat : "",
checkInLon: e.checkInLon, checkInLon: e.checkInLon ? e.checkInLon : "",
checkOutDate: e.checkOutDate && date2Thai(e.checkOutDate), checkOutDate: e.checkOutDate && date2Thai(e.checkOutDate),
checkOutLocation: e.checkOutLocation, checkOutLocation: e.checkOutLocation ? e.checkOutLocation : "-",
checkOutTime: e.checkOutTime, checkOutTime: e.checkOutTime ? e.checkOutTime : "-",
checkOutLat: e.checkOutLat, checkOutLat: e.checkOutLat ? e.checkOutLat : "",
checkOutLon: e.checkOutLon, checkOutLon: e.checkOutLon ? e.checkOutLon : "",
})); }));
} else { } else {
rows.value = []; rows.value = [];

View file

@ -123,7 +123,7 @@ onMounted(() => {
{{ props.rowIndex + 1 }} {{ props.rowIndex + 1 }}
</div> </div>
<div v-else-if="col.name == 'checkInLocation'"> <div v-else-if="col.name == 'checkInLocation'">
<q-item style="padding: 0" v-if="props.row.checkInLocation"> <q-item style="padding: 0">
<q-item-section> <q-item-section>
<q-item-label> {{ props.row.checkInLocation }}</q-item-label> <q-item-label> {{ props.row.checkInLocation }}</q-item-label>
<q-item-label caption lines="2">{{ <q-item-label caption lines="2">{{
@ -131,10 +131,9 @@ onMounted(() => {
}}</q-item-label> }}</q-item-label>
</q-item-section> </q-item-section>
</q-item> </q-item>
<div v-else>-</div>
</div> </div>
<div v-else-if="col.name == 'checkOutLocation'"> <div v-else-if="col.name == 'checkOutLocation'">
<q-item style="padding: 0" v-if="props.row.checkOutLocation"> <q-item style="padding: 0">
<q-item-section> <q-item-section>
<q-item-label> {{ props.row.checkOutLocation }}</q-item-label> <q-item-label> {{ props.row.checkOutLocation }}</q-item-label>
<q-item-label caption lines="2">{{ <q-item-label caption lines="2">{{
@ -142,7 +141,6 @@ onMounted(() => {
}}</q-item-label> }}</q-item-label>
</q-item-section> </q-item-section>
</q-item> </q-item>
<div v-else>-</div>
</div> </div>
<div v-else-if="col.name == 'profileType'"> <div v-else-if="col.name == 'profileType'">
{{ {{
@ -152,7 +150,7 @@ onMounted(() => {
}} }}
</div> </div>
<div v-else> <div v-else>
{{ col.value ? col.value : "-" }} {{ col.value }}
</div> </div>
</q-td> </q-td>
</q-tr> </q-tr>

View file

@ -228,12 +228,8 @@ onMounted(async () => {
<template> <template>
<div class="q-mt-sm"> <div class="q-mt-sm">
<div class="row q-gutter-sm"> <div class="row q-gutter-sm">
<div class="col-3 scrollable-list"> <div class="col-3">
<q-list <q-list bordered class="rounded-borders">
bordered
class="rounded-borders"
style="max-height: 80vh; overflow-y: auto"
>
<q-item <q-item
v-for="(item, i) in filterLists" v-for="(item, i) in filterLists"
:key="i" :key="i"

View file

@ -82,7 +82,6 @@ const formData = reactive<FormData>({
status: "", // status: "", //
leaveLimit: 0, //() leaveLimit: 0, //()
leaveSummary: 0, //() leaveSummary: 0, //()
leaveWaitingSummary: 0, //()
leaveRemain: 0, //() leaveRemain: 0, //()
leaveWrote: "", // leaveWrote: "", //
leaveAddress: "", // leaveAddress: "", //
@ -150,41 +149,21 @@ const rows = ref<RowsType>();
// //
const idCheck = computed(() => { const idCheck = computed(() => {
if (typeAdd.value == "COMMANDER") { if (typeAdd.value == "COMMANDER") {
return rows.value?.commanders.map((items: SeqTypeRow) => items.keyId); return rows.value?.commanders.map((items: SeqTypeRow) => items.profileId);
} else if (typeAdd.value == "APPROVER") { } else if (typeAdd.value == "APPROVER") {
return rows.value?.approvers.map((items: SeqTypeRow) => items.keyId); return rows.value?.approvers.map((items: SeqTypeRow) => items.profileId);
} }
}); });
// //
const commanderList = computed(() => { const commanderList = computed(() => {
if (typeAdd.value === "COMMANDER") { if (typeAdd.value == "COMMANDER") {
return rows.value?.approvers.map((items: SeqTypeRow) => ({ return rows.value?.approvers.map((items: SeqTypeRow) => items.profileId);
profileId: items.profileId, } else if (typeAdd.value == "APPROVER") {
isAct: items.isAct, return rows.value?.commanders.map((items: SeqTypeRow) => items.profileId);
}));
} else if (typeAdd.value === "APPROVER") {
return rows.value?.commanders.map((items: SeqTypeRow) => ({
profileId: items.profileId,
isAct: items.isAct,
}));
} }
return [];
}); });
//
const isAct = computed(() => {
if (typeAdd.value === "COMMANDER") {
return rows.value?.commanders && rows.value.commanders.length > 0
? rows.value.commanders[0].isAct
: false;
} else if (typeAdd.value === "APPROVER") {
return rows.value?.approvers && rows.value.approvers.length > 0
? rows.value.approvers[0].isAct
: false;
}
return false;
});
// //
const approveCheck = computed(() => { const approveCheck = computed(() => {
const commanders = rows.value?.commanders; const commanders = rows.value?.commanders;
@ -412,9 +391,6 @@ async function fetchDetailLeave(paramsId: string) {
formData.leaveRange = data.leaveRange; formData.leaveRange = data.leaveRange;
formData.commanderPosition = data.commanderPosition; formData.commanderPosition = data.commanderPosition;
formData.leaveRangeEnd = data.leaveRangeEnd; formData.leaveRangeEnd = data.leaveRangeEnd;
formData.leaveWaitingSummary = data.leaveWaitingSummary
? data.leaveWaitingSummary
: "0";
keycloakUserId.value = data.keycloakUserId; keycloakUserId.value = data.keycloakUserId;
rows.value = { rows.value = {
commanders: data.commanders, commanders: data.commanders,
@ -692,10 +668,10 @@ onMounted(async () => {
fetchOptionType(), fetchOptionType(),
fetchKeycloakPositionData(), fetchKeycloakPositionData(),
fetchDetailLeave(paramsId), fetchDetailLeave(paramsId),
checkOfficer(),
]); ]);
await checkLeaveType(formData.leaveTypeId, formData); await checkLeaveType(formData.leaveTypeId, formData);
await checkOfficer();
} finally { } finally {
hideLoader(); hideLoader();
} }
@ -797,50 +773,40 @@ onMounted(async () => {
</div> </div>
</div> </div>
<div class="col-xs-12 col-sm-7 row"> <div class="col-xs-12 col-sm-7 row">
<div class="row col-12 q-col-gutter-md"> <div class="row col-12 q-gutter-md">
<div <div
v-if="formData.leaveTypeName == 'ลาพักผ่อน'" v-if="formData.leaveTypeName == 'ลาพักผ่อน'"
class="col-md-3 col-xs-6" class="col-3"
> >
<q-card bordered class="items-center row col-12 q-pa-md"> <q-card bordered class="items-center row col-12 q-pa-md">
<div class="text-h6 text-weight-bold text-blue-10"> <div class="text-h6 text-weight-bold text-blue-10">
{{ formData.leaveLimit }} {{ formData.leaveLimit }}
</div> </div>
<div class="col-12 text-subtitle2 text-weight-regular"> <div class="col-12 text-subtitle2 text-weight-regular">
<span>ได</span> <span class="gt-xs">ได</span>
</div> </div>
</q-card> </q-card>
</div> </div>
<div class="col-md-3 col-xs-6"> <div class="col-3">
<q-card bordered class="items-center row col-12 q-pa-md"> <q-card bordered class="items-center row col-12 q-pa-md">
<div class="text-h6 text-weight-bold text-light-blue-6"> <div class="text-h6 text-weight-bold text-light-blue-6">
{{ formData.leaveSummary }} {{ formData.leaveSummary }}
</div> </div>
<div class="col-12 text-subtitle2 text-weight-regular"> <div class="col-12 text-subtitle2 text-weight-regular">
<span>ใชไป</span> <span class="gt-xs">ใชไป</span>
</div> </div>
</q-card> </q-card>
</div> </div>
<div <div
v-if="formData.leaveTypeName == 'ลาพักผ่อน'" v-if="formData.leaveTypeName == 'ลาพักผ่อน'"
class="col-md-3 col-xs-6" class="col-3"
> >
<q-card bordered class="items-center row col-12 q-pa-md"> <q-card bordered class="items-center row col-12 q-pa-md">
<div class="text-h6 text-weight-bold text-indigo-7"> <div class="text-h6 text-weight-bold text-indigo-7">
{{ formData.leaveRemain }} {{ formData.leaveRemain }}
</div> </div>
<div class="col-12 text-subtitle2 text-weight-regular"> <div class="col-12 text-subtitle2 text-weight-regular">
<span>คงเหล</span> <span class="gt-xs">คงเหล</span>
</div>
</q-card>
</div>
<div class="col-md-3 col-xs-6">
<q-card bordered class="items-center row col-12 q-pa-md">
<div class="text-h6 text-weight-bold text-light-blue-6">
{{ formData.leaveWaitingSummary }}
</div>
<div class="col-12 text-subtitle2 text-weight-regular">
<span>อยระหวางการพจารณา</span>
</div> </div>
</q-card> </q-card>
</div> </div>
@ -1292,6 +1258,5 @@ onMounted(async () => {
:id-check="idCheck" :id-check="idCheck"
:keycloak-user-id="keycloakUserId" :keycloak-user-id="keycloakUserId"
:commanders-list="commanderList" :commanders-list="commanderList"
:commanders-is-act="isAct"
/> />
</template> </template>

View file

@ -151,7 +151,6 @@ const formData = reactive<FormData>({
leaveSubTypeName: "", leaveSubTypeName: "",
commanderPosition: "", commanderPosition: "",
leaveRangeEnd: "", leaveRangeEnd: "",
leaveWaitingSummary: 0, //()
}); });
const isLoadData = ref<boolean>(false); const isLoadData = ref<boolean>(false);
@ -218,9 +217,6 @@ async function fetchDetailLeave(paramsId: string) {
formData.leaveLimit = data.leaveLimit; formData.leaveLimit = data.leaveLimit;
formData.leaveSummary = data.leaveSummary; formData.leaveSummary = data.leaveSummary;
formData.leaveRemain = data.leaveRemain; formData.leaveRemain = data.leaveRemain;
formData.leaveWaitingSummary = data.leaveWaitingSummary
? data.leaveWaitingSummary
: 0;
formData.leaveWrote = data.leaveWrote; formData.leaveWrote = data.leaveWrote;
formData.leaveAddress = data.leaveAddress; formData.leaveAddress = data.leaveAddress;
formData.leaveNumber = data.leaveNumber; formData.leaveNumber = data.leaveNumber;
@ -630,50 +626,40 @@ onMounted(async () => {
</div> </div>
</div> </div>
<div class="col-xs-12 col-sm-7 row"> <div class="col-xs-12 col-sm-7 row">
<div class="row col-12 q-col-gutter-md"> <div class="row col-12 q-gutter-md">
<div <div
v-if="formData.leaveTypeName == 'ลาพักผ่อน'" v-if="formData.leaveTypeName == 'ลาพักผ่อน'"
class="col-md-3 col-xs-6" class="col-3"
> >
<q-card bordered class="items-center row col-12 q-pa-md"> <q-card bordered class="items-center row col-12 q-pa-md">
<div class="text-h6 text-weight-bold text-blue-10"> <div class="text-h6 text-weight-bold text-blue-10">
{{ formData.leaveLimit }} {{ formData.leaveLimit }}
</div> </div>
<div class="col-12 text-subtitle2 text-weight-regular"> <div class="col-12 text-subtitle2 text-weight-regular">
<span>ได</span> <span class="gt-xs">ได</span>
</div> </div>
</q-card> </q-card>
</div> </div>
<div class="col-md-3 col-xs-6"> <div class="col-3">
<q-card bordered class="items-center row col-12 q-pa-md"> <q-card bordered class="items-center row col-12 q-pa-md">
<div class="text-h6 text-weight-bold text-light-blue-6"> <div class="text-h6 text-weight-bold text-light-blue-6">
{{ formData.leaveSummary }} {{ formData.leaveSummary }}
</div> </div>
<div class="col-12 text-subtitle2 text-weight-regular"> <div class="col-12 text-subtitle2 text-weight-regular">
<span>ใชไป</span> <span class="gt-xs">ใชไป</span>
</div> </div>
</q-card> </q-card>
</div> </div>
<div <div
v-if="formData.leaveTypeName == 'ลาพักผ่อน'" v-if="formData.leaveTypeName == 'ลาพักผ่อน'"
class="col-md-3 col-xs-6" class="col-3"
> >
<q-card bordered class="items-center row col-12 q-pa-md"> <q-card bordered class="items-center row col-12 q-pa-md">
<div class="text-h6 text-weight-bold text-indigo-7"> <div class="text-h6 text-weight-bold text-indigo-7">
{{ formData.leaveRemain }} {{ formData.leaveRemain }}
</div> </div>
<div class="col-12 text-subtitle2 text-weight-regular"> <div class="col-12 text-subtitle2 text-weight-regular">
<span>คงเหล</span> <span class="gt-xs">คงเหล</span>
</div>
</q-card>
</div>
<div class="col-md-3 col-xs-6">
<q-card bordered class="items-center row col-12 q-pa-md">
<div class="text-h6 text-weight-bold text-light-blue-6">
{{ formData.leaveWaitingSummary }}
</div>
<div class="col-12 text-subtitle2 text-weight-regular">
<span>อยระหวางการพจารณา</span>
</div> </div>
</q-card> </q-card>
</div> </div>

View file

@ -4,6 +4,7 @@ import { useQuasar, type QTableProps } from "quasar";
import http from "@/plugins/http"; import http from "@/plugins/http";
import config from "@/app.config"; import config from "@/app.config";
import { getColumnLabel } from "@/utils/function";
import { useCounterMixin } from "@/stores/mixin"; import { useCounterMixin } from "@/stores/mixin";
import DialogHeader from "@/components/DialogHeader.vue"; import DialogHeader from "@/components/DialogHeader.vue";
@ -29,7 +30,6 @@ const props = defineProps({
fetchDetailLeave: Function, fetchDetailLeave: Function,
idCheck: Array, idCheck: Array,
commandersList: Array, commandersList: Array,
commandersIsAct: Boolean,
}); });
const pageId = ref<string>(route.params.id as string); const pageId = ref<string>(route.params.id as string);
@ -117,7 +117,7 @@ async function getData() {
total.value = data.total; total.value = data.total;
rows.value = data.data; rows.value = data.data;
selected.value = data.data.filter((items: any) => { selected.value = data.data.filter((items: any) => {
return props.idCheck?.some((i: any) => i === items.key); return props.idCheck?.some((i: any) => i === items.id);
}); });
}) })
.catch((err) => { .catch((err) => {
@ -153,14 +153,10 @@ function onSubmit() {
] ]
.filter(Boolean) .filter(Boolean)
.join(" "), .join(" "),
isAct: isAct.value,
keyId: items.key,
})); }));
const hasCommander = selected.value.some((e) => const hasCommander = selected.value.some((e) =>
props.commandersList?.some( props.commandersList?.some((i: any) => i === e.id)
(i: any) => i.profileId === e.id && i.isAct === isAct.value
)
); );
if (hasCommander) { if (hasCommander) {
@ -209,7 +205,6 @@ watch(
() => modal.value, () => modal.value,
() => { () => {
if (modal.value) { if (modal.value) {
isAct.value = props.commandersIsAct ?? false;
getSearch(); getSearch();
} }
} }
@ -238,7 +233,6 @@ watch(
label="ค้นหา" label="ค้นหา"
v-model="keyword" v-model="keyword"
style="width: 300px" style="width: 300px"
@keydown.enter.prevent="onSearchData()"
> >
<template v-slot:append> <template v-slot:append>
<q-icon name="search"></q-icon> <q-icon name="search"></q-icon>
@ -318,7 +312,9 @@ watch(
:key="col.name" :key="col.name"
:props="props" :props="props"
> >
<span class="text-weight-medium">{{ col.label }}</span> <span class="text-weight-medium">{{
getColumnLabel(col, isAct)
}}</span>
</q-th> </q-th>
</q-tr> </q-tr>
</template> </template>

View file

@ -1,59 +1,17 @@
<script setup lang="ts"> <script setup lang="ts">
import { ref, onMounted, computed } from "vue"; import { ref, onMounted, watch } from "vue";
import { useQuasar } from "quasar";
import type { QTableProps } from "quasar"; import type { QTableProps } from "quasar";
import { useRouter, useRoute } from "vue-router"; import { useRouter } from "vue-router";
import { checkPermission } from "@/utils/permissions"; import { checkPermission } from "@/utils/permissions";
import { useLeavelistDataStore } from "@/modules/09_leave/stores/LeaveStore"; import { useLeavelistDataStore } from "@/modules/09_leave/stores/LeaveStore";
import { useCounterMixin } from "@/stores/mixin";
import http from "@/plugins/http";
import config from "@/app.config";
const $q = useQuasar();
const leaveStore = useLeavelistDataStore(); const leaveStore = useLeavelistDataStore();
const route = useRoute();
const router = useRouter(); const router = useRouter();
const { showLoader, hideLoader, messageError, dialogRemove, success } =
useCounterMixin();
const total = defineModel<number>("total", { required: true }); const total = defineModel<number>("total", { required: true });
const totalList = defineModel<number>("totalList", { required: true }); const totalList = defineModel<number>("totalList", { required: true });
const pagination = defineModel<any>("pagination", { required: true }); const pagination = defineModel<any>("pagination", { required: true });
const props = defineProps({
getList: Function,
rows: {
type: Object,
require: true,
},
page: {
type: Number,
require: true,
},
rowsPerPage: {
type: Number,
require: true,
},
maxPage: {
type: Number,
require: true,
},
totalList: {
type: Number,
require: true,
},
dataToobar: Object,
});
const isPermissionDelete = computed(() => {
return (status: string) => {
return (
checkPermission(route)?.attrOwnership === "OWNER" &&
!leaveStore.statusDelete.includes(status) &&
leaveStore.tabMenu === "1"
);
};
});
/** ข้อมูลหัวตาราง รายการลา */ /** ข้อมูลหัวตาราง รายการลา */
const columnsLeave = ref<QTableProps["columns"]>([ const columnsLeave = ref<QTableProps["columns"]>([
@ -254,6 +212,31 @@ const visibleReject = ref<string[]>([
"status", "status",
]); ]);
const props = defineProps({
getList: Function,
rows: {
type: Object,
require: true,
},
page: {
type: Number,
require: true,
},
rowsPerPage: {
type: Number,
require: true,
},
maxPage: {
type: Number,
require: true,
},
totalList: {
type: Number,
require: true,
},
dataToobar: Object,
});
/** ไปหน้ารายละเอียด */ /** ไปหน้ารายละเอียด */
function redirectToDetail(id: string) { function redirectToDetail(id: string) {
const routePrefix = leaveStore.tabMenu === "1" ? "/leave" : "/leave-reject"; const routePrefix = leaveStore.tabMenu === "1" ? "/leave" : "/leave-reject";
@ -265,37 +248,6 @@ function updatePagination(newPagination: any) {
pagination.value.rowsPerPage = newPagination.rowsPerPage; pagination.value.rowsPerPage = newPagination.rowsPerPage;
} }
/**
* งกนสำหรบดงช CSS Class ตามสถานะ
* @param statusText สถานะของรายการ
*/
function getStatusColor(statusText: string) {
const statusMap: Record<string, string> = {
APPROVE: "text-green-6",
REJECT: "text-red",
NEW: "text-blue",
PENDING: "text-warning",
// DELETE DELETING
};
return statusMap[statusText.toUpperCase()] ?? "";
}
function handleDelete(id: string) {
dialogRemove($q, async () => {
try {
showLoader();
await http.delete(config.API.leaveList() + `/${id}`);
await props.getList?.();
success($q, "ลบข้อมูลสำเร็จ");
} catch (error) {
messageError($q, error);
} finally {
hideLoader();
}
});
}
/** Hook*/ /** Hook*/
onMounted(() => { onMounted(() => {
if (leaveStore.tabMenu === "1") { if (leaveStore.tabMenu === "1") {
@ -345,23 +297,23 @@ onMounted(() => {
> >
<q-tooltip>รายละเอยด</q-tooltip> <q-tooltip>รายละเอยด</q-tooltip>
</q-btn> </q-btn>
<q-btn
v-if="isPermissionDelete(props.row.statusText)"
flat
round
dense
icon="mdi-delete"
color="red"
@click.prevent="handleDelete(props.row.id)"
>
<q-tooltip>ลบขอม</q-tooltip>
</q-btn>
</q-td> </q-td>
<q-td <q-td
v-for="col in props.cols" v-for="col in props.cols"
:key="col.name" :key="col.name"
:props="props" :props="props"
:class="getStatusColor(props.row.statusText)" :class="
props.row.statusText == 'REJECT'
? 'text-red'
: props.row.statusText == 'NEW'
? 'text-blue'
: props.row.statusText == 'PENDING'
? 'text-warning'
: props.row.statusText == 'DELETE' ||
props.row.statusText == 'DELETING'
? 'text-orange'
: ''
"
> >
<div v-if="col.name == 'no'"> <div v-if="col.name == 'no'">
{{ {{

View file

@ -10,7 +10,6 @@ import { useCounterMixin } from "@/stores/mixin";
import { useLeaveHistoryDataStore } from "@/modules/09_leave/stores/LeaveHistoryStore"; import { useLeaveHistoryDataStore } from "@/modules/09_leave/stores/LeaveHistoryStore";
import { calculateFiscalYear } from "@/utils/function"; import { calculateFiscalYear } from "@/utils/function";
import { usePagination } from "@/composables/usePagination"; import { usePagination } from "@/composables/usePagination";
import { checkPermission } from "@/utils/permissions";
import type { QTableColumn } from "quasar"; import type { QTableColumn } from "quasar";
import type { DataOption } from "@/modules/09_leave/interface/index/Main"; import type { DataOption } from "@/modules/09_leave/interface/index/Main";
@ -105,18 +104,10 @@ async function onSubmit() {
leaveTypeId: formData.leaveTypeId, leaveTypeId: formData.leaveTypeId,
leaveYear: formData.leaveYear, leaveYear: formData.leaveYear,
leaveDays: formData.leaveDays ? Number(formData.leaveDays) : 0, leaveDays: formData.leaveDays ? Number(formData.leaveDays) : 0,
leaveDaysUsed: leaveDaysUsed: formData.leaveDaysUsed
checkPermission(route)?.attrOwnership === "OWNER" ? Number(formData.leaveDaysUsed)
? formData.leaveDaysUsed : 0,
? Number(formData.leaveDaysUsed) leaveCount: formData.leaveCount ? Number(formData.leaveCount) : 0,
: 0
: undefined,
leaveCount:
checkPermission(route)?.attrOwnership === "OWNER"
? formData.leaveCount
? Number(formData.leaveCount)
: 0
: undefined,
beginningLeaveDays: formData.beginningLeaveDays beginningLeaveDays: formData.beginningLeaveDays
? Number(formData.beginningLeaveDays) ? Number(formData.beginningLeaveDays)
: 0, : 0,
@ -322,7 +313,6 @@ watch(modal, async (val) => {
hide-bottom-space hide-bottom-space
dense dense
label="คำค้น" label="คำค้น"
@keydown.enter.prevent="onSearchData(true)"
> >
<template v-slot:after> <template v-slot:after>
<q-btn <q-btn
@ -396,7 +386,7 @@ watch(modal, async (val) => {
<q-separator vertical /> <q-separator vertical />
<!-- input --> <!-- input -->
<div class="col overflow-auto q-pa-md"> <div class="col overflow-hidden q-pa-md">
<div class="row q-col-gutter-sm"> <div class="row q-col-gutter-sm">
<div class="col-12"> <div class="col-12">
<datepicker <datepicker
@ -407,7 +397,6 @@ watch(modal, async (val) => {
year-picker year-picker
:enableTimePicker="false" :enableTimePicker="false"
:max-date="`${calculateFiscalYear(new Date())}, 12, 31`" :max-date="`${calculateFiscalYear(new Date())}, 12, 31`"
:class="classInput(true)"
> >
<template #year="{ year }">{{ year + 543 }}</template> <template #year="{ year }">{{ year + 543 }}</template>
<template #year-overlay-value="{ value }">{{ <template #year-overlay-value="{ value }">{{
@ -464,7 +453,7 @@ watch(modal, async (val) => {
</template> </template>
</q-select> </q-select>
</div> </div>
<div v-show="leaveTypeCode === 'LV-005'" class="col-12"> <div class="col-12">
<q-input <q-input
:class="classInput(leaveTypeCode == 'LV-005')" :class="classInput(leaveTypeCode == 'LV-005')"
:readonly="leaveTypeCode !== 'LV-005'" :readonly="leaveTypeCode !== 'LV-005'"
@ -472,10 +461,33 @@ watch(modal, async (val) => {
dense dense
outlined outlined
label="จำนวนสิทธิ์การลา" label="จำนวนสิทธิ์การลา"
hint="* สำหรับลาพักผ่อนเท่านั้น คือสิทธิ์ลาประจำปี + สิทธิ์สะสม" hide-bottom-space
:rules="[(val: string) => !val || /^\d+(\.\d*)?$/.test(val) || 'กรุณากรอกเฉพาะตัวเลข']" :rules="[(val: string) => !val || /^\d+(\.\d*)?$/.test(val) || 'กรุณากรอกเฉพาะตัวเลข']"
/> />
</div> </div>
<div class="col-12">
<q-input
:class="classInput(true)"
v-model="formData.leaveDaysUsed"
dense
outlined
label="ที่ใช้ไป (วัน)"
hide-bottom-space
:rules="[(val: string) => !val || /^\d+(\.\d*)?$/.test(val) || 'กรุณากรอกเฉพาะตัวเลข']"
/>
</div>
<div class="col-12">
<q-input
:class="classInput(true)"
v-model="formData.leaveCount"
dense
outlined
label="ที่ใช้ไป (ครั้ง)"
hide-bottom-space
mask="#"
reverse-fill-mask
/>
</div>
<div class="col-12"> <div class="col-12">
<q-input <q-input
@ -483,9 +495,9 @@ watch(modal, async (val) => {
v-model="formData.beginningLeaveDays" v-model="formData.beginningLeaveDays"
dense dense
outlined outlined
label="จำนวนวันลาก่อนใช้งานระบบ" label="ยกมา (วัน)"
hide-bottom-space
:rules="[(val: string) => !val || /^\d+(\.\d*)?$/.test(val) || 'กรุณากรอกเฉพาะตัวเลข']" :rules="[(val: string) => !val || /^\d+(\.\d*)?$/.test(val) || 'กรุณากรอกเฉพาะตัวเลข']"
hint="* จำนวนวันรวมการลาในปีงบประมาณนี้ที่เกิดขึ้นก่อนเริ่มใช้ระบบ"
/> />
</div> </div>
@ -495,38 +507,10 @@ watch(modal, async (val) => {
v-model="formData.beginningLeaveCount" v-model="formData.beginningLeaveCount"
dense dense
outlined outlined
label="จำนวนครั้งที่ลาก่อนใช้งานระบบ" label="ยกมา (ครั้ง)"
:rules="[(val: string) => !val || /^\d+$/.test(val) || 'กรุณากรอกเฉพาะตัวเลขที่เป็นจำนวนเต็ม']"
hint="* จำนวนครั้งของการลาในปีงบประมาณนี้ที่เกิดขึ้นก่อนเริ่มใช้ระบบ"
/>
</div>
<div
v-if="checkPermission(route)?.attrOwnership === 'OWNER'"
class="col-12"
>
<q-input
:class="classInput(true)"
v-model="formData.leaveDaysUsed"
dense
outlined
label="ที่ใช้ไปทั้งหมด (วัน)"
hide-bottom-space hide-bottom-space
:rules="[(val: string) => !val || /^\d+(\.\d*)?$/.test(val) || 'กรุณากรอกเฉพาะตัวเลข']" mask="#"
/> reverse-fill-mask
</div>
<div
v-if="checkPermission(route)?.attrOwnership === 'OWNER'"
class="col-12"
>
<q-input
:class="classInput(true)"
v-model="formData.leaveCount"
dense
outlined
label="ที่ใช้ไปทั้งหมด (ครั้ง)"
hide-bottom-space
:rules="[(val: string) => !val || /^\d+$/.test(val) || 'กรุณากรอกเฉพาะตัวเลขที่เป็นจำนวนเต็ม']"
/> />
</div> </div>
</div> </div>

View file

@ -4,8 +4,8 @@ interface DataPost {
lastName: string; lastName: string;
page: number; page: number;
pageSize: number; pageSize: number;
selectedNodeId?: string | null; selectedNodeId: string | null;
selectedNode?: string; selectedNode: string;
} }
interface DataOption { interface DataOption {

View file

@ -41,7 +41,6 @@ interface FormData {
status: string; //สถานะการของลา status: string; //สถานะการของลา
leaveLimit: number; //โควต้าลา(แต่ละประเภท)หน่วยเป็นวัน leaveLimit: number; //โควต้าลา(แต่ละประเภท)หน่วยเป็นวัน
leaveSummary: number; //ลาป่วยไปแล้ว(แต่ละประเภท)หน่วยเป็นวัน leaveSummary: number; //ลาป่วยไปแล้ว(แต่ละประเภท)หน่วยเป็นวัน
leaveWaitingSummary: number; //ลาอยู่ระหว่างการพิจารณา(แต่ละประเภท)หน่วยเป็นวัน
leaveRemain: number; //คงเหลือโควต้า(แต่ละประเภท)หน่วยเป็นวัน leaveRemain: number; //คงเหลือโควต้า(แต่ละประเภท)หน่วยเป็นวัน
// leaveStartDate: Date | null; //*วัน เดือน ปีเริ่มต้นลา // leaveStartDate: Date | null; //*วัน เดือน ปีเริ่มต้นลา
// leaveEndDate: Date | null; //*วัน เดือน ปีสิ้นสุดลา // leaveEndDate: Date | null; //*วัน เดือน ปีสิ้นสุดลา

View file

@ -50,8 +50,6 @@ interface SeqTypeRow {
keycloakId: string; keycloakId: string;
approveStatus: string; approveStatus: string;
comment: string; comment: string;
keyId?: string;
isAct?: boolean;
} }
interface DataDateMonthObject { interface DataDateMonthObject {
month: number; month: number;

View file

@ -44,8 +44,6 @@ export const useLeavelistDataStore = defineStore("leave", () => {
const leaveType = ref<LeaveType[]>([]); const leaveType = ref<LeaveType[]>([]);
const statusDelete = ["APPROVE", "DELETING", "DELETE"];
/** /**
* fetchListLeave * fetchListLeave
* @param data Page * @param data Page
@ -266,6 +264,5 @@ export const useLeavelistDataStore = defineStore("leave", () => {
leaveTypeOption, leaveTypeOption,
leaveTypeList, leaveTypeList,
fetchKeycloakPosition, fetchKeycloakPosition,
statusDelete,
}; };
}); });

View file

@ -11,14 +11,7 @@ export const useSpecialTimeStore = defineStore("LeaveSpecialTime", () => {
{ id: "NOT_COMPLETE", name: "ปฏิบัติงานไม่ครบตามกำหนดเวลา" }, { id: "NOT_COMPLETE", name: "ปฏิบัติงานไม่ครบตามกำหนดเวลา" },
]); ]);
const optionStatusMain = ref<DataOption[]>([ // convertSatatus
{ id: "ALL", name: "ทั้งหมด" },
{ id: "PENDING", name: "รอดำเนินการ" },
{ id: "APPROVE", name: "อนุมัติ" },
{ id: "REJECT", name: "ไม่อนุมัติ" },
]);
// convertStatus
function convertStatus(val: string) { function convertStatus(val: string) {
const value = val ? val.toUpperCase() : null; const value = val ? val.toUpperCase() : null;
switch (value) { switch (value) {
@ -36,6 +29,5 @@ export const useSpecialTimeStore = defineStore("LeaveSpecialTime", () => {
return { return {
optionStatus, optionStatus,
convertStatus, convertStatus,
optionStatusMain,
}; };
}); });

View file

@ -410,7 +410,6 @@ onMounted(() => {
@click="selectAllRows()" @click="selectAllRows()"
/> --> /> -->
<q-btn <q-btn
v-if="checkPermission($route)?.attrIsUpdate"
:disable="selected.length === 0 || isLoadingAll" :disable="selected.length === 0 || isLoadingAll"
:color="selected.length === 0 ? 'grey' : 'info'" :color="selected.length === 0 ? 'grey' : 'info'"
dense dense

View file

@ -10,10 +10,8 @@ import { useSpecialTimeStore } from "@/modules/09_leave/stores/SpecialTimeStore"
import { checkPermission } from "@/utils/permissions"; import { checkPermission } from "@/utils/permissions";
import { usePagination } from "@/composables/usePagination"; import { usePagination } from "@/composables/usePagination";
import type { import type { DataDateMonthObject } from "@/modules/09_leave/interface/request/specialTime";
DataSpecialTime, import type { DataSpecialTime } from "@/modules/09_leave/interface/index/Main";
DataOption,
} from "@/modules/09_leave/interface/index/Main";
import DialogReason from "@/components/Dialogs/PopupReason.vue"; import DialogReason from "@/components/Dialogs/PopupReason.vue";
import DialogApprove from "@/modules/09_leave/components/04_SpecialTime/DialogApprove.vue"; import DialogApprove from "@/modules/09_leave/components/04_SpecialTime/DialogApprove.vue";
@ -23,17 +21,23 @@ const mixin = useCounterMixin();
const store = useSpecialTimeStore(); const store = useSpecialTimeStore();
const { const {
hideLoader, hideLoader,
monthYear2Thai,
messageError, messageError,
showLoader, showLoader,
success, success,
date2Thai, date2Thai,
dialogConfirm, dialogConfirm,
dateToISO,
} = mixin; } = mixin;
const { pagination, params, onRequest } = usePagination("", fetchData); const { pagination, params, onRequest } = usePagination("", fetchData);
const emit = defineEmits(["update:change-page"]); const emit = defineEmits(["update:change-page"]);
const toDay = ref<Date>(new Date());
const monthToday = toDay.value.getMonth();
const yearToday = toDay.value.getFullYear();
const month = ref<number>(monthToday + 1);
const year = ref<number>(yearToday);
const description = ref<string>(""); const description = ref<string>("");
/**ตัวแปรที่ใช้ */ /**ตัวแปรที่ใช้ */
@ -47,12 +51,16 @@ const name = ref<string>("");
const id = ref<string>(""); const id = ref<string>("");
const dateDialog = ref<string>(""); const dateDialog = ref<string>("");
const dateFixDialog = ref<string>(""); const dateFixDialog = ref<string>("");
const dateYear = ref<number>(new Date().getFullYear());
/** Function Date */
const dateMonth = ref<DataDateMonthObject>({
month: new Date().getMonth(),
year: new Date().getFullYear(),
});
// //
const filterKeyword = ref<string>(""); const filterKeyword = ref<string>("");
const filterStatus = ref<string>("PENDING");
const filterDate = ref<[Date, Date] | null>([new Date(), new Date()]); //
const optionStatus = ref<DataOption[]>(store.optionStatusMain);
const rows = ref<DataSpecialTime[]>([]); const rows = ref<DataSpecialTime[]>([]);
const visibleColumns = ref<String[]>([ const visibleColumns = ref<String[]>([
"no", "no",
@ -136,10 +144,9 @@ async function fetchData() {
.get(config.API.specialTime(), { .get(config.API.specialTime(), {
params: { params: {
...params.value, ...params.value,
year: year.value,
month: month.value,
keyword: filterKeyword.value.trim(), keyword: filterKeyword.value.trim(),
status: filterStatus.value != "ALL" ? filterStatus.value : null,
startDate: filterDate.value ? dateToISO(filterDate.value[0]) : null,
endDate: filterDate.value ? dateToISO(filterDate.value[1]) : null,
}, },
}) })
.then(async (res) => { .then(async (res) => {
@ -239,43 +246,40 @@ async function clickSave(reason: string) {
}); });
} }
/**
* งขอมลตามป
* @param e
*/
async function updateMonth(e: DataDateMonthObject) {
if (e != null) {
dateYear.value = e.year;
dateYear.value = year.value;
month.value = dateMonth.value.month + 1;
year.value = dateMonth.value.year;
onSearchData();
}
}
//
function monthYearThai(val: DataDateMonthObject) {
if (val == null) return "";
else return monthYear2Thai(val.month, val.year);
}
/** ฟังก์ชั่นค้นหาข้อมูล */ /** ฟังก์ชั่นค้นหาข้อมูล */
function onSearchData() { function onSearchData() {
pagination.value.page = 1; pagination.value.page = 1;
fetchData(); fetchData();
} }
/**
* งกนคนหาขอมลของ Option Filter
* @param val คำทนหา
* @param update Function
* @param typeOp ประเภทของ Select
*/
function filterOption(val: string, update: Function) {
update(() => {
const needle = val.toLowerCase();
optionStatus.value = store.optionStatusMain.filter(
(v: DataOption) => v.name.toLowerCase().indexOf(needle) > -1
);
});
}
/**
* แปลงชวงวนทา2คาเปนวนเดยวกนจะโชววนเดยวแตาไมเทากนจะแสดงเปนชวง
* @param val วงวนท
*/
function dateThaiRange(val: [Date, Date]) {
if (val === null) {
return "";
} else if (date2Thai(val[0], true) === date2Thai(val[1], true)) {
return `${date2Thai(val[0], true)}`;
} else {
return `${date2Thai(val[0], true)} - ${date2Thai(val[1], true)}`;
}
}
/**Hook */ /**Hook */
onMounted(async () => { onMounted(async () => {
//
const toDay = ref<Date>(new Date());
const monthToday = toDay.value.getMonth();
const yearToday = toDay.value.getFullYear();
month.value = monthToday + 1;
year.value = yearToday;
await fetchData(); await fetchData();
}); });
</script> </script>
@ -287,31 +291,25 @@ onMounted(async () => {
<q-card flat bordered class="col-12 q-pa-md"> <q-card flat bordered class="col-12 q-pa-md">
<div class="row col-12 q-col-gutter-sm q-mb-sm"> <div class="row col-12 q-col-gutter-sm q-mb-sm">
<div class="col-xs-12 col-sm-6 col-md-3"> <div class="col-xs-12 col-sm-3 col-md-2">
<datepicker <datepicker
v-model="filterDate" v-model="dateMonth"
:locale="'th'" :locale="'th'"
autoApply autoApply
borderless month-picker
range
:enableTimePicker="false" :enableTimePicker="false"
week-start="0" @update:modelValue="updateMonth"
@update:modelValue="onSearchData()"
> >
<template #year="{ year }"> <template #year="{ year }">{{ year + 543 }}</template>
{{ year + 543 }} <template #year-overlay-value="{ value }">{{
</template> parseInt(value + 543)
<template #year-overlay-value="{ value }"> }}</template>
{{ parseInt(value + 543) }}
</template>
<template #trigger> <template #trigger>
<q-input <q-input
outlined :model-value="monthYearThai(dateMonth)"
dense dense
class="full-width datepicker" outlined
:model-value=" hide-bottom-space
filterDate != null ? dateThaiRange(filterDate) : null
"
> >
<template v-slot:prepend> <template v-slot:prepend>
<q-icon <q-icon
@ -325,37 +323,9 @@ onMounted(async () => {
</template> </template>
</datepicker> </datepicker>
</div> </div>
<div class="col-xs-12 col-sm-6 col-md-2">
<q-select
hide-bottom-space
dense
lazy-rules
outlined
onlind
hide-selected
v-model="filterStatus"
label="สถานะ"
emit-value
map-options
:options="optionStatus"
option-value="id"
use-input
fill-input
option-label="name"
@update:modelValue="onSearchData()"
@filter="(inputValue:string, doneFn:Function) =>
filterOption(inputValue, doneFn,)"
>
<template v-slot:no-option>
<q-item>
<q-item-section class="text-grey"> ไมอม </q-item-section>
</q-item>
</template>
</q-select>
</div>
<q-space v-if="!$q.screen.xs && !$q.screen.sm" /> <q-space />
<div class="col-xs-12 col-sm-6 col-md-2"> <div class="col-xs-12 col-sm-3 col-md-2">
<q-input <q-input
standout standout
dense dense
@ -370,7 +340,7 @@ onMounted(async () => {
</q-input> </q-input>
</div> </div>
<div class="col-xs-12 col-sm-6 col-md-2"> <div class="col-xs-12 col-sm-3 col-md-2">
<q-select <q-select
v-model="visibleColumns" v-model="visibleColumns"
multiple multiple

View file

@ -83,7 +83,7 @@ const columns = ref<QTableColumn[]>([
{ {
name: "leaveDaysUsed", name: "leaveDaysUsed",
align: "left", align: "left",
label: "ที่ใช้ไปทั้งหมด (วัน)", label: "ที่ใช้ไป (วัน)",
sortable: true, sortable: true,
field: "leaveDaysUsed", field: "leaveDaysUsed",
headerStyle: "font-size: 14px", headerStyle: "font-size: 14px",
@ -92,7 +92,7 @@ const columns = ref<QTableColumn[]>([
{ {
name: "leaveCount", name: "leaveCount",
align: "left", align: "left",
label: "ที่ใช้ไปทั้งหมด (ครั้ง)", label: "ที่ใช้ไป (ครั้ง)",
sortable: true, sortable: true,
field: "leaveCount", field: "leaveCount",
headerStyle: "font-size: 14px", headerStyle: "font-size: 14px",
@ -101,7 +101,7 @@ const columns = ref<QTableColumn[]>([
{ {
name: "beginningLeaveDays", name: "beginningLeaveDays",
align: "left", align: "left",
label: "จำนวนวันลาก่อนใช้งานระบบ", label: "ยกมา (วัน)",
sortable: true, sortable: true,
field: "beginningLeaveDays", field: "beginningLeaveDays",
headerStyle: "font-size: 14px", headerStyle: "font-size: 14px",
@ -110,7 +110,7 @@ const columns = ref<QTableColumn[]>([
{ {
name: "beginningLeaveCount", name: "beginningLeaveCount",
align: "left", align: "left",
label: "จำนวนครั้งที่ลาก่อนใช้งานระบบ", label: "ยกมา (ครั้ง)",
sortable: true, sortable: true,
field: "beginningLeaveCount", field: "beginningLeaveCount",
headerStyle: "font-size: 14px", headerStyle: "font-size: 14px",

View file

@ -2,7 +2,6 @@
import { ref } from "vue"; import { ref } from "vue";
import { useRoute } from "vue-router"; import { useRoute } from "vue-router";
import { useQuasar } from "quasar"; import { useQuasar } from "quasar";
import type { QTableColumn } from "quasar";
import http from "@/plugins/http"; import http from "@/plugins/http";
import config from "@/app.config"; import config from "@/app.config";
@ -23,12 +22,21 @@ const checkRoutePermisson = ref<boolean>(
route.name == "disciplineInvestigatefactsDetail" route.name == "disciplineInvestigatefactsDetail"
); );
const props = defineProps<{ const props = defineProps({
rows?: any[]; rows: {
columns?: QTableColumn[]; type: Array,
visibleColumns?: string[]; default: [],
fetchData?: () => void; },
}>(); columns: {
type: Array,
default: [],
},
visibleColumns: {
type: Array,
default: [],
},
fetchData: Function,
});
const remark = ref<string>(""); const remark = ref<string>("");
const selected = ref<any[]>([]); const selected = ref<any[]>([]);
@ -92,7 +100,7 @@ function onSubmit() {
<q-card-section class="q-pa-xs"> <q-card-section class="q-pa-xs">
<q-table <q-table
:columns="props.columns" :columns="props.columns"
:rows="props.rows ?? []" :rows="rows"
row-key="personId" row-key="personId"
flat flat
bordered bordered

View file

@ -10,7 +10,7 @@ import { useInvestigateFactStore } from "@/modules/11_discipline/store/Investiga
import { useDisciplineMainStore } from "@/modules/11_discipline/store/Main"; import { useDisciplineMainStore } from "@/modules/11_discipline/store/Main";
import type { ArrayPersonAdd } from "@/modules/11_discipline/interface/response/investigate"; import type { ArrayPersonAdd } from "@/modules/11_discipline/interface/response/investigate";
import type { FormData } from "@/modules/11_discipline/interface/request/investigateFact"; import type { FormData } from "@/modules/11_discipline/interface/request/InvestigateFact";
import type { import type {
FormData as FormDataComplaint, FormData as FormDataComplaint,
ArrayPerson, ArrayPerson,

View file

@ -11,7 +11,7 @@ import { useInvestigateDisStore } from "@/modules/11_discipline/store/Investigat
import { useInvestigateFactStore } from "@/modules/11_discipline/store/InvestigateFactStore"; import { useInvestigateFactStore } from "@/modules/11_discipline/store/InvestigateFactStore";
import { useDisciplineMainStore } from "@/modules/11_discipline/store/Main"; import { useDisciplineMainStore } from "@/modules/11_discipline/store/Main";
import type { FormData } from "@/modules/11_discipline/interface/request/investigateFact"; import type { FormData } from "@/modules/11_discipline/interface/request/InvestigateFact";
import type { OptionData } from "@/modules/07_insignia/interface/index/Main"; import type { OptionData } from "@/modules/07_insignia/interface/index/Main";
import CalandarDialog from "@/modules/11_discipline/components/2_InvestigateFacts/CalandarDialog.vue"; import CalandarDialog from "@/modules/11_discipline/components/2_InvestigateFacts/CalandarDialog.vue";

Some files were not shown because too many files have changed in this diff Show more