From e973d69c61fb0b8ad30d6fffe66685eca1ca5543 Mon Sep 17 00:00:00 2001 From: Alex Gleason Date: Sat, 19 Nov 2022 14:36:58 -0600 Subject: [PATCH 1/7] Stop mouseUp propagation in statuses --- app/soapbox/components/account.tsx | 161 +++++++++--------- app/soapbox/components/polls/poll.tsx | 6 +- app/soapbox/components/quoted-status.tsx | 85 ++++----- app/soapbox/components/status-action-bar.tsx | 148 +++++++--------- app/soapbox/components/status-media.tsx | 11 +- app/soapbox/components/status.tsx | 6 +- .../statuses/sensitive-content-overlay.tsx | 77 +++++---- app/soapbox/components/stop-propagation.tsx | 28 +++ app/soapbox/components/ui/button/button.tsx | 4 +- app/soapbox/components/ui/hstack/hstack.tsx | 2 +- 10 files changed, 269 insertions(+), 259 deletions(-) create mode 100644 app/soapbox/components/stop-propagation.tsx diff --git a/app/soapbox/components/account.tsx b/app/soapbox/components/account.tsx index 006584e8d..db8d4b306 100644 --- a/app/soapbox/components/account.tsx +++ b/app/soapbox/components/account.tsx @@ -9,6 +9,7 @@ import { getAcct } from 'soapbox/utils/accounts'; import { displayFqn } from 'soapbox/utils/state'; import RelativeTimestamp from './relative-timestamp'; +import StopPropagation from './stop-propagation'; import { Avatar, Emoji, HStack, Icon, IconButton, Stack, Text } from './ui'; import type { Account as AccountEntity } from 'soapbox/types/entities'; @@ -21,8 +22,6 @@ const InstanceFavicon: React.FC = ({ account }) => { const history = useHistory(); const handleClick: React.MouseEventHandler = (e) => { - e.stopPropagation(); - const timelineUrl = `/timeline/${account.domain}`; if (!(e.ctrlKey || e.metaKey)) { history.push(timelineUrl); @@ -167,106 +166,100 @@ const Account = ({ const LinkEl: any = withLinkToProfile ? Link : 'div'; return ( -
- - - {children}} - > - event.stopPropagation()} - > - - {emoji && ( - - )} - - - -
+ +
+ + {children}} + wrapper={(children) => {children}} > - event.stopPropagation()} - > -
- + + {emoji && ( + - - {account.verified && } -
+ )}
- - - @{username} +
+ {children}} + > + +
+ - {account.favicon && ( - - )} + {account.verified && } +
+
+
- {(timestamp) ? ( - <> - · + + + @{username} - {timestampUrl ? ( - event.stopPropagation()}> + {account.favicon && ( + + )} + + {(timestamp) ? ( + <> + · + + {timestampUrl ? ( + + + + ) : ( - - ) : ( - - )} - - ) : null} + )} + + ) : null} - {showEdit ? ( - <> - · + {showEdit ? ( + <> + · - - - ) : null} + + + ) : null} - {actionType === 'muting' && account.mute_expires_at ? ( - <> - · + {actionType === 'muting' && account.mute_expires_at ? ( + <> + · - - - ) : null} - + + + ) : null} + - {withAccountNote && ( - - )} - + {withAccountNote && ( + + )} + +
+
+ +
+ {withRelationship ? renderAction() : null}
- -
- {withRelationship ? renderAction() : null} -
-
-
+
+ ); }; diff --git a/app/soapbox/components/polls/poll.tsx b/app/soapbox/components/polls/poll.tsx index 2b6f99392..2df985b7e 100644 --- a/app/soapbox/components/polls/poll.tsx +++ b/app/soapbox/components/polls/poll.tsx @@ -5,6 +5,7 @@ import { openModal } from 'soapbox/actions/modals'; import { vote } from 'soapbox/actions/polls'; import { useAppDispatch, useAppSelector } from 'soapbox/hooks'; +import StopPropagation from '../stop-propagation'; import { Stack, Text } from '../ui'; import PollFooter from './poll-footer'; @@ -64,8 +65,7 @@ const Poll: React.FC = ({ id, status }): JSX.Element | null => { const showResults = poll.voted || poll.expired; return ( - // eslint-disable-next-line jsx-a11y/no-static-element-interactions -
e.stopPropagation()}> + {!showResults && poll.multiple && ( {intl.formatMessage(messages.multiple)} @@ -93,7 +93,7 @@ const Poll: React.FC = ({ id, status }): JSX.Element | null => { selected={selected} /> -
+ ); }; diff --git a/app/soapbox/components/quoted-status.tsx b/app/soapbox/components/quoted-status.tsx index 2c976bf86..925f2115e 100644 --- a/app/soapbox/components/quoted-status.tsx +++ b/app/soapbox/components/quoted-status.tsx @@ -13,6 +13,7 @@ import OutlineBox from './outline-box'; import StatusContent from './status-content'; import StatusReplyMentions from './status-reply-mentions'; import SensitiveContentOverlay from './statuses/sensitive-content-overlay'; +import StopPropagation from './stop-propagation'; import type { Account as AccountEntity, Status as StatusEntity } from 'soapbox/types/entities'; @@ -91,58 +92,60 @@ const QuotedStatus: React.FC = ({ status, onCancel, compose }) => } return ( - - + - - - - - {(status.hidden) && ( - - )} + - - + - {(status.card || status.media_attachments.size > 0) && ( - + {(status.hidden) && ( + )} + + + + + {(status.card || status.media_attachments.size > 0) && ( + + )} + - - + + ); }; diff --git a/app/soapbox/components/status-action-bar.tsx b/app/soapbox/components/status-action-bar.tsx index 011fc63a8..54c181c71 100644 --- a/app/soapbox/components/status-action-bar.tsx +++ b/app/soapbox/components/status-action-bar.tsx @@ -1,4 +1,3 @@ -import classNames from 'clsx'; import { List as ImmutableList } from 'immutable'; import React from 'react'; import { defineMessages, FormattedMessage, useIntl } from 'react-intl'; @@ -16,6 +15,7 @@ import { initReport } from 'soapbox/actions/reports'; import { deleteStatus, editStatus, toggleMuteStatus } from 'soapbox/actions/statuses'; import EmojiButtonWrapper from 'soapbox/components/emoji-button-wrapper'; import StatusActionButton from 'soapbox/components/status-action-button'; +import { HStack } from 'soapbox/components/ui'; import DropdownMenuContainer from 'soapbox/containers/dropdown-menu-container'; import { useAppDispatch, useAppSelector, useFeatures, useOwnAccount, useSettings, useSoapboxConfig } from 'soapbox/hooks'; import { isLocal } from 'soapbox/utils/accounts'; @@ -127,8 +127,6 @@ const StatusActionBar: React.FC = ({ } else { onOpenUnauthorizedModal('REPLY'); } - - e.stopPropagation(); }; const handleShareClick = () => { @@ -146,18 +144,13 @@ const StatusActionBar: React.FC = ({ } else { onOpenUnauthorizedModal('FAVOURITE'); } - - e.stopPropagation(); }; const handleBookmarkClick: React.EventHandler = (e) => { - e.stopPropagation(); dispatch(toggleBookmark(status)); }; const handleReblogClick: React.EventHandler = e => { - e.stopPropagation(); - if (me) { const modalReblog = () => dispatch(toggleReblog(status)); const boostModal = settings.get('boostModal'); @@ -172,8 +165,6 @@ const StatusActionBar: React.FC = ({ }; const handleQuoteClick: React.EventHandler = (e) => { - e.stopPropagation(); - if (me) { dispatch(quoteCompose(status)); } else { @@ -199,12 +190,10 @@ const StatusActionBar: React.FC = ({ }; const handleDeleteClick: React.EventHandler = (e) => { - e.stopPropagation(); doDeleteStatus(); }; const handleRedraftClick: React.EventHandler = (e) => { - e.stopPropagation(); doDeleteStatus(true); }; @@ -213,35 +202,29 @@ const StatusActionBar: React.FC = ({ }; const handlePinClick: React.EventHandler = (e) => { - e.stopPropagation(); dispatch(togglePin(status)); }; const handleMentionClick: React.EventHandler = (e) => { - e.stopPropagation(); dispatch(mentionCompose(status.account as Account)); }; const handleDirectClick: React.EventHandler = (e) => { - e.stopPropagation(); dispatch(directCompose(status.account as Account)); }; const handleChatClick: React.EventHandler = (e) => { - e.stopPropagation(); const account = status.account as Account; dispatch(launchChat(account.id, history)); }; const handleMuteClick: React.EventHandler = (e) => { - e.stopPropagation(); dispatch(initMuteModal(status.account as Account)); }; const handleBlockClick: React.EventHandler = (e) => { - e.stopPropagation(); - const account = status.get('account') as Account; + dispatch(openModal('CONFIRM', { icon: require('@tabler/icons/ban.svg'), heading: , @@ -257,7 +240,6 @@ const StatusActionBar: React.FC = ({ }; const handleOpen: React.EventHandler = (e) => { - e.stopPropagation(); history.push(`/@${status.getIn(['account', 'acct'])}/posts/${status.id}`); }; @@ -269,12 +251,10 @@ const StatusActionBar: React.FC = ({ }; const handleReport: React.EventHandler = (e) => { - e.stopPropagation(); dispatch(initReport(status.account as Account, status)); }; const handleConversationMuteClick: React.EventHandler = (e) => { - e.stopPropagation(); dispatch(toggleMuteStatus(status)); }; @@ -282,8 +262,6 @@ const StatusActionBar: React.FC = ({ const { uri } = status; const textarea = document.createElement('textarea'); - e.stopPropagation(); - textarea.textContent = uri; textarea.style.position = 'fixed'; @@ -300,18 +278,15 @@ const StatusActionBar: React.FC = ({ }; const onModerate: React.MouseEventHandler = (e) => { - e.stopPropagation(); const account = status.account as Account; dispatch(openModal('ACCOUNT_MODERATION', { accountId: account.id })); }; const handleDeleteStatus: React.EventHandler = (e) => { - e.stopPropagation(); dispatch(deleteStatusModal(intl, status.id)); }; const handleToggleStatusSensitivity: React.EventHandler = (e) => { - e.stopPropagation(); dispatch(toggleStatusSensitivityModal(intl, status.id, status.sensitive)); }; @@ -550,74 +525,77 @@ const StatusActionBar: React.FC = ({ const canShare = ('share' in navigator) && status.visibility === 'public'; return ( -
- + + e.stopPropagation()} + onMouseDown={e => e.stopPropagation()} + onClick={e => e.stopPropagation()} + > + - {(features.quotePosts && me) ? ( - - {reblogButton} - - ) : ( - reblogButton - )} + {(features.quotePosts && me) ? ( + + {reblogButton} + + ) : ( + reblogButton + )} - {features.emojiReacts ? ( - + {features.emojiReacts ? ( + + + + ) : ( - - ) : ( - - )} + )} - {canShare && ( - - )} + {canShare && ( + + )} - - - -
+ + + +
+
); }; diff --git a/app/soapbox/components/status-media.tsx b/app/soapbox/components/status-media.tsx index 71177cb12..f3d418d30 100644 --- a/app/soapbox/components/status-media.tsx +++ b/app/soapbox/components/status-media.tsx @@ -2,6 +2,7 @@ import React, { useState } from 'react'; import { openModal } from 'soapbox/actions/modals'; import AttachmentThumbs from 'soapbox/components/attachment-thumbs'; +import StopPropagation from 'soapbox/components/stop-propagation'; import PlaceholderCard from 'soapbox/features/placeholder/components/placeholder-card'; import Card from 'soapbox/features/status/components/card'; import Bundle from 'soapbox/features/ui/components/bundle'; @@ -173,7 +174,15 @@ const StatusMedia: React.FC = ({ ); } - return media; + if (media) { + return ( + + {media} + + ); + } else { + return null; + } }; export default StatusMedia; diff --git a/app/soapbox/components/status.tsx b/app/soapbox/components/status.tsx index 846735dc3..31af2cbaa 100644 --- a/app/soapbox/components/status.tsx +++ b/app/soapbox/components/status.tsx @@ -235,7 +235,8 @@ const Status: React.FC = (props) => { reblogElement = ( event.stopPropagation()} + onClick={e => e.stopPropagation()} + onMouseUp={e => e.stopPropagation()} className='hidden sm:flex items-center text-gray-700 dark:text-gray-600 text-xs font-medium space-x-1 hover:underline' > @@ -258,7 +259,8 @@ const Status: React.FC = (props) => {
event.stopPropagation()} + onClick={e => e.stopPropagation()} + onMouseUp={e => e.stopPropagation()} className='flex items-center text-gray-700 dark:text-gray-600 text-xs font-medium space-x-1 hover:underline' > diff --git a/app/soapbox/components/statuses/sensitive-content-overlay.tsx b/app/soapbox/components/statuses/sensitive-content-overlay.tsx index 59faef161..21411bf5c 100644 --- a/app/soapbox/components/statuses/sensitive-content-overlay.tsx +++ b/app/soapbox/components/statuses/sensitive-content-overlay.tsx @@ -5,6 +5,7 @@ import { defineMessages, useIntl } from 'react-intl'; import { useSettings, useSoapboxConfig } from 'soapbox/hooks'; import { defaultMediaVisibility } from 'soapbox/utils/status'; +import StopPropagation from '../stop-propagation'; import { Button, HStack, Text } from '../ui'; import type { Status as StatusEntity } from 'soapbox/types/entities'; @@ -38,9 +39,7 @@ const SensitiveContentOverlay = React.forwardRef(defaultMediaVisibility(status, displayMedia)); - const toggleVisibility = (event: React.MouseEvent) => { - event.stopPropagation(); - + const toggleVisibility = () => { if (onToggleVisibility) { onToggleVisibility(); } else { @@ -64,13 +63,15 @@ const SensitiveContentOverlay = React.forwardRef {visible ? ( - - - )} - - ) : null} + + {isUnderReview ? ( + <> + {links.get('support') && ( + + + + )} + + ) : null} - + +
)} diff --git a/app/soapbox/components/stop-propagation.tsx b/app/soapbox/components/stop-propagation.tsx new file mode 100644 index 000000000..614fa129c --- /dev/null +++ b/app/soapbox/components/stop-propagation.tsx @@ -0,0 +1,28 @@ +import React from 'react'; + +interface IStopPropagation { + children: React.ReactNode, +} + +/** + * Prevent mouse events from bubbling up. + * + * Why is this needed? Because `onClick`, `onMouseDown`, and `onMouseUp` are 3 separate events. + * To prevent a lot of code duplication, this component can stop all mouse events. + * Plus, placing it in the component tree makes it more readable. + */ +const StopPropagation: React.FC = ({ children }) => { + + const handler: React.MouseEventHandler = (e) => { + e.stopPropagation(); + }; + + return ( + // eslint-disable-next-line jsx-a11y/no-static-element-interactions +
+ {children} +
+ ); +}; + +export default StopPropagation; \ No newline at end of file diff --git a/app/soapbox/components/ui/button/button.tsx b/app/soapbox/components/ui/button/button.tsx index 97de8862b..1b0528efb 100644 --- a/app/soapbox/components/ui/button/button.tsx +++ b/app/soapbox/components/ui/button/button.tsx @@ -8,7 +8,7 @@ import { useButtonStyles } from './useButtonStyles'; import type { ButtonSizes, ButtonThemes } from './useButtonStyles'; -interface IButton { +interface IButton extends Pick, 'onClick' | 'onMouseUp'> { /** Whether this button expands the width of its container. */ block?: boolean, /** Elements inside the + + + ); } return ( - + + + ); }; From 0cea66dcefabdc352abd53614795437644628732 Mon Sep 17 00:00:00 2001 From: Alex Gleason Date: Sat, 19 Nov 2022 15:09:01 -0600 Subject: [PATCH 3/7] ProfileDropdown: allow clicking accounts again --- app/soapbox/features/ui/components/profile-dropdown.tsx | 7 ++++++- 1 file changed, 6 insertions(+), 1 deletion(-) diff --git a/app/soapbox/features/ui/components/profile-dropdown.tsx b/app/soapbox/features/ui/components/profile-dropdown.tsx index 0a9a3f297..4122727ab 100644 --- a/app/soapbox/features/ui/components/profile-dropdown.tsx +++ b/app/soapbox/features/ui/components/profile-dropdown.tsx @@ -58,7 +58,12 @@ const ProfileDropdown: React.FC = ({ account, children }) => { const renderAccount = (account: AccountEntity) => { return ( - +
+ {/* HACK: The component stops click events, so insert this div as something to click. */} +
+ + +
); }; From 6a8d7e4ed3ec4dc778374c84dffdd048215b64e5 Mon Sep 17 00:00:00 2001 From: Alex Gleason Date: Sat, 19 Nov 2022 15:26:07 -0600 Subject: [PATCH 4/7] StopPropagation: let it be disabled --- app/soapbox/components/stop-propagation.tsx | 9 +++++++-- 1 file changed, 7 insertions(+), 2 deletions(-) diff --git a/app/soapbox/components/stop-propagation.tsx b/app/soapbox/components/stop-propagation.tsx index 614fa129c..f8eff7b4e 100644 --- a/app/soapbox/components/stop-propagation.tsx +++ b/app/soapbox/components/stop-propagation.tsx @@ -1,7 +1,10 @@ import React from 'react'; interface IStopPropagation { + /** Children to render within the bubble. */ children: React.ReactNode, + /** Whether to prevent mouse events from bubbling. (default: `true`) */ + enabled?: boolean, } /** @@ -11,10 +14,12 @@ interface IStopPropagation { * To prevent a lot of code duplication, this component can stop all mouse events. * Plus, placing it in the component tree makes it more readable. */ -const StopPropagation: React.FC = ({ children }) => { +const StopPropagation: React.FC = ({ children, enabled = true }) => { const handler: React.MouseEventHandler = (e) => { - e.stopPropagation(); + if (enabled) { + e.stopPropagation(); + } }; return ( From 4c579689147039eeb82ec3725a7e17c022904e04 Mon Sep 17 00:00:00 2001 From: Alex Gleason Date: Sat, 19 Nov 2022 15:30:09 -0600 Subject: [PATCH 5/7] StatusContent: prevent bubbling of mentions, links, hashtags, and "Read more" button --- app/soapbox/components/status-content.tsx | 15 +++++++++++---- 1 file changed, 11 insertions(+), 4 deletions(-) diff --git a/app/soapbox/components/status-content.tsx b/app/soapbox/components/status-content.tsx index a495a9f82..2b2754176 100644 --- a/app/soapbox/components/status-content.tsx +++ b/app/soapbox/components/status-content.tsx @@ -12,6 +12,7 @@ import { isRtl } from '../rtl'; import Poll from './polls/poll'; import './status-content.css'; +import StopPropagation from './stop-propagation'; import type { Status, Mention } from 'soapbox/types/entities'; @@ -29,10 +30,12 @@ interface IReadMoreButton { /** Button to expand a truncated status (due to too much content) */ const ReadMoreButton: React.FC = ({ onClick }) => ( - + + + ); interface IStatusContent { @@ -103,6 +106,10 @@ const StatusContent: React.FC = ({ status, onClick, collapsable link.setAttribute('title', link.href); link.addEventListener('click', onLinkClick.bind(link), false); } + + // Prevent bubbling + link.addEventListener('mouseup', e => e.stopPropagation()); + link.addEventListener('mousedown', e => e.stopPropagation()); }); }; From f66b50361d24c01a47dc3efb14111dc6e3d37e6f Mon Sep 17 00:00:00 2001 From: Alex Gleason Date: Sat, 19 Nov 2022 15:33:30 -0600 Subject: [PATCH 6/7] StatusReplyMentions: prevent bubbling --- .../components/status-reply-mentions.tsx | 59 ++++++++++--------- 1 file changed, 30 insertions(+), 29 deletions(-) diff --git a/app/soapbox/components/status-reply-mentions.tsx b/app/soapbox/components/status-reply-mentions.tsx index 0964c84ae..3adde6b65 100644 --- a/app/soapbox/components/status-reply-mentions.tsx +++ b/app/soapbox/components/status-reply-mentions.tsx @@ -5,6 +5,7 @@ import { Link } from 'react-router-dom'; import { openModal } from 'soapbox/actions/modals'; import HoverRefWrapper from 'soapbox/components/hover-ref-wrapper'; import HoverStatusWrapper from 'soapbox/components/hover-status-wrapper'; +import StopPropagation from 'soapbox/components/stop-propagation'; import { useAppDispatch } from 'soapbox/hooks'; import type { Account, Status } from 'soapbox/types/entities'; @@ -18,8 +19,6 @@ const StatusReplyMentions: React.FC = ({ status, hoverable const dispatch = useAppDispatch(); const handleOpenMentionsModal: React.MouseEventHandler = (e) => { - e.stopPropagation(); - const account = status.account as Account; dispatch(openModal('MENTIONS', { @@ -50,7 +49,7 @@ const StatusReplyMentions: React.FC = ({ status, hoverable // The typical case with a reply-to and a list of mentions. const accounts = to.slice(0, 2).map(account => { const link = ( - e.stopPropagation()}>@{account.username} + @{account.username} ); if (hoverable) { @@ -73,32 +72,34 @@ const StatusReplyMentions: React.FC = ({ status, hoverable } return ( -
- , - hover: (children: React.ReactNode) => { - if (hoverable) { - return ( - - - {children} - - - ); - } else { - return children; - } - }, - }} - /> -
+ +
+ , + hover: (children: React.ReactNode) => { + if (hoverable) { + return ( + + + {children} + + + ); + } else { + return children; + } + }, + }} + /> +
+
); }; From 990e28ccc81fa4f3dddfb1cc8cf400a7c3ce152c Mon Sep 17 00:00:00 2001 From: Alex Gleason Date: Sat, 19 Nov 2022 15:55:55 -0600 Subject: [PATCH 7/7] Remove unnecessary stopPropagation() calls --- app/soapbox/features/audio/index.tsx | 1 - app/soapbox/features/video/index.tsx | 3 --- 2 files changed, 4 deletions(-) diff --git a/app/soapbox/features/audio/index.tsx b/app/soapbox/features/audio/index.tsx index ecbae5b16..0bef3c3d9 100644 --- a/app/soapbox/features/audio/index.tsx +++ b/app/soapbox/features/audio/index.tsx @@ -449,7 +449,6 @@ const Audio: React.FC = (props) => { onMouseLeave={handleMouseLeave} tabIndex={0} onKeyDown={handleKeyDown} - onClick={e => e.stopPropagation()} >