[🐴] Mutate data instead of invalidating queries when muting or unmuting (#3946)
* mutate for mutes * mutate data for mutes * add initial data, `useConvoQuery` in `ConvoMenu` * `useInitialData` * don't use `identifier` for notifications, use `dates` instead * better implementation * simplify * simplify * fix typeszio/stable
parent
8f56f79c6c
commit
f928e0a547
|
@ -2,17 +2,18 @@ import React, {useCallback} from 'react'
|
||||||
import {Keyboard, Pressable, View} from 'react-native'
|
import {Keyboard, Pressable, View} from 'react-native'
|
||||||
import {AppBskyActorDefs} from '@atproto/api'
|
import {AppBskyActorDefs} from '@atproto/api'
|
||||||
import {ChatBskyConvoDefs} from '@atproto-labs/api'
|
import {ChatBskyConvoDefs} from '@atproto-labs/api'
|
||||||
|
import {ConvoView} from '@atproto-labs/api/dist/client/types/chat/bsky/convo/defs'
|
||||||
import {msg, Trans} from '@lingui/macro'
|
import {msg, Trans} from '@lingui/macro'
|
||||||
import {useLingui} from '@lingui/react'
|
import {useLingui} from '@lingui/react'
|
||||||
import {useNavigation} from '@react-navigation/native'
|
import {useNavigation} from '@react-navigation/native'
|
||||||
|
|
||||||
import {NavigationProp} from '#/lib/routes/types'
|
import {NavigationProp} from '#/lib/routes/types'
|
||||||
import {useMarkAsReadMutation} from '#/state/queries/messages/conversation'
|
|
||||||
import {useLeaveConvo} from '#/state/queries/messages/leave-conversation'
|
|
||||||
import {
|
import {
|
||||||
useMuteConvo,
|
useConvoQuery,
|
||||||
useUnmuteConvo,
|
useMarkAsReadMutation,
|
||||||
} from '#/state/queries/messages/mute-conversation'
|
} from '#/state/queries/messages/conversation'
|
||||||
|
import {useLeaveConvo} from '#/state/queries/messages/leave-conversation'
|
||||||
|
import {useMuteConvo} from '#/state/queries/messages/mute-conversation'
|
||||||
import * as Toast from '#/view/com/util/Toast'
|
import * as Toast from '#/view/com/util/Toast'
|
||||||
import {atoms as a, useTheme} from '#/alf'
|
import {atoms as a, useTheme} from '#/alf'
|
||||||
import {ArrowBoxLeft_Stroke2_Corner0_Rounded as ArrowBoxLeft} from '#/components/icons/ArrowBoxLeft'
|
import {ArrowBoxLeft_Stroke2_Corner0_Rounded as ArrowBoxLeft} from '#/components/icons/ArrowBoxLeft'
|
||||||
|
@ -28,16 +29,15 @@ import * as Prompt from '#/components/Prompt'
|
||||||
import {Bubble_Stroke2_Corner2_Rounded as Bubble} from '../icons/Bubble'
|
import {Bubble_Stroke2_Corner2_Rounded as Bubble} from '../icons/Bubble'
|
||||||
|
|
||||||
let ConvoMenu = ({
|
let ConvoMenu = ({
|
||||||
convo,
|
convo: initialConvo,
|
||||||
profile,
|
profile,
|
||||||
onUpdateConvo,
|
|
||||||
control,
|
control,
|
||||||
currentScreen,
|
currentScreen,
|
||||||
showMarkAsRead,
|
showMarkAsRead,
|
||||||
hideTrigger,
|
hideTrigger,
|
||||||
triggerOpacity,
|
triggerOpacity,
|
||||||
}: {
|
}: {
|
||||||
convo: ChatBskyConvoDefs.ConvoView
|
convo: ConvoView
|
||||||
profile: AppBskyActorDefs.ProfileViewBasic
|
profile: AppBskyActorDefs.ProfileViewBasic
|
||||||
onUpdateConvo?: (convo: ChatBskyConvoDefs.ConvoView) => void
|
onUpdateConvo?: (convo: ChatBskyConvoDefs.ConvoView) => void
|
||||||
control?: Menu.MenuControlProps
|
control?: Menu.MenuControlProps
|
||||||
|
@ -52,31 +52,26 @@ let ConvoMenu = ({
|
||||||
const leaveConvoControl = Prompt.usePromptControl()
|
const leaveConvoControl = Prompt.usePromptControl()
|
||||||
const {mutate: markAsRead} = useMarkAsReadMutation()
|
const {mutate: markAsRead} = useMarkAsReadMutation()
|
||||||
|
|
||||||
|
const {data: convo} = useConvoQuery(initialConvo)
|
||||||
|
|
||||||
const onNavigateToProfile = useCallback(() => {
|
const onNavigateToProfile = useCallback(() => {
|
||||||
navigation.navigate('Profile', {name: profile.did})
|
navigation.navigate('Profile', {name: profile.did})
|
||||||
}, [navigation, profile.did])
|
}, [navigation, profile.did])
|
||||||
|
|
||||||
const {mutate: muteConvo} = useMuteConvo(convo.id, {
|
const {mutate: muteConvo} = useMuteConvo(convo?.id, {
|
||||||
onSuccess: data => {
|
onSuccess: data => {
|
||||||
onUpdateConvo?.(data.convo)
|
if (data.convo.muted) {
|
||||||
Toast.show(_(msg`Chat muted`))
|
Toast.show(_(msg`Chat muted`))
|
||||||
|
} else {
|
||||||
|
Toast.show(_(msg`Chat unmuted`))
|
||||||
|
}
|
||||||
},
|
},
|
||||||
onError: () => {
|
onError: () => {
|
||||||
Toast.show(_(msg`Could not mute chat`))
|
Toast.show(_(msg`Could not mute chat`))
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|
||||||
const {mutate: unmuteConvo} = useUnmuteConvo(convo.id, {
|
const {mutate: leaveConvo} = useLeaveConvo(convo?.id, {
|
||||||
onSuccess: data => {
|
|
||||||
onUpdateConvo?.(data.convo)
|
|
||||||
Toast.show(_(msg`Chat unmuted`))
|
|
||||||
},
|
|
||||||
onError: () => {
|
|
||||||
Toast.show(_(msg`Could not unmute chat`))
|
|
||||||
},
|
|
||||||
})
|
|
||||||
|
|
||||||
const {mutate: leaveConvo} = useLeaveConvo(convo.id, {
|
|
||||||
onSuccess: () => {
|
onSuccess: () => {
|
||||||
if (currentScreen === 'conversation') {
|
if (currentScreen === 'conversation') {
|
||||||
navigation.replace('Messages')
|
navigation.replace('Messages')
|
||||||
|
@ -121,7 +116,7 @@ let ConvoMenu = ({
|
||||||
label={_(msg`Mark as read`)}
|
label={_(msg`Mark as read`)}
|
||||||
onPress={() =>
|
onPress={() =>
|
||||||
markAsRead({
|
markAsRead({
|
||||||
convoId: convo.id,
|
convoId: convo?.id,
|
||||||
})
|
})
|
||||||
}>
|
}>
|
||||||
<Menu.ItemText>
|
<Menu.ItemText>
|
||||||
|
@ -140,7 +135,7 @@ let ConvoMenu = ({
|
||||||
</Menu.Item>
|
</Menu.Item>
|
||||||
<Menu.Item
|
<Menu.Item
|
||||||
label={_(msg`Mute notifications`)}
|
label={_(msg`Mute notifications`)}
|
||||||
onPress={() => (convo?.muted ? unmuteConvo() : muteConvo())}>
|
onPress={() => muteConvo({mute: !convo?.muted})}>
|
||||||
<Menu.ItemText>
|
<Menu.ItemText>
|
||||||
{convo?.muted ? (
|
{convo?.muted ? (
|
||||||
<Trans>Unmute notifications</Trans>
|
<Trans>Unmute notifications</Trans>
|
||||||
|
|
|
@ -56,7 +56,7 @@ export function MessagesConversationScreen({route}: Props) {
|
||||||
|
|
||||||
function Inner() {
|
function Inner() {
|
||||||
const t = useTheme()
|
const t = useTheme()
|
||||||
const convo = useConvo()
|
const convoState = useConvo()
|
||||||
const {_} = useLingui()
|
const {_} = useLingui()
|
||||||
|
|
||||||
const [hasInitiallyRendered, setHasInitiallyRendered] = React.useState(false)
|
const [hasInitiallyRendered, setHasInitiallyRendered] = React.useState(false)
|
||||||
|
@ -72,23 +72,23 @@ function Inner() {
|
||||||
React.useEffect(() => {
|
React.useEffect(() => {
|
||||||
if (
|
if (
|
||||||
!hasInitiallyRendered &&
|
!hasInitiallyRendered &&
|
||||||
convo.status === ConvoStatus.Ready &&
|
convoState.status === ConvoStatus.Ready &&
|
||||||
!convo.isFetchingHistory
|
!convoState.isFetchingHistory
|
||||||
) {
|
) {
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
setHasInitiallyRendered(true)
|
setHasInitiallyRendered(true)
|
||||||
}, 15)
|
}, 15)
|
||||||
}
|
}
|
||||||
}, [convo.isFetchingHistory, convo.items, convo.status, hasInitiallyRendered])
|
}, [convoState.isFetchingHistory, convoState.status, hasInitiallyRendered])
|
||||||
|
|
||||||
if (convo.status === ConvoStatus.Error) {
|
if (convoState.status === ConvoStatus.Error) {
|
||||||
return (
|
return (
|
||||||
<CenteredView style={a.flex_1} sideBorders>
|
<CenteredView style={a.flex_1} sideBorders>
|
||||||
<Header />
|
<Header />
|
||||||
<Error
|
<Error
|
||||||
title={_(msg`Something went wrong`)}
|
title={_(msg`Something went wrong`)}
|
||||||
message={_(msg`We couldn't load this conversation`)}
|
message={_(msg`We couldn't load this conversation`)}
|
||||||
onRetry={() => convo.error.retry()}
|
onRetry={() => convoState.error.retry()}
|
||||||
/>
|
/>
|
||||||
</CenteredView>
|
</CenteredView>
|
||||||
)
|
)
|
||||||
|
@ -106,9 +106,9 @@ function Inner() {
|
||||||
behavior="padding"
|
behavior="padding"
|
||||||
contentContainerStyle={a.flex_1}>
|
contentContainerStyle={a.flex_1}>
|
||||||
<CenteredView style={a.flex_1} sideBorders>
|
<CenteredView style={a.flex_1} sideBorders>
|
||||||
<Header profile={convo.recipients?.[0]} />
|
<Header profile={convoState.recipients?.[0]} />
|
||||||
<View style={[a.flex_1]}>
|
<View style={[a.flex_1]}>
|
||||||
{convo.status !== ConvoStatus.Ready ? (
|
{convoState.status !== ConvoStatus.Ready ? (
|
||||||
<ListMaybePlaceholder isLoading />
|
<ListMaybePlaceholder isLoading />
|
||||||
) : (
|
) : (
|
||||||
<MessagesList />
|
<MessagesList />
|
||||||
|
@ -145,7 +145,7 @@ let Header = ({
|
||||||
const {_} = useLingui()
|
const {_} = useLingui()
|
||||||
const {gtTablet} = useBreakpoints()
|
const {gtTablet} = useBreakpoints()
|
||||||
const navigation = useNavigation<NavigationProp>()
|
const navigation = useNavigation<NavigationProp>()
|
||||||
const convo = useConvo()
|
const convoState = useConvo()
|
||||||
|
|
||||||
const onPressBack = useCallback(() => {
|
const onPressBack = useCallback(() => {
|
||||||
if (isWeb) {
|
if (isWeb) {
|
||||||
|
@ -155,10 +155,6 @@ let Header = ({
|
||||||
}
|
}
|
||||||
}, [navigation])
|
}, [navigation])
|
||||||
|
|
||||||
const onUpdateConvo = useCallback(() => {
|
|
||||||
// TODO eric update muted state
|
|
||||||
}, [])
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<View
|
<View
|
||||||
style={[
|
style={[
|
||||||
|
@ -234,11 +230,10 @@ let Header = ({
|
||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
</View>
|
</View>
|
||||||
{convo.status === ConvoStatus.Ready && profile ? (
|
{convoState.status === ConvoStatus.Ready && profile ? (
|
||||||
<ConvoMenu
|
<ConvoMenu
|
||||||
convo={convo.convo}
|
convo={convoState.convo}
|
||||||
profile={profile}
|
profile={profile}
|
||||||
onUpdateConvo={onUpdateConvo}
|
|
||||||
currentScreen="conversation"
|
currentScreen="conversation"
|
||||||
/>
|
/>
|
||||||
) : (
|
) : (
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
import {BskyAgent} from '@atproto-labs/api'
|
import {BskyAgent} from '@atproto-labs/api'
|
||||||
|
import {ConvoView} from '@atproto-labs/api/dist/client/types/chat/bsky/convo/defs'
|
||||||
import {useMutation, useQuery, useQueryClient} from '@tanstack/react-query'
|
import {useMutation, useQuery, useQueryClient} from '@tanstack/react-query'
|
||||||
|
|
||||||
import {useOnMarkAsRead} from '#/state/queries/messages/list-converations'
|
import {useOnMarkAsRead} from '#/state/queries/messages/list-converations'
|
||||||
|
@ -9,20 +10,21 @@ import {useHeaders} from './temp-headers'
|
||||||
const RQKEY_ROOT = 'convo'
|
const RQKEY_ROOT = 'convo'
|
||||||
export const RQKEY = (convoId: string) => [RQKEY_ROOT, convoId]
|
export const RQKEY = (convoId: string) => [RQKEY_ROOT, convoId]
|
||||||
|
|
||||||
export function useConvoQuery(convoId: string) {
|
export function useConvoQuery(convo: ConvoView) {
|
||||||
const headers = useHeaders()
|
const headers = useHeaders()
|
||||||
const {serviceUrl} = useDmServiceUrlStorage()
|
const {serviceUrl} = useDmServiceUrlStorage()
|
||||||
|
|
||||||
return useQuery({
|
return useQuery({
|
||||||
queryKey: RQKEY(convoId),
|
queryKey: RQKEY(convo.id),
|
||||||
queryFn: async () => {
|
queryFn: async () => {
|
||||||
const agent = new BskyAgent({service: serviceUrl})
|
const agent = new BskyAgent({service: serviceUrl})
|
||||||
const {data} = await agent.api.chat.bsky.convo.getConvo(
|
const {data} = await agent.api.chat.bsky.convo.getConvo(
|
||||||
{convoId},
|
{convoId: convo.id},
|
||||||
{headers},
|
{headers},
|
||||||
)
|
)
|
||||||
return data.convo
|
return data.convo
|
||||||
},
|
},
|
||||||
|
initialData: convo,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -37,9 +39,11 @@ export function useMarkAsReadMutation() {
|
||||||
convoId,
|
convoId,
|
||||||
messageId,
|
messageId,
|
||||||
}: {
|
}: {
|
||||||
convoId: string
|
convoId?: string
|
||||||
messageId?: string
|
messageId?: string
|
||||||
}) => {
|
}) => {
|
||||||
|
if (!convoId) throw new Error('No convoId provided')
|
||||||
|
|
||||||
const agent = new BskyAgent({service: serviceUrl})
|
const agent = new BskyAgent({service: serviceUrl})
|
||||||
await agent.api.chat.bsky.convo.updateRead(
|
await agent.api.chat.bsky.convo.updateRead(
|
||||||
{
|
{
|
||||||
|
@ -53,6 +57,7 @@ export function useMarkAsReadMutation() {
|
||||||
)
|
)
|
||||||
},
|
},
|
||||||
onMutate({convoId}) {
|
onMutate({convoId}) {
|
||||||
|
if (!convoId) throw new Error('No convoId provided')
|
||||||
optimisticUpdate(convoId)
|
optimisticUpdate(convoId)
|
||||||
},
|
},
|
||||||
onSettled() {
|
onSettled() {
|
||||||
|
|
|
@ -11,7 +11,7 @@ import {RQKEY as CONVO_LIST_KEY} from './list-converations'
|
||||||
import {useHeaders} from './temp-headers'
|
import {useHeaders} from './temp-headers'
|
||||||
|
|
||||||
export function useLeaveConvo(
|
export function useLeaveConvo(
|
||||||
convoId: string,
|
convoId: string | undefined,
|
||||||
{
|
{
|
||||||
onSuccess,
|
onSuccess,
|
||||||
onError,
|
onError,
|
||||||
|
@ -26,6 +26,8 @@ export function useLeaveConvo(
|
||||||
|
|
||||||
return useMutation({
|
return useMutation({
|
||||||
mutationFn: async () => {
|
mutationFn: async () => {
|
||||||
|
if (!convoId) throw new Error('No convoId provided')
|
||||||
|
|
||||||
const agent = new BskyAgent({service: serviceUrl})
|
const agent = new BskyAgent({service: serviceUrl})
|
||||||
const {data} = await agent.api.chat.bsky.convo.leaveConvo(
|
const {data} = await agent.api.chat.bsky.convo.leaveConvo(
|
||||||
{convoId},
|
{convoId},
|
||||||
|
@ -41,7 +43,6 @@ export function useLeaveConvo(
|
||||||
pageParams: Array<string | undefined>
|
pageParams: Array<string | undefined>
|
||||||
pages: Array<ChatBskyConvoListConvos.OutputSchema>
|
pages: Array<ChatBskyConvoListConvos.OutputSchema>
|
||||||
}) => {
|
}) => {
|
||||||
console.log('old', old)
|
|
||||||
if (!old) return old
|
if (!old) return old
|
||||||
return {
|
return {
|
||||||
...old,
|
...old,
|
||||||
|
|
|
@ -1,18 +1,18 @@
|
||||||
import {
|
import {
|
||||||
BskyAgent,
|
BskyAgent,
|
||||||
|
ChatBskyConvoDefs,
|
||||||
|
ChatBskyConvoListConvos,
|
||||||
ChatBskyConvoMuteConvo,
|
ChatBskyConvoMuteConvo,
|
||||||
ChatBskyConvoUnmuteConvo,
|
|
||||||
} from '@atproto-labs/api'
|
} from '@atproto-labs/api'
|
||||||
import {useMutation, useQueryClient} from '@tanstack/react-query'
|
import {InfiniteData, useMutation, useQueryClient} from '@tanstack/react-query'
|
||||||
|
|
||||||
import {logger} from '#/logger'
|
|
||||||
import {useDmServiceUrlStorage} from '#/screens/Messages/Temp/useDmServiceUrlStorage'
|
import {useDmServiceUrlStorage} from '#/screens/Messages/Temp/useDmServiceUrlStorage'
|
||||||
import {RQKEY as CONVO_KEY} from './conversation'
|
import {RQKEY as CONVO_KEY} from './conversation'
|
||||||
import {RQKEY as CONVO_LIST_KEY} from './list-converations'
|
import {RQKEY as CONVO_LIST_KEY} from './list-converations'
|
||||||
import {useHeaders} from './temp-headers'
|
import {useHeaders} from './temp-headers'
|
||||||
|
|
||||||
export function useMuteConvo(
|
export function useMuteConvo(
|
||||||
convoId: string,
|
convoId: string | undefined,
|
||||||
{
|
{
|
||||||
onSuccess,
|
onSuccess,
|
||||||
onError,
|
onError,
|
||||||
|
@ -26,59 +26,58 @@ export function useMuteConvo(
|
||||||
const {serviceUrl} = useDmServiceUrlStorage()
|
const {serviceUrl} = useDmServiceUrlStorage()
|
||||||
|
|
||||||
return useMutation({
|
return useMutation({
|
||||||
mutationFn: async () => {
|
mutationFn: async ({mute}: {mute: boolean}) => {
|
||||||
const agent = new BskyAgent({service: serviceUrl})
|
if (!convoId) throw new Error('No convoId provided')
|
||||||
const {data} = await agent.api.chat.bsky.convo.muteConvo(
|
|
||||||
{convoId},
|
|
||||||
{headers, encoding: 'application/json'},
|
|
||||||
)
|
|
||||||
|
|
||||||
return data
|
const agent = new BskyAgent({service: serviceUrl})
|
||||||
|
if (mute) {
|
||||||
|
const {data} = await agent.api.chat.bsky.convo.muteConvo(
|
||||||
|
{convoId},
|
||||||
|
{headers, encoding: 'application/json'},
|
||||||
|
)
|
||||||
|
return data
|
||||||
|
} else {
|
||||||
|
const {data} = await agent.api.chat.bsky.convo.unmuteConvo(
|
||||||
|
{convoId},
|
||||||
|
{headers, encoding: 'application/json'},
|
||||||
|
)
|
||||||
|
return data
|
||||||
|
}
|
||||||
},
|
},
|
||||||
onSuccess: data => {
|
onSuccess: (data, params) => {
|
||||||
queryClient.invalidateQueries({queryKey: CONVO_LIST_KEY})
|
queryClient.setQueryData<ChatBskyConvoDefs.ConvoView>(
|
||||||
queryClient.invalidateQueries({queryKey: CONVO_KEY(convoId)})
|
CONVO_KEY(data.convo.id),
|
||||||
|
prev => {
|
||||||
|
if (!prev) return
|
||||||
|
return {
|
||||||
|
...prev,
|
||||||
|
muted: params.mute,
|
||||||
|
}
|
||||||
|
},
|
||||||
|
)
|
||||||
|
queryClient.setQueryData<
|
||||||
|
InfiniteData<ChatBskyConvoListConvos.OutputSchema>
|
||||||
|
>(CONVO_LIST_KEY, prev => {
|
||||||
|
if (!prev?.pages) return
|
||||||
|
return {
|
||||||
|
...prev,
|
||||||
|
pages: prev.pages.map(page => ({
|
||||||
|
...page,
|
||||||
|
convos: page.convos.map(convo => {
|
||||||
|
if (convo.id !== data.convo.id) return convo
|
||||||
|
return {
|
||||||
|
...convo,
|
||||||
|
muted: params.mute,
|
||||||
|
}
|
||||||
|
}),
|
||||||
|
})),
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
onSuccess?.(data)
|
onSuccess?.(data)
|
||||||
},
|
},
|
||||||
onError: error => {
|
onError: e => {
|
||||||
logger.error(error)
|
onError?.(e)
|
||||||
onError?.(error)
|
|
||||||
},
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
export function useUnmuteConvo(
|
|
||||||
convoId: string,
|
|
||||||
{
|
|
||||||
onSuccess,
|
|
||||||
onError,
|
|
||||||
}: {
|
|
||||||
onSuccess?: (data: ChatBskyConvoUnmuteConvo.OutputSchema) => void
|
|
||||||
onError?: (error: Error) => void
|
|
||||||
},
|
|
||||||
) {
|
|
||||||
const queryClient = useQueryClient()
|
|
||||||
const headers = useHeaders()
|
|
||||||
const {serviceUrl} = useDmServiceUrlStorage()
|
|
||||||
|
|
||||||
return useMutation({
|
|
||||||
mutationFn: async () => {
|
|
||||||
const agent = new BskyAgent({service: serviceUrl})
|
|
||||||
const {data} = await agent.api.chat.bsky.convo.unmuteConvo(
|
|
||||||
{convoId},
|
|
||||||
{headers, encoding: 'application/json'},
|
|
||||||
)
|
|
||||||
|
|
||||||
return data
|
|
||||||
},
|
|
||||||
onSuccess: data => {
|
|
||||||
queryClient.invalidateQueries({queryKey: CONVO_LIST_KEY})
|
|
||||||
queryClient.invalidateQueries({queryKey: CONVO_KEY(convoId)})
|
|
||||||
onSuccess?.(data)
|
|
||||||
},
|
|
||||||
onError: error => {
|
|
||||||
logger.error(error)
|
|
||||||
onError?.(error)
|
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue