1
+ import { useCallback } from "react"
2
+ import { useNavigation } from "@react-navigation/native"
3
+ import { captureError } from "@/utils/capture-error"
4
+ import { GenericError } from "@/utils/error"
5
+ import { IXmtpInboxId , IXmtpConversationId } from "@/features/xmtp/xmtp.types"
6
+ import { logger } from "@/utils/logger"
7
+ import { checkConversationExists } from "./conversation-links"
8
+
9
+ /**
10
+ * Custom hook to handle conversation deep links
11
+ * This is used by the DeepLinkHandler component to navigate to conversations from deep links
12
+ */
13
+ export function useConversationDeepLinkHandler ( ) {
14
+ const navigation = useNavigation ( )
15
+
16
+ /**
17
+ * Process an inbox ID from a deep link and navigate to the appropriate conversation
18
+ * @param inboxId The inbox ID from the deep link
19
+ * @param composerTextPrefill Optional text to prefill in the composer
20
+ */
21
+ const handleConversationDeepLink = useCallback ( async (
22
+ inboxId : IXmtpInboxId ,
23
+ composerTextPrefill ?: string
24
+ ) => {
25
+ if ( ! inboxId ) {
26
+ logger . warn ( "Cannot handle conversation deep link - missing inboxId" )
27
+ return
28
+ }
29
+
30
+ try {
31
+ logger . info ( `Handling conversation deep link for inboxId: ${ inboxId } ${ composerTextPrefill ? ' with prefill text' : '' } ` )
32
+
33
+ // Check if the conversation exists
34
+ const { exists, conversationId } = await checkConversationExists ( inboxId )
35
+
36
+ if ( exists && conversationId ) {
37
+ // We have an existing conversation - navigate to it
38
+ logger . info ( `Found existing conversation with ID: ${ conversationId } ` )
39
+
40
+ navigation . navigate ( "Conversation" , {
41
+ xmtpConversationId : conversationId ,
42
+ isNew : false ,
43
+ composerTextPrefill
44
+ } )
45
+ } else {
46
+ // No existing conversation - start a new one
47
+ logger . info ( `No existing conversation found with inboxId: ${ inboxId } , creating new conversation` )
48
+
49
+ navigation . navigate ( "Conversation" , {
50
+ searchSelectedUserInboxIds : [ inboxId ] ,
51
+ isNew : true ,
52
+ composerTextPrefill
53
+ } )
54
+ }
55
+ } catch ( error ) {
56
+ captureError (
57
+ new GenericError ( {
58
+ error,
59
+ additionalMessage : `Failed to handle conversation deep link for inboxId: ${ inboxId } ` ,
60
+ extra : { inboxId }
61
+ } )
62
+ )
63
+ }
64
+ } , [ navigation ] )
65
+
66
+ return { handleConversationDeepLink }
67
+ }
68
+
69
+ /**
70
+ * Global function to process a new deep link that uses the ConversationScreenConfig format
71
+ * This function is called by the navigation library when it receives a deep link
72
+ */
73
+ export function processConversationDeepLink (
74
+ params : Record < string , string | undefined >
75
+ ) : Promise < boolean > {
76
+ return new Promise ( async ( resolve ) => {
77
+ const { inboxId, composerTextPrefill } = params
78
+
79
+ if ( ! inboxId ) {
80
+ // Skip if no inboxId
81
+ logger . warn ( "Cannot process conversation deep link - missing inboxId" )
82
+ resolve ( false )
83
+ return
84
+ }
85
+
86
+ try {
87
+ logger . info ( `Processing Conversation deep link via navigation for inboxId: ${ inboxId } ${ composerTextPrefill ? ' with prefill text' : '' } ` )
88
+
89
+ // Check if the conversation exists
90
+ const { exists, conversationId } = await checkConversationExists ( inboxId as IXmtpInboxId )
91
+
92
+ if ( exists && conversationId ) {
93
+ logger . info ( `Navigation found existing conversation with ID: ${ conversationId } ` )
94
+ resolve ( true )
95
+ return
96
+ }
97
+
98
+ // We didn't find an existing conversation, so we'll create a new one
99
+ logger . info ( `No existing conversation found with inboxId: ${ inboxId } , navigation will create a new conversation` )
100
+ resolve ( true )
101
+ } catch ( error ) {
102
+ logger . error ( `Error in processConversationDeepLink: ${ error } ` )
103
+ // Still return true to indicate we're handling it, even though there was an error
104
+ resolve ( true )
105
+ }
106
+ } )
107
+ }
0 commit comments