5 Commits

Author SHA1 Message Date
Adrian Zürcher
39e5479947 fix wrong filter parameter 2025-11-16 08:42:54 +01:00
Adrian Zürcher
5c15fb5599 change padding to userfriendlier size on phone close #23 2025-11-16 08:42:28 +01:00
Adrian Zürcher
dce451bfc2 add new page header and decrease marging 2025-11-16 08:40:42 +01:00
Adrian Zürcher
bba4486136 fix translation 2025-11-16 08:39:31 +01:00
Adrian Zürcher
a18b01450d fix wrong input field to select clos #25 2025-11-16 08:39:17 +01:00
19 changed files with 370 additions and 326 deletions

View File

@@ -86,9 +86,9 @@ members: Mitglider
attendanceTable: Anweseheits Tabelle attendanceTable: Anweseheits Tabelle
excursionTable: Usflugs Tabelle excursionTable: Usflugs Tabelle
updated: aktualisiert updated: aktualisiert
events: Veranstalige events: Verastautige
eventNameIsRequired: Verastatigsname isch erforderlich eventNameIsRequired: Verastautigssname isch erforderlich
eventName: Verastatigsname eventName: Verastautigssname
attendees: Teilnähmer attendees: Teilnähmer
now: Jetzt now: Jetzt
addToEvent: Füge zu Veranstautig addToEvent: Füge zu Veranstautig

View File

@@ -60,12 +60,17 @@
filled filled
v-model="localMember.email" v-model="localMember.email"
></q-input> ></q-input>
<q-input <q-select
class="q-ml-md col-5" class="q-ml-md col-5"
:label="$t('group')" :label="$t('group')"
:options="props.group"
filled filled
emit-value
map-options
option-value="name"
option-label="name"
v-model="localMember.group" v-model="localMember.group"
></q-input> ></q-select>
<q-select <q-select
class="q-ml-md col-5" class="q-ml-md col-5"
:label="$t('responsible')" :label="$t('responsible')"
@@ -118,6 +123,9 @@ const props = defineProps({
responsibles: { responsibles: {
type: Object as PropType<Members>, type: Object as PropType<Members>,
}, },
group: {
type: Array,
},
}); });
const emit = defineEmits(['update']); const emit = defineEmits(['update']);

View File

@@ -1,11 +1,11 @@
<template> <template>
<q-page> <q-page>
<h4 class="text-primary text-bold text-center">{{ $t('events') }}</h4> <SiteTitle :title="$t('events')" />
<EventsTable /> <EventsTable />
</q-page> </q-page>
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import SiteTitle from 'src/vueLib/general/SiteTitle.vue';
import EventsTable from 'src/vueLib/tables/events/EventsTable.vue'; import EventsTable from 'src/vueLib/tables/events/EventsTable.vue';
</script> </script>

View File

@@ -1,10 +1,11 @@
<template> <template>
<q-page> <q-page>
<h4 class="text-primary text-bold text-center">{{ $t('groups') }}</h4> <SiteTitle :title="$t('groups')" />
<GroupTable /> <GroupTable />
</q-page> </q-page>
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import SiteTitle from 'src/vueLib/general/SiteTitle.vue';
import GroupTable from 'src/vueLib/tables/group/GroupTable.vue'; import GroupTable from 'src/vueLib/tables/group/GroupTable.vue';
</script> </script>

View File

@@ -1,10 +1,13 @@
<template> <template>
<q-page> <q-page>
<h4 class="text-primary text-bold text-center">{{ $t('members') }}</h4> <SiteTitle :title="$t('members')" />
<MembersTable /> <div class="q-pa-md">
<MembersTable />
</div>
</q-page> </q-page>
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import SiteTitle from 'src/vueLib/general/SiteTitle.vue';
import MembersTable from 'src/vueLib/tables/members/MembersTable.vue'; import MembersTable from 'src/vueLib/tables/members/MembersTable.vue';
</script> </script>

View File

@@ -1,11 +1,11 @@
<template> <template>
<q-page> <q-page>
<h4 class="text-primary text-bold text-center">{{ $t('responsibles') }}</h4> <SiteTitle :title="$t('responsibles')" />
<ResponsibleTable /> <ResponsibleTable />
</q-page> </q-page>
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import SiteTitle from 'src/vueLib/general/SiteTitle.vue';
import ResponsibleTable from 'src/vueLib/tables/responsible/ResponsibleTable.vue'; import ResponsibleTable from 'src/vueLib/tables/responsible/ResponsibleTable.vue';
</script> </script>

View File

@@ -1,6 +1,5 @@
<template> <template>
<h4 class="text-primary text-bold text-center">{{ $t('userSettings') }}</h4> <SiteTitle :title="$t('userSettings')" />
<div class="text-h2 flex flex-center"> <div class="text-h2 flex flex-center">
<q-card class="q-gutter-md"> <q-card class="q-gutter-md">
<div> <div>
@@ -152,6 +151,7 @@ import { useNotify } from 'src/vueLib/general/useNotify';
import { type Settings } from 'src/vueLib/models/settings'; import { type Settings } from 'src/vueLib/models/settings';
import { useUserStore } from 'src/vueLib/login/userStore'; import { useUserStore } from 'src/vueLib/login/userStore';
import { setLocalSettings } from 'src/localstorage/localStorage'; import { setLocalSettings } from 'src/localstorage/localStorage';
import SiteTitle from 'src/vueLib/general/SiteTitle.vue';
const { NotifyResponse } = useNotify(); const { NotifyResponse } = useNotify();
const colorGroup = ref(false); const colorGroup = ref(false);

View File

@@ -1,6 +1,5 @@
<template> <template>
<h4 class="text-primary text-bold text-center">{{ $t('userSettings') }}</h4> <SiteTitle :title="$t('userSettings')" />
<div class="q-pa-md"> <div class="q-pa-md">
<div class="q-gutter-y-md"> <div class="q-gutter-y-md">
<q-card> <q-card>
@@ -35,6 +34,7 @@
import { ref } from 'vue'; import { ref } from 'vue';
import UserTable from 'src/vueLib/tables/users/UserTable.vue'; import UserTable from 'src/vueLib/tables/users/UserTable.vue';
import RoleTable from 'src/vueLib/tables/roles/RoleTable.vue'; import RoleTable from 'src/vueLib/tables/roles/RoleTable.vue';
import SiteTitle from 'src/vueLib/general/SiteTitle.vue';
const tab = ref('users'); const tab = ref('users');
</script> </script>

View File

@@ -25,12 +25,12 @@ const routes: RouteRecordRaw[] = [
}, },
{ {
path: 'responsible', path: 'responsible',
component: () => import('pages/ResponsibleTable.vue'), component: () => import('src/pages/ResponsibleTable.vue'),
meta: { requiresAuth: true, requiresAdmin: true }, meta: { requiresAuth: true, requiresAdmin: true },
}, },
{ {
path: 'group', path: 'group',
component: () => import('pages/GroupTable.vue'), component: () => import('src/pages/GroupTable.vue'),
meta: { requiresAuth: true, requiresAdmin: true }, meta: { requiresAuth: true, requiresAdmin: true },
}, },
{ {

View File

@@ -25,7 +25,7 @@
<q-separator color="black" /> <q-separator color="black" />
<!-- Content Slot --> <!-- Content Slot -->
<div class="scrollArea"> <div class="scrollArea" :style="'padding: ' + props.innerPadding + 'px'">
<slot /> <slot />
</div> </div>
@@ -48,6 +48,7 @@ const props = defineProps({
headerTitle: { type: String, default: '' }, headerTitle: { type: String, default: '' },
width: { type: Number, default: 400 }, width: { type: Number, default: 400 },
height: { type: Number, default: 250 }, height: { type: Number, default: 250 },
innerPadding: { type: Number, default: 16 },
}); });
// Fullscreen toggle // Fullscreen toggle
@@ -153,7 +154,6 @@ const cardStyle = computed(() => {
flex: 1 1 auto; flex: 1 1 auto;
min-height: 0; min-height: 0;
overflow-y: auto; overflow-y: auto;
padding: 16px;
} }
/* Resize handle in bottom right */ /* Resize handle in bottom right */

View File

@@ -0,0 +1,12 @@
<template>
<h4 class="text-primary text-bold text-center" :style="{ margin: props.marging + 'px' }">
{{ props.title }}
</h4>
</template>
<script setup lang="ts">
const props = defineProps({
title: { type: String },
marging: { type: Number, default: 28 },
});
</script>

View File

@@ -1,97 +1,96 @@
<template> <template>
<div class="q-pa-md"> <q-table
<q-table flat
flat bordered
bordered ref="tableRef"
ref="tableRef" :title="$t('attendees')"
:title="$t('attendees')" title-class="text-bold text-blue-9"
title-class="text-bold text-blue-9" :no-data-label="$t('noDataAvailable')"
:no-data-label="$t('noDataAvailable')" :loading-label="$t('loading')"
:loading-label="$t('loading')" :rows-per-page-label="$t('recordsPerPage')"
:rows-per-page-label="$t('recordsPerPage')" :selected-rows-label="(val) => val + ' ' + $t('recordSelected')"
:selected-rows-label="(val) => val + ' ' + $t('recordSelected')" :rows="attendees"
:rows="attendees" :columns="columns"
:columns="columns" row-key="id"
row-key="id" v-model:pagination="pagination"
v-model:pagination="pagination" :loading="loading"
:loading="loading" :filter="filter"
:filter="filter" :selection="selectOption ? 'multiple' : 'none'"
:selection="selectOption ? 'multiple' : 'none'" v-model:selected="selected"
v-model:selected="selected" binary-state-sort
binary-state-sort dense
class="bigger-table-text" class="bigger-table-text"
> >
<template v-slot:top-left> <template v-slot:top-left>
<q-btn-group push flat style="color: grey"> <q-btn-group push flat style="color: grey">
<q-btn <q-btn
v-if="user.isPermittedTo('events', 'write')" v-if="user.isPermittedTo('events', 'write')"
dense dense
flat flat
icon="person" icon="person"
@click="openAllValueDialog" @click="openAllValueDialog"
><q-badge floating transparent color="primary" text-color="primary-text">+</q-badge> ><q-badge floating transparent color="primary" text-color="primary-text">+</q-badge>
<q-tooltip>{{ $t('addNewAttendees') }}</q-tooltip> <q-tooltip>{{ $t('addNewAttendees') }}</q-tooltip>
</q-btn> </q-btn>
<q-btn <q-btn
v-if="user.isPermittedTo('events', 'write')" v-if="user.isPermittedTo('events', 'write')"
dense dense
flat flat
style="color: grey" style="color: grey"
:icon="selectOption ? 'check_box' : 'check_box_outline_blank'" :icon="selectOption ? 'check_box' : 'check_box_outline_blank'"
@click="selectOption = !selectOption" @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-tooltip>{{ $t('selectAttendeesOptions') }}</q-tooltip> </q-menu>
</q-btn> </div>
</q-btn-group> <div v-if="selectOption && selected.length > 0" class="q-ml-md text-weight-bold">
<div v-if="selectOption && selected.length > 0"> {{ $t('selected') }}: {{ selected.length }}
<q-btn flat dense icon="more_vert" @click="openSubmenu = true" /> </div>
<q-menu v-if="openSubmenu" anchor="bottom middle" self="top middle"> </template>
<q-item <template v-slot:top-right>
clickable <q-input filled dense debounce="300" v-model="filter" :placeholder="$t('search')">
v-close-popup <template v-slot:append>
@click="openRemoveDialog(...selected)" <q-icon name="search" />
class="text-negative" </template>
>{{ $t('delete') }}</q-item </q-input>
> </template>
</q-menu> <template v-slot:body-cell="props">
</div> <q-td v-if="props.col.field === 'attendees'" :props="props">
<div v-if="selectOption && selected.length > 0" class="q-ml-md text-weight-bold"> <q-btn v-if="props.value !== null && props.value.length > 0" dense flat icon="people"
{{ $t('selected') }}: {{ selected.length }} ><q-badge color="primary" text-color="primary-text" floating transparent>{{
</div> props.row.count
</template> }}</q-badge></q-btn
<template v-slot:top-right> >
<q-input filled dense debounce="300" v-model="filter" :placeholder="$t('search')"> </q-td>
<template v-slot:append> <q-td v-else :props="props">
<q-icon name="search" /> {{ props.value }}
</template> </q-td>
</q-input> </template>
</template> <template v-slot:body-cell-option="props">
<template v-slot:body-cell="props"> <q-td :props="props">
<q-td v-if="props.col.field === 'attendees'" :props="props"> <q-btn
<q-btn v-if="props.value !== null && props.value.length > 0" dense flat icon="people" v-if="user.isPermittedTo('events', 'delete')"
><q-badge color="primary" text-color="primary-text" floating transparent>{{ flat
props.row.count dense
}}</q-badge></q-btn @click="openRemoveDialog(props.row)"
> color="negative"
</q-td> icon="delete"
<q-td v-else :props="props"> />
{{ props.value }} </q-td>
</q-td> </template>
</template> </q-table>
<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" :height="500"> <DialogFrame ref="memberTableDialog" :header-title="$t('members')" :width="700" :height="500">
<MembersTable <MembersTable
add-attendees add-attendees

View File

@@ -1,5 +1,11 @@
<template> <template>
<DialogFrame ref="dialog" :header-title="$t('attendees')" :width="700" :height="600"> <DialogFrame
ref="dialog"
:header-title="$t('attendees')"
:width="700"
:height="600"
:inner-padding="0"
>
<q-card> <q-card>
<q-tabs <q-tabs
v-model="tab" v-model="tab"
@@ -17,10 +23,10 @@
<q-separator /> <q-separator />
<q-tab-panels v-model="tab" animated> <q-tab-panels v-model="tab" animated>
<q-tab-panel name="attendance"> <q-tab-panel name="attendance" style="padding: 0px">
<AttendeesTable :event="localEvent!" v-on:update="updateTable" /> <AttendeesTable :event="localEvent!" v-on:update="updateTable" />
</q-tab-panel> </q-tab-panel>
<q-tab-panel name="noneAttendees"> <q-tab-panel name="noneAttendees" style="padding: 0px">
<MembersTable <MembersTable
add-attendees add-attendees
:compare-members="attendees" :compare-members="attendees"

View File

@@ -19,6 +19,7 @@
:selection="selectOption ? 'multiple' : 'none'" :selection="selectOption ? 'multiple' : 'none'"
v-model:selected="selected" v-model:selected="selected"
binary-state-sort binary-state-sort
dense
class="bigger-table-text" class="bigger-table-text"
> >
<template v-slot:top-left> <template v-slot:top-left>

View File

@@ -19,6 +19,7 @@
:selection="selectOption ? 'multiple' : 'none'" :selection="selectOption ? 'multiple' : 'none'"
v-model:selected="selected" v-model:selected="selected"
binary-state-sort binary-state-sort
dense
class="bigger-table-text" class="bigger-table-text"
> >
<template v-slot:top-left> <template v-slot:top-left>

View File

@@ -241,8 +241,9 @@ export function useMemberTable() {
.finally(() => { .finally(() => {
loading.value = false; loading.value = false;
//filter same members out so list is shorter //filter same members out so list is shorter
filteredMembers.value = allMembers.value;
if (filter) { if (filter) {
filteredMembers.value = allMembers.value.filter( filteredMembers.value = filteredMembers.value.filter(
(m1) => (m1) =>
!filter.some((m2) => { !filter.some((m2) => {
if (filterbyName) { if (filterbyName) {
@@ -256,7 +257,7 @@ export function useMemberTable() {
//second filter //second filter
const list = filterList.value ?? []; const list = filterList.value ?? [];
if (filterList.value && filterList.value.length > 0) { if (filterList.value && filterList.value.length > 0) {
filteredMembers.value = allMembers.value.filter((member) => filteredMembers.value = filteredMembers.value.filter((member) =>
list.every((filterItem) => { list.every((filterItem) => {
const keys = filterItem.keys ?? []; const keys = filterItem.keys ?? [];
if (keys.includes('null')) return true; if (keys.includes('null')) return true;

View File

@@ -1,224 +1,233 @@
<template> <template>
<div class="q-pa-md"> <q-table
<q-table flat
flat bordered
bordered ref="tableRef"
ref="tableRef" title="Members"
title="Members" title-class="text-bold text-blue-9"
title-class="text-bold text-blue-9" :no-data-label="$t('noDataAvailable')"
:no-data-label="$t('noDataAvailable')" :loading-label="$t('loading')"
:loading-label="$t('loading')" :rows-per-page-label="$t('recordsPerPage')"
:rows-per-page-label="$t('recordsPerPage')" :selected-rows-label="(val) => val + ' ' + $t('recordSelected')"
:selected-rows-label="(val) => val + ' ' + $t('recordSelected')" :rows="filteredMembers"
:rows="filteredMembers" :columns="columns"
:columns="columns" row-key="id"
row-key="id" v-model:pagination="pagination"
v-model:pagination="pagination" :loading="loading"
:loading="loading" :filter="filter"
:filter="filter" :selection="selectOption ? 'multiple' : 'none'"
:selection="selectOption ? 'multiple' : 'none'" v-model:selected="selected"
v-model:selected="selected" binary-state-sort
binary-state-sort dense
class="bigger-table-text" class="bigger-table-text"
> >
<template v-slot:top-left> <!-- top left of table-->
<div>
<q-btn-group push flat style="color: grey">
<q-btn
v-if="user.isPermittedTo('members', 'write')"
dense
flat
icon="add"
@click="openAllValueDialog(null)"
>
<q-tooltip>{{ $t('addNewMember') }}</q-tooltip>
</q-btn>
<q-btn
v-if="
user.isPermittedTo('members', 'write') || user.isPermittedTo('members', 'delete')
"
dense
flat
style="color: grey"
:icon="selectOption ? 'check_box' : 'check_box_outline_blank'"
@click="selectOption = !selectOption"
>
<q-tooltip>{{ $t('selectMemberOptions') }}</q-tooltip>
</q-btn>
<q-btn
v-if="user.isPermittedTo('members', 'import')"
dense
flat
icon="upload"
@click="openUploadDialog"
>
<q-tooltip>{{ $t('importCSV') }}</q-tooltip>
</q-btn>
<q-btn
v-if="user.isPermittedTo('members', 'export')"
dense
flat
icon="download"
@click="exportCsv"
>
<q-tooltip>{{ $t('exportCSV') }}</q-tooltip>
</q-btn>
</q-btn-group>
<div v-if="selectOption && selected.length > 0">
<q-btn
v-if="inProps.addAttendees || inProps.addResponsible"
dense
color="grey-7"
flat
icon="person"
@click="addMemberTo"
>
<q-badge floating transparent color="primary" text-color="primary-text">+</q-badge>
</q-btn>
<q-btn v-else flat dense icon="more_vert" @click="openSubmenu = true" />
<q-menu v-if="openSubmenu" anchor="bottom middle" self="top middle"> <template v-slot:top-left>
<q-item clickable v-close-popup @click="addToEvent" class="text-primary">{{ <div>
$t('addToEvent') <q-btn-group push flat style="color: grey">
}}</q-item> <q-btn
<q-item v-if="user.isPermittedTo('members', 'write')"
v-if="user.isPermittedTo('members', 'delete')" dense
clickable flat
v-close-popup icon="add"
@click="openRemoveDialog(...selected)" @click="openAllValueDialog(null)"
class="text-negative" >
>{{ $t('delete') }}</q-item <q-tooltip>{{ $t('addNewMember') }}</q-tooltip>
> </q-btn>
</q-menu>
</div>
<q-card flat class="q-pa-sm">
<q-select
:label="$t('filterByColumn')"
dense
v-model="selectedColumnFilter"
option-label="label"
option-value="name"
map-options
emit-value
clearable
:options="columns.filter((col) => col.label !== '')"
v-on:clear="selectedColumnOptions = []"
@update:model-value="
filterMembers(selectedColumnFilter, ...(selectedColumnOptions || []))
"
class="q-mt-xs"
/>
<q-select
v-if="selectedColumnFilter"
:label="$t('filterByColumnValue')"
dense
v-model="selectedColumnOptions"
:options="setColumnOptions(selectedColumnFilter)"
class="q-mt-xs"
multiple
clearable
@update:model-value="
filterMembers(selectedColumnFilter, ...(selectedColumnOptions || []))
"
/>
</q-card>
</div>
<div v-if="selectOption && selected.length > 0" class="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
:props="props"
:class="getRowClass(props.row)"
:style="user.isPermittedTo('members', 'write') ? 'cursor: pointer' : ''"
@click="
user.isPermittedTo('members', 'write') &&
openSingleValueDialog(props.col.label, props.col.name, props.row)
"
>
{{ props.value }}
</q-td>
</template>
<template v-slot:body-cell-cake="props">
<q-td :props="props" :class="getRowClass(props.row)">
<q-icon
:name="isXDaysBeforeAnnualDate(props.row.birthday, 1) ? 'cake' : ''"
:color="'red'"
size="md"
/>
</q-td>
</template>
<template v-slot:body-cell-group="props">
<q-td :props="props">
<q-select
v-if="groups.length > 0"
:readonly="!user.isPermittedTo('members', 'write')"
:options="groups"
emit-value
map-options
option-value="name"
option-label="name"
v-model="props.row.group"
@update:model-value="updateMember(props.row)"
></q-select>
</q-td>
</template>
<template v-slot:body-cell-responsiblePerson="props">
<q-td :props="props">
<q-select
v-if="responsibles.length > 0"
:readonly="!user.isPermittedTo('members', 'write')"
:options="responsibles"
:option-label="(opt) => opt.firstName + ' ' + opt.lastName"
v-model="props.row.responsiblePerson"
@update:model-value="updateMember(props.row)"
></q-select>
</q-td>
</template>
<template v-slot:body-cell-option="props">
<q-td :props="props">
<q-btn <q-btn
v-if="user.isPermittedTo('members', 'write') || user.isPermittedTo('members', 'delete')" v-if="user.isPermittedTo('members', 'write') || user.isPermittedTo('members', 'delete')"
flat
dense dense
icon="more_vert" flat
@click="openSubmenu = true" style="color: grey"
/> :icon="selectOption ? 'check_box' : 'check_box_outline_blank'"
<q-menu v-if="openSubmenu" anchor="top right" self="top left"> @click="selectOption = !selectOption"
>
<q-tooltip>{{ $t('selectMemberOptions') }}</q-tooltip>
</q-btn>
<q-btn
v-if="user.isPermittedTo('members', 'import')"
dense
flat
icon="upload"
@click="openUploadDialog"
>
<q-tooltip>{{ $t('importCSV') }}</q-tooltip>
</q-btn>
<q-btn
v-if="user.isPermittedTo('members', 'export')"
dense
flat
icon="download"
@click="exportCsv"
>
<q-tooltip>{{ $t('exportCSV') }}</q-tooltip>
</q-btn>
</q-btn-group>
<div v-if="selectOption && selected.length > 0">
<q-btn
v-if="inProps.addAttendees || inProps.addResponsible"
dense
color="grey-7"
flat
icon="person"
@click="addMemberTo"
>
<q-badge floating transparent color="primary" text-color="primary-text">+</q-badge>
</q-btn>
<q-btn v-else 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="addToEvent" class="text-primary">{{
$t('addToEvent')
}}</q-item>
<q-item <q-item
v-if="user.isPermittedTo('members', 'write')" v-if="user.isPermittedTo('members', 'delete')"
clickable clickable
v-close-popup v-close-popup
@click="openAllValueDialog(props.row)" @click="openRemoveDialog(...selected)"
class="text-primary"
>{{ $t('edit') }}</q-item
>
<q-item
v-if="
user.isPermittedTo('members', 'delete') &&
!inProps.addAttendees &&
!inProps.addResponsible
"
clickable
v-close-popup
@click="openRemoveDialog(props.row)"
class="text-negative" class="text-negative"
title="zu"
>{{ $t('delete') }}</q-item >{{ $t('delete') }}</q-item
> >
</q-menu> </q-menu>
</q-td> </div>
</template> <q-card flat class="q-pa-sm">
</q-table> <q-select
</div> :label="$t('filterByColumn')"
dense
v-model="selectedColumnFilter"
option-label="label"
option-value="name"
map-options
emit-value
clearable
:options="columns.filter((col) => col.label !== '')"
v-on:clear="selectedColumnOptions = []"
@update:model-value="
filterMembers(selectedColumnFilter, ...(selectedColumnOptions || []))
"
class="q-mt-xs"
/>
<q-select
v-if="selectedColumnFilter"
:label="$t('filterByColumnValue')"
dense
v-model="selectedColumnOptions"
:options="setColumnOptions(selectedColumnFilter)"
class="q-mt-xs"
multiple
clearable
@update:model-value="
filterMembers(selectedColumnFilter, ...(selectedColumnOptions || []))
"
/>
</q-card>
</div>
<div v-if="selectOption && selected.length > 0" class="text-weight-bold">
{{ $t('selected') }}: {{ selected.length }}
</div>
</template>
<!-- top right of table-->
<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>
<!-- table body content-->
<template v-slot:body-cell="props">
<q-td
:props="props"
:class="getRowClass(props.row)"
:style="user.isPermittedTo('members', 'write') ? 'cursor: pointer' : ''"
@click="
user.isPermittedTo('members', 'write') &&
openSingleValueDialog(props.col.label, props.col.name, props.row)
"
>
{{ props.value }}
</q-td>
</template>
<template v-slot:body-cell-cake="props">
<q-td :props="props" :class="getRowClass(props.row)">
<q-icon
:name="isXDaysBeforeAnnualDate(props.row.birthday, 1) ? 'cake' : ''"
:color="'red'"
size="md"
/>
</q-td>
</template>
<template v-slot:body-cell-group="props">
<q-td :props="props">
<q-select
v-if="groups.length > 0"
dense
:readonly="!user.isPermittedTo('members', 'write')"
:options="groups"
emit-value
map-options
option-value="name"
option-label="name"
v-model="props.row.group"
@update:model-value="updateMember(props.row)"
></q-select>
</q-td>
</template>
<template v-slot:body-cell-responsiblePerson="props">
<q-td :props="props">
<q-select
dense
v-if="responsibles.length > 0"
:readonly="!user.isPermittedTo('members', 'write')"
:options="responsibles"
:option-label="(opt) => opt.firstName + ' ' + opt.lastName"
v-model="props.row.responsiblePerson"
@update:model-value="updateMember(props.row)"
></q-select>
</q-td>
</template>
<template v-slot:body-cell-option="props">
<q-td :props="props">
<q-btn
v-if="user.isPermittedTo('members', 'write') || user.isPermittedTo('members', 'delete')"
flat
dense
icon="more_vert"
@click="openSubmenu = true"
/>
<q-menu v-if="openSubmenu" anchor="top right" self="top left">
<q-item
dense
v-if="user.isPermittedTo('members', 'write')"
clickable
v-close-popup
@click="openAllValueDialog(props.row)"
class="text-primary"
>{{ $t('edit') }}</q-item
>
<q-item
v-if="
user.isPermittedTo('members', 'delete') &&
!inProps.addAttendees &&
!inProps.addResponsible
"
dense
clickable
v-close-popup
@click="openRemoveDialog(props.row)"
class="text-negative"
title="zu"
>{{ $t('delete') }}</q-item
>
</q-menu>
</q-td>
</template>
</q-table>
<EditOneDialog <EditOneDialog
ref="editOneDialog" ref="editOneDialog"
endpoint="members/edit" endpoint="members/edit"
@@ -228,6 +237,7 @@
<EditAllDialog <EditAllDialog
ref="editAllDialog" ref="editAllDialog"
:responsibles="responsibles" :responsibles="responsibles"
:group="groups"
v-on:update="updateMember" v-on:update="updateMember"
></EditAllDialog> ></EditAllDialog>
<OkDialog <OkDialog

View File

@@ -19,6 +19,7 @@
:selection="selectOption ? 'multiple' : 'none'" :selection="selectOption ? 'multiple' : 'none'"
v-model:selected="selected" v-model:selected="selected"
binary-state-sort binary-state-sort
dense
class="bigger-table-text" class="bigger-table-text"
> >
<template v-slot:top-left> <template v-slot:top-left>

View File

@@ -19,6 +19,7 @@
:selection="selectOption ? 'multiple' : 'none'" :selection="selectOption ? 'multiple' : 'none'"
v-model:selected="selected" v-model:selected="selected"
binary-state-sort binary-state-sort
dense
class="bigger-table-text" class="bigger-table-text"
> >
<template v-slot:top-left> <template v-slot:top-left>