1
0
Fork 0
mirror of https://github.com/mealie-recipes/mealie.git synced 2025-07-21 14:19:41 +02:00
mealie/frontend/pages/g/[groupSlug]/cookbooks/index.vue

263 lines
7.9 KiB
Vue
Raw Normal View History

<template>
<div>
<!-- Create Dialog -->
<BaseDialog
v-if="createTarget"
v-model="dialogStates.create"
width="100%"
max-width="1100px"
:icon="$globals.icons.pages"
:title="$t('cookbook.create-a-cookbook')"
:submit-icon="$globals.icons.save"
:submit-text="$t('general.save')"
:submit-disabled="!createTarget.queryFilterString"
can-submit
@submit="actions.updateOne(createTarget)"
@cancel="deleteCreateTarget()"
>
<v-card-text>
<CookbookEditor :key="createTargetKey" v-model="createTarget" :actions="actions" />
</v-card-text>
</BaseDialog>
<!-- Delete Dialog -->
<BaseDialog
v-model="dialogStates.delete"
:title="$t('general.delete-with-name', { name: $t('cookbook.cookbook') })"
:icon="$globals.icons.alertCircle"
color="error"
can-confirm
@confirm="deleteCookbook()"
>
<v-card-text>
<p>{{ $t("general.confirm-delete-generic-with-name", { name: $t("cookbook.cookbook") }) }}</p>
<p v-if="deleteTarget" class="mt-4 ml-4">
{{ deleteTarget.name }}
</p>
</v-card-text>
</BaseDialog>
<!-- Cookbook Page -->
<!-- Page Title -->
<v-container class="px-12" max-width="1000">
<BasePageTitle divider>
<template #header>
<v-img width="100%" max-height="100" max-width="100" :src="require('~/static/svgs/manage-cookbooks.svg')" />
</template>
<template #title>
{{ $t("cookbook.cookbooks") }}
</template>
{{ $t("cookbook.description") }}
</BasePageTitle>
<div class="my-6">
<v-checkbox
v-model="cookbookPreferences.hideOtherHouseholds"
:label="$t('cookbook.hide-cookbooks-from-other-households')"
hide-details
color="primary"
/>
<div class="ml-10 mt-n3">
<p class="text-subtitle-2 my-0 py-0">
{{ $t("cookbook.hide-cookbooks-from-other-households-description") }}
</p>
</div>
</div>
<!-- Create New -->
<BaseButton create @click="createCookbook" />
<!-- Cookbook List -->
<v-expansion-panels class="mt-2">
<VueDraggable
v-model="myCookbooks"
handle=".handle"
:delay="250"
:delay-on-touch-only="true"
style="width: 100%"
@end="actions.updateOrder(myCookbooks)"
>
<v-expansion-panel
v-for="(cookbook, index) in myCookbooks"
:key="cookbook.id"
class="my-2 left-border rounded"
>
<v-expansion-panel-title disable-icon-rotate class="text-h6 opacity-80">
<div class="d-flex align-center">
<v-icon size="large" start>
{{ $globals.icons.pages }}
</v-icon>
{{ cookbook.name }}
</div>
<template #actions>
<div class="d-flex align-center">
<v-btn icon variant="text" class="ml-2">
<v-icon>
{{ $globals.icons.edit }}
</v-icon>
</v-btn>
<v-icon class="handle" :size="40">
{{ $globals.icons.arrowUpDown }}
</v-icon>
</div>
</template>
</v-expansion-panel-title>
<v-expansion-panel-text>
<CookbookEditor
v-model="myCookbooks[index]"
:actions="actions"
:collapsable="false"
@delete="deleteEventHandler"
/>
<v-card-actions>
<v-spacer />
<BaseButtonGroup
:buttons="[
{
icon: $globals.icons.delete,
text: $t('general.delete'),
event: 'delete',
},
{
icon: $globals.icons.save,
text: $t('general.save'),
event: 'save',
disabled: !cookbook.queryFilterString,
},
]"
@delete="deleteEventHandler(myCookbooks[index])"
@save="actions.updateOne(myCookbooks[index])"
/>
</v-card-actions>
</v-expansion-panel-text>
</v-expansion-panel>
</VueDraggable>
</v-expansion-panels>
</v-container>
</div>
</template>
<script lang="ts">
import { VueDraggable } from "vue-draggable-plus";
import { useCookbookStore } from "~/composables/store/use-cookbook-store";
import { useHouseholdSelf } from "@/composables/use-households";
import CookbookEditor from "~/components/Domain/Cookbook/CookbookEditor.vue";
import type { CreateCookBook, ReadCookBook } from "~/lib/api/types/cookbook";
import { useCookbookPreferences } from "~/composables/use-users/preferences";
export default defineNuxtComponent({
components: { CookbookEditor, VueDraggable },
middleware: ["sidebase-auth", "group-only"],
setup() {
const dialogStates = reactive({
create: false,
delete: false,
});
const i18n = useI18n();
// Set page title
useSeoMeta({
title: i18n.t("cookbook.cookbooks"),
});
const $auth = useMealieAuth();
const { store: allCookbooks, actions } = useCookbookStore();
// Make a local reactive copy of myCookbooks
const myCookbooks = ref<ReadCookBook[]>([]);
watch(
allCookbooks,
(cookbooks) => {
myCookbooks.value
= cookbooks?.filter(
cookbook => cookbook.householdId === $auth.user.value?.householdId,
) ?? [];
},
{ immediate: true },
);
const { household } = useHouseholdSelf();
const cookbookPreferences = useCookbookPreferences();
// create
const createTargetKey = ref(0);
const createTarget = ref<ReadCookBook | null>(null);
async function createCookbook() {
const name = i18n.t("cookbook.household-cookbook-name", [
household.value?.name || "",
String((myCookbooks.value?.length ?? 0) + 1),
]) as string;
const data = { name } as CreateCookBook;
await actions.createOne(data).then((cookbook) => {
if (!cookbook) {
return;
}
myCookbooks.value.push(cookbook);
createTarget.value = cookbook as ReadCookBook;
createTargetKey.value++;
});
dialogStates.create = true;
}
// delete
const deleteTarget = ref<ReadCookBook | null>(null);
function deleteEventHandler(item: ReadCookBook) {
deleteTarget.value = item;
dialogStates.delete = true;
}
async function deleteCookbook() {
if (!deleteTarget.value) {
return;
}
await actions.deleteOne(deleteTarget.value.id);
myCookbooks.value = myCookbooks.value.filter(c => c.id !== deleteTarget.value?.id);
dialogStates.delete = false;
deleteTarget.value = null;
}
async function deleteCreateTarget() {
if (!createTarget.value?.id) {
return;
}
await actions.deleteOne(createTarget.value.id);
myCookbooks.value = myCookbooks.value.filter(c => c.id !== createTarget.value?.id);
dialogStates.create = false;
createTarget.value = null;
}
function handleUnmount() {
if (!createTarget.value?.id || createTarget.value.queryFilterString) {
return;
}
deleteCreateTarget();
}
onMounted(() => {
window.addEventListener("beforeunload", handleUnmount);
});
onBeforeUnmount(() => {
handleUnmount();
window.removeEventListener("beforeunload", handleUnmount);
});
return {
myCookbooks,
cookbookPreferences,
actions,
dialogStates,
// create
createTargetKey,
createTarget,
createCookbook,
// delete
deleteTarget,
deleteEventHandler,
deleteCookbook,
deleteCreateTarget,
};
},
});
</script>