xref: /MusicFree/src/core/config.ts (revision 0dd58d38ce353e06cabdd6631dd92e8c9780cc3e)
1b7048bd1S猫头猫// import {Quality} from '@/constants/commonConst';
2a27adc20S猫头猫import {CustomizedColors} from '@/hooks/useColors';
3e22d5e4fS猫头猫import {getStorage, setStorage} from '@/utils/storage';
4e22d5e4fS猫头猫import produce from 'immer';
5e22d5e4fS猫头猫import {useEffect, useState} from 'react';
6e22d5e4fS猫头猫
7b7048bd1S猫头猫type ExceptionType = IMusic.IMusicItem | IMusic.IMusicItem[] | IMusic.IQuality;
8e22d5e4fS猫头猫interface IConfig {
9e22d5e4fS猫头猫    setting: {
10e22d5e4fS猫头猫        basic: {
11aaa0db32S猫头猫            autoPlayWhenAppStart: boolean;
12cfa0fc07S猫头猫            /** 使用移动网络播放 */
13cfa0fc07S猫头猫            useCelluarNetworkPlay: boolean;
14cfa0fc07S猫头猫            /** 使用移动网络下载 */
15cfa0fc07S猫头猫            useCelluarNetworkDownload: boolean;
16e22d5e4fS猫头猫            /** 最大同时下载 */
17e22d5e4fS猫头猫            maxDownload: number | string;
189297b0a0S猫头猫            /** 播放歌曲行为 */
199297b0a0S猫头猫            clickMusicInSearch: '播放歌曲' | '播放歌曲并替换播放列表';
20cbb92902S猫头猫            /** 点击专辑单曲 */
21cbb92902S猫头猫            clickMusicInAlbum: '播放专辑' | '播放单曲';
22b6261296S猫头猫            /** 下载文件夹 */
23b6261296S猫头猫            downloadPath: string;
24e22d5e4fS猫头猫            /** 同时播放 */
25e22d5e4fS猫头猫            notInterrupt: boolean;
26266badb7S猫头猫            /** 打断时 */
27266badb7S猫头猫            tempRemoteDuck: '暂停' | '降低音量';
28e22d5e4fS猫头猫            /** 播放错误时自动停止 */
29e22d5e4fS猫头猫            autoStopWhenError: boolean;
30cfa0fc07S猫头猫            /** 插件缓存策略 todo */
31cfa0fc07S猫头猫            pluginCacheControl: string;
32cfa0fc07S猫头猫            /** 最大音乐缓存 */
33cfa0fc07S猫头猫            maxCacheSize: number;
34e08d37a3S猫头猫            /** 默认播放音质 */
35b7048bd1S猫头猫            defaultPlayQuality: IMusic.IQualityKey;
36e08d37a3S猫头猫            /** 音质顺序 */
37e08d37a3S猫头猫            playQualityOrder: 'asc' | 'desc';
38e08d37a3S猫头猫            /** 默认下载音质 */
39b7048bd1S猫头猫            defaultDownloadQuality: IMusic.IQualityKey;
40e08d37a3S猫头猫            /** 下载音质顺序 */
41e08d37a3S猫头猫            downloadQualityOrder: 'asc' | 'desc';
42c342e600S猫头猫            /** 歌曲详情页 */
43c342e600S猫头猫            musicDetailDefault: 'album' | 'lyric';
44fbdbd2d3S猫头猫            /** 歌曲详情页常亮 */
45fbdbd2d3S猫头猫            musicDetailAwake: boolean;
46e22d5e4fS猫头猫            debug: {
47e22d5e4fS猫头猫                errorLog: boolean;
48e22d5e4fS猫头猫                traceLog: boolean;
49ea6d708fS猫头猫                devLog: boolean;
50e22d5e4fS猫头猫            };
51b3a3a048S猫头猫            /** 最大历史记录条目 */
520412c91bS猫头猫            maxHistoryLen: number;
53b3a3a048S猫头猫            /** 启动时自动更新插件 */
54b3a3a048S猫头猫            autoUpdatePlugin: boolean;
55c2b3a262S猫头猫            // 不检查插件版本号
56c2b3a262S猫头猫            notCheckPluginVersion: boolean;
57944c56ffS猫头猫            /** 关联歌词方式 */
58944c56ffS猫头猫            associateLyricType: 'input' | 'search';
59*0dd58d38S猫头猫            // 是否展示退出按钮
60*0dd58d38S猫头猫            showExitOnNotification: boolean;
61e22d5e4fS猫头猫        };
6257277364S猫头猫        /** 歌词 */
6357277364S猫头猫        lyric: {
6457277364S猫头猫            showStatusBarLyric: boolean;
6557277364S猫头猫            topPercent: number;
6657277364S猫头猫            leftPercent: number;
6757277364S猫头猫            align: number;
6857277364S猫头猫            color: string;
6957277364S猫头猫            backgroundColor: string;
7057277364S猫头猫            widthPercent: number;
7157277364S猫头猫            fontSize: number;
726f57784cS猫头猫            // 详情页的字体大小
736f57784cS猫头猫            detailFontSize: number;
741057be29S猫头猫            // 自动搜索歌词
751057be29S猫头猫            autoSearchLyric: boolean;
7657277364S猫头猫        };
77e22d5e4fS猫头猫
78e22d5e4fS猫头猫        /** 主题 */
79e22d5e4fS猫头猫        theme: {
80e22d5e4fS猫头猫            background: string;
81e22d5e4fS猫头猫            backgroundOpacity: number;
82e22d5e4fS猫头猫            backgroundBlur: number;
83a27adc20S猫头猫            colors: CustomizedColors;
84d1fd80edS猫头猫            customColors?: CustomizedColors;
856cfecf1cS猫头猫            followSystem: boolean;
866cfecf1cS猫头猫            selectedTheme: string;
87e22d5e4fS猫头猫        };
888b7ddca8S猫头猫
89a7b42a4cS猫头猫        backup: {
90a7b42a4cS猫头猫            resumeMode: 'append' | 'overwrite';
91a7b42a4cS猫头猫        };
92a7b42a4cS猫头猫
938b7ddca8S猫头猫        plugin: {
948b7ddca8S猫头猫            subscribeUrl: string;
958b7ddca8S猫头猫        };
9628ccceb7S猫头猫        webdav: {
9728ccceb7S猫头猫            url: string;
9828ccceb7S猫头猫            username: string;
9928ccceb7S猫头猫            password: string;
10028ccceb7S猫头猫        };
101e22d5e4fS猫头猫    };
102e22d5e4fS猫头猫    status: {
103e22d5e4fS猫头猫        music: {
104e22d5e4fS猫头猫            /** 当前的音乐 */
105e22d5e4fS猫头猫            track: IMusic.IMusicItem;
106e22d5e4fS猫头猫            /** 进度 */
107e22d5e4fS猫头猫            progress: number;
108e22d5e4fS猫头猫            /** 模式 */
109e22d5e4fS猫头猫            repeatMode: string;
110e22d5e4fS猫头猫            /** 列表 */
111e22d5e4fS猫头猫            musicQueue: IMusic.IMusicItem[];
112fcc60afbS猫头猫            /** 速度 */
113fcc60afbS猫头猫            rate: number;
114e22d5e4fS猫头猫        };
115f6017740S猫头猫        app: {
116f6017740S猫头猫            /** 跳过特定版本 */
117f6017740S猫头猫            skipVersion: string;
118f6017740S猫头猫        };
119e22d5e4fS猫头猫    };
120e22d5e4fS猫头猫}
121e22d5e4fS猫头猫
122e22d5e4fS猫头猫type FilterType<T, R = never> = T extends Record<string | number, any>
123e22d5e4fS猫头猫    ? {
124e22d5e4fS猫头猫          [P in keyof T]: T[P] extends ExceptionType ? R : T[P];
125e22d5e4fS猫头猫      }
126e22d5e4fS猫头猫    : never;
127e22d5e4fS猫头猫
128e22d5e4fS猫头猫type KeyPaths<
129e22d5e4fS猫头猫    T extends object,
130e22d5e4fS猫头猫    Root extends boolean = true,
131e22d5e4fS猫头猫    R = FilterType<T, ''>,
132e22d5e4fS猫头猫    K extends keyof R = keyof R,
133e22d5e4fS猫头猫> = K extends string | number
134e22d5e4fS猫头猫    ?
135e22d5e4fS猫头猫          | (Root extends true ? `${K}` : `.${K}`)
136e22d5e4fS猫头猫          | (R[K] extends Record<string | number, any>
1374060c00aS猫头猫                ? `${Root extends true ? `${K}` : `.${K}`}${KeyPaths<
1384060c00aS猫头猫                      R[K],
1394060c00aS猫头猫                      false
1404060c00aS猫头猫                  >}`
141e22d5e4fS猫头猫                : never)
142e22d5e4fS猫头猫    : never;
143e22d5e4fS猫头猫
144e22d5e4fS猫头猫type KeyPathValue<T extends object, K extends string> = T extends Record<
145e22d5e4fS猫头猫    string | number,
146e22d5e4fS猫头猫    any
147e22d5e4fS猫头猫>
148e22d5e4fS猫头猫    ? K extends `${infer S}.${infer R}`
149e22d5e4fS猫头猫        ? KeyPathValue<T[S], R>
150e22d5e4fS猫头猫        : T[K]
151e22d5e4fS猫头猫    : never;
152e22d5e4fS猫头猫
153e22d5e4fS猫头猫type KeyPathsObj<
154e22d5e4fS猫头猫    T extends object,
155e22d5e4fS猫头猫    K extends string = KeyPaths<T>,
156e22d5e4fS猫头猫> = T extends Record<string | number, any>
157e22d5e4fS猫头猫    ? {
158e22d5e4fS猫头猫          [R in K]: KeyPathValue<T, R>;
159e22d5e4fS猫头猫      }
160e22d5e4fS猫头猫    : never;
161e22d5e4fS猫头猫
162e22d5e4fS猫头猫type DeepPartial<T> = {
163e22d5e4fS猫头猫    [K in keyof T]?: T[K] extends Record<string | number, any>
164e22d5e4fS猫头猫        ? T[K] extends ExceptionType
165e22d5e4fS猫头猫            ? T[K]
166e22d5e4fS猫头猫            : DeepPartial<T[K]>
167e22d5e4fS猫头猫        : T[K];
168e22d5e4fS猫头猫};
169e22d5e4fS猫头猫
170cfa0fc07S猫头猫export type IConfigPaths = KeyPaths<IConfig>;
171e22d5e4fS猫头猫type PartialConfig = DeepPartial<IConfig> | null;
172e22d5e4fS猫头猫type IConfigPathsObj = KeyPathsObj<DeepPartial<IConfig>, IConfigPaths>;
173e22d5e4fS猫头猫
174e22d5e4fS猫头猫let config: PartialConfig = null;
175e22d5e4fS猫头猫/** 初始化config */
1761f829e09S猫头猫async function setup() {
177e22d5e4fS猫头猫    config = (await getStorage('local-config')) ?? {};
178e22d5e4fS猫头猫    // await checkValidPath(['setting.theme.background']);
179e22d5e4fS猫头猫    notify();
180e22d5e4fS猫头猫}
181e22d5e4fS猫头猫
182e22d5e4fS猫头猫/** 设置config */
183e22d5e4fS猫头猫async function setConfig<T extends IConfigPaths>(
184e22d5e4fS猫头猫    key: T,
185e22d5e4fS猫头猫    value: IConfigPathsObj[T],
186e22d5e4fS猫头猫    shouldNotify = true,
187e22d5e4fS猫头猫) {
188e22d5e4fS猫头猫    if (config === null) {
189e22d5e4fS猫头猫        return;
190e22d5e4fS猫头猫    }
191e22d5e4fS猫头猫    const keys = key.split('.');
192e22d5e4fS猫头猫
193e22d5e4fS猫头猫    const result = produce(config, draft => {
194e22d5e4fS猫头猫        draft[keys[0] as keyof IConfig] = draft[keys[0] as keyof IConfig] ?? {};
195e22d5e4fS猫头猫        let conf: any = draft[keys[0] as keyof IConfig];
196e22d5e4fS猫头猫        for (let i = 1; i < keys.length - 1; ++i) {
197e22d5e4fS猫头猫            if (!conf?.[keys[i]]) {
198e22d5e4fS猫头猫                conf[keys[i]] = {};
199e22d5e4fS猫头猫            }
200e22d5e4fS猫头猫            conf = conf[keys[i]];
201e22d5e4fS猫头猫        }
202e22d5e4fS猫头猫        conf[keys[keys.length - 1]] = value;
203e22d5e4fS猫头猫        return draft;
204e22d5e4fS猫头猫    });
205e22d5e4fS猫头猫
206e22d5e4fS猫头猫    setStorage('local-config', result);
207e22d5e4fS猫头猫    config = result;
208e22d5e4fS猫头猫    if (shouldNotify) {
209e22d5e4fS猫头猫        notify();
210e22d5e4fS猫头猫    }
211e22d5e4fS猫头猫}
212e22d5e4fS猫头猫
213b7048bd1S猫头猫// todo: 获取兜底
214e22d5e4fS猫头猫/** 获取config */
215e22d5e4fS猫头猫function getConfig(): PartialConfig;
216e22d5e4fS猫头猫function getConfig<T extends IConfigPaths>(key: T): IConfigPathsObj[T];
217e22d5e4fS猫头猫function getConfig(key?: string) {
218e22d5e4fS猫头猫    let result: any = config;
219e22d5e4fS猫头猫    if (key && config) {
220e22d5e4fS猫头猫        result = getPathValue(config, key);
221e22d5e4fS猫头猫    }
222e22d5e4fS猫头猫
223e22d5e4fS猫头猫    return result;
224e22d5e4fS猫头猫}
225e22d5e4fS猫头猫
226e22d5e4fS猫头猫/** 通过path获取值 */
227e22d5e4fS猫头猫function getPathValue(obj: Record<string, any>, path: string) {
228e22d5e4fS猫头猫    const keys = path.split('.');
229e22d5e4fS猫头猫    let tmp = obj;
230e22d5e4fS猫头猫    for (let i = 0; i < keys.length; ++i) {
231e22d5e4fS猫头猫        tmp = tmp?.[keys[i]];
232e22d5e4fS猫头猫    }
233e22d5e4fS猫头猫    return tmp;
234e22d5e4fS猫头猫}
235e22d5e4fS猫头猫
236e22d5e4fS猫头猫/** 同步hook */
237e22d5e4fS猫头猫const notifyCbs = new Set<() => void>();
238e22d5e4fS猫头猫function notify() {
239e22d5e4fS猫头猫    notifyCbs.forEach(_ => _?.());
240e22d5e4fS猫头猫}
241e22d5e4fS猫头猫
242e22d5e4fS猫头猫/** hook */
243e22d5e4fS猫头猫function useConfig(): PartialConfig;
244e22d5e4fS猫头猫function useConfig<T extends IConfigPaths>(key: T): IConfigPathsObj[T];
245e22d5e4fS猫头猫function useConfig(key?: string) {
246a27adc20S猫头猫    // TODO: 应该有性能损失
247e22d5e4fS猫头猫    const [_cfg, _setCfg] = useState<PartialConfig>(config);
248e22d5e4fS猫头猫    function setCfg() {
249e22d5e4fS猫头猫        _setCfg(config);
250e22d5e4fS猫头猫    }
251e22d5e4fS猫头猫    useEffect(() => {
252e22d5e4fS猫头猫        notifyCbs.add(setCfg);
253e22d5e4fS猫头猫        return () => {
254e22d5e4fS猫头猫            notifyCbs.delete(setCfg);
255e22d5e4fS猫头猫        };
256e22d5e4fS猫头猫    }, []);
257e22d5e4fS猫头猫
258e22d5e4fS猫头猫    if (key) {
259e22d5e4fS猫头猫        return _cfg ? getPathValue(_cfg, key) : undefined;
260e22d5e4fS猫头猫    } else {
261e22d5e4fS猫头猫        return _cfg;
262e22d5e4fS猫头猫    }
263e22d5e4fS猫头猫}
264e22d5e4fS猫头猫
265e22d5e4fS猫头猫const Config = {
266e22d5e4fS猫头猫    get: getConfig,
267e22d5e4fS猫头猫    set: setConfig,
268e22d5e4fS猫头猫    useConfig,
2691f829e09S猫头猫    setup,
270e22d5e4fS猫头猫};
271e22d5e4fS猫头猫
272e22d5e4fS猫头猫export default Config;
273