import {useMemo} from 'react' import {type StyleProp, type TextStyle} from 'react-native' import {AppBskyRichtextFacet, RichText as RichTextAPI} from '@atproto/api' import {toShortUrl} from '#/lib/strings/url-helpers' import {atoms as a, flatten, type TextStyleProp} from '#/alf' import {isOnlyEmoji} from '#/alf/typography' import {InlineLinkText, type LinkProps} from '#/components/Link' import {ProfileHoverCard} from '#/components/ProfileHoverCard' import {RichTextTag} from '#/components/RichTextTag' import {Text, type TextProps} from '#/components/Typography' const WORD_WRAP = {wordWrap: 1} // lifted from facet detection in `RichText` impl, _without_ `gm` flags const URL_REGEX = /(^|\s|\()(?!javascript:)([a-z][a-z0-9+.-]*:\/\/[\S]+|(?:[a-z0-9]+\.)+[a-z0-9]+(:[0-9]+)?[\S]*|[a-z][a-z0-9+.-]*:[^\s()]+)/i export type RichTextProps = TextStyleProp & Pick & { value: RichTextAPI | string testID?: string numberOfLines?: number disableLinks?: boolean enableTags?: boolean authorHandle?: string onLinkPress?: LinkProps['onPress'] interactiveStyle?: StyleProp emojiMultiplier?: number shouldProxyLinks?: boolean /** * DANGEROUS: Disable facet lexicon validation * * `detectFacetsWithoutResolution()` generates technically invalid facets, * with a handle in place of the DID. This means that RichText that uses it * won't be able to render links. * * Use with care - only use if you're rendering facets you're generating yourself. */ disableMentionFacetValidation?: true } export function RichText({ testID, value, style, numberOfLines, disableLinks, selectable, enableTags = false, authorHandle, onLinkPress, interactiveStyle, emojiMultiplier = 1.85, onLayout, onTextLayout, shouldProxyLinks, disableMentionFacetValidation, }: RichTextProps) { const richText = useMemo(() => { if (value instanceof RichTextAPI) { return value } else { const rt = new RichTextAPI({text: value}) rt.detectFacetsWithoutResolution() return rt } }, [value]) const plainStyles = [a.leading_snug, style] const interactiveStyles = [plainStyles, interactiveStyle] const {text, facets} = richText if (!facets?.length) { if (isOnlyEmoji(text)) { const flattenedStyle = flatten(style) ?? {} const fontSize = (flattenedStyle.fontSize ?? a.text_sm.fontSize) * emojiMultiplier return ( {text} ) } return ( {text} ) } const els = [] let key = 0 // N.B. must access segments via `richText.segments`, not via destructuring for (const segment of richText.segments()) { const link = segment.link const mention = segment.mention const tag = segment.tag if ( mention && (disableMentionFacetValidation || AppBskyRichtextFacet.validateMention(mention).success) && !disableLinks ) { els.push( {segment.text} , ) } else if (link && AppBskyRichtextFacet.validateLink(link).success) { const isValidLink = URL_REGEX.test(link.uri) if (!isValidLink || disableLinks) { els.push(toShortUrl(segment.text)) } else { els.push( {toShortUrl(segment.text)} , ) } } else if ( !disableLinks && enableTags && tag && AppBskyRichtextFacet.validateTag(tag).success ) { els.push( , ) } else { els.push(segment.text) } key++ } return ( {els} ) }