Merge branch 'remote-timeline-styles' into 'develop'
Remote timeline styles Closes #1182 See merge request soapbox-pub/soapbox!1952
This commit is contained in:
commit
2d97ae399a
3 changed files with 115 additions and 115 deletions
|
@ -4,7 +4,7 @@ import React from 'react';
|
||||||
import { FormattedMessage } from 'react-intl';
|
import { FormattedMessage } from 'react-intl';
|
||||||
|
|
||||||
import Icon from 'soapbox/components/icon';
|
import Icon from 'soapbox/components/icon';
|
||||||
import { Text } from 'soapbox/components/ui';
|
import { HStack, Stack, Text } from 'soapbox/components/ui';
|
||||||
import { useAppSelector } from 'soapbox/hooks';
|
import { useAppSelector } from 'soapbox/hooks';
|
||||||
|
|
||||||
import type { Map as ImmutableMap } from 'immutable';
|
import type { Map as ImmutableMap } from 'immutable';
|
||||||
|
@ -16,6 +16,23 @@ const hasRestrictions = (remoteInstance: ImmutableMap<string, any>): boolean =>
|
||||||
.reduce((acc: boolean, value: boolean) => acc || value, false);
|
.reduce((acc: boolean, value: boolean) => acc || value, false);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
interface IRestriction {
|
||||||
|
icon: string,
|
||||||
|
children: React.ReactNode,
|
||||||
|
}
|
||||||
|
|
||||||
|
const Restriction: React.FC<IRestriction> = ({ icon, children }) => {
|
||||||
|
return (
|
||||||
|
<HStack space={3}>
|
||||||
|
<Icon className='flex-none w-5 h-5' src={icon} />
|
||||||
|
|
||||||
|
<Text theme='muted'>
|
||||||
|
{children}
|
||||||
|
</Text>
|
||||||
|
</HStack>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
interface IInstanceRestrictions {
|
interface IInstanceRestrictions {
|
||||||
remoteInstance: ImmutableMap<string, any>,
|
remoteInstance: ImmutableMap<string, any>,
|
||||||
}
|
}
|
||||||
|
@ -40,57 +57,52 @@ const InstanceRestrictions: React.FC<IInstanceRestrictions> = ({ remoteInstance
|
||||||
|
|
||||||
if (followers_only) {
|
if (followers_only) {
|
||||||
items.push((
|
items.push((
|
||||||
<Text key='followers_only' className='flex items-center gap-2' theme='muted'>
|
<Restriction key='followersOnly' icon={require('@tabler/icons/lock.svg')}>
|
||||||
<Icon src={require('@tabler/icons/lock.svg')} />
|
|
||||||
<FormattedMessage
|
<FormattedMessage
|
||||||
id='federation_restriction.followers_only'
|
id='federation_restriction.followers_only'
|
||||||
defaultMessage='Hidden except to followers'
|
defaultMessage='Hidden except to followers'
|
||||||
/>
|
/>
|
||||||
</Text>
|
</Restriction>
|
||||||
));
|
));
|
||||||
} else if (federated_timeline_removal) {
|
} else if (federated_timeline_removal) {
|
||||||
items.push((
|
items.push((
|
||||||
<Text key='federated_timeline_removal' className='flex items-center gap-2' theme='muted'>
|
<Restriction key='federatedTimelineRemoval' icon={require('@tabler/icons/lock-open.svg')}>
|
||||||
<Icon src={require('@tabler/icons/lock-open.svg')} />
|
|
||||||
<FormattedMessage
|
<FormattedMessage
|
||||||
id='federation_restriction.federated_timeline_removal'
|
id='federation_restriction.federated_timeline_removal'
|
||||||
defaultMessage='Fediverse timeline removal'
|
defaultMessage='Fediverse timeline removal'
|
||||||
/>
|
/>
|
||||||
</Text>
|
</Restriction>
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (fullMediaRemoval) {
|
if (fullMediaRemoval) {
|
||||||
items.push((
|
items.push((
|
||||||
<Text key='full_media_removal' className='flex items-center gap-2' theme='muted'>
|
<Restriction key='fullMediaRemoval' icon={require('@tabler/icons/photo-off.svg')}>
|
||||||
<Icon src={require('@tabler/icons/photo-off.svg')} />
|
|
||||||
<FormattedMessage
|
<FormattedMessage
|
||||||
id='federation_restriction.full_media_removal'
|
id='federation_restriction.full_media_removal'
|
||||||
defaultMessage='Full media removal'
|
defaultMessage='Full media removal'
|
||||||
/>
|
/>
|
||||||
</Text>
|
</Restriction>
|
||||||
));
|
));
|
||||||
} else if (partialMediaRemoval) {
|
} else if (partialMediaRemoval) {
|
||||||
items.push((
|
items.push((
|
||||||
<Text key='partial_media_removal' className='flex items-center gap-2' theme='muted'>
|
<Restriction key='partialMediaRemoval' icon={require('@tabler/icons/photo-off.svg')}>
|
||||||
<Icon src={require('@tabler/icons/photo-off.svg')} />
|
|
||||||
<FormattedMessage
|
<FormattedMessage
|
||||||
id='federation_restriction.partial_media_removal'
|
id='federation_restriction.partial_media_removal'
|
||||||
defaultMessage='Partial media removal'
|
defaultMessage='Partial media removal'
|
||||||
/>
|
/>
|
||||||
</Text>
|
</Restriction>
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!fullMediaRemoval && media_nsfw) {
|
if (!fullMediaRemoval && media_nsfw) {
|
||||||
items.push((
|
items.push((
|
||||||
<Text key='media_nsfw' className='flex items-center gap-2' theme='muted'>
|
<Restriction key='mediaNsfw' icon={require('@tabler/icons/eye-off.svg')}>
|
||||||
<Icon src={require('@tabler/icons/eye-off.svg')} />
|
|
||||||
<FormattedMessage
|
<FormattedMessage
|
||||||
id='federation_restriction.media_nsfw'
|
id='federation_restriction.media_nsfw'
|
||||||
defaultMessage='Attachments marked NSFW'
|
defaultMessage='Attachments marked NSFW'
|
||||||
/>
|
/>
|
||||||
</Text>
|
</Restriction>
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -105,46 +117,45 @@ const InstanceRestrictions: React.FC<IInstanceRestrictions> = ({ remoteInstance
|
||||||
|
|
||||||
if (remoteInstance.getIn(['federation', 'reject']) === true) {
|
if (remoteInstance.getIn(['federation', 'reject']) === true) {
|
||||||
return (
|
return (
|
||||||
<Text className='flex items-center gap-2' theme='muted'>
|
<Restriction icon={require('@tabler/icons/shield-x.svg')}>
|
||||||
<Icon src={require('@tabler/icons/x.svg')} />
|
|
||||||
<FormattedMessage
|
<FormattedMessage
|
||||||
id='remote_instance.federation_panel.restricted_message'
|
id='remote_instance.federation_panel.restricted_message'
|
||||||
defaultMessage='{siteTitle} blocks all activities from {host}.'
|
defaultMessage='{siteTitle} blocks all activities from {host}.'
|
||||||
values={{ host, siteTitle }}
|
values={{ host, siteTitle }}
|
||||||
/>
|
/>
|
||||||
</Text>
|
</Restriction>
|
||||||
);
|
);
|
||||||
} else if (hasRestrictions(remoteInstance)) {
|
} else if (hasRestrictions(remoteInstance)) {
|
||||||
return [
|
return (
|
||||||
(
|
<>
|
||||||
<Text theme='muted'>
|
<Restriction icon={require('@tabler/icons/shield-lock.svg')}>
|
||||||
<FormattedMessage
|
<FormattedMessage
|
||||||
id='remote_instance.federation_panel.some_restrictions_message'
|
id='remote_instance.federation_panel.some_restrictions_message'
|
||||||
defaultMessage='{siteTitle} has placed some restrictions on {host}.'
|
defaultMessage='{siteTitle} has placed some restrictions on {host}.'
|
||||||
values={{ host, siteTitle }}
|
values={{ host, siteTitle }}
|
||||||
/>
|
/>
|
||||||
</Text>
|
</Restriction>
|
||||||
),
|
|
||||||
renderRestrictions(),
|
{renderRestrictions()}
|
||||||
];
|
</>
|
||||||
|
);
|
||||||
} else {
|
} else {
|
||||||
return (
|
return (
|
||||||
<Text className='flex items-center gap-2' theme='muted'>
|
<Restriction icon={require('@tabler/icons/shield-check.svg')}>
|
||||||
<Icon src={require('@tabler/icons/check.svg')} />
|
|
||||||
<FormattedMessage
|
<FormattedMessage
|
||||||
id='remote_instance.federation_panel.no_restrictions_message'
|
id='remote_instance.federation_panel.no_restrictions_message'
|
||||||
defaultMessage='{siteTitle} has placed no restrictions on {host}.'
|
defaultMessage='{siteTitle} has placed no restrictions on {host}.'
|
||||||
values={{ host, siteTitle }}
|
values={{ host, siteTitle }}
|
||||||
/>
|
/>
|
||||||
</Text>
|
</Restriction>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className='py-1 pl-4 mb-4 border-solid border-l-[3px] border-gray-300 dark:border-gray-500'>
|
<Stack space={3}>
|
||||||
{renderContent()}
|
{renderContent()}
|
||||||
</div>
|
</Stack>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -5,8 +5,8 @@ import { useHistory } from 'react-router-dom';
|
||||||
import { connectRemoteStream } from 'soapbox/actions/streaming';
|
import { connectRemoteStream } from 'soapbox/actions/streaming';
|
||||||
import { expandRemoteTimeline } from 'soapbox/actions/timelines';
|
import { expandRemoteTimeline } from 'soapbox/actions/timelines';
|
||||||
import IconButton from 'soapbox/components/icon-button';
|
import IconButton from 'soapbox/components/icon-button';
|
||||||
import { HStack, Text } from 'soapbox/components/ui';
|
import SubNavigation from 'soapbox/components/sub-navigation';
|
||||||
import Column from 'soapbox/features/ui/components/column';
|
import { Column, HStack, Text } from 'soapbox/components/ui';
|
||||||
import { useAppDispatch, useSettings } from 'soapbox/hooks';
|
import { useAppDispatch, useSettings } from 'soapbox/hooks';
|
||||||
|
|
||||||
import Timeline from '../ui/components/timeline';
|
import Timeline from '../ui/components/timeline';
|
||||||
|
@ -14,7 +14,7 @@ import Timeline from '../ui/components/timeline';
|
||||||
import PinnedHostsPicker from './components/pinned-hosts-picker';
|
import PinnedHostsPicker from './components/pinned-hosts-picker';
|
||||||
|
|
||||||
const messages = defineMessages({
|
const messages = defineMessages({
|
||||||
title: { id: 'column.remote', defaultMessage: 'Federated timeline' },
|
heading: { id: 'column.remote', defaultMessage: 'Federated timeline' },
|
||||||
});
|
});
|
||||||
|
|
||||||
interface IRemoteTimeline {
|
interface IRemoteTimeline {
|
||||||
|
@ -65,18 +65,26 @@ const RemoteTimeline: React.FC<IRemoteTimeline> = ({ params }) => {
|
||||||
}, [onlyMedia]);
|
}, [onlyMedia]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Column label={intl.formatMessage(messages.title)} heading={instance} transparent withHeader={false}>
|
<Column label={intl.formatMessage(messages.heading)} transparent withHeader={false}>
|
||||||
{instance && <PinnedHostsPicker host={instance} />}
|
<div className='px-4 pt-4 sm:p-0'>
|
||||||
{!pinned && <HStack className='mb-4 px-2' space={2}>
|
<SubNavigation message={instance} />
|
||||||
<IconButton iconClassName='h-5 w-5' src={require('@tabler/icons/x.svg')} onClick={handleCloseClick} />
|
|
||||||
<Text>
|
{instance && <PinnedHostsPicker host={instance} />}
|
||||||
<FormattedMessage
|
|
||||||
id='remote_timeline.filter_message'
|
{!pinned && (
|
||||||
defaultMessage='You are viewing the timeline of {instance}.'
|
<HStack className='mb-4 px-2' space={2}>
|
||||||
values={{ instance }}
|
<IconButton iconClassName='h-5 w-5' src={require('@tabler/icons/x.svg')} onClick={handleCloseClick} />
|
||||||
/>
|
<Text>
|
||||||
</Text>
|
<FormattedMessage
|
||||||
</HStack>}
|
id='remote_timeline.filter_message'
|
||||||
|
defaultMessage='You are viewing the timeline of {instance}.'
|
||||||
|
values={{ instance }}
|
||||||
|
/>
|
||||||
|
</Text>
|
||||||
|
</HStack>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
<Timeline
|
<Timeline
|
||||||
scrollKey={`${timelineId}_${instance}_timeline`}
|
scrollKey={`${timelineId}_${instance}_timeline`}
|
||||||
timelineId={`${timelineId}${onlyMedia ? ':media' : ''}:${instance}`}
|
timelineId={`${timelineId}${onlyMedia ? ':media' : ''}:${instance}`}
|
||||||
|
|
|
@ -1,17 +1,15 @@
|
||||||
import { Map as ImmutableMap } from 'immutable';
|
import { Map as ImmutableMap } from 'immutable';
|
||||||
import React, { useState, useEffect } from 'react';
|
import React, { useState, useEffect, useCallback } from 'react';
|
||||||
import { defineMessages, FormattedMessage, useIntl } from 'react-intl';
|
import { defineMessages, FormattedMessage, useIntl } from 'react-intl';
|
||||||
import Toggle from 'react-toggle';
|
import Toggle from 'react-toggle';
|
||||||
|
|
||||||
import { updateMrf } from 'soapbox/actions/mrf';
|
import { updateMrf } from 'soapbox/actions/mrf';
|
||||||
import snackbar from 'soapbox/actions/snackbar';
|
import snackbar from 'soapbox/actions/snackbar';
|
||||||
import { HStack, Modal, Stack, Text } from 'soapbox/components/ui';
|
import List, { ListItem } from 'soapbox/components/list';
|
||||||
import { SimpleForm } from 'soapbox/features/forms';
|
import { Modal } from 'soapbox/components/ui';
|
||||||
import { useAppSelector, useAppDispatch } from 'soapbox/hooks';
|
import { useAppSelector, useAppDispatch } from 'soapbox/hooks';
|
||||||
import { makeGetRemoteInstance } from 'soapbox/selectors';
|
import { makeGetRemoteInstance } from 'soapbox/selectors';
|
||||||
|
|
||||||
const getRemoteInstance = makeGetRemoteInstance();
|
|
||||||
|
|
||||||
const messages = defineMessages({
|
const messages = defineMessages({
|
||||||
mediaRemoval: { id: 'edit_federation.media_removal', defaultMessage: 'Strip media' },
|
mediaRemoval: { id: 'edit_federation.media_removal', defaultMessage: 'Strip media' },
|
||||||
forceNsfw: { id: 'edit_federation.force_nsfw', defaultMessage: 'Force attachments to be marked sensitive' },
|
forceNsfw: { id: 'edit_federation.force_nsfw', defaultMessage: 'Force attachments to be marked sensitive' },
|
||||||
|
@ -31,6 +29,7 @@ const EditFederationModal: React.FC<IEditFederationModal> = ({ host, onClose })
|
||||||
const intl = useIntl();
|
const intl = useIntl();
|
||||||
const dispatch = useAppDispatch();
|
const dispatch = useAppDispatch();
|
||||||
|
|
||||||
|
const getRemoteInstance = useCallback(makeGetRemoteInstance(), []);
|
||||||
const remoteInstance = useAppSelector(state => getRemoteInstance(state, host));
|
const remoteInstance = useAppSelector(state => getRemoteInstance(state, host));
|
||||||
|
|
||||||
const [data, setData] = useState(ImmutableMap<string, any>());
|
const [data, setData] = useState(ImmutableMap<string, any>());
|
||||||
|
@ -82,74 +81,56 @@ const EditFederationModal: React.FC<IEditFederationModal> = ({ host, onClose })
|
||||||
confirmationAction={handleSubmit}
|
confirmationAction={handleSubmit}
|
||||||
confirmationText={intl.formatMessage(messages.save)}
|
confirmationText={intl.formatMessage(messages.save)}
|
||||||
>
|
>
|
||||||
<SimpleForm onSubmit={handleSubmit}>
|
<List>
|
||||||
<Stack space={2}>
|
<ListItem label={<FormattedMessage id='edit_federation.reject' defaultMessage='Reject all activities' />}>
|
||||||
<HStack space={2} alignItems='center'>
|
<Toggle
|
||||||
<Toggle
|
checked={reject}
|
||||||
checked={reject}
|
onChange={handleDataChange('reject')}
|
||||||
onChange={handleDataChange('reject')}
|
icons={false}
|
||||||
icons={false}
|
id='reject'
|
||||||
id='reject'
|
/>
|
||||||
/>
|
</ListItem>
|
||||||
|
|
||||||
<Text theme='muted' tag='label' size='sm' htmlFor='reject'>
|
<ListItem label={<FormattedMessage id='edit_federation.media_removal' defaultMessage='Strip media' />}>
|
||||||
<FormattedMessage id='edit_federation.reject' defaultMessage='Reject all activities' />
|
<Toggle
|
||||||
</Text>
|
checked={fullMediaRemoval}
|
||||||
</HStack>
|
onChange={handleMediaRemoval}
|
||||||
<HStack space={2} alignItems='center'>
|
icons={false}
|
||||||
<Toggle
|
id='media_removal'
|
||||||
checked={fullMediaRemoval}
|
disabled={reject}
|
||||||
onChange={handleMediaRemoval}
|
/>
|
||||||
icons={false}
|
</ListItem>
|
||||||
id='media_removal'
|
|
||||||
disabled={reject}
|
|
||||||
/>
|
|
||||||
|
|
||||||
<Text theme='muted' tag='label' size='sm' htmlFor='media_removal'>
|
<ListItem label={<FormattedMessage id='edit_federation.force_nsfw' defaultMessage='Force attachments to be marked sensitive' />}>
|
||||||
<FormattedMessage id='edit_federation.media_removal' defaultMessage='Strip media' />
|
<Toggle
|
||||||
</Text>
|
checked={media_nsfw}
|
||||||
</HStack>
|
onChange={handleDataChange('media_nsfw')}
|
||||||
<HStack space={2} alignItems='center'>
|
icons={false}
|
||||||
<Toggle
|
id='media_nsfw'
|
||||||
checked={media_nsfw}
|
disabled={reject || media_removal}
|
||||||
onChange={handleDataChange('media_nsfw')}
|
/>
|
||||||
icons={false}
|
</ListItem>
|
||||||
id='media_nsfw'
|
|
||||||
disabled={reject || media_removal}
|
|
||||||
/>
|
|
||||||
|
|
||||||
<Text theme='muted' tag='label' size='sm' htmlFor='media_nsfw'>
|
<ListItem label={<FormattedMessage id='edit_federation.followers_only' defaultMessage='Hide posts except to followers' />}>
|
||||||
<FormattedMessage id='edit_federation.force_nsfw' defaultMessage='Force attachments to be marked sensitive' />
|
<Toggle
|
||||||
</Text>
|
checked={followers_only}
|
||||||
</HStack>
|
onChange={handleDataChange('followers_only')}
|
||||||
<HStack space={2} alignItems='center'>
|
icons={false}
|
||||||
<Toggle
|
id='followers_only'
|
||||||
checked={followers_only}
|
disabled={reject}
|
||||||
onChange={handleDataChange('followers_only')}
|
/>
|
||||||
icons={false}
|
</ListItem>
|
||||||
id='followers_only'
|
|
||||||
disabled={reject}
|
|
||||||
/>
|
|
||||||
|
|
||||||
<Text theme='muted' tag='label' size='sm' htmlFor='followers_only'>
|
<ListItem label={<FormattedMessage id='edit_federation.unlisted' defaultMessage='Force posts unlisted' />}>
|
||||||
<FormattedMessage id='edit_federation.followers_only' defaultMessage='Hide posts except to followers' />
|
<Toggle
|
||||||
</Text>
|
checked={federated_timeline_removal}
|
||||||
</HStack>
|
onChange={handleDataChange('federated_timeline_removal')}
|
||||||
<HStack space={2} alignItems='center'>
|
icons={false}
|
||||||
<Toggle
|
id='federated_timeline_removal'
|
||||||
checked={federated_timeline_removal}
|
disabled={reject || followers_only}
|
||||||
onChange={handleDataChange('federated_timeline_removal')}
|
/>
|
||||||
icons={false}
|
</ListItem>
|
||||||
id='federated_timeline_removal'
|
</List>
|
||||||
disabled={reject || followers_only}
|
|
||||||
/>
|
|
||||||
|
|
||||||
<Text theme='muted' tag='label' size='sm' htmlFor='federated_timeline_removal'>
|
|
||||||
<FormattedMessage id='edit_federation.unlisted' defaultMessage='Force posts unlisted' />
|
|
||||||
</Text>
|
|
||||||
</HStack>
|
|
||||||
</Stack>
|
|
||||||
</SimpleForm>
|
|
||||||
</Modal>
|
</Modal>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
Loading…
Reference in a new issue