import Icon from '@components/Icon' import openLink from '@components/openLink' import ParseEmojis from '@components/Parse/Emojis' import { useNavigation } from '@react-navigation/native' import { StyleConstants } from '@utils/styles/constants' import { useTheme } from '@utils/styles/ThemeManager' import { LinearGradient } from 'expo-linear-gradient' import React, { useCallback, useMemo, useState } from 'react' import { Pressable, Text, View } from 'react-native' import HTMLView from 'react-native-htmlview' import Animated, { measure, useAnimatedRef, useAnimatedStyle, useDerivedValue, useSharedValue, withSpring, withTiming } from 'react-native-reanimated' // Prevent going to the same hashtag multiple times const renderNode = ({ theme, node, index, size, navigation, mentions, tags, showFullLink, disableDetails }: { theme: any node: any index: number size: 'M' | 'L' navigation: any mentions?: Mastodon.Mention[] tags?: Mastodon.Tag[] showFullLink: boolean disableDetails: boolean }) => { if (node.name == 'a') { const classes = node.attribs.class const href = node.attribs.href if (classes) { if (classes.includes('hashtag')) { return ( { const tag = href.split(new RegExp(/\/tag\/(.*)|\/tags\/(.*)/)) !disableDetails && navigation.push('Screen-Shared-Hashtag', { hashtag: tag[1] || tag[2] }) }} > {node.children[0].data} {node.children[1]?.children[0].data} ) } else if (classes.includes('mention') && mentions) { const accountIndex = mentions.findIndex(mention => mention.url === href) return ( { accountIndex !== -1 && !disableDetails && navigation.push('Screen-Shared-Account', { account: mentions[accountIndex] }) }} > {node.children[0].data} {node.children[1]?.children[0].data} ) } } else { const domain = href.split(new RegExp(/:\/\/(.[^\/]+)/)) // Need example here const content = node.children && node.children[0] && node.children[0].data const shouldBeTag = tags && tags.filter(tag => `#${tag.name}` === content).length > 0 return ( !disableDetails && !shouldBeTag ? await openLink(href) : navigation.push('Screen-Shared-Hashtag', { hashtag: content.substring(1) }) } > {!shouldBeTag ? ( ) : null} {content || (showFullLink ? href : domain[1])} ) } } else { if (node.name === 'p') { if (!node.children.length) { return // bug when the tag is empty } } } } export interface Props { content: string size?: 'M' | 'L' emojis?: Mastodon.Emoji[] mentions?: Mastodon.Mention[] tags?: Mastodon.Tag[] showFullLink?: boolean numberOfLines?: number expandHint?: string disableDetails?: boolean } const ParseHTML: React.FC = ({ content, size = 'M', emojis, mentions, tags, showFullLink = false, numberOfLines = 10, expandHint = '全文', disableDetails = false }) => { const navigation = useNavigation() const { theme } = useTheme() const renderNodeCallback = useCallback( (node, index) => renderNode({ theme, node, index, size, navigation, mentions, tags, showFullLink, disableDetails }), [] ) const textComponent = useCallback(({ children }) => { if (children) { return ( ) } else { return null } }, []) const rootComponent = useCallback( ({ children }) => { const lineHeight = StyleConstants.Font.LineHeight[size] const [lines, setLines] = useState(undefined) const [heightOriginal, setHeightOriginal] = useState() const [heightTruncated, setHeightTruncated] = useState() const [expandAllow, setExpandAllow] = useState(false) const [expanded, setExpanded] = useState(false) const viewHeight = useDerivedValue(() => { if (expandAllow) { if (expanded) { return heightOriginal as number } else { return heightTruncated as number } } else { return heightOriginal as number } }, [heightOriginal, heightTruncated, expandAllow, expanded]) const ViewHeight = useAnimatedStyle(() => { return { height: expandAllow ? expanded ? withTiming(viewHeight.value) : withTiming(viewHeight.value) : viewHeight.value } }) const onLayout = useCallback( ({ nativeEvent }) => { if (!heightOriginal) { setHeightOriginal(nativeEvent.layout.height) setLines(numberOfLines === 0 ? 1 : numberOfLines) } else { if (!heightTruncated) { setHeightTruncated(nativeEvent.layout.height) setLines(undefined) } else { if (heightOriginal > heightTruncated) { setExpandAllow(true) } } } }, [heightOriginal, heightTruncated] ) return ( {expandAllow ? ( setExpanded(!expanded)} style={{ marginTop: expanded ? 0 : -lineHeight * 1.25 }} > {`${expanded ? '折叠' : '展开'}${expandHint}`} ) : null} ) }, [theme] ) return ( ) } export default ParseHTML