[ Index ]

PHP Cross Reference of WordPress Trunk (Updated Daily)

Search

title

Body

[close]

/wp-admin/js/ -> common.js (source)

   1  /**
   2   * @output wp-admin/js/common.js
   3   */
   4  
   5  /* global setUserSetting, ajaxurl, alert, confirm, pagenow */
   6  /* global columns, screenMeta */
   7  
   8  /**
   9   *  Adds common WordPress functionality to the window.
  10   *
  11   *  @param {jQuery} $        jQuery object.
  12   *  @param {Object} window   The window object.
  13   *  @param {mixed} undefined Unused.
  14   */
  15  ( function( $, window, undefined ) {
  16      var $document = $( document ),
  17          $window = $( window ),
  18          $body = $( document.body ),
  19          __ = wp.i18n.__,
  20          sprintf = wp.i18n.sprintf;
  21  
  22  /**
  23   * Throws an error for a deprecated property.
  24   *
  25   * @since 5.5.1
  26   *
  27   * @param {string} propName    The property that was used.
  28   * @param {string} version     The version of WordPress that deprecated the property.
  29   * @param {string} replacement The property that should have been used.
  30   */
  31  function deprecatedProperty( propName, version, replacement ) {
  32      var message;
  33  
  34      if ( 'undefined' !== typeof replacement ) {
  35          message = sprintf(
  36              /* translators: 1: Deprecated property name, 2: Version number, 3: Alternative property name. */
  37              __( '%1$s is deprecated since version %2$s! Use %3$s instead.' ),
  38              propName,
  39              version,
  40              replacement
  41          );
  42      } else {
  43          message = sprintf(
  44              /* translators: 1: Deprecated property name, 2: Version number. */
  45              __( '%1$s is deprecated since version %2$s with no alternative available.' ),
  46              propName,
  47              version
  48          );
  49      }
  50  
  51      window.console.warn( message );
  52  }
  53  
  54  /**
  55   * Deprecate all properties on an object.
  56   *
  57   * @since 5.5.1
  58   * @since 5.6.0 Added the `version` parameter.
  59   *
  60   * @param {string} name       The name of the object, i.e. commonL10n.
  61   * @param {object} l10nObject The object to deprecate the properties on.
  62   * @param {string} version    The version of WordPress that deprecated the property.
  63   *
  64   * @return {object} The object with all its properties deprecated.
  65   */
  66  function deprecateL10nObject( name, l10nObject, version ) {
  67      var deprecatedObject = {};
  68  
  69      Object.keys( l10nObject ).forEach( function( key ) {
  70          var prop = l10nObject[ key ];
  71          var propName = name + '.' + key;
  72  
  73          if ( 'object' === typeof prop ) {
  74              Object.defineProperty( deprecatedObject, key, { get: function() {
  75                  deprecatedProperty( propName, version, prop.alternative );
  76                  return prop.func();
  77              } } );
  78          } else {
  79              Object.defineProperty( deprecatedObject, key, { get: function() {
  80                  deprecatedProperty( propName, version, 'wp.i18n' );
  81                  return prop;
  82              } } );
  83          }
  84      } );
  85  
  86      return deprecatedObject;
  87  }
  88  
  89  window.wp.deprecateL10nObject = deprecateL10nObject;
  90  
  91  /**
  92   * Removed in 5.5.0, needed for back-compatibility.
  93   *
  94   * @since 2.6.0
  95   * @deprecated 5.5.0
  96   */
  97  window.commonL10n = window.commonL10n || {
  98      warnDelete: '',
  99      dismiss: '',
 100      collapseMenu: '',
 101      expandMenu: ''
 102  };
 103  
 104  window.commonL10n = deprecateL10nObject( 'commonL10n', window.commonL10n, '5.5.0' );
 105  
 106  /**
 107   * Removed in 5.5.0, needed for back-compatibility.
 108   *
 109   * @since 3.3.0
 110   * @deprecated 5.5.0
 111   */
 112  window.wpPointerL10n = window.wpPointerL10n || {
 113      dismiss: ''
 114  };
 115  
 116  window.wpPointerL10n = deprecateL10nObject( 'wpPointerL10n', window.wpPointerL10n, '5.5.0' );
 117  
 118  /**
 119   * Removed in 5.5.0, needed for back-compatibility.
 120   *
 121   * @since 4.3.0
 122   * @deprecated 5.5.0
 123   */
 124  window.userProfileL10n = window.userProfileL10n || {
 125      warn: '',
 126      warnWeak: '',
 127      show: '',
 128      hide: '',
 129      cancel: '',
 130      ariaShow: '',
 131      ariaHide: ''
 132  };
 133  
 134  window.userProfileL10n = deprecateL10nObject( 'userProfileL10n', window.userProfileL10n, '5.5.0' );
 135  
 136  /**
 137   * Removed in 5.5.0, needed for back-compatibility.
 138   *
 139   * @since 4.9.6
 140   * @deprecated 5.5.0
 141   */
 142  window.privacyToolsL10n = window.privacyToolsL10n || {
 143      noDataFound: '',
 144      foundAndRemoved: '',
 145      noneRemoved: '',
 146      someNotRemoved: '',
 147      removalError: '',
 148      emailSent: '',
 149      noExportFile: '',
 150      exportError: ''
 151  };
 152  
 153  window.privacyToolsL10n = deprecateL10nObject( 'privacyToolsL10n', window.privacyToolsL10n, '5.5.0' );
 154  
 155  /**
 156   * Removed in 5.5.0, needed for back-compatibility.
 157   *
 158   * @since 3.6.0
 159   * @deprecated 5.5.0
 160   */
 161  window.authcheckL10n = {
 162      beforeunload: ''
 163  };
 164  
 165  window.authcheckL10n = window.authcheckL10n || deprecateL10nObject( 'authcheckL10n', window.authcheckL10n, '5.5.0' );
 166  
 167  /**
 168   * Removed in 5.5.0, needed for back-compatibility.
 169   *
 170   * @since 2.8.0
 171   * @deprecated 5.5.0
 172   */
 173  window.tagsl10n = {
 174      noPerm: '',
 175      broken: ''
 176  };
 177  
 178  window.tagsl10n = window.tagsl10n || deprecateL10nObject( 'tagsl10n', window.tagsl10n, '5.5.0' );
 179  
 180  /**
 181   * Removed in 5.5.0, needed for back-compatibility.
 182   *
 183   * @since 2.5.0
 184   * @deprecated 5.5.0
 185   */
 186  window.adminCommentsL10n = window.adminCommentsL10n || {
 187      hotkeys_highlight_first: {
 188          alternative: 'window.adminCommentsSettings.hotkeys_highlight_first',
 189          func: function() { return window.adminCommentsSettings.hotkeys_highlight_first; }
 190      },
 191      hotkeys_highlight_last: {
 192          alternative: 'window.adminCommentsSettings.hotkeys_highlight_last',
 193          func: function() { return window.adminCommentsSettings.hotkeys_highlight_last; }
 194      },
 195      replyApprove: '',
 196      reply: '',
 197      warnQuickEdit: '',
 198      warnCommentChanges: '',
 199      docTitleComments: '',
 200      docTitleCommentsCount: ''
 201  };
 202  
 203  window.adminCommentsL10n = deprecateL10nObject( 'adminCommentsL10n', window.adminCommentsL10n, '5.5.0' );
 204  
 205  /**
 206   * Removed in 5.5.0, needed for back-compatibility.
 207   *
 208   * @since 2.5.0
 209   * @deprecated 5.5.0
 210   */
 211  window.tagsSuggestL10n = window.tagsSuggestL10n || {
 212      tagDelimiter: '',
 213      removeTerm: '',
 214      termSelected: '',
 215      termAdded: '',
 216      termRemoved: ''
 217  };
 218  
 219  window.tagsSuggestL10n = deprecateL10nObject( 'tagsSuggestL10n', window.tagsSuggestL10n, '5.5.0' );
 220  
 221  /**
 222   * Removed in 5.5.0, needed for back-compatibility.
 223   *
 224   * @since 3.5.0
 225   * @deprecated 5.5.0
 226   */
 227  window.wpColorPickerL10n = window.wpColorPickerL10n || {
 228      clear: '',
 229      clearAriaLabel: '',
 230      defaultString: '',
 231      defaultAriaLabel: '',
 232      pick: '',
 233      defaultLabel: ''
 234  };
 235  
 236  window.wpColorPickerL10n = deprecateL10nObject( 'wpColorPickerL10n', window.wpColorPickerL10n, '5.5.0' );
 237  
 238  /**
 239   * Removed in 5.5.0, needed for back-compatibility.
 240   *
 241   * @since 2.7.0
 242   * @deprecated 5.5.0
 243   */
 244  window.attachMediaBoxL10n = window.attachMediaBoxL10n || {
 245      error: ''
 246  };
 247  
 248  window.attachMediaBoxL10n = deprecateL10nObject( 'attachMediaBoxL10n', window.attachMediaBoxL10n, '5.5.0' );
 249  
 250  /**
 251   * Removed in 5.5.0, needed for back-compatibility.
 252   *
 253   * @since 2.5.0
 254   * @deprecated 5.5.0
 255   */
 256  window.postL10n = window.postL10n || {
 257      ok: '',
 258      cancel: '',
 259      publishOn: '',
 260      publishOnFuture: '',
 261      publishOnPast: '',
 262      dateFormat: '',
 263      showcomm: '',
 264      endcomm: '',
 265      publish: '',
 266      schedule: '',
 267      update: '',
 268      savePending: '',
 269      saveDraft: '',
 270      'private': '',
 271      'public': '',
 272      publicSticky: '',
 273      password: '',
 274      privatelyPublished: '',
 275      published: '',
 276      saveAlert: '',
 277      savingText: '',
 278      permalinkSaved: ''
 279  };
 280  
 281  window.postL10n = deprecateL10nObject( 'postL10n', window.postL10n, '5.5.0' );
 282  
 283  /**
 284   * Removed in 5.5.0, needed for back-compatibility.
 285   *
 286   * @since 2.7.0
 287   * @deprecated 5.5.0
 288   */
 289  window.inlineEditL10n = window.inlineEditL10n || {
 290      error: '',
 291      ntdeltitle: '',
 292      notitle: '',
 293      comma: '',
 294      saved: ''
 295  };
 296  
 297  window.inlineEditL10n = deprecateL10nObject( 'inlineEditL10n', window.inlineEditL10n, '5.5.0' );
 298  
 299  /**
 300   * Removed in 5.5.0, needed for back-compatibility.
 301   *
 302   * @since 2.7.0
 303   * @deprecated 5.5.0
 304   */
 305  window.plugininstallL10n = window.plugininstallL10n || {
 306      plugin_information: '',
 307      plugin_modal_label: '',
 308      ays: ''
 309  };
 310  
 311  window.plugininstallL10n = deprecateL10nObject( 'plugininstallL10n', window.plugininstallL10n, '5.5.0' );
 312  
 313  /**
 314   * Removed in 5.5.0, needed for back-compatibility.
 315   *
 316   * @since 3.0.0
 317   * @deprecated 5.5.0
 318   */
 319  window.navMenuL10n = window.navMenuL10n || {
 320      noResultsFound: '',
 321      warnDeleteMenu: '',
 322      saveAlert: '',
 323      untitled: ''
 324  };
 325  
 326  window.navMenuL10n = deprecateL10nObject( 'navMenuL10n', window.navMenuL10n, '5.5.0' );
 327  
 328  /**
 329   * Removed in 5.5.0, needed for back-compatibility.
 330   *
 331   * @since 2.5.0
 332   * @deprecated 5.5.0
 333   */
 334  window.commentL10n = window.commentL10n || {
 335      submittedOn: '',
 336      dateFormat: ''
 337  };
 338  
 339  window.commentL10n = deprecateL10nObject( 'commentL10n', window.commentL10n, '5.5.0' );
 340  
 341  /**
 342   * Removed in 5.5.0, needed for back-compatibility.
 343   *
 344   * @since 2.9.0
 345   * @deprecated 5.5.0
 346   */
 347  window.setPostThumbnailL10n = window.setPostThumbnailL10n || {
 348      setThumbnail: '',
 349      saving: '',
 350      error: '',
 351      done: ''
 352  };
 353  
 354  window.setPostThumbnailL10n = deprecateL10nObject( 'setPostThumbnailL10n', window.setPostThumbnailL10n, '5.5.0' );
 355  
 356  /**
 357   * Removed in 6.5.0, needed for back-compatibility.
 358   *
 359   * @since 4.5.0
 360   * @deprecated 6.5.0
 361   */
 362  window.uiAutocompleteL10n = window.uiAutocompleteL10n || {
 363      noResults: '',
 364      oneResult: '',
 365      manyResults: '',
 366      itemSelected: ''
 367  };
 368  
 369  window.uiAutocompleteL10n = deprecateL10nObject( 'uiAutocompleteL10n', window.uiAutocompleteL10n, '6.5.0' );
 370  
 371  /**
 372   * Removed in 3.3.0, needed for back-compatibility.
 373   *
 374   * @since 2.7.0
 375   * @deprecated 3.3.0
 376   */
 377  window.adminMenu = {
 378      init : function() {},
 379      fold : function() {},
 380      restoreMenuState : function() {},
 381      toggle : function() {},
 382      favorites : function() {}
 383  };
 384  
 385  // Show/hide/save table columns.
 386  window.columns = {
 387  
 388      /**
 389       * Initializes the column toggles in the screen options.
 390       *
 391       * Binds an onClick event to the checkboxes to show or hide the table columns
 392       * based on their toggled state. And persists the toggled state.
 393       *
 394       * @since 2.7.0
 395       *
 396       * @return {void}
 397       */
 398      init : function() {
 399          var that = this;
 400          $('.hide-column-tog', '#adv-settings').on( 'click', function() {
 401              var $t = $(this), column = $t.val();
 402              if ( $t.prop('checked') )
 403                  that.checked(column);
 404              else
 405                  that.unchecked(column);
 406  
 407              columns.saveManageColumnsState();
 408          });
 409      },
 410  
 411      /**
 412       * Saves the toggled state for the columns.
 413       *
 414       * Saves whether the columns should be shown or hidden on a page.
 415       *
 416       * @since 3.0.0
 417       *
 418       * @return {void}
 419       */
 420      saveManageColumnsState : function() {
 421          var hidden = this.hidden();
 422          $.post(
 423              ajaxurl,
 424              {
 425                  action: 'hidden-columns',
 426                  hidden: hidden,
 427                  screenoptionnonce: $('#screenoptionnonce').val(),
 428                  page: pagenow
 429              },
 430              function() {
 431                  wp.a11y.speak( __( 'Screen Options updated.' ) );
 432              }
 433          );
 434      },
 435  
 436      /**
 437       * Makes a column visible and adjusts the column span for the table.
 438       *
 439       * @since 3.0.0
 440       * @param {string} column The column name.
 441       *
 442       * @return {void}
 443       */
 444      checked : function(column) {
 445          $('.column-' + column).removeClass( 'hidden' );
 446          this.colSpanChange(+1);
 447      },
 448  
 449      /**
 450       * Hides a column and adjusts the column span for the table.
 451       *
 452       * @since 3.0.0
 453       * @param {string} column The column name.
 454       *
 455       * @return {void}
 456       */
 457      unchecked : function(column) {
 458          $('.column-' + column).addClass( 'hidden' );
 459          this.colSpanChange(-1);
 460      },
 461  
 462      /**
 463       * Gets all hidden columns.
 464       *
 465       * @since 3.0.0
 466       *
 467       * @return {string} The hidden column names separated by a comma.
 468       */
 469      hidden : function() {
 470          return $( '.manage-column[id]' ).filter( '.hidden' ).map(function() {
 471              return this.id;
 472          }).get().join( ',' );
 473      },
 474  
 475      /**
 476       * Gets the checked column toggles from the screen options.
 477       *
 478       * @since 3.0.0
 479       *
 480       * @return {string} String containing the checked column names.
 481       */
 482      useCheckboxesForHidden : function() {
 483          this.hidden = function(){
 484              return $('.hide-column-tog').not(':checked').map(function() {
 485                  var id = this.id;
 486                  return id.substring( id, id.length - 5 );
 487              }).get().join(',');
 488          };
 489      },
 490  
 491      /**
 492       * Adjusts the column span for the table.
 493       *
 494       * @since 3.1.0
 495       *
 496       * @param {number} diff The modifier for the column span.
 497       */
 498      colSpanChange : function(diff) {
 499          var $t = $('table').find('.colspanchange'), n;
 500          if ( !$t.length )
 501              return;
 502          n = parseInt( $t.attr('colspan'), 10 ) + diff;
 503          $t.attr('colspan', n.toString());
 504      }
 505  };
 506  
 507  $( function() { columns.init(); } );
 508  
 509  /**
 510   * Validates that the required form fields are not empty.
 511   *
 512   * @since 2.9.0
 513   *
 514   * @param {jQuery} form The form to validate.
 515   *
 516   * @return {boolean} Returns true if all required fields are not an empty string.
 517   */
 518  window.validateForm = function( form ) {
 519      return !$( form )
 520          .find( '.form-required' )
 521          .filter( function() { return $( ':input:visible', this ).val() === ''; } )
 522          .addClass( 'form-invalid' )
 523          .find( ':input:visible' )
 524          .on( 'change', function() { $( this ).closest( '.form-invalid' ).removeClass( 'form-invalid' ); } )
 525          .length;
 526  };
 527  
 528  // Stub for doing better warnings.
 529  /**
 530   * Shows message pop-up notice or confirmation message.
 531   *
 532   * @since 2.7.0
 533   *
 534   * @type {{warn: showNotice.warn, note: showNotice.note}}
 535   *
 536   * @return {void}
 537   */
 538  window.showNotice = {
 539  
 540      /**
 541       * Shows a delete confirmation pop-up message.
 542       *
 543       * @since 2.7.0
 544       *
 545       * @return {boolean} Returns true if the message is confirmed.
 546       */
 547      warn : function() {
 548          if ( confirm( __( 'You are about to permanently delete these items from your site.\nThis action cannot be undone.\n\'Cancel\' to stop, \'OK\' to delete.' ) ) ) {
 549              return true;
 550          }
 551  
 552          return false;
 553      },
 554  
 555      /**
 556       * Shows an alert message.
 557       *
 558       * @since 2.7.0
 559       *
 560       * @param text The text to display in the message.
 561       */
 562      note : function(text) {
 563          alert(text);
 564      }
 565  };
 566  
 567  /**
 568   * Represents the functions for the meta screen options panel.
 569   *
 570   * @since 3.2.0
 571   *
 572   * @type {{element: null, toggles: null, page: null, init: screenMeta.init,
 573   *         toggleEvent: screenMeta.toggleEvent, open: screenMeta.open,
 574   *         close: screenMeta.close}}
 575   *
 576   * @return {void}
 577   */
 578  window.screenMeta = {
 579      element: null, // #screen-meta
 580      toggles: null, // .screen-meta-toggle
 581      page:    null, // #wpcontent
 582  
 583      /**
 584       * Initializes the screen meta options panel.
 585       *
 586       * @since 3.2.0
 587       *
 588       * @return {void}
 589       */
 590      init: function() {
 591          this.element = $('#screen-meta');
 592          this.toggles = $( '#screen-meta-links' ).find( '.show-settings' );
 593          this.page    = $('#wpcontent');
 594  
 595          this.toggles.on( 'click', this.toggleEvent );
 596      },
 597  
 598      /**
 599       * Toggles the screen meta options panel.
 600       *
 601       * @since 3.2.0
 602       *
 603       * @return {void}
 604       */
 605      toggleEvent: function() {
 606          var panel = $( '#' + $( this ).attr( 'aria-controls' ) );
 607  
 608          if ( !panel.length )
 609              return;
 610  
 611          if ( panel.is(':visible') )
 612              screenMeta.close( panel, $(this) );
 613          else
 614              screenMeta.open( panel, $(this) );
 615      },
 616  
 617      /**
 618       * Opens the screen meta options panel.
 619       *
 620       * @since 3.2.0
 621       *
 622       * @param {jQuery} panel  The screen meta options panel div.
 623       * @param {jQuery} button The toggle button.
 624       *
 625       * @return {void}
 626       */
 627      open: function( panel, button ) {
 628  
 629          $( '#screen-meta-links' ).find( '.screen-meta-toggle' ).not( button.parent() ).css( 'visibility', 'hidden' );
 630  
 631          panel.parent().show();
 632  
 633          /**
 634           * Sets the focus to the meta options panel and adds the necessary CSS classes.
 635           *
 636           * @since 3.2.0
 637           *
 638           * @return {void}
 639           */
 640          panel.slideDown( 'fast', function() {
 641              panel.removeClass( 'hidden' ).trigger( 'focus' );
 642              button.addClass( 'screen-meta-active' ).attr( 'aria-expanded', true );
 643          });
 644  
 645          $document.trigger( 'screen:options:open' );
 646      },
 647  
 648      /**
 649       * Closes the screen meta options panel.
 650       *
 651       * @since 3.2.0
 652       *
 653       * @param {jQuery} panel  The screen meta options panel div.
 654       * @param {jQuery} button The toggle button.
 655       *
 656       * @return {void}
 657       */
 658      close: function( panel, button ) {
 659          /**
 660           * Hides the screen meta options panel.
 661           *
 662           * @since 3.2.0
 663           *
 664           * @return {void}
 665           */
 666          panel.slideUp( 'fast', function() {
 667              button.removeClass( 'screen-meta-active' ).attr( 'aria-expanded', false );
 668              $('.screen-meta-toggle').css('visibility', '');
 669              panel.parent().hide();
 670              panel.addClass( 'hidden' );
 671          });
 672  
 673          $document.trigger( 'screen:options:close' );
 674      }
 675  };
 676  
 677  /**
 678   * Initializes the help tabs in the help panel.
 679   *
 680   * @param {Event} e The event object.
 681   *
 682   * @return {void}
 683   */
 684  $('.contextual-help-tabs').on( 'click', 'a', function(e) {
 685      var link = $(this),
 686          panel;
 687  
 688      e.preventDefault();
 689  
 690      // Don't do anything if the click is for the tab already showing.
 691      if ( link.is('.active a') )
 692          return false;
 693  
 694      // Links.
 695      $('.contextual-help-tabs .active').removeClass('active');
 696      link.parent('li').addClass('active');
 697  
 698      panel = $( link.attr('href') );
 699  
 700      // Panels.
 701      $('.help-tab-content').not( panel ).removeClass('active').hide();
 702      panel.addClass('active').show();
 703  });
 704  
 705  /**
 706   * Update custom permalink structure via buttons.
 707   */
 708  var permalinkStructureFocused = false,
 709      $permalinkStructure       = $( '#permalink_structure' ),
 710      $permalinkStructureInputs = $( '.permalink-structure input:radio' ),
 711      $permalinkCustomSelection = $( '#custom_selection' ),
 712      $availableStructureTags   = $( '.form-table.permalink-structure .available-structure-tags button' );
 713  
 714  // Change permalink structure input when selecting one of the common structures.
 715  $permalinkStructureInputs.on( 'change', function() {
 716      if ( 'custom' === this.value ) {
 717          return;
 718      }
 719  
 720      $permalinkStructure.val( this.value );
 721  
 722      // Update button states after selection.
 723      $availableStructureTags.each( function() {
 724          changeStructureTagButtonState( $( this ) );
 725      } );
 726  } );
 727  
 728  $permalinkStructure.on( 'click input', function() {
 729      $permalinkCustomSelection.prop( 'checked', true );
 730  } );
 731  
 732  // Check if the permalink structure input field has had focus at least once.
 733  $permalinkStructure.on( 'focus', function( event ) {
 734      permalinkStructureFocused = true;
 735      $( this ).off( event );
 736  } );
 737  
 738  /**
 739   * Enables or disables a structure tag button depending on its usage.
 740   *
 741   * If the structure is already used in the custom permalink structure,
 742   * it will be disabled.
 743   *
 744   * @param {Object} button Button jQuery object.
 745   */
 746  function changeStructureTagButtonState( button ) {
 747      if ( -1 !== $permalinkStructure.val().indexOf( button.text().trim() ) ) {
 748          button.attr( 'data-label', button.attr( 'aria-label' ) );
 749          button.attr( 'aria-label', button.attr( 'data-used' ) );
 750          button.attr( 'aria-pressed', true );
 751          button.addClass( 'active' );
 752      } else if ( button.attr( 'data-label' ) ) {
 753          button.attr( 'aria-label', button.attr( 'data-label' ) );
 754          button.attr( 'aria-pressed', false );
 755          button.removeClass( 'active' );
 756      }
 757  }
 758  
 759  // Check initial button state.
 760  $availableStructureTags.each( function() {
 761      changeStructureTagButtonState( $( this ) );
 762  } );
 763  
 764  // Observe permalink structure field and disable buttons of tags that are already present.
 765  $permalinkStructure.on( 'change', function() {
 766      $availableStructureTags.each( function() {
 767          changeStructureTagButtonState( $( this ) );
 768      } );
 769  } );
 770  
 771  $availableStructureTags.on( 'click', function() {
 772      var permalinkStructureValue = $permalinkStructure.val(),
 773          selectionStart          = $permalinkStructure[ 0 ].selectionStart,
 774          selectionEnd            = $permalinkStructure[ 0 ].selectionEnd,
 775          textToAppend            = $( this ).text().trim(),
 776          textToAnnounce,
 777          newSelectionStart;
 778  
 779      if ( $( this ).hasClass( 'active' ) ) {
 780          textToAnnounce = $( this ).attr( 'data-removed' );
 781      } else {
 782          textToAnnounce = $( this ).attr( 'data-added' );
 783      }
 784  
 785      // Remove structure tag if already part of the structure.
 786      if ( -1 !== permalinkStructureValue.indexOf( textToAppend ) ) {
 787          permalinkStructureValue = permalinkStructureValue.replace( textToAppend + '/', '' );
 788  
 789          $permalinkStructure.val( '/' === permalinkStructureValue ? '' : permalinkStructureValue );
 790  
 791          // Announce change to screen readers.
 792          $( '#custom_selection_updated' ).text( textToAnnounce );
 793  
 794          // Disable button.
 795          changeStructureTagButtonState( $( this ) );
 796  
 797          return;
 798      }
 799  
 800      // Input field never had focus, move selection to end of input.
 801      if ( ! permalinkStructureFocused && 0 === selectionStart && 0 === selectionEnd ) {
 802          selectionStart = selectionEnd = permalinkStructureValue.length;
 803      }
 804  
 805      $permalinkCustomSelection.prop( 'checked', true );
 806  
 807      // Prepend and append slashes if necessary.
 808      if ( '/' !== permalinkStructureValue.substr( 0, selectionStart ).substr( -1 ) ) {
 809          textToAppend = '/' + textToAppend;
 810      }
 811  
 812      if ( '/' !== permalinkStructureValue.substr( selectionEnd, 1 ) ) {
 813          textToAppend = textToAppend + '/';
 814      }
 815  
 816      // Insert structure tag at the specified position.
 817      $permalinkStructure.val( permalinkStructureValue.substr( 0, selectionStart ) + textToAppend + permalinkStructureValue.substr( selectionEnd ) );
 818  
 819      // Announce change to screen readers.
 820      $( '#custom_selection_updated' ).text( textToAnnounce );
 821  
 822      // Disable button.
 823      changeStructureTagButtonState( $( this ) );
 824  
 825      // If input had focus give it back with cursor right after appended text.
 826      if ( permalinkStructureFocused && $permalinkStructure[0].setSelectionRange ) {
 827          newSelectionStart = ( permalinkStructureValue.substr( 0, selectionStart ) + textToAppend ).length;
 828          $permalinkStructure[0].setSelectionRange( newSelectionStart, newSelectionStart );
 829          $permalinkStructure.trigger( 'focus' );
 830      }
 831  } );
 832  
 833  $( function() {
 834      var checks, first, last, checked, sliced, mobileEvent, transitionTimeout, focusedRowActions,
 835          lastClicked = false,
 836          pageInput = $('input.current-page'),
 837          currentPage = pageInput.val(),
 838          isIOS = /iPhone|iPad|iPod/.test( navigator.userAgent ),
 839          isAndroid = navigator.userAgent.indexOf( 'Android' ) !== -1,
 840          $adminMenuWrap = $( '#adminmenuwrap' ),
 841          $wpwrap = $( '#wpwrap' ),
 842          $adminmenu = $( '#adminmenu' ),
 843          $overlay = $( '#wp-responsive-overlay' ),
 844          $toolbar = $( '#wp-toolbar' ),
 845          $toolbarPopups = $toolbar.find( 'a[aria-haspopup="true"]' ),
 846          $sortables = $('.meta-box-sortables'),
 847          wpResponsiveActive = false,
 848          $adminbar = $( '#wpadminbar' ),
 849          lastScrollPosition = 0,
 850          pinnedMenuTop = false,
 851          pinnedMenuBottom = false,
 852          menuTop = 0,
 853          menuState,
 854          menuIsPinned = false,
 855          height = {
 856              window: $window.height(),
 857              wpwrap: $wpwrap.height(),
 858              adminbar: $adminbar.height(),
 859              menu: $adminMenuWrap.height()
 860          },
 861          $headerEnd = $( '.wp-header-end' );
 862  
 863      /**
 864       * Makes the fly-out submenu header clickable, when the menu is folded.
 865       *
 866       * @param {Event} e The event object.
 867       *
 868       * @return {void}
 869       */
 870      $adminmenu.on('click.wp-submenu-head', '.wp-submenu-head', function(e){
 871          $(e.target).parent().siblings('a').get(0).click();
 872      });
 873  
 874      /**
 875       * Collapses the admin menu.
 876       *
 877       * @return {void}
 878       */
 879      $( '#collapse-button' ).on( 'click.collapse-menu', function() {
 880          var viewportWidth = getViewportWidth() || 961;
 881  
 882          // Reset any compensation for submenus near the bottom of the screen.
 883          $('#adminmenu div.wp-submenu').css('margin-top', '');
 884  
 885          if ( viewportWidth <= 960 ) {
 886              if ( $body.hasClass('auto-fold') ) {
 887                  $body.removeClass('auto-fold').removeClass('folded');
 888                  setUserSetting('unfold', 1);
 889                  setUserSetting('mfold', 'o');
 890                  menuState = 'open';
 891              } else {
 892                  $body.addClass('auto-fold');
 893                  setUserSetting('unfold', 0);
 894                  menuState = 'folded';
 895              }
 896          } else {
 897              if ( $body.hasClass('folded') ) {
 898                  $body.removeClass('folded');
 899                  setUserSetting('mfold', 'o');
 900                  menuState = 'open';
 901              } else {
 902                  $body.addClass('folded');
 903                  setUserSetting('mfold', 'f');
 904                  menuState = 'folded';
 905              }
 906          }
 907  
 908          $document.trigger( 'wp-collapse-menu', { state: menuState } );
 909      });
 910  
 911      /**
 912       * Ensures an admin submenu is within the visual viewport.
 913       *
 914       * @since 4.1.0
 915       *
 916       * @param {jQuery} $menuItem The parent menu item containing the submenu.
 917       *
 918       * @return {void}
 919       */
 920  	function adjustSubmenu( $menuItem ) {
 921          var bottomOffset, pageHeight, adjustment, theFold, menutop, wintop, maxtop,
 922              $submenu = $menuItem.find( '.wp-submenu' );
 923  
 924          menutop = $menuItem.offset().top;
 925          wintop = $window.scrollTop();
 926          maxtop = menutop - wintop - 30; // max = make the top of the sub almost touch admin bar.
 927  
 928          bottomOffset = menutop + $submenu.height() + 1; // Bottom offset of the menu.
 929          pageHeight = $wpwrap.height();                  // Height of the entire page.
 930          adjustment = 60 + bottomOffset - pageHeight;
 931          theFold = $window.height() + wintop - 50;       // The fold.
 932  
 933          if ( theFold < ( bottomOffset - adjustment ) ) {
 934              adjustment = bottomOffset - theFold;
 935          }
 936  
 937          if ( adjustment > maxtop ) {
 938              adjustment = maxtop;
 939          }
 940  
 941          if ( adjustment > 1 && $('#wp-admin-bar-menu-toggle').is(':hidden') ) {
 942              $submenu.css( 'margin-top', '-' + adjustment + 'px' );
 943          } else {
 944              $submenu.css( 'margin-top', '' );
 945          }
 946      }
 947  
 948      if ( 'ontouchstart' in window || /IEMobile\/[1-9]/.test(navigator.userAgent) ) { // Touch screen device.
 949          // iOS Safari works with touchstart, the rest work with click.
 950          mobileEvent = isIOS ? 'touchstart' : 'click';
 951  
 952          /**
 953           * Closes any open submenus when touch/click is not on the menu.
 954           *
 955           * @param {Event} e The event object.
 956           *
 957           * @return {void}
 958           */
 959          $body.on( mobileEvent+'.wp-mobile-hover', function(e) {
 960              if ( $adminmenu.data('wp-responsive') ) {
 961                  return;
 962              }
 963  
 964              if ( ! $( e.target ).closest( '#adminmenu' ).length ) {
 965                  $adminmenu.find( 'li.opensub' ).removeClass( 'opensub' );
 966              }
 967          });
 968  
 969          /**
 970           * Handles the opening or closing the submenu based on the mobile click|touch event.
 971           *
 972           * @param {Event} event The event object.
 973           *
 974           * @return {void}
 975           */
 976          $adminmenu.find( 'a.wp-has-submenu' ).on( mobileEvent + '.wp-mobile-hover', function( event ) {
 977              var $menuItem = $(this).parent();
 978  
 979              if ( $adminmenu.data( 'wp-responsive' ) ) {
 980                  return;
 981              }
 982  
 983              /*
 984               * Show the sub instead of following the link if:
 985               *     - the submenu is not open.
 986               *     - the submenu is not shown inline or the menu is not folded.
 987               */
 988              if ( ! $menuItem.hasClass( 'opensub' ) && ( ! $menuItem.hasClass( 'wp-menu-open' ) || $menuItem.width() < 40 ) ) {
 989                  event.preventDefault();
 990                  adjustSubmenu( $menuItem );
 991                  $adminmenu.find( 'li.opensub' ).removeClass( 'opensub' );
 992                  $menuItem.addClass('opensub');
 993              }
 994          });
 995      }
 996  
 997      if ( ! isIOS && ! isAndroid ) {
 998          $adminmenu.find( 'li.wp-has-submenu' ).hoverIntent({
 999  
1000              /**
1001               * Opens the submenu when hovered over the menu item for desktops.
1002               *
1003               * @return {void}
1004               */
1005              over: function() {
1006                  var $menuItem = $( this ),
1007                      $submenu = $menuItem.find( '.wp-submenu' ),
1008                      top = parseInt( $submenu.css( 'top' ), 10 );
1009  
1010                  if ( isNaN( top ) || top > -5 ) { // The submenu is visible.
1011                      return;
1012                  }
1013  
1014                  if ( $adminmenu.data( 'wp-responsive' ) ) {
1015                      // The menu is in responsive mode, bail.
1016                      return;
1017                  }
1018  
1019                  adjustSubmenu( $menuItem );
1020                  $adminmenu.find( 'li.opensub' ).removeClass( 'opensub' );
1021                  $menuItem.addClass( 'opensub' );
1022              },
1023  
1024              /**
1025               * Closes the submenu when no longer hovering the menu item.
1026               *
1027               * @return {void}
1028               */
1029              out: function(){
1030                  if ( $adminmenu.data( 'wp-responsive' ) ) {
1031                      // The menu is in responsive mode, bail.
1032                      return;
1033                  }
1034  
1035                  $( this ).removeClass( 'opensub' ).find( '.wp-submenu' ).css( 'margin-top', '' );
1036              },
1037              timeout: 200,
1038              sensitivity: 7,
1039              interval: 90
1040          });
1041  
1042          /**
1043           * Opens the submenu on when focused on the menu item.
1044           *
1045           * @param {Event} event The event object.
1046           *
1047           * @return {void}
1048           */
1049          $adminmenu.on( 'focus.adminmenu', '.wp-submenu a', function( event ) {
1050              if ( $adminmenu.data( 'wp-responsive' ) ) {
1051                  // The menu is in responsive mode, bail.
1052                  return;
1053              }
1054  
1055              $( event.target ).closest( 'li.menu-top' ).addClass( 'opensub' );
1056  
1057              /**
1058               * Closes the submenu on blur from the menu item.
1059               *
1060               * @param {Event} event The event object.
1061               *
1062               * @return {void}
1063               */
1064          }).on( 'blur.adminmenu', '.wp-submenu a', function( event ) {
1065              if ( $adminmenu.data( 'wp-responsive' ) ) {
1066                  return;
1067              }
1068  
1069              $( event.target ).closest( 'li.menu-top' ).removeClass( 'opensub' );
1070  
1071              /**
1072               * Adjusts the size for the submenu.
1073               *
1074               * @return {void}
1075               */
1076          }).find( 'li.wp-has-submenu.wp-not-current-submenu' ).on( 'focusin.adminmenu', function() {
1077              adjustSubmenu( $( this ) );
1078          });
1079      }
1080  
1081      /*
1082       * The `.below-h2` class is here just for backward compatibility with plugins
1083       * that are (incorrectly) using it. Do not use. Use `.inline` instead. See #34570.
1084       * If '.wp-header-end' is found, append the notices after it otherwise
1085       * after the first h1 or h2 heading found within the main content.
1086       */
1087      if ( ! $headerEnd.length ) {
1088          $headerEnd = $( '.wrap h1, .wrap h2' ).first();
1089      }
1090      $( 'div.updated, div.error, div.notice' ).not( '.inline, .below-h2' ).insertAfter( $headerEnd );
1091  
1092      /**
1093       * Makes notices dismissible.
1094       *
1095       * @since 4.4.0
1096       *
1097       * @return {void}
1098       */
1099  	function makeNoticesDismissible() {
1100          $( '.notice.is-dismissible' ).each( function() {
1101              var $el = $( this ),
1102                  $button = $( '<button type="button" class="notice-dismiss"><span class="screen-reader-text"></span></button>' );
1103  
1104              if ( $el.find( '.notice-dismiss' ).length ) {
1105                  return;
1106              }
1107  
1108              // Ensure plain text.
1109              $button.find( '.screen-reader-text' ).text( __( 'Dismiss this notice.' ) );
1110              $button.on( 'click.wp-dismiss-notice', function( event ) {
1111                  event.preventDefault();
1112                  $el.fadeTo( 100, 0, function() {
1113                      $el.slideUp( 100, function() {
1114                          $el.remove();
1115                      });
1116                  });
1117              });
1118  
1119              $el.append( $button );
1120          });
1121      }
1122  
1123      $document.on( 'wp-updates-notice-added wp-plugin-install-error wp-plugin-update-error wp-plugin-delete-error wp-theme-install-error wp-theme-delete-error wp-notice-added', makeNoticesDismissible );
1124  
1125      // Init screen meta.
1126      screenMeta.init();
1127  
1128      /**
1129       * Checks a checkbox.
1130       *
1131       * This event needs to be delegated. Ticket #37973.
1132       *
1133       * @return {boolean} Returns whether a checkbox is checked or not.
1134       */
1135      $body.on( 'click', 'tbody > tr > .check-column :checkbox', function( event ) {
1136          // Shift click to select a range of checkboxes.
1137          if ( 'undefined' == event.shiftKey ) { return true; }
1138          if ( event.shiftKey ) {
1139              if ( !lastClicked ) { return true; }
1140              checks = $( lastClicked ).closest( 'form' ).find( ':checkbox' ).filter( ':visible:enabled' );
1141              first = checks.index( lastClicked );
1142              last = checks.index( this );
1143              checked = $(this).prop('checked');
1144              if ( 0 < first && 0 < last && first != last ) {
1145                  sliced = ( last > first ) ? checks.slice( first, last ) : checks.slice( last, first );
1146                  sliced.prop( 'checked', function() {
1147                      if ( $(this).closest('tr').is(':visible') )
1148                          return checked;
1149  
1150                      return false;
1151                  });
1152              }
1153          }
1154          lastClicked = this;
1155  
1156          // Toggle the "Select all" checkboxes depending if the other ones are all checked or not.
1157          var unchecked = $(this).closest('tbody').find('tr').find(':checkbox').filter(':visible:enabled').not(':checked');
1158  
1159          /**
1160           * Determines if all checkboxes are checked.
1161           *
1162           * @return {boolean} Returns true if there are no unchecked checkboxes.
1163           */
1164          $(this).closest('table').children('thead, tfoot').find(':checkbox').prop('checked', function() {
1165              return ( 0 === unchecked.length );
1166          });
1167  
1168          return true;
1169      });
1170  
1171      /**
1172       * Controls all the toggles on bulk toggle change.
1173       *
1174       * When the bulk checkbox is changed, all the checkboxes in the tables are changed accordingly.
1175       * When the shift-button is pressed while changing the bulk checkbox the checkboxes in the table are inverted.
1176       *
1177       * This event needs to be delegated. Ticket #37973.
1178       *
1179       * @param {Event} event The event object.
1180       *
1181       * @return {boolean}
1182       */
1183      $body.on( 'click.wp-toggle-checkboxes', 'thead .check-column :checkbox, tfoot .check-column :checkbox', function( event ) {
1184          var $this = $(this),
1185              $table = $this.closest( 'table' ),
1186              controlChecked = $this.prop('checked'),
1187              toggle = event.shiftKey || $this.data('wp-toggle');
1188  
1189          $table.children( 'tbody' ).filter(':visible')
1190              .children().children('.check-column').find(':checkbox')
1191              /**
1192               * Updates the checked state on the checkbox in the table.
1193               *
1194               * @return {boolean} True checks the checkbox, False unchecks the checkbox.
1195               */
1196              .prop('checked', function() {
1197                  if ( $(this).is(':hidden,:disabled') ) {
1198                      return false;
1199                  }
1200  
1201                  if ( toggle ) {
1202                      return ! $(this).prop( 'checked' );
1203                  } else if ( controlChecked ) {
1204                      return true;
1205                  }
1206  
1207                  return false;
1208              });
1209  
1210          $table.children('thead,  tfoot').filter(':visible')
1211              .children().children('.check-column').find(':checkbox')
1212  
1213              /**
1214               * Syncs the bulk checkboxes on the top and bottom of the table.
1215               *
1216               * @return {boolean} True checks the checkbox, False unchecks the checkbox.
1217               */
1218              .prop('checked', function() {
1219                  if ( toggle ) {
1220                      return false;
1221                  } else if ( controlChecked ) {
1222                      return true;
1223                  }
1224  
1225                  return false;
1226              });
1227      });
1228  
1229      /**
1230       * Marries a secondary control to its primary control.
1231       *
1232       * @param {jQuery} topSelector    The top selector element.
1233       * @param {jQuery} topSubmit      The top submit element.
1234       * @param {jQuery} bottomSelector The bottom selector element.
1235       * @param {jQuery} bottomSubmit   The bottom submit element.
1236       * @return {void}
1237       */
1238  	function marryControls( topSelector, topSubmit, bottomSelector, bottomSubmit ) {
1239          /**
1240           * Updates the primary selector when the secondary selector is changed.
1241           *
1242           * @since 5.7.0
1243           *
1244           * @return {void}
1245           */
1246  		function updateTopSelector() {
1247              topSelector.val($(this).val());
1248          }
1249          bottomSelector.on('change', updateTopSelector);
1250  
1251          /**
1252           * Updates the secondary selector when the primary selector is changed.
1253           *
1254           * @since 5.7.0
1255           *
1256           * @return {void}
1257           */
1258  		function updateBottomSelector() {
1259              bottomSelector.val($(this).val());
1260          }
1261          topSelector.on('change', updateBottomSelector);
1262  
1263          /**
1264           * Triggers the primary submit when then secondary submit is clicked.
1265           *
1266           * @since 5.7.0
1267           *
1268           * @return {void}
1269           */
1270  		function triggerSubmitClick(e) {
1271              e.preventDefault();
1272              e.stopPropagation();
1273  
1274              topSubmit.trigger('click');
1275          }
1276          bottomSubmit.on('click', triggerSubmitClick);
1277      }
1278  
1279      // Marry the secondary "Bulk actions" controls to the primary controls:
1280      marryControls( $('#bulk-action-selector-top'), $('#doaction'), $('#bulk-action-selector-bottom'), $('#doaction2') );
1281  
1282      // Marry the secondary "Change role to" controls to the primary controls:
1283      marryControls( $('#new_role'), $('#changeit'), $('#new_role2'), $('#changeit2') );
1284  
1285      var addAdminNotice = function( data ) {
1286          var $notice = $( data.selector ),
1287              $headerEnd = $( '.wp-header-end' ),
1288              type,
1289              dismissible,
1290              $adminNotice;
1291  
1292          delete data.selector;
1293  
1294          dismissible = ( data.dismissible && data.dismissible === true ) ? ' is-dismissible' : '';
1295          type        = ( data.type ) ? data.type : 'info';
1296  
1297          $adminNotice = '<div id="' + data.id + '" class="notice notice-' + data.type + dismissible + '"><p>' + data.message + '</p></div>';
1298  
1299          // Check if this admin notice already exists.
1300          if ( ! $notice.length ) {
1301              $notice = $( '#' + data.id );
1302          }
1303  
1304          if ( $notice.length ) {
1305              $notice.replaceWith( $adminNotice );
1306          } else if ( $headerEnd.length ) {
1307              $headerEnd.after( $adminNotice );
1308          } else {
1309              if ( 'customize' === pagenow ) {
1310                  $( '.customize-themes-notifications' ).append( $adminNotice );
1311              } else {
1312                  $( '.wrap' ).find( '> h1' ).after( $adminNotice );
1313              }
1314          }
1315  
1316          $document.trigger( 'wp-notice-added' );
1317      };
1318  
1319      $( '.bulkactions' ).parents( 'form' ).on( 'submit', function( event ) {
1320          var form = this,
1321              submitterName = event.originalEvent && event.originalEvent.submitter ? event.originalEvent.submitter.name : false,
1322              currentPageSelector = form.querySelector( '#current-page-selector' );
1323  
1324          if ( currentPageSelector && currentPageSelector.defaultValue !== currentPageSelector.value ) {
1325              return; // Pagination form submission.
1326          }
1327  
1328          // Observe submissions from posts lists for 'bulk_action' or users lists for 'new_role'.
1329          var bulkFieldRelations = {
1330              'bulk_action' : 'action',
1331              'changeit' : 'new_role'
1332          };
1333          if ( ! Object.keys( bulkFieldRelations ).includes( submitterName ) ) {
1334              return;
1335          }
1336  
1337          var values = new FormData(form);
1338          var value = values.get( bulkFieldRelations[ submitterName ] ) || '-1';
1339  
1340          // Check that the action is not the default one.
1341          if ( value !== '-1' ) {
1342              // Check that at least one item is selected.
1343              var itemsSelected = form.querySelectorAll( '.wp-list-table tbody .check-column input[type="checkbox"]:checked' );
1344  
1345              if ( itemsSelected.length > 0 ) {
1346                  return;
1347              }
1348          }
1349          event.preventDefault();
1350          event.stopPropagation();
1351          $( 'html, body' ).animate( { scrollTop: 0 } );
1352  
1353          var errorMessage = __( 'Please select at least one item to perform this action on.' );
1354          addAdminNotice( {
1355              id: 'no-items-selected',
1356              type: 'error',
1357              message: errorMessage,
1358              dismissible: true,
1359          } );
1360  
1361          wp.a11y.speak( errorMessage );
1362      });
1363  
1364      /**
1365       * Shows row actions on focus of its parent container element or any other elements contained within.
1366       *
1367       * @return {void}
1368       */
1369      $( '#wpbody-content' ).on({
1370          focusin: function() {
1371              clearTimeout( transitionTimeout );
1372              focusedRowActions = $( this ).find( '.row-actions' );
1373              // transitionTimeout is necessary for Firefox, but Chrome won't remove the CSS class without a little help.
1374              $( '.row-actions' ).not( this ).removeClass( 'visible' );
1375              focusedRowActions.addClass( 'visible' );
1376          },
1377          focusout: function() {
1378              // Tabbing between post title and .row-actions links needs a brief pause, otherwise
1379              // the .row-actions div gets hidden in transit in some browsers (ahem, Firefox).
1380              transitionTimeout = setTimeout( function() {
1381                  focusedRowActions.removeClass( 'visible' );
1382              }, 30 );
1383          }
1384      }, '.table-view-list .has-row-actions' );
1385  
1386      // Toggle list table rows on small screens.
1387      $( 'tbody' ).on( 'click', '.toggle-row', function() {
1388          $( this ).closest( 'tr' ).toggleClass( 'is-expanded' );
1389      });
1390  
1391      $('#default-password-nag-no').on( 'click', function() {
1392          setUserSetting('default_password_nag', 'hide');
1393          $('div.default-password-nag').hide();
1394          return false;
1395      });
1396  
1397      /**
1398       * Handles tab keypresses in theme and plugin file editor textareas.
1399       *
1400       * @param {Event} e The event object.
1401       *
1402       * @return {void}
1403       */
1404      $('#newcontent').on('keydown.wpevent_InsertTab', function(e) {
1405          var el = e.target, selStart, selEnd, val, scroll, sel;
1406  
1407          // After pressing escape key (keyCode: 27), the tab key should tab out of the textarea.
1408          if ( e.keyCode == 27 ) {
1409              // When pressing Escape: Opera 12 and 27 blur form fields, IE 8 clears them.
1410              e.preventDefault();
1411              $(el).data('tab-out', true);
1412              return;
1413          }
1414  
1415          // Only listen for plain tab key (keyCode: 9) without any modifiers.
1416          if ( e.keyCode != 9 || e.ctrlKey || e.altKey || e.shiftKey )
1417              return;
1418  
1419          // After tabbing out, reset it so next time the tab key can be used again.
1420          if ( $(el).data('tab-out') ) {
1421              $(el).data('tab-out', false);
1422              return;
1423          }
1424  
1425          selStart = el.selectionStart;
1426          selEnd = el.selectionEnd;
1427          val = el.value;
1428  
1429          // If any text is selected, replace the selection with a tab character.
1430          if ( document.selection ) {
1431              el.focus();
1432              sel = document.selection.createRange();
1433              sel.text = '\t';
1434          } else if ( selStart >= 0 ) {
1435              scroll = this.scrollTop;
1436              el.value = val.substring(0, selStart).concat('\t', val.substring(selEnd) );
1437              el.selectionStart = el.selectionEnd = selStart + 1;
1438              this.scrollTop = scroll;
1439          }
1440  
1441          // Cancel the regular tab functionality, to prevent losing focus of the textarea.
1442          if ( e.stopPropagation )
1443              e.stopPropagation();
1444          if ( e.preventDefault )
1445              e.preventDefault();
1446      });
1447  
1448      // Reset page number variable for new filters/searches but not for bulk actions. See #17685.
1449      if ( pageInput.length ) {
1450  
1451          /**
1452           * Handles pagination variable when filtering the list table.
1453           *
1454           * Set the pagination argument to the first page when the post-filter form is submitted.
1455           * This happens when pressing the 'filter' button on the list table page.
1456           *
1457           * The pagination argument should not be touched when the bulk action dropdowns are set to do anything.
1458           *
1459           * The form closest to the pageInput is the post-filter form.
1460           *
1461           * @return {void}
1462           */
1463          pageInput.closest('form').on( 'submit', function() {
1464              /*
1465               * action = bulk action dropdown at the top of the table
1466               */
1467              if ( $('select[name="action"]').val() == -1 && pageInput.val() == currentPage )
1468                  pageInput.val('1');
1469          });
1470      }
1471  
1472      /**
1473       * Resets the bulk actions when the search button is clicked.
1474       *
1475       * @return {void}
1476       */
1477      $('.search-box input[type="search"], .search-box input[type="submit"]').on( 'mousedown', function () {
1478          $('select[name^="action"]').val('-1');
1479      });
1480  
1481      /**
1482       * Scrolls into view when focus.scroll-into-view is triggered.
1483       *
1484       * @param {Event} e The event object.
1485       *
1486       * @return {void}
1487        */
1488      $('#contextual-help-link, #show-settings-link').on( 'focus.scroll-into-view', function(e){
1489          if ( e.target.scrollIntoViewIfNeeded )
1490              e.target.scrollIntoViewIfNeeded(false);
1491      });
1492  
1493      /**
1494       * Disables the submit upload buttons when no data is entered.
1495       *
1496       * @return {void}
1497       */
1498      (function(){
1499          var button, input, form = $('form.wp-upload-form');
1500  
1501          // Exit when no upload form is found.
1502          if ( ! form.length )
1503              return;
1504  
1505          button = form.find('input[type="submit"]');
1506          input = form.find('input[type="file"]');
1507  
1508          /**
1509           * Determines if any data is entered in any file upload input.
1510           *
1511           * @since 3.5.0
1512           *
1513           * @return {void}
1514           */
1515  		function toggleUploadButton() {
1516              // When no inputs have a value, disable the upload buttons.
1517              button.prop('disabled', '' === input.map( function() {
1518                  return $(this).val();
1519              }).get().join(''));
1520          }
1521  
1522          // Update the status initially.
1523          toggleUploadButton();
1524          // Update the status when any file input changes.
1525          input.on('change', toggleUploadButton);
1526      })();
1527  
1528      /**
1529       * Pins the menu while distraction-free writing is enabled.
1530       *
1531       * @param {Event} event Event data.
1532       *
1533       * @since 4.1.0
1534       *
1535       * @return {void}
1536       */
1537  	function pinMenu( event ) {
1538          var windowPos = $window.scrollTop(),
1539              resizing = ! event || event.type !== 'scroll';
1540  
1541          if ( isIOS || $adminmenu.data( 'wp-responsive' ) ) {
1542              return;
1543          }
1544  
1545          /*
1546           * When the menu is higher than the window and smaller than the entire page.
1547           * It should be adjusted to be able to see the entire menu.
1548           *
1549           * Otherwise it can be accessed normally.
1550           */
1551          if ( height.menu + height.adminbar < height.window ||
1552              height.menu + height.adminbar + 20 > height.wpwrap ) {
1553              unpinMenu();
1554              return;
1555          }
1556  
1557          menuIsPinned = true;
1558  
1559          // If the menu is higher than the window, compensate on scroll.
1560          if ( height.menu + height.adminbar > height.window ) {
1561              // Check for overscrolling, this happens when swiping up at the top of the document in modern browsers.
1562              if ( windowPos < 0 ) {
1563                  // Stick the menu to the top.
1564                  if ( ! pinnedMenuTop ) {
1565                      pinnedMenuTop = true;
1566                      pinnedMenuBottom = false;
1567  
1568                      $adminMenuWrap.css({
1569                          position: 'fixed',
1570                          top: '',
1571                          bottom: ''
1572                      });
1573                  }
1574  
1575                  return;
1576              } else if ( windowPos + height.window > $document.height() - 1 ) {
1577                  // When overscrolling at the bottom, stick the menu to the bottom.
1578                  if ( ! pinnedMenuBottom ) {
1579                      pinnedMenuBottom = true;
1580                      pinnedMenuTop = false;
1581  
1582                      $adminMenuWrap.css({
1583                          position: 'fixed',
1584                          top: '',
1585                          bottom: 0
1586                      });
1587                  }
1588  
1589                  return;
1590              }
1591  
1592              if ( windowPos > lastScrollPosition ) {
1593                  // When a down scroll has been detected.
1594  
1595                  // If it was pinned to the top, unpin and calculate relative scroll.
1596                  if ( pinnedMenuTop ) {
1597                      pinnedMenuTop = false;
1598                      // Calculate new offset position.
1599                      menuTop = $adminMenuWrap.offset().top - height.adminbar - ( windowPos - lastScrollPosition );
1600  
1601                      if ( menuTop + height.menu + height.adminbar < windowPos + height.window ) {
1602                          menuTop = windowPos + height.window - height.menu - height.adminbar;
1603                      }
1604  
1605                      $adminMenuWrap.css({
1606                          position: 'absolute',
1607                          top: menuTop,
1608                          bottom: ''
1609                      });
1610                  } else if ( ! pinnedMenuBottom && $adminMenuWrap.offset().top + height.menu < windowPos + height.window ) {
1611                      // Pin it to the bottom.
1612                      pinnedMenuBottom = true;
1613  
1614                      $adminMenuWrap.css({
1615                          position: 'fixed',
1616                          top: '',
1617                          bottom: 0
1618                      });
1619                  }
1620              } else if ( windowPos < lastScrollPosition ) {
1621                  // When a scroll up is detected.
1622  
1623                  // If it was pinned to the bottom, unpin and calculate relative scroll.
1624                  if ( pinnedMenuBottom ) {
1625                      pinnedMenuBottom = false;
1626  
1627                      // Calculate new offset position.
1628                      menuTop = $adminMenuWrap.offset().top - height.adminbar + ( lastScrollPosition - windowPos );
1629  
1630                      if ( menuTop + height.menu > windowPos + height.window ) {
1631                          menuTop = windowPos;
1632                      }
1633  
1634                      $adminMenuWrap.css({
1635                          position: 'absolute',
1636                          top: menuTop,
1637                          bottom: ''
1638                      });
1639                  } else if ( ! pinnedMenuTop && $adminMenuWrap.offset().top >= windowPos + height.adminbar ) {
1640  
1641                      // Pin it to the top.
1642                      pinnedMenuTop = true;
1643  
1644                      $adminMenuWrap.css({
1645                          position: 'fixed',
1646                          top: '',
1647                          bottom: ''
1648                      });
1649                  }
1650              } else if ( resizing ) {
1651                  // Window is being resized.
1652  
1653                  pinnedMenuTop = pinnedMenuBottom = false;
1654  
1655                  // Calculate the new offset.
1656                  menuTop = windowPos + height.window - height.menu - height.adminbar - 1;
1657  
1658                  if ( menuTop > 0 ) {
1659                      $adminMenuWrap.css({
1660                          position: 'absolute',
1661                          top: menuTop,
1662                          bottom: ''
1663                      });
1664                  } else {
1665                      unpinMenu();
1666                  }
1667              }
1668          }
1669  
1670          lastScrollPosition = windowPos;
1671      }
1672  
1673      /**
1674       * Determines the height of certain elements.
1675       *
1676       * @since 4.1.0
1677       *
1678       * @return {void}
1679       */
1680  	function resetHeights() {
1681          height = {
1682              window: $window.height(),
1683              wpwrap: $wpwrap.height(),
1684              adminbar: $adminbar.height(),
1685              menu: $adminMenuWrap.height()
1686          };
1687      }
1688  
1689      /**
1690       * Unpins the menu.
1691       *
1692       * @since 4.1.0
1693       *
1694       * @return {void}
1695       */
1696  	function unpinMenu() {
1697          if ( isIOS || ! menuIsPinned ) {
1698              return;
1699          }
1700  
1701          pinnedMenuTop = pinnedMenuBottom = menuIsPinned = false;
1702          $adminMenuWrap.css({
1703              position: '',
1704              top: '',
1705              bottom: ''
1706          });
1707      }
1708  
1709      /**
1710       * Pins and unpins the menu when applicable.
1711       *
1712       * @since 4.1.0
1713       *
1714       * @return {void}
1715       */
1716  	function setPinMenu() {
1717          resetHeights();
1718  
1719          if ( $adminmenu.data('wp-responsive') ) {
1720              $body.removeClass( 'sticky-menu' );
1721              unpinMenu();
1722          } else if ( height.menu + height.adminbar > height.window ) {
1723              pinMenu();
1724              $body.removeClass( 'sticky-menu' );
1725          } else {
1726              $body.addClass( 'sticky-menu' );
1727              unpinMenu();
1728          }
1729      }
1730  
1731      if ( ! isIOS ) {
1732          $window.on( 'scroll.pin-menu', pinMenu );
1733          $document.on( 'tinymce-editor-init.pin-menu', function( event, editor ) {
1734              editor.on( 'wp-autoresize', resetHeights );
1735          });
1736      }
1737  
1738      /**
1739       * Changes the sortables and responsiveness of metaboxes.
1740       *
1741       * @since 3.8.0
1742       *
1743       * @return {void}
1744       */
1745      window.wpResponsive = {
1746  
1747          /**
1748           * Initializes the wpResponsive object.
1749           *
1750           * @since 3.8.0
1751           *
1752           * @return {void}
1753           */
1754          init: function() {
1755              var self = this;
1756  
1757              this.maybeDisableSortables = this.maybeDisableSortables.bind( this );
1758  
1759              // Modify functionality based on custom activate/deactivate event.
1760              $document.on( 'wp-responsive-activate.wp-responsive', function() {
1761                  self.activate();
1762                  self.toggleAriaHasPopup( 'add' );
1763              }).on( 'wp-responsive-deactivate.wp-responsive', function() {
1764                  self.deactivate();
1765                  self.toggleAriaHasPopup( 'remove' );
1766              });
1767  
1768              $( '#wp-admin-bar-menu-toggle a' ).attr( 'aria-expanded', 'false' );
1769  
1770              // Toggle sidebar when toggle is clicked.
1771              $( '#wp-admin-bar-menu-toggle' ).on( 'click.wp-responsive', function( event ) {
1772                  event.preventDefault();
1773  
1774                  // Close any open toolbar submenus.
1775                  $adminbar.find( '.hover' ).removeClass( 'hover' );
1776  
1777                  $wpwrap.toggleClass( 'wp-responsive-open' );
1778                  if ( $wpwrap.hasClass( 'wp-responsive-open' ) ) {
1779                      $(this).find('a').attr( 'aria-expanded', 'true' );
1780                      $( '#adminmenu a:first' ).trigger( 'focus' );
1781                  } else {
1782                      $(this).find('a').attr( 'aria-expanded', 'false' );
1783                  }
1784              } );
1785  
1786              // Close sidebar when target moves outside of toggle and sidebar.
1787              $( document ).on( 'click', function( event ) {
1788                  if ( ! $wpwrap.hasClass( 'wp-responsive-open' ) || ! document.hasFocus() ) {
1789                      return;
1790                  }
1791  
1792                  var focusIsInToggle  = $.contains( $( '#wp-admin-bar-menu-toggle' )[0], event.target );
1793                  var focusIsInSidebar = $.contains( $( '#adminmenuwrap' )[0], event.target );
1794  
1795                  if ( ! focusIsInToggle && ! focusIsInSidebar ) {
1796                      $( '#wp-admin-bar-menu-toggle' ).trigger( 'click.wp-responsive' );
1797                  }
1798              } );
1799  
1800              // Close sidebar when a keypress completes outside of toggle and sidebar.
1801              $( document ).on( 'keyup', function( event ) {
1802                  var toggleButton   = $( '#wp-admin-bar-menu-toggle' )[0];
1803                  if ( ! $wpwrap.hasClass( 'wp-responsive-open' ) ) {
1804                      return;
1805                  }
1806                  if ( 27 === event.keyCode ) {
1807                      $( toggleButton ).trigger( 'click.wp-responsive' );
1808                      $( toggleButton ).find( 'a' ).trigger( 'focus' );
1809                  } else {
1810                      if ( 9 === event.keyCode ) {
1811                          var sidebar        = $( '#adminmenuwrap' )[0];
1812                          var focusedElement = event.relatedTarget || document.activeElement;
1813                          // A brief delay is required to allow focus to switch to another element.
1814                          setTimeout( function() {
1815                              var focusIsInToggle  = $.contains( toggleButton, focusedElement );
1816                              var focusIsInSidebar = $.contains( sidebar, focusedElement );
1817  
1818                              if ( ! focusIsInToggle && ! focusIsInSidebar ) {
1819                                  $( toggleButton ).trigger( 'click.wp-responsive' );
1820                              }
1821                          }, 10 );
1822                      }
1823                  }
1824              });
1825  
1826              // Add menu events.
1827              $adminmenu.on( 'click.wp-responsive', 'li.wp-has-submenu > a', function( event ) {
1828                  if ( ! $adminmenu.data('wp-responsive') ) {
1829                      return;
1830                  }
1831                  let state = ( 'false' === $( this ).attr( 'aria-expanded' ) ) ? 'true' : 'false';
1832                  $( this ).parent( 'li' ).toggleClass( 'selected' );
1833                  $( this ).attr( 'aria-expanded', state );
1834                  $( this ).trigger( 'focus' );
1835                  event.preventDefault();
1836              });
1837  
1838              self.trigger();
1839              $document.on( 'wp-window-resized.wp-responsive', this.trigger.bind( this ) );
1840  
1841              // This needs to run later as UI Sortable may be initialized when the document is ready.
1842              $window.on( 'load.wp-responsive', this.maybeDisableSortables );
1843              $document.on( 'postbox-toggled', this.maybeDisableSortables );
1844  
1845              // When the screen columns are changed, potentially disable sortables.
1846              $( '#screen-options-wrap input' ).on( 'click', this.maybeDisableSortables );
1847          },
1848  
1849          /**
1850           * Disable sortables if there is only one metabox, or the screen is in one column mode. Otherwise, enable sortables.
1851           *
1852           * @since 5.3.0
1853           *
1854           * @return {void}
1855           */
1856          maybeDisableSortables: function() {
1857              var width = navigator.userAgent.indexOf('AppleWebKit/') > -1 ? $window.width() : window.innerWidth;
1858  
1859              if (
1860                  ( width <= 782 ) ||
1861                  ( 1 >= $sortables.find( '.ui-sortable-handle:visible' ).length && jQuery( '.columns-prefs-1 input' ).prop( 'checked' ) )
1862              ) {
1863                  this.disableSortables();
1864              } else {
1865                  this.enableSortables();
1866              }
1867          },
1868  
1869          /**
1870           * Changes properties of body and admin menu.
1871           *
1872           * Pins and unpins the menu and adds the auto-fold class to the body.
1873           * Makes the admin menu responsive and disables the metabox sortables.
1874           *
1875           * @since 3.8.0
1876           *
1877           * @return {void}
1878           */
1879          activate: function() {
1880              setPinMenu();
1881  
1882              if ( ! $body.hasClass( 'auto-fold' ) ) {
1883                  $body.addClass( 'auto-fold' );
1884              }
1885  
1886              $adminmenu.data( 'wp-responsive', 1 );
1887              this.disableSortables();
1888          },
1889  
1890          /**
1891           * Changes properties of admin menu and enables metabox sortables.
1892           *
1893           * Pin and unpin the menu.
1894           * Removes the responsiveness of the admin menu and enables the metabox sortables.
1895           *
1896           * @since 3.8.0
1897           *
1898           * @return {void}
1899           */
1900          deactivate: function() {
1901              setPinMenu();
1902              $adminmenu.removeData('wp-responsive');
1903  
1904              this.maybeDisableSortables();
1905          },
1906  
1907          /**
1908           * Toggles the aria-haspopup attribute for the responsive admin menu.
1909           *
1910           * The aria-haspopup attribute is only necessary for the responsive menu.
1911           * See ticket https://core.trac.wordpress.org/ticket/43095
1912           *
1913           * @since 6.6.0
1914           *
1915           * @param {string} action Whether to add or remove the aria-haspopup attribute.
1916           *
1917           * @return {void}
1918           */
1919          toggleAriaHasPopup: function( action ) {
1920              var elements = $adminmenu.find( '[data-ariahaspopup]' );
1921  
1922              if ( action === 'add' ) {
1923                  elements.each( function() {
1924                      $( this ).attr( 'aria-haspopup', 'menu' ).attr( 'aria-expanded', 'false' );
1925                  } );
1926  
1927                  return;
1928              }
1929  
1930              elements.each( function() {
1931                  $( this ).removeAttr( 'aria-haspopup' ).removeAttr( 'aria-expanded' );
1932              } );
1933          },
1934  
1935          /**
1936           * Sets the responsiveness and enables the overlay based on the viewport width.
1937           *
1938           * @since 3.8.0
1939           *
1940           * @return {void}
1941           */
1942          trigger: function() {
1943              var viewportWidth = getViewportWidth();
1944  
1945              // Exclude IE < 9, it doesn't support @media CSS rules.
1946              if ( ! viewportWidth ) {
1947                  return;
1948              }
1949  
1950              if ( viewportWidth <= 782 ) {
1951                  if ( ! wpResponsiveActive ) {
1952                      $document.trigger( 'wp-responsive-activate' );
1953                      wpResponsiveActive = true;
1954                  }
1955              } else {
1956                  if ( wpResponsiveActive ) {
1957                      $document.trigger( 'wp-responsive-deactivate' );
1958                      wpResponsiveActive = false;
1959                  }
1960              }
1961  
1962              if ( viewportWidth <= 480 ) {
1963                  this.enableOverlay();
1964              } else {
1965                  this.disableOverlay();
1966              }
1967  
1968              this.maybeDisableSortables();
1969          },
1970  
1971          /**
1972           * Inserts a responsive overlay and toggles the window.
1973           *
1974           * @since 3.8.0
1975           *
1976           * @return {void}
1977           */
1978          enableOverlay: function() {
1979              if ( $overlay.length === 0 ) {
1980                  $overlay = $( '<div id="wp-responsive-overlay"></div>' )
1981                      .insertAfter( '#wpcontent' )
1982                      .hide()
1983                      .on( 'click.wp-responsive', function() {
1984                          $toolbar.find( '.menupop.hover' ).removeClass( 'hover' );
1985                          $( this ).hide();
1986                      });
1987              }
1988  
1989              $toolbarPopups.on( 'click.wp-responsive', function() {
1990                  $overlay.show();
1991              });
1992          },
1993  
1994          /**
1995           * Disables the responsive overlay and removes the overlay.
1996           *
1997           * @since 3.8.0
1998           *
1999           * @return {void}
2000           */
2001          disableOverlay: function() {
2002              $toolbarPopups.off( 'click.wp-responsive' );
2003              $overlay.hide();
2004          },
2005  
2006          /**
2007           * Disables sortables.
2008           *
2009           * @since 3.8.0
2010           *
2011           * @return {void}
2012           */
2013          disableSortables: function() {
2014              if ( $sortables.length ) {
2015                  try {
2016                      $sortables.sortable( 'disable' );
2017                      $sortables.find( '.ui-sortable-handle' ).addClass( 'is-non-sortable' );
2018                  } catch ( e ) {}
2019              }
2020          },
2021  
2022          /**
2023           * Enables sortables.
2024           *
2025           * @since 3.8.0
2026           *
2027           * @return {void}
2028           */
2029          enableSortables: function() {
2030              if ( $sortables.length ) {
2031                  try {
2032                      $sortables.sortable( 'enable' );
2033                      $sortables.find( '.ui-sortable-handle' ).removeClass( 'is-non-sortable' );
2034                  } catch ( e ) {}
2035              }
2036          }
2037      };
2038  
2039      /**
2040       * Add an ARIA role `button` to elements that behave like UI controls when JavaScript is on.
2041       *
2042       * @since 4.5.0
2043       *
2044       * @return {void}
2045       */
2046  	function aria_button_if_js() {
2047          $( '.aria-button-if-js' ).attr( 'role', 'button' );
2048      }
2049  
2050      $( document ).on( 'ajaxComplete', function() {
2051          aria_button_if_js();
2052      });
2053  
2054      /**
2055       * Get the viewport width.
2056       *
2057       * @since 4.7.0
2058       *
2059       * @return {number|boolean} The current viewport width or false if the
2060       *                          browser doesn't support innerWidth (IE < 9).
2061       */
2062  	function getViewportWidth() {
2063          var viewportWidth = false;
2064  
2065          if ( window.innerWidth ) {
2066              // On phones, window.innerWidth is affected by zooming.
2067              viewportWidth = Math.max( window.innerWidth, document.documentElement.clientWidth );
2068          }
2069  
2070          return viewportWidth;
2071      }
2072  
2073      /**
2074       * Sets the admin menu collapsed/expanded state.
2075       *
2076       * Sets the global variable `menuState` and triggers a custom event passing
2077       * the current menu state.
2078       *
2079       * @since 4.7.0
2080       *
2081       * @return {void}
2082       */
2083  	function setMenuState() {
2084          var viewportWidth = getViewportWidth() || 961;
2085  
2086          if ( viewportWidth <= 782  ) {
2087              menuState = 'responsive';
2088          } else if ( $body.hasClass( 'folded' ) || ( $body.hasClass( 'auto-fold' ) && viewportWidth <= 960 && viewportWidth > 782 ) ) {
2089              menuState = 'folded';
2090          } else {
2091              menuState = 'open';
2092          }
2093  
2094          $document.trigger( 'wp-menu-state-set', { state: menuState } );
2095      }
2096  
2097      // Set the menu state when the window gets resized.
2098      $document.on( 'wp-window-resized.set-menu-state', setMenuState );
2099  
2100      /**
2101       * Sets ARIA attributes on the collapse/expand menu button.
2102       *
2103       * When the admin menu is open or folded, updates the `aria-expanded` and
2104       * `aria-label` attributes of the button to give feedback to assistive
2105       * technologies. In the responsive view, the button is always hidden.
2106       *
2107       * @since 4.7.0
2108       *
2109       * @return {void}
2110       */
2111      $document.on( 'wp-menu-state-set wp-collapse-menu', function( event, eventData ) {
2112          var $collapseButton = $( '#collapse-button' ),
2113              ariaExpanded, ariaLabelText;
2114  
2115          if ( 'folded' === eventData.state ) {
2116              ariaExpanded = 'false';
2117              ariaLabelText = __( 'Expand Main menu' );
2118          } else {
2119              ariaExpanded = 'true';
2120              ariaLabelText = __( 'Collapse Main menu' );
2121          }
2122  
2123          $collapseButton.attr({
2124              'aria-expanded': ariaExpanded,
2125              'aria-label': ariaLabelText
2126          });
2127      });
2128  
2129      window.wpResponsive.init();
2130      setPinMenu();
2131      setMenuState();
2132      makeNoticesDismissible();
2133      aria_button_if_js();
2134  
2135      $document.on( 'wp-pin-menu wp-window-resized.pin-menu postboxes-columnchange.pin-menu postbox-toggled.pin-menu wp-collapse-menu.pin-menu wp-scroll-start.pin-menu', setPinMenu );
2136  
2137      // Set initial focus on a specific element.
2138      $( '.wp-initial-focus' ).trigger( 'focus' );
2139  
2140      // Toggle update details on update-core.php.
2141      $body.on( 'click', '.js-update-details-toggle', function() {
2142          var $updateNotice = $( this ).closest( '.js-update-details' ),
2143              $progressDiv = $( '#' + $updateNotice.data( 'update-details' ) );
2144  
2145          /*
2146           * When clicking on "Show details" move the progress div below the update
2147           * notice. Make sure it gets moved just the first time.
2148           */
2149          if ( ! $progressDiv.hasClass( 'update-details-moved' ) ) {
2150              $progressDiv.insertAfter( $updateNotice ).addClass( 'update-details-moved' );
2151          }
2152  
2153          // Toggle the progress div visibility.
2154          $progressDiv.toggle();
2155          // Toggle the Show Details button expanded state.
2156          $( this ).attr( 'aria-expanded', $progressDiv.is( ':visible' ) );
2157      });
2158  });
2159  
2160  /**
2161   * Hides the update button for expired plugin or theme uploads.
2162   *
2163   * On the "Update plugin/theme from uploaded zip" screen, once the upload has expired,
2164   * hides the "Replace current with uploaded" button and displays a warning.
2165   *
2166   * @since 5.5.0
2167   */
2168  $( function( $ ) {
2169      var $overwrite, $warning;
2170  
2171      if ( ! $body.hasClass( 'update-php' ) ) {
2172          return;
2173      }
2174  
2175      $overwrite = $( 'a.update-from-upload-overwrite' );
2176      $warning   = $( '.update-from-upload-expired' );
2177  
2178      if ( ! $overwrite.length || ! $warning.length ) {
2179          return;
2180      }
2181  
2182      window.setTimeout(
2183          function() {
2184              $overwrite.hide();
2185              $warning.removeClass( 'hidden' );
2186  
2187              if ( window.wp && window.wp.a11y ) {
2188                  window.wp.a11y.speak( $warning.text() );
2189              }
2190          },
2191          7140000 // 119 minutes. The uploaded file is deleted after 2 hours.
2192      );
2193  } );
2194  
2195  // Fire a custom jQuery event at the end of window resize.
2196  ( function() {
2197      var timeout;
2198  
2199      /**
2200       * Triggers the WP window-resize event.
2201       *
2202       * @since 3.8.0
2203       *
2204       * @return {void}
2205       */
2206  	function triggerEvent() {
2207          $document.trigger( 'wp-window-resized' );
2208      }
2209  
2210      /**
2211       * Fires the trigger event again after 200 ms.
2212       *
2213       * @since 3.8.0
2214       *
2215       * @return {void}
2216       */
2217  	function fireOnce() {
2218          window.clearTimeout( timeout );
2219          timeout = window.setTimeout( triggerEvent, 200 );
2220      }
2221  
2222      $window.on( 'resize.wp-fire-once', fireOnce );
2223  }());
2224  
2225  // Make Windows 8 devices play along nicely.
2226  (function(){
2227      if ( '-ms-user-select' in document.documentElement.style && navigator.userAgent.match(/IEMobile\/10\.0/) ) {
2228          var msViewportStyle = document.createElement( 'style' );
2229          msViewportStyle.appendChild(
2230              document.createTextNode( '@-ms-viewport{width:auto!important}' )
2231          );
2232          document.getElementsByTagName( 'head' )[0].appendChild( msViewportStyle );
2233      }
2234  })();
2235  
2236  }( jQuery, window ));
2237  
2238  /**
2239   * Freeze animated plugin icons when reduced motion is enabled.
2240   *
2241   * When the user has enabled the 'prefers-reduced-motion' setting, this module
2242   * stops animations for all GIFs on the page with the class 'plugin-icon' or
2243   * plugin icon images in the update plugins table.
2244   *
2245   * @since 6.4.0
2246   */
2247  (function() {
2248      // Private variables and methods.
2249      var priv = {},
2250          pub = {},
2251          mediaQuery;
2252  
2253      // Initialize pauseAll to false; it will be set to true if reduced motion is preferred.
2254      priv.pauseAll = false;
2255      if ( window.matchMedia ) {
2256          mediaQuery = window.matchMedia( '(prefers-reduced-motion: reduce)' );
2257          if ( ! mediaQuery || mediaQuery.matches ) {
2258              priv.pauseAll = true;
2259          }
2260      }
2261  
2262      // Method to replace animated GIFs with a static frame.
2263      priv.freezeAnimatedPluginIcons = function( img ) {
2264          var coverImage = function() {
2265              var width = img.width;
2266              var height = img.height;
2267              var canvas = document.createElement( 'canvas' );
2268  
2269              // Set canvas dimensions.
2270              canvas.width = width;
2271              canvas.height = height;
2272  
2273              // Copy classes from the image to the canvas.
2274              canvas.className = img.className;
2275  
2276              // Check if the image is inside a specific table.
2277              var isInsideUpdateTable = img.closest( '#update-plugins-table' );
2278  
2279              if ( isInsideUpdateTable ) {
2280                  // Transfer computed styles from image to canvas.
2281                  var computedStyles = window.getComputedStyle( img ),
2282                      i, max;
2283                  for ( i = 0, max = computedStyles.length; i < max; i++ ) {
2284                      var propName = computedStyles[ i ];
2285                      var propValue = computedStyles.getPropertyValue( propName );
2286                      canvas.style[ propName ] = propValue;
2287                  }
2288              }
2289  
2290              // Draw the image onto the canvas.
2291              canvas.getContext( '2d' ).drawImage( img, 0, 0, width, height );
2292  
2293              // Set accessibility attributes on canvas.
2294              canvas.setAttribute( 'aria-hidden', 'true' );
2295              canvas.setAttribute( 'role', 'presentation' );
2296  
2297              // Insert canvas before the image and set the image to be near-invisible.
2298              var parent = img.parentNode;
2299              parent.insertBefore( canvas, img );
2300              img.style.opacity = 0.01;
2301              img.style.width = '0px';
2302              img.style.height = '0px';
2303          };
2304  
2305          // If the image is already loaded, apply the coverImage function.
2306          if ( img.complete ) {
2307              coverImage();
2308          } else {
2309              // Otherwise, wait for the image to load.
2310              img.addEventListener( 'load', coverImage, true );
2311          }
2312      };
2313  
2314      // Public method to freeze all relevant GIFs on the page.
2315      pub.freezeAll = function() {
2316          var images = document.querySelectorAll( '.plugin-icon, #update-plugins-table img' );
2317          for ( var x = 0; x < images.length; x++ ) {
2318              if ( /\.gif(?:\?|$)/i.test( images[ x ].src ) ) {
2319                  priv.freezeAnimatedPluginIcons( images[ x ] );
2320              }
2321          }
2322      };
2323  
2324      // Only run the freezeAll method if the user prefers reduced motion.
2325      if ( true === priv.pauseAll ) {
2326          pub.freezeAll();
2327      }
2328  
2329      // Listen for jQuery AJAX events.
2330      ( function( $ ) {
2331          if ( window.pagenow === 'plugin-install' ) {
2332              // Only listen for ajaxComplete if this is the plugin-install.php page.
2333              $( document ).ajaxComplete( function( event, xhr, settings ) {
2334  
2335                  // Check if this is the 'search-install-plugins' request.
2336                  if ( settings.data && typeof settings.data === 'string' && settings.data.includes( 'action=search-install-plugins' ) ) {
2337                      // Recheck if the user prefers reduced motion.
2338                      if ( window.matchMedia ) {
2339                          var mediaQuery = window.matchMedia( '(prefers-reduced-motion: reduce)' );
2340                          if ( mediaQuery.matches ) {
2341                              pub.freezeAll();
2342                          }
2343                      } else {
2344                          // Fallback for browsers that don't support matchMedia.
2345                          if ( true === priv.pauseAll ) {
2346                              pub.freezeAll();
2347                          }
2348                      }
2349                  }
2350              } );
2351          }
2352      } )( jQuery );
2353  
2354      // Expose public methods.
2355      return pub;
2356  })();


Generated : Thu Apr 3 08:20:01 2025 Cross-referenced by PHPXref