| 
									
										
										
										
											2020-10-30 15:34:04 -05:00
										 |  |  | // Copyright 2014-2020 Signal Messenger, LLC
 | 
					
						
							|  |  |  | // SPDX-License-Identifier: AGPL-3.0-only
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-14 13:49:58 -08:00
										 |  |  | /* global | 
					
						
							|  |  |  |   ConversationController, | 
					
						
							|  |  |  |   i18n, | 
					
						
							|  |  |  |   Whisper, | 
					
						
							| 
									
										
										
										
											2021-02-26 13:06:37 -08:00
										 |  |  |   Signal, | 
					
						
							|  |  |  |   $ | 
					
						
							| 
									
										
										
										
											2019-01-14 13:49:58 -08:00
										 |  |  | */ | 
					
						
							| 
									
										
										
										
											2018-03-02 15:54:15 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | // eslint-disable-next-line func-names
 | 
					
						
							| 
									
										
										
										
											2020-11-18 07:15:42 -08:00
										 |  |  | (function () { | 
					
						
							| 
									
										
										
										
											2018-03-02 15:54:15 -05:00
										 |  |  |   window.Whisper = window.Whisper || {}; | 
					
						
							| 
									
										
										
										
											2014-11-16 15:30:40 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  |   Whisper.StickerPackInstallFailedToast = Whisper.ToastView.extend({ | 
					
						
							|  |  |  |     render_attributes() { | 
					
						
							|  |  |  |       return { toastMessage: i18n('stickers--toast--InstallFailed') }; | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-02 15:54:15 -05:00
										 |  |  |   Whisper.ConversationStack = Whisper.View.extend({ | 
					
						
							|  |  |  |     className: 'conversation-stack', | 
					
						
							| 
									
										
										
										
											2019-06-27 16:35:21 -04:00
										 |  |  |     lastConversation: null, | 
					
						
							| 
									
										
										
										
											2019-05-31 15:42:01 -07:00
										 |  |  |     open(conversation, messageId) { | 
					
						
							| 
									
										
										
										
											2018-03-02 15:54:15 -05:00
										 |  |  |       const id = `conversation-${conversation.cid}`; | 
					
						
							| 
									
										
										
										
											2019-05-31 15:42:01 -07:00
										 |  |  |       if (id !== this.el.lastChild.id) { | 
					
						
							|  |  |  |         const view = new Whisper.ConversationView({ | 
					
						
							|  |  |  |           model: conversation, | 
					
						
							|  |  |  |           window: this.model.window, | 
					
						
							|  |  |  |         }); | 
					
						
							| 
									
										
										
										
											2019-10-17 10:35:35 -07:00
										 |  |  |         this.listenTo(conversation, 'unload', () => | 
					
						
							|  |  |  |           this.onUnload(conversation) | 
					
						
							|  |  |  |         ); | 
					
						
							| 
									
										
										
										
											2019-05-31 15:42:01 -07:00
										 |  |  |         view.$el.appendTo(this.el); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-07 13:36:16 -08:00
										 |  |  |         if (this.lastConversation && this.lastConversation !== conversation) { | 
					
						
							| 
									
										
										
										
											2019-05-31 15:42:01 -07:00
										 |  |  |           this.lastConversation.trigger( | 
					
						
							|  |  |  |             'unload', | 
					
						
							|  |  |  |             'opened another conversation' | 
					
						
							|  |  |  |           ); | 
					
						
							| 
									
										
										
										
											2019-10-17 10:35:35 -07:00
										 |  |  |           this.stopListening(this.lastConversation); | 
					
						
							|  |  |  |           this.lastConversation = null; | 
					
						
							| 
									
										
										
										
											2015-12-07 12:36:30 -08:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-05-31 15:42:01 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         this.lastConversation = conversation; | 
					
						
							|  |  |  |         conversation.trigger('opened', messageId); | 
					
						
							|  |  |  |       } else if (messageId) { | 
					
						
							|  |  |  |         conversation.trigger('scroll-to-message', messageId); | 
					
						
							| 
									
										
										
										
											2019-06-27 16:35:21 -04:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2019-05-31 15:42:01 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  |       // Make sure poppers are positioned properly
 | 
					
						
							|  |  |  |       window.dispatchEvent(new Event('resize')); | 
					
						
							| 
									
										
										
										
											2018-03-02 15:54:15 -05:00
										 |  |  |     }, | 
					
						
							| 
									
										
										
										
											2021-05-07 13:07:24 -07:00
										 |  |  |     unload() { | 
					
						
							|  |  |  |       const { lastConversation } = this; | 
					
						
							|  |  |  |       if (!lastConversation) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       lastConversation.trigger('unload', 'force unload requested'); | 
					
						
							|  |  |  |     }, | 
					
						
							| 
									
										
										
										
											2019-10-17 10:35:35 -07:00
										 |  |  |     onUnload(conversation) { | 
					
						
							|  |  |  |       if (this.lastConversation === conversation) { | 
					
						
							|  |  |  |         this.stopListening(this.lastConversation); | 
					
						
							| 
									
										
										
										
											2019-08-20 12:34:52 -07:00
										 |  |  |         this.lastConversation = null; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }, | 
					
						
							| 
									
										
										
										
											2018-03-02 15:54:15 -05:00
										 |  |  |   }); | 
					
						
							| 
									
										
										
										
											2014-11-16 15:30:40 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-02 15:54:15 -05:00
										 |  |  |   Whisper.AppLoadingScreen = Whisper.View.extend({ | 
					
						
							| 
									
										
										
										
											2021-02-26 13:06:37 -08:00
										 |  |  |     template: () => $('#app-loading-screen').html(), | 
					
						
							| 
									
										
										
										
											2018-03-02 15:54:15 -05:00
										 |  |  |     className: 'app-loading-screen', | 
					
						
							|  |  |  |     updateProgress(count) { | 
					
						
							|  |  |  |       if (count > 0) { | 
					
						
							| 
									
										
										
										
											2020-07-29 16:20:05 -07:00
										 |  |  |         const message = i18n('loadingMessages', [count.toString()]); | 
					
						
							| 
									
										
										
										
											2018-03-02 15:54:15 -05:00
										 |  |  |         this.$('.message').text(message); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     render_attributes: { | 
					
						
							|  |  |  |       message: i18n('loading'), | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2014-11-16 15:30:40 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-02 15:54:15 -05:00
										 |  |  |   Whisper.InboxView = Whisper.View.extend({ | 
					
						
							| 
									
										
										
										
											2021-02-26 13:06:37 -08:00
										 |  |  |     template: () => $('#two-column').html(), | 
					
						
							| 
									
										
										
										
											2018-03-02 15:54:15 -05:00
										 |  |  |     className: 'inbox index', | 
					
						
							|  |  |  |     initialize(options = {}) { | 
					
						
							|  |  |  |       this.ready = false; | 
					
						
							|  |  |  |       this.render(); | 
					
						
							| 
									
										
										
										
											2017-07-24 18:43:35 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-02 15:54:15 -05:00
										 |  |  |       this.conversation_stack = new Whisper.ConversationStack({ | 
					
						
							|  |  |  |         el: this.$('.conversation-stack'), | 
					
						
							|  |  |  |         model: { window: options.window }, | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2017-01-03 21:37:56 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-10 11:28:49 -07:00
										 |  |  |       Whisper.events.on('refreshConversation', ({ oldId, newId }) => { | 
					
						
							|  |  |  |         const convo = this.conversation_stack.lastConversation; | 
					
						
							|  |  |  |         if (convo && convo.get('id') === oldId) { | 
					
						
							|  |  |  |           this.conversation_stack.open(newId); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-07 13:07:24 -07:00
										 |  |  |       // Close current opened conversation to reload the group information once
 | 
					
						
							|  |  |  |       // linked.
 | 
					
						
							|  |  |  |       Whisper.events.on('setupAsNewDevice', () => { | 
					
						
							|  |  |  |         this.conversation_stack.unload(); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-02 15:54:15 -05:00
										 |  |  |       if (!options.initialLoadComplete) { | 
					
						
							|  |  |  |         this.appLoadingScreen = new Whisper.AppLoadingScreen(); | 
					
						
							|  |  |  |         this.appLoadingScreen.render(); | 
					
						
							|  |  |  |         this.appLoadingScreen.$el.prependTo(this.el); | 
					
						
							|  |  |  |         this.startConnectionListener(); | 
					
						
							| 
									
										
										
										
											2019-08-15 14:42:43 -07:00
										 |  |  |       } else { | 
					
						
							|  |  |  |         this.setupLeftPane(); | 
					
						
							| 
									
										
										
										
											2020-06-04 11:16:19 -07:00
										 |  |  |         this.setupCallManagerUI(); | 
					
						
							| 
									
										
										
										
											2018-03-02 15:54:15 -05:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2015-10-15 12:10:03 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 15:32:11 -07:00
										 |  |  |       Whisper.events.on('pack-install-failed', () => { | 
					
						
							|  |  |  |         const toast = new Whisper.StickerPackInstallFailedToast(); | 
					
						
							|  |  |  |         toast.$el.appendTo(this.$el); | 
					
						
							|  |  |  |         toast.render(); | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2018-03-02 15:54:15 -05:00
										 |  |  |     }, | 
					
						
							|  |  |  |     render_attributes: { | 
					
						
							|  |  |  |       welcomeToSignal: i18n('welcomeToSignal'), | 
					
						
							| 
									
										
										
										
											2021-04-20 16:16:49 -07:00
										 |  |  |       // TODO DESKTOP-1451: add back the selectAContact message
 | 
					
						
							|  |  |  |       selectAContact: '', | 
					
						
							| 
									
										
										
										
											2018-03-02 15:54:15 -05:00
										 |  |  |     }, | 
					
						
							|  |  |  |     events: { | 
					
						
							|  |  |  |       click: 'onClick', | 
					
						
							| 
									
										
										
										
											2019-01-14 13:49:58 -08:00
										 |  |  |     }, | 
					
						
							| 
									
										
										
										
											2020-06-04 11:16:19 -07:00
										 |  |  |     setupCallManagerUI() { | 
					
						
							|  |  |  |       if (this.callManagerView) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       this.callManagerView = new Whisper.ReactWrapperView({ | 
					
						
							|  |  |  |         className: 'call-manager-wrapper', | 
					
						
							|  |  |  |         JSX: Signal.State.Roots.createCallManager(window.reduxStore), | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |       this.$('.call-manager-placeholder').append(this.callManagerView.el); | 
					
						
							|  |  |  |     }, | 
					
						
							| 
									
										
										
										
											2019-01-14 13:49:58 -08:00
										 |  |  |     setupLeftPane() { | 
					
						
							| 
									
										
										
										
											2019-08-15 14:42:43 -07:00
										 |  |  |       if (this.leftPaneView) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2019-01-14 13:49:58 -08:00
										 |  |  |       this.leftPaneView = new Whisper.ReactWrapperView({ | 
					
						
							|  |  |  |         className: 'left-pane-wrapper', | 
					
						
							| 
									
										
										
										
											2019-05-31 15:42:01 -07:00
										 |  |  |         JSX: Signal.State.Roots.createLeftPane(window.reduxStore), | 
					
						
							| 
									
										
										
										
											2019-01-14 13:49:58 -08:00
										 |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       this.$('.left-pane-placeholder').append(this.leftPaneView.el); | 
					
						
							| 
									
										
										
										
											2018-03-02 15:54:15 -05:00
										 |  |  |     }, | 
					
						
							|  |  |  |     startConnectionListener() { | 
					
						
							|  |  |  |       this.interval = setInterval(() => { | 
					
						
							|  |  |  |         const status = window.getSocketStatus(); | 
					
						
							|  |  |  |         switch (status) { | 
					
						
							|  |  |  |           case WebSocket.CONNECTING: | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |           case WebSocket.OPEN: | 
					
						
							|  |  |  |             clearInterval(this.interval); | 
					
						
							|  |  |  |             // if we've connected, we can wait for real empty event
 | 
					
						
							|  |  |  |             this.interval = null; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |           case WebSocket.CLOSING: | 
					
						
							|  |  |  |           case WebSocket.CLOSED: | 
					
						
							|  |  |  |             clearInterval(this.interval); | 
					
						
							|  |  |  |             this.interval = null; | 
					
						
							|  |  |  |             // if we failed to connect, we pretend we got an empty event
 | 
					
						
							|  |  |  |             this.onEmpty(); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |           default: | 
					
						
							| 
									
										
										
										
											2020-09-03 21:25:19 -04:00
										 |  |  |             window.log.warn( | 
					
						
							|  |  |  |               'startConnectionListener: Found unexpected socket status; calling onEmpty() manually.' | 
					
						
							|  |  |  |             ); | 
					
						
							| 
									
										
										
										
											2019-02-26 14:14:19 -08:00
										 |  |  |             this.onEmpty(); | 
					
						
							| 
									
										
										
										
											2018-03-02 15:54:15 -05:00
										 |  |  |             break; | 
					
						
							| 
									
										
										
										
											2015-12-07 12:36:30 -08:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-03-02 15:54:15 -05:00
										 |  |  |       }, 1000); | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     onEmpty() { | 
					
						
							| 
									
										
										
										
											2019-08-15 14:42:43 -07:00
										 |  |  |       this.setupLeftPane(); | 
					
						
							| 
									
										
										
										
											2020-06-04 11:16:19 -07:00
										 |  |  |       this.setupCallManagerUI(); | 
					
						
							| 
									
										
										
										
											2019-08-15 14:42:43 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-02 15:54:15 -05:00
										 |  |  |       const view = this.appLoadingScreen; | 
					
						
							|  |  |  |       if (view) { | 
					
						
							|  |  |  |         this.appLoadingScreen = null; | 
					
						
							|  |  |  |         view.remove(); | 
					
						
							| 
									
										
										
										
											2019-11-07 13:36:16 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         const searchInput = document.querySelector( | 
					
						
							|  |  |  |           '.module-main-header__search__input' | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  |         if (searchInput && searchInput.focus) { | 
					
						
							|  |  |  |           searchInput.focus(); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-03-02 15:54:15 -05:00
										 |  |  |       } | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     onProgress(count) { | 
					
						
							|  |  |  |       const view = this.appLoadingScreen; | 
					
						
							|  |  |  |       if (view) { | 
					
						
							|  |  |  |         view.updateProgress(count); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     focusConversation(e) { | 
					
						
							|  |  |  |       if (e && this.$(e.target).closest('.placeholder').length) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2014-11-16 15:30:40 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-02 15:54:15 -05:00
										 |  |  |       this.$('#header, .gutter').addClass('inactive'); | 
					
						
							|  |  |  |       this.$('.conversation-stack').removeClass('inactive'); | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     focusHeader() { | 
					
						
							|  |  |  |       this.$('.conversation-stack').addClass('inactive'); | 
					
						
							|  |  |  |       this.$('#header, .gutter').removeClass('inactive'); | 
					
						
							|  |  |  |       this.$('.conversation:first .menu').trigger('close'); | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     reloadBackgroundPage() { | 
					
						
							|  |  |  |       window.location.reload(); | 
					
						
							|  |  |  |     }, | 
					
						
							| 
									
										
										
										
											2019-01-14 13:49:58 -08:00
										 |  |  |     async openConversation(id, messageId) { | 
					
						
							| 
									
										
										
										
											2019-03-11 17:20:16 -07:00
										 |  |  |       const conversation = await ConversationController.getOrCreateAndWait( | 
					
						
							| 
									
										
										
										
											2019-01-14 13:49:58 -08:00
										 |  |  |         id, | 
					
						
							|  |  |  |         'private' | 
					
						
							|  |  |  |       ); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-28 15:54:32 -07:00
										 |  |  |       conversation.setMarkedUnread(false); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 17:50:36 -07:00
										 |  |  |       const { openConversationExternal } = window.reduxActions.conversations; | 
					
						
							|  |  |  |       if (openConversationExternal) { | 
					
						
							| 
									
										
										
										
											2021-02-23 14:34:28 -06:00
										 |  |  |         openConversationExternal(conversation.id, messageId); | 
					
						
							| 
									
										
										
										
											2018-03-02 15:54:15 -05:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2019-01-14 13:49:58 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-31 15:42:01 -07:00
										 |  |  |       this.conversation_stack.open(conversation, messageId); | 
					
						
							| 
									
										
										
										
											2019-01-14 13:49:58 -08:00
										 |  |  |       this.focusConversation(); | 
					
						
							| 
									
										
										
										
											2018-03-02 15:54:15 -05:00
										 |  |  |     }, | 
					
						
							|  |  |  |     closeRecording(e) { | 
					
						
							|  |  |  |       if (e && this.$(e.target).closest('.capture-audio').length > 0) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       this.$('.conversation:first .recorder').trigger('close'); | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     onClick(e) { | 
					
						
							|  |  |  |       this.closeRecording(e); | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2018-04-27 17:25:04 -04:00
										 |  |  | })(); |