1e22d5e4fS猫头猫import {getStorage, setStorage} from '@/utils/storage'; 2e22d5e4fS猫头猫import produce from 'immer'; 3e22d5e4fS猫头猫import {useEffect, useState} from 'react'; 4e22d5e4fS猫头猫 5e22d5e4fS猫头猫type ExceptionType = IMusic.IMusicItem | IMusic.IMusicItem[]; 6e22d5e4fS猫头猫interface IConfig { 7e22d5e4fS猫头猫 setting: { 8e22d5e4fS猫头猫 basic: { 9cfa0fc07S猫头猫 /** 使用移动网络播放 */ 10cfa0fc07S猫头猫 useCelluarNetworkPlay: boolean; 11cfa0fc07S猫头猫 /** 使用移动网络下载 */ 12cfa0fc07S猫头猫 useCelluarNetworkDownload: boolean; 13e22d5e4fS猫头猫 /** 最大同时下载 */ 14e22d5e4fS猫头猫 maxDownload: number | string; 15cbb92902S猫头猫 /** 点击专辑单曲 */ 16cbb92902S猫头猫 clickMusicInAlbum: '播放专辑' | '播放单曲'; 17e22d5e4fS猫头猫 /** 同时播放 */ 18e22d5e4fS猫头猫 notInterrupt: boolean; 19e22d5e4fS猫头猫 /** 播放错误时自动停止 */ 20e22d5e4fS猫头猫 autoStopWhenError: boolean; 21cfa0fc07S猫头猫 /** 插件缓存策略 todo */ 22cfa0fc07S猫头猫 pluginCacheControl: string; 23cfa0fc07S猫头猫 /** 最大音乐缓存 */ 24cfa0fc07S猫头猫 maxCacheSize: number; 25e22d5e4fS猫头猫 26e22d5e4fS猫头猫 debug: { 27e22d5e4fS猫头猫 errorLog: boolean; 28e22d5e4fS猫头猫 traceLog: boolean; 29e22d5e4fS猫头猫 }; 30e22d5e4fS猫头猫 }; 31e22d5e4fS猫头猫 32e22d5e4fS猫头猫 /** 主题 */ 33e22d5e4fS猫头猫 theme: { 34e22d5e4fS猫头猫 mode: 'light' | 'dark' | 'custom-light' | 'custom-dark'; 35e22d5e4fS猫头猫 background: string; 36e22d5e4fS猫头猫 backgroundOpacity: number; 37e22d5e4fS猫头猫 backgroundBlur: number; 38e22d5e4fS猫头猫 colors: { 39e22d5e4fS猫头猫 primary: string; 40e22d5e4fS猫头猫 secondary: string; 41e22d5e4fS猫头猫 textHighlight: string; 42e22d5e4fS猫头猫 pageBackground: string; 43f415a09cS猫头猫 accent: string; 44e22d5e4fS猫头猫 }; 45e22d5e4fS猫头猫 }; 46*8b7ddca8S猫头猫 47*8b7ddca8S猫头猫 plugin: { 48*8b7ddca8S猫头猫 subscribeUrl: string; 49*8b7ddca8S猫头猫 }; 50e22d5e4fS猫头猫 }; 51e22d5e4fS猫头猫 status: { 52e22d5e4fS猫头猫 music: { 53e22d5e4fS猫头猫 /** 当前的音乐 */ 54e22d5e4fS猫头猫 track: IMusic.IMusicItem; 55e22d5e4fS猫头猫 /** 进度 */ 56e22d5e4fS猫头猫 progress: number; 57e22d5e4fS猫头猫 /** 模式 */ 58e22d5e4fS猫头猫 repeatMode: string; 59e22d5e4fS猫头猫 /** 列表 */ 60e22d5e4fS猫头猫 musicQueue: IMusic.IMusicItem[]; 61e22d5e4fS猫头猫 }; 62e22d5e4fS猫头猫 }; 63e22d5e4fS猫头猫} 64e22d5e4fS猫头猫 65e22d5e4fS猫头猫type FilterType<T, R = never> = T extends Record<string | number, any> 66e22d5e4fS猫头猫 ? { 67e22d5e4fS猫头猫 [P in keyof T]: T[P] extends ExceptionType ? R : T[P]; 68e22d5e4fS猫头猫 } 69e22d5e4fS猫头猫 : never; 70e22d5e4fS猫头猫 71e22d5e4fS猫头猫type KeyPaths< 72e22d5e4fS猫头猫 T extends object, 73e22d5e4fS猫头猫 Root extends boolean = true, 74e22d5e4fS猫头猫 R = FilterType<T, ''>, 75e22d5e4fS猫头猫 K extends keyof R = keyof R, 76e22d5e4fS猫头猫> = K extends string | number 77e22d5e4fS猫头猫 ? 78e22d5e4fS猫头猫 | (Root extends true ? `${K}` : `.${K}`) 79e22d5e4fS猫头猫 | (R[K] extends Record<string | number, any> 804060c00aS猫头猫 ? `${Root extends true ? `${K}` : `.${K}`}${KeyPaths< 814060c00aS猫头猫 R[K], 824060c00aS猫头猫 false 834060c00aS猫头猫 >}` 84e22d5e4fS猫头猫 : never) 85e22d5e4fS猫头猫 : never; 86e22d5e4fS猫头猫 87e22d5e4fS猫头猫type KeyPathValue<T extends object, K extends string> = T extends Record< 88e22d5e4fS猫头猫 string | number, 89e22d5e4fS猫头猫 any 90e22d5e4fS猫头猫> 91e22d5e4fS猫头猫 ? K extends `${infer S}.${infer R}` 92e22d5e4fS猫头猫 ? KeyPathValue<T[S], R> 93e22d5e4fS猫头猫 : T[K] 94e22d5e4fS猫头猫 : never; 95e22d5e4fS猫头猫 96e22d5e4fS猫头猫type KeyPathsObj< 97e22d5e4fS猫头猫 T extends object, 98e22d5e4fS猫头猫 K extends string = KeyPaths<T>, 99e22d5e4fS猫头猫> = T extends Record<string | number, any> 100e22d5e4fS猫头猫 ? { 101e22d5e4fS猫头猫 [R in K]: KeyPathValue<T, R>; 102e22d5e4fS猫头猫 } 103e22d5e4fS猫头猫 : never; 104e22d5e4fS猫头猫 105e22d5e4fS猫头猫type DeepPartial<T> = { 106e22d5e4fS猫头猫 [K in keyof T]?: T[K] extends Record<string | number, any> 107e22d5e4fS猫头猫 ? T[K] extends ExceptionType 108e22d5e4fS猫头猫 ? T[K] 109e22d5e4fS猫头猫 : DeepPartial<T[K]> 110e22d5e4fS猫头猫 : T[K]; 111e22d5e4fS猫头猫}; 112e22d5e4fS猫头猫 113cfa0fc07S猫头猫export type IConfigPaths = KeyPaths<IConfig>; 114e22d5e4fS猫头猫type PartialConfig = DeepPartial<IConfig> | null; 115e22d5e4fS猫头猫type IConfigPathsObj = KeyPathsObj<DeepPartial<IConfig>, IConfigPaths>; 116e22d5e4fS猫头猫 117e22d5e4fS猫头猫let config: PartialConfig = null; 118e22d5e4fS猫头猫/** 初始化config */ 1191f829e09S猫头猫async function setup() { 120e22d5e4fS猫头猫 config = (await getStorage('local-config')) ?? {}; 121e22d5e4fS猫头猫 // await checkValidPath(['setting.theme.background']); 122e22d5e4fS猫头猫 notify(); 123e22d5e4fS猫头猫} 124e22d5e4fS猫头猫 125e22d5e4fS猫头猫/** 设置config */ 126e22d5e4fS猫头猫async function setConfig<T extends IConfigPaths>( 127e22d5e4fS猫头猫 key: T, 128e22d5e4fS猫头猫 value: IConfigPathsObj[T], 129e22d5e4fS猫头猫 shouldNotify = true, 130e22d5e4fS猫头猫) { 131e22d5e4fS猫头猫 if (config === null) { 132e22d5e4fS猫头猫 return; 133e22d5e4fS猫头猫 } 134e22d5e4fS猫头猫 const keys = key.split('.'); 135e22d5e4fS猫头猫 136e22d5e4fS猫头猫 const result = produce(config, draft => { 137e22d5e4fS猫头猫 draft[keys[0] as keyof IConfig] = draft[keys[0] as keyof IConfig] ?? {}; 138e22d5e4fS猫头猫 let conf: any = draft[keys[0] as keyof IConfig]; 139e22d5e4fS猫头猫 for (let i = 1; i < keys.length - 1; ++i) { 140e22d5e4fS猫头猫 if (!conf?.[keys[i]]) { 141e22d5e4fS猫头猫 conf[keys[i]] = {}; 142e22d5e4fS猫头猫 } 143e22d5e4fS猫头猫 conf = conf[keys[i]]; 144e22d5e4fS猫头猫 } 145e22d5e4fS猫头猫 conf[keys[keys.length - 1]] = value; 146e22d5e4fS猫头猫 return draft; 147e22d5e4fS猫头猫 }); 148e22d5e4fS猫头猫 149e22d5e4fS猫头猫 setStorage('local-config', result); 150e22d5e4fS猫头猫 config = result; 151e22d5e4fS猫头猫 if (shouldNotify) { 152e22d5e4fS猫头猫 notify(); 153e22d5e4fS猫头猫 } 154e22d5e4fS猫头猫} 155e22d5e4fS猫头猫 156e22d5e4fS猫头猫/** 获取config */ 157e22d5e4fS猫头猫function getConfig(): PartialConfig; 158e22d5e4fS猫头猫function getConfig<T extends IConfigPaths>(key: T): IConfigPathsObj[T]; 159e22d5e4fS猫头猫function getConfig(key?: string) { 160e22d5e4fS猫头猫 let result: any = config; 161e22d5e4fS猫头猫 if (key && config) { 162e22d5e4fS猫头猫 result = getPathValue(config, key); 163e22d5e4fS猫头猫 } 164e22d5e4fS猫头猫 165e22d5e4fS猫头猫 return result; 166e22d5e4fS猫头猫} 167e22d5e4fS猫头猫 168e22d5e4fS猫头猫/** 通过path获取值 */ 169e22d5e4fS猫头猫function getPathValue(obj: Record<string, any>, path: string) { 170e22d5e4fS猫头猫 const keys = path.split('.'); 171e22d5e4fS猫头猫 let tmp = obj; 172e22d5e4fS猫头猫 for (let i = 0; i < keys.length; ++i) { 173e22d5e4fS猫头猫 tmp = tmp?.[keys[i]]; 174e22d5e4fS猫头猫 } 175e22d5e4fS猫头猫 return tmp; 176e22d5e4fS猫头猫} 177e22d5e4fS猫头猫 178e22d5e4fS猫头猫/** 同步hook */ 179e22d5e4fS猫头猫const notifyCbs = new Set<() => void>(); 180e22d5e4fS猫头猫function notify() { 181e22d5e4fS猫头猫 notifyCbs.forEach(_ => _?.()); 182e22d5e4fS猫头猫} 183e22d5e4fS猫头猫 184e22d5e4fS猫头猫/** hook */ 185e22d5e4fS猫头猫function useConfig(): PartialConfig; 186e22d5e4fS猫头猫function useConfig<T extends IConfigPaths>(key: T): IConfigPathsObj[T]; 187e22d5e4fS猫头猫function useConfig(key?: string) { 188e22d5e4fS猫头猫 const [_cfg, _setCfg] = useState<PartialConfig>(config); 189e22d5e4fS猫头猫 function setCfg() { 190e22d5e4fS猫头猫 _setCfg(config); 191e22d5e4fS猫头猫 } 192e22d5e4fS猫头猫 useEffect(() => { 193e22d5e4fS猫头猫 notifyCbs.add(setCfg); 194e22d5e4fS猫头猫 return () => { 195e22d5e4fS猫头猫 notifyCbs.delete(setCfg); 196e22d5e4fS猫头猫 }; 197e22d5e4fS猫头猫 }, []); 198e22d5e4fS猫头猫 199e22d5e4fS猫头猫 if (key) { 200e22d5e4fS猫头猫 return _cfg ? getPathValue(_cfg, key) : undefined; 201e22d5e4fS猫头猫 } else { 202e22d5e4fS猫头猫 return _cfg; 203e22d5e4fS猫头猫 } 204e22d5e4fS猫头猫} 205e22d5e4fS猫头猫 206e22d5e4fS猫头猫const Config = { 207e22d5e4fS猫头猫 get: getConfig, 208e22d5e4fS猫头猫 set: setConfig, 209e22d5e4fS猫头猫 useConfig, 2101f829e09S猫头猫 setup, 211e22d5e4fS猫头猫}; 212e22d5e4fS猫头猫 213e22d5e4fS猫头猫export default Config; 214