[Clipclops] New clipclop dialog (#3750)
* add new routes with placeholder screens * add clops list * add a clop input * add some better padding to the clops * some more adjustments * add rnkc * implement rnkc * implement rnkc * be a little less weird about it * rename clop stuff * rename more clop * one more * add codegenerated lexicon * replace hailey's types * use codegen'd types in components * fix error + throw if fetch failed * remove bad imports * update messageslist and messageitem * import useState * replace hailey's types * use codegen'd types in components * add FAB * new chat dialog * error + default search term * fix typo * fix web styles * optimistically set chat data * use cursor instead of last rev * [Clipclops] Temp codegenerated lexicon (#3749) * add codegenerated lexicon * replace hailey's types * use codegen'd types in components * fix error + throw if fetch failed * remove bad imports * update messageslist and messageitem * import useState * add clop service URL hook * add dm service url storage * use context * use context for service url (temp) * remove log * cleanup merge * fix merge error * disable hack * sender-based message styles * temporary filter * merge cleanup * add `hideBackButton` * rm unneeded return * tried to be smart * hide go back button * use `searchActorTypeahead` instead --------- Co-authored-by: Hailey <me@haileyok.com>zio/stable
parent
2b7d796ca9
commit
bcd3678067
|
@ -17,12 +17,14 @@ export function Error({
|
||||||
message,
|
message,
|
||||||
onRetry,
|
onRetry,
|
||||||
onGoBack: onGoBackProp,
|
onGoBack: onGoBackProp,
|
||||||
|
hideBackButton,
|
||||||
sideBorders = true,
|
sideBorders = true,
|
||||||
}: {
|
}: {
|
||||||
title?: string
|
title?: string
|
||||||
message?: string
|
message?: string
|
||||||
onRetry?: () => unknown
|
onRetry?: () => unknown
|
||||||
onGoBack?: () => unknown
|
onGoBack?: () => unknown
|
||||||
|
hideBackButton?: boolean
|
||||||
sideBorders?: boolean
|
sideBorders?: boolean
|
||||||
}) {
|
}) {
|
||||||
const navigation = useNavigation<NavigationProp>()
|
const navigation = useNavigation<NavigationProp>()
|
||||||
|
@ -89,6 +91,7 @@ export function Error({
|
||||||
</ButtonText>
|
</ButtonText>
|
||||||
</Button>
|
</Button>
|
||||||
)}
|
)}
|
||||||
|
{!hideBackButton && (
|
||||||
<Button
|
<Button
|
||||||
variant="solid"
|
variant="solid"
|
||||||
color={onRetry ? 'secondary' : 'primary'}
|
color={onRetry ? 'secondary' : 'primary'}
|
||||||
|
@ -100,6 +103,7 @@ export function Error({
|
||||||
<Trans>Go Back</Trans>
|
<Trans>Go Back</Trans>
|
||||||
</ButtonText>
|
</ButtonText>
|
||||||
</Button>
|
</Button>
|
||||||
|
)}
|
||||||
</View>
|
</View>
|
||||||
</CenteredView>
|
</CenteredView>
|
||||||
)
|
)
|
||||||
|
|
|
@ -134,6 +134,7 @@ let ListMaybePlaceholder = ({
|
||||||
emptyType = 'page',
|
emptyType = 'page',
|
||||||
onRetry,
|
onRetry,
|
||||||
onGoBack,
|
onGoBack,
|
||||||
|
hideBackButton,
|
||||||
sideBorders,
|
sideBorders,
|
||||||
}: {
|
}: {
|
||||||
isLoading: boolean
|
isLoading: boolean
|
||||||
|
@ -146,6 +147,7 @@ let ListMaybePlaceholder = ({
|
||||||
emptyType?: 'page' | 'results'
|
emptyType?: 'page' | 'results'
|
||||||
onRetry?: () => Promise<unknown>
|
onRetry?: () => Promise<unknown>
|
||||||
onGoBack?: () => void
|
onGoBack?: () => void
|
||||||
|
hideBackButton?: boolean
|
||||||
sideBorders?: boolean
|
sideBorders?: boolean
|
||||||
}): React.ReactNode => {
|
}): React.ReactNode => {
|
||||||
const t = useTheme()
|
const t = useTheme()
|
||||||
|
@ -179,6 +181,7 @@ let ListMaybePlaceholder = ({
|
||||||
onRetry={onRetry}
|
onRetry={onRetry}
|
||||||
onGoBack={onGoBack}
|
onGoBack={onGoBack}
|
||||||
sideBorders={sideBorders}
|
sideBorders={sideBorders}
|
||||||
|
hideBackButton={hideBackButton}
|
||||||
/>
|
/>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
@ -198,6 +201,7 @@ let ListMaybePlaceholder = ({
|
||||||
}
|
}
|
||||||
onRetry={onRetry}
|
onRetry={onRetry}
|
||||||
onGoBack={onGoBack}
|
onGoBack={onGoBack}
|
||||||
|
hideBackButton={hideBackButton}
|
||||||
sideBorders={sideBorders}
|
sideBorders={sideBorders}
|
||||||
/>
|
/>
|
||||||
)
|
)
|
||||||
|
|
|
@ -0,0 +1,233 @@
|
||||||
|
import React, {useCallback, useMemo, useRef, useState} from 'react'
|
||||||
|
import {Keyboard, View} from 'react-native'
|
||||||
|
import {AppBskyActorDefs, moderateProfile} from '@atproto/api'
|
||||||
|
import {BottomSheetFlatListMethods} from '@discord/bottom-sheet'
|
||||||
|
import {msg, Trans} from '@lingui/macro'
|
||||||
|
import {useLingui} from '@lingui/react'
|
||||||
|
|
||||||
|
import {sanitizeDisplayName} from '#/lib/strings/display-names'
|
||||||
|
import {sanitizeHandle} from '#/lib/strings/handles'
|
||||||
|
import {isWeb} from '#/platform/detection'
|
||||||
|
import {useModerationOpts} from '#/state/queries/preferences'
|
||||||
|
import {useActorAutocompleteQuery} from 'state/queries/actor-autocomplete'
|
||||||
|
import {FAB} from '#/view/com/util/fab/FAB'
|
||||||
|
import * as Toast from '#/view/com/util/Toast'
|
||||||
|
import {UserAvatar} from '#/view/com/util/UserAvatar'
|
||||||
|
import {atoms as a, useTheme, web} from '#/alf'
|
||||||
|
import * as Dialog from '#/components/Dialog'
|
||||||
|
import * as TextField from '#/components/forms/TextField'
|
||||||
|
import {MagnifyingGlass2_Stroke2_Corner0_Rounded as Search} from '#/components/icons/MagnifyingGlass2'
|
||||||
|
import {useGetChatFromMembers} from '../../screens/Messages/Temp/query/query'
|
||||||
|
import {Button} from '../Button'
|
||||||
|
import {Envelope_Stroke2_Corner0_Rounded as Envelope} from '../icons/Envelope'
|
||||||
|
import {ListMaybePlaceholder} from '../Lists'
|
||||||
|
import {Text} from '../Typography'
|
||||||
|
|
||||||
|
export function NewChat({onNewChat}: {onNewChat: (chatId: string) => void}) {
|
||||||
|
const control = Dialog.useDialogControl()
|
||||||
|
const t = useTheme()
|
||||||
|
const {_} = useLingui()
|
||||||
|
|
||||||
|
const {mutate: createChat} = useGetChatFromMembers({
|
||||||
|
onSuccess: data => {
|
||||||
|
onNewChat(data.chat.id)
|
||||||
|
},
|
||||||
|
onError: error => {
|
||||||
|
Toast.show(error.message)
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
const onCreateChat = useCallback(
|
||||||
|
(did: string) => {
|
||||||
|
control.close(() => createChat([did]))
|
||||||
|
},
|
||||||
|
[control, createChat],
|
||||||
|
)
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<FAB
|
||||||
|
testID="newChatFAB"
|
||||||
|
onPress={control.open}
|
||||||
|
icon={<Envelope size="xl" fill={t.palette.white} />}
|
||||||
|
accessibilityRole="button"
|
||||||
|
accessibilityLabel={_(msg`New chat`)}
|
||||||
|
accessibilityHint=""
|
||||||
|
/>
|
||||||
|
|
||||||
|
<Dialog.Outer
|
||||||
|
control={control}
|
||||||
|
testID="newChatDialog"
|
||||||
|
nativeOptions={{sheet: {snapPoints: ['100%']}}}>
|
||||||
|
<Dialog.Handle />
|
||||||
|
<SearchablePeopleList onCreateChat={onCreateChat} />
|
||||||
|
</Dialog.Outer>
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
function SearchablePeopleList({
|
||||||
|
onCreateChat,
|
||||||
|
}: {
|
||||||
|
onCreateChat: (did: string) => void
|
||||||
|
}) {
|
||||||
|
const t = useTheme()
|
||||||
|
const {_} = useLingui()
|
||||||
|
const moderationOpts = useModerationOpts()
|
||||||
|
const control = Dialog.useDialogContext()
|
||||||
|
const listRef = useRef<BottomSheetFlatListMethods>(null)
|
||||||
|
|
||||||
|
const [searchText, setSearchText] = useState('')
|
||||||
|
|
||||||
|
const {
|
||||||
|
data: actorAutocompleteData,
|
||||||
|
isFetching,
|
||||||
|
isError,
|
||||||
|
refetch,
|
||||||
|
} = useActorAutocompleteQuery(searchText, true)
|
||||||
|
|
||||||
|
const renderItem = useCallback(
|
||||||
|
({item: profile}: {item: AppBskyActorDefs.ProfileView}) => {
|
||||||
|
if (!moderationOpts) return null
|
||||||
|
const moderation = moderateProfile(profile, moderationOpts)
|
||||||
|
return (
|
||||||
|
<Button
|
||||||
|
label={profile.displayName || sanitizeHandle(profile.handle)}
|
||||||
|
onPress={() => onCreateChat(profile.did)}>
|
||||||
|
{({hovered, pressed}) => (
|
||||||
|
<View
|
||||||
|
style={[
|
||||||
|
a.flex_1,
|
||||||
|
a.px_md,
|
||||||
|
a.py_sm,
|
||||||
|
a.gap_md,
|
||||||
|
a.align_center,
|
||||||
|
a.flex_row,
|
||||||
|
a.rounded_sm,
|
||||||
|
pressed
|
||||||
|
? t.atoms.bg_contrast_25
|
||||||
|
: hovered
|
||||||
|
? t.atoms.bg_contrast_50
|
||||||
|
: t.atoms.bg,
|
||||||
|
]}>
|
||||||
|
<UserAvatar
|
||||||
|
size={40}
|
||||||
|
avatar={profile.avatar}
|
||||||
|
moderation={moderation.ui('avatar')}
|
||||||
|
type={profile.associated?.labeler ? 'labeler' : 'user'}
|
||||||
|
/>
|
||||||
|
<View style={{flex: 1}}>
|
||||||
|
<Text
|
||||||
|
style={[t.atoms.text, a.font_bold, a.leading_snug]}
|
||||||
|
numberOfLines={1}>
|
||||||
|
{sanitizeDisplayName(
|
||||||
|
profile.displayName || sanitizeHandle(profile.handle),
|
||||||
|
moderation.ui('displayName'),
|
||||||
|
)}
|
||||||
|
</Text>
|
||||||
|
<Text style={t.atoms.text_contrast_high} numberOfLines={1}>
|
||||||
|
{sanitizeHandle(profile.handle, '@')}
|
||||||
|
</Text>
|
||||||
|
</View>
|
||||||
|
</View>
|
||||||
|
)}
|
||||||
|
</Button>
|
||||||
|
)
|
||||||
|
},
|
||||||
|
[
|
||||||
|
moderationOpts,
|
||||||
|
onCreateChat,
|
||||||
|
t.atoms.bg_contrast_25,
|
||||||
|
t.atoms.bg_contrast_50,
|
||||||
|
t.atoms.bg,
|
||||||
|
t.atoms.text,
|
||||||
|
t.atoms.text_contrast_high,
|
||||||
|
],
|
||||||
|
)
|
||||||
|
|
||||||
|
const listHeader = useMemo(() => {
|
||||||
|
return (
|
||||||
|
<View style={[a.relative, a.mb_lg]}>
|
||||||
|
{/* cover top corners */}
|
||||||
|
<View
|
||||||
|
style={[
|
||||||
|
a.absolute,
|
||||||
|
a.inset_0,
|
||||||
|
{
|
||||||
|
borderBottomLeftRadius: 8,
|
||||||
|
borderBottomRightRadius: 8,
|
||||||
|
},
|
||||||
|
t.atoms.bg,
|
||||||
|
]}
|
||||||
|
/>
|
||||||
|
<Dialog.Close />
|
||||||
|
<Text
|
||||||
|
style={[
|
||||||
|
a.text_2xl,
|
||||||
|
a.font_bold,
|
||||||
|
a.leading_tight,
|
||||||
|
a.pb_lg,
|
||||||
|
web(a.pt_lg),
|
||||||
|
]}>
|
||||||
|
<Trans>Start a new chat</Trans>
|
||||||
|
</Text>
|
||||||
|
<TextField.Root>
|
||||||
|
<TextField.Icon icon={Search} />
|
||||||
|
<TextField.Input
|
||||||
|
label={_(msg`Search profiles`)}
|
||||||
|
placeholder={_(msg`Search`)}
|
||||||
|
value={searchText}
|
||||||
|
onChangeText={text => {
|
||||||
|
setSearchText(text)
|
||||||
|
listRef.current?.scrollToOffset({offset: 0, animated: false})
|
||||||
|
}}
|
||||||
|
returnKeyType="search"
|
||||||
|
clearButtonMode="while-editing"
|
||||||
|
maxLength={50}
|
||||||
|
onKeyPress={({nativeEvent}) => {
|
||||||
|
if (nativeEvent.key === 'Escape') {
|
||||||
|
control.close()
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
autoCorrect={false}
|
||||||
|
autoComplete="off"
|
||||||
|
autoCapitalize="none"
|
||||||
|
/>
|
||||||
|
</TextField.Root>
|
||||||
|
</View>
|
||||||
|
)
|
||||||
|
}, [t.atoms.bg, _, control, searchText])
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Dialog.InnerFlatList
|
||||||
|
ref={listRef}
|
||||||
|
data={actorAutocompleteData}
|
||||||
|
renderItem={renderItem}
|
||||||
|
ListHeaderComponent={
|
||||||
|
<>
|
||||||
|
{listHeader}
|
||||||
|
{searchText.length > 0 && !actorAutocompleteData?.length && (
|
||||||
|
<ListMaybePlaceholder
|
||||||
|
isLoading={isFetching}
|
||||||
|
isError={isError}
|
||||||
|
onRetry={refetch}
|
||||||
|
hideBackButton={true}
|
||||||
|
emptyType="results"
|
||||||
|
sideBorders={false}
|
||||||
|
emptyMessage={
|
||||||
|
isError
|
||||||
|
? _(msg`No search results found for "${searchText}".`)
|
||||||
|
: _(msg`Could not load profiles. Please try again later.`)
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
}
|
||||||
|
stickyHeaderIndices={[0]}
|
||||||
|
keyExtractor={(item: AppBskyActorDefs.ProfileView) => item.did}
|
||||||
|
// @ts-expect-error web only
|
||||||
|
style={isWeb && {minHeight: '100vh'}}
|
||||||
|
onScrollBeginDrag={() => Keyboard.dismiss()}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
|
@ -1,12 +1,16 @@
|
||||||
import React from 'react'
|
import React from 'react'
|
||||||
import {View} from 'react-native'
|
import {View} from 'react-native'
|
||||||
|
|
||||||
|
import {useAgent} from '#/state/session'
|
||||||
import {atoms as a, useTheme} from '#/alf'
|
import {atoms as a, useTheme} from '#/alf'
|
||||||
import {Text} from '#/components/Typography'
|
import {Text} from '#/components/Typography'
|
||||||
import * as TempDmChatDefs from '#/temp/dm/defs'
|
import * as TempDmChatDefs from '#/temp/dm/defs'
|
||||||
|
|
||||||
export function MessageItem({item}: {item: TempDmChatDefs.MessageView}) {
|
export function MessageItem({item}: {item: TempDmChatDefs.MessageView}) {
|
||||||
const t = useTheme()
|
const t = useTheme()
|
||||||
|
const {getAgent} = useAgent()
|
||||||
|
|
||||||
|
const fromMe = item.sender?.did === getAgent().session?.did
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<View
|
<View
|
||||||
|
@ -15,13 +19,17 @@ export function MessageItem({item}: {item: TempDmChatDefs.MessageView}) {
|
||||||
a.px_md,
|
a.px_md,
|
||||||
a.my_xs,
|
a.my_xs,
|
||||||
a.rounded_md,
|
a.rounded_md,
|
||||||
|
fromMe ? a.self_end : a.self_start,
|
||||||
{
|
{
|
||||||
backgroundColor: t.palette.primary_500,
|
backgroundColor: fromMe
|
||||||
|
? t.palette.primary_500
|
||||||
|
: t.palette.contrast_50,
|
||||||
maxWidth: '65%',
|
maxWidth: '65%',
|
||||||
borderRadius: 17,
|
borderRadius: 17,
|
||||||
},
|
},
|
||||||
]}>
|
]}>
|
||||||
<Text style={[a.text_md, {lineHeight: 1.2, color: 'white'}]}>
|
<Text
|
||||||
|
style={[a.text_md, a.leading_snug, fromMe && {color: t.palette.white}]}>
|
||||||
{item.text}
|
{item.text}
|
||||||
</Text>
|
</Text>
|
||||||
</View>
|
</View>
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
import React, {useCallback, useMemo, useRef, useState} from 'react'
|
import React, {useCallback, useMemo, useRef, useState} from 'react'
|
||||||
import {Alert, FlatList, View, ViewToken} from 'react-native'
|
import {FlatList, View, ViewToken} from 'react-native'
|
||||||
|
import {Alert} from 'react-native'
|
||||||
import {KeyboardAvoidingView} from 'react-native-keyboard-controller'
|
import {KeyboardAvoidingView} from 'react-native-keyboard-controller'
|
||||||
|
|
||||||
import {isWeb} from 'platform/detection'
|
import {isWeb} from 'platform/detection'
|
||||||
|
@ -64,6 +65,7 @@ export function MessagesList({chatId}: {chatId: string}) {
|
||||||
const totalMessages = useRef(10)
|
const totalMessages = useRef(10)
|
||||||
|
|
||||||
// TODO later
|
// TODO later
|
||||||
|
|
||||||
const [_, setShowSpinner] = useState(false)
|
const [_, setShowSpinner] = useState(false)
|
||||||
|
|
||||||
// Query Data
|
// Query Data
|
||||||
|
@ -147,6 +149,8 @@ export function MessagesList({chatId}: {chatId: string}) {
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
totalMessages.current = filtered.length
|
totalMessages.current = filtered.length
|
||||||
|
|
||||||
|
return filtered
|
||||||
}, [chat])
|
}, [chat])
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
@ -162,7 +166,7 @@ export function MessagesList({chatId}: {chatId: string}) {
|
||||||
contentContainerStyle={{paddingHorizontal: 10}}
|
contentContainerStyle={{paddingHorizontal: 10}}
|
||||||
// In the future, we might want to adjust this value. Not very concerning right now as long as we are only
|
// In the future, we might want to adjust this value. Not very concerning right now as long as we are only
|
||||||
// dealing with text. But whenever we have images or other media and things are taller, we will want to lower
|
// dealing with text. But whenever we have images or other media and things are taller, we will want to lower
|
||||||
// this...probably
|
// this...probably.
|
||||||
initialNumToRender={20}
|
initialNumToRender={20}
|
||||||
// Same with the max to render per batch. Let's be safe for now though.
|
// Same with the max to render per batch. Let's be safe for now though.
|
||||||
maxToRenderPerBatch={25}
|
maxToRenderPerBatch={25}
|
||||||
|
@ -175,7 +179,6 @@ export function MessagesList({chatId}: {chatId: string}) {
|
||||||
maintainVisibleContentPosition={{
|
maintainVisibleContentPosition={{
|
||||||
minIndexForVisible: 0,
|
minIndexForVisible: 0,
|
||||||
}}
|
}}
|
||||||
// This is actually a header since we are inverted!
|
|
||||||
ListFooterComponent={<MaybeLoader isLoading={false} />}
|
ListFooterComponent={<MaybeLoader isLoading={false} />}
|
||||||
removeClippedSubviews={true}
|
removeClippedSubviews={true}
|
||||||
ref={flatListRef}
|
ref={flatListRef}
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
import React, {useCallback, useState} from 'react'
|
import React, {useCallback, useMemo, useState} from 'react'
|
||||||
import {View} from 'react-native'
|
import {View} from 'react-native'
|
||||||
import {msg} from '@lingui/macro'
|
import {msg} from '@lingui/macro'
|
||||||
import {useLingui} from '@lingui/react'
|
import {useLingui} from '@lingui/react'
|
||||||
|
@ -20,10 +20,11 @@ import {SettingsSliderVertical_Stroke2_Corner0_Rounded as SettingsSlider} from '
|
||||||
import {Link} from '#/components/Link'
|
import {Link} from '#/components/Link'
|
||||||
import {ListFooter, ListMaybePlaceholder} from '#/components/Lists'
|
import {ListFooter, ListMaybePlaceholder} from '#/components/Lists'
|
||||||
import {Text} from '#/components/Typography'
|
import {Text} from '#/components/Typography'
|
||||||
|
import {NewChat} from '../../../components/dms/NewChat'
|
||||||
import {ClipClopGate} from '../gate'
|
import {ClipClopGate} from '../gate'
|
||||||
|
|
||||||
type Props = NativeStackScreenProps<MessagesTabNavigatorParams, 'MessagesList'>
|
type Props = NativeStackScreenProps<MessagesTabNavigatorParams, 'MessagesList'>
|
||||||
export function MessagesListScreen({}: Props) {
|
export function MessagesListScreen({navigation}: Props) {
|
||||||
const {_} = useLingui()
|
const {_} = useLingui()
|
||||||
const t = useTheme()
|
const t = useTheme()
|
||||||
|
|
||||||
|
@ -53,14 +54,14 @@ export function MessagesListScreen({}: Props) {
|
||||||
|
|
||||||
const isError = !!error
|
const isError = !!error
|
||||||
|
|
||||||
const conversations = React.useMemo(() => {
|
const conversations = useMemo(() => {
|
||||||
if (data?.pages) {
|
if (data?.pages) {
|
||||||
return data.pages.flat()
|
return data.pages.flat()
|
||||||
}
|
}
|
||||||
return []
|
return []
|
||||||
}, [data])
|
}, [data])
|
||||||
|
|
||||||
const onRefresh = React.useCallback(async () => {
|
const onRefresh = useCallback(async () => {
|
||||||
setIsPTRing(true)
|
setIsPTRing(true)
|
||||||
try {
|
try {
|
||||||
await refetch()
|
await refetch()
|
||||||
|
@ -70,7 +71,7 @@ export function MessagesListScreen({}: Props) {
|
||||||
setIsPTRing(false)
|
setIsPTRing(false)
|
||||||
}, [refetch, setIsPTRing])
|
}, [refetch, setIsPTRing])
|
||||||
|
|
||||||
const onEndReached = React.useCallback(async () => {
|
const onEndReached = useCallback(async () => {
|
||||||
if (isFetchingNextPage || !hasNextPage || isError) return
|
if (isFetchingNextPage || !hasNextPage || isError) return
|
||||||
try {
|
try {
|
||||||
await fetchNextPage()
|
await fetchNextPage()
|
||||||
|
@ -79,11 +80,18 @@ export function MessagesListScreen({}: Props) {
|
||||||
}
|
}
|
||||||
}, [isFetchingNextPage, hasNextPage, isError, fetchNextPage])
|
}, [isFetchingNextPage, hasNextPage, isError, fetchNextPage])
|
||||||
|
|
||||||
|
const onNewChat = useCallback(
|
||||||
|
(conversation: string) =>
|
||||||
|
navigation.navigate('MessagesConversation', {conversation}),
|
||||||
|
[navigation],
|
||||||
|
)
|
||||||
|
|
||||||
const gate = useGate()
|
const gate = useGate()
|
||||||
if (!gate('dms')) return <ClipClopGate />
|
if (!gate('dms')) return <ClipClopGate />
|
||||||
|
|
||||||
if (conversations.length < 1) {
|
if (conversations.length < 1) {
|
||||||
return (
|
return (
|
||||||
|
<>
|
||||||
<ListMaybePlaceholder
|
<ListMaybePlaceholder
|
||||||
isLoading={isLoading}
|
isLoading={isLoading}
|
||||||
isError={isError}
|
isError={isError}
|
||||||
|
@ -94,11 +102,13 @@ export function MessagesListScreen({}: Props) {
|
||||||
errorMessage={cleanError(error)}
|
errorMessage={cleanError(error)}
|
||||||
onRetry={isError ? refetch : undefined}
|
onRetry={isError ? refetch : undefined}
|
||||||
/>
|
/>
|
||||||
|
<NewChat onNewChat={onNewChat} />
|
||||||
|
</>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<View>
|
<View style={a.flex_1}>
|
||||||
<ViewHeader
|
<ViewHeader
|
||||||
title={_(msg`Messages`)}
|
title={_(msg`Messages`)}
|
||||||
showOnDesktop
|
showOnDesktop
|
||||||
|
@ -106,6 +116,7 @@ export function MessagesListScreen({}: Props) {
|
||||||
showBorder
|
showBorder
|
||||||
canGoBack={false}
|
canGoBack={false}
|
||||||
/>
|
/>
|
||||||
|
<NewChat onNewChat={onNewChat} />
|
||||||
<List
|
<List
|
||||||
data={conversations}
|
data={conversations}
|
||||||
renderItem={({item}) => {
|
renderItem={({item}) => {
|
||||||
|
|
|
@ -1,20 +1,24 @@
|
||||||
import {useMutation, useQuery, useQueryClient} from '@tanstack/react-query'
|
import {useMutation, useQuery, useQueryClient} from '@tanstack/react-query'
|
||||||
|
|
||||||
import {useSession} from 'state/session'
|
import {useAgent} from '#/state/session'
|
||||||
import {useDmServiceUrlStorage} from '#/screens/Messages/Temp/useDmServiceUrlStorage'
|
|
||||||
import * as TempDmChatDefs from '#/temp/dm/defs'
|
import * as TempDmChatDefs from '#/temp/dm/defs'
|
||||||
import * as TempDmChatGetChat from '#/temp/dm/getChat'
|
import * as TempDmChatGetChat from '#/temp/dm/getChat'
|
||||||
|
import * as TempDmChatGetChatForMembers from '#/temp/dm/getChatForMembers'
|
||||||
import * as TempDmChatGetChatLog from '#/temp/dm/getChatLog'
|
import * as TempDmChatGetChatLog from '#/temp/dm/getChatLog'
|
||||||
import * as TempDmChatGetChatMessages from '#/temp/dm/getChatMessages'
|
import * as TempDmChatGetChatMessages from '#/temp/dm/getChatMessages'
|
||||||
|
import {useDmServiceUrlStorage} from '../useDmServiceUrlStorage'
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* TEMPORARY, PLEASE DO NOT JUDGE ME REACT QUERY OVERLORDS 🙏
|
* TEMPORARY, PLEASE DO NOT JUDGE ME REACT QUERY OVERLORDS 🙏
|
||||||
* (and do not try this at home)
|
* (and do not try this at home)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
function createHeaders(did: string) {
|
const useHeaders = () => {
|
||||||
|
const {getAgent} = useAgent()
|
||||||
return {
|
return {
|
||||||
Authorization: did,
|
get Authorization() {
|
||||||
|
return getAgent().session!.did
|
||||||
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -27,10 +31,8 @@ type Chat = {
|
||||||
|
|
||||||
export function useChat(chatId: string) {
|
export function useChat(chatId: string) {
|
||||||
const queryClient = useQueryClient()
|
const queryClient = useQueryClient()
|
||||||
|
const headers = useHeaders()
|
||||||
const {serviceUrl} = useDmServiceUrlStorage()
|
const {serviceUrl} = useDmServiceUrlStorage()
|
||||||
const {currentAccount} = useSession()
|
|
||||||
const did = currentAccount?.did ?? ''
|
|
||||||
|
|
||||||
return useQuery({
|
return useQuery({
|
||||||
queryKey: ['chat', chatId],
|
queryKey: ['chat', chatId],
|
||||||
|
@ -44,7 +46,7 @@ export function useChat(chatId: string) {
|
||||||
const messagesResponse = await fetch(
|
const messagesResponse = await fetch(
|
||||||
`${serviceUrl}/xrpc/temp.dm.getChatMessages?chatId=${chatId}`,
|
`${serviceUrl}/xrpc/temp.dm.getChatMessages?chatId=${chatId}`,
|
||||||
{
|
{
|
||||||
headers: createHeaders(did),
|
headers,
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -56,7 +58,7 @@ export function useChat(chatId: string) {
|
||||||
const chatResponse = await fetch(
|
const chatResponse = await fetch(
|
||||||
`${serviceUrl}/xrpc/temp.dm.getChat?chatId=${chatId}`,
|
`${serviceUrl}/xrpc/temp.dm.getChat?chatId=${chatId}`,
|
||||||
{
|
{
|
||||||
headers: createHeaders(did),
|
headers,
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -90,10 +92,8 @@ export function createTempId() {
|
||||||
|
|
||||||
export function useSendMessageMutation(chatId: string) {
|
export function useSendMessageMutation(chatId: string) {
|
||||||
const queryClient = useQueryClient()
|
const queryClient = useQueryClient()
|
||||||
|
const headers = useHeaders()
|
||||||
const {serviceUrl} = useDmServiceUrlStorage()
|
const {serviceUrl} = useDmServiceUrlStorage()
|
||||||
const {currentAccount} = useSession()
|
|
||||||
const did = currentAccount?.did ?? ''
|
|
||||||
|
|
||||||
return useMutation<
|
return useMutation<
|
||||||
TempDmChatDefs.Message,
|
TempDmChatDefs.Message,
|
||||||
|
@ -108,7 +108,7 @@ export function useSendMessageMutation(chatId: string) {
|
||||||
{
|
{
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
headers: {
|
headers: {
|
||||||
...createHeaders(did),
|
...headers,
|
||||||
'Content-Type': 'application/json',
|
'Content-Type': 'application/json',
|
||||||
},
|
},
|
||||||
body: JSON.stringify({
|
body: JSON.stringify({
|
||||||
|
@ -130,8 +130,10 @@ export function useSendMessageMutation(chatId: string) {
|
||||||
...prev,
|
...prev,
|
||||||
messages: [
|
messages: [
|
||||||
{
|
{
|
||||||
|
$type: 'temp.dm.defs#messageView',
|
||||||
id: variables.tempId,
|
id: variables.tempId,
|
||||||
text: variables.message,
|
text: variables.message,
|
||||||
|
sender: {did: headers.Authorization}, // TODO a real DID get
|
||||||
},
|
},
|
||||||
...prev.messages,
|
...prev.messages,
|
||||||
],
|
],
|
||||||
|
@ -165,10 +167,8 @@ export function useSendMessageMutation(chatId: string) {
|
||||||
|
|
||||||
export function useChatLogQuery() {
|
export function useChatLogQuery() {
|
||||||
const queryClient = useQueryClient()
|
const queryClient = useQueryClient()
|
||||||
|
const headers = useHeaders()
|
||||||
const {serviceUrl} = useDmServiceUrlStorage()
|
const {serviceUrl} = useDmServiceUrlStorage()
|
||||||
const {currentAccount} = useSession()
|
|
||||||
const did = currentAccount?.did ?? ''
|
|
||||||
|
|
||||||
return useQuery({
|
return useQuery({
|
||||||
queryKey: ['chatLog'],
|
queryKey: ['chatLog'],
|
||||||
|
@ -183,7 +183,7 @@ export function useChatLogQuery() {
|
||||||
prevLog?.cursor ?? ''
|
prevLog?.cursor ?? ''
|
||||||
}`,
|
}`,
|
||||||
{
|
{
|
||||||
headers: createHeaders(did),
|
headers,
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -193,13 +193,10 @@ export function useChatLogQuery() {
|
||||||
(await response.json()) as TempDmChatGetChatLog.OutputSchema
|
(await response.json()) as TempDmChatGetChatLog.OutputSchema
|
||||||
|
|
||||||
for (const log of json.logs) {
|
for (const log of json.logs) {
|
||||||
if (TempDmChatDefs.isLogDeleteMessage(log)) {
|
if (TempDmChatDefs.isLogCreateMessage(log)) {
|
||||||
queryClient.setQueryData(['chat', log.chatId], (prev: Chat) => {
|
queryClient.setQueryData(['chat', log.chatId], (prev: Chat) => {
|
||||||
// What to do in this case
|
// TODO hack filter out duplicates
|
||||||
if (!prev) return
|
if (prev?.messages.find(m => m.id === log.message.id)) return
|
||||||
|
|
||||||
// HACK we don't know who the creator of a message is, so just filter by id for now
|
|
||||||
if (prev.messages.find(m => m.id === log.message.id)) return prev
|
|
||||||
|
|
||||||
return {
|
return {
|
||||||
...prev,
|
...prev,
|
||||||
|
@ -217,3 +214,39 @@ export function useChatLogQuery() {
|
||||||
refetchInterval: 5000,
|
refetchInterval: 5000,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function useGetChatFromMembers({
|
||||||
|
onSuccess,
|
||||||
|
onError,
|
||||||
|
}: {
|
||||||
|
onSuccess?: (data: TempDmChatGetChatForMembers.OutputSchema) => void
|
||||||
|
onError?: (error: Error) => void
|
||||||
|
}) {
|
||||||
|
const queryClient = useQueryClient()
|
||||||
|
const headers = useHeaders()
|
||||||
|
const {serviceUrl} = useDmServiceUrlStorage()
|
||||||
|
|
||||||
|
return useMutation({
|
||||||
|
mutationFn: async (members: string[]) => {
|
||||||
|
const response = await fetch(
|
||||||
|
`${serviceUrl}/xrpc/temp.dm.getChatForMembers?members=${members.join(
|
||||||
|
',',
|
||||||
|
)}`,
|
||||||
|
{headers},
|
||||||
|
)
|
||||||
|
|
||||||
|
if (!response.ok) throw new Error('Failed to fetch chat')
|
||||||
|
|
||||||
|
return (await response.json()) as TempDmChatGetChatForMembers.OutputSchema
|
||||||
|
},
|
||||||
|
onSuccess: data => {
|
||||||
|
queryClient.setQueryData(['chat', data.chat.id], {
|
||||||
|
chatId: data.chat.id,
|
||||||
|
messages: [],
|
||||||
|
lastRev: data.chat.rev,
|
||||||
|
})
|
||||||
|
onSuccess?.(data)
|
||||||
|
},
|
||||||
|
onError,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
|
@ -791,7 +791,7 @@ export function SettingsScreen({}: Props) {
|
||||||
<TextField.Input
|
<TextField.Input
|
||||||
value={dmServiceUrl}
|
value={dmServiceUrl}
|
||||||
onChangeText={(text: string) => {
|
onChangeText={(text: string) => {
|
||||||
if (text.endsWith('/')) {
|
if (text.length > 9 && text.endsWith('/')) {
|
||||||
text = text.slice(0, -1)
|
text = text.slice(0, -1)
|
||||||
}
|
}
|
||||||
setDmServiceUrl(text)
|
setDmServiceUrl(text)
|
||||||
|
|
Loading…
Reference in New Issue