自定义源列显示源版本号、作者名字,优化自定义源脚本数据存储

pull/1761/head
lyswhut 2023-11-25 12:19:36 +08:00
parent ff98e06afa
commit 3ba51fd6fe
7 changed files with 94 additions and 20 deletions

View File

@ -15,6 +15,7 @@
### 优化
- 更新zh-tw翻译
- 自定义源列显示源版本号、作者名字
### 修复

View File

@ -13,7 +13,7 @@ declare namespace LX {
type UserApiSources = Record<LX.Source, UserApiSourceInfo>
interface UserApiInfo {
interface UserApiInfoFull {
id: string
name: string
description: string
@ -25,6 +25,8 @@ declare namespace LX {
sources?: UserApiSources
}
type UserApiInfo = Omit<UserApiInfoFull, 'script'>
interface UserApiStatus {
status: boolean
message?: string

View File

@ -1,2 +1,2 @@
export const userApis: LX.UserApi.UserApiInfo[] = []
export const userApis: LX.UserApi.UserApiInfoFull[] = []

View File

@ -6,9 +6,9 @@ let userApiId: string | null
export const getApiList = getUserApis
export const importApi = (script: string): LX.UserApi.ImportUserApi => {
export const importApi = async(script: string): Promise<LX.UserApi.ImportUserApi> => {
return {
apiInfo: handleImportApi(script),
apiInfo: await handleImportApi(script),
apiList: getUserApis(),
}
}

View File

@ -5,6 +5,7 @@ import path from 'node:path'
import { openDevTools as handleOpenDevTools } from '@main/utils'
import { encodePath } from '@common/utils/electron'
import USER_API_RENDERER_EVENT_NAME from './rendererEvent/name'
import { getScript } from './utils'
let browserWindow: Electron.BrowserWindow | null = null
@ -91,8 +92,8 @@ export const createWindow = async(userApi: LX.UserApi.UserApiInfo) => {
// const randomNum = Math.random().toString().substring(2, 10)
await browserWindow.loadURL('data:text/html;charset=UTF-8,' + encodeURIComponent(html))
browserWindow.on('ready-to-show', () => {
sendEvent(USER_API_RENDERER_EVENT_NAME.initEnv, userApi)
browserWindow.on('ready-to-show', async() => {
sendEvent(USER_API_RENDERER_EVENT_NAME.initEnv, { ...userApi, script: await getScript(userApi.id) })
})
// global.modules.userApiWindow.loadFile(join(dir, 'renderer/user-api.html'))
@ -106,7 +107,7 @@ export const closeWindow = async() => {
browserWindow.webContents.session.clearStorageData(),
browserWindow.webContents.session.clearCache(),
])
browserWindow.destroy()
browserWindow?.destroy()
browserWindow = null
}

View File

@ -1,20 +1,52 @@
import { userApis as defaultUserApis } from './config'
import { STORE_NAMES } from '@common/constants'
import getStore from '@main/utils/store'
import zlib from 'node:zlib'
let userApis: LX.UserApi.UserApiInfo[] | null
let scripts = new Map<string, string>()
const saveData = () => {
getStore(STORE_NAMES.USER_API).set('userApis', userApis!.map(api => {
return {
...api,
script: scripts.get(api.id),
}
}))
}
export const getUserApis = (): LX.UserApi.UserApiInfo[] => {
const electronStore_userApi = getStore(STORE_NAMES.USER_API)
if (userApis) return userApis
userApis = electronStore_userApi.get('userApis') as LX.UserApi.UserApiInfo[]
if (!userApis) {
userApis = defaultUserApis
const electronStore_userApi = getStore(STORE_NAMES.USER_API)
let infoFull = electronStore_userApi.get('userApis') as LX.UserApi.UserApiInfoFull[]
let requiredUpdate = false
if (infoFull) {
for (let i = 0; i < infoFull.length; i++) {
const api = infoFull[i]
if (api.version != null) continue
requiredUpdate ||= true
try {
infoFull.splice(i, 1, {
...parseScriptInfo(api.script),
...api,
})
} catch (e) {
infoFull.splice(i, 1)
i--
}
}
} else {
infoFull = defaultUserApis
electronStore_userApi.set('userApis', userApis)
}
for (const api of userApis) {
userApis = infoFull.map(api => {
if (api.allowShowUpdateAlert == null) api.allowShowUpdateAlert = false
}
const { script, ...info } = api
scripts.set(api.id, script)
return info
})
if (requiredUpdate) saveData()
return userApis
}
@ -46,22 +78,47 @@ const matchInfo = (scriptInfo: string) => {
return infos as Record<keyof typeof INFO_NAMES, string>
}
export const importApi = (script: string): LX.UserApi.UserApiInfo => {
const parseScriptInfo = (script: string) => {
const result = /^\/\*[\S|\s]+?\*\//.exec(script)
if (!result) throw new Error('无效的自定义源文件')
let scriptInfo = matchInfo(result[0])
scriptInfo.name ||= `user_api_${new Date().toLocaleString()}`
return scriptInfo
}
const deflateScript = async(script: string) => new Promise<string>((resolve, reject) => {
zlib.deflate(Buffer.from(script, 'utf8'), (err, buf) => {
if (err) {
reject(err)
return
}
resolve('gz_' + buf.toString('base64'))
})
})
const inflateScript = async(script: string) => new Promise<string>((resolve, reject) => {
if (script.startsWith('gz_')) {
zlib.inflate(Buffer.from(script.substring(3), 'base64'), (err, buf) => {
if (err) {
reject(err)
return
}
resolve(buf.toString('utf8'))
})
} else resolve(script)
})
export const importApi = async(scriptRaw: string): Promise<LX.UserApi.UserApiInfo> => {
let scriptInfo = parseScriptInfo(scriptRaw)
const apiInfo = {
id: `user_api_${Math.random().toString().substring(2, 5)}_${Date.now()}`,
...scriptInfo,
script,
allowShowUpdateAlert: true,
}
userApis ??= []
userApis.push(apiInfo)
getStore(STORE_NAMES.USER_API).set('userApis', userApis)
const script = await deflateScript(scriptRaw)
scripts.set(apiInfo.id, script)
saveData()
return apiInfo
}
@ -69,16 +126,21 @@ export const removeApi = (ids: string[]) => {
if (!userApis) return
for (let index = userApis.length - 1; index > -1; index--) {
if (ids.includes(userApis[index].id)) {
scripts.delete(userApis[index].id)
userApis.splice(index, 1)
ids.splice(index, 1)
}
}
getStore(STORE_NAMES.USER_API).set('userApis', userApis)
saveData()
}
export const setAllowShowUpdateAlert = (id: string, enable: boolean) => {
const targetApi = userApis?.find(api => api.id == id)
if (!targetApi) return
targetApi.allowShowUpdateAlert = enable
getStore(STORE_NAMES.USER_API).set('userApis', userApis)
saveData()
}
export const getScript = async(id: string) => {
return inflateScript(scripts.get(id) ?? '')
}

View File

@ -5,7 +5,10 @@ material-modal(:show="modelValue" bg-close teleport="#view" @close="handleClose"
ul.scroll(v-if="apiList.length" :class="$style.content")
li(v-for="(api, index) in apiList" :key="api.id" :class="[$style.listItem, {[$style.active]: appSetting['common.apiSource'] == api.id}]")
div(:class="$style.listLeft")
h3 {{ api.name }}
h3
| {{ api.name }}
span(v-if="api.version") {{ /^\d/.test(api.version) ? `v${api.version}` : api.version }}
span(v-if="api.author") {{ api.author }}
p {{ api.description }}
div
base-checkbox(:id="`user_api_${api.id}`" v-model="api.allowShowUpdateAlert" :class="$style.checkbox" :label="$t('user_api__allow_show_update_alert')" @change="handleChangeAllowUpdateAlert(api, $event)")
@ -145,7 +148,7 @@ export default {
flex-flow: row nowrap;
align-items: center;
transition: background-color 0.2s ease;
padding: 10px;
padding: 15px 10px;
border-radius: @radius-border;
&:hover {
background-color: var(--color-primary-background-hover);
@ -157,6 +160,11 @@ export default {
font-size: 15px;
color: var(--color-font);
word-break: break-all;
span {
font-size: 12px;
color: var(--color-font-label);
margin-left: 6px;
}
}
p {
margin-top: 5px;