import {memo, useState} from 'react' import {type StyleProp, View, type ViewStyle} from 'react-native' import { type AppBskyFeedDefs, type AppBskyFeedPost, type AppBskyFeedThreadgate, type RichText as RichTextAPI, } from '@atproto/api' import {msg, plural} from '@lingui/macro' import {useLingui} from '@lingui/react' import {CountWheel} from '#/lib/custom-animations/CountWheel' import {AnimatedLikeIcon} from '#/lib/custom-animations/LikeIcon' import {useHaptics} from '#/lib/haptics' import {useOpenComposer} from '#/lib/hooks/useOpenComposer' import {type Shadow} from '#/state/cache/types' import {useFeedFeedbackContext} from '#/state/feed-feedback' import { usePostLikeMutationQueue, usePostRepostMutationQueue, } from '#/state/queries/post' import {useRequireAuth} from '#/state/session' import { ProgressGuideAction, useProgressGuideControls, } from '#/state/shell/progress-guide' import {formatCount} from '#/view/com/util/numeric/format' import * as Toast from '#/view/com/util/Toast' import {atoms as a, useBreakpoints} from '#/alf' import {Bubble_Stroke2_Corner2_Rounded as Bubble} from '#/components/icons/Bubble' import { PostControlButton, PostControlButtonIcon, PostControlButtonText, } from './PostControlButton' import {PostMenuButton} from './PostMenu' import {RepostButton} from './RepostButton' import {ShareMenuButton} from './ShareMenu' let PostControls = ({ big, post, record, richText, feedContext, reqId, style, onPressReply, onPostReply, logContext, threadgateRecord, onShowLess, viaRepost, }: { big?: boolean post: Shadow record: AppBskyFeedPost.Record richText: RichTextAPI feedContext?: string | undefined reqId?: string | undefined style?: StyleProp onPressReply: () => void onPostReply?: (postUri: string | undefined) => void logContext: 'FeedItem' | 'PostThreadItem' | 'Post' | 'ImmersiveVideo' threadgateRecord?: AppBskyFeedThreadgate.Record onShowLess?: (interaction: AppBskyFeedDefs.Interaction) => void viaRepost?: {uri: string; cid: string} }): React.ReactNode => { const {_, i18n} = useLingui() const {gtMobile} = useBreakpoints() const {openComposer} = useOpenComposer() const [queueLike, queueUnlike] = usePostLikeMutationQueue( post, viaRepost, logContext, ) const [queueRepost, queueUnrepost] = usePostRepostMutationQueue( post, viaRepost, logContext, ) const requireAuth = useRequireAuth() const {sendInteraction} = useFeedFeedbackContext() const {captureAction} = useProgressGuideControls() const playHaptic = useHaptics() const isBlocked = Boolean( post.author.viewer?.blocking || post.author.viewer?.blockedBy || post.author.viewer?.blockingByList, ) const replyDisabled = post.viewer?.replyDisabled const [hasLikeIconBeenToggled, setHasLikeIconBeenToggled] = useState(false) const onPressToggleLike = async () => { if (isBlocked) { Toast.show( _(msg`Cannot interact with a blocked user`), 'exclamation-circle', ) return } try { setHasLikeIconBeenToggled(true) if (!post.viewer?.like) { playHaptic('Light') sendInteraction({ item: post.uri, event: 'app.bsky.feed.defs#interactionLike', feedContext, reqId, }) captureAction(ProgressGuideAction.Like) await queueLike() } else { await queueUnlike() } } catch (e: any) { if (e?.name !== 'AbortError') { throw e } } } const onRepost = async () => { if (isBlocked) { Toast.show( _(msg`Cannot interact with a blocked user`), 'exclamation-circle', ) return } try { if (!post.viewer?.repost) { sendInteraction({ item: post.uri, event: 'app.bsky.feed.defs#interactionRepost', feedContext, reqId, }) await queueRepost() } else { await queueUnrepost() } } catch (e: any) { if (e?.name !== 'AbortError') { throw e } } } const onQuote = () => { if (isBlocked) { Toast.show( _(msg`Cannot interact with a blocked user`), 'exclamation-circle', ) return } sendInteraction({ item: post.uri, event: 'app.bsky.feed.defs#interactionQuote', feedContext, reqId, }) openComposer({ quote: post, onPost: onPostReply, }) } const onShare = () => { sendInteraction({ item: post.uri, event: 'app.bsky.feed.defs#interactionShare', feedContext, reqId, }) } return ( requireAuth(() => onPressReply()) : undefined } label={_( msg({ message: `Reply (${plural(post.replyCount || 0, { one: '# reply', other: '# replies', })})`, comment: 'Accessibility label for the reply button, verb form followed by number of replies and noun form', }), )} big={big}> {typeof post.replyCount !== 'undefined' && post.replyCount > 0 && ( {formatCount(i18n, post.replyCount)} )} requireAuth(() => onPressToggleLike())} label={ post.viewer?.like ? _( msg({ message: `Unlike (${plural(post.likeCount || 0, { one: '# like', other: '# likes', })})`, comment: 'Accessibility label for the like button when the post has been liked, verb followed by number of likes and noun', }), ) : _( msg({ message: `Like (${plural(post.likeCount || 0, { one: '# like', other: '# likes', })})`, comment: 'Accessibility label for the like button when the post has not been liked, verb form followed by number of likes and noun form', }), ) }> ) } PostControls = memo(PostControls) export {PostControls}