2022-05-05 12:25:57 -07:00
import { Map as ImmutableMap , List as ImmutableList , fromJS } from 'immutable' ;
import React , { useState , useEffect , useMemo } from 'react' ;
import { defineMessages , useIntl , FormattedMessage } from 'react-intl' ;
2022-12-17 11:05:50 -08:00
import { updateSoapboxConfig } from 'soapbox/actions/admin' ;
2022-05-05 12:25:57 -07:00
import { uploadMedia } from 'soapbox/actions/media' ;
2022-05-05 16:30:25 -07:00
import List , { ListItem } from 'soapbox/components/list' ;
2022-05-05 16:46:16 -07:00
import {
2022-11-16 09:30:24 -08:00
Accordion ,
2022-11-25 10:28:43 -08:00
Button ,
2022-05-05 16:46:16 -07:00
Column ,
CardHeader ,
CardTitle ,
2022-11-25 10:28:43 -08:00
FileInput ,
2022-05-05 16:46:16 -07:00
Form ,
FormActions ,
FormGroup ,
Input ,
2022-11-25 10:28:43 -08:00
Streamfield ,
2022-05-05 16:46:16 -07:00
Textarea ,
2022-05-06 12:55:16 -07:00
Toggle ,
2022-05-05 16:46:16 -07:00
} from 'soapbox/components/ui' ;
2022-05-05 16:12:08 -07:00
import ThemeSelector from 'soapbox/features/ui/components/theme-selector' ;
2022-12-08 07:30:48 -08:00
import { useAppSelector , useAppDispatch , useFeatures } from 'soapbox/hooks' ;
2022-05-05 12:25:57 -07:00
import { normalizeSoapboxConfig } from 'soapbox/normalizers' ;
2022-12-20 07:47:46 -08:00
import toast from 'soapbox/toast' ;
2022-05-05 12:25:57 -07:00
2022-05-05 13:12:38 -07:00
import CryptoAddressInput from './components/crypto-address-input' ;
2022-05-05 14:18:43 -07:00
import FooterLinkInput from './components/footer-link-input' ;
2022-05-05 13:57:30 -07:00
import PromoPanelInput from './components/promo-panel-input' ;
2022-05-05 12:25:57 -07:00
import SitePreview from './components/site-preview' ;
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.' } ,
2023-01-10 10:41:32 -08:00
rawJSONInvalid : { id : 'soapbox_config.raw_json_invalid' , defaultMessage : 'is invalid' } ,
2022-05-05 12:25:57 -07:00
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.' } ,
2022-10-09 15:41:47 -07:00
displayCtaLabel : { id : 'soapbox_config.cta_label' , defaultMessage : 'Display call to action panels if not authenticated' } ,
2023-01-02 09:52:05 -08:00
mediaPreviewLabel : { id : 'soapbox_config.media_preview_label' , defaultMessage : 'Prefer preview media for thumbnails' } ,
mediaPreviewHint : { id : 'soapbox_config.media_preview_hint' , defaultMessage : 'Some backends provide an optimized version of media for display in timelines. However, these preview images may be too small without additional configuration.' } ,
2022-11-27 19:05:39 -08:00
feedInjectionLabel : { id : 'soapbox_config.feed_injection_label' , defaultMessage : 'Feed injection' } ,
feedInjectionHint : { id : 'soapbox_config.feed_injection_hint' , defaultMessage : 'Inject the feed with additional content, such as suggested profiles.' } ,
2022-12-08 07:30:48 -08:00
tileServerLabel : { id : 'soapbox_config.tile_server_label' , defaultMessage : 'Map tile server' } ,
tileServerAttributionLabel : { id : 'soapbox_config.tile_server_attribution_label' , defaultMessage : 'Map tiles attribution' } ,
2023-01-11 17:11:54 -08:00
redirectRootNoLoginLabel : { id : 'soapbox_config.redirect_root_no_login_label' , defaultMessage : 'Redirect homepage' } ,
redirectRootNoLoginHint : { id : 'soapbox_config.redirect_root_no_login_hint' , defaultMessage : 'Path to redirect the homepage when a user is not logged in.' } ,
2022-05-05 12:25:57 -07:00
} ) ;
type ValueGetter < T = Element > = ( e : React.ChangeEvent < T > ) = > any ;
type Template = ImmutableMap < string , any > ;
type ConfigPath = Array < string | number > ;
2022-05-05 16:12:08 -07:00
type ThemeChangeHandler = ( theme : string ) = > void ;
2022-05-05 12:25:57 -07:00
const templates : Record < string , Template > = {
promoPanelItem : ImmutableMap ( { icon : '' , text : '' , url : '' } ) ,
footerItem : ImmutableMap ( { title : '' , url : '' } ) ,
cryptoAddress : ImmutableMap ( { ticker : '' , address : '' , note : '' } ) ,
} ;
const SoapboxConfig : React.FC = ( ) = > {
const intl = useIntl ( ) ;
const dispatch = useAppDispatch ( ) ;
2022-12-08 07:30:48 -08:00
const features = useFeatures ( ) ;
2022-05-05 12:25:57 -07:00
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 < string > ( 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 handleSubmit : React.FormEventHandler = ( e ) = > {
2022-12-17 11:05:50 -08:00
dispatch ( updateSoapboxConfig ( data . toJS ( ) ) ) . then ( ( ) = > {
2022-05-05 12:25:57 -07:00
setLoading ( false ) ;
2022-12-20 07:47:46 -08:00
toast . success ( intl . formatMessage ( messages . saved ) ) ;
2022-05-05 12:25:57 -07:00
} ) . catch ( ( ) = > {
setLoading ( false ) ;
} ) ;
setLoading ( true ) ;
e . preventDefault ( ) ;
} ;
const handleChange = ( path : ConfigPath , getValue : ValueGetter < any > ) : React . ChangeEventHandler = > {
return e = > {
setConfig ( path , getValue ( e ) ) ;
} ;
} ;
2022-05-05 16:12:08 -07:00
const handleThemeChange = ( path : ConfigPath ) : ThemeChangeHandler = > {
return theme = > {
setConfig ( path , theme ) ;
} ;
} ;
2022-05-05 12:25:57 -07:00
const handleFileChange = ( path : ConfigPath ) : React . ChangeEventHandler < HTMLInputElement > = > {
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 ) ;
}
} ;
} ;
2022-05-05 13:52:25 -07:00
const handleStreamItemChange = ( path : ConfigPath ) = > {
return ( values : any [ ] ) = > {
setConfig ( path , ImmutableList ( values ) ) ;
} ;
2022-05-05 12:25:57 -07:00
} ;
2022-05-05 13:52:25 -07:00
const addStreamItem = ( path : ConfigPath , template : Template ) = > {
return ( ) = > {
2022-09-13 12:17:54 -07:00
const items = data . getIn ( path ) || ImmutableList ( ) ;
2022-05-05 13:52:25 -07:00
setConfig ( path , items . push ( template ) ) ;
} ;
2022-05-05 13:06:52 -07:00
} ;
2022-05-05 13:52:25 -07:00
const deleteStreamItem = ( path : ConfigPath ) = > {
return ( i : number ) = > {
2022-05-05 14:18:43 -07:00
const newData = data . deleteIn ( [ . . . path , i ] ) ;
setData ( newData ) ;
2022-05-05 13:52:25 -07:00
} ;
2022-05-05 13:06:52 -07:00
} ;
2022-05-05 12:25:57 -07:00
const handleEditJSON : React.ChangeEventHandler < HTMLTextAreaElement > = 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 (
< Column label = { intl . formatMessage ( messages . heading ) } >
2022-05-05 15:37:25 -07:00
< Form onSubmit = { handleSubmit } >
2022-05-05 14:56:34 -07:00
< fieldset className = 'space-y-6' disabled = { isLoading } >
2022-05-05 12:25:57 -07:00
< SitePreview soapbox = { soapbox } / >
2022-05-05 14:56:34 -07:00
2022-05-05 16:36:27 -07:00
< FormGroup
labelText = { < FormattedMessage id = 'soapbox_config.fields.logo_label' defaultMessage = 'Logo' / > }
hintText = { < FormattedMessage id = 'soapbox_config.hints.logo' defaultMessage = 'SVG. At most 2 MB. Will be displayed to 50px height, maintaining aspect ratio' / > }
>
2022-05-16 16:33:44 -07:00
< FileInput
2022-05-05 16:36:27 -07:00
onChange = { handleFileChange ( [ 'logo' ] ) }
2022-11-25 13:01:49 -08:00
accept = 'image/svg+xml,image/png'
2022-05-05 16:36:27 -07:00
/ >
< / FormGroup >
2022-05-05 16:46:16 -07:00
< CardHeader >
< CardTitle title = { < FormattedMessage id = 'soapbox_config.headings.theme' defaultMessage = 'Theme' / > } / >
< / CardHeader >
2022-05-05 16:36:27 -07:00
< List >
< ListItem label = { < FormattedMessage id = 'soapbox_config.fields.theme_label' defaultMessage = 'Default theme' / > } >
< ThemeSelector
value = { soapbox . defaultSettings . get ( 'themeMode' ) }
onChange = { handleThemeChange ( [ 'defaultSettings' , 'themeMode' ] ) }
/ >
< / ListItem >
2022-12-17 11:10:59 -08:00
< ListItem
label = { < FormattedMessage id = 'soapbox_config.fields.edit_theme_label' defaultMessage = 'Edit theme' / > }
2023-08-25 14:19:56 -07:00
to = '/soapbox/admin/theme'
2022-12-17 11:10:59 -08:00
/ >
2022-05-05 16:36:27 -07:00
< / List >
2022-05-05 14:35:30 -07:00
2022-05-05 16:46:16 -07:00
< CardHeader >
< CardTitle title = { < FormattedMessage id = 'soapbox_config.headings.options' defaultMessage = 'Options' / > } / >
< / CardHeader >
2022-05-05 14:35:30 -07:00
2022-05-05 16:30:25 -07:00
< List >
< ListItem label = { intl . formatMessage ( messages . verifiedCanEditNameLabel ) } >
< Toggle
checked = { soapbox . verifiedCanEditName === true }
onChange = { handleChange ( [ 'verifiedCanEditName' ] , ( e ) = > e . target . checked ) }
/ >
< / ListItem >
< ListItem label = { intl . formatMessage ( messages . displayFqnLabel ) } >
< Toggle
checked = { soapbox . displayFqn === true }
onChange = { handleChange ( [ 'displayFqn' ] , ( e ) = > e . target . checked ) }
/ >
< / ListItem >
< ListItem label = { intl . formatMessage ( messages . greentextLabel ) } >
< Toggle
checked = { soapbox . greentext === true }
onChange = { handleChange ( [ 'greentext' ] , ( e ) = > e . target . checked ) }
/ >
< / ListItem >
2022-11-27 19:05:39 -08:00
< ListItem
label = { intl . formatMessage ( messages . feedInjectionLabel ) }
hint = { intl . formatMessage ( messages . feedInjectionHint ) }
>
< Toggle
checked = { soapbox . feedInjection === true }
onChange = { handleChange ( [ 'feedInjection' ] , ( e ) = > e . target . checked ) }
/ >
< / ListItem >
2023-01-02 09:52:05 -08:00
< ListItem
label = { intl . formatMessage ( messages . mediaPreviewLabel ) }
hint = { intl . formatMessage ( messages . mediaPreviewHint ) }
>
< Toggle
checked = { soapbox . mediaPreview === true }
onChange = { handleChange ( [ 'mediaPreview' ] , ( e ) = > e . target . checked ) }
/ >
< / ListItem >
2022-10-09 15:41:47 -07:00
< ListItem label = { intl . formatMessage ( messages . displayCtaLabel ) } >
< Toggle
checked = { soapbox . displayCta === true }
onChange = { handleChange ( [ 'displayCta' ] , ( e ) = > e . target . checked ) }
/ >
< / ListItem >
2022-05-05 16:30:25 -07:00
< ListItem
2022-05-05 12:25:57 -07:00
label = { intl . formatMessage ( messages . authenticatedProfileLabel ) }
hint = { intl . formatMessage ( messages . authenticatedProfileHint ) }
2022-05-05 16:30:25 -07:00
>
< Toggle
checked = { soapbox . authenticatedProfile === true }
onChange = { handleChange ( [ 'authenticatedProfile' ] , ( e ) = > e . target . checked ) }
/ >
< / ListItem >
2023-01-05 11:52:37 -08:00
< ListItem
label = { intl . formatMessage ( messages . redirectRootNoLoginLabel ) }
hint = { intl . formatMessage ( messages . redirectRootNoLoginHint ) }
>
< Input
type = 'text'
2023-01-15 11:23:28 -08:00
placeholder = '/timeline/local'
2023-01-15 11:22:03 -08:00
value = { String ( data . get ( 'redirectRootNoLogin' , '' ) ) }
2023-01-05 11:52:37 -08:00
onChange = { handleChange ( [ 'redirectRootNoLogin' ] , ( e ) = > e . target . value ) }
/ >
< / ListItem >
2022-05-05 16:30:25 -07:00
< / List >
2022-05-05 13:52:25 -07:00
2022-05-05 16:46:16 -07:00
< CardHeader >
< CardTitle title = { < FormattedMessage id = 'soapbox_config.headings.navigation' defaultMessage = 'Navigation' / > } / >
< / CardHeader >
2022-05-05 13:52:25 -07:00
< Streamfield
label = { < FormattedMessage id = 'soapbox_config.fields.promo_panel_fields_label' defaultMessage = 'Promo panel items' / > }
hint = { < FormattedMessage id = 'soapbox_config.hints.promo_panel_fields' defaultMessage = 'You can have custom defined links displayed on the right panel of the timelines page.' / > }
component = { PromoPanelInput }
values = { soapbox . promoPanel . items . toArray ( ) }
onChange = { handleStreamItemChange ( [ 'promoPanel' , 'items' ] ) }
onAddItem = { addStreamItem ( [ 'promoPanel' , 'items' ] , templates . promoPanel ) }
onRemoveItem = { deleteStreamItem ( [ 'promoPanel' , 'items' ] ) }
/ >
2022-05-05 14:18:43 -07:00
< Streamfield
label = { < FormattedMessage id = 'soapbox_config.fields.home_footer_fields_label' defaultMessage = 'Home footer items' / > }
hint = { < FormattedMessage id = 'soapbox_config.hints.home_footer_fields' defaultMessage = 'You can have custom defined links displayed on the footer of your static pages' / > }
component = { FooterLinkInput }
values = { soapbox . navlinks . get ( 'homeFooter' ) ? . toArray ( ) || [ ] }
onChange = { handleStreamItemChange ( [ 'navlinks' , 'homeFooter' ] ) }
onAddItem = { addStreamItem ( [ 'navlinks' , 'homeFooter' ] , templates . footerItem ) }
onRemoveItem = { deleteStreamItem ( [ 'navlinks' , 'homeFooter' ] ) }
/ >
2022-05-05 13:06:52 -07:00
2022-05-05 16:46:16 -07:00
< FormGroup labelText = { intl . formatMessage ( messages . copyrightFooterLabel ) } >
< Input
type = 'text'
placeholder = { intl . formatMessage ( messages . copyrightFooterLabel ) }
value = { soapbox . copyright }
onChange = { handleChange ( [ 'copyright' ] , ( e ) = > e . target . value ) }
/ >
< / FormGroup >
2022-12-08 07:30:48 -08:00
{ features . events && (
< >
< CardHeader >
< CardTitle title = { < FormattedMessage id = 'soapbox_config.headings.events' defaultMessage = 'Events' / > } / >
< / CardHeader >
< FormGroup labelText = { intl . formatMessage ( messages . tileServerLabel ) } >
< Input
type = 'text'
placeholder = { intl . formatMessage ( messages . tileServerLabel ) }
value = { soapbox . tileServer }
onChange = { handleChange ( [ 'tileServer' ] , ( e ) = > e . target . value ) }
/ >
< / FormGroup >
< FormGroup labelText = { intl . formatMessage ( messages . tileServerAttributionLabel ) } >
< Input
type = 'text'
placeholder = { intl . formatMessage ( messages . tileServerAttributionLabel ) }
value = { soapbox . tileServerAttribution }
onChange = { handleChange ( [ 'tileServerAttribution' ] , ( e ) = > e . target . value ) }
/ >
< / FormGroup >
< / >
) }
2022-05-05 16:46:16 -07:00
< CardHeader >
< CardTitle title = { < FormattedMessage id = 'soapbox_config.headings.cryptocurrency' defaultMessage = 'Cryptocurrency' / > } / >
< / CardHeader >
2022-05-05 13:06:52 -07:00
< Streamfield
label = { < FormattedMessage id = 'soapbox_config.fields.crypto_addresses_label' defaultMessage = 'Cryptocurrency addresses' / > }
hint = { < FormattedMessage id = 'soapbox_config.hints.crypto_addresses' defaultMessage = 'Add cryptocurrency addresses so users of your site can donate to you. Order matters, and you must use lowercase ticker values.' / > }
component = { CryptoAddressInput }
values = { soapbox . cryptoAddresses . toArray ( ) }
2022-05-05 13:52:25 -07:00
onChange = { handleStreamItemChange ( [ 'cryptoAddresses' ] ) }
onAddItem = { addStreamItem ( [ 'cryptoAddresses' ] , templates . cryptoAddress ) }
onRemoveItem = { deleteStreamItem ( [ 'cryptoAddresses' ] ) }
2022-05-05 13:06:52 -07:00
/ >
2022-05-05 14:35:30 -07:00
< FormGroup labelText = { intl . formatMessage ( messages . cryptoDonatePanelLimitLabel ) } >
< Input
2022-05-05 12:25:57 -07:00
type = 'number'
min = { 0 }
pattern = '[0-9]+'
placeholder = { intl . formatMessage ( messages . cryptoDonatePanelLimitLabel ) }
value = { soapbox . cryptoDonatePanel . get ( 'limit' ) }
onChange = { handleChange ( [ 'cryptoDonatePanel' , 'limit' ] , ( e ) = > Number ( e . target . value ) ) }
/ >
2022-05-05 14:35:30 -07:00
< / FormGroup >
2022-05-05 16:46:16 -07:00
< CardHeader >
< CardTitle title = { < FormattedMessage id = 'soapbox_config.headings.advanced' defaultMessage = 'Advanced' / > } / >
< / CardHeader >
2022-05-05 12:25:57 -07:00
< Accordion
headline = { intl . formatMessage ( messages . rawJSONLabel ) }
expanded = { jsonEditorExpanded }
onToggle = { toggleJSONEditor }
>
2023-01-10 10:41:32 -08:00
< FormGroup
hintText = { intl . formatMessage ( messages . rawJSONHint ) }
errors = { jsonValid ? undefined : [ intl . formatMessage ( messages . rawJSONInvalid ) ] }
>
2022-05-05 15:45:32 -07:00
< Textarea
2022-05-05 12:25:57 -07:00
value = { rawJSON }
onChange = { handleEditJSON }
2022-05-05 15:45:32 -07:00
isCodeEditor
2022-05-05 12:25:57 -07:00
rows = { 12 }
/ >
2022-05-05 15:45:32 -07:00
< / FormGroup >
2022-05-05 12:25:57 -07:00
< / Accordion >
< / fieldset >
2022-05-05 15:51:07 -07:00
2022-05-05 13:52:25 -07:00
< FormActions >
< Button type = 'submit' >
2022-05-05 12:25:57 -07:00
< FormattedMessage id = 'soapbox_config.save' defaultMessage = 'Save' / >
2022-05-05 13:52:25 -07:00
< / Button >
< / FormActions >
< / Form >
2022-05-05 12:25:57 -07:00
< / Column >
) ;
} ;
export default SoapboxConfig ;