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