diff --git a/src/components/ThemeSettings/UserCSSModal.tsx b/src/components/ThemeSettings/UserCSSModal.tsx index 9880c3b20..6cf3c9b15 100644 --- a/src/components/ThemeSettings/UserCSSModal.tsx +++ b/src/components/ThemeSettings/UserCSSModal.tsx @@ -6,9 +6,10 @@ import { useSettings } from "@api/Settings"; import { Flex } from "@components/Flex"; +import { copyWithToast } from "@utils/misc"; import { ModalCloseButton, ModalContent, ModalHeader, ModalProps, ModalRoot } from "@utils/modal"; -import { Text } from "@webpack/common"; -import type { ReactNode } from "react"; +import { Button, showToast, Text, Toasts } from "@webpack/common"; +import { type ReactNode } from "react"; import { UserstyleHeader } from "usercss-meta"; import { SettingBooleanComponent, SettingColorComponent, SettingNumberComponent, SettingRangeComponent, SettingSelectComponent, SettingTextComponent } from "./components"; @@ -107,7 +108,36 @@ export function UserCSSSettingsModal({ modalProps, theme }: UserCSSSettingsModal - {controls} + + + { + copyWithToast(JSON.stringify(themeSettings), "Copied theme settings to clipboard."); + }}>Export + { + const clip = (await navigator.clipboard.read())[0]; + + if (!clip) return showToast("Your clipboard is empty.", Toasts.Type.FAILURE); + + if (!clip.types.includes("text/plain")) + return showToast("Your clipboard doesn't have valid settings data.", Toasts.Type.FAILURE); + + try { + var potentialSettings: Record = + JSON.parse(await clip.getType("text/plain").then(b => b.text())); + } catch (e) { + return showToast("Your clipboard doesn't have valid settings data.", Toasts.Type.FAILURE); + } + + for (const [key, value] of Object.entries(potentialSettings)) { + if (Object.prototype.hasOwnProperty.call(themeSettings, key)) + themeSettings[key] = value; + } + + showToast("Imported theme settings from clipboard.", Toasts.Type.SUCCESS); + }}>Import + + {controls} + ); diff --git a/src/components/ThemeSettings/components/SettingBooleanComponent.tsx b/src/components/ThemeSettings/components/SettingBooleanComponent.tsx index a0d3239b6..6b49ae43a 100644 --- a/src/components/ThemeSettings/components/SettingBooleanComponent.tsx +++ b/src/components/ThemeSettings/components/SettingBooleanComponent.tsx @@ -4,7 +4,7 @@ * SPDX-License-Identifier: GPL-3.0-or-later */ -import { Forms, Switch, useState } from "@webpack/common"; +import { Forms, Switch } from "@webpack/common"; interface Props { label: string; @@ -13,13 +13,9 @@ interface Props { } export function SettingBooleanComponent({ label, name, themeSettings }: Props) { - const [value, setValue] = useState(themeSettings[name]); - function handleChange(value: boolean) { const corrected = value ? "1" : "0"; - setValue(corrected); - themeSettings[name] = corrected; } @@ -27,7 +23,7 @@ export function SettingBooleanComponent({ label, name, themeSettings }: Props) { parseInt(TinyColor(value).toHex(), 16), [value]); + const normalizedValue = useMemo(() => parseInt(TinyColor(themeSettings[name]).toHex(), 16), [themeSettings[name]]); return ( diff --git a/src/components/ThemeSettings/components/SettingNumberComponent.tsx b/src/components/ThemeSettings/components/SettingNumberComponent.tsx index 35f57ed44..d59c21e0d 100644 --- a/src/components/ThemeSettings/components/SettingNumberComponent.tsx +++ b/src/components/ThemeSettings/components/SettingNumberComponent.tsx @@ -4,7 +4,7 @@ * SPDX-License-Identifier: GPL-3.0-or-later */ -import { Forms, TextInput, useState } from "@webpack/common"; +import { Forms, TextInput } from "@webpack/common"; interface Props { label: string; @@ -13,11 +13,7 @@ interface Props { } export function SettingNumberComponent({ label, name, themeSettings }: Props) { - const [value, setValue] = useState(themeSettings[name]); - function handleChange(value: string) { - setValue(value); - themeSettings[name] = value; } @@ -28,7 +24,7 @@ export function SettingNumberComponent({ label, name, themeSettings }: Props) { type="number" pattern="-?[0-9]+" key={name} - value={value} + value={themeSettings[name]} onChange={handleChange} /> diff --git a/src/components/ThemeSettings/components/SettingRangeComponent.tsx b/src/components/ThemeSettings/components/SettingRangeComponent.tsx index 6bb15ff8b..b6437582d 100644 --- a/src/components/ThemeSettings/components/SettingRangeComponent.tsx +++ b/src/components/ThemeSettings/components/SettingRangeComponent.tsx @@ -4,7 +4,7 @@ * SPDX-License-Identifier: GPL-3.0-or-later */ -import { Forms, Slider, useMemo, useState } from "@webpack/common"; +import { Forms, Slider, useMemo } from "@webpack/common"; interface Props { label: string; @@ -17,13 +17,9 @@ interface Props { } export function SettingRangeComponent({ label, name, default: def, min, max, step, themeSettings }: Props) { - const [value, setValue] = useState(themeSettings[name]); - function handleChange(value: number) { const corrected = value.toString(); - setValue(corrected); - themeSettings[name] = corrected; } @@ -42,7 +38,7 @@ export function SettingRangeComponent({ label, name, default: def, min, max, ste {label} v === value} + isSelected={v => v === themeSettings[name]} serialize={identity} /> diff --git a/src/components/ThemeSettings/components/SettingTextComponent.tsx b/src/components/ThemeSettings/components/SettingTextComponent.tsx index a568d6336..be3ccf42e 100644 --- a/src/components/ThemeSettings/components/SettingTextComponent.tsx +++ b/src/components/ThemeSettings/components/SettingTextComponent.tsx @@ -4,7 +4,7 @@ * SPDX-License-Identifier: GPL-3.0-or-later */ -import { Forms, TextInput, useState } from "@webpack/common"; +import { Forms, TextInput } from "@webpack/common"; interface Props { label: string; @@ -13,11 +13,7 @@ interface Props { } export function SettingTextComponent({ label, name, themeSettings }: Props) { - const [value, setValue] = useState(themeSettings[name]); - function handleChange(value: string) { - setValue(value); - themeSettings[name] = value; } @@ -26,7 +22,7 @@ export function SettingTextComponent({ label, name, themeSettings }: Props) { {label} diff --git a/src/components/ThemeSettings/themesStyles.css b/src/components/ThemeSettings/themesStyles.css index 6038274f8..62e95f001 100644 --- a/src/components/ThemeSettings/themesStyles.css +++ b/src/components/ThemeSettings/themesStyles.css @@ -27,3 +27,9 @@ .vc-settings-theme-author::before { content: "by "; } + +.vc-settings-usercss-ie-grid { + display: grid; + grid-template-columns: repeat(2, 1fr); + grid-gap: 12px; /* matching the flex gap */ +}