import { Map as ImmutableMap, List as ImmutableList, fromJS } from 'immutable'; import React, { useState, useEffect, useMemo } from 'react'; import { defineMessages, useIntl, FormattedMessage } from 'react-intl'; import { updateConfig } from 'soapbox/actions/admin'; import { uploadMedia } from 'soapbox/actions/media'; import snackbar from 'soapbox/actions/snackbar'; import List, { ListItem } from 'soapbox/components/list'; import { Column, CardHeader, CardTitle, Form, FormActions, FormGroup, Input, FileInput, Textarea, Button, Toggle, } from 'soapbox/components/ui'; import Streamfield from 'soapbox/components/ui/streamfield/streamfield'; import ThemeSelector from 'soapbox/features/ui/components/theme-selector'; import { useAppSelector, useAppDispatch } from 'soapbox/hooks'; import { normalizeSoapboxConfig } from 'soapbox/normalizers'; import Accordion from '../ui/components/accordion'; import ColorWithPicker from './components/color-with-picker'; import CryptoAddressInput from './components/crypto-address-input'; import FooterLinkInput from './components/footer-link-input'; import PromoPanelInput from './components/promo-panel-input'; import SitePreview from './components/site-preview'; import type { ColorChangeHandler, ColorResult } from 'react-color'; const messages = defineMessages({ heading: { id: 'column.soapbox_config', defaultMessage: 'Soapbox config' }, saved: { id: 'soapbox_config.saved', defaultMessage: 'Soapbox config saved!' }, copyrightFooterLabel: { id: 'soapbox_config.copyright_footer.meta_fields.label_placeholder', defaultMessage: 'Copyright footer' }, cryptoDonatePanelLimitLabel: { id: 'soapbox_config.crypto_donate_panel_limit.meta_fields.limit_placeholder', defaultMessage: 'Number of items to display in the crypto homepage widget' }, customCssLabel: { id: 'soapbox_config.custom_css.meta_fields.url_placeholder', defaultMessage: 'URL' }, rawJSONLabel: { id: 'soapbox_config.raw_json_label', defaultMessage: 'Advanced: Edit raw JSON data' }, rawJSONHint: { id: 'soapbox_config.raw_json_hint', defaultMessage: 'Edit the settings data directly. Changes made directly to the JSON file will override the form fields above. Click "Save" to apply your changes.' }, verifiedCanEditNameLabel: { id: 'soapbox_config.verified_can_edit_name_label', defaultMessage: 'Allow verified users to edit their own display name.' }, displayFqnLabel: { id: 'soapbox_config.display_fqn_label', defaultMessage: 'Display domain (eg @user@domain) for local accounts.' }, greentextLabel: { id: 'soapbox_config.greentext_label', defaultMessage: 'Enable greentext support' }, promoPanelIconsLink: { id: 'soapbox_config.hints.promo_panel_icons.link', defaultMessage: 'Soapbox Icons List' }, authenticatedProfileLabel: { id: 'soapbox_config.authenticated_profile_label', defaultMessage: 'Profiles require authentication' }, authenticatedProfileHint: { id: 'soapbox_config.authenticated_profile_hint', defaultMessage: 'Users must be logged-in to view replies and media on user profiles.' }, singleUserModeLabel: { id: 'soapbox_config.single_user_mode_label', defaultMessage: 'Single user mode' }, singleUserModeHint: { id: 'soapbox_config.single_user_mode_hint', defaultMessage: 'Front page will redirect to a given user profile.' }, singleUserModeProfileLabel: { id: 'soapbox_config.single_user_mode_profile_label', defaultMessage: 'Main user handle' }, singleUserModeProfileHint: { id: 'soapbox_config.single_user_mode_profile_hint', defaultMessage: '@handle' }, }); type ValueGetter = (e: React.ChangeEvent) => any; type ColorValueGetter = (color: ColorResult, event: React.ChangeEvent) => any; type Template = ImmutableMap; type ConfigPath = Array; type ThemeChangeHandler = (theme: string) => void; const templates: Record = { promoPanelItem: ImmutableMap({ icon: '', text: '', url: '' }), footerItem: ImmutableMap({ title: '', url: '' }), cryptoAddress: ImmutableMap({ ticker: '', address: '', note: '' }), }; const SoapboxConfig: React.FC = () => { const intl = useIntl(); const dispatch = useAppDispatch(); const initialData = useAppSelector(state => state.soapbox); const [isLoading, setLoading] = useState(false); const [data, setData] = useState(initialData); const [jsonEditorExpanded, setJsonEditorExpanded] = useState(false); const [rawJSON, setRawJSON] = useState(JSON.stringify(initialData, null, 2)); const [jsonValid, setJsonValid] = useState(true); const soapbox = useMemo(() => { return normalizeSoapboxConfig(data); }, [data]); const setConfig = (path: ConfigPath, value: any) => { const newData = data.setIn(path, value); setData(newData); setJsonValid(true); }; const putConfig = (newData: any) => { setData(newData); setJsonValid(true); }; const getParams = () => { return [{ group: ':pleroma', key: ':frontend_configurations', value: [{ tuple: [':soapbox_fe', data.toJS()], }], }]; }; const handleSubmit: React.FormEventHandler = (e) => { dispatch(updateConfig(getParams())).then(() => { setLoading(false); dispatch(snackbar.success(intl.formatMessage(messages.saved))); }).catch(() => { setLoading(false); }); setLoading(true); e.preventDefault(); }; const handleChange = (path: ConfigPath, getValue: ValueGetter): React.ChangeEventHandler => { return e => { setConfig(path, getValue(e)); }; }; const handleThemeChange = (path: ConfigPath): ThemeChangeHandler => { return theme => { setConfig(path, theme); }; }; const handleColorChange = (path: ConfigPath, getValue: ColorValueGetter): ColorChangeHandler => { return (color, event) => { setConfig(path, getValue(color, event)); }; }; const handleFileChange = (path: ConfigPath): React.ChangeEventHandler => { return e => { const data = new FormData(); const file = e.target.files?.item(0); if (file) { data.append('file', file); dispatch(uploadMedia(data)).then(({ data }: any) => { handleChange(path, () => data.url)(e); }).catch(console.error); } }; }; const handleStreamItemChange = (path: ConfigPath) => { return (values: any[]) => { setConfig(path, ImmutableList(values)); }; }; const addStreamItem = (path: ConfigPath, template: Template) => { return () => { const items = data.getIn(path) || ImmutableList(); setConfig(path, items.push(template)); }; }; const deleteStreamItem = (path: ConfigPath) => { return (i: number) => { const newData = data.deleteIn([...path, i]); setData(newData); }; }; const handleEditJSON: React.ChangeEventHandler = e => { setRawJSON(e.target.value); }; const toggleJSONEditor = (expanded: boolean) => setJsonEditorExpanded(expanded); useEffect(() => { putConfig(initialData); }, [initialData]); useEffect(() => { setRawJSON(JSON.stringify(data, null, 2)); }, [data]); useEffect(() => { try { const data = fromJS(JSON.parse(rawJSON)); putConfig(data); } catch { setJsonValid(false); } }, [rawJSON]); return (
} hintText={ } /> }> }> color.hex)} /> }> color.hex)} /> } /> e.target.checked)} /> e.target.checked)} /> e.target.checked)} /> e.target.checked)} /> e.target.checked)} /> {soapbox.get('singleUserMode') && ( e.target.value)} /> )} } /> } hint={} component={PromoPanelInput} values={soapbox.promoPanel.items.toArray()} onChange={handleStreamItemChange(['promoPanel', 'items'])} onAddItem={addStreamItem(['promoPanel', 'items'], templates.promoPanel)} onRemoveItem={deleteStreamItem(['promoPanel', 'items'])} /> } hint={} component={FooterLinkInput} values={soapbox.navlinks.get('homeFooter')?.toArray() || []} onChange={handleStreamItemChange(['navlinks', 'homeFooter'])} onAddItem={addStreamItem(['navlinks', 'homeFooter'], templates.footerItem)} onRemoveItem={deleteStreamItem(['navlinks', 'homeFooter'])} /> e.target.value)} /> } /> } hint={} component={CryptoAddressInput} values={soapbox.cryptoAddresses.toArray()} onChange={handleStreamItemChange(['cryptoAddresses'])} onAddItem={addStreamItem(['cryptoAddresses'], templates.cryptoAddress)} onRemoveItem={deleteStreamItem(['cryptoAddresses'])} /> Number(e.target.value))} /> } />