2024-04-03 18:28:59 +07:00
|
|
|
<script setup lang="ts">
|
2024-06-27 08:32:31 +00:00
|
|
|
import { ref, onMounted, watch, computed } from 'vue';
|
2024-07-16 03:24:13 +00:00
|
|
|
import useUtilsStore, { dialog } from 'src/stores/utils';
|
|
|
|
|
import { calculateAge } from 'src/utils/datetime';
|
2024-07-09 11:21:25 +07:00
|
|
|
import type { QTableProps } from 'quasar';
|
|
|
|
|
import { storeToRefs } from 'pinia';
|
2024-06-27 08:32:31 +00:00
|
|
|
import { useI18n } from 'vue-i18n';
|
|
|
|
|
import useFlowStore from 'src/stores/flow';
|
2024-04-10 14:00:55 +07:00
|
|
|
import useUserStore from 'stores/user';
|
|
|
|
|
import useBranchStore from 'src/stores/branch';
|
|
|
|
|
|
2024-04-19 13:54:57 +07:00
|
|
|
import {
|
|
|
|
|
User,
|
|
|
|
|
UserAttachmentCreate,
|
|
|
|
|
UserCreate,
|
|
|
|
|
UserTypeStats,
|
|
|
|
|
} from 'src/stores/user/types';
|
2024-04-10 14:00:55 +07:00
|
|
|
import { BranchUserStats } from 'src/stores/branch/types';
|
2024-04-18 18:27:22 +07:00
|
|
|
import useAddressStore from 'src/stores/address';
|
2024-06-28 14:14:23 +07:00
|
|
|
import ButtonAddComponent from 'components/ButtonAddCompoent.vue';
|
2024-04-05 17:36:54 +07:00
|
|
|
import PersonCard from 'components/home/PersonCard.vue';
|
2024-04-03 18:28:59 +07:00
|
|
|
import AppBox from 'components/app/AppBox.vue';
|
2024-04-05 17:36:54 +07:00
|
|
|
import StatCardComponent from 'components/StatCardComponent.vue';
|
2024-04-10 20:26:32 +07:00
|
|
|
import AddButton from 'components/AddButton.vue';
|
2024-04-05 17:43:16 +07:00
|
|
|
import TooltipComponent from 'components/TooltipComponent.vue';
|
2024-06-28 14:14:23 +07:00
|
|
|
import FormDialog from 'components/FormDialog.vue';
|
|
|
|
|
import FormInformation from 'components/02_personnel-management/FormInformation.vue';
|
|
|
|
|
import FormPerson from 'components/02_personnel-management/FormPerson.vue';
|
|
|
|
|
import FormByType from 'components/02_personnel-management/FormByType.vue';
|
|
|
|
|
import DrawerInfo from 'components/DrawerInfo.vue';
|
|
|
|
|
import InfoForm from 'components/02_personnel-management/InfoForm.vue';
|
|
|
|
|
import NoData from 'components/NoData.vue';
|
|
|
|
|
import ProfileUpload from 'components/ProfileUpload.vue';
|
2024-07-01 13:01:57 +07:00
|
|
|
import PaginationComponent from 'src/components/PaginationComponent.vue';
|
2024-04-04 15:03:39 +07:00
|
|
|
|
2024-07-09 11:21:25 +07:00
|
|
|
const { locale, t } = useI18n();
|
2024-06-26 17:03:49 +07:00
|
|
|
|
2024-07-02 09:11:39 +00:00
|
|
|
const utilsStore = useUtilsStore();
|
2024-06-26 17:03:49 +07:00
|
|
|
const flowStore = useFlowStore();
|
2024-04-05 17:26:40 +07:00
|
|
|
const userStore = useUserStore();
|
2024-04-10 17:00:28 +07:00
|
|
|
const branchStore = useBranchStore();
|
2024-04-18 18:27:22 +07:00
|
|
|
const adrressStore = useAddressStore();
|
2024-04-05 17:26:40 +07:00
|
|
|
const { data: userData } = storeToRefs(userStore);
|
2024-04-03 18:28:59 +07:00
|
|
|
|
2024-04-10 14:00:55 +07:00
|
|
|
const defaultFormData = {
|
|
|
|
|
provinceId: null,
|
|
|
|
|
districtId: null,
|
|
|
|
|
subDistrictId: null,
|
|
|
|
|
telephoneNo: '',
|
|
|
|
|
email: '',
|
|
|
|
|
zipCode: '',
|
|
|
|
|
gender: '',
|
|
|
|
|
addressEN: '',
|
|
|
|
|
address: '',
|
|
|
|
|
trainingPlace: null,
|
|
|
|
|
importNationality: null,
|
|
|
|
|
sourceNationality: null,
|
|
|
|
|
licenseExpireDate: null,
|
|
|
|
|
licenseIssueDate: null,
|
|
|
|
|
licenseNo: null,
|
2024-04-18 09:06:21 +07:00
|
|
|
discountCondition: null,
|
2024-04-10 14:00:55 +07:00
|
|
|
retireDate: null,
|
|
|
|
|
startDate: null,
|
|
|
|
|
registrationNo: null,
|
|
|
|
|
lastNameEN: '',
|
|
|
|
|
lastName: '',
|
|
|
|
|
firstNameEN: '',
|
|
|
|
|
firstName: '',
|
|
|
|
|
userRole: '',
|
|
|
|
|
userType: '',
|
|
|
|
|
profileImage: null,
|
2024-04-11 22:17:24 +07:00
|
|
|
birthDate: null,
|
2024-04-18 09:06:21 +07:00
|
|
|
responsibleArea: null,
|
|
|
|
|
username: '',
|
2024-04-18 18:27:22 +07:00
|
|
|
status: 'CREATED',
|
2024-04-19 13:54:57 +07:00
|
|
|
checkpoint: null,
|
|
|
|
|
checkpointEN: null,
|
2024-04-10 14:00:55 +07:00
|
|
|
};
|
2024-04-10 17:00:28 +07:00
|
|
|
|
2024-07-09 11:58:51 +07:00
|
|
|
const modeView = ref(false);
|
2024-07-09 11:21:25 +07:00
|
|
|
|
|
|
|
|
const fieldSelectedOption = ref<{ label: string; value: string }[]>([
|
|
|
|
|
{
|
|
|
|
|
label: 'fullname',
|
|
|
|
|
value: 'name',
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
label: 'type',
|
|
|
|
|
value: 'type',
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
label: 'telephone',
|
|
|
|
|
value: 'telephoneNo',
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
label: 'personnelCardAge',
|
|
|
|
|
value: 'birthDate',
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
label: 'formDialogInputEmail',
|
|
|
|
|
value: 'email',
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
label: 'userRole',
|
|
|
|
|
value: 'userRole',
|
|
|
|
|
},
|
|
|
|
|
]);
|
|
|
|
|
|
|
|
|
|
const fieldSelected = ref<
|
|
|
|
|
('name' | 'type' | 'telephoneNo' | 'birthDate' | 'email' | 'userRole')[]
|
|
|
|
|
>(['name', 'type', 'telephoneNo', 'birthDate', 'email', 'userRole']);
|
2024-07-05 11:01:40 +00:00
|
|
|
const fieldDisplay = ref();
|
2024-07-05 03:29:33 +00:00
|
|
|
const hideStat = ref(false);
|
2024-04-22 14:04:06 +07:00
|
|
|
const userCode = ref<string>();
|
2024-04-18 18:27:22 +07:00
|
|
|
const currentUser = ref<User>();
|
2024-04-25 04:07:12 +00:00
|
|
|
const infoDrawerEdit = ref(false);
|
2024-04-19 09:06:38 +07:00
|
|
|
const infoPersonId = ref<string>('');
|
2024-04-18 18:27:22 +07:00
|
|
|
const infoPersonCard = ref();
|
|
|
|
|
const infoDrawer = ref(false);
|
2024-04-18 09:06:21 +07:00
|
|
|
const profileSubmit = ref(false);
|
2024-04-10 23:17:39 +07:00
|
|
|
const urlProfile = ref<string>();
|
2024-04-10 17:00:28 +07:00
|
|
|
const isEdit = ref(false);
|
2024-04-05 19:05:51 +07:00
|
|
|
const modal = ref(false);
|
2024-04-18 18:27:22 +07:00
|
|
|
const statusToggle = ref(true);
|
2024-04-18 16:35:47 +07:00
|
|
|
const statusFilter = ref<'all' | 'statusACTIVE' | 'statusINACTIVE'>('all');
|
|
|
|
|
const inputSearch = ref('');
|
2024-04-18 18:27:22 +07:00
|
|
|
const userId = ref<string>();
|
2024-07-05 11:01:40 +00:00
|
|
|
const selectorLabel = ref<string>('ALL');
|
2024-04-18 18:27:22 +07:00
|
|
|
const hqId = ref<string>();
|
|
|
|
|
const brId = ref<string>();
|
2024-04-10 23:17:39 +07:00
|
|
|
const formDialogRef = ref();
|
|
|
|
|
const userStats = ref<BranchUserStats[]>();
|
2024-04-22 17:52:37 +07:00
|
|
|
const sortedUserStats = computed(() => {
|
|
|
|
|
return userStats.value?.slice().sort((a, b) => b.count - a.count);
|
|
|
|
|
});
|
2024-04-10 23:17:39 +07:00
|
|
|
const typeStats = ref<UserTypeStats>();
|
2024-04-19 13:54:57 +07:00
|
|
|
const agencyFile = ref<File[]>([]);
|
|
|
|
|
const agencyFileList = ref<{ name: string; url: string }[]>([]);
|
2024-04-10 17:00:28 +07:00
|
|
|
const formData = ref<UserCreate>({
|
2024-07-03 09:21:53 +00:00
|
|
|
branchId: '',
|
2024-04-10 17:00:28 +07:00
|
|
|
provinceId: null,
|
|
|
|
|
districtId: null,
|
|
|
|
|
subDistrictId: null,
|
|
|
|
|
telephoneNo: '',
|
|
|
|
|
email: '',
|
|
|
|
|
zipCode: '',
|
|
|
|
|
gender: '',
|
|
|
|
|
addressEN: '',
|
|
|
|
|
address: '',
|
|
|
|
|
trainingPlace: null,
|
|
|
|
|
importNationality: null,
|
|
|
|
|
sourceNationality: null,
|
|
|
|
|
licenseExpireDate: null,
|
|
|
|
|
licenseIssueDate: null,
|
|
|
|
|
licenseNo: null,
|
2024-04-18 18:27:22 +07:00
|
|
|
discountCondition: null,
|
2024-04-10 17:00:28 +07:00
|
|
|
retireDate: null,
|
|
|
|
|
startDate: null,
|
|
|
|
|
registrationNo: null,
|
|
|
|
|
lastNameEN: '',
|
|
|
|
|
lastName: '',
|
|
|
|
|
firstNameEN: '',
|
|
|
|
|
firstName: '',
|
|
|
|
|
userRole: '',
|
|
|
|
|
userType: '',
|
|
|
|
|
profileImage: null,
|
|
|
|
|
birthDate: null,
|
2024-04-18 09:06:21 +07:00
|
|
|
responsibleArea: null,
|
|
|
|
|
checkpoint: null,
|
|
|
|
|
checkpointEN: null,
|
|
|
|
|
username: '',
|
2024-04-18 18:27:22 +07:00
|
|
|
status: 'CREATED',
|
2024-04-10 17:00:28 +07:00
|
|
|
});
|
2024-04-05 19:05:51 +07:00
|
|
|
|
2024-04-10 19:32:15 +07:00
|
|
|
const profileFile = ref<File | undefined>(undefined);
|
|
|
|
|
const inputFile = document.createElement('input');
|
|
|
|
|
inputFile.type = 'file';
|
|
|
|
|
inputFile.accept = 'image/*';
|
|
|
|
|
|
2024-04-17 11:32:18 +07:00
|
|
|
const reader = new FileReader();
|
|
|
|
|
|
2024-07-09 11:21:25 +07:00
|
|
|
const columns = [
|
|
|
|
|
{
|
|
|
|
|
name: 'name',
|
|
|
|
|
align: 'left',
|
|
|
|
|
label: 'fullname',
|
|
|
|
|
field: 'firstName',
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
name: 'type',
|
|
|
|
|
align: 'left',
|
|
|
|
|
label: 'type',
|
|
|
|
|
field: 'type',
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
name: 'telephoneNo',
|
|
|
|
|
align: 'left',
|
|
|
|
|
label: 'telephone',
|
|
|
|
|
field: 'telephoneNo',
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
name: 'birthDate',
|
|
|
|
|
align: 'left',
|
|
|
|
|
label: 'personnelCardAge',
|
|
|
|
|
field: 'birthDate',
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
name: 'email',
|
|
|
|
|
align: 'left',
|
|
|
|
|
label: 'formDialogInputEmail',
|
|
|
|
|
field: 'email',
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
name: 'userRole',
|
|
|
|
|
align: 'left',
|
|
|
|
|
label: 'userRole',
|
|
|
|
|
field: 'userRole',
|
|
|
|
|
},
|
|
|
|
|
] satisfies QTableProps['columns'];
|
|
|
|
|
|
2024-04-17 11:32:18 +07:00
|
|
|
reader.addEventListener('load', () => {
|
|
|
|
|
if (typeof reader.result === 'string') {
|
|
|
|
|
urlProfile.value = reader.result;
|
|
|
|
|
}
|
2024-06-27 08:32:31 +00:00
|
|
|
|
|
|
|
|
if (infoDrawerEdit.value) infoPersonCard.value[0].img = urlProfile.value;
|
2024-04-17 11:32:18 +07:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
watch(profileFile, () => {
|
|
|
|
|
if (profileFile.value) reader.readAsDataURL(profileFile.value);
|
|
|
|
|
});
|
|
|
|
|
|
2024-04-10 19:32:15 +07:00
|
|
|
inputFile.addEventListener('change', (e) => {
|
|
|
|
|
profileFile.value = (e.currentTarget as HTMLInputElement).files?.[0];
|
|
|
|
|
});
|
|
|
|
|
|
2024-04-17 17:06:32 +07:00
|
|
|
const selectorList = computed(() => [
|
|
|
|
|
{ label: 'USER', count: typeStats.value?.USER || 0 },
|
|
|
|
|
{ label: 'MESSENGER', count: typeStats.value?.MESSENGER || 0 },
|
|
|
|
|
{ label: 'DELEGATE', count: typeStats.value?.DELEGATE || 0 },
|
|
|
|
|
{ label: 'AGENCY', count: typeStats.value?.AGENCY || 0 },
|
|
|
|
|
]);
|
2024-04-10 14:00:55 +07:00
|
|
|
|
2024-06-25 16:26:46 +07:00
|
|
|
async function openDialog(
|
|
|
|
|
action?: 'FORM' | 'INFO',
|
2024-07-05 16:29:06 +07:00
|
|
|
id?: string,
|
2024-06-25 16:26:46 +07:00
|
|
|
isPersonEdit: boolean = false,
|
|
|
|
|
) {
|
2024-04-22 14:04:06 +07:00
|
|
|
if (userStore.userOption.hqOpts.length === 0) {
|
|
|
|
|
await userStore.fetchHqOption();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (userStore.userOption.roleOpts.length === 0) {
|
|
|
|
|
await userStore.fetchRoleOption();
|
|
|
|
|
}
|
|
|
|
|
|
2024-07-05 16:29:06 +07:00
|
|
|
if (id && userData.value) {
|
|
|
|
|
assignFormData(id);
|
2024-04-22 14:04:06 +07:00
|
|
|
isEdit.value = true;
|
|
|
|
|
|
|
|
|
|
if (formData.value.userType === 'AGENCY') {
|
2024-07-05 16:29:06 +07:00
|
|
|
const result = await userStore.fetchAttachment(id);
|
2024-04-22 14:04:06 +07:00
|
|
|
|
|
|
|
|
if (result) {
|
|
|
|
|
agencyFileList.value = result;
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-07-16 02:11:18 +00:00
|
|
|
}
|
|
|
|
|
if (userStore.userOption.hqOpts.length !== 0) {
|
2024-07-12 11:07:45 +07:00
|
|
|
hqId.value = userStore.userOption.hqOpts[0].value;
|
|
|
|
|
}
|
2024-07-16 02:11:18 +00:00
|
|
|
if (userStore.userOption.hqOpts.length === 0) {
|
2024-07-12 11:07:45 +07:00
|
|
|
console.log('no hq');
|
|
|
|
|
|
2024-07-16 02:11:18 +00:00
|
|
|
dialog({
|
|
|
|
|
color: 'warning',
|
|
|
|
|
icon: 'mdi-alert',
|
|
|
|
|
title: t('warning'),
|
|
|
|
|
actionText: t('agree'),
|
|
|
|
|
persistent: true,
|
|
|
|
|
message: t('headquartersNotEstablished'),
|
|
|
|
|
action: async () => {},
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
return;
|
2024-07-12 11:07:45 +07:00
|
|
|
}
|
|
|
|
|
|
2024-04-18 18:27:22 +07:00
|
|
|
if (action === 'FORM') {
|
|
|
|
|
modal.value = true;
|
|
|
|
|
} else if (action === 'INFO') {
|
2024-04-22 14:04:06 +07:00
|
|
|
if (!userData.value) return;
|
2024-06-25 16:26:46 +07:00
|
|
|
|
|
|
|
|
infoDrawerEdit.value = isPersonEdit ? true : false;
|
2024-04-18 18:27:22 +07:00
|
|
|
infoDrawer.value = true;
|
2024-07-05 16:29:06 +07:00
|
|
|
const user = userData.value.result.find((x) => x.id === id);
|
2024-04-18 18:27:22 +07:00
|
|
|
infoPersonCard.value = user
|
|
|
|
|
? [
|
|
|
|
|
{
|
|
|
|
|
id: user.id,
|
|
|
|
|
img: `${user.profileImageUrl}`,
|
|
|
|
|
name:
|
|
|
|
|
locale.value === 'en-US'
|
|
|
|
|
? `${user.firstNameEN} ${user.lastNameEN}`
|
|
|
|
|
: `${user.firstName} ${user.lastName}`,
|
|
|
|
|
male: user.gender === 'male',
|
|
|
|
|
female: user.gender === 'female',
|
|
|
|
|
badge: user.code,
|
|
|
|
|
disabled: user.status === 'INACTIVE',
|
|
|
|
|
},
|
|
|
|
|
]
|
|
|
|
|
: [];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
statusToggle.value = true;
|
2024-04-11 22:17:24 +07:00
|
|
|
userStore.userOption.brOpts = [];
|
2024-04-05 19:05:51 +07:00
|
|
|
}
|
2024-04-10 14:00:55 +07:00
|
|
|
|
2024-04-19 09:06:38 +07:00
|
|
|
function undo() {
|
2024-04-19 14:19:08 +07:00
|
|
|
if (!infoPersonId.value) return;
|
2024-04-25 04:07:12 +00:00
|
|
|
infoDrawerEdit.value = false;
|
2024-04-19 09:06:38 +07:00
|
|
|
assignFormData(infoPersonId.value);
|
|
|
|
|
}
|
|
|
|
|
|
2024-04-10 14:00:55 +07:00
|
|
|
function onClose() {
|
|
|
|
|
hqId.value = '';
|
|
|
|
|
brId.value = '';
|
|
|
|
|
userId.value = '';
|
2024-04-22 14:04:06 +07:00
|
|
|
userCode.value = '';
|
2024-06-27 08:32:31 +00:00
|
|
|
urlProfile.value = undefined;
|
|
|
|
|
profileFile.value = undefined;
|
2024-04-19 13:54:57 +07:00
|
|
|
agencyFile.value = [];
|
2024-04-11 22:17:24 +07:00
|
|
|
modal.value = false;
|
|
|
|
|
isEdit.value = false;
|
2024-04-22 14:04:06 +07:00
|
|
|
infoDrawer.value = false;
|
|
|
|
|
profileSubmit.value = false;
|
2024-04-18 18:27:22 +07:00
|
|
|
statusToggle.value = true;
|
2024-04-11 22:17:24 +07:00
|
|
|
Object.assign(formData.value, defaultFormData);
|
2024-04-18 09:06:21 +07:00
|
|
|
mapUserType(selectorLabel.value);
|
2024-06-26 17:03:49 +07:00
|
|
|
flowStore.rotate();
|
2024-04-10 14:00:55 +07:00
|
|
|
}
|
|
|
|
|
|
2024-04-10 17:00:28 +07:00
|
|
|
async function onSubmit() {
|
2024-04-19 14:19:08 +07:00
|
|
|
if (profileSubmit.value) {
|
|
|
|
|
formData.value.profileImage = profileFile.value as File;
|
|
|
|
|
} else {
|
|
|
|
|
formData.value.profileImage = null;
|
|
|
|
|
}
|
|
|
|
|
|
2024-04-22 10:54:32 +07:00
|
|
|
if (isEdit.value && userId.value) {
|
2024-06-27 08:32:31 +00:00
|
|
|
if (!userId.value) return;
|
|
|
|
|
|
2024-07-03 09:21:53 +00:00
|
|
|
formData.value.branchId = brId.value
|
|
|
|
|
? brId.value
|
|
|
|
|
: hqId.value
|
|
|
|
|
? hqId.value
|
|
|
|
|
: '';
|
2024-06-27 08:32:31 +00:00
|
|
|
const formDataEdit = {
|
|
|
|
|
...formData.value,
|
|
|
|
|
status: !statusToggle.value ? 'INACTIVE' : 'ACTIVE',
|
|
|
|
|
} as const;
|
|
|
|
|
|
2024-07-03 09:21:53 +00:00
|
|
|
await userStore.editById(userId.value, formDataEdit);
|
2024-06-27 08:32:31 +00:00
|
|
|
|
|
|
|
|
if (userId.value && formDataEdit.userType === 'AGENCY') {
|
|
|
|
|
if (!agencyFile.value) return;
|
|
|
|
|
const payload: UserAttachmentCreate = {
|
|
|
|
|
file: agencyFile.value,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
if (payload?.file) {
|
|
|
|
|
await userStore.addAttachment(userId.value, payload);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-06-28 14:19:46 +07:00
|
|
|
await fetchUserList();
|
2024-06-27 08:32:31 +00:00
|
|
|
typeStats.value = await userStore.typeStats();
|
|
|
|
|
const res = await branchStore.userStats(formData.value.userType);
|
|
|
|
|
if (res) {
|
|
|
|
|
userStats.value = res;
|
|
|
|
|
}
|
|
|
|
|
onClose();
|
2024-04-10 19:10:13 +07:00
|
|
|
} else {
|
2024-06-27 08:32:31 +00:00
|
|
|
if (!hqId.value) return;
|
|
|
|
|
|
|
|
|
|
statusToggle.value
|
|
|
|
|
? (formData.value.status = 'CREATED')
|
|
|
|
|
: (formData.value.status = 'INACTIVE');
|
|
|
|
|
|
2024-07-03 09:21:53 +00:00
|
|
|
formData.value.branchId = brId.value
|
|
|
|
|
? brId.value
|
|
|
|
|
: hqId.value
|
|
|
|
|
? hqId.value
|
|
|
|
|
: '';
|
|
|
|
|
|
2024-06-27 08:32:31 +00:00
|
|
|
const result = await userStore.create(formData.value);
|
|
|
|
|
|
|
|
|
|
if (result && formData.value.userType === 'AGENCY') {
|
|
|
|
|
if (!agencyFile.value) return;
|
|
|
|
|
|
|
|
|
|
const payload: UserAttachmentCreate = {
|
|
|
|
|
file: agencyFile.value,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
if (payload?.file) {
|
|
|
|
|
await userStore.addAttachment(result.id, payload);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
selectorLabel.value = formData.value.userType;
|
2024-06-28 14:19:46 +07:00
|
|
|
|
|
|
|
|
await fetchUserList();
|
|
|
|
|
|
2024-06-27 08:32:31 +00:00
|
|
|
typeStats.value = await userStore.typeStats();
|
|
|
|
|
const res = await branchStore.userStats(formData.value.userType);
|
|
|
|
|
if (res) {
|
|
|
|
|
userStats.value = res;
|
|
|
|
|
}
|
|
|
|
|
onClose();
|
2024-04-10 14:00:55 +07:00
|
|
|
}
|
2024-06-26 17:03:49 +07:00
|
|
|
flowStore.rotate();
|
2024-04-10 14:00:55 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async function onDelete(id: string) {
|
2024-04-11 00:37:07 +07:00
|
|
|
dialog({
|
|
|
|
|
color: 'negative',
|
|
|
|
|
icon: 'mdi-trash-can-outline',
|
2024-07-12 11:07:45 +07:00
|
|
|
title: t('deleteConfirmTitle'),
|
|
|
|
|
actionText: t('agree'),
|
2024-04-11 00:37:07 +07:00
|
|
|
persistent: true,
|
2024-07-12 11:07:45 +07:00
|
|
|
message: t('deleteConfirmMessage'),
|
2024-04-11 00:37:07 +07:00
|
|
|
action: async () => {
|
|
|
|
|
await userStore.deleteById(id);
|
2024-06-28 14:19:46 +07:00
|
|
|
|
|
|
|
|
await fetchUserList();
|
2024-04-18 18:27:22 +07:00
|
|
|
typeStats.value = await userStore.typeStats();
|
2024-06-26 17:03:49 +07:00
|
|
|
flowStore.rotate();
|
2024-04-11 00:37:07 +07:00
|
|
|
},
|
|
|
|
|
});
|
2024-04-10 14:00:55 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function mapUserType(label: string) {
|
2024-04-18 09:06:21 +07:00
|
|
|
const userTypeMap: { [key: string]: string } = {
|
|
|
|
|
USER: 'USER',
|
|
|
|
|
MESSENGER: 'MESSENGER',
|
|
|
|
|
DELEGATE: 'DELEGATE',
|
|
|
|
|
AGENCY: 'AGENCY',
|
|
|
|
|
};
|
|
|
|
|
formData.value.userType = userTypeMap[label];
|
2024-04-10 14:00:55 +07:00
|
|
|
}
|
|
|
|
|
|
2024-04-18 16:35:47 +07:00
|
|
|
async function toggleStatus(id: string) {
|
|
|
|
|
const record = userData.value?.result.find((v) => v.id === id);
|
|
|
|
|
|
|
|
|
|
if (!record) return;
|
|
|
|
|
|
|
|
|
|
const res = await userStore.editById(record.id, {
|
|
|
|
|
status: record.status !== 'INACTIVE' ? 'INACTIVE' : 'ACTIVE',
|
|
|
|
|
});
|
|
|
|
|
if (res) record.status = res.status;
|
|
|
|
|
}
|
|
|
|
|
|
2024-04-18 18:27:22 +07:00
|
|
|
async function assignFormData(idEdit: string) {
|
|
|
|
|
if (!userData.value) return;
|
|
|
|
|
const foundUser = userData.value.result.find((user) => user.id === idEdit);
|
2024-04-22 14:04:06 +07:00
|
|
|
|
2024-04-18 18:27:22 +07:00
|
|
|
if (foundUser) {
|
|
|
|
|
currentUser.value = foundUser;
|
2024-04-19 09:06:38 +07:00
|
|
|
infoPersonId.value = currentUser.value.id;
|
2024-04-18 18:27:22 +07:00
|
|
|
|
|
|
|
|
formData.value = {
|
2024-07-05 16:29:06 +07:00
|
|
|
branchId: foundUser.branch[0]?.id,
|
2024-04-18 18:27:22 +07:00
|
|
|
provinceId: foundUser.provinceId,
|
|
|
|
|
districtId: foundUser.districtId,
|
|
|
|
|
subDistrictId: foundUser.subDistrictId,
|
|
|
|
|
telephoneNo: foundUser.telephoneNo,
|
|
|
|
|
email: foundUser.email,
|
|
|
|
|
zipCode: foundUser.zipCode,
|
|
|
|
|
gender: foundUser.gender,
|
|
|
|
|
addressEN: foundUser.addressEN,
|
|
|
|
|
address: foundUser.address,
|
|
|
|
|
trainingPlace: foundUser.trainingPlace,
|
|
|
|
|
importNationality: foundUser.importNationality,
|
|
|
|
|
sourceNationality: foundUser.sourceNationality,
|
|
|
|
|
licenseNo: foundUser.licenseNo,
|
|
|
|
|
discountCondition: foundUser.discountCondition,
|
|
|
|
|
registrationNo: foundUser.registrationNo,
|
|
|
|
|
lastNameEN: foundUser.lastNameEN,
|
|
|
|
|
lastName: foundUser.lastName,
|
|
|
|
|
firstNameEN: foundUser.firstNameEN,
|
|
|
|
|
firstName: foundUser.firstName,
|
|
|
|
|
userRole: foundUser.userRole,
|
|
|
|
|
userType: foundUser.userType,
|
|
|
|
|
username: foundUser.username,
|
2024-04-19 13:54:57 +07:00
|
|
|
checkpoint: foundUser.checkpoint,
|
|
|
|
|
checkpointEN: foundUser.checkpointEN,
|
2024-04-18 18:27:22 +07:00
|
|
|
responsibleArea: foundUser.responsibleArea,
|
|
|
|
|
status: foundUser.status,
|
|
|
|
|
licenseExpireDate:
|
|
|
|
|
(foundUser.licenseExpireDate &&
|
|
|
|
|
new Date(foundUser.licenseExpireDate)) ||
|
|
|
|
|
null,
|
|
|
|
|
licenseIssueDate:
|
|
|
|
|
(foundUser.licenseIssueDate && new Date(foundUser.licenseIssueDate)) ||
|
|
|
|
|
null,
|
|
|
|
|
retireDate:
|
|
|
|
|
(foundUser.retireDate && new Date(foundUser.retireDate)) || null,
|
|
|
|
|
startDate: (foundUser.startDate && new Date(foundUser.startDate)) || null,
|
|
|
|
|
birthDate: (foundUser.birthDate && new Date(foundUser.birthDate)) || null,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
formData.value.status === 'ACTIVE' || 'CREATED'
|
|
|
|
|
? (statusToggle.value = true)
|
|
|
|
|
: (statusToggle.value = false);
|
|
|
|
|
userId.value = foundUser.id;
|
2024-04-22 10:54:32 +07:00
|
|
|
|
|
|
|
|
if (foundUser.branch[0]) {
|
|
|
|
|
if (foundUser.branch[0].isHeadOffice) {
|
|
|
|
|
hqId.value = foundUser.branch[0].id as string;
|
|
|
|
|
} else {
|
|
|
|
|
hqId.value = foundUser.branch[0].headOfficeId as string;
|
|
|
|
|
brId.value = foundUser.branch[0].id;
|
|
|
|
|
}
|
2024-04-18 18:27:22 +07:00
|
|
|
}
|
2024-04-22 10:54:32 +07:00
|
|
|
|
2024-04-22 14:04:06 +07:00
|
|
|
userCode.value = foundUser.code;
|
2024-04-18 18:27:22 +07:00
|
|
|
urlProfile.value = foundUser.profileImageUrl;
|
2024-06-27 08:32:31 +00:00
|
|
|
|
2024-04-18 18:27:22 +07:00
|
|
|
isEdit.value = true;
|
|
|
|
|
profileSubmit.value = true;
|
2024-04-22 10:54:32 +07:00
|
|
|
hqId.value && (await userStore.fetchBrOption(hqId.value));
|
2024-06-27 08:32:31 +00:00
|
|
|
if (infoPersonCard.value) {
|
|
|
|
|
infoPersonCard.value[0].img = foundUser.profileImageUrl;
|
|
|
|
|
}
|
2024-04-18 18:27:22 +07:00
|
|
|
if (formData.value.districtId) {
|
|
|
|
|
await adrressStore.fetchSubDistrictByProvinceId(
|
|
|
|
|
formData.value.districtId,
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-04-10 14:00:55 +07:00
|
|
|
onMounted(async () => {
|
2024-07-02 09:11:39 +00:00
|
|
|
utilsStore.currentTitle.title = 'personnelManagement';
|
2024-07-08 14:08:57 +07:00
|
|
|
utilsStore.currentTitle.path = [{ text: 'personnelManagementCaption' }];
|
2024-07-02 09:11:39 +00:00
|
|
|
|
2024-06-28 14:19:46 +07:00
|
|
|
await fetchUserList();
|
|
|
|
|
|
2024-04-11 22:17:24 +07:00
|
|
|
userStore.userOption.roleOpts.length === 0
|
|
|
|
|
? await userStore.fetchRoleOption()
|
|
|
|
|
: '';
|
2024-04-10 14:00:55 +07:00
|
|
|
typeStats.value = await userStore.typeStats();
|
2024-04-17 17:19:58 +07:00
|
|
|
|
2024-07-05 11:01:40 +00:00
|
|
|
// const firstTypeIncludeUser = Object.entries(typeStats.value || {}).find(
|
|
|
|
|
// (v) => v[1] > 0,
|
|
|
|
|
// );
|
2024-04-17 17:19:58 +07:00
|
|
|
|
2024-07-05 11:01:40 +00:00
|
|
|
// firstTypeIncludeUser && (selectorLabel.value = firstTypeIncludeUser[0]);
|
2024-04-17 17:19:58 +07:00
|
|
|
|
2024-04-10 19:10:13 +07:00
|
|
|
const res = await branchStore.userStats(formData.value.userType);
|
|
|
|
|
if (res) {
|
|
|
|
|
userStats.value = res;
|
2024-04-10 14:00:55 +07:00
|
|
|
}
|
2024-06-26 17:03:49 +07:00
|
|
|
flowStore.rotate();
|
2024-04-10 14:00:55 +07:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
watch(
|
|
|
|
|
() => selectorLabel.value,
|
|
|
|
|
async (label) => {
|
|
|
|
|
mapUserType(label);
|
2024-06-28 14:19:46 +07:00
|
|
|
await fetchUserList();
|
2024-04-10 19:10:13 +07:00
|
|
|
const res = await branchStore.userStats(label);
|
|
|
|
|
if (res) {
|
|
|
|
|
userStats.value = res;
|
2024-04-10 14:00:55 +07:00
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
);
|
2024-04-19 11:45:51 +07:00
|
|
|
|
2024-04-19 14:54:07 +07:00
|
|
|
watch(
|
|
|
|
|
() => formData.value.userType,
|
|
|
|
|
async () => {
|
2024-07-17 10:49:21 +00:00
|
|
|
if (!infoDrawerEdit.value) return;
|
2024-04-19 14:54:07 +07:00
|
|
|
formData.value.registrationNo = null;
|
|
|
|
|
formData.value.startDate = null;
|
|
|
|
|
formData.value.retireDate = null;
|
|
|
|
|
formData.value.responsibleArea = null;
|
|
|
|
|
formData.value.discountCondition = null;
|
|
|
|
|
formData.value.sourceNationality = null;
|
|
|
|
|
formData.value.importNationality = null;
|
|
|
|
|
formData.value.trainingPlace = null;
|
|
|
|
|
formData.value.checkpoint = null;
|
|
|
|
|
formData.value.checkpointEN = null;
|
|
|
|
|
agencyFile.value = [];
|
|
|
|
|
},
|
|
|
|
|
);
|
2024-06-28 14:19:46 +07:00
|
|
|
|
|
|
|
|
const currentPage = ref(1);
|
2024-07-04 14:25:24 +07:00
|
|
|
const pageSize = ref(30);
|
2024-06-28 14:19:46 +07:00
|
|
|
const currentMaxPage = computed(() =>
|
2024-07-04 14:25:24 +07:00
|
|
|
userData.value ? Math.ceil(userData.value?.total / pageSize.value) : 1,
|
2024-06-28 14:19:46 +07:00
|
|
|
);
|
|
|
|
|
|
|
|
|
|
async function fetchUserList() {
|
|
|
|
|
await userStore.fetchList({
|
|
|
|
|
includeBranch: true,
|
2024-07-04 14:25:24 +07:00
|
|
|
pageSize: pageSize.value,
|
2024-06-28 14:19:46 +07:00
|
|
|
page: currentPage.value,
|
|
|
|
|
query: !!inputSearch.value ? inputSearch.value : undefined,
|
2024-07-05 11:01:40 +00:00
|
|
|
userType: selectorLabel.value === 'ALL' ? undefined : selectorLabel.value,
|
2024-06-28 14:19:46 +07:00
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
watch(inputSearch, async () => await fetchUserList());
|
2024-04-03 18:28:59 +07:00
|
|
|
</script>
|
|
|
|
|
|
|
|
|
|
<template>
|
2024-06-13 17:13:09 +07:00
|
|
|
<ButtonAddComponent style="z-index: 999">
|
|
|
|
|
<q-fab-action
|
2024-06-14 13:37:11 +07:00
|
|
|
id="btn-add-personne"
|
2024-06-13 17:13:09 +07:00
|
|
|
:label="$t('personnelAdd')"
|
|
|
|
|
external-label
|
|
|
|
|
label-position="left"
|
|
|
|
|
@click="openDialog('FORM')"
|
|
|
|
|
color="primary"
|
|
|
|
|
padding="xs"
|
|
|
|
|
icon="mdi-account-plus"
|
|
|
|
|
></q-fab-action>
|
|
|
|
|
</ButtonAddComponent>
|
|
|
|
|
|
2024-07-04 13:14:21 +07:00
|
|
|
<!-- stat -->
|
2024-07-04 09:53:33 +00:00
|
|
|
<div class="column full-height no-wrap">
|
2024-07-05 03:29:33 +00:00
|
|
|
<div class="text-body-2 q-mb-xs flex items-center">
|
|
|
|
|
{{ $t('personnelStatTitle') }}
|
2024-07-05 11:01:40 +00:00
|
|
|
<q-badge
|
|
|
|
|
v-if="typeStats"
|
|
|
|
|
rounded
|
|
|
|
|
class="q-ml-sm"
|
|
|
|
|
style="
|
|
|
|
|
background-color: hsla(var(--info-bg) / 0.15);
|
|
|
|
|
color: hsl(var(--info-bg));
|
|
|
|
|
"
|
|
|
|
|
>
|
|
|
|
|
{{
|
|
|
|
|
selectorLabel === 'ALL'
|
|
|
|
|
? Object.values(typeStats).reduce((acc, val) => acc + val, 0)
|
|
|
|
|
: typeStats[selectorLabel]
|
|
|
|
|
}}
|
|
|
|
|
</q-badge>
|
2024-07-05 03:29:33 +00:00
|
|
|
<q-btn
|
2024-07-05 11:01:40 +00:00
|
|
|
class="q-ml-sm"
|
2024-07-05 03:29:33 +00:00
|
|
|
icon="mdi-pin"
|
|
|
|
|
color="primary"
|
|
|
|
|
size="sm"
|
|
|
|
|
flat
|
|
|
|
|
dense
|
|
|
|
|
rounded
|
|
|
|
|
@click="hideStat = !hideStat"
|
|
|
|
|
:style="hideStat ? 'rotate: 90deg' : ''"
|
|
|
|
|
style="transition: 0.1s ease-in-out"
|
|
|
|
|
/>
|
2024-07-04 13:14:21 +07:00
|
|
|
</div>
|
2024-07-02 09:11:39 +00:00
|
|
|
|
2024-07-05 03:29:33 +00:00
|
|
|
<transition name="slide">
|
|
|
|
|
<div v-if="!hideStat" class="scroll q-mb-md">
|
|
|
|
|
<div style="display: inline-block">
|
|
|
|
|
<StatCardComponent
|
2024-07-05 11:01:40 +00:00
|
|
|
v-if="typeStats && userData?.result"
|
|
|
|
|
labelI18n
|
2024-07-05 03:29:33 +00:00
|
|
|
:branch="
|
2024-07-05 11:01:40 +00:00
|
|
|
selectorLabel === 'ALL'
|
|
|
|
|
? Object.entries(typeStats).map(([key, val]) => ({
|
|
|
|
|
count: val,
|
|
|
|
|
label: key,
|
|
|
|
|
icon: 'mdi-account',
|
|
|
|
|
color:
|
|
|
|
|
(
|
|
|
|
|
{
|
|
|
|
|
USER: 'cyan',
|
|
|
|
|
MESSENGER: 'yellow',
|
|
|
|
|
DELEGATE: 'red',
|
|
|
|
|
AGENCY: 'magenta',
|
|
|
|
|
} as const
|
|
|
|
|
)[key] || 'pink',
|
|
|
|
|
}))
|
|
|
|
|
: [
|
2024-07-05 16:29:06 +07:00
|
|
|
{
|
2024-07-05 11:01:40 +00:00
|
|
|
label: selectorLabel,
|
|
|
|
|
count: typeStats[selectorLabel],
|
|
|
|
|
icon: 'mdi-account',
|
|
|
|
|
color:
|
|
|
|
|
selectorLabel === 'USER'
|
|
|
|
|
? 'cyan'
|
|
|
|
|
: selectorLabel === 'MESSENGER'
|
|
|
|
|
? 'yellow'
|
|
|
|
|
: selectorLabel === 'DELEGATE'
|
|
|
|
|
? 'red'
|
|
|
|
|
: 'magenta',
|
|
|
|
|
},
|
|
|
|
|
]
|
2024-07-05 03:29:33 +00:00
|
|
|
"
|
|
|
|
|
:dark="$q.dark.isActive"
|
|
|
|
|
/>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</transition>
|
|
|
|
|
|
2024-07-04 09:53:33 +00:00
|
|
|
<!-- main -->
|
2024-07-04 14:25:24 +07:00
|
|
|
<div
|
2024-07-16 02:11:18 +00:00
|
|
|
class="col surface-2 rounded justify-between column no-wrap bordered"
|
2024-07-05 11:10:16 +07:00
|
|
|
style="overflow: hidden"
|
2024-07-04 14:25:24 +07:00
|
|
|
>
|
2024-07-05 11:01:40 +00:00
|
|
|
<div class="column">
|
|
|
|
|
<div
|
2024-07-17 08:31:31 +00:00
|
|
|
class="row surface-3 justify-between full-width items-center bordered-b"
|
2024-07-05 11:01:40 +00:00
|
|
|
style="z-index: 1"
|
|
|
|
|
>
|
|
|
|
|
<div class="row q-py-sm q-px-md justify-between full-width">
|
|
|
|
|
<q-input
|
|
|
|
|
for="input-search"
|
|
|
|
|
outlined
|
|
|
|
|
dense
|
|
|
|
|
:label="$t('search')"
|
2024-07-16 02:52:53 +00:00
|
|
|
class="q-mr-md col-12 col-md-3"
|
2024-07-05 11:01:40 +00:00
|
|
|
:bg-color="$q.dark.isActive ? 'dark' : 'white'"
|
|
|
|
|
v-model="inputSearch"
|
|
|
|
|
debounce="200"
|
|
|
|
|
>
|
|
|
|
|
<template #prepend>
|
|
|
|
|
<q-icon name="mdi-magnify" />
|
|
|
|
|
</template>
|
|
|
|
|
</q-input>
|
|
|
|
|
|
2024-07-16 02:52:53 +00:00
|
|
|
<div
|
2024-07-17 08:19:21 +00:00
|
|
|
class="row col-12 col-md-5"
|
2024-07-16 02:52:53 +00:00
|
|
|
:class="{ 'q-pt-xs': $q.screen.lt.md }"
|
2024-07-17 08:19:21 +00:00
|
|
|
style="white-space: nowrap"
|
2024-07-16 02:52:53 +00:00
|
|
|
>
|
2024-07-05 11:01:40 +00:00
|
|
|
<q-select
|
|
|
|
|
v-model="statusFilter"
|
|
|
|
|
outlined
|
|
|
|
|
dense
|
|
|
|
|
option-value="value"
|
|
|
|
|
option-label="label"
|
2024-07-16 02:52:53 +00:00
|
|
|
class="col"
|
2024-07-05 11:01:40 +00:00
|
|
|
map-options
|
|
|
|
|
emit-value
|
2024-07-18 04:51:09 +00:00
|
|
|
:hide-dropdown-icon="$q.screen.lt.sm"
|
2024-07-05 11:01:40 +00:00
|
|
|
:options="[
|
|
|
|
|
{ label: $t('all'), value: 'all' },
|
|
|
|
|
{ label: $t('statusACTIVE'), value: 'statusACTIVE' },
|
|
|
|
|
{ label: $t('statusINACTIVE'), value: 'statusINACTIVE' },
|
|
|
|
|
]"
|
|
|
|
|
></q-select>
|
|
|
|
|
|
|
|
|
|
<q-select
|
2024-07-09 11:21:25 +07:00
|
|
|
v-if="!modeView"
|
2024-07-05 11:01:40 +00:00
|
|
|
id="select-field"
|
|
|
|
|
for="select-field"
|
2024-07-16 02:52:53 +00:00
|
|
|
class="col q-mx-sm"
|
2024-07-09 11:21:25 +07:00
|
|
|
:options="
|
|
|
|
|
fieldSelectedOption.map((v) => ({
|
|
|
|
|
...v,
|
|
|
|
|
label: $t(v.label),
|
|
|
|
|
}))
|
|
|
|
|
"
|
2024-07-05 11:01:40 +00:00
|
|
|
:display-value="$t('displayField')"
|
2024-07-18 04:51:09 +00:00
|
|
|
:hide-dropdown-icon="$q.screen.lt.sm"
|
2024-07-05 11:01:40 +00:00
|
|
|
v-model="fieldSelected"
|
|
|
|
|
option-label="label"
|
|
|
|
|
option-value="value"
|
|
|
|
|
map-options
|
|
|
|
|
emit-value
|
|
|
|
|
outlined
|
|
|
|
|
multiple
|
|
|
|
|
dense
|
|
|
|
|
/>
|
|
|
|
|
|
|
|
|
|
<q-btn-toggle
|
|
|
|
|
id="btn-mode"
|
|
|
|
|
v-model="modeView"
|
|
|
|
|
dense
|
2024-07-16 02:52:53 +00:00
|
|
|
class="no-shadow bordered rounded surface-1"
|
2024-07-05 11:01:40 +00:00
|
|
|
:toggle-color="$q.dark.isActive ? 'grey-9' : 'grey-2'"
|
|
|
|
|
size="xs"
|
|
|
|
|
:options="[
|
|
|
|
|
{ value: true, slot: 'folder' },
|
|
|
|
|
{ value: false, slot: 'list' },
|
|
|
|
|
]"
|
|
|
|
|
>
|
|
|
|
|
<template v-slot:folder>
|
|
|
|
|
<q-icon
|
|
|
|
|
name="mdi-view-grid-outline"
|
|
|
|
|
size="16px"
|
|
|
|
|
class="q-px-sm q-py-xs rounded"
|
|
|
|
|
:style="{
|
|
|
|
|
color: $q.dark.isActive
|
|
|
|
|
? modeView
|
|
|
|
|
? '#C9D3DB '
|
|
|
|
|
: '#787B7C'
|
|
|
|
|
: modeView
|
|
|
|
|
? '#787B7C'
|
|
|
|
|
: '#C9D3DB',
|
|
|
|
|
}"
|
|
|
|
|
/>
|
|
|
|
|
</template>
|
|
|
|
|
<template v-slot:list>
|
|
|
|
|
<q-icon
|
|
|
|
|
name="mdi-format-list-bulleted"
|
|
|
|
|
class="q-px-sm q-py-xs rounded"
|
|
|
|
|
size="16px"
|
|
|
|
|
:style="{
|
|
|
|
|
color: $q.dark.isActive
|
|
|
|
|
? modeView === false
|
|
|
|
|
? '#C9D3DB'
|
|
|
|
|
: '#787B7C'
|
|
|
|
|
: modeView === false
|
|
|
|
|
? '#787B7C'
|
|
|
|
|
: '#C9D3DB',
|
|
|
|
|
}"
|
|
|
|
|
/>
|
|
|
|
|
</template>
|
|
|
|
|
</q-btn-toggle>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div class="surface-2 bordered-b q-px-md">
|
|
|
|
|
<q-tabs
|
2024-07-04 09:53:33 +00:00
|
|
|
dense
|
2024-07-05 11:01:40 +00:00
|
|
|
v-model="selectorLabel"
|
|
|
|
|
align="left"
|
|
|
|
|
class="full-height"
|
|
|
|
|
active-color="info"
|
2024-07-04 09:53:33 +00:00
|
|
|
>
|
2024-07-05 11:01:40 +00:00
|
|
|
<q-tab
|
2024-07-05 11:10:46 +00:00
|
|
|
name="ALL"
|
2024-07-05 11:01:40 +00:00
|
|
|
@click="
|
|
|
|
|
async () => {
|
|
|
|
|
currentPage = 1;
|
|
|
|
|
inputSearch = '';
|
|
|
|
|
selectorLabel = 'ALL';
|
|
|
|
|
statusFilter = 'all';
|
|
|
|
|
flowStore.rotate();
|
|
|
|
|
}
|
|
|
|
|
"
|
|
|
|
|
>
|
|
|
|
|
<div
|
|
|
|
|
class="row"
|
|
|
|
|
:class="
|
|
|
|
|
selectorLabel === 'ALL' ? 'text-bold' : 'app-text-muted'
|
|
|
|
|
"
|
|
|
|
|
>
|
|
|
|
|
{{ $t('all') }}
|
|
|
|
|
</div>
|
|
|
|
|
</q-tab>
|
|
|
|
|
<q-tab
|
2024-07-05 11:10:46 +00:00
|
|
|
name="USER"
|
2024-07-05 11:01:40 +00:00
|
|
|
@click="
|
|
|
|
|
async () => {
|
|
|
|
|
selectorLabel = 'USER';
|
|
|
|
|
statusFilter = 'all';
|
|
|
|
|
currentPage = 1;
|
|
|
|
|
inputSearch = '';
|
|
|
|
|
flowStore.rotate();
|
|
|
|
|
}
|
|
|
|
|
"
|
|
|
|
|
>
|
|
|
|
|
<div
|
|
|
|
|
class="row"
|
|
|
|
|
:class="
|
|
|
|
|
selectorLabel === 'USER' ? 'text-bold' : 'app-text-muted'
|
|
|
|
|
"
|
|
|
|
|
>
|
|
|
|
|
{{ $t('USER') }}
|
|
|
|
|
</div>
|
|
|
|
|
</q-tab>
|
|
|
|
|
<q-tab
|
|
|
|
|
name="MESSENGER"
|
|
|
|
|
@click="
|
|
|
|
|
async () => {
|
|
|
|
|
selectorLabel = 'MESSENGER';
|
|
|
|
|
statusFilter = 'all';
|
|
|
|
|
currentPage = 1;
|
|
|
|
|
inputSearch = '';
|
|
|
|
|
flowStore.rotate();
|
|
|
|
|
}
|
|
|
|
|
"
|
|
|
|
|
>
|
|
|
|
|
<div
|
|
|
|
|
class="row"
|
|
|
|
|
:class="
|
|
|
|
|
selectorLabel === 'MESSENGER' ? 'text-bold' : 'app-text-muted'
|
|
|
|
|
"
|
|
|
|
|
>
|
|
|
|
|
{{ $t('MESSENGER') }}
|
|
|
|
|
</div>
|
|
|
|
|
</q-tab>
|
|
|
|
|
<q-tab
|
|
|
|
|
name="DELEGATE"
|
|
|
|
|
@click="
|
|
|
|
|
async () => {
|
|
|
|
|
selectorLabel = 'DELEGATE';
|
|
|
|
|
statusFilter = 'all';
|
|
|
|
|
currentPage = 1;
|
|
|
|
|
inputSearch = '';
|
|
|
|
|
flowStore.rotate();
|
|
|
|
|
}
|
|
|
|
|
"
|
|
|
|
|
>
|
|
|
|
|
<div
|
|
|
|
|
class="row"
|
|
|
|
|
:class="
|
|
|
|
|
selectorLabel === 'DELEGATE' ? 'text-bold' : 'app-text-muted'
|
|
|
|
|
"
|
|
|
|
|
>
|
|
|
|
|
{{ $t('DELEGATE') }}
|
|
|
|
|
</div>
|
|
|
|
|
</q-tab>
|
|
|
|
|
<q-tab
|
|
|
|
|
name="AGENCY"
|
|
|
|
|
@click="
|
|
|
|
|
async () => {
|
|
|
|
|
selectorLabel = 'AGENCY';
|
|
|
|
|
statusFilter = 'all';
|
|
|
|
|
currentPage = 1;
|
|
|
|
|
inputSearch = '';
|
|
|
|
|
flowStore.rotate();
|
|
|
|
|
}
|
|
|
|
|
"
|
|
|
|
|
>
|
|
|
|
|
<div
|
|
|
|
|
class="row"
|
|
|
|
|
:class="
|
|
|
|
|
selectorLabel === 'AGENCY' ? 'text-bold' : 'app-text-muted'
|
|
|
|
|
"
|
|
|
|
|
>
|
|
|
|
|
{{ $t('AGENCY') }}
|
|
|
|
|
</div>
|
|
|
|
|
</q-tab>
|
|
|
|
|
</q-tabs>
|
2024-07-04 09:53:33 +00:00
|
|
|
</div>
|
2024-07-05 11:10:16 +07:00
|
|
|
</div>
|
2024-07-05 11:01:40 +00:00
|
|
|
|
2024-07-18 02:26:03 +00:00
|
|
|
<div class="col scroll q-pa-md full-width">
|
2024-07-09 11:21:25 +07:00
|
|
|
<div v-if="userData && userData.total > 0 && !inputSearch">
|
|
|
|
|
<q-table
|
|
|
|
|
flat
|
|
|
|
|
bordered
|
|
|
|
|
:grid="modeView"
|
|
|
|
|
:rows="
|
|
|
|
|
userData.result.filter((v) => {
|
|
|
|
|
if (
|
|
|
|
|
statusFilter === 'statusACTIVE' &&
|
|
|
|
|
v.status === 'INACTIVE'
|
|
|
|
|
) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (
|
|
|
|
|
statusFilter === 'statusINACTIVE' &&
|
|
|
|
|
v.status !== 'INACTIVE'
|
|
|
|
|
) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
})
|
|
|
|
|
"
|
|
|
|
|
:columns="columns"
|
2024-07-10 09:44:10 +07:00
|
|
|
class="full-width"
|
|
|
|
|
card-container-class="q-gutter-md "
|
2024-07-09 11:21:25 +07:00
|
|
|
row-key="name"
|
|
|
|
|
:rows-per-page-options="[0]"
|
|
|
|
|
hide-pagination
|
|
|
|
|
:visible-columns="fieldSelected"
|
|
|
|
|
>
|
|
|
|
|
<template v-slot:header="props">
|
|
|
|
|
<q-tr class="surface-2" :props="props">
|
|
|
|
|
<q-th v-for="col in props.cols" :key="col.name" :props="props">
|
|
|
|
|
{{ $t(col.label) }}
|
|
|
|
|
</q-th>
|
|
|
|
|
<q-th auto-width />
|
|
|
|
|
</q-tr>
|
|
|
|
|
</template>
|
|
|
|
|
|
|
|
|
|
<template v-slot:body="props">
|
|
|
|
|
<q-tr
|
|
|
|
|
:class="{
|
|
|
|
|
'app-text-muted': props.row.status === 'INACTIVE',
|
|
|
|
|
'status-active': props.row.status !== 'INACTIVE',
|
|
|
|
|
'status-inactive': props.row.status === 'INACTIVE',
|
|
|
|
|
}"
|
|
|
|
|
:props="props"
|
|
|
|
|
>
|
|
|
|
|
<q-td v-if="fieldSelected.includes('name')">
|
|
|
|
|
<div class="row items-center">
|
|
|
|
|
<div
|
|
|
|
|
style="
|
|
|
|
|
width: 50px;
|
|
|
|
|
display: flex;
|
|
|
|
|
margin-bottom: var(--size-2);
|
|
|
|
|
"
|
|
|
|
|
>
|
|
|
|
|
<div class="branch-card__icon">
|
|
|
|
|
<q-avatar size="md">
|
|
|
|
|
<q-img
|
|
|
|
|
:src="
|
|
|
|
|
props.row.profileImageUrl ?? '/no-profile.png'
|
|
|
|
|
"
|
|
|
|
|
>
|
|
|
|
|
<template #error>
|
|
|
|
|
<q-img src="/no-profile.png" />
|
|
|
|
|
</template>
|
|
|
|
|
</q-img>
|
|
|
|
|
</q-avatar>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
<div class="col">
|
|
|
|
|
<div class="col">
|
|
|
|
|
{{
|
|
|
|
|
$i18n.locale === 'en-US'
|
|
|
|
|
? `${props.row.firstNameEN} ${props.row.lastNameEN}`.trim()
|
|
|
|
|
: `${props.row.firstName} ${props.row.lastName}`.trim()
|
|
|
|
|
}}
|
|
|
|
|
|
|
|
|
|
<q-icon
|
|
|
|
|
class="q-pl-xs"
|
|
|
|
|
:class="{
|
|
|
|
|
'symbol-gender': props.row.gender,
|
|
|
|
|
'symbol-gender__male': props.row.gender === 'male',
|
|
|
|
|
'symbol-gender__female':
|
|
|
|
|
props.row.gender === 'female',
|
|
|
|
|
}"
|
|
|
|
|
:name="`mdi-gender-${props.row.gender === 'male' ? 'male' : 'female'}`"
|
|
|
|
|
width="24px"
|
|
|
|
|
/>
|
|
|
|
|
</div>
|
|
|
|
|
<div class="col app-text-muted">
|
|
|
|
|
{{ props.row.code || '-' }}
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</q-td>
|
|
|
|
|
<q-td v-if="fieldSelected.includes('type')">
|
|
|
|
|
<span
|
|
|
|
|
class="tags"
|
|
|
|
|
:class="{
|
|
|
|
|
[`tags__${
|
|
|
|
|
{
|
|
|
|
|
USER: 'cyan',
|
|
|
|
|
MESSENGER: 'yellow',
|
|
|
|
|
DELEGATE: 'red',
|
|
|
|
|
AGENCY: 'magenta',
|
|
|
|
|
}[props.row.userType as string] || 'pink'
|
|
|
|
|
} }`]: true,
|
|
|
|
|
}"
|
|
|
|
|
>
|
|
|
|
|
{{
|
|
|
|
|
$t(
|
|
|
|
|
{
|
|
|
|
|
USER: 'USER',
|
|
|
|
|
MESSENGER: 'MESSENGER',
|
|
|
|
|
DELEGATE: 'DELEGATE',
|
|
|
|
|
AGENCY: 'AGENCY',
|
|
|
|
|
}[props.row.userType as string] || 'USER',
|
|
|
|
|
)
|
|
|
|
|
}}
|
|
|
|
|
</span>
|
|
|
|
|
</q-td>
|
|
|
|
|
|
|
|
|
|
<q-td v-if="fieldSelected.includes('telephoneNo')">
|
|
|
|
|
{{ props.row.telephoneNo || '-' }}
|
|
|
|
|
</q-td>
|
|
|
|
|
|
|
|
|
|
<q-td v-if="fieldSelected.includes('birthDate')">
|
|
|
|
|
{{ calculateAge(props.row.birthDate) || '-' }}
|
|
|
|
|
</q-td>
|
|
|
|
|
|
|
|
|
|
<q-td v-if="fieldSelected.includes('email')">
|
|
|
|
|
{{ props.row.email || '-' }}
|
|
|
|
|
</q-td>
|
|
|
|
|
|
|
|
|
|
<q-td v-if="fieldSelected.includes('userRole')">
|
|
|
|
|
{{ props.row.userRole || '-' }}
|
|
|
|
|
</q-td>
|
|
|
|
|
|
|
|
|
|
<q-td>
|
|
|
|
|
<q-btn
|
|
|
|
|
icon="mdi-eye-outline"
|
|
|
|
|
size="sm"
|
|
|
|
|
dense
|
|
|
|
|
round
|
|
|
|
|
flat
|
|
|
|
|
@click.stop="
|
|
|
|
|
() => {
|
|
|
|
|
openDialog('INFO', props.row.id);
|
|
|
|
|
}
|
|
|
|
|
"
|
|
|
|
|
/>
|
|
|
|
|
|
|
|
|
|
<q-btn
|
|
|
|
|
icon="mdi-dots-vertical"
|
|
|
|
|
size="sm"
|
|
|
|
|
dense
|
|
|
|
|
round
|
|
|
|
|
flat
|
|
|
|
|
@click.stop
|
|
|
|
|
:key="props.row.id"
|
|
|
|
|
>
|
|
|
|
|
<q-menu class="bordered">
|
|
|
|
|
<q-list v-close-popup>
|
|
|
|
|
<q-item
|
|
|
|
|
:id="`view-detail-btn-${props.row.name}-view`"
|
|
|
|
|
@click.stop="
|
|
|
|
|
() => {
|
|
|
|
|
openDialog('INFO', props.row.id);
|
|
|
|
|
}
|
|
|
|
|
"
|
2024-07-16 02:11:18 +00:00
|
|
|
v-close-popup
|
|
|
|
|
clickable
|
2024-07-09 11:21:25 +07:00
|
|
|
dense
|
|
|
|
|
class="row q-py-sm"
|
|
|
|
|
style="white-space: nowrap"
|
|
|
|
|
>
|
|
|
|
|
<q-icon
|
|
|
|
|
name="mdi-eye-outline"
|
|
|
|
|
class="col-3"
|
|
|
|
|
size="xs"
|
|
|
|
|
style="color: hsl(var(--green-6-hsl))"
|
|
|
|
|
/>
|
|
|
|
|
<span class="col-9 q-px-md flex items-center">
|
|
|
|
|
{{ $t('viewDetail') }}
|
|
|
|
|
</span>
|
|
|
|
|
</q-item>
|
|
|
|
|
|
|
|
|
|
<q-item
|
|
|
|
|
:id="`view-detail-btn-${props.row.name}-edit`"
|
2024-07-16 02:11:18 +00:00
|
|
|
v-close-popup
|
|
|
|
|
clickable
|
2024-07-09 11:21:25 +07:00
|
|
|
dense
|
|
|
|
|
class="row q-py-sm"
|
|
|
|
|
style="white-space: nowrap"
|
|
|
|
|
@click="
|
|
|
|
|
() => {
|
|
|
|
|
openDialog('INFO', props.row.id, true);
|
|
|
|
|
}
|
|
|
|
|
"
|
|
|
|
|
>
|
|
|
|
|
<q-icon
|
|
|
|
|
name="mdi-pencil-outline"
|
|
|
|
|
class="col-3"
|
|
|
|
|
size="xs"
|
|
|
|
|
style="color: hsl(var(--cyan-6-hsl))"
|
|
|
|
|
/>
|
|
|
|
|
<span class="col-9 q-px-md flex items-center">
|
|
|
|
|
{{ $t('edit') }}
|
|
|
|
|
</span>
|
|
|
|
|
</q-item>
|
|
|
|
|
<q-item
|
|
|
|
|
:id="`view-detail-btn-${props.row.name}-delete`"
|
|
|
|
|
dense
|
2024-07-16 02:11:18 +00:00
|
|
|
v-close-popup
|
2024-07-09 11:21:25 +07:00
|
|
|
:clickable="props.row.status === 'CREATED'"
|
|
|
|
|
class="row"
|
|
|
|
|
:class="{
|
|
|
|
|
'surface-3': props.row.status !== 'CREATED',
|
|
|
|
|
'app-text-muted': props.row.status !== 'CREATED',
|
|
|
|
|
}"
|
|
|
|
|
style="white-space: nowrap"
|
|
|
|
|
@click="onDelete(props.row.id)"
|
|
|
|
|
>
|
|
|
|
|
<q-icon
|
|
|
|
|
name="mdi-trash-can-outline"
|
|
|
|
|
size="xs"
|
|
|
|
|
class="col-3"
|
|
|
|
|
:class="{
|
|
|
|
|
'app-text-negative':
|
|
|
|
|
props.row.status === 'CREATED',
|
|
|
|
|
}"
|
|
|
|
|
/>
|
|
|
|
|
<span class="col-9 q-px-md flex items-center">
|
|
|
|
|
{{ $t('delete') }}
|
|
|
|
|
</span>
|
|
|
|
|
</q-item>
|
|
|
|
|
|
|
|
|
|
<q-item dense>
|
|
|
|
|
<q-item-section class="q-py-sm">
|
|
|
|
|
<div class="q-pa-sm surface-2 rounded">
|
|
|
|
|
<q-toggle
|
|
|
|
|
:id="`view-detail-btn-${props.row.name}-status`"
|
|
|
|
|
dense
|
|
|
|
|
size="sm"
|
|
|
|
|
:label="
|
|
|
|
|
props.row.status !== 'INACTIVE'
|
|
|
|
|
? $t('switchOnLabel')
|
|
|
|
|
: $t('switchOffLabel')
|
|
|
|
|
"
|
|
|
|
|
@click="
|
|
|
|
|
async () => {
|
|
|
|
|
toggleStatus(props.row.id);
|
|
|
|
|
}
|
|
|
|
|
"
|
|
|
|
|
:model-value="props.row.status !== 'INACTIVE'"
|
|
|
|
|
/>
|
|
|
|
|
</div>
|
|
|
|
|
</q-item-section>
|
|
|
|
|
</q-item>
|
|
|
|
|
</q-list>
|
|
|
|
|
</q-menu>
|
|
|
|
|
</q-btn>
|
|
|
|
|
</q-td>
|
|
|
|
|
</q-tr>
|
|
|
|
|
</template>
|
|
|
|
|
|
|
|
|
|
<template v-slot:item="props">
|
|
|
|
|
<div class="col-3">
|
|
|
|
|
<PersonCard
|
|
|
|
|
:data="{
|
|
|
|
|
code: props.row.code,
|
|
|
|
|
name:
|
|
|
|
|
$i18n.locale === 'en-US'
|
|
|
|
|
? `${props.row.firstNameEN} ${props.row.lastNameEN}`.trim()
|
|
|
|
|
: `${props.row.firstName} ${props.row.lastName}`.trim(),
|
|
|
|
|
img: props.row.profileImageUrl,
|
|
|
|
|
male: props.row.gender === 'male',
|
|
|
|
|
female: props.row.gender === 'female',
|
|
|
|
|
detail: [
|
|
|
|
|
{
|
|
|
|
|
icon: 'mdi-phone',
|
|
|
|
|
value: props.row.telephoneNo,
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
icon: 'mdi-clock-outline',
|
|
|
|
|
value:
|
|
|
|
|
(props.row.birthDate &&
|
|
|
|
|
calculateAge(props.row.birthDate)) ||
|
|
|
|
|
'-',
|
|
|
|
|
},
|
|
|
|
|
],
|
|
|
|
|
}"
|
|
|
|
|
:tag="[
|
|
|
|
|
{
|
|
|
|
|
color:
|
|
|
|
|
{
|
|
|
|
|
USER: 'cyan',
|
|
|
|
|
MESSENGER: 'yellow',
|
|
|
|
|
DELEGATE: 'red',
|
|
|
|
|
AGENCY: 'magenta',
|
|
|
|
|
}[props.row.userType as string] || 'pink',
|
|
|
|
|
value: $t(props.row.userType),
|
|
|
|
|
},
|
|
|
|
|
]"
|
|
|
|
|
:disabled="props.row.status === 'INACTIVE'"
|
|
|
|
|
@update-card="(a, b) => openDialog(a, props.row.id, b)"
|
|
|
|
|
@delete-card="onDelete(props.row.id)"
|
|
|
|
|
@enter-card="(a) => openDialog(a, props.row.id)"
|
|
|
|
|
@toggle-status="toggleStatus(props.row.id)"
|
|
|
|
|
/>
|
|
|
|
|
</div>
|
|
|
|
|
</template>
|
|
|
|
|
</q-table>
|
|
|
|
|
|
|
|
|
|
<!-- <div
|
2024-07-05 16:29:06 +07:00
|
|
|
v-for="v in userData?.result.filter((v) => {
|
|
|
|
|
if (statusFilter === 'statusACTIVE' && v.status === 'INACTIVE') {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (
|
|
|
|
|
statusFilter === 'statusINACTIVE' &&
|
|
|
|
|
v.status !== 'INACTIVE'
|
|
|
|
|
) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
})"
|
|
|
|
|
:key="v.id"
|
|
|
|
|
class="col-3"
|
|
|
|
|
>
|
|
|
|
|
<PersonCard
|
|
|
|
|
:data="{
|
|
|
|
|
code: v.code,
|
2024-07-04 09:53:33 +00:00
|
|
|
name:
|
|
|
|
|
$i18n.locale === 'en-US'
|
2024-07-05 16:29:06 +07:00
|
|
|
? `${v.firstNameEN} ${v.lastNameEN}`.trim()
|
|
|
|
|
: `${v.firstName} ${v.lastName}`.trim(),
|
|
|
|
|
img: v.profileImageUrl,
|
2024-07-04 09:53:33 +00:00
|
|
|
male: v.gender === 'male',
|
|
|
|
|
female: v.gender === 'female',
|
|
|
|
|
detail: [
|
|
|
|
|
{
|
2024-07-05 16:29:06 +07:00
|
|
|
icon: 'mdi-phone',
|
|
|
|
|
value: v.telephoneNo,
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
icon: 'mdi-clock-outline',
|
|
|
|
|
value: (v.birthDate && calculateAge(v.birthDate)) || '-',
|
2024-07-04 09:53:33 +00:00
|
|
|
},
|
|
|
|
|
],
|
2024-07-05 16:29:06 +07:00
|
|
|
}"
|
|
|
|
|
:tag="[
|
|
|
|
|
{
|
|
|
|
|
color:
|
|
|
|
|
(
|
|
|
|
|
{
|
|
|
|
|
USER: 'cyan',
|
|
|
|
|
MESSENGER: 'yellow',
|
|
|
|
|
DELEGATE: 'red',
|
|
|
|
|
AGENCY: 'magenta',
|
|
|
|
|
} as const
|
|
|
|
|
)[v.userType] || 'pink',
|
|
|
|
|
value: $t(v.userType),
|
|
|
|
|
},
|
|
|
|
|
]"
|
|
|
|
|
:disabled="v.status === 'INACTIVE'"
|
|
|
|
|
@update-card="(a, b) => openDialog(a, v.id, b)"
|
|
|
|
|
@delete-card="onDelete(v.id)"
|
|
|
|
|
@enter-card="(a) => openDialog(a, v.id)"
|
|
|
|
|
@toggle-status="toggleStatus(v.id)"
|
|
|
|
|
/>
|
2024-07-09 11:21:25 +07:00
|
|
|
</div> -->
|
2024-07-05 16:29:06 +07:00
|
|
|
</div>
|
2024-07-05 11:01:40 +00:00
|
|
|
|
|
|
|
|
<div
|
|
|
|
|
v-if="userData && userData.total === 0 && !inputSearch"
|
2024-07-18 02:26:03 +00:00
|
|
|
class="full-height column"
|
2024-07-05 11:01:40 +00:00
|
|
|
>
|
2024-07-17 08:19:21 +00:00
|
|
|
<div class="flex justify-end">
|
2024-07-05 11:01:40 +00:00
|
|
|
<TooltipComponent
|
|
|
|
|
title="personnelTooltipTitle"
|
|
|
|
|
caption="personnelTooltipCaption"
|
|
|
|
|
imgSrc="personnel-table-"
|
|
|
|
|
/>
|
2024-07-04 14:25:24 +07:00
|
|
|
</div>
|
2024-07-05 11:01:40 +00:00
|
|
|
|
|
|
|
|
<div class="col items-center flex justify-center">
|
2024-07-04 09:53:33 +00:00
|
|
|
<AddButton
|
|
|
|
|
:label="'personnelAdd'"
|
|
|
|
|
:cyanOn="true"
|
|
|
|
|
@trigger="openDialog('FORM')"
|
|
|
|
|
/>
|
|
|
|
|
</div>
|
2024-07-05 11:01:40 +00:00
|
|
|
</div>
|
2024-07-04 09:53:33 +00:00
|
|
|
|
|
|
|
|
<div
|
|
|
|
|
v-if="userData?.total === 0 && !!inputSearch"
|
2024-07-05 11:01:40 +00:00
|
|
|
class="row col full-width items-center justify-center"
|
2024-07-04 09:53:33 +00:00
|
|
|
style="min-height: 250px"
|
|
|
|
|
>
|
|
|
|
|
<NoData :not-found="!!inputSearch" />
|
2024-07-04 14:25:24 +07:00
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
|
2024-07-04 09:53:33 +00:00
|
|
|
<div
|
2024-07-05 11:10:16 +07:00
|
|
|
class="row justify-between items-center q-px-md q-py-sm"
|
2024-07-04 09:53:33 +00:00
|
|
|
v-if="currentMaxPage > 0"
|
|
|
|
|
>
|
|
|
|
|
<div class="col-4">
|
|
|
|
|
<div class="row items-center">
|
|
|
|
|
<div class="app-text-muted" style="width: 80px">
|
|
|
|
|
{{ $t('showing') }}
|
|
|
|
|
</div>
|
|
|
|
|
<div>
|
2024-07-05 11:10:16 +07:00
|
|
|
<q-btn-dropdown
|
|
|
|
|
dense
|
|
|
|
|
unelevated
|
|
|
|
|
:label="pageSize"
|
|
|
|
|
class="bordered q-pl-md"
|
|
|
|
|
>
|
2024-07-04 09:53:33 +00:00
|
|
|
<q-list>
|
|
|
|
|
<q-item
|
|
|
|
|
v-for="v in [10, 30, 50, 100, 500, 1000]"
|
|
|
|
|
:key="v"
|
|
|
|
|
clickable
|
|
|
|
|
v-close-popup
|
|
|
|
|
@click="
|
|
|
|
|
async () => {
|
|
|
|
|
pageSize = v;
|
|
|
|
|
await fetchUserList();
|
|
|
|
|
}
|
|
|
|
|
"
|
|
|
|
|
>
|
|
|
|
|
<q-item-section>
|
|
|
|
|
<q-item-label>{{ v }}</q-item-label>
|
|
|
|
|
</q-item-section>
|
|
|
|
|
</q-item>
|
|
|
|
|
</q-list>
|
|
|
|
|
</q-btn-dropdown>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div class="col-4 row justify-center app-text-muted">
|
|
|
|
|
{{
|
|
|
|
|
$t('recordsPage', {
|
|
|
|
|
resultcurrentPage: userData?.result.length,
|
|
|
|
|
total: userData?.total,
|
|
|
|
|
})
|
|
|
|
|
}}
|
|
|
|
|
</div>
|
|
|
|
|
<div class="col-4 row justify-end">
|
|
|
|
|
<PaginationComponent
|
|
|
|
|
v-model:current-page="currentPage"
|
|
|
|
|
v-model:max-page="currentMaxPage"
|
|
|
|
|
:fetch-data="async () => await fetchUserList()"
|
|
|
|
|
/>
|
|
|
|
|
</div>
|
2024-07-04 14:25:24 +07:00
|
|
|
</div>
|
2024-07-02 09:11:39 +00:00
|
|
|
</div>
|
2024-04-03 18:28:59 +07:00
|
|
|
</div>
|
2024-04-05 19:05:51 +07:00
|
|
|
|
2024-04-18 18:27:22 +07:00
|
|
|
<DrawerInfo
|
2024-04-19 09:06:38 +07:00
|
|
|
v-if="currentUser"
|
2024-04-22 17:52:37 +07:00
|
|
|
:category="$t('personnelTitle')"
|
2024-04-19 10:49:14 +07:00
|
|
|
bg-on
|
2024-06-27 08:32:31 +00:00
|
|
|
:badgeClass="
|
|
|
|
|
formData.gender === 'male'
|
|
|
|
|
? 'app-bg-male text-white'
|
2024-07-03 02:46:27 +00:00
|
|
|
: formData.gender === 'female'
|
|
|
|
|
? 'app-bg-female text-white'
|
|
|
|
|
: ''
|
2024-06-27 08:32:31 +00:00
|
|
|
"
|
2024-04-22 14:04:06 +07:00
|
|
|
:badgeLabel="userCode"
|
2024-04-25 04:07:12 +00:00
|
|
|
:isEdit="infoDrawerEdit"
|
2024-04-18 18:27:22 +07:00
|
|
|
:title="
|
|
|
|
|
$i18n.locale === 'en-US'
|
2024-04-19 09:06:38 +07:00
|
|
|
? `${currentUser.firstNameEN} ${currentUser.lastNameEN}`
|
|
|
|
|
: `${currentUser.firstName} ${currentUser.lastName}`
|
2024-04-18 18:27:22 +07:00
|
|
|
"
|
|
|
|
|
v-model:drawerOpen="infoDrawer"
|
2024-04-19 09:06:38 +07:00
|
|
|
:deleteData="() => onDelete(infoPersonId)"
|
|
|
|
|
:submit="() => onSubmit()"
|
2024-04-18 18:27:22 +07:00
|
|
|
:close="() => onClose()"
|
2024-04-19 09:06:38 +07:00
|
|
|
:undo="() => undo()"
|
2024-04-25 04:07:12 +00:00
|
|
|
:editData="() => (infoDrawerEdit = true)"
|
2024-04-18 18:27:22 +07:00
|
|
|
>
|
|
|
|
|
<template #info>
|
2024-04-19 13:54:57 +07:00
|
|
|
<InfoForm
|
2024-04-25 04:07:12 +00:00
|
|
|
:readonly="!infoDrawerEdit"
|
2024-04-18 18:27:22 +07:00
|
|
|
v-model:address="formData.address"
|
|
|
|
|
v-model:addressEN="formData.addressEN"
|
|
|
|
|
v-model:provinceId="formData.provinceId"
|
|
|
|
|
v-model:districtId="formData.districtId"
|
|
|
|
|
v-model:subDistrictId="formData.subDistrictId"
|
|
|
|
|
v-model:zipCode="formData.zipCode"
|
|
|
|
|
>
|
|
|
|
|
<template #person-card>
|
|
|
|
|
<div class="q-ma-md">
|
|
|
|
|
<AppBox class="surface-1" style="padding: 0">
|
|
|
|
|
<PersonCard
|
2024-06-27 08:32:31 +00:00
|
|
|
:can-edit-profile="infoDrawerEdit"
|
2024-07-05 16:29:06 +07:00
|
|
|
:data="{
|
|
|
|
|
code: currentUser.code,
|
|
|
|
|
name:
|
|
|
|
|
$i18n.locale === 'en-US'
|
|
|
|
|
? `${currentUser.firstNameEN} ${currentUser.lastNameEN}`.trim()
|
|
|
|
|
: `${currentUser.firstName} ${currentUser.lastName}`.trim(),
|
|
|
|
|
img: currentUser.profileImageUrl,
|
|
|
|
|
male: currentUser.gender === 'male',
|
|
|
|
|
female: currentUser.gender === 'female',
|
|
|
|
|
}"
|
2024-04-23 11:13:57 +00:00
|
|
|
no-hover
|
|
|
|
|
no-action
|
2024-04-18 18:27:22 +07:00
|
|
|
no-bg
|
2024-07-05 16:29:06 +07:00
|
|
|
@edit-profile="inputFile.click()"
|
2024-04-18 18:27:22 +07:00
|
|
|
/>
|
|
|
|
|
</AppBox>
|
|
|
|
|
</div>
|
|
|
|
|
</template>
|
|
|
|
|
<template #information>
|
|
|
|
|
<FormInformation
|
|
|
|
|
dense
|
|
|
|
|
outlined
|
|
|
|
|
separator
|
2024-04-25 04:07:12 +00:00
|
|
|
:readonly="!infoDrawerEdit"
|
2024-04-18 18:27:22 +07:00
|
|
|
:usernameReadonly="isEdit"
|
|
|
|
|
v-model:hqId="hqId"
|
|
|
|
|
v-model:brId="brId"
|
|
|
|
|
v-model:userType="formData.userType"
|
|
|
|
|
v-model:userRole="formData.userRole"
|
|
|
|
|
v-model:username="formData.username"
|
2024-04-22 14:04:06 +07:00
|
|
|
v-model:userCode="userCode"
|
2024-04-18 18:27:22 +07:00
|
|
|
/>
|
|
|
|
|
</template>
|
|
|
|
|
<template #person>
|
|
|
|
|
<FormPerson
|
|
|
|
|
dense
|
|
|
|
|
outlined
|
|
|
|
|
separator
|
2024-04-25 04:07:12 +00:00
|
|
|
:readonly="!infoDrawerEdit"
|
2024-04-18 18:27:22 +07:00
|
|
|
v-model:firstName="formData.firstName"
|
|
|
|
|
v-model:lastName="formData.lastName"
|
|
|
|
|
v-model:firstNameEN="formData.firstNameEN"
|
|
|
|
|
v-model:lastNameEN="formData.lastNameEN"
|
|
|
|
|
v-model:telephoneNo="formData.telephoneNo"
|
|
|
|
|
v-model:email="formData.email"
|
|
|
|
|
v-model:gender="formData.gender"
|
|
|
|
|
v-model:birthDate="formData.birthDate"
|
|
|
|
|
/>
|
|
|
|
|
</template>
|
|
|
|
|
<template #by-type>
|
|
|
|
|
<FormByType
|
|
|
|
|
dense
|
|
|
|
|
outlined
|
|
|
|
|
separator
|
2024-04-25 04:07:12 +00:00
|
|
|
:readonly="!infoDrawerEdit"
|
2024-04-18 18:27:22 +07:00
|
|
|
v-model:userType="formData.userType"
|
|
|
|
|
v-model:registrationNo="formData.registrationNo"
|
|
|
|
|
v-model:startDate="formData.startDate"
|
|
|
|
|
v-model:retireDate="formData.retireDate"
|
|
|
|
|
v-model:responsibleArea="formData.responsibleArea"
|
|
|
|
|
v-model:discountCondition="formData.discountCondition"
|
|
|
|
|
v-model:sourceNationality="formData.sourceNationality"
|
|
|
|
|
v-model:importNationality="formData.importNationality"
|
|
|
|
|
v-model:trainingPlace="formData.trainingPlace"
|
2024-04-19 13:54:57 +07:00
|
|
|
v-model:checkpoint="formData.checkpoint"
|
|
|
|
|
v-model:checkpointEN="formData.checkpointEN"
|
|
|
|
|
v-model:agencyFile="agencyFile"
|
|
|
|
|
v-model:agencyFileList="agencyFileList"
|
|
|
|
|
v-model:userId="userId"
|
2024-04-18 18:27:22 +07:00
|
|
|
/>
|
|
|
|
|
</template>
|
2024-04-19 13:54:57 +07:00
|
|
|
</InfoForm>
|
2024-04-18 18:27:22 +07:00
|
|
|
</template>
|
|
|
|
|
</DrawerInfo>
|
|
|
|
|
|
2024-04-10 14:00:55 +07:00
|
|
|
<!-- form -->
|
2024-04-05 19:05:51 +07:00
|
|
|
<FormDialog
|
2024-04-18 18:27:22 +07:00
|
|
|
removeDialog
|
2024-04-10 23:17:39 +07:00
|
|
|
ref="formDialogRef"
|
2024-04-22 14:04:06 +07:00
|
|
|
:badgeClass="formData.gender === 'male' ? 'app-bg-male' : 'app-bg-female'"
|
|
|
|
|
:badgeLabel="userCode"
|
2024-04-22 10:54:32 +07:00
|
|
|
:title="$t('personnelAdd')"
|
2024-04-18 09:06:21 +07:00
|
|
|
v-model:modal="modal"
|
|
|
|
|
v-model:address="formData.address"
|
|
|
|
|
v-model:addressEN="formData.addressEN"
|
|
|
|
|
v-model:provinceId="formData.provinceId"
|
|
|
|
|
v-model:districtId="formData.districtId"
|
|
|
|
|
v-model:subDistrictId="formData.subDistrictId"
|
|
|
|
|
v-model:zipCode="formData.zipCode"
|
|
|
|
|
:addressSeparator="formData.userType !== ''"
|
2024-04-10 17:00:28 +07:00
|
|
|
:submit="() => onSubmit()"
|
2024-04-10 14:00:55 +07:00
|
|
|
:close="() => onClose()"
|
2024-04-05 19:05:51 +07:00
|
|
|
>
|
|
|
|
|
<template #prepend>
|
2024-06-27 08:32:31 +00:00
|
|
|
<ProfileUpload
|
|
|
|
|
v-model:url-profile="urlProfile"
|
|
|
|
|
v-model:status-toggle="statusToggle"
|
|
|
|
|
v-model:profile-submit="profileSubmit"
|
|
|
|
|
@input-file="inputFile.click()"
|
|
|
|
|
/>
|
2024-04-05 19:05:51 +07:00
|
|
|
</template>
|
|
|
|
|
|
2024-04-17 15:44:13 +07:00
|
|
|
<template #information>
|
|
|
|
|
<FormInformation
|
|
|
|
|
dense
|
|
|
|
|
outlined
|
|
|
|
|
separator
|
2024-04-18 09:06:21 +07:00
|
|
|
:usernameReadonly="isEdit"
|
2024-04-17 15:44:13 +07:00
|
|
|
v-model:hqId="hqId"
|
|
|
|
|
v-model:brId="brId"
|
|
|
|
|
v-model:userType="formData.userType"
|
|
|
|
|
v-model:userRole="formData.userRole"
|
2024-04-18 09:06:21 +07:00
|
|
|
v-model:username="formData.username"
|
2024-04-22 14:04:06 +07:00
|
|
|
v-model:userCode="userCode"
|
2024-04-17 15:44:13 +07:00
|
|
|
/>
|
|
|
|
|
</template>
|
|
|
|
|
<template #person>
|
|
|
|
|
<FormPerson
|
2024-04-10 14:00:55 +07:00
|
|
|
dense
|
|
|
|
|
outlined
|
2024-04-17 15:44:13 +07:00
|
|
|
separator
|
2024-04-11 22:17:24 +07:00
|
|
|
v-model:firstName="formData.firstName"
|
|
|
|
|
v-model:lastName="formData.lastName"
|
|
|
|
|
v-model:firstNameEN="formData.firstNameEN"
|
|
|
|
|
v-model:lastNameEN="formData.lastNameEN"
|
2024-04-17 15:44:13 +07:00
|
|
|
v-model:telephoneNo="formData.telephoneNo"
|
|
|
|
|
v-model:email="formData.email"
|
|
|
|
|
v-model:gender="formData.gender"
|
|
|
|
|
v-model:birthDate="formData.birthDate"
|
2024-04-10 14:00:55 +07:00
|
|
|
/>
|
2024-04-05 19:05:51 +07:00
|
|
|
</template>
|
2024-04-17 15:44:13 +07:00
|
|
|
<template #by-type>
|
2024-04-11 22:17:24 +07:00
|
|
|
<FormByType
|
2024-04-05 19:05:51 +07:00
|
|
|
dense
|
|
|
|
|
outlined
|
2024-04-17 15:44:13 +07:00
|
|
|
separator
|
2024-04-11 22:17:24 +07:00
|
|
|
v-model:userType="formData.userType"
|
|
|
|
|
v-model:registrationNo="formData.registrationNo"
|
|
|
|
|
v-model:startDate="formData.startDate"
|
|
|
|
|
v-model:retireDate="formData.retireDate"
|
|
|
|
|
v-model:responsibleArea="formData.responsibleArea"
|
|
|
|
|
v-model:discountCondition="formData.discountCondition"
|
|
|
|
|
v-model:sourceNationality="formData.sourceNationality"
|
|
|
|
|
v-model:importNationality="formData.importNationality"
|
|
|
|
|
v-model:trainingPlace="formData.trainingPlace"
|
2024-04-19 13:54:57 +07:00
|
|
|
v-model:checkpoint="formData.checkpoint"
|
|
|
|
|
v-model:checkpointEN="formData.checkpointEN"
|
|
|
|
|
v-model:agencyFile="agencyFile"
|
2024-04-05 19:05:51 +07:00
|
|
|
/>
|
2024-04-10 14:00:55 +07:00
|
|
|
</template>
|
2024-04-05 19:05:51 +07:00
|
|
|
</FormDialog>
|
2024-04-03 18:28:59 +07:00
|
|
|
</template>
|
2024-07-09 11:21:25 +07:00
|
|
|
<style scoped>
|
2024-04-10 14:00:55 +07:00
|
|
|
.upload-img-preview {
|
|
|
|
|
border: 1px solid var(--border-color);
|
|
|
|
|
border-radius: var(--radius-2);
|
|
|
|
|
height: 12vw;
|
2024-04-18 09:06:21 +07:00
|
|
|
background-color: var(--surface-1);
|
2024-04-10 14:00:55 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.upload-img-btn {
|
|
|
|
|
color: hsl(var(--info-bg));
|
|
|
|
|
border: 1px solid hsl(var(--info-bg));
|
|
|
|
|
background-color: hsla(var(--info-bg) / 0.1);
|
|
|
|
|
border-radius: var(--radius-2);
|
|
|
|
|
|
|
|
|
|
&.dark {
|
|
|
|
|
background-color: transparent;
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-04-10 23:17:39 +07:00
|
|
|
|
|
|
|
|
.edit-img-btn {
|
|
|
|
|
color: hsl(var(--info-bg));
|
|
|
|
|
border: 1px solid hsl(var(--info-bg));
|
|
|
|
|
background-color: transparent;
|
|
|
|
|
border-radius: var(--radius-2);
|
|
|
|
|
}
|
2024-04-18 09:06:21 +07:00
|
|
|
|
|
|
|
|
.cancel-img-btn {
|
|
|
|
|
color: hsl(var(--negative-bg));
|
|
|
|
|
border: 1px solid hsl(var(--negative-bg));
|
|
|
|
|
background-color: transparent;
|
|
|
|
|
border-radius: var(--radius-2);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.submit-img-btn {
|
|
|
|
|
color: var(--surface-1);
|
|
|
|
|
border: 1px solid hsl(var(--positive-bg));
|
|
|
|
|
background-color: hsl(var(--positive-bg));
|
|
|
|
|
border-radius: var(--radius-2);
|
|
|
|
|
}
|
2024-07-05 03:29:33 +00:00
|
|
|
|
|
|
|
|
.slide-enter-active {
|
|
|
|
|
transition: all 0.1s ease-out;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.slide-leave-active {
|
|
|
|
|
transition: all 0.1s cubic-bezier(1, 0.5, 0.8, 1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.slide-enter-from,
|
|
|
|
|
.slide-leave-to {
|
|
|
|
|
transform: translateY(-20px);
|
|
|
|
|
opacity: 0;
|
|
|
|
|
}
|
2024-07-09 11:21:25 +07:00
|
|
|
|
|
|
|
|
.status-active {
|
|
|
|
|
--_branch-status-color: var(--green-6-hsl);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.status-inactive {
|
|
|
|
|
--_branch-status-color: var(--red-4-hsl);
|
|
|
|
|
--_branch-badge-bg: var(--red-4-hsl);
|
|
|
|
|
filter: grayscale(0.5);
|
|
|
|
|
opacity: 0.5;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.branch-card__icon {
|
|
|
|
|
background-color: hsla(var(--_branch-card-bg) / 0.15);
|
|
|
|
|
border-radius: 50%;
|
|
|
|
|
padding: var(--size-1);
|
|
|
|
|
position: relative;
|
|
|
|
|
transform: rotate(45deg);
|
|
|
|
|
|
|
|
|
|
&::after {
|
|
|
|
|
content: ' ';
|
|
|
|
|
display: block;
|
|
|
|
|
block-size: 0.5rem;
|
|
|
|
|
aspect-ratio: 1;
|
|
|
|
|
position: absolute;
|
|
|
|
|
border-radius: 50%;
|
|
|
|
|
right: -0.1rem;
|
|
|
|
|
top: calc(50% - 0.25rem);
|
|
|
|
|
bottom: calc(50% - 0.25rem);
|
|
|
|
|
background-color: hsla(var(--_branch-status-color) / 1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
& :deep(.q-avatar) {
|
|
|
|
|
transform: rotate(-45deg);
|
|
|
|
|
color: hsla(var(--_branch-card-bg) / 1);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.tags {
|
|
|
|
|
display: inline-block;
|
|
|
|
|
color: hsla(var(--_color) / 1);
|
|
|
|
|
background: hsla(var(--_color) / 0.15);
|
|
|
|
|
border-radius: var(--radius-2);
|
|
|
|
|
padding-inline: var(--size-2);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.tags__cyan {
|
|
|
|
|
--_color: var(--cyan-7-hsl);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.tags__yellow {
|
|
|
|
|
--_color: var(--orange-4-hsl);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.tags__red {
|
|
|
|
|
--_color: var(--red-6-hsl);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.tags__magenta {
|
|
|
|
|
--_color: var(--pink-8-hsl);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.tags__pink {
|
|
|
|
|
--_color: var(--pink-6-hsl);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.dark .tags__magenta {
|
|
|
|
|
--_color: var(--pink-7-hsl);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
& .symbol-gender {
|
|
|
|
|
color: hsla(var(--_fg));
|
|
|
|
|
|
|
|
|
|
&.symbol-gender__male {
|
|
|
|
|
--_fg: var(--gender-male);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
&.symbol-gender__female {
|
|
|
|
|
--_fg: var(--gender-female);
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-04-05 19:05:51 +07:00
|
|
|
</style>
|