mirror of
https://github.com/jeffvli/feishin.git
synced 2026-05-13 20:10:07 +02:00
ca0a1569f8
Co-authored-by: Ross <ro@noirlab.edu>
390 lines
13 KiB
TypeScript
390 lines
13 KiB
TypeScript
import type { SettingsState } from './settings.store';
|
|
|
|
import { sanitizeCss } from '/@/renderer/utils/sanitize';
|
|
|
|
const APP_THEMES = new Set([
|
|
'ayuDark',
|
|
'ayuLight',
|
|
'catppuccinLatte',
|
|
'catppuccinMocha',
|
|
'defaultDark',
|
|
'defaultLight',
|
|
'dracula',
|
|
'everforestDark',
|
|
'everforestLight',
|
|
'githubDark',
|
|
'githubLight',
|
|
'glassyDark',
|
|
'gruvboxDark',
|
|
'gruvboxLight',
|
|
'highContrastDark',
|
|
'highContrastLight',
|
|
'materialDark',
|
|
'materialLight',
|
|
'monokai',
|
|
'nightOwl',
|
|
'nord',
|
|
'oneDark',
|
|
'rosePine',
|
|
'rosePineDawn',
|
|
'rosePineMoon',
|
|
'shadesOfPurple',
|
|
'solarizedDark',
|
|
'solarizedLight',
|
|
'tokyoNight',
|
|
'vscodeDarkPlus',
|
|
'vscodeLightPlus',
|
|
]);
|
|
|
|
const DISCORD_DISPLAY_TYPES = new Set(['artist', 'feishin', 'song']);
|
|
const DISCORD_LINK_TYPES = new Set(['last_fm', 'musicbrainz', 'musicbrainz_last_fm', 'none']);
|
|
const LYRICS_ALIGNMENTS = new Set(['center', 'left', 'right']);
|
|
const FONT_TYPES = new Set(['builtIn', 'custom', 'system']);
|
|
const HOME_FEATURE_STYLES = new Set(['multiple', 'single']);
|
|
const SIDE_QUEUE_TYPES = new Set(['sideDrawerQueue', 'sideQueue']);
|
|
const SIDE_QUEUE_LAYOUTS = new Set(['horizontal', 'vertical']);
|
|
|
|
export type EnvSettingsOverrides = DeepPartial<
|
|
Pick<SettingsState, 'autoDJ' | 'css' | 'discord' | 'font' | 'general' | 'lyrics' | 'playback'>
|
|
>;
|
|
|
|
type DeepPartial<T> = {
|
|
[P in keyof T]?: T[P] extends object ? DeepPartial<T[P]> : T[P];
|
|
};
|
|
|
|
interface EnvSettingSpec {
|
|
enumSet?: Set<string>;
|
|
key: string;
|
|
path: [string, string, string] | [string, string];
|
|
skipIfEmpty?: boolean;
|
|
transform?: (raw: string) => unknown;
|
|
type: 'bool' | 'enum' | 'num' | 'string';
|
|
}
|
|
|
|
function setAtPath(
|
|
obj: EnvSettingsOverrides,
|
|
path: [string, string, string] | [string, string],
|
|
value: unknown,
|
|
): void {
|
|
const [a, b, c] = path;
|
|
const root = (obj as Record<string, unknown>)[a] ?? {};
|
|
(obj as Record<string, unknown>)[a] = root;
|
|
const branch = root as Record<string, unknown>;
|
|
if (c === undefined) {
|
|
branch[b] = value;
|
|
} else {
|
|
const nested = branch[b] ?? {};
|
|
branch[b] = nested;
|
|
(nested as Record<string, unknown>)[c] = value;
|
|
}
|
|
}
|
|
|
|
const RGB_ACCENT_REGEX = /^rgb\(\s*(\d{1,3})\s*,\s*(\d{1,3})\s*,\s*(\d{1,3})\s*\)$/;
|
|
|
|
const ENV_SETTING_SPECS: EnvSettingSpec[] = [
|
|
{
|
|
key: 'FS_GENERAL_ACCENT',
|
|
path: ['general', 'accent'],
|
|
transform: (s) => (RGB_ACCENT_REGEX.test(s) ? s : undefined),
|
|
type: 'string',
|
|
},
|
|
{ key: 'FS_GENERAL_ALBUM_BACKGROUND', path: ['general', 'albumBackground'], type: 'bool' },
|
|
{
|
|
key: 'FS_GENERAL_ALBUM_BACKGROUND_BLUR',
|
|
path: ['general', 'albumBackgroundBlur'],
|
|
type: 'num',
|
|
},
|
|
{ key: 'FS_GENERAL_ARTIST_BACKGROUND', path: ['general', 'artistBackground'], type: 'bool' },
|
|
{
|
|
key: 'FS_GENERAL_ARTIST_BACKGROUND_BLUR',
|
|
path: ['general', 'artistBackgroundBlur'],
|
|
type: 'num',
|
|
},
|
|
{
|
|
key: 'FS_GENERAL_BLUR_EXPLICIT_IMAGES',
|
|
path: ['general', 'blurExplicitImages'],
|
|
type: 'bool',
|
|
},
|
|
{
|
|
key: 'FS_GENERAL_COMBINED_LYRICS_AND_VISUALIZER',
|
|
path: ['general', 'combinedLyricsAndVisualizer'],
|
|
type: 'bool',
|
|
},
|
|
{
|
|
key: 'FS_GENERAL_ENABLE_GRID_MULTI_SELECT',
|
|
path: ['general', 'enableGridMultiSelect'],
|
|
type: 'bool',
|
|
},
|
|
{ key: 'FS_GENERAL_FOLLOW_CURRENT_SONG', path: ['general', 'followCurrentSong'], type: 'bool' },
|
|
{ key: 'FS_GENERAL_HOME_FEATURE', path: ['general', 'homeFeature'], type: 'bool' },
|
|
{
|
|
enumSet: HOME_FEATURE_STYLES,
|
|
key: 'FS_GENERAL_HOME_FEATURE_STYLE',
|
|
path: ['general', 'homeFeatureStyle'],
|
|
type: 'enum',
|
|
},
|
|
{
|
|
key: 'FS_GENERAL_LANGUAGE',
|
|
path: ['general', 'language'],
|
|
skipIfEmpty: true,
|
|
type: 'string',
|
|
},
|
|
{
|
|
key: 'FS_GENERAL_PRIMARY_SHADE',
|
|
path: ['general', 'primaryShade'],
|
|
transform: (s) => {
|
|
const n = parseNum(s);
|
|
return n !== undefined ? Math.min(9, Math.max(0, Math.round(n))) : undefined;
|
|
},
|
|
type: 'num',
|
|
},
|
|
{ enumSet: APP_THEMES, key: 'FS_GENERAL_THEME', path: ['general', 'theme'], type: 'enum' },
|
|
{
|
|
enumSet: APP_THEMES,
|
|
key: 'FS_GENERAL_THEME_DARK',
|
|
path: ['general', 'themeDark'],
|
|
type: 'enum',
|
|
},
|
|
{
|
|
enumSet: APP_THEMES,
|
|
key: 'FS_GENERAL_THEME_LIGHT',
|
|
path: ['general', 'themeLight'],
|
|
type: 'enum',
|
|
},
|
|
{ key: 'FS_GENERAL_FOLLOW_SYSTEM_THEME', path: ['general', 'followSystemTheme'], type: 'bool' },
|
|
{ key: 'FS_GENERAL_PATH_REPLACE', path: ['general', 'pathReplace'], type: 'string' },
|
|
{ key: 'FS_GENERAL_PATH_REPLACE_WITH', path: ['general', 'pathReplaceWith'], type: 'string' },
|
|
{ key: 'FS_GENERAL_LASTFM_API_KEY', path: ['general', 'lastfmApiKey'], type: 'string' },
|
|
{ key: 'FS_GENERAL_LAST_FM', path: ['general', 'lastFM'], type: 'bool' },
|
|
{ key: 'FS_GENERAL_LISTEN_BRAINZ', path: ['general', 'listenBrainz'], type: 'bool' },
|
|
{ key: 'FS_GENERAL_MUSIC_BRAINZ', path: ['general', 'musicBrainz'], type: 'bool' },
|
|
{ key: 'FS_GENERAL_QOBUZ', path: ['general', 'qobuz'], type: 'bool' },
|
|
{ key: 'FS_GENERAL_SPOTIFY', path: ['general', 'spotify'], type: 'bool' },
|
|
{ key: 'FS_GENERAL_SPOTIFY_NATIVE_APP', path: ['general', 'nativeSpotify'], type: 'bool' },
|
|
{ key: 'FS_GENERAL_NATIVE_ASPECT_RATIO', path: ['general', 'nativeAspectRatio'], type: 'bool' },
|
|
{
|
|
key: 'FS_GENERAL_PLAYERBAR_OPEN_DRAWER',
|
|
path: ['general', 'playerbarOpenDrawer'],
|
|
type: 'bool',
|
|
},
|
|
{ key: 'FS_GENERAL_EXTERNAL_LINKS', path: ['general', 'externalLinks'], type: 'bool' },
|
|
{
|
|
key: 'FS_GENERAL_SHOW_LYRICS_IN_SIDEBAR',
|
|
path: ['general', 'showLyricsInSidebar'],
|
|
type: 'bool',
|
|
},
|
|
{ key: 'FS_GENERAL_SHOW_RATINGS', path: ['general', 'showRatings'], type: 'bool' },
|
|
{
|
|
key: 'FS_GENERAL_SHOW_VISUALIZER_IN_SIDEBAR',
|
|
path: ['general', 'showVisualizerInSidebar'],
|
|
type: 'bool',
|
|
},
|
|
{
|
|
key: 'FS_GENERAL_SIDEBAR_COLLAPSED_NAVIGATION',
|
|
path: ['general', 'sidebarCollapsedNavigation'],
|
|
type: 'bool',
|
|
},
|
|
{
|
|
key: 'FS_GENERAL_SIDEBAR_COLLAPSE_SHARED',
|
|
path: ['general', 'sidebarCollapseShared'],
|
|
type: 'bool',
|
|
},
|
|
{
|
|
key: 'FS_GENERAL_SIDEBAR_PLAYLIST_LIST',
|
|
path: ['general', 'sidebarPlaylistList'],
|
|
type: 'bool',
|
|
},
|
|
{
|
|
key: 'FS_GENERAL_SIDEBAR_PLAYLIST_SORTING',
|
|
path: ['general', 'sidebarPlaylistSorting'],
|
|
type: 'bool',
|
|
},
|
|
{
|
|
enumSet: SIDE_QUEUE_TYPES,
|
|
key: 'FS_GENERAL_SIDE_QUEUE_TYPE',
|
|
path: ['general', 'sideQueueType'],
|
|
type: 'enum',
|
|
},
|
|
{
|
|
enumSet: SIDE_QUEUE_LAYOUTS,
|
|
key: 'FS_GENERAL_SIDE_QUEUE_LAYOUT',
|
|
path: ['general', 'sideQueueLayout'],
|
|
type: 'enum',
|
|
},
|
|
{ key: 'FS_GENERAL_RESUME', path: ['general', 'resume'], type: 'bool' },
|
|
{
|
|
key: 'FS_GENERAL_USE_THEME_ACCENT_COLOR',
|
|
path: ['general', 'useThemeAccentColor'],
|
|
type: 'bool',
|
|
},
|
|
{
|
|
key: 'FS_GENERAL_USE_THEME_PRIMARY_SHADE',
|
|
path: ['general', 'useThemePrimaryShade'],
|
|
type: 'bool',
|
|
},
|
|
{ key: 'FS_GENERAL_ZOOM_FACTOR', path: ['general', 'zoomFactor'], type: 'num' },
|
|
{ key: 'FS_PLAYBACK_MEDIA_SESSION', path: ['playback', 'mediaSession'], type: 'bool' },
|
|
{ key: 'FS_PLAYBACK_WEB_AUDIO', path: ['playback', 'webAudio'], type: 'bool' },
|
|
{
|
|
key: 'FS_PLAYBACK_AUDIO_FADE_ON_STATUS_CHANGE',
|
|
path: ['playback', 'audioFadeOnStatusChange'],
|
|
type: 'bool',
|
|
},
|
|
{ key: 'FS_PLAYBACK_PRESERVE_PITCH', path: ['playback', 'preservePitch'], type: 'bool' },
|
|
{
|
|
key: 'FS_PLAYBACK_SCROBBLE_ENABLED',
|
|
path: ['playback', 'scrobble', 'enabled'],
|
|
type: 'bool',
|
|
},
|
|
{ key: 'FS_PLAYBACK_SCROBBLE_NOTIFY', path: ['playback', 'scrobble', 'notify'], type: 'bool' },
|
|
{
|
|
key: 'FS_PLAYBACK_SCROBBLE_AT_DURATION',
|
|
path: ['playback', 'scrobble', 'scrobbleAtDuration'],
|
|
type: 'num',
|
|
},
|
|
{
|
|
key: 'FS_PLAYBACK_SCROBBLE_AT_PERCENTAGE',
|
|
path: ['playback', 'scrobble', 'scrobbleAtPercentage'],
|
|
type: 'num',
|
|
},
|
|
{
|
|
key: 'FS_PLAYBACK_TRANSCODE_ENABLED',
|
|
path: ['playback', 'transcode', 'enabled'],
|
|
type: 'bool',
|
|
},
|
|
{ key: 'FS_DISCORD_ENABLED', path: ['discord', 'enabled'], type: 'bool' },
|
|
{
|
|
key: 'FS_DISCORD_CLIENT_ID',
|
|
path: ['discord', 'clientId'],
|
|
skipIfEmpty: true,
|
|
type: 'string',
|
|
},
|
|
{
|
|
enumSet: DISCORD_DISPLAY_TYPES,
|
|
key: 'FS_DISCORD_DISPLAY_TYPE',
|
|
path: ['discord', 'displayType'],
|
|
type: 'enum',
|
|
},
|
|
{
|
|
enumSet: DISCORD_LINK_TYPES,
|
|
key: 'FS_DISCORD_LINK_TYPE',
|
|
path: ['discord', 'linkType'],
|
|
type: 'enum',
|
|
},
|
|
{ key: 'FS_DISCORD_SHOW_AS_LISTENING', path: ['discord', 'showAsListening'], type: 'bool' },
|
|
{ key: 'FS_DISCORD_SHOW_PAUSED', path: ['discord', 'showPaused'], type: 'bool' },
|
|
{ key: 'FS_DISCORD_SHOW_SERVER_IMAGE', path: ['discord', 'showServerImage'], type: 'bool' },
|
|
{ key: 'FS_DISCORD_SHOW_STATE_ICON', path: ['discord', 'showStateIcon'], type: 'bool' },
|
|
{ key: 'FS_LYRICS_FETCH', path: ['lyrics', 'fetch'], type: 'bool' },
|
|
{ key: 'FS_LYRICS_FOLLOW', path: ['lyrics', 'follow'], type: 'bool' },
|
|
{ key: 'FS_LYRICS_DELAY_MS', path: ['lyrics', 'delayMs'], type: 'num' },
|
|
{ key: 'FS_LYRICS_PREFER_LOCAL', path: ['lyrics', 'preferLocalLyrics'], type: 'bool' },
|
|
{ key: 'FS_LYRICS_SHOW_MATCH', path: ['lyrics', 'showMatch'], type: 'bool' },
|
|
{ key: 'FS_LYRICS_SHOW_PROVIDER', path: ['lyrics', 'showProvider'], type: 'bool' },
|
|
{
|
|
key: 'FS_LYRICS_ENABLE_AUTO_TRANSLATION',
|
|
path: ['lyrics', 'enableAutoTranslation'],
|
|
type: 'bool',
|
|
},
|
|
{ key: 'FS_LYRICS_TRANSLATION_API_KEY', path: ['lyrics', 'translationApiKey'], type: 'string' },
|
|
{
|
|
key: 'FS_LYRICS_TRANSLATION_TARGET_LANGUAGE',
|
|
path: ['lyrics', 'translationTargetLanguage'],
|
|
skipIfEmpty: true,
|
|
type: 'string',
|
|
},
|
|
{
|
|
enumSet: LYRICS_ALIGNMENTS,
|
|
key: 'FS_LYRICS_ALIGNMENT',
|
|
path: ['lyrics', 'alignment'],
|
|
type: 'enum',
|
|
},
|
|
{ key: 'FS_AUTO_DJ_ENABLED', path: ['autoDJ', 'enabled'], type: 'bool' },
|
|
{ key: 'FS_AUTO_DJ_ITEM_COUNT', path: ['autoDJ', 'itemCount'], type: 'num' },
|
|
{ key: 'FS_AUTO_DJ_TIMING', path: ['autoDJ', 'timing'], type: 'num' },
|
|
{
|
|
key: 'FS_CSS_CONTENT',
|
|
path: ['css', 'content'],
|
|
transform: (s) => (s.trim() === '' ? undefined : sanitizeCss(`<style>${s}`)),
|
|
type: 'string',
|
|
},
|
|
{ key: 'FS_CSS_ENABLED', path: ['css', 'enabled'], type: 'bool' },
|
|
{ enumSet: FONT_TYPES, key: 'FS_FONT_TYPE', path: ['font', 'type'], type: 'enum' },
|
|
{ key: 'FS_FONT_BUILT_IN', path: ['font', 'builtIn'], skipIfEmpty: true, type: 'string' },
|
|
{
|
|
key: 'FS_FONT_SYSTEM',
|
|
path: ['font', 'system'],
|
|
transform: (s) => (s === '' ? null : s),
|
|
type: 'string',
|
|
},
|
|
];
|
|
|
|
export function getEnvSettingsOverrides(): EnvSettingsOverrides {
|
|
const w = getWin();
|
|
const get = (key: string): string | undefined => {
|
|
const v = w[key];
|
|
if (typeof v !== 'string') return undefined;
|
|
if (isUnsubstitutedPlaceholder(v)) return undefined;
|
|
return v;
|
|
};
|
|
|
|
const overrides: EnvSettingsOverrides = {};
|
|
|
|
for (const spec of ENV_SETTING_SPECS) {
|
|
const raw = get(spec.key);
|
|
const value = parseValue(raw, spec);
|
|
if (value !== undefined) {
|
|
setAtPath(overrides, spec.path, value);
|
|
}
|
|
}
|
|
|
|
return overrides;
|
|
}
|
|
|
|
function getWin(): Record<string, unknown> & Window {
|
|
if (typeof window === 'undefined') return {} as Record<string, unknown> & Window;
|
|
return window as unknown as Record<string, unknown> & Window;
|
|
}
|
|
|
|
function isUnsubstitutedPlaceholder(s: string): boolean {
|
|
return s.length > 0 && s.startsWith('${FS_') && s.endsWith('}');
|
|
}
|
|
|
|
function parseBool(s: string | undefined): boolean | undefined {
|
|
if (s === undefined || s === '') return undefined;
|
|
const lower = s.toLowerCase();
|
|
if (lower === 'true' || lower === '1') return true;
|
|
if (lower === 'false' || lower === '0') return false;
|
|
return undefined;
|
|
}
|
|
|
|
function parseEnum<T extends string>(s: string | undefined, allowed: Set<string>): T | undefined {
|
|
if (s === undefined || s === '') return undefined;
|
|
const v = s.trim();
|
|
return allowed.has(v) ? (v as T) : undefined;
|
|
}
|
|
|
|
function parseNum(s: string | undefined): number | undefined {
|
|
if (s === undefined || s === '') return undefined;
|
|
const n = Number(s);
|
|
return Number.isFinite(n) ? n : undefined;
|
|
}
|
|
|
|
function parseValue(raw: string | undefined, spec: EnvSettingSpec): unknown {
|
|
if (raw === undefined) return undefined;
|
|
if (spec.transform) return spec.transform(raw);
|
|
switch (spec.type) {
|
|
case 'bool':
|
|
return parseBool(raw);
|
|
case 'enum':
|
|
return spec.enumSet ? parseEnum(raw, spec.enumSet) : undefined;
|
|
case 'num':
|
|
return parseNum(raw);
|
|
case 'string':
|
|
if (spec.skipIfEmpty && raw === '') return undefined;
|
|
return raw;
|
|
default:
|
|
return undefined;
|
|
}
|
|
}
|