<template> <q-card bordered> <q-dialog :model-value="dialog"> <q-card style="min-width: 320px"> <q-card-section class="row items-center q-pb-none"> <div class="text-h6"> {{ userDisplay(service.userid) }} {{ asDate(modelValue.start) }} </div> <q-space /> <q-btn icon="mdi-close" flat round dense @click="dialog = false" /> </q-card-section> <q-card-section> {{ modelValue.type.name }} {{ asHour(modelValue.start) }}{{ modelValue.end ? ` - ${asHour(modelValue.end)}` : '' }} Uhr </q-card-section> <q-card-actions align="around"> <q-btn style="width: 47.5%" color="primary" label="Eintragen" @click="enroll()" /> <q-toggle v-model="service.is_backup" style="width: 47.5%" color="primary" label="Als Backup" /> </q-card-actions> <q-card-actions v-if="!enrolled(service.userid)" align="around"> <q-btn v-if="isEnrolled" style="width: 47.5%" color="negative" label="Übertragen" @click="enroll(true)" /> <q-btn v-if="!iam(service.userid)" style="width: 47.5%" color="secondary" label="Einladen" @click="enroll(false, true)" /> </q-card-actions> </q-card> </q-dialog> <div class="text-weight-medium q-px-xs"> {{ asHour(modelValue.start) }} <template v-if="modelValue.end">- {{ asHour(modelValue.end) }}</template> </div> <div class="q-px-xs"> {{ modelValue.type.name }} </div> <div class="col-auto q-px-xs" style="font-size: 10px"> {{ modelValue.comment }} </div> <div> <q-select :model-value="modelValue.services" :options="options" :option-label="(v) => userDisplay(v.userid)" filled multiple stack-label label="Dienste" class="col-auto q-px-xs" style="font-size: 6px" counter :max-values="modelValue.required_services" @add="enrollDialog" > <template #selected-item="scope"> <q-chip :removable="canEdit(scope.opt.userid)" dense :tabindex="scope.tabindex" color="white" :text-color="scope.opt.is_backup ? 'primary' : 'secondary'" class="q-ma-none" @remove="remove(scope.opt, scope.removeAtIndex, scope.index)" > <q-avatar :color="scope.opt.is_backup ? 'primary' : 'secondary'" text-color="white"> <img :src="userAvatar(scope.opt.userid)" onerror="this.src=' ';" /> </q-avatar> {{ userDisplay(scope.opt.userid) }} </q-chip> </template> </q-select> </div> </q-card> </template> <script lang="ts"> import { defineComponent, onBeforeMount, computed, ref, PropType } from 'vue'; import { Notify } from 'quasar'; import { asHour, asDate } from 'src/utils/datetime'; import { useUserStore } from 'src/plugins/user/store'; import { useMainStore } from 'src/stores'; import { useScheduleStore } from 'src/plugins/events/store'; import { PERMISSIONS } from 'src/plugins/events/permissions'; import { hasPermission } from 'src/utils/permission'; export default defineComponent({ name: 'JobSlot', props: { modelValue: { required: true, type: Object as PropType<FG.Job>, }, eventId: { required: true, type: Number, }, }, emits: { 'update:modelValue': (v: FG.Job) => !!v }, setup(props, { emit }) { const store = useScheduleStore(); const mainStore = useMainStore(); const userStore = useUserStore(); const availableUsers = null; const dialog = ref(false); const service = ref<FG.Service>(); onBeforeMount(async () => userStore.getUsers()); function userDisplay(uid: string) { return userStore.findUser(uid)?.display_name || uid; } function userAvatar(uid: string) { return userStore.findUser(uid)?.avatar_url; } function enrollDialog(details: { index: number; value: FG.User }) { service.value = { userid: details.value.userid, is_backup: false, value: 1, }; dialog.value = true; } function iam(uid: string) { return uid === mainStore.currentUser.userid; } async function enroll(transfer = false, invite = false) { try { dialog.value = false; if (transfer) service.value = Object.assign({}, service.value, { replace: mainStore.currentUser.userid, }); // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access else if (invite) delete (<any>service.value).value; const job = await store.updateJob(props.eventId, props.modelValue.id, { user: <FG.Service>service.value, }); emit('update:modelValue', job); } catch (error) { console.warn(error); Notify.create({ group: false, type: 'negative', message: 'Fehler beim Eintragen als Dienst', timeout: 10000, progress: true, actions: [{ icon: 'mdi-close', color: 'white' }], }); } } async function remove(service: FG.Service, rem: (i: number) => void, index: number) { if ( service.userid === mainStore.currentUser.userid || hasPermission(PERMISSIONS.ASSIGN_OTHER) ) { rem(index); try { const job = await store.updateJob(props.eventId, props.modelValue.id, { user: Object.assign({}, service, { value: -service.value }), }); emit('update:modelValue', job); } catch (error) { console.warn(error); Notify.create({ group: false, type: 'negative', message: 'Fehler beim Austragen als Dienst', timeout: 10000, progress: true, actions: [{ icon: 'mdi-close', color: 'white' }], }); } } } const options = computed(() => hasPermission(PERMISSIONS.ASSIGN_OTHER) ? userStore.users.filter((v) => !enrolled(v.userid)) : !isEnrolled.value ? [mainStore.currentUser] : [] ); function canEdit(uid: string) { return uid === mainStore.currentUser.userid || hasPermission(PERMISSIONS.ASSIGN_OTHER); } function enrolled(userid: string) { return props.modelValue.services.findIndex((service) => service.userid == userid) !== -1; } const isEnrolled = computed(() => enrolled(mainStore.currentUser.userid)); const canEnroll = computed(() => { const is = isEnrolled.value; let sum = 0; props.modelValue.services.forEach((s) => (sum += s.value)); return sum < props.modelValue.required_services && !is; }); return { enroll, availableUsers, canEdit, isEnrolled, enrolled, enrollDialog, canEnroll, remove, userAvatar, userDisplay, options, iam, dialog, service, asHour, asDate, }; }, }); </script> <style scoped></style>