Compare commits

..

197 commits
v1.1.13 ... dev

Author SHA1 Message Date
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
20f75d24b0 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 2m51s
2026-05-12 18:07:41 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
50fdb30920 fix(receive): Display columns posTypeOld 2026-05-12 18:07:29 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
ae8e667922 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m0s
2026-05-12 17:46:12 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
5ead16e528 fix(receive): diaplay rank 2026-05-12 17:45:54 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
9cbe56fb1e Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 2m56s
2026-05-12 17:15:38 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
95f37cf68d fix(leave): fetch checkOfficer 2026-05-12 17:14:53 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
bee7aa54f0 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 2m41s
2026-05-12 15:25:34 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
762ed36190 Merge branch 'feat/receive' into develop 2026-05-12 15:25:10 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
e14ba58df0 refactor(receive): display prefix 2026-05-12 15:24:53 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
8d83bd2b1e Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 2m56s
2026-05-12 14:25:51 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
9228132052 refactor: display getColumnLabel 2026-05-12 14:25:33 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
8f7a5c058b Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 2m55s
2026-05-12 12:03:31 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
94d80409c3 fix(discipline): import path interface 2026-05-12 12:03:03 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
eed4258617 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 2m57s
2026-05-12 11:42:28 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
b1a7983b00 refactor: add keydown event listener to search input 2026-05-12 11:42:03 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
1b2636cea2 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m6s
2026-05-11 17:35:03 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
3c91c0db16 fix(resign): add @keydown search keyword 2026-05-11 17:34:45 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
dd875380df Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 2m47s
2026-05-11 15:56:07 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
8dc473eec0 refactor(leave): validate isAct commander 2026-05-11 15:54:10 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
852f3226c1 refactor(qualify-period): hide type column in Table Position 2026-05-11 15:07:00 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
4c7ab0f7e6 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 2m53s
2026-05-08 11:01:50 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
e998de2b95 fix: remove max-height style from TableCandidate 2026-05-08 11:01:33 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
084cf7bcfa Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m0s
2026-05-08 10:41:31 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
ba392dec41 fix(issues): statusOptions value 2026-05-08 10:41:14 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
735617b541 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 2m53s
2026-05-08 10:19:36 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
b22ae37eec reafactor(issues): add status HELPDESK_IN_PROGRESS and REPLIED 2026-05-08 10:19:21 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
cf58e28a11 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m3s
2026-05-07 16:24:51 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
3cc8d9e974 refactor(timestamp-special): filter date and status 2026-05-07 16:24:30 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
0739f2b9d7 feat(receive): add input rank 2026-05-07 10:40:58 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
7848323b1f Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 2m53s
2026-05-06 10:55:15 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
29b45f77ea refactor(placement): interface DataList 2026-05-06 10:54:37 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
5ac50fc1f2 refactor(placement): PersonalList DialogSelectOrg 2026-05-06 10:53:53 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
a262319c20 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m1s
2026-05-06 09:48:29 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
ebdea05f4d fix 2026-05-06 09:48:19 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
22a9f0f935 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m8s
2026-05-06 09:40:31 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
e513586b65 refactor(DialogOrgSelect): fetch data 2026-05-06 09:39:53 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
46285f5d18 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m3s
2026-05-05 17:40:21 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
999a2d1b45 refactor(DialogOrgSelect): reset positionData to default values 2026-05-05 17:39:52 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
26f5d286d1 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m8s
2026-05-05 17:13:41 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
a7341c7440 refactor(placement): DialogSelectOrg LoadPosition 2026-05-05 17:12:56 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
0d1dea5289 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 2m49s
2026-05-05 15:56:32 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
7bbd9ae933 fix(DialogOrgSelect): fetch position 2026-05-05 15:56:14 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
55d3897b2d Merge branch 'develop' into dev 2026-05-05 10:35:54 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
44487139ea feat(file):add validateFileSize 2026-05-05 10:35:29 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
5ef1059659 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m28s
2026-05-01 17:53:02 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
c69c4c1fa8 fix 2026-05-01 17:52:41 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
dc71d0bb59 Merge branch 'develop' into dev 2026-05-01 16:35:38 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
05afc1fe5b refactor(exam): API candidate-exam 2026-05-01 16:21:44 +07:00
de95ae61c0 แก้ฟอร์มสิทธิ์การลา staff และ OWNER 2026-05-01 16:00:08 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
4827823e82 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m16s
2026-05-01 15:31:43 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
c20e67d1a1 fix(command): loadFile 2026-05-01 15:31:18 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
a71cdec921 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m5s
2026-04-30 16:07:22 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
598492b61e refactor(change-round): add permission check using attrIsUpdate for change round button
Co-authored-by: Copilot <copilot@github.com>
2026-04-30 16:07:02 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
aad1009c0a Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m9s
2026-04-29 13:58:16 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
9e550fc5bf refactor(placement-emp): hide positionDate input when typeCommand is MOVE 2026-04-29 13:58:01 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
6f798eed20 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m2s
2026-04-29 13:46:45 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
ebf34aa0c2 refactor(placement): hide positionDate input when typeCommand is MOVE 2026-04-29 13:46:31 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
9cae94ae75 Merge branch 'develop' into dev 2026-04-29 13:31:51 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
bad9bd3adf refactor(leave-history): change leaveDays from integer to float 2026-04-29 13:31:31 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
16707fee64 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m11s
2026-04-28 13:55:32 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
1f0ec397ff fix: delete hide-bottom Table Expand 2026-04-28 13:55:11 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
2a49e67022 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m4s
2026-04-28 11:02:57 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
df5268705c refactor(UI):display checkOutLocationName 2026-04-28 11:02:34 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
7c7f344a4a Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m6s
2026-04-28 10:07:33 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
69859517fb Merge branch 'develop' of https://github.com/Frappet/bma-ehr-frontend into develop 2026-04-28 10:07:12 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
8d5384c1d3 refactor(UI work-list): display column time
Co-authored-by: Copilot <copilot@github.com>
2026-04-28 10:07:00 +07:00
4486a32fd8 Merge branch 'dev' of https://forgejo.chamomind.com/hrms-bangkok/hrms-mgt into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m19s
* 'dev' of https://forgejo.chamomind.com/hrms-bangkok/hrms-mgt:
2026-04-24 16:26:29 +07:00
fce944d90f Merge branch 'develop' into dev
* develop:
  fix display gov age
  refactor(worklist): formatDate colunm checkInTime  checkOutTime
2026-04-24 16:26:24 +07:00
83b6d723ba Merge branch 'develop' of github.com:Frappet/hrms-mgt into develop
* 'develop' of github.com:Frappet/hrms-mgt:
  refactor(worklist): formatDate colunm checkInTime  checkOutTime
2026-04-24 16:25:36 +07:00
5b71a6a37b fix display gov age 2026-04-24 16:25:24 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
130adabc27 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m18s
2026-04-24 15:59:23 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
8a1fb4f9ed refactor(worklist): formatDate colunm checkInTime checkOutTime 2026-04-24 15:59:07 +07:00
e1bfc1b3bf Merge branch 'dev' of https://forgejo.chamomind.com/hrms-bangkok/hrms-mgt into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m42s
* 'dev' of https://forgejo.chamomind.com/hrms-bangkok/hrms-mgt:
2026-04-24 15:02:34 +07:00
e1db58294b Merge branch 'develop' into dev
* develop:
  แก้ไขฟอร์มสิทธิ์การลา
  refactor: dialogConfirm  onConfirmUpload
  refactor(leave-history): replace overflow-hidden with overflow-auto for better scrollability
  feat(registry-edit): previewFile excel Position
  feat(leave): UI Show Card leaveWaitingSummary
  refactor(registry-edit): permisson handUploadFile
  refactor(registry-edit): exportExcel
  feat(registry-edit): uploadFile profileSalaryTemp
  fixed สิทธิ์การลา
  refactor(retirement): download retirementReport
2026-04-24 15:02:26 +07:00
bf5979aa2d Merge branch 'develop' of github.com:Frappet/hrms-mgt into develop
* 'develop' of github.com:Frappet/hrms-mgt:
  refactor: dialogConfirm  onConfirmUpload
  refactor(leave-history): replace overflow-hidden with overflow-auto for better scrollability
  feat(registry-edit): previewFile excel Position
  feat(leave): UI Show Card leaveWaitingSummary
  refactor(registry-edit): permisson handUploadFile
2026-04-24 15:02:01 +07:00
41a9f9cb89 แก้ไขฟอร์มสิทธิ์การลา 2026-04-24 15:01:50 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
c3dc4ce774 Merge branch 'develop' into dev 2026-04-24 14:51:47 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
52f4db00a9 refactor: dialogConfirm onConfirmUpload 2026-04-24 14:51:31 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
9463d8c9f9 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m29s
2026-04-24 14:29:53 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
7211081d14 refactor(leave-history): replace overflow-hidden with overflow-auto for better scrollability 2026-04-24 14:29:32 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
bfe71c1a23 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m29s
2026-04-24 14:06:16 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
9132efed11 feat(registry-edit): previewFile excel Position 2026-04-24 14:06:01 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
69f9657719 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m45s
2026-04-24 09:53:02 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
433a0ce9b8 feat(leave): UI Show Card leaveWaitingSummary 2026-04-24 09:52:14 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
230dba078c Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m27s
2026-04-24 09:24:06 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
874bedf7eb refactor(registry-edit): permisson handUploadFile 2026-04-24 09:23:46 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
d296b2e4ac Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m58s
2026-04-23 17:22:25 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
24d08ac0cf refactor(registry-edit): exportExcel 2026-04-23 17:22:01 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
88667b8ea8 Merge branch 'develop' of https://github.com/Frappet/bma-ehr-frontend into develop 2026-04-23 16:42:38 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
cf053cb306 feat(registry-edit): uploadFile profileSalaryTemp 2026-04-23 16:42:31 +07:00
5bf1f2f197 fixed สิทธิ์การลา 2026-04-23 10:43:08 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
c6bb8fffe2 Merge branch 'dev' of https://forgejo.chamomind.com/hrms-bangkok/hrms-mgt into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 2m58s
2026-04-21 14:15:06 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
34cf21da65 Merge branch 'develop' into dev 2026-04-21 14:14:59 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
1de3d35752 refactor(retirement): download retirementReport 2026-04-21 14:14:38 +07:00
ac7da9c363 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m4s
* develop:
  fixed format email
2026-04-20 09:55:37 +07:00
5193720567 fixed format email 2026-04-20 09:55:27 +07:00
f7412ca1a8 Merge branch 'develop' into dev
* develop:
  fixed#1565 ระบบแจ้งปัญหาบังคับให้กรอกอีเมล และระบุข้อความว่า admin จะติดต่อกลับทางอีเมลเป็นลำดับแรกกรุณาตรวจสอบที่อีเมล
  fixed#2435 ระบบแก้ไขทะเบียนประวัติตำแหน่ง/เงินเดือน >> เพิ่มตำแหน่ง/เงินเดือน (ชื่อช่องกรอกข้อมูลหาย)
2026-04-20 09:47:36 +07:00
0ccda33b37 fixed#1565
ระบบแจ้งปัญหาบังคับให้กรอกอีเมล และระบุข้อความว่า admin จะติดต่อกลับทางอีเมลเป็นลำดับแรกกรุณาตรวจสอบที่อีเมล
2026-04-20 09:47:20 +07:00
cba8f4b703 fixed#2435
ระบบแก้ไขทะเบียนประวัติตำแหน่ง/เงินเดือน >> เพิ่มตำแหน่ง/เงินเดือน (ชื่อช่องกรอกข้อมูลหาย)
2026-04-20 09:40:16 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
926c47f273 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m27s
2026-04-17 15:29:31 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
d6e75d6966 refactor(command): dialogMessageNotify messageWarning 2026-04-17 15:29:11 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
170568384d Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m33s
2026-04-17 15:00:02 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
7f25ed4ef1 Merge branch 'develop' of https://github.com/Frappet/bma-ehr-frontend into develop 2026-04-17 14:59:29 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
8a0a6ea873 fix(command): validate if isIdofficer isBangkok fields to show warnings 2026-04-17 14:59:22 +07:00
833718d276 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 5m4s
* develop:
  add permission owner
2026-04-16 20:56:07 +07:00
73335d7dd5 add permission owner 2026-04-16 20:55:51 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
07659ecc6c Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m0s
2026-04-16 16:14:51 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
9d9cd92d6b refactor(retire-old): standardize API response by renaming data to result 2026-04-16 16:14:33 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
40ebaa646f Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m8s
2026-04-16 13:41:07 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
c700cf0abd feat(placement): add UpdateDraftStatus menu option 2026-04-16 13:40:48 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
b0b834cb9b Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 2m54s
2026-04-10 16:06:04 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
0d5cb36fb2 feat(registry): enable edit for items from command list 2026-04-10 16:05:30 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
cf468e14cb Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 2m57s
2026-04-09 13:43:02 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
c21cfa838f refactor(personal-detail): resetData befor fetchData 2026-04-09 13:42:40 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
376fba6059 feat(personal-detail): add display section for personal info 2026-04-09 12:09:33 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
3f5a4783c1 refactor(reportOrg): change getReport API method from GET to POST 2026-04-09 10:05:42 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
1de7a11721 refactor(evaluate): convert table headers to data cells 2026-04-09 09:09:01 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
b233b424d3 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m16s
2026-04-08 10:54:35 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
7da397a1fe fix(command): handle dialog notification for Bangkok officers 2026-04-08 10:54:08 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
c8a8321014 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m3s
2026-04-03 17:33:02 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
e11d9b18dc Merge branch 'feat/evaluate' into develop 2026-04-03 17:32:42 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
006c2bc0ba feat(evaluate): implement delete endpoints for meeting and director 2026-04-03 17:32:32 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
f408889000 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m3s
2026-04-03 10:08:35 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
f5e1c0eca6 refactor: convert displayAdd to a computed property 2026-04-03 10:08:12 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
ae495a90e1 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 2m48s
2026-04-02 15:50:36 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
26977319ed fix: mask leaveDaysUsed add beginningLeaveDays 2026-04-02 15:49:48 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
e6921f4166 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m1s
2026-04-02 09:22:58 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
f403813099 fix(leave): diaplay columns completedDate 2026-04-02 09:22:30 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
5641a14cb7 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m0s
2026-04-01 17:03:34 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
503d111634 fix: fetch permission.attrOwnership 2026-04-01 16:59:59 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
fe30d01330 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 2m59s
2026-04-01 16:17:42 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
17c05d60d7 fix(ui): class q-pt-none 2026-04-01 16:17:14 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
546942f8fb feat(): add handleDelete director meeting 2026-04-01 13:40:50 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
f12d999bbc Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m25s
2026-04-01 11:33:56 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
a6018507c9 fix(checkin-report): resolve preview display issue for typeReport 4 2026-04-01 11:33:33 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
9275ef6bfe Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m32s
2026-04-01 11:10:01 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
514de15f09 feat(leave): add Processed Late tab to work-list 2026-04-01 11:09:33 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
0303046e7a Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 2m58s
2026-03-30 10:41:33 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
e9bcebee6d fix(exam): add qualify period validation for announcement type 2026-03-30 10:40:57 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
c00b6f89fe Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 2m56s
2026-03-30 09:18:32 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
70d2c59f65 fix: display 2026-03-28 11:27:13 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
f1e8bc0997 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m25s
2026-03-27 16:26:13 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
abce475de0 fix(retirement): checkRoutePermisson 2026-03-27 16:25:56 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
b1ae19afa7 fix(placement): condition edit 2026-03-27 16:07:14 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
e151217c72 Merge branch 'develop' into dev 2026-03-27 10:45:15 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
24c307e076 refactor(ui report): remove unused code and clean up 2026-03-27 10:44:41 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
689c29ada4 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m21s
2026-03-26 11:12:19 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
db4df70a83 fix(salary): filter data to show only records with groupid 2026-03-26 11:11:59 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
c7accb6044 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m40s
2026-03-26 10:18:08 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
dd5b2e0676 fix(discipline): clear selected employees on class change 2026-03-26 10:16:46 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
3c15bb3b0b fix(salary): add null check for id before processing 2026-03-26 09:59:29 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
4cfcd48a20 Merge branch 'develop' into dev 2026-03-26 09:49:31 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
26ae535b15 Merge branch 'develop' of https://github.com/Frappet/bma-ehr-frontend into develop 2026-03-26 09:49:14 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
ec5f113922 fix(registry): handle router.push based on isLeave status 2026-03-26 09:49:02 +07:00
514f2c548d Merge branch 'dev' of https://forgejo.chamomind.com/hrms-bangkok/hrms-mgt into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m20s
* 'dev' of https://forgejo.chamomind.com/hrms-bangkok/hrms-mgt:
2026-03-25 17:49:33 +07:00
6967d90c25 Merge branch 'develop' into dev
* develop:
  update selected and load data
  fix: insigniaForm.year
  refactor(ui): apply +543 year formatting to columns
  fix: payload multiple
  fix: rowsPerPage 100
  fix
  fix:add filter rowsPerPage = 3000
  fix : orgTreeDnaId
  feat:Change Round Multiple
2026-03-25 17:49:26 +07:00
1d9086fe7f
Merge pull request #1558 from Frappet/feat/change-round
feat:Change Round Multiple
2026-03-25 17:48:45 +07:00
bebf684069 update selected and load data 2026-03-25 17:48:24 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
faec83aacc Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m0s
2026-03-25 17:08:48 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
beef941dfd fix: insigniaForm.year 2026-03-25 17:08:34 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
b26be66f89 Merge branch 'dev' of https://forgejo.chamomind.com/hrms-bangkok/hrms-mgt into dev 2026-03-25 16:55:55 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
b9059c486e Merge branch 'develop' into dev 2026-03-25 16:55:44 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
5830d24118 refactor(ui): apply +543 year formatting to columns 2026-03-25 16:55:15 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
263e122b60 fix: payload multiple 2026-03-25 15:05:49 +07:00
069a221b29 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m24s
* develop:
  fix: style
  fix:tel
  fix:tel
  fix
  feat: add contact banner
2026-03-25 12:58:06 +07:00
30b63f1b52
Merge pull request #1561 from Frappet/feat/banner
feat: add contact banner
2026-03-25 12:55:56 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
c066ee577a fix: style 2026-03-25 12:10:53 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
59c188a3a9 fix:tel 2026-03-25 11:59:14 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
2b0d3340ee fix:tel 2026-03-25 11:54:34 +07:00
3afa3ce3e4 Merge branch 'dev' of https://forgejo.chamomind.com/hrms-bangkok/hrms-mgt into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 2m48s
* 'dev' of https://forgejo.chamomind.com/hrms-bangkok/hrms-mgt:
2026-03-25 11:07:19 +07:00
d6c57c5680 Merge branch 'develop' into dev
* develop:
  fix layout and select input
  fix: payload profileEmployeeId
  feat:absentlate
  fix(registry-officer): add field leaveSubTypeName  and coupleDayLevelCountry  form leave
  fix(registry): add reactive-rules rule prefix rank
2026-03-25 11:07:13 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
140553666d fix 2026-03-25 11:05:28 +07:00
d087fea744 Merge branch 'develop' into feat/change-round
* develop:
  fix layout and select input
  fix: payload profileEmployeeId
  feat:absentlate
2026-03-25 10:34:07 +07:00
31cb5827e1
Merge pull request #1560 from Frappet/feat/absentlate
Feat/absentlate
2026-03-25 10:28:02 +07:00
5f48063ae7 fix layout and select input 2026-03-25 10:27:45 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
e9c545a18d feat: add contact banner 2026-03-25 10:20:48 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
8359d1fbca fix: rowsPerPage 100 2026-03-25 09:32:02 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
c6ea0527c7 fix 2026-03-25 09:23:19 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
0a5f64c17c fix: payload profileEmployeeId 2026-03-25 09:18:51 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
6f495cca9d fix:add filter rowsPerPage = 3000 2026-03-24 17:35:55 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
b70d9948c0 fix : orgTreeDnaId 2026-03-24 17:30:10 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
ecc10e16c9 feat:absentlate 2026-03-24 16:50:26 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
f899f05527 feat:Change Round Multiple 2026-03-24 13:08:17 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
8d4f7bc077 Merge branch 'develop' into dev 2026-03-24 10:01:55 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
1d99705b65 fix(registry-officer): add field leaveSubTypeName and coupleDayLevelCountry form leave 2026-03-24 10:01:34 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
eeaa9bfff2 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m7s
2026-03-16 10:14:31 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
7471836167 Merge branch 'develop' of https://github.com/Frappet/bma-ehr-frontend into develop 2026-03-16 10:14:08 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
52cfa6e0af fix(registry): add reactive-rules rule prefix rank 2026-03-16 10:14:03 +07:00
3856eb8489 Merge branch 'dev' of https://forgejo.chamomind.com/hrms-bangkok/hrms-mgt into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 2m35s
* 'dev' of https://forgejo.chamomind.com/hrms-bangkok/hrms-mgt:
2026-03-13 16:19:10 +07:00
5ad9010654 Merge branch 'develop' into dev
* develop:
  fix ข้อความ
  fix(command): disable button delete
2026-03-13 16:19:04 +07:00
c7a784adc5 fix ข้อความ 2026-03-13 16:18:48 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
d7ea297e61 Merge branch 'develop' into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m0s
2026-03-13 14:02:01 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
b0b6eada67 Merge branch 'develop' of https://github.com/Frappet/bma-ehr-frontend into develop 2026-03-13 14:01:43 +07:00
DESKTOP-1R2VSQH\Lenovo ThinkPad E490
6ef832d84b fix(command): disable button delete 2026-03-13 14:01:33 +07:00
e2b50bdb22 Merge branch 'dev' of https://forgejo.chamomind.com/hrms-bangkok/hrms-mgt into dev
All checks were successful
Build & Deploy on Dev / build (push) Successful in 3m33s
* 'dev' of https://forgejo.chamomind.com/hrms-bangkok/hrms-mgt:
2026-03-12 20:36:01 +07:00
ae47c7cf34 Merge branch 'develop' into dev
* develop:
  แก้คำผิด
  fix(development - record): add button delete
  fix(registry-edit) : input autocomplete
  fix : registry edit
2026-03-12 20:35:55 +07:00
f1a2c753bb แก้คำผิด 2026-03-12 20:26:26 +07:00
122 changed files with 4071 additions and 930 deletions

View file

@ -30,6 +30,7 @@
"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",

View file

@ -192,6 +192,8 @@ 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}`,
workflowCommanderOperate: `${workflow}/commander/operate`, workflowCommanderOperate: `${workflow}/commander/operate`,
workflowCommanderSign: `${workflow}/commander/sign`, workflowCommanderSign: `${workflow}/commander/sign`,

View file

@ -27,6 +27,7 @@ export default {
placementDefermentInfo: (id: string) => `${placement}/pass/deferment/${id}`, placementDefermentInfo: (id: string) => `${placement}/pass/deferment/${id}`,
placementDisclaimInfo: (id: string) => `${placement}/pass/disclaim/${id}`, placementDisclaimInfo: (id: string) => `${placement}/pass/disclaim/${id}`,
placementUpdatePass: `${placement}/pass/update-status`, placementUpdatePass: `${placement}/pass/update-status`,
placementUpdateDraftStatus: `${placement}/update/draft-status`,
//personal //personal
placementPersonalId: (personalId: string) => placementPersonalId: (personalId: string) =>

View file

@ -61,4 +61,6 @@ export default {
leaveReportAPI: (type: string) => leaveReportAPI: (type: string) =>
`${leave}/report/download/time-records/${type}`, `${leave}/report/download/time-records/${type}`,
leaveTask: `${leave}/admin/leave-task/process`,
}; };

View file

@ -96,4 +96,8 @@ 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

@ -272,4 +272,10 @@ export default {
profileAssistanceReturn: `${env.API_URI}/placement/repatriation`, profileAssistanceReturn: `${env.API_URI}/placement/repatriation`,
profileAssistanceUpdateDelete: (type: string) => profileAssistanceUpdateDelete: (type: string) =>
`${registryNew}${type}/assistance/update-delete/`, `${registryNew}${type}/assistance/update-delete/`,
profileAbsentLate: (type: string) => `${registryNew}${type}/absent-late`,
profileAbsentLateUpdateDelete: (type: string) => `${registryNew}${type}/absent-late/update-delete`,
profileAbsentLateHistory: (id: string, type: string) =>
`${registryNew}${type}/absent-late/history/${id}`,
}; };

View file

@ -43,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.prefix ?? ""}${data.firstName ?? ""} ${ profile.fullName = `${data.rank ? data.rank : data.prefix ?? ""}${
data.lastName ?? "" data.firstName ?? ""
} `; } ${data.lastName ?? ""} `;
if (data["posTypeNameOld"] !== undefined) { if (data["posTypeNameOld"] !== undefined) {
profile.position = profile.position =

View file

@ -305,7 +305,7 @@ watch(
outlined outlined
option-label="name" option-label="name"
option-value="id" option-value="id"
@update:model-value="rows = []" @update:model-value="(rows = []), (selected = [])"
/> />
</div> </div>
</div> </div>
@ -332,6 +332,7 @@ 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

@ -11,6 +11,7 @@ import http from "@/plugins/http";
import config from "@/app.config"; import config from "@/app.config";
import DialogHeader from "@/components/DialogHeader.vue"; import DialogHeader from "@/components/DialogHeader.vue";
import FooterContact from "@/components/FooterContact.vue";
const $q = useQuasar(); const $q = useQuasar();
const store = usePositionKeycloakStore(); const store = usePositionKeycloakStore();
@ -353,6 +354,11 @@ 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
@ -362,10 +368,11 @@ function onClose() {
class="inputgreen" class="inputgreen"
hide-bottom-space hide-bottom-space
:rules="[ :rules="[
() => (val: string) => !!val || 'กรุณากรอกที่อยู่อีเมล',
!!formData.email || (val: string) => {
!!formData.phone || const emailPattern = /^[^\s@]+@[^\s@]+\.[^\s@]+$/;
'กรุณากรอกอีเมลหรือเบอร์โทรติดต่อกลับ', return emailPattern.test(val) || 'กรุณากรอกที่อยู่อีเมลในรูปแบบที่ถูกต้อง';
}
]" ]"
/> />
</div> </div>
@ -377,12 +384,6 @@ 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>
@ -391,7 +392,11 @@ function onClose() {
</q-card-section> </q-card-section>
<q-separator /> <q-separator />
<q-card-actions align="right"> <q-card-actions class="q-px-sm items-center">
<div class="row items-center q-pa-sm">
<FooterContact />
</div>
<q-space />
<q-btn <q-btn
type="submit" type="submit"
for="#submitForm" for="#submitForm"

View file

@ -24,14 +24,14 @@ import DialogHeader from "@/components/DialogHeader.vue";
const props = defineProps({ const props = defineProps({
dataRows: { dataRows: {
type: Object, type: Object,
require: true, default: () => ({}),
}, },
onSubmit: Function, onSubmit: Function,
}); });
const $q = useQuasar(); const $q = useQuasar();
const route = useRoute(); const route = useRoute();
const storeTree = useStructureTree(); const storeTree = useStructureTree();
const { fetchStructureTree } = useStructureTree(); const { fetchStructureTree } = storeTree;
const mixin = useCounterMixin(); const mixin = useCounterMixin();
const { const {
dialogConfirm, dialogConfirm,
@ -47,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<any>("type", { required: true }); const type = defineModel<string | null>("type", { required: true });
const posType = defineModel<any>("posType", { required: true }); const posType = defineModel<string | null>("posType", { required: true });
const posLevel = defineModel<any>("posLevel", { required: true }); const posLevel = defineModel<string | null>("posLevel", { required: true });
const position = defineModel<any>("position", { required: true }); const position = defineModel<string | null>("position", { required: true });
// const routeName = ref<string>(route?.name); // const routeName = ref<string>(route?.name);
const orgRevisionId = ref<string>(""); const orgRevisionId = ref<string>("");
@ -64,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 seletcId = ref<string>(""); const selectId = 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[]>([]);
@ -79,7 +79,6 @@ 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 +149,7 @@ const columns = ref<QTableProps["columns"]>([
style: "font-size: 14px", style: "font-size: 14px",
}, },
]); ]);
const columnsPostition = ref<QTableProps["columns"]>([ const columnsPosition = ref<QTableProps["columns"]>([
{ {
name: "no", name: "no",
align: "left", align: "left",
@ -181,7 +180,7 @@ const columnsPostition = 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,10 +243,16 @@ 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,
@ -261,7 +266,7 @@ async function getDataTable(id: string, level: number = 0) {
await http await http
.post(config.API.orgPosPlacement, body) .post(config.API.orgPosPlacement, body)
.then((res) => { .then(async (res) => {
const dataMain: PositionMain[] = []; const dataMain: PositionMain[] = [];
posMasterMain.value = res.data.result.data; posMasterMain.value = res.data.result.data;
@ -299,7 +304,6 @@ 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(
@ -313,14 +317,13 @@ 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(() => {
setTimeout(() => { hideLoader();
hideLoader();
}, 1000);
}); });
} }
@ -331,11 +334,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;
seletcId.value = props?.dataRows?.positionId; selectId.value = props?.dataRows?.positionId;
datePos.value = props?.dataRows?.reportingDate; datePos.value = props?.dataRows?.reportingDate;
} else { } else {
positionId.value = ""; positionId.value = "";
seletcId.value = ""; selectId.value = "";
selectedPos.value = []; selectedPos.value = [];
datePos.value = new Date(); datePos.value = new Date();
} }
@ -382,9 +385,9 @@ async function onClickSelectPos(id: string) {
// //
if (position) { if (position) {
rowsPosition.value = position.positions; rowsPosition.value = position.positions;
if (seletcId.value) { if (selectId.value) {
selectedPos.value = rowsPosition.value.filter( selectedPos.value = rowsPosition.value.filter(
(e) => e.id === seletcId.value (e) => e.id === selectId.value
); );
} }
} }
@ -403,18 +406,19 @@ async function fetchPosFind(level: number, id: string) {
}; };
await http await http
.post(config.API.orgPosFind, body) .post(config.API.orgPosFind, body)
.then((res) => { .then(async (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;
seletcId.value = props?.dataRows?.positionId; selectId.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();
}); });
} }
@ -427,11 +431,6 @@ 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 = [];
} }
@ -463,19 +462,15 @@ function fetchPositionUes() {
watch( watch(
() => isAll.value, () => isAll.value,
(value, oldVal) => { () => {
if (value !== oldVal) { getDataTable(nodeId.value, nodeLevel.value);
getDataTable(nodeId.value, nodeLevel.value);
}
} }
); );
watch( watch(
() => isBlank.value, () => isBlank.value,
(value, oldVal) => { () => {
if (value !== oldVal) { getDataTable(nodeId.value, nodeLevel.value);
getDataTable(nodeId.value, nodeLevel.value);
}
} }
); );
@ -491,35 +486,42 @@ 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, "กรุณาเลือกตำแหน่ง");
} else { return;
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, //
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() {
@ -534,6 +536,7 @@ 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">
@ -565,7 +568,7 @@ onMounted(async () => {
<q-tree <q-tree
class="q-pa-sm q-gutter-sm" class="q-pa-sm q-gutter-sm"
dense dense
:nodes="lazy" :nodes="nodes"
node-key="orgTreeId" node-key="orgTreeId"
label-key="labelName" label-key="labelName"
:filter="filter" :filter="filter"
@ -831,7 +834,7 @@ onMounted(async () => {
</q-toolbar> </q-toolbar>
<d-table <d-table
ref="table" ref="table"
:columns="columnsPostition" :columns="columnsPosition"
:rows="rowsPosition" :rows="rowsPosition"
row-key="id" row-key="id"
flat flat
@ -929,6 +932,7 @@ 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

@ -0,0 +1,17 @@
<template>
<div class="row items-center justify-center q-gutter-sm">
<q-icon name="support_agent" color="primary" size="sm" />
<span class="text-body2">
พบปญหาการใชงานกรณาตดตอผแลระบบ
<span class="text-weight-medium text-primary"
><a href="tel:0882649800" style="text-decoration: none; color: inherit"
>088-264-9800</a
>
</span>
</span>
</div>
</template>
<script setup lang="ts"></script>
<style scoped></style>

View file

@ -5,7 +5,6 @@ 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";
@ -187,6 +186,7 @@ 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">
{{ getColumnLabel(col, isAct) }} {{ col.label }}
</span> </span>
</q-th> </q-th>
</q-tr> </q-tr>

View file

@ -1004,7 +1004,6 @@ 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"

View file

@ -365,7 +365,7 @@ async function downloadFileDashboard() {
async function clickPassExam() { async function clickPassExam() {
showLoader(); showLoader();
await http await http
.get(config.API.exportExamPassExamList(examId.value)) .get(config.API.downloadPassExam(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.exportExamCandidateList(examId.value)) .get(config.API.downloadCandidateExam(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,8 +397,6 @@ async function clickCandidateList() {
} }
async function onCheckShowExaminfo() { async function onCheckShowExaminfo() {
console.log(props.isShowExaminfo);
dialogMessage( dialogMessage(
$q, $q,
`ยืนยันการ${props.isShowExaminfo ? "ปิด" : "เปิด"}`, `ยืนยันการ${props.isShowExaminfo ? "ปิด" : "เปิด"}`,

View file

@ -403,16 +403,6 @@ function clearFilter() {
</div> </div>
</div> </div>
</q-form> </q-form>
<div class="q-pa-sm q-gutter-sm">
<q-card flat bordered class="col-12">
<div class="row q-col-gutter-sm q-pa-sm">
<div class="row col-12 q-col-gutter-sm">
<q-space />
</div>
</div>
</q-card>
</div>
</template> </template>
<style scoped></style> <style scoped></style>

View file

@ -112,7 +112,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.year + 543}`; return `${row.year ? row.year + 543 : "-"}`;
}, },
sort: (a: number, b: number) => b - a, sort: (a: number, b: number) => b - a,
}, },

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(() => {
@ -400,7 +400,7 @@ async function checkSave() {
// validation form // validation form
const isPositionFormValid = await myFormPosition.value?.validate(); const isPositionFormValid = await myFormPosition.value?.validate();
if (!isPositionFormValid) return; if (!isPositionFormValid && announcementExam.value) return;
// //
if (announcementExam.value && rowsPosition.value.length === 0) { if (announcementExam.value && rowsPosition.value.length === 0) {
@ -889,7 +889,7 @@ function fetchPosition(level: number) {
* @param val าประเภทตำแหน 0 = ประเภททวไป ,1 = ประเภทวชาการ * @param val าประเภทตำแหน 0 = ประเภททวไป ,1 = ประเภทวชาการ
* @param index ตำแหนงของขอม * @param index ตำแหนงของขอม
*/ */
function onUpdateHighDegree(val: string, index: string) { function onUpdateHighDegree(val: string, index: number) {
rowsPosition.value[index].position = null; rowsPosition.value[index].position = null;
rowsPosition.value[index].level = rowsPosition.value[index].level =
val === "0" ? optionPosLevel1.value[0] : optionPosLevel2.value[0]; val === "0" ? optionPosLevel1.value[0] : optionPosLevel2.value[0];
@ -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,7 +260,6 @@ 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({
@ -402,6 +401,7 @@ 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,6 +603,7 @@ 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

@ -92,7 +92,7 @@ const baseColumns = ref<QTableColumn[]>([
field: "year", field: "year",
headerStyle: "font-size: 14px", headerStyle: "font-size: 14px",
style: "font-size: 14px", style: "font-size: 14px",
format: (v) => v + 543, format: (v) => (v ? v + 543 : "-"),
sort: (a: string, b: string) => sort: (a: string, b: string) =>
a a
.toString() .toString()
@ -920,9 +920,11 @@ onMounted(async () => {
:locale="'th'" :locale="'th'"
:enableTimePicker="false" :enableTimePicker="false"
> >
<template #year="{ year }">{{ year + 543 }}</template> <template #year="{ year }">{{
year ? year + 543 : "-"
}}</template>
<template #year-overlay-value="{ value }">{{ <template #year-overlay-value="{ value }">{{
parseInt(value + 543) value ? parseInt(value + 543) : "-"
}}</template> }}</template>
<template #trigger> <template #trigger>
<q-input <q-input
@ -930,7 +932,7 @@ onMounted(async () => {
outlined outlined
hide-bottom-space hide-bottom-space
class="inputgreen" class="inputgreen"
:model-value="insigniaForm.year !== 0 ? (insigniaForm.year as number) + 543 : null" :model-value="insigniaForm.year != null && insigniaForm.year !== 0 ? (insigniaForm.year as number) + 543 : null"
:rules="[ :rules="[
(val:string) => (val:string) =>
!!val || !!val ||

View file

@ -50,18 +50,10 @@ 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
@ -165,10 +157,22 @@ 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);
@ -455,25 +459,19 @@ 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>{{ <span>{{ formMain.ageAll }}</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> <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>{{ <span>{{ formMain.govAgeBkk }}</span>
formMain.govAgeBkk
? `${formMain.govAgeBkk.year} ปี ${formMain.govAgeBkk.month} เดือน ${formMain.govAgeBkk.day} วัน`
: "-"
}}</span>
</div> </div>
</div> </div>
</div> </div>

View file

@ -202,6 +202,17 @@ const pagination = ref({
const columnsHistory = ref<QTableColumn[]>(baseColumns.value); const columnsHistory = ref<QTableColumn[]>(baseColumns.value);
const visibleColumnsHistory = ref<string[]>(baseVisibleColumns.value); const visibleColumnsHistory = ref<string[]>(baseVisibleColumns.value);
/** รายการประเภทการลาของ ลาไปศึกษา ฝึกอบรม ปฎิบัติการวิจัย หรือดูงาน*/
const leaveSubTypeName = ref<string>("");
const optionSubTypeName = ref<string[]>([
"ศึกษาต่อ",
"ฝึกอบรม",
"ปฎิบัติการวิจัย",
"ดูงาน",
]);
const coupleDayLevelCountry = ref<string>("");
/** function fetch ข้อมูลรายการลา*/ /** function fetch ข้อมูลรายการลา*/
async function getData() { async function getData() {
showLoader(); showLoader();
@ -213,6 +224,7 @@ async function getData() {
...item, ...item,
id: item.id, id: item.id,
typeLeave: item.leaveType.name, typeLeave: item.leaveType.name,
codeLeave: item.leaveType.code,
code: item.leaveType.refCommandDate, code: item.leaveType.refCommandDate,
dateStartLeave: item.dateLeaveStart, dateStartLeave: item.dateLeaveStart,
dateEndLeave: item.dateLeaveEnd, dateEndLeave: item.dateLeaveEnd,
@ -292,8 +304,10 @@ function openDialogEdit(props: DetailData) {
typeLeave.value = { typeLeave.value = {
id: props.typeLeaveId, id: props.typeLeaveId,
name: props.typeLeave, name: props.typeLeave,
code: props.code, code: props.codeLeave,
}; };
leaveSubTypeName.value = props.leaveSubTypeName;
coupleDayLevelCountry.value = props.coupleDayLevelCountry;
statLeave.value = props.status; statLeave.value = props.status;
reason.value = props.reason; reason.value = props.reason;
dateRange.value = [ dateRange.value = [
@ -316,6 +330,10 @@ function onSubmit() {
const body = { const body = {
leaveTypeId: typeLeave.value?.id, leaveTypeId: typeLeave.value?.id,
leaveSubTypeName:
typeLeave.value?.code === "LV-008" ? leaveSubTypeName.value : undefined,
coupleDayLevelCountry:
typeLeave.value?.code === "LV-010" ? coupleDayLevelCountry.value : undefined,
dateLeaveStart: dateToISO(dateRange.value[0]), dateLeaveStart: dateToISO(dateRange.value[0]),
dateLeaveEnd: dateToISO(dateRange.value[1]), dateLeaveEnd: dateToISO(dateRange.value[1]),
leaveDays: numLeave.value, leaveDays: numLeave.value,
@ -451,6 +469,8 @@ function closeDialog() {
dateRange.value = [new Date(), new Date()]; dateRange.value = [new Date(), new Date()];
numLeave.value = 1; numLeave.value = 1;
numUsedLeave.value = 0; numUsedLeave.value = 0;
leaveSubTypeName.value = "";
coupleDayLevelCountry.value = "";
} }
function statusLeave(val: string) { function statusLeave(val: string) {
@ -741,6 +761,43 @@ onMounted(() => {
) " ) "
/> />
</div> </div>
<div
class="col-xs-6 col-sm-6 col-md-6"
v-if="
typeLeave?.code === 'LV-008' || typeLeave?.code === 'LV-010'
"
>
<q-select
v-if="typeLeave?.code === 'LV-008'"
ref="typeLeaveRef"
class="full-width inputgreen cursor-pointer"
outlined
dense
lazy-rules
:label="`${'ประเภท'}`"
:rules="[(val:string) => !!val || `${'กรุณาเลือกประเภท'}`]"
v-model="leaveSubTypeName"
:options="optionSubTypeName"
option-value="id"
option-label="name"
emit-value
map-options
hide-bottom-space
/>
<q-input
v-if="typeLeave?.code === 'LV-010'"
ref="numLeaveRef"
class="full-width inputgreen cursor-pointer"
outlined
dense
lazy-rules
v-model="coupleDayLevelCountry"
:rules="[(val:string) => !!val || `${'กรุณากรอกประเทศที่ลาติดตามคู่สมรส'}`]"
hide-bottom-space
:label="`${'ประเทศที่ลาติดตามคู่สมรส'}`"
/>
</div>
<div class="col-xs-6 col-sm-6 col-md-6"> <div class="col-xs-6 col-sm-6 col-md-6">
<datepicker <datepicker
:readonly="!typeLeave" :readonly="!typeLeave"

View file

@ -402,6 +402,7 @@ const modalDialogSalary = ref<boolean>(false); //แสดง popup ตำแห
const isStatusEdit = ref<boolean>(false); // const isStatusEdit = ref<boolean>(false); //
const salaryId = ref<string>(""); //id const salaryId = ref<string>(""); //id
const dataLevel = ref<DataPosType[]>([]); // const dataLevel = ref<DataPosType[]>([]); //
const idCommandId = ref<boolean>(false); // commandId
const commandCodeOptions = ref<DataOption[]>(store.commandCodeData); // const commandCodeOptions = ref<DataOption[]>(store.commandCodeData); //
const posTypeOptions = ref<DataOption[]>(store.posTypeData); // | const posTypeOptions = ref<DataOption[]>(store.posTypeData); // |
@ -658,6 +659,7 @@ async function onClickOpenDialog(
} else { } else {
await fetchOptionGroup(); await fetchOptionGroup();
} }
idCommandId.value = statusEdit ? (data.commandId ? true : false) : false;
commandCodeOptions.value = store.commandCodeData; commandCodeOptions.value = store.commandCodeData;
posTypeOptions.value = store.posTypeData; posTypeOptions.value = store.posTypeData;
posLevelOptions.value = store.posLevelData; posLevelOptions.value = store.posLevelData;
@ -978,6 +980,7 @@ onMounted(async () => {
<q-tooltip>ประวแกไขตำแหน/เงนเดอน</q-tooltip> <q-tooltip>ประวแกไขตำแหน/เงนเดอน</q-tooltip>
</q-btn> </q-btn>
<!-- :disable="(props.row.commandId !== null && props.row.commandId !== '') || props.row.commandType === 'C-PM-47'" -->
<q-btn <q-btn
v-if=" v-if="
!isLeave && !isLeave &&
@ -985,11 +988,7 @@ onMounted(async () => {
checkPermission($route)?.attrOwnership === 'OWNER' checkPermission($route)?.attrOwnership === 'OWNER'
" "
flat flat
:disable=" color="edit"
(props.row.commandId !== null && props.row.commandId !== '') ||
props.row.commandType === 'C-PM-47'
"
:color="props.row.commandId ? 'grey' : 'edit'"
dense dense
round round
icon="edit" icon="edit"
@ -1154,7 +1153,8 @@ onMounted(async () => {
<div class="row q-col-gutter-sm"> <div class="row q-col-gutter-sm">
<div class="col-6"> <div class="col-6">
<q-input <q-input
:class="classInput(true)" :class="classInput(!idCommandId)"
:readonly="idCommandId"
outlined outlined
dense dense
lazy-rules lazy-rules
@ -1174,7 +1174,8 @@ onMounted(async () => {
autoApply autoApply
year-picker year-picker
:enableTimePicker="false" :enableTimePicker="false"
class="inputgreen" :class="classInput(!idCommandId)"
:disabled="idCommandId"
> >
<template #year="{ year }">{{ year + 543 }}</template> <template #year="{ year }">{{ year + 543 }}</template>
<template #year-overlay-value="{ value }">{{ <template #year-overlay-value="{ value }">{{
@ -1191,6 +1192,8 @@ onMounted(async () => {
: formData.commandYear + 543 : formData.commandYear + 543
" "
label="ปี พ.ศ." label="ปี พ.ศ."
:class="classInput(!idCommandId)"
:readonly="idCommandId"
> >
<template v-slot:prepend> <template v-slot:prepend>
<q-icon <q-icon

View file

@ -0,0 +1,424 @@
<script setup lang="ts">
import { ref, onMounted, reactive } from "vue";
import { useQuasar } from "quasar";
import { useRoute } from "vue-router";
import { checkPermission } from "@/utils/permissions";
import { useCounterMixin } from "@/stores/mixin";
import { useAbsentLateStore } from "@/modules/04_registryPerson/stores/AbsentLate";
import http from "@/plugins/http";
import config from "@/app.config";
import type { QTableColumn } from "quasar";
import type { ResAbsentLateData} from "@/modules/04_registryPerson/interface/response/Government";
import DialogAbsentLate from "@/modules/04_registryPerson/components/detail/GovernmentInformation/08_DialogAbsentLate.vue";
import DialogHistory from "@/modules/04_registryPerson/components/detail/DialogHistory.vue";
const route = useRoute();
const absentLateStore = useAbsentLateStore();
const $q = useQuasar();
const {
date2Thai,
dialogConfirm,
showLoader,
hideLoader,
messageError,
success,
pathRegistryEmp,
onSearchDataTable,
convertDateToAPI,
dialogRemove,
} = useCounterMixin();
const profileId = ref<string>(
route.params.id ? route.params.id.toString() : ""
);
const empType = ref<string>(pathRegistryEmp(route.name?.toString() ?? ""));
const isLeave = defineModel<boolean>("isLeave", {
required: true,
});
const baseColumns = ref<QTableColumn[]>([
{
name: "status",
align: "left",
label: "มาสาย/ ขาดราชการ",
sortable: true,
field: "status",
headerStyle: "font-size: 14px",
style: "font-size: 14px",
format(val, row) {
const status = absentLateStore.statusOps.find(
(option) => option.id === val
);
return status ? status.name : val;
},
},
{
name: "stampDate",
align: "left",
label: "วันที่ลงเวลา",
sortable: true,
field: "stampDate",
format: (v) => date2Thai(v),
headerStyle: "font-size: 14px",
style: "font-size: 14px",
},
{
name: "stampType",
align: "left",
label: "ประเภทการลงเวลา",
sortable: true,
field: "stampType",
headerStyle: "font-size: 14px",
style: "font-size: 14px",
format(val) {
const type = absentLateStore.stampTypeOps.find(
(option) => option.id === val
);
return type ? type.name : val;
},
},
{
name: "stampAmount",
align: "left",
label: "จำนวน",
sortable: true,
field: "stampAmount",
headerStyle: "font-size: 14px",
style: "font-size: 14px",
},
{
name: "remark",
align: "left",
label: "หมายเหตุ",
sortable: true,
field: "remark",
headerStyle: "font-size: 14px",
style: "font-size: 14px",
},
]);
const baseVisibleColumns = ref<string[]>([
"status",
"stampDate",
"stampType",
"stampAmount",
"remark",
]);
/** Table*/
const rows = ref<ResAbsentLateData[]>([]);
const rowsMain = ref<ResAbsentLateData[]>([]);
const mode = ref<string>("table"); // Table card
const filterKeyword = ref<string>(""); //
const columns = ref<QTableColumn[]>(
baseColumns.value.filter((e: QTableColumn) => e.name !== "lastUpdateFullName")
);
const visibleColumns = ref<string[]>(
baseVisibleColumns.value.filter((e: string) => e !== "lastUpdateFullName")
);
const pagination = ref({
sortBy: "lastUpdatedAt",
});
const columnsHistory = ref<QTableColumn[]>(baseColumns.value);
const visibleColumnsHistory = ref<string[]>(baseVisibleColumns.value);
/** Dialog*/
const isStatusEdit = ref<boolean>(false);
const modal = ref<boolean>(false);
const modalHistory = ref<boolean>(false);
const rowId = ref<string>("");
const dataAbsentLate = ref<ResAbsentLateData | null>(null);
async function fetchData() {
showLoader();
try {
const res = await http.get(
config.API.profileAbsentLate(empType.value) + `/${profileId.value}`
);
const data = res.data.result;
rowsMain.value = data;
serchDataTable();
} catch (err) {
messageError($q, err);
} finally {
hideLoader();
}
}
/** function fetch ข้อมูลประวัติการแก้ไขข้อมูล*/
async function fetchDataHistory() {
showLoader();
try {
const res = await http.get(
config.API.profileAbsentLateHistory(rowId.value, empType.value)
);
return res.data.result;
} catch (err) {
messageError($q, err);
} finally {
hideLoader();
}
}
function openEditDialog(data: any) {
modal.value = true;
isStatusEdit.value = true;
rowId.value = data.id;
dataAbsentLate.value = data;
}
function showHistoryDialog(id: string) {
modalHistory.value = true;
rowId.value = id;
}
/** ฟังก์ค้นหาข้อมูลขาดราชการ/มาสาย*/
function serchDataTable() {
rows.value = onSearchDataTable(
filterKeyword.value,
rowsMain.value,
columns.value ? columns.value : []
);
}
function onDelete(rowId: string) {
dialogRemove($q, async () => {
showLoader();
try {
await http.patch(
config.API.profileAbsentLateUpdateDelete(empType.value) + `/${rowId}`
);
await fetchData();
await success($q, "ลบข้อมูลสำเร็จ");
} catch (err) {
messageError($q, err);
} finally {
hideLoader();
}
});
}
onMounted(() => {
fetchData();
});
</script>
<template>
<div class="row items-center q-gutter-x-sm q-pb-sm">
<q-btn
v-if="!isLeave && checkPermission($route)?.attrIsUpdate"
dense
color="primary"
icon="add"
flat
round
@click.stop.prevent="(modal = true), (isStatusEdit = false)"
>
<q-tooltip>เพมขอม</q-tooltip>
</q-btn>
<q-space />
<q-input
standout
dense
v-model="filterKeyword"
ref="filterRef"
outlined
placeholder="ค้นหา"
@keydown.enter.prevent="serchDataTable"
>
<template v-slot:append>
<q-icon name="search" />
</template>
</q-input>
<q-select
v-model="visibleColumns"
multiple
outlined
dense
options-dense
:display-value="$q.lang.table.columns"
emit-value
map-options
:options="columns"
option-value="name"
style="min-width: 140px"
/>
<q-btn-toggle
v-model="mode"
dense
class="no-shadow toggle-borderd"
toggle-color="grey-4"
:options="[
{ value: 'table', slot: 'table' },
{ value: 'card', slot: 'card' },
]"
>
<template v-slot:table>
<q-icon
name="format_list_bulleted"
size="24px"
:style="{
color: mode === 'table' ? '#787B7C' : '#C9D3DB',
}"
/>
</template>
<template v-slot:card>
<q-icon
name="mdi-view-grid-outline"
size="24px"
:style="{
color: mode === 'card' ? '#787B7C' : '#C9D3DB',
}"
/>
</template>
</q-btn-toggle>
</div>
<d-table
:grid="mode === 'card'"
ref="table"
row-key="id"
flat
bordered
dense
:columns="columns"
:rows="rows"
:visible-columns="visibleColumns"
v-model:pagination="pagination"
>
>
<template v-slot:header="props">
<q-tr :props="props">
<q-th auto-width></q-th>
<q-th v-for="col in props.cols" :key="col.name" :props="props">
<span class="text-weight-medium">{{ col.label }}</span>
</q-th>
</q-tr>
</template>
<template v-slot:body="props" v-if="mode === 'table'">
<q-tr :props="props">
<q-td auto-width>
<q-btn
flat
dense
round
color="deep-purple"
icon="mdi-history"
@click.stop.prevent="showHistoryDialog(props.row.id)"
>
<q-tooltip>ประวแกไขขาดราชการ/มาสาย</q-tooltip>
</q-btn>
<q-btn
v-if="!isLeave && checkPermission($route)?.attrIsUpdate"
flat
dense
round
color="edit"
icon="edit"
@click.stop.prevent="openEditDialog(props.row)"
>
<q-tooltip>แกไขขอม</q-tooltip>
</q-btn>
<q-btn
v-if="!isLeave && checkPermission($route)?.attrIsDelete"
flat
dense
round
color="red"
icon="delete"
@click.stop.prevent="onDelete(props.row.id)"
>
<q-tooltip>ลบขอม</q-tooltip>
</q-btn>
</q-td>
<q-td v-for="col in props.cols" :key="col.id">
<div>
{{ col.value ? col.value : "-" }}
</div>
</q-td>
</q-tr>
</template>
<template v-slot:item="props" v-else>
<div class="q-pa-xs col-xs-12 col-sm-4 col-md-3">
<q-card flat bordered>
<q-card-actions align="right" class="bg-grey-3">
<q-btn
color="deep-purple"
icon="mdi-history"
flat
round
@click.stop.prevent="showHistoryDialog(props.row.id)"
>
<q-tooltip>ประวแกไขขาดราชการ/มาสาย</q-tooltip>
</q-btn>
<q-btn
v-if="isLeave === false && checkPermission($route)?.attrIsUpdate"
:color="props.row.commandId ? 'grey-5' : 'edit'"
:disable="props.row.commandId !== null"
icon="edit"
flat
round
@click.stop.prevent="openEditDialog(props.row)"
>
<q-tooltip>แกไขขอม</q-tooltip>
</q-btn>
<q-btn
v-if="isLeave === false && checkPermission($route)?.attrIsDelete"
color="red"
icon="delete"
flat
round
@click.stop.prevent="onDelete(props.row.id)"
>
<q-tooltip>ลบขอม</q-tooltip>
</q-btn>
</q-card-actions>
<q-separator />
<q-list>
<div
:class="`row q-pa-sm`"
:style="`background-color: ${index % 2 !== 0 ? '#FAFAFA' : ''}`"
v-for="(col, index) in props.cols"
:key="col.name"
>
<div class="col text-grey-6">
<div>{{ col.label }}</div>
</div>
<div class="col">
<div>{{ col.value ? col.value : "-" }}</div>
</div>
</div>
</q-list>
</q-card>
</div>
</template>
</d-table>
<DialogAbsentLate
v-model:modal="modal"
v-model:isStatusEdit="isStatusEdit"
:fetchData="fetchData"
:rowId="rowId"
:dataAbsentLate="dataAbsentLate"
/>
<DialogHistory
v-model:modal="modalHistory"
:title="`ประวัติแก้ไขขาดราชการ/มาสาย`"
:columns="columnsHistory"
:visible-columns="visibleColumnsHistory"
:fetch-data="fetchDataHistory"
/>
</template>
<style scoped></style>

View file

@ -0,0 +1,263 @@
<script setup lang="ts">
import { reactive, ref, computed, watch } from "vue";
import { useQuasar } from "quasar";
import { useRoute } from "vue-router";
import { useCounterMixin } from "@/stores/mixin";
import { useAbsentLateStore } from "@/modules/04_registryPerson/stores/AbsentLate";
import http from "@/plugins/http";
import config from "@/app.config";
/** import components*/
import DialogHeader from "@/components/DialogHeader.vue";
//use
const $q = useQuasar();
const route = useRoute();
const absentLateStore = useAbsentLateStore();
const {
showLoader,
hideLoader,
messageError,
success,
date2Thai,
convertDateToAPI,
dialogConfirm,
pathRegistryEmp,
} = useCounterMixin();
//props
const modal = defineModel<boolean>("modal", { required: true });
const isStatusEdit = defineModel<boolean>("isStatusEdit", { required: true });
const rowId = defineModel<string>("rowId", { required: true });
const props = defineProps<{
fetchData: () => Promise<void>;
dataAbsentLate: any;
}>();
const profileId = ref<string>(route.params.id?.toString() ?? ""); //ProfileId
const empType = ref<string>(pathRegistryEmp(route.name?.toString() ?? ""));
const form = reactive({
status: "",
stampDate: new Date(),
stampType: "FULL_DAY",
stampAmount: "1.0",
remark: "",
});
const tittle = computed(() => {
return isStatusEdit.value
? "แก้ไขข้อมูลขาดราชการ/มาสาย"
: "เพิ่มข้อมูลขาดราชการ/มาสาย";
});
function onSubmit() {
dialogConfirm($q, async () => {
try {
showLoader();
const payload = {
...form,
stampDate: convertDateToAPI(form.stampDate),
profileId:
!isStatusEdit.value && empType.value === ""
? profileId.value
: undefined,
profileEmployeeId:
!isStatusEdit.value && empType.value !== ""
? profileId.value
: undefined,
};
const method = isStatusEdit.value ? "patch" : "post";
const url = isStatusEdit.value
? config.API.profileAbsentLate(empType.value) + `/${rowId.value}`
: config.API.profileAbsentLate(empType.value);
await http[method](url, payload);
success($q, "บันทึกข้อมูลสำเร็จ");
props.fetchData();
closeDialog();
} catch (error) {
messageError($q, error);
} finally {
hideLoader();
}
});
}
/** function ปิด popup*/
function closeDialog() {
modal.value = false;
}
watch(
() => form.stampType,
(stampType) => {
if (stampType === "FULL_DAY") {
form.stampAmount = "1.0";
} else if (stampType === "MORNING" || stampType === "AFTERNOON") {
form.stampAmount = "0.5";
}
}
);
watch(
() => modal.value,
(newVal) => {
if (newVal) {
form.status = isStatusEdit.value ? props.dataAbsentLate.status : "";
form.stampDate = isStatusEdit.value
? new Date(props.dataAbsentLate.stampDate)
: new Date();
form.stampType = isStatusEdit.value
? props.dataAbsentLate.stampType
: "FULL_DAY";
form.stampAmount = isStatusEdit.value
? props.dataAbsentLate.stampAmount
: "1.0";
form.remark = isStatusEdit.value ? props.dataAbsentLate.remark : "";
}
}
);
</script>
<template>
<q-dialog v-model="modal" persistent>
<q-card style="min-width: 50vw">
<q-form greedy @submit.prevent="onSubmit">
<DialogHeader :tittle="tittle" :close="closeDialog" />
<q-separator />
<q-card-section class="q-pa-md">
<div class="col-12 row q-col-gutter-sm">
<div class="col-12 row">
<div class="col-md-4 col-sm-12">
<q-select
class="full-width inputgreen cursor-pointer"
outlined
dense
lazy-rules
v-model="form.status"
:rules="[(val:string) => !!val || `${'กรุณาเลือกสถานะ'}`]"
hide-bottom-space
:label="`${'สถานะ'}`"
map-options
emit-value
option-label="name"
option-value="id"
:options="absentLateStore.statusOps"
use-input
hide-selected
fill-input
input-debounce="0"
/>
</div>
</div>
<div class="col-md-4 col-sm-12">
<datepicker
class="inputgreen"
menu-class-name="modalfix"
v-model="form.stampDate"
:locale="'th'"
autoApply
borderless
:enableTimePicker="false"
week-start="0"
>
<template #year="{ year }">
{{ year + 543 }}
</template>
<template #year-overlay-value="{ value }">
{{ parseInt(value + 543) }}
</template>
<template #trigger>
<q-input
outlined
dense
hide-bottom-space
class="full-width"
:model-value="
form.stampDate != null ? date2Thai(form.stampDate) : null
"
:label="`${'วันที่ลงเวลา'}`"
:rules="[
(val: string) =>
!!val || `${'กรุณาเลือกวันที่ลงเวลา'}`,
]"
>
<template v-slot:prepend>
<q-icon
name="event"
class="cursor-pointer"
style="color: var(--q-primary)"
>
</q-icon>
</template>
</q-input>
</template>
</datepicker>
</div>
<div class="col-md-4 col-xs-12">
<q-select
class="full-width inputgreen cursor-pointer"
outlined
dense
lazy-rules
v-model="form.stampType"
:rules="[(val:string) => !!val || `${'กรุณาเลือกประเภทการลงเวลา'}`]"
hide-bottom-space
:label="`${'ประเภทการลงเวลา'}`"
map-options
emit-value
option-label="name"
:options="absentLateStore.stampTypeOps"
option-value="id"
use-input
hide-selected
fill-input
input-debounce="0"
/>
</div>
<div class="col-md-4 col-xs-12">
<q-input
class="inputgreen"
dense
outlined
v-model="form.stampAmount"
label="จำนวน"
hide-bottom-space
readonly
:rules="[(val:string) => !!val ||
`${'กรุณากรอกจำนวน'}`]"
mask="#.#"
/>
</div>
<div class="col-12">
<q-input
class="inputgreen"
dense
outlined
v-model="form.remark"
label="เหตุผล"
hide-bottom-space
type="textarea"
/>
</div>
</div>
</q-card-section>
<q-separator />
<q-card-actions align="right">
<q-btn type="submit" :label="`บันทึก`" color="public">
<q-tooltip>นท</q-tooltip>
</q-btn>
</q-card-actions>
</q-form>
</q-card>
</q-dialog>
</template>
<style scoped></style>

View file

@ -14,6 +14,7 @@ import PerformSpecialWork from "@/modules/04_registryPerson/components/detail/Go
import ActingPos from "@/modules/04_registryPerson/components/detail/GovernmentInformation/05_ActingPos.vue"; // import ActingPos from "@/modules/04_registryPerson/components/detail/GovernmentInformation/05_ActingPos.vue"; //
import HelpGovernmentDetail from "@/modules/04_registryPerson/components/detail/GovernmentInformation/06_HelpGovernment.vue"; // import HelpGovernmentDetail from "@/modules/04_registryPerson/components/detail/GovernmentInformation/06_HelpGovernment.vue"; //
import Postion from "@/modules/04_registryPerson/components/detail/GovernmentInformation/07_Position.vue"; import Postion from "@/modules/04_registryPerson/components/detail/GovernmentInformation/07_Position.vue";
import AbsentLate from "@/modules/04_registryPerson/components/detail/GovernmentInformation/08_AbsentLate.vue";
import { useRegistryNewDataStore } from "@/modules/04_registryPerson/store"; import { useRegistryNewDataStore } from "@/modules/04_registryPerson/store";
const empType = ref<string>(pathRegistryEmp(route.name?.toString() ?? "")); const empType = ref<string>(pathRegistryEmp(route.name?.toString() ?? ""));
@ -47,6 +48,7 @@ const storeRegistry = useRegistryNewDataStore();
<q-tab v-if="empType != '-employee'" name="6" label="ช่วยราชการ" /> <q-tab v-if="empType != '-employee'" name="6" label="ช่วยราชการ" />
<q-tab name="2" label="วินัย" /> <q-tab name="2" label="วินัย" />
<q-tab name="3" label="การลา" /> <q-tab name="3" label="การลา" />
<q-tab name="8" label="ขาดราชการ/มาสาย" />
<q-tab name="4" label="ปฏิบัติราชการพิเศษ" /> <q-tab name="4" label="ปฏิบัติราชการพิเศษ" />
</q-tabs> </q-tabs>
<q-separator /> <q-separator />
@ -76,6 +78,9 @@ const storeRegistry = useRegistryNewDataStore();
:citizen-id="storeRegistry.citizenId" :citizen-id="storeRegistry.citizenId"
/> />
</q-tab-panel> </q-tab-panel>
<q-tab-panel name="8">
<AbsentLate :is-leave="storeRegistry.isLeave" />
</q-tab-panel>
<q-tab-panel name="7"> <q-tab-panel name="7">
<Postion <Postion
:is-leave="storeRegistry.isLeave" :is-leave="storeRegistry.isLeave"

View file

@ -406,6 +406,12 @@ function calculateMinDate() {
return today; return today;
} }
function prefixRankRule() {
return [
() => !!formData.rank || !!formData.prefix || "กรุณาเลือกคำนำหน้าชื่อ หรือยศ",
];
}
/** ดูการเปลี่ยนแปลงของวันเกิดเมื่อมีการเปลี่ยนแปลงจะคำนวนอายูใหม่*/ /** ดูการเปลี่ยนแปลงของวันเกิดเมื่อมีการเปลี่ยนแปลงจะคำนวนอายูใหม่*/
watch( watch(
() => formData.birthDate, () => formData.birthDate,
@ -599,7 +605,8 @@ onMounted(() => {
class="inputgreen" class="inputgreen"
:options="store.Ops.prefixOps" :options="store.Ops.prefixOps"
:label="dataLabel.prefix" :label="dataLabel.prefix"
:rules="[(val: string) => !!formData.rank || !!formData.prefix || `${'กรุณาเลือกคำนำหน้าชื่อ หรือยศ'}`]" :rules="prefixRankRule()"
reactive-rules
@filter="(inputValue: string, @filter="(inputValue: string,
doneFn: Function) => filterSelector(inputValue, doneFn, 'prefixOps' doneFn: Function) => filterSelector(inputValue, doneFn, 'prefixOps'
)" )"
@ -620,7 +627,8 @@ onMounted(() => {
input-debounce="0" input-debounce="0"
option-label="name" option-label="name"
option-value="name" option-value="name"
:rules="[(val: string) => !!formData.rank || !!formData.prefix || `${'กรุณาเลือกคำนำหน้าชื่อ หรือยศ'}`]" :rules="prefixRankRule()"
reactive-rules
v-model="formData.rank" v-model="formData.rank"
class="inputgreen" class="inputgreen"
:options="store.Ops.rankOps" :options="store.Ops.rankOps"

View file

@ -66,6 +66,7 @@ const keyword = ref<string>(""); //คำค้นหา
const modalCommand = ref<boolean>(false); const modalCommand = ref<boolean>(false);
const command = ref<string>(""); const command = ref<string>("");
const commandId = ref<string>(""); const commandId = ref<string>("");
const idCommandId = ref<boolean>(false); //
const baseColumns = ref<QTableColumn[]>([ const baseColumns = ref<QTableColumn[]>([
{ {
name: "commandDateAffect", name: "commandDateAffect",
@ -638,6 +639,7 @@ async function onClickOpenDialog(
} else { } else {
await fetchOptionGroup(); await fetchOptionGroup();
} }
idCommandId.value = statusEdit && data.commandId ? true : false;
commandCodeOptions.value = store.commandCodeData; commandCodeOptions.value = store.commandCodeData;
posTypeOptions.value = store.posTypeData; posTypeOptions.value = store.posTypeData;
posLevelOptions.value = store.posLevelData; posLevelOptions.value = store.posLevelData;
@ -939,6 +941,10 @@ onMounted(async () => {
> >
<q-tooltip>ประวแกไขตำแหน/เงนเดอน</q-tooltip> <q-tooltip>ประวแกไขตำแหน/เงนเดอน</q-tooltip>
</q-btn> </q-btn>
<!-- :disable="
(props.row.commandId !== null && props.row.commandId !== '') ||
props.row.commandType === 'C-PM-47'
" -->
<q-btn <q-btn
v-if=" v-if="
!isLeave && !isLeave &&
@ -946,11 +952,7 @@ onMounted(async () => {
checkPermission($route)?.attrOwnership === 'OWNER' checkPermission($route)?.attrOwnership === 'OWNER'
" "
flat flat
:disable=" color="edit"
(props.row.commandId !== null && props.row.commandId !== '') ||
props.row.commandType === 'C-PM-47'
"
:color="props.row.commandId ? 'grey' : 'edit'"
dense dense
round round
icon="edit" icon="edit"
@ -1119,7 +1121,8 @@ onMounted(async () => {
<div class="row q-col-gutter-sm"> <div class="row q-col-gutter-sm">
<div class="col-6"> <div class="col-6">
<q-input <q-input
:class="classInput(true)" :class="classInput(!idCommandId)"
:readonly="idCommandId"
outlined outlined
dense dense
lazy-rules lazy-rules
@ -1139,7 +1142,8 @@ onMounted(async () => {
autoApply autoApply
year-picker year-picker
:enableTimePicker="false" :enableTimePicker="false"
class="inputgreen" :class="classInput(!idCommandId)"
:disable="idCommandId"
> >
<template #year="{ year }">{{ year + 543 }}</template> <template #year="{ year }">{{ year + 543 }}</template>
<template #year-overlay-value="{ value }">{{ <template #year-overlay-value="{ value }">{{
@ -1156,6 +1160,8 @@ onMounted(async () => {
: formData.commandYear + 543 : formData.commandYear + 543
" "
label="ปี พ.ศ." label="ปี พ.ศ."
:class="classInput(!idCommandId)"
:readonly="idCommandId"
> >
<template v-slot:prepend> <template v-slot:prepend>
<q-icon <q-icon

View file

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

View file

@ -8,6 +8,9 @@ interface DetailData {
reason: string; reason: string;
typeLeaveId: string; typeLeaveId: string;
code: string; code: string;
codeLeave:string
leaveSubTypeName:string
coupleDayLevelCountry:string
} }
interface FormFilter { interface FormFilter {

View file

@ -70,6 +70,7 @@ 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

@ -44,4 +44,21 @@ interface ResFileData {
pathname: string; pathname: string;
} }
export type { ResActingPosData, ResAssistanceData, ResFileData }; interface ResAbsentLateData {
createdAt: string;
createdFullName: string;
createdUserId: string;
id: string;
isDeleted: boolean;
lastUpdateFullName: string;
lastUpdateUserId: string;
lastUpdatedAt: string;
profileId: string;
remark: string;
stampAmount: string;
stampDate: string;
stampType: string;
status: string;
}
export type { ResActingPosData, ResAssistanceData, ResFileData, ResAbsentLateData };

View file

@ -0,0 +1,19 @@
import { ref } from "vue";
import { defineStore } from "pinia";
import type { DataOption } from "@/modules/04_registryPerson/interface/index/Main";
export const useAbsentLateStore = defineStore("absentLate", () => {
const statusOps = ref<DataOption[]>([
{ name: "ขาดราชการ", id: "ABSENT" },
{ name: "มาสาย", id: "LATE" },
]);
const stampTypeOps = ref<DataOption[]>([
{ name: "เต็มวัน", id: "FULL_DAY" },
{ name: "ครึ่งเช้า", id: "MORNING" },
{ name: "ครึ่งบ่าย ", id: "AFTERNOON" },
]);
return { statusOps, stampTypeOps };
});

View file

@ -0,0 +1,177 @@
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

@ -0,0 +1,157 @@
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

@ -806,9 +806,13 @@ function handleBackNavigation() {
router.go(-1); router.go(-1);
} else { } else {
if (empType.value === "") { if (empType.value === "") {
router.push("/registry-officer"); storeRegistry.isLeave
? router.push("/registry-retire-officer")
: router.push("/registry-officer");
} else if (empType.value === "-employee") { } else if (empType.value === "-employee") {
router.push("/registry-employee"); storeRegistry.isLeave
? router.push("/registry-retire-employee")
: router.push("/registry-employee");
} }
} }
} }

View file

@ -0,0 +1,270 @@
<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,6 +706,7 @@ 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,6 +20,7 @@ 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();
@ -52,6 +53,11 @@ 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);
@ -430,66 +436,67 @@ function classColorRow(isDelete: boolean, isEdit: boolean, isEntry: boolean) {
/** ฟังก์ชันดาวน์โหลดไฟล Excel */ /** ฟังก์ชันดาวน์โหลดไฟล Excel */
function exportToExcel() { function exportToExcel() {
const newData = rows.value.map((e: DataPosition) => { exportToExcelPosition(rows.value);
return { // const newData = rows.value.map((e: DataPosition) => {
commandDateAffect: date2Thai(e.commandDateAffect), // return {
positionName: e.positionName, // commandDateAffect: date2Thai(e.commandDateAffect),
positionType: e.positionType, // positionName: e.positionName,
positionLevel: e.positionLevel // positionType: e.positionType,
? e.positionLevel // positionLevel: e.positionLevel
: e.positionCee // ? e.positionLevel
? e.positionCee // : e.positionCee
: "", // ? e.positionCee
positionExecutive: e.positionExecutive, // : "",
amount: e.amount, // positionExecutive: e.positionExecutive,
mouthSalaryAmount: e.mouthSalaryAmount, // amount: e.amount,
positionSalaryAmount: e.positionSalaryAmount, // mouthSalaryAmount: e.mouthSalaryAmount,
organization: findOrgName({ // positionSalaryAmount: e.positionSalaryAmount,
root: e.orgRoot, // organization: findOrgName({
child1: e.orgChild1, // root: e.orgRoot,
child2: e.orgChild2, // child1: e.orgChild1,
child3: e.orgChild3, // child2: e.orgChild2,
child4: e.orgChild4, // child3: e.orgChild3,
}), // child4: e.orgChild4,
posNo: // }),
e.posNoAbb && e.posNo // posNo:
? `${e.posNoAbb} ${e.posNo}` // e.posNoAbb && e.posNo
: e.posNo // ? `${e.posNoAbb} ${e.posNo}`
? e.posNo // : e.posNo
: "", // ? e.posNo
posNumCodeSit: // : "",
e.posNumCodeSitAbb && e.posNumCodeSit // posNumCodeSit:
? `${e.posNumCodeSit} (${e.posNumCodeSitAbb})` // e.posNumCodeSitAbb && e.posNumCodeSit
: e.posNumCodeSit // ? `${e.posNumCodeSit} (${e.posNumCodeSitAbb})`
? e.posNumCodeSit // : e.posNumCodeSit
: "", // ? e.posNumCodeSit
commandNo: // : "",
e.commandNo && e.commandYear // commandNo:
? `${e.commandNo}/${Number(e.commandYear) + 543}` // e.commandNo && e.commandYear
: "", // ? `${e.commandNo}/${Number(e.commandYear) + 543}`
commandDateSign: date2Thai(e.commandDateSign), // : "",
commandCode: store.convertCommandCodeName(e.commandCode), // commandDateSign: date2Thai(e.commandDateSign),
remark: e.remark, // commandCode: store.convertCommandCodeName(e.commandCode),
}; // remark: e.remark,
}); // };
// });
const headers = columns.value.map((item: any) => item.label) || []; // // const headers = columns.value.map((item: any) => item.label) || []; //
const worksheet = XLSX.utils.json_to_sheet(newData, { // const worksheet = XLSX.utils.json_to_sheet(newData, {
header: visibleColumns.value, // header: visibleColumns.value,
}); // });
// ( A1, B1, C1 ) // // ( A1, B1, C1 )
XLSX.utils.sheet_add_aoa(worksheet, [headers], { origin: "A1" }); // XLSX.utils.sheet_add_aoa(worksheet, [headers], { origin: "A1" });
// Create a new workbook and append the worksheet // // Create a new workbook and append the worksheet
const workbook = XLSX.utils.book_new(); // const workbook = XLSX.utils.book_new();
XLSX.utils.book_append_sheet( // XLSX.utils.book_append_sheet(
workbook, // workbook,
worksheet, // worksheet,
`รายการประวัติตำแหน่งเงินเดือน` // ``
); // );
XLSX.writeFile(workbook, "รายการประวัติตำแหน่งเงินเดือน.xlsx"); // XLSX.writeFile(workbook, ".xlsx");
} }
const commandCodeOptions = ref<DataOption[]>(store.commandCodeData); // const commandCodeOptions = ref<DataOption[]>(store.commandCodeData); //
@ -847,6 +854,83 @@ 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;
}
onMounted(async () => { onMounted(async () => {
await Promise.all([fetchData(), fetchType()]); await Promise.all([fetchData(), fetchType()]);
}); });
@ -886,6 +970,20 @@ onMounted(async () => {
> >
<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
@ -893,7 +991,9 @@ 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
@ -1724,6 +1824,13 @@ 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,6 +54,7 @@ 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() {
@ -79,6 +80,7 @@ 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);
@ -323,7 +325,10 @@ onMounted(async () => {
</div> </div>
<div class="col-12"><q-separator /></div> <div class="col-12"><q-separator /></div>
<div class="col-xs-6 col-sm-6 row items-center"> <div
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,6 +45,7 @@ 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() {
@ -61,6 +62,7 @@ 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;
@ -313,7 +315,10 @@ onMounted(() => {
</div> </div>
<div class="col-12"><q-separator /></div> <div class="col-12"><q-separator /></div>
<div class="col-xs-6 col-sm-6 row items-center"> <div
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

@ -53,6 +53,7 @@ const saveData = async () => {
changeBtn(); changeBtn();
onEdit.value = false; onEdit.value = false;
edit.value = false; edit.value = false;
emit("update:statusEdit", false);
}) })
.catch((e: any) => { .catch((e: any) => {
messageError($q, e); messageError($q, e);

View file

@ -1,7 +1,7 @@
<!-- วนหวของ อมลสวนต และ อย --> <!-- วนหวของ อมลสวนต และ อย -->
<script setup lang="ts"> <script setup lang="ts">
import { checkPermission } from "@/utils/permissions"; import { checkPermission } from "@/utils/permissions";
import { ref } from "vue"; import { computed } from "vue";
import { useRoute } from "vue-router"; import { useRoute } from "vue-router";
const route = useRoute(); const route = useRoute();
@ -62,9 +62,9 @@ const props = defineProps({
}); });
const emit = defineEmits(["update:edit"]); const emit = defineEmits(["update:edit"]);
const isEdit = ref<boolean>( const isEdit = computed(() => {
checkPermission(route)?.attrIsUpdate ? true : false return checkPermission(route)?.attrIsUpdate ? true : false;
); });
const updateEdit = (value: any) => { const updateEdit = (value: any) => {
emit("update:edit", value); emit("update:edit", value);

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 seletcId = defineModel<string>("seletcId", { required: true }); // const selectId = defineModel<string>("selectId", { required: true }); //
const date = defineModel<Date>("datePos", { required: true }); // const date = defineModel<Date | null>("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>("posType", { required: true }); // const posType = defineModel<FormPosType | null>("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,6 +47,7 @@ const props = defineProps({
onPosType: Function, onPosType: Function,
nodeId: String, nodeId: String,
nodeLevel: Number, nodeLevel: Number,
isLoadPosition: Boolean,
}); });
//Table //Table
@ -213,9 +214,9 @@ async function onClickSelectPos(id: string) {
// //
if (position) { if (position) {
rowsPosition.value = position.positions; rowsPosition.value = position.positions;
if (seletcId.value) { if (selectId.value) {
selected.value = rowsPosition.value.filter( selected.value = rowsPosition.value.filter(
(e) => e.id === seletcId.value (e) => e.id === selectId.value
); );
} }
} }
@ -248,15 +249,20 @@ 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

@ -3,6 +3,7 @@ import { ref, watch } from "vue";
import { useQuasar } from "quasar"; import { useQuasar } from "quasar";
import { useCounterMixin } from "@/stores/mixin"; import { useCounterMixin } from "@/stores/mixin";
import { calculateAge } from "@/utils/function";
import http from "@/plugins/http"; import http from "@/plugins/http";
import config from "@/app.config"; import config from "@/app.config";
@ -38,6 +39,20 @@ const props = defineProps({
const rows = ref<DataEducation[]>([]); const rows = ref<DataEducation[]>([]);
const personalForm = ref<DataPerson>(); const personalForm = ref<DataPerson>();
const age = ref<string | null>(""); //
// ID
const currentProvinceName = ref<string>("");
const currentDistrictName = ref<string>("");
const currentSubDistrictName = ref<string>("");
const registProvinceName = ref<string>("");
const registDistrictName = ref<string>("");
const registSubDistrictName = ref<string>("");
// Cache
const provincesCache = ref<any[]>([]);
const districtsCache = ref<Map<string, any[]>>(new Map()); // key: provinceId, value: districts[]
const subDistrictsCache = ref<Map<string, any[]>>(new Map()); // key: districtId, value: subDistricts[]
/**หัวตาราง */ /**หัวตาราง */
const columns = ref<QTableProps["columns"]>([ const columns = ref<QTableProps["columns"]>([
@ -87,11 +102,20 @@ const columns = ref<QTableProps["columns"]>([
* งกนดงขอมลรายละเอยด * งกนดงขอมลรายละเอยด
*/ */
async function fetchData() { async function fetchData() {
// Reset
resetData();
showLoader(); showLoader();
await http await http
.get(config.API.getDatapersonal(props.personalId)) .get(config.API.getDatapersonal(props.personalId))
.then((res) => { .then((res) => {
personalForm.value = res.data.result; personalForm.value = res.data.result;
if (res.data.result.dateOfBirth) {
//
age.value = calculateAge(res.data.result.dateOfBirth);
} else {
age.value = null;
}
personalForm.value?.education.map((e: Education) => { personalForm.value?.education.map((e: Education) => {
rows.value.push({ rows.value.push({
university: e.institute, university: e.institute,
@ -107,6 +131,159 @@ async function fetchData() {
.finally(() => { .finally(() => {
hideLoader(); hideLoader();
}); });
// ID
await convertAddressIds();
}
/**
* Reset อมลทกครงทเปดคนใหม
*/
function resetData() {
// Reset
personalForm.value = undefined;
age.value = "";
rows.value = [];
// Reset
currentProvinceName.value = "";
currentDistrictName.value = "";
currentSubDistrictName.value = "";
registProvinceName.value = "";
registDistrictName.value = "";
registSubDistrictName.value = "";
}
/**
* งกนแปลง ID เปนชอจรงของทอย (แบบ optimize)
*/
async function convertAddressIds() {
if (!personalForm.value) return;
try {
//
await loadProvinces();
//
if (personalForm.value.currentProvinceId && personalForm.value.currentProvinceId.trim() !== "") {
currentProvinceName.value = getProvinceNameFromCache(
personalForm.value.currentProvinceId
);
if (personalForm.value.currentDistrictId && personalForm.value.currentDistrictId.trim() !== "") {
currentDistrictName.value = await getDistrictNameOptimized(
personalForm.value.currentProvinceId,
personalForm.value.currentDistrictId
);
if (personalForm.value.currentSubDistrictId && personalForm.value.currentSubDistrictId.trim() !== "") {
currentSubDistrictName.value = await getSubDistrictNameOptimized(
personalForm.value.currentDistrictId,
personalForm.value.currentSubDistrictId
);
}
}
}
//
if (personalForm.value.registProvinceId && personalForm.value.registProvinceId.trim() !== "") {
registProvinceName.value = getProvinceNameFromCache(
personalForm.value.registProvinceId
);
if (personalForm.value.registDistrictId && personalForm.value.registDistrictId.trim() !== "") {
registDistrictName.value = await getDistrictNameOptimized(
personalForm.value.registProvinceId,
personalForm.value.registDistrictId
);
if (personalForm.value.registSubDistrictId && personalForm.value.registSubDistrictId.trim() !== "") {
registSubDistrictName.value = await getSubDistrictNameOptimized(
personalForm.value.registDistrictId,
personalForm.value.registSubDistrictId
);
}
}
}
} catch (error) {
console.error("Error converting address IDs:", error);
}
}
/**
* โหลดขอมลจงหวดทงหมด (ครงเดยว)
*/
async function loadProvinces() {
if (provincesCache.value.length === 0) {
try {
const res = await http.get(config.API.province);
provincesCache.value = res.data.result;
} catch (error) {
console.error("Error loading provinces:", error);
}
}
}
/**
* งชอจงหวดจาก cache
*/
function getProvinceNameFromCache(provinceId: string): string {
const province = provincesCache.value.find(
(p: any) => p.id.toString() === provinceId
);
return province ? province.name : "-";
}
/**
* งชออำเภอแบบ optimize (เรยก API เฉพาะจงหวดทองการ)
* @param provinceId ID ของจงหว
* @param districtId ID ของอำเภอ
*/
async function getDistrictNameOptimized(
provinceId: string,
districtId: string
): Promise<string> {
try {
// cache districts province
if (!districtsCache.value.has(provinceId)) {
const res = await http.get(config.API.listDistrict(provinceId));
districtsCache.value.set(provinceId, res.data.result.districts);
}
const districts = districtsCache.value.get(provinceId) || [];
const district = districts.find((d: any) => d.id.toString() === districtId);
return district ? district.name : "-";
} catch (error) {
console.error("Error loading district:", error);
return "-";
}
}
/**
* งชอตำบลแบบ optimize (เรยก API เฉพาะอำเภอทองการ)
* @param districtId ID ของอำเภอ
* @param subDistrictId ID ของตำบล
*/
async function getSubDistrictNameOptimized(
districtId: string,
subDistrictId: string
): Promise<string> {
try {
// cache subDistricts district
if (!subDistrictsCache.value.has(districtId)) {
const res = await http.get(config.API.listSubDistrict(districtId));
subDistrictsCache.value.set(districtId, res.data.result.subDistricts);
}
const subDistricts = subDistrictsCache.value.get(districtId) || [];
const subDistrict = subDistricts.find(
(sd: any) => sd.id.toString() === subDistrictId
);
return subDistrict ? subDistrict.name : "-";
} catch (error) {
console.error("Error loading subdistrict:", error);
return "-";
}
} }
/** /**
@ -131,7 +308,7 @@ function formBmaofficer(val: string) {
*/ */
async function close() { async function close() {
props.close(); props.close();
rows.value = []; resetData(); // Reset dialog
} }
/** /**
@ -170,30 +347,58 @@ watch(props, () => {
<div class="row q-pa-xs"> <div class="row q-pa-xs">
<div class="col-3 header-sub-text"> <div class="col-3 header-sub-text">
<div class="q-pb-md">เลขประจำตวประชาชน</div> <div class="q-pb-sm">เลขประจำตวประชาชน</div>
<div>/เดอน/เก</div> <div class="q-pb-sm">/เดอน/เก</div>
<div class="q-pb-sm">อาย</div>
<div class="q-pb-sm">ญชาต</div>
<div>หมเลอด</div>
</div> </div>
<div class="col-4 sub-text"> <div class="col-4 sub-text">
<div class="q-pb-md"> <div class="q-pb-sm">
{{ personalForm?.idCard }} {{ personalForm?.idCard }}
</div> </div>
<div> <div class="q-pb-sm">
{{ date2Thai(personalForm?.dateOfBirth) }} {{ date2Thai(personalForm?.dateOfBirth) }}
</div> </div>
<div class="q-pb-sm">
{{ age ? age : "-" }}
</div>
<div class="q-pb-sm">
{{
personalForm?.nationality ? personalForm.nationality : "-"
}}
</div>
<div>
{{
personalForm?.bloodGroup ? personalForm.bloodGroup : "-"
}}
</div>
</div> </div>
<div class="col-2 header-sub-text"> <div class="col-2 header-sub-text">
<div class="q-pb-md">-นามสก</div> <div class="q-pb-sm">-นามสก</div>
<div>เพศ</div> <div class="q-pb-sm">เพศ</div>
<div class="q-pb-sm">เชอชาต</div>
<div class="q-pb-sm">ศาสนา</div>
<div>เบอรโทรศพท</div>
</div> </div>
<div class="col-3 sub-text"> <div class="col-3 sub-text">
<div class="q-pb-md"> <div class="q-pb-sm">
{{ personalForm?.fullName }} {{ personalForm?.fullName }}
</div> </div>
<div> <div class="q-pb-sm">
{{ personalForm?.gender }} {{ personalForm?.gender }}
</div> </div>
<div class="q-pb-sm">
{{ personalForm?.race ? personalForm.race : "-" }}
</div>
<div class="q-pb-sm">
{{ personalForm?.religion ? personalForm.religion : "-" }}
</div>
<div class="q-mt-sm">
{{ personalForm?.telephone ? personalForm.telephone : "-" }}
</div>
</div> </div>
</div> </div>
</q-card> </q-card>
@ -203,9 +408,90 @@ watch(props, () => {
<q-card bordered class="card-panding"> <q-card bordered class="card-panding">
<div class="row items-center q-pa-xs header-text">ลำเนา</div> <div class="row items-center q-pa-xs header-text">ลำเนา</div>
<div class="row q-pa-xs"> <div class="row q-pa-xs">
<div class="col-3 header-sub-text">อย</div> <!-- อยจจ -->
<div class="col-9 sub-text"> <div class="col-6">
{{ personalForm?.registAddress }} <div class="q-pb-sm text-weight-medium text-primary">
อยจจ
</div>
<div class="row">
<div class="col-4 header-sub-text">
<div class="q-pb-sm">อย</div>
<div class="q-pb-sm">งหว</div>
<div class="q-pb-sm">เขต/อำเภอ</div>
<div class="q-pb-sm">แขวง/ตำบล</div>
<div>รหสไปรษณ</div>
</div>
<div class="col-8 sub-text">
<div class="q-pb-sm">
{{
personalForm?.currentAddress
? personalForm.currentAddress
: "-"
}}
</div>
<div class="q-pb-sm">
{{ currentProvinceName ? currentProvinceName : "-" }}
</div>
<div class="q-pb-sm">
{{ currentDistrictName ? currentDistrictName : "-" }}
</div>
<div class="q-pb-sm">
{{
currentSubDistrictName ? currentSubDistrictName : "-"
}}
</div>
<div class="q-mt-sm">
{{
personalForm?.currentZipCode
? personalForm.currentZipCode
: "-"
}}
</div>
</div>
</div>
</div>
<!-- อยตามทะเบยนบาน -->
<div class="col-6">
<div class="q-pb-sm text-weight-medium text-primary">
อยตามทะเบยนบาน
</div>
<div class="row">
<div class="col-4 header-sub-text">
<div class="q-pb-sm">อย</div>
<div class="q-pb-sm">งหว</div>
<div class="q-pb-sm">เขต/อำเภอ</div>
<div class="q-pb-sm">แขวง/ตำบล</div>
<div>รหสไปรษณ</div>
</div>
<div class="col-8 sub-text">
<div class="q-pb-sm">
{{
personalForm?.registAddress
? personalForm.registAddress
: "-"
}}
</div>
<div class="q-pb-sm">
{{ registProvinceName ? registProvinceName : "-" }}
</div>
<div class="q-pb-sm">
{{ registDistrictName ? registDistrictName : "-" }}
</div>
<div class="q-pb-sm">
{{
registSubDistrictName ? registSubDistrictName : "-"
}}
</div>
<div class="q-mt-sm">
{{
personalForm?.registZipCode
? personalForm.registZipCode
: "-"
}}
</div>
</div>
</div>
</div> </div>
</div> </div>
</q-card> </q-card>

View file

@ -1,5 +1,5 @@
<script setup lang="ts"> <script setup lang="ts">
import { onMounted, ref, watch, reactive } from "vue"; import { onMounted, ref, watch } from "vue";
import { useQuasar } from "quasar"; import { useQuasar } from "quasar";
import { useRoute } from "vue-router"; import { useRoute } from "vue-router";
@ -11,6 +11,7 @@ import { useStructureTree } from "@/stores/structureTree";
/** importType*/ /** importType*/
import type { import type {
DataList,
PositionMaim, PositionMaim,
PositionNo, PositionNo,
Positions, Positions,
@ -41,17 +42,14 @@ const {
/**props*/ /**props*/
const modal = defineModel<boolean>("modal", { required: true }); const modal = defineModel<boolean>("modal", { required: true });
const props = defineProps({
dataRow: { interface Props {
type: Object, dataRow?: DataList;
require: true, fetchTable?: () => Promise<void>;
}, fetchStatCard?: () => Promise<void>;
fetchTable: { }
type: Function,
require: true, const props = defineProps<Props>();
},
fetchStatCard: { type: Function, require: true },
});
/** Tree*/ /** Tree*/
const nodeId = ref<string>(""); const nodeId = ref<string>("");
@ -65,15 +63,16 @@ 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 seletcId = ref<string>(""); const selectId = 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<any[]>([]); const selectedPos = ref<Positions[]>([]);
const datePos = ref<Date>(new Date()); const datePos = ref<Date | null>(new Date());
const posMasterMain = ref<any[]>([]); const posMasterMain = ref<PositionMaim[]>([]);
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() {
@ -91,12 +90,14 @@ 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 ?? "";
seletcId.value = props?.dataRow?.positionId; selectId.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 = "";
seletcId.value = ""; selectId.value = "";
selectedPos.value = []; selectedPos.value = [];
datePos.value = new Date(); datePos.value = new Date();
} }
@ -117,6 +118,8 @@ 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,
@ -150,7 +153,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: any = { ...e, ...rest }; const data: PositionMaim = { ...e, ...rest } as PositionMaim;
dataMain.push(data); dataMain.push(data);
} }
}); });
@ -162,21 +165,20 @@ async function fetchDataTable(id: string, level: number = 0) {
(e) => e !== props.dataRow?.posmasterId (e) => e !== props.dataRow?.posmasterId
); );
positionNo.value = posMain.filter((e: any) => !newUse.includes(e.id)); positionNo.value = posMain.filter((e: DataPositionNo) => !newUse.includes(e.id));
} else { } else {
positionNo.value = posMain.filter( positionNo.value = posMain.filter(
(e: any) => !positionUse.value.includes(e.id) (e: DataPositionNo) => !positionUse.value.includes(e.id)
); );
} }
isLoadPosition.value = true;
}) })
.catch((err) => { .catch((err) => {
messageError($q, err); messageError($q, err);
hideLoader();
}) })
.finally(() => { .finally(() => {
setTimeout(() => { hideLoader();
hideLoader();
}, 1000);
}); });
} }
@ -198,9 +200,11 @@ 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 ?? "";
seletcId.value = props?.dataRow?.positionId; selectId.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);
}) })
@ -212,12 +216,14 @@ async function fetchPosFind(level: number, id: string) {
/** function บันทึกข้อมูลตำแหน่ง*/ /** function บันทึกข้อมูลตำแหน่ง*/
async function onClickSubmit() { async function onClickSubmit() {
const dataPosMaster = await posMasterMain.value?.find( const dataPosMaster = posMasterMain.value?.find(
(e: any) => e.id === positionId.value (e) => 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();
@ -272,6 +278,9 @@ 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*/
@ -280,9 +289,8 @@ 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(props?.dataRow?.node, props?.dataRow?.nodeId); await fetchPosFind(props?.dataRow?.node ?? 0, props?.dataRow?.nodeId ?? "");
} else { } else {
expanded.value = []; expanded.value = [];
} }
@ -323,8 +331,7 @@ async function getOrgPosType() {
}) })
.catch((e) => { .catch((e) => {
messageError($q, e); messageError($q, e);
}) });
.finally(() => {});
} }
function onPosType() { function onPosType() {
@ -336,28 +343,15 @@ function onPosType() {
} }
watch( watch(
() => isAll.value, [isAll, isBlank, () => isPosition.value],
(value, oldVal) => { ([newAll, newBlank, newPos], [oldAll, oldBlank, oldPos]) => {
if (value !== oldVal) { const shouldFetch = (newAll !== oldAll) || (newBlank !== oldBlank);
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 +452,15 @@ onMounted(() => {
:name="item" :name="item"
> >
<CardPosition <CardPosition
v-model:position="positionNo as []" v-model:position="positionNo"
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:seletcId="seletcId" v-model:selectId="selectId"
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 as FormPosType" v-model:pos-type="posType"
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,7 +468,8 @@ 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

@ -9,6 +9,8 @@ import { useRoute, useRouter } from "vue-router";
import { checkPermission } from "@/utils/permissions"; 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 { 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";
@ -27,6 +29,7 @@ import DialogPreviewCommand from "@/modules/18_command/components/DialogPreviewC
const $q = useQuasar(); // show dialog const $q = useQuasar(); // show dialog
const DataStore = usePlacementDataStore(); const DataStore = usePlacementDataStore();
const storeMenu = useMenuDataStore();
const route = useRoute(); const route = useRoute();
const router = useRouter(); const router = useRouter();
const mixin = useCounterMixin(); // const mixin = useCounterMixin(); //
@ -55,7 +58,14 @@ const command = ref<string>("");
const commandId = ref<string>(""); const commandId = ref<string>("");
const commandCitizenId = ref<string>(""); const commandCitizenId = ref<string>("");
let roleAdmin = ref<boolean>(false); // ref computed property
const roleAdmin = computed(() => {
if (!storeMenu.permissions) return false;
const permission = checkPermission(route);
return DataStore.isOfficer || permission?.attrOwnership == "OWNER";
});
const edit = ref<boolean>(true); const edit = ref<boolean>(true);
const modalPersonal = ref<boolean>(false); const modalPersonal = ref<boolean>(false);
const modal = ref<boolean>(false); //modal + const modal = ref<boolean>(false); //modal +
@ -84,7 +94,6 @@ const appointModal = ref<boolean>(false);
const getNumFile = ref(0); const getNumFile = ref(0);
const dataRes = ref<any>([]); const dataRes = ref<any>([]);
const personal = ref<any>([]); const personal = ref<any>([]);
const displayAdd = ref<boolean>(true);
const containStatus = ref<boolean>(false); const containStatus = ref<boolean>(false);
const modaladdlist = ref<boolean>(false); const modaladdlist = ref<boolean>(false);
const selected = ref<any>([]); const selected = ref<any>([]);
@ -92,6 +101,13 @@ const personal_selected = ref<any>([]);
const filterlistAdd = ref<string>(""); const filterlistAdd = ref<string>("");
const paging = ref<boolean>(true); const paging = ref<boolean>(true);
const displayAdd = computed(() => {
if (!storeMenu.permissions) return true;
const permission = checkPermission(route);
return roleAdmin.value || permission?.attrOwnership === "OWNER";
});
// //
const checkSelected = computed(() => { const checkSelected = computed(() => {
if (selected.value.length === 0) { if (selected.value.length === 0) {
@ -248,16 +264,17 @@ const columnsBase = ref<QTableProps["columns"]>([
}, },
]); ]);
const columns = computed(() => const columns = computed(() => {
roleAdmin.value || checkPermission(route)?.attrOwnership == "OWNER" const permission = checkPermission(route);
return roleAdmin.value || permission?.attrOwnership == "OWNER"
? columnsBase.value ? columnsBase.value
: columnsBase.value?.filter( : columnsBase.value?.filter(
(col) => (col) =>
col.name !== "no" && col.name !== "no" &&
col.name !== "draft" && col.name !== "draft" &&
col.name !== "refCommandNo" col.name !== "refCommandNo"
) );
); });
/** /**
* แปลงสถานะพนกงาน * แปลงสถานะพนกงาน
@ -406,8 +423,9 @@ async function getTable() {
rowsAll.value.push(rowData); rowsAll.value.push(rowData);
}); });
const permission = checkPermission(route);
const rowData = await (roleAdmin.value || const rowData = await (roleAdmin.value ||
checkPermission(route)?.attrOwnership == "OWNER" permission?.attrOwnership == "OWNER"
? rowsAll.value ? rowsAll.value
: rowsAll.value.filter((x: any) => x.isDraft === true)); : rowsAll.value.filter((x: any) => x.isDraft === true));
@ -554,8 +572,9 @@ function getClass(val: boolean) {
* @param draft status * @param draft status
*/ */
function selectData(pid: string, draft: string) { function selectData(pid: string, draft: string) {
const permission = checkPermission(route);
if ( if (
(roleAdmin.value || checkPermission(route)?.attrOwnership == "OWNER") && (roleAdmin.value || permission?.attrOwnership == "OWNER") &&
draft === "ส่งตัวแล้ว" draft === "ส่งตัวแล้ว"
) { ) {
personalId.value = pid; personalId.value = pid;
@ -858,34 +877,33 @@ async function filterRowsMain(type: string, pos: string) {
}); });
} }
// Helper function permissions load
function waitForPermissions(): Promise<void> {
return new Promise((resolve) => {
const checkPermissions = () => {
if (storeMenu.permissions) {
resolve();
} else {
setTimeout(checkPermissions, 100); // 100ms
}
};
checkPermissions();
});
}
// getWorkFlow function
async function getWorkFlow() { async function getWorkFlow() {
showLoader(); showLoader();
// permissions load
await waitForPermissions();
await http await http
.get(config.API.workflowKeycloakSystem("SYS_PLACEMENT_PASS")) .get(config.API.workflowKeycloakSystem("SYS_PLACEMENT_PASS"))
.then(async (res) => { .then(async (res) => {
const data = await res.data.result; const data = await res.data.result;
DataStore.isOfficer = data.isOfficer; DataStore.isOfficer = data.isOfficer;
DataStore.isStaff = data.isStaff; DataStore.isStaff = data.isStaff;
roleAdmin.value =
data.isOfficer || checkPermission(route)?.attrOwnership == "OWNER";
if (
roleAdmin.value === false &&
checkPermission(route)?.attrOwnership !== "OWNER"
) {
displayAdd.value = false;
// visibleColumns.value = [
// "position",
// "fullName",
// "examNumber",
// "idCard",
// "positionNumber",
// "organizationName",
// "reportingDate",
// "bmaOfficer",
// "statusName",
// "positionCandidate",
// ];
}
}) })
.catch((e) => { .catch((e) => {
messageError($q, e); messageError($q, e);
@ -950,6 +968,28 @@ function onUpdateStatus(id: string) {
); );
} }
function onUpdateDraftStatus(id: string) {
dialogConfirm(
$q,
async () => {
showLoader();
try {
await http.post(config.API.placementUpdateDraftStatus, {
personalId: id,
});
await success($q, "บันทึกสำเร็จ");
await getTable();
} catch (error) {
messageError($q, error);
} finally {
hideLoader();
}
},
"ยืนยันการยกเลิกการส่งตัว",
"ต้องการยกเลิกการส่งตัวรายชื่อนี้ใช่หรือไม่ ?"
);
}
onMounted(async () => { onMounted(async () => {
await getWorkFlow(); await getWorkFlow();
await getTable(); await getTable();
@ -1012,6 +1052,31 @@ onMounted(async () => {
</q-item-section> </q-item-section>
<q-item-section>รายละเอยด</q-item-section> <q-item-section>รายละเอยด</q-item-section>
</q-item> </q-item>
<q-item
v-if="
checkPermission($route)?.attrIsUpdate &&
props.row.isDraft &&
props.row.statusId === 'PREPARE-CONTAIN' &&
(DataStore.isOfficer ||
checkPermission($route)?.attrOwnership == 'OWNER')
"
clickable
v-close-popup
@click="onUpdateDraftStatus(props.row.personalId)"
>
<q-item-section
style="min-width: 0px"
avatar
class="q-py-sm"
>
<q-icon
color="red"
size="xs"
name="mdi-account-arrow-left-outline"
/>
</q-item-section>
<q-item-section>ยกเลกการสงต</q-item-section>
</q-item>
<q-item <q-item
v-if=" v-if="
@ -1480,7 +1545,10 @@ onMounted(async () => {
:label="`${'เลือกไฟล์เอกสารหลักฐาน'}`" :label="`${'เลือกไฟล์เอกสารหลักฐาน'}`"
outlined outlined
use-chips use-chips
:rules="[(val:string) => !!val || 'กรุณาเลือกไฟล์เอกสารหลักฐาน']" :rules="[
(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

@ -89,7 +89,9 @@ 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.prefix ?? ""}${row.firstName ?? ""} ${row.lastName ?? ""}`; return `${row.rank ? row.rank : row.prefix ?? ""}${row.firstName ?? ""} ${
row.lastName ?? ""
}`;
}, },
}, },
{ {
@ -110,9 +112,11 @@ 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 return row.positionTypeOld && row.positionTypeOl !== "-"
? `${row.positionTypeOld}${ ? `${row.positionTypeOld}${
row.positionLevelOld ? `(${row.positionLevelOld})` : "" row.positionLevelOld && row.positionLevelOld !== "-"
? `(${row.positionLevelOld})`
: ""
}` }`
: ""; : "";
}, },

View file

@ -48,6 +48,7 @@ const informaData = ref<FormAddPerson>({
employeeType: null, employeeType: null,
employeeClass: null, employeeClass: null,
profileType: null, profileType: null,
rank: "",
}); });
// //
@ -60,6 +61,7 @@ const Ops = ref<InformationOps>({
religionOps: [], religionOps: [],
employeeClassOps: [], employeeClassOps: [],
employeeTypeOps: [], employeeTypeOps: [],
rankOps: [],
}); });
// //
const OpsFilter = ref<InformationOps>({ const OpsFilter = ref<InformationOps>({
@ -71,6 +73,7 @@ const OpsFilter = ref<InformationOps>({
religionOps: [], religionOps: [],
employeeClassOps: [], employeeClassOps: [],
employeeTypeOps: [], employeeTypeOps: [],
rankOps: [],
}); });
// profile // profile
@ -135,6 +138,16 @@ 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);
@ -204,7 +217,13 @@ 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;
} }
@ -227,7 +246,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) if (informaData.value.prefix != undefined && informaData.value.prefix != "")
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);
@ -256,6 +275,8 @@ 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();
@ -289,6 +310,15 @@ function updateBirthDate(v: Date) {
age.value = calculateAge(v); age.value = calculateAge(v);
} }
function prefixRankRule() {
return [
() =>
!!informaData.value.rank ||
!!informaData.value.prefix ||
"กรุณาเลือกคำนำหน้าชื่อ หรือยศ",
];
}
/** /**
* ทำงานเมอมการเรยกใช Components * ทำงานเมอมการเรยกใช Components
*/ */
@ -383,7 +413,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-3"> <div class="col-2">
<q-input <q-input
bg-color="white" bg-color="white"
outlined outlined
@ -404,13 +434,14 @@ onMounted(async () => {
mask="#############" mask="#############"
/> />
</div> </div>
<div class="col-3"> <div class="col-2">
<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"
@ -418,11 +449,8 @@ onMounted(async () => {
option-value="name" option-value="name"
map-options map-options
hide-bottom-space hide-bottom-space
:rules="[ :rules="prefixRankRule()"
(val:string) => { reactive-rules
return val.length > 0 || 'กรุณาเลือกคำนำหน้าชื่อ';
},
]"
emit-value emit-value
use-input use-input
hide-selected hide-selected
@ -432,6 +460,32 @@ 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,6 +113,7 @@ 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: [],
@ -129,6 +130,7 @@ const OpsFilter = ref<InformationOps>({
{ id: "gov", name: "งบประมาณเงินอุดหนุนรัฐบาล" }, { id: "gov", name: "งบประมาณเงินอุดหนุนรัฐบาล" },
{ id: "bkk", name: "งบประมาณกรุงเทพมหานคร" }, { id: "bkk", name: "งบประมาณกรุงเทพมหานคร" },
], ],
rankOps: [],
}); });
/** ฟังก์ชันดึงข้อมูลรายการข้อมูลเกี่ยวกับบุคคล (dropdown list)*/ /** ฟังก์ชันดึงข้อมูลรายการข้อมูลเกี่ยวกับบุคคล (dropdown list)*/
@ -186,6 +188,16 @@ 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);
@ -215,9 +227,9 @@ async function getData() {
} }
rows.value = list; rows.value = list;
profileId.value = data.profileId; profileId.value = data.profileId;
title.value.fullname = `${data.prefix ?? ""}${data.firstName ?? ""} ${ title.value.fullname = `${data.rank ? data.rank : data.prefix ?? ""}${
data.lastName ?? "" data.firstName ?? ""
}`; } ${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 ?? "-";
@ -230,6 +242,7 @@ 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:
@ -295,6 +308,7 @@ 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,
@ -436,6 +450,14 @@ 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;
} }
@ -470,6 +492,7 @@ 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
@ -517,6 +540,15 @@ 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();
@ -598,7 +630,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-3 col-md-3"> <div class="col-xs-6 col-sm-2 col-md-2">
<q-input <q-input
:class="getClass(edit)" :class="getClass(edit)"
hide-bottom-space hide-bottom-space
@ -614,7 +646,7 @@ onMounted(async () => {
mask="#############" mask="#############"
/> />
</div> </div>
<div class="col-xs-6 col-sm-3 col-md-3"> <div class="col-xs-6 col-sm-2 col-md-2">
<selector <selector
:hide-dropdown-icon="!edit" :hide-dropdown-icon="!edit"
hide-bottom-space hide-bottom-space
@ -634,10 +666,42 @@ 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,6 +201,7 @@ 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,6 +7,7 @@ interface InformationOps {
religionOps: DataOption[]; religionOps: DataOption[];
employeeClassOps: DataOption[]; employeeClassOps: DataOption[];
employeeTypeOps: DataOption[]; employeeTypeOps: DataOption[];
rankOps: DataOption[];
} }
//ข้อมูลส่วนตัว //ข้อมูลส่วนตัว
@ -63,6 +64,7 @@ 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 = {
@ -83,6 +85,7 @@ const defaultInformation: Information = {
employeeType: null, employeeType: null,
employeeClass: null, employeeClass: null,
profileType: null, profileType: null,
rank: null,
}; };
export { defaultInformation }; export { defaultInformation };

View file

@ -1,17 +1,24 @@
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

@ -110,7 +110,9 @@ 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.prefix ?? ""}${row.firstName ?? ""} ${row.lastName ?? ""}`; return ` ${row.rank ? row.rank : row.prefix ?? ""}${
row.firstName ?? ""
} ${row.lastName ?? ""}`;
}, },
}, },
{ {
@ -131,9 +133,11 @@ 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 return row.positionTypeOld && row.positionTypeOld !== "-"
? `${row.positionTypeOld}${ ? `${row.positionTypeOld}${
row.positionLevelOld ? `(${row.positionLevelOld})` : "" row.positionLevelOld && row.positionLevelOld !== "-"
? `(${row.positionLevelOld})`
: ""
}` }`
: ""; : "";
}, },

View file

@ -381,7 +381,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

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

View file

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

@ -771,6 +771,7 @@ onMounted(async () => {
<q-space /> <q-space />
<div <div
v-if=" v-if="
!checkRoutePermisson &&
isStaff && isStaff &&
status === 'WAITTING' && status === 'WAITTING' &&
checkPermission($route)?.attrIsUpdate checkPermission($route)?.attrIsUpdate

View file

@ -36,7 +36,7 @@ const {
dialogConfirm, dialogConfirm,
} = useCounterMixin(); } = useCounterMixin();
const checkRoutePermisson = ref<boolean>(route.name == "resignDetailReject"); const checkRoutePermisson = ref<boolean>(route.name == "resignDetailreject");
/** ตัวแปร */ /** ตัวแปร */
const roleUser = ref<string>(""); const roleUser = ref<string>("");

View file

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

View file

@ -29,7 +29,7 @@ const router = useRouter();
const store = useRetirementDataStore(); const store = useRetirementDataStore();
const { fetchDataCheckIsoffice } = useRoleWorkflowDataStore(); const { fetchDataCheckIsoffice } = useRoleWorkflowDataStore();
const { convertStatusText } = store; const { convertStatusText } = store;
const checkRoutePermisson = ref<boolean>(route.name == "resignDetailRejectEMP"); const checkRoutePermisson = ref<boolean>(route.name == "resignDetailrejectEMP");
const mixin = useCounterMixin(); const mixin = useCounterMixin();
const { const {
messageError, messageError,

View file

@ -4,7 +4,6 @@ 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";
@ -229,6 +228,7 @@ 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,9 +308,7 @@ watch(
:key="col.name" :key="col.name"
:props="props" :props="props"
> >
<span class="text-weight-medium">{{ <span class="text-weight-medium">{{ col.label }}</span>
getColumnLabel(col, isAct)
}}</span>
</q-th> </q-th>
</q-tr> </q-tr>
</template> </template>

View file

@ -151,7 +151,7 @@ async function fetchDataRetirement() {
firstNameTH: filter.value.firstNameTH.trim(), firstNameTH: filter.value.firstNameTH.trim(),
lastNameTH: filter.value.lastNameTH.trim(), lastNameTH: filter.value.lastNameTH.trim(),
}); });
const data: RetirementOld = response.data.data; const data: RetirementOld = response.data.result;
pagination.value.rowsNumber = data.totalRecords; pagination.value.rowsNumber = data.totalRecords;
rows.value = data.dataRecords; rows.value = data.dataRecords;
} catch (error) { } catch (error) {

View file

@ -35,7 +35,7 @@ const columns = ref<QTableProps["columns"]>([
field: "year", field: "year",
headerStyle: "font-size: 14px", headerStyle: "font-size: 14px",
style: "font-size: 14px", style: "font-size: 14px",
format: (v) => v + 543, format: (v) => (v ? v + 543 : "-"),
}, },
{ {
name: "receiveDate", name: "receiveDate",

View file

@ -267,7 +267,13 @@ 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">{{ formData.checkInTime }}</div> <div class="col">
{{
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>
@ -337,17 +343,23 @@ 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">{{ formData.checkOutTime }}</div> <div class="col">
{{
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.checkInLocationName formData.checkOutLocationName
? `นอกสถานที่ (${formData.checkInLocationName})` ? `นอกสถานที่ (${formData.checkOutLocationName})`
: "" : ""
}} }}
</div> </div>

View file

@ -74,6 +74,12 @@ 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",
@ -101,6 +107,12 @@ 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",
@ -160,19 +172,19 @@ async function fetchListTimeRecord() {
id: e.id, id: e.id,
fullName: e.fullName, fullName: e.fullName,
profileType: e.profileType, profileType: e.profileType,
checkInDate: e.checkInDate ? date2Thai(e.checkInDate) : "-", checkInDate: date2Thai(e.checkInDate),
checkInTime: e.checkInTime, checkInTime: e.checkInTime,
checkInLocation: e.checkInLocation, checkInLocation: e.checkInLocation,
checkInLat: e.checkInLat ? e.checkInLat : "", checkInLat: e.checkInLat,
checkInLon: e.checkInLon ? e.checkInLon : "", checkInLon: e.checkInLon,
checkInStatus: e.checkInStatus checkInStatus: e.checkInStatus
? workStore.convertSatatus(e.checkInStatus) ? workStore.convertSatatus(e.checkInStatus)
: "-", : "-",
checkOutDate: e.checkOutDate ? date2Thai(e.checkOutDate) : "-", checkOutDate: date2Thai(e.checkOutDate),
checkOutLocation: e.checkOutLocation ? e.checkOutLocation : "-", checkOutLocation: e.checkOutLocation,
checkOutTime: e.checkOutTime ? e.checkOutTime : "-", checkOutTime: e.checkOutTime,
checkOutLat: e.checkOutLat ? e.checkOutLat : "", checkOutLat: e.checkOutLat,
checkOutLon: e.checkOutLat ? e.checkOutLon : "", checkOutLon: e.checkOutLon,
checkOutStatus: e.checkOutStatus checkOutStatus: e.checkOutStatus
? workStore.convertSatatus(e.checkOutStatus) ? workStore.convertSatatus(e.checkOutStatus)
: "-", : "-",

View file

@ -70,6 +70,12 @@ 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",
@ -88,6 +94,12 @@ 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",
@ -133,15 +145,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 ? e.checkInTime : "-", checkInTime: e.checkInTime,
checkInLocation: e.checkInLocation ? e.checkInLocation : "-", checkInLocation: e.checkInLocation,
checkInLat: e.checkInLat ? e.checkInLat : "", checkInLat: e.checkInLat,
checkInLon: e.checkInLon ? e.checkInLon : "", checkInLon: e.checkInLon,
checkOutDate: e.checkOutDate && date2Thai(e.checkOutDate), checkOutDate: e.checkOutDate && date2Thai(e.checkOutDate),
checkOutLocation: e.checkOutLocation ? e.checkOutLocation : "-", checkOutLocation: e.checkOutLocation,
checkOutTime: e.checkOutTime ? e.checkOutTime : "-", checkOutTime: e.checkOutTime,
checkOutLat: e.checkOutLat ? e.checkOutLat : "", checkOutLat: e.checkOutLat,
checkOutLon: e.checkOutLon ? e.checkOutLon : "", checkOutLon: e.checkOutLon,
})); }));
} else { } else {
rows.value = []; rows.value = [];

View file

@ -0,0 +1,491 @@
<script setup lang="ts">
import { onMounted, reactive, ref } from "vue";
import { useQuasar } from "quasar";
import http from "@/plugins/http";
import config from "@/app.config";
import { useCounterMixin } from "@/stores/mixin";
import type { QTableProps } from "quasar";
import type { FormDataProcess } from "@/modules/09_leave/interface/request/work";
import type { DataProcess } from "@/modules/09_leave/interface/response/work";
import HeaderDialog from "@/components/DialogHeader.vue";
const $q = useQuasar();
const {
dialogConfirm,
showLoader,
hideLoader,
messageError,
success,
date2Thai,
onSearchDataTable,
convertDateToAPI,
dialogRemove,
} = useCounterMixin();
/** ข้อมูลตาราง*/
const keyword = ref<string>("");
const rowsMain = ref<DataProcess[]>([]);
const rows = ref<DataProcess[]>([]);
const columns = ref<QTableProps["columns"]>([
{
name: "no",
align: "left",
label: "ลำดับ",
sortable: false,
field: "no",
headerStyle: "font-size: 14px",
style: "font-size: 14px",
},
{
name: "startDate",
align: "left",
label: "วันเริ่มต้น",
sortable: true,
field: "startDate",
headerStyle: "font-size: 14px",
style: "font-size: 14px",
format(val, row) {
return val ? date2Thai(val) : "-";
},
},
{
name: "endDate",
align: "left",
label: "วันสิ้นสุด",
sortable: true,
field: "endDate",
headerStyle: "font-size: 14px",
style: "font-size: 14px",
format(val, row) {
return val ? date2Thai(val) : "-";
},
},
{
name: "processingDate",
align: "left",
label: "วันที่ประมวลผล",
sortable: true,
field: "processingDate",
headerStyle: "font-size: 14px",
style: "font-size: 14px",
format(val, row) {
return val ? date2Thai(val, false, true) : "-";
},
},
{
name: "completedDate",
align: "left",
label: "วันที่เสร็จสิ้น",
sortable: true,
field: "completedDate",
headerStyle: "font-size: 14px",
style: "font-size: 14px",
format(val, row) {
return val ? date2Thai(val, false, true) : "-";
},
},
{
name: "status",
align: "left",
label: "สถานะ",
sortable: true,
field: "status",
headerStyle: "font-size: 14px",
style: "font-size: 14px",
format(val, row) {
return convertStatus(val);
},
},
{
name: "createdAt",
align: "left",
label: "วันที่สร้าง",
sortable: true,
field: "createdAt",
headerStyle: "font-size: 14px",
style: "font-size: 14px",
format(val, row) {
return val ? date2Thai(val, false, true) : "-";
},
},
{
name: "createdFullName",
align: "left",
label: "ผู้สร้าง",
sortable: true,
field: "createdFullName",
headerStyle: "font-size: 14px",
style: "font-size: 14px",
},
]);
const visibleColumns = ref<string[]>([
"no",
"startDate",
"endDate",
"processingDate",
"completedDate",
"status",
"createdAt",
"createdFullName",
]);
const modal = ref<boolean>(false); // dialog
const isEdit = ref<boolean>(false); //
const editId = ref<string>(""); // id
const formData = reactive<FormDataProcess>({
startDate: null,
endDate: null,
});
/** ฟังก์ชันดึงข้อมูลจาก API */
async function fetchData() {
try {
showLoader();
// API
const res = await http.get(`${config.API.leaveTask}`);
rowsMain.value = res.data.result;
serchDataTable();
} catch (error) {
messageError($q, error);
} finally {
hideLoader();
}
}
/** ฟังก์ชันค้นหาข้อมูลในตาราง */
function serchDataTable() {
rows.value = onSearchDataTable(
keyword.value,
rowsMain.value,
columns.value ? columns.value : []
);
}
/** ฟังก์ชันเปิด dialog เพื่อเพิ่มข้อมูล */
function handleOpenDialog() {
modal.value = true;
isEdit.value = false;
}
/** ฟังก์ชันปิด dialog และรีเซ็ตข้อมูล */
function handleCloseDialog() {
modal.value = false;
formData.startDate = null;
formData.endDate = null;
editId.value = "";
}
/** ฟังก์ชันส่งข้อมูล */
function onSubmit() {
dialogConfirm($q, async () => {
showLoader();
// payload API
const payload = {
startDate: convertDateToAPI(formData.startDate),
endDate: convertDateToAPI(formData.endDate),
};
// URL method API
const url = isEdit.value
? `${config.API.leaveTask}/${editId.value}`
: config.API.leaveTask;
// method API
const method = isEdit.value ? "put" : "post";
try {
await http[method](url, payload);
success($q, "บันทึกข้อมูลสำเร็จ");
fetchData();
handleCloseDialog();
} catch (error) {
messageError($q, error);
} finally {
hideLoader();
}
});
}
/** ฟังก์ชันตรวจสอบและอัปเดตวันที่สิ้นสุดให้เป็น null หากวันที่เริ่มต้นมากกว่าวันที่สิ้นสุด*/
function updateDate() {
if (
formData.startDate &&
formData.endDate &&
new Date(formData.startDate) > new Date(formData.endDate)
) {
formData.endDate = null;
}
}
/**
* งกนแกไขขอม
* @param data อมลทองการแกไข
*/
function handleEdit(data: DataProcess) {
isEdit.value = true;
editId.value = data.id;
formData.startDate = data.startDate;
formData.endDate = data.endDate;
modal.value = true;
}
/**
* งกนลบขอม
* @param id องการลบ
*/
function handleDelete(id: string) {
dialogRemove($q, async () => {
showLoader();
try {
await http.delete(`${config.API.leaveTask}/${id}`);
success($q, "ลบข้อมูลสำเร็จ");
fetchData();
} catch (error) {
messageError($q, error);
} finally {
hideLoader();
}
});
}
/**
* งกนแปลงสถานะเปนขอความ
* @param val
*/
function convertStatus(val: string) {
switch (val) {
case "PENDING":
return "รอดำเนินการ";
case "COMPLETED":
return "ดำเนินการเสร็จสิ้น";
default:
break;
}
}
onMounted(() => {
fetchData();
});
</script>
<template>
<div class="row items-center q-gutter-x-sm q-pb-sm">
<q-btn
round
dense
flat
color="primary"
icon="add"
@click="handleOpenDialog"
>
<q-tooltip>เพมขอม</q-tooltip>
</q-btn>
<q-space />
<q-input
dense
outlined
v-model="keyword"
label="ค้นหา"
@keydown.enter.prevent="serchDataTable"
>
<template v-slot:append>
<q-icon name="search" />
</template>
</q-input>
<q-select
v-model="visibleColumns"
multiple
outlined
dense
options-dense
:display-value="$q.lang.table.columns"
emit-value
map-options
:options="columns"
option-value="name"
style="min-width: 140px"
/>
</div>
<d-table
:columns="columns"
:rows="rows"
row-key="id"
flat
bordered
:paging="true"
dense
class="custom-header-table"
:visible-columns="visibleColumns"
>
<template v-slot:header="props">
<q-tr :props="props">
<q-th auto-width />
<q-th
v-for="col in props.cols"
:key="col.name"
:props="props"
style="color: #000000; font-weight: 500"
>
<span class="text-weight-medium">{{ col.label }}</span>
</q-th>
</q-tr>
</template>
<template v-slot:body="props">
<q-tr :props="props" class="cursor-pointer">
<q-td auto-width>
<q-btn
v-if="props.row.status === 'PENDING'"
dense
flat
round
color="edit"
icon="edit"
@click.stop.prevent="handleEdit(props.row)"
>
<q-tooltip>แกไข</q-tooltip>
</q-btn>
<q-btn
v-if="props.row.status === 'PENDING'"
dense
flat
round
color="red"
icon="delete"
@click.stop.prevent="handleDelete(props.row.id)"
>
<q-tooltip>ลบ</q-tooltip>
</q-btn>
</q-td>
<q-td v-for="col in props.cols" :key="col.name" :props="props">
<div v-if="col.name == 'no'">
{{ props.rowIndex + 1 }}
</div>
<div v-else>
{{ col.value ? col.value : "-" }}
</div>
</q-td>
</q-tr>
</template>
</d-table>
<q-dialog v-model="modal" persistent>
<q-card style="width: 320px">
<q-form greedy @submit.prevent @validation-success="onSubmit">
<HeaderDialog
:tittle="'ประมวลผลการขาดราชการ/มาสาย'"
:close="handleCloseDialog"
/>
<q-separator />
<q-card-section>
<div class="row q-col-gutter-md">
<div class="col-12">
<datepicker
menu-class-name="modalfix"
v-model="formData.startDate"
:locale="'th'"
autoApply
borderless
:enableTimePicker="false"
week-start="0"
@update:model-value="updateDate"
>
<template #year="{ year }">
{{ year + 543 }}
</template>
<template #year-overlay-value="{ value }">
{{ parseInt(value + 543) }}
</template>
<template #trigger>
<q-input
ref="dateRef"
outlined
dense
hide-bottom-space
:model-value="
formData.startDate != null
? date2Thai(formData.startDate)
: null
"
label="วันที่เริ่มต้น"
:rules="[
(val:string) => !!val || `${'กรุณาเลือกวันที่เริ่มต้น'}`,
]"
>
<template v-slot:prepend>
<q-icon
name="event"
class="cursor-pointer"
style="color: var(--q-primary)"
>
</q-icon>
</template>
</q-input>
</template>
</datepicker>
</div>
<div class="col-12">
<datepicker
menu-class-name="modalfix"
v-model="formData.endDate"
:locale="'th'"
autoApply
borderless
:enableTimePicker="false"
week-start="0"
:min-date="formData.startDate"
>
<template #year="{ year }">
{{ year + 543 }}
</template>
<template #year-overlay-value="{ value }">
{{ parseInt(value + 543) }}
</template>
<template #trigger>
<q-input
ref="dateRef"
outlined
dense
hide-bottom-space
:model-value="
formData.endDate != null
? date2Thai(formData.endDate)
: null
"
label="วันที่สิ้นสุด"
:rules="[
(val:string) => !!val || `${'กรุณาเลือกวันที่สิ้นสุด'}`,
]"
>
<template v-slot:prepend>
<q-icon
name="event"
class="cursor-pointer"
style="color: var(--q-primary)"
>
</q-icon>
</template>
</q-input>
</template>
</datepicker>
</div>
</div>
</q-card-section>
<q-separator />
<q-card-actions align="right">
<q-btn color="secondary" label="บันทึก" type="submit" />
</q-card-actions>
</q-form>
</q-card>
</q-dialog>
</template>
<style scoped></style>

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"> <q-item style="padding: 0" v-if="props.row.checkInLocation">
<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,9 +131,10 @@ 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"> <q-item style="padding: 0" v-if="props.row.checkOutLocation">
<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">{{
@ -141,6 +142,7 @@ 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'">
{{ {{
@ -150,7 +152,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

@ -27,7 +27,11 @@ const {
convertDateToAPI, convertDateToAPI,
} = mixin; } = mixin;
const emit = defineEmits(["update:change-page"]); const emit = defineEmits(["update:change-page", "update:selected"]);
const isMultiple = defineModel<boolean>("isMultiple", {
default: false,
});
/**Props */ /**Props */
const props = defineProps({ const props = defineProps({
@ -40,6 +44,10 @@ const props = defineProps({
type: String, type: String,
default: "", default: "",
}, },
selectedMultiple: {
type: Array,
default: () => [],
},
}); });
/**FormData */ /**FormData */
@ -118,18 +126,43 @@ function onSubmit() {
async function changeRound() { async function changeRound() {
const formattedDateForAPI = await convertDateToAPI(formData.effectiveDate); const formattedDateForAPI = await convertDateToAPI(formData.effectiveDate);
const url = const urlAPI =
props.type == "emp" ? config.API.leaveRoundEMP() : config.API.leaveRound(); props.type == "emp" ? config.API.leaveRoundEMP() : config.API.leaveRound();
showLoader(); showLoader();
await http
.post(url, { const urlFull = isMultiple.value ? urlAPI + `/multiple ` : urlAPI;
let payload: any;
if (isMultiple.value && props.selectedMultiple.length > 0) {
payload = props.selectedMultiple.map((item: any) => ({
profileId: item.profileId,
roundId: formData.round,
effectiveDate: formattedDateForAPI,
remark: formData.reson,
firstName: item.firstName,
lastName: item.lastName,
prefix: item.prefix,
rootDnaId: item.rootDnaId,
child1DnaId: item.child1DnaId,
child2DnaId: item.child2DnaId,
child3DnaId: item.child3DnaId,
child4DnaId: item.child4DnaId,
}));
} else {
payload = {
profileId: props.personId, profileId: props.personId,
roundId: formData.round, roundId: formData.round,
effectiveDate: formattedDateForAPI, effectiveDate: formattedDateForAPI,
remark: formData.reson, remark: formData.reson,
}) };
}
await http
.post(urlFull, payload)
.then(() => { .then(() => {
success($q, "บันทึกข้อมูลเปลี่ยนรอบเวลา"); success($q, "บันทึกข้อมูลเปลี่ยนรอบเวลา");
if (isMultiple.value) {
emit("update:selected");
isMultiple.value = false;
}
props.closeDialog?.(); props.closeDialog?.();
}) })
.catch((err) => { .catch((err) => {
@ -242,10 +275,10 @@ watch(
? "เปลี่ยนรอบการปฏิบัติงาน" ? "เปลี่ยนรอบการปฏิบัติงาน"
: "ประวัติการเปลี่ยนรอบการปฏิบัติงาน" : "ประวัติการเปลี่ยนรอบการปฏิบัติงาน"
}} }}
<span class="text-teal-6">{{ <span class="text-teal-6" v-if="!isMultiple">
props.DataRow ? props.DataRow.fullName : "" {{ props.DataRow ? props.DataRow.fullName : "" }}
}}</span></q-toolbar-title </span>
> </q-toolbar-title>
<q-btn <q-btn
icon="close" icon="close"
unelevated unelevated
@ -259,7 +292,7 @@ watch(
<q-separator /> <q-separator />
<q-card-section style="max-height: 50vh" class="scroll q-pa-none"> <q-card-section style="max-height: 50vh" class="scroll q-pa-none">
<div class="q-pa-md"> <div class="q-pa-md">
<div class="row"> <div class="row" v-if="!isMultiple">
<q-icon <q-icon
name="mdi-label-variant" name="mdi-label-variant"
class="cursor-pointer self-center" class="cursor-pointer self-center"
@ -267,12 +300,12 @@ watch(
size="md" size="md"
> >
</q-icon> </q-icon>
<span class="self-center text-bold text-blue text-subtitle1" <span class="self-center text-bold text-blue text-subtitle1">
>รอบปจจ</span รอบปจจ
> </span>
<span class="self-center text-subtitle1 q-ml-sm">{{ <span class="self-center text-subtitle1 q-ml-sm">
props.DataRow ? `${props.DataRow.currentRound} น.` : "" {{ props.DataRow ? `${props.DataRow.currentRound} น.` : "" }}
}}</span> </span>
</div> </div>
<div class="row q-mt-sm q-col-gutter-sm"> <div class="row q-mt-sm q-col-gutter-sm">
<div class="col-6"> <div class="col-6">

View file

@ -82,6 +82,7 @@ const formData = reactive<FormData>({
status: "", // status: "", //
leaveLimit: 0, //() leaveLimit: 0, //()
leaveSummary: 0, //() leaveSummary: 0, //()
leaveWaitingSummary: 0, //()
leaveRemain: 0, //() leaveRemain: 0, //()
leaveWrote: "", // leaveWrote: "", //
leaveAddress: "", // leaveAddress: "", //
@ -149,21 +150,41 @@ 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.profileId); return rows.value?.commanders.map((items: SeqTypeRow) => items.keyId);
} else if (typeAdd.value == "APPROVER") { } else if (typeAdd.value == "APPROVER") {
return rows.value?.approvers.map((items: SeqTypeRow) => items.profileId); return rows.value?.approvers.map((items: SeqTypeRow) => items.keyId);
} }
}); });
// //
const commanderList = computed(() => { const commanderList = computed(() => {
if (typeAdd.value == "COMMANDER") { if (typeAdd.value === "COMMANDER") {
return rows.value?.approvers.map((items: SeqTypeRow) => items.profileId); return rows.value?.approvers.map((items: SeqTypeRow) => ({
} else if (typeAdd.value == "APPROVER") { profileId: items.profileId,
return rows.value?.commanders.map((items: SeqTypeRow) => items.profileId); isAct: items.isAct,
}));
} 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;
@ -391,6 +412,9 @@ 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,
@ -668,10 +692,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();
} }
@ -773,40 +797,50 @@ 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-gutter-md"> <div class="row col-12 q-col-gutter-md">
<div <div
v-if="formData.leaveTypeName == 'ลาพักผ่อน'" v-if="formData.leaveTypeName == 'ลาพักผ่อน'"
class="col-3" class="col-md-3 col-xs-6"
> >
<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 class="gt-xs">ได</span> <span>ได</span>
</div> </div>
</q-card> </q-card>
</div> </div>
<div class="col-3"> <div class="col-md-3 col-xs-6">
<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 class="gt-xs">ใชไป</span> <span>ใชไป</span>
</div> </div>
</q-card> </q-card>
</div> </div>
<div <div
v-if="formData.leaveTypeName == 'ลาพักผ่อน'" v-if="formData.leaveTypeName == 'ลาพักผ่อน'"
class="col-3" class="col-md-3 col-xs-6"
> >
<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 class="gt-xs">คงเหล</span> <span>คงเหล</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>
@ -1258,5 +1292,6 @@ 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,6 +151,7 @@ const formData = reactive<FormData>({
leaveSubTypeName: "", leaveSubTypeName: "",
commanderPosition: "", commanderPosition: "",
leaveRangeEnd: "", leaveRangeEnd: "",
leaveWaitingSummary: 0, //()
}); });
const isLoadData = ref<boolean>(false); const isLoadData = ref<boolean>(false);
@ -217,6 +218,9 @@ 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;
@ -626,40 +630,50 @@ 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-gutter-md"> <div class="row col-12 q-col-gutter-md">
<div <div
v-if="formData.leaveTypeName == 'ลาพักผ่อน'" v-if="formData.leaveTypeName == 'ลาพักผ่อน'"
class="col-3" class="col-md-3 col-xs-6"
> >
<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 class="gt-xs">ได</span> <span>ได</span>
</div> </div>
</q-card> </q-card>
</div> </div>
<div class="col-3"> <div class="col-md-3 col-xs-6">
<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 class="gt-xs">ใชไป</span> <span>ใชไป</span>
</div> </div>
</q-card> </q-card>
</div> </div>
<div <div
v-if="formData.leaveTypeName == 'ลาพักผ่อน'" v-if="formData.leaveTypeName == 'ลาพักผ่อน'"
class="col-3" class="col-md-3 col-xs-6"
> >
<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 class="gt-xs">คงเหล</span> <span>คงเหล</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,7 +4,6 @@ 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";
@ -30,6 +29,7 @@ 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.id); return props.idCheck?.some((i: any) => i === items.key);
}); });
}) })
.catch((err) => { .catch((err) => {
@ -153,10 +153,14 @@ 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((i: any) => i === e.id) props.commandersList?.some(
(i: any) => i.profileId === e.id && i.isAct === isAct.value
)
); );
if (hasCommander) { if (hasCommander) {
@ -205,6 +209,7 @@ watch(
() => modal.value, () => modal.value,
() => { () => {
if (modal.value) { if (modal.value) {
isAct.value = props.commandersIsAct ?? false;
getSearch(); getSearch();
} }
} }
@ -233,6 +238,7 @@ 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>
@ -312,9 +318,7 @@ watch(
:key="col.name" :key="col.name"
:props="props" :props="props"
> >
<span class="text-weight-medium">{{ <span class="text-weight-medium">{{ col.label }}</span>
getColumnLabel(col, isAct)
}}</span>
</q-th> </q-th>
</q-tr> </q-tr>
</template> </template>

View file

@ -10,6 +10,7 @@ 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";
@ -104,10 +105,18 @@ 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: formData.leaveDaysUsed leaveDaysUsed:
? Number(formData.leaveDaysUsed) checkPermission(route)?.attrOwnership === "OWNER"
: 0, ? formData.leaveDaysUsed
leaveCount: formData.leaveCount ? Number(formData.leaveCount) : 0, ? Number(formData.leaveDaysUsed)
: 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,
@ -265,7 +274,6 @@ watch(modal, async (val) => {
await Promise.all([ await Promise.all([
filterLeaveTypeData(), filterLeaveTypeData(),
isStatusEdit.value && defineDataLeaveBeginning(rowData.value), isStatusEdit.value && defineDataLeaveBeginning(rowData.value),
console.log(rowData.value),
]); ]);
} finally { } finally {
hideLoader(); hideLoader();
@ -314,6 +322,7 @@ 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
@ -387,7 +396,7 @@ watch(modal, async (val) => {
<q-separator vertical /> <q-separator vertical />
<!-- input --> <!-- input -->
<div class="col overflow-hidden q-pa-md"> <div class="col overflow-auto 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
@ -398,6 +407,7 @@ 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 }">{{
@ -454,7 +464,7 @@ watch(modal, async (val) => {
</template> </template>
</q-select> </q-select>
</div> </div>
<div class="col-12"> <div v-show="leaveTypeCode === 'LV-005'" 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'"
@ -462,31 +472,8 @@ watch(modal, async (val) => {
dense dense
outlined outlined
label="จำนวนสิทธิ์การลา" label="จำนวนสิทธิ์การลา"
hide-bottom-space hint="* สำหรับลาพักผ่อนเท่านั้น คือสิทธิ์ลาประจำปี + สิทธิ์สะสม"
/> :rules="[(val: string) => !val || /^\d+(\.\d*)?$/.test(val) || 'กรุณากรอกเฉพาะตัวเลข']"
</div>
<div class="col-12">
<q-input
:class="classInput(true)"
v-model="formData.leaveDaysUsed"
dense
outlined
label="ที่ใช้ไป (วัน)"
hide-bottom-space
mask="#"
reverse-fill-mask
/>
</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>
@ -496,10 +483,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) || 'กรุณากรอกเฉพาะตัวเลข']"
mask="#" hint="* จำนวนวันรวมการลาในปีงบประมาณนี้ที่เกิดขึ้นก่อนเริ่มใช้ระบบ"
reverse-fill-mask
/> />
</div> </div>
@ -509,10 +495,38 @@ 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
mask="#" :rules="[(val: string) => !val || /^\d+(\.\d*)?$/.test(val) || 'กรุณากรอกเฉพาะตัวเลข']"
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,6 +4,8 @@ interface DataPost {
lastName: string; lastName: string;
page: number; page: number;
pageSize: number; pageSize: number;
selectedNodeId?: string | null;
selectedNode?: string;
} }
interface DataOption { interface DataOption {

View file

@ -41,6 +41,7 @@ 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

@ -0,0 +1,6 @@
interface FormDataProcess {
startDate: Date | null;
endDate: Date | null;
}
export type { FormDataProcess };

View file

@ -50,6 +50,8 @@ 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

@ -85,4 +85,24 @@ interface FormDetail {
checkInLocationName: string; checkInLocationName: string;
checkOutLocationName: string; checkOutLocationName: string;
} }
export type { TableRows, DataResLog, DataResTime, TableRowsTime, FormDetail };
interface DataProcess {
id: string;
createdFullName: string;
createdAt: Date | null;
status: string;
startDate: Date | null;
endDate: Date | null;
processingDate: Date | null;
completedDate: Date | null;
errorMessage: string | null;
}
export type {
TableRows,
DataResLog,
DataResTime,
TableRowsTime,
FormDetail,
DataProcess,
};

View file

@ -125,7 +125,7 @@ export const useChangeRoundDataStore = defineStore(
async function fetchDataForCardId(dataDetail: any, type?: string) { async function fetchDataForCardId(dataDetail: any, type?: string) {
if (dataDetail) { if (dataDetail) {
showLoader(); // showLoader();
const url = const url =
type && type == "emp" type && type == "emp"
? config.API.leaveSearchEMP() ? config.API.leaveSearchEMP()
@ -138,6 +138,8 @@ export const useChangeRoundDataStore = defineStore(
page: dataDetail.page, //หน้า page: dataDetail.page, //หน้า
pageSize: dataDetail.pageSize || 10, //จำนวนแถวต่อหน้า pageSize: dataDetail.pageSize || 10, //จำนวนแถวต่อหน้า
keyword: dataDetail.keyword || "", //keyword ค้นหา keyword: dataDetail.keyword || "", //keyword ค้นหา
selectedNodeId: dataDetail.selectedNodeId, //id ต้นไม้ที่เลือก
selectedNode: dataDetail.selectedNode, //ระดับต้นไม้ที่เลือก
}) })
.then((res) => { .then((res) => {
const apiData = res.data.result.data; const apiData = res.data.result.data;
@ -148,6 +150,7 @@ export const useChangeRoundDataStore = defineStore(
if (apiData.length > 0) { if (apiData.length > 0) {
checkCilck.value = false; checkCilck.value = false;
rows.value = apiData.map((e: any) => ({ rows.value = apiData.map((e: any) => ({
...e,
profileId: e.profileId, profileId: e.profileId,
cardId: e.citizenId, cardId: e.citizenId,
fullName: e.fullName, fullName: e.fullName,
@ -167,7 +170,7 @@ export const useChangeRoundDataStore = defineStore(
console.log(e); console.log(e);
}) })
.finally(() => { .finally(() => {
hideLoader(); // hideLoader();
}); });
} }
} }

View file

@ -11,7 +11,14 @@ export const useSpecialTimeStore = defineStore("LeaveSpecialTime", () => {
{ id: "NOT_COMPLETE", name: "ปฏิบัติงานไม่ครบตามกำหนดเวลา" }, { id: "NOT_COMPLETE", name: "ปฏิบัติงานไม่ครบตามกำหนดเวลา" },
]); ]);
// convertSatatus const optionStatusMain = ref<DataOption[]>([
{ 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) {
@ -29,5 +36,6 @@ export const useSpecialTimeStore = defineStore("LeaveSpecialTime", () => {
return { return {
optionStatus, optionStatus,
convertStatus, convertStatus,
optionStatusMain,
}; };
}); });

View file

@ -1,15 +1,26 @@
<script setup lang="ts"> <script setup lang="ts">
import { ref, toRefs } from "vue"; import { computed, ref, toRefs } from "vue";
import { useWorklistDataStore } from "@/modules/09_leave/stores/WorkStore"; import { useWorklistDataStore } from "@/modules/09_leave/stores/WorkStore";
import { checkPermission } from "@/utils/permissions";
import { useRoute } from "vue-router";
/** import Components */ /** import Components */
import Tab1 from "@/modules/09_leave/components/02_WorkList/Tab1.vue"; import Tab1 from "@/modules/09_leave/components/02_WorkList/Tab1.vue";
import Tab2 from "@/modules/09_leave/components/02_WorkList/Tab2.vue"; import Tab2 from "@/modules/09_leave/components/02_WorkList/Tab2.vue";
import Tab3 from "@/modules/09_leave/components/02_WorkList/Tab3_Processed_Late.vue";
const stores = useWorklistDataStore(); const stores = useWorklistDataStore();
const route = useRoute();
const { tabs } = toRefs(stores); const { tabs } = toRefs(stores);
const isPermissionTab3 = computed(() => {
return (
checkPermission(route)?.attrOwnership === "OWNER" ||
(checkPermission(route)?.attrOwnership === "STAFF" &&
checkPermission(route)?.attrIsUpdate)
);
});
</script> </script>
<template> <template>
@ -30,6 +41,12 @@ const { tabs } = toRefs(stores);
> >
<q-tab name="1" label="รายการลงเวลาที่ประมวลผลแล้ว" /> <q-tab name="1" label="รายการลงเวลาที่ประมวลผลแล้ว" />
<q-tab name="2" label="รายการลงเวลา" /> <q-tab name="2" label="รายการลงเวลา" />
<q-tab
v-if="isPermissionTab3"
name="3"
label="ประมวลผลการขาดราชการ/มาสาย"
/>
<!-- เพมแทบใหม -->
</q-tabs> </q-tabs>
<q-separator /> <q-separator />
@ -42,6 +59,9 @@ const { tabs } = toRefs(stores);
<q-tab-panel name="2"> <q-tab-panel name="2">
<Tab2 /> <Tab2 />
</q-tab-panel> </q-tab-panel>
<q-tab-panel name="3">
<Tab3 />
</q-tab-panel>
</q-tab-panels> </q-tab-panels>
</div> </div>
</q-card> </q-card>

View file

@ -1,5 +1,5 @@
<script setup lang="ts"> <script setup lang="ts">
import { ref, watch, reactive, onMounted } from "vue"; import { ref, watch, reactive, onMounted, nextTick } from "vue";
import { useQuasar } from "quasar"; import { useQuasar } from "quasar";
import http from "@/plugins/http"; import http from "@/plugins/http";
@ -7,22 +7,20 @@ import config from "@/app.config";
import { useCounterMixin } from "@/stores/mixin"; import { useCounterMixin } from "@/stores/mixin";
import { useChangeRoundDataStore } from "@/modules/09_leave/stores/ChangeRoundStore"; import { useChangeRoundDataStore } from "@/modules/09_leave/stores/ChangeRoundStore";
import { checkPermission } from "@/utils/permissions"; import { checkPermission } from "@/utils/permissions";
import { useStructureTree } from "@/stores/structureTree";
import { useRoute } from "vue-router";
import type { DataPost } from "@/modules/09_leave/interface/request/changeRound"; import type { DataPost } from "@/modules/09_leave/interface/request/changeRound";
import Dialogform from "@/modules/09_leave/components/03_ChangeRound/DialogForm.vue"; import Dialogform from "@/modules/09_leave/components/03_ChangeRound/DialogForm.vue";
/** useStore */ /** useStore */
const route = useRoute();
const mixin = useCounterMixin(); const mixin = useCounterMixin();
const {
showLoader, const { showLoader, hideLoader, success, messageError, dialogConfirm } = mixin;
hideLoader,
success,
messageError,
dialogMessageNotify,
dialogConfirm,
} = mixin;
const dataStore = useChangeRoundDataStore(); const dataStore = useChangeRoundDataStore();
const { fetchStructureTree } = useStructureTree();
/** use */ /** use */
const $q = useQuasar(); const $q = useQuasar();
@ -37,9 +35,42 @@ const formData = reactive<DataPost>({
firstName: "", firstName: "",
lastName: "", lastName: "",
page: 1, page: 1,
pageSize: 10, pageSize: 100,
selectedNodeId: null,
selectedNode: "",
}); });
const pagination = ref({
page: 1,
rowsPerPage: 0,
});
/** โครงสร้างข้อมูลต้นไม้ขององค์กร **/
const nodeTree = ref<any[]>([]);
const expanded = ref<string[]>([]);
const orgTreeId = ref<string | null>(null);
const filter = ref<string>("");
const selected = ref<any[]>([]);
const isMultiple = ref<boolean>(false);
/** client-side data & batch loading **/
const allRows = ref<any[]>([]);
const isLoadingAll = ref<boolean>(false);
const loadAllProgress = ref<number>(0);
const totalToLoad = ref<number | null>(null);
const BATCH_SIZE = 250;
function waitForUi() {
return new Promise((resolve) => {
setTimeout(resolve, 0);
});
}
/** function fetch ข้อมูลของ Tree*/
async function fetchDataTree() {
nodeTree.value = await fetchStructureTree(route.meta.Key as string, true);
}
/** /**
* Function openPopup * Function openPopup
* @param check action edit,history * @param check action edit,history
@ -87,6 +118,7 @@ async function OpenmodalFix(detail: any) {
function closeDialog() { function closeDialog() {
modal.value = false; modal.value = false;
modalFix.value = false; modalFix.value = false;
isMultiple.value = false;
} }
function save() { function save() {
@ -110,214 +142,416 @@ function save() {
}); });
} }
/** /** Function โหลดข้อมูลทั้งหมดแบบ batch แล้วให้ QTable ทำ pagination เอง */
* function updatePagination async function fetchAllData() {
* @param newPagination อม Pagination ใหม isLoadingAll.value = true;
*/ allRows.value = [];
function updatePagination(newPagination: any) { loadAllProgress.value = 0;
formData.pageSize = newPagination.rowsPerPage; totalToLoad.value = null;
selected.value = [];
// UI API
await nextTick();
await waitForUi();
try {
await dataStore.fetchDataForCardId({
...formData,
page: 1,
pageSize: BATCH_SIZE,
});
allRows.value = dataStore.rows.slice();
const total = dataStore.totalListMain;
totalToLoad.value = total;
const totalPages = Math.ceil(total / BATCH_SIZE);
loadAllProgress.value =
totalPages <= 1 ? 100 : Math.round(100 / totalPages);
await nextTick();
for (let page = 2; page <= totalPages; page++) {
await dataStore.fetchDataForCardId({
...formData,
page,
pageSize: BATCH_SIZE,
});
allRows.value.push(...dataStore.rows);
loadAllProgress.value = Math.round((page / totalPages) * 100);
await nextTick();
await waitForUi();
}
} finally {
isLoadingAll.value = false;
totalToLoad.value = null;
}
} }
/** Function ค้นหาข้อมูล */ /** Function ค้นหาข้อมูล */
async function searchData() { async function searchData() {
if (formData.cardId || formData.firstName || formData.lastName) { await fetchAllData();
await dataStore.fetchDataForCardId(formData); }
} else {
dialogMessageNotify($q, "กรุณากรอกข้อมูลอย่างน้อย 1 ช่อง"); function submitSearchByEnter() {
} if (isLoadingAll.value) return;
if (!checkPermission(route)?.attrIsGet) return;
formData.page = 1;
searchData();
}
/** Function เลือกทั้งหมด */
function selectAllRows() {
selected.value = [...allRows.value];
}
function onSelectedOrgTree(data: any) {
if (isLoadingAll.value) return;
selected.value = [];
allRows.value = [];
orgTreeId.value = data.orgTreeId;
formData.selectedNodeId = data.orgTreeDnaId;
formData.selectedNode = data.orgLevel;
formData.page = 1;
}
function handleSelectMultiple() {
modal.value = true;
isMultiple.value = true;
editCheck.value = "edit";
}
function resetSelected() {
selected.value = [];
} }
watch( watch(
() => formData.pageSize, () => formData.pageSize,
() => { () => {
formData.page = 1; // pageSize QTable (client-side) fetch API
searchData();
} }
); );
onMounted(() => { onMounted(() => {
fetchDataTree();
dataStore.rows = []; dataStore.rows = [];
allRows.value = [];
}); });
</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-pa-md">
<div class="row col-12 q-mb-sm"> <q-card>
<q-card flat bordered class="bg-grey-2 col-12 bg-white q-pa-lg"> <q-card-section :horizontal="$q.screen.gt.xs">
<div class="text-dark col-12 text-weight-bold text-subtitle1"> <q-card-section class="col-lg-3 col-md-4 col-xs-12 q-gutter-sm">
นหารายช <div>
<q-input dense outlined v-model="filter" label="ค้นหา">
<template v-slot:append>
<q-icon name="search" />
</template>
</q-input>
</div> </div>
<div class="row justify-between q-gutter-y-sm"> <div class="bg-white tree-container q-pa-xs">
<q-input <q-tree
:readonly="!checkPermission($route)?.attrIsGet" class="q-pa-sm q-gutter-sm"
v-model="formData.cardId"
outlined
label="เลขประจำตัวประชาชน"
class="col-6 col-md-4 bg-white inputgreen"
dense dense
hide-bottom-space :nodes="nodeTree"
maxlength="13" node-key="orgTreeId"
/> label-key="labelName"
<q-input :filter="filter.trim()"
:readonly="!checkPermission($route)?.attrIsGet" no-results-label="ไม่พบข้อมูลที่ค้นหา"
v-model="formData.firstName" no-nodes-label="ไม่มีข้อมูล"
outlined v-model:expanded="expanded"
label="ชื่อ" >
class="col-5 col-md-3 bg-white inputgreen" <template v-slot:default-header="prop">
dense <q-item
hide-bottom-space clickable
/> :active="orgTreeId == prop.node.orgTreeId"
<q-input @click.stop="onSelectedOrgTree(prop.node)"
:readonly="!checkPermission($route)?.attrIsGet" active-class="my-list-link text-primary text-weight-medium"
v-model="formData.lastName" class="row col-12 text-dark items-center q-py-xs q-pl-sm rounded-borders my-list"
outlined >
label="นามสกุล" <div>
class="col-6 col-md-3 bg-white inputgreen" <div class="text-weight-medium">
dense {{ prop.node.orgTreeName }}
hide-bottom-space </div>
/> <div class="text-weight-light text-grey-8">
<q-btn {{ prop.node.orgCode == null ? null : prop.node.orgCode }}
v-if="checkPermission($route)?.attrIsGet" {{
@click="(formData.page = 1), searchData()" prop.node.orgTreeShortName == null
for="#search" ? null
dense : prop.node.orgTreeShortName
unelevated }}
color="primary" </div>
class="q-px-sm col-5 col-md-1" </div>
style="max-height: 40px" </q-item>
>นหา</q-btn </template>
</q-tree>
</div>
</q-card-section>
<q-separator :vertical="$q.screen.gt.xs" />
<q-card-section
class="col-lg-9 col-md-8 col-xs-12 q-gutter-sm scroll"
style="height: 80vh"
>
<div class="row col-xs-12 col-sm-9">
<q-card flat bordered class="bg-grey-2 col-12 bg-white q-pa-lg">
<div class="text-dark col-12 text-weight-bold text-subtitle1">
นหารายช
</div>
<div class="row justify-between q-gutter-y-sm">
<q-input
:readonly="!checkPermission($route)?.attrIsGet || isLoadingAll"
v-model="formData.cardId"
outlined
label="เลขประจำตัวประชาชน"
class="col-6 col-md-4 bg-white inputgreen"
dense
hide-bottom-space
maxlength="13"
@keyup.enter="submitSearchByEnter"
/>
<q-input
:readonly="!checkPermission($route)?.attrIsGet || isLoadingAll"
v-model="formData.firstName"
outlined
label="ชื่อ"
class="col-5 col-md-3 bg-white inputgreen"
dense
hide-bottom-space
@keyup.enter="submitSearchByEnter"
/>
<q-input
:readonly="!checkPermission($route)?.attrIsGet || isLoadingAll"
v-model="formData.lastName"
outlined
label="นามสกุล"
class="col-6 col-md-3 bg-white inputgreen"
dense
hide-bottom-space
@keyup.enter="submitSearchByEnter"
/>
<q-btn
v-if="checkPermission($route)?.attrIsGet"
@click="(formData.page = 1), searchData()"
:disable="isLoadingAll"
for="#search"
dense
unelevated
color="primary"
class="q-px-sm col-5 col-md-1"
style="max-height: 40px"
>นหา</q-btn
>
</div>
</q-card>
</div>
<div
v-if="
allRows.length === 0 &&
dataStore.checkCilck === true &&
!isLoadingAll
"
>
<q-card
flat
bordered
class="bg-grey-2 col-12 q-pa-lg text-center text-subtitle1 text-bold"
>ไมพบขอม</q-card
> >
</div> </div>
</q-card> <div v-if="allRows.length !== 0 || isLoadingAll" class="col-12">
</div> <q-banner
<div v-if="dataStore.rows.length === 0 && dataStore.checkCilck === true"> v-if="isLoadingAll"
<q-card rounded
flat class="bg-blue-1 text-primary q-mb-sm"
bordered >
class="bg-grey-2 col-12 q-pa-lg text-center text-subtitle1 text-bold" <div class="row items-center q-gutter-sm">
>ไมพบขอม</q-card <q-spinner color="primary" size="22px" />
> <div v-if="totalToLoad === null || totalToLoad <= 500">
</div> กำลงคนหาขอม กรณารอสกคร
<div v-if="dataStore.rows.length !== 0" class="col-12 q-mt-xl"> </div>
<d-table <div v-else>
ref="table" กำลงโหลดขอมลจำนวนมาก กรณารอสกคร
:columns="dataStore.columns" <div class="text-caption text-grey-7">
:rows="dataStore.rows" ระบบกำลงดงขอมลทงหมด
row-key="interrogated" {{ totalToLoad.toLocaleString() }} รายการ
flat </div>
bordered </div>
dense </div>
class="custom-header-table" </q-banner>
:visible-columns="dataStore.visibleColumns" <q-linear-progress
:rows-per-page-options="[10, 25, 50, 100]" v-if="isLoadingAll && totalToLoad === null"
@update:pagination="updatePagination" indeterminate
> color="primary"
<!-- :paging="true" --> class="q-mb-xs"
<template v-slot:header="props"> />
<q-tr :props="props"> <q-linear-progress
<q-th auto-width /> v-if="isLoadingAll && totalToLoad !== null"
<q-th :value="loadAllProgress / 100"
v-for="col in props.cols" color="primary"
:key="col.name" class="q-mb-xs"
:props="props" />
style="color: #000000; font-weight: 500" <div
> v-if="isLoadingAll && totalToLoad !== null"
<span class="text-weight-medium">{{ col.label }}</span> class="text-caption text-grey-6 q-mb-xs"
</q-th> >
</q-tr> กำลงโหลด... {{ allRows.length.toLocaleString() }} /
</template> {{ totalToLoad.toLocaleString() }} รายการ
<template v-slot:body="props"> </div>
<q-tr :props="props"> <div class="row justify-between items-center q-mb-sm">
<q-td> <div class="row q-gutter-sm">
<div> <!-- <q-btn
<q-btn color="secondary"
flat dense
icon="mdi-dots-horizontal-circle-outline" icon="mdi-checkbox-multiple-marked-outline"
color="secondary" label="เลือกทั้งหมด"
for="#cancel" :disable="isLoadingAll || allRows.length === 0"
dense @click="selectAllRows()"
round /> -->
<q-btn
v-if="checkPermission($route)?.attrIsUpdate"
:disable="selected.length === 0 || isLoadingAll"
:color="selected.length === 0 ? 'grey' : 'info'"
dense
icon="mdi-shuffle-variant"
:label="`เปลี่ยนรอบการลงเวลา${
selected.length > 0 ? ` (${selected.length})` : ''
}`"
@click="handleSelectMultiple()"
/>
</div>
</div>
<d-table
ref="table"
:columns="dataStore.columns"
:rows="allRows"
row-key="profileId"
flat
bordered
dense
virtual-scroll
table-style="max-height: 58vh"
class="custom-header-table"
:visible-columns="dataStore.visibleColumns"
hide-pagination
v-model:pagination="pagination"
selection="multiple"
v-model:selected="selected"
>
<template v-slot:header="props">
<q-tr :props="props">
<q-th auto-width>
<q-checkbox
keep-color
color="primary"
dense
:disable="isLoadingAll"
v-model="props.selected"
/>
</q-th>
<q-th auto-width />
<q-th
v-for="col in props.cols"
:key="col.name"
:props="props"
style="color: #000000; font-weight: 500"
> >
<q-menu> <span class="text-weight-medium">{{ col.label }}</span>
<q-list> </q-th>
<q-item </q-tr>
v-if="checkPermission($route)?.attrIsUpdate" </template>
clickable <template v-slot:body="props">
v-close-popup <q-tr :props="props">
@click="Openmodal('edit', props.row)" <q-td class="text-center">
> <q-checkbox
<q-item-section style="min-width: 0px" avatar> keep-color
<q-icon color="primary"
color="primary" dense
name="mdi-shuffle-variant" :disable="isLoadingAll"
size="xs" v-model="props.selected"
/> />
</q-item-section> </q-td>
<q-td>
<div>
<q-btn
flat
icon="mdi-dots-horizontal-circle-outline"
color="secondary"
for="#cancel"
dense
round
>
<q-menu>
<q-list>
<q-item
v-if="checkPermission($route)?.attrIsUpdate"
clickable
v-close-popup
@click="Openmodal('edit', props.row)"
>
<q-item-section style="min-width: 0px" avatar>
<q-icon
color="primary"
name="mdi-shuffle-variant"
size="xs"
/>
</q-item-section>
<q-item-section> <q-item-section>
<q-item-label>เปลยนรอบการลงเวลา</q-item-label> <q-item-label>เปลยนรอบการลงเวลา</q-item-label>
</q-item-section> </q-item-section>
</q-item> </q-item>
<q-item <q-item
v-if="checkPermission($route)?.attrIsUpdate" v-if="checkPermission($route)?.attrIsUpdate"
clickable clickable
v-close-popup v-close-popup
@click="OpenmodalFix(props.row)" @click="OpenmodalFix(props.row)"
> >
<q-item-section style="min-width: 0px" avatar> <q-item-section style="min-width: 0px" avatar>
<q-icon color="edit" name="edit" size="xs" /> <q-icon color="edit" name="edit" size="xs" />
</q-item-section> </q-item-section>
<q-item-section> <q-item-section>
<q-item-label>แกไขปฏนวนทำงาน</q-item-label> <q-item-label>แกไขปฏนวนทำงาน</q-item-label>
</q-item-section> </q-item-section>
</q-item> </q-item>
<q-item <q-item
clickable clickable
v-close-popup v-close-popup
@click="Openmodal('history', props.row)" @click="Openmodal('history', props.row)"
> >
<q-item-section style="min-width: 0px" avatar> <q-item-section style="min-width: 0px" avatar>
<q-icon <q-icon
color="deep-purple" color="deep-purple"
name="mdi-history" name="mdi-history"
size="xs" size="xs"
/> />
</q-item-section> </q-item-section>
<q-item-section> <q-item-section>
<q-item-label>ประวการเปลยนรอบ</q-item-label> <q-item-label>ประวการเปลยนรอบ</q-item-label>
</q-item-section> </q-item-section>
</q-item> </q-item>
</q-list> </q-list>
</q-menu> </q-menu>
</q-btn> </q-btn>
</div> </div>
</q-td> </q-td>
<q-td v-for="col in props.cols" :key="col.name" :props="props"> <q-td v-for="col in props.cols" :key="col.name" :props="props">
<div> <div>
{{ col.value ?? "-" }} {{ col.value ?? "-" }}
</div> </div>
</q-td> </q-td>
</q-tr> </q-tr>
</template> </template>
<template v-slot:pagination="scope"> </d-table>
งหมด {{ dataStore.totalListMain }} รายการ <div class="text-caption text-grey-7 q-mt-sm">
<q-pagination งหมด {{ allRows.length }} รายการ
v-model="formData.page" </div>
active-color="primary" </div>
color="dark" </q-card-section>
:max="Number(dataStore.maxPageMain)" </q-card-section>
size="sm"
boundary-links
direction-links
:max-pages="5"
@update:model-value="dataStore.fetchDataForCardId(formData)"
></q-pagination>
</template>
</d-table>
</div>
</q-card> </q-card>
<!-- popup เปลยนรอบการปฏงาน ,ประวการเปลยนรอบการปฏงาน --> <!-- popup เปลยนรอบการปฏงาน ,ประวการเปลยนรอบการปฏงาน -->
@ -328,6 +562,9 @@ onMounted(() => {
:DataRow="DataRow" :DataRow="DataRow"
:personId="DataRow == null ? '' : DataRow.profileId" :personId="DataRow == null ? '' : DataRow.profileId"
@update:change-page="dataStore.changePage" @update:change-page="dataStore.changePage"
v-model:isMultiple="isMultiple"
:selectedMultiple="selected"
@update:selected="resetSelected"
/> />
<!-- แกไขปฏนวนทำงาน --> <!-- แกไขปฏนวนทำงาน -->
@ -384,4 +621,19 @@ onMounted(() => {
.q-table tbody td:before.no-background { .q-table tbody td:before.no-background {
background: none; background: none;
} }
.tree-container {
overflow: auto;
height: 75vh;
border: 1px solid #e6e6e7;
border-radius: 10px;
}
.my-list-link {
color: rgb(118, 168, 222);
border-radius: 5px;
background: #a3d3fb48 !important;
font-weight: 600;
border: 1px solid rgba(175, 185, 196, 0.217);
}
</style> </style>

View file

@ -10,8 +10,10 @@ 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 { DataDateMonthObject } from "@/modules/09_leave/interface/request/specialTime"; import type {
import type { DataSpecialTime } from "@/modules/09_leave/interface/index/Main"; DataSpecialTime,
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";
@ -21,23 +23,17 @@ 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>("");
/**ตัวแปรที่ใช้ */ /**ตัวแปรที่ใช้ */
@ -51,16 +47,12 @@ 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",
@ -144,9 +136,10 @@ 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) => {
@ -246,40 +239,43 @@ 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>
@ -291,25 +287,31 @@ 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-3 col-md-2"> <div class="col-xs-12 col-sm-6 col-md-3">
<datepicker <datepicker
v-model="dateMonth" v-model="filterDate"
:locale="'th'" :locale="'th'"
autoApply autoApply
month-picker borderless
range
:enableTimePicker="false" :enableTimePicker="false"
@update:modelValue="updateMonth" week-start="0"
@update:modelValue="onSearchData()"
> >
<template #year="{ year }">{{ year + 543 }}</template> <template #year="{ year }">
<template #year-overlay-value="{ value }">{{ {{ year + 543 }}
parseInt(value + 543) </template>
}}</template> <template #year-overlay-value="{ value }">
{{ parseInt(value + 543) }}
</template>
<template #trigger> <template #trigger>
<q-input <q-input
:model-value="monthYearThai(dateMonth)"
dense
outlined outlined
hide-bottom-space dense
class="full-width datepicker"
:model-value="
filterDate != null ? dateThaiRange(filterDate) : null
"
> >
<template v-slot:prepend> <template v-slot:prepend>
<q-icon <q-icon
@ -323,9 +325,37 @@ 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 /> <q-space v-if="!$q.screen.xs && !$q.screen.sm" />
<div class="col-xs-12 col-sm-3 col-md-2"> <div class="col-xs-12 col-sm-6 col-md-2">
<q-input <q-input
standout standout
dense dense
@ -340,7 +370,7 @@ onMounted(async () => {
</q-input> </q-input>
</div> </div>
<div class="col-xs-12 col-sm-3 col-md-2"> <div class="col-xs-12 col-sm-6 col-md-2">
<q-select <q-select
v-model="visibleColumns" v-model="visibleColumns"
multiple multiple

View file

@ -269,6 +269,7 @@ async function fetchLeaveday(
* @param data อมลบญชนลา * @param data อมลบญชนลา
*/ */
async function fetchDocumentTemplate(data: any) { async function fetchDocumentTemplate(data: any) {
if (typeReport.value === 4) return;
await axios await axios
.post(`${config.API.reportTemplate}/xlsx`, data, { .post(`${config.API.reportTemplate}/xlsx`, data, {
headers: { headers: {
@ -352,7 +353,7 @@ function clearData() {
}; };
} }
function onSearch() { async function onSearch() {
isReport.value = false; isReport.value = false;
isLoadPDF.value = true; isLoadPDF.value = true;
pdfSrc.value = undefined; pdfSrc.value = undefined;
@ -467,6 +468,19 @@ const reportName = () => {
return reportNameVal + employeeClassName; return reportNameVal + employeeClassName;
}; };
async function handleDownload() {
updateLeaveday();
await fetchLeaveday(
employeeClass.value,
typeReport.value == 3 || typeReport.value == 4
? leaveType.value
: yearType.value,
dateStart.value,
dateEnd.value
);
await genReportXLSX(detailReport.value, `${reportName()}`);
}
onMounted(() => { onMounted(() => {
fetchDataTree(); fetchDataTree();
}); });
@ -512,7 +526,11 @@ onMounted(() => {
round round
color="primary" color="primary"
icon="download" icon="download"
v-if="checkPermission($route)?.attrIsGet && typeReport !== 3" v-if="
checkPermission($route)?.attrIsGet &&
typeReport !== 3 &&
typeReport !== 4
"
> >
<q-menu> <q-menu>
<q-list style="min-width: 150px"> <q-list style="min-width: 150px">
@ -547,8 +565,11 @@ onMounted(() => {
round round
color="primary" color="primary"
icon="download" icon="download"
v-if="checkPermission($route)?.attrIsGet && typeReport == 3" v-if="
@click="getReport()" checkPermission($route)?.attrIsGet &&
(typeReport == 3 || typeReport == 4)
"
@click="typeReport == 3 ? getReport() : handleDownload()"
> >
</q-btn> </q-btn>
</div> </div>
@ -957,7 +978,7 @@ onMounted(() => {
<q-separator /> <q-separator />
<q-card-actions align="right"> <q-card-actions align="right">
<q-btn <q-btn
v-if="typeReport !== 3" v-if="typeReport !== 3 && typeReport !== 4"
dense dense
class="q-px-md" class="q-px-md"
label="ค้นหา" label="ค้นหา"
@ -976,7 +997,7 @@ onMounted(() => {
</div> </div>
<div class="col-lg-9 col-md-9 col-sm-9 col-xs-12 col-xs-12 flex"> <div class="col-lg-9 col-md-9 col-sm-9 col-xs-12 col-xs-12 flex">
<q-splitter <q-splitter
v-if="typeReport !== 3" v-if="typeReport !== 3 && typeReport !== 4"
disable disable
v-model="splitterModel" v-model="splitterModel"
horizontal horizontal

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,6 +2,7 @@
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";
@ -22,21 +23,12 @@ const checkRoutePermisson = ref<boolean>(
route.name == "disciplineInvestigatefactsDetail" route.name == "disciplineInvestigatefactsDetail"
); );
const props = defineProps({ const props = defineProps<{
rows: { rows?: any[];
type: Array, columns?: QTableColumn[];
default: [], visibleColumns?: string[];
}, 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[]>([]);
@ -100,7 +92,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="rows" :rows="props.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";

View file

@ -15,7 +15,7 @@ import type {
FormData, FormData,
Director, Director,
PersonsArray, PersonsArray,
} from "@/modules/11_discipline/interface/request/Disciplinary"; } from "@/modules/11_discipline/interface/request/disciplinary";
import type { import type {
DataOption, DataOption,
FileLists, FileLists,

View file

@ -260,7 +260,7 @@ watch(
keep-color keep-color
color="primary" color="primary"
dense dense
:disable="commandType" :disable="commandType === ''"
v-model="scope.selected" v-model="scope.selected"
/> />
</template> </template>
@ -271,7 +271,7 @@ watch(
keep-color keep-color
color="primary" color="primary"
dense dense
:disable="commandType" :disable="commandType === ''"
v-model="props.selected" v-model="props.selected"
/> />
</q-td> </q-td>

View file

@ -9,13 +9,13 @@ import { useCounterMixin } from "@/stores/mixin";
import { useDisciplineResultStore } from "@/modules/11_discipline/store/ResultStore"; import { useDisciplineResultStore } from "@/modules/11_discipline/store/ResultStore";
import { useDisciplineMainStore } from "@/modules/11_discipline/store/Main"; import { useDisciplineMainStore } from "@/modules/11_discipline/store/Main";
import type { DataListRow } from "@/modules/11_discipline/interface/request/Result"; import type { DataListRow } from "@/modules/11_discipline/interface/request/result";
import type { import type {
FormData as FormDataComplaint, FormData as FormDataComplaint,
ArrayPerson, ArrayPerson,
ArrayFileList, ArrayFileList,
} from "@/modules/11_discipline/interface/request/complaint"; } from "@/modules/11_discipline/interface/request/complaint";
import type { FormData as FormInvestigateFact } from "@/modules/11_discipline/interface/request/InvestigateFact"; import type { FormData as FormInvestigateFact } from "@/modules/11_discipline/interface/request/investigateFact";
import DialogSendToCommand from "@/modules/11_discipline/components/4_Result/DialogSendToCommand.vue"; import DialogSendToCommand from "@/modules/11_discipline/components/4_Result/DialogSendToCommand.vue";
import FormComplaints from "@/modules/11_discipline/components/1_Complaint/Form.vue"; // import FormComplaints from "@/modules/11_discipline/components/1_Complaint/Form.vue"; //

View file

@ -98,7 +98,7 @@ watch(props, () => {
v-ripple v-ripple
:active="listCheck === index" :active="listCheck === index"
active-class="my-menu-link" active-class="my-menu-link"
@click="clickList(index, item.director)" @click="clickList(Number(index), item.director)"
> >
<q-item-section>{{ item.title }}</q-item-section> <q-item-section>{{ item.title }}</q-item-section>
</q-item> </q-item>

View file

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

View file

@ -9,7 +9,7 @@ import config from "@/app.config";
/** importType*/ /** importType*/
import type { QTableProps } from "quasar"; import type { QTableProps } from "quasar";
import type { Director } from "@/modules/11_discipline/interface/request/Disciplinary"; import type { Director } from "@/modules/11_discipline/interface/request/disciplinary";
import type { Directors } from "@/modules/12_evaluatePersonal/interface/response/Main"; import type { Directors } from "@/modules/12_evaluatePersonal/interface/response/Main";
/** importComponents*/ /** importComponents*/
@ -20,7 +20,14 @@ import DialogDuty from "@/modules/12_evaluatePersonal/components/Detail/viewTab2
const $q = useQuasar(); const $q = useQuasar();
const route = useRoute(); const route = useRoute();
const mixin = useCounterMixin(); const mixin = useCounterMixin();
const { showLoader, hideLoader, messageError, dialogConfirm, success } = mixin; const {
showLoader,
hideLoader,
messageError,
dialogConfirm,
success,
dialogRemove,
} = mixin;
const props = defineProps({ const props = defineProps({
data: { data: {
@ -201,6 +208,21 @@ function onEditDuty(data: Director) {
modalDuty.value = true; modalDuty.value = true;
} }
function handleDelete(id: string) {
dialogRemove($q, async () => {
showLoader();
try {
await http.delete(config.API.evaluationMain() + `/del-director/${id}`);
await props.fetchData();
await success($q, "ลบสำเร็จ");
} catch (error) {
messageError($q, error);
} finally {
hideLoader();
}
});
}
/** /**
* ทำงานเม props.data การเปลยนแปลง * ทำงานเม props.data การเปลยนแปลง
*/ */
@ -268,17 +290,30 @@ watch(
</template> </template>
<template v-slot:body="props"> <template v-slot:body="props">
<q-tr :props="props"> <q-tr :props="props">
<q-td> <q-td auto-width>
<q-btn <q-btn
v-if="checkPermission($route)?.attrIsUpdate"
flat flat
round round
denes dense
icon="edit" icon="edit"
color="edit" color="edit"
@click.stop.prevent="onEditDuty(props.row)" @click.stop.prevent="onEditDuty(props.row)"
> >
<q-tooltip>แกไขหนาท</q-tooltip> <q-tooltip>แกไขหนาท</q-tooltip>
</q-btn> </q-btn>
<q-btn
v-if="checkPermission($route)?.attrIsDelete"
flat
round
dense
icon="delete"
color="red"
@click="handleDelete(props.row.id)"
>
<q-tooltip>ลบ</q-tooltip>
</q-btn>
</q-td> </q-td>
<q-td v-for="col in props.cols" :key="col.name" :props="props"> <q-td v-for="col in props.cols" :key="col.name" :props="props">
<div v-if="col.name == 'no'"> <div v-if="col.name == 'no'">

View file

@ -27,6 +27,7 @@ const {
dialogConfirm, dialogConfirm,
date2Thai, date2Thai,
success, success,
dialogRemove,
} = mixin; } = mixin;
/** props*/ /** props*/
@ -208,6 +209,23 @@ async function getList() {
}); });
} }
function handleDelete(meetingId: string) {
dialogRemove($q, async () => {
showLoader();
try {
await http.delete(
config.API.evaluationMain() + `/del-meeting/${id.value}/${meetingId}`
);
await props.fetchData();
await success($q, "ลบสำเร็จ");
} catch (error) {
messageError($q, error);
} finally {
hideLoader();
}
});
}
watch( watch(
() => props.data, () => props.data,
() => { () => {
@ -265,6 +283,7 @@ watch(
> >
<template v-slot:header="props"> <template v-slot:header="props">
<q-tr :props="props"> <q-tr :props="props">
<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">{{ col.label }}</span> <span class="text-weight-medium">{{ col.label }}</span>
</q-th> </q-th>
@ -272,6 +291,19 @@ watch(
</template> </template>
<template v-slot:body="props"> <template v-slot:body="props">
<q-tr :props="props"> <q-tr :props="props">
<q-td auto-width>
<q-btn
v-if="checkPermission($route)?.attrIsUpdate"
flat
round
dense
icon="delete"
color="red"
@click="handleDelete(props.row.id)"
>
<q-tooltip>ลบ</q-tooltip>
</q-btn>
</q-td>
<q-td v-for="col in props.cols" :key="col.name" :props="props"> <q-td v-for="col in props.cols" :key="col.name" :props="props">
<div v-if="col.name == 'no'"> <div v-if="col.name == 'no'">
{{ props.rowIndex + 1 }} {{ props.rowIndex + 1 }}

View file

@ -280,6 +280,7 @@ onMounted(async () => {
hide-bottom-space hide-bottom-space
dense dense
label="คำค้น" label="คำค้น"
@keydown.enter.prevent="(pagination.page = 1), searchInput()"
> >
<template v-slot:after> <template v-slot:after>
<q-btn <q-btn

View file

@ -222,6 +222,7 @@ const itemsCard = ref([
* @param id กล * @param id กล
*/ */
async function fetchDataQuota(id: string) { async function fetchDataQuota(id: string) {
if (!id) return;
await http await http
.get(config.API.salaryListPeriodQuota(id)) .get(config.API.salaryListPeriodQuota(id))
.then((res) => { .then((res) => {
@ -254,6 +255,7 @@ async function fetchDataQuota(id: string) {
* @param id กล * @param id กล
*/ */
async function fetchDataPeriod(id: string, force: boolean = false) { async function fetchDataPeriod(id: string, force: boolean = false) {
if (!id) return;
force && showLoader(); force && showLoader();
let formData = { let formData = {
...params.value, ...params.value,

View file

@ -215,6 +215,7 @@ const itemsCard = ref([
* @param id กล * @param id กล
*/ */
async function fetchDataQuota(id: string) { async function fetchDataQuota(id: string) {
if (!id) return;
await http await http
.get(config.API.salaryListPeriodQuotaEmp(id)) .get(config.API.salaryListPeriodQuotaEmp(id))
.then((res) => { .then((res) => {
@ -246,6 +247,7 @@ async function fetchDataQuota(id: string) {
* @param id กล * @param id กล
*/ */
async function fetchDataPeriod(id: string, force: boolean = false) { async function fetchDataPeriod(id: string, force: boolean = false) {
if (!id) return;
force && showLoader(); force && showLoader();
let formData = { let formData = {
...params.value, ...params.value,

View file

@ -308,6 +308,7 @@ async function fetchSalalyPeriod(
if (!data.group1id) { if (!data.group1id) {
hideLoader(); hideLoader();
} }
isLoad.value = data.group1id ? true : false;
}) })
.catch((err) => { .catch((err) => {
messageError($q, err); messageError($q, err);

View file

@ -44,7 +44,7 @@ const columns = ref<QTableProps["columns"]>([
field: "year", field: "year",
headerStyle: "font-size: 14px", headerStyle: "font-size: 14px",
style: "font-size: 14px", style: "font-size: 14px",
format: (val) => val + 543, format: (val) => (val ? val + 543 : "-"),
}, },
{ {
name: "durationKPI", name: "durationKPI",

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