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