添加语言 languageSchema
This commit is contained in:
parent
10580a0d63
commit
ddaa02da3d
43
languageSchema.json
Normal file
43
languageSchema.json
Normal file
@ -0,0 +1,43 @@
|
|||||||
|
{
|
||||||
|
"$schema": "http://json-schema.org/draft-07/schema#",
|
||||||
|
"$id": "https://example.com/language-config.schema.json",
|
||||||
|
"title": "Language Configuration",
|
||||||
|
"description": "Schema for validating language configuration files",
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"languages": {
|
||||||
|
"type": "array",
|
||||||
|
"minItems": 1,
|
||||||
|
"items": {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"title": {
|
||||||
|
"type": "string",
|
||||||
|
"description": "Display name of the language",
|
||||||
|
"pattern": "^[\\w\\s-]+$"
|
||||||
|
},
|
||||||
|
"id": {
|
||||||
|
"type": "string",
|
||||||
|
"description": "Language identifier in locale format",
|
||||||
|
"pattern": "^[a-z]{2}-[a-z]{2}$"
|
||||||
|
},
|
||||||
|
"file": {
|
||||||
|
"type": "string",
|
||||||
|
"description": "Language file name",
|
||||||
|
"pattern": "^[A-Z]{2}_[A-Z]{2}\\.json$"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"required": [
|
||||||
|
"title",
|
||||||
|
"id",
|
||||||
|
"file"
|
||||||
|
],
|
||||||
|
"additionalProperties": false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"required": [
|
||||||
|
"languages"
|
||||||
|
],
|
||||||
|
"additionalProperties": false
|
||||||
|
}
|
@ -26,6 +26,7 @@
|
|||||||
"unplugin-auto-import": "^19.3.0",
|
"unplugin-auto-import": "^19.3.0",
|
||||||
"unplugin-vue-components": "^28.8.0",
|
"unplugin-vue-components": "^28.8.0",
|
||||||
"vue": "^3.5.17",
|
"vue": "^3.5.17",
|
||||||
|
"vue-i18n": "12.0.0-alpha.2",
|
||||||
"vue-router": "^4.5.1"
|
"vue-router": "^4.5.1"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
|
@ -1,7 +1,18 @@
|
|||||||
<template>
|
<template>
|
||||||
|
<n-config-provider :theme="useSettingStore.theme === 'dark' ? darkTheme : lightTheme" abstract>
|
||||||
|
<n-loading-bar-provider>
|
||||||
|
<n-message-provider>
|
||||||
<Index />
|
<Index />
|
||||||
|
</n-message-provider>
|
||||||
|
</n-loading-bar-provider>
|
||||||
|
</n-config-provider>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts" setup>
|
<script lang="ts" setup>
|
||||||
import Index from '@/layout/index.vue';
|
import Index from '@/layout/index.vue';
|
||||||
|
import { darkTheme, lightTheme } from 'naive-ui';
|
||||||
|
|
||||||
|
import { UseSettingStore } from '@/plugin';
|
||||||
|
|
||||||
|
const useSettingStore = UseSettingStore();
|
||||||
</script>
|
</script>
|
||||||
|
@ -4,7 +4,7 @@
|
|||||||
<meta charset="UTF-8">
|
<meta charset="UTF-8">
|
||||||
<link href="./assets/index.svg" rel="icon">
|
<link href="./assets/index.svg" rel="icon">
|
||||||
<meta content="width=device-width, initial-scale=0.8" name="viewport">
|
<meta content="width=device-width, initial-scale=0.8" name="viewport">
|
||||||
<title>Vite App</title>
|
<title>pan-disk</title>
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
<main id="app"></main>
|
<main id="app"></main>
|
||||||
|
15
web-src/language/index.json
Normal file
15
web-src/language/index.json
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
{
|
||||||
|
"$schema": "https://git.mingliqiye.com/mingliqiye/pan-disk/raw/branch/master/languageSchema.json",
|
||||||
|
"languages": [
|
||||||
|
{
|
||||||
|
"title": "中国-汉语",
|
||||||
|
"id": "zh-cn",
|
||||||
|
"file": "ZH_CN.json"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": "US-English",
|
||||||
|
"id": "en-us",
|
||||||
|
"file": "EN_US.json"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
@ -19,12 +19,22 @@
|
|||||||
主页
|
主页
|
||||||
<div class="after"></div>
|
<div class="after"></div>
|
||||||
</div>
|
</div>
|
||||||
<div class="router-item">
|
<div
|
||||||
|
v-bind:class="{
|
||||||
|
'router-item-atc': router.currentRoute.value.path.startsWith('/file'),
|
||||||
|
'router-item': true,
|
||||||
|
}"
|
||||||
|
@click="openFile()">
|
||||||
<Icon icon="material-symbols:folder-rounded" />
|
<Icon icon="material-symbols:folder-rounded" />
|
||||||
文件
|
文件
|
||||||
<div class="after"></div>
|
<div class="after"></div>
|
||||||
</div>
|
</div>
|
||||||
<div class="router-item">
|
<div
|
||||||
|
v-bind:class="{
|
||||||
|
'router-item-atc': router.currentRoute.value.name === 'user',
|
||||||
|
'router-item': true,
|
||||||
|
}"
|
||||||
|
@click="router.push({ name: 'user' })">
|
||||||
<Icon icon="material-symbols:person-rounded" />
|
<Icon icon="material-symbols:person-rounded" />
|
||||||
用户
|
用户
|
||||||
<div class="after"></div>
|
<div class="after"></div>
|
||||||
@ -68,7 +78,7 @@
|
|||||||
<script lang="ts" setup>
|
<script lang="ts" setup>
|
||||||
import Icon from '@/components/Icon.vue';
|
import Icon from '@/components/Icon.vue';
|
||||||
import { UseBoolRef } from '@/util';
|
import { UseBoolRef } from '@/util';
|
||||||
import { router, UseAuthStore, UseSettingStore } from '@/plugin';
|
import { message, router, UseAuthStore, UseSettingStore } from '@/plugin';
|
||||||
import { getSysInfo } from '@/api/system.ts';
|
import { getSysInfo } from '@/api/system.ts';
|
||||||
|
|
||||||
const useSettingStore = UseSettingStore();
|
const useSettingStore = UseSettingStore();
|
||||||
@ -78,8 +88,22 @@ getSysInfo().then((r) => {
|
|||||||
const data = r.json().data;
|
const data = r.json().data;
|
||||||
useSettingStore.appName = data.appName;
|
useSettingStore.appName = data.appName;
|
||||||
useSettingStore.appVersion = data.appVersion;
|
useSettingStore.appVersion = data.appVersion;
|
||||||
|
document.title = data.appName;
|
||||||
});
|
});
|
||||||
useAuthStore.woIsMe();
|
useAuthStore.woIsMe();
|
||||||
|
|
||||||
|
function openFile() {
|
||||||
|
if (useAuthStore.isLogin) {
|
||||||
|
router.push({
|
||||||
|
path: `/file/${useAuthStore.username}`,
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
message.error('请先登录');
|
||||||
|
router.push({
|
||||||
|
path: `/login`,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
</script>
|
</script>
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
html.dark {
|
html.dark {
|
||||||
|
@ -1,15 +1,12 @@
|
|||||||
<template>
|
<template>
|
||||||
<Head />
|
<Head />
|
||||||
|
<n-back-top />
|
||||||
|
<router-view v-slot="{ Component }">
|
||||||
<div style="margin: 10px">
|
<div style="margin: 10px">
|
||||||
<RouterView />
|
<component :is="Component" />
|
||||||
</div>
|
</div>
|
||||||
|
</router-view>
|
||||||
</template>
|
</template>
|
||||||
<script lang="ts" setup>
|
<script lang="ts" setup>
|
||||||
import { RouterView } from 'vue-router';
|
|
||||||
import { useMessage } from 'naive-ui';
|
|
||||||
import Head from '@/layout/Head.vue';
|
import Head from '@/layout/Head.vue';
|
||||||
import { setMessage } from '@/plugin';
|
|
||||||
|
|
||||||
setMessage(useMessage());
|
|
||||||
</script>
|
</script>
|
||||||
<style scoped></style>
|
|
||||||
|
@ -1,15 +1,11 @@
|
|||||||
<template>
|
<template>
|
||||||
<n-config-provider :theme="useSettingStore.theme === 'dark' ? darkTheme : lightTheme" abstract>
|
|
||||||
<n-message-provider>
|
|
||||||
<RouterView />
|
<RouterView />
|
||||||
</n-message-provider>
|
|
||||||
</n-config-provider>
|
|
||||||
</template>
|
</template>
|
||||||
<script lang="ts" setup>
|
<script lang="ts" setup>
|
||||||
import { UseSettingStore } from '@/plugin';
|
import { useLoadingBar, useMessage } from 'naive-ui';
|
||||||
import { RouterView } from 'vue-router';
|
import { setLoadingBar, setMessage } from '@/plugin';
|
||||||
import { darkTheme, lightTheme } from 'naive-ui';
|
|
||||||
|
|
||||||
const useSettingStore = UseSettingStore();
|
setMessage(useMessage());
|
||||||
|
setLoadingBar(useLoadingBar());
|
||||||
</script>
|
</script>
|
||||||
<style scoped></style>
|
<style scoped></style>
|
||||||
|
12
web-src/plugin/i18n/index.ts
Normal file
12
web-src/plugin/i18n/index.ts
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
import { createI18n } from 'vue-i18n';
|
||||||
|
|
||||||
|
export type languageType = { [key: string]: string | languageType };
|
||||||
|
const messages: languageType | any = {};
|
||||||
|
|
||||||
|
const i18n = createI18n({
|
||||||
|
legacy: false, // 使用 Composition API
|
||||||
|
locale: 'zh', // 默认语言
|
||||||
|
messages,
|
||||||
|
});
|
||||||
|
|
||||||
|
export default i18n;
|
@ -1,11 +1,17 @@
|
|||||||
import type { MessageApiInjection } from 'naive-ui/es/message/src/MessageProvider';
|
import type { MessageApiInjection } from 'naive-ui/es/message/src/MessageProvider';
|
||||||
|
import type { LoadingBarApiInjection } from 'naive-ui/es/loading-bar/src/LoadingBarProvider';
|
||||||
|
|
||||||
export * from './router';
|
export * from './router';
|
||||||
export * from './stores';
|
export * from './stores';
|
||||||
export * from './alova';
|
export * from './alova';
|
||||||
|
|
||||||
export let message: MessageApiInjection;
|
export let message: MessageApiInjection;
|
||||||
|
export let loadingBar: LoadingBarApiInjection;
|
||||||
|
|
||||||
export function setMessage(messages: MessageApiInjection) {
|
export function setMessage(messages: MessageApiInjection) {
|
||||||
message = messages;
|
message = messages;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function setLoadingBar(loadingBars: LoadingBarApiInjection) {
|
||||||
|
loadingBar = loadingBars;
|
||||||
|
}
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
import { createRouter, createWebHistory } from 'vue-router';
|
import { createRouter, createWebHistory } from 'vue-router';
|
||||||
|
import { loadingBar } from '@/plugin';
|
||||||
|
|
||||||
const router = createRouter({
|
const router = createRouter({
|
||||||
history: createWebHistory(import.meta.env.BASE_URL),
|
history: createWebHistory(import.meta.env.BASE_URL),
|
||||||
@ -12,6 +13,26 @@ const router = createRouter({
|
|||||||
name: 'home',
|
name: 'home',
|
||||||
component: () => import('@/views/HomeView.vue'),
|
component: () => import('@/views/HomeView.vue'),
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
path: '/file/:pathMatch(.*)*',
|
||||||
|
name: 'file',
|
||||||
|
component: () => import('@/views/FileView.vue'),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: '/user',
|
||||||
|
name: 'user',
|
||||||
|
component: () => import('@/views/UserView.vue'),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: '/login',
|
||||||
|
name: 'login',
|
||||||
|
component: () => import('@/views/loginView.vue'),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: '/register',
|
||||||
|
name: 'register',
|
||||||
|
component: () => import('@/views/RegisterView.vue'),
|
||||||
|
},
|
||||||
{
|
{
|
||||||
path: '/swagger',
|
path: '/swagger',
|
||||||
name: 'api',
|
name: 'api',
|
||||||
@ -22,4 +43,10 @@ const router = createRouter({
|
|||||||
],
|
],
|
||||||
});
|
});
|
||||||
|
|
||||||
|
router.beforeEach((to, from, next) => {
|
||||||
|
loadingBar.start();
|
||||||
|
next();
|
||||||
|
setTimeout(() => loadingBar.finish(), 100);
|
||||||
|
});
|
||||||
|
|
||||||
export { router };
|
export { router };
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
import { defineStore } from 'pinia';
|
import { defineStore } from 'pinia';
|
||||||
import { login, woIsMe } from '@/api';
|
import { login, logout, woIsMe } from '@/api';
|
||||||
import { LocalStorageApi } from '@/util';
|
import { LocalStorageApi } from '@/util';
|
||||||
|
|
||||||
export const UseAuthStore = defineStore('auth', {
|
export const UseAuthStore = defineStore('auth', {
|
||||||
@ -39,6 +39,9 @@ export const UseAuthStore = defineStore('auth', {
|
|||||||
})
|
})
|
||||||
.catch(() => this.$reset());
|
.catch(() => this.$reset());
|
||||||
},
|
},
|
||||||
|
logout() {
|
||||||
|
logout().then(() => this.$reset());
|
||||||
|
},
|
||||||
},
|
},
|
||||||
persist: {
|
persist: {
|
||||||
storage: LocalStorageApi.StorageApi,
|
storage: LocalStorageApi.StorageApi,
|
||||||
|
9
web-src/views/FileView.vue
Normal file
9
web-src/views/FileView.vue
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
<template>
|
||||||
|
<div></div>
|
||||||
|
</template>
|
||||||
|
<script lang="ts" setup>
|
||||||
|
import { router } from '@/plugin';
|
||||||
|
|
||||||
|
console.log(router.currentRoute);
|
||||||
|
</script>
|
||||||
|
<style scoped></style>
|
@ -1,10 +1,11 @@
|
|||||||
<template>
|
<template>
|
||||||
|
<div>
|
||||||
<NButton @click="useAuthStore.login('admin', 'admin')">登录</NButton>
|
<NButton @click="useAuthStore.login('admin', 'admin')">登录</NButton>
|
||||||
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<script lang="ts" setup>
|
<script lang="ts" setup>
|
||||||
import { UseAuthStore } from '@/plugin';
|
import { UseAuthStore } from '@/plugin';
|
||||||
|
|
||||||
const useAuthStore = UseAuthStore();
|
const useAuthStore = UseAuthStore();
|
||||||
console.log(useAuthStore);
|
|
||||||
</script>
|
</script>
|
||||||
<style scoped></style>
|
<style scoped></style>
|
||||||
|
3
web-src/views/RegisterView.vue
Normal file
3
web-src/views/RegisterView.vue
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
<template></template>
|
||||||
|
<script lang="ts" setup></script>
|
||||||
|
<style scoped></style>
|
19
web-src/views/UserView.vue
Normal file
19
web-src/views/UserView.vue
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
<n-button @click="useAuthStore.logout()">注销</n-button>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
<script lang="ts" setup>
|
||||||
|
import { router, UseAuthStore } from '@/plugin';
|
||||||
|
import { onMounted } from 'vue';
|
||||||
|
|
||||||
|
const useAuthStore = UseAuthStore();
|
||||||
|
|
||||||
|
onMounted(() => {
|
||||||
|
if (!useAuthStore.isLogin)
|
||||||
|
router.push({
|
||||||
|
name: 'login',
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
<style scoped></style>
|
25
web-src/views/loginView.vue
Normal file
25
web-src/views/loginView.vue
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
<template>
|
||||||
|
<div c-c f style="height: calc(100vh - 66px - 20px)">
|
||||||
|
<n-spin :delay="1000" :show="false">
|
||||||
|
<n-card
|
||||||
|
:segmented="{
|
||||||
|
content: true,
|
||||||
|
footer: 'soft',
|
||||||
|
}"
|
||||||
|
style="width: 430px"
|
||||||
|
title="登录">
|
||||||
|
<n-form border label-align="right" label-placement="left" label-width="auto">
|
||||||
|
<n-form-item label="用户名">
|
||||||
|
<n-input />
|
||||||
|
</n-form-item>
|
||||||
|
<n-form-item label="密码">
|
||||||
|
<n-input />
|
||||||
|
</n-form-item>
|
||||||
|
</n-form>
|
||||||
|
<n-button block secondary strong type="primary">登录</n-button>
|
||||||
|
</n-card>
|
||||||
|
</n-spin>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
<script lang="ts" setup></script>
|
||||||
|
<style scoped></style>
|
Loading…
x
Reference in New Issue
Block a user