matrix-react-sdk
Version:
SDK for matrix.org using React
293 lines (277 loc) • 49 kB
JavaScript
;
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.ThemeChoicePanel = ThemeChoicePanel;
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
var _react = _interopRequireWildcard(require("react"));
var _compoundWeb = require("@vector-im/compound-web");
var _delete = _interopRequireDefault(require("@vector-im/compound-design-tokens/assets/web/icons/delete"));
var _classnames = _interopRequireDefault(require("classnames"));
var _logger = require("matrix-js-sdk/src/logger");
var _languageHandler = require("../../../languageHandler");
var _SettingsSubsection = _interopRequireDefault(require("./shared/SettingsSubsection"));
var _ThemeWatcher = _interopRequireDefault(require("../../../settings/watchers/ThemeWatcher"));
var _SettingsStore = _interopRequireDefault(require("../../../settings/SettingsStore"));
var _SettingLevel = require("../../../settings/SettingLevel");
var _dispatcher = _interopRequireDefault(require("../../../dispatcher/dispatcher"));
var _actions = require("../../../dispatcher/actions");
var _useTheme = require("../../../hooks/useTheme");
var _theme2 = require("../../../theme");
var _useSettings = require("../../../hooks/useSettings");
function _getRequireWildcardCache(e) { if ("function" != typeof WeakMap) return null; var r = new WeakMap(), t = new WeakMap(); return (_getRequireWildcardCache = function (e) { return e ? t : r; })(e); }
function _interopRequireWildcard(e, r) { if (!r && e && e.__esModule) return e; if (null === e || "object" != typeof e && "function" != typeof e) return { default: e }; var t = _getRequireWildcardCache(r); if (t && t.has(e)) return t.get(e); var n = { __proto__: null }, a = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var u in e) if ("default" !== u && {}.hasOwnProperty.call(e, u)) { var i = a ? Object.getOwnPropertyDescriptor(e, u) : null; i && (i.get || i.set) ? Object.defineProperty(n, u, i) : n[u] = e[u]; } return n.default = e, t && t.set(e, n), n; }
function ownKeys(e, r) { var t = Object.keys(e); if (Object.getOwnPropertySymbols) { var o = Object.getOwnPropertySymbols(e); r && (o = o.filter(function (r) { return Object.getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; }
function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? ownKeys(Object(t), !0).forEach(function (r) { (0, _defineProperty2.default)(e, r, t[r]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(e, Object.getOwnPropertyDescriptors(t)) : ownKeys(Object(t)).forEach(function (r) { Object.defineProperty(e, r, Object.getOwnPropertyDescriptor(t, r)); }); } return e; } /*
* Copyright 2024 New Vector Ltd.
* Copyright 2024 The Matrix.org Foundation C.I.C.
*
* SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only
* Please see LICENSE files in the repository root for full details.
*/
/**
* Panel to choose the theme
*/
function ThemeChoicePanel() {
const themeState = (0, _useTheme.useTheme)();
const themeWatcher = (0, _react.useRef)(new _ThemeWatcher.default());
const customThemeEnabled = (0, _useSettings.useSettingValue)("feature_custom_themes");
return /*#__PURE__*/_react.default.createElement(_SettingsSubsection.default, {
heading: (0, _languageHandler._t)("common|theme"),
legacy: false,
"data-testid": "themePanel"
}, themeWatcher.current.isSystemThemeSupported() && /*#__PURE__*/_react.default.createElement(SystemTheme, {
systemThemeActivated: themeState.systemThemeActivated
}), /*#__PURE__*/_react.default.createElement(ThemeSelectors, {
theme: themeState.theme,
disabled: themeState.systemThemeActivated
}), customThemeEnabled && /*#__PURE__*/_react.default.createElement(CustomTheme, {
theme: themeState.theme
}));
}
/**
* Component to toggle the system theme
*/
/**
* Component to toggle the system theme
*/
function SystemTheme({
systemThemeActivated
}) {
return /*#__PURE__*/_react.default.createElement(_compoundWeb.Root, {
onChange: async evt => {
const checked = new FormData(evt.currentTarget).get("systemTheme") === "on";
await _SettingsStore.default.setValue("use_system_theme", null, _SettingLevel.SettingLevel.DEVICE, checked);
_dispatcher.default.dispatch({
action: _actions.Action.RecheckTheme
});
}
}, /*#__PURE__*/_react.default.createElement(_compoundWeb.InlineField, {
name: "systemTheme",
control: /*#__PURE__*/_react.default.createElement(_compoundWeb.ToggleControl, {
name: "systemTheme",
defaultChecked: systemThemeActivated
})
}, /*#__PURE__*/_react.default.createElement(_compoundWeb.Label, null, _SettingsStore.default.getDisplayName("use_system_theme"))));
}
/**
* Component to select the theme
*/
/**
* Component to select the theme
*/
function ThemeSelectors({
theme,
disabled
}) {
const themes = useThemes();
return /*#__PURE__*/_react.default.createElement(_compoundWeb.Root, {
className: "mx_ThemeChoicePanel_ThemeSelectors",
onChange: async evt => {
// We don't have any file in the form, we can cast it as string safely
const newTheme = new FormData(evt.currentTarget).get("themeSelector");
// Do nothing if the same theme is selected
if (!newTheme || theme === newTheme) return;
// doing getValue in the .catch will still return the value we failed to set,
_SettingsStore.default.setValue("theme", null, _SettingLevel.SettingLevel.DEVICE, newTheme).catch(() => {
_dispatcher.default.dispatch({
action: _actions.Action.RecheckTheme
});
});
// The settings watcher doesn't fire until the echo comes back from the
// server, so to make the theme change immediately we need to manually
// do the dispatch now
// XXX: The local echoed value appears to be unreliable, in particular
// when settings custom themes(!) so adding forceTheme to override
// the value from settings.
_dispatcher.default.dispatch({
action: _actions.Action.RecheckTheme,
forceTheme: newTheme
});
}
}, themes.map(_theme => {
const isChecked = theme === _theme.id;
return /*#__PURE__*/_react.default.createElement(_compoundWeb.InlineField, {
className: (0, _classnames.default)("mx_ThemeChoicePanel_themeSelector", {
[`mx_ThemeChoicePanel_themeSelector_enabled`]: !disabled && theme === _theme.id,
[`mx_ThemeChoicePanel_themeSelector_disabled`]: disabled,
// We need to force the compound theme to be light or dark
// The theme selection doesn't depend on the current theme
// For example when the light theme is used, the dark theme selector should be dark
"cpd-theme-light": !_theme.isDark,
"cpd-theme-dark": _theme.isDark
}),
name: "themeSelector",
key: _theme.id,
control: /*#__PURE__*/_react.default.createElement(_compoundWeb.RadioControl, {
name: "themeSelector",
checked: !disabled && isChecked,
disabled: disabled,
value: _theme.id
})
}, /*#__PURE__*/_react.default.createElement(_compoundWeb.Label, {
className: "mx_ThemeChoicePanel_themeSelector_Label"
}, _theme.name));
}));
}
/**
* Return all the available themes
*/
function useThemes() {
const customThemes = (0, _useSettings.useSettingValue)("custom_themes");
return (0, _react.useMemo)(() => {
// Put the custom theme into a map
// To easily find the theme by name when going through the themes list
const checkedCustomThemes = customThemes || [];
const customThemeMap = checkedCustomThemes.reduce((map, theme) => map.set(theme.name, theme), new Map());
const themes = (0, _theme2.getOrderedThemes)();
// Separate the built-in themes from the custom themes
// To insert the high contrast theme between them
const builtInThemes = themes.filter(theme => !customThemeMap.has(theme.name));
const otherThemes = themes.filter(theme => customThemeMap.has(theme.name));
const highContrastTheme = makeHighContrastTheme();
if (highContrastTheme) builtInThemes.push(highContrastTheme);
const allThemes = builtInThemes.concat(otherThemes);
// Check if the themes are dark
return allThemes.map(theme => {
const customTheme = customThemeMap.get(theme.name);
const isDark = (customTheme ? customTheme.is_dark : theme.id.includes("dark")) || false;
return _objectSpread(_objectSpread({}, theme), {}, {
isDark
});
});
}, [customThemes]);
}
/**
* Create the light high contrast theme
*/
function makeHighContrastTheme() {
const lightHighContrastId = (0, _theme2.findHighContrastTheme)("light");
if (lightHighContrastId) {
return {
name: (0, _languageHandler._t)("settings|appearance|high_contrast"),
id: lightHighContrastId
};
}
}
/**
* Add and manager custom themes
*/
function CustomTheme({
theme
}) {
const [customTheme, setCustomTheme] = (0, _react.useState)("");
const [error, setError] = (0, _react.useState)();
const clear = (0, _react.useCallback)(() => {
setError(undefined);
setCustomTheme("");
}, [setError, setCustomTheme]);
return /*#__PURE__*/_react.default.createElement("div", {
className: "mx_ThemeChoicePanel_CustomTheme"
}, /*#__PURE__*/_react.default.createElement(_compoundWeb.EditInPlace, {
className: "mx_ThemeChoicePanel_CustomTheme_EditInPlace",
label: (0, _languageHandler._t)("settings|appearance|custom_theme_add"),
cancelButtonLabel: (0, _languageHandler._t)("action|cancel"),
saveButtonLabel: (0, _languageHandler._t)("settings|appearance|custom_theme_add"),
savingLabel: (0, _languageHandler._t)("settings|appearance|custom_theme_downloading"),
value: customTheme,
onChange: e => {
setError(undefined);
setCustomTheme(e.target.value);
},
onSave: async () => {
// The field empty is empty
if (!customTheme) return;
// Get the custom themes and do a cheap clone
// To avoid to mutate the original array in the settings
const currentThemes = _SettingsStore.default.getValue("custom_themes").map(t => t) || [];
try {
const r = await fetch(customTheme);
// XXX: need some schema for this
const themeInfo = await r.json();
if (!themeInfo || typeof themeInfo["name"] !== "string" || typeof themeInfo["colors"] !== "object") {
setError((0, _languageHandler._t)("settings|appearance|custom_theme_invalid"));
return;
}
// Check if the theme is already existing
const isAlreadyExisting = Boolean(currentThemes.find(t => t.name === themeInfo.name));
if (isAlreadyExisting) {
clear();
return;
}
currentThemes.push(themeInfo);
} catch (e) {
_logger.logger.error(e);
setError((0, _languageHandler._t)("settings|appearance|custom_theme_error_downloading"));
return;
}
// Reset the error
clear();
await _SettingsStore.default.setValue("custom_themes", null, _SettingLevel.SettingLevel.ACCOUNT, currentThemes);
},
onCancel: clear
}, /*#__PURE__*/_react.default.createElement(_compoundWeb.HelpMessage, null, (0, _languageHandler._t)("settings|appearance|custom_theme_help")), error && /*#__PURE__*/_react.default.createElement(_compoundWeb.ErrorMessage, null, error)), /*#__PURE__*/_react.default.createElement(CustomThemeList, {
theme: theme
}));
}
/**
* List of the custom themes
*/
function CustomThemeList({
theme: currentTheme
}) {
const customThemes = (0, _useSettings.useSettingValue)("custom_themes") || [];
return /*#__PURE__*/_react.default.createElement("ul", {
className: "mx_ThemeChoicePanel_CustomThemeList"
}, customThemes.map(theme => {
return /*#__PURE__*/_react.default.createElement("li", {
key: theme.name,
className: "mx_ThemeChoicePanel_CustomThemeList_theme",
"aria-label": theme.name
}, /*#__PURE__*/_react.default.createElement("span", {
className: "mx_ThemeChoicePanel_CustomThemeList_name"
}, theme.name), /*#__PURE__*/_react.default.createElement(_compoundWeb.IconButton, {
destructive: true,
"aria-label": (0, _languageHandler._t)("action|delete"),
tooltip: (0, _languageHandler._t)("action|delete"),
onClick: async () => {
// Get the custom themes and do a cheap clone
// To avoid to mutate the original array in the settings
const currentThemes = _SettingsStore.default.getValue("custom_themes").map(t => t) || [];
// Remove the theme from the list
const newThemes = currentThemes.filter(t => t.name !== theme.name);
await _SettingsStore.default.setValue("custom_themes", null, _SettingLevel.SettingLevel.ACCOUNT, newThemes);
// If the delete custom theme is the current theme, reset the theme to the default theme
// By settings the theme at null at the device level, we are getting the default theme
if (currentTheme === `custom-${theme.name}`) {
await _SettingsStore.default.setValue("theme", null, _SettingLevel.SettingLevel.DEVICE, null);
_dispatcher.default.dispatch({
action: _actions.Action.RecheckTheme
});
}
}
}, /*#__PURE__*/_react.default.createElement(_delete.default, null)));
}));
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_react","_interopRequireWildcard","require","_compoundWeb","_delete","_interopRequireDefault","_classnames","_logger","_languageHandler","_SettingsSubsection","_ThemeWatcher","_SettingsStore","_SettingLevel","_dispatcher","_actions","_useTheme","_theme2","_useSettings","_getRequireWildcardCache","e","WeakMap","r","t","__esModule","default","has","get","n","__proto__","a","Object","defineProperty","getOwnPropertyDescriptor","u","hasOwnProperty","call","i","set","ownKeys","keys","getOwnPropertySymbols","o","filter","enumerable","push","apply","_objectSpread","arguments","length","forEach","_defineProperty2","getOwnPropertyDescriptors","defineProperties","ThemeChoicePanel","themeState","useTheme","themeWatcher","useRef","ThemeWatcher","customThemeEnabled","useSettingValue","createElement","heading","_t","legacy","current","isSystemThemeSupported","SystemTheme","systemThemeActivated","ThemeSelectors","theme","disabled","CustomTheme","Root","onChange","evt","checked","FormData","currentTarget","SettingsStore","setValue","SettingLevel","DEVICE","dis","dispatch","action","Action","RecheckTheme","InlineField","name","control","ToggleControl","defaultChecked","Label","getDisplayName","themes","useThemes","className","newTheme","catch","forceTheme","map","_theme","isChecked","id","classNames","isDark","key","RadioControl","value","customThemes","useMemo","checkedCustomThemes","customThemeMap","reduce","Map","getOrderedThemes","builtInThemes","otherThemes","highContrastTheme","makeHighContrastTheme","allThemes","concat","customTheme","is_dark","includes","lightHighContrastId","findHighContrastTheme","setCustomTheme","useState","error","setError","clear","useCallback","undefined","EditInPlace","label","cancelButtonLabel","saveButtonLabel","savingLabel","target","onSave","currentThemes","getValue","fetch","themeInfo","json","isAlreadyExisting","Boolean","find","logger","ACCOUNT","onCancel","HelpMessage","ErrorMessage","CustomThemeList","currentTheme","IconButton","destructive","tooltip","onClick","newThemes"],"sources":["../../../../src/components/views/settings/ThemeChoicePanel.tsx"],"sourcesContent":["/*\n * Copyright 2024 New Vector Ltd.\n * Copyright 2024 The Matrix.org Foundation C.I.C.\n *\n * SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only\n * Please see LICENSE files in the repository root for full details.\n */\n\nimport React, { ChangeEvent, JSX, useCallback, useMemo, useRef, useState } from \"react\";\nimport {\n    InlineField,\n    ToggleControl,\n    Label,\n    Root,\n    RadioControl,\n    EditInPlace,\n    IconButton,\n    ErrorMessage,\n    HelpMessage,\n} from \"@vector-im/compound-web\";\nimport DeleteIcon from \"@vector-im/compound-design-tokens/assets/web/icons/delete\";\nimport classNames from \"classnames\";\nimport { logger } from \"matrix-js-sdk/src/logger\";\n\nimport { _t } from \"../../../languageHandler\";\nimport SettingsSubsection from \"./shared/SettingsSubsection\";\nimport ThemeWatcher from \"../../../settings/watchers/ThemeWatcher\";\nimport SettingsStore from \"../../../settings/SettingsStore\";\nimport { SettingLevel } from \"../../../settings/SettingLevel\";\nimport dis from \"../../../dispatcher/dispatcher\";\nimport { RecheckThemePayload } from \"../../../dispatcher/payloads/RecheckThemePayload\";\nimport { Action } from \"../../../dispatcher/actions\";\nimport { useTheme } from \"../../../hooks/useTheme\";\nimport { findHighContrastTheme, getOrderedThemes, CustomTheme as CustomThemeType, ITheme } from \"../../../theme\";\nimport { useSettingValue } from \"../../../hooks/useSettings\";\n\n/**\n * Panel to choose the theme\n */\nexport function ThemeChoicePanel(): JSX.Element {\n    const themeState = useTheme();\n    const themeWatcher = useRef(new ThemeWatcher());\n    const customThemeEnabled = useSettingValue<boolean>(\"feature_custom_themes\");\n\n    return (\n        <SettingsSubsection heading={_t(\"common|theme\")} legacy={false} data-testid=\"themePanel\">\n            {themeWatcher.current.isSystemThemeSupported() && (\n                <SystemTheme systemThemeActivated={themeState.systemThemeActivated} />\n            )}\n            <ThemeSelectors theme={themeState.theme} disabled={themeState.systemThemeActivated} />\n            {customThemeEnabled && <CustomTheme theme={themeState.theme} />}\n        </SettingsSubsection>\n    );\n}\n\n/**\n * Component to toggle the system theme\n */\ninterface SystemThemeProps {\n    /* Whether the system theme is activated */\n    systemThemeActivated: boolean;\n}\n\n/**\n * Component to toggle the system theme\n */\nfunction SystemTheme({ systemThemeActivated }: SystemThemeProps): JSX.Element {\n    return (\n        <Root\n            onChange={async (evt) => {\n                const checked = new FormData(evt.currentTarget).get(\"systemTheme\") === \"on\";\n                await SettingsStore.setValue(\"use_system_theme\", null, SettingLevel.DEVICE, checked);\n                dis.dispatch<RecheckThemePayload>({ action: Action.RecheckTheme });\n            }}\n        >\n            <InlineField\n                name=\"systemTheme\"\n                control={<ToggleControl name=\"systemTheme\" defaultChecked={systemThemeActivated} />}\n            >\n                <Label>{SettingsStore.getDisplayName(\"use_system_theme\")}</Label>\n            </InlineField>\n        </Root>\n    );\n}\n\n/**\n * Component to select the theme\n */\ninterface ThemeSelectorProps {\n    /* The current theme */\n    theme: string;\n    /* The theme can't be selected */\n    disabled: boolean;\n}\n\n/**\n * Component to select the theme\n */\nfunction ThemeSelectors({ theme, disabled }: ThemeSelectorProps): JSX.Element {\n    const themes = useThemes();\n\n    return (\n        <Root\n            className=\"mx_ThemeChoicePanel_ThemeSelectors\"\n            onChange={async (evt) => {\n                // We don't have any file in the form, we can cast it as string safely\n                const newTheme = new FormData(evt.currentTarget).get(\"themeSelector\") as string | null;\n\n                // Do nothing if the same theme is selected\n                if (!newTheme || theme === newTheme) return;\n\n                // doing getValue in the .catch will still return the value we failed to set,\n                SettingsStore.setValue(\"theme\", null, SettingLevel.DEVICE, newTheme).catch(() => {\n                    dis.dispatch<RecheckThemePayload>({ action: Action.RecheckTheme });\n                });\n\n                // The settings watcher doesn't fire until the echo comes back from the\n                // server, so to make the theme change immediately we need to manually\n                // do the dispatch now\n                // XXX: The local echoed value appears to be unreliable, in particular\n                // when settings custom themes(!) so adding forceTheme to override\n                // the value from settings.\n                dis.dispatch<RecheckThemePayload>({ action: Action.RecheckTheme, forceTheme: newTheme });\n            }}\n        >\n            {themes.map((_theme) => {\n                const isChecked = theme === _theme.id;\n                return (\n                    <InlineField\n                        className={classNames(\"mx_ThemeChoicePanel_themeSelector\", {\n                            [`mx_ThemeChoicePanel_themeSelector_enabled`]: !disabled && theme === _theme.id,\n                            [`mx_ThemeChoicePanel_themeSelector_disabled`]: disabled,\n                            // We need to force the compound theme to be light or dark\n                            // The theme selection doesn't depend on the current theme\n                            // For example when the light theme is used, the dark theme selector should be dark\n                            \"cpd-theme-light\": !_theme.isDark,\n                            \"cpd-theme-dark\": _theme.isDark,\n                        })}\n                        name=\"themeSelector\"\n                        key={_theme.id}\n                        control={\n                            <RadioControl\n                                name=\"themeSelector\"\n                                checked={!disabled && isChecked}\n                                disabled={disabled}\n                                value={_theme.id}\n                            />\n                        }\n                    >\n                        <Label className=\"mx_ThemeChoicePanel_themeSelector_Label\">{_theme.name}</Label>\n                    </InlineField>\n                );\n            })}\n        </Root>\n    );\n}\n\n/**\n * Return all the available themes\n */\nfunction useThemes(): Array<ITheme & { isDark: boolean }> {\n    const customThemes = useSettingValue<CustomThemeType[] | undefined>(\"custom_themes\");\n    return useMemo(() => {\n        // Put the custom theme into a map\n        // To easily find the theme by name when going through the themes list\n        const checkedCustomThemes = customThemes || [];\n        const customThemeMap = checkedCustomThemes.reduce(\n            (map, theme) => map.set(theme.name, theme),\n            new Map<string, CustomThemeType>(),\n        );\n\n        const themes = getOrderedThemes();\n        // Separate the built-in themes from the custom themes\n        // To insert the high contrast theme between them\n        const builtInThemes = themes.filter((theme) => !customThemeMap.has(theme.name));\n        const otherThemes = themes.filter((theme) => customThemeMap.has(theme.name));\n\n        const highContrastTheme = makeHighContrastTheme();\n        if (highContrastTheme) builtInThemes.push(highContrastTheme);\n\n        const allThemes = builtInThemes.concat(otherThemes);\n\n        // Check if the themes are dark\n        return allThemes.map((theme) => {\n            const customTheme = customThemeMap.get(theme.name);\n            const isDark = (customTheme ? customTheme.is_dark : theme.id.includes(\"dark\")) || false;\n            return { ...theme, isDark };\n        });\n    }, [customThemes]);\n}\n\n/**\n * Create the light high contrast theme\n */\nfunction makeHighContrastTheme(): ITheme | undefined {\n    const lightHighContrastId = findHighContrastTheme(\"light\");\n    if (lightHighContrastId) {\n        return {\n            name: _t(\"settings|appearance|high_contrast\"),\n            id: lightHighContrastId,\n        };\n    }\n}\n\ninterface CustomThemeProps {\n    /**\n     * The current theme\n     */\n    theme: string;\n}\n\n/**\n * Add and manager custom themes\n */\nfunction CustomTheme({ theme }: CustomThemeProps): JSX.Element {\n    const [customTheme, setCustomTheme] = useState<string>(\"\");\n    const [error, setError] = useState<string>();\n    const clear = useCallback(() => {\n        setError(undefined);\n        setCustomTheme(\"\");\n    }, [setError, setCustomTheme]);\n\n    return (\n        <div className=\"mx_ThemeChoicePanel_CustomTheme\">\n            <EditInPlace\n                className=\"mx_ThemeChoicePanel_CustomTheme_EditInPlace\"\n                label={_t(\"settings|appearance|custom_theme_add\")}\n                cancelButtonLabel={_t(\"action|cancel\")}\n                saveButtonLabel={_t(\"settings|appearance|custom_theme_add\")}\n                savingLabel={_t(\"settings|appearance|custom_theme_downloading\")}\n                value={customTheme}\n                onChange={(e: ChangeEvent<HTMLInputElement>) => {\n                    setError(undefined);\n                    setCustomTheme(e.target.value);\n                }}\n                onSave={async () => {\n                    // The field empty is empty\n                    if (!customTheme) return;\n\n                    // Get the custom themes and do a cheap clone\n                    // To avoid to mutate the original array in the settings\n                    const currentThemes =\n                        SettingsStore.getValue<CustomThemeType[]>(\"custom_themes\").map((t) => t) || [];\n\n                    try {\n                        const r = await fetch(customTheme);\n                        // XXX: need some schema for this\n                        const themeInfo = await r.json();\n                        if (\n                            !themeInfo ||\n                            typeof themeInfo[\"name\"] !== \"string\" ||\n                            typeof themeInfo[\"colors\"] !== \"object\"\n                        ) {\n                            setError(_t(\"settings|appearance|custom_theme_invalid\"));\n                            return;\n                        }\n\n                        // Check if the theme is already existing\n                        const isAlreadyExisting = Boolean(currentThemes.find((t) => t.name === themeInfo.name));\n                        if (isAlreadyExisting) {\n                            clear();\n                            return;\n                        }\n\n                        currentThemes.push(themeInfo);\n                    } catch (e) {\n                        logger.error(e);\n                        setError(_t(\"settings|appearance|custom_theme_error_downloading\"));\n                        return;\n                    }\n\n                    // Reset the error\n                    clear();\n                    await SettingsStore.setValue(\"custom_themes\", null, SettingLevel.ACCOUNT, currentThemes);\n                }}\n                onCancel={clear}\n            >\n                <HelpMessage>{_t(\"settings|appearance|custom_theme_help\")}</HelpMessage>\n                {error && <ErrorMessage>{error}</ErrorMessage>}\n            </EditInPlace>\n            <CustomThemeList theme={theme} />\n        </div>\n    );\n}\n\ninterface CustomThemeListProps {\n    /*\n     * The current theme\n     */\n    theme: string;\n}\n\n/**\n * List of the custom themes\n */\nfunction CustomThemeList({ theme: currentTheme }: CustomThemeListProps): JSX.Element {\n    const customThemes = useSettingValue<CustomThemeType[]>(\"custom_themes\") || [];\n\n    return (\n        <ul className=\"mx_ThemeChoicePanel_CustomThemeList\">\n            {customThemes.map((theme) => {\n                return (\n                    <li key={theme.name} className=\"mx_ThemeChoicePanel_CustomThemeList_theme\" aria-label={theme.name}>\n                        <span className=\"mx_ThemeChoicePanel_CustomThemeList_name\">{theme.name}</span>\n                        <IconButton\n                            destructive={true}\n                            aria-label={_t(\"action|delete\")}\n                            tooltip={_t(\"action|delete\")}\n                            onClick={async () => {\n                                // Get the custom themes and do a cheap clone\n                                // To avoid to mutate the original array in the settings\n                                const currentThemes =\n                                    SettingsStore.getValue<CustomThemeType[]>(\"custom_themes\").map((t) => t) || [];\n\n                                // Remove the theme from the list\n                                const newThemes = currentThemes.filter((t) => t.name !== theme.name);\n                                await SettingsStore.setValue(\"custom_themes\", null, SettingLevel.ACCOUNT, newThemes);\n\n                                // If the delete custom theme is the current theme, reset the theme to the default theme\n                                // By settings the theme at null at the device level, we are getting the default theme\n                                if (currentTheme === `custom-${theme.name}`) {\n                                    await SettingsStore.setValue(\"theme\", null, SettingLevel.DEVICE, null);\n                                    dis.dispatch<RecheckThemePayload>({\n                                        action: Action.RecheckTheme,\n                                    });\n                                }\n                            }}\n                        >\n                            <DeleteIcon />\n                        </IconButton>\n                    </li>\n                );\n            })}\n        </ul>\n    );\n}\n"],"mappings":"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","ignoreList":[]}