import React, { useRef, useState } from 'react'; import { defineMessages, FormattedMessage, useIntl } from 'react-intl'; import { v4 as uuidv4 } from 'uuid'; import { updateSoapboxConfig } from 'soapbox/actions/admin'; import { getHost } from 'soapbox/actions/instance'; import { fetchSoapboxConfig } from 'soapbox/actions/soapbox'; import DropdownMenu from 'soapbox/components/dropdown-menu'; import List, { ListItem } from 'soapbox/components/list'; import { Button, Column, Form, FormActions } from 'soapbox/components/ui'; import ColorWithPicker from 'soapbox/features/soapbox-config/components/color-with-picker'; import { useAppDispatch, useAppSelector, useSoapboxConfig } from 'soapbox/hooks'; import { normalizeSoapboxConfig } from 'soapbox/normalizers'; import toast from 'soapbox/toast'; import { download } from 'soapbox/utils/download'; import Palette, { ColorGroup } from './components/palette'; import type { ColorChangeHandler } from 'react-color'; const messages = defineMessages({ title: { id: 'admin.theme.title', defaultMessage: 'Theme' }, saved: { id: 'theme_editor.saved', defaultMessage: 'Theme updated!' }, restore: { id: 'theme_editor.restore', defaultMessage: 'Restore default theme' }, export: { id: 'theme_editor.export', defaultMessage: 'Export theme' }, import: { id: 'theme_editor.import', defaultMessage: 'Import theme' }, importSuccess: { id: 'theme_editor.import_success', defaultMessage: 'Theme was successfully imported!' }, }); interface IThemeEditor { } /** UI for editing Tailwind theme colors. */ const ThemeEditor: React.FC = () => { const intl = useIntl(); const dispatch = useAppDispatch(); const soapbox = useSoapboxConfig(); const host = useAppSelector(state => getHost(state)); const rawConfig = useAppSelector(state => state.soapbox); const [colors, setColors] = useState(soapbox.colors.toJS() as any); const [submitting, setSubmitting] = useState(false); const [resetKey, setResetKey] = useState(uuidv4()); const fileInput = useRef(null); const updateColors = (key: string) => { return (newColors: ColorGroup) => { setColors({ ...colors, [key]: { ...colors[key], ...newColors, }, }); }; }; const updateColor = (key: string) => { return (hex: string) => { setColors({ ...colors, [key]: hex, }); }; }; const setTheme = (theme: any) => { setResetKey(uuidv4()); setTimeout(() => setColors(theme)); }; const resetTheme = () => { setTheme(soapbox.colors.toJS() as any); }; const updateTheme = async () => { const params = rawConfig.set('colors', colors).toJS(); await dispatch(updateSoapboxConfig(params)); }; const restoreDefaultTheme = () => { const colors = normalizeSoapboxConfig({ brandColor: '#0482d8' }).colors.toJS(); setTheme(colors); }; const exportTheme = () => { const data = JSON.stringify(colors, null, 2); download(data, 'theme.json'); }; const importTheme = () => { fileInput.current?.click(); }; const handleSelectFile: React.ChangeEventHandler = async (e) => { const file = e.target.files?.item(0); if (file) { const text = await file.text(); const json = JSON.parse(text); const colors = normalizeSoapboxConfig({ colors: json }).colors.toJS(); setTheme(colors); toast.success(intl.formatMessage(messages.importSuccess)); } }; const handleSubmit = async() => { setSubmitting(true); try { await dispatch(fetchSoapboxConfig(host)); await updateTheme(); toast.success(intl.formatMessage(messages.saved)); setSubmitting(false); } catch (e) { setSubmitting(false); } }; return (
); }; interface IPaletteListItem { label: React.ReactNode, palette: ColorGroup, onChange: (palette: ColorGroup) => void, resetKey?: string, } /** Palette editor inside a ListItem. */ const PaletteListItem: React.FC = ({ label, palette, onChange, resetKey }) => { return ( {label}}> ); }; interface IColorListItem { label: React.ReactNode, value: string, onChange: (hex: string) => void, } /** Single-color picker. */ const ColorListItem: React.FC = ({ label, value, onChange }) => { const handleChange: ColorChangeHandler = (color, _e) => { onChange(color.hex); }; return ( ); }; export default ThemeEditor;