[ Index ]

PHP Cross Reference of WordPress Trunk (Updated Daily)

Search

title

Body

[close]

/wp-admin/js/ -> user-profile.js (source)

   1  /**
   2   * @output wp-admin/js/user-profile.js
   3   */
   4  
   5  /* global ajaxurl, pwsL10n, userProfileL10n, ClipboardJS */
   6  (function($) {
   7      var updateLock = false,
   8          isSubmitting = false,
   9          __ = wp.i18n.__,
  10          clipboard = new ClipboardJS( '.application-password-display .copy-button' ),
  11          $pass1Row,
  12          $pass1,
  13          $pass2,
  14          $weakRow,
  15          $weakCheckbox,
  16          $toggleButton,
  17          $submitButtons,
  18          $submitButton,
  19          currentPass,
  20          $form,
  21          originalFormContent,
  22          $passwordWrapper,
  23          successTimeout;
  24  
  25  	function generatePassword() {
  26          if ( typeof zxcvbn !== 'function' ) {
  27              setTimeout( generatePassword, 50 );
  28              return;
  29          } else if ( ! $pass1.val() || $passwordWrapper.hasClass( 'is-open' ) ) {
  30              // zxcvbn loaded before user entered password, or generating new password.
  31              $pass1.val( $pass1.data( 'pw' ) );
  32              $pass1.trigger( 'pwupdate' );
  33              showOrHideWeakPasswordCheckbox();
  34          } else {
  35              // zxcvbn loaded after the user entered password, check strength.
  36              check_pass_strength();
  37              showOrHideWeakPasswordCheckbox();
  38          }
  39  
  40          /*
  41           * This works around a race condition when zxcvbn loads quickly and
  42           * causes `generatePassword()` to run prior to the toggle button being
  43           * bound.
  44           */
  45          bindToggleButton();
  46  
  47          // Install screen.
  48          if ( 1 !== parseInt( $toggleButton.data( 'start-masked' ), 10 ) ) {
  49              // Show the password not masked if admin_password hasn't been posted yet.
  50              $pass1.attr( 'type', 'text' );
  51          } else {
  52              // Otherwise, mask the password.
  53              $toggleButton.trigger( 'click' );
  54          }
  55  
  56          // Once zxcvbn loads, passwords strength is known.
  57          $( '#pw-weak-text-label' ).text( __( 'Confirm use of weak password' ) );
  58  
  59          // Focus the password field.
  60          if ( 'mailserver_pass' !== $pass1.prop('id' ) ) {
  61              $( $pass1 ).trigger( 'focus' );
  62          }
  63      }
  64  
  65  	function bindPass1() {
  66          currentPass = $pass1.val();
  67  
  68          if ( 1 === parseInt( $pass1.data( 'reveal' ), 10 ) ) {
  69              generatePassword();
  70          }
  71  
  72          $pass1.on( 'input' + ' pwupdate', function () {
  73              if ( $pass1.val() === currentPass ) {
  74                  return;
  75              }
  76  
  77              currentPass = $pass1.val();
  78  
  79              // Refresh password strength area.
  80              $pass1.removeClass( 'short bad good strong' );
  81              showOrHideWeakPasswordCheckbox();
  82          } );
  83      }
  84  
  85  	function resetToggle( show ) {
  86          $toggleButton
  87              .attr({
  88                  'aria-label': show ? __( 'Show password' ) : __( 'Hide password' )
  89              })
  90              .find( '.text' )
  91                  .text( show ? __( 'Show' ) : __( 'Hide' ) )
  92              .end()
  93              .find( '.dashicons' )
  94                  .removeClass( show ? 'dashicons-hidden' : 'dashicons-visibility' )
  95                  .addClass( show ? 'dashicons-visibility' : 'dashicons-hidden' );
  96      }
  97  
  98  	function bindToggleButton() {
  99          if ( !! $toggleButton ) {
 100              // Do not rebind.
 101              return;
 102          }
 103          $toggleButton = $pass1Row.find('.wp-hide-pw');
 104          $toggleButton.show().on( 'click', function () {
 105              if ( 'password' === $pass1.attr( 'type' ) ) {
 106                  $pass1.attr( 'type', 'text' );
 107                  resetToggle( false );
 108              } else {
 109                  $pass1.attr( 'type', 'password' );
 110                  resetToggle( true );
 111              }
 112          });
 113      }
 114  
 115      /**
 116       * Handle the password reset button. Sets up an ajax callback to trigger sending
 117       * a password reset email.
 118       */
 119  	function bindPasswordResetLink() {
 120          $( '#generate-reset-link' ).on( 'click', function() {
 121              var $this  = $(this),
 122                  data = {
 123                      'user_id': userProfileL10n.user_id, // The user to send a reset to.
 124                      'nonce':   userProfileL10n.nonce    // Nonce to validate the action.
 125                  };
 126  
 127                  // Remove any previous error messages.
 128                  $this.parent().find( '.notice-error' ).remove();
 129  
 130                  // Send the reset request.
 131                  var resetAction =  wp.ajax.post( 'send-password-reset', data );
 132  
 133                  // Handle reset success.
 134                  resetAction.done( function( response ) {
 135                      addInlineNotice( $this, true, response );
 136                  } );
 137  
 138                  // Handle reset failure.
 139                  resetAction.fail( function( response ) {
 140                      addInlineNotice( $this, false, response );
 141                  } );
 142  
 143          });
 144  
 145      }
 146  
 147      /**
 148       * Helper function to insert an inline notice of success or failure.
 149       *
 150       * @param {jQuery Object} $this   The button element: the message will be inserted
 151       *                                above this button
 152       * @param {bool}          success Whether the message is a success message.
 153       * @param {string}        message The message to insert.
 154       */
 155  	function addInlineNotice( $this, success, message ) {
 156          var resultDiv = $( '<div />', {
 157              role: 'alert'
 158          } );
 159  
 160          // Set up the notice div.
 161          resultDiv.addClass( 'notice inline' );
 162  
 163          // Add a class indicating success or failure.
 164          resultDiv.addClass( 'notice-' + ( success ? 'success' : 'error' ) );
 165  
 166          // Add the message, wrapping in a p tag, with a fadein to highlight each message.
 167          resultDiv.text( $( $.parseHTML( message ) ).text() ).wrapInner( '<p />');
 168  
 169          // Disable the button when the callback has succeeded.
 170          $this.prop( 'disabled', success );
 171  
 172          // Remove any previous notices.
 173          $this.siblings( '.notice' ).remove();
 174  
 175          // Insert the notice.
 176          $this.before( resultDiv );
 177      }
 178  
 179  	function bindPasswordForm() {
 180          var $generateButton,
 181              $cancelButton;
 182  
 183          $pass1Row = $( '.user-pass1-wrap, .user-pass-wrap, .mailserver-pass-wrap, .reset-pass-submit' );
 184  
 185          // Hide the confirm password field when JavaScript support is enabled.
 186          $('.user-pass2-wrap').hide();
 187  
 188          $submitButton = $( '#submit, #wp-submit' ).on( 'click', function () {
 189              updateLock = false;
 190          });
 191  
 192          $submitButtons = $submitButton.add( ' #createusersub' );
 193  
 194          $weakRow = $( '.pw-weak' );
 195          $weakCheckbox = $weakRow.find( '.pw-checkbox' );
 196          $weakCheckbox.on( 'change', function() {
 197              $submitButtons.prop( 'disabled', ! $weakCheckbox.prop( 'checked' ) );
 198          } );
 199  
 200          $pass1 = $('#pass1, #mailserver_pass');
 201          if ( $pass1.length ) {
 202              bindPass1();
 203          } else {
 204              // Password field for the login form.
 205              $pass1 = $( '#user_pass' );
 206          }
 207  
 208          /*
 209           * Fix a LastPass mismatch issue, LastPass only changes pass2.
 210           *
 211           * This fixes the issue by copying any changes from the hidden
 212           * pass2 field to the pass1 field, then running check_pass_strength.
 213           */
 214          $pass2 = $( '#pass2' ).on( 'input', function () {
 215              if ( $pass2.val().length > 0 ) {
 216                  $pass1.val( $pass2.val() );
 217                  $pass2.val('');
 218                  currentPass = '';
 219                  $pass1.trigger( 'pwupdate' );
 220              }
 221          } );
 222  
 223          // Disable hidden inputs to prevent autofill and submission.
 224          if ( $pass1.is( ':hidden' ) ) {
 225              $pass1.prop( 'disabled', true );
 226              $pass2.prop( 'disabled', true );
 227          }
 228  
 229          $passwordWrapper = $pass1Row.find( '.wp-pwd' );
 230          $generateButton  = $pass1Row.find( 'button.wp-generate-pw' );
 231  
 232          bindToggleButton();
 233  
 234          $generateButton.show();
 235          $generateButton.on( 'click', function () {
 236              updateLock = true;
 237  
 238              // Make sure the password fields are shown.
 239              $generateButton.not( '.skip-aria-expanded' ).attr( 'aria-expanded', 'true' );
 240              $passwordWrapper
 241                  .show()
 242                  .addClass( 'is-open' );
 243  
 244              // Enable the inputs when showing.
 245              $pass1.attr( 'disabled', false );
 246              $pass2.attr( 'disabled', false );
 247  
 248              // Set the password to the generated value.
 249              generatePassword();
 250  
 251              // Show generated password in plaintext by default.
 252              resetToggle ( false );
 253  
 254              // Generate the next password and cache.
 255              wp.ajax.post( 'generate-password' )
 256                  .done( function( data ) {
 257                      $pass1.data( 'pw', data );
 258                  } );
 259          } );
 260  
 261          $cancelButton = $pass1Row.find( 'button.wp-cancel-pw' );
 262          $cancelButton.on( 'click', function () {
 263              updateLock = false;
 264  
 265              // Disable the inputs when hiding to prevent autofill and submission.
 266              $pass1.prop( 'disabled', true );
 267              $pass2.prop( 'disabled', true );
 268  
 269              // Clear password field and update the UI.
 270              $pass1.val( '' ).trigger( 'pwupdate' );
 271              resetToggle( false );
 272  
 273              // Hide password controls.
 274              $passwordWrapper
 275                  .hide()
 276                  .removeClass( 'is-open' );
 277  
 278              // Stop an empty password from being submitted as a change.
 279              $submitButtons.prop( 'disabled', false );
 280  
 281              $generateButton.attr( 'aria-expanded', 'false' );
 282          } );
 283  
 284          $pass1Row.closest( 'form' ).on( 'submit', function () {
 285              updateLock = false;
 286  
 287              $pass1.prop( 'disabled', false );
 288              $pass2.prop( 'disabled', false );
 289              $pass2.val( $pass1.val() );
 290          });
 291      }
 292  
 293  	function check_pass_strength() {
 294          var pass1 = $('#pass1').val(), strength;
 295  
 296          $('#pass-strength-result').removeClass('short bad good strong empty');
 297          if ( ! pass1 || '' ===  pass1.trim() ) {
 298              $( '#pass-strength-result' ).addClass( 'empty' ).html( '&nbsp;' );
 299              return;
 300          }
 301  
 302          strength = wp.passwordStrength.meter( pass1, wp.passwordStrength.userInputDisallowedList(), pass1 );
 303  
 304          switch ( strength ) {
 305              case -1:
 306                  $( '#pass-strength-result' ).addClass( 'bad' ).html( pwsL10n.unknown );
 307                  break;
 308              case 2:
 309                  $('#pass-strength-result').addClass('bad').html( pwsL10n.bad );
 310                  break;
 311              case 3:
 312                  $('#pass-strength-result').addClass('good').html( pwsL10n.good );
 313                  break;
 314              case 4:
 315                  $('#pass-strength-result').addClass('strong').html( pwsL10n.strong );
 316                  break;
 317              case 5:
 318                  $('#pass-strength-result').addClass('short').html( pwsL10n.mismatch );
 319                  break;
 320              default:
 321                  $('#pass-strength-result').addClass('short').html( pwsL10n.short );
 322          }
 323      }
 324  
 325  	function showOrHideWeakPasswordCheckbox() {
 326          var passStrengthResult = $('#pass-strength-result');
 327  
 328          if ( passStrengthResult.length ) {
 329              var passStrength = passStrengthResult[0];
 330  
 331              if ( passStrength.className ) {
 332                  $pass1.addClass( passStrength.className );
 333                  if ( $( passStrength ).is( '.short, .bad' ) ) {
 334                      if ( ! $weakCheckbox.prop( 'checked' ) ) {
 335                          $submitButtons.prop( 'disabled', true );
 336                      }
 337                      $weakRow.show();
 338                  } else {
 339                      if ( $( passStrength ).is( '.empty' ) ) {
 340                          $submitButtons.prop( 'disabled', true );
 341                          $weakCheckbox.prop( 'checked', false );
 342                      } else {
 343                          $submitButtons.prop( 'disabled', false );
 344                      }
 345                      $weakRow.hide();
 346                  }
 347              }
 348          }
 349      }
 350  
 351      // Debug information copy section.
 352      clipboard.on( 'success', function( e ) {
 353          var triggerElement = $( e.trigger ),
 354              successElement = $( '.success', triggerElement.closest( '.application-password-display' ) );
 355  
 356          // Clear the selection and move focus back to the trigger.
 357          e.clearSelection();
 358  
 359          // Show success visual feedback.
 360          clearTimeout( successTimeout );
 361          successElement.removeClass( 'hidden' );
 362  
 363          // Hide success visual feedback after 3 seconds since last success.
 364          successTimeout = setTimeout( function() {
 365              successElement.addClass( 'hidden' );
 366          }, 3000 );
 367  
 368          // Handle success audible feedback.
 369          wp.a11y.speak( __( 'Application password has been copied to your clipboard.' ) );
 370      } );
 371  
 372      $( function() {
 373          var $colorpicker, $stylesheet, user_id, current_user_id,
 374              select       = $( '#display_name' ),
 375              current_name = select.val(),
 376              greeting     = $( '#wp-admin-bar-my-account' ).find( '.display-name' );
 377  
 378          $( '#pass1' ).val( '' ).on( 'input' + ' pwupdate', check_pass_strength );
 379          $('#pass-strength-result').show();
 380          $('.color-palette').on( 'click', function() {
 381              $(this).siblings('input[name="admin_color"]').prop('checked', true);
 382          });
 383  
 384          if ( select.length ) {
 385              $('#first_name, #last_name, #nickname').on( 'blur.user_profile', function() {
 386                  var dub = [],
 387                      inputs = {
 388                          display_nickname  : $('#nickname').val() || '',
 389                          display_username  : $('#user_login').val() || '',
 390                          display_firstname : $('#first_name').val() || '',
 391                          display_lastname  : $('#last_name').val() || ''
 392                      };
 393  
 394                  if ( inputs.display_firstname && inputs.display_lastname ) {
 395                      inputs.display_firstlast = inputs.display_firstname + ' ' + inputs.display_lastname;
 396                      inputs.display_lastfirst = inputs.display_lastname + ' ' + inputs.display_firstname;
 397                  }
 398  
 399                  $.each( $('option', select), function( i, el ){
 400                      dub.push( el.value );
 401                  });
 402  
 403                  $.each(inputs, function( id, value ) {
 404                      if ( ! value ) {
 405                          return;
 406                      }
 407  
 408                      var val = value.replace(/<\/?[a-z][^>]*>/gi, '');
 409  
 410                      if ( inputs[id].length && $.inArray( val, dub ) === -1 ) {
 411                          dub.push(val);
 412                          $('<option />', {
 413                              'text': val
 414                          }).appendTo( select );
 415                      }
 416                  });
 417              });
 418  
 419              /**
 420               * Replaces "Howdy, *" in the admin toolbar whenever the display name dropdown is updated for one's own profile.
 421               */
 422              select.on( 'change', function() {
 423                  if ( user_id !== current_user_id ) {
 424                      return;
 425                  }
 426  
 427                  var display_name = this.value.trim() || current_name;
 428  
 429                  greeting.text( display_name );
 430              } );
 431          }
 432  
 433          $colorpicker = $( '#color-picker' );
 434          $stylesheet = $( '#colors-css' );
 435          user_id = $( 'input#user_id' ).val();
 436          current_user_id = $( 'input[name="checkuser_id"]' ).val();
 437  
 438          $colorpicker.on( 'click.colorpicker', '.color-option', function() {
 439              var colors,
 440                  $this = $(this);
 441  
 442              if ( $this.hasClass( 'selected' ) ) {
 443                  return;
 444              }
 445  
 446              $this.siblings( '.selected' ).removeClass( 'selected' );
 447              $this.addClass( 'selected' ).find( 'input[type="radio"]' ).prop( 'checked', true );
 448  
 449              // Set color scheme.
 450              if ( user_id === current_user_id ) {
 451                  // Load the colors stylesheet.
 452                  // The default color scheme won't have one, so we'll need to create an element.
 453                  if ( 0 === $stylesheet.length ) {
 454                      $stylesheet = $( '<link rel="stylesheet" />' ).appendTo( 'head' );
 455                  }
 456                  $stylesheet.attr( 'href', $this.children( '.css_url' ).val() );
 457  
 458                  // Repaint icons.
 459                  if ( typeof wp !== 'undefined' && wp.svgPainter ) {
 460                      try {
 461                          colors = JSON.parse( $this.children( '.icon_colors' ).val() );
 462                      } catch ( error ) {}
 463  
 464                      if ( colors ) {
 465                          wp.svgPainter.setColors( colors );
 466                          wp.svgPainter.paint();
 467                      }
 468                  }
 469  
 470                  // Update user option.
 471                  $.post( ajaxurl, {
 472                      action:       'save-user-color-scheme',
 473                      color_scheme: $this.children( 'input[name="admin_color"]' ).val(),
 474                      nonce:        $('#color-nonce').val()
 475                  }).done( function( response ) {
 476                      if ( response.success ) {
 477                          $( 'body' ).removeClass( response.data.previousScheme ).addClass( response.data.currentScheme );
 478                      }
 479                  });
 480              }
 481          });
 482  
 483          bindPasswordForm();
 484          bindPasswordResetLink();
 485          $submitButtons.on( 'click', function() {
 486              isSubmitting = true;
 487          });
 488  
 489          $form = $( '#your-profile, #createuser' );
 490          originalFormContent = $form.serialize();
 491      });
 492  
 493      $( '#destroy-sessions' ).on( 'click', function( e ) {
 494          var $this = $(this);
 495  
 496          wp.ajax.post( 'destroy-sessions', {
 497              nonce: $( '#_wpnonce' ).val(),
 498              user_id: $( '#user_id' ).val()
 499          }).done( function( response ) {
 500              $this.prop( 'disabled', true );
 501              $this.siblings( '.notice' ).remove();
 502              $this.before( '<div class="notice notice-success inline" role="alert"><p>' + response.message + '</p></div>' );
 503          }).fail( function( response ) {
 504              $this.siblings( '.notice' ).remove();
 505              $this.before( '<div class="notice notice-error inline" role="alert"><p>' + response.message + '</p></div>' );
 506          });
 507  
 508          e.preventDefault();
 509      });
 510  
 511      window.generatePassword = generatePassword;
 512  
 513      // Warn the user if password was generated but not saved.
 514      $( window ).on( 'beforeunload', function () {
 515          if ( true === updateLock ) {
 516              return __( 'Your new password has not been saved.' );
 517          }
 518          if ( originalFormContent !== $form.serialize() && ! isSubmitting ) {
 519              return __( 'The changes you made will be lost if you navigate away from this page.' );
 520          }
 521      });
 522  
 523      /*
 524       * We need to generate a password as soon as the Reset Password page is loaded,
 525       * to avoid double clicking the button to retrieve the first generated password.
 526       * See ticket #39638.
 527       */
 528      $( function() {
 529          if ( $( '.reset-pass-submit' ).length ) {
 530              $( '.reset-pass-submit button.wp-generate-pw' ).trigger( 'click' );
 531          }
 532      });
 533  
 534  })(jQuery);


Generated : Tue Jan 21 08:20:01 2025 Cross-referenced by PHPXref