2021-07-31 14:45:28 -08:00
|
|
|
<template>
|
2025-06-20 00:09:12 +07:00
|
|
|
<v-container
|
|
|
|
fluid
|
|
|
|
class="narrow-container"
|
|
|
|
>
|
2023-12-11 19:49:26 +01:00
|
|
|
<!-- Image -->
|
2021-10-03 18:38:45 -08:00
|
|
|
<BasePageTitle divider>
|
|
|
|
<template #header>
|
2025-06-20 00:09:12 +07:00
|
|
|
<v-img
|
|
|
|
width="100%"
|
|
|
|
max-height="200"
|
|
|
|
max-width="150"
|
|
|
|
:src="require('~/static/svgs/admin-site-settings.svg')"
|
|
|
|
/>
|
|
|
|
</template>
|
|
|
|
<template #title>
|
|
|
|
{{ $t("settings.site-settings") }}
|
2021-10-03 18:38:45 -08:00
|
|
|
</template>
|
|
|
|
</BasePageTitle>
|
2021-10-04 20:16:37 -08:00
|
|
|
|
2023-12-11 19:49:26 +01:00
|
|
|
<!-- Bug Report -->
|
2025-06-20 00:09:12 +07:00
|
|
|
<BaseDialog
|
|
|
|
v-model="bugReportDialog"
|
|
|
|
:title="$t('settings.bug-report')"
|
|
|
|
:width="800"
|
|
|
|
:icon="$globals.icons.github"
|
|
|
|
>
|
2022-05-08 17:43:03 -08:00
|
|
|
<v-card-text>
|
|
|
|
<div class="pb-4">
|
2023-03-21 20:45:27 +01:00
|
|
|
{{ $t('settings.bug-report-information') }}
|
2022-05-08 17:43:03 -08:00
|
|
|
</div>
|
2025-06-20 00:09:12 +07:00
|
|
|
<v-textarea
|
|
|
|
v-model="bugReportText"
|
|
|
|
variant="outlined"
|
|
|
|
rows="18"
|
|
|
|
readonly
|
|
|
|
/>
|
|
|
|
<div
|
|
|
|
class="d-flex justify-end"
|
|
|
|
style="gap: 5px"
|
|
|
|
>
|
|
|
|
<BaseButton
|
|
|
|
color="gray"
|
|
|
|
secondary
|
|
|
|
target="_blank"
|
|
|
|
href="https://github.com/hay-kot/mealie/issues/new/choose"
|
|
|
|
>
|
|
|
|
<template #icon>
|
|
|
|
{{ $globals.icons.github }}
|
|
|
|
</template>
|
2023-03-21 20:45:27 +01:00
|
|
|
{{ $t('settings.tracker') }}
|
2022-05-08 17:43:03 -08:00
|
|
|
</BaseButton>
|
2025-06-20 00:09:12 +07:00
|
|
|
<AppButtonCopy
|
|
|
|
:copy-text="bugReportText"
|
|
|
|
color="info"
|
|
|
|
:icon="false"
|
|
|
|
/>
|
2022-05-08 17:43:03 -08:00
|
|
|
</div>
|
|
|
|
</v-card-text>
|
|
|
|
</BaseDialog>
|
|
|
|
|
|
|
|
<div class="d-flex justify-end">
|
|
|
|
<BaseButton
|
|
|
|
color="info"
|
|
|
|
@click="
|
|
|
|
bugReportDialog = true;
|
|
|
|
"
|
|
|
|
>
|
2025-06-20 00:09:12 +07:00
|
|
|
<template #icon>
|
|
|
|
{{ $globals.icons.github }}
|
|
|
|
</template>
|
2023-03-21 20:45:27 +01:00
|
|
|
{{ $t('settings.bug-report') }}
|
2022-05-08 17:43:03 -08:00
|
|
|
</BaseButton>
|
|
|
|
</div>
|
|
|
|
|
2023-12-11 19:49:26 +01:00
|
|
|
<!-- Configuration -->
|
2021-10-04 20:16:37 -08:00
|
|
|
<section>
|
2025-06-20 00:09:12 +07:00
|
|
|
<BaseCardSectionTitle
|
|
|
|
class="pb-0"
|
|
|
|
:icon="$globals.icons.cog"
|
|
|
|
:title="$t('settings.configuration')"
|
|
|
|
/>
|
2022-04-02 16:35:53 -08:00
|
|
|
<v-card class="mb-4">
|
2025-06-20 00:09:12 +07:00
|
|
|
<template
|
|
|
|
v-for="(check, idx) in simpleChecks"
|
|
|
|
:key="`list-item-${idx}`"
|
|
|
|
>
|
|
|
|
<v-list-item :title="check.text">
|
|
|
|
<template #prepend>
|
|
|
|
<v-icon :color="check.color" class="opacity-100">
|
2022-04-02 16:35:53 -08:00
|
|
|
{{ check.icon }}
|
|
|
|
</v-icon>
|
2025-06-20 00:09:12 +07:00
|
|
|
</template>
|
|
|
|
<v-list-item-subtitle class="wrap-word">
|
|
|
|
{{ check.status ? check.successText : check.errorText }}
|
|
|
|
</v-list-item-subtitle>
|
2022-04-02 16:35:53 -08:00
|
|
|
</v-list-item>
|
2025-06-20 00:09:12 +07:00
|
|
|
<v-divider />
|
2022-04-02 16:35:53 -08:00
|
|
|
</template>
|
|
|
|
</v-card>
|
|
|
|
</section>
|
2021-12-04 14:18:46 -09:00
|
|
|
|
2023-12-11 19:49:26 +01:00
|
|
|
<!-- Email -->
|
2021-10-04 20:16:37 -08:00
|
|
|
<section>
|
2025-06-20 00:09:12 +07:00
|
|
|
<BaseCardSectionTitle
|
|
|
|
class="pt-2"
|
|
|
|
:icon="$globals.icons.email"
|
|
|
|
:title="$t('user.email')"
|
|
|
|
/>
|
|
|
|
<v-alert
|
|
|
|
border="start"
|
|
|
|
:border-color="appConfig.emailReady ? 'success' : 'error'"
|
|
|
|
variant="text"
|
|
|
|
elevation="2"
|
|
|
|
>
|
|
|
|
<template #prepend>
|
|
|
|
<v-icon :color="appConfig.emailReady ? 'success' : 'warning'">
|
|
|
|
{{ appConfig.emailReady ? $globals.icons.checkboxMarkedCircle : $globals.icons.alertCircle }}
|
|
|
|
</v-icon>
|
|
|
|
</template>
|
|
|
|
<div class="font-weight-medium">
|
|
|
|
{{ $t('settings.email-configuration-status') }}
|
|
|
|
</div>
|
2021-12-04 14:18:46 -09:00
|
|
|
<div>
|
2025-06-20 00:09:12 +07:00
|
|
|
{{ appConfig.emailReady ? $t('settings.ready') : $t('settings.not-ready') }}
|
2021-12-04 14:18:46 -09:00
|
|
|
</div>
|
|
|
|
<div>
|
2025-06-20 00:09:12 +07:00
|
|
|
<v-text-field
|
|
|
|
v-model="address"
|
|
|
|
class="mr-4"
|
|
|
|
:label="$t('user.email')"
|
|
|
|
:rules="[validators.email]"
|
|
|
|
/>
|
2021-12-04 14:18:46 -09:00
|
|
|
<BaseButton
|
|
|
|
color="info"
|
2025-06-20 00:09:12 +07:00
|
|
|
variant="elevated"
|
2021-12-04 14:18:46 -09:00
|
|
|
:disabled="!appConfig.emailReady || !validEmail"
|
|
|
|
:loading="loading"
|
2025-06-20 00:09:12 +07:00
|
|
|
class="opacity-100"
|
2021-12-04 14:18:46 -09:00
|
|
|
@click="testEmail"
|
|
|
|
>
|
2025-06-20 00:09:12 +07:00
|
|
|
<template #icon>
|
|
|
|
{{ $globals.icons.email }}
|
|
|
|
</template>
|
2021-12-04 14:18:46 -09:00
|
|
|
{{ $t("general.test") }}
|
|
|
|
</BaseButton>
|
|
|
|
<template v-if="tested">
|
2025-06-20 00:09:12 +07:00
|
|
|
<v-divider class="my-x mt-6" />
|
2022-02-23 15:04:45 -09:00
|
|
|
<v-card-text class="px-0">
|
2025-06-20 00:09:12 +07:00
|
|
|
<h4> {{ $t("settings.email-test-results") }}</h4>
|
2022-02-23 15:04:45 -09:00
|
|
|
<span class="pl-4">
|
2023-03-21 20:45:27 +01:00
|
|
|
{{ success ? $t('settings.succeeded') : $t('settings.failed') }}
|
2022-02-23 15:04:45 -09:00
|
|
|
</span>
|
2021-12-04 14:18:46 -09:00
|
|
|
</v-card-text>
|
|
|
|
</template>
|
|
|
|
</div>
|
|
|
|
</v-alert>
|
2021-10-04 20:16:37 -08:00
|
|
|
</section>
|
2022-05-08 17:43:03 -08:00
|
|
|
|
|
|
|
<!-- General App Info -->
|
2021-11-05 15:48:10 -08:00
|
|
|
<section class="mt-4">
|
2025-06-20 00:09:12 +07:00
|
|
|
<BaseCardSectionTitle
|
|
|
|
class="pb-0"
|
|
|
|
:icon="$globals.icons.cog"
|
|
|
|
:title="$t('settings.general-about')"
|
|
|
|
/>
|
2021-11-05 15:48:10 -08:00
|
|
|
<v-card class="mb-4">
|
2023-12-11 03:56:22 +11:00
|
|
|
<template v-if="appInfo && appInfo.length">
|
2025-06-20 00:09:12 +07:00
|
|
|
<template
|
|
|
|
v-for="(property, idx) in appInfo"
|
|
|
|
:key="property.name"
|
|
|
|
>
|
|
|
|
<v-list-item
|
|
|
|
:title="property.name"
|
|
|
|
:prepend-icon="property.icon || $globals.icons.user"
|
|
|
|
>
|
|
|
|
<template v-if="property.slot === 'recipe-scraper'">
|
|
|
|
<v-list-item-subtitle>
|
|
|
|
<a
|
|
|
|
target="_blank"
|
|
|
|
:href="`https://github.com/hhursev/recipe-scrapers/releases/tag/${property.value}`"
|
|
|
|
>
|
2023-02-26 11:51:04 -09:00
|
|
|
{{ property.value }}
|
2025-06-20 00:09:12 +07:00
|
|
|
</a>
|
|
|
|
</v-list-item-subtitle>
|
|
|
|
</template>
|
|
|
|
<template v-else-if="property.slot === 'build'">
|
|
|
|
<v-list-item-subtitle>
|
|
|
|
<a
|
|
|
|
target="_blank"
|
|
|
|
:href="`https://github.com/mealie-recipes/mealie/commit/${property.value}`"
|
|
|
|
>
|
|
|
|
{{ property.value }}
|
|
|
|
</a>
|
|
|
|
</v-list-item-subtitle>
|
|
|
|
</template>
|
|
|
|
<template v-else>
|
|
|
|
<v-list-item-subtitle>
|
|
|
|
{{ property.value }}
|
|
|
|
</v-list-item-subtitle>
|
|
|
|
</template>
|
2023-12-11 03:56:22 +11:00
|
|
|
</v-list-item>
|
2025-06-20 00:09:12 +07:00
|
|
|
<v-divider
|
|
|
|
v-if="appInfo && idx !== appInfo.length - 1"
|
|
|
|
:key="`divider-${property.name}`"
|
|
|
|
/>
|
2023-12-11 03:56:22 +11:00
|
|
|
</template>
|
|
|
|
</template>
|
|
|
|
<template v-else>
|
|
|
|
<div class="mb-3 text-center">
|
2025-06-20 00:09:12 +07:00
|
|
|
<AppLoader :waiting-text="$t('general.loading')" />
|
2023-12-11 03:56:22 +11:00
|
|
|
</div>
|
2022-05-08 17:43:03 -08:00
|
|
|
</template>
|
2021-11-05 15:48:10 -08:00
|
|
|
</v-card>
|
|
|
|
</section>
|
2021-08-06 16:28:12 -08:00
|
|
|
</v-container>
|
2021-08-01 19:24:47 -08:00
|
|
|
</template>
|
2022-01-09 07:15:23 +01:00
|
|
|
|
2021-08-01 19:24:47 -08:00
|
|
|
<script lang="ts">
|
2025-06-20 00:09:12 +07:00
|
|
|
import type { TranslateResult } from "vue-i18n";
|
2021-11-06 11:28:47 -08:00
|
|
|
import { useAdminApi, useUserApi } from "~/composables/api";
|
2021-10-03 18:38:45 -08:00
|
|
|
import { validators } from "~/composables/use-validators";
|
2021-11-05 15:48:10 -08:00
|
|
|
import { useAsyncKey } from "~/composables/use-utils";
|
2025-06-20 00:09:12 +07:00
|
|
|
import type { CheckAppConfig } from "~/lib/api/types/admin";
|
2023-12-11 03:56:22 +11:00
|
|
|
import AppLoader from "~/components/global/AppLoader.vue";
|
2022-04-02 16:35:53 -08:00
|
|
|
|
|
|
|
enum DockerVolumeState {
|
|
|
|
Unknown = "unknown",
|
|
|
|
Success = "success",
|
|
|
|
Error = "error",
|
|
|
|
}
|
2021-08-01 19:24:47 -08:00
|
|
|
|
2021-10-30 15:46:44 -08:00
|
|
|
interface SimpleCheck {
|
2023-03-21 20:45:27 +01:00
|
|
|
id: string;
|
|
|
|
text: TranslateResult;
|
2022-04-02 16:35:53 -08:00
|
|
|
status: boolean | undefined;
|
2023-03-21 20:45:27 +01:00
|
|
|
successText: TranslateResult;
|
|
|
|
errorText: TranslateResult;
|
2022-04-02 16:35:53 -08:00
|
|
|
color: string;
|
|
|
|
icon: string;
|
|
|
|
}
|
|
|
|
|
|
|
|
interface CheckApp extends CheckAppConfig {
|
|
|
|
isSiteSecure?: boolean;
|
2021-10-30 15:46:44 -08:00
|
|
|
}
|
|
|
|
|
2025-06-20 00:09:12 +07:00
|
|
|
export default defineNuxtComponent({
|
|
|
|
components: { AppLoader },
|
|
|
|
setup() {
|
|
|
|
definePageMeta({
|
|
|
|
layout: "admin",
|
|
|
|
});
|
|
|
|
|
|
|
|
const { $globals } = useNuxtApp();
|
|
|
|
const i18n = useI18n();
|
|
|
|
|
|
|
|
const state = reactive({
|
|
|
|
loading: false,
|
|
|
|
address: "",
|
|
|
|
success: false,
|
|
|
|
error: "",
|
|
|
|
tested: false,
|
|
|
|
});
|
|
|
|
|
|
|
|
// Set page title
|
|
|
|
useSeoMeta({
|
|
|
|
title: i18n.t("settings.site-settings"),
|
|
|
|
});
|
|
|
|
|
|
|
|
const appConfig = ref<CheckApp>({
|
|
|
|
emailReady: true,
|
|
|
|
baseUrlSet: true,
|
|
|
|
isSiteSecure: true,
|
|
|
|
isUpToDate: false,
|
|
|
|
ldapReady: false,
|
|
|
|
oidcReady: false,
|
|
|
|
enableOpenai: false,
|
|
|
|
});
|
|
|
|
function isLocalHostOrHttps() {
|
|
|
|
return window.location.hostname === "localhost" || window.location.protocol === "https:";
|
|
|
|
}
|
|
|
|
const api = useUserApi();
|
|
|
|
const adminApi = useAdminApi();
|
|
|
|
onMounted(async () => {
|
|
|
|
const { data } = await adminApi.about.checkApp();
|
|
|
|
if (data) {
|
|
|
|
appConfig.value = { ...data, isSiteSecure: false };
|
|
|
|
}
|
|
|
|
appConfig.value.isSiteSecure = isLocalHostOrHttps();
|
|
|
|
});
|
|
|
|
const simpleChecks = computed<SimpleCheck[]>(() => {
|
|
|
|
const goodIcon = $globals.icons.checkboxMarkedCircle;
|
|
|
|
const badIcon = $globals.icons.alert;
|
|
|
|
const warningIcon = $globals.icons.alertCircle;
|
|
|
|
const goodColor = "success";
|
|
|
|
const badColor = "error";
|
|
|
|
const warningColor = "warning";
|
|
|
|
const data: SimpleCheck[] = [
|
|
|
|
{
|
|
|
|
id: "application-version",
|
|
|
|
text: i18n.t("settings.application-version"),
|
|
|
|
status: appConfig.value.isUpToDate,
|
|
|
|
errorText: i18n.t("settings.application-version-error-text", [rawAppInfo.value.version, rawAppInfo.value.versionLatest]),
|
|
|
|
successText: i18n.t("settings.mealie-is-up-to-date"),
|
|
|
|
color: appConfig.value.isUpToDate ? goodColor : warningColor,
|
|
|
|
icon: appConfig.value.isUpToDate ? goodIcon : warningIcon,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
id: "secure-site",
|
|
|
|
text: i18n.t("settings.secure-site"),
|
|
|
|
status: appConfig.value.isSiteSecure,
|
|
|
|
errorText: i18n.t("settings.secure-site-error-text"),
|
|
|
|
successText: i18n.t("settings.secure-site-success-text"),
|
|
|
|
color: appConfig.value.isSiteSecure ? goodColor : badColor,
|
|
|
|
icon: appConfig.value.isSiteSecure ? goodIcon : badIcon,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
id: "server-side-base-url",
|
|
|
|
text: i18n.t("settings.server-side-base-url"),
|
|
|
|
status: appConfig.value.baseUrlSet,
|
|
|
|
errorText: i18n.t("settings.server-side-base-url-error-text"),
|
|
|
|
successText: i18n.t("settings.server-side-base-url-success-text"),
|
|
|
|
color: appConfig.value.baseUrlSet ? goodColor : badColor,
|
|
|
|
icon: appConfig.value.baseUrlSet ? goodIcon : badIcon,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
id: "ldap-ready",
|
|
|
|
text: i18n.t("settings.ldap-ready"),
|
|
|
|
status: appConfig.value.ldapReady,
|
|
|
|
errorText: i18n.t("settings.ldap-ready-error-text"),
|
|
|
|
successText: i18n.t("settings.ldap-ready-success-text"),
|
|
|
|
color: appConfig.value.ldapReady ? goodColor : warningColor,
|
|
|
|
icon: appConfig.value.ldapReady ? goodIcon : warningIcon,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
id: "oidc-ready",
|
|
|
|
text: i18n.t("settings.oidc-ready"),
|
|
|
|
status: appConfig.value.oidcReady,
|
|
|
|
errorText: i18n.t("settings.oidc-ready-error-text"),
|
|
|
|
successText: i18n.t("settings.oidc-ready-success-text"),
|
|
|
|
color: appConfig.value.oidcReady ? goodColor : warningColor,
|
|
|
|
icon: appConfig.value.oidcReady ? goodIcon : warningIcon,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
id: "openai-ready",
|
|
|
|
text: i18n.t("settings.openai-ready"),
|
|
|
|
status: appConfig.value.enableOpenai,
|
|
|
|
errorText: i18n.t("settings.openai-ready-error-text"),
|
|
|
|
successText: i18n.t("settings.openai-ready-success-text"),
|
|
|
|
color: appConfig.value.enableOpenai ? goodColor : warningColor,
|
|
|
|
icon: appConfig.value.enableOpenai ? goodIcon : warningIcon,
|
|
|
|
},
|
|
|
|
];
|
|
|
|
return data;
|
|
|
|
});
|
|
|
|
async function testEmail() {
|
|
|
|
state.loading = true;
|
|
|
|
state.tested = false;
|
|
|
|
const { data } = await api.email.test({ email: state.address });
|
|
|
|
if (data) {
|
|
|
|
if (data.success) {
|
|
|
|
state.success = true;
|
2021-11-05 15:48:10 -08:00
|
|
|
}
|
2025-06-20 00:09:12 +07:00
|
|
|
else {
|
|
|
|
state.error = data.error ?? "";
|
|
|
|
state.success = false;
|
2022-05-08 17:43:03 -08:00
|
|
|
}
|
2025-06-20 00:09:12 +07:00
|
|
|
}
|
|
|
|
state.loading = false;
|
|
|
|
state.tested = true;
|
|
|
|
}
|
|
|
|
const validEmail = computed(() => {
|
|
|
|
if (state.address === "") {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
const valid = validators.email(state.address);
|
|
|
|
// Explicit bool check because validators.email sometimes returns a string
|
|
|
|
if (valid === true) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
});
|
|
|
|
// ============================================================
|
|
|
|
// General About Info
|
|
|
|
const rawAppInfo = ref({
|
|
|
|
version: "null",
|
|
|
|
versionLatest: "null",
|
|
|
|
});
|
|
|
|
function getAppInfo() {
|
|
|
|
const { data: statistics } = useAsyncData(useAsyncKey(), async () => {
|
|
|
|
const { data } = await adminApi.about.about();
|
|
|
|
if (data) {
|
|
|
|
rawAppInfo.value.version = data.version;
|
|
|
|
rawAppInfo.value.versionLatest = data.versionLatest;
|
|
|
|
const prettyInfo = [
|
|
|
|
{
|
|
|
|
name: i18n.t("about.version"),
|
|
|
|
icon: $globals.icons.information,
|
|
|
|
value: data.version,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
slot: "build",
|
|
|
|
name: i18n.t("settings.build"),
|
|
|
|
icon: $globals.icons.information,
|
|
|
|
value: data.buildId,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: i18n.t("about.application-mode"),
|
|
|
|
icon: $globals.icons.devTo,
|
|
|
|
value: data.production ? i18n.t("about.production") : i18n.t("about.development"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: i18n.t("about.demo-status"),
|
|
|
|
icon: $globals.icons.testTube,
|
|
|
|
value: data.demoStatus ? i18n.t("about.demo") : i18n.t("about.not-demo"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: i18n.t("about.api-port"),
|
|
|
|
icon: $globals.icons.api,
|
|
|
|
value: data.apiPort,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: i18n.t("about.api-docs"),
|
|
|
|
icon: $globals.icons.file,
|
|
|
|
value: data.apiDocs ? i18n.t("general.enabled") : i18n.t("general.disabled"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: i18n.t("about.database-type"),
|
|
|
|
icon: $globals.icons.database,
|
|
|
|
value: data.dbType,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: i18n.t("about.database-url"),
|
|
|
|
icon: $globals.icons.database,
|
|
|
|
value: data.dbUrl,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: i18n.t("about.default-group"),
|
|
|
|
icon: $globals.icons.group,
|
|
|
|
value: data.defaultGroup,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: i18n.t("about.default-household"),
|
|
|
|
icon: $globals.icons.household,
|
|
|
|
value: data.defaultHousehold,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
slot: "recipe-scraper",
|
|
|
|
name: i18n.t("settings.recipe-scraper-version"),
|
|
|
|
icon: $globals.icons.primary,
|
|
|
|
value: data.recipeScraperVersion,
|
|
|
|
},
|
|
|
|
];
|
|
|
|
return prettyInfo;
|
2022-05-08 17:43:03 -08:00
|
|
|
}
|
2025-06-20 00:09:12 +07:00
|
|
|
return data;
|
|
|
|
});
|
|
|
|
return statistics;
|
2023-12-09 14:41:26 +00:00
|
|
|
}
|
2025-06-20 00:09:12 +07:00
|
|
|
const appInfo = getAppInfo();
|
|
|
|
const bugReportDialog = ref(false);
|
|
|
|
const bugReportText = computed(() => {
|
|
|
|
const ignore = {
|
|
|
|
[i18n.t("about.database-url")]: true,
|
|
|
|
[i18n.t("about.default-group")]: true,
|
|
|
|
};
|
|
|
|
let text = "**Details**\n";
|
|
|
|
appInfo.value?.forEach((item) => {
|
|
|
|
if (ignore[item.name as string]) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
text += `${item.name as string}: ${item.value as string}\n`;
|
|
|
|
});
|
|
|
|
const ignoreChecks: {
|
|
|
|
[key: string]: boolean;
|
|
|
|
} = {
|
|
|
|
"application-version": true,
|
|
|
|
};
|
|
|
|
text += "\n**Checks**\n";
|
|
|
|
simpleChecks.value.forEach((item) => {
|
|
|
|
if (ignoreChecks[item.id]) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const status = item.status ? i18n.t("general.yes") : i18n.t("general.no");
|
|
|
|
text += `${item.text.toString()}: ${status}\n`;
|
|
|
|
});
|
|
|
|
text += `${i18n.t("settings.email-configured")}: ${appConfig.value.emailReady ? i18n.t("general.yes") : i18n.t("general.no")}\n`;
|
|
|
|
return text;
|
|
|
|
});
|
|
|
|
return {
|
|
|
|
bugReportDialog,
|
|
|
|
bugReportText,
|
|
|
|
DockerVolumeState,
|
|
|
|
simpleChecks,
|
|
|
|
appConfig,
|
|
|
|
validEmail,
|
|
|
|
validators,
|
|
|
|
...toRefs(state),
|
|
|
|
testEmail,
|
|
|
|
appInfo,
|
|
|
|
};
|
|
|
|
},
|
2021-08-01 19:24:47 -08:00
|
|
|
});
|
|
|
|
</script>
|
2022-01-09 07:15:23 +01:00
|
|
|
|
2022-04-02 16:35:53 -08:00
|
|
|
<style scoped>
|
|
|
|
.wrap-word {
|
|
|
|
white-space: normal;
|
|
|
|
word-wrap: break-word;
|
|
|
|
}
|
|
|
|
</style>
|