// @flow import * as React from 'react'; import Card from 'component/common/card'; import TagsSearch from 'component/tagsSearch'; import Page from 'component/page'; import ChannelSelector from 'component/channelSelector'; import SearchChannelField from 'component/searchChannelField'; import SettingsRow from 'component/settingsRow'; import Spinner from 'component/spinner'; import { FormField } from 'component/common/form-components/form-field'; import LbcSymbol from 'component/common/lbc-symbol'; import I18nMessage from 'component/i18nMessage'; import { parseURI } from 'util/lbryURI'; import debounce from 'util/debounce'; const DEBOUNCE_REFRESH_MS = 1000; const LBC_MAX = 21000000; const LBC_MIN = 0; const LBC_STEP = 1.0; // **************************************************************************** // **************************************************************************** type Props = { activeChannelClaim: ChannelClaim, settingsByChannelId: { [string]: PerChannelSettings }, fetchingCreatorSettings: boolean, fetchingBlockedWords: boolean, moderationDelegatesById: { [string]: Array<{ channelId: string, channelName: string }> }, commentBlockWords: (ChannelClaim, Array) => void, commentUnblockWords: (ChannelClaim, Array) => void, commentModAddDelegate: (string, string, ChannelClaim) => void, commentModRemoveDelegate: (string, string, ChannelClaim) => void, commentModListDelegates: (ChannelClaim) => void, fetchCreatorSettings: (channelId: string) => void, updateCreatorSettings: (ChannelClaim, PerChannelSettings) => void, doToast: ({ message: string }) => void, }; export default function SettingsCreatorPage(props: Props) { const { activeChannelClaim, settingsByChannelId, moderationDelegatesById, commentBlockWords, commentUnblockWords, commentModAddDelegate, commentModRemoveDelegate, commentModListDelegates, fetchCreatorSettings, updateCreatorSettings, } = props; const [commentsEnabled, setCommentsEnabled] = React.useState(true); const [mutedWordTags, setMutedWordTags] = React.useState([]); const [moderatorUris, setModeratorUris] = React.useState([]); const [minTip, setMinTip] = React.useState(0); const [minSuper, setMinSuper] = React.useState(0); const [slowModeMin, setSlowModeMin] = React.useState(0); const [lastUpdated, setLastUpdated] = React.useState(1); const pushSlowModeMinDebounced = React.useMemo(() => debounce(pushSlowModeMin, 1000), []); // eslint-disable-line react-hooks/exhaustive-deps const pushMinTipDebounced = React.useMemo(() => debounce(pushMinTip, 1000), []); // eslint-disable-line react-hooks/exhaustive-deps const pushMinSuperDebounced = React.useMemo(() => debounce(pushMinSuper, 1000), []); // eslint-disable-line react-hooks/exhaustive-deps // ************************************************************************** // ************************************************************************** /** * Updates corresponding GUI states with the given PerChannelSettings values. * * @param settings * @param fullSync If true, update all states and consider 'undefined' * settings as "cleared/false"; if false, only update defined settings. */ function settingsToStates(settings: PerChannelSettings, fullSync: boolean) { const doSetMutedWordTags = (words: Array) => { const tagArray = Array.from(new Set(words)); setMutedWordTags( tagArray .filter((t) => t !== '') .map((x) => { return { name: x }; }) ); }; if (fullSync) { setCommentsEnabled(settings.comments_enabled || false); setMinTip(settings.min_tip_amount_comment || 0); setMinSuper(settings.min_tip_amount_super_chat || 0); setSlowModeMin(settings.slow_mode_min_gap || 0); doSetMutedWordTags(settings.words || []); } else { if (settings.comments_enabled !== undefined) { setCommentsEnabled(settings.comments_enabled); } if (settings.min_tip_amount_comment !== undefined) { setMinTip(settings.min_tip_amount_comment); } if (settings.min_tip_amount_super_chat !== undefined) { setMinSuper(settings.min_tip_amount_super_chat); } if (settings.slow_mode_min_gap !== undefined) { setSlowModeMin(settings.slow_mode_min_gap); } if (settings.words) { doSetMutedWordTags(settings.words); } } } function setSettings(newSettings: PerChannelSettings) { settingsToStates(newSettings, false); updateCreatorSettings(activeChannelClaim, newSettings); setLastUpdated(Date.now()); } function pushSlowModeMin(value: number, activeChannelClaim: ChannelClaim) { updateCreatorSettings(activeChannelClaim, { slow_mode_min_gap: value }); } function pushMinTip(value: number, activeChannelClaim: ChannelClaim) { updateCreatorSettings(activeChannelClaim, { min_tip_amount_comment: value }); } function pushMinSuper(value: number, activeChannelClaim: ChannelClaim) { updateCreatorSettings(activeChannelClaim, { min_tip_amount_super_chat: value }); } function parseModUri(uri) { try { return parseURI(uri); } catch (e) {} return undefined; } function handleModeratorAdded(channelUri: string) { setModeratorUris([...moderatorUris, channelUri]); const parsedUri = parseModUri(channelUri); if (parsedUri && parsedUri.claimId && parsedUri.claimName) { commentModAddDelegate(parsedUri.claimId, parsedUri.claimName, activeChannelClaim); setLastUpdated(Date.now()); } } function handleModeratorRemoved(channelUri: string) { setModeratorUris(moderatorUris.filter((x) => x !== channelUri)); const parsedUri = parseModUri(channelUri); if (parsedUri && parsedUri.claimId && parsedUri.claimName) { commentModRemoveDelegate(parsedUri.claimId, parsedUri.claimName, activeChannelClaim); setLastUpdated(Date.now()); } } function addMutedWords(newTags: Array) { const validatedNewTags = []; newTags.forEach((newTag) => { if (!mutedWordTags.some((tag) => tag.name === newTag.name)) { validatedNewTags.push(newTag); } }); if (validatedNewTags.length !== 0) { setMutedWordTags([...mutedWordTags, ...validatedNewTags]); commentBlockWords( activeChannelClaim, validatedNewTags.map((x) => x.name) ); setLastUpdated(Date.now()); } } function removeMutedWord(tagToRemove: Tag) { const newMutedWordTags = mutedWordTags.slice().filter((t) => t.name !== tagToRemove.name); setMutedWordTags(newMutedWordTags); commentUnblockWords(activeChannelClaim, ['', tagToRemove.name]); setLastUpdated(Date.now()); } // ************************************************************************** // ************************************************************************** // Update local moderator states with data from API. React.useEffect(() => { commentModListDelegates(activeChannelClaim); }, [activeChannelClaim, commentModListDelegates]); React.useEffect(() => { if (activeChannelClaim) { const delegates = moderationDelegatesById[activeChannelClaim.claim_id]; if (delegates) { setModeratorUris(delegates.map((d) => `${d.channelName}#${d.channelId}`)); } else { setModeratorUris([]); } } }, [activeChannelClaim, moderationDelegatesById]); // Update local states with data from API. React.useEffect(() => { if (lastUpdated !== 0 && Date.now() - lastUpdated < DEBOUNCE_REFRESH_MS) { // Still debouncing. Skip update. return; } if (activeChannelClaim && settingsByChannelId && settingsByChannelId[activeChannelClaim.claim_id]) { const channelSettings = settingsByChannelId[activeChannelClaim.claim_id]; settingsToStates(channelSettings, true); } }, [activeChannelClaim, settingsByChannelId, lastUpdated]); // Re-sync list on first idle time; mainly to correct any invalid settings. React.useEffect(() => { if (lastUpdated && activeChannelClaim) { const timer = setTimeout(() => { fetchCreatorSettings(activeChannelClaim.claim_id); }, DEBOUNCE_REFRESH_MS); return () => clearTimeout(timer); } }, [lastUpdated, activeChannelClaim, fetchCreatorSettings]); // ************************************************************************** // ************************************************************************** const isBusy = !activeChannelClaim || !settingsByChannelId || settingsByChannelId[activeChannelClaim.claim_id] === undefined; const isDisabled = activeChannelClaim && settingsByChannelId && settingsByChannelId[activeChannelClaim.claim_id] === null; return (
{isBusy && (
)} {isDisabled && ( )} {!isBusy && !isDisabled && ( <> setSettings({ comments_enabled: !commentsEnabled })} /> { const value = parseInt(e.target.value); setSlowModeMin(value); pushSlowModeMinDebounced(value, activeChannelClaim); }} onBlur={() => setLastUpdated(Date.now())} /> }}>Minimum %lbc% tip amount for comments } subtitle={__(HELP.MIN_TIP)} > { const newMinTip = parseFloat(e.target.value); setMinTip(newMinTip); pushMinTipDebounced(newMinTip, activeChannelClaim); if (newMinTip !== 0 && minSuper !== 0) { setMinSuper(0); pushMinSuperDebounced(0, activeChannelClaim); } }} onBlur={() => setLastUpdated(Date.now())} /> }}>Minimum %lbc% tip amount for hyperchats } subtitle={ <> {__(HELP.MIN_SUPER)} {minTip !== 0 && (

{__(HELP.MIN_SUPER_OFF)}

)} } > { const newMinSuper = parseFloat(e.target.value); setMinSuper(newMinSuper); pushMinSuperDebounced(newMinSuper, activeChannelClaim); }} onBlur={() => setLastUpdated(Date.now())} />
} /> )}
); } // prettier-ignore const HELP = { SLOW_MODE: 'Minimum time gap in seconds between comments.', MIN_TIP: 'Enabling a minimum amount to comment will force all comments to have tips associated with them. This can help prevent spam.', MIN_SUPER: 'Enabling a minimum amount to hyperchat will force all TIPPED comments to have this value in order to be shown. This still allows regular comments to be posted.', MIN_SUPER_OFF: '(This settings is not applicable if all comments require a tip.)', BLOCKED_WORDS: 'Comments containing these words will be blocked.', MODERATORS: 'Moderators can block channels on your behalf. Blocked channels will appear in your "Blocked and Muted" list.', MODERATOR_SEARCH: 'Enter a channel name or URL to add as a moderator.\nExamples:\n - @channel\n - @channel#3\n - https://odysee.com/@Odysee:8\n - lbry://@Odysee#8', };