diff --git a/src/components/ParseContent.tsx b/src/components/ParseContent.tsx index 3acc50ac..f5b8a1cb 100644 --- a/src/components/ParseContent.tsx +++ b/src/components/ParseContent.tsx @@ -1,5 +1,5 @@ import React, { useCallback, useState } from 'react' -import { Pressable, StyleSheet, Text } from 'react-native' +import { Pressable, Text } from 'react-native' import HTMLView from 'react-native-htmlview' import { useNavigation } from '@react-navigation/native' @@ -22,7 +22,7 @@ const renderNode = ({ theme: any node: any index: number - size: number + size: 'M' | 'L' navigation: any mentions?: Mastodon.Mention[] showFullLink: boolean @@ -35,7 +35,10 @@ const renderNode = ({ return ( { const tag = href.split(new RegExp(/\/tag\/(.*)|\/tags\/(.*)/)) navigation.push('Screen-Shared-Hashtag', { @@ -51,7 +54,10 @@ const renderNode = ({ return ( { const username = href.split(new RegExp(/@(.*)/)) const usernameIndex = mentions.findIndex( @@ -72,7 +78,10 @@ const renderNode = ({ return ( { navigation.navigate('Screen-Shared-Webview', { uri: href, @@ -80,7 +89,11 @@ const renderNode = ({ }) }} > - {' '} + {' '} {showFullLink ? href : domain[1]} ) @@ -90,7 +103,7 @@ const renderNode = ({ export interface Props { content: string - size: number + size: 'M' | 'L' emojis?: Mastodon.Emoji[] mentions?: Mastodon.Mention[] showFullLink?: boolean @@ -124,7 +137,11 @@ const ParseContent: React.FC = ({ const textComponent = useCallback( ({ children }) => emojis && children ? ( - + ) : ( {children} ), @@ -143,7 +160,7 @@ const ParseContent: React.FC = ({ numberOfLines={ totalLines && totalLines > numberOfLines ? shownLines : totalLines } - style={styles.root} + style={{ lineHeight: StyleConstants.Font.LineHeight[size] }} onTextLayout={({ nativeEvent }) => { if (!textLoaded) { setTextLoaded(true) @@ -200,10 +217,4 @@ const ParseContent: React.FC = ({ ) } -const styles = StyleSheet.create({ - root: { - lineHeight: StyleConstants.Font.LineHeight.M - } -}) - export default ParseContent diff --git a/src/components/Timelines/Timeline.tsx b/src/components/Timelines/Timeline.tsx index ab02c67f..989ec54f 100644 --- a/src/components/Timelines/Timeline.tsx +++ b/src/components/Timelines/Timeline.tsx @@ -1,5 +1,5 @@ -import React, { useCallback, useEffect, useRef } from 'react' -import { ActivityIndicator, AppState, FlatList, StyleSheet } from 'react-native' +import React, { useCallback, useEffect, useMemo, useRef } from 'react' +import { AppState, FlatList, StyleSheet } from 'react-native' import { setFocusHandler, useInfiniteQuery } from 'react-query' import TimelineNotifications from 'src/components/Timelines/Timeline/Notifications' @@ -75,10 +75,33 @@ const Timeline: React.FC = ({ case 'Notifications': return default: - return + return ( + + ) } }, []) - const flItemSeparatorComponent = useCallback(() => , []) + const flItemSeparatorComponent = useCallback( + ({ leadingItem }) => ( + + ), + [] + ) + const flItemEmptyComponent = useMemo( + () => ( + + ), + [isLoading, isError] + ) const flOnRefresh = useCallback( () => !disableRefresh && @@ -102,11 +125,12 @@ const Timeline: React.FC = ({ [flattenData] ) const flFooter = useCallback(() => { - if (isFetchingMore) { - return - } else { - return - } + return + // if (isFetchingMore) { + // return + // } else { + // return + // } }, [isFetchingMore]) const onScrollToIndexFailed = useCallback(error => { const offset = error.averageItemLength * error.index @@ -127,18 +151,12 @@ const Timeline: React.FC = ({ renderItem={flRenderItem} onEndReached={flOnEndReach} keyExtractor={flKeyExtrator} - ListFooterComponent={flFooter} scrollEnabled={scrollEnabled} // For timeline in Account view + ListFooterComponent={flFooter} + ListEmptyComponent={flItemEmptyComponent} ItemSeparatorComponent={flItemSeparatorComponent} onEndReachedThreshold={!disableRefresh ? 0.75 : null} refreshing={!disableRefresh && isLoading && flattenData.length > 0} - ListEmptyComponent={ - - } {...(toot && isSuccess && { onScrollToIndexFailed })} /> ) diff --git a/src/components/Timelines/Timeline/Default.tsx b/src/components/Timelines/Timeline/Default.tsx index 88d744fc..528ddfae 100644 --- a/src/components/Timelines/Timeline/Default.tsx +++ b/src/components/Timelines/Timeline/Default.tsx @@ -16,18 +16,25 @@ import { StyleConstants } from 'src/utils/styles/constants' export interface Props { item: Mastodon.Status queryKey: App.QueryKey + highlighted?: boolean } // When the poll is long -const TimelineDefault: React.FC = ({ item, queryKey }) => { +const TimelineDefault: React.FC = ({ + item, + queryKey, + highlighted = false +}) => { const navigation = useNavigation() let actualStatus = item.reblog ? item.reblog : item - const contentWidth = - Dimensions.get('window').width - - StyleConstants.Spacing.Global.PagePadding * 2 - // Global page padding on both sides - StyleConstants.Avatar.S - // Avatar width - StyleConstants.Spacing.S // Avatar margin to the right + const contentWidth = highlighted + ? Dimensions.get('window').width - + StyleConstants.Spacing.Global.PagePadding * 2 // Global page padding on both sides + : Dimensions.get('window').width - + StyleConstants.Spacing.Global.PagePadding * 2 - // Global page padding on both sides + StyleConstants.Avatar.S - // Avatar width + StyleConstants.Spacing.S // Avatar margin to the right const tootOnPress = useCallback( () => @@ -38,9 +45,15 @@ const TimelineDefault: React.FC = ({ item, queryKey }) => { ) const tootChildren = useMemo( () => ( - <> + {actualStatus.content.length > 0 && ( - + )} {actualStatus.poll && ( @@ -49,7 +62,7 @@ const TimelineDefault: React.FC = ({ item, queryKey }) => { )} {actualStatus.card && } - + ), [actualStatus.poll?.voted] ) @@ -59,13 +72,19 @@ const TimelineDefault: React.FC = ({ item, queryKey }) => { {item.reblog && ( )} - + - - - - - + + + + + ) @@ -73,17 +92,17 @@ const TimelineDefault: React.FC = ({ item, queryKey }) => { const styles = StyleSheet.create({ statusView: { - flex: 1, - flexDirection: 'column', padding: StyleConstants.Spacing.Global.PagePadding, paddingBottom: StyleConstants.Spacing.M }, - status: { + header: { flex: 1, - flexDirection: 'row' + width: '100%', + flexDirection: 'row', + marginBottom: StyleConstants.Spacing.S }, - details: { - flex: 1 + content: { + paddingLeft: StyleConstants.Avatar.S + StyleConstants.Spacing.S } }) diff --git a/src/components/Timelines/Timeline/Separator.tsx b/src/components/Timelines/Timeline/Separator.tsx index f7f871d0..8ddbdbf4 100644 --- a/src/components/Timelines/Timeline/Separator.tsx +++ b/src/components/Timelines/Timeline/Separator.tsx @@ -4,20 +4,34 @@ import { StyleSheet, View } from 'react-native' import { useTheme } from 'src/utils/styles/ThemeManager' import { StyleConstants } from 'src/utils/styles/constants' -const TimelineSeparator = () => { +export interface Props { + highlighted?: boolean +} + +const TimelineSeparator: React.FC = ({ highlighted = false }) => { const { theme } = useTheme() - return + return ( + + ) } const styles = StyleSheet.create({ base: { borderTopWidth: 1, - marginLeft: - StyleConstants.Spacing.M + - StyleConstants.Avatar.S + - StyleConstants.Spacing.S, - marginRight: StyleConstants.Spacing.M + marginRight: StyleConstants.Spacing.Global.PagePadding } }) diff --git a/src/components/Timelines/Timeline/Shared/Avatar.tsx b/src/components/Timelines/Timeline/Shared/Avatar.tsx index c98bf41e..b5c68cd2 100644 --- a/src/components/Timelines/Timeline/Shared/Avatar.tsx +++ b/src/components/Timelines/Timeline/Shared/Avatar.tsx @@ -25,7 +25,7 @@ const TimelineAvatar: React.FC = ({ account }) => { const styles = StyleSheet.create({ avatar: { - width: StyleConstants.Avatar.S, + flexBasis: StyleConstants.Avatar.S, height: StyleConstants.Avatar.S, marginRight: StyleConstants.Spacing.S }, diff --git a/src/components/Timelines/Timeline/Shared/Content.tsx b/src/components/Timelines/Timeline/Shared/Content.tsx index bfe91be1..ac17fac5 100644 --- a/src/components/Timelines/Timeline/Shared/Content.tsx +++ b/src/components/Timelines/Timeline/Shared/Content.tsx @@ -11,9 +11,14 @@ import { LinearGradient } from 'expo-linear-gradient' export interface Props { status: Mastodon.Status numberOfLines?: number + highlighted?: boolean } -const TimelineContent: React.FC = ({ status, numberOfLines }) => { +const TimelineContent: React.FC = ({ + status, + numberOfLines, + highlighted = false +}) => { const { theme } = useTheme() const [spoilerCollapsed, setSpoilerCollapsed] = useState(true) const lineHeight = 28 @@ -24,13 +29,13 @@ const TimelineContent: React.FC = ({ status, numberOfLines }) => { <> = ({ status, numberOfLines }) => { ) : ( { +export interface Props { + isFetchingMore: false | 'previous' | 'next' | undefined +} + +const TimelineEnd: React.FC = ({ isFetchingMore }) => { const { theme } = useTheme() return ( - - 居然刷到底了,喝杯{' '} - {' '} - 吧 - + {isFetchingMore ? ( + + ) : ( + + 居然刷到底了,喝杯{' '} + {' '} + 吧 + + )} ) } diff --git a/src/components/Timelines/Timeline/Shared/HeaderDefault.tsx b/src/components/Timelines/Timeline/Shared/HeaderDefault.tsx index 4d3c5135..611dd90c 100644 --- a/src/components/Timelines/Timeline/Shared/HeaderDefault.tsx +++ b/src/components/Timelines/Timeline/Shared/HeaderDefault.tsx @@ -60,7 +60,7 @@ const TimelineHeaderDefault: React.FC = ({ queryKey, status }) => { ) return ( - + {emojis?.length ? ( @@ -156,13 +156,17 @@ const TimelineHeaderDefault: React.FC = ({ queryKey, status }) => { } const styles = StyleSheet.create({ + base: { + flex: 1 + }, nameAndAction: { - width: '100%', + flex: 1, + flexBasis: '100%', flexDirection: 'row', - justifyContent: 'space-between' + alignItems: 'flex-start' }, name: { - flexBasis: '90%', + flexBasis: '85%', flexDirection: 'row' }, nameWithoutEmoji: { @@ -170,7 +174,9 @@ const styles = StyleSheet.create({ fontWeight: StyleConstants.Font.Weight.Bold }, action: { - alignItems: 'flex-end' + flex: 1, + flexDirection: 'row', + justifyContent: 'center' }, account: { flexShrink: 1, diff --git a/src/utils/styles/constants.ts b/src/utils/styles/constants.ts index 6e7892ee..0d46fd37 100644 --- a/src/utils/styles/constants.ts +++ b/src/utils/styles/constants.ts @@ -7,7 +7,7 @@ export const StyleConstants = { M: 16, L: 18 }, - LineHeight: { M: 20 }, + LineHeight: { M: 20, L: 24 }, Weight: { Bold: '600' as '600' }