mirror of
https://github.com/HangarMC/Hangar.git
synced 2024-12-09 06:32:43 +08:00
149 lines
5.5 KiB
Vue
149 lines
5.5 KiB
Vue
<template>
|
|
<v-row>
|
|
<v-col v-if="!$fetchState.pending" cols="12" md="7">
|
|
<h1>{{ $t('notifications.title') }}</h1>
|
|
<v-select v-model="filters.notification" :items="filter"></v-select>
|
|
<v-btn v-if="filteredNotifications.length && filters.notification === 'unread'">{{ $t('notifications.readAll') }}</v-btn>
|
|
<v-list v-if="filteredNotifications.length">
|
|
<v-list-item v-for="notification in filteredNotifications" :key="notification.id">
|
|
<v-list-item-title>
|
|
{{ $t(notification.message[0], notification.message.slice(1)) }}
|
|
</v-list-item-title>
|
|
<v-list-item-action v-if="!notification.read">
|
|
<v-icon @click="markNotificationRead(notification)">mdi-check</v-icon>
|
|
</v-list-item-action>
|
|
</v-list-item>
|
|
</v-list>
|
|
<div v-else>
|
|
{{ $t(`notifications.empty.${filters.notification}`) }}
|
|
</div>
|
|
</v-col>
|
|
<v-col v-if="!$fetchState.pending" cols="12" md="5">
|
|
<h1>{{ $t('notifications.invites') }}</h1>
|
|
<v-select v-model="filters.invite" :items="inviteFilter"></v-select>
|
|
<v-list v-if="filteredInvites.length">
|
|
<v-list-item v-for="(invite, index) in filteredInvites" :key="index">
|
|
<v-list-item-title>
|
|
{{ $t('notifications.invited', [invite.name]) }}
|
|
</v-list-item-title>
|
|
<v-list-item-action>
|
|
<v-btn icon class="success" light @click="updateInvite(invite, 'accept')">
|
|
<v-icon>mdi-check</v-icon>
|
|
</v-btn>
|
|
</v-list-item-action>
|
|
<v-list-item-action>
|
|
<v-btn icon color="error" @click="updateInvite(invite, 'decline')">
|
|
<v-icon>mdi-close</v-icon>
|
|
</v-btn>
|
|
</v-list-item-action>
|
|
</v-list-item>
|
|
</v-list>
|
|
<div v-else>
|
|
{{ $t('notifications.empty.invites') }}
|
|
</div>
|
|
</v-col>
|
|
</v-row>
|
|
</template>
|
|
|
|
<script lang="ts">
|
|
import remove from 'lodash-es/remove';
|
|
import { Component, Vue } from 'nuxt-property-decorator';
|
|
import { HangarNotification, Invite, Invites } from 'hangar-internal';
|
|
import { LoggedIn } from '~/utils/perms';
|
|
// TODO implement NotificationsPage
|
|
|
|
@Component({
|
|
head: {
|
|
title: 'Notifications',
|
|
},
|
|
})
|
|
@LoggedIn
|
|
export default class NotificationsPage extends Vue {
|
|
notifications: HangarNotification[] = [];
|
|
invites = {} as Invites;
|
|
filters = {
|
|
notification: 'unread' as 'unread' | 'read' | 'all',
|
|
invite: 'all' as 'organizations' | 'projects' | 'all',
|
|
};
|
|
|
|
get filteredNotifications(): HangarNotification[] {
|
|
switch (this.filters.notification) {
|
|
case 'unread':
|
|
return this.notifications.filter((n) => !n.read);
|
|
case 'read':
|
|
return this.notifications.filter((n) => n.read);
|
|
case 'all':
|
|
return this.notifications;
|
|
}
|
|
}
|
|
|
|
get filteredInvites(): Invite[] {
|
|
switch (this.filters.invite) {
|
|
case 'projects':
|
|
return this.invites.projects;
|
|
case 'organizations':
|
|
return this.invites.organizations;
|
|
case 'all':
|
|
return [...this.invites.projects, ...this.invites.organizations];
|
|
}
|
|
}
|
|
|
|
get filter() {
|
|
return [
|
|
{ text: this.$t('notifications.unread'), value: 'unread' },
|
|
{ text: this.$t('notifications.read'), value: 'read' },
|
|
{ text: this.$t('notifications.all'), value: 'all' },
|
|
];
|
|
}
|
|
|
|
get inviteFilter() {
|
|
return [
|
|
{ text: this.$t('notifications.invite.organizations'), value: 'organizations' },
|
|
{ text: this.$t('notifications.invite.projects'), value: 'projects' },
|
|
{ text: this.$t('notifications.invite.all'), value: 'all' },
|
|
];
|
|
}
|
|
|
|
markNotificationRead(notification: HangarNotification) {
|
|
this.$api
|
|
.requestInternal(`notifications/${notification.id}`, true, 'post')
|
|
.then(() => {
|
|
remove(this.notifications, (n: HangarNotification) => n.id === notification.id);
|
|
if (notification.action) {
|
|
this.$router.push(notification.action);
|
|
}
|
|
})
|
|
.catch(() => {
|
|
// TODO error popup
|
|
console.error('Could not mark as read');
|
|
});
|
|
}
|
|
|
|
updateInvite(invite: Invite, status: 'accept' | 'decline' | 'unaccept') {
|
|
this.$api
|
|
.requestInternal(`invites/${invite.type}/${invite.roleTableId}/${status}`)
|
|
.then(() => {
|
|
console.log('successful');
|
|
// TODO actions on save
|
|
})
|
|
.catch(() => {
|
|
// TODO error popup
|
|
});
|
|
}
|
|
|
|
async fetch() {
|
|
this.notifications = await this.$api.requestInternal<HangarNotification[]>('notifications');
|
|
this.invites = await this.$api.requestInternal<Invites>('invites');
|
|
// TODO remove this test article
|
|
this.invites.projects.push({
|
|
name: 'TestProject',
|
|
roleTableId: 4,
|
|
type: 'project',
|
|
url: '/Machine_Maker/TestProject',
|
|
} as Invite);
|
|
}
|
|
}
|
|
</script>
|
|
|
|
<style lang="scss" scoped></style>
|