@selfcommunity/react-ui
Version:
React UI Components to integrate a Community created with SelfCommunity Platform.
149 lines (140 loc) • 6.84 kB
JavaScript
import { __rest } from "tslib";
import { jsx as _jsx, Fragment as _Fragment, jsxs as _jsxs } from "react/jsx-runtime";
import { useMemo, useRef, useState } from 'react';
import { styled } from '@mui/material/styles';
import { LoadingButton } from '@mui/lab';
import classNames from 'classnames';
import { useThemeProps } from '@mui/system';
import Icon from '@mui/material/Icon';
import { IconButton, Paper, Popper, Tooltip, useMediaQuery, useTheme } from '@mui/material';
import { UserUtils, useSCContext, useSCFetchVote, useSCUser } from '@selfcommunity/react-core';
import { FormattedMessage } from 'react-intl';
import { useSnackbar } from 'notistack';
import { catchUnauthorizedActionByBlockedUser } from '../../utils/errors';
const PREFIX = 'SCVoteButton';
const classes = {
root: `${PREFIX}-root`,
voted: `${PREFIX}-voted`,
popperRoot: `${PREFIX}-popper-root`,
reactionList: `${PREFIX}-reaction-list`,
reaction: `${PREFIX}-reaction`
};
const Root = styled(LoadingButton, {
name: PREFIX,
slot: 'Root',
overridesResolver: (props, styles) => [styles.root, styles.voted]
})(({ theme }) => ({}));
const PopperRoot = styled(Popper, {
name: PREFIX,
slot: 'Root',
overridesResolver: (props, styles) => styles.popperRoot
})(() => ({}));
/**
* > API documentation for the Community-JS Vote Button component. Learn about the available props and the CSS API.
#### Import
```jsx
import {VoteAudienceButton} from '@selfcommunity/react-ui';
```
#### Component Name
The name `SCVoteButton` can be used when providing style overrides in the theme.
#### CSS
|Rule Name|Global class|Description|
|---|---|---|
|root|.SCVoteButton-root|Styles applied to the root element.|
|voted|.SCVoteButton-voted|Styles applied to the root element when the user has vote the contribution.|
* @param inProps
*/
export default function VoteButton(inProps) {
// PROPS
const props = useThemeProps({
props: inProps,
name: PREFIX
});
const { className, contributionId, contributionType, contribution = null, onVote } = props, rest = __rest(props, ["className", "contributionId", "contributionType", "contribution", "onVote"]);
// STATE
const [anchorEl, setAnchorEl] = useState(null);
// REF
const timeoutRef = useRef(null);
// CONTEXT
const scContext = useSCContext();
const scUserContext = useSCUser();
const { enqueueSnackbar } = useSnackbar();
// HANDLERS
const handleMouseEnter = (event) => {
handleClearTimeout();
timeoutRef.current = setTimeout(() => setAnchorEl(event.target), 1000);
};
const handleMouseLeave = (event) => {
handleClearTimeout();
timeoutRef.current = setTimeout(() => setAnchorEl(null), 500);
};
const handleClearTimeout = () => {
timeoutRef.current && clearTimeout(timeoutRef.current);
timeoutRef.current = null;
};
/**
* Perform vote action
* @param reaction
*/
const handleVoteAction = (reaction) => {
if (!scUserContext.user) {
scContext.settings.handleAnonymousAction();
}
else if (UserUtils.isBlocked(scUserContext.user)) {
enqueueSnackbar(_jsx(FormattedMessage, { id: "ui.common.userBlocked", defaultMessage: "ui.common.userBlocked" }), {
variant: 'warning',
autoHideDuration: 3000
});
}
else {
handleVote(reaction);
}
};
/**
* Handle callback onVote
* @param contribution
* @param error
*/
const handleOnVote = (contribution, error) => {
setAnchorEl(null);
if (error) {
if (!catchUnauthorizedActionByBlockedUser(error, scUserContext.managers.blockedUsers.isBlocked(contribution.author), enqueueSnackbar)) {
enqueueSnackbar(_jsx(FormattedMessage, { id: "ui.common.error.action", defaultMessage: "ui.common.error.action" }), {
variant: 'error',
autoHideDuration: 3000
});
}
}
else {
onVote && onVote(contribution);
}
};
// HOOKS
const { isLoading, isVoting, contributionVoted, contributionReaction, reactions, handleVote, error } = useSCFetchVote({
id: contributionId,
contributionType,
contribution,
onVote: handleOnVote
});
const theme = useTheme();
const isMobile = useMediaQuery(theme.breakpoints.down('md'));
// MEMO
const rootProps = useMemo(() => {
if (!reactions.default) {
return {};
}
return {
onMouseEnter: handleMouseEnter,
onMouseLeave: handleMouseLeave
};
}, [reactions]);
// RENDER
const button = (_jsx(Root, Object.assign({ onClick: isMobile && reactions.reactions
? handleMouseEnter
: () => handleVoteAction(contributionReaction ? contributionReaction : reactions.default ? reactions.default : null), disabled: isLoading || Boolean(error), loading: isVoting, className: classNames(classes.root, className, {
[classes.voted]: scUserContext.user && contributionVoted
}) }, rest, rootProps, { children: scUserContext.user && contributionVoted ? (contributionReaction ? (_jsx(Icon, { children: _jsx("img", { alt: contributionReaction.label, src: contributionReaction.image, width: "100%", height: "100%" }) })) : (_jsx(Icon, { children: "thumb_up" }))) : (_jsx(Icon, { children: "thumb_up_off_alt" })) })));
return (_jsxs(_Fragment, { children: [reactions.default ? (button) : (_jsx(Tooltip, Object.assign({ title: contributionVoted ? (_jsx(FormattedMessage, { id: "ui.voteButton.voteDown", defaultMessage: "ui.voteButton.voteDown" })) : (_jsx(FormattedMessage, { id: "ui.voteButton.voteUp", defaultMessage: "ui.voteButton.voteUp" })) }, { children: _jsx("span", { children: button }) }))), reactions.default && Boolean(anchorEl) && !isVoting && !isLoading && (_jsx(PopperRoot, Object.assign({ id: `vote_${contributionId}_${contributionType}_popper`, className: classes.popperRoot, open: true, anchorEl: anchorEl, placement: "top", keepMounted: true }, { children: _jsx(Paper, Object.assign({ className: classes.reactionList, onMouseEnter: handleClearTimeout, onMouseLeave: handleMouseLeave }, { children: reactions.reactions
.filter((reaction) => !contributionVoted || (contributionVoted && contributionReaction.id !== reaction.id) ? reaction.active : reaction)
.map((reaction) => (_jsx(IconButton, Object.assign({ className: classes.reaction, onClick: () => handleVoteAction(reaction) }, { children: _jsx(Icon, { children: _jsx("img", { alt: reaction.label, src: reaction.image, width: "100%", height: "100%" }) }) }), reaction.id))) })) })))] }));
}