first commit
This commit is contained in:
commit
e8ec46d19f
60 changed files with 13652 additions and 0 deletions
76
src/components/DrawerComponent.vue
Normal file
76
src/components/DrawerComponent.vue
Normal file
|
|
@ -0,0 +1,76 @@
|
|||
<script setup lang="ts">
|
||||
import { ref } from 'vue';
|
||||
|
||||
const currentRoute = ref<string>('');
|
||||
const labelMenu: {
|
||||
label: string;
|
||||
icon: string;
|
||||
}[] = [
|
||||
{ label: 'Dashboard', icon: 'file-account-outline' },
|
||||
{ label: 'จัดการสาขา', icon: 'add' },
|
||||
{ label: 'จัดการผู้ใช้งาน', icon: 'add' },
|
||||
{ label: 'จัดการหลักสูตร', icon: 'add' },
|
||||
];
|
||||
|
||||
const leftDrawerOpen = defineModel<boolean>('leftDrawerOpen', {
|
||||
default: false,
|
||||
});
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<!-- Drawer -->
|
||||
<q-drawer v-model="leftDrawerOpen" side="left" show-if-above>
|
||||
<div class="main-bar">
|
||||
<div
|
||||
class="column items-center justify-center q-pa-xl cursor-pointer"
|
||||
@click="$router.push('/')"
|
||||
id="btn-drawer-home"
|
||||
>
|
||||
<q-img src="/logo.png" />
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div id="drawer-menu" class="q-pr-sm">
|
||||
<q-item
|
||||
v-for="v in labelMenu"
|
||||
:key="v.label"
|
||||
clickable
|
||||
@click="currentRoute = v.label"
|
||||
:class="{ active: currentRoute === v.label, dark: $q.dark.isActive }"
|
||||
>
|
||||
<q-item-section id="btn-drawer-back " class="test">
|
||||
<q-item-label>
|
||||
<!-- <q-icon :name="v.icon" size="sm" class="q-mr-xs" /> -->
|
||||
{{ v.label }}
|
||||
</q-item-label>
|
||||
</q-item-section>
|
||||
</q-item>
|
||||
</div>
|
||||
</q-drawer>
|
||||
</template>
|
||||
|
||||
<style lang="scss" scoped>
|
||||
#drawer-menu :deep(.q-item) {
|
||||
color: var(--gray-6);
|
||||
|
||||
border-top-right-radius: 10px;
|
||||
border-bottom-right-radius: 10px;
|
||||
}
|
||||
#drawer-menu :deep(.q-item.active) {
|
||||
--_drawer-item-background-color: var(--brand-1) !important;
|
||||
background-color: var(--_drawer-item-background-color) !important;
|
||||
|
||||
color: white;
|
||||
border-left: 10px solid $secondary;
|
||||
|
||||
&.dark {
|
||||
--_drawer-item-background-color: var(--gray-10) !important;
|
||||
border: 1px solid var(--brand-1);
|
||||
border-left: 10px solid $secondary;
|
||||
}
|
||||
}
|
||||
|
||||
.test {
|
||||
border: 1px solid blue;
|
||||
}
|
||||
</style>
|
||||
59
src/components/GlobalDialog.vue
Normal file
59
src/components/GlobalDialog.vue
Normal file
|
|
@ -0,0 +1,59 @@
|
|||
<script setup lang="ts">
|
||||
defineProps<{
|
||||
title: string;
|
||||
message: string;
|
||||
color?: string;
|
||||
icon?: string;
|
||||
actionText?: string;
|
||||
cancelText?: string;
|
||||
persistent?: boolean;
|
||||
action?: (...args: unknown[]) => void;
|
||||
cancel?: (...args: unknown[]) => void;
|
||||
}>();
|
||||
</script>
|
||||
<template>
|
||||
<q-dialog ref="dialogRef" :persistent="persistent || false">
|
||||
<q-card class="q-pa-sm" style="min-width: 300px; max-width: 80%">
|
||||
<q-card-section class="row q-pa-sm">
|
||||
<div
|
||||
class="q-pr-md"
|
||||
v-if="icon"
|
||||
style="display: flex; align-items: center"
|
||||
>
|
||||
<q-avatar
|
||||
:icon="icon"
|
||||
size="xl"
|
||||
font-size="25px"
|
||||
color="grey-2"
|
||||
:text-color="color || 'primary'"
|
||||
/>
|
||||
</div>
|
||||
<div class="col text-dark">
|
||||
<span class="text-bold block">{{ title }}</span>
|
||||
<span class="block">{{ message }}</span>
|
||||
</div>
|
||||
</q-card-section>
|
||||
<q-card-actions align="right" v-if="action || cancel">
|
||||
<q-btn
|
||||
id="btn-cancel-dialog"
|
||||
v-if="cancel"
|
||||
@click="cancel"
|
||||
:label="cancelText || $t('cancel')"
|
||||
:color="color || 'primary'"
|
||||
v-close-popup
|
||||
flat
|
||||
/>
|
||||
<q-btn
|
||||
id="btn-ok-dialog"
|
||||
v-if="action"
|
||||
@click="action"
|
||||
:label="actionText || $t('ok')"
|
||||
:color="color || 'primary'"
|
||||
v-close-popup
|
||||
/>
|
||||
</q-card-actions>
|
||||
</q-card>
|
||||
</q-dialog>
|
||||
</template>
|
||||
|
||||
<style lang="scss"></style>
|
||||
20
src/components/GlobalLoading.vue
Normal file
20
src/components/GlobalLoading.vue
Normal file
|
|
@ -0,0 +1,20 @@
|
|||
<script setup lang="ts">
|
||||
defineProps<{
|
||||
visibility: boolean;
|
||||
}>();
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<q-inner-loading :showing="visibility" class="loader q-gutter-y-xl">
|
||||
<q-spinner size="80px" color="primary" />
|
||||
<span class="text-h5 text-weight-bold text-primary">
|
||||
{{ $t('loading') }}
|
||||
</span>
|
||||
</q-inner-loading>
|
||||
</template>
|
||||
|
||||
<style scoped>
|
||||
.loader {
|
||||
z-index: 1000;
|
||||
}
|
||||
</style>
|
||||
214
src/components/UsersDetailCardComponent.vue
Normal file
214
src/components/UsersDetailCardComponent.vue
Normal file
|
|
@ -0,0 +1,214 @@
|
|||
<script setup lang="ts">
|
||||
import AppBox from './app/AppBox.vue';
|
||||
|
||||
withDefaults(
|
||||
defineProps<{
|
||||
dark?: boolean;
|
||||
width?: string;
|
||||
color: 'purple' | 'green';
|
||||
}>(),
|
||||
{
|
||||
width: '300px',
|
||||
deletable: false,
|
||||
color: 'green',
|
||||
},
|
||||
);
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<div class="container">
|
||||
<div style="aspect-ratio: 9/16" class="flip">
|
||||
<AppBox
|
||||
style="padding: 0"
|
||||
:class="`${$q.dark.isActive ? 'dark ' : ''} color__${color}`"
|
||||
class="surface-1 front"
|
||||
bordered
|
||||
>
|
||||
<div class="row justify-center">
|
||||
<q-card-section class="q-pt-xl img">
|
||||
<q-avatar size="100px">
|
||||
<img src="https://cdn.quasar.dev/img/avatar1.jpg" />
|
||||
</q-avatar>
|
||||
</q-card-section>
|
||||
</div>
|
||||
<div class="box-title">
|
||||
<div class="rounded title">นิติบุคคล</div>
|
||||
</div>
|
||||
<q-card-section class="no-padding">
|
||||
<div class="column items-center justify-center">
|
||||
<div class="row">นางสาวสุดใจ แสนดี</div>
|
||||
<div class="row q-mb-md">HQ0001</div>
|
||||
</div>
|
||||
</q-card-section>
|
||||
|
||||
<div>
|
||||
<q-scroll-area
|
||||
class="front-scroll"
|
||||
style="border-radius: 0px 0px 12px 12px"
|
||||
>
|
||||
<q-card-section v-for="v in [1, 2]" :key="v" class="overflow">
|
||||
<div class="color-front-text">ชื่อ บริษัท/นิติบุคคล ภาษาไทย</div>
|
||||
<div>บริษัทเฟรปเป้</div>
|
||||
</q-card-section>
|
||||
</q-scroll-area>
|
||||
</div>
|
||||
</AppBox>
|
||||
|
||||
<AppBox
|
||||
style="padding: 0"
|
||||
class="back"
|
||||
:class="`${$q.dark.isActive ? 'dark ' : ''} color__${color}`"
|
||||
bordered
|
||||
>
|
||||
<div
|
||||
class="row q-pl-md q-pb-md items-center"
|
||||
:class="{
|
||||
'surface-1': !$q.dark.isActive,
|
||||
'surface-2': $q.dark.isActive,
|
||||
}"
|
||||
style="border-radius: 12px 12px 0px 0px"
|
||||
>
|
||||
<q-card-section class="q-pt-xl img">
|
||||
<q-avatar size="50px">
|
||||
<img src="https://cdn.quasar.dev/img/avatar1.jpg" />
|
||||
</q-avatar>
|
||||
</q-card-section>
|
||||
|
||||
<div class="col-7 q-pl-md">
|
||||
<div class="row">นางสาวสุขใจ แสนดี</div>
|
||||
<div class="row">HQ0001</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<q-scroll-area
|
||||
class="back-scroll"
|
||||
:class="{
|
||||
'surface-1': $q.dark.isActive,
|
||||
'surface-2': !$q.dark.isActive,
|
||||
}"
|
||||
>
|
||||
<q-card-section class="q-pa-md">
|
||||
<div
|
||||
v-for="v in [1, 2]"
|
||||
:key="v"
|
||||
class="bordered row q-pa-sm q-mb-sm rounded bg-color-text-1"
|
||||
>
|
||||
<div class="col-2 flex flex-center">{{ v }}</div>
|
||||
<div class="col-10 rounded q-pa-sm bg-color-text-2">HQ0001</div>
|
||||
</div>
|
||||
</q-card-section>
|
||||
</q-scroll-area>
|
||||
</AppBox>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<style scoped>
|
||||
.color-front-text {
|
||||
color: var(--color-front-title);
|
||||
}
|
||||
|
||||
.container {
|
||||
width: 300px;
|
||||
height: 500px;
|
||||
}
|
||||
|
||||
.flip {
|
||||
cursor: pointer;
|
||||
width: 100%;
|
||||
position: relative;
|
||||
transition: transform 0.4s;
|
||||
transform-style: preserve-3d;
|
||||
|
||||
&:hover {
|
||||
transform: rotateY(180deg);
|
||||
}
|
||||
}
|
||||
|
||||
.front,
|
||||
.back {
|
||||
width: 100%;
|
||||
height: min-content;
|
||||
position: absolute;
|
||||
backface-visibility: hidden;
|
||||
}
|
||||
|
||||
.front,
|
||||
.back {
|
||||
--_color: var(--teal-6);
|
||||
--_color-dark: var(--_color);
|
||||
--_bg-front-scroll: hsla(var(--_color) / 0.05);
|
||||
--_bg-back-detail-1: white;
|
||||
--_bg-back-detail-2: hsla(var(--_color) / 0.05) !important;
|
||||
|
||||
.img {
|
||||
background: hsl(var(--_color)) !important;
|
||||
border-bottom-left-radius: 100px;
|
||||
border-bottom-right-radius: 100px;
|
||||
}
|
||||
|
||||
.title {
|
||||
width: 50%;
|
||||
height: 25px;
|
||||
background: hsl(var(--_color)) !important;
|
||||
position: relative;
|
||||
bottom: 16px;
|
||||
display: flex;
|
||||
justify-content: center;
|
||||
align-items: center;
|
||||
color: white;
|
||||
}
|
||||
|
||||
.front-scroll {
|
||||
border-radius: 0px 0px 5px 5px;
|
||||
background: var(--_bg-front-scroll) !important;
|
||||
height: 254px;
|
||||
max-width: 300px;
|
||||
}
|
||||
|
||||
.back-scroll {
|
||||
height: 370px;
|
||||
max-width: 300px;
|
||||
|
||||
border-radius: 0px 0px 15px 15px;
|
||||
|
||||
.bg-color-text-1 {
|
||||
background: var(--_bg-back-detail-1);
|
||||
}
|
||||
.bg-color-text-2 {
|
||||
background: var(--_bg-back-detail-2) !important;
|
||||
}
|
||||
}
|
||||
|
||||
&.dark {
|
||||
color: hsl(--_color-dark) !important;
|
||||
background-color: transparent !important;
|
||||
|
||||
--_bg-back-avatar: var(--gray-11) !important;
|
||||
--_bg-back-detail-1: var(--gray-9) !important;
|
||||
--_bg-front-scroll: var(--gray-11);
|
||||
--_bg-back-detail-2: hsla(var(--_color) / 0.1) !important;
|
||||
}
|
||||
&.color__purple {
|
||||
--_color: var(--purple-11-hsl);
|
||||
}
|
||||
|
||||
&.color__green {
|
||||
--_color: var(--green-9-hsl);
|
||||
}
|
||||
}
|
||||
|
||||
.back {
|
||||
background: #f5f5f5;
|
||||
transform: rotateY(180deg);
|
||||
}
|
||||
|
||||
.rounded {
|
||||
border-radius: 5px 5px 5px 5px;
|
||||
}
|
||||
|
||||
.box-title {
|
||||
display: flex;
|
||||
justify-content: center;
|
||||
}
|
||||
</style>
|
||||
23
src/components/app/AppBox.vue
Normal file
23
src/components/app/AppBox.vue
Normal file
|
|
@ -0,0 +1,23 @@
|
|||
<script lang="ts" setup>
|
||||
defineProps<{ dark?: boolean; bordered?: boolean }>();
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<div class="app-box" :class="{ dark, 'app-box__bordered': bordered }">
|
||||
<slot />
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<style scoped>
|
||||
.app-box {
|
||||
overflow: hidden;
|
||||
background-color: var(--surface-1);
|
||||
border-radius: var(--radius-3);
|
||||
padding: var(--size-4);
|
||||
box-shadow: var(--shadow-1);
|
||||
}
|
||||
|
||||
.app-box__bordered {
|
||||
border: 1px solid var(--border-color);
|
||||
}
|
||||
</style>
|
||||
25
src/components/app/AppCircle.vue
Normal file
25
src/components/app/AppCircle.vue
Normal file
|
|
@ -0,0 +1,25 @@
|
|||
<script lang="ts" setup>
|
||||
defineProps<{ bordered?: boolean; dark?: boolean }>();
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<div class="app-circle" :class="{ dark, 'app-circle__bordered': bordered }">
|
||||
<slot />
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<style scoped>
|
||||
.app-circle {
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
overflow: hidden;
|
||||
border-radius: var(--radius-round);
|
||||
background-color: var(--surface-1);
|
||||
aspect-ratio: var(--ratio-square);
|
||||
}
|
||||
|
||||
.app-circle__bordered {
|
||||
border: 1px solid var(--border-color);
|
||||
}
|
||||
</style>
|
||||
112
src/components/home/MenuItem.vue
Normal file
112
src/components/home/MenuItem.vue
Normal file
|
|
@ -0,0 +1,112 @@
|
|||
<script setup lang="ts">
|
||||
import AppBox from 'components/app/AppBox.vue';
|
||||
import AppCircle from 'components/app/AppCircle.vue';
|
||||
|
||||
defineProps<{
|
||||
list: {
|
||||
icon: string;
|
||||
title: string;
|
||||
caption: string;
|
||||
color:
|
||||
| 'green'
|
||||
| 'red'
|
||||
| 'orange'
|
||||
| 'cyan'
|
||||
| 'camo'
|
||||
| 'purple'
|
||||
| 'violet'
|
||||
| 'indigo'
|
||||
| 'lime';
|
||||
}[];
|
||||
}>();
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<div class="menu-container">
|
||||
<AppBox
|
||||
class="column inline-flex items-center"
|
||||
v-for="(v, i) in list"
|
||||
:key="i"
|
||||
:bordered="$q.dark.isActive"
|
||||
>
|
||||
<AppCircle
|
||||
:class="`q-pa-sm menu-icon menu-icon__${v.color}${($q.dark.isActive && ' dark') || ''}`"
|
||||
:bordered="$q.dark.isActive"
|
||||
>
|
||||
<q-icon size="3rem" :name="v.icon" />
|
||||
</AppCircle>
|
||||
<div class="column items-center q-mt-md text-center">
|
||||
<span style="font-weight: var(--font-weight-8)">{{ v.title }}</span>
|
||||
<span style="color: rgba(130 130 130 / 0.7)">{{ v.caption }}</span>
|
||||
</div>
|
||||
</AppBox>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<style scoped>
|
||||
.menu-container {
|
||||
display: grid;
|
||||
grid-template-columns: repeat(4, 1fr);
|
||||
gap: var(--size-6);
|
||||
& > * {
|
||||
transition: 100ms border-color ease-in-out;
|
||||
border: 1px solid transparent;
|
||||
|
||||
&:hover {
|
||||
border-color: var(--brand-1);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
.menu-icon {
|
||||
--_color: var(--gray-7-hsl);
|
||||
--_color-dark: var(--_color);
|
||||
color: hsl(var(--_color)) !important;
|
||||
background-color: hsla(var(--_color) / 0.1) !important;
|
||||
|
||||
&.dark {
|
||||
color: hsl(--_color-dark) !important;
|
||||
background-color: transparent !important;
|
||||
}
|
||||
|
||||
&.menu-icon__green {
|
||||
--_color: var(--green-6-hsl);
|
||||
}
|
||||
|
||||
&.menu-icon__red {
|
||||
--_color: var(--red-8-hsl);
|
||||
}
|
||||
|
||||
&.menu-icon__orange {
|
||||
--_color: var(--orange-6-hsl);
|
||||
--_color-dark: var(--orange-5-hsl);
|
||||
}
|
||||
|
||||
&.menu-icon__cyan {
|
||||
--_color: var(--cyan-5-hsl);
|
||||
--_color-dark: var(--cyan-4-hsl);
|
||||
}
|
||||
|
||||
&.menu-icon__camo {
|
||||
--_color: var(--camo-5-hsl);
|
||||
--_color-dark: var(--cyan-4-hsl);
|
||||
}
|
||||
|
||||
&.menu-icon__purple {
|
||||
--_color: var(--purple-3-hsl);
|
||||
--_color-dark: var(--purple-5-hsl);
|
||||
}
|
||||
|
||||
&.menu-icon__violet {
|
||||
--_color: var(--violet-5-hsl);
|
||||
}
|
||||
|
||||
&.menu-icon__indigo {
|
||||
--_color: var(--indigo-9-hsl);
|
||||
}
|
||||
|
||||
&.menu-icon__lime {
|
||||
--_color: var(--lime-11-hsl);
|
||||
}
|
||||
}
|
||||
</style>
|
||||
112
src/components/home/PersonCard.vue
Normal file
112
src/components/home/PersonCard.vue
Normal file
|
|
@ -0,0 +1,112 @@
|
|||
<script lang="ts" setup>
|
||||
import AppBox from 'components/app/AppBox.vue';
|
||||
import AppCircle from '../app/AppCircle.vue';
|
||||
|
||||
defineProps<{
|
||||
list: {
|
||||
name: string;
|
||||
badge: string;
|
||||
detail: { label: string; value: string }[];
|
||||
male?: boolean;
|
||||
female?: boolean;
|
||||
disabled?: boolean;
|
||||
}[];
|
||||
}>();
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<div class="person-container">
|
||||
<AppBox
|
||||
:bordered="$q.dark.isActive"
|
||||
class="column"
|
||||
style="padding: 0"
|
||||
v-for="(v, i) in list"
|
||||
:key="i"
|
||||
>
|
||||
<div class="q-pa-md column items-center">
|
||||
<AppCircle class="surface-2 avatar q-mb-md" bordered>Avatar</AppCircle>
|
||||
|
||||
<span>{{ v.name }}</span>
|
||||
<span
|
||||
class="badge"
|
||||
:class="{
|
||||
'bg-gender': v.male || v.female,
|
||||
'bg-gender__male': v.male,
|
||||
'bg-gender__female': v.female,
|
||||
}"
|
||||
>
|
||||
{{ v.badge }}
|
||||
</span>
|
||||
</div>
|
||||
<q-separator />
|
||||
<div
|
||||
class="q-py-sm q-px-xl person-detail rounded-b"
|
||||
:class="{
|
||||
'bg-gender': v.male || v.female,
|
||||
'bg-gender__light': v.male || v.female,
|
||||
'bg-gender__male': v.male,
|
||||
'bg-gender__female': v.female,
|
||||
}"
|
||||
>
|
||||
<div v-for="(d, j) in v.detail" :key="j">
|
||||
<span>{{ d.label }}</span>
|
||||
<span>{{ d.value }}</span>
|
||||
</div>
|
||||
</div>
|
||||
</AppBox>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<style scoped>
|
||||
.person-container {
|
||||
display: grid;
|
||||
grid-template-columns: repeat(3, 1fr);
|
||||
gap: var(--size-6);
|
||||
}
|
||||
|
||||
.avatar {
|
||||
block-size: 7rem;
|
||||
}
|
||||
|
||||
.badge {
|
||||
display: inline-block;
|
||||
border-radius: var(--radius-6);
|
||||
background-color: var(--surface-2);
|
||||
padding: 0 var(--size-6);
|
||||
}
|
||||
|
||||
.person-detail {
|
||||
display: grid;
|
||||
flex-grow: 1;
|
||||
grid-template-columns: repeat(2, 1fr);
|
||||
gap: var(--size-2);
|
||||
|
||||
& > * {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
|
||||
& > :first-child {
|
||||
font-size: var(--font-size-0);
|
||||
}
|
||||
}
|
||||
}
|
||||
.bg-gender {
|
||||
color: hsla(var(--_fg));
|
||||
background-color: hsl(var(--_bg));
|
||||
|
||||
&.bg-gender__light {
|
||||
color: unset;
|
||||
background-color: hsla(var(--_bg) / 0.1);
|
||||
}
|
||||
|
||||
&.bg-gender__male {
|
||||
--_fg: 0 100 100%;
|
||||
--_bg: var(--blue-5-hsl);
|
||||
}
|
||||
|
||||
&.bg-gender__female {
|
||||
--_fg: 0 100 100%;
|
||||
--_bg: var(--pink-7-hsl);
|
||||
}
|
||||
}
|
||||
</style>
|
||||
Loading…
Add table
Add a link
Reference in a new issue