add new event and attendance table with automatic now timestamp
All checks were successful
Build Quasar SPA and Go Backend for memberApp / build-spa (push) Successful in 2m34s
Build Quasar SPA and Go Backend for memberApp / build-backend (amd64, .exe, windows) (push) Successful in 5m39s
Build Quasar SPA and Go Backend for memberApp / build-backend (amd64, , linux) (push) Successful in 5m46s
Build Quasar SPA and Go Backend for memberApp / build-backend (arm, 6, , linux) (push) Successful in 5m32s
Build Quasar SPA and Go Backend for memberApp / build-backend (arm64, , linux) (push) Successful in 5m35s
All checks were successful
Build Quasar SPA and Go Backend for memberApp / build-spa (push) Successful in 2m34s
Build Quasar SPA and Go Backend for memberApp / build-backend (amd64, .exe, windows) (push) Successful in 5m39s
Build Quasar SPA and Go Backend for memberApp / build-backend (amd64, , linux) (push) Successful in 5m46s
Build Quasar SPA and Go Backend for memberApp / build-backend (arm, 6, , linux) (push) Successful in 5m32s
Build Quasar SPA and Go Backend for memberApp / build-backend (arm64, , linux) (push) Successful in 5m35s
This commit is contained in:
77
src/vueLib/tables/attendees/AttendeesTable.ts
Normal file
77
src/vueLib/tables/attendees/AttendeesTable.ts
Normal file
@@ -0,0 +1,77 @@
|
||||
import { appApi } from 'src/boot/axios';
|
||||
import { ref, computed } from 'vue';
|
||||
import type { Attendees } from 'src/vueLib/models/attendee';
|
||||
import { useNotify } from 'src/vueLib/general/useNotify';
|
||||
import { i18n } from 'boot/lang';
|
||||
import type { Events } from 'src/vueLib/models/event';
|
||||
|
||||
export function useAttendeesTable() {
|
||||
const attendees = ref<Attendees>([]);
|
||||
|
||||
const pagination = ref({
|
||||
sortBy: 'firstName',
|
||||
descending: false,
|
||||
page: 1,
|
||||
rowsPerPage: 20,
|
||||
});
|
||||
|
||||
const columns = computed(() => [
|
||||
{
|
||||
name: 'firstName',
|
||||
align: 'left' as const,
|
||||
label: i18n.global.t('prename'),
|
||||
field: 'firstName',
|
||||
sortable: true,
|
||||
},
|
||||
{
|
||||
name: 'lastName',
|
||||
align: 'left' as const,
|
||||
label: i18n.global.t('lastName'),
|
||||
field: 'lastName',
|
||||
sortable: true,
|
||||
},
|
||||
{ name: 'option', align: 'center' as const, label: '', field: 'option', icon: 'option' },
|
||||
]);
|
||||
|
||||
const { NotifyResponse } = useNotify();
|
||||
|
||||
const loading = ref(false);
|
||||
|
||||
//updates Attendees list from database
|
||||
async function updateAttendees() {
|
||||
loading.value = true;
|
||||
|
||||
let events: Events | undefined;
|
||||
|
||||
await appApi
|
||||
.get('events')
|
||||
.then((resp) => {
|
||||
if (resp.data === null) {
|
||||
attendees.value = [];
|
||||
return;
|
||||
}
|
||||
events = resp.data as Events;
|
||||
})
|
||||
|
||||
.catch((err) => {
|
||||
NotifyResponse(err, 'error');
|
||||
})
|
||||
.finally(() => {
|
||||
loading.value = false;
|
||||
});
|
||||
|
||||
if (!events || events.length === 0 || !events[0]?.attendees || events[0].attendees === null) {
|
||||
attendees.value = [];
|
||||
return;
|
||||
}
|
||||
attendees.value = events[0].attendees ?? [];
|
||||
}
|
||||
|
||||
return {
|
||||
attendees,
|
||||
pagination,
|
||||
columns,
|
||||
loading,
|
||||
updateAttendees,
|
||||
};
|
||||
}
|
||||
233
src/vueLib/tables/attendees/AttendeesTable.vue
Normal file
233
src/vueLib/tables/attendees/AttendeesTable.vue
Normal file
@@ -0,0 +1,233 @@
|
||||
<template>
|
||||
<DialogFrame ref="dialog" :header-title="$t('attendees')" :width="600">
|
||||
<div class="q-pa-md">
|
||||
<q-table
|
||||
flat
|
||||
bordered
|
||||
ref="tableRef"
|
||||
:title="$t('attendees')"
|
||||
title-class="text-bold text-blue-9"
|
||||
:no-data-label="$t('noDataAvailable')"
|
||||
:loading-label="$t('loading')"
|
||||
:rows-per-page-label="$t('recordsPerPage')"
|
||||
:selected-rows-label="(val) => val + ' ' + $t('recordSelected')"
|
||||
:rows="attendees"
|
||||
:columns="columns"
|
||||
row-key="id"
|
||||
v-model:pagination="pagination"
|
||||
:loading="loading"
|
||||
:filter="filter"
|
||||
:selection="selectOption ? 'multiple' : 'none'"
|
||||
v-model:selected="selected"
|
||||
binary-state-sort
|
||||
class="bigger-table-text"
|
||||
>
|
||||
<template v-slot:top-left>
|
||||
<q-btn-group push flat style="color: grey">
|
||||
<q-btn
|
||||
v-if="user.isPermittedTo('events', 'write')"
|
||||
dense
|
||||
flat
|
||||
icon="person"
|
||||
@click="openAllValueDialog"
|
||||
><q-badge floating transparent color="primary" text-color="primary-text">+</q-badge>
|
||||
<q-tooltip>{{ $t('addNewAttendees') }}</q-tooltip>
|
||||
</q-btn>
|
||||
<q-btn
|
||||
v-if="user.isPermittedTo('events', 'write')"
|
||||
dense
|
||||
flat
|
||||
style="color: grey"
|
||||
:icon="selectOption ? 'check_box' : 'check_box_outline_blank'"
|
||||
@click="selectOption = !selectOption"
|
||||
>
|
||||
<q-tooltip>{{ $t('selectAttendeesOptions') }}</q-tooltip>
|
||||
</q-btn>
|
||||
</q-btn-group>
|
||||
<div v-if="selectOption && selected.length > 0">
|
||||
<q-btn flat dense icon="more_vert" @click="openSubmenu = true" />
|
||||
<q-menu v-if="openSubmenu" anchor="bottom middle" self="top middle">
|
||||
<q-item
|
||||
clickable
|
||||
v-close-popup
|
||||
@click="openRemoveDialog(...selected)"
|
||||
class="text-negative"
|
||||
>{{ $t('delete') }}</q-item
|
||||
>
|
||||
</q-menu>
|
||||
</div>
|
||||
<div v-if="selectOption && selected.length > 0" class="q-ml-md text-weight-bold">
|
||||
{{ $t('selected') }}: {{ selected.length }}
|
||||
</div>
|
||||
</template>
|
||||
<template v-slot:top-right>
|
||||
<q-input filled dense debounce="300" v-model="filter" :placeholder="$t('search')">
|
||||
<template v-slot:append>
|
||||
<q-icon name="search" />
|
||||
</template>
|
||||
</q-input>
|
||||
</template>
|
||||
<template v-slot:body-cell="props">
|
||||
<q-td v-if="props.col.field === 'attendees'" :props="props">
|
||||
<q-btn v-if="props.value !== null && props.value.length > 0" dense flat icon="people"
|
||||
><q-badge color="primary" text-color="primary-text" floating transparent>{{
|
||||
props.row.count
|
||||
}}</q-badge></q-btn
|
||||
>
|
||||
</q-td>
|
||||
<q-td v-else :props="props">
|
||||
{{ props.value }}
|
||||
</q-td>
|
||||
</template>
|
||||
<template v-slot:body-cell-option="props">
|
||||
<q-td :props="props">
|
||||
<q-btn
|
||||
v-if="user.isPermittedTo('events', 'delete')"
|
||||
flat
|
||||
dense
|
||||
@click="openRemoveDialog(props.row)"
|
||||
color="negative"
|
||||
icon="delete"
|
||||
/>
|
||||
</q-td>
|
||||
</template>
|
||||
</q-table>
|
||||
</div>
|
||||
<DialogFrame ref="memberTableDialog" :header-title="$t('members')" :width="700">
|
||||
<MembersTable add-attendees v-on:update-event="updateTable" :event-id="localEvent?.id ?? 0" />
|
||||
</DialogFrame>
|
||||
|
||||
<OkDialog
|
||||
ref="okDialog"
|
||||
:dialog-label="$t('delete')"
|
||||
:text="$t('doYouWantToDelete') + ' ' + deleteText"
|
||||
label-color="red"
|
||||
:button-cancel-label="$t('cancel')"
|
||||
:button-ok-label="$t('confirm')"
|
||||
:button-ok-flat="false"
|
||||
button-ok-color="red"
|
||||
v-on:update-confirm="(val) => removeAttendees(...val)"
|
||||
></OkDialog>
|
||||
</DialogFrame>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
import { appApi } from 'src/boot/axios';
|
||||
import { ref } from 'vue';
|
||||
import type { Attendees } from 'src/vueLib/models/attendee';
|
||||
import DialogFrame from 'src/vueLib/dialog/DialogFrame.vue';
|
||||
import OkDialog from 'src/components/dialog/OkDialog.vue';
|
||||
import { useNotify } from 'src/vueLib/general/useNotify';
|
||||
import { useAttendeesTable } from '../attendees/AttendeesTable';
|
||||
import { useUserStore } from 'src/vueLib/login/userStore';
|
||||
import type { Event } from 'src/vueLib/models/event';
|
||||
import MembersTable from '../members/MembersTable.vue';
|
||||
import { i18n } from 'src/boot/lang';
|
||||
|
||||
export interface AttendeesDialog {
|
||||
getSelected: () => Attendees;
|
||||
}
|
||||
|
||||
const emit = defineEmits(['update']);
|
||||
|
||||
const { NotifyResponse } = useNotify();
|
||||
const memberTableDialog = ref();
|
||||
const dialog = ref();
|
||||
const okDialog = ref();
|
||||
const deleteText = ref('');
|
||||
const selectOption = ref(false);
|
||||
const localEvent = ref<Event>();
|
||||
const selected = ref<Attendees>([]);
|
||||
const openSubmenu = ref(false);
|
||||
const filter = ref('');
|
||||
const user = useUserStore();
|
||||
|
||||
const { attendees, pagination, loading, columns, updateAttendees } = useAttendeesTable();
|
||||
|
||||
const open = (event: Event) => {
|
||||
localEvent.value = event;
|
||||
attendees.value = event.attendees ?? [];
|
||||
dialog.value.open();
|
||||
};
|
||||
|
||||
//opens dialog for one value
|
||||
function openAllValueDialog() {
|
||||
memberTableDialog.value?.open();
|
||||
}
|
||||
|
||||
//opens remove dialog
|
||||
function openRemoveDialog(...attendees: Attendees) {
|
||||
if (attendees.length === 1) {
|
||||
deleteText.value = "'";
|
||||
if (attendees[0]?.firstName && attendees[0]?.lastName) {
|
||||
deleteText.value += attendees[0]?.firstName + ' ' + attendees[0]?.lastName;
|
||||
}
|
||||
deleteText.value += "'";
|
||||
} else {
|
||||
deleteText.value = String(attendees.length) + ' attendees';
|
||||
}
|
||||
okDialog.value?.open(attendees);
|
||||
}
|
||||
|
||||
//remove Attendees from database
|
||||
async function removeAttendees(...removeAttendees: Attendees) {
|
||||
if (!localEvent.value) {
|
||||
console.error('event is empty');
|
||||
return;
|
||||
}
|
||||
|
||||
localEvent.value.attendees = removeAttendees;
|
||||
|
||||
await appApi
|
||||
.post('events/delete/attendees', localEvent.value)
|
||||
.then(() => {
|
||||
selected.value = [];
|
||||
if (localEvent.value?.attendees !== undefined && localEvent.value?.attendees.length > 1) {
|
||||
NotifyResponse(i18n.global.t('deleteAttendees'), 'warning');
|
||||
} else {
|
||||
NotifyResponse(i18n.global.t('deleteAttendee'), 'warning');
|
||||
}
|
||||
})
|
||||
.catch((err) => NotifyResponse(err, 'error'))
|
||||
.finally(() => {
|
||||
loading.value = false;
|
||||
});
|
||||
emit('update');
|
||||
await updateAttendees();
|
||||
}
|
||||
|
||||
async function updateTable() {
|
||||
await updateAttendees();
|
||||
emit('update');
|
||||
}
|
||||
|
||||
defineExpose({
|
||||
open,
|
||||
});
|
||||
</script>
|
||||
|
||||
<style>
|
||||
@keyframes blink-yellow {
|
||||
0%,
|
||||
100% {
|
||||
background-color: yellow;
|
||||
}
|
||||
50% {
|
||||
background-color: transparent;
|
||||
}
|
||||
}
|
||||
|
||||
.blink-yellow {
|
||||
animation: blink-yellow 1.5s step-start 6 !important;
|
||||
}
|
||||
|
||||
.bigger-table-text .q-table__middle td {
|
||||
font-size: 14px;
|
||||
}
|
||||
|
||||
.bigger-table-text .q-table__top,
|
||||
.bigger-table-text .q-table__bottom,
|
||||
.bigger-table-text th {
|
||||
font-size: 14px;
|
||||
}
|
||||
</style>
|
||||
79
src/vueLib/tables/events/EventsTable.ts
Normal file
79
src/vueLib/tables/events/EventsTable.ts
Normal file
@@ -0,0 +1,79 @@
|
||||
import { appApi } from 'src/boot/axios';
|
||||
import { ref, computed } from 'vue';
|
||||
import type { Events } from 'src/vueLib/models/event';
|
||||
import { useNotify } from 'src/vueLib/general/useNotify';
|
||||
import { i18n } from 'boot/lang';
|
||||
|
||||
export function useEventTable() {
|
||||
const Events = ref<Events>([]);
|
||||
|
||||
const pagination = ref({
|
||||
sortBy: 'firstName',
|
||||
descending: false,
|
||||
page: 1,
|
||||
rowsPerPage: 20,
|
||||
});
|
||||
|
||||
const columns = computed(() => [
|
||||
{
|
||||
name: 'name',
|
||||
align: 'left' as const,
|
||||
label: i18n.global.t('name'),
|
||||
field: 'name',
|
||||
sortable: true,
|
||||
},
|
||||
{
|
||||
name: 'attendees',
|
||||
align: 'center' as const,
|
||||
label: i18n.global.t('attendees'),
|
||||
field: 'attendees',
|
||||
sortable: true,
|
||||
},
|
||||
{
|
||||
name: 'date',
|
||||
align: 'left' as const,
|
||||
label: i18n.global.t('dateAndTime'),
|
||||
field: 'date',
|
||||
sortable: true,
|
||||
},
|
||||
{ name: 'option', align: 'center' as const, label: '', field: 'option', icon: 'option' },
|
||||
]);
|
||||
|
||||
const { NotifyResponse } = useNotify();
|
||||
|
||||
const loading = ref(false);
|
||||
|
||||
//updates Event list from database
|
||||
function updateEvents() {
|
||||
loading.value = true;
|
||||
|
||||
appApi
|
||||
.get('events')
|
||||
.then((resp) => {
|
||||
if (resp.data === null) {
|
||||
Events.value = [];
|
||||
return;
|
||||
}
|
||||
Events.value = resp.data as Events;
|
||||
if (Events.value === null) {
|
||||
Events.value = [];
|
||||
return;
|
||||
}
|
||||
})
|
||||
|
||||
.catch((err) => {
|
||||
NotifyResponse(err, 'error');
|
||||
})
|
||||
.finally(() => {
|
||||
loading.value = false;
|
||||
});
|
||||
}
|
||||
|
||||
return {
|
||||
Events,
|
||||
pagination,
|
||||
columns,
|
||||
loading,
|
||||
updateEvents,
|
||||
};
|
||||
}
|
||||
268
src/vueLib/tables/events/EventsTable.vue
Normal file
268
src/vueLib/tables/events/EventsTable.vue
Normal file
@@ -0,0 +1,268 @@
|
||||
<template>
|
||||
<div class="q-pa-md">
|
||||
<q-table
|
||||
flat
|
||||
bordered
|
||||
ref="tableRef"
|
||||
title="Events"
|
||||
title-class="text-bold text-blue-9"
|
||||
:no-data-label="$t('noDataAvailable')"
|
||||
:loading-label="$t('loading')"
|
||||
:rows-per-page-label="$t('recordsPerPage')"
|
||||
:selected-rows-label="(val) => val + ' ' + $t('recordSelected')"
|
||||
:rows="Events"
|
||||
:columns="columns"
|
||||
row-key="id"
|
||||
v-model:pagination="pagination"
|
||||
:loading="loading"
|
||||
:filter="filter"
|
||||
:selection="selectOption ? 'multiple' : 'none'"
|
||||
v-model:selected="selected"
|
||||
binary-state-sort
|
||||
class="bigger-table-text"
|
||||
>
|
||||
<template v-slot:top-left>
|
||||
<q-btn-group push flat style="color: grey">
|
||||
<q-btn
|
||||
v-if="user.isPermittedTo('events', 'write')"
|
||||
dense
|
||||
flat
|
||||
icon="add"
|
||||
@click="openAllValueDialog(null)"
|
||||
>
|
||||
<q-tooltip>{{ $t('addNewEvent') }}</q-tooltip>
|
||||
</q-btn>
|
||||
<q-btn
|
||||
v-if="user.isPermittedTo('events', 'write')"
|
||||
dense
|
||||
flat
|
||||
style="color: grey"
|
||||
:icon="selectOption ? 'check_box' : 'check_box_outline_blank'"
|
||||
@click="selectOption = !selectOption"
|
||||
>
|
||||
<q-tooltip>{{ $t('selectEventOptions') }}</q-tooltip>
|
||||
</q-btn>
|
||||
</q-btn-group>
|
||||
<div v-if="selectOption && selected.length > 0">
|
||||
<q-btn flat dense icon="more_vert" @click="openSubmenu = true" />
|
||||
<q-menu v-if="openSubmenu" anchor="bottom middle" self="top middle">
|
||||
<q-item
|
||||
clickable
|
||||
v-close-popup
|
||||
@click="openRemoveDialog(...selected)"
|
||||
class="text-negative"
|
||||
>{{ $t('delete') }}</q-item
|
||||
>
|
||||
</q-menu>
|
||||
</div>
|
||||
<div v-if="selectOption && selected.length > 0" class="q-ml-md text-weight-bold">
|
||||
{{ $t('selected') }}: {{ selected.length }}
|
||||
</div>
|
||||
</template>
|
||||
<template v-slot:top-right>
|
||||
<q-input filled dense debounce="300" v-model="filter" :placeholder="$t('search')">
|
||||
<template v-slot:append>
|
||||
<q-icon name="search" />
|
||||
</template>
|
||||
</q-input>
|
||||
</template>
|
||||
<template v-slot:body-cell="props">
|
||||
<q-td v-if="props.col.field === 'attendees'" :props="props">
|
||||
<q-btn dense flat icon="people" @click="openAttendees(props.row)"
|
||||
><q-badge color="primary" text-color="primary-text" floating transparent>{{
|
||||
props.row.count
|
||||
}}</q-badge></q-btn
|
||||
>
|
||||
</q-td>
|
||||
<q-td
|
||||
v-else
|
||||
:props="props"
|
||||
:style="
|
||||
user.isPermittedTo('events', 'write') && props.col.field !== 'count'
|
||||
? 'cursor: pointer'
|
||||
: ''
|
||||
"
|
||||
@click="
|
||||
user.isPermittedTo('events', 'write') &&
|
||||
props.col.field !== 'count' &&
|
||||
openSingleValueDialog(props.col.label, props.col.name, props.row)
|
||||
"
|
||||
>
|
||||
{{ props.value }}
|
||||
</q-td>
|
||||
</template>
|
||||
<template v-slot:body-cell-option="props">
|
||||
<q-td :props="props">
|
||||
<q-btn
|
||||
v-if="user.isPermittedTo('events', 'write') || user.isPermittedTo('events', 'delete')"
|
||||
flat
|
||||
dense
|
||||
icon="more_vert"
|
||||
@click="openSubmenu = true"
|
||||
/>
|
||||
<q-menu v-if="openSubmenu" anchor="top right" self="top left">
|
||||
<q-item
|
||||
v-if="user.isPermittedTo('events', 'write')"
|
||||
clickable
|
||||
v-close-popup
|
||||
@click="openAllValueDialog(props.row)"
|
||||
class="text-primary"
|
||||
>{{ $t('edit') }}</q-item
|
||||
>
|
||||
<q-item
|
||||
v-if="user.isPermittedTo('events', 'delete')"
|
||||
clickable
|
||||
v-close-popup
|
||||
@click="openRemoveDialog(props.row)"
|
||||
class="text-negative"
|
||||
title="zu"
|
||||
>{{ $t('delete') }}</q-item
|
||||
>
|
||||
</q-menu>
|
||||
</q-td>
|
||||
</template>
|
||||
</q-table>
|
||||
</div>
|
||||
<EditOneDialog
|
||||
ref="editOneDialog"
|
||||
endpoint="Events/edit"
|
||||
query-id
|
||||
v-on:update="updateEvents"
|
||||
></EditOneDialog>
|
||||
<EditAllDialog ref="editAllDialog" v-on:update="updateEvents"></EditAllDialog>
|
||||
<AttendeesTable ref="attendeesDialog" v-on:update="updateEvents" />
|
||||
<OkDialog
|
||||
ref="okDialog"
|
||||
:dialog-label="$t('delete')"
|
||||
:text="$t('doYouWantToDelete') + ' ' + deleteText"
|
||||
label-color="red"
|
||||
:button-cancel-label="$t('cancel')"
|
||||
:button-ok-label="$t('confirm')"
|
||||
:button-ok-flat="false"
|
||||
button-ok-color="red"
|
||||
v-on:update-confirm="(val) => removeEvent(...val)"
|
||||
></OkDialog>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
import { appApi } from 'src/boot/axios';
|
||||
import { ref, onMounted } from 'vue';
|
||||
import type { Event, Events } from 'src/vueLib/models/event';
|
||||
import EditOneDialog from 'src/components/EditOneDialog.vue';
|
||||
import EditAllDialog from 'src/components/EventEditAllDialog.vue';
|
||||
import OkDialog from 'src/components/dialog/OkDialog.vue';
|
||||
import { useNotify } from 'src/vueLib/general/useNotify';
|
||||
import { useEventTable } from './EventsTable';
|
||||
import { databaseName } from '../members/MembersTable';
|
||||
import { useUserStore } from 'src/vueLib/login/userStore';
|
||||
import AttendeesTable from '../attendees/AttendeesTable.vue';
|
||||
import type { Attendees } from 'src/vueLib/models/attendee';
|
||||
|
||||
export interface EventDialog {
|
||||
getSelected: () => Events;
|
||||
}
|
||||
|
||||
const { NotifyResponse } = useNotify();
|
||||
const editOneDialog = ref();
|
||||
const editAllDialog = ref();
|
||||
const attendeesDialog = ref();
|
||||
const okDialog = ref();
|
||||
const deleteText = ref('');
|
||||
const selectOption = ref(false);
|
||||
const selected = ref<Events>([]);
|
||||
const openSubmenu = ref(false);
|
||||
const filter = ref('');
|
||||
const user = useUserStore();
|
||||
|
||||
const { Events, pagination, loading, columns, updateEvents } = useEventTable();
|
||||
|
||||
//load on mounting page
|
||||
onMounted(() => {
|
||||
loading.value = true;
|
||||
|
||||
appApi
|
||||
.post('database/open', { dbPath: databaseName.value, create: true })
|
||||
.then(() => {
|
||||
updateEvents();
|
||||
})
|
||||
.catch((err) => NotifyResponse(err, 'error'))
|
||||
|
||||
.finally(() => {
|
||||
loading.value = false;
|
||||
});
|
||||
});
|
||||
|
||||
// opens dialog for all Event values
|
||||
function openSingleValueDialog(label: string, field: string, Event: Event) {
|
||||
editOneDialog.value?.open(label, field, Event);
|
||||
}
|
||||
|
||||
//opens dialog for one value
|
||||
function openAllValueDialog(Event: Event | null) {
|
||||
editAllDialog.value?.open(Event);
|
||||
}
|
||||
|
||||
//opens remove dialog
|
||||
function openRemoveDialog(...Events: Events) {
|
||||
if (Events.length === 1) {
|
||||
deleteText.value = "'";
|
||||
if (Events[0]?.name !== undefined) {
|
||||
deleteText.value += Events[0]?.name + ' ';
|
||||
}
|
||||
deleteText.value += "'";
|
||||
} else {
|
||||
deleteText.value = String(Events.length) + ' Events';
|
||||
}
|
||||
okDialog.value?.open(Events);
|
||||
}
|
||||
|
||||
function openAttendees(attendees: Attendees | null) {
|
||||
attendeesDialog.value.open(attendees);
|
||||
}
|
||||
|
||||
//remove Event from database
|
||||
function removeEvent(...removeEvents: Events) {
|
||||
const EventIds: number[] = [];
|
||||
|
||||
removeEvents.forEach((Event: Event) => {
|
||||
EventIds.push(Event.id);
|
||||
});
|
||||
|
||||
appApi
|
||||
.post('events/delete', { ids: EventIds })
|
||||
.then(() => {
|
||||
updateEvents();
|
||||
selected.value = [];
|
||||
})
|
||||
.catch((err) => NotifyResponse(err, 'error'))
|
||||
.finally(() => {
|
||||
loading.value = false;
|
||||
});
|
||||
}
|
||||
</script>
|
||||
|
||||
<style>
|
||||
@keyframes blink-yellow {
|
||||
0%,
|
||||
100% {
|
||||
background-color: yellow;
|
||||
}
|
||||
50% {
|
||||
background-color: transparent;
|
||||
}
|
||||
}
|
||||
|
||||
.blink-yellow {
|
||||
animation: blink-yellow 1.5s step-start 6 !important;
|
||||
}
|
||||
|
||||
.bigger-table-text .q-table__middle td {
|
||||
font-size: 14px;
|
||||
}
|
||||
|
||||
.bigger-table-text .q-table__top,
|
||||
.bigger-table-text .q-table__bottom,
|
||||
.bigger-table-text th {
|
||||
font-size: 14px;
|
||||
}
|
||||
</style>
|
||||
@@ -16,101 +16,122 @@ export function useMemberTable() {
|
||||
rowsPerPage: 20,
|
||||
});
|
||||
|
||||
const columns = computed(() => [
|
||||
{ name: 'cake', align: 'center' as const, label: '', field: 'cake', icon: 'cake' },
|
||||
{
|
||||
name: 'firstName',
|
||||
align: 'left' as const,
|
||||
label: i18n.global.t('prename'),
|
||||
field: 'firstName',
|
||||
sortable: true,
|
||||
},
|
||||
{
|
||||
name: 'lastName',
|
||||
align: 'left' as const,
|
||||
label: i18n.global.t('lastName'),
|
||||
field: 'lastName',
|
||||
sortable: true,
|
||||
},
|
||||
{
|
||||
name: 'birthday',
|
||||
align: 'left' as const,
|
||||
label: i18n.global.t('birthday'),
|
||||
field: 'birthday',
|
||||
sortable: true,
|
||||
},
|
||||
{
|
||||
name: 'age',
|
||||
align: 'left' as const,
|
||||
label: i18n.global.t('age'),
|
||||
field: 'age',
|
||||
sortable: true,
|
||||
},
|
||||
{
|
||||
name: 'address',
|
||||
align: 'left' as const,
|
||||
label: i18n.global.t('address'),
|
||||
field: 'address',
|
||||
sortable: true,
|
||||
},
|
||||
{
|
||||
name: 'town',
|
||||
align: 'left' as const,
|
||||
label: i18n.global.t('town'),
|
||||
field: 'town',
|
||||
sortable: true,
|
||||
},
|
||||
{
|
||||
name: 'zip',
|
||||
align: 'left' as const,
|
||||
label: i18n.global.t('zipCode'),
|
||||
field: 'zip',
|
||||
sortable: true,
|
||||
},
|
||||
{
|
||||
name: 'phone',
|
||||
align: 'left' as const,
|
||||
label: i18n.global.t('phone'),
|
||||
field: 'phone',
|
||||
sortable: true,
|
||||
},
|
||||
{
|
||||
name: 'email',
|
||||
align: 'left' as const,
|
||||
label: i18n.global.t('email'),
|
||||
field: 'email',
|
||||
sortable: true,
|
||||
},
|
||||
{
|
||||
name: 'group',
|
||||
align: 'left' as const,
|
||||
label: i18n.global.t('group'),
|
||||
field: 'group',
|
||||
sortable: true,
|
||||
},
|
||||
{
|
||||
name: 'responsiblePerson',
|
||||
align: 'left' as const,
|
||||
label: i18n.global.t('responsible'),
|
||||
field: 'responsiblePerson',
|
||||
sortable: true,
|
||||
},
|
||||
{
|
||||
name: 'firstVisit',
|
||||
align: 'left' as const,
|
||||
label: i18n.global.t('firstVisit'),
|
||||
field: 'firstVisit',
|
||||
sortable: true,
|
||||
},
|
||||
{
|
||||
name: 'lastVisit',
|
||||
align: 'left' as const,
|
||||
label: i18n.global.t('lastVisit'),
|
||||
field: 'lastVisit',
|
||||
sortable: true,
|
||||
},
|
||||
{ name: 'option', align: 'center' as const, label: '', field: 'option', icon: 'option' },
|
||||
]);
|
||||
//add enabling of each columns
|
||||
const enabledColumns = ref<Record<string, boolean>>({
|
||||
cake: true,
|
||||
firstName: true,
|
||||
lastName: true,
|
||||
birthday: true,
|
||||
age: true,
|
||||
address: false,
|
||||
town: true,
|
||||
zip: true,
|
||||
phone: true,
|
||||
email: true,
|
||||
group: true,
|
||||
responsiblePerson: true,
|
||||
firstVisit: true,
|
||||
lastVisit: true,
|
||||
option: true,
|
||||
});
|
||||
|
||||
const columns = computed(() =>
|
||||
[
|
||||
{ name: 'cake', align: 'center' as const, label: '', field: 'cake', icon: 'cake' },
|
||||
{
|
||||
name: 'firstName',
|
||||
align: 'left' as const,
|
||||
label: i18n.global.t('prename'),
|
||||
field: 'firstName',
|
||||
sortable: true,
|
||||
},
|
||||
{
|
||||
name: 'lastName',
|
||||
align: 'left' as const,
|
||||
label: i18n.global.t('lastName'),
|
||||
field: 'lastName',
|
||||
sortable: true,
|
||||
},
|
||||
{
|
||||
name: 'birthday',
|
||||
align: 'left' as const,
|
||||
label: i18n.global.t('birthday'),
|
||||
field: 'birthday',
|
||||
sortable: true,
|
||||
},
|
||||
{
|
||||
name: 'age',
|
||||
align: 'left' as const,
|
||||
label: i18n.global.t('age'),
|
||||
field: 'age',
|
||||
sortable: true,
|
||||
},
|
||||
{
|
||||
name: 'address',
|
||||
align: 'left' as const,
|
||||
label: i18n.global.t('address'),
|
||||
field: 'address',
|
||||
sortable: true,
|
||||
},
|
||||
{
|
||||
name: 'town',
|
||||
align: 'left' as const,
|
||||
label: i18n.global.t('town'),
|
||||
field: 'town',
|
||||
sortable: true,
|
||||
},
|
||||
{
|
||||
name: 'zip',
|
||||
align: 'left' as const,
|
||||
label: i18n.global.t('zipCode'),
|
||||
field: 'zip',
|
||||
sortable: true,
|
||||
},
|
||||
{
|
||||
name: 'phone',
|
||||
align: 'left' as const,
|
||||
label: i18n.global.t('phone'),
|
||||
field: 'phone',
|
||||
sortable: true,
|
||||
},
|
||||
{
|
||||
name: 'email',
|
||||
align: 'left' as const,
|
||||
label: i18n.global.t('email'),
|
||||
field: 'email',
|
||||
sortable: true,
|
||||
},
|
||||
{
|
||||
name: 'group',
|
||||
align: 'left' as const,
|
||||
label: i18n.global.t('group'),
|
||||
field: 'group',
|
||||
sortable: true,
|
||||
},
|
||||
{
|
||||
name: 'responsiblePerson',
|
||||
align: 'left' as const,
|
||||
label: i18n.global.t('responsible'),
|
||||
field: 'responsiblePerson',
|
||||
sortable: true,
|
||||
},
|
||||
{
|
||||
name: 'firstVisit',
|
||||
align: 'left' as const,
|
||||
label: i18n.global.t('firstVisit'),
|
||||
field: 'firstVisit',
|
||||
sortable: true,
|
||||
},
|
||||
{
|
||||
name: 'lastVisit',
|
||||
align: 'left' as const,
|
||||
label: i18n.global.t('lastVisit'),
|
||||
field: 'lastVisit',
|
||||
sortable: true,
|
||||
},
|
||||
{ name: 'option', align: 'center' as const, label: '', field: 'option', icon: 'option' },
|
||||
].filter((c) => enabledColumns.value[c.name]),
|
||||
);
|
||||
|
||||
const { NotifyResponse } = useNotify();
|
||||
|
||||
@@ -197,6 +218,13 @@ export function useMemberTable() {
|
||||
});
|
||||
}
|
||||
|
||||
function disableColumns(...columns: string[]) {
|
||||
columns.forEach((col) => {
|
||||
if (col in enabledColumns.value) {
|
||||
enabledColumns.value[col] = false;
|
||||
}
|
||||
});
|
||||
}
|
||||
return {
|
||||
members,
|
||||
pagination,
|
||||
@@ -205,5 +233,6 @@ export function useMemberTable() {
|
||||
getRowClass,
|
||||
updateMembers,
|
||||
isXDaysBeforeAnnualDate,
|
||||
disableColumns,
|
||||
};
|
||||
}
|
||||
|
||||
@@ -9,7 +9,7 @@
|
||||
:no-data-label="$t('noDataAvailable')"
|
||||
:loading-label="$t('loading')"
|
||||
:rows-per-page-label="$t('recordsPerPage')"
|
||||
:selected-rows-label="(val) => val + $t('recordSelected')"
|
||||
:selected-rows-label="(val) => val + ' ' + $t('recordSelected')"
|
||||
:rows="members"
|
||||
:columns="columns"
|
||||
row-key="id"
|
||||
@@ -24,7 +24,7 @@
|
||||
<template v-slot:top-left>
|
||||
<q-btn-group push flat style="color: grey">
|
||||
<q-btn
|
||||
v-if="user.isPermittedTo('userSettings', 'write')"
|
||||
v-if="user.isPermittedTo('members', 'write')"
|
||||
dense
|
||||
flat
|
||||
icon="add"
|
||||
@@ -33,7 +33,7 @@
|
||||
<q-tooltip>{{ $t('addNewMember') }}</q-tooltip>
|
||||
</q-btn>
|
||||
<q-btn
|
||||
v-if="user.isPermittedTo('userSettings', 'write')"
|
||||
v-if="user.isPermittedTo('members', 'write') || user.isPermittedTo('members', 'delete')"
|
||||
dense
|
||||
flat
|
||||
style="color: grey"
|
||||
@@ -43,7 +43,7 @@
|
||||
<q-tooltip>{{ $t('selectMemberOptions') }}</q-tooltip>
|
||||
</q-btn>
|
||||
<q-btn
|
||||
v-if="user.isPermittedTo('userSettings', 'write')"
|
||||
v-if="user.isPermittedTo('members', 'write')"
|
||||
dense
|
||||
flat
|
||||
icon="upload"
|
||||
@@ -53,9 +53,22 @@
|
||||
</q-btn>
|
||||
</q-btn-group>
|
||||
<div v-if="selectOption && selected.length > 0">
|
||||
<q-btn flat dense icon="more_vert" @click="openSubmenu = true" />
|
||||
<q-btn
|
||||
v-if="!inProps.addAttendees"
|
||||
flat
|
||||
dense
|
||||
icon="more_vert"
|
||||
@click="openSubmenu = true"
|
||||
/>
|
||||
<q-btn v-else dense color="grey-7" flat icon="person" @click="addEventAttendees">
|
||||
<q-badge floating transparent color="primary" text-color="primary-text">+</q-badge>
|
||||
</q-btn>
|
||||
<q-menu v-if="openSubmenu" anchor="bottom middle" self="top middle">
|
||||
<q-item clickable v-close-popup @click="addToEvent" class="text-primary">{{
|
||||
$t('addToEvent')
|
||||
}}</q-item>
|
||||
<q-item
|
||||
v-if="user.isPermittedTo('members', 'delete')"
|
||||
clickable
|
||||
v-close-popup
|
||||
@click="openRemoveDialog(...selected)"
|
||||
@@ -64,7 +77,7 @@
|
||||
>
|
||||
</q-menu>
|
||||
</div>
|
||||
<div v-if="selectOption && selected.length > 0" class="q-ml-md text-weight-bold">
|
||||
<div v-if="selectOption && selected.length > 0" class="text-weight-bold">
|
||||
{{ $t('selected') }}: {{ selected.length }}
|
||||
</div>
|
||||
</template>
|
||||
@@ -79,9 +92,9 @@
|
||||
<q-td
|
||||
:props="props"
|
||||
:class="getRowClass(props.row)"
|
||||
:style="user.isPermittedTo('userSettings', 'write') ? 'cursor: pointer' : ''"
|
||||
:style="user.isPermittedTo('members', 'write') ? 'cursor: pointer' : ''"
|
||||
@click="
|
||||
user.isPermittedTo('userSettings', 'write') &&
|
||||
user.isPermittedTo('members', 'write') &&
|
||||
openSingleValueDialog(props.col.label, props.col.name, props.row)
|
||||
"
|
||||
>
|
||||
@@ -100,10 +113,7 @@
|
||||
<template v-slot:body-cell-option="props">
|
||||
<q-td :props="props">
|
||||
<q-btn
|
||||
v-if="
|
||||
user.isPermittedTo('userSettings', 'write') ||
|
||||
user.isPermittedTo('userSettings', 'delete')
|
||||
"
|
||||
v-if="user.isPermittedTo('members', 'write') || user.isPermittedTo('members', 'delete')"
|
||||
flat
|
||||
dense
|
||||
icon="more_vert"
|
||||
@@ -111,7 +121,7 @@
|
||||
/>
|
||||
<q-menu v-if="openSubmenu" anchor="top right" self="top left">
|
||||
<q-item
|
||||
v-if="user.isPermittedTo('userSettings', 'write')"
|
||||
v-if="user.isPermittedTo('members', 'write')"
|
||||
clickable
|
||||
v-close-popup
|
||||
@click="openAllValueDialog(props.row)"
|
||||
@@ -119,7 +129,7 @@
|
||||
>{{ $t('edit') }}</q-item
|
||||
>
|
||||
<q-item
|
||||
v-if="user.isPermittedTo('userSettings', 'delete')"
|
||||
v-if="user.isPermittedTo('members', 'delete') && !inProps.addAttendees"
|
||||
clickable
|
||||
v-close-popup
|
||||
@click="openRemoveDialog(props.row)"
|
||||
@@ -151,6 +161,11 @@
|
||||
v-on:update-confirm="(val) => removeMember(...val)"
|
||||
></OkDialog>
|
||||
<UploadDialog ref="uploadDialog" @update-upload="updateMembers"> </UploadDialog>
|
||||
<AddToEvent
|
||||
ref="addToEventDialog"
|
||||
endpoint="events/add/attendees"
|
||||
v-on:update-event="(val) => updateMemberLastVisit(val)"
|
||||
/>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
@@ -163,16 +178,22 @@ import OkDialog from 'src/components/dialog/OkDialog.vue';
|
||||
import { useNotify } from 'src/vueLib/general/useNotify';
|
||||
import { useMemberTable } from './MembersTable';
|
||||
import UploadDialog from 'src/components/UploadDialog.vue';
|
||||
import AddToEvent from 'src/components/AddToEvent.vue';
|
||||
import { databaseName } from './MembersTable';
|
||||
import { useUserStore } from 'src/vueLib/login/userStore';
|
||||
import { i18n } from 'src/boot/lang';
|
||||
|
||||
const inProps = defineProps({ addAttendees: { type: Boolean }, eventId: { type: Number } });
|
||||
export interface MemberDialog {
|
||||
getSelected: () => Members;
|
||||
}
|
||||
|
||||
const emit = defineEmits(['update-event']);
|
||||
|
||||
const { NotifyResponse } = useNotify();
|
||||
const editOneDialog = ref();
|
||||
const editAllDialog = ref();
|
||||
const addToEventDialog = ref();
|
||||
const uploadDialog = ref();
|
||||
const okDialog = ref();
|
||||
const deleteText = ref('');
|
||||
@@ -190,10 +211,28 @@ const {
|
||||
getRowClass,
|
||||
updateMembers,
|
||||
isXDaysBeforeAnnualDate,
|
||||
disableColumns,
|
||||
} = useMemberTable();
|
||||
|
||||
//load on mounting page
|
||||
onMounted(() => {
|
||||
if (inProps.addAttendees) {
|
||||
selectOption.value = true;
|
||||
disableColumns(
|
||||
'birthday',
|
||||
'age',
|
||||
'town',
|
||||
'zip',
|
||||
'email',
|
||||
'address',
|
||||
'phone',
|
||||
'group',
|
||||
'responsiblePerson',
|
||||
'firstVisit',
|
||||
'lastVisit',
|
||||
);
|
||||
}
|
||||
|
||||
loading.value = true;
|
||||
|
||||
appApi
|
||||
@@ -260,24 +299,62 @@ function removeMember(...removeMembers: Members) {
|
||||
});
|
||||
}
|
||||
|
||||
//const blinkingId = ref<number | null>(null);
|
||||
|
||||
// function triggerBlink(id: number) {
|
||||
// blinkingId.value = id;
|
||||
|
||||
// // Optional: stop blinking after 3 seconds
|
||||
// setTimeout(() => {
|
||||
// blinkingId.value = null;
|
||||
// }, 3000);
|
||||
// }
|
||||
|
||||
function getSelected(): Members {
|
||||
if (selected.value.length === 0) return [];
|
||||
return selected.value;
|
||||
function addToEvent() {
|
||||
addToEventDialog.value?.open(i18n.global.t('addToEvent'), selected.value);
|
||||
}
|
||||
|
||||
async function addEventAttendees() {
|
||||
await appApi
|
||||
.post('events/add/attendees', {
|
||||
id: inProps.eventId,
|
||||
attendees: [...selected.value],
|
||||
})
|
||||
.then(() => {
|
||||
if (selected.value.length > 1) {
|
||||
NotifyResponse(i18n.global.t('attendeeAdded'));
|
||||
} else {
|
||||
NotifyResponse(i18n.global.t('attendeesAdded'));
|
||||
}
|
||||
})
|
||||
.catch((err) => {
|
||||
NotifyResponse(err, 'error');
|
||||
});
|
||||
|
||||
console.log(56, selected.value);
|
||||
await updateMemberLastVisit(selected.value);
|
||||
|
||||
//update member last last visit
|
||||
console.log(57, selected.value);
|
||||
}
|
||||
|
||||
async function updateMemberLastVisit(members: Members) {
|
||||
const now = new Date();
|
||||
|
||||
const year = now.getFullYear();
|
||||
const month = String(now.getMonth() + 1).padStart(2, '0'); // Months are 0-based
|
||||
const day = String(now.getDate()).padStart(2, '0');
|
||||
const hours = String(now.getHours()).padStart(2, '0');
|
||||
const minutes = String(now.getMinutes()).padStart(2, '0');
|
||||
const seconds = String(now.getSeconds()).padStart(2, '0');
|
||||
|
||||
const dateTimeNow = `${year}-${month}-${day} ${hours}:${minutes}:${seconds}`;
|
||||
members.forEach((mem) => {
|
||||
mem.lastVisit = dateTimeNow;
|
||||
});
|
||||
|
||||
await appApi
|
||||
.post('members/edit', members)
|
||||
.then(() => {
|
||||
if (members.length > 1) {
|
||||
NotifyResponse(i18n.global.t('membersUpdated'));
|
||||
} else {
|
||||
NotifyResponse(i18n.global.t('memberUpdated'));
|
||||
}
|
||||
})
|
||||
.catch((err) => NotifyResponse(err, 'error'));
|
||||
|
||||
emit('update-event');
|
||||
}
|
||||
defineExpose({
|
||||
getSelected,
|
||||
});
|
||||
</script>
|
||||
|
||||
<style>
|
||||
|
||||
@@ -77,11 +77,11 @@ export function useRoleTable() {
|
||||
.then((resp) => {
|
||||
userStore
|
||||
.setUser({ id: resp.data.id, username: resp.data.username, role: resp.data.role })
|
||||
.catch((err) => console.log(err));
|
||||
login.refresh().catch((err) => console.error(err));
|
||||
.catch((err) => NotifyResponse(err, 'error'));
|
||||
login.refresh().catch((err) => NotifyResponse(err, 'error'));
|
||||
})
|
||||
.catch(() => {
|
||||
login.logout().catch((err) => console.error(err));
|
||||
login.logout().catch((err) => NotifyResponse(err, 'error'));
|
||||
});
|
||||
}
|
||||
return {
|
||||
|
||||
@@ -9,7 +9,7 @@
|
||||
:no-data-label="$t('noDataAvailable')"
|
||||
:loading-label="$t('loading')"
|
||||
:rows-per-page-label="$t('recordsPerPage')"
|
||||
:selected-rows-label="(val) => val + $t('recordSelected')"
|
||||
:selected-rows-label="(val) => val + ' ' + $t('recordSelected')"
|
||||
:rows="roles"
|
||||
:columns="columns"
|
||||
row-key="id"
|
||||
@@ -151,7 +151,6 @@ import EditAllDialog from 'src/components/RoleEditAllDialog.vue';
|
||||
import OkDialog from 'src/components/dialog/OkDialog.vue';
|
||||
import { useNotify } from 'src/vueLib/general/useNotify';
|
||||
import { useRoleTable } from './RoleTable';
|
||||
import { useLogin } from 'src/vueLib/login/useLogin';
|
||||
import { i18n } from 'src/boot/lang';
|
||||
import { QTable } from 'quasar';
|
||||
import { useUserStore } from 'src/vueLib/login/userStore';
|
||||
@@ -173,9 +172,8 @@ const { roles, pagination, loading, columns, updateRoles } = useRoleTable();
|
||||
//load on mounting page
|
||||
onMounted(() => {
|
||||
loading.value = true;
|
||||
const login = useLogin();
|
||||
currentUser.value = login.getUser();
|
||||
updateRoles().catch((err) => console.error(err));
|
||||
currentUser.value = user.user;
|
||||
updateRoles().catch((err) => NotifyResponse(err, 'error'));
|
||||
});
|
||||
|
||||
function autorized(role: Role): boolean {
|
||||
@@ -217,12 +215,12 @@ function removeRole(...removeRoles: Roles) {
|
||||
|
||||
appApi
|
||||
.post('roles/delete?role=' + currentUser.value.role, { roles: roles })
|
||||
.then((resp) => {
|
||||
updateRoles().catch((err) => console.error(err));
|
||||
.then(() => {
|
||||
updateRoles().catch((err) => NotifyResponse(err, 'error'));
|
||||
if (roles.length === 1) {
|
||||
NotifyResponse("'" + roles[0] + "' " + i18n.global.t('deleted'), 'warning');
|
||||
} else {
|
||||
NotifyResponse(resp.data, 'warning');
|
||||
NotifyResponse(i18n.global.t('deleteRoles'), 'warning');
|
||||
}
|
||||
selected.value = [];
|
||||
})
|
||||
@@ -231,15 +229,6 @@ function removeRole(...removeRoles: Roles) {
|
||||
loading.value = false;
|
||||
});
|
||||
}
|
||||
|
||||
function getSelected(): Roles {
|
||||
if (selected.value.length === 0) return [];
|
||||
return selected.value;
|
||||
}
|
||||
|
||||
defineExpose({
|
||||
getSelected,
|
||||
});
|
||||
</script>
|
||||
|
||||
<style>
|
||||
|
||||
@@ -9,7 +9,7 @@
|
||||
:no-data-label="$t('noDataAvailable')"
|
||||
:loading-label="$t('loading')"
|
||||
:rows-per-page-label="$t('recordsPerPage')"
|
||||
:selected-rows-label="(val) => val + $t('recordSelected')"
|
||||
:selected-rows-label="(val) => val + ' ' + $t('recordSelected')"
|
||||
:rows="users"
|
||||
:columns="columns"
|
||||
row-key="id"
|
||||
@@ -139,7 +139,6 @@ import EditAllDialog from 'src/components/UserEditAllDialog.vue';
|
||||
import OkDialog from 'src/components/dialog/OkDialog.vue';
|
||||
import { useNotify } from 'src/vueLib/general/useNotify';
|
||||
import { useUserTable } from './UserTable';
|
||||
import { useLogin } from 'src/vueLib/login/useLogin';
|
||||
import { roles, useRoleTable } from '../roles/RoleTable';
|
||||
import { i18n } from 'src/boot/lang';
|
||||
import { useUserStore } from 'src/vueLib/login/userStore';
|
||||
@@ -164,10 +163,9 @@ const user = useUserStore();
|
||||
//load on mounting page
|
||||
onMounted(() => {
|
||||
loading.value = true;
|
||||
const login = useLogin();
|
||||
currentUser.value = login.getUser();
|
||||
currentUser.value = user.user;
|
||||
updateUsers();
|
||||
updateRoles().catch((err) => console.error(err));
|
||||
updateRoles().catch((err) => NotifyResponse(err, 'error'));
|
||||
});
|
||||
|
||||
//check authorization
|
||||
@@ -224,18 +222,9 @@ function removeUser(...removeUsers: Users) {
|
||||
function updateUser(user: User) {
|
||||
appApi
|
||||
.post('/users/update', user)
|
||||
.then((resp) => console.log(100, resp))
|
||||
.catch((err) => console.log(101, err));
|
||||
.then(() => NotifyResponse(i18n.global.t('userUpdated')))
|
||||
.catch((err) => NotifyResponse(err, 'error'));
|
||||
}
|
||||
|
||||
function getSelected(): Users {
|
||||
if (selected.value.length === 0) return [];
|
||||
return selected.value;
|
||||
}
|
||||
|
||||
defineExpose({
|
||||
getSelected,
|
||||
});
|
||||
</script>
|
||||
|
||||
<style>
|
||||
|
||||
Reference in New Issue
Block a user