1b7048bd1S猫头猫// import {Quality} from '@/constants/commonConst'; 2*819a9075Smaotoumaoimport { CustomizedColors } from "@/hooks/useColors"; 3*819a9075Smaotoumaoimport { getStorage, setStorage } from "@/utils/storage"; 4*819a9075Smaotoumaoimport { produce } from "immer"; 5*819a9075Smaotoumaoimport { useEffect, useState } from "react"; 6*819a9075Smaotoumaoimport { ResumeMode, SortType } from "@/constants/commonConst.ts"; 7e22d5e4fS猫头猫 8b7048bd1S猫头猫type ExceptionType = IMusic.IMusicItem | IMusic.IMusicItem[] | IMusic.IQuality; 9e22d5e4fS猫头猫interface IConfig { 10e22d5e4fS猫头猫 setting: { 11e22d5e4fS猫头猫 basic: { 12aaa0db32S猫头猫 autoPlayWhenAppStart: boolean; 13cfa0fc07S猫头猫 /** 使用移动网络播放 */ 14cfa0fc07S猫头猫 useCelluarNetworkPlay: boolean; 15cfa0fc07S猫头猫 /** 使用移动网络下载 */ 16cfa0fc07S猫头猫 useCelluarNetworkDownload: boolean; 17e22d5e4fS猫头猫 /** 最大同时下载 */ 18e22d5e4fS猫头猫 maxDownload: number | string; 199297b0a0S猫头猫 /** 播放歌曲行为 */ 209297b0a0S猫头猫 clickMusicInSearch: '播放歌曲' | '播放歌曲并替换播放列表'; 21cbb92902S猫头猫 /** 点击专辑单曲 */ 22cbb92902S猫头猫 clickMusicInAlbum: '播放专辑' | '播放单曲'; 23b6261296S猫头猫 /** 下载文件夹 */ 24b6261296S猫头猫 downloadPath: string; 25e22d5e4fS猫头猫 /** 同时播放 */ 26e22d5e4fS猫头猫 notInterrupt: boolean; 27266badb7S猫头猫 /** 打断时 */ 28266badb7S猫头猫 tempRemoteDuck: '暂停' | '降低音量'; 29e22d5e4fS猫头猫 /** 播放错误时自动停止 */ 30e22d5e4fS猫头猫 autoStopWhenError: boolean; 31cfa0fc07S猫头猫 /** 插件缓存策略 todo */ 32cfa0fc07S猫头猫 pluginCacheControl: string; 33cfa0fc07S猫头猫 /** 最大音乐缓存 */ 34cfa0fc07S猫头猫 maxCacheSize: number; 35e08d37a3S猫头猫 /** 默认播放音质 */ 36b7048bd1S猫头猫 defaultPlayQuality: IMusic.IQualityKey; 37e08d37a3S猫头猫 /** 音质顺序 */ 38e08d37a3S猫头猫 playQualityOrder: 'asc' | 'desc'; 39e08d37a3S猫头猫 /** 默认下载音质 */ 40b7048bd1S猫头猫 defaultDownloadQuality: IMusic.IQualityKey; 41e08d37a3S猫头猫 /** 下载音质顺序 */ 42e08d37a3S猫头猫 downloadQualityOrder: 'asc' | 'desc'; 43c342e600S猫头猫 /** 歌曲详情页 */ 44c342e600S猫头猫 musicDetailDefault: 'album' | 'lyric'; 45fbdbd2d3S猫头猫 /** 歌曲详情页常亮 */ 46fbdbd2d3S猫头猫 musicDetailAwake: boolean; 47e22d5e4fS猫头猫 debug: { 48e22d5e4fS猫头猫 errorLog: boolean; 49e22d5e4fS猫头猫 traceLog: boolean; 50ea6d708fS猫头猫 devLog: boolean; 51e22d5e4fS猫头猫 }; 52b3a3a048S猫头猫 /** 最大历史记录条目 */ 530412c91bS猫头猫 maxHistoryLen: number; 54b3a3a048S猫头猫 /** 启动时自动更新插件 */ 55b3a3a048S猫头猫 autoUpdatePlugin: boolean; 56c2b3a262S猫头猫 // 不检查插件版本号 57c2b3a262S猫头猫 notCheckPluginVersion: boolean; 58944c56ffS猫头猫 /** 关联歌词方式 */ 59944c56ffS猫头猫 associateLyricType: 'input' | 'search'; 600dd58d38S猫头猫 // 是否展示退出按钮 610dd58d38S猫头猫 showExitOnNotification: boolean; 6295b36719S猫头猫 // 本地歌单添加歌曲顺序 63740e3947S猫头猫 musicOrderInLocalSheet: SortType; 643991724eS猫头猫 // 自动换源 653991724eS猫头猫 tryChangeSourceWhenPlayFail: boolean; 66e22d5e4fS猫头猫 }; 6757277364S猫头猫 /** 歌词 */ 6857277364S猫头猫 lyric: { 6957277364S猫头猫 showStatusBarLyric: boolean; 7057277364S猫头猫 topPercent: number; 7157277364S猫头猫 leftPercent: number; 7257277364S猫头猫 align: number; 7357277364S猫头猫 color: string; 7457277364S猫头猫 backgroundColor: string; 7557277364S猫头猫 widthPercent: number; 7657277364S猫头猫 fontSize: number; 776f57784cS猫头猫 // 详情页的字体大小 786f57784cS猫头猫 detailFontSize: number; 791057be29S猫头猫 // 自动搜索歌词 801057be29S猫头猫 autoSearchLyric: boolean; 8157277364S猫头猫 }; 82e22d5e4fS猫头猫 83e22d5e4fS猫头猫 /** 主题 */ 84e22d5e4fS猫头猫 theme: { 85e22d5e4fS猫头猫 background: string; 86e22d5e4fS猫头猫 backgroundOpacity: number; 87e22d5e4fS猫头猫 backgroundBlur: number; 88a27adc20S猫头猫 colors: CustomizedColors; 89d1fd80edS猫头猫 customColors?: CustomizedColors; 906cfecf1cS猫头猫 followSystem: boolean; 916cfecf1cS猫头猫 selectedTheme: string; 92e22d5e4fS猫头猫 }; 938b7ddca8S猫头猫 94a7b42a4cS猫头猫 backup: { 95adf41771S猫头猫 resumeMode: ResumeMode; 96a7b42a4cS猫头猫 }; 97a7b42a4cS猫头猫 988b7ddca8S猫头猫 plugin: { 998b7ddca8S猫头猫 subscribeUrl: string; 1008b7ddca8S猫头猫 }; 10128ccceb7S猫头猫 webdav: { 10228ccceb7S猫头猫 url: string; 10328ccceb7S猫头猫 username: string; 10428ccceb7S猫头猫 password: string; 10528ccceb7S猫头猫 }; 106e22d5e4fS猫头猫 }; 107e22d5e4fS猫头猫} 108e22d5e4fS猫头猫 109e22d5e4fS猫头猫type FilterType<T, R = never> = T extends Record<string | number, any> 110e22d5e4fS猫头猫 ? { 111e22d5e4fS猫头猫 [P in keyof T]: T[P] extends ExceptionType ? R : T[P]; 112e22d5e4fS猫头猫 } 113e22d5e4fS猫头猫 : never; 114e22d5e4fS猫头猫 115e22d5e4fS猫头猫type KeyPaths< 116e22d5e4fS猫头猫 T extends object, 117e22d5e4fS猫头猫 Root extends boolean = true, 118e22d5e4fS猫头猫 R = FilterType<T, ''>, 119e22d5e4fS猫头猫 K extends keyof R = keyof R, 120e22d5e4fS猫头猫> = K extends string | number 121e22d5e4fS猫头猫 ? 122e22d5e4fS猫头猫 | (Root extends true ? `${K}` : `.${K}`) 123e22d5e4fS猫头猫 | (R[K] extends Record<string | number, any> 1244060c00aS猫头猫 ? `${Root extends true ? `${K}` : `.${K}`}${KeyPaths< 1254060c00aS猫头猫 R[K], 1264060c00aS猫头猫 false 1274060c00aS猫头猫 >}` 128e22d5e4fS猫头猫 : never) 129e22d5e4fS猫头猫 : never; 130e22d5e4fS猫头猫 131e22d5e4fS猫头猫type KeyPathValue<T extends object, K extends string> = T extends Record< 132e22d5e4fS猫头猫 string | number, 133e22d5e4fS猫头猫 any 134e22d5e4fS猫头猫> 135e22d5e4fS猫头猫 ? K extends `${infer S}.${infer R}` 136e22d5e4fS猫头猫 ? KeyPathValue<T[S], R> 137e22d5e4fS猫头猫 : T[K] 138e22d5e4fS猫头猫 : never; 139e22d5e4fS猫头猫 140e22d5e4fS猫头猫type KeyPathsObj< 141e22d5e4fS猫头猫 T extends object, 142e22d5e4fS猫头猫 K extends string = KeyPaths<T>, 143e22d5e4fS猫头猫> = T extends Record<string | number, any> 144e22d5e4fS猫头猫 ? { 145e22d5e4fS猫头猫 [R in K]: KeyPathValue<T, R>; 146e22d5e4fS猫头猫 } 147e22d5e4fS猫头猫 : never; 148e22d5e4fS猫头猫 149e22d5e4fS猫头猫type DeepPartial<T> = { 150e22d5e4fS猫头猫 [K in keyof T]?: T[K] extends Record<string | number, any> 151e22d5e4fS猫头猫 ? T[K] extends ExceptionType 152e22d5e4fS猫头猫 ? T[K] 153e22d5e4fS猫头猫 : DeepPartial<T[K]> 154e22d5e4fS猫头猫 : T[K]; 155e22d5e4fS猫头猫}; 156e22d5e4fS猫头猫 157cfa0fc07S猫头猫export type IConfigPaths = KeyPaths<IConfig>; 158e22d5e4fS猫头猫type PartialConfig = DeepPartial<IConfig> | null; 159e22d5e4fS猫头猫type IConfigPathsObj = KeyPathsObj<DeepPartial<IConfig>, IConfigPaths>; 160e22d5e4fS猫头猫 161e22d5e4fS猫头猫let config: PartialConfig = null; 162e22d5e4fS猫头猫/** 初始化config */ 1631f829e09S猫头猫async function setup() { 164e22d5e4fS猫头猫 config = (await getStorage('local-config')) ?? {}; 165e22d5e4fS猫头猫 // await checkValidPath(['setting.theme.background']); 166e22d5e4fS猫头猫 notify(); 167e22d5e4fS猫头猫} 168e22d5e4fS猫头猫 169e22d5e4fS猫头猫/** 设置config */ 170e22d5e4fS猫头猫async function setConfig<T extends IConfigPaths>( 171e22d5e4fS猫头猫 key: T, 172e22d5e4fS猫头猫 value: IConfigPathsObj[T], 173e22d5e4fS猫头猫 shouldNotify = true, 174e22d5e4fS猫头猫) { 175e22d5e4fS猫头猫 if (config === null) { 176e22d5e4fS猫头猫 return; 177e22d5e4fS猫头猫 } 178e22d5e4fS猫头猫 const keys = key.split('.'); 179e22d5e4fS猫头猫 180e22d5e4fS猫头猫 const result = produce(config, draft => { 181e22d5e4fS猫头猫 draft[keys[0] as keyof IConfig] = draft[keys[0] as keyof IConfig] ?? {}; 182e22d5e4fS猫头猫 let conf: any = draft[keys[0] as keyof IConfig]; 183e22d5e4fS猫头猫 for (let i = 1; i < keys.length - 1; ++i) { 184e22d5e4fS猫头猫 if (!conf?.[keys[i]]) { 185e22d5e4fS猫头猫 conf[keys[i]] = {}; 186e22d5e4fS猫头猫 } 187e22d5e4fS猫头猫 conf = conf[keys[i]]; 188e22d5e4fS猫头猫 } 189e22d5e4fS猫头猫 conf[keys[keys.length - 1]] = value; 190e22d5e4fS猫头猫 return draft; 191e22d5e4fS猫头猫 }); 192e22d5e4fS猫头猫 193e22d5e4fS猫头猫 setStorage('local-config', result); 194e22d5e4fS猫头猫 config = result; 195e22d5e4fS猫头猫 if (shouldNotify) { 196e22d5e4fS猫头猫 notify(); 197e22d5e4fS猫头猫 } 198e22d5e4fS猫头猫} 199e22d5e4fS猫头猫 200b7048bd1S猫头猫// todo: 获取兜底 201e22d5e4fS猫头猫/** 获取config */ 202e22d5e4fS猫头猫function getConfig(): PartialConfig; 203e22d5e4fS猫头猫function getConfig<T extends IConfigPaths>(key: T): IConfigPathsObj[T]; 204e22d5e4fS猫头猫function getConfig(key?: string) { 205e22d5e4fS猫头猫 let result: any = config; 206e22d5e4fS猫头猫 if (key && config) { 207e22d5e4fS猫头猫 result = getPathValue(config, key); 208e22d5e4fS猫头猫 } 209e22d5e4fS猫头猫 210e22d5e4fS猫头猫 return result; 211e22d5e4fS猫头猫} 212e22d5e4fS猫头猫 213e22d5e4fS猫头猫/** 通过path获取值 */ 214e22d5e4fS猫头猫function getPathValue(obj: Record<string, any>, path: string) { 215e22d5e4fS猫头猫 const keys = path.split('.'); 216e22d5e4fS猫头猫 let tmp = obj; 217e22d5e4fS猫头猫 for (let i = 0; i < keys.length; ++i) { 218e22d5e4fS猫头猫 tmp = tmp?.[keys[i]]; 219e22d5e4fS猫头猫 } 220e22d5e4fS猫头猫 return tmp; 221e22d5e4fS猫头猫} 222e22d5e4fS猫头猫 223e22d5e4fS猫头猫/** 同步hook */ 224e22d5e4fS猫头猫const notifyCbs = new Set<() => void>(); 225e22d5e4fS猫头猫function notify() { 226e22d5e4fS猫头猫 notifyCbs.forEach(_ => _?.()); 227e22d5e4fS猫头猫} 228e22d5e4fS猫头猫 229e22d5e4fS猫头猫/** hook */ 230e22d5e4fS猫头猫function useConfig(): PartialConfig; 231e22d5e4fS猫头猫function useConfig<T extends IConfigPaths>(key: T): IConfigPathsObj[T]; 232e22d5e4fS猫头猫function useConfig(key?: string) { 233a27adc20S猫头猫 // TODO: 应该有性能损失 234e22d5e4fS猫头猫 const [_cfg, _setCfg] = useState<PartialConfig>(config); 235e22d5e4fS猫头猫 function setCfg() { 236e22d5e4fS猫头猫 _setCfg(config); 237e22d5e4fS猫头猫 } 238e22d5e4fS猫头猫 useEffect(() => { 239e22d5e4fS猫头猫 notifyCbs.add(setCfg); 240e22d5e4fS猫头猫 return () => { 241e22d5e4fS猫头猫 notifyCbs.delete(setCfg); 242e22d5e4fS猫头猫 }; 243e22d5e4fS猫头猫 }, []); 244e22d5e4fS猫头猫 245e22d5e4fS猫头猫 if (key) { 246e22d5e4fS猫头猫 return _cfg ? getPathValue(_cfg, key) : undefined; 247e22d5e4fS猫头猫 } else { 248e22d5e4fS猫头猫 return _cfg; 249e22d5e4fS猫头猫 } 250e22d5e4fS猫头猫} 251e22d5e4fS猫头猫 252e22d5e4fS猫头猫const Config = { 253e22d5e4fS猫头猫 get: getConfig, 254e22d5e4fS猫头猫 set: setConfig, 255e22d5e4fS猫头猫 useConfig, 2561f829e09S猫头猫 setup, 257e22d5e4fS猫头猫}; 258e22d5e4fS猫头猫 259e22d5e4fS猫头猫export default Config; 260