import React from 'react' import { StyleProp, StyleSheet, TouchableOpacity, View, ViewStyle, } from 'react-native' import { AppBskyEmbedExternal, AppBskyEmbedImages, AppBskyEmbedRecord, AppBskyEmbedRecordWithMedia, AppBskyEmbedVideo, AppBskyFeedDefs, AppBskyFeedPost, ModerationDecision, RichText as RichTextAPI, } from '@atproto/api' import {AtUri} from '@atproto/api' import {FontAwesomeIcon} from '@fortawesome/react-native-fontawesome' import {msg, Trans} from '@lingui/macro' import {useLingui} from '@lingui/react' import {useQueryClient} from '@tanstack/react-query' import {HITSLOP_20} from '#/lib/constants' import {moderatePost_wrapped} from '#/lib/moderatePost_wrapped' import {s} from '#/lib/styles' import {useModerationOpts} from '#/state/preferences/moderation-opts' import {useSession} from '#/state/session' import {usePalette} from 'lib/hooks/usePalette' import {InfoCircleIcon} from 'lib/icons' import {makeProfileLink} from 'lib/routes/links' import {precacheProfile} from 'state/queries/profile' import {ComposerOptsQuote} from 'state/shell/composer' import {atoms as a} from '#/alf' import {RichText} from '#/components/RichText' import {ContentHider} from '../../../../components/moderation/ContentHider' import {PostAlerts} from '../../../../components/moderation/PostAlerts' import {Link} from '../Link' import {PostMeta} from '../PostMeta' import {Text} from '../text/Text' import {PostEmbeds} from '.' export function MaybeQuoteEmbed({ embed, onOpen, style, allowNestedQuotes, }: { embed: AppBskyEmbedRecord.View onOpen?: () => void style?: StyleProp allowNestedQuotes?: boolean }) { const pal = usePalette('default') const {currentAccount} = useSession() if ( AppBskyEmbedRecord.isViewRecord(embed.record) && AppBskyFeedPost.isRecord(embed.record.value) && AppBskyFeedPost.validateRecord(embed.record.value).success ) { return ( ) } else if (AppBskyEmbedRecord.isViewBlocked(embed.record)) { return ( Blocked ) } else if (AppBskyEmbedRecord.isViewNotFound(embed.record)) { return ( Deleted ) } else if (AppBskyEmbedRecord.isViewDetached(embed.record)) { const isViewerOwner = currentAccount?.did ? embed.record.uri.includes(currentAccount.did) : false return ( {isViewerOwner ? ( Removed by you ) : ( Removed by author )} ) } return null } function QuoteEmbedModerated({ viewRecord, postRecord, onOpen, style, allowNestedQuotes, }: { viewRecord: AppBskyEmbedRecord.ViewRecord postRecord: AppBskyFeedPost.Record onOpen?: () => void style?: StyleProp allowNestedQuotes?: boolean }) { const moderationOpts = useModerationOpts() const moderation = React.useMemo(() => { return moderationOpts ? moderatePost_wrapped(viewRecordToPostView(viewRecord), moderationOpts) : undefined }, [viewRecord, moderationOpts]) const quote = { author: viewRecord.author, cid: viewRecord.cid, uri: viewRecord.uri, indexedAt: viewRecord.indexedAt, text: postRecord.text, facets: postRecord.facets, embeds: viewRecord.embeds, } return ( ) } export function QuoteEmbed({ quote, moderation, onOpen, style, allowNestedQuotes, }: { quote: ComposerOptsQuote moderation?: ModerationDecision onOpen?: () => void style?: StyleProp allowNestedQuotes?: boolean }) { const queryClient = useQueryClient() const pal = usePalette('default') const itemUrip = new AtUri(quote.uri) const itemHref = makeProfileLink(quote.author, 'post', itemUrip.rkey) const itemTitle = `Post by ${quote.author.handle}` const richText = React.useMemo( () => quote.text.trim() ? new RichTextAPI({text: quote.text, facets: quote.facets}) : undefined, [quote.text, quote.facets], ) const embed = React.useMemo(() => { const e = quote.embeds?.[0] if (allowNestedQuotes) { return e } else { if ( AppBskyEmbedImages.isView(e) || AppBskyEmbedExternal.isView(e) || AppBskyEmbedVideo.isView(e) ) { return e } else if ( AppBskyEmbedRecordWithMedia.isView(e) && (AppBskyEmbedImages.isView(e.media) || AppBskyEmbedExternal.isView(e.media) || AppBskyEmbedVideo.isView(e.media)) ) { return e.media } } }, [quote.embeds, allowNestedQuotes]) const onBeforePress = React.useCallback(() => { precacheProfile(queryClient, quote.author) onOpen?.() }, [queryClient, quote.author, onOpen]) return ( {moderation ? ( ) : null} {richText ? ( ) : null} {embed && } ) } export function QuoteX({onRemove}: {onRemove: () => void}) { const {_} = useLingui() return ( ) } function viewRecordToPostView( viewRecord: AppBskyEmbedRecord.ViewRecord, ): AppBskyFeedDefs.PostView { const {value, embeds, ...rest} = viewRecord return { ...rest, $type: 'app.bsky.feed.defs#postView', record: value, embed: embeds?.[0], } } const styles = StyleSheet.create({ container: { borderRadius: 8, marginTop: 8, paddingVertical: 12, paddingHorizontal: 12, borderWidth: StyleSheet.hairlineWidth, }, errorContainer: { flexDirection: 'row', alignItems: 'center', gap: 4, borderRadius: 8, marginTop: 8, paddingVertical: 14, paddingHorizontal: 14, borderWidth: StyleSheet.hairlineWidth, }, alert: { marginBottom: 6, }, })