(function() { 'use strict'; window.Whisper = window.Whisper || {}; var Steps = { INSTALL_SIGNAL: 2, SCAN_QR_CODE: 3, ENTER_NAME: 4, PROGRESS_BAR: 5, TOO_MANY_DEVICES: 'TooManyDevices', NETWORK_ERROR: 'NetworkError', }; var DEVICE_NAME_SELECTOR = 'input.device-name'; var CONNECTION_ERROR = -1; var TOO_MANY_DEVICES = 411; Whisper.InstallView = Whisper.View.extend({ templateName: 'link-flow-template', className: 'main full-screen-flow', events: { 'click .try-again': 'connect', 'click .finish': 'finishLinking', // the actual next step happens in confirmNumber() on submit form #link-phone }, initialize: function(options) { options = options || {}; this.selectStep(Steps.SCAN_QR_CODE); this.connect(); this.on('disconnected', this.reconnect); // Keep data around if it's a re-link, or the middle of a light import this.shouldRetainData = Whisper.Registration.everDone() || options.hasExistingData; }, render_attributes: function() { var errorMessage; if (this.error) { if ( this.error.name === 'HTTPError' && this.error.code == TOO_MANY_DEVICES ) { errorMessage = i18n('installTooManyDevices'); } else if ( this.error.name === 'HTTPError' && this.error.code == CONNECTION_ERROR ) { errorMessage = i18n('installConnectionFailed'); } else if (this.error.message === 'websocket closed') { // AccountManager.registerSecondDevice uses this specific // 'websocket closed' error message errorMessage = i18n('installConnectionFailed'); } return { isError: true, errorHeader: 'Something went wrong!', errorMessage, errorButton: 'Try again', }; } return { isStep3: this.step === Steps.SCAN_QR_CODE, linkYourPhone: i18n('linkYourPhone'), signalSettings: i18n('signalSettings'), linkedDevices: i18n('linkedDevices'), androidFinalStep: i18n('plusButton'), appleFinalStep: i18n('linkNewDevice'), isStep4: this.step === Steps.ENTER_NAME, chooseName: i18n('chooseDeviceName'), finishLinkingPhoneButton: i18n('finishLinkingPhone'), isStep5: this.step === Steps.PROGRESS_BAR, syncing: i18n('initialSync'), }; }, selectStep: function(step) { this.step = step; this.render(); }, connect: function() { this.error = null; this.selectStep(Steps.SCAN_QR_CODE); this.clearQR(); if (this.timeout) { clearTimeout(this.timeout); this.timeout = null; } var accountManager = getAccountManager(); accountManager .registerSecondDevice( this.setProvisioningUrl.bind(this), this.confirmNumber.bind(this) ) .catch(this.handleDisconnect.bind(this)); }, handleDisconnect: function(e) { console.log('provisioning failed', e.stack); this.error = e; this.render(); if (e.message === 'websocket closed') { this.trigger('disconnected'); } else if ( e.name !== 'HTTPError' || (e.code !== CONNECTION_ERROR && e.code !== TOO_MANY_DEVICES) ) { throw e; } }, reconnect: function() { if (this.timeout) { clearTimeout(this.timeout); this.timeout = null; } this.timeout = setTimeout(this.connect.bind(this), 10000); }, clearQR: function() { this.$('#qr img').remove(); this.$('#qr canvas').remove(); this.$('#qr .container').show(); this.$('#qr').removeClass('ready'); }, setProvisioningUrl: function(url) { if ($('#qr').length === 0) { console.log('Did not find #qr element in the DOM!'); return; } this.$('#qr .container').hide(); this.qr = new QRCode(this.$('#qr')[0]).makeCode(url); this.$('#qr').removeAttr('title'); this.$('#qr').addClass('ready'); }, setDeviceNameDefault: function() { var deviceName = textsecure.storage.user.getDeviceName(); this.$(DEVICE_NAME_SELECTOR).val(deviceName || window.getHostName()); this.$(DEVICE_NAME_SELECTOR).focus(); }, finishLinking: function() { // We use a form so we get submit-on-enter behavior this.$('#link-phone').submit(); }, confirmNumber: function(number) { var tsp = textsecure.storage.protocol; window.removeSetupMenuItems(); this.selectStep(Steps.ENTER_NAME); this.setDeviceNameDefault(); return new Promise( function(resolve, reject) { this.$('#link-phone').submit( function(e) { e.stopPropagation(); e.preventDefault(); var name = this.$(DEVICE_NAME_SELECTOR).val(); name = name.replace(/\0/g, ''); // strip unicode null if (name.trim().length === 0) { this.$(DEVICE_NAME_SELECTOR).focus(); return; } this.selectStep(Steps.PROGRESS_BAR); var finish = function() { resolve(name); }; // Delete all data from database unless we're in the middle // of a re-link, or we are finishing a light import. Without this, // app restarts at certain times can cause weird things to happen, // like data from a previous incomplete light import showing up // after a new install. if (this.shouldRetainData) { return finish(); } tsp.removeAllData().then(finish, function(error) { console.log( 'confirmNumber: error clearing database', error && error.stack ? error.stack : error ); finish(); }); }.bind(this) ); }.bind(this) ); }, }); })();