feat: toggle grid customer branch

This commit is contained in:
Methapon2001 2024-07-30 14:24:35 +07:00
parent be397112f6
commit 5ac705af0e
2 changed files with 369 additions and 153 deletions

View file

@ -11,6 +11,7 @@ import { CustomerBranch, CustomerType } from 'src/stores/customer/types';
import BranchCardCustomer from 'components/03_customer-management/BranchCardCustomer.vue'; import BranchCardCustomer from 'components/03_customer-management/BranchCardCustomer.vue';
import PaginationComponent from 'src/components/PaginationComponent.vue'; import PaginationComponent from 'src/components/PaginationComponent.vue';
import NoData from 'components/NoData.vue'; import NoData from 'components/NoData.vue';
import { QTableProps } from 'quasar';
const flowStore = useFlowStore(); const flowStore = useFlowStore();
@ -29,6 +30,7 @@ const currentCustomerUrlImage = defineModel<string | null>(
const currentStatus = ref<Status | 'All'>('All'); const currentStatus = ref<Status | 'All'>('All');
const totalBranch = ref(0); const totalBranch = ref(0);
const modeView = ref(false);
const currentPageBranch = ref<number>(1); const currentPageBranch = ref<number>(1);
const maxPageBranch = ref<number>(1); const maxPageBranch = ref<number>(1);
const pageSizeBranch = ref<number>(30); const pageSizeBranch = ref<number>(30);
@ -55,6 +57,57 @@ onMounted(async () => {
await fetchList(); await fetchList();
}); });
const columns = [
{
name: 'branchLabelName',
align: 'left',
label: 'office',
field: 'name',
sortable: true,
},
{
name: 'branchLabelAddress',
align: 'left',
label: 'address',
field: 'address',
sortable: true,
},
{
name: 'branchLabelTel',
align: 'left',
label: 'formDialogInputTelephone',
field: 'telephoneNo',
},
] satisfies QTableProps['columns'];
const fieldDisplay = ref<
(
| 'branchName'
| 'customerBranchFormTab'
| 'address'
| 'telephone'
| 'businessTypePure'
| 'totalEmployee'
)[]
>([
'branchName',
'customerBranchFormTab',
'address',
'telephone',
'businessTypePure',
'totalEmployee',
]);
const branchFieldSelected = ref<
(
| 'customerBranchFormTab'
| 'branchName'
| 'address'
| 'telephone'
| 'businessTypePure'
| 'totalEmployee'
)[]
>(fieldDisplay.value);
async function fetchList() { async function fetchList() {
const result = await fetchListCustomeBranch({ const result = await fetchListCustomeBranch({
customerId: prop.customerId, customerId: prop.customerId,
@ -76,20 +129,12 @@ async function fetchList() {
} }
} }
watch(inputSearch, async () => { watch([inputSearch, currentStatus], async () => {
await fetchList();
});
watch(currentStatus, async () => {
await fetchList(); await fetchList();
}); });
</script> </script>
<template> <template>
<div
class="bordered surface-1 col column surface-2 rounded no-wrap"
style="overflow: hidden"
>
<div <div
class="row justify-between bordered-b surface-3 full-width no-wrap" class="row justify-between bordered-b surface-3 full-width no-wrap"
style="z-index: 1" style="z-index: 1"
@ -151,45 +196,79 @@ watch(currentStatus, async () => {
<q-icon name="mdi-magnify" /> <q-icon name="mdi-magnify" />
</template> </template>
</q-input> </q-input>
<q-btn
padding="10px 16px" <q-select
icon="mdi-tune-vertical-variant" lazy-rules="ondemand"
size="sm" id="select-status"
:color="$q.dark.isActive ? 'dark' : 'white'" for="select-status"
:text-color="$q.dark.isActive ? 'white' : 'dark'" v-model="currentStatus"
class="bordered rounded q-ml-sm col-1" outlined
unelevated dense
option-value="value"
option-label="label"
class="col q-pl-sm"
:class="{ 'offset-md-5': modeView }"
map-options
emit-value
:hide-dropdown-icon="$q.screen.lt.sm"
:options="[
{ label: $t('all'), value: 'All' },
{ label: $t('statusACTIVE'), value: 'ACTIVE' },
{ label: $t('statusINACTIVE'), value: 'INACTIVE' },
]"
></q-select>
<q-btn-toggle
id="btn-mode"
v-model="modeView"
dense
class="no-shadow bordered rounded surface-1 q-ml-sm"
:toggle-color="$q.dark.isActive ? 'grey-9' : 'grey-2'"
size="xs"
:options="[
{ value: true, slot: 'folder' },
{ value: false, slot: 'list' },
]"
> >
<q-menu class="bordered"> <template v-slot:folder>
<q-list v-close-popup dense> <q-icon
<q-item id="icon-mode-grid"
clickable name="mdi-view-grid-outline"
class="flex items-center" size="16px"
@click="currentStatus = 'All'" class="q-px-sm q-py-sm rounded"
> :style="{
{{ $t('all') }} color: $q.dark.isActive
</q-item> ? modeView
<q-item ? '#C9D3DB'
clickable : '#787B7C'
class="flex items-center" : modeView
@click="currentStatus = 'ACTIVE'" ? '#787B7C'
> : '#C9D3DB',
{{ $t('statusACTIVE') }} }"
</q-item> />
<q-item </template>
clickable <template v-slot:list>
class="flex items-center" <q-icon
@click="currentStatus = 'INACTIVE'" id="icon-mode-list"
> name="mdi-format-list-bulleted"
{{ $t('statusINACTIVE') }} class="q-px-sm q-py-xs rounded"
</q-item> size="16px"
</q-list> :style="{
</q-menu> color: $q.dark.isActive
</q-btn> ? modeView === true
? '#C9D3DB'
: '#787B7C'
: modeView === false
? '#787B7C'
: '#C9D3DB',
}"
/>
</template>
</q-btn-toggle>
</div> </div>
</div> </div>
<div <div
class="row q-pa-md q-col-gutter-md col scroll" class="row q-pa-md col scroll full-width"
:class="{ :class="{
'justify-center': totalBranch === 0, 'justify-center': totalBranch === 0,
'items-center': totalBranch === 0, 'items-center': totalBranch === 0,
@ -197,29 +276,143 @@ watch(currentStatus, async () => {
> >
<NoData v-if="totalBranch === 0" :not-found="!!inputSearch" /> <NoData v-if="totalBranch === 0" :not-found="!!inputSearch" />
<div v-for="(br, i) in branch" :key="i" class="col-md-6 col-12"> <div style="width: 100%">
<BranchCardCustomer <q-table
class="surface-1" v-if="branch"
:inactive="br.status === 'INACTIVE'" flat
:color="customerType === 'CORP' ? 'corp' : 'pers'" class="full-width"
:badgeField="['status']" bordered
:data="{ :rows-per-page-options="[0]"
customerBranchFormTab: br.branchNo, :rows="branch"
branchName: br.name, :columns="columns"
address: :grid="modeView"
$i18n.locale === 'en-US' card-container-class="row q-col-gutter-md"
? `${br.addressEN || ''} ${br.subDistrict?.nameEN || ''} ${br.district?.nameEN || ''} ${br.province?.nameEN || ''}` row-key="name"
: `${br.address || ''} ${br.subDistrict?.name || ''} ${br.district?.name || ''} ${br.province?.name || ''}`, hide-pagination
telephone: br.telephoneNo, :visible-columns="branchFieldSelected"
businessTypePure: useOptionStore().mapOption(br.bussinessType), >
totalEmployee: br._count?.employee, <template #header>
<q-tr class="surface-3">
<q-th
v-for="(v, i) in fieldDisplay"
:key="v"
:class="{ 'text-left': i === 0 }"
>
{{ $t(v) }}
</q-th>
<q-th auto-width />
</q-tr>
</template>
<template #body="props">
<q-tr
:class="{
'app-text-muted': props.row.status === 'INACTIVE',
'cursor-pointer': props.row._count.branch !== 0,
}" }"
@view-detail="$emit('viewDetail', br, i)" :props="props"
@click="$emit('viewDetail', props.row, props.rowIndex)"
>
<q-td v-if="branchFieldSelected.includes('branchName')">
<div class="row items-center no-wrap">
<div
:class="{
'status-active': props.row.status !== 'INACTIVE',
'status-inactive': props.row.status === 'INACTIVE',
}"
class="q-mr-md"
style="display: flex; margin-bottom: var(--size-2)"
>
<div
class="branch-card__icon"
:class="{ 'branch-card__dark': $q.dark.isActive }"
>
<q-icon
size="md"
style="scale: 0.8"
name="mdi-office-building-outline"
/> />
</div> </div>
</div> </div>
<div class="col" style="min-width: fit-content">
<div class="col">
{{
$i18n.locale === 'en-US'
? props.row.nameEN
: props.row.name
}}
</div>
<div class="col app-text-muted">
{{ props.row.code }}
</div>
</div>
</div>
</q-td>
<q-td
v-if="branchFieldSelected.includes('customerBranchFormTab')"
class="text-center"
>
{{ props.row.branchNo || '-' }}
</q-td>
<q-td
v-if="branchFieldSelected.includes('address')"
class="text-center"
>
{{
$i18n.locale === 'en-US'
? `${props.row.addressEN || ''} ${props.row.subDistrict?.nameEN || ''} ${props.row.district?.nameEN || ''} ${props.row.province?.nameEN || ''}`
: `${props.row.address || ''} ${props.row.subDistrict?.name || ''} ${props.row.district?.name || ''} ${props.row.province?.name || ''}`
}}
</q-td>
<q-td
v-if="branchFieldSelected.includes('telephone')"
class="text-center"
>
{{ props.row.telephoneNo || '-' }}
</q-td>
<q-td
v-if="branchFieldSelected.includes('businessTypePure')"
class="text-center"
>
{{ useOptionStore().mapOption(props.row.bussinessType) || '-' }}
</q-td>
<q-td
v-if="branchFieldSelected.includes('totalEmployee')"
class="text-center"
>
{{ props.row._count?.employee }}
</q-td>
</q-tr>
</template>
<template #item="props">
<div class="col-12 col-md-6">
<BranchCardCustomer
class="surface-1"
:inactive="props.row.status === 'INACTIVE'"
:color="customerType === 'CORP' ? 'corp' : 'pers'"
:badgeField="['status']"
:data="{
customerBranchFormTab: props.row.branchNo,
branchName: props.row.name,
address:
$i18n.locale === 'en-US'
? `${props.row.addressEN || ''} ${props.row.subDistrict?.nameEN || ''} ${props.row.district?.nameEN || ''} ${props.row.province?.nameEN || ''}`
: `${props.row.address || ''} ${props.row.subDistrict?.name || ''} ${props.row.district?.name || ''} ${props.row.province?.name || ''}`,
telephone: props.row.telephoneNo,
businessTypePure: useOptionStore().mapOption(
props.row.bussinessType,
),
totalEmployee: props.row._count?.employee,
}"
:visible-columns="branchFieldSelected"
@view-detail="$emit('viewDetail', props.row, props.rowIndex)"
/>
</div>
</template>
</q-table>
</div>
</div>
<div class="row justify-between items-center q-px-md q-py-sm"> <div class="row justify-between items-center q-px-md q-py-sm surface-2">
<div class="app-text-muted col offset-md-4"> <div class="app-text-muted col offset-md-4">
{{ {{
$t('recordsPage', { $t('recordsPage', {
@ -241,15 +434,38 @@ watch(currentStatus, async () => {
/> />
</div> </div>
</div> </div>
</div>
</template> </template>
<style scoped> <style scoped>
.color__purple { .branch-card__icon {
background: hsl(var(--purple-11-hsl)); --_branch-card-bg: var(--pink-6-hsl);
background-color: hsla(var(--_branch-card-bg) / 0.15);
border-radius: 50%;
padding: var(--size-1);
position: relative;
transform: rotate(45deg);
&.branch-card__dark {
--_branch-card-bg: var(--pink-5-hsl);
} }
.color__green { &::after {
background: hsl(var(--teal-9-hsl)); content: ' ';
display: block;
block-size: 0.5rem;
aspect-ratio: 1;
position: absolute;
border-radius: 50%;
right: -0.25rem;
top: calc(50% - 0.25rem);
bottom: calc(50% - 0.25rem);
background-color: hsla(var(--_branch-status-color) / 1);
}
& :deep(.q-icon) {
transform: rotate(-45deg);
color: hsla(var(--_branch-card-bg) / 1);
}
} }
</style> </style>

View file

@ -2908,7 +2908,7 @@ watch(isMainPage, () => {
</q-splitter> </q-splitter>
</div> </div>
</div> </div>
<div class="col column" v-else> <div class="col column surface-1" v-else>
<CustomerInfoComponent <CustomerInfoComponent
:customer-type="customerType" :customer-type="customerType"
:customer-id="currentCustomerId" :customer-id="currentCustomerId"