You cannot select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
	
	
		
			229 lines
		
	
	
		
			5.9 KiB
		
	
	
	
		
			JavaScript
		
	
			
		
		
	
	
			229 lines
		
	
	
		
			5.9 KiB
		
	
	
	
		
			JavaScript
		
	
/* global Whisper, i18n, _, Signal, passwordUtil */
 | 
						|
 | 
						|
// eslint-disable-next-line func-names
 | 
						|
(function() {
 | 
						|
  'use strict';
 | 
						|
 | 
						|
  window.Whisper = window.Whisper || {};
 | 
						|
 | 
						|
  const PasswordDialogView = Whisper.View.extend({
 | 
						|
    className: 'loki-dialog password-dialog modal',
 | 
						|
    templateName: 'password-dialog',
 | 
						|
    initialize(options) {
 | 
						|
      this.type = options.type;
 | 
						|
      this.resolve = options.resolve;
 | 
						|
      this.okText = options.okText || i18n('ok');
 | 
						|
 | 
						|
      this.reject = options.reject;
 | 
						|
      this.cancelText = options.cancelText || i18n('cancel');
 | 
						|
 | 
						|
      this.title = options.title;
 | 
						|
 | 
						|
      this.render();
 | 
						|
      this.updateUI();
 | 
						|
    },
 | 
						|
    events: {
 | 
						|
      keyup: 'onKeyup',
 | 
						|
      'click .ok': 'ok',
 | 
						|
      'click .cancel': 'cancel',
 | 
						|
    },
 | 
						|
    render_attributes() {
 | 
						|
      return {
 | 
						|
        showCancel: !this.hideCancel,
 | 
						|
        cancel: this.cancelText,
 | 
						|
        ok: this.okText,
 | 
						|
        title: this.title,
 | 
						|
      };
 | 
						|
    },
 | 
						|
    async updateUI() {
 | 
						|
      if (this.disableOkButton()) {
 | 
						|
        this.$('.ok').prop('disabled', true);
 | 
						|
      } else {
 | 
						|
        this.$('.ok').prop('disabled', false);
 | 
						|
      }
 | 
						|
    },
 | 
						|
    disableOkButton() {
 | 
						|
      const password = this.$('#password').val();
 | 
						|
      return _.isEmpty(password);
 | 
						|
    },
 | 
						|
    async validate() {
 | 
						|
      const password = this.$('#password').val();
 | 
						|
      const passwordConfirmation = this.$('#password-confirmation').val();
 | 
						|
 | 
						|
      const pairValidation = this.validatePasswordPair(
 | 
						|
        password,
 | 
						|
        passwordConfirmation
 | 
						|
      );
 | 
						|
      const hashValidation = await this.validatePasswordHash(password);
 | 
						|
 | 
						|
      return pairValidation || hashValidation;
 | 
						|
    },
 | 
						|
    async validatePasswordHash(password) {
 | 
						|
      // Check if the password matches the hash we have stored
 | 
						|
      const hash = await Signal.Data.getPasswordHash();
 | 
						|
      if (hash && !passwordUtil.matchesHash(password, hash)) {
 | 
						|
        return i18n('invalidPassword');
 | 
						|
      }
 | 
						|
      return null;
 | 
						|
    },
 | 
						|
    validatePasswordPair(password, passwordConfirmation) {
 | 
						|
      if (!_.isEmpty(password)) {
 | 
						|
        // Check if the password is first valid
 | 
						|
        const passwordValidation = passwordUtil.validatePassword(
 | 
						|
          password,
 | 
						|
          i18n
 | 
						|
        );
 | 
						|
        if (passwordValidation) {
 | 
						|
          return passwordValidation;
 | 
						|
        }
 | 
						|
 | 
						|
        // Check if the confirmation password is the same
 | 
						|
        if (
 | 
						|
          !passwordConfirmation ||
 | 
						|
          password.trim() !== passwordConfirmation.trim()
 | 
						|
        ) {
 | 
						|
          return i18n('passwordsDoNotMatch');
 | 
						|
        }
 | 
						|
      }
 | 
						|
      return null;
 | 
						|
    },
 | 
						|
    okPressed() {
 | 
						|
      const password = this.$('#password').val();
 | 
						|
      if (this.type === 'set') {
 | 
						|
        window.setPassword(password.trim());
 | 
						|
      } else if (this.type === 'remove') {
 | 
						|
        window.setPassword(null, password.trim());
 | 
						|
      }
 | 
						|
    },
 | 
						|
    okErrored() {
 | 
						|
      if (this.type === 'set') {
 | 
						|
        this.showError(i18n('setPasswordFail'));
 | 
						|
      } else if (this.type === 'remove') {
 | 
						|
        this.showError(i18n('removePasswordFail'));
 | 
						|
      }
 | 
						|
    },
 | 
						|
    async ok() {
 | 
						|
      const error = await this.validate();
 | 
						|
      if (error) {
 | 
						|
        this.showError(error);
 | 
						|
        return;
 | 
						|
      }
 | 
						|
 | 
						|
      // Clear any errors
 | 
						|
      this.showError(null);
 | 
						|
 | 
						|
      try {
 | 
						|
        this.okPressed();
 | 
						|
 | 
						|
        this.remove();
 | 
						|
        if (this.resolve) {
 | 
						|
          this.resolve();
 | 
						|
        }
 | 
						|
      } catch (e) {
 | 
						|
        this.okErrored();
 | 
						|
      }
 | 
						|
    },
 | 
						|
    cancel() {
 | 
						|
      this.remove();
 | 
						|
      if (this.reject) {
 | 
						|
        this.reject();
 | 
						|
      }
 | 
						|
    },
 | 
						|
    onKeyup(event) {
 | 
						|
      this.updateUI();
 | 
						|
      switch (event.key) {
 | 
						|
        case 'Enter':
 | 
						|
          this.ok();
 | 
						|
          break;
 | 
						|
        case 'Escape':
 | 
						|
        case 'Esc':
 | 
						|
          this.cancel();
 | 
						|
          break;
 | 
						|
        default:
 | 
						|
          return;
 | 
						|
      }
 | 
						|
      event.preventDefault();
 | 
						|
    },
 | 
						|
    focusCancel() {
 | 
						|
      this.$('.cancel').focus();
 | 
						|
    },
 | 
						|
    showError(message) {
 | 
						|
      if (_.isEmpty(message)) {
 | 
						|
        this.$('.error').text('');
 | 
						|
        this.$('.error').hide();
 | 
						|
      } else {
 | 
						|
        this.$('.error').text(`Error: ${message}`);
 | 
						|
        this.$('.error').show();
 | 
						|
      }
 | 
						|
    },
 | 
						|
  });
 | 
						|
 | 
						|
  const ChangePasswordDialogView = PasswordDialogView.extend({
 | 
						|
    templateName: 'password-change-dialog',
 | 
						|
    disableOkButton() {
 | 
						|
      const oldPassword = this.$('#old-password').val();
 | 
						|
      const newPassword = this.$('#new-password').val();
 | 
						|
      return _.isEmpty(oldPassword) || _.isEmpty(newPassword);
 | 
						|
    },
 | 
						|
    async validate() {
 | 
						|
      const oldPassword = this.$('#old-password').val();
 | 
						|
 | 
						|
      // Validate the old password
 | 
						|
      if (!_.isEmpty(oldPassword)) {
 | 
						|
        const oldPasswordValidation = passwordUtil.validatePassword(
 | 
						|
          oldPassword,
 | 
						|
          i18n
 | 
						|
        );
 | 
						|
        if (oldPasswordValidation) {
 | 
						|
          return oldPasswordValidation;
 | 
						|
        }
 | 
						|
      } else {
 | 
						|
        return i18n('typeInOldPassword');
 | 
						|
      }
 | 
						|
 | 
						|
      const password = this.$('#new-password').val();
 | 
						|
      const passwordConfirmation = this.$('#new-password-confirmation').val();
 | 
						|
 | 
						|
      const pairValidation = this.validatePasswordPair(
 | 
						|
        password,
 | 
						|
        passwordConfirmation
 | 
						|
      );
 | 
						|
      const hashValidation = await this.validatePasswordHash(oldPassword);
 | 
						|
 | 
						|
      return pairValidation || hashValidation;
 | 
						|
    },
 | 
						|
    okPressed() {
 | 
						|
      const oldPassword = this.$('#old-password').val();
 | 
						|
      const newPassword = this.$('#new-password').val();
 | 
						|
      window.setPassword(newPassword.trim(), oldPassword.trim());
 | 
						|
    },
 | 
						|
    okErrored() {
 | 
						|
      this.showError(i18n('changePasswordFail'));
 | 
						|
    },
 | 
						|
  });
 | 
						|
 | 
						|
  Whisper.getPasswordDialogView = (type, resolve, reject) => {
 | 
						|
    // This is a differently styled dialog
 | 
						|
    if (type === 'change') {
 | 
						|
      return new ChangePasswordDialogView({
 | 
						|
        title: i18n('changePassword'),
 | 
						|
        okTitle: i18n('change'),
 | 
						|
        resolve,
 | 
						|
        reject,
 | 
						|
      });
 | 
						|
    }
 | 
						|
 | 
						|
    // Set and Remove is basically the same UI
 | 
						|
    const title =
 | 
						|
      type === 'remove' ? i18n('removePassword') : i18n('setPassword');
 | 
						|
    const okTitle = type === 'remove' ? i18n('remove') : i18n('set');
 | 
						|
    return new PasswordDialogView({
 | 
						|
      title,
 | 
						|
      okTitle,
 | 
						|
      type,
 | 
						|
      resolve,
 | 
						|
      reject,
 | 
						|
    });
 | 
						|
  };
 | 
						|
})();
 |