recipePage use v-model instead of prop and turn into script setup

This commit is contained in:
Kuchenpirat 2025-06-18 11:53:38 +00:00
commit 5fadd299dd
3 changed files with 152 additions and 196 deletions

View file

@ -149,7 +149,7 @@
</div> </div>
</template> </template>
<script lang="ts"> <script setup lang="ts">
import { invoke, until } from "@vueuse/core"; import { invoke, until } from "@vueuse/core";
import RecipeIngredients from "../RecipeIngredients.vue"; import RecipeIngredients from "../RecipeIngredients.vue";
import RecipePageEditorToolbar from "./RecipePageParts/RecipePageEditorToolbar.vue"; import RecipePageEditorToolbar from "./RecipePageParts/RecipePageEditorToolbar.vue";
@ -186,30 +186,8 @@ const EDITOR_OPTIONS = {
mainMenuBar: false, mainMenuBar: false,
}; };
export default defineNuxtComponent({ const recipe = defineModel<NoUndefinedField<Recipe>>({ required: true });
components: {
RecipePageHeader,
RecipePrintContainer,
RecipePageComments,
RecipePageInfoEditor,
RecipePageEditorToolbar,
RecipePageIngredientEditor,
RecipePageOrganizers,
RecipePageScale,
RecipePageIngredientToolsView,
RecipeDialogBulkAdd,
RecipeNotes,
RecipePageInstructions,
RecipePageFooter,
RecipeIngredients,
},
props: {
recipe: {
type: Object as () => NoUndefinedField<Recipe>,
required: true,
},
},
setup(props) {
const { $vuetify } = useNuxtApp(); const { $vuetify } = useNuxtApp();
const i18n = useI18n(); const i18n = useI18n();
const $auth = useMealieAuth(); const $auth = useMealieAuth();
@ -221,11 +199,11 @@ export default defineNuxtComponent({
const router = useRouter(); const router = useRouter();
const api = useUserApi(); const api = useUserApi();
const { pageMode, editMode, setMode, isEditForm, isEditJSON, isCookMode, isEditMode, toggleCookMode } const { pageMode, editMode, setMode, isEditForm, isEditJSON, isCookMode, isEditMode, toggleCookMode }
= usePageState(props.recipe.slug); = usePageState(recipe.value.slug);
const { deactivateNavigationWarning } = useNavigationWarning(); const { deactivateNavigationWarning } = useNavigationWarning();
const notLinkedIngredients = computed(() => { const notLinkedIngredients = computed(() => {
return props.recipe.recipeIngredient.filter((ingredient) => { return recipe.value.recipeIngredient.filter((ingredient) => {
return !props.recipe.recipeInstructions.some(step => return !recipe.value.recipeInstructions.some(step =>
step.ingredientReferences?.map(ref => ref.referenceId).includes(ingredient.referenceId), step.ingredientReferences?.map(ref => ref.referenceId).includes(ingredient.referenceId),
); );
}); });
@ -239,27 +217,27 @@ export default defineNuxtComponent({
const originalRecipe = ref<Recipe | null>(null); const originalRecipe = ref<Recipe | null>(null);
invoke(async () => { invoke(async () => {
await until(props.recipe).not.toBeNull(); await until(recipe.value).not.toBeNull();
originalRecipe.value = deepCopy(props.recipe); originalRecipe.value = deepCopy(recipe.value);
}); });
onUnmounted(async () => { onUnmounted(async () => {
const isSame = JSON.stringify(props.recipe) === JSON.stringify(originalRecipe.value); const isSame = JSON.stringify(recipe.value) === JSON.stringify(originalRecipe.value);
if (isEditMode.value && !isSame && props.recipe?.slug !== undefined) { if (isEditMode.value && !isSame && recipe.value?.slug !== undefined) {
const save = window.confirm(i18n.t("general.unsaved-changes")); const save = window.confirm(i18n.t("general.unsaved-changes"));
if (save) { if (save) {
await api.recipes.updateOne(props.recipe.slug, props.recipe); await api.recipes.updateOne(recipe.value.slug, recipe.value);
} }
} }
deactivateNavigationWarning(); deactivateNavigationWarning();
toggleCookMode(); toggleCookMode();
clearPageState(props.recipe.slug || ""); clearPageState(recipe.value.slug || "");
console.debug("reset RecipePage state during unmount"); console.debug("reset RecipePage state during unmount");
}); });
const hasLinkedIngredients = computed(() => { const hasLinkedIngredients = computed(() => {
return props.recipe.recipeInstructions.some( return recipe.value.recipeInstructions.some(
step => step.ingredientReferences && step.ingredientReferences.length > 0, step => step.ingredientReferences && step.ingredientReferences.length > 0,
); );
}); });
@ -282,7 +260,7 @@ export default defineNuxtComponent({
*/ */
async function saveRecipe() { async function saveRecipe() {
const { data } = await api.recipes.updateOne(props.recipe.slug, props.recipe); const { data } = await api.recipes.updateOne(recipe.value.slug, recipe.value);
setMode(PageMode.VIEW); setMode(PageMode.VIEW);
if (data?.slug) { if (data?.slug) {
router.push(`/g/${groupSlug.value}/r/` + data.slug); router.push(`/g/${groupSlug.value}/r/` + data.slug);
@ -290,7 +268,7 @@ export default defineNuxtComponent({
} }
async function deleteRecipe() { async function deleteRecipe() {
const { data } = await api.recipes.deleteOne(props.recipe.slug); const { data } = await api.recipes.deleteOne(recipe.value.slug);
if (data?.slug) { if (data?.slug) {
router.push(`/g/${groupSlug.value}`); router.push(`/g/${groupSlug.value}`);
} }
@ -300,7 +278,7 @@ export default defineNuxtComponent({
* View Preferences * View Preferences
*/ */
const landscape = computed(() => { const landscape = computed(() => {
const preferLandscape = props.recipe.settings.landscapeView; const preferLandscape = recipe.value.settings.landscapeView;
const smallScreen = !$vuetify.display.smAndUp.value; const smallScreen = !$vuetify.display.smAndUp.value;
if (preferLandscape) { if (preferLandscape) {
@ -319,7 +297,7 @@ export default defineNuxtComponent({
*/ */
function addStep(steps: Array<string> | null = null) { function addStep(steps: Array<string> | null = null) {
if (!props.recipe.recipeInstructions) { if (!recipe.value.recipeInstructions) {
return; return;
} }
@ -328,10 +306,10 @@ export default defineNuxtComponent({
return { id: uuid4(), text: step, title: "", ingredientReferences: [] }; return { id: uuid4(), text: step, title: "", ingredientReferences: [] };
}); });
props.recipe.recipeInstructions.push(...cleanedSteps); recipe.value.recipeInstructions.push(...cleanedSteps);
} }
else { else {
props.recipe.recipeInstructions.push({ recipe.value.recipeInstructions.push({
id: uuid4(), id: uuid4(),
text: "", text: "",
title: "", title: "",
@ -357,32 +335,10 @@ export default defineNuxtComponent({
router.push(`/g/${groupSlug.value}?${itemType}=${item.id}`); router.push(`/g/${groupSlug.value}?${itemType}=${item.id}`);
} }
return { const scale = ref(1);
user,
isOwnGroup,
api,
scale: ref(1),
EDITOR_OPTIONS,
landscape,
pageMode, // expose to template
editMode, // (all variables used in template are top-level in <script setup>)
PageMode,
EditorMode,
isEditMode,
isEditForm,
isEditJSON,
isCookMode,
toggleCookMode,
saveRecipe,
deleteRecipe,
addStep,
hasLinkedIngredients,
notLinkedIngredients,
chipClicked,
};
},
});
</script> </script>
<style lang="css"> <style lang="css">

View file

@ -2,7 +2,7 @@
<div> <div>
<RecipePage <RecipePage
v-if="recipe" v-if="recipe"
:recipe="recipe" v-model="recipe"
/> />
</div> </div>
</template> </template>

View file

@ -3,7 +3,7 @@
<client-only> <client-only>
<RecipePage <RecipePage
v-if="recipe" v-if="recipe"
:recipe="recipe" v-model="recipe"
/> />
</client-only> </client-only>
</div> </div>