You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
160 lines
5.0 KiB
TypeScript
160 lines
5.0 KiB
TypeScript
import React, { ReactElement, useRef, useState } from 'react';
|
|
import { SortedReactionList } from '../../../../types/Reaction';
|
|
import { UserUtils } from '../../../../session/utils';
|
|
import { abbreviateNumber } from '../../../../util/abbreviateNumber';
|
|
import { nativeEmojiData } from '../../../../util/emoji';
|
|
import styled from 'styled-components';
|
|
import { useMouse } from 'react-use';
|
|
import { ReactionPopup, TipPosition } from './ReactionPopup';
|
|
import { popupXDefault, popupYDefault } from '../message-content/MessageReactions';
|
|
import { isUsAnySogsFromCache } from '../../../../session/apis/open_group_api/sogsv3/knownBlindedkeys';
|
|
|
|
const StyledReaction = styled.button<{ selected: boolean; inModal: boolean; showCount: boolean }>`
|
|
display: flex;
|
|
justify-content: ${props => (props.showCount ? 'flex-start' : 'center')};
|
|
align-items: center;
|
|
|
|
background-color: var(--color-received-message-background);
|
|
border-width: 1px;
|
|
border-style: solid;
|
|
border-color: ${props => (props.selected ? 'var(--color-accent)' : 'transparent')};
|
|
border-radius: 11px;
|
|
box-sizing: border-box;
|
|
padding: 0 7px;
|
|
margin: 0 4px var(--margins-sm);
|
|
height: 24px;
|
|
min-width: ${props => (props.showCount ? '48px' : '24px')};
|
|
${props => props.inModal && 'width: 100%;'}
|
|
|
|
span {
|
|
width: 100%;
|
|
}
|
|
`;
|
|
|
|
const StyledReactionContainer = styled.div<{
|
|
inModal: boolean;
|
|
}>`
|
|
position: relative;
|
|
${props => props.inModal && 'margin-right: 8px;'}
|
|
`;
|
|
|
|
export type ReactionProps = {
|
|
emoji: string;
|
|
messageId: string;
|
|
reactions: SortedReactionList;
|
|
inModal: boolean;
|
|
inGroup: boolean;
|
|
handlePopupX: (x: number) => void;
|
|
handlePopupY: (y: number) => void;
|
|
onClick: (emoji: string) => void;
|
|
popupReaction?: string;
|
|
onSelected?: (emoji: string) => boolean;
|
|
handlePopupReaction?: (emoji: string) => void;
|
|
handlePopupClick?: () => void;
|
|
};
|
|
|
|
export const Reaction = (props: ReactionProps): ReactElement => {
|
|
const {
|
|
emoji,
|
|
messageId,
|
|
reactions,
|
|
inModal,
|
|
inGroup,
|
|
handlePopupX,
|
|
handlePopupY,
|
|
onClick,
|
|
popupReaction,
|
|
onSelected,
|
|
handlePopupReaction,
|
|
handlePopupClick,
|
|
} = props;
|
|
const reactionsMap = (reactions && Object.fromEntries(reactions)) || {};
|
|
const senders = reactionsMap[emoji]?.senders || [];
|
|
const count = reactionsMap[emoji]?.count;
|
|
const showCount = count !== undefined && (count > 1 || inGroup);
|
|
|
|
const reactionRef = useRef<HTMLDivElement>(null);
|
|
const { docX, elW } = useMouse(reactionRef);
|
|
|
|
const gutterWidth = 380;
|
|
const tooltipMidPoint = 108; // width is 216px;
|
|
const [tooltipPosition, setTooltipPosition] = useState<TipPosition>('center');
|
|
|
|
const me = UserUtils.getOurPubKeyStrFromCache();
|
|
const isBlindedMe =
|
|
senders && senders.length > 0 && senders.filter(isUsAnySogsFromCache).length > 0;
|
|
|
|
const selected = () => {
|
|
if (onSelected) {
|
|
return onSelected(emoji);
|
|
}
|
|
|
|
return senders && senders.length > 0 && (senders.includes(me) || isBlindedMe);
|
|
};
|
|
|
|
const handleReactionClick = () => {
|
|
onClick(emoji);
|
|
};
|
|
|
|
return (
|
|
<StyledReactionContainer ref={reactionRef} inModal={inModal}>
|
|
<StyledReaction
|
|
showCount={showCount}
|
|
selected={selected()}
|
|
inModal={inModal}
|
|
onClick={handleReactionClick}
|
|
onMouseEnter={() => {
|
|
if (inGroup) {
|
|
const { innerWidth: windowWidth } = window;
|
|
if (handlePopupReaction) {
|
|
// overflow on far right means we shift left
|
|
if (docX + tooltipMidPoint > windowWidth) {
|
|
handlePopupX(Math.abs(popupXDefault) * 1.5 * -1);
|
|
setTooltipPosition('right');
|
|
// overflow onto conversations means we lock to the right
|
|
} else if (docX - elW <= gutterWidth + tooltipMidPoint) {
|
|
const offset = -12.5;
|
|
handlePopupX(offset);
|
|
setTooltipPosition('left');
|
|
} else {
|
|
handlePopupX(popupXDefault);
|
|
setTooltipPosition('center');
|
|
}
|
|
|
|
handlePopupReaction(emoji);
|
|
}
|
|
}
|
|
}}
|
|
>
|
|
<span
|
|
role={'img'}
|
|
aria-label={nativeEmojiData?.ariaLabels ? nativeEmojiData.ariaLabels[emoji] : undefined}
|
|
>
|
|
{emoji}
|
|
{showCount && `\u00A0\u00A0${abbreviateNumber(count)}`}
|
|
</span>
|
|
</StyledReaction>
|
|
{inGroup && popupReaction && popupReaction === emoji && (
|
|
<ReactionPopup
|
|
messageId={messageId}
|
|
emoji={popupReaction}
|
|
count={reactionsMap[popupReaction]?.count}
|
|
senders={reactionsMap[popupReaction]?.senders}
|
|
tooltipPosition={tooltipPosition}
|
|
onClick={() => {
|
|
if (handlePopupReaction) {
|
|
handlePopupReaction('');
|
|
}
|
|
handlePopupX(popupXDefault);
|
|
handlePopupY(popupYDefault);
|
|
setTooltipPosition('center');
|
|
if (handlePopupClick) {
|
|
handlePopupClick();
|
|
}
|
|
}}
|
|
/>
|
|
)}
|
|
</StyledReactionContainer>
|
|
);
|
|
};
|