mirror of
https://github.com/mealie-recipes/mealie.git
synced 2026-04-11 23:45:36 -04:00
137 lines
3.8 KiB
Vue
137 lines
3.8 KiB
Vue
<template>
|
|
<BaseDialog
|
|
v-if="currentAnnouncement"
|
|
v-model="dialog"
|
|
:title="$t('announcements.announcements')"
|
|
:icon="$globals.icons.bullhornVariant"
|
|
:cancel-text="$t('general.done')"
|
|
width="100%"
|
|
max-width="1200"
|
|
>
|
|
<div class="d-flex">
|
|
<!-- Nav list -->
|
|
<v-list
|
|
nav
|
|
density="compact"
|
|
color="primary"
|
|
class="overflow-y-auto border-e flex-shrink-0"
|
|
style="width: 200px; max-height: 60vh"
|
|
>
|
|
<v-list-item
|
|
v-for="announcement in allAnnouncements.toReversed()"
|
|
:key="announcement.key"
|
|
:active="currentAnnouncement.key === announcement.key"
|
|
rounded
|
|
@click="currentAnnouncement = announcement"
|
|
>
|
|
<v-list-item-title class="text-body-2">
|
|
{{ announcement.meta?.title }}
|
|
</v-list-item-title>
|
|
<v-list-item-subtitle v-if="announcement.date">
|
|
{{ $d(announcement.date) }}
|
|
</v-list-item-subtitle>
|
|
</v-list-item>
|
|
</v-list>
|
|
|
|
<!-- Main content -->
|
|
<div class="flex-grow-1 overflow-y-auto" style="max-height: 60vh">
|
|
<v-card-title>
|
|
<v-chip v-if="currentAnnouncement.date" label large class="me-1">
|
|
<v-icon class="me-1">
|
|
{{ $globals.icons.calendar }}
|
|
</v-icon>
|
|
{{ $d(currentAnnouncement.date) }}
|
|
</v-chip>
|
|
{{ currentAnnouncement.meta?.title }}
|
|
</v-card-title>
|
|
<v-card-text>
|
|
<component :is="currentAnnouncement.component" />
|
|
</v-card-text>
|
|
</div>
|
|
</div>
|
|
<template #custom-card-action>
|
|
<div v-if="newAnnouncements.length">
|
|
<BaseButton
|
|
color="success"
|
|
:icon="$globals.icons.textBoxCheckOutline"
|
|
:text="$t('announcements.mark-all-as-read')"
|
|
class="mx-4"
|
|
@click="markAllAsRead"
|
|
/>
|
|
<BaseButton
|
|
color="info"
|
|
:icon="$globals.icons.arrowRightBold"
|
|
icon-right
|
|
:text="$t('general.next')"
|
|
@click="nextAnnouncement"
|
|
/>
|
|
</div>
|
|
</template>
|
|
</BaseDialog>
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
import { useAnnouncements } from "~/composables/use-announcements";
|
|
import type { Announcement } from "~/composables/use-announcements";
|
|
import { useUserApi } from "~/composables/api";
|
|
|
|
const dialog = defineModel<boolean>({ default: false });
|
|
|
|
const auth = useMealieAuth();
|
|
const api = useUserApi();
|
|
const { newAnnouncements, allAnnouncements } = useAnnouncements();
|
|
|
|
const currentAnnouncement = shallowRef<Announcement | undefined>();
|
|
watch(
|
|
dialog,
|
|
() => {
|
|
// Once the dialog is opened, show the next announcement
|
|
if (dialog.value) {
|
|
nextAnnouncement();
|
|
|
|
// If there are no new announcements, this is never set, so show the newest one
|
|
if (!currentAnnouncement.value) {
|
|
currentAnnouncement.value = allAnnouncements.at(-1);
|
|
}
|
|
}
|
|
},
|
|
);
|
|
|
|
async function setLastRead(key: string) {
|
|
const user = auth.user.value!;
|
|
if (user.lastReadAnnouncement && key <= user.lastReadAnnouncement) {
|
|
// Don't update the last read announcement if it's older than the current one
|
|
return;
|
|
}
|
|
|
|
await api.users.updateOne(
|
|
user.id,
|
|
{
|
|
...user,
|
|
lastReadAnnouncement: null, // TODO: switch back to key
|
|
},
|
|
{ suppressAlert: true },
|
|
);
|
|
}
|
|
|
|
function markAllAsRead() {
|
|
newAnnouncements.value = [];
|
|
|
|
const newestAnnouncement = allAnnouncements.at(-1)!;
|
|
setLastRead(newestAnnouncement.key);
|
|
}
|
|
|
|
function nextAnnouncement() {
|
|
const nextAnnouncement = newAnnouncements.value.at(0);
|
|
newAnnouncements.value = newAnnouncements.value.slice(1);
|
|
|
|
if (!nextAnnouncement) {
|
|
markAllAsRead();
|
|
return;
|
|
}
|
|
|
|
currentAnnouncement.value = nextAnnouncement;
|
|
setLastRead(currentAnnouncement.value.key);
|
|
}
|
|
</script>
|