feat: 后台自定义代码 & 后端语言优先

This commit is contained in:
naiba
2024-12-15 15:08:57 +08:00
parent 830992a74d
commit 8e45f8ca6f
6 changed files with 92 additions and 65 deletions

View File

@@ -1,11 +1,7 @@
import { ModelSettingForm, ModelSettingResponse } from "@/types" import { ModelSettingForm } from "@/types"
import { FetcherMethod, fetcher } from "./api" import { FetcherMethod, fetcher } from "./api"
export const updateSettings = async (data: ModelSettingForm): Promise<void> => { export const updateSettings = async (data: ModelSettingForm): Promise<void> => {
return fetcher<void>(FetcherMethod.PATCH, `/api/v1/setting`, data) return fetcher<void>(FetcherMethod.PATCH, `/api/v1/setting`, data)
} }
export const getSettings = async (): Promise<ModelSettingResponse> => {
return fetcher<ModelSettingResponse>(FetcherMethod.GET, "/api/v1/setting", null)
}

View File

@@ -3,6 +3,5 @@ import { ModelSettingResponse } from "@/types"
import useSWR from "swr" import useSWR from "swr"
export default function useSetting() { export default function useSetting() {
const { data } = useSWR<ModelSettingResponse>("/api/v1/setting", swrFetcher) return useSWR<ModelSettingResponse>("/api/v1/setting", swrFetcher)
return data
} }

View File

@@ -22,7 +22,7 @@ const resources = {
} }
const getStoredLanguage = () => { const getStoredLanguage = () => {
return localStorage.getItem("language") || "zh-CN" return localStorage.getItem("language") || "en-US"
} }
i18n.use(initReactI18next).init({ i18n.use(initReactI18next).init({

View File

@@ -1,4 +1,3 @@
import { StrictMode } from "react"
import { createRoot } from "react-dom/client" import { createRoot } from "react-dom/client"
import { RouterProvider, createBrowserRouter } from "react-router-dom" import { RouterProvider, createBrowserRouter } from "react-router-dom"

View File

@@ -2,17 +2,68 @@ import Header from "@/components/header"
import { ThemeProvider } from "@/components/theme-provider" import { ThemeProvider } from "@/components/theme-provider"
import { Toaster } from "@/components/ui/sonner" import { Toaster } from "@/components/ui/sonner"
import useSetting from "@/hooks/useSetting" import useSetting from "@/hooks/useSetting"
import { useEffect } from "react" import i18n from "@/lib/i18n"
import { useCallback, useEffect } from "react"
import { useTranslation } from "react-i18next" import { useTranslation } from "react-i18next"
import { Outlet } from "react-router-dom" import { Outlet } from "react-router-dom"
export default function Root() { export default function Root() {
const { t } = useTranslation() const { t } = useTranslation()
const settings = useSetting() const { data: settingData, error } = useSetting()
useEffect(() => { useEffect(() => {
document.title = settings?.site_name || "哪吒监控 Nezha Monitoring" document.title = settingData?.site_name || "哪吒监控 Nezha Monitoring"
}, [settings]) }, [settingData])
const InjectContext = useCallback((content: string) => {
const tempDiv = document.createElement("div")
tempDiv.innerHTML = content
const handlers: { [key: string]: (element: HTMLElement) => void } = {
SCRIPT: (element) => {
const script = document.createElement("script")
if ((element as HTMLScriptElement).src) {
script.src = (element as HTMLScriptElement).src
} else {
script.textContent = element.textContent
}
document.body.appendChild(script)
},
STYLE: (element) => {
const style = document.createElement("style")
style.textContent = element.textContent
document.head.appendChild(style)
},
DEFAULT: (element) => {
document.body.appendChild(element)
},
}
Array.from(tempDiv.childNodes).forEach((node) => {
if (node.nodeType === Node.ELEMENT_NODE) {
const element = node as HTMLElement
; (handlers[element.tagName] || handlers.DEFAULT)(element)
} else if (node.nodeType === Node.TEXT_NODE) {
document.body.appendChild(document.createTextNode(node.textContent || ""))
}
})
}, [])
if (error) {
throw error
}
if (!settingData) {
return null
}
if (settingData?.language && !localStorage.getItem("language")) {
i18n.changeLanguage(settingData?.language)
}
if (settingData?.custom_code_dashboard) {
InjectContext(settingData?.custom_code_dashboard)
}
return ( return (
<ThemeProvider defaultTheme="dark" storageKey="vite-ui-theme"> <ThemeProvider defaultTheme="dark" storageKey="vite-ui-theme">
@@ -24,7 +75,7 @@ export default function Root() {
</div> </div>
</div> </div>
<footer className="mx-5 pb-5 text-foreground/50 font-light text-xs text-center"> <footer className="mx-5 pb-5 text-foreground/50 font-light text-xs text-center">
&copy; 2019-2024 {t("nezha")} {settings?.version} &copy; 2019-2024 {t("nezha")} {settingData?.version}
</footer> </footer>
</section> </section>
<Toaster /> <Toaster />

View File

@@ -1,4 +1,4 @@
import { getSettings, updateSettings } from "@/api/settings" import { updateSettings } from "@/api/settings"
import { SettingsTab } from "@/components/settings-tab" import { SettingsTab } from "@/components/settings-tab"
import { Button } from "@/components/ui/button" import { Button } from "@/components/ui/button"
import { Card, CardContent } from "@/components/ui/card" import { Card, CardContent } from "@/components/ui/card"
@@ -21,10 +21,11 @@ import {
SelectValue, SelectValue,
} from "@/components/ui/select" } from "@/components/ui/select"
import { Textarea } from "@/components/ui/textarea" import { Textarea } from "@/components/ui/textarea"
import useSetting from "@/hooks/useSetting"
import { asOptionalField } from "@/lib/utils" import { asOptionalField } from "@/lib/utils"
import { ModelSettingResponse, nezhaLang, settingCoverageTypes } from "@/types" import { nezhaLang, settingCoverageTypes } from "@/types"
import { zodResolver } from "@hookform/resolvers/zod" import { zodResolver } from "@hookform/resolvers/zod"
import { useEffect, useState } from "react" import { useEffect } from "react"
import { useForm } from "react-hook-form" import { useForm } from "react-hook-form"
import { useTranslation } from "react-i18next" import { useTranslation } from "react-i18next"
import { toast } from "sonner" import { toast } from "sonner"
@@ -50,49 +51,27 @@ const settingFormSchema = z.object({
export default function SettingsPage() { export default function SettingsPage() {
const { t, i18n } = useTranslation() const { t, i18n } = useTranslation()
const [config, setConfig] = useState<ModelSettingResponse>() const { data: config, mutate } = useSetting()
const [error, setError] = useState<Error>()
useEffect(() => {
if (error)
toast(t("Error"), {
description: t("Results.ErrorFetchingResource", {
error: error.message,
}),
})
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [error])
useEffect(() => {
;(async () => {
try {
const c = await getSettings()
setConfig(c)
} catch (e) {
if (e instanceof Error) setError(e)
}
})()
}, [])
const form = useForm<z.infer<typeof settingFormSchema>>({ const form = useForm<z.infer<typeof settingFormSchema>>({
resolver: zodResolver(settingFormSchema), resolver: zodResolver(settingFormSchema),
defaultValues: config defaultValues: config
? { ? {
...config, ...config,
language: config.language.replace("_", "-"), language: config.language,
site_name: config.site_name || "", site_name: config.site_name || "",
user_template: user_template:
config.user_template || config.user_template ||
Object.keys(config.frontend_templates.filter((t) => !t.is_admin) || {})[0] || Object.keys(config.frontend_templates.filter((t) => !t.is_admin) || {})[0] ||
"user-dist", "user-dist",
} }
: { : {
ip_change_notification_group_id: 0, ip_change_notification_group_id: 0,
cover: 1, cover: 1,
site_name: "", site_name: "",
language: "", language: "",
user_template: "user-dist", user_template: "user-dist",
}, },
resetOptions: { resetOptions: {
keepDefaultValues: false, keepDefaultValues: false,
}, },
@@ -107,11 +86,14 @@ export default function SettingsPage() {
const onSubmit = async (values: z.infer<typeof settingFormSchema>) => { const onSubmit = async (values: z.infer<typeof settingFormSchema>) => {
try { try {
await updateSettings(values) await updateSettings(values)
const newConfig = await getSettings() await mutate()
setConfig(newConfig)
form.reset() form.reset()
} catch (e) { } catch (e) {
if (e instanceof Error) setError(e) toast(t("Error"), {
description: t("Results.ErrorFetchingResource", {
error: e?.toString(),
}),
})
return return
} finally { } finally {
if (values.language != i18n.language) { if (values.language != i18n.language) {
@@ -237,15 +219,15 @@ export default function SettingsPage() {
{!config?.frontend_templates?.find( {!config?.frontend_templates?.find(
(t) => t.path === field.value, (t) => t.path === field.value,
)?.is_official && ( )?.is_official && (
<div className="mt-2 text-sm text-yellow-700 dark:text-yellow-200 bg-yellow-100 dark:bg-yellow-900 border border-yellow-200 dark:border-yellow-700 rounded-md p-2"> <div className="mt-2 text-sm text-yellow-700 dark:text-yellow-200 bg-yellow-100 dark:bg-yellow-900 border border-yellow-200 dark:border-yellow-700 rounded-md p-2">
<div className="font-medium text-lg mb-1"> <div className="font-medium text-lg mb-1">
{t("CommunityThemeWarning")} {t("CommunityThemeWarning")}
</div>
<div className="text-yellow-700 dark:text-yellow-200">
{t("CommunityThemeDescription")}
</div>
</div> </div>
<div className="text-yellow-700 dark:text-yellow-200"> )}
{t("CommunityThemeDescription")}
</div>
</div>
)}
</FormItem> </FormItem>
)} )}
/> />