import React, { 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 snackbar from 'soapbox/actions/snackbar'; import { fetchSoapboxConfig } from 'soapbox/actions/soapbox'; import List, { ListItem } from 'soapbox/components/list'; import { Button, Column, Form, FormActions } from 'soapbox/components/ui'; import DropdownMenuContainer from 'soapbox/containers/dropdown-menu-container'; import { useAppDispatch, useAppSelector, useSoapboxConfig } from 'soapbox/hooks'; import { download } from 'soapbox/utils/download'; import Palette, { ColorGroup } from './components/palette'; const messages = defineMessages({ title: { id: 'admin.theme.title', defaultMessage: 'Theme' }, saved: { id: 'theme_editor.saved', defaultMessage: 'Theme updated!' }, export: { id: 'theme_editor.export', defaultMessage: 'Export theme' }, }); 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 updateColors = (key: string) => { return (newColors: ColorGroup) => { setColors({ ...colors, [key]: { ...colors[key], ...newColors, }, }); }; }; const resetTheme = () => { setResetKey(uuidv4()); setTimeout(() => setColors(soapbox.colors.toJS() as any)); }; const updateTheme = async () => { const params = rawConfig.set('colors', colors).toJS(); await dispatch(updateSoapboxConfig(params)); }; const exportTheme = () => { const data = JSON.stringify(colors, null, 2); download(data, 'theme.json'); }; const handleSubmit = async() => { setSubmitting(true); try { await dispatch(fetchSoapboxConfig(host)); await updateTheme(); dispatch(snackbar.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}}> ); }; export default ThemeEditor;