[ Index ]

PHP Cross Reference of WordPress Trunk (Updated Daily)

Search

title

Body

[close]

/wp-admin/js/ -> customize-nav-menus.js (source)

   1  /**
   2   * @output wp-admin/js/customize-nav-menus.js
   3   */
   4  
   5  /* global menus, _wpCustomizeNavMenusSettings, wpNavMenu, console */
   6  ( function( api, wp, $ ) {
   7      'use strict';
   8  
   9      /**
  10       * Set up wpNavMenu for drag and drop.
  11       */
  12      wpNavMenu.originalInit = wpNavMenu.init;
  13      wpNavMenu.options.menuItemDepthPerLevel = 20;
  14      wpNavMenu.options.sortableItems         = '> .customize-control-nav_menu_item';
  15      wpNavMenu.options.targetTolerance       = 10;
  16      wpNavMenu.init = function() {
  17          this.jQueryExtensions();
  18      };
  19  
  20      /**
  21       * @namespace wp.customize.Menus
  22       */
  23      api.Menus = api.Menus || {};
  24  
  25      // Link settings.
  26      api.Menus.data = {
  27          itemTypes: [],
  28          l10n: {},
  29          settingTransport: 'refresh',
  30          phpIntMax: 0,
  31          defaultSettingValues: {
  32              nav_menu: {},
  33              nav_menu_item: {}
  34          },
  35          locationSlugMappedToName: {}
  36      };
  37      if ( 'undefined' !== typeof _wpCustomizeNavMenusSettings ) {
  38          $.extend( api.Menus.data, _wpCustomizeNavMenusSettings );
  39      }
  40  
  41      /**
  42       * Newly-created Nav Menus and Nav Menu Items have negative integer IDs which
  43       * serve as placeholders until Save & Publish happens.
  44       *
  45       * @alias wp.customize.Menus.generatePlaceholderAutoIncrementId
  46       *
  47       * @return {number}
  48       */
  49      api.Menus.generatePlaceholderAutoIncrementId = function() {
  50          return -Math.ceil( api.Menus.data.phpIntMax * Math.random() );
  51      };
  52  
  53      /**
  54       * wp.customize.Menus.AvailableItemModel
  55       *
  56       * A single available menu item model. See PHP's WP_Customize_Nav_Menu_Item_Setting class.
  57       *
  58       * @class    wp.customize.Menus.AvailableItemModel
  59       * @augments Backbone.Model
  60       */
  61      api.Menus.AvailableItemModel = Backbone.Model.extend( $.extend(
  62          {
  63              id: null // This is only used by Backbone.
  64          },
  65          api.Menus.data.defaultSettingValues.nav_menu_item
  66      ) );
  67  
  68      /**
  69       * wp.customize.Menus.AvailableItemCollection
  70       *
  71       * Collection for available menu item models.
  72       *
  73       * @class    wp.customize.Menus.AvailableItemCollection
  74       * @augments Backbone.Collection
  75       */
  76      api.Menus.AvailableItemCollection = Backbone.Collection.extend(/** @lends wp.customize.Menus.AvailableItemCollection.prototype */{
  77          model: api.Menus.AvailableItemModel,
  78  
  79          sort_key: 'order',
  80  
  81          comparator: function( item ) {
  82              return -item.get( this.sort_key );
  83          },
  84  
  85          sortByField: function( fieldName ) {
  86              this.sort_key = fieldName;
  87              this.sort();
  88          }
  89      });
  90      api.Menus.availableMenuItems = new api.Menus.AvailableItemCollection( api.Menus.data.availableMenuItems );
  91  
  92      /**
  93       * Insert a new `auto-draft` post.
  94       *
  95       * @since 4.7.0
  96       * @alias wp.customize.Menus.insertAutoDraftPost
  97       *
  98       * @param {Object} params - Parameters for the draft post to create.
  99       * @param {string} params.post_type - Post type to add.
 100       * @param {string} params.post_title - Post title to use.
 101       * @return {jQuery.promise} Promise resolved with the added post.
 102       */
 103      api.Menus.insertAutoDraftPost = function insertAutoDraftPost( params ) {
 104          var request, deferred = $.Deferred();
 105  
 106          request = wp.ajax.post( 'customize-nav-menus-insert-auto-draft', {
 107              'customize-menus-nonce': api.settings.nonce['customize-menus'],
 108              'wp_customize': 'on',
 109              'customize_changeset_uuid': api.settings.changeset.uuid,
 110              'params': params
 111          } );
 112  
 113          request.done( function( response ) {
 114              if ( response.post_id ) {
 115                  api( 'nav_menus_created_posts' ).set(
 116                      api( 'nav_menus_created_posts' ).get().concat( [ response.post_id ] )
 117                  );
 118  
 119                  if ( 'page' === params.post_type ) {
 120  
 121                      // Activate static front page controls as this could be the first page created.
 122                      if ( api.section.has( 'static_front_page' ) ) {
 123                          api.section( 'static_front_page' ).activate();
 124                      }
 125  
 126                      // Add new page to dropdown-pages controls.
 127                      api.control.each( function( control ) {
 128                          var select;
 129                          if ( 'dropdown-pages' === control.params.type ) {
 130                              select = control.container.find( 'select[name^="_customize-dropdown-pages-"]' );
 131                              select.append( new Option( params.post_title, response.post_id ) );
 132                          }
 133                      } );
 134                  }
 135                  deferred.resolve( response );
 136              }
 137          } );
 138  
 139          request.fail( function( response ) {
 140              var error = response || '';
 141  
 142              if ( 'undefined' !== typeof response.message ) {
 143                  error = response.message;
 144              }
 145  
 146              console.error( error );
 147              deferred.rejectWith( error );
 148          } );
 149  
 150          return deferred.promise();
 151      };
 152  
 153      api.Menus.AvailableMenuItemsPanelView = wp.Backbone.View.extend(/** @lends wp.customize.Menus.AvailableMenuItemsPanelView.prototype */{
 154  
 155          el: '#available-menu-items',
 156  
 157          events: {
 158              'input #menu-items-search': 'debounceSearch',
 159              'focus .menu-item-tpl': 'focus',
 160              'click .menu-item-tpl': '_submit',
 161              'click #custom-menu-item-submit': '_submitLink',
 162              'keypress #custom-menu-item-name': '_submitLink',
 163              'click .new-content-item .add-content': '_submitNew',
 164              'keypress .create-item-input': '_submitNew',
 165              'keydown': 'keyboardAccessible'
 166          },
 167  
 168          // Cache current selected menu item.
 169          selected: null,
 170  
 171          // Cache menu control that opened the panel.
 172          currentMenuControl: null,
 173          debounceSearch: null,
 174          $search: null,
 175          $clearResults: null,
 176          searchTerm: '',
 177          rendered: false,
 178          pages: {},
 179          sectionContent: '',
 180          loading: false,
 181          addingNew: false,
 182  
 183          /**
 184           * wp.customize.Menus.AvailableMenuItemsPanelView
 185           *
 186           * View class for the available menu items panel.
 187           *
 188           * @constructs wp.customize.Menus.AvailableMenuItemsPanelView
 189           * @augments   wp.Backbone.View
 190           */
 191          initialize: function() {
 192              var self = this;
 193  
 194              if ( ! api.panel.has( 'nav_menus' ) ) {
 195                  return;
 196              }
 197  
 198              this.$search = $( '#menu-items-search' );
 199              this.$clearResults = this.$el.find( '.clear-results' );
 200              this.sectionContent = this.$el.find( '.available-menu-items-list' );
 201  
 202              this.debounceSearch = _.debounce( self.search, 500 );
 203  
 204              _.bindAll( this, 'close' );
 205  
 206              /*
 207               * If the available menu items panel is open and the customize controls
 208               * are interacted with (other than an item being deleted), then close
 209               * the available menu items panel. Also close on back button click.
 210               */
 211              $( '#customize-controls, .customize-section-back' ).on( 'click keydown', function( e ) {
 212                  var isDeleteBtn = $( e.target ).is( '.item-delete, .item-delete *' ),
 213                      isAddNewBtn = $( e.target ).is( '.add-new-menu-item, .add-new-menu-item *' );
 214                  if ( $( 'body' ).hasClass( 'adding-menu-items' ) && ! isDeleteBtn && ! isAddNewBtn ) {
 215                      self.close();
 216                  }
 217              } );
 218  
 219              // Clear the search results and trigger an `input` event to fire a new search.
 220              this.$clearResults.on( 'click', function() {
 221                  self.$search.val( '' ).trigger( 'focus' ).trigger( 'input' );
 222              } );
 223  
 224              this.$el.on( 'input', '#custom-menu-item-name.invalid, #custom-menu-item-url.invalid', function() {
 225                  $( this ).removeClass( 'invalid' );
 226              });
 227  
 228              // Load available items if it looks like we'll need them.
 229              api.panel( 'nav_menus' ).container.on( 'expanded', function() {
 230                  if ( ! self.rendered ) {
 231                      self.initList();
 232                      self.rendered = true;
 233                  }
 234              });
 235  
 236              // Load more items.
 237              this.sectionContent.on( 'scroll', function() {
 238                  var totalHeight = self.$el.find( '.accordion-section.open .available-menu-items-list' ).prop( 'scrollHeight' ),
 239                      visibleHeight = self.$el.find( '.accordion-section.open' ).height();
 240  
 241                  if ( ! self.loading && $( this ).scrollTop() > 3 / 4 * totalHeight - visibleHeight ) {
 242                      var type = $( this ).data( 'type' ),
 243                          object = $( this ).data( 'object' );
 244  
 245                      if ( 'search' === type ) {
 246                          if ( self.searchTerm ) {
 247                              self.doSearch( self.pages.search );
 248                          }
 249                      } else {
 250                          self.loadItems( [
 251                              { type: type, object: object }
 252                          ] );
 253                      }
 254                  }
 255              });
 256  
 257              // Close the panel if the URL in the preview changes.
 258              api.previewer.bind( 'url', this.close );
 259  
 260              self.delegateEvents();
 261          },
 262  
 263          // Search input change handler.
 264          search: function( event ) {
 265              var $searchSection = $( '#available-menu-items-search' ),
 266                  $otherSections = $( '#available-menu-items .accordion-section' ).not( $searchSection );
 267  
 268              if ( ! event ) {
 269                  return;
 270              }
 271  
 272              if ( this.searchTerm === event.target.value ) {
 273                  return;
 274              }
 275  
 276              if ( '' !== event.target.value && ! $searchSection.hasClass( 'open' ) ) {
 277                  $otherSections.fadeOut( 100 );
 278                  $searchSection.find( '.accordion-section-content' ).slideDown( 'fast' );
 279                  $searchSection.addClass( 'open' );
 280                  this.$clearResults.addClass( 'is-visible' );
 281              } else if ( '' === event.target.value ) {
 282                  $searchSection.removeClass( 'open' );
 283                  $otherSections.show();
 284                  this.$clearResults.removeClass( 'is-visible' );
 285              }
 286  
 287              this.searchTerm = event.target.value;
 288              this.pages.search = 1;
 289              this.doSearch( 1 );
 290          },
 291  
 292          // Get search results.
 293          doSearch: function( page ) {
 294              var self = this, params,
 295                  $section = $( '#available-menu-items-search' ),
 296                  $content = $section.find( '.accordion-section-content' ),
 297                  itemTemplate = wp.template( 'available-menu-item' );
 298  
 299              if ( self.currentRequest ) {
 300                  self.currentRequest.abort();
 301              }
 302  
 303              if ( page < 0 ) {
 304                  return;
 305              } else if ( page > 1 ) {
 306                  $section.addClass( 'loading-more' );
 307                  $content.attr( 'aria-busy', 'true' );
 308                  wp.a11y.speak( api.Menus.data.l10n.itemsLoadingMore );
 309              } else if ( '' === self.searchTerm ) {
 310                  $content.html( '' );
 311                  wp.a11y.speak( '' );
 312                  return;
 313              }
 314  
 315              $section.addClass( 'loading' );
 316              self.loading = true;
 317  
 318              params = api.previewer.query( { excludeCustomizedSaved: true } );
 319              _.extend( params, {
 320                  'customize-menus-nonce': api.settings.nonce['customize-menus'],
 321                  'wp_customize': 'on',
 322                  'search': self.searchTerm,
 323                  'page': page
 324              } );
 325  
 326              self.currentRequest = wp.ajax.post( 'search-available-menu-items-customizer', params );
 327  
 328              self.currentRequest.done(function( data ) {
 329                  var items;
 330                  if ( 1 === page ) {
 331                      // Clear previous results as it's a new search.
 332                      $content.empty();
 333                  }
 334                  $section.removeClass( 'loading loading-more' );
 335                  $content.attr( 'aria-busy', 'false' );
 336                  $section.addClass( 'open' );
 337                  self.loading = false;
 338                  items = new api.Menus.AvailableItemCollection( data.items );
 339                  self.collection.add( items.models );
 340                  items.each( function( menuItem ) {
 341                      $content.append( itemTemplate( menuItem.attributes ) );
 342                  } );
 343                  if ( 20 > items.length ) {
 344                      self.pages.search = -1; // Up to 20 posts and 20 terms in results, if <20, no more results for either.
 345                  } else {
 346                      self.pages.search = self.pages.search + 1;
 347                  }
 348                  if ( items && page > 1 ) {
 349                      wp.a11y.speak( api.Menus.data.l10n.itemsFoundMore.replace( '%d', items.length ) );
 350                  } else if ( items && page === 1 ) {
 351                      wp.a11y.speak( api.Menus.data.l10n.itemsFound.replace( '%d', items.length ) );
 352                  }
 353              });
 354  
 355              self.currentRequest.fail(function( data ) {
 356                  // data.message may be undefined, for example when typing slow and the request is aborted.
 357                  if ( data.message ) {
 358                      $content.empty().append( $( '<li class="nothing-found"></li>' ).text( data.message ) );
 359                      wp.a11y.speak( data.message );
 360                  }
 361                  self.pages.search = -1;
 362              });
 363  
 364              self.currentRequest.always(function() {
 365                  $section.removeClass( 'loading loading-more' );
 366                  $content.attr( 'aria-busy', 'false' );
 367                  self.loading = false;
 368                  self.currentRequest = null;
 369              });
 370          },
 371  
 372          // Render the individual items.
 373          initList: function() {
 374              var self = this;
 375  
 376              // Render the template for each item by type.
 377              _.each( api.Menus.data.itemTypes, function( itemType ) {
 378                  self.pages[ itemType.type + ':' + itemType.object ] = 0;
 379              } );
 380              self.loadItems( api.Menus.data.itemTypes );
 381          },
 382  
 383          /**
 384           * Load available nav menu items.
 385           *
 386           * @since 4.3.0
 387           * @since 4.7.0 Changed function signature to take list of item types instead of single type/object.
 388           * @access private
 389           *
 390           * @param {Array.<Object>} itemTypes List of objects containing type and key.
 391           * @param {string} deprecated Formerly the object parameter.
 392           * @return {void}
 393           */
 394          loadItems: function( itemTypes, deprecated ) {
 395              var self = this, _itemTypes, requestItemTypes = [], params, request, itemTemplate, availableMenuItemContainers = {};
 396              itemTemplate = wp.template( 'available-menu-item' );
 397  
 398              if ( _.isString( itemTypes ) && _.isString( deprecated ) ) {
 399                  _itemTypes = [ { type: itemTypes, object: deprecated } ];
 400              } else {
 401                  _itemTypes = itemTypes;
 402              }
 403  
 404              _.each( _itemTypes, function( itemType ) {
 405                  var container, name = itemType.type + ':' + itemType.object;
 406                  if ( -1 === self.pages[ name ] ) {
 407                      return; // Skip types for which there are no more results.
 408                  }
 409                  container = $( '#available-menu-items-' + itemType.type + '-' + itemType.object );
 410                  container.find( '.accordion-section-title' ).addClass( 'loading' );
 411                  availableMenuItemContainers[ name ] = container;
 412  
 413                  requestItemTypes.push( {
 414                      object: itemType.object,
 415                      type: itemType.type,
 416                      page: self.pages[ name ]
 417                  } );
 418              } );
 419  
 420              if ( 0 === requestItemTypes.length ) {
 421                  return;
 422              }
 423  
 424              self.loading = true;
 425  
 426              params = api.previewer.query( { excludeCustomizedSaved: true } );
 427              _.extend( params, {
 428                  'customize-menus-nonce': api.settings.nonce['customize-menus'],
 429                  'wp_customize': 'on',
 430                  'item_types': requestItemTypes
 431              } );
 432  
 433              request = wp.ajax.post( 'load-available-menu-items-customizer', params );
 434  
 435              request.done(function( data ) {
 436                  var typeInner;
 437                  _.each( data.items, function( typeItems, name ) {
 438                      if ( 0 === typeItems.length ) {
 439                          if ( 0 === self.pages[ name ] ) {
 440                              availableMenuItemContainers[ name ].find( '.accordion-section-title' )
 441                                  .addClass( 'cannot-expand' )
 442                                  .removeClass( 'loading' )
 443                                  .find( '.accordion-section-title > button' )
 444                                  .prop( 'tabIndex', -1 );
 445                          }
 446                          self.pages[ name ] = -1;
 447                          return;
 448                      } else if ( ( 'post_type:page' === name ) && ( ! availableMenuItemContainers[ name ].hasClass( 'open' ) ) ) {
 449                          availableMenuItemContainers[ name ].find( '.accordion-section-title > button' ).trigger( 'click' );
 450                      }
 451                      typeItems = new api.Menus.AvailableItemCollection( typeItems ); // @todo Why is this collection created and then thrown away?
 452                      self.collection.add( typeItems.models );
 453                      typeInner = availableMenuItemContainers[ name ].find( '.available-menu-items-list' );
 454                      typeItems.each( function( menuItem ) {
 455                          typeInner.append( itemTemplate( menuItem.attributes ) );
 456                      } );
 457                      self.pages[ name ] += 1;
 458                  });
 459              });
 460              request.fail(function( data ) {
 461                  if ( typeof console !== 'undefined' && console.error ) {
 462                      console.error( data );
 463                  }
 464              });
 465              request.always(function() {
 466                  _.each( availableMenuItemContainers, function( container ) {
 467                      container.find( '.accordion-section-title' ).removeClass( 'loading' );
 468                  } );
 469                  self.loading = false;
 470              });
 471          },
 472  
 473          // Adjust the height of each section of items to fit the screen.
 474          itemSectionHeight: function() {
 475              var sections, lists, totalHeight, accordionHeight, diff;
 476              totalHeight = window.innerHeight;
 477              sections = this.$el.find( '.accordion-section:not( #available-menu-items-search ) .accordion-section-content' );
 478              lists = this.$el.find( '.accordion-section:not( #available-menu-items-search ) .available-menu-items-list:not(":only-child")' );
 479              accordionHeight =  46 * ( 1 + sections.length ) + 14; // Magic numbers.
 480              diff = totalHeight - accordionHeight;
 481              if ( 120 < diff && 290 > diff ) {
 482                  sections.css( 'max-height', diff );
 483                  lists.css( 'max-height', ( diff - 60 ) );
 484              }
 485          },
 486  
 487          // Highlights a menu item.
 488          select: function( menuitemTpl ) {
 489              this.selected = $( menuitemTpl );
 490              this.selected.siblings( '.menu-item-tpl' ).removeClass( 'selected' );
 491              this.selected.addClass( 'selected' );
 492          },
 493  
 494          // Highlights a menu item on focus.
 495          focus: function( event ) {
 496              this.select( $( event.currentTarget ) );
 497          },
 498  
 499          // Submit handler for keypress and click on menu item.
 500          _submit: function( event ) {
 501              // Only proceed with keypress if it is Enter or Spacebar.
 502              if ( 'keypress' === event.type && ( 13 !== event.which && 32 !== event.which ) ) {
 503                  return;
 504              }
 505  
 506              this.submit( $( event.currentTarget ) );
 507          },
 508  
 509          // Adds a selected menu item to the menu.
 510          submit: function( menuitemTpl ) {
 511              var menuitemId, menu_item;
 512  
 513              if ( ! menuitemTpl ) {
 514                  menuitemTpl = this.selected;
 515              }
 516  
 517              if ( ! menuitemTpl || ! this.currentMenuControl ) {
 518                  return;
 519              }
 520  
 521              this.select( menuitemTpl );
 522  
 523              menuitemId = $( this.selected ).data( 'menu-item-id' );
 524              menu_item = this.collection.findWhere( { id: menuitemId } );
 525              if ( ! menu_item ) {
 526                  return;
 527              }
 528  
 529              this.currentMenuControl.addItemToMenu( menu_item.attributes );
 530  
 531              $( menuitemTpl ).find( '.menu-item-handle' ).addClass( 'item-added' );
 532          },
 533  
 534          // Submit handler for keypress and click on custom menu item.
 535          _submitLink: function( event ) {
 536              // Only proceed with keypress if it is Enter.
 537              if ( 'keypress' === event.type && 13 !== event.which ) {
 538                  return;
 539              }
 540  
 541              this.submitLink();
 542          },
 543  
 544          // Adds the custom menu item to the menu.
 545          submitLink: function() {
 546              var menuItem,
 547                  itemName = $( '#custom-menu-item-name' ),
 548                  itemUrl = $( '#custom-menu-item-url' ),
 549                  url = itemUrl.val().trim(),
 550                  urlRegex;
 551  
 552              if ( ! this.currentMenuControl ) {
 553                  return;
 554              }
 555  
 556              /*
 557               * Allow URLs including:
 558               * - http://example.com/
 559               * - //example.com
 560               * - /directory/
 561               * - ?query-param
 562               * - #target
 563               * - mailto:foo@example.com
 564               *
 565               * Any further validation will be handled on the server when the setting is attempted to be saved,
 566               * so this pattern does not need to be complete.
 567               */
 568              urlRegex = /^((\w+:)?\/\/\w.*|\w+:(?!\/\/$)|\/|\?|#)/;
 569  
 570              if ( '' === itemName.val() ) {
 571                  itemName.addClass( 'invalid' );
 572                  return;
 573              } else if ( ! urlRegex.test( url ) ) {
 574                  itemUrl.addClass( 'invalid' );
 575                  return;
 576              }
 577  
 578              menuItem = {
 579                  'title': itemName.val(),
 580                  'url': url,
 581                  'type': 'custom',
 582                  'type_label': api.Menus.data.l10n.custom_label,
 583                  'object': 'custom'
 584              };
 585  
 586              this.currentMenuControl.addItemToMenu( menuItem );
 587  
 588              // Reset the custom link form.
 589              itemUrl.val( '' ).attr( 'placeholder', 'https://' );
 590              itemName.val( '' );
 591          },
 592  
 593          /**
 594           * Submit handler for keypress (enter) on field and click on button.
 595           *
 596           * @since 4.7.0
 597           * @private
 598           *
 599           * @param {jQuery.Event} event Event.
 600           * @return {void}
 601           */
 602          _submitNew: function( event ) {
 603              var container;
 604  
 605              // Only proceed with keypress if it is Enter.
 606              if ( 'keypress' === event.type && 13 !== event.which ) {
 607                  return;
 608              }
 609  
 610              if ( this.addingNew ) {
 611                  return;
 612              }
 613  
 614              container = $( event.target ).closest( '.accordion-section' );
 615  
 616              this.submitNew( container );
 617          },
 618  
 619          /**
 620           * Creates a new object and adds an associated menu item to the menu.
 621           *
 622           * @since 4.7.0
 623           * @private
 624           *
 625           * @param {jQuery} container
 626           * @return {void}
 627           */
 628          submitNew: function( container ) {
 629              var panel = this,
 630                  itemName = container.find( '.create-item-input' ),
 631                  title = itemName.val(),
 632                  dataContainer = container.find( '.available-menu-items-list' ),
 633                  itemType = dataContainer.data( 'type' ),
 634                  itemObject = dataContainer.data( 'object' ),
 635                  itemTypeLabel = dataContainer.data( 'type_label' ),
 636                  promise;
 637  
 638              if ( ! this.currentMenuControl ) {
 639                  return;
 640              }
 641  
 642              // Only posts are supported currently.
 643              if ( 'post_type' !== itemType ) {
 644                  return;
 645              }
 646  
 647              if ( '' === itemName.val().trim() ) {
 648                  itemName.addClass( 'invalid' );
 649                  itemName.focus();
 650                  return;
 651              } else {
 652                  itemName.removeClass( 'invalid' );
 653                  container.find( '.accordion-section-title' ).addClass( 'loading' );
 654              }
 655  
 656              panel.addingNew = true;
 657              itemName.attr( 'disabled', 'disabled' );
 658              promise = api.Menus.insertAutoDraftPost( {
 659                  post_title: title,
 660                  post_type: itemObject
 661              } );
 662              promise.done( function( data ) {
 663                  var availableItem, $content, itemElement;
 664                  availableItem = new api.Menus.AvailableItemModel( {
 665                      'id': 'post-' + data.post_id, // Used for available menu item Backbone models.
 666                      'title': itemName.val(),
 667                      'type': itemType,
 668                      'type_label': itemTypeLabel,
 669                      'object': itemObject,
 670                      'object_id': data.post_id,
 671                      'url': data.url
 672                  } );
 673  
 674                  // Add new item to menu.
 675                  panel.currentMenuControl.addItemToMenu( availableItem.attributes );
 676  
 677                  // Add the new item to the list of available items.
 678                  api.Menus.availableMenuItemsPanel.collection.add( availableItem );
 679                  $content = container.find( '.available-menu-items-list' );
 680                  itemElement = $( wp.template( 'available-menu-item' )( availableItem.attributes ) );
 681                  itemElement.find( '.menu-item-handle:first' ).addClass( 'item-added' );
 682                  $content.prepend( itemElement );
 683                  $content.scrollTop();
 684  
 685                  // Reset the create content form.
 686                  itemName.val( '' ).removeAttr( 'disabled' );
 687                  panel.addingNew = false;
 688                  container.find( '.accordion-section-title' ).removeClass( 'loading' );
 689              } );
 690          },
 691  
 692          // Opens the panel.
 693          open: function( menuControl ) {
 694              var panel = this, close;
 695  
 696              this.currentMenuControl = menuControl;
 697  
 698              this.itemSectionHeight();
 699  
 700              if ( api.section.has( 'publish_settings' ) ) {
 701                  api.section( 'publish_settings' ).collapse();
 702              }
 703  
 704              $( 'body' ).addClass( 'adding-menu-items' );
 705  
 706              close = function() {
 707                  panel.close();
 708                  $( this ).off( 'click', close );
 709              };
 710              $( '#customize-preview' ).on( 'click', close );
 711  
 712              // Collapse all controls.
 713              _( this.currentMenuControl.getMenuItemControls() ).each( function( control ) {
 714                  control.collapseForm();
 715              } );
 716  
 717              this.$el.find( '.selected' ).removeClass( 'selected' );
 718  
 719              this.$search.trigger( 'focus' );
 720          },
 721  
 722          // Closes the panel.
 723          close: function( options ) {
 724              options = options || {};
 725  
 726              if ( options.returnFocus && this.currentMenuControl ) {
 727                  this.currentMenuControl.container.find( '.add-new-menu-item' ).focus();
 728              }
 729  
 730              this.currentMenuControl = null;
 731              this.selected = null;
 732  
 733              $( 'body' ).removeClass( 'adding-menu-items' );
 734              $( '#available-menu-items .menu-item-handle.item-added' ).removeClass( 'item-added' );
 735  
 736              this.$search.val( '' ).trigger( 'input' );
 737          },
 738  
 739          // Add a few keyboard enhancements to the panel.
 740          keyboardAccessible: function( event ) {
 741              var isEnter = ( 13 === event.which ),
 742                  isEsc = ( 27 === event.which ),
 743                  isBackTab = ( 9 === event.which && event.shiftKey ),
 744                  isSearchFocused = $( event.target ).is( this.$search );
 745  
 746              // If enter pressed but nothing entered, don't do anything.
 747              if ( isEnter && ! this.$search.val() ) {
 748                  return;
 749              }
 750  
 751              if ( isSearchFocused && isBackTab ) {
 752                  this.currentMenuControl.container.find( '.add-new-menu-item' ).focus();
 753                  event.preventDefault(); // Avoid additional back-tab.
 754              } else if ( isEsc ) {
 755                  this.close( { returnFocus: true } );
 756              }
 757          }
 758      });
 759  
 760      /**
 761       * wp.customize.Menus.MenusPanel
 762       *
 763       * Customizer panel for menus. This is used only for screen options management.
 764       * Note that 'menus' must match the WP_Customize_Menu_Panel::$type.
 765       *
 766       * @class    wp.customize.Menus.MenusPanel
 767       * @augments wp.customize.Panel
 768       */
 769      api.Menus.MenusPanel = api.Panel.extend(/** @lends wp.customize.Menus.MenusPanel.prototype */{
 770  
 771          attachEvents: function() {
 772              api.Panel.prototype.attachEvents.call( this );
 773  
 774              var panel = this,
 775                  panelMeta = panel.container.find( '.panel-meta' ),
 776                  help = panelMeta.find( '.customize-help-toggle' ),
 777                  content = panelMeta.find( '.customize-panel-description' ),
 778                  options = $( '#screen-options-wrap' ),
 779                  button = panelMeta.find( '.customize-screen-options-toggle' );
 780              button.on( 'click keydown', function( event ) {
 781                  if ( api.utils.isKeydownButNotEnterEvent( event ) ) {
 782                      return;
 783                  }
 784                  event.preventDefault();
 785  
 786                  // Hide description.
 787                  if ( content.not( ':hidden' ) ) {
 788                      content.slideUp( 'fast' );
 789                      help.attr( 'aria-expanded', 'false' );
 790                  }
 791  
 792                  if ( 'true' === button.attr( 'aria-expanded' ) ) {
 793                      button.attr( 'aria-expanded', 'false' );
 794                      panelMeta.removeClass( 'open' );
 795                      panelMeta.removeClass( 'active-menu-screen-options' );
 796                      options.slideUp( 'fast' );
 797                  } else {
 798                      button.attr( 'aria-expanded', 'true' );
 799                      panelMeta.addClass( 'open' );
 800                      panelMeta.addClass( 'active-menu-screen-options' );
 801                      options.slideDown( 'fast' );
 802                  }
 803  
 804                  return false;
 805              } );
 806  
 807              // Help toggle.
 808              help.on( 'click keydown', function( event ) {
 809                  if ( api.utils.isKeydownButNotEnterEvent( event ) ) {
 810                      return;
 811                  }
 812                  event.preventDefault();
 813  
 814                  if ( 'true' === button.attr( 'aria-expanded' ) ) {
 815                      button.attr( 'aria-expanded', 'false' );
 816                      help.attr( 'aria-expanded', 'true' );
 817                      panelMeta.addClass( 'open' );
 818                      panelMeta.removeClass( 'active-menu-screen-options' );
 819                      options.slideUp( 'fast' );
 820                      content.slideDown( 'fast' );
 821                  }
 822              } );
 823          },
 824  
 825          /**
 826           * Update field visibility when clicking on the field toggles.
 827           */
 828          ready: function() {
 829              var panel = this;
 830              panel.container.find( '.hide-column-tog' ).on( 'click', function() {
 831                  panel.saveManageColumnsState();
 832              });
 833  
 834              // Inject additional heading into the menu locations section's head container.
 835              api.section( 'menu_locations', function( section ) {
 836                  section.headContainer.prepend(
 837                      wp.template( 'nav-menu-locations-header' )( api.Menus.data )
 838                  );
 839              } );
 840          },
 841  
 842          /**
 843           * Save hidden column states.
 844           *
 845           * @since 4.3.0
 846           * @private
 847           *
 848           * @return {void}
 849           */
 850          saveManageColumnsState: _.debounce( function() {
 851              var panel = this;
 852              if ( panel._updateHiddenColumnsRequest ) {
 853                  panel._updateHiddenColumnsRequest.abort();
 854              }
 855  
 856              panel._updateHiddenColumnsRequest = wp.ajax.post( 'hidden-columns', {
 857                  hidden: panel.hidden(),
 858                  screenoptionnonce: $( '#screenoptionnonce' ).val(),
 859                  page: 'nav-menus'
 860              } );
 861              panel._updateHiddenColumnsRequest.always( function() {
 862                  panel._updateHiddenColumnsRequest = null;
 863              } );
 864          }, 2000 ),
 865  
 866          /**
 867           * @deprecated Since 4.7.0 now that the nav_menu sections are responsible for toggling the classes on their own containers.
 868           */
 869          checked: function() {},
 870  
 871          /**
 872           * @deprecated Since 4.7.0 now that the nav_menu sections are responsible for toggling the classes on their own containers.
 873           */
 874          unchecked: function() {},
 875  
 876          /**
 877           * Get hidden fields.
 878           *
 879           * @since 4.3.0
 880           * @private
 881           *
 882           * @return {Array} Fields (columns) that are hidden.
 883           */
 884          hidden: function() {
 885              return $( '.hide-column-tog' ).not( ':checked' ).map( function() {
 886                  var id = this.id;
 887                  return id.substring( 0, id.length - 5 );
 888              }).get().join( ',' );
 889          }
 890      } );
 891  
 892      /**
 893       * wp.customize.Menus.MenuSection
 894       *
 895       * Customizer section for menus. This is used only for lazy-loading child controls.
 896       * Note that 'nav_menu' must match the WP_Customize_Menu_Section::$type.
 897       *
 898       * @class    wp.customize.Menus.MenuSection
 899       * @augments wp.customize.Section
 900       */
 901      api.Menus.MenuSection = api.Section.extend(/** @lends wp.customize.Menus.MenuSection.prototype */{
 902  
 903          /**
 904           * Initialize.
 905           *
 906           * @since 4.3.0
 907           *
 908           * @param {string} id
 909           * @param {Object} options
 910           */
 911          initialize: function( id, options ) {
 912              var section = this;
 913              api.Section.prototype.initialize.call( section, id, options );
 914              section.deferred.initSortables = $.Deferred();
 915          },
 916  
 917          /**
 918           * Ready.
 919           */
 920          ready: function() {
 921              var section = this, fieldActiveToggles, handleFieldActiveToggle;
 922  
 923              if ( 'undefined' === typeof section.params.menu_id ) {
 924                  throw new Error( 'params.menu_id was not defined' );
 925              }
 926  
 927              /*
 928               * Since newly created sections won't be registered in PHP, we need to prevent the
 929               * preview's sending of the activeSections to result in this control
 930               * being deactivated when the preview refreshes. So we can hook onto
 931               * the setting that has the same ID and its presence can dictate
 932               * whether the section is active.
 933               */
 934              section.active.validate = function() {
 935                  if ( ! api.has( section.id ) ) {
 936                      return false;
 937                  }
 938                  return !! api( section.id ).get();
 939              };
 940  
 941              section.populateControls();
 942  
 943              section.navMenuLocationSettings = {};
 944              section.assignedLocations = new api.Value( [] );
 945  
 946              api.each(function( setting, id ) {
 947                  var matches = id.match( /^nav_menu_locations\[(.+?)]/ );
 948                  if ( matches ) {
 949                      section.navMenuLocationSettings[ matches[1] ] = setting;
 950                      setting.bind( function() {
 951                          section.refreshAssignedLocations();
 952                      });
 953                  }
 954              });
 955  
 956              section.assignedLocations.bind(function( to ) {
 957                  section.updateAssignedLocationsInSectionTitle( to );
 958              });
 959  
 960              section.refreshAssignedLocations();
 961  
 962              api.bind( 'pane-contents-reflowed', function() {
 963                  // Skip menus that have been removed.
 964                  if ( ! section.contentContainer.parent().length ) {
 965                      return;
 966                  }
 967                  section.container.find( '.menu-item .menu-item-reorder-nav button' ).attr({ 'tabindex': '0', 'aria-hidden': 'false' });
 968                  section.container.find( '.menu-item.move-up-disabled .menus-move-up' ).attr({ 'tabindex': '-1', 'aria-hidden': 'true' });
 969                  section.container.find( '.menu-item.move-down-disabled .menus-move-down' ).attr({ 'tabindex': '-1', 'aria-hidden': 'true' });
 970                  section.container.find( '.menu-item.move-left-disabled .menus-move-left' ).attr({ 'tabindex': '-1', 'aria-hidden': 'true' });
 971                  section.container.find( '.menu-item.move-right-disabled .menus-move-right' ).attr({ 'tabindex': '-1', 'aria-hidden': 'true' });
 972              } );
 973  
 974              /**
 975               * Update the active field class for the content container for a given checkbox toggle.
 976               *
 977               * @this {jQuery}
 978               * @return {void}
 979               */
 980              handleFieldActiveToggle = function() {
 981                  var className = 'field-' + $( this ).val() + '-active';
 982                  section.contentContainer.toggleClass( className, $( this ).prop( 'checked' ) );
 983              };
 984              fieldActiveToggles = api.panel( 'nav_menus' ).contentContainer.find( '.metabox-prefs:first' ).find( '.hide-column-tog' );
 985              fieldActiveToggles.each( handleFieldActiveToggle );
 986              fieldActiveToggles.on( 'click', handleFieldActiveToggle );
 987          },
 988  
 989          populateControls: function() {
 990              var section = this,
 991                  menuNameControlId,
 992                  menuLocationsControlId,
 993                  menuAutoAddControlId,
 994                  menuDeleteControlId,
 995                  menuControl,
 996                  menuNameControl,
 997                  menuLocationsControl,
 998                  menuAutoAddControl,
 999                  menuDeleteControl;
1000  
1001              // Add the control for managing the menu name.
1002              menuNameControlId = section.id + '[name]';
1003              menuNameControl = api.control( menuNameControlId );
1004              if ( ! menuNameControl ) {
1005                  menuNameControl = new api.controlConstructor.nav_menu_name( menuNameControlId, {
1006                      type: 'nav_menu_name',
1007                      label: api.Menus.data.l10n.menuNameLabel,
1008                      section: section.id,
1009                      priority: 0,
1010                      settings: {
1011                          'default': section.id
1012                      }
1013                  } );
1014                  api.control.add( menuNameControl );
1015                  menuNameControl.active.set( true );
1016              }
1017  
1018              // Add the menu control.
1019              menuControl = api.control( section.id );
1020              if ( ! menuControl ) {
1021                  menuControl = new api.controlConstructor.nav_menu( section.id, {
1022                      type: 'nav_menu',
1023                      section: section.id,
1024                      priority: 998,
1025                      settings: {
1026                          'default': section.id
1027                      },
1028                      menu_id: section.params.menu_id
1029                  } );
1030                  api.control.add( menuControl );
1031                  menuControl.active.set( true );
1032              }
1033  
1034              // Add the menu locations control.
1035              menuLocationsControlId = section.id + '[locations]';
1036              menuLocationsControl = api.control( menuLocationsControlId );
1037              if ( ! menuLocationsControl ) {
1038                  menuLocationsControl = new api.controlConstructor.nav_menu_locations( menuLocationsControlId, {
1039                      section: section.id,
1040                      priority: 999,
1041                      settings: {
1042                          'default': section.id
1043                      },
1044                      menu_id: section.params.menu_id
1045                  } );
1046                  api.control.add( menuLocationsControl.id, menuLocationsControl );
1047                  menuControl.active.set( true );
1048              }
1049  
1050              // Add the control for managing the menu auto_add.
1051              menuAutoAddControlId = section.id + '[auto_add]';
1052              menuAutoAddControl = api.control( menuAutoAddControlId );
1053              if ( ! menuAutoAddControl ) {
1054                  menuAutoAddControl = new api.controlConstructor.nav_menu_auto_add( menuAutoAddControlId, {
1055                      type: 'nav_menu_auto_add',
1056                      label: '',
1057                      section: section.id,
1058                      priority: 1000,
1059                      settings: {
1060                          'default': section.id
1061                      }
1062                  } );
1063                  api.control.add( menuAutoAddControl );
1064                  menuAutoAddControl.active.set( true );
1065              }
1066  
1067              // Add the control for deleting the menu.
1068              menuDeleteControlId = section.id + '[delete]';
1069              menuDeleteControl = api.control( menuDeleteControlId );
1070              if ( ! menuDeleteControl ) {
1071                  menuDeleteControl = new api.Control( menuDeleteControlId, {
1072                      section: section.id,
1073                      priority: 1001,
1074                      templateId: 'nav-menu-delete-button'
1075                  } );
1076                  api.control.add( menuDeleteControl.id, menuDeleteControl );
1077                  menuDeleteControl.active.set( true );
1078                  menuDeleteControl.deferred.embedded.done( function () {
1079                      menuDeleteControl.container.find( 'button' ).on( 'click', function() {
1080                          var menuId = section.params.menu_id;
1081                          var menuControl = api.Menus.getMenuControl( menuId );
1082                          menuControl.setting.set( false );
1083                      });
1084                  } );
1085              }
1086          },
1087  
1088          /**
1089           *
1090           */
1091          refreshAssignedLocations: function() {
1092              var section = this,
1093                  menuTermId = section.params.menu_id,
1094                  currentAssignedLocations = [];
1095              _.each( section.navMenuLocationSettings, function( setting, themeLocation ) {
1096                  if ( setting() === menuTermId ) {
1097                      currentAssignedLocations.push( themeLocation );
1098                  }
1099              });
1100              section.assignedLocations.set( currentAssignedLocations );
1101          },
1102  
1103          /**
1104           * @param {Array} themeLocationSlugs Theme location slugs.
1105           */
1106          updateAssignedLocationsInSectionTitle: function( themeLocationSlugs ) {
1107              var section = this,
1108                  $title;
1109  
1110              $title = section.container.find( '.accordion-section-title button:first' );
1111              $title.find( '.menu-in-location' ).remove();
1112              _.each( themeLocationSlugs, function( themeLocationSlug ) {
1113                  var $label, locationName;
1114                  $label = $( '<span class="menu-in-location"></span>' );
1115                  locationName = api.Menus.data.locationSlugMappedToName[ themeLocationSlug ];
1116                  $label.text( api.Menus.data.l10n.menuLocation.replace( '%s', locationName ) );
1117                  $title.append( $label );
1118              });
1119  
1120              section.container.toggleClass( 'assigned-to-menu-location', 0 !== themeLocationSlugs.length );
1121  
1122          },
1123  
1124          onChangeExpanded: function( expanded, args ) {
1125              var section = this, completeCallback;
1126  
1127              if ( expanded ) {
1128                  wpNavMenu.menuList = section.contentContainer;
1129                  wpNavMenu.targetList = wpNavMenu.menuList;
1130  
1131                  // Add attributes needed by wpNavMenu.
1132                  $( '#menu-to-edit' ).removeAttr( 'id' );
1133                  wpNavMenu.menuList.attr( 'id', 'menu-to-edit' ).addClass( 'menu' );
1134  
1135                  api.Menus.MenuItemControl.prototype.initAccessibility();
1136  
1137                  _.each( api.section( section.id ).controls(), function( control ) {
1138                      if ( 'nav_menu_item' === control.params.type ) {
1139                          control.actuallyEmbed();
1140                      }
1141                  } );
1142  
1143                  // Make sure Sortables is initialized after the section has been expanded to prevent `offset` issues.
1144                  if ( args.completeCallback ) {
1145                      completeCallback = args.completeCallback;
1146                  }
1147                  args.completeCallback = function() {
1148                      if ( 'resolved' !== section.deferred.initSortables.state() ) {
1149                          wpNavMenu.initSortables(); // Depends on menu-to-edit ID being set above.
1150                          section.deferred.initSortables.resolve( wpNavMenu.menuList ); // Now MenuControl can extend the sortable.
1151  
1152                          // @todo Note that wp.customize.reflowPaneContents() is debounced,
1153                          // so this immediate change will show a slight flicker while priorities get updated.
1154                          api.control( 'nav_menu[' + String( section.params.menu_id ) + ']' ).reflowMenuItems();
1155                      }
1156                      if ( _.isFunction( completeCallback ) ) {
1157                          completeCallback();
1158                      }
1159                  };
1160              }
1161              api.Section.prototype.onChangeExpanded.call( section, expanded, args );
1162          },
1163  
1164          /**
1165           * Highlight how a user may create new menu items.
1166           *
1167           * This method reminds the user to create new menu items and how.
1168           * It's exposed this way because this class knows best which UI needs
1169           * highlighted but those expanding this section know more about why and
1170           * when the affordance should be highlighted.
1171           *
1172           * @since 4.9.0
1173           *
1174           * @return {void}
1175           */
1176          highlightNewItemButton: function() {
1177              api.utils.highlightButton( this.contentContainer.find( '.add-new-menu-item' ), { delay: 2000 } );
1178          }
1179      });
1180  
1181      /**
1182       * Create a nav menu setting and section.
1183       *
1184       * @since 4.9.0
1185       *
1186       * @param {string} [name=''] Nav menu name.
1187       * @return {wp.customize.Menus.MenuSection} Added nav menu.
1188       */
1189      api.Menus.createNavMenu = function createNavMenu( name ) {
1190          var customizeId, placeholderId, setting;
1191          placeholderId = api.Menus.generatePlaceholderAutoIncrementId();
1192  
1193          customizeId = 'nav_menu[' + String( placeholderId ) + ']';
1194  
1195          // Register the menu control setting.
1196          setting = api.create( customizeId, customizeId, {}, {
1197              type: 'nav_menu',
1198              transport: api.Menus.data.settingTransport,
1199              previewer: api.previewer
1200          } );
1201          setting.set( $.extend(
1202              {},
1203              api.Menus.data.defaultSettingValues.nav_menu,
1204              {
1205                  name: name || ''
1206              }
1207          ) );
1208  
1209          /*
1210           * Add the menu section (and its controls).
1211           * Note that this will automatically create the required controls
1212           * inside via the Section's ready method.
1213           */
1214          return api.section.add( new api.Menus.MenuSection( customizeId, {
1215              panel: 'nav_menus',
1216              title: displayNavMenuName( name ),
1217              customizeAction: api.Menus.data.l10n.customizingMenus,
1218              priority: 10,
1219              menu_id: placeholderId
1220          } ) );
1221      };
1222  
1223      /**
1224       * wp.customize.Menus.NewMenuSection
1225       *
1226       * Customizer section for new menus.
1227       *
1228       * @class    wp.customize.Menus.NewMenuSection
1229       * @augments wp.customize.Section
1230       */
1231      api.Menus.NewMenuSection = api.Section.extend(/** @lends wp.customize.Menus.NewMenuSection.prototype */{
1232  
1233          /**
1234           * Add behaviors for the accordion section.
1235           *
1236           * @since 4.3.0
1237           */
1238          attachEvents: function() {
1239              var section = this,
1240                  container = section.container,
1241                  contentContainer = section.contentContainer,
1242                  navMenuSettingPattern = /^nav_menu\[/;
1243  
1244              section.headContainer.find( '.accordion-section-title' ).replaceWith(
1245                  wp.template( 'nav-menu-create-menu-section-title' )
1246              );
1247  
1248              /*
1249               * We have to manually handle section expanded because we do not
1250               * apply the `accordion-section-title` class to this button-driven section.
1251               */
1252              container.on( 'click', '.customize-add-menu-button', function() {
1253                  section.expand();
1254              });
1255  
1256              contentContainer.on( 'keydown', '.menu-name-field', function( event ) {
1257                  if ( 13 === event.which ) { // Enter.
1258                      section.submit();
1259                  }
1260              } );
1261              contentContainer.on( 'click', '#customize-new-menu-submit', function( event ) {
1262                  section.submit();
1263                  event.stopPropagation();
1264                  event.preventDefault();
1265              } );
1266  
1267              /**
1268               * Get number of non-deleted nav menus.
1269               *
1270               * @since 4.9.0
1271               * @return {number} Count.
1272               */
1273  			function getNavMenuCount() {
1274                  var count = 0;
1275                  api.each( function( setting ) {
1276                      if ( navMenuSettingPattern.test( setting.id ) && false !== setting.get() ) {
1277                          count += 1;
1278                      }
1279                  } );
1280                  return count;
1281              }
1282  
1283              /**
1284               * Update visibility of notice to prompt users to create menus.
1285               *
1286               * @since 4.9.0
1287               * @return {void}
1288               */
1289  			function updateNoticeVisibility() {
1290                  container.find( '.add-new-menu-notice' ).prop( 'hidden', getNavMenuCount() > 0 );
1291              }
1292  
1293              /**
1294               * Handle setting addition.
1295               *
1296               * @since 4.9.0
1297               * @param {wp.customize.Setting} setting - Added setting.
1298               * @return {void}
1299               */
1300  			function addChangeEventListener( setting ) {
1301                  if ( navMenuSettingPattern.test( setting.id ) ) {
1302                      setting.bind( updateNoticeVisibility );
1303                      updateNoticeVisibility();
1304                  }
1305              }
1306  
1307              /**
1308               * Handle setting removal.
1309               *
1310               * @since 4.9.0
1311               * @param {wp.customize.Setting} setting - Removed setting.
1312               * @return {void}
1313               */
1314  			function removeChangeEventListener( setting ) {
1315                  if ( navMenuSettingPattern.test( setting.id ) ) {
1316                      setting.unbind( updateNoticeVisibility );
1317                      updateNoticeVisibility();
1318                  }
1319              }
1320  
1321              api.each( addChangeEventListener );
1322              api.bind( 'add', addChangeEventListener );
1323              api.bind( 'removed', removeChangeEventListener );
1324              updateNoticeVisibility();
1325  
1326              api.Section.prototype.attachEvents.apply( section, arguments );
1327          },
1328  
1329          /**
1330           * Set up the control.
1331           *
1332           * @since 4.9.0
1333           */
1334          ready: function() {
1335              this.populateControls();
1336          },
1337  
1338          /**
1339           * Create the controls for this section.
1340           *
1341           * @since 4.9.0
1342           */
1343          populateControls: function() {
1344              var section = this,
1345                  menuNameControlId,
1346                  menuLocationsControlId,
1347                  newMenuSubmitControlId,
1348                  menuNameControl,
1349                  menuLocationsControl,
1350                  newMenuSubmitControl;
1351  
1352              menuNameControlId = section.id + '[name]';
1353              menuNameControl = api.control( menuNameControlId );
1354              if ( ! menuNameControl ) {
1355                  menuNameControl = new api.controlConstructor.nav_menu_name( menuNameControlId, {
1356                      label: api.Menus.data.l10n.menuNameLabel,
1357                      description: api.Menus.data.l10n.newMenuNameDescription,
1358                      section: section.id,
1359                      priority: 0
1360                  } );
1361                  api.control.add( menuNameControl.id, menuNameControl );
1362                  menuNameControl.active.set( true );
1363              }
1364  
1365              menuLocationsControlId = section.id + '[locations]';
1366              menuLocationsControl = api.control( menuLocationsControlId );
1367              if ( ! menuLocationsControl ) {
1368                  menuLocationsControl = new api.controlConstructor.nav_menu_locations( menuLocationsControlId, {
1369                      section: section.id,
1370                      priority: 1,
1371                      menu_id: '',
1372                      isCreating: true
1373                  } );
1374                  api.control.add( menuLocationsControlId, menuLocationsControl );
1375                  menuLocationsControl.active.set( true );
1376              }
1377  
1378              newMenuSubmitControlId = section.id + '[submit]';
1379              newMenuSubmitControl = api.control( newMenuSubmitControlId );
1380              if ( !newMenuSubmitControl ) {
1381                  newMenuSubmitControl = new api.Control( newMenuSubmitControlId, {
1382                      section: section.id,
1383                      priority: 1,
1384                      templateId: 'nav-menu-submit-new-button'
1385                  } );
1386                  api.control.add( newMenuSubmitControlId, newMenuSubmitControl );
1387                  newMenuSubmitControl.active.set( true );
1388              }
1389          },
1390  
1391          /**
1392           * Create the new menu with name and location supplied by the user.
1393           *
1394           * @since 4.9.0
1395           */
1396          submit: function() {
1397              var section = this,
1398                  contentContainer = section.contentContainer,
1399                  nameInput = contentContainer.find( '.menu-name-field' ).first(),
1400                  name = nameInput.val(),
1401                  menuSection;
1402  
1403              if ( ! name ) {
1404                  nameInput.addClass( 'invalid' );
1405                  nameInput.focus();
1406                  return;
1407              }
1408  
1409              menuSection = api.Menus.createNavMenu( name );
1410  
1411              // Clear name field.
1412              nameInput.val( '' );
1413              nameInput.removeClass( 'invalid' );
1414  
1415              contentContainer.find( '.assigned-menu-location input[type=checkbox]' ).each( function() {
1416                  var checkbox = $( this ),
1417                  navMenuLocationSetting;
1418  
1419                  if ( checkbox.prop( 'checked' ) ) {
1420                      navMenuLocationSetting = api( 'nav_menu_locations[' + checkbox.data( 'location-id' ) + ']' );
1421                      navMenuLocationSetting.set( menuSection.params.menu_id );
1422  
1423                      // Reset state for next new menu.
1424                      checkbox.prop( 'checked', false );
1425                  }
1426              } );
1427  
1428              wp.a11y.speak( api.Menus.data.l10n.menuAdded );
1429  
1430              // Focus on the new menu section.
1431              menuSection.focus( {
1432                  completeCallback: function() {
1433                      menuSection.highlightNewItemButton();
1434                  }
1435              } );
1436          },
1437  
1438          /**
1439           * Select a default location.
1440           *
1441           * This method selects a single location by default so we can support
1442           * creating a menu for a specific menu location.
1443           *
1444           * @since 4.9.0
1445           *
1446           * @param {string|null} locationId - The ID of the location to select. `null` clears all selections.
1447           * @return {void}
1448           */
1449          selectDefaultLocation: function( locationId ) {
1450              var locationControl = api.control( this.id + '[locations]' ),
1451                  locationSelections = {};
1452  
1453              if ( locationId !== null ) {
1454                  locationSelections[ locationId ] = true;
1455              }
1456  
1457              locationControl.setSelections( locationSelections );
1458          }
1459      });
1460  
1461      /**
1462       * wp.customize.Menus.MenuLocationControl
1463       *
1464       * Customizer control for menu locations (rendered as a <select>).
1465       * Note that 'nav_menu_location' must match the WP_Customize_Nav_Menu_Location_Control::$type.
1466       *
1467       * @class    wp.customize.Menus.MenuLocationControl
1468       * @augments wp.customize.Control
1469       */
1470      api.Menus.MenuLocationControl = api.Control.extend(/** @lends wp.customize.Menus.MenuLocationControl.prototype */{
1471          initialize: function( id, options ) {
1472              var control = this,
1473                  matches = id.match( /^nav_menu_locations\[(.+?)]/ );
1474              control.themeLocation = matches[1];
1475              api.Control.prototype.initialize.call( control, id, options );
1476          },
1477  
1478          ready: function() {
1479              var control = this, navMenuIdRegex = /^nav_menu\[(-?\d+)]/;
1480  
1481              // @todo It would be better if this was added directly on the setting itself, as opposed to the control.
1482              control.setting.validate = function( value ) {
1483                  if ( '' === value ) {
1484                      return 0;
1485                  } else {
1486                      return parseInt( value, 10 );
1487                  }
1488              };
1489  
1490              // Create and Edit menu buttons.
1491              control.container.find( '.create-menu' ).on( 'click', function() {
1492                  var addMenuSection = api.section( 'add_menu' );
1493                  addMenuSection.selectDefaultLocation( this.dataset.locationId );
1494                  addMenuSection.focus();
1495              } );
1496              control.container.find( '.edit-menu' ).on( 'click', function() {
1497                  var menuId = control.setting();
1498                  api.section( 'nav_menu[' + menuId + ']' ).focus();
1499              });
1500              control.setting.bind( 'change', function() {
1501                  var menuIsSelected = 0 !== control.setting();
1502                  control.container.find( '.create-menu' ).toggleClass( 'hidden', menuIsSelected );
1503                  control.container.find( '.edit-menu' ).toggleClass( 'hidden', ! menuIsSelected );
1504              });
1505  
1506              // Add/remove menus from the available options when they are added and removed.
1507              api.bind( 'add', function( setting ) {
1508                  var option, menuId, matches = setting.id.match( navMenuIdRegex );
1509                  if ( ! matches || false === setting() ) {
1510                      return;
1511                  }
1512                  menuId = matches[1];
1513                  option = new Option( displayNavMenuName( setting().name ), menuId );
1514                  control.container.find( 'select' ).append( option );
1515              });
1516              api.bind( 'remove', function( setting ) {
1517                  var menuId, matches = setting.id.match( navMenuIdRegex );
1518                  if ( ! matches ) {
1519                      return;
1520                  }
1521                  menuId = parseInt( matches[1], 10 );
1522                  if ( control.setting() === menuId ) {
1523                      control.setting.set( '' );
1524                  }
1525                  control.container.find( 'option[value=' + menuId + ']' ).remove();
1526              });
1527              api.bind( 'change', function( setting ) {
1528                  var menuId, matches = setting.id.match( navMenuIdRegex );
1529                  if ( ! matches ) {
1530                      return;
1531                  }
1532                  menuId = parseInt( matches[1], 10 );
1533                  if ( false === setting() ) {
1534                      if ( control.setting() === menuId ) {
1535                          control.setting.set( '' );
1536                      }
1537                      control.container.find( 'option[value=' + menuId + ']' ).remove();
1538                  } else {
1539                      control.container.find( 'option[value=' + menuId + ']' ).text( displayNavMenuName( setting().name ) );
1540                  }
1541              });
1542          }
1543      });
1544  
1545      api.Menus.MenuItemControl = api.Control.extend(/** @lends wp.customize.Menus.MenuItemControl.prototype */{
1546  
1547          /**
1548           * wp.customize.Menus.MenuItemControl
1549           *
1550           * Customizer control for menu items.
1551           * Note that 'menu_item' must match the WP_Customize_Menu_Item_Control::$type.
1552           *
1553           * @constructs wp.customize.Menus.MenuItemControl
1554           * @augments   wp.customize.Control
1555           *
1556           * @inheritDoc
1557           */
1558          initialize: function( id, options ) {
1559              var control = this;
1560              control.expanded = new api.Value( false );
1561              control.expandedArgumentsQueue = [];
1562              control.expanded.bind( function( expanded ) {
1563                  var args = control.expandedArgumentsQueue.shift();
1564                  args = $.extend( {}, control.defaultExpandedArguments, args );
1565                  control.onChangeExpanded( expanded, args );
1566              });
1567              api.Control.prototype.initialize.call( control, id, options );
1568              control.active.validate = function() {
1569                  var value, section = api.section( control.section() );
1570                  if ( section ) {
1571                      value = section.active();
1572                  } else {
1573                      value = false;
1574                  }
1575                  return value;
1576              };
1577          },
1578  
1579          /**
1580           * Set up the initial state of the screen reader accessibility information for menu items.
1581           *
1582           * @since 6.6.0
1583           */
1584          initAccessibility: function() {
1585              var control = this,
1586                  menu = $( '#menu-to-edit' );
1587  
1588              // Refresh the accessibility when the user comes close to the item in any way.
1589              menu.on( 'mouseenter.refreshAccessibility focus.refreshAccessibility touchstart.refreshAccessibility', '.menu-item', function(){
1590                  control.refreshAdvancedAccessibilityOfItem( $( this ).find( 'button.item-edit' ) );
1591              } );
1592  
1593              // We have to update on click as well because we might hover first, change the item, and then click.
1594              menu.on( 'click', 'button.item-edit', function() {
1595                  control.refreshAdvancedAccessibilityOfItem( $( this ) );
1596              } );
1597          },
1598  
1599          /**
1600           * refreshAdvancedAccessibilityOfItem( [itemToRefresh] )
1601           *
1602           * Refreshes advanced accessibility buttons for one menu item.
1603           * Shows or hides buttons based on the location of the menu item.
1604           *
1605           * @param {Object} itemToRefresh The menu item that might need its advanced accessibility buttons refreshed
1606           * 
1607           * @since 6.6.0
1608           */
1609          refreshAdvancedAccessibilityOfItem: function( itemToRefresh ) {
1610              // Only refresh accessibility when necessary.
1611              if ( true !== $( itemToRefresh ).data( 'needs_accessibility_refresh' ) ) {
1612                  return;
1613              }
1614  
1615              var primaryItems, itemPosition, title,
1616                  parentItem, parentItemId, parentItemName, subItems, totalSubItems,
1617                  $this = $( itemToRefresh ),
1618                  menuItem = $this.closest( 'li.menu-item' ).first(),
1619                  depth = menuItem.menuItemDepth(),
1620                  isPrimaryMenuItem = ( 0 === depth ),
1621                  itemName = $this.closest( '.menu-item-handle' ).find( '.menu-item-title' ).text(),
1622                  menuItemType = $this.closest( '.menu-item-handle' ).find( '.item-type' ).text(),
1623                  totalMenuItems = $( '#menu-to-edit li' ).length;
1624  
1625              if ( isPrimaryMenuItem ) {
1626                  primaryItems = $( '.menu-item-depth-0' ),
1627                  itemPosition = primaryItems.index( menuItem ) + 1,
1628                  totalMenuItems = primaryItems.length,
1629                  // String together help text for primary menu items.
1630                  title = menus.menuFocus.replace( '%1$s', itemName ).replace( '%2$s', menuItemType ).replace( '%3$d', itemPosition ).replace( '%4$d', totalMenuItems );
1631              } else {
1632                  parentItem = menuItem.prevAll( '.menu-item-depth-' + parseInt( depth - 1, 10 ) ).first(),
1633                  parentItemId = parentItem.find( '.menu-item-data-db-id' ).val(),
1634                  parentItemName = parentItem.find( '.menu-item-title' ).text(),
1635                  subItems = $( '.menu-item .menu-item-data-parent-id[value="' + parentItemId + '"]' ),
1636                  totalSubItems = subItems.length,
1637                  itemPosition = $( subItems.parents( '.menu-item' ).get().reverse() ).index( menuItem ) + 1;
1638  
1639                  // String together help text for sub menu items.
1640                  if ( depth < 2 ) {
1641                      title = menus.subMenuFocus.replace( '%1$s', itemName ).replace( '%2$s', menuItemType ).replace( '%3$d', itemPosition ).replace( '%4$d', totalSubItems ).replace( '%5$s', parentItemName );
1642                  } else {
1643                      title = menus.subMenuMoreDepthFocus.replace( '%1$s', itemName ).replace( '%2$s', menuItemType ).replace( '%3$d', itemPosition ).replace( '%4$d', totalSubItems ).replace( '%5$s', parentItemName ).replace( '%6$d', depth );
1644                  }
1645              }
1646  
1647              $this.find( '.screen-reader-text' ).text( title );
1648  
1649              // Mark this item's accessibility as refreshed.
1650              $this.data( 'needs_accessibility_refresh', false );
1651          },
1652  
1653          /**
1654           * Override the embed() method to do nothing,
1655           * so that the control isn't embedded on load,
1656           * unless the containing section is already expanded.
1657           *
1658           * @since 4.3.0
1659           */
1660          embed: function() {
1661              var control = this,
1662                  sectionId = control.section(),
1663                  section;
1664              if ( ! sectionId ) {
1665                  return;
1666              }
1667              section = api.section( sectionId );
1668              if ( ( section && section.expanded() ) || api.settings.autofocus.control === control.id ) {
1669                  control.actuallyEmbed();
1670              }
1671          },
1672  
1673          /**
1674           * This function is called in Section.onChangeExpanded() so the control
1675           * will only get embedded when the Section is first expanded.
1676           *
1677           * @since 4.3.0
1678           */
1679          actuallyEmbed: function() {
1680              var control = this;
1681              if ( 'resolved' === control.deferred.embedded.state() ) {
1682                  return;
1683              }
1684              control.renderContent();
1685              control.deferred.embedded.resolve(); // This triggers control.ready().
1686              
1687              // Mark all menu items as unprocessed.
1688              $( 'button.item-edit' ).data( 'needs_accessibility_refresh', true );
1689          },
1690  
1691          /**
1692           * Set up the control.
1693           */
1694          ready: function() {
1695              if ( 'undefined' === typeof this.params.menu_item_id ) {
1696                  throw new Error( 'params.menu_item_id was not defined' );
1697              }
1698  
1699              this._setupControlToggle();
1700              this._setupReorderUI();
1701              this._setupUpdateUI();
1702              this._setupRemoveUI();
1703              this._setupLinksUI();
1704              this._setupTitleUI();
1705          },
1706  
1707          /**
1708           * Show/hide the settings when clicking on the menu item handle.
1709           */
1710          _setupControlToggle: function() {
1711              var control = this;
1712  
1713              this.container.find( '.menu-item-handle' ).on( 'click', function( e ) {
1714                  e.preventDefault();
1715                  e.stopPropagation();
1716                  var menuControl = control.getMenuControl(),
1717                      isDeleteBtn = $( e.target ).is( '.item-delete, .item-delete *' ),
1718                      isAddNewBtn = $( e.target ).is( '.add-new-menu-item, .add-new-menu-item *' );
1719  
1720                  if ( $( 'body' ).hasClass( 'adding-menu-items' ) && ! isDeleteBtn && ! isAddNewBtn ) {
1721                      api.Menus.availableMenuItemsPanel.close();
1722                  }
1723  
1724                  if ( menuControl.isReordering || menuControl.isSorting ) {
1725                      return;
1726                  }
1727                  control.toggleForm();
1728              } );
1729          },
1730  
1731          /**
1732           * Set up the menu-item-reorder-nav
1733           */
1734          _setupReorderUI: function() {
1735              var control = this, template, $reorderNav;
1736  
1737              template = wp.template( 'menu-item-reorder-nav' );
1738  
1739              // Add the menu item reordering elements to the menu item control.
1740              control.container.find( '.item-controls' ).after( template );
1741  
1742              // Handle clicks for up/down/left-right on the reorder nav.
1743              $reorderNav = control.container.find( '.menu-item-reorder-nav' );
1744              $reorderNav.find( '.menus-move-up, .menus-move-down, .menus-move-left, .menus-move-right' ).on( 'click', function() {
1745                  var moveBtn = $( this );
1746                  control.params.depth = control.getDepth();
1747  
1748                  moveBtn.focus();
1749  
1750                  var isMoveUp = moveBtn.is( '.menus-move-up' ),
1751                      isMoveDown = moveBtn.is( '.menus-move-down' ),
1752                      isMoveLeft = moveBtn.is( '.menus-move-left' ),
1753                      isMoveRight = moveBtn.is( '.menus-move-right' );
1754  
1755                  if ( isMoveUp ) {
1756                      control.moveUp();
1757                  } else if ( isMoveDown ) {
1758                      control.moveDown();
1759                  } else if ( isMoveLeft ) {
1760                      control.moveLeft();
1761                  } else if ( isMoveRight ) {
1762                      control.moveRight();
1763                      control.params.depth += 1;
1764                  }
1765                  
1766                  moveBtn.focus(); // Re-focus after the container was moved.
1767  
1768                  // Mark all menu items as unprocessed.
1769                  $( 'button.item-edit' ).data( 'needs_accessibility_refresh', true );
1770              } );
1771          },
1772  
1773          /**
1774           * Set up event handlers for menu item updating.
1775           */
1776          _setupUpdateUI: function() {
1777              var control = this,
1778                  settingValue = control.setting(),
1779                  updateNotifications;
1780  
1781              control.elements = {};
1782              control.elements.url = new api.Element( control.container.find( '.edit-menu-item-url' ) );
1783              control.elements.title = new api.Element( control.container.find( '.edit-menu-item-title' ) );
1784              control.elements.attr_title = new api.Element( control.container.find( '.edit-menu-item-attr-title' ) );
1785              control.elements.target = new api.Element( control.container.find( '.edit-menu-item-target' ) );
1786              control.elements.classes = new api.Element( control.container.find( '.edit-menu-item-classes' ) );
1787              control.elements.xfn = new api.Element( control.container.find( '.edit-menu-item-xfn' ) );
1788              control.elements.description = new api.Element( control.container.find( '.edit-menu-item-description' ) );
1789              // @todo Allow other elements, added by plugins, to be automatically picked up here;
1790              // allow additional values to be added to setting array.
1791  
1792              _.each( control.elements, function( element, property ) {
1793                  element.bind(function( value ) {
1794                      if ( element.element.is( 'input[type=checkbox]' ) ) {
1795                          value = ( value ) ? element.element.val() : '';
1796                      }
1797  
1798                      var settingValue = control.setting();
1799                      if ( settingValue && settingValue[ property ] !== value ) {
1800                          settingValue = _.clone( settingValue );
1801                          settingValue[ property ] = value;
1802                          control.setting.set( settingValue );
1803                      }
1804                  });
1805                  if ( settingValue ) {
1806                      if ( ( property === 'classes' || property === 'xfn' ) && _.isArray( settingValue[ property ] ) ) {
1807                          element.set( settingValue[ property ].join( ' ' ) );
1808                      } else {
1809                          element.set( settingValue[ property ] );
1810                      }
1811                  }
1812              });
1813  
1814              control.setting.bind(function( to, from ) {
1815                  var itemId = control.params.menu_item_id,
1816                      followingSiblingItemControls = [],
1817                      childrenItemControls = [],
1818                      menuControl;
1819  
1820                  if ( false === to ) {
1821                      menuControl = api.control( 'nav_menu[' + String( from.nav_menu_term_id ) + ']' );
1822                      control.container.remove();
1823  
1824                      _.each( menuControl.getMenuItemControls(), function( otherControl ) {
1825                          if ( from.menu_item_parent === otherControl.setting().menu_item_parent && otherControl.setting().position > from.position ) {
1826                              followingSiblingItemControls.push( otherControl );
1827                          } else if ( otherControl.setting().menu_item_parent === itemId ) {
1828                              childrenItemControls.push( otherControl );
1829                          }
1830                      });
1831  
1832                      // Shift all following siblings by the number of children this item has.
1833                      _.each( followingSiblingItemControls, function( followingSiblingItemControl ) {
1834                          var value = _.clone( followingSiblingItemControl.setting() );
1835                          value.position += childrenItemControls.length;
1836                          followingSiblingItemControl.setting.set( value );
1837                      });
1838  
1839                      // Now move the children up to be the new subsequent siblings.
1840                      _.each( childrenItemControls, function( childrenItemControl, i ) {
1841                          var value = _.clone( childrenItemControl.setting() );
1842                          value.position = from.position + i;
1843                          value.menu_item_parent = from.menu_item_parent;
1844                          childrenItemControl.setting.set( value );
1845                      });
1846  
1847                      menuControl.debouncedReflowMenuItems();
1848                  } else {
1849                      // Update the elements' values to match the new setting properties.
1850                      _.each( to, function( value, key ) {
1851                          if ( control.elements[ key] ) {
1852                              control.elements[ key ].set( to[ key ] );
1853                          }
1854                      } );
1855                      control.container.find( '.menu-item-data-parent-id' ).val( to.menu_item_parent );
1856  
1857                      // Handle UI updates when the position or depth (parent) change.
1858                      if ( to.position !== from.position || to.menu_item_parent !== from.menu_item_parent ) {
1859                          control.getMenuControl().debouncedReflowMenuItems();
1860                      }
1861                  }
1862              });
1863  
1864              // Style the URL field as invalid when there is an invalid_url notification.
1865              updateNotifications = function() {
1866                  control.elements.url.element.toggleClass( 'invalid', control.setting.notifications.has( 'invalid_url' ) );
1867              };
1868              control.setting.notifications.bind( 'add', updateNotifications );
1869              control.setting.notifications.bind( 'removed', updateNotifications );
1870          },
1871  
1872          /**
1873           * Set up event handlers for menu item deletion.
1874           */
1875          _setupRemoveUI: function() {
1876              var control = this, $removeBtn;
1877  
1878              // Configure delete button.
1879              $removeBtn = control.container.find( '.item-delete' );
1880  
1881              $removeBtn.on( 'click', function() {
1882                  // Find an adjacent element to add focus to when this menu item goes away.
1883                  var addingItems = true, $adjacentFocusTarget, $next, $prev,
1884                      instanceCounter = 0, // Instance count of the menu item deleted.
1885                      deleteItemOriginalItemId = control.params.original_item_id,
1886                      addedItems = control.getMenuControl().$sectionContent.find( '.menu-item' ),
1887                      availableMenuItem;
1888  
1889                  if ( ! $( 'body' ).hasClass( 'adding-menu-items' ) ) {
1890                      addingItems = false;
1891                  }
1892  
1893                  $next = control.container.nextAll( '.customize-control-nav_menu_item:visible' ).first();
1894                  $prev = control.container.prevAll( '.customize-control-nav_menu_item:visible' ).first();
1895  
1896                  if ( $next.length ) {
1897                      $adjacentFocusTarget = $next.find( false === addingItems ? '.item-edit' : '.item-delete' ).first();
1898                  } else if ( $prev.length ) {
1899                      $adjacentFocusTarget = $prev.find( false === addingItems ? '.item-edit' : '.item-delete' ).first();
1900                  } else {
1901                      $adjacentFocusTarget = control.container.nextAll( '.customize-control-nav_menu' ).find( '.add-new-menu-item' ).first();
1902                  }
1903  
1904                  /*
1905                   * If the menu item deleted is the only of its instance left,
1906                   * remove the check icon of this menu item in the right panel.
1907                   */
1908                  _.each( addedItems, function( addedItem ) {
1909                      var menuItemId, menuItemControl, matches;
1910  
1911                      // This is because menu item that's deleted is just hidden.
1912                      if ( ! $( addedItem ).is( ':visible' ) ) {
1913                          return;
1914                      }
1915  
1916                      matches = addedItem.getAttribute( 'id' ).match( /^customize-control-nav_menu_item-(-?\d+)$/, '' );
1917                      if ( ! matches ) {
1918                          return;
1919                      }
1920  
1921                      menuItemId      = parseInt( matches[1], 10 );
1922                      menuItemControl = api.control( 'nav_menu_item[' + String( menuItemId ) + ']' );
1923  
1924                      // Check for duplicate menu items.
1925                      if ( menuItemControl && deleteItemOriginalItemId == menuItemControl.params.original_item_id ) {
1926                          instanceCounter++;
1927                      }
1928                  } );
1929  
1930                  if ( instanceCounter <= 1 ) {
1931                      // Revert the check icon to add icon.
1932                      availableMenuItem = $( '#menu-item-tpl-' + control.params.original_item_id );
1933                      availableMenuItem.removeClass( 'selected' );
1934                      availableMenuItem.find( '.menu-item-handle' ).removeClass( 'item-added' );
1935                  }
1936  
1937                  control.container.slideUp( function() {
1938                      control.setting.set( false );
1939                      wp.a11y.speak( api.Menus.data.l10n.itemDeleted );
1940                      $adjacentFocusTarget.focus(); // Keyboard accessibility.
1941                  } );
1942  
1943                  control.setting.set( false );
1944              } );
1945          },
1946  
1947          _setupLinksUI: function() {
1948              var $origBtn;
1949  
1950              // Configure original link.
1951              $origBtn = this.container.find( 'a.original-link' );
1952  
1953              $origBtn.on( 'click', function( e ) {
1954                  e.preventDefault();
1955                  api.previewer.previewUrl( e.target.toString() );
1956              } );
1957          },
1958  
1959          /**
1960           * Update item handle title when changed.
1961           */
1962          _setupTitleUI: function() {
1963              var control = this, titleEl;
1964  
1965              // Ensure that whitespace is trimmed on blur so placeholder can be shown.
1966              control.container.find( '.edit-menu-item-title' ).on( 'blur', function() {
1967                  $( this ).val( $( this ).val().trim() );
1968              } );
1969  
1970              titleEl = control.container.find( '.menu-item-title' );
1971              control.setting.bind( function( item ) {
1972                  var trimmedTitle, titleText;
1973                  if ( ! item ) {
1974                      return;
1975                  }
1976                  item.title = item.title || '';
1977                  trimmedTitle = item.title.trim();
1978  
1979                  titleText = trimmedTitle || item.original_title || api.Menus.data.l10n.untitled;
1980  
1981                  if ( item._invalid ) {
1982                      titleText = api.Menus.data.l10n.invalidTitleTpl.replace( '%s', titleText );
1983                  }
1984  
1985                  // Don't update to an empty title.
1986                  if ( trimmedTitle || item.original_title ) {
1987                      titleEl
1988                          .text( titleText )
1989                          .removeClass( 'no-title' );
1990                  } else {
1991                      titleEl
1992                          .text( titleText )
1993                          .addClass( 'no-title' );
1994                  }
1995              } );
1996          },
1997  
1998          /**
1999           *
2000           * @return {number}
2001           */
2002          getDepth: function() {
2003              var control = this, setting = control.setting(), depth = 0;
2004              if ( ! setting ) {
2005                  return 0;
2006              }
2007              while ( setting && setting.menu_item_parent ) {
2008                  depth += 1;
2009                  control = api.control( 'nav_menu_item[' + setting.menu_item_parent + ']' );
2010                  if ( ! control ) {
2011                      break;
2012                  }
2013                  setting = control.setting();
2014              }
2015              return depth;
2016          },
2017  
2018          /**
2019           * Amend the control's params with the data necessary for the JS template just in time.
2020           */
2021          renderContent: function() {
2022              var control = this,
2023                  settingValue = control.setting(),
2024                  containerClasses;
2025  
2026              control.params.title = settingValue.title || '';
2027              control.params.depth = control.getDepth();
2028              control.container.data( 'item-depth', control.params.depth );
2029              containerClasses = [
2030                  'menu-item',
2031                  'menu-item-depth-' + String( control.params.depth ),
2032                  'menu-item-' + settingValue.object,
2033                  'menu-item-edit-inactive'
2034              ];
2035  
2036              if ( settingValue._invalid ) {
2037                  containerClasses.push( 'menu-item-invalid' );
2038                  control.params.title = api.Menus.data.l10n.invalidTitleTpl.replace( '%s', control.params.title );
2039              } else if ( 'draft' === settingValue.status ) {
2040                  containerClasses.push( 'pending' );
2041                  control.params.title = api.Menus.data.pendingTitleTpl.replace( '%s', control.params.title );
2042              }
2043  
2044              control.params.el_classes = containerClasses.join( ' ' );
2045              control.params.item_type_label = settingValue.type_label;
2046              control.params.item_type = settingValue.type;
2047              control.params.url = settingValue.url;
2048              control.params.target = settingValue.target;
2049              control.params.attr_title = settingValue.attr_title;
2050              control.params.classes = _.isArray( settingValue.classes ) ? settingValue.classes.join( ' ' ) : settingValue.classes;
2051              control.params.xfn = settingValue.xfn;
2052              control.params.description = settingValue.description;
2053              control.params.parent = settingValue.menu_item_parent;
2054              control.params.original_title = settingValue.original_title || '';
2055  
2056              control.container.addClass( control.params.el_classes );
2057  
2058              api.Control.prototype.renderContent.call( control );
2059          },
2060  
2061          /***********************************************************************
2062           * Begin public API methods
2063           **********************************************************************/
2064  
2065          /**
2066           * @return {wp.customize.controlConstructor.nav_menu|null}
2067           */
2068          getMenuControl: function() {
2069              var control = this, settingValue = control.setting();
2070              if ( settingValue && settingValue.nav_menu_term_id ) {
2071                  return api.control( 'nav_menu[' + settingValue.nav_menu_term_id + ']' );
2072              } else {
2073                  return null;
2074              }
2075          },
2076  
2077          /**
2078           * Expand the accordion section containing a control
2079           */
2080          expandControlSection: function() {
2081              var $section = this.container.closest( '.accordion-section' );
2082              if ( ! $section.hasClass( 'open' ) ) {
2083                  $section.find( '.accordion-section-title:first' ).trigger( 'click' );
2084              }
2085          },
2086  
2087          /**
2088           * @since 4.6.0
2089           *
2090           * @param {Boolean} expanded
2091           * @param {Object} [params]
2092           * @return {Boolean} False if state already applied.
2093           */
2094          _toggleExpanded: api.Section.prototype._toggleExpanded,
2095  
2096          /**
2097           * @since 4.6.0
2098           *
2099           * @param {Object} [params]
2100           * @return {Boolean} False if already expanded.
2101           */
2102          expand: api.Section.prototype.expand,
2103  
2104          /**
2105           * Expand the menu item form control.
2106           *
2107           * @since 4.5.0 Added params.completeCallback.
2108           *
2109           * @param {Object}   [params] - Optional params.
2110           * @param {Function} [params.completeCallback] - Function to call when the form toggle has finished animating.
2111           */
2112          expandForm: function( params ) {
2113              this.expand( params );
2114          },
2115  
2116          /**
2117           * @since 4.6.0
2118           *
2119           * @param {Object} [params]
2120           * @return {Boolean} False if already collapsed.
2121           */
2122          collapse: api.Section.prototype.collapse,
2123  
2124          /**
2125           * Collapse the menu item form control.
2126           *
2127           * @since 4.5.0 Added params.completeCallback.
2128           *
2129           * @param {Object}   [params] - Optional params.
2130           * @param {Function} [params.completeCallback] - Function to call when the form toggle has finished animating.
2131           */
2132          collapseForm: function( params ) {
2133              this.collapse( params );
2134          },
2135  
2136          /**
2137           * Expand or collapse the menu item control.
2138           *
2139           * @deprecated this is poor naming, and it is better to directly set control.expanded( showOrHide )
2140           * @since 4.5.0 Added params.completeCallback.
2141           *
2142           * @param {boolean}  [showOrHide] - If not supplied, will be inverse of current visibility
2143           * @param {Object}   [params] - Optional params.
2144           * @param {Function} [params.completeCallback] - Function to call when the form toggle has finished animating.
2145           */
2146          toggleForm: function( showOrHide, params ) {
2147              if ( typeof showOrHide === 'undefined' ) {
2148                  showOrHide = ! this.expanded();
2149              }
2150              if ( showOrHide ) {
2151                  this.expand( params );
2152              } else {
2153                  this.collapse( params );
2154              }
2155          },
2156  
2157          /**
2158           * Expand or collapse the menu item control.
2159           *
2160           * @since 4.6.0
2161           * @param {boolean}  [showOrHide] - If not supplied, will be inverse of current visibility
2162           * @param {Object}   [params] - Optional params.
2163           * @param {Function} [params.completeCallback] - Function to call when the form toggle has finished animating.
2164           */
2165          onChangeExpanded: function( showOrHide, params ) {
2166              var self = this, $menuitem, $inside, complete;
2167  
2168              $menuitem = this.container;
2169              $inside = $menuitem.find( '.menu-item-settings:first' );
2170              if ( 'undefined' === typeof showOrHide ) {
2171                  showOrHide = ! $inside.is( ':visible' );
2172              }
2173  
2174              // Already expanded or collapsed.
2175              if ( $inside.is( ':visible' ) === showOrHide ) {
2176                  if ( params && params.completeCallback ) {
2177                      params.completeCallback();
2178                  }
2179                  return;
2180              }
2181  
2182              if ( showOrHide ) {
2183                  // Close all other menu item controls before expanding this one.
2184                  api.control.each( function( otherControl ) {
2185                      if ( self.params.type === otherControl.params.type && self !== otherControl ) {
2186                          otherControl.collapseForm();
2187                      }
2188                  } );
2189  
2190                  complete = function() {
2191                      $menuitem
2192                          .removeClass( 'menu-item-edit-inactive' )
2193                          .addClass( 'menu-item-edit-active' );
2194                      self.container.trigger( 'expanded' );
2195  
2196                      if ( params && params.completeCallback ) {
2197                          params.completeCallback();
2198                      }
2199                  };
2200  
2201                  $menuitem.find( '.item-edit' ).attr( 'aria-expanded', 'true' );
2202                  $inside.slideDown( 'fast', complete );
2203  
2204                  self.container.trigger( 'expand' );
2205              } else {
2206                  complete = function() {
2207                      $menuitem
2208                          .addClass( 'menu-item-edit-inactive' )
2209                          .removeClass( 'menu-item-edit-active' );
2210                      self.container.trigger( 'collapsed' );
2211  
2212                      if ( params && params.completeCallback ) {
2213                          params.completeCallback();
2214                      }
2215                  };
2216  
2217                  self.container.trigger( 'collapse' );
2218  
2219                  $menuitem.find( '.item-edit' ).attr( 'aria-expanded', 'false' );
2220                  $inside.slideUp( 'fast', complete );
2221              }
2222          },
2223  
2224          /**
2225           * Expand the containing menu section, expand the form, and focus on
2226           * the first input in the control.
2227           *
2228           * @since 4.5.0 Added params.completeCallback.
2229           *
2230           * @param {Object}   [params] - Params object.
2231           * @param {Function} [params.completeCallback] - Optional callback function when focus has completed.
2232           */
2233          focus: function( params ) {
2234              params = params || {};
2235              var control = this, originalCompleteCallback = params.completeCallback, focusControl;
2236  
2237              focusControl = function() {
2238                  control.expandControlSection();
2239  
2240                  params.completeCallback = function() {
2241                      var focusable;
2242  
2243                      // Note that we can't use :focusable due to a jQuery UI issue. See: https://github.com/jquery/jquery-ui/pull/1583
2244                      focusable = control.container.find( '.menu-item-settings' ).find( 'input, select, textarea, button, object, a[href], [tabindex]' ).filter( ':visible' );
2245                      focusable.first().focus();
2246  
2247                      if ( originalCompleteCallback ) {
2248                          originalCompleteCallback();
2249                      }
2250                  };
2251  
2252                  control.expandForm( params );
2253              };
2254  
2255              if ( api.section.has( control.section() ) ) {
2256                  api.section( control.section() ).expand( {
2257                      completeCallback: focusControl
2258                  } );
2259              } else {
2260                  focusControl();
2261              }
2262          },
2263  
2264          /**
2265           * Move menu item up one in the menu.
2266           */
2267          moveUp: function() {
2268              this._changePosition( -1 );
2269              wp.a11y.speak( api.Menus.data.l10n.movedUp );
2270          },
2271  
2272          /**
2273           * Move menu item up one in the menu.
2274           */
2275          moveDown: function() {
2276              this._changePosition( 1 );
2277              wp.a11y.speak( api.Menus.data.l10n.movedDown );
2278          },
2279          /**
2280           * Move menu item and all children up one level of depth.
2281           */
2282          moveLeft: function() {
2283              this._changeDepth( -1 );
2284              wp.a11y.speak( api.Menus.data.l10n.movedLeft );
2285          },
2286  
2287          /**
2288           * Move menu item and children one level deeper, as a submenu of the previous item.
2289           */
2290          moveRight: function() {
2291              this._changeDepth( 1 );
2292              wp.a11y.speak( api.Menus.data.l10n.movedRight );
2293          },
2294  
2295          /**
2296           * Note that this will trigger a UI update, causing child items to
2297           * move as well and cardinal order class names to be updated.
2298           *
2299           * @private
2300           *
2301           * @param {number} offset 1|-1
2302           */
2303          _changePosition: function( offset ) {
2304              var control = this,
2305                  adjacentSetting,
2306                  settingValue = _.clone( control.setting() ),
2307                  siblingSettings = [],
2308                  realPosition;
2309  
2310              if ( 1 !== offset && -1 !== offset ) {
2311                  throw new Error( 'Offset changes by 1 are only supported.' );
2312              }
2313  
2314              // Skip moving deleted items.
2315              if ( ! control.setting() ) {
2316                  return;
2317              }
2318  
2319              // Locate the other items under the same parent (siblings).
2320              _( control.getMenuControl().getMenuItemControls() ).each(function( otherControl ) {
2321                  if ( otherControl.setting().menu_item_parent === settingValue.menu_item_parent ) {
2322                      siblingSettings.push( otherControl.setting );
2323                  }
2324              });
2325              siblingSettings.sort(function( a, b ) {
2326                  return a().position - b().position;
2327              });
2328  
2329              realPosition = _.indexOf( siblingSettings, control.setting );
2330              if ( -1 === realPosition ) {
2331                  throw new Error( 'Expected setting to be among siblings.' );
2332              }
2333  
2334              // Skip doing anything if the item is already at the edge in the desired direction.
2335              if ( ( realPosition === 0 && offset < 0 ) || ( realPosition === siblingSettings.length - 1 && offset > 0 ) ) {
2336                  // @todo Should we allow a menu item to be moved up to break it out of a parent? Adopt with previous or following parent?
2337                  return;
2338              }
2339  
2340              // Update any adjacent menu item setting to take on this item's position.
2341              adjacentSetting = siblingSettings[ realPosition + offset ];
2342              if ( adjacentSetting ) {
2343                  adjacentSetting.set( $.extend(
2344                      _.clone( adjacentSetting() ),
2345                      {
2346                          position: settingValue.position
2347                      }
2348                  ) );
2349              }
2350  
2351              settingValue.position += offset;
2352              control.setting.set( settingValue );
2353          },
2354  
2355          /**
2356           * Note that this will trigger a UI update, causing child items to
2357           * move as well and cardinal order class names to be updated.
2358           *
2359           * @private
2360           *
2361           * @param {number} offset 1|-1
2362           */
2363          _changeDepth: function( offset ) {
2364              if ( 1 !== offset && -1 !== offset ) {
2365                  throw new Error( 'Offset changes by 1 are only supported.' );
2366              }
2367              var control = this,
2368                  settingValue = _.clone( control.setting() ),
2369                  siblingControls = [],
2370                  realPosition,
2371                  siblingControl,
2372                  parentControl;
2373  
2374              // Locate the other items under the same parent (siblings).
2375              _( control.getMenuControl().getMenuItemControls() ).each(function( otherControl ) {
2376                  if ( otherControl.setting().menu_item_parent === settingValue.menu_item_parent ) {
2377                      siblingControls.push( otherControl );
2378                  }
2379              });
2380              siblingControls.sort(function( a, b ) {
2381                  return a.setting().position - b.setting().position;
2382              });
2383  
2384              realPosition = _.indexOf( siblingControls, control );
2385              if ( -1 === realPosition ) {
2386                  throw new Error( 'Expected control to be among siblings.' );
2387              }
2388  
2389              if ( -1 === offset ) {
2390                  // Skip moving left an item that is already at the top level.
2391                  if ( ! settingValue.menu_item_parent ) {
2392                      return;
2393                  }
2394  
2395                  parentControl = api.control( 'nav_menu_item[' + settingValue.menu_item_parent + ']' );
2396  
2397                  // Make this control the parent of all the following siblings.
2398                  _( siblingControls ).chain().slice( realPosition ).each(function( siblingControl, i ) {
2399                      siblingControl.setting.set(
2400                          $.extend(
2401                              {},
2402                              siblingControl.setting(),
2403                              {
2404                                  menu_item_parent: control.params.menu_item_id,
2405                                  position: i
2406                              }
2407                          )
2408                      );
2409                  });
2410  
2411                  // Increase the positions of the parent item's subsequent children to make room for this one.
2412                  _( control.getMenuControl().getMenuItemControls() ).each(function( otherControl ) {
2413                      var otherControlSettingValue, isControlToBeShifted;
2414                      isControlToBeShifted = (
2415                          otherControl.setting().menu_item_parent === parentControl.setting().menu_item_parent &&
2416                          otherControl.setting().position > parentControl.setting().position
2417                      );
2418                      if ( isControlToBeShifted ) {
2419                          otherControlSettingValue = _.clone( otherControl.setting() );
2420                          otherControl.setting.set(
2421                              $.extend(
2422                                  otherControlSettingValue,
2423                                  { position: otherControlSettingValue.position + 1 }
2424                              )
2425                          );
2426                      }
2427                  });
2428  
2429                  // Make this control the following sibling of its parent item.
2430                  settingValue.position = parentControl.setting().position + 1;
2431                  settingValue.menu_item_parent = parentControl.setting().menu_item_parent;
2432                  control.setting.set( settingValue );
2433  
2434              } else if ( 1 === offset ) {
2435                  // Skip moving right an item that doesn't have a previous sibling.
2436                  if ( realPosition === 0 ) {
2437                      return;
2438                  }
2439  
2440                  // Make the control the last child of the previous sibling.
2441                  siblingControl = siblingControls[ realPosition - 1 ];
2442                  settingValue.menu_item_parent = siblingControl.params.menu_item_id;
2443                  settingValue.position = 0;
2444                  _( control.getMenuControl().getMenuItemControls() ).each(function( otherControl ) {
2445                      if ( otherControl.setting().menu_item_parent === settingValue.menu_item_parent ) {
2446                          settingValue.position = Math.max( settingValue.position, otherControl.setting().position );
2447                      }
2448                  });
2449                  settingValue.position += 1;
2450                  control.setting.set( settingValue );
2451              }
2452          }
2453      } );
2454  
2455      /**
2456       * wp.customize.Menus.MenuNameControl
2457       *
2458       * Customizer control for a nav menu's name.
2459       *
2460       * @class    wp.customize.Menus.MenuNameControl
2461       * @augments wp.customize.Control
2462       */
2463      api.Menus.MenuNameControl = api.Control.extend(/** @lends wp.customize.Menus.MenuNameControl.prototype */{
2464  
2465          ready: function() {
2466              var control = this;
2467  
2468              if ( control.setting ) {
2469                  var settingValue = control.setting();
2470  
2471                  control.nameElement = new api.Element( control.container.find( '.menu-name-field' ) );
2472  
2473                  control.nameElement.bind(function( value ) {
2474                      var settingValue = control.setting();
2475                      if ( settingValue && settingValue.name !== value ) {
2476                          settingValue = _.clone( settingValue );
2477                          settingValue.name = value;
2478                          control.setting.set( settingValue );
2479                      }
2480                  });
2481                  if ( settingValue ) {
2482                      control.nameElement.set( settingValue.name );
2483                  }
2484  
2485                  control.setting.bind(function( object ) {
2486                      if ( object ) {
2487                          control.nameElement.set( object.name );
2488                      }
2489                  });
2490              }
2491          }
2492      });
2493  
2494      /**
2495       * wp.customize.Menus.MenuLocationsControl
2496       *
2497       * Customizer control for a nav menu's locations.
2498       *
2499       * @since 4.9.0
2500       * @class    wp.customize.Menus.MenuLocationsControl
2501       * @augments wp.customize.Control
2502       */
2503      api.Menus.MenuLocationsControl = api.Control.extend(/** @lends wp.customize.Menus.MenuLocationsControl.prototype */{
2504  
2505          /**
2506           * Set up the control.
2507           *
2508           * @since 4.9.0
2509           */
2510          ready: function () {
2511              var control = this;
2512  
2513              control.container.find( '.assigned-menu-location' ).each(function() {
2514                  var container = $( this ),
2515                      checkbox = container.find( 'input[type=checkbox]' ),
2516                      element = new api.Element( checkbox ),
2517                      navMenuLocationSetting = api( 'nav_menu_locations[' + checkbox.data( 'location-id' ) + ']' ),
2518                      isNewMenu = control.params.menu_id === '',
2519                      updateCheckbox = isNewMenu ? _.noop : function( checked ) {
2520                          element.set( checked );
2521                      },
2522                      updateSetting = isNewMenu ? _.noop : function( checked ) {
2523                          navMenuLocationSetting.set( checked ? control.params.menu_id : 0 );
2524                      },
2525                      updateSelectedMenuLabel = function( selectedMenuId ) {
2526                          var menuSetting = api( 'nav_menu[' + String( selectedMenuId ) + ']' );
2527                          if ( ! selectedMenuId || ! menuSetting || ! menuSetting() ) {
2528                              container.find( '.theme-location-set' ).hide();
2529                          } else {
2530                              container.find( '.theme-location-set' ).show().find( 'span' ).text( displayNavMenuName( menuSetting().name ) );
2531                          }
2532                      };
2533  
2534                  updateCheckbox( navMenuLocationSetting.get() === control.params.menu_id );
2535  
2536                  checkbox.on( 'change', function() {
2537                      // Note: We can't use element.bind( function( checked ){ ... } ) here because it will trigger a change as well.
2538                      updateSetting( this.checked );
2539                  } );
2540  
2541                  navMenuLocationSetting.bind( function( selectedMenuId ) {
2542                      updateCheckbox( selectedMenuId === control.params.menu_id );
2543                      updateSelectedMenuLabel( selectedMenuId );
2544                  } );
2545                  updateSelectedMenuLabel( navMenuLocationSetting.get() );
2546              });
2547          },
2548  
2549          /**
2550           * Set the selected locations.
2551           *
2552           * This method sets the selected locations and allows us to do things like
2553           * set the default location for a new menu.
2554           *
2555           * @since 4.9.0
2556           *
2557           * @param {Object.<string,boolean>} selections - A map of location selections.
2558           * @return {void}
2559           */
2560          setSelections: function( selections ) {
2561              this.container.find( '.menu-location' ).each( function( i, checkboxNode ) {
2562                  var locationId = checkboxNode.dataset.locationId;
2563                  checkboxNode.checked = locationId in selections ? selections[ locationId ] : false;
2564              } );
2565          }
2566      });
2567  
2568      /**
2569       * wp.customize.Menus.MenuAutoAddControl
2570       *
2571       * Customizer control for a nav menu's auto add.
2572       *
2573       * @class    wp.customize.Menus.MenuAutoAddControl
2574       * @augments wp.customize.Control
2575       */
2576      api.Menus.MenuAutoAddControl = api.Control.extend(/** @lends wp.customize.Menus.MenuAutoAddControl.prototype */{
2577  
2578          ready: function() {
2579              var control = this,
2580                  settingValue = control.setting();
2581  
2582              /*
2583               * Since the control is not registered in PHP, we need to prevent the
2584               * preview's sending of the activeControls to result in this control
2585               * being deactivated.
2586               */
2587              control.active.validate = function() {
2588                  var value, section = api.section( control.section() );
2589                  if ( section ) {
2590                      value = section.active();
2591                  } else {
2592                      value = false;
2593                  }
2594                  return value;
2595              };
2596  
2597              control.autoAddElement = new api.Element( control.container.find( 'input[type=checkbox].auto_add' ) );
2598  
2599              control.autoAddElement.bind(function( value ) {
2600                  var settingValue = control.setting();
2601                  if ( settingValue && settingValue.name !== value ) {
2602                      settingValue = _.clone( settingValue );
2603                      settingValue.auto_add = value;
2604                      control.setting.set( settingValue );
2605                  }
2606              });
2607              if ( settingValue ) {
2608                  control.autoAddElement.set( settingValue.auto_add );
2609              }
2610  
2611              control.setting.bind(function( object ) {
2612                  if ( object ) {
2613                      control.autoAddElement.set( object.auto_add );
2614                  }
2615              });
2616          }
2617  
2618      });
2619  
2620      /**
2621       * wp.customize.Menus.MenuControl
2622       *
2623       * Customizer control for menus.
2624       * Note that 'nav_menu' must match the WP_Menu_Customize_Control::$type
2625       *
2626       * @class    wp.customize.Menus.MenuControl
2627       * @augments wp.customize.Control
2628       */
2629      api.Menus.MenuControl = api.Control.extend(/** @lends wp.customize.Menus.MenuControl.prototype */{
2630          /**
2631           * Set up the control.
2632           */
2633          ready: function() {
2634              var control = this,
2635                  section = api.section( control.section() ),
2636                  menuId = control.params.menu_id,
2637                  menu = control.setting(),
2638                  name,
2639                  widgetTemplate,
2640                  select;
2641  
2642              if ( 'undefined' === typeof this.params.menu_id ) {
2643                  throw new Error( 'params.menu_id was not defined' );
2644              }
2645  
2646              /*
2647               * Since the control is not registered in PHP, we need to prevent the
2648               * preview's sending of the activeControls to result in this control
2649               * being deactivated.
2650               */
2651              control.active.validate = function() {
2652                  var value;
2653                  if ( section ) {
2654                      value = section.active();
2655                  } else {
2656                      value = false;
2657                  }
2658                  return value;
2659              };
2660  
2661              control.$controlSection = section.headContainer;
2662              control.$sectionContent = control.container.closest( '.accordion-section-content' );
2663  
2664              this._setupModel();
2665  
2666              api.section( control.section(), function( section ) {
2667                  section.deferred.initSortables.done(function( menuList ) {
2668                      control._setupSortable( menuList );
2669                  });
2670              } );
2671  
2672              this._setupAddition();
2673              this._setupTitle();
2674  
2675              // Add menu to Navigation Menu widgets.
2676              if ( menu ) {
2677                  name = displayNavMenuName( menu.name );
2678  
2679                  // Add the menu to the existing controls.
2680                  api.control.each( function( widgetControl ) {
2681                      if ( ! widgetControl.extended( api.controlConstructor.widget_form ) || 'nav_menu' !== widgetControl.params.widget_id_base ) {
2682                          return;
2683                      }
2684                      widgetControl.container.find( '.nav-menu-widget-form-controls:first' ).show();
2685                      widgetControl.container.find( '.nav-menu-widget-no-menus-message:first' ).hide();
2686  
2687                      select = widgetControl.container.find( 'select' );
2688                      if ( 0 === select.find( 'option[value=' + String( menuId ) + ']' ).length ) {
2689                          select.append( new Option( name, menuId ) );
2690                      }
2691                  } );
2692  
2693                  // Add the menu to the widget template.
2694                  widgetTemplate = $( '#available-widgets-list .widget-tpl:has( input.id_base[ value=nav_menu ] )' );
2695                  widgetTemplate.find( '.nav-menu-widget-form-controls:first' ).show();
2696                  widgetTemplate.find( '.nav-menu-widget-no-menus-message:first' ).hide();
2697                  select = widgetTemplate.find( '.widget-inside select:first' );
2698                  if ( 0 === select.find( 'option[value=' + String( menuId ) + ']' ).length ) {
2699                      select.append( new Option( name, menuId ) );
2700                  }
2701              }
2702  
2703              /*
2704               * Wait for menu items to be added.
2705               * Ideally, we'd bind to an event indicating construction is complete,
2706               * but deferring appears to be the best option today.
2707               */
2708              _.defer( function () {
2709                  control.updateInvitationVisibility();
2710              } );
2711          },
2712  
2713          /**
2714           * Update ordering of menu item controls when the setting is updated.
2715           */
2716          _setupModel: function() {
2717              var control = this,
2718                  menuId = control.params.menu_id;
2719  
2720              control.setting.bind( function( to ) {
2721                  var name;
2722                  if ( false === to ) {
2723                      control._handleDeletion();
2724                  } else {
2725                      // Update names in the Navigation Menu widgets.
2726                      name = displayNavMenuName( to.name );
2727                      api.control.each( function( widgetControl ) {
2728                          if ( ! widgetControl.extended( api.controlConstructor.widget_form ) || 'nav_menu' !== widgetControl.params.widget_id_base ) {
2729                              return;
2730                          }
2731                          var select = widgetControl.container.find( 'select' );
2732                          select.find( 'option[value=' + String( menuId ) + ']' ).text( name );
2733                      });
2734                  }
2735              } );
2736          },
2737  
2738          /**
2739           * Allow items in each menu to be re-ordered, and for the order to be previewed.
2740           *
2741           * Notice that the UI aspects here are handled by wpNavMenu.initSortables()
2742           * which is called in MenuSection.onChangeExpanded()
2743           *
2744           * @param {Object} menuList - The element that has sortable().
2745           */
2746          _setupSortable: function( menuList ) {
2747              var control = this;
2748  
2749              if ( ! menuList.is( control.$sectionContent ) ) {
2750                  throw new Error( 'Unexpected menuList.' );
2751              }
2752  
2753              menuList.on( 'sortstart', function() {
2754                  control.isSorting = true;
2755              });
2756  
2757              menuList.on( 'sortstop', function() {
2758                  setTimeout( function() { // Next tick.
2759                      var menuItemContainerIds = control.$sectionContent.sortable( 'toArray' ),
2760                          menuItemControls = [],
2761                          position = 0,
2762                          priority = 10;
2763  
2764                      control.isSorting = false;
2765  
2766                      // Reset horizontal scroll position when done dragging.
2767                      control.$sectionContent.scrollLeft( 0 );
2768  
2769                      _.each( menuItemContainerIds, function( menuItemContainerId ) {
2770                          var menuItemId, menuItemControl, matches;
2771                          matches = menuItemContainerId.match( /^customize-control-nav_menu_item-(-?\d+)$/, '' );
2772                          if ( ! matches ) {
2773                              return;
2774                          }
2775                          menuItemId = parseInt( matches[1], 10 );
2776                          menuItemControl = api.control( 'nav_menu_item[' + String( menuItemId ) + ']' );
2777                          if ( menuItemControl ) {
2778                              menuItemControls.push( menuItemControl );
2779                          }
2780                      } );
2781  
2782                      _.each( menuItemControls, function( menuItemControl ) {
2783                          if ( false === menuItemControl.setting() ) {
2784                              // Skip deleted items.
2785                              return;
2786                          }
2787                          var setting = _.clone( menuItemControl.setting() );
2788                          position += 1;
2789                          priority += 1;
2790                          setting.position = position;
2791                          menuItemControl.priority( priority );
2792  
2793                          // Note that wpNavMenu will be setting this .menu-item-data-parent-id input's value.
2794                          setting.menu_item_parent = parseInt( menuItemControl.container.find( '.menu-item-data-parent-id' ).val(), 10 );
2795                          if ( ! setting.menu_item_parent ) {
2796                              setting.menu_item_parent = 0;
2797                          }
2798  
2799                          menuItemControl.setting.set( setting );
2800                      });
2801  
2802                      // Mark all menu items as unprocessed.
2803                      $( 'button.item-edit' ).data( 'needs_accessibility_refresh', true );
2804                  });
2805  
2806              });
2807              control.isReordering = false;
2808  
2809              /**
2810               * Keyboard-accessible reordering.
2811               */
2812              this.container.find( '.reorder-toggle' ).on( 'click', function() {
2813                  control.toggleReordering( ! control.isReordering );
2814              } );
2815          },
2816  
2817          /**
2818           * Set up UI for adding a new menu item.
2819           */
2820          _setupAddition: function() {
2821              var self = this;
2822  
2823              this.container.find( '.add-new-menu-item' ).on( 'click', function( event ) {
2824                  if ( self.$sectionContent.hasClass( 'reordering' ) ) {
2825                      return;
2826                  }
2827  
2828                  if ( ! $( 'body' ).hasClass( 'adding-menu-items' ) ) {
2829                      $( this ).attr( 'aria-expanded', 'true' );
2830                      api.Menus.availableMenuItemsPanel.open( self );
2831                  } else {
2832                      $( this ).attr( 'aria-expanded', 'false' );
2833                      api.Menus.availableMenuItemsPanel.close();
2834                      event.stopPropagation();
2835                  }
2836              } );
2837          },
2838  
2839          _handleDeletion: function() {
2840              var control = this,
2841                  section,
2842                  menuId = control.params.menu_id,
2843                  removeSection,
2844                  widgetTemplate,
2845                  navMenuCount = 0;
2846              section = api.section( control.section() );
2847              removeSection = function() {
2848                  section.container.remove();
2849                  api.section.remove( section.id );
2850              };
2851  
2852              if ( section && section.expanded() ) {
2853                  section.collapse({
2854                      completeCallback: function() {
2855                          removeSection();
2856                          wp.a11y.speak( api.Menus.data.l10n.menuDeleted );
2857                          api.panel( 'nav_menus' ).focus();
2858                      }
2859                  });
2860              } else {
2861                  removeSection();
2862              }
2863  
2864              api.each(function( setting ) {
2865                  if ( /^nav_menu\[/.test( setting.id ) && false !== setting() ) {
2866                      navMenuCount += 1;
2867                  }
2868              });
2869  
2870              // Remove the menu from any Navigation Menu widgets.
2871              api.control.each(function( widgetControl ) {
2872                  if ( ! widgetControl.extended( api.controlConstructor.widget_form ) || 'nav_menu' !== widgetControl.params.widget_id_base ) {
2873                      return;
2874                  }
2875                  var select = widgetControl.container.find( 'select' );
2876                  if ( select.val() === String( menuId ) ) {
2877                      select.prop( 'selectedIndex', 0 ).trigger( 'change' );
2878                  }
2879  
2880                  widgetControl.container.find( '.nav-menu-widget-form-controls:first' ).toggle( 0 !== navMenuCount );
2881                  widgetControl.container.find( '.nav-menu-widget-no-menus-message:first' ).toggle( 0 === navMenuCount );
2882                  widgetControl.container.find( 'option[value=' + String( menuId ) + ']' ).remove();
2883              });
2884  
2885              // Remove the menu to the nav menu widget template.
2886              widgetTemplate = $( '#available-widgets-list .widget-tpl:has( input.id_base[ value=nav_menu ] )' );
2887              widgetTemplate.find( '.nav-menu-widget-form-controls:first' ).toggle( 0 !== navMenuCount );
2888              widgetTemplate.find( '.nav-menu-widget-no-menus-message:first' ).toggle( 0 === navMenuCount );
2889              widgetTemplate.find( 'option[value=' + String( menuId ) + ']' ).remove();
2890          },
2891  
2892          /**
2893           * Update Section Title as menu name is changed.
2894           */
2895          _setupTitle: function() {
2896              var control = this;
2897  
2898              control.setting.bind( function( menu ) {
2899                  if ( ! menu ) {
2900                      return;
2901                  }
2902  
2903                  var section = api.section( control.section() ),
2904                      menuId = control.params.menu_id,
2905                      controlTitle = section.headContainer.find( '.accordion-section-title' ),
2906                      sectionTitle = section.contentContainer.find( '.customize-section-title h3' ),
2907                      location = section.headContainer.find( '.menu-in-location' ),
2908                      action = sectionTitle.find( '.customize-action' ),
2909                      name = displayNavMenuName( menu.name );
2910  
2911                  // Update the control title.
2912                  controlTitle.text( name );
2913                  if ( location.length ) {
2914                      location.appendTo( controlTitle );
2915                  }
2916  
2917                  // Update the section title.
2918                  sectionTitle.text( name );
2919                  if ( action.length ) {
2920                      action.prependTo( sectionTitle );
2921                  }
2922  
2923                  // Update the nav menu name in location selects.
2924                  api.control.each( function( control ) {
2925                      if ( /^nav_menu_locations\[/.test( control.id ) ) {
2926                          control.container.find( 'option[value=' + menuId + ']' ).text( name );
2927                      }
2928                  } );
2929  
2930                  // Update the nav menu name in all location checkboxes.
2931                  section.contentContainer.find( '.customize-control-checkbox input' ).each( function() {
2932                      if ( $( this ).prop( 'checked' ) ) {
2933                          $( '.current-menu-location-name-' + $( this ).data( 'location-id' ) ).text( name );
2934                      }
2935                  } );
2936              } );
2937          },
2938  
2939          /***********************************************************************
2940           * Begin public API methods
2941           **********************************************************************/
2942  
2943          /**
2944           * Enable/disable the reordering UI
2945           *
2946           * @param {boolean} showOrHide to enable/disable reordering
2947           */
2948          toggleReordering: function( showOrHide ) {
2949              var addNewItemBtn = this.container.find( '.add-new-menu-item' ),
2950                  reorderBtn = this.container.find( '.reorder-toggle' ),
2951                  itemsTitle = this.$sectionContent.find( '.item-title' );
2952  
2953              showOrHide = Boolean( showOrHide );
2954  
2955              if ( showOrHide === this.$sectionContent.hasClass( 'reordering' ) ) {
2956                  return;
2957              }
2958  
2959              this.isReordering = showOrHide;
2960              this.$sectionContent.toggleClass( 'reordering', showOrHide );
2961              this.$sectionContent.sortable( this.isReordering ? 'disable' : 'enable' );
2962              if ( this.isReordering ) {
2963                  addNewItemBtn.attr({ 'tabindex': '-1', 'aria-hidden': 'true' });
2964                  reorderBtn.attr( 'aria-label', api.Menus.data.l10n.reorderLabelOff );
2965                  wp.a11y.speak( api.Menus.data.l10n.reorderModeOn );
2966                  itemsTitle.attr( 'aria-hidden', 'false' );
2967              } else {
2968                  addNewItemBtn.removeAttr( 'tabindex aria-hidden' );
2969                  reorderBtn.attr( 'aria-label', api.Menus.data.l10n.reorderLabelOn );
2970                  wp.a11y.speak( api.Menus.data.l10n.reorderModeOff );
2971                  itemsTitle.attr( 'aria-hidden', 'true' );
2972              }
2973  
2974              if ( showOrHide ) {
2975                  _( this.getMenuItemControls() ).each( function( formControl ) {
2976                      formControl.collapseForm();
2977                  } );
2978              }
2979          },
2980  
2981          /**
2982           * @return {wp.customize.controlConstructor.nav_menu_item[]}
2983           */
2984          getMenuItemControls: function() {
2985              var menuControl = this,
2986                  menuItemControls = [],
2987                  menuTermId = menuControl.params.menu_id;
2988  
2989              api.control.each(function( control ) {
2990                  if ( 'nav_menu_item' === control.params.type && control.setting() && menuTermId === control.setting().nav_menu_term_id ) {
2991                      menuItemControls.push( control );
2992                  }
2993              });
2994  
2995              return menuItemControls;
2996          },
2997  
2998          /**
2999           * Make sure that each menu item control has the proper depth.
3000           */
3001          reflowMenuItems: function() {
3002              var menuControl = this,
3003                  menuItemControls = menuControl.getMenuItemControls(),
3004                  reflowRecursively;
3005  
3006              reflowRecursively = function( context ) {
3007                  var currentMenuItemControls = [],
3008                      thisParent = context.currentParent;
3009                  _.each( context.menuItemControls, function( menuItemControl ) {
3010                      if ( thisParent === menuItemControl.setting().menu_item_parent ) {
3011                          currentMenuItemControls.push( menuItemControl );
3012                          // @todo We could remove this item from menuItemControls now, for efficiency.
3013                      }
3014                  });
3015                  currentMenuItemControls.sort( function( a, b ) {
3016                      return a.setting().position - b.setting().position;
3017                  });
3018  
3019                  _.each( currentMenuItemControls, function( menuItemControl ) {
3020                      // Update position.
3021                      context.currentAbsolutePosition += 1;
3022                      menuItemControl.priority.set( context.currentAbsolutePosition ); // This will change the sort order.
3023  
3024                      // Update depth.
3025                      if ( ! menuItemControl.container.hasClass( 'menu-item-depth-' + String( context.currentDepth ) ) ) {
3026                          _.each( menuItemControl.container.prop( 'className' ).match( /menu-item-depth-\d+/g ), function( className ) {
3027                              menuItemControl.container.removeClass( className );
3028                          });
3029                          menuItemControl.container.addClass( 'menu-item-depth-' + String( context.currentDepth ) );
3030                      }
3031                      menuItemControl.container.data( 'item-depth', context.currentDepth );
3032  
3033                      // Process any children items.
3034                      context.currentDepth += 1;
3035                      context.currentParent = menuItemControl.params.menu_item_id;
3036                      reflowRecursively( context );
3037                      context.currentDepth -= 1;
3038                      context.currentParent = thisParent;
3039                  });
3040  
3041                  // Update class names for reordering controls.
3042                  if ( currentMenuItemControls.length ) {
3043                      _( currentMenuItemControls ).each(function( menuItemControl ) {
3044                          menuItemControl.container.removeClass( 'move-up-disabled move-down-disabled move-left-disabled move-right-disabled' );
3045                          if ( 0 === context.currentDepth ) {
3046                              menuItemControl.container.addClass( 'move-left-disabled' );
3047                          } else if ( 10 === context.currentDepth ) {
3048                              menuItemControl.container.addClass( 'move-right-disabled' );
3049                          }
3050                      });
3051  
3052                      currentMenuItemControls[0].container
3053                          .addClass( 'move-up-disabled' )
3054                          .addClass( 'move-right-disabled' )
3055                          .toggleClass( 'move-down-disabled', 1 === currentMenuItemControls.length );
3056                      currentMenuItemControls[ currentMenuItemControls.length - 1 ].container
3057                          .addClass( 'move-down-disabled' )
3058                          .toggleClass( 'move-up-disabled', 1 === currentMenuItemControls.length );
3059                  }
3060              };
3061  
3062              reflowRecursively( {
3063                  menuItemControls: menuItemControls,
3064                  currentParent: 0,
3065                  currentDepth: 0,
3066                  currentAbsolutePosition: 0
3067              } );
3068  
3069              menuControl.updateInvitationVisibility( menuItemControls );
3070              menuControl.container.find( '.reorder-toggle' ).toggle( menuItemControls.length > 1 );
3071          },
3072  
3073          /**
3074           * Note that this function gets debounced so that when a lot of setting
3075           * changes are made at once, for instance when moving a menu item that
3076           * has child items, this function will only be called once all of the
3077           * settings have been updated.
3078           */
3079          debouncedReflowMenuItems: _.debounce( function() {
3080              this.reflowMenuItems.apply( this, arguments );
3081          }, 0 ),
3082  
3083          /**
3084           * Add a new item to this menu.
3085           *
3086           * @param {Object} item - Value for the nav_menu_item setting to be created.
3087           * @return {wp.customize.Menus.controlConstructor.nav_menu_item} The newly-created nav_menu_item control instance.
3088           */
3089          addItemToMenu: function( item ) {
3090              var menuControl = this, customizeId, settingArgs, setting, menuItemControl, placeholderId, position = 0, priority = 10,
3091                  originalItemId = item.id || '';
3092  
3093              _.each( menuControl.getMenuItemControls(), function( control ) {
3094                  if ( false === control.setting() ) {
3095                      return;
3096                  }
3097                  priority = Math.max( priority, control.priority() );
3098                  if ( 0 === control.setting().menu_item_parent ) {
3099                      position = Math.max( position, control.setting().position );
3100                  }
3101              });
3102              position += 1;
3103              priority += 1;
3104  
3105              item = $.extend(
3106                  {},
3107                  api.Menus.data.defaultSettingValues.nav_menu_item,
3108                  item,
3109                  {
3110                      nav_menu_term_id: menuControl.params.menu_id,
3111                      original_title: item.title,
3112                      position: position
3113                  }
3114              );
3115              delete item.id; // Only used by Backbone.
3116  
3117              placeholderId = api.Menus.generatePlaceholderAutoIncrementId();
3118              customizeId = 'nav_menu_item[' + String( placeholderId ) + ']';
3119              settingArgs = {
3120                  type: 'nav_menu_item',
3121                  transport: api.Menus.data.settingTransport,
3122                  previewer: api.previewer
3123              };
3124              setting = api.create( customizeId, customizeId, {}, settingArgs );
3125              setting.set( item ); // Change from initial empty object to actual item to mark as dirty.
3126  
3127              // Add the menu item control.
3128              menuItemControl = new api.controlConstructor.nav_menu_item( customizeId, {
3129                  type: 'nav_menu_item',
3130                  section: menuControl.id,
3131                  priority: priority,
3132                  settings: {
3133                      'default': customizeId
3134                  },
3135                  menu_item_id: placeholderId,
3136                  original_item_id: originalItemId
3137              } );
3138  
3139              api.control.add( menuItemControl );
3140              setting.preview();
3141              menuControl.debouncedReflowMenuItems();
3142  
3143              wp.a11y.speak( api.Menus.data.l10n.itemAdded );
3144  
3145              return menuItemControl;
3146          },
3147  
3148          /**
3149           * Show an invitation to add new menu items when there are no menu items.
3150           *
3151           * @since 4.9.0
3152           *
3153           * @param {wp.customize.controlConstructor.nav_menu_item[]} optionalMenuItemControls
3154           */
3155          updateInvitationVisibility: function ( optionalMenuItemControls ) {
3156              var menuItemControls = optionalMenuItemControls || this.getMenuItemControls();
3157  
3158              this.container.find( '.new-menu-item-invitation' ).toggle( menuItemControls.length === 0 );
3159          }
3160      } );
3161  
3162      /**
3163       * Extends wp.customize.controlConstructor with control constructor for
3164       * menu_location, menu_item, nav_menu, and new_menu.
3165       */
3166      $.extend( api.controlConstructor, {
3167          nav_menu_location: api.Menus.MenuLocationControl,
3168          nav_menu_item: api.Menus.MenuItemControl,
3169          nav_menu: api.Menus.MenuControl,
3170          nav_menu_name: api.Menus.MenuNameControl,
3171          nav_menu_locations: api.Menus.MenuLocationsControl,
3172          nav_menu_auto_add: api.Menus.MenuAutoAddControl
3173      });
3174  
3175      /**
3176       * Extends wp.customize.panelConstructor with section constructor for menus.
3177       */
3178      $.extend( api.panelConstructor, {
3179          nav_menus: api.Menus.MenusPanel
3180      });
3181  
3182      /**
3183       * Extends wp.customize.sectionConstructor with section constructor for menu.
3184       */
3185      $.extend( api.sectionConstructor, {
3186          nav_menu: api.Menus.MenuSection,
3187          new_menu: api.Menus.NewMenuSection
3188      });
3189  
3190      /**
3191       * Init Customizer for menus.
3192       */
3193      api.bind( 'ready', function() {
3194  
3195          // Set up the menu items panel.
3196          api.Menus.availableMenuItemsPanel = new api.Menus.AvailableMenuItemsPanelView({
3197              collection: api.Menus.availableMenuItems
3198          });
3199  
3200          api.bind( 'saved', function( data ) {
3201              if ( data.nav_menu_updates || data.nav_menu_item_updates ) {
3202                  api.Menus.applySavedData( data );
3203              }
3204          } );
3205  
3206          /*
3207           * Reset the list of posts created in the customizer once published.
3208           * The setting is updated quietly (bypassing events being triggered)
3209           * so that the customized state doesn't become immediately dirty.
3210           */
3211          api.state( 'changesetStatus' ).bind( function( status ) {
3212              if ( 'publish' === status ) {
3213                  api( 'nav_menus_created_posts' )._value = [];
3214              }
3215          } );
3216  
3217          // Open and focus menu control.
3218          api.previewer.bind( 'focus-nav-menu-item-control', api.Menus.focusMenuItemControl );
3219      } );
3220  
3221      /**
3222       * When customize_save comes back with a success, make sure any inserted
3223       * nav menus and items are properly re-added with their newly-assigned IDs.
3224       *
3225       * @alias wp.customize.Menus.applySavedData
3226       *
3227       * @param {Object} data
3228       * @param {Array} data.nav_menu_updates
3229       * @param {Array} data.nav_menu_item_updates
3230       */
3231      api.Menus.applySavedData = function( data ) {
3232  
3233          var insertedMenuIdMapping = {}, insertedMenuItemIdMapping = {};
3234  
3235          _( data.nav_menu_updates ).each(function( update ) {
3236              var oldCustomizeId, newCustomizeId, customizeId, oldSetting, newSetting, setting, settingValue, oldSection, newSection, wasSaved, widgetTemplate, navMenuCount, shouldExpandNewSection;
3237              if ( 'inserted' === update.status ) {
3238                  if ( ! update.previous_term_id ) {
3239                      throw new Error( 'Expected previous_term_id' );
3240                  }
3241                  if ( ! update.term_id ) {
3242                      throw new Error( 'Expected term_id' );
3243                  }
3244                  oldCustomizeId = 'nav_menu[' + String( update.previous_term_id ) + ']';
3245                  if ( ! api.has( oldCustomizeId ) ) {
3246                      throw new Error( 'Expected setting to exist: ' + oldCustomizeId );
3247                  }
3248                  oldSetting = api( oldCustomizeId );
3249                  if ( ! api.section.has( oldCustomizeId ) ) {
3250                      throw new Error( 'Expected control to exist: ' + oldCustomizeId );
3251                  }
3252                  oldSection = api.section( oldCustomizeId );
3253  
3254                  settingValue = oldSetting.get();
3255                  if ( ! settingValue ) {
3256                      throw new Error( 'Did not expect setting to be empty (deleted).' );
3257                  }
3258                  settingValue = $.extend( _.clone( settingValue ), update.saved_value );
3259  
3260                  insertedMenuIdMapping[ update.previous_term_id ] = update.term_id;
3261                  newCustomizeId = 'nav_menu[' + String( update.term_id ) + ']';
3262                  newSetting = api.create( newCustomizeId, newCustomizeId, settingValue, {
3263                      type: 'nav_menu',
3264                      transport: api.Menus.data.settingTransport,
3265                      previewer: api.previewer
3266                  } );
3267  
3268                  shouldExpandNewSection = oldSection.expanded();
3269                  if ( shouldExpandNewSection ) {
3270                      oldSection.collapse();
3271                  }
3272  
3273                  // Add the menu section.
3274                  newSection = new api.Menus.MenuSection( newCustomizeId, {
3275                      panel: 'nav_menus',
3276                      title: settingValue.name,
3277                      customizeAction: api.Menus.data.l10n.customizingMenus,
3278                      type: 'nav_menu',
3279                      priority: oldSection.priority.get(),
3280                      menu_id: update.term_id
3281                  } );
3282  
3283                  // Add new control for the new menu.
3284                  api.section.add( newSection );
3285  
3286                  // Update the values for nav menus in Navigation Menu controls.
3287                  api.control.each( function( setting ) {
3288                      if ( ! setting.extended( api.controlConstructor.widget_form ) || 'nav_menu' !== setting.params.widget_id_base ) {
3289                          return;
3290                      }
3291                      var select, oldMenuOption, newMenuOption;
3292                      select = setting.container.find( 'select' );
3293                      oldMenuOption = select.find( 'option[value=' + String( update.previous_term_id ) + ']' );
3294                      newMenuOption = select.find( 'option[value=' + String( update.term_id ) + ']' );
3295                      newMenuOption.prop( 'selected', oldMenuOption.prop( 'selected' ) );
3296                      oldMenuOption.remove();
3297                  } );
3298  
3299                  // Delete the old placeholder nav_menu.
3300                  oldSetting.callbacks.disable(); // Prevent setting triggering Customizer dirty state when set.
3301                  oldSetting.set( false );
3302                  oldSetting.preview();
3303                  newSetting.preview();
3304                  oldSetting._dirty = false;
3305  
3306                  // Remove nav_menu section.
3307                  oldSection.container.remove();
3308                  api.section.remove( oldCustomizeId );
3309  
3310                  // Update the nav_menu widget to reflect removed placeholder menu.
3311                  navMenuCount = 0;
3312                  api.each(function( setting ) {
3313                      if ( /^nav_menu\[/.test( setting.id ) && false !== setting() ) {
3314                          navMenuCount += 1;
3315                      }
3316                  });
3317                  widgetTemplate = $( '#available-widgets-list .widget-tpl:has( input.id_base[ value=nav_menu ] )' );
3318                  widgetTemplate.find( '.nav-menu-widget-form-controls:first' ).toggle( 0 !== navMenuCount );
3319                  widgetTemplate.find( '.nav-menu-widget-no-menus-message:first' ).toggle( 0 === navMenuCount );
3320                  widgetTemplate.find( 'option[value=' + String( update.previous_term_id ) + ']' ).remove();
3321  
3322                  // Update the nav_menu_locations[...] controls to remove the placeholder menus from the dropdown options.
3323                  wp.customize.control.each(function( control ){
3324                      if ( /^nav_menu_locations\[/.test( control.id ) ) {
3325                          control.container.find( 'option[value=' + String( update.previous_term_id ) + ']' ).remove();
3326                      }
3327                  });
3328  
3329                  // Update nav_menu_locations to reference the new ID.
3330                  api.each( function( setting ) {
3331                      var wasSaved = api.state( 'saved' ).get();
3332                      if ( /^nav_menu_locations\[/.test( setting.id ) && setting.get() === update.previous_term_id ) {
3333                          setting.set( update.term_id );
3334                          setting._dirty = false; // Not dirty because this is has also just been done on server in WP_Customize_Nav_Menu_Setting::update().
3335                          api.state( 'saved' ).set( wasSaved );
3336                          setting.preview();
3337                      }
3338                  } );
3339  
3340                  if ( shouldExpandNewSection ) {
3341                      newSection.expand();
3342                  }
3343              } else if ( 'updated' === update.status ) {
3344                  customizeId = 'nav_menu[' + String( update.term_id ) + ']';
3345                  if ( ! api.has( customizeId ) ) {
3346                      throw new Error( 'Expected setting to exist: ' + customizeId );
3347                  }
3348  
3349                  // Make sure the setting gets updated with its sanitized server value (specifically the conflict-resolved name).
3350                  setting = api( customizeId );
3351                  if ( ! _.isEqual( update.saved_value, setting.get() ) ) {
3352                      wasSaved = api.state( 'saved' ).get();
3353                      setting.set( update.saved_value );
3354                      setting._dirty = false;
3355                      api.state( 'saved' ).set( wasSaved );
3356                  }
3357              }
3358          } );
3359  
3360          // Build up mapping of nav_menu_item placeholder IDs to inserted IDs.
3361          _( data.nav_menu_item_updates ).each(function( update ) {
3362              if ( update.previous_post_id ) {
3363                  insertedMenuItemIdMapping[ update.previous_post_id ] = update.post_id;
3364              }
3365          });
3366  
3367          _( data.nav_menu_item_updates ).each(function( update ) {
3368              var oldCustomizeId, newCustomizeId, oldSetting, newSetting, settingValue, oldControl, newControl;
3369              if ( 'inserted' === update.status ) {
3370                  if ( ! update.previous_post_id ) {
3371                      throw new Error( 'Expected previous_post_id' );
3372                  }
3373                  if ( ! update.post_id ) {
3374                      throw new Error( 'Expected post_id' );
3375                  }
3376                  oldCustomizeId = 'nav_menu_item[' + String( update.previous_post_id ) + ']';
3377                  if ( ! api.has( oldCustomizeId ) ) {
3378                      throw new Error( 'Expected setting to exist: ' + oldCustomizeId );
3379                  }
3380                  oldSetting = api( oldCustomizeId );
3381                  if ( ! api.control.has( oldCustomizeId ) ) {
3382                      throw new Error( 'Expected control to exist: ' + oldCustomizeId );
3383                  }
3384                  oldControl = api.control( oldCustomizeId );
3385  
3386                  settingValue = oldSetting.get();
3387                  if ( ! settingValue ) {
3388                      throw new Error( 'Did not expect setting to be empty (deleted).' );
3389                  }
3390                  settingValue = _.clone( settingValue );
3391  
3392                  // If the parent menu item was also inserted, update the menu_item_parent to the new ID.
3393                  if ( settingValue.menu_item_parent < 0 ) {
3394                      if ( ! insertedMenuItemIdMapping[ settingValue.menu_item_parent ] ) {
3395                          throw new Error( 'inserted ID for menu_item_parent not available' );
3396                      }
3397                      settingValue.menu_item_parent = insertedMenuItemIdMapping[ settingValue.menu_item_parent ];
3398                  }
3399  
3400                  // If the menu was also inserted, then make sure it uses the new menu ID for nav_menu_term_id.
3401                  if ( insertedMenuIdMapping[ settingValue.nav_menu_term_id ] ) {
3402                      settingValue.nav_menu_term_id = insertedMenuIdMapping[ settingValue.nav_menu_term_id ];
3403                  }
3404  
3405                  newCustomizeId = 'nav_menu_item[' + String( update.post_id ) + ']';
3406                  newSetting = api.create( newCustomizeId, newCustomizeId, settingValue, {
3407                      type: 'nav_menu_item',
3408                      transport: api.Menus.data.settingTransport,
3409                      previewer: api.previewer
3410                  } );
3411  
3412                  // Add the menu control.
3413                  newControl = new api.controlConstructor.nav_menu_item( newCustomizeId, {
3414                      type: 'nav_menu_item',
3415                      menu_id: update.post_id,
3416                      section: 'nav_menu[' + String( settingValue.nav_menu_term_id ) + ']',
3417                      priority: oldControl.priority.get(),
3418                      settings: {
3419                          'default': newCustomizeId
3420                      },
3421                      menu_item_id: update.post_id
3422                  } );
3423  
3424                  // Remove old control.
3425                  oldControl.container.remove();
3426                  api.control.remove( oldCustomizeId );
3427  
3428                  // Add new control to take its place.
3429                  api.control.add( newControl );
3430  
3431                  // Delete the placeholder and preview the new setting.
3432                  oldSetting.callbacks.disable(); // Prevent setting triggering Customizer dirty state when set.
3433                  oldSetting.set( false );
3434                  oldSetting.preview();
3435                  newSetting.preview();
3436                  oldSetting._dirty = false;
3437  
3438                  newControl.container.toggleClass( 'menu-item-edit-inactive', oldControl.container.hasClass( 'menu-item-edit-inactive' ) );
3439              }
3440          });
3441  
3442          /*
3443           * Update the settings for any nav_menu widgets that had selected a placeholder ID.
3444           */
3445          _.each( data.widget_nav_menu_updates, function( widgetSettingValue, widgetSettingId ) {
3446              var setting = api( widgetSettingId );
3447              if ( setting ) {
3448                  setting._value = widgetSettingValue;
3449                  setting.preview(); // Send to the preview now so that menu refresh will use the inserted menu.
3450              }
3451          });
3452      };
3453  
3454      /**
3455       * Focus a menu item control.
3456       *
3457       * @alias wp.customize.Menus.focusMenuItemControl
3458       *
3459       * @param {string} menuItemId
3460       */
3461      api.Menus.focusMenuItemControl = function( menuItemId ) {
3462          var control = api.Menus.getMenuItemControl( menuItemId );
3463          if ( control ) {
3464              control.focus();
3465          }
3466      };
3467  
3468      /**
3469       * Get the control for a given menu.
3470       *
3471       * @alias wp.customize.Menus.getMenuControl
3472       *
3473       * @param menuId
3474       * @return {wp.customize.controlConstructor.menus[]}
3475       */
3476      api.Menus.getMenuControl = function( menuId ) {
3477          return api.control( 'nav_menu[' + menuId + ']' );
3478      };
3479  
3480      /**
3481       * Given a menu item ID, get the control associated with it.
3482       *
3483       * @alias wp.customize.Menus.getMenuItemControl
3484       *
3485       * @param {string} menuItemId
3486       * @return {Object|null}
3487       */
3488      api.Menus.getMenuItemControl = function( menuItemId ) {
3489          return api.control( menuItemIdToSettingId( menuItemId ) );
3490      };
3491  
3492      /**
3493       * @alias wp.customize.Menus~menuItemIdToSettingId
3494       *
3495       * @param {string} menuItemId
3496       */
3497  	function menuItemIdToSettingId( menuItemId ) {
3498          return 'nav_menu_item[' + menuItemId + ']';
3499      }
3500  
3501      /**
3502       * Apply sanitize_text_field()-like logic to the supplied name, returning a
3503       * "unnammed" fallback string if the name is then empty.
3504       *
3505       * @alias wp.customize.Menus~displayNavMenuName
3506       *
3507       * @param {string} name
3508       * @return {string}
3509       */
3510  	function displayNavMenuName( name ) {
3511          name = name || '';
3512          name = wp.sanitize.stripTagsAndEncodeText( name ); // Remove any potential tags from name.
3513          name = name.toString().trim();
3514          return name || api.Menus.data.l10n.unnamed;
3515      }
3516  
3517  })( wp.customize, wp, jQuery );


Generated : Sat Dec 21 08:20:01 2024 Cross-referenced by PHPXref