chore: Nuxt 4 upgrade (#7426)

This commit is contained in:
Kuchenpirat
2026-04-08 17:25:41 +02:00
committed by GitHub
parent 70a251a331
commit d3e41582ae
561 changed files with 1840 additions and 2750 deletions

View File

@@ -1,74 +0,0 @@
<template>
<v-tooltip
v-if="userId"
:disabled="!user || !tooltip"
location="end"
>
<template #activator="{ props: tooltipProps }">
<v-avatar
v-if="list"
v-bind="tooltipProps"
>
<v-img
:src="imageURL"
:alt="userId"
@load="error = false"
@error="error = true"
/>
</v-avatar>
<v-avatar
v-else
:size="size"
v-bind="tooltipProps"
>
<v-img
:src="imageURL"
:alt="userId"
@load="error = false"
@error="error = true"
/>
</v-avatar>
</template>
<span v-if="user">
{{ user.fullName }}
</span>
</v-tooltip>
</template>
<script setup lang="ts">
import { useUserStore } from "~/composables/store/use-user-store";
const props = defineProps({
userId: {
type: String,
required: true,
},
list: {
type: Boolean,
default: false,
},
size: {
type: String,
default: "42",
},
tooltip: {
type: Boolean,
default: true,
},
});
const error = ref(false);
const auth = useMealieAuth();
const { store: users } = useUserStore();
const user = computed(() => {
return users.value.find(user => user.id === props.userId);
});
const imageURL = computed(() => {
// Note: auth.user is a ref now
const authUser = auth.user.value;
const key = authUser?.cacheKey ?? "";
return `/api/media/users/${props.userId}/profile.webp?cacheKey=${key}`;
});
</script>

View File

@@ -1,179 +0,0 @@
<template>
<BaseDialog
v-model="inviteDialog"
:title="$t('profile.get-invite-link')"
:icon="$globals.icons.accountPlusOutline"
color="primary"
>
<v-container>
<v-form class="mt-5">
<v-select
v-if="groups && groups.length"
v-model="selectedGroup"
:items="groups"
item-title="name"
item-value="id"
:return-object="false"
variant="filled"
:label="$t('group.user-group')"
:rules="[validators.required]"
/>
<v-select
v-if="households && households.length"
v-model="selectedHousehold"
:items="filteredHouseholds"
item-title="name"
item-value="id"
:return-object="false"
variant="filled"
:label="$t('household.user-household')"
:rules="[validators.required]"
/>
<v-row>
<v-col cols="9">
<v-text-field
v-model="generatedSignupLink"
:label="$t('profile.invite-link')"
type="text"
readonly
variant="filled"
/>
</v-col>
<v-col
cols="3"
class="pl-1 mt-3"
>
<AppButtonCopy
:icon="false"
color="info"
:copy-text="generatedSignupLink"
:disabled="generatedSignupLink"
/>
</v-col>
</v-row>
<v-text-field
v-model="sendTo"
:label="$t('user.email')"
:rules="[validators.email]"
variant="outlined"
@keydown.enter="sendInvite"
/>
</v-form>
</v-container>
<template #custom-card-action>
<BaseButton
:disabled="!validEmail"
:loading="loading"
:icon="$globals.icons.email"
@click="sendInvite"
>
{{ $t("group.invite") }}
</BaseButton>
</template>
</BaseDialog>
</template>
<script setup lang="ts">
import { useUserApi } from "@/composables/api";
import BaseDialog from "~/components/global/BaseDialog.vue";
import AppButtonCopy from "~/components/global/AppButtonCopy.vue";
import BaseButton from "~/components/global/BaseButton.vue";
import { validators } from "~/composables/use-validators";
import { alert } from "~/composables/use-toast";
import type { GroupInDB } from "~/lib/api/types/user";
import type { HouseholdInDB } from "~/lib/api/types/household";
import { useGroups } from "~/composables/use-groups";
import { useAdminHouseholds } from "~/composables/use-households";
const inviteDialog = defineModel<boolean>("modelValue", { type: Boolean, default: false });
const i18n = useI18n();
const auth = useMealieAuth();
const isAdmin = computed(() => auth.user.value?.admin);
const token = ref("");
const selectedGroup = ref<string | null>(null);
const selectedHousehold = ref<string | null>(null);
const groups = ref<GroupInDB[]>([]);
const households = ref<HouseholdInDB[]>([]);
const api = useUserApi();
const fetchGroupsAndHouseholds = () => {
if (isAdmin.value) {
const groupsResponse = useGroups();
const householdsResponse = useAdminHouseholds();
watchEffect(() => {
groups.value = groupsResponse.groups.value || [];
households.value = householdsResponse.households.value || [];
});
}
};
async function getSignupLink(group: string | null = null, household: string | null = null) {
const payload = (group && household) ? { uses: 1, group_id: group, household_id: household } : { uses: 1 };
const { data } = await api.households.createInvitation(payload);
if (data) {
token.value = data.token;
}
}
const filteredHouseholds = computed(() => {
if (!selectedGroup.value) return [];
return households.value?.filter(household => household.groupId === selectedGroup.value);
});
function constructLink(tokenVal: string) {
return tokenVal ? `${window.location.origin}/register?token=${tokenVal}` : "";
}
const generatedSignupLink = computed(() => constructLink(token.value));
// Email Invitation
const state = reactive({
loading: false,
sendTo: "",
});
const { loading, sendTo } = toRefs(state);
async function sendInvite() {
state.loading = true;
if (!token.value) {
getSignupLink(selectedGroup.value, selectedHousehold.value);
}
const { data } = await api.email.sendInvitation({
email: state.sendTo,
token: token.value,
});
if (data && data.success) {
alert.success(i18n.t("profile.email-sent"));
}
else {
alert.error(i18n.t("profile.error-sending-email"));
}
state.loading = false;
inviteDialog.value = false;
}
const validEmail = computed(() => {
if (sendTo.value === "") return false;
const valid = validators.email(sendTo.value);
return valid === true;
});
// Watchers (replacing options API watchers)
watch(inviteDialog, (val) => {
if (val && !isAdmin.value) {
getSignupLink();
}
});
watch(selectedHousehold, (newVal) => {
if (newVal && selectedGroup.value) {
getSignupLink(selectedGroup.value, selectedHousehold.value);
}
});
// initial fetch
fetchGroupsAndHouseholds();
</script>

View File

@@ -1,22 +0,0 @@
<template>
<div class="d-flex pb-6 mt-n1 ml-10">
<div style="flex-basis: 500px">
<strong> {{ $t("user.password-strength", { strength: pwStrength.strength.value }) }}</strong>
<v-progress-linear
v-model="pwStrength.score.value"
rounded
:color="pwStrength.color.value"
height="15"
/>
</div>
</div>
</template>
<script setup lang="ts">
import { usePasswordStrength } from "~/composables/use-passwords";
const modelValue = defineModel<string>({ default: "" });
const i18n = useI18n();
const pwStrength = usePasswordStrength(modelValue, i18n);
</script>

View File

@@ -1,73 +0,0 @@
<template>
<v-card
variant="outlined"
style="border-color: lightgrey;"
:to="link.to"
height="100%"
class="d-flex flex-column mt-4"
>
<div
v-if="$vuetify.display.smAndDown"
class="pa-2 mx-auto"
>
<v-img
width="150px"
height="125"
:src="image"
/>
</div>
<div class="d-flex justify-space-between">
<div>
<v-card-title class="text-subtitle-1 pb-0">
<slot name="title" />
</v-card-title>
<div class="d-flex justify-center align-center">
<v-card-text class="d-flex flex-row mb-auto">
<slot name="default" />
</v-card-text>
</div>
</div>
<div
v-if="$vuetify.display.mdAndUp"
class="py-2 px-10 my-auto"
>
<v-img
width="150px"
height="125"
:src="image"
/>
</div>
</div>
<v-spacer />
<v-divider />
<v-card-actions>
<v-btn
variant="text"
color="info"
:to="link.to"
>
{{ link.text }}
</v-btn>
</v-card-actions>
</v-card>
</template>
<script setup lang="ts">
interface LinkProp {
text: string;
url?: string;
to: string;
}
defineProps({
link: {
type: Object as () => LinkProp,
required: true,
},
image: {
type: String,
required: false,
default: "",
},
});
</script>

View File

@@ -1,142 +0,0 @@
<template>
<div>
<v-card-title class="pt-0">
<v-icon
size="large"
class="mr-3"
>
{{ $globals.icons.user }}
</v-icon>
<span class="headline"> {{ $t("user-registration.account-details") }}</span>
</v-card-title>
<v-divider />
<v-card-text class="mt-2">
<v-form
ref="domAccountForm"
@submit.prevent
>
<v-text-field
v-model="accountDetails.username.value"
autofocus
v-bind="inputAttrs"
:label="$t('user.username')"
:prepend-icon="$globals.icons.user"
:rules="[validators.required]"
:error-messages="usernameErrorMessages"
@blur="validateUsername"
/>
<v-text-field
v-model="accountDetails.fullName.value"
v-bind="inputAttrs"
:label="$t('user.full-name')"
:prepend-icon="$globals.icons.user"
:rules="[validators.required]"
/>
<v-text-field
v-model="accountDetails.email.value"
v-bind="inputAttrs"
:prepend-icon="$globals.icons.email"
:label="$t('user.email')"
:rules="[validators.required, validators.email]"
:error-messages="emailErrorMessages"
@blur="validateEmail"
/>
<v-text-field
v-model="credentials.password1.value"
v-bind="inputAttrs"
:type="pwFields.inputType.value"
:append-inner-icon="pwFields.passwordIcon.value"
:prepend-icon="$globals.icons.lock"
:label="$t('user.password')"
:rules="[validators.required, validators.minLength(8), validators.maxLength(258)]"
@click:append-inner="pwFields.togglePasswordShow"
/>
<UserPasswordStrength v-model="credentials.password1.value" />
<v-text-field
v-model="credentials.password2.value"
v-bind="inputAttrs"
:type="pwFields.inputType.value"
:append-inner-icon="pwFields.passwordIcon.value"
:prepend-icon="$globals.icons.lock"
:label="$t('user.confirm-password')"
:rules="[validators.required, credentials.passwordMatch]"
@click:append-inner="pwFields.togglePasswordShow"
/>
<div class="px-2">
<v-checkbox
v-model="accountDetails.advancedOptions.value"
:label="$t('user.enable-advanced-content')"
/>
<p class="text-caption mt-n4">
{{ $t("user.enable-advanced-content-description") }}
</p>
</div>
</v-form>
</v-card-text>
</div>
</template>
<script setup lang="ts">
import { validators } from "~/composables/use-validators";
import { useUserRegistrationForm } from "~/composables/use-users/user-registration-form";
import { usePasswordField } from "~/composables/use-passwords";
import UserPasswordStrength from "~/components/Domain/User/UserPasswordStrength.vue";
definePageMeta({ layout: "blank" });
const inputAttrs = {
validateOnBlur: true,
class: "pb-1",
variant: "solo-filled" as any,
};
const pwFields = usePasswordField();
const {
accountDetails,
credentials,
emailErrorMessages,
usernameErrorMessages,
validateUsername,
validateEmail,
domAccountForm,
} = useUserRegistrationForm();
</script>
<style lang="css" scoped>
.icon-primary {
fill: var(--v-primary-base);
}
.icon-white {
fill: white;
}
.icon-container {
display: flex;
flex-direction: column;
align-items: center;
width: 100%;
position: relative;
margin-top: 2.5rem;
}
.icon-divider {
width: 100%;
margin-bottom: -2.5rem;
}
.icon-avatar {
border-color: rgba(0, 0, 0, 0.12);
border: 2px;
}
.bg-off-white {
background: #f5f8fa;
}
.preferred-width {
width: 840px;
}
</style>