@@ -41,6 +41,7 @@ export type TransportConnectCallback = () => void;
41
41
export type Transport = {
42
42
send ( data : any ) : void ;
43
43
onMessage ( callback : TransportMessageCallback ) : void ;
44
+ hasMessageListener ( ) : boolean ;
44
45
onConnect ( callback : TransportConnectCallback ) : void ; // Socket connection cb
45
46
clear ( ) : void ;
46
47
address ( ) : string ;
@@ -53,13 +54,13 @@ type Log = (...args: any[]) => void;
53
54
54
55
export class RealtimeListener implements WebxdcRealtimeListener {
55
56
private trashed = false ;
56
- private listener : ( data : Uint8Array ) => void = ( ) => { } ;
57
+ private listener : ( data : Uint8Array ) => void = ( ) => { } ;
57
58
58
59
constructor (
59
- public sendHook : ( data : Uint8Array ) => void = ( ) => { } ,
60
- public setListenerHook : ( ) => void = ( ) => { } ,
61
- private leaveHook : ( ) => void = ( ) => { } ,
62
- ) { }
60
+ public sendHook : ( data : Uint8Array ) => void = ( ) => { } ,
61
+ public setListenerHook : ( ) => void = ( ) => { } ,
62
+ private leaveHook : ( ) => void = ( ) => { } ,
63
+ ) { }
63
64
64
65
is_trashed ( ) : boolean {
65
66
return this . trashed ;
@@ -94,11 +95,11 @@ export class RealtimeListener implements WebxdcRealtimeListener {
94
95
95
96
export function createWebXdc (
96
97
transport : Transport ,
97
- log : Log = ( ) => { } ,
98
+ log : Log = ( ) => { } ,
98
99
) : Webxdc < any > {
99
100
let resolveUpdateListenerPromise : ( ( ) => void ) | null = null ;
100
101
let realtime : RealtimeListener | null = null ;
101
-
102
+
102
103
const webXdc : Webxdc < any > = {
103
104
sendUpdate : ( update ) => {
104
105
transport . send ( { type : "sendUpdate" , update } ) ;
@@ -116,10 +117,11 @@ export function createWebXdc(
116
117
resolveUpdateListenerPromise = null ;
117
118
}
118
119
} else if ( isRealtimeMessage ( message ) ) {
119
- // TODO: move this out of setUpdateListener because otherwise
120
- // You have to set an update listener such that realtime works
121
120
// Conversion to any because the actual data is a dict representation of Uint8Array
122
121
// This is due to JSON.stringify conversion.
122
+ if ( realtime === null ) {
123
+ return
124
+ }
123
125
realtime ! . receive ( new Uint8Array ( Object . values ( message . data as any ) ) ) ;
124
126
} else if ( isClearMessage ( message ) ) {
125
127
log ( "clear" ) ;
@@ -199,13 +201,11 @@ export function createWebXdc(
199
201
) ;
200
202
}
201
203
}
202
- const msg = `The app would now close and the user would select a chat to send this message:\nText: ${
203
- content . text ? `"${ content . text } "` : "No Text"
204
- } \nFile: ${
205
- content . file
204
+ const msg = `The app would now close and the user would select a chat to send this message:\nText: ${ content . text ? `"${ content . text } "` : "No Text"
205
+ } \nFile: ${ content . file
206
206
? `${ content . file . name } - ${ base64Content . length } bytes`
207
207
: "No File"
208
- } `;
208
+ } `;
209
209
if ( content . file ) {
210
210
const confirmed = confirm (
211
211
msg + "\n\nDownload the file in the browser instead?" ,
@@ -249,8 +249,22 @@ export function createWebXdc(
249
249
} ,
250
250
251
251
joinRealtimeChannel : ( ) => {
252
+ if ( ! transport . hasMessageListener ( ) ) {
253
+ // we can only have one message listener with the current implementation,
254
+ // so we need to set it here to receive realtime data. When `setUpdateListener`
255
+ // is called, the callback is overwritten but the new value also looks for
256
+ // realtime data.
257
+ transport . onMessage ( ( message ) => {
258
+ if ( isRealtimeMessage ( message ) ) {
259
+ if ( realtime === null ) {
260
+ return
261
+ }
262
+ realtime ! . receive ( new Uint8Array ( Object . values ( message . data as any ) ) ) ;
263
+ }
264
+ } )
265
+ }
252
266
realtime = new RealtimeListener (
253
- ( ) => { } ,
267
+ ( ) => { } ,
254
268
( ) => {
255
269
transport . send ( { type : "setRealtimeListener" } ) ;
256
270
} ,
0 commit comments