Add ability to report a Chat Message
This commit is contained in:
parent
002fef27a3
commit
0a8fa3e635
6 changed files with 82 additions and 19 deletions
|
@ -4,7 +4,7 @@ import { openModal } from './modals';
|
||||||
|
|
||||||
import type { AxiosError } from 'axios';
|
import type { AxiosError } from 'axios';
|
||||||
import type { AppDispatch, RootState } from 'soapbox/store';
|
import type { AppDispatch, RootState } from 'soapbox/store';
|
||||||
import type { Account, Status } from 'soapbox/types/entities';
|
import type { Account, ChatMessage, Status } from 'soapbox/types/entities';
|
||||||
|
|
||||||
const REPORT_INIT = 'REPORT_INIT';
|
const REPORT_INIT = 'REPORT_INIT';
|
||||||
const REPORT_CANCEL = 'REPORT_CANCEL';
|
const REPORT_CANCEL = 'REPORT_CANCEL';
|
||||||
|
@ -20,17 +20,25 @@ const REPORT_BLOCK_CHANGE = 'REPORT_BLOCK_CHANGE';
|
||||||
|
|
||||||
const REPORT_RULE_CHANGE = 'REPORT_RULE_CHANGE';
|
const REPORT_RULE_CHANGE = 'REPORT_RULE_CHANGE';
|
||||||
|
|
||||||
const initReport = (account: Account, status?: Status) =>
|
type ReportedEntity = {
|
||||||
(dispatch: AppDispatch) => {
|
status?: Status,
|
||||||
|
chatMessage?: ChatMessage
|
||||||
|
}
|
||||||
|
|
||||||
|
const initReport = (account: Account, entities?: ReportedEntity) => (dispatch: AppDispatch) => {
|
||||||
|
const { status, chatMessage } = entities || {};
|
||||||
|
|
||||||
dispatch({
|
dispatch({
|
||||||
type: REPORT_INIT,
|
type: REPORT_INIT,
|
||||||
account,
|
account,
|
||||||
status,
|
status,
|
||||||
|
chatMessage,
|
||||||
});
|
});
|
||||||
|
|
||||||
return dispatch(openModal('REPORT'));
|
return dispatch(openModal('REPORT'));
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// TODO: no longer used. Can be removed.
|
||||||
const initReportById = (accountId: string) =>
|
const initReportById = (accountId: string) =>
|
||||||
(dispatch: AppDispatch, getState: () => RootState) => {
|
(dispatch: AppDispatch, getState: () => RootState) => {
|
||||||
dispatch({
|
dispatch({
|
||||||
|
@ -59,6 +67,7 @@ const submitReport = () =>
|
||||||
return api(getState).post('/api/v1/reports', {
|
return api(getState).post('/api/v1/reports', {
|
||||||
account_id: reports.getIn(['new', 'account_id']),
|
account_id: reports.getIn(['new', 'account_id']),
|
||||||
status_ids: reports.getIn(['new', 'status_ids']),
|
status_ids: reports.getIn(['new', 'status_ids']),
|
||||||
|
message_ids: [reports.getIn(['new', 'chat_message', 'id'])],
|
||||||
rule_ids: reports.getIn(['new', 'rule_ids']),
|
rule_ids: reports.getIn(['new', 'rule_ids']),
|
||||||
comment: reports.getIn(['new', 'comment']),
|
comment: reports.getIn(['new', 'comment']),
|
||||||
forward: reports.getIn(['new', 'forward']),
|
forward: reports.getIn(['new', 'forward']),
|
||||||
|
|
|
@ -251,7 +251,7 @@ const StatusActionBar: React.FC<IStatusActionBar> = ({
|
||||||
secondary: intl.formatMessage(messages.blockAndReport),
|
secondary: intl.formatMessage(messages.blockAndReport),
|
||||||
onSecondary: () => {
|
onSecondary: () => {
|
||||||
dispatch(blockAccount(account.id));
|
dispatch(blockAccount(account.id));
|
||||||
dispatch(initReport(account, status));
|
dispatch(initReport(account, { status }));
|
||||||
},
|
},
|
||||||
}));
|
}));
|
||||||
};
|
};
|
||||||
|
@ -270,7 +270,7 @@ const StatusActionBar: React.FC<IStatusActionBar> = ({
|
||||||
|
|
||||||
const handleReport: React.EventHandler<React.MouseEvent> = (e) => {
|
const handleReport: React.EventHandler<React.MouseEvent> = (e) => {
|
||||||
e.stopPropagation();
|
e.stopPropagation();
|
||||||
dispatch(initReport(status.account as Account, status));
|
dispatch(initReport(status.account as Account, { status }));
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleConversationMuteClick: React.EventHandler<React.MouseEvent> = (e) => {
|
const handleConversationMuteClick: React.EventHandler<React.MouseEvent> = (e) => {
|
||||||
|
|
|
@ -18,7 +18,7 @@ const Divider = ({ text, textSize = 'md' }: IDivider) => (
|
||||||
|
|
||||||
{text && (
|
{text && (
|
||||||
<div className='relative flex justify-center'>
|
<div className='relative flex justify-center'>
|
||||||
<span className='px-2 bg-white dark:bg-gray-900 text-gray-400' data-testid='divider-text'>
|
<span className='px-2 bg-white dark:bg-gray-900 text-gray-700 dark:text-gray-600' data-testid='divider-text'>
|
||||||
<Text size={textSize} tag='span' theme='inherit'>{text}</Text>
|
<Text size={textSize} tag='span' theme='inherit'>{text}</Text>
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -7,6 +7,7 @@ import React, { useState, useEffect, useRef } from 'react';
|
||||||
import { useIntl, defineMessages } from 'react-intl';
|
import { useIntl, defineMessages } from 'react-intl';
|
||||||
|
|
||||||
import { openModal } from 'soapbox/actions/modals';
|
import { openModal } from 'soapbox/actions/modals';
|
||||||
|
import { initReport } from 'soapbox/actions/reports';
|
||||||
import { Avatar, Button, Divider, HStack, Spinner, Stack, Text } from 'soapbox/components/ui';
|
import { Avatar, Button, Divider, HStack, Spinner, Stack, Text } from 'soapbox/components/ui';
|
||||||
import DropdownMenuContainer from 'soapbox/containers/dropdown_menu_container';
|
import DropdownMenuContainer from 'soapbox/containers/dropdown_menu_container';
|
||||||
// import emojify from 'soapbox/features/emoji/emoji';
|
// import emojify from 'soapbox/features/emoji/emoji';
|
||||||
|
@ -14,6 +15,7 @@ import PlaceholderChatMessage from 'soapbox/features/placeholder/components/plac
|
||||||
import Bundle from 'soapbox/features/ui/components/bundle';
|
import Bundle from 'soapbox/features/ui/components/bundle';
|
||||||
import { MediaGallery } from 'soapbox/features/ui/util/async-components';
|
import { MediaGallery } from 'soapbox/features/ui/util/async-components';
|
||||||
import { useAppSelector, useAppDispatch, useOwnAccount } from 'soapbox/hooks';
|
import { useAppSelector, useAppDispatch, useOwnAccount } from 'soapbox/hooks';
|
||||||
|
import { normalizeAccount } from 'soapbox/normalizers';
|
||||||
import { chatKeys, IChat, IChatMessage, useChat, useChatMessages } from 'soapbox/queries/chats';
|
import { chatKeys, IChat, IChatMessage, useChat, useChatMessages } from 'soapbox/queries/chats';
|
||||||
import { queryClient } from 'soapbox/queries/client';
|
import { queryClient } from 'soapbox/queries/client';
|
||||||
import { onlyEmoji } from 'soapbox/utils/rich_content';
|
import { onlyEmoji } from 'soapbox/utils/rich_content';
|
||||||
|
@ -246,7 +248,7 @@ const ChatMessageList: React.FC<IChatMessageList> = ({ chat, autosize }) => {
|
||||||
} else {
|
} else {
|
||||||
menu.push({
|
menu.push({
|
||||||
text: intl.formatMessage(messages.report),
|
text: intl.formatMessage(messages.report),
|
||||||
action: () => null, // TODO: implement once API is available
|
action: () => dispatch(initReport(normalizeAccount(chat.account) as any, { chatMessage })),
|
||||||
icon: require('@tabler/icons/flag.svg'),
|
icon: require('@tabler/icons/flag.svg'),
|
||||||
});
|
});
|
||||||
menu.push({
|
menu.push({
|
||||||
|
|
|
@ -6,7 +6,7 @@ import { submitReport, submitReportSuccess, submitReportFail } from 'soapbox/act
|
||||||
import { expandAccountTimeline } from 'soapbox/actions/timelines';
|
import { expandAccountTimeline } from 'soapbox/actions/timelines';
|
||||||
import AttachmentThumbs from 'soapbox/components/attachment-thumbs';
|
import AttachmentThumbs from 'soapbox/components/attachment-thumbs';
|
||||||
import StatusContent from 'soapbox/components/status_content';
|
import StatusContent from 'soapbox/components/status_content';
|
||||||
import { Modal, ProgressBar, Stack, Text } from 'soapbox/components/ui';
|
import { Avatar, HStack, Modal, ProgressBar, Stack, Text } from 'soapbox/components/ui';
|
||||||
import AccountContainer from 'soapbox/containers/account_container';
|
import AccountContainer from 'soapbox/containers/account_container';
|
||||||
import { useAccount, useAppDispatch, useAppSelector } from 'soapbox/hooks';
|
import { useAccount, useAppDispatch, useAppSelector } from 'soapbox/hooks';
|
||||||
|
|
||||||
|
@ -74,6 +74,12 @@ interface IReportModal {
|
||||||
onClose: () => void
|
onClose: () => void
|
||||||
}
|
}
|
||||||
|
|
||||||
|
enum ReportedEntities {
|
||||||
|
Account = 'Account',
|
||||||
|
Status = 'Status',
|
||||||
|
ChatMessage = 'ChatMessage'
|
||||||
|
}
|
||||||
|
|
||||||
const ReportModal = ({ onClose }: IReportModal) => {
|
const ReportModal = ({ onClose }: IReportModal) => {
|
||||||
const dispatch = useAppDispatch();
|
const dispatch = useAppDispatch();
|
||||||
const intl = useIntl();
|
const intl = useIntl();
|
||||||
|
@ -86,10 +92,23 @@ const ReportModal = ({ onClose }: IReportModal) => {
|
||||||
const rules = useAppSelector((state) => state.rules.items);
|
const rules = useAppSelector((state) => state.rules.items);
|
||||||
const ruleIds = useAppSelector((state) => state.reports.new.rule_ids);
|
const ruleIds = useAppSelector((state) => state.reports.new.rule_ids);
|
||||||
const selectedStatusIds = useAppSelector((state) => state.reports.new.status_ids);
|
const selectedStatusIds = useAppSelector((state) => state.reports.new.status_ids);
|
||||||
|
const selectedChatMessage = useAppSelector((state) => state.reports.new.chat_message);
|
||||||
|
|
||||||
const isReportingAccount = useMemo(() => selectedStatusIds.size === 0, []);
|
|
||||||
const shouldRequireRule = rules.length > 0;
|
const shouldRequireRule = rules.length > 0;
|
||||||
|
|
||||||
|
const reportedEntity = useMemo(() => {
|
||||||
|
if (selectedStatusIds.size === 0 && !selectedChatMessage) {
|
||||||
|
return ReportedEntities.Account;
|
||||||
|
} else if (selectedChatMessage) {
|
||||||
|
return ReportedEntities.ChatMessage;
|
||||||
|
} else {
|
||||||
|
return ReportedEntities.Status;
|
||||||
|
}
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const isReportingAccount = reportedEntity === ReportedEntities.Account;
|
||||||
|
const isReportingStatus = reportedEntity === ReportedEntities.Status;
|
||||||
|
|
||||||
const [currentStep, setCurrentStep] = useState<Steps>(Steps.ONE);
|
const [currentStep, setCurrentStep] = useState<Steps>(Steps.ONE);
|
||||||
|
|
||||||
const handleSubmit = () => {
|
const handleSubmit = () => {
|
||||||
|
@ -132,6 +151,31 @@ const ReportModal = ({ onClose }: IReportModal) => {
|
||||||
}
|
}
|
||||||
}, [selectedStatusIds.size]);
|
}, [selectedStatusIds.size]);
|
||||||
|
|
||||||
|
const renderSelectedChatMessage = () => {
|
||||||
|
if (account) {
|
||||||
|
return (
|
||||||
|
<HStack alignItems='center' space={4} className='rounded-md border dark:border-2 border-solid border-gray-400 dark:border-gray-800 p-4'>
|
||||||
|
<div>
|
||||||
|
<Avatar src={account.avatar} className='w-8 h-8' />
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className='bg-gray-200 dark:bg-primary-800 rounded-md p-4 flex-grow'>
|
||||||
|
<Text dangerouslySetInnerHTML={{ __html: selectedChatMessage?.content as string }} />
|
||||||
|
</div>
|
||||||
|
</HStack>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const renderSelectedEntity = () => {
|
||||||
|
switch (reportedEntity) {
|
||||||
|
case ReportedEntities.Status:
|
||||||
|
return renderSelectedStatuses();
|
||||||
|
case ReportedEntities.ChatMessage:
|
||||||
|
return renderSelectedChatMessage();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
const confirmationText = useMemo(() => {
|
const confirmationText = useMemo(() => {
|
||||||
switch (currentStep) {
|
switch (currentStep) {
|
||||||
case Steps.TWO:
|
case Steps.TWO:
|
||||||
|
@ -148,8 +192,8 @@ const ReportModal = ({ onClose }: IReportModal) => {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
return isSubmitting || (shouldRequireRule && ruleIds.isEmpty()) || (!isReportingAccount && selectedStatusIds.size === 0);
|
return isSubmitting || (shouldRequireRule && ruleIds.isEmpty()) || (isReportingStatus && selectedStatusIds.size === 0);
|
||||||
}, [currentStep, isSubmitting, shouldRequireRule, ruleIds, selectedStatusIds.size, isReportingAccount]);
|
}, [currentStep, isSubmitting, shouldRequireRule, ruleIds, selectedStatusIds.size, isReportingStatus]);
|
||||||
|
|
||||||
const calculateProgress = useCallback(() => {
|
const calculateProgress = useCallback(() => {
|
||||||
switch (currentStep) {
|
switch (currentStep) {
|
||||||
|
@ -189,7 +233,7 @@ const ReportModal = ({ onClose }: IReportModal) => {
|
||||||
<Stack space={4}>
|
<Stack space={4}>
|
||||||
<ProgressBar progress={calculateProgress()} />
|
<ProgressBar progress={calculateProgress()} />
|
||||||
|
|
||||||
{(currentStep !== Steps.THREE && !isReportingAccount) && renderSelectedStatuses()}
|
{(currentStep !== Steps.THREE && !isReportingAccount) && renderSelectedEntity()}
|
||||||
|
|
||||||
<StepToRender account={account} />
|
<StepToRender account={account} />
|
||||||
</Stack>
|
</Stack>
|
||||||
|
|
|
@ -1,5 +1,7 @@
|
||||||
import { Record as ImmutableRecord, Set as ImmutableSet } from 'immutable';
|
import { Record as ImmutableRecord, Set as ImmutableSet } from 'immutable';
|
||||||
|
|
||||||
|
import { ChatMessage } from 'soapbox/types/entities';
|
||||||
|
|
||||||
import {
|
import {
|
||||||
REPORT_INIT,
|
REPORT_INIT,
|
||||||
REPORT_SUBMIT_REQUEST,
|
REPORT_SUBMIT_REQUEST,
|
||||||
|
@ -19,6 +21,7 @@ const NewReportRecord = ImmutableRecord({
|
||||||
isSubmitting: false,
|
isSubmitting: false,
|
||||||
account_id: null as string | null,
|
account_id: null as string | null,
|
||||||
status_ids: ImmutableSet<string>(),
|
status_ids: ImmutableSet<string>(),
|
||||||
|
chat_message: null as null | ChatMessage,
|
||||||
comment: '',
|
comment: '',
|
||||||
forward: false,
|
forward: false,
|
||||||
block: false,
|
block: false,
|
||||||
|
@ -38,6 +41,10 @@ export default function reports(state: State = ReducerRecord(), action: AnyActio
|
||||||
map.setIn(['new', 'isSubmitting'], false);
|
map.setIn(['new', 'isSubmitting'], false);
|
||||||
map.setIn(['new', 'account_id'], action.account.id);
|
map.setIn(['new', 'account_id'], action.account.id);
|
||||||
|
|
||||||
|
if (action.chatMessage) {
|
||||||
|
map.setIn(['new', 'chat_message'], action.chatMessage);
|
||||||
|
}
|
||||||
|
|
||||||
if (state.new.account_id !== action.account.id) {
|
if (state.new.account_id !== action.account.id) {
|
||||||
map.setIn(['new', 'status_ids'], action.status ? ImmutableSet([action.status.reblog?.id || action.status.id]) : ImmutableSet());
|
map.setIn(['new', 'status_ids'], action.status ? ImmutableSet([action.status.reblog?.id || action.status.id]) : ImmutableSet());
|
||||||
map.setIn(['new', 'comment'], '');
|
map.setIn(['new', 'comment'], '');
|
||||||
|
@ -76,6 +83,7 @@ export default function reports(state: State = ReducerRecord(), action: AnyActio
|
||||||
return state.withMutations(map => {
|
return state.withMutations(map => {
|
||||||
map.setIn(['new', 'account_id'], null);
|
map.setIn(['new', 'account_id'], null);
|
||||||
map.setIn(['new', 'status_ids'], ImmutableSet());
|
map.setIn(['new', 'status_ids'], ImmutableSet());
|
||||||
|
map.setIn(['new', 'chat_message'], null);
|
||||||
map.setIn(['new', 'comment'], '');
|
map.setIn(['new', 'comment'], '');
|
||||||
map.setIn(['new', 'isSubmitting'], false);
|
map.setIn(['new', 'isSubmitting'], false);
|
||||||
map.setIn(['new', 'rule_ids'], ImmutableSet());
|
map.setIn(['new', 'rule_ids'], ImmutableSet());
|
||||||
|
|
Loading…
Reference in a new issue