feat: employee history dialog & table

This commit is contained in:
puriphatt 2024-06-28 08:10:14 +00:00
parent 39272000a7
commit 3458517de4
3 changed files with 349 additions and 50 deletions

View file

@ -1,14 +1,24 @@
<script setup lang="ts">
import { useI18n } from 'vue-i18n';
import { QTableColumn } from 'quasar';
import { EmployeeHistory, NewEmployeeHistory } from 'src/stores/employee/types';
import { dateFormat } from 'src/utils/datetime';
import NoData from '../NoData.vue';
import { computed, onMounted, ref, watch } from 'vue';
import useOptionStore from 'src/stores/options';
const { t } = useI18n();
const optionStore = useOptionStore();
const historyList = defineModel<EmployeeHistory[]>('historyList', {
required: true,
});
const columns: QTableColumn[] = [
{
name: 'time',
name: 'updatedAt',
label: t('time'),
field: 'time',
field: 'updatedAt',
align: 'left',
headerStyle: 'font-weight: bold',
},
@ -42,25 +52,254 @@ const columns: QTableColumn[] = [
},
];
const rows = [
const currentDate = ref();
const currentData = ref();
const currentIndex = ref(0);
const formatList = ref<NewEmployeeHistory[]>([]);
const fieldName = [
{
time: '17:00:29',
editBy: 'สุขใจ แสนดี',
history: [
{ title: 'ข้อมูลส่วนตัว', caption: 'อัพโหลดรูปภาพ' },
{ title: 'ข้อมูลผลตรวจสุขภาพ', caption: 'ผลตรวจสุขภาพ' },
],
valueBefore: [{ title: 'ผิดปกติ' }, { title: 'ผิดปกติ' }],
valueAfter: [{ title: 'ปกติ' }, { title: 'ปกติ' }],
name: 'customerBranchId',
title: 'formDialogTitleInformation',
i18n: 'formDialogEmployerID',
},
{
time: '15:20:01',
editBy: 'สุขใจ แสนดี',
history: [{ title: 'ข้อมูลส่วนตัว', caption: 'อัพโหลดรูปภาพ' }],
valueBefore: [{ title: '-' }],
valueAfter: [{ title: '1.jpg' }],
name: 'nrcNo',
title: 'formDialogTitleInformation',
i18n: 'formDialogEmployeeNRCNo',
},
{
name: 'firstName',
title: 'personalInfo',
i18n: 'formDialogInputFirstName',
},
{
name: 'firstNameEN',
title: 'personalInfo',
i18n: 'formDialogInputFirstNameEN',
},
{ name: 'lastName', title: 'personalInfo', i18n: 'formDialogInputLastName' },
{
name: 'lastNameEN',
title: 'personalInfo',
i18n: 'formDialogInputLastNameEN',
},
{
name: 'dateOfBirth',
title: 'personalInfo',
i18n: 'formDialogInputBirthDate',
},
{ name: 'gender', title: 'personalInfo', i18n: 'formDialogInputGender' },
{
name: 'nationality',
title: 'personalInfo',
i18n: 'formDialogInputNationality',
},
{
name: 'address',
title: 'formDialogTitlePersonnelAddress',
i18n: 'formDialogTitleAddressPure',
},
{
name: 'addressEN',
title: 'formDialogTitlePersonnelAddress',
i18n: 'formDialogTitleAddressPureEN',
},
{
name: 'provinceId',
title: 'formDialogTitlePersonnelAddress',
i18n: 'province',
},
{
name: 'districtId',
title: 'formDialogTitlePersonnelAddress',
i18n: 'district',
},
{
name: 'subDistrictId',
title: 'formDialogTitlePersonnelAddress',
i18n: 'subDistrict',
},
{
name: 'passportType',
title: 'formDialogTitlePassport',
i18n: 'formDialogInputPassportType',
},
{
name: 'passportNumber',
title: 'formDialogTitlePassport',
i18n: 'formDialogInputPassportNo',
},
{
name: 'previousPassportReference',
title: 'formDialogTitlePassport',
i18n: 'formDialogInputPassportRef',
},
{
name: 'passportIssuingPlace',
title: 'formDialogTitlePassport',
i18n: 'formDialogInputWPassportPlace',
},
{
name: 'passportIssuingCountry',
title: 'formDialogTitlePassport',
i18n: 'formDialogInputPassportCountry',
},
{
name: 'passportIssueDate',
title: 'formDialogTitlePassport',
i18n: 'formDialogInputPassportIssuance',
},
{
name: 'passportExpiryDate',
title: 'formDialogTitlePassport',
i18n: 'formDialogInputPassportExpire',
},
{
name: 'visaType',
title: 'formDialogTitleVisa',
i18n: 'formDialogInputVisaType',
},
{
name: 'visaNumber',
title: 'formDialogTitleVisa',
i18n: 'formDialogInputVisaNo',
},
{
name: 'visaIssueDate',
title: 'formDialogTitleVisa',
i18n: 'formDialogInputVisaIssuance',
},
{
name: 'visaExpiryDate',
title: 'formDialogTitleVisa',
i18n: 'formDialogInputVisaExpire',
},
{
name: 'visaIssuingPlace',
title: 'formDialogTitleVisa',
i18n: 'formDialogInputVisaPlace',
},
{
name: 'visaStayUntilDate',
title: 'formDialogTitleVisa',
i18n: 'formDialogInputVisaStayUntil',
},
{
name: 'tm6Number',
title: 'formDialogTitleVisa',
i18n: 'formDialogInputVisaTM6',
},
{
name: 'entryDate',
title: 'formDialogTitleVisa',
i18n: 'formDialogInputVisaEnter',
},
];
function isValidDate(dateString: string): boolean {
if (typeof dateString !== 'string' || dateString.length < 24) {
return false;
}
const date = new Date(dateString);
return !isNaN(date.getTime()) && dateString === date.toISOString();
}
function mapName(field: string): { title: string; i18n: string } {
const fieldData = fieldName.find((item) => item.name === field);
if (fieldData) {
return { title: fieldData.title, i18n: fieldData.i18n };
}
return { title: '-', i18n: '-' };
}
async function groupEmployeeHistory(
historyList: EmployeeHistory[],
): Promise<NewEmployeeHistory[]> {
const grouped = historyList.reduce((acc, curr) => {
const updatedAt = new Date(curr.updatedAt);
const dateKey = `${updatedAt.getFullYear()}-${updatedAt.getMonth() + 1}-${updatedAt.getDate()}`;
const existingEntry = acc.find((entry) => entry.date === dateKey);
if (existingEntry) {
const existingData = existingEntry.data.find(
(data) => data.updatedAt.getTime() === updatedAt.getTime(),
);
if (existingData) {
existingData.history.push({
valueAfter: curr.valueAfter,
valueBefore: curr.valueBefore,
field: curr.field,
});
} else {
existingEntry.data.push({
masterId: curr.masterId,
updatedBy: curr.updatedBy,
updatedByUserId: curr.updatedByUserId,
timestamp: curr.timestamp,
updatedAt: updatedAt,
id: curr.id,
history: [
{
valueAfter: curr.valueAfter,
valueBefore: curr.valueBefore,
field: curr.field,
},
],
});
}
} else {
acc.push({
date: dateKey,
data: [
{
masterId: curr.masterId,
updatedBy: curr.updatedBy,
updatedByUserId: curr.updatedByUserId,
timestamp: curr.timestamp,
updatedAt: updatedAt,
id: curr.id,
history: [
{
valueAfter: curr.valueAfter,
valueBefore: curr.valueBefore,
field: curr.field,
},
],
},
],
});
}
return acc;
}, [] as NewEmployeeHistory[]);
return grouped;
}
onMounted(async () => {
const newList = await groupEmployeeHistory(historyList.value);
formatList.value = newList;
currentDate.value = formatList.value[currentIndex.value].date;
currentData.value = formatList.value[currentIndex.value].data;
});
watch(
() => currentIndex.value,
(i) => {
currentDate.value = formatList.value[i].date;
currentData.value = formatList.value[i].data;
},
);
</script>
<template>
@ -70,41 +309,57 @@ const rows = [
style="background: hsla(var(--info-bg) / 0.1)"
>
<div class="surface-1 q-py-sm q-px-sm row items-center">
<q-btn flat padding="0" icon="mdi-chevron-left" />
<q-btn
flat
:disable="currentIndex === formatList.length - 1"
color="info"
padding="0"
icon="mdi-chevron-left"
@click="currentIndex++"
/>
<span class="text-weight-medium q-px-xl">
{{ '20 มีนาคม 2567' }}
{{ dateFormat(currentDate) }}
</span>
<q-btn flat padding="0" icon="mdi-chevron-right" />
<q-btn
flat
:disable="currentIndex === 0"
color="info"
padding="0"
icon="mdi-chevron-right"
@click="currentIndex--"
/>
</div>
</div>
<q-table
v-if="currentData?.length > 0"
flat
class="table-border"
table-header-class="surface-2"
:rows="rows"
:rows="currentData"
:columns="columns"
row-key="name"
>
<template v-slot:body="props">
<q-tr
:props="props"
:style="`background-color: ${props.rowIndex / 2 === 0 ? '' : 'var(--surface-2)'}`"
:style="`background-color: ${props.rowIndex % 2 === 0 ? '' : 'var(--surface-2)'}`"
>
<q-td key="time" :props="props">
{{ props.row.time }}
<q-td key="updatedAt" :props="props">
{{ dateFormat(props.row.updatedAt, false, true, true) }}
</q-td>
<q-td key="editBy" :props="props">
<div class="row items-center no-wrap">
<q-avatar>
<img src="https://cdn.quasar.dev/img/avatar.png" />
<q-avatar class="surface-tab">
<img v-if="false" src="https://cdn.quasar.dev/img/avatar.png" />
<q-icon v-else name="mdi-account"></q-icon>
</q-avatar>
<div class="column q-pl-md items-start">
<span class="text-weight-bold">
{{ props.row.editBy }}
{{ props.row.editBy ?? '-' }}
</span>
<span class="text-caption">กบรหาร</span>
<!-- <span class="text-caption">กบรหาร</span> -->
</div>
</div>
</q-td>
@ -112,38 +367,51 @@ const rows = [
<q-td key="history" :props="props">
<q-stepper vertical flat>
<q-step
v-for="(item, index) in props.row.history.slice().reverse()"
v-for="(item, index) in props.row.history"
:key="index"
:name="1"
:title="item.title"
:caption="item.caption"
:title="$t(mapName(item.field).title)"
:caption="$t(mapName(item.field).i18n)"
:icon="`mdi-numeric-${props.row.history.length - index}`"
/>
>
asd
</q-step>
</q-stepper>
</q-td>
<q-td key="valueBefore" :props="props">
<div
v-for="(item, index) in props.row.valueBefore.slice().reverse()"
v-for="(i, index) in props.row.history"
:key="index"
class="q-py-md"
>
{{ item.title }}
{{
isValidDate(i.valueBefore) === true
? dateFormat(i.valueBefore)
: optionStore.mapOption(i.valueBefore)
}}
</div>
</q-td>
<q-td key="valueAfter" :props="props">
<div
v-for="(item, index) in props.row.valueAfter.slice().reverse()"
v-for="(i, index) in props.row.history"
:key="index"
class="q-py-md"
>
{{ item.title }}
{{
isValidDate(i.valueAfter) === true
? dateFormat(i.valueAfter)
: optionStore.mapOption(i.valueAfter)
}}
</div>
</q-td>
</q-tr>
</template>
</q-table>
<div v-else class="table-border flex items-center justify-center q-py-lg">
<NoData />
</div>
</div>
</template>

View file

@ -1,19 +1,22 @@
<script lang="ts" setup>
import { computed, onMounted, ref, watch } from 'vue';
import SignaturePad from 'signature_pad';
import { useQuasar } from 'quasar';
import SignaturePad from 'signature_pad';
import Cropper from 'cropperjs';
defineExpose({ clearCanvas, clearUpload });
const $q = useQuasar();
const isDarkActive = computed(() => $q.dark.isActive);
const canvasRef = ref<HTMLCanvasElement>();
const signaturePad = ref();
const currentColor = ref('blue');
const imageRef = ref<HTMLImageElement>();
const cropper = ref();
const tab = ref('draw');
const isDarkActive = computed(() => $q.dark.isActive);
const uploadFile = ref<File | undefined>(undefined);
const profileUrl = ref<string | null>('');
const inputFile = (() => {
@ -36,7 +39,7 @@ const inputFile = (() => {
return element;
})();
function initializeSignaturePad(canva?: HTMLCanvasElement) {
async function initializeSignaturePad(canva?: HTMLCanvasElement) {
if (canva) {
signaturePad.value = new SignaturePad(canva, {
backgroundColor: isDarkActive.value
@ -49,6 +52,26 @@ function initializeSignaturePad(canva?: HTMLCanvasElement) {
}
}
async function initializeCropper(image?: HTMLImageElement) {
console.log(image);
if (image) {
cropper.value = new Cropper(image, {
aspectRatio: 16 / 9,
crop(event) {
console.log(event.detail.x);
console.log(event.detail.y);
console.log(event.detail.width);
console.log(event.detail.height);
console.log(event.detail.rotate);
console.log(event.detail.scaleX);
console.log(event.detail.scaleY);
},
});
} else {
console.warn('Canvas reference not found. Cropper not initialized.');
}
}
function changeColor(color: string) {
signaturePad.value.penColor = color;
currentColor.value = color;
@ -65,12 +88,14 @@ function clearUpload() {
watch(
() => tab.value,
async () => {
initializeSignaturePad(canvasRef.value);
await initializeSignaturePad(canvasRef.value);
await initializeCropper(imageRef.value);
},
);
onMounted(() => {
initializeSignaturePad(canvasRef.value);
onMounted(async () => {
await initializeSignaturePad(canvasRef.value);
await initializeCropper(imageRef.value);
});
</script>
<template>
@ -161,17 +186,18 @@ onMounted(() => {
</div>
</div>
<div v-if="tab === 'upload'" class="q-pa-md">
<div v-show="tab === 'upload'" class="q-pa-md">
<div
class="bordered upload-border rounded column items-center justify-center"
style="height: 312px"
>
<q-img
v-if="profileUrl"
:src="profileUrl"
v-show="profileUrl"
ref="imageRef"
:src="profileUrl ?? ''"
style="object-fit: cover; width: 100%; height: 100%"
/>
<div v-else>
<div v-if="!profileUrl">
<q-icon
name="mdi-cloud-upload"
size="10rem"

View file

@ -44,6 +44,7 @@ import {
Employee,
EmployeeWork,
EmployeeCheckup,
EmployeeHistory,
} from 'stores/employee/types';
import { onMounted } from 'vue';
@ -113,7 +114,6 @@ const formData = ref<CustomerCreate>({
image: null,
});
const employeeHistoryDialog = ref(false);
const inputSearch = ref<string>();
const fieldSelectedCustomer = ref<{ label: string; value: string }>({
label: t('all'),
@ -363,6 +363,9 @@ const selectorList = computed(() => [
const customerType = ref<CustomerType>('CORP');
const employeeHistoryDialog = ref(false);
const employeeHistory = ref<EmployeeHistory[]>();
function triggerCreate(type: CustomerType) {
customerType.value = type;
openDialogInputForm();
@ -1072,7 +1075,8 @@ async function checkEmployeeForm() {
}
async function openHistory(id: string) {
console.log(id);
const res = await employeeStore.getEditHistory(id);
employeeHistory.value = res.reverse();
employeeHistoryDialog.value = true;
}
@ -2883,8 +2887,9 @@ watch([inputSearch, currentStatus], async () => {
:title="$t('editHistory')"
v-model:modal="employeeHistoryDialog"
:close="() => (employeeHistoryDialog = false)"
v-if="employeeHistory"
>
<HistoryEditComponent />
<HistoryEditComponent v-model:history-list="employeeHistory" />
</FormDialog>
</template>