pleroma/app/soapbox/components/polls/poll.tsx

102 lines
2.5 KiB
TypeScript
Raw Normal View History

2022-06-15 12:59:17 -07:00
import classNames from 'classnames';
import React, { useState } from 'react';
2022-06-21 04:35:19 -07:00
import { defineMessages, useIntl } from 'react-intl';
2022-06-15 12:59:17 -07:00
import { openModal } from 'soapbox/actions/modals';
import { vote } from 'soapbox/actions/polls';
import { useAppDispatch, useAppSelector } from 'soapbox/hooks';
2022-06-21 04:35:19 -07:00
import { Stack, Text } from '../ui';
2022-06-15 12:59:17 -07:00
import PollFooter from './poll-footer';
import PollOption from './poll-option';
export type Selected = Record<number, boolean>;
interface IPoll {
id: string,
status?: string,
}
2022-06-21 04:35:19 -07:00
const messages = defineMessages({
multiple: { id: 'poll.chooseMultiple', defaultMessage: 'Choose as many as you\'d like.' },
});
2022-06-15 12:59:17 -07:00
const Poll: React.FC<IPoll> = ({ id, status }): JSX.Element | null => {
const dispatch = useAppDispatch();
2022-06-21 04:35:19 -07:00
const intl = useIntl();
2022-06-15 12:59:17 -07:00
const isLoggedIn = useAppSelector((state) => state.me);
const poll = useAppSelector((state) => state.polls.get(id));
const [selected, setSelected] = useState({} as Selected);
const openUnauthorizedModal = () =>
dispatch(openModal('UNAUTHORIZED', {
action: 'POLL_VOTE',
ap_id: status,
}));
2022-06-21 09:25:20 -07:00
const handleVote = (selectedId: number) => dispatch(vote(id, [String(selectedId)]));
2022-06-15 12:59:17 -07:00
const toggleOption = (value: number) => {
if (isLoggedIn) {
if (poll?.multiple) {
const tmp = { ...selected };
if (tmp[value]) {
delete tmp[value];
} else {
tmp[value] = true;
}
setSelected(tmp);
} else {
const tmp: Selected = {};
tmp[value] = true;
setSelected(tmp);
handleVote(value);
}
} else {
openUnauthorizedModal();
}
};
if (!poll) return null;
const showResults = poll.voted || poll.expired;
return (
// eslint-disable-next-line jsx-a11y/no-static-element-interactions
<div onClick={e => e.stopPropagation()}>
2022-06-21 04:35:19 -07:00
{poll.multiple && (
<Text theme='muted' size='sm'>
{intl.formatMessage(messages.multiple)}
</Text>
)}
2022-06-16 06:20:33 -07:00
<Stack space={4} className={classNames('mt-4')}>
2022-06-15 12:59:17 -07:00
<Stack space={2}>
{poll.options.map((option, i) => (
<PollOption
key={i}
poll={poll}
option={option}
index={i}
showResults={showResults}
active={!!selected[i]}
onToggle={toggleOption}
/>
))}
</Stack>
<PollFooter
poll={poll}
showResults={showResults}
selected={selected}
/>
</Stack>
</div>
);
};
export default Poll;