mirror of
https://github.com/mealie-recipes/mealie.git
synced 2025-05-24 01:12:54 -04:00
* add groupSlug to most routes * fixed more routing issues * fixed jank and incorrect routes * remove public explore links * remove unused groupSlug and explore routes * nuked explore pages * fixed public toolstore bug * fixed various routes missing group slug * restored public app header menu * fix janky login redirect * 404 recipe API call returns to login * removed unused explore layout * force redirect when using the wrong group slug * fixed dead admin links * removed unused middleware from earlier attempt * 🧹 * improve cookbooks sidebar fixed sidebar link not working fixed sidebar link target hide cookbooks header when there are none * added group slug to user * fix $auth typehints * vastly simplified groupSlug logic * allow logged-in users to view other groups * fixed some edgecases that bypassed isOwnGroup * fixed static home ref * 🧹 * fixed redirect logic * lint warning * removed group slug from group and user pages refactored all components to use route groupSlug or user group slug moved some group pages to recipe pages * fixed some bad types * 🧹 * moved groupSlug routes under /g/groupSlug * move /recipe/ to /r/ * fix backend url generation and metadata injection * moved shopping lists to root/other route fixes * changed shared from /recipes/ to /r/ * fixed 404 redirect not awaiting * removed unused import * fix doc links * fix public recipe setting not affecting public API * fixed backend tests * fix nuxt-generate command --------- Co-authored-by: Hayden <64056131+hay-kot@users.noreply.github.com>
154 lines
4.6 KiB
Vue
154 lines
4.6 KiB
Vue
<template>
|
|
<div>
|
|
<div class="d-flex justify-end flex-wrap align-stretch">
|
|
<v-card v-if="!landscape" width="50%" flat class="d-flex flex-column justify-center align-center">
|
|
<v-card-text>
|
|
<v-card-title class="headline pa-0 flex-column align-center">
|
|
{{ recipe.name }}
|
|
<RecipeRating :key="recipe.slug" v-model="recipe.rating" :name="recipe.name" :slug="recipe.slug" />
|
|
</v-card-title>
|
|
<v-divider class="my-2"></v-divider>
|
|
<SafeMarkdown :source="recipe.description" />
|
|
<v-divider></v-divider>
|
|
<div v-if="isOwnGroup" class="d-flex justify-center mt-5">
|
|
<RecipeLastMade
|
|
v-model="recipe.lastMade"
|
|
:recipe="recipe"
|
|
class="d-flex justify-center flex-wrap"
|
|
:class="true ? undefined : 'force-bottom'"
|
|
/>
|
|
</div>
|
|
<div class="d-flex justify-center mt-5">
|
|
<RecipeTimeCard
|
|
class="d-flex justify-center flex-wrap"
|
|
:class="true ? undefined : 'force-bottom'"
|
|
:prep-time="recipe.prepTime"
|
|
:total-time="recipe.totalTime"
|
|
:perform-time="recipe.performTime"
|
|
/>
|
|
</div>
|
|
</v-card-text>
|
|
</v-card>
|
|
<v-img
|
|
:key="imageKey"
|
|
:max-width="landscape ? null : '50%'"
|
|
min-height="50"
|
|
:height="hideImage ? undefined : imageHeight"
|
|
:src="recipeImageUrl"
|
|
class="d-print-none"
|
|
@error="hideImage = true"
|
|
>
|
|
</v-img>
|
|
</div>
|
|
<v-divider></v-divider>
|
|
<RecipeActionMenu
|
|
:recipe="recipe"
|
|
:slug="recipe.slug"
|
|
:recipe-scale="recipeScale"
|
|
:locked="isOwnGroup && user.id !== recipe.userId && recipe.settings.locked"
|
|
:name="recipe.name"
|
|
:logged-in="isOwnGroup"
|
|
:open="isEditMode"
|
|
:recipe-id="recipe.id"
|
|
:show-ocr-button="recipe.isOcrRecipe"
|
|
class="ml-auto mt-n8 pb-4"
|
|
@close="setMode(PageMode.VIEW)"
|
|
@json="toggleEditMode()"
|
|
@edit="setMode(PageMode.EDIT)"
|
|
@save="$emit('save')"
|
|
@delete="$emit('delete')"
|
|
@print="printRecipe"
|
|
@ocr="goToOcrEditor"
|
|
/>
|
|
</div>
|
|
</template>
|
|
|
|
<script lang="ts">
|
|
import { defineComponent, useContext, computed, ref, watch, useRouter, useRoute } from "@nuxtjs/composition-api";
|
|
import { useLoggedInState } from "~/composables/use-logged-in-state";
|
|
import RecipeRating from "~/components/Domain/Recipe/RecipeRating.vue";
|
|
import RecipeLastMade from "~/components/Domain/Recipe/RecipeLastMade.vue";
|
|
import RecipeActionMenu from "~/components/Domain/Recipe/RecipeActionMenu.vue";
|
|
import RecipeTimeCard from "~/components/Domain/Recipe/RecipeTimeCard.vue";
|
|
import { useStaticRoutes } from "~/composables/api";
|
|
import { Recipe } from "~/lib/api/types/recipe";
|
|
import { NoUndefinedField } from "~/lib/api/types/non-generated";
|
|
import { usePageState, usePageUser, PageMode, EditorMode } from "~/composables/recipe-page/shared-state";
|
|
export default defineComponent({
|
|
components: {
|
|
RecipeTimeCard,
|
|
RecipeActionMenu,
|
|
RecipeRating,
|
|
RecipeLastMade,
|
|
},
|
|
props: {
|
|
recipe: {
|
|
type: Object as () => NoUndefinedField<Recipe>,
|
|
required: true,
|
|
},
|
|
recipeScale: {
|
|
type: Number,
|
|
default: 1,
|
|
},
|
|
landscape: {
|
|
type: Boolean,
|
|
default: false,
|
|
},
|
|
},
|
|
setup(props) {
|
|
const { $auth, $vuetify } = useContext();
|
|
const { recipeImage } = useStaticRoutes();
|
|
const { imageKey, pageMode, editMode, setMode, toggleEditMode, isEditMode } = usePageState(props.recipe.slug);
|
|
const { user } = usePageUser();
|
|
const { isOwnGroup } = useLoggedInState();
|
|
|
|
const route = useRoute();
|
|
const groupSlug = computed(() => route.value.params.groupSlug || $auth.user?.groupSlug || "");
|
|
const router = useRouter();
|
|
|
|
function printRecipe() {
|
|
window.print();
|
|
}
|
|
|
|
const hideImage = ref(false);
|
|
const imageHeight = computed(() => {
|
|
return $vuetify.breakpoint.xs ? "200" : "400";
|
|
});
|
|
|
|
const recipeImageUrl = computed(() => {
|
|
return recipeImage(props.recipe.id, props.recipe.image, imageKey.value);
|
|
});
|
|
|
|
function goToOcrEditor() {
|
|
router.push(`/g/${groupSlug.value}/r/${props.recipe.slug}/ocr-editor`);
|
|
}
|
|
|
|
watch(
|
|
() => recipeImageUrl.value,
|
|
() => {
|
|
hideImage.value = false;
|
|
}
|
|
);
|
|
|
|
return {
|
|
isOwnGroup,
|
|
setMode,
|
|
toggleEditMode,
|
|
recipeImage,
|
|
imageKey,
|
|
user,
|
|
PageMode,
|
|
pageMode,
|
|
EditorMode,
|
|
editMode,
|
|
printRecipe,
|
|
imageHeight,
|
|
hideImage,
|
|
isEditMode,
|
|
recipeImageUrl,
|
|
goToOcrEditor,
|
|
};
|
|
},
|
|
});
|
|
</script>
|