From c7cc3f1f27b81fa06cf5804f318e2bb33da21c79 Mon Sep 17 00:00:00 2001 From: Zhiyuan Zheng Date: Sun, 27 Dec 2020 16:25:29 +0100 Subject: [PATCH] Rewrite account page --- src/components/Button.tsx | 2 +- src/components/Timelines/Timeline.tsx | 4 +- .../Timelines/Timeline/Conversation.tsx | 58 ++- .../Timeline/Shared/HeaderConversation.tsx | 48 +- src/screens/Me/Root.tsx | 1 - src/screens/Me/Root/Login.tsx | 4 + src/screens/Shared/Account.tsx | 1 - src/screens/Shared/Account/Header.tsx | 14 +- src/screens/Shared/Account/Information.tsx | 440 ++---------------- .../Shared/Account/Information/Account.tsx | 64 +++ .../Shared/Account/Information/Actions.tsx | 160 +++++++ .../Shared/Account/Information/Avatar.tsx | 47 ++ .../Shared/Account/Information/Created.tsx | 68 +++ .../Shared/Account/Information/Fields.tsx | 83 ++++ .../Shared/Account/Information/Name.tsx | 64 +++ .../Shared/Account/Information/Notes.tsx | 28 ++ .../Shared/Account/Information/Stats.tsx | 95 ++++ src/screens/Shared/Announcements.tsx | 48 +- src/screens/Shared/Compose/Attachments.tsx | 1 + src/utils/styles/themes.ts | 21 +- 20 files changed, 785 insertions(+), 466 deletions(-) create mode 100644 src/screens/Shared/Account/Information/Account.tsx create mode 100644 src/screens/Shared/Account/Information/Actions.tsx create mode 100644 src/screens/Shared/Account/Information/Avatar.tsx create mode 100644 src/screens/Shared/Account/Information/Created.tsx create mode 100644 src/screens/Shared/Account/Information/Fields.tsx create mode 100644 src/screens/Shared/Account/Information/Name.tsx create mode 100644 src/screens/Shared/Account/Information/Notes.tsx create mode 100644 src/screens/Shared/Account/Information/Stats.tsx diff --git a/src/components/Button.tsx b/src/components/Button.tsx index 88b9cfc1..881e15aa 100644 --- a/src/components/Button.tsx +++ b/src/components/Button.tsx @@ -46,7 +46,7 @@ const Button: React.FC = ({ }) => { const { theme } = useTheme() - useLayoutEffect(() => layoutAnimation(), [loading, disabled]) + useLayoutEffect(() => layoutAnimation(), [content, loading, disabled]) const loadingSpinkit = useMemo( () => ( diff --git a/src/components/Timelines/Timeline.tsx b/src/components/Timelines/Timeline.tsx index 8440b6d1..78f444af 100644 --- a/src/components/Timelines/Timeline.tsx +++ b/src/components/Timelines/Timeline.tsx @@ -100,7 +100,9 @@ const Timeline: React.FC = ({ ({ item, index }) => { switch (page) { case 'Conversations': - return + return ( + + ) case 'Notifications': return ( diff --git a/src/components/Timelines/Timeline/Conversation.tsx b/src/components/Timelines/Timeline/Conversation.tsx index 1a8868c7..330feb23 100644 --- a/src/components/Timelines/Timeline/Conversation.tsx +++ b/src/components/Timelines/Timeline/Conversation.tsx @@ -7,32 +7,55 @@ import TimelineHeaderConversation from '@components/Timelines/Timeline/Shared/He import TimelineContent from '@components/Timelines/Timeline/Shared/Content' import { StyleConstants } from '@utils/styles/constants' import TimelineActions from '@components/Timelines/Timeline/Shared/Actions' +import client from '@root/api/client' +import { useMutation, useQueryClient } from 'react-query' export interface Props { - item: Mastodon.Conversation + conversation: Mastodon.Conversation queryKey: QueryKey.Timeline highlighted?: boolean } -// Unread and mark as unread + +const fireMutation = async ({ id }: { id: Mastodon.Conversation['id'] }) => { + const res = await client({ + method: 'post', + instance: 'local', + url: `conversations/${id}/read` + }) + + if (res.body.id === id) { + return Promise.resolve() + } else { + return Promise.reject() + } +} + const TimelineConversation: React.FC = ({ - item, + conversation, queryKey, highlighted = false }) => { + const queryClient = useQueryClient() + const { mutate } = useMutation(fireMutation, { + onSettled: () => { + queryClient.invalidateQueries(queryKey) + } + }) + const navigation = useNavigation() - const conversationOnPress = useCallback( - () => - item.last_status && + const conversationOnPress = useCallback(() => { + if (conversation.last_status) { + conversation.unread && mutate({ id: conversation.id }) navigation.navigate('Screen-Shared-Toot', { - toot: item.last_status - }), - [] - ) + toot: conversation.last_status + }) + } + }, []) const conversationChildren = useMemo(() => { return ( - item.last_status && ( + conversation.last_status && ( = ({ }} > @@ -53,12 +76,13 @@ const TimelineConversation: React.FC = ({ return ( - + @@ -76,7 +100,7 @@ const TimelineConversation: React.FC = ({ > diff --git a/src/components/Timelines/Timeline/Shared/HeaderConversation.tsx b/src/components/Timelines/Timeline/Shared/HeaderConversation.tsx index 443f3782..da2ee989 100644 --- a/src/components/Timelines/Timeline/Shared/HeaderConversation.tsx +++ b/src/components/Timelines/Timeline/Shared/HeaderConversation.tsx @@ -12,9 +12,7 @@ import Emojis from '@components/Timelines/Timeline/Shared/Emojis' export interface Props { queryKey: QueryKey.Timeline - id: string - account: Mastodon.Account - created_at?: Mastodon.Status['created_at'] + conversation: Mastodon.Conversation } const fireMutation = async ({ id }: { id: string }) => { @@ -37,12 +35,7 @@ const fireMutation = async ({ id }: { id: string }) => { } } -const HeaderConversation: React.FC = ({ - queryKey, - id, - account, - created_at -}) => { +const HeaderConversation: React.FC = ({ queryKey, conversation }) => { const queryClient = useQueryClient() const { mutate } = useMutation(fireMutation, { onMutate: () => { @@ -51,7 +44,9 @@ const HeaderConversation: React.FC = ({ queryClient.setQueryData(queryKey, (old: any) => old.pages.map((paging: any) => ({ - toots: paging.toots.filter((toot: any) => toot.id !== id), + toots: paging.toots.filter( + (toot: Mastodon.Conversation) => toot.id !== conversation.id + ), pointer: paging.pointer })) ) @@ -66,7 +61,7 @@ const HeaderConversation: React.FC = ({ const { theme } = useTheme() - const actionOnPress = useCallback(() => mutate({ id }), []) + const actionOnPress = useCallback(() => mutate({ id: conversation.id }), []) const actionChildren = useMemo( () => ( @@ -83,10 +78,13 @@ const HeaderConversation: React.FC = ({ - {account.emojis ? ( + {conversation.accounts[0].emojis ? ( @@ -95,24 +93,27 @@ const HeaderConversation: React.FC = ({ numberOfLines={1} style={[styles.nameWithoutEmoji, { color: theme.primary }]} > - {account.display_name || account.username} + {conversation.accounts[0].display_name || + conversation.accounts[0].username} )} - @{account.acct} + @{conversation.accounts[0].acct} - - {created_at && ( - + + {conversation.last_status?.created_at && ( - {relativeTime(created_at)} + {relativeTime(conversation.last_status?.created_at)} - - )} + )} + {conversation.unread && ( + + )} + { - layoutAnimation() const localRegistered = useSelector(getLocalUrl) const scrollRef = useRef(null) diff --git a/src/screens/Me/Root/Login.tsx b/src/screens/Me/Root/Login.tsx index 32f463a7..30689166 100644 --- a/src/screens/Me/Root/Login.tsx +++ b/src/screens/Me/Root/Login.tsx @@ -138,6 +138,7 @@ const Login: React.FC = () => { StyleConstants.Spacing.Global.PagePadding * 4 } height={StyleConstants.Font.Size.M} + shimmerColors={theme.shimmer} > @@ -217,6 +218,7 @@ const Login: React.FC = () => { stopAutoRun width={StyleConstants.Font.Size.M * 4} height={StyleConstants.Font.Size.M} + shimmerColors={theme.shimmer} > { stopAutoRun width={StyleConstants.Font.Size.M * 4} height={StyleConstants.Font.Size.M} + shimmerColors={theme.shimmer} > { stopAutoRun width={StyleConstants.Font.Size.M * 4} height={StyleConstants.Font.Size.M} + shimmerColors={theme.shimmer} > = ({ }, navigation }) => { - layoutAnimation() const localAccountId = useSelector(getLocalAccountId) const { data } = useQuery(['Account', { id: account.id }], accountFetch) diff --git a/src/screens/Shared/Account/Header.tsx b/src/screens/Shared/Account/Header.tsx index c8357d1a..dabff21c 100644 --- a/src/screens/Shared/Account/Header.tsx +++ b/src/screens/Shared/Account/Header.tsx @@ -1,3 +1,4 @@ +import { useTheme } from '@root/utils/styles/ThemeManager' import React, { Dispatch, useEffect, useState } from 'react' import { Dimensions, Image, StyleSheet, View } from 'react-native' import { AccountAction, AccountState } from '../Account' @@ -15,6 +16,7 @@ const AccountHeader: React.FC = ({ account, limitHeight = false }) => { + const { theme } = useTheme() const [ratio, setRatio] = useState(accountState.headerRatio) let isMounted = false @@ -43,23 +45,25 @@ const AccountHeader: React.FC = ({ setRatio(limitHeight ? accountState.headerRatio : height / width) }) } else { - isMounted && setRatio(1 / 5) + isMounted && setRatio(1 / 3) } }, [account, isMounted]) const windowWidth = Dimensions.get('window').width return ( - + ) } const styles = StyleSheet.create({ - imageContainer: { - backgroundColor: 'lightgray' - }, image: { width: '100%', height: '100%' diff --git a/src/screens/Shared/Account/Information.tsx b/src/screens/Shared/Account/Information.tsx index 30c3a2cc..03c07b8e 100644 --- a/src/screens/Shared/Account/Information.tsx +++ b/src/screens/Shared/Account/Information.tsx @@ -1,50 +1,15 @@ -import React, { createRef, Dispatch, useEffect, useMemo, useState } from 'react' -import { Animated, Image, StyleSheet, Text, View } from 'react-native' -import { createShimmerPlaceholder } from 'react-native-shimmer-placeholder' -import { Feather } from '@expo/vector-icons' - -import ParseContent from '@components/ParseContent' -import { useTheme } from '@utils/styles/ThemeManager' import { StyleConstants } from '@utils/styles/constants' -import { useTranslation } from 'react-i18next' -import Emojis from '@components/Timelines/Timeline/Shared/Emojis' -import { LinearGradient } from 'expo-linear-gradient' +import React, { createRef, Dispatch, useCallback, useEffect } from 'react' +import { Animated, StyleSheet, View } from 'react-native' +import AccountInformationAvatar from './Information/Avatar' +import AccountInformationName from './Information/Name' +import AccountInformationAccount from './Information/Account' +import AccountInformationCreated from './Information/Created' +import AccountInformationStats from './Information/Stats' +import AccountInformationActions from './Information/Actions' +import AccountInformationFields from './Information/Fields' +import AccountInformationNotes from './Information/Notes' import { AccountAction } from '../Account' -import Button from '@root/components/Button' -import { useMutation, useQuery, useQueryClient } from 'react-query' -import { relationshipFetch } from '@root/utils/fetches/relationshipFetch' -import client from '@root/api/client' -import { useNavigation } from '@react-navigation/native' -import getCurrentTab from '@root/utils/getCurrentTab' - -const fireMutation = async ({ - type, - id, - stateKey, - prevState -}: { - type: 'follow' - id: string - stateKey: 'following' - prevState: boolean -}) => { - let res - switch (type) { - case 'follow': - res = await client({ - method: 'post', - instance: 'local', - url: `accounts/${id}/${prevState ? 'un' : ''}${type}` - }) - - if (res.body[stateKey] === !prevState) { - return Promise.resolve() - } else { - return Promise.reject() - } - break - } -} export interface Props { accountDispatch?: Dispatch @@ -57,325 +22,63 @@ const AccountInformation: React.FC = ({ account, disableActions = false }) => { - const navigation = useNavigation() - const { t } = useTranslation('sharedAccount') - const { theme } = useTheme() - const [avatarLoaded, setAvatarLoaded] = useState(false) - - const relationshipQueryKey = ['Relationship', { id: account?.id }] - const { status, data, refetch } = useQuery( - relationshipQueryKey, - relationshipFetch, - { - enabled: false - } - ) - useEffect(() => { - if (account?.id) { - refetch() - } - }, [account]) - const queryClient = useQueryClient() - const { mutate, status: mutateStatus } = useMutation(fireMutation, { - onMutate: () => { - queryClient.cancelQueries(relationshipQueryKey) - const oldData = queryClient.getQueryData(relationshipQueryKey) - - queryClient.setQueryData(relationshipQueryKey, (old: any) => { - old && (old.following = !old?.following) - return old - }) - - return oldData - }, - onError: (err, _, oldData) => { - queryClient.setQueryData(relationshipQueryKey, oldData) - }, - onSettled: () => { - queryClient.invalidateQueries(['Following']) - } - }) - - const ShimmerPlaceholder = createShimmerPlaceholder(LinearGradient) const shimmerAvatarRef = createRef() const shimmerNameRef = createRef() const shimmerAccountRef = createRef() const shimmerCreatedRef = createRef() - const shimmerStatTootRef = createRef() - const shimmerStatFolloingRef = createRef() - const shimmerStatFollowerRef = createRef() + const shimmerStatsRef = createRef() useEffect(() => { const informationAnimated = Animated.stagger(400, [ Animated.parallel([ - shimmerAvatarRef.current!.getAnimated(), - shimmerNameRef.current!.getAnimated(), - shimmerAccountRef.current!.getAnimated(), - shimmerCreatedRef.current!.getAnimated(), - shimmerStatTootRef.current!.getAnimated(), - shimmerStatFolloingRef.current!.getAnimated(), - shimmerStatFollowerRef.current!.getAnimated() + shimmerAvatarRef.current?.getAnimated(), + shimmerNameRef.current?.getAnimated(), + shimmerAccountRef.current?.getAnimated(), + shimmerCreatedRef.current?.getAnimated(), + shimmerStatsRef.current?.ref1.getAnimated(), + shimmerStatsRef.current?.ref2.getAnimated(), + shimmerStatsRef.current?.ref3.getAnimated() ]) ]) Animated.loop(informationAnimated).start() }, []) - const followingButton = useMemo( - () => ( -