release v2.0.0 #4
			
				
			
		
		
		
	|  | @ -313,7 +313,7 @@ async function getBackend() { | ||||||
|  */ |  */ | ||||||
| export default boot(async ({ router, app }) => { | export default boot(async ({ router, app }) => { | ||||||
|   const backend = await getBackend(); |   const backend = await getBackend(); | ||||||
|   if (!backend) { |   if (backend === null) { | ||||||
|     void router.push({ name: 'error' }); |     void router.push({ name: 'error' }); | ||||||
|     return; |     return; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  | @ -0,0 +1,45 @@ | ||||||
|  | <template> | ||||||
|  |   <q-card bordered class="row q-ma-xs q-pa-xs" style="position: relative"> | ||||||
|  |     <div class="col-12 text-weight-light">{{ dateString }}</div> | ||||||
|  |     <div class="col-12">{{ modelValue.text }}</div> | ||||||
|  |     <q-btn | ||||||
|  |       round | ||||||
|  |       dense | ||||||
|  |       icon="close" | ||||||
|  |       size="sm" | ||||||
|  |       color="negative" | ||||||
|  |       class="q-ma-xs" | ||||||
|  |       style="position: absolute; top: 0; right: 0" | ||||||
|  |       @click="remove" | ||||||
|  |     /> | ||||||
|  |   </q-card> | ||||||
|  | </template> | ||||||
|  | 
 | ||||||
|  | <script lang="ts"> | ||||||
|  | import { defineComponent, PropType, computed } from 'vue'; | ||||||
|  | import { formatDateTime } from 'src/utils/datetime'; | ||||||
|  | 
 | ||||||
|  | export default defineComponent({ | ||||||
|  |   name: 'Notification', | ||||||
|  |   props: { | ||||||
|  |     modelValue: { | ||||||
|  |       required: true, | ||||||
|  |       type: Object as PropType<FG.Notification>, | ||||||
|  |     }, | ||||||
|  |   }, | ||||||
|  |   emits: { | ||||||
|  |     remove: (id: number) => !!id, | ||||||
|  |   }, | ||||||
|  |   setup(props, { emit }) { | ||||||
|  |     const dateString = computed(() => formatDateTime(props.modelValue.time, true, true)); | ||||||
|  | 
 | ||||||
|  |     function remove() { | ||||||
|  |       emit('remove', props.modelValue.id); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return { dateString, remove }; | ||||||
|  |   }, | ||||||
|  | }); | ||||||
|  | </script> | ||||||
|  | 
 | ||||||
|  | <style scoped></style> | ||||||
|  | @ -1,11 +1,10 @@ | ||||||
| declare namespace FG { | declare namespace FG { | ||||||
|   interface Session { |   interface Notification { | ||||||
|     expires: Date; |     id: number; | ||||||
|     token: string; |     plugin: string; | ||||||
|     lifetime: number; |     text: string; | ||||||
|     browser: string; |     data?: unknown; | ||||||
|     platform: string; |     time: Date; | ||||||
|     userid: string; |  | ||||||
|   } |   } | ||||||
|   interface User { |   interface User { | ||||||
|     userid: string; |     userid: string; | ||||||
|  | @ -18,6 +17,14 @@ declare namespace FG { | ||||||
|     permissions?: Array<string>; |     permissions?: Array<string>; | ||||||
|     avatar_url?: string; |     avatar_url?: string; | ||||||
|   } |   } | ||||||
|  |   interface Session { | ||||||
|  |     expires: Date; | ||||||
|  |     token: string; | ||||||
|  |     lifetime: number; | ||||||
|  |     browser: string; | ||||||
|  |     platform: string; | ||||||
|  |     userid: string; | ||||||
|  |   } | ||||||
|   type Permission = string; |   type Permission = string; | ||||||
|   interface Role { |   interface Role { | ||||||
|     id: number; |     id: number; | ||||||
|  | @ -69,6 +76,7 @@ declare namespace FG { | ||||||
|   } |   } | ||||||
|   interface Service { |   interface Service { | ||||||
|     userid: string; |     userid: string; | ||||||
|  |     is_backup: boolean; | ||||||
|     value: number; |     value: number; | ||||||
|   } |   } | ||||||
|   interface Drink { |   interface Drink { | ||||||
|  |  | ||||||
|  | @ -21,6 +21,25 @@ | ||||||
|         </q-toolbar-title> |         </q-toolbar-title> | ||||||
| 
 | 
 | ||||||
|         <!-- Hier kommen die Shortlinks hin --> |         <!-- Hier kommen die Shortlinks hin --> | ||||||
|  |         <q-btn icon="message" flat dense | ||||||
|  |           ><q-badge color="negative" floating>{{ notifications.length }}</q-badge> | ||||||
|  |           <q-menu style="max-height: 400px; overflow: auto"> | ||||||
|  |             <q-btn | ||||||
|  |               v-if="noPermission" | ||||||
|  |               label="Benachrichtigungen erlauben" | ||||||
|  |               @click="requestPermission" | ||||||
|  |             /> | ||||||
|  |             <template v-if="notifications.length > 0"> | ||||||
|  |               <Notification | ||||||
|  |                 v-for="(notification, index) in notifications" | ||||||
|  |                 :key="index" | ||||||
|  |                 :model-value="notification" | ||||||
|  |                 @remove="remove" | ||||||
|  |               /> | ||||||
|  |             </template> | ||||||
|  |             <div v-else class="q-pa-sm">Keine neuen Benachrichtigungen</div> | ||||||
|  |           </q-menu> | ||||||
|  |         </q-btn> | ||||||
|         <shortcut-link |         <shortcut-link | ||||||
|           v-for="(shortcut, index) in shortcuts" |           v-for="(shortcut, index) in shortcuts" | ||||||
|           :key="'shortcut' + index" |           :key="'shortcut' + index" | ||||||
|  | @ -84,8 +103,9 @@ | ||||||
| <script lang="ts"> | <script lang="ts"> | ||||||
| import EssentialLink from 'src/components/navigation/EssentialLink.vue'; | import EssentialLink from 'src/components/navigation/EssentialLink.vue'; | ||||||
| import ShortcutLink from 'src/components/navigation/ShortcutLink.vue'; | import ShortcutLink from 'src/components/navigation/ShortcutLink.vue'; | ||||||
|  | import Notification from 'src/components/Notification.vue'; | ||||||
| import { Screen } from 'quasar'; | import { Screen } from 'quasar'; | ||||||
| import { defineComponent, ref, inject, computed } from 'vue'; | import { defineComponent, ref, inject, computed, onBeforeMount } from 'vue'; | ||||||
| import { useMainStore } from 'src/store'; | import { useMainStore } from 'src/store'; | ||||||
| import { FG_Plugin } from 'src/plugins'; | import { FG_Plugin } from 'src/plugins'; | ||||||
| import { useRouter } from 'vue-router'; | import { useRouter } from 'vue-router'; | ||||||
|  | @ -100,7 +120,7 @@ const essentials: FG_Plugin.MenuLink[] = [ | ||||||
| 
 | 
 | ||||||
| export default defineComponent({ | export default defineComponent({ | ||||||
|   name: 'MainLayout', |   name: 'MainLayout', | ||||||
|   components: { EssentialLink, ShortcutLink }, |   components: { EssentialLink, ShortcutLink, Notification }, | ||||||
|   setup() { |   setup() { | ||||||
|     const router = useRouter(); |     const router = useRouter(); | ||||||
|     const mainStore = useMainStore(); |     const mainStore = useMainStore(); | ||||||
|  | @ -108,6 +128,16 @@ export default defineComponent({ | ||||||
|     const leftDrawer = ref(false); |     const leftDrawer = ref(false); | ||||||
|     const shortcuts = flaschengeist?.shortcuts; |     const shortcuts = flaschengeist?.shortcuts; | ||||||
|     const mainLinks = flaschengeist?.menuLinks; |     const mainLinks = flaschengeist?.menuLinks; | ||||||
|  |     const notifications = computed(() => mainStore.notifications.slice().reverse()); | ||||||
|  |     const noPermission = ref(window.Notification.permission !== 'granted'); | ||||||
|  | 
 | ||||||
|  |     function requestPermission() { | ||||||
|  |       void window.Notification.requestPermission().then( | ||||||
|  |         (p) => (noPermission.value = p !== 'granted') | ||||||
|  |       ); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     onBeforeMount(() => window.setInterval(() => void mainStore.loadNotifications(), 30000)); | ||||||
| 
 | 
 | ||||||
|     const leftDrawerOpen = computed({ |     const leftDrawerOpen = computed({ | ||||||
|       get: () => (leftDrawer.value || Screen.gt.sm ? true : false), |       get: () => (leftDrawer.value || Screen.gt.sm ? true : false), | ||||||
|  | @ -131,6 +161,10 @@ export default defineComponent({ | ||||||
|       void mainStore.logout(); |       void mainStore.logout(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     async function remove(id: number) { | ||||||
|  |       await mainStore.removeNotification(id); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     return { |     return { | ||||||
|       essentials, |       essentials, | ||||||
|       leftDrawerOpen, |       leftDrawerOpen, | ||||||
|  | @ -138,6 +172,10 @@ export default defineComponent({ | ||||||
|       leftDrawerClicker, |       leftDrawerClicker, | ||||||
|       logout, |       logout, | ||||||
|       mainLinks, |       mainLinks, | ||||||
|  |       notifications, | ||||||
|  |       noPermission, | ||||||
|  |       remove, | ||||||
|  |       requestPermission, | ||||||
|       shortcuts, |       shortcuts, | ||||||
|       subLinks, |       subLinks, | ||||||
|     }; |     }; | ||||||
|  |  | ||||||
|  | @ -84,6 +84,7 @@ export default defineComponent({ | ||||||
|     async function enrollForJob() { |     async function enrollForJob() { | ||||||
|       const newService: FG.Service = { |       const newService: FG.Service = { | ||||||
|         userid: mainStore.currentUser.userid, |         userid: mainStore.currentUser.userid, | ||||||
|  |         is_backup: false, | ||||||
|         value: 1, |         value: 1, | ||||||
|       }; |       }; | ||||||
|       try { |       try { | ||||||
|  | @ -104,6 +105,7 @@ export default defineComponent({ | ||||||
|     async function signOutFromJob() { |     async function signOutFromJob() { | ||||||
|       const newService: FG.Service = { |       const newService: FG.Service = { | ||||||
|         userid: mainStore.currentUser.userid, |         userid: mainStore.currentUser.userid, | ||||||
|  |         is_backup: false, | ||||||
|         value: -1, |         value: -1, | ||||||
|       }; |       }; | ||||||
|       try { |       try { | ||||||
|  |  | ||||||
|  | @ -41,6 +41,7 @@ export const useMainStore = defineStore({ | ||||||
|   state: () => ({ |   state: () => ({ | ||||||
|     session: loadCurrentSession(), |     session: loadCurrentSession(), | ||||||
|     user: loadUser(), |     user: loadUser(), | ||||||
|  |     notifications: [] as Array<FG.Notification>, | ||||||
|   }), |   }), | ||||||
| 
 | 
 | ||||||
|   getters: { |   getters: { | ||||||
|  | @ -123,5 +124,33 @@ export const useMainStore = defineStore({ | ||||||
|           response && 'status' in response ? (<AxiosResponse>response).status : false |           response && 'status' in response ? (<AxiosResponse>response).status : false | ||||||
|         ); |         ); | ||||||
|     }, |     }, | ||||||
|  | 
 | ||||||
|  |     async loadNotifications() { | ||||||
|  |       const params = | ||||||
|  |         this.notifications.length > 0 | ||||||
|  |           ? { from: this.notifications[this.notifications.length - 1].time } | ||||||
|  |           : {}; | ||||||
|  |       const { data } = await api.get<FG.Notification[]>('/notifications', { params: params }); | ||||||
|  |       data.forEach((n) => { | ||||||
|  |         n.time = new Date(n.time); | ||||||
|  |         if (window.Notification.permission === 'granted') | ||||||
|  |           new window.Notification(n.text, { | ||||||
|  |             timestamp: n.time.getTime(), | ||||||
|  |           }); | ||||||
|  |       }); | ||||||
|  |       this.notifications.push(...data); | ||||||
|  |     }, | ||||||
|  | 
 | ||||||
|  |     async removeNotification(id: number) { | ||||||
|  |       const idx = this.notifications.findIndex((n) => n.id === id); | ||||||
|  |       if (idx >= 0) | ||||||
|  |         try { | ||||||
|  |           this.notifications.splice(idx, 1); | ||||||
|  |           await api.delete(`/notifications/${id}`); | ||||||
|  |         } catch (error) { | ||||||
|  |           if (this.notifications.length > idx) | ||||||
|  |             this.notifications.splice(idx, this.notifications.length - idx - 1); | ||||||
|  |         } | ||||||
|  |     }, | ||||||
|   }, |   }, | ||||||
| }); | }); | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue