mirror of
				https://github.com/tooot-app/app
				synced 2025-06-05 22:19:13 +02:00 
			
		
		
		
	Expandable content done
This commit is contained in:
		
							
								
								
									
										1
									
								
								App.tsx
									
									
									
									
									
								
							
							
						
						
									
										1
									
								
								App.tsx
									
									
									
									
									
								
							| @@ -33,7 +33,6 @@ const App: React.FC = () => { | |||||||
|           <PersistGate persistor={persistor}> |           <PersistGate persistor={persistor}> | ||||||
|             {bootstrapped => { |             {bootstrapped => { | ||||||
|               if (bootstrapped) { |               if (bootstrapped) { | ||||||
|                 console.log('Bootstrapped!') |  | ||||||
|                 require('src/i18n/i18n') |                 require('src/i18n/i18n') | ||||||
|                 return ( |                 return ( | ||||||
|                   <ThemeManager> |                   <ThemeManager> | ||||||
|   | |||||||
| @@ -18,6 +18,7 @@ | |||||||
|     "expo-auth-session": "~2.0.0", |     "expo-auth-session": "~2.0.0", | ||||||
|     "expo-av": "~8.6.0", |     "expo-av": "~8.6.0", | ||||||
|     "expo-image-picker": "~9.1.1", |     "expo-image-picker": "~9.1.1", | ||||||
|  |     "expo-linear-gradient": "~8.3.0", | ||||||
|     "expo-localization": "^9.0.0", |     "expo-localization": "^9.0.0", | ||||||
|     "expo-secure-store": "~9.2.0", |     "expo-secure-store": "~9.2.0", | ||||||
|     "expo-splash-screen": "~0.6.1", |     "expo-splash-screen": "~0.6.1", | ||||||
|   | |||||||
							
								
								
									
										1
									
								
								src/@types/untyped.d.ts
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								src/@types/untyped.d.ts
									
									
									
									
										vendored
									
									
								
							| @@ -1 +1,2 @@ | |||||||
| declare module 'react-native-toast-message' | declare module 'react-native-toast-message' | ||||||
|  | declare module 'react-native-htmlview' | ||||||
|   | |||||||
| @@ -31,7 +31,7 @@ const MenuContainer: React.FC<Props> = ({ ...props }) => { | |||||||
| const styles = StyleSheet.create({ | const styles = StyleSheet.create({ | ||||||
|   base: { |   base: { | ||||||
|     borderTopWidth: 1, |     borderTopWidth: 1, | ||||||
|     marginBottom: StyleConstants.Spacing.Global.PagePadding |     marginBottom: StyleConstants.Spacing.L | ||||||
|   } |   } | ||||||
| }) | }) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| import React, { useCallback } from 'react' | import React, { useCallback, useState } from 'react' | ||||||
| import { StyleSheet, Text, View } from 'react-native' | import { Pressable, StyleSheet, Text } from 'react-native' | ||||||
| import HTMLView from 'react-native-htmlview' | import HTMLView from 'react-native-htmlview' | ||||||
| import { useNavigation } from '@react-navigation/native' | import { useNavigation } from '@react-navigation/native' | ||||||
|  |  | ||||||
| @@ -7,12 +7,14 @@ import Emojis from 'src/components/Timelines/Timeline/Shared/Emojis' | |||||||
| import { useTheme } from 'src/utils/styles/ThemeManager' | import { useTheme } from 'src/utils/styles/ThemeManager' | ||||||
| import { Feather } from '@expo/vector-icons' | import { Feather } from '@expo/vector-icons' | ||||||
| import { StyleConstants } from 'src/utils/styles/constants' | import { StyleConstants } from 'src/utils/styles/constants' | ||||||
|  | import { LinearGradient } from 'expo-linear-gradient' | ||||||
|  |  | ||||||
| // Prevent going to the same hashtag multiple times | // Prevent going to the same hashtag multiple times | ||||||
| const renderNode = ({ | const renderNode = ({ | ||||||
|   theme, |   theme, | ||||||
|   node, |   node, | ||||||
|   index, |   index, | ||||||
|  |   size, | ||||||
|   navigation, |   navigation, | ||||||
|   mentions, |   mentions, | ||||||
|   showFullLink |   showFullLink | ||||||
| @@ -20,6 +22,7 @@ const renderNode = ({ | |||||||
|   theme: any |   theme: any | ||||||
|   node: any |   node: any | ||||||
|   index: number |   index: number | ||||||
|  |   size: number | ||||||
|   navigation: any |   navigation: any | ||||||
|   mentions?: Mastodon.Mention[] |   mentions?: Mastodon.Mention[] | ||||||
|   showFullLink: boolean |   showFullLink: boolean | ||||||
| @@ -32,7 +35,7 @@ const renderNode = ({ | |||||||
|         return ( |         return ( | ||||||
|           <Text |           <Text | ||||||
|             key={index} |             key={index} | ||||||
|             style={{ color: theme.link }} |             style={{ color: theme.link, fontSize: size }} | ||||||
|             onPress={() => { |             onPress={() => { | ||||||
|               const tag = href.split(new RegExp(/\/tag\/(.*)|\/tags\/(.*)/)) |               const tag = href.split(new RegExp(/\/tag\/(.*)|\/tags\/(.*)/)) | ||||||
|               navigation.push('Screen-Shared-Hashtag', { |               navigation.push('Screen-Shared-Hashtag', { | ||||||
| @@ -48,7 +51,7 @@ const renderNode = ({ | |||||||
|         return ( |         return ( | ||||||
|           <Text |           <Text | ||||||
|             key={index} |             key={index} | ||||||
|             style={{ color: theme.link }} |             style={{ color: theme.link, fontSize: size }} | ||||||
|             onPress={() => { |             onPress={() => { | ||||||
|               const username = href.split(new RegExp(/@(.*)/)) |               const username = href.split(new RegExp(/@(.*)/)) | ||||||
|               const usernameIndex = mentions.findIndex( |               const usernameIndex = mentions.findIndex( | ||||||
| @@ -69,7 +72,7 @@ const renderNode = ({ | |||||||
|       return ( |       return ( | ||||||
|         <Text |         <Text | ||||||
|           key={index} |           key={index} | ||||||
|           style={{ color: theme.link }} |           style={{ color: theme.link, fontSize: size }} | ||||||
|           onPress={() => { |           onPress={() => { | ||||||
|             navigation.navigate('Screen-Shared-Webview', { |             navigation.navigate('Screen-Shared-Webview', { | ||||||
|               uri: href, |               uri: href, | ||||||
| @@ -77,11 +80,7 @@ const renderNode = ({ | |||||||
|             }) |             }) | ||||||
|           }} |           }} | ||||||
|         > |         > | ||||||
|           <Feather |           <Feather name='external-link' size={size} color={theme.link} />{' '} | ||||||
|             name='external-link' |  | ||||||
|             size={StyleConstants.Font.Size.M} |  | ||||||
|             color={theme.link} |  | ||||||
|           />{' '} |  | ||||||
|           {showFullLink ? href : domain[1]} |           {showFullLink ? href : domain[1]} | ||||||
|         </Text> |         </Text> | ||||||
|       ) |       ) | ||||||
| @@ -111,7 +110,15 @@ const ParseContent: React.FC<Props> = ({ | |||||||
|  |  | ||||||
|   const renderNodeCallback = useCallback( |   const renderNodeCallback = useCallback( | ||||||
|     (node, index) => |     (node, index) => | ||||||
|       renderNode({ theme, node, index, navigation, mentions, showFullLink }), |       renderNode({ | ||||||
|  |         theme, | ||||||
|  |         node, | ||||||
|  |         index, | ||||||
|  |         size, | ||||||
|  |         navigation, | ||||||
|  |         mentions, | ||||||
|  |         showFullLink | ||||||
|  |       }), | ||||||
|     [] |     [] | ||||||
|   ) |   ) | ||||||
|   const textComponent = useCallback( |   const textComponent = useCallback( | ||||||
| @@ -124,10 +131,62 @@ const ParseContent: React.FC<Props> = ({ | |||||||
|     [] |     [] | ||||||
|   ) |   ) | ||||||
|   const rootComponent = useCallback(({ children }) => { |   const rootComponent = useCallback(({ children }) => { | ||||||
|  |     const { theme } = useTheme() | ||||||
|  |     const [textLoaded, setTextLoaded] = useState(false) | ||||||
|  |     const [totalLines, setTotalLines] = useState<number | undefined>() | ||||||
|  |     const [lineHeight, setLineHeight] = useState<number | undefined>() | ||||||
|  |     const [shownLines, setShownLines] = useState(numberOfLines) | ||||||
|  |  | ||||||
|     return ( |     return ( | ||||||
|       <Text numberOfLines={numberOfLines} style={styles.root}> |       <> | ||||||
|  |         <Text | ||||||
|  |           numberOfLines={ | ||||||
|  |             totalLines && totalLines > numberOfLines ? shownLines : totalLines | ||||||
|  |           } | ||||||
|  |           style={styles.root} | ||||||
|  |           onTextLayout={({ nativeEvent }) => { | ||||||
|  |             if (!textLoaded) { | ||||||
|  |               setTextLoaded(true) | ||||||
|  |               setTotalLines(nativeEvent.lines.length) | ||||||
|  |               setLineHeight(nativeEvent.lines[0].height) | ||||||
|  |             } | ||||||
|  |           }} | ||||||
|  |         > | ||||||
|           {children} |           {children} | ||||||
|         </Text> |         </Text> | ||||||
|  |         {totalLines && lineHeight && totalLines > shownLines && ( | ||||||
|  |           <Pressable | ||||||
|  |             onPress={() => { | ||||||
|  |               setShownLines(totalLines) | ||||||
|  |             }} | ||||||
|  |             style={{ | ||||||
|  |               marginTop: -lineHeight | ||||||
|  |             }} | ||||||
|  |           > | ||||||
|  |             <LinearGradient | ||||||
|  |               colors={[ | ||||||
|  |                 theme.backgroundGradientStart, | ||||||
|  |                 theme.backgroundGradientEnd | ||||||
|  |               ]} | ||||||
|  |               locations={[0, lineHeight / (StyleConstants.Font.Size.S * 5)]} | ||||||
|  |               style={{ | ||||||
|  |                 paddingTop: StyleConstants.Font.Size.S * 2, | ||||||
|  |                 paddingBottom: StyleConstants.Font.Size.S | ||||||
|  |               }} | ||||||
|  |             > | ||||||
|  |               <Text | ||||||
|  |                 style={{ | ||||||
|  |                   textAlign: 'center', | ||||||
|  |                   fontSize: StyleConstants.Font.Size.S, | ||||||
|  |                   color: theme.primary | ||||||
|  |                 }} | ||||||
|  |               > | ||||||
|  |                 展开全文 | ||||||
|  |               </Text> | ||||||
|  |             </LinearGradient> | ||||||
|  |           </Pressable> | ||||||
|  |         )} | ||||||
|  |       </> | ||||||
|     ) |     ) | ||||||
|   }, []) |   }, []) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -12,13 +12,15 @@ export interface Props { | |||||||
|   emojis: Mastodon.Emoji[] |   emojis: Mastodon.Emoji[] | ||||||
|   mentions: Mastodon.Mention[] |   mentions: Mastodon.Mention[] | ||||||
|   spoiler_text?: string |   spoiler_text?: string | ||||||
|  |   numberOfLines?: number | ||||||
| } | } | ||||||
|  |  | ||||||
| const Content: React.FC<Props> = ({ | const Content: React.FC<Props> = ({ | ||||||
|   content, |   content, | ||||||
|   emojis, |   emojis, | ||||||
|   mentions, |   mentions, | ||||||
|   spoiler_text |   spoiler_text, | ||||||
|  |   numberOfLines | ||||||
| }) => { | }) => { | ||||||
|   const { theme } = useTheme() |   const { theme } = useTheme() | ||||||
|   const [spoilerCollapsed, setSpoilerCollapsed] = useState(true) |   const [spoilerCollapsed, setSpoilerCollapsed] = useState(true) | ||||||
| @@ -28,11 +30,13 @@ const Content: React.FC<Props> = ({ | |||||||
|       {content && |       {content && | ||||||
|         (spoiler_text ? ( |         (spoiler_text ? ( | ||||||
|           <> |           <> | ||||||
|             <Text> |             <Text style={{ fontSize: StyleConstants.Font.Size.M }}> | ||||||
|               {spoiler_text}{' '} |               {spoiler_text}{' '} | ||||||
|               <Text |               <Text | ||||||
|                 onPress={() => setSpoilerCollapsed(!spoilerCollapsed)} |                 onPress={() => setSpoilerCollapsed(!spoilerCollapsed)} | ||||||
|                 style={{ color: theme.link }} |                 style={{ | ||||||
|  |                   color: theme.link | ||||||
|  |                 }} | ||||||
|               > |               > | ||||||
|                 {spoilerCollapsed ? '点击展开' : '点击收起'} |                 {spoilerCollapsed ? '点击展开' : '点击收起'} | ||||||
|               </Text> |               </Text> | ||||||
| @@ -43,6 +47,7 @@ const Content: React.FC<Props> = ({ | |||||||
|                 size={StyleConstants.Font.Size.M} |                 size={StyleConstants.Font.Size.M} | ||||||
|                 emojis={emojis} |                 emojis={emojis} | ||||||
|                 mentions={mentions} |                 mentions={mentions} | ||||||
|  |                 {...(numberOfLines && { numberOfLines: numberOfLines })} | ||||||
|               /> |               /> | ||||||
|             </Collapsible> |             </Collapsible> | ||||||
|           </> |           </> | ||||||
| @@ -52,6 +57,7 @@ const Content: React.FC<Props> = ({ | |||||||
|             size={StyleConstants.Font.Size.M} |             size={StyleConstants.Font.Size.M} | ||||||
|             emojis={emojis} |             emojis={emojis} | ||||||
|             mentions={mentions} |             mentions={mentions} | ||||||
|  |             {...(numberOfLines && { numberOfLines: numberOfLines })} | ||||||
|           /> |           /> | ||||||
|         ))} |         ))} | ||||||
|     </> |     </> | ||||||
|   | |||||||
| @@ -5,6 +5,8 @@ export type ColorDefinitions = | |||||||
|   | 'secondary' |   | 'secondary' | ||||||
|   | 'disabled' |   | 'disabled' | ||||||
|   | 'background' |   | 'background' | ||||||
|  |   | 'backgroundGradientStart' | ||||||
|  |   | 'backgroundGradientEnd' | ||||||
|   | 'link' |   | 'link' | ||||||
|   | 'border' |   | 'border' | ||||||
|   | 'separator' |   | 'separator' | ||||||
| @@ -34,6 +36,14 @@ const themeColors: { | |||||||
|     light: 'rgb(255, 255, 255)', |     light: 'rgb(255, 255, 255)', | ||||||
|     dark: 'rgb(0, 0, 0)' |     dark: 'rgb(0, 0, 0)' | ||||||
|   }, |   }, | ||||||
|  |   backgroundGradientStart: { | ||||||
|  |     light: 'rgba(255, 255, 255, 0.5)', | ||||||
|  |     dark: 'rgba(0, 0, 0, 0.5)' | ||||||
|  |   }, | ||||||
|  |   backgroundGradientEnd: { | ||||||
|  |     light: 'rgba(255, 255, 255, 1)', | ||||||
|  |     dark: 'rgba(0, 0, 0, 1)' | ||||||
|  |   }, | ||||||
|   link: { |   link: { | ||||||
|     light: 'rgb(0, 122, 255)', |     light: 'rgb(0, 122, 255)', | ||||||
|     dark: 'rgb(10, 132, 255)' |     dark: 'rgb(10, 132, 255)' | ||||||
|   | |||||||
| @@ -1481,11 +1481,6 @@ | |||||||
|   resolved "https://registry.yarnpkg.com/@types/stack-utils/-/stack-utils-1.0.1.tgz#0a851d3bd96498fa25c33ab7278ed3bd65f06c3e" |   resolved "https://registry.yarnpkg.com/@types/stack-utils/-/stack-utils-1.0.1.tgz#0a851d3bd96498fa25c33ab7278ed3bd65f06c3e" | ||||||
|   integrity sha512-l42BggppR6zLmpfU6fq9HEa2oGPEI8yrSPL3GITjfRInppYFahObbIQOQK3UGxEnyQpltZLaPe75046NOZQikw== |   integrity sha512-l42BggppR6zLmpfU6fq9HEa2oGPEI8yrSPL3GITjfRInppYFahObbIQOQK3UGxEnyQpltZLaPe75046NOZQikw== | ||||||
|  |  | ||||||
| "@types/webpack-env@^1.16.0": |  | ||||||
|   version "1.16.0" |  | ||||||
|   resolved "https://registry.yarnpkg.com/@types/webpack-env/-/webpack-env-1.16.0.tgz#8c0a9435dfa7b3b1be76562f3070efb3f92637b4" |  | ||||||
|   integrity sha512-Fx+NpfOO0CpeYX2g9bkvX8O5qh9wrU1sOF4g8sft4Mu7z+qfe387YlyY8w8daDyDsKY5vUxM0yxkAYnbkRbZEw== |  | ||||||
|  |  | ||||||
| "@types/yargs-parser@*": | "@types/yargs-parser@*": | ||||||
|   version "15.0.0" |   version "15.0.0" | ||||||
|   resolved "https://registry.yarnpkg.com/@types/yargs-parser/-/yargs-parser-15.0.0.tgz#cb3f9f741869e20cce330ffbeb9271590483882d" |   resolved "https://registry.yarnpkg.com/@types/yargs-parser/-/yargs-parser-15.0.0.tgz#cb3f9f741869e20cce330ffbeb9271590483882d" | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user