mirror of
https://github.com/yangjian102621/geekai.git
synced 2026-04-26 13:04:30 +08:00
dev 环境下自动使用 API、代理
This commit is contained in:
@@ -6,94 +6,106 @@
|
|||||||
// * +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
|
// * +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
|
||||||
|
|
||||||
import axios from 'axios'
|
import axios from 'axios'
|
||||||
import {getAdminToken, getUserToken, removeAdminToken, removeUserToken} from "@/store/session";
|
import { getAdminToken, getUserToken, removeAdminToken, removeUserToken } from '@/store/session'
|
||||||
|
|
||||||
axios.defaults.timeout = 180000
|
axios.defaults.timeout = 180000
|
||||||
axios.defaults.baseURL = process.env.VUE_APP_API_HOST
|
// axios.defaults.baseURL = process.env.VUE_APP_API_HOST
|
||||||
axios.defaults.withCredentials = true;
|
axios.defaults.withCredentials = true
|
||||||
//axios.defaults.headers.post['Content-Type'] = 'application/json'
|
//axios.defaults.headers.post['Content-Type'] = 'application/json'
|
||||||
|
|
||||||
// HTTP拦截器
|
// HTTP拦截器
|
||||||
axios.interceptors.request.use(
|
axios.interceptors.request.use(
|
||||||
config => {
|
(config) => {
|
||||||
// set token
|
// set token
|
||||||
config.headers['Authorization'] = getUserToken();
|
config.headers['Authorization'] = getUserToken()
|
||||||
config.headers['Admin-Authorization'] = getAdminToken();
|
config.headers['Admin-Authorization'] = getAdminToken()
|
||||||
return config
|
return config
|
||||||
}, error => {
|
},
|
||||||
return Promise.reject(error)
|
(error) => {
|
||||||
})
|
return Promise.reject(error)
|
||||||
|
}
|
||||||
|
)
|
||||||
axios.interceptors.response.use(
|
axios.interceptors.response.use(
|
||||||
response => {
|
(response) => {
|
||||||
return response
|
return response
|
||||||
}, error => {
|
},
|
||||||
if (error.response.status === 401) {
|
(error) => {
|
||||||
if (error.response.request.responseURL.indexOf("/api/admin") !== -1) {
|
if (error.response.status === 401) {
|
||||||
removeAdminToken()
|
if (error.response.request.responseURL.indexOf('/api/admin') !== -1) {
|
||||||
} else {
|
removeAdminToken()
|
||||||
removeUserToken()
|
} else {
|
||||||
}
|
removeUserToken()
|
||||||
error.response.data.message = "请先登录"
|
}
|
||||||
return Promise.reject(error.response.data)
|
error.response.data.message = '请先登录'
|
||||||
}
|
return Promise.reject(error.response.data)
|
||||||
if (error.response.status === 400) {
|
}
|
||||||
return Promise.reject(new Error(error.response.data.message))
|
if (error.response.status === 400) {
|
||||||
} else {
|
return Promise.reject(new Error(error.response.data.message))
|
||||||
return Promise.reject(error)
|
} else {
|
||||||
}
|
return Promise.reject(error)
|
||||||
})
|
}
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
// send a http get request
|
// send a http get request
|
||||||
export function httpGet(url, params = {}) {
|
export function httpGet(url, params = {}) {
|
||||||
return new Promise((resolve, reject) => {
|
return new Promise((resolve, reject) => {
|
||||||
axios.get(url, {
|
axios
|
||||||
params: params
|
.get(url, {
|
||||||
}).then(response => {
|
params: params,
|
||||||
resolve(response.data)
|
})
|
||||||
}).catch(err => {
|
.then((response) => {
|
||||||
reject(err)
|
resolve(response.data)
|
||||||
})
|
})
|
||||||
})
|
.catch((err) => {
|
||||||
|
reject(err)
|
||||||
|
})
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// send a http post request
|
// send a http post request
|
||||||
export function httpPost(url, data = {}, options = {}) {
|
export function httpPost(url, data = {}, options = {}) {
|
||||||
return new Promise((resolve, reject) => {
|
return new Promise((resolve, reject) => {
|
||||||
axios.post(url, data, options).then(response => {
|
axios
|
||||||
resolve(response.data)
|
.post(url, data, options)
|
||||||
}).catch(err => {
|
.then((response) => {
|
||||||
reject(err)
|
resolve(response.data)
|
||||||
})
|
})
|
||||||
})
|
.catch((err) => {
|
||||||
|
reject(err)
|
||||||
|
})
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
export function httpDownload(url) {
|
export function httpDownload(url) {
|
||||||
return new Promise((resolve, reject) => {
|
return new Promise((resolve, reject) => {
|
||||||
axios({
|
axios({
|
||||||
method: 'GET',
|
method: 'GET',
|
||||||
url: url,
|
url: url,
|
||||||
responseType: 'blob' // 将响应类型设置为 `blob`
|
responseType: 'blob', // 将响应类型设置为 `blob`
|
||||||
}).then(response => {
|
|
||||||
resolve(response)
|
|
||||||
}).catch(err => {
|
|
||||||
reject(err)
|
|
||||||
})
|
|
||||||
})
|
})
|
||||||
|
.then((response) => {
|
||||||
|
resolve(response)
|
||||||
|
})
|
||||||
|
.catch((err) => {
|
||||||
|
reject(err)
|
||||||
|
})
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
export function httpPostDownload(url, data) {
|
export function httpPostDownload(url, data) {
|
||||||
return new Promise((resolve, reject) => {
|
return new Promise((resolve, reject) => {
|
||||||
axios({
|
axios({
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
url: url,
|
url: url,
|
||||||
data: data,
|
data: data,
|
||||||
responseType: 'blob' // 将响应类型设置为 `blob`
|
responseType: 'blob', // 将响应类型设置为 `blob`
|
||||||
}).then(response => {
|
|
||||||
resolve(response)
|
|
||||||
}).catch(err => {
|
|
||||||
reject(err)
|
|
||||||
})
|
|
||||||
})
|
})
|
||||||
|
.then((response) => {
|
||||||
|
resolve(response)
|
||||||
|
})
|
||||||
|
.catch((err) => {
|
||||||
|
reject(err)
|
||||||
|
})
|
||||||
|
})
|
||||||
}
|
}
|
||||||
@@ -8,252 +8,250 @@
|
|||||||
/**
|
/**
|
||||||
* Util lib functions
|
* Util lib functions
|
||||||
*/
|
*/
|
||||||
import { showConfirmDialog } from "vant";
|
import { showConfirmDialog } from 'vant'
|
||||||
|
|
||||||
// generate a random string
|
// generate a random string
|
||||||
export function randString(length) {
|
export function randString(length) {
|
||||||
const str = "0123456789abcdefghijklmnopqrstuvwxyz";
|
const str = '0123456789abcdefghijklmnopqrstuvwxyz'
|
||||||
const size = str.length;
|
const size = str.length
|
||||||
let buf = [];
|
let buf = []
|
||||||
for (let i = 0; i < length; i++) {
|
for (let i = 0; i < length; i++) {
|
||||||
const rand = Math.random() * size;
|
const rand = Math.random() * size
|
||||||
buf.push(str.charAt(rand));
|
buf.push(str.charAt(rand))
|
||||||
}
|
}
|
||||||
return buf.join("");
|
return buf.join('')
|
||||||
}
|
}
|
||||||
|
|
||||||
export function UUID() {
|
export function UUID() {
|
||||||
let d = new Date().getTime();
|
let d = new Date().getTime()
|
||||||
return "xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx".replace(/[xy]/g, function (c) {
|
return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, function (c) {
|
||||||
const r = (d + Math.random() * 16) % 16 | 0;
|
const r = (d + Math.random() * 16) % 16 | 0
|
||||||
d = Math.floor(d / 16);
|
d = Math.floor(d / 16)
|
||||||
return (c === "x" ? r : (r & 0x3) | 0x8).toString(16);
|
return (c === 'x' ? r : (r & 0x3) | 0x8).toString(16)
|
||||||
});
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
// 判断是否是移动设备
|
// 判断是否是移动设备
|
||||||
export function isMobile() {
|
export function isMobile() {
|
||||||
const userAgent = navigator.userAgent;
|
const userAgent = navigator.userAgent
|
||||||
const mobileRegex = /Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini|Mobile|mobile|CriOS/i;
|
const mobileRegex =
|
||||||
return mobileRegex.test(userAgent);
|
/Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini|Mobile|mobile|CriOS/i
|
||||||
|
return mobileRegex.test(userAgent)
|
||||||
}
|
}
|
||||||
|
|
||||||
// 格式化日期
|
// 格式化日期
|
||||||
export function dateFormat(timestamp, format) {
|
export function dateFormat(timestamp, format) {
|
||||||
if (!timestamp) {
|
if (!timestamp) {
|
||||||
return "";
|
return ''
|
||||||
} else if (timestamp < 9680917502) {
|
} else if (timestamp < 9680917502) {
|
||||||
timestamp = timestamp * 1000;
|
timestamp = timestamp * 1000
|
||||||
}
|
}
|
||||||
let year, month, day, HH, mm, ss;
|
let year, month, day, HH, mm, ss
|
||||||
let time = new Date(timestamp);
|
let time = new Date(timestamp)
|
||||||
let timeDate;
|
let timeDate
|
||||||
year = time.getFullYear(); // 年
|
year = time.getFullYear() // 年
|
||||||
month = time.getMonth() + 1; // 月
|
month = time.getMonth() + 1 // 月
|
||||||
day = time.getDate(); // 日
|
day = time.getDate() // 日
|
||||||
HH = time.getHours(); // 时
|
HH = time.getHours() // 时
|
||||||
mm = time.getMinutes(); // 分
|
mm = time.getMinutes() // 分
|
||||||
ss = time.getSeconds(); // 秒
|
ss = time.getSeconds() // 秒
|
||||||
|
|
||||||
month = month < 10 ? "0" + month : month;
|
month = month < 10 ? '0' + month : month
|
||||||
day = day < 10 ? "0" + day : day;
|
day = day < 10 ? '0' + day : day
|
||||||
HH = HH < 10 ? "0" + HH : HH; // 时
|
HH = HH < 10 ? '0' + HH : HH // 时
|
||||||
mm = mm < 10 ? "0" + mm : mm; // 分
|
mm = mm < 10 ? '0' + mm : mm // 分
|
||||||
ss = ss < 10 ? "0" + ss : ss; // 秒
|
ss = ss < 10 ? '0' + ss : ss // 秒
|
||||||
|
|
||||||
switch (format) {
|
switch (format) {
|
||||||
case "yyyy":
|
case 'yyyy':
|
||||||
timeDate = String(year);
|
timeDate = String(year)
|
||||||
break;
|
break
|
||||||
case "yyyy-MM":
|
case 'yyyy-MM':
|
||||||
timeDate = year + "-" + month;
|
timeDate = year + '-' + month
|
||||||
break;
|
break
|
||||||
case "yyyy-MM-dd":
|
case 'yyyy-MM-dd':
|
||||||
timeDate = year + "-" + month + "-" + day;
|
timeDate = year + '-' + month + '-' + day
|
||||||
break;
|
break
|
||||||
case "yyyy/MM/dd":
|
case 'yyyy/MM/dd':
|
||||||
timeDate = year + "/" + month + "/" + day;
|
timeDate = year + '/' + month + '/' + day
|
||||||
break;
|
break
|
||||||
case "yyyy-MM-dd HH:mm:ss":
|
case 'yyyy-MM-dd HH:mm:ss':
|
||||||
timeDate = year + "-" + month + "-" + day + " " + HH + ":" + mm + ":" + ss;
|
timeDate = year + '-' + month + '-' + day + ' ' + HH + ':' + mm + ':' + ss
|
||||||
break;
|
break
|
||||||
case "HH:mm:ss":
|
case 'HH:mm:ss':
|
||||||
timeDate = HH + ":" + mm + ":" + ss;
|
timeDate = HH + ':' + mm + ':' + ss
|
||||||
break;
|
break
|
||||||
case "MM":
|
case 'MM':
|
||||||
timeDate = String(month);
|
timeDate = String(month)
|
||||||
break;
|
break
|
||||||
default:
|
default:
|
||||||
timeDate = year + "-" + month + "-" + day + " " + HH + ":" + mm + ":" + ss;
|
timeDate = year + '-' + month + '-' + day + ' ' + HH + ':' + mm + ':' + ss
|
||||||
break;
|
break
|
||||||
}
|
}
|
||||||
return timeDate;
|
return timeDate
|
||||||
}
|
}
|
||||||
|
|
||||||
export function formatTime(time) {
|
export function formatTime(time) {
|
||||||
const minutes = Math.floor(time / 60);
|
const minutes = Math.floor(time / 60)
|
||||||
const seconds = Math.floor(time % 60);
|
const seconds = Math.floor(time % 60)
|
||||||
return `${minutes}:${seconds < 10 ? "0" : ""}${seconds}`;
|
return `${minutes}:${seconds < 10 ? '0' : ''}${seconds}`
|
||||||
}
|
}
|
||||||
|
|
||||||
// 判断数组中是否包含某个元素
|
// 判断数组中是否包含某个元素
|
||||||
export function arrayContains(array, value, compare) {
|
export function arrayContains(array, value, compare) {
|
||||||
if (!array) {
|
if (!array) {
|
||||||
return false;
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
if (typeof compare !== "function") {
|
if (typeof compare !== 'function') {
|
||||||
compare = function (v1, v2) {
|
compare = function (v1, v2) {
|
||||||
return v1 === v2;
|
return v1 === v2
|
||||||
};
|
}
|
||||||
}
|
}
|
||||||
for (let i = 0; i < array.length; i++) {
|
for (let i = 0; i < array.length; i++) {
|
||||||
if (compare(array[i], value)) {
|
if (compare(array[i], value)) {
|
||||||
return true;
|
return true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return false;
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
// 删除数组中指定的元素
|
// 删除数组中指定的元素
|
||||||
export function removeArrayItem(array, value, compare) {
|
export function removeArrayItem(array, value, compare) {
|
||||||
if (typeof compare !== "function") {
|
if (typeof compare !== 'function') {
|
||||||
compare = function (v1, v2) {
|
compare = function (v1, v2) {
|
||||||
return v1 === v2;
|
return v1 === v2
|
||||||
};
|
}
|
||||||
}
|
}
|
||||||
for (let i = 0; i < array.length; i++) {
|
for (let i = 0; i < array.length; i++) {
|
||||||
if (compare(array[i], value)) {
|
if (compare(array[i], value)) {
|
||||||
array.splice(i, 1);
|
array.splice(i, 1)
|
||||||
break;
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return array;
|
return array
|
||||||
}
|
}
|
||||||
|
|
||||||
// 渲染输入的换行符
|
// 渲染输入的换行符
|
||||||
export function renderInputText(text) {
|
export function renderInputText(text) {
|
||||||
const replaceRegex = /(\n\r|\r\n|\r|\n)/g;
|
const replaceRegex = /(\n\r|\r\n|\r|\n)/g
|
||||||
text = text || "";
|
text = text || ''
|
||||||
return text.replace(replaceRegex, "<br/>");
|
return text.replace(replaceRegex, '<br/>')
|
||||||
}
|
}
|
||||||
|
|
||||||
// 拷贝对象
|
// 拷贝对象
|
||||||
export function copyObj(origin) {
|
export function copyObj(origin) {
|
||||||
return JSON.parse(JSON.stringify(origin));
|
return JSON.parse(JSON.stringify(origin))
|
||||||
}
|
}
|
||||||
|
|
||||||
export function disabledDate(time) {
|
export function disabledDate(time) {
|
||||||
return time.getTime() < Date.now();
|
return time.getTime() < Date.now()
|
||||||
}
|
}
|
||||||
|
|
||||||
// 字符串截取
|
// 字符串截取
|
||||||
export function substr(str, length) {
|
export function substr(str, length) {
|
||||||
let result = "";
|
let result = ''
|
||||||
let count = 0;
|
let count = 0
|
||||||
|
|
||||||
for (let i = 0; i < str.length; i++) {
|
for (let i = 0; i < str.length; i++) {
|
||||||
const char = str.charAt(i);
|
const char = str.charAt(i)
|
||||||
const charCode = str.charCodeAt(i);
|
const charCode = str.charCodeAt(i)
|
||||||
|
|
||||||
// 判断字符是否为中文字符
|
// 判断字符是否为中文字符
|
||||||
if (charCode >= 0x4e00 && charCode <= 0x9fff) {
|
if (charCode >= 0x4e00 && charCode <= 0x9fff) {
|
||||||
// 中文字符算两个字符
|
// 中文字符算两个字符
|
||||||
count += 2;
|
count += 2
|
||||||
} else {
|
} else {
|
||||||
count++;
|
count++
|
||||||
}
|
}
|
||||||
|
|
||||||
if (count <= length) {
|
if (count <= length) {
|
||||||
result += char;
|
result += char
|
||||||
} else {
|
} else {
|
||||||
result += " ...";
|
result += ' ...'
|
||||||
break;
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return result;
|
return result
|
||||||
}
|
}
|
||||||
|
|
||||||
export function isImage(url) {
|
export function isImage(url) {
|
||||||
const expr = /\.(jpg|jpeg|png|gif|bmp|svg)$/i;
|
const expr = /\.(jpg|jpeg|png|gif|bmp|svg)$/i
|
||||||
return expr.test(url);
|
return expr.test(url)
|
||||||
}
|
}
|
||||||
|
|
||||||
export function processContent(content) {
|
export function processContent(content) {
|
||||||
if (!content) {
|
if (!content) {
|
||||||
return "";
|
return ''
|
||||||
}
|
}
|
||||||
|
|
||||||
// 如果是图片链接地址,则直接替换成图片标签
|
// 如果是图片链接地址,则直接替换成图片标签
|
||||||
const linkRegex = /(https?:\/\/\S+)/g;
|
const linkRegex = /(https?:\/\/\S+)/g
|
||||||
const links = content.match(linkRegex);
|
const links = content.match(linkRegex)
|
||||||
if (links) {
|
if (links) {
|
||||||
for (let link of links) {
|
for (let link of links) {
|
||||||
if (isImage(link)) {
|
if (isImage(link)) {
|
||||||
const index = content.indexOf(link);
|
const index = content.indexOf(link)
|
||||||
if (content.substring(index - 1, 2) !== "]") {
|
if (content.substring(index - 1, 2) !== ']') {
|
||||||
content = content.replace(link, "\n\n");
|
content = content.replace(link, '\n\n')
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// 处理推理标签
|
// 处理推理标签
|
||||||
if (content.includes("<think>")) {
|
if (content.includes('<think>')) {
|
||||||
content = content.replace(/<think>(.*?)<\/think>/gs, (match, content) => {
|
content = content.replace(/<think>(.*?)<\/think>/gs, (match, content) => {
|
||||||
if (content.length > 10) {
|
if (content.length > 10) {
|
||||||
return `<blockquote>${content}</blockquote>`;
|
return `<blockquote>${content}</blockquote>`
|
||||||
}
|
}
|
||||||
return "";
|
return ''
|
||||||
});
|
})
|
||||||
content = content.replace(/<think>(.*?)$/gs, (match, content) => {
|
content = content.replace(/<think>(.*?)$/gs, (match, content) => {
|
||||||
if (content.length > 10) {
|
if (content.length > 10) {
|
||||||
return `<blockquote>${content}</blockquote>`;
|
return `<blockquote>${content}</blockquote>`
|
||||||
}
|
}
|
||||||
return "";
|
return ''
|
||||||
});
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
// 支持 \[ 公式标签
|
// 支持 \[ 公式标签
|
||||||
content = content.replace(/\\\[/g, "$$").replace(/\\\]/g, "$$");
|
content = content.replace(/\\\[/g, '$$').replace(/\\\]/g, '$$')
|
||||||
content = content.replace(/\\\(\\boxed\{(\d+)\}\\\)/g, '<span class="boxed">$1</span>');
|
content = content.replace(/\\\(\\boxed\{(\d+)\}\\\)/g, '<span class="boxed">$1</span>')
|
||||||
return content;
|
return content
|
||||||
}
|
}
|
||||||
|
|
||||||
export function processPrompt(prompt) {
|
export function processPrompt(prompt) {
|
||||||
return prompt.replace(/&/g, "&").replace(/</g, "<").replace(/>/g, ">");
|
return prompt.replace(/&/g, '&').replace(/</g, '<').replace(/>/g, '>')
|
||||||
}
|
}
|
||||||
|
|
||||||
// 判断是否为微信浏览器
|
// 判断是否为微信浏览器
|
||||||
export function isWeChatBrowser() {
|
export function isWeChatBrowser() {
|
||||||
return /MicroMessenger/i.test(navigator.userAgent);
|
return /MicroMessenger/i.test(navigator.userAgent)
|
||||||
}
|
}
|
||||||
|
|
||||||
export function showLoginDialog(router) {
|
export function showLoginDialog(router) {
|
||||||
showConfirmDialog({
|
showConfirmDialog({
|
||||||
title: "登录",
|
title: '登录',
|
||||||
message: "此操作需要登录才能进行,前往登录?",
|
message: '此操作需要登录才能进行,前往登录?',
|
||||||
})
|
})
|
||||||
.then(() => {
|
.then(() => {
|
||||||
router.push("/mobile/login");
|
router.push('/mobile/login')
|
||||||
})
|
})
|
||||||
.catch(() => {
|
.catch(() => {
|
||||||
// on cancel
|
// on cancel
|
||||||
});
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
export const replaceImg = (img) => {
|
export const replaceImg = (img) => {
|
||||||
if (!img.startsWith("http")) {
|
if (img.startsWith('http')) {
|
||||||
img = `${location.protocol}//${location.host}${img}`;
|
return img
|
||||||
}
|
}
|
||||||
const devHost = process.env.VUE_APP_API_HOST;
|
return `${location.protocol}//${location.host}${img}`
|
||||||
const localhost = "http://localhost:5678";
|
}
|
||||||
if (img.includes(localhost)) {
|
|
||||||
return img?.replace(localhost, devHost);
|
// 判断是否 google 浏览器
|
||||||
}
|
export function isChrome() {
|
||||||
return img;
|
const userAgent = navigator.userAgent.toLowerCase()
|
||||||
};
|
return /chrome/.test(userAgent) && !/edg/.test(userAgent)
|
||||||
export function isChrome() {
|
|
||||||
const userAgent = navigator.userAgent.toLowerCase();
|
|
||||||
return /chrome/.test(userAgent) && !/edg/.test(userAgent);
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,7 +3,13 @@
|
|||||||
<el-tabs v-model="activeName" class="sys-tabs">
|
<el-tabs v-model="activeName" class="sys-tabs">
|
||||||
<el-tab-pane label="系统配置" name="basic">
|
<el-tab-pane label="系统配置" name="basic">
|
||||||
<div class="container">
|
<div class="container">
|
||||||
<el-form :model="system" label-width="150px" label-position="right" ref="systemFormRef" :rules="rules">
|
<el-form
|
||||||
|
:model="system"
|
||||||
|
label-width="150px"
|
||||||
|
label-position="right"
|
||||||
|
ref="systemFormRef"
|
||||||
|
:rules="rules"
|
||||||
|
>
|
||||||
<el-tabs type="border-card">
|
<el-tabs type="border-card">
|
||||||
<el-tab-pane label="基础配置">
|
<el-tab-pane label="基础配置">
|
||||||
<el-form-item label="网站标题" prop="title">
|
<el-form-item label="网站标题" prop="title">
|
||||||
@@ -18,7 +24,12 @@
|
|||||||
<el-form-item label="圆形 LOGO" prop="logo">
|
<el-form-item label="圆形 LOGO" prop="logo">
|
||||||
<el-input v-model="system['logo']" placeholder="正方形或者圆形 Logo">
|
<el-input v-model="system['logo']" placeholder="正方形或者圆形 Logo">
|
||||||
<template #append>
|
<template #append>
|
||||||
<el-upload :auto-upload="true" :show-file-list="false" @click="beforeUpload('logo')" :http-request="uploadImg">
|
<el-upload
|
||||||
|
:auto-upload="true"
|
||||||
|
:show-file-list="false"
|
||||||
|
@click="beforeUpload('logo')"
|
||||||
|
:http-request="uploadImg"
|
||||||
|
>
|
||||||
<el-icon class="uploader-icon">
|
<el-icon class="uploader-icon">
|
||||||
<UploadFilled />
|
<UploadFilled />
|
||||||
</el-icon>
|
</el-icon>
|
||||||
@@ -29,7 +40,12 @@
|
|||||||
<el-form-item label="条形 LOGO" prop="logo">
|
<el-form-item label="条形 LOGO" prop="logo">
|
||||||
<el-input v-model="system['bar_logo']" placeholder="长方形 Logo">
|
<el-input v-model="system['bar_logo']" placeholder="长方形 Logo">
|
||||||
<template #append>
|
<template #append>
|
||||||
<el-upload :auto-upload="true" :show-file-list="false" @click="beforeUpload('bar_logo')" :http-request="uploadImg">
|
<el-upload
|
||||||
|
:auto-upload="true"
|
||||||
|
:show-file-list="false"
|
||||||
|
@click="beforeUpload('bar_logo')"
|
||||||
|
:http-request="uploadImg"
|
||||||
|
>
|
||||||
<el-icon class="uploader-icon">
|
<el-icon class="uploader-icon">
|
||||||
<UploadFilled />
|
<UploadFilled />
|
||||||
</el-icon>
|
</el-icon>
|
||||||
@@ -41,20 +57,39 @@
|
|||||||
<template #label>
|
<template #label>
|
||||||
<div class="label-title">
|
<div class="label-title">
|
||||||
首页导航菜单
|
首页导航菜单
|
||||||
<el-tooltip effect="dark" content="被选中的菜单将会在首页导航栏显示" raw-content placement="right">
|
<el-tooltip
|
||||||
|
effect="dark"
|
||||||
|
content="被选中的菜单将会在首页导航栏显示"
|
||||||
|
raw-content
|
||||||
|
placement="right"
|
||||||
|
>
|
||||||
<el-icon>
|
<el-icon>
|
||||||
<InfoFilled />
|
<InfoFilled />
|
||||||
</el-icon>
|
</el-icon>
|
||||||
</el-tooltip>
|
</el-tooltip>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<el-select v-model="system['index_navs']" multiple :filterable="true" placeholder="请选择菜单,多选" style="width: 100%">
|
<el-select
|
||||||
<el-option v-for="item in menus" :key="item.id" :label="item.name" :value="item.id" />
|
v-model="system['index_navs']"
|
||||||
|
multiple
|
||||||
|
:filterable="true"
|
||||||
|
placeholder="请选择菜单,多选"
|
||||||
|
style="width: 100%"
|
||||||
|
>
|
||||||
|
<el-option
|
||||||
|
v-for="item in menus"
|
||||||
|
:key="item.id"
|
||||||
|
:label="item.name"
|
||||||
|
:value="item.id"
|
||||||
|
/>
|
||||||
</el-select>
|
</el-select>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
|
|
||||||
<el-form-item label="版权信息" prop="copyright">
|
<el-form-item label="版权信息" prop="copyright">
|
||||||
<el-input v-model="system['copyright']" placeholder="更改此选项需要获取 License 授权" />
|
<el-input
|
||||||
|
v-model="system['copyright']"
|
||||||
|
placeholder="更改此选项需要获取 License 授权"
|
||||||
|
/>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
|
|
||||||
<el-form-item label="默认昵称" prop="default_nickname">
|
<el-form-item label="默认昵称" prop="default_nickname">
|
||||||
@@ -69,7 +104,12 @@
|
|||||||
<template #label>
|
<template #label>
|
||||||
<div class="label-title">
|
<div class="label-title">
|
||||||
开放注册
|
开放注册
|
||||||
<el-tooltip effect="dark" content="关闭注册之后只能通过管理后台添加用户" raw-content placement="right">
|
<el-tooltip
|
||||||
|
effect="dark"
|
||||||
|
content="关闭注册之后只能通过管理后台添加用户"
|
||||||
|
raw-content
|
||||||
|
placement="right"
|
||||||
|
>
|
||||||
<el-icon>
|
<el-icon>
|
||||||
<InfoFilled />
|
<InfoFilled />
|
||||||
</el-icon>
|
</el-icon>
|
||||||
@@ -113,7 +153,12 @@
|
|||||||
<el-form-item label="微信客服二维码" prop="wechat_card_url">
|
<el-form-item label="微信客服二维码" prop="wechat_card_url">
|
||||||
<el-input v-model="system['wechat_card_url']" placeholder="微信客服二维码">
|
<el-input v-model="system['wechat_card_url']" placeholder="微信客服二维码">
|
||||||
<template #append>
|
<template #append>
|
||||||
<el-upload :auto-upload="true" :show-file-list="false" @click="beforeUpload('wechat_card_url')" :http-request="uploadImg">
|
<el-upload
|
||||||
|
:auto-upload="true"
|
||||||
|
:show-file-list="false"
|
||||||
|
@click="beforeUpload('wechat_card_url')"
|
||||||
|
:http-request="uploadImg"
|
||||||
|
>
|
||||||
<el-icon class="uploader-icon">
|
<el-icon class="uploader-icon">
|
||||||
<UploadFilled />
|
<UploadFilled />
|
||||||
</el-icon>
|
</el-icon>
|
||||||
@@ -125,15 +170,30 @@
|
|||||||
<template #label>
|
<template #label>
|
||||||
<div class="label-title">
|
<div class="label-title">
|
||||||
默认翻译模型
|
默认翻译模型
|
||||||
<el-tooltip effect="dark" content="选择一个默认模型来翻译提示词" raw-content placement="right">
|
<el-tooltip
|
||||||
|
effect="dark"
|
||||||
|
content="选择一个默认模型来翻译提示词"
|
||||||
|
raw-content
|
||||||
|
placement="right"
|
||||||
|
>
|
||||||
<el-icon>
|
<el-icon>
|
||||||
<InfoFilled />
|
<InfoFilled />
|
||||||
</el-icon>
|
</el-icon>
|
||||||
</el-tooltip>
|
</el-tooltip>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<el-select v-model.number="system['translate_model_id']" :filterable="true" placeholder="选择一个默认模型来翻译提示词" style="width: 100%">
|
<el-select
|
||||||
<el-option v-for="item in models" :key="item.id" :label="item.name" :value="item.id" />
|
v-model.number="system['translate_model_id']"
|
||||||
|
:filterable="true"
|
||||||
|
placeholder="选择一个默认模型来翻译提示词"
|
||||||
|
style="width: 100%"
|
||||||
|
>
|
||||||
|
<el-option
|
||||||
|
v-for="item in models"
|
||||||
|
:key="item.id"
|
||||||
|
:label="item.name"
|
||||||
|
:value="item.id"
|
||||||
|
/>
|
||||||
</el-select>
|
</el-select>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
|
|
||||||
@@ -144,8 +204,8 @@
|
|||||||
<div class="tip-input-line">
|
<div class="tip-input-line">
|
||||||
<el-input-number v-model="system['context_deep']" :min="0" :max="10" />
|
<el-input-number v-model="system['context_deep']" :min="0" :max="10" />
|
||||||
<div class="tip">
|
<div class="tip">
|
||||||
会话上下文深度:在老会话中继续会话,默认加载多少条聊天记录作为上下文。如果设置为 0
|
会话上下文深度:在老会话中继续会话,默认加载多少条聊天记录作为上下文。如果设置为
|
||||||
则不加载聊天记录,仅仅使用当前角色的上下文。该配置参数必须设置需要为偶数。
|
0 则不加载聊天记录,仅仅使用当前角色的上下文。该配置参数必须设置需要为偶数。
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
@@ -154,55 +214,98 @@
|
|||||||
<template #label>
|
<template #label>
|
||||||
<div class="label-title">
|
<div class="label-title">
|
||||||
SD反向提示词
|
SD反向提示词
|
||||||
<el-tooltip effect="dark" content="Stable-Diffusion 绘画默认反向提示词" raw-content placement="right">
|
<el-tooltip
|
||||||
|
effect="dark"
|
||||||
|
content="Stable-Diffusion 绘画默认反向提示词"
|
||||||
|
raw-content
|
||||||
|
placement="right"
|
||||||
|
>
|
||||||
<el-icon>
|
<el-icon>
|
||||||
<InfoFilled />
|
<InfoFilled />
|
||||||
</el-icon>
|
</el-icon>
|
||||||
</el-tooltip>
|
</el-tooltip>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<el-input type="textarea" :rows="2" v-model="system['sd_neg_prompt']" placeholder="" />
|
<el-input
|
||||||
|
type="textarea"
|
||||||
|
:rows="2"
|
||||||
|
v-model="system['sd_neg_prompt']"
|
||||||
|
placeholder=""
|
||||||
|
/>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
|
|
||||||
<el-form-item label="会员充值说明" prop="order_pay_timeout">
|
<el-form-item label="会员充值说明" prop="order_pay_timeout">
|
||||||
<template #label>
|
<template #label>
|
||||||
<div class="label-title">
|
<div class="label-title">
|
||||||
会员充值说明
|
会员充值说明
|
||||||
<el-tooltip effect="dark" content="会员充值页面的充值说明文字" raw-content placement="right">
|
<el-tooltip
|
||||||
|
effect="dark"
|
||||||
|
content="会员充值页面的充值说明文字"
|
||||||
|
raw-content
|
||||||
|
placement="right"
|
||||||
|
>
|
||||||
<el-icon>
|
<el-icon>
|
||||||
<InfoFilled />
|
<InfoFilled />
|
||||||
</el-icon>
|
</el-icon>
|
||||||
</el-tooltip>
|
</el-tooltip>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<el-input type="textarea" :rows="2" v-model="system['vip_info_text']" placeholder="" />
|
<el-input
|
||||||
|
type="textarea"
|
||||||
|
:rows="2"
|
||||||
|
v-model="system['vip_info_text']"
|
||||||
|
placeholder=""
|
||||||
|
/>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
|
|
||||||
<el-form-item label="MJ默认API模式" prop="mj_mode">
|
<el-form-item label="MJ默认API模式" prop="mj_mode">
|
||||||
<el-select v-model="system['mj_mode']" placeholder="请选择模式">
|
<el-select v-model="system['mj_mode']" placeholder="请选择模式">
|
||||||
<el-option v-for="item in mjModels" :value="item.value" :label="item.name" :key="item.value">{{ item.name }} </el-option>
|
<el-option
|
||||||
|
v-for="item in mjModels"
|
||||||
|
:value="item.value"
|
||||||
|
:label="item.name"
|
||||||
|
:key="item.value"
|
||||||
|
>{{ item.name }}
|
||||||
|
</el-option>
|
||||||
</el-select>
|
</el-select>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
<el-form-item label="上传文件限制" prop="max_file_size">
|
<el-form-item label="上传文件限制" prop="max_file_size">
|
||||||
<el-input v-model.number="system['max_file_size']" placeholder="最大上传文件大小,单位:MB" />
|
<el-input
|
||||||
|
v-model.number="system['max_file_size']"
|
||||||
|
placeholder="最大上传文件大小,单位:MB"
|
||||||
|
/>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
</el-tab-pane>
|
</el-tab-pane>
|
||||||
|
|
||||||
<el-tab-pane label="算力配置">
|
<el-tab-pane label="算力配置">
|
||||||
<el-form-item label="注册赠送算力" prop="init_power">
|
<el-form-item label="注册赠送算力" prop="init_power">
|
||||||
<el-input v-model.number="system['init_power']" placeholder="新用户注册赠送算力" />
|
<el-input
|
||||||
|
v-model.number="system['init_power']"
|
||||||
|
placeholder="新用户注册赠送算力"
|
||||||
|
/>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
<el-form-item label="邀请赠送算力" prop="invite_power">
|
<el-form-item label="邀请赠送算力" prop="invite_power">
|
||||||
<el-input v-model.number="system['invite_power']" placeholder="邀请新用户注册赠送算力" />
|
<el-input
|
||||||
|
v-model.number="system['invite_power']"
|
||||||
|
placeholder="邀请新用户注册赠送算力"
|
||||||
|
/>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
<el-form-item label="VIP每月赠送算力" prop="vip_month_power">
|
<el-form-item label="VIP每月赠送算力" prop="vip_month_power">
|
||||||
<el-input v-model.number="system['vip_month_power']" placeholder="VIP用户每月赠送算力" />
|
<el-input
|
||||||
|
v-model.number="system['vip_month_power']"
|
||||||
|
placeholder="VIP用户每月赠送算力"
|
||||||
|
/>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
<el-form-item>
|
<el-form-item>
|
||||||
<template #label>
|
<template #label>
|
||||||
<div class="label-title">
|
<div class="label-title">
|
||||||
签到赠送算力
|
签到赠送算力
|
||||||
<el-tooltip effect="dark" content="每日签到赠送算力" raw-content placement="right">
|
<el-tooltip
|
||||||
|
effect="dark"
|
||||||
|
content="每日签到赠送算力"
|
||||||
|
raw-content
|
||||||
|
placement="right"
|
||||||
|
>
|
||||||
<el-icon>
|
<el-icon>
|
||||||
<InfoFilled />
|
<InfoFilled />
|
||||||
</el-icon>
|
</el-icon>
|
||||||
@@ -215,7 +318,12 @@
|
|||||||
<template #label>
|
<template #label>
|
||||||
<div class="label-title">
|
<div class="label-title">
|
||||||
MJ绘图算力
|
MJ绘图算力
|
||||||
<el-tooltip effect="dark" content="使用MidJourney画一张图消耗算力" raw-content placement="right">
|
<el-tooltip
|
||||||
|
effect="dark"
|
||||||
|
content="使用MidJourney画一张图消耗算力"
|
||||||
|
raw-content
|
||||||
|
placement="right"
|
||||||
|
>
|
||||||
<el-icon>
|
<el-icon>
|
||||||
<InfoFilled />
|
<InfoFilled />
|
||||||
</el-icon>
|
</el-icon>
|
||||||
@@ -226,32 +334,54 @@
|
|||||||
</el-form-item>
|
</el-form-item>
|
||||||
|
|
||||||
<el-form-item label="Stable-Diffusion算力" prop="sd_power">
|
<el-form-item label="Stable-Diffusion算力" prop="sd_power">
|
||||||
<el-input v-model.number="system['sd_power']" placeholder="使用Stable-Diffusion画一张图消耗算力" />
|
<el-input
|
||||||
|
v-model.number="system['sd_power']"
|
||||||
|
placeholder="使用Stable-Diffusion画一张图消耗算力"
|
||||||
|
/>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
<el-form-item>
|
<el-form-item>
|
||||||
<template #label>
|
<template #label>
|
||||||
<div class="label-title">
|
<div class="label-title">
|
||||||
DALL-E-3算力
|
DALL-E-3算力
|
||||||
<el-tooltip effect="dark" content="使用DALL-E-3画一张图消耗算力" raw-content placement="right">
|
<el-tooltip
|
||||||
|
effect="dark"
|
||||||
|
content="使用DALL-E-3画一张图消耗算力"
|
||||||
|
raw-content
|
||||||
|
placement="right"
|
||||||
|
>
|
||||||
<el-icon>
|
<el-icon>
|
||||||
<InfoFilled />
|
<InfoFilled />
|
||||||
</el-icon>
|
</el-icon>
|
||||||
</el-tooltip>
|
</el-tooltip>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<el-input v-model.number="system['dall_power']" placeholder="使用DALL-E-3画一张图消耗算力" />
|
<el-input
|
||||||
|
v-model.number="system['dall_power']"
|
||||||
|
placeholder="使用DALL-E-3画一张图消耗算力"
|
||||||
|
/>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
<el-form-item label="Suno 算力" prop="suno_power">
|
<el-form-item label="Suno 算力" prop="suno_power">
|
||||||
<el-input v-model.number="system['suno_power']" placeholder="使用 Suno 生成一首音乐消耗算力" />
|
<el-input
|
||||||
|
v-model.number="system['suno_power']"
|
||||||
|
placeholder="使用 Suno 生成一首音乐消耗算力"
|
||||||
|
/>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
<el-form-item label="Luma 算力" prop="luma_power">
|
<el-form-item label="Luma 算力" prop="luma_power">
|
||||||
<el-input v-model.number="system['luma_power']" placeholder="使用 Luma 生成一段视频消耗算力" />
|
<el-input
|
||||||
|
v-model.number="system['luma_power']"
|
||||||
|
placeholder="使用 Luma 生成一段视频消耗算力"
|
||||||
|
/>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
<el-form-item>
|
<el-form-item>
|
||||||
<template #label>
|
<template #label>
|
||||||
<div class="label-title">
|
<div class="label-title">
|
||||||
可灵算力
|
可灵算力
|
||||||
<el-tooltip effect="dark" content="可灵每个模型价格不一样,具体请参考:https://api.geekai.pro/models" raw-content placement="right">
|
<el-tooltip
|
||||||
|
effect="dark"
|
||||||
|
content="可灵每个模型价格不一样,具体请参考:https://api.geekai.pro/models"
|
||||||
|
raw-content
|
||||||
|
placement="right"
|
||||||
|
>
|
||||||
<el-icon>
|
<el-icon>
|
||||||
<InfoFilled />
|
<InfoFilled />
|
||||||
</el-icon>
|
</el-icon>
|
||||||
@@ -259,7 +389,11 @@
|
|||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<el-row :gutter="20" v-if="system['keling_powers']">
|
<el-row :gutter="20" v-if="system['keling_powers']">
|
||||||
<el-col :span="6" v-for="[key] in Object.entries(system['keling_powers'])" :key="key">
|
<el-col
|
||||||
|
:span="6"
|
||||||
|
v-for="[key] in Object.entries(system['keling_powers'])"
|
||||||
|
:key="key"
|
||||||
|
>
|
||||||
<el-form-item :label="key" label-position="left">
|
<el-form-item :label="key" label-position="left">
|
||||||
<el-input v-model.number="system['keling_powers'][key]" size="small" />
|
<el-input v-model.number="system['keling_powers'][key]" size="small" />
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
@@ -270,7 +404,12 @@
|
|||||||
<template #label>
|
<template #label>
|
||||||
<div class="label-title">
|
<div class="label-title">
|
||||||
高级语音算力
|
高级语音算力
|
||||||
<el-tooltip effect="dark" content="使用一次 OpenAI 高级语音对话消耗的算力" raw-content placement="right">
|
<el-tooltip
|
||||||
|
effect="dark"
|
||||||
|
content="使用一次 OpenAI 高级语音对话消耗的算力"
|
||||||
|
raw-content
|
||||||
|
placement="right"
|
||||||
|
>
|
||||||
<el-icon>
|
<el-icon>
|
||||||
<InfoFilled />
|
<InfoFilled />
|
||||||
</el-icon>
|
</el-icon>
|
||||||
@@ -283,7 +422,12 @@
|
|||||||
<template #label>
|
<template #label>
|
||||||
<div class="label-title">
|
<div class="label-title">
|
||||||
提示词算力
|
提示词算力
|
||||||
<el-tooltip effect="dark" content="生成AI绘图提示词,歌词,视频描述消耗的算力" raw-content placement="right">
|
<el-tooltip
|
||||||
|
effect="dark"
|
||||||
|
content="生成AI绘图提示词,歌词,视频描述消耗的算力"
|
||||||
|
raw-content
|
||||||
|
placement="right"
|
||||||
|
>
|
||||||
<el-icon>
|
<el-icon>
|
||||||
<InfoFilled />
|
<InfoFilled />
|
||||||
</el-icon>
|
</el-icon>
|
||||||
@@ -304,7 +448,12 @@
|
|||||||
</div>
|
</div>
|
||||||
</el-tab-pane>
|
</el-tab-pane>
|
||||||
<el-tab-pane label="公告配置" name="notice">
|
<el-tab-pane label="公告配置" name="notice">
|
||||||
<md-editor class="mgb20" v-model="notice" :theme="store.theme" @on-upload-img="onUploadImg" />
|
<md-editor
|
||||||
|
class="mgb20"
|
||||||
|
v-model="notice"
|
||||||
|
:theme="store.theme"
|
||||||
|
@on-upload-img="onUploadImg"
|
||||||
|
/>
|
||||||
<el-form-item>
|
<el-form-item>
|
||||||
<div style="padding-top: 10px; margin-left: 150px">
|
<div style="padding-top: 10px; margin-left: 150px">
|
||||||
<el-button type="primary" @click="save('notice')">保存</el-button>
|
<el-button type="primary" @click="save('notice')">保存</el-button>
|
||||||
@@ -313,7 +462,12 @@
|
|||||||
</el-tab-pane>
|
</el-tab-pane>
|
||||||
|
|
||||||
<el-tab-pane label="用户协议" name="agreement">
|
<el-tab-pane label="用户协议" name="agreement">
|
||||||
<md-editor class="mgb20" v-model="agreement" :theme="store.theme" @on-upload-img="onUploadImg" />
|
<md-editor
|
||||||
|
class="mgb20"
|
||||||
|
v-model="agreement"
|
||||||
|
:theme="store.theme"
|
||||||
|
@on-upload-img="onUploadImg"
|
||||||
|
/>
|
||||||
<el-form-item>
|
<el-form-item>
|
||||||
<div style="padding-top: 10px; margin-left: 150px">
|
<div style="padding-top: 10px; margin-left: 150px">
|
||||||
<el-button type="primary" @click="save('agreement')">保存</el-button>
|
<el-button type="primary" @click="save('agreement')">保存</el-button>
|
||||||
@@ -322,7 +476,12 @@
|
|||||||
</el-tab-pane>
|
</el-tab-pane>
|
||||||
|
|
||||||
<el-tab-pane label="隐私声明" name="privacy">
|
<el-tab-pane label="隐私声明" name="privacy">
|
||||||
<md-editor class="mgb20" v-model="privacy" :theme="store.theme" @on-upload-img="onUploadImg" />
|
<md-editor
|
||||||
|
class="mgb20"
|
||||||
|
v-model="privacy"
|
||||||
|
:theme="store.theme"
|
||||||
|
@on-upload-img="onUploadImg"
|
||||||
|
/>
|
||||||
<el-form-item>
|
<el-form-item>
|
||||||
<div style="padding-top: 10px; margin-left: 150px">
|
<div style="padding-top: 10px; margin-left: 150px">
|
||||||
<el-button type="primary" @click="save('privacy')">保存</el-button>
|
<el-button type="primary" @click="save('privacy')">保存</el-button>
|
||||||
@@ -331,7 +490,12 @@
|
|||||||
</el-tab-pane>
|
</el-tab-pane>
|
||||||
|
|
||||||
<el-tab-pane label="思维导图" name="mark_map">
|
<el-tab-pane label="思维导图" name="mark_map">
|
||||||
<md-editor class="mgb20" :theme="store.theme" v-model="system['mark_map_text']" @on-upload-img="onUploadImg" />
|
<md-editor
|
||||||
|
class="mgb20"
|
||||||
|
:theme="store.theme"
|
||||||
|
v-model="system['mark_map_text']"
|
||||||
|
@on-upload-img="onUploadImg"
|
||||||
|
/>
|
||||||
<el-form-item>
|
<el-form-item>
|
||||||
<div style="padding-top: 10px; margin-left: 150px">
|
<div style="padding-top: 10px; margin-left: 150px">
|
||||||
<el-button type="primary" @click="save('system')">保存</el-button>
|
<el-button type="primary" @click="save('system')">保存</el-button>
|
||||||
@@ -344,7 +508,13 @@
|
|||||||
|
|
||||||
<el-tab-pane label="授权激活" name="license">
|
<el-tab-pane label="授权激活" name="license">
|
||||||
<div class="container">
|
<div class="container">
|
||||||
<el-descriptions v-if="license.is_active" class="margin-top" title="已授权信息" :column="1" border>
|
<el-descriptions
|
||||||
|
v-if="license.is_active"
|
||||||
|
class="margin-top"
|
||||||
|
title="已授权信息"
|
||||||
|
:column="1"
|
||||||
|
border
|
||||||
|
>
|
||||||
<el-descriptions-item>
|
<el-descriptions-item>
|
||||||
<template #label>
|
<template #label>
|
||||||
<div class="cell-item">License Key</div>
|
<div class="cell-item">License Key</div>
|
||||||
@@ -400,8 +570,8 @@
|
|||||||
<el-tab-pane label="修复数据" name="fixData">
|
<el-tab-pane label="修复数据" name="fixData">
|
||||||
<div class="container">
|
<div class="container">
|
||||||
<p class="text">
|
<p class="text">
|
||||||
有些版本升级的时候更新了数据库的结构,比如字段名字改了,需要把之前的字段的值转移到其他字段,这些无法通过简单的 SQL
|
有些版本升级的时候更新了数据库的结构,比如字段名字改了,需要把之前的字段的值转移到其他字段,这些无法通过简单的
|
||||||
语句可以实现的,需要手动写程序修正数据。
|
SQL 语句可以实现的,需要手动写程序修正数据。
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
<!-- <p class="text">当前版本 v4.1.4 需要修正用户数据,增加了 mobile 和 email 字段,需要把之前用手机号或者邮箱注册的用户的 username 字段数据初始化到 mobile 或者 email 字段。另外,需要把订单的支付渠道从名字称修正为 key。</p>-->
|
<!-- <p class="text">当前版本 v4.1.4 需要修正用户数据,增加了 mobile 和 email 字段,需要把之前用手机号或者邮箱注册的用户的 username 字段数据初始化到 mobile 或者 email 字段。另外,需要把订单的支付渠道从名字称修正为 key。</p>-->
|
||||||
@@ -418,183 +588,196 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup>
|
<script setup>
|
||||||
import { onMounted, reactive, ref } from "vue";
|
import { onMounted, reactive, ref } from 'vue'
|
||||||
import { httpGet, httpPost } from "@/utils/http";
|
import { httpGet, httpPost } from '@/utils/http'
|
||||||
import Compressor from "compressorjs";
|
import Compressor from 'compressorjs'
|
||||||
import { ElMessage, ElMessageBox } from "element-plus";
|
import { ElMessage, ElMessageBox } from 'element-plus'
|
||||||
import { CloseBold, InfoFilled, Select, UploadFilled } from "@element-plus/icons-vue";
|
import { CloseBold, InfoFilled, Select, UploadFilled } from '@element-plus/icons-vue'
|
||||||
import MdEditor from "md-editor-v3";
|
import MdEditor from 'md-editor-v3'
|
||||||
import "md-editor-v3/lib/style.css";
|
import 'md-editor-v3/lib/style.css'
|
||||||
import Menu from "@/views/admin/Menu.vue";
|
import Menu from '@/views/admin/Menu.vue'
|
||||||
import { copyObj, dateFormat } from "@/utils/libs";
|
import { copyObj, dateFormat } from '@/utils/libs'
|
||||||
import ItemsInput from "@/components/ui/ItemsInput.vue";
|
import ItemsInput from '@/components/ui/ItemsInput.vue'
|
||||||
import { useSharedStore } from "@/store/sharedata";
|
import { useSharedStore } from '@/store/sharedata'
|
||||||
|
|
||||||
const activeName = ref("basic");
|
const activeName = ref('basic')
|
||||||
const system = ref({ models: [] });
|
const system = ref({ models: [] })
|
||||||
const configBak = ref({});
|
const configBak = ref({})
|
||||||
const loading = ref(true);
|
const loading = ref(true)
|
||||||
const systemFormRef = ref(null);
|
const systemFormRef = ref(null)
|
||||||
const models = ref([]);
|
const models = ref([])
|
||||||
const notice = ref("");
|
const notice = ref('')
|
||||||
const agreement = ref("");
|
const agreement = ref('')
|
||||||
const privacy = ref("");
|
const privacy = ref('')
|
||||||
const license = ref({ is_active: false });
|
const license = ref({ is_active: false })
|
||||||
const menus = ref([]);
|
const menus = ref([])
|
||||||
const mjModels = ref([
|
const mjModels = ref([
|
||||||
{ name: "慢速(Relax)", value: "relax" },
|
{ name: '慢速(Relax)', value: 'relax' },
|
||||||
{ name: "快速(Fast)", value: "fast" },
|
{ name: '快速(Fast)', value: 'fast' },
|
||||||
{ name: "急速(Turbo)", value: "turbo" },
|
{ name: '急速(Turbo)', value: 'turbo' },
|
||||||
]);
|
])
|
||||||
const store = useSharedStore();
|
const store = useSharedStore()
|
||||||
|
|
||||||
onMounted(() => {
|
onMounted(() => {
|
||||||
// 加载系统配置
|
// 加载系统配置
|
||||||
httpGet("/api/admin/config/get?key=system")
|
httpGet('/api/admin/config/get?key=system')
|
||||||
.then((res) => {
|
.then((res) => {
|
||||||
system.value = res.data;
|
system.value = res.data
|
||||||
system.value.keling_powers = system.value.keling_powers || {
|
system.value.keling_powers = system.value.keling_powers || {
|
||||||
"kling-v1-6_std_5": 240,
|
'kling-v1-6_std_5': 240,
|
||||||
"kling-v1-6_std_10": 480,
|
'kling-v1-6_std_10': 480,
|
||||||
"kling-v1-6_pro_5": 420,
|
'kling-v1-6_pro_5': 420,
|
||||||
"kling-v1-6_pro_10": 840,
|
'kling-v1-6_pro_10': 840,
|
||||||
"kling-v1-5_std_5": 240,
|
'kling-v1-5_std_5': 240,
|
||||||
"kling-v1-5_std_10": 480,
|
'kling-v1-5_std_10': 480,
|
||||||
"kling-v1-5_pro_5": 420,
|
'kling-v1-5_pro_5': 420,
|
||||||
"kling-v1-5_pro_10": 840,
|
'kling-v1-5_pro_10': 840,
|
||||||
"kling-v1_std_5": 120,
|
'kling-v1_std_5': 120,
|
||||||
"kling-v1_std_10": 240,
|
'kling-v1_std_10': 240,
|
||||||
"kling-v1_pro_5": 420,
|
'kling-v1_pro_5': 420,
|
||||||
"kling-v1_pro_10": 840,
|
'kling-v1_pro_10': 840,
|
||||||
};
|
}
|
||||||
configBak.value = copyObj(system.value);
|
configBak.value = copyObj(system.value)
|
||||||
})
|
})
|
||||||
.catch((e) => {
|
.catch((e) => {
|
||||||
ElMessage.error("加载系统配置失败: " + e.message);
|
ElMessage.error('加载系统配置失败: ' + e.message)
|
||||||
});
|
})
|
||||||
// 加载聊天配置
|
// 加载聊天配置
|
||||||
httpGet("/api/admin/config/get?key=notice")
|
httpGet('/api/admin/config/get?key=notice')
|
||||||
.then((res) => {
|
.then((res) => {
|
||||||
notice.value = res.data["content"];
|
notice.value = res.data['content']
|
||||||
})
|
})
|
||||||
.catch((e) => {
|
.catch((e) => {
|
||||||
ElMessage.error("公告信息失败: " + e.message);
|
ElMessage.error('公告信息失败: ' + e.message)
|
||||||
});
|
})
|
||||||
|
|
||||||
// 加载用户协议
|
// 加载用户协议
|
||||||
httpGet("/api/admin/config/get?key=agreement")
|
httpGet('/api/admin/config/get?key=agreement')
|
||||||
.then((res) => {
|
.then((res) => {
|
||||||
agreement.value = res.data["content"] || '';
|
agreement.value = res.data['content'] || ''
|
||||||
})
|
})
|
||||||
.catch((e) => {
|
.catch((e) => {
|
||||||
ElMessage.error("加载用户协议失败: " + e.message);
|
console.warn('加载用户协议失败: ' + e.message)
|
||||||
});
|
})
|
||||||
|
|
||||||
// 加载隐私政策
|
// 加载隐私政策
|
||||||
httpGet("/api/admin/config/get?key=privacy")
|
httpGet('/api/admin/config/get?key=privacy')
|
||||||
.then((res) => {
|
.then((res) => {
|
||||||
privacy.value = res.data["content"] || '';
|
privacy.value = res.data['content'] || ''
|
||||||
})
|
})
|
||||||
.catch((e) => {
|
.catch((e) => {
|
||||||
ElMessage.error("加载隐私政策失败: " + e.message);
|
console.warn('加载隐私政策失败: ' + e.message)
|
||||||
});
|
})
|
||||||
|
|
||||||
httpGet("/api/admin/model/list")
|
httpGet('/api/admin/model/list')
|
||||||
.then((res) => {
|
.then((res) => {
|
||||||
models.value = res.data;
|
models.value = res.data
|
||||||
loading.value = false;
|
loading.value = false
|
||||||
})
|
})
|
||||||
.catch((e) => {
|
.catch((e) => {
|
||||||
ElMessage.error("获取模型失败:" + e.message);
|
ElMessage.error('获取模型失败:' + e.message)
|
||||||
});
|
})
|
||||||
|
|
||||||
httpGet("/api/admin/menu/list")
|
httpGet('/api/admin/menu/list')
|
||||||
.then((res) => {
|
.then((res) => {
|
||||||
menus.value = res.data;
|
menus.value = res.data
|
||||||
})
|
})
|
||||||
.catch((e) => {
|
.catch((e) => {
|
||||||
ElMessage.error("获取模型失败:" + e.message);
|
ElMessage.error('获取模型失败:' + e.message)
|
||||||
});
|
})
|
||||||
|
|
||||||
fetchLicense();
|
fetchLicense()
|
||||||
});
|
})
|
||||||
|
|
||||||
const fetchLicense = () => {
|
const fetchLicense = () => {
|
||||||
httpGet("/api/admin/config/license")
|
httpGet('/api/admin/config/license')
|
||||||
.then((res) => {
|
.then((res) => {
|
||||||
license.value = res.data;
|
license.value = res.data
|
||||||
})
|
})
|
||||||
.catch((e) => {
|
.catch((e) => {
|
||||||
ElMessage.error("获取 License 失败:" + e.message);
|
ElMessage.error('获取 License 失败:' + e.message)
|
||||||
});
|
})
|
||||||
};
|
}
|
||||||
|
|
||||||
const rules = reactive({
|
const rules = reactive({
|
||||||
title: [{ required: true, message: "请输入网站标题", trigger: "blur" }],
|
title: [{ required: true, message: '请输入网站标题', trigger: 'blur' }],
|
||||||
admin_title: [{ required: true, message: "请输入控制台标题", trigger: "blur" }],
|
admin_title: [{ required: true, message: '请输入控制台标题', trigger: 'blur' }],
|
||||||
init_chat_calls: [{ required: true, message: "请输入赠送对话次数", trigger: "blur" }],
|
init_chat_calls: [{ required: true, message: '请输入赠送对话次数', trigger: 'blur' }],
|
||||||
user_img_calls: [{ required: true, message: "请输入赠送绘图次数", trigger: "blur" }],
|
user_img_calls: [{ required: true, message: '请输入赠送绘图次数', trigger: 'blur' }],
|
||||||
});
|
})
|
||||||
const save = function (key) {
|
const save = function (key) {
|
||||||
if (key === "system") {
|
if (key === 'system') {
|
||||||
systemFormRef.value.validate((valid) => {
|
systemFormRef.value.validate((valid) => {
|
||||||
if (valid) {
|
if (valid) {
|
||||||
httpPost("/api/admin/config/update", { key: key, config: system.value, config_bak: configBak.value })
|
httpPost('/api/admin/config/update', {
|
||||||
|
key: key,
|
||||||
|
config: system.value,
|
||||||
|
config_bak: configBak.value,
|
||||||
|
})
|
||||||
.then(() => {
|
.then(() => {
|
||||||
ElMessage.success("操作成功!");
|
ElMessage.success('操作成功!')
|
||||||
})
|
})
|
||||||
.catch((e) => {
|
.catch((e) => {
|
||||||
ElMessage.error("操作失败:" + e.message);
|
ElMessage.error('操作失败:' + e.message)
|
||||||
});
|
})
|
||||||
}
|
}
|
||||||
});
|
})
|
||||||
} else if (key === "notice") {
|
} else if (key === 'notice') {
|
||||||
httpPost("/api/admin/config/update", { key: key, config: { content: notice.value, updated: true } })
|
httpPost('/api/admin/config/update', {
|
||||||
|
key: key,
|
||||||
|
config: { content: notice.value, updated: true },
|
||||||
|
})
|
||||||
.then(() => {
|
.then(() => {
|
||||||
ElMessage.success("操作成功!");
|
ElMessage.success('操作成功!')
|
||||||
})
|
})
|
||||||
.catch((e) => {
|
.catch((e) => {
|
||||||
ElMessage.error("操作失败:" + e.message);
|
ElMessage.error('操作失败:' + e.message)
|
||||||
});
|
})
|
||||||
} else if (key === "agreement") {
|
} else if (key === 'agreement') {
|
||||||
httpPost("/api/admin/config/update", { key: key, config: { content: agreement.value, updated: true } })
|
httpPost('/api/admin/config/update', {
|
||||||
|
key: key,
|
||||||
|
config: { content: agreement.value, updated: true },
|
||||||
|
})
|
||||||
.then(() => {
|
.then(() => {
|
||||||
ElMessage.success("操作成功!");
|
ElMessage.success('操作成功!')
|
||||||
})
|
})
|
||||||
.catch((e) => {
|
.catch((e) => {
|
||||||
ElMessage.error("操作失败:" + e.message);
|
ElMessage.error('操作失败:' + e.message)
|
||||||
});
|
})
|
||||||
} else if (key === "privacy") {
|
} else if (key === 'privacy') {
|
||||||
httpPost("/api/admin/config/update", { key: key, config: { content: privacy.value, updated: true } })
|
httpPost('/api/admin/config/update', {
|
||||||
|
key: key,
|
||||||
|
config: { content: privacy.value, updated: true },
|
||||||
|
})
|
||||||
.then(() => {
|
.then(() => {
|
||||||
ElMessage.success("操作成功!");
|
ElMessage.success('操作成功!')
|
||||||
})
|
})
|
||||||
.catch((e) => {
|
.catch((e) => {
|
||||||
ElMessage.error("操作失败:" + e.message);
|
ElMessage.error('操作失败:' + e.message)
|
||||||
});
|
})
|
||||||
}
|
}
|
||||||
};
|
}
|
||||||
|
|
||||||
// 激活授权
|
// 激活授权
|
||||||
const licenseKey = ref("");
|
const licenseKey = ref('')
|
||||||
const active = () => {
|
const active = () => {
|
||||||
if (licenseKey.value === "") {
|
if (licenseKey.value === '') {
|
||||||
return ElMessage.error("请输入授权码");
|
return ElMessage.error('请输入授权码')
|
||||||
}
|
}
|
||||||
httpPost("/api/admin/config/active", { license: licenseKey.value })
|
httpPost('/api/admin/config/active', { license: licenseKey.value })
|
||||||
.then((res) => {
|
.then((res) => {
|
||||||
ElMessage.success("授权成功,机器编码为:" + res.data);
|
ElMessage.success('授权成功,机器编码为:' + res.data)
|
||||||
fetchLicense();
|
fetchLicense()
|
||||||
})
|
})
|
||||||
.catch((e) => {
|
.catch((e) => {
|
||||||
ElMessage.error(e.message);
|
ElMessage.error(e.message)
|
||||||
});
|
})
|
||||||
};
|
}
|
||||||
|
|
||||||
const configKey = ref("");
|
const configKey = ref('')
|
||||||
const beforeUpload = (key) => {
|
const beforeUpload = (key) => {
|
||||||
configKey.value = key;
|
configKey.value = key
|
||||||
};
|
}
|
||||||
|
|
||||||
// 图片上传
|
// 图片上传
|
||||||
const uploadImg = (file) => {
|
const uploadImg = (file) => {
|
||||||
@@ -602,65 +785,65 @@ const uploadImg = (file) => {
|
|||||||
new Compressor(file.file, {
|
new Compressor(file.file, {
|
||||||
quality: 0.6,
|
quality: 0.6,
|
||||||
success(result) {
|
success(result) {
|
||||||
const formData = new FormData();
|
const formData = new FormData()
|
||||||
formData.append("file", result, result.name);
|
formData.append('file', result, result.name)
|
||||||
// 执行上传操作
|
// 执行上传操作
|
||||||
httpPost("/api/admin/upload", formData)
|
httpPost('/api/admin/upload', formData)
|
||||||
.then((res) => {
|
.then((res) => {
|
||||||
system.value[configKey.value] = res.data.url;
|
system.value[configKey.value] = res.data.url
|
||||||
ElMessage.success("上传成功");
|
ElMessage.success('上传成功')
|
||||||
})
|
})
|
||||||
.catch((e) => {
|
.catch((e) => {
|
||||||
ElMessage.error("上传失败:" + e.message);
|
ElMessage.error('上传失败:' + e.message)
|
||||||
});
|
})
|
||||||
},
|
},
|
||||||
error(e) {
|
error(e) {
|
||||||
ElMessage.error("上传失败:" + e.message);
|
ElMessage.error('上传失败:' + e.message)
|
||||||
},
|
},
|
||||||
});
|
})
|
||||||
};
|
}
|
||||||
|
|
||||||
// 编辑期文件上传处理
|
// 编辑期文件上传处理
|
||||||
const onUploadImg = (files, callback) => {
|
const onUploadImg = (files, callback) => {
|
||||||
Promise.all(
|
Promise.all(
|
||||||
files.map((file) => {
|
files.map((file) => {
|
||||||
return new Promise((rev, rej) => {
|
return new Promise((rev, rej) => {
|
||||||
const formData = new FormData();
|
const formData = new FormData()
|
||||||
formData.append("file", file, file.name);
|
formData.append('file', file, file.name)
|
||||||
// 执行上传操作
|
// 执行上传操作
|
||||||
httpPost("/api/admin/upload", formData)
|
httpPost('/api/admin/upload', formData)
|
||||||
.then((res) => rev(res))
|
.then((res) => rev(res))
|
||||||
.catch((error) => rej(error));
|
.catch((error) => rej(error))
|
||||||
});
|
})
|
||||||
})
|
})
|
||||||
)
|
)
|
||||||
.then((res) => {
|
.then((res) => {
|
||||||
ElMessage.success({ message: "上传成功", duration: 500 });
|
ElMessage.success({ message: '上传成功', duration: 500 })
|
||||||
callback(res.map((item) => item.data.url));
|
callback(res.map((item) => item.data.url))
|
||||||
})
|
})
|
||||||
.catch((e) => {
|
.catch((e) => {
|
||||||
ElMessage.error("图片上传失败:" + e.message);
|
ElMessage.error('图片上传失败:' + e.message)
|
||||||
});
|
})
|
||||||
};
|
}
|
||||||
|
|
||||||
const fixData = () => {
|
const fixData = () => {
|
||||||
ElMessageBox.confirm("在修复数据前,请先备份好数据库,以免数据丢失!是否继续操作?", "警告", {
|
ElMessageBox.confirm('在修复数据前,请先备份好数据库,以免数据丢失!是否继续操作?', '警告', {
|
||||||
confirmButtonText: "确定",
|
confirmButtonText: '确定',
|
||||||
cancelButtonText: "取消",
|
cancelButtonText: '取消',
|
||||||
type: "warning",
|
type: 'warning',
|
||||||
}).then(() => {
|
}).then(() => {
|
||||||
loading.value = true;
|
loading.value = true
|
||||||
httpGet("/api/admin/config/fixData")
|
httpGet('/api/admin/config/fixData')
|
||||||
.then(() => {
|
.then(() => {
|
||||||
ElMessage.success("数据修复成功");
|
ElMessage.success('数据修复成功')
|
||||||
loading.value = false;
|
loading.value = false
|
||||||
})
|
})
|
||||||
.catch((e) => {
|
.catch((e) => {
|
||||||
loading.value = false;
|
loading.value = false
|
||||||
ElMessage.error("数据修复失败:" + e.message);
|
ElMessage.error('数据修复失败:' + e.message)
|
||||||
});
|
})
|
||||||
});
|
})
|
||||||
};
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="stylus" scoped>
|
<style lang="stylus" scoped>
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
const { defineConfig } = require("@vue/cli-service");
|
const { defineConfig } = require('@vue/cli-service')
|
||||||
const path = require("path");
|
const path = require('path')
|
||||||
let webpack = require("webpack");
|
let webpack = require('webpack')
|
||||||
module.exports = defineConfig({
|
module.exports = defineConfig({
|
||||||
transpileDependencies: true,
|
transpileDependencies: true,
|
||||||
lintOnSave: false, //关闭eslint校验
|
lintOnSave: false, //关闭eslint校验
|
||||||
@@ -14,26 +14,31 @@ module.exports = defineConfig({
|
|||||||
plugins: [new webpack.optimize.MinChunkSizePlugin({ minChunkSize: 10000 })],
|
plugins: [new webpack.optimize.MinChunkSizePlugin({ minChunkSize: 10000 })],
|
||||||
resolve: {
|
resolve: {
|
||||||
alias: {
|
alias: {
|
||||||
"@": path.resolve(__dirname, "src"),
|
'@': path.resolve(__dirname, 'src'),
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
publicPath: "/",
|
publicPath: '/',
|
||||||
|
|
||||||
outputDir: "dist",
|
outputDir: 'dist',
|
||||||
crossorigin: "anonymous",
|
crossorigin: 'anonymous',
|
||||||
devServer: {
|
devServer: {
|
||||||
client: {
|
client: {
|
||||||
overlay: false // 关闭错误覆盖层
|
overlay: false, // 关闭错误覆盖层
|
||||||
},
|
},
|
||||||
allowedHosts: "all",
|
allowedHosts: 'all',
|
||||||
port: 8888,
|
port: 8888,
|
||||||
proxy: {
|
proxy: {
|
||||||
"/static/upload/": {
|
'/api': {
|
||||||
|
target: process.env.VUE_APP_API_HOST,
|
||||||
|
changeOrigin: true,
|
||||||
|
ws: true,
|
||||||
|
},
|
||||||
|
'/static/upload/': {
|
||||||
target: process.env.VUE_APP_API_HOST,
|
target: process.env.VUE_APP_API_HOST,
|
||||||
changeOrigin: true,
|
changeOrigin: true,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
});
|
})
|
||||||
|
|||||||
Reference in New Issue
Block a user