/** * @output wp-includes/js/customize-preview-nav-menus.js */ /* global _wpCustomizePreviewNavMenusExports */ /** @namespace wp.customize.navMenusPreview */ wp.customize.navMenusPreview = wp.customize.MenusCustomizerPreview = ( function( $, _, wp, api ) { 'use strict'; var self = { data: { navMenuInstanceArgs: {} } }; if ( 'undefined' !== typeof _wpCustomizePreviewNavMenusExports ) { _.extend( self.data, _wpCustomizePreviewNavMenusExports ); } /** * Initialize nav menus preview. */ self.init = function() { var self = this, synced = false; /* * Keep track of whether we synced to determine whether or not bindSettingListener * should also initially fire the listener. This initial firing needs to wait until * after all of the settings have been synced from the pane in order to prevent * an infinite selective fallback-refresh. Note that this sync handler will be * added after the sync handler in customize-preview.js, so it will be triggered * after all of the settings are added. */ api.preview.bind( 'sync', function() { synced = true; } ); if ( api.selectiveRefresh ) { // Listen for changes to settings related to nav menus. api.each( function( setting ) { self.bindSettingListener( setting ); } ); api.bind( 'add', function( setting ) { /* * Handle case where an invalid nav menu item (one for which its associated object has been deleted) * is synced from the controls into the preview. Since invalid nav menu items are filtered out from * being exported to the frontend by the _is_valid_nav_menu_item filter in wp_get_nav_menu_items(), * the customizer controls will have a nav_menu_item setting where the preview will have none, and * this can trigger an infinite fallback refresh when the nav menu item lacks any valid items. */ if ( setting.get() && ! setting.get()._invalid ) { self.bindSettingListener( setting, { fire: synced } ); } } ); api.bind( 'remove', function( setting ) { self.unbindSettingListener( setting ); } ); /* * Ensure that wp_nav_menu() instances nested inside of other partials * will be recognized as being present on the page. */ api.selectiveRefresh.bind( 'render-partials-response', function( response ) { if ( response.nav_menu_instance_args ) { _.extend( self.data.navMenuInstanceArgs, response.nav_menu_instance_args ); } } ); } api.preview.bind( 'active', function() { self.highlightControls(); } ); }; if ( api.selectiveRefresh ) { /** * Partial representing an invocation of wp_nav_menu(). * * @memberOf wp.customize.navMenusPreview * @alias wp.customize.navMenusPreview.NavMenuInstancePartial * * @class * @augments wp.customize.selectiveRefresh.Partial * @since 4.5.0 */ self.NavMenuInstancePartial = api.selectiveRefresh.Partial.extend(/** @lends wp.customize.navMenusPreview.NavMenuInstancePartial.prototype */{ /** * Constructor. * * @since 4.5.0 * @param {string} id - Partial ID. * @param {Object} options * @param {Object} options.params * @param {Object} options.params.navMenuArgs * @param {string} options.params.navMenuArgs.args_hmac * @param {string} [options.params.navMenuArgs.theme_location] * @param {number} [options.params.navMenuArgs.menu] * @param {Object} [options.constructingContainerContext] */ initialize: function( id, options ) { var partial = this, matches, argsHmac; matches = id.match( /^nav_menu_instance\[([0-9a-f]{32})]$/ ); if ( ! matches ) { throw new Error( 'Illegal id for nav_menu_instance partial. The key corresponds with the args HMAC.' ); } argsHmac = matches[1]; options = options || {}; options.params = _.extend( { selector: '[data-customize-partial-id="' + id + '"]', navMenuArgs: options.constructingContainerContext || {}, containerInclusive: true }, options.params || {} ); api.selectiveRefresh.Partial.prototype.initialize.call( partial, id, options ); if ( ! _.isObject( partial.params.navMenuArgs ) ) { throw new Error( 'Missing navMenuArgs' ); } if ( partial.params.navMenuArgs.args_hmac !== argsHmac ) { throw new Error( 'args_hmac mismatch with id' ); } }, /** * Return whether the setting is related to this partial. * * @since 4.5.0 * @param {wp.customize.Value|string} setting - Object or ID. * @param {number|Object|false|null} newValue - New value, or null if the setting was just removed. * @param {number|Object|false|null} oldValue - Old value, or null if the setting was just added. * @return {boolean} */ isRelatedSetting: function( setting, newValue, oldValue ) { var partial = this, navMenuLocationSetting, navMenuId, isNavMenuItemSetting, _newValue, _oldValue, urlParser; if ( _.isString( setting ) ) { setting = api( setting ); } /* * Prevent nav_menu_item changes only containing type_label differences triggering a refresh. * These settings in the preview do not include type_label property, and so if one of these * nav_menu_item settings is dirty, after a refresh the nav menu instance would do a selective * refresh immediately because the setting from the pane would have the type_label whereas * the setting in the preview would not, thus triggering a change event. The following * condition short-circuits this unnecessary selective refresh and also prevents an infinite * loop in the case where a nav_menu_instance partial had done a fallback refresh. * @todo Nav menu item settings should not include a type_label property to begin with. */ isNavMenuItemSetting = /^nav_menu_item\[/.test( setting.id ); if ( isNavMenuItemSetting && _.isObject( newValue ) && _.isObject( oldValue ) ) { _newValue = _.clone( newValue ); _oldValue = _.clone( oldValue ); delete _newValue.type_label; delete _oldValue.type_label; // Normalize URL scheme when parent frame is HTTPS to prevent selective refresh upon initial page load. if ( 'https' === api.preview.scheme.get() ) { urlParser = document.createElement( 'a' ); urlParser.href = _newValue.url; urlParser.protocol = 'https:'; _newValue.url = urlParser.href; urlParser.href = _oldValue.url; urlParser.protocol = 'https:'; _oldValue.url = urlParser.href; } // Prevent original_title differences from causing refreshes if title is present. if ( newValue.title ) { delete _oldValue.original_title; delete _newValue.original_title; } if ( _.isEqual( _oldValue, _newValue ) ) { return false; } } if ( partial.params.navMenuArgs.theme_location ) { if ( 'nav_menu_locations[' + partial.params.navMenuArgs.theme_location + ']' === setting.id ) { return true; } navMenuLocationSetting = api( 'nav_menu_locations[' + partial.params.navMenuArgs.theme_location + ']' ); } navMenuId = partial.params.navMenuArgs.menu; if ( ! navMenuId && navMenuLocationSetting ) { navMenuId = navMenuLocationSetting(); } if ( ! navMenuId ) { return false; } return ( ( 'nav_menu[' + navMenuId + ']' === setting.id ) || ( isNavMenuItemSetting && ( ( newValue && newValue.nav_menu_term_id === navMenuId ) || ( oldValue && oldValue.nav_menu_term_id === navMenuId ) ) ) ); }, /** * Make sure that partial fallback behavior is invoked if there is no associated menu. * * @since 4.5.0 * * @return {Promise} */ refresh: function() { var partial = this, menuId, deferred = $.Deferred(); // Make sure the fallback behavior is invoked when the partial is no longer associated with a menu. if ( _.isNumber( partial.params.navMenuArgs.menu ) ) { menuId = partial.params.navMenuArgs.menu; } else if ( partial.params.navMenuArgs.theme_location && api.has( 'nav_menu_locations[' + partial.params.navMenuArgs.theme_location + ']' ) ) { menuId = api( 'nav_menu_locations[' + partial.params.navMenuArgs.theme_location + ']' ).get(); } if ( ! menuId ) { partial.fallback(); deferred.reject(); return deferred.promise(); } return api.selectiveRefresh.Partial.prototype.refresh.call( partial ); }, /** * Render content. * * @inheritdoc * @param {wp.customize.selectiveRefresh.Placement} placement */ renderContent: function( placement ) { var partial = this, previousContainer = placement.container; // Do fallback behavior to refresh preview if menu is now empty. if ( '' === placement.addedContent ) { placement.partial.fallback(); } if ( api.selectiveRefresh.Partial.prototype.renderContent.call( partial, placement ) ) { // Trigger deprecated event. $( document ).trigger( 'customize-preview-menu-refreshed', [ { instanceNumber: null, // @deprecated wpNavArgs: placement.context, // @deprecated wpNavMenuArgs: placement.context, oldContainer: previousContainer, newContainer: placement.container } ] ); } } }); api.selectiveRefresh.partialConstructor.nav_menu_instance = self.NavMenuInstancePartial; /** * Request full refresh if there are nav menu instances that lack partials which also match the supplied args. * * @param {Object} navMenuInstanceArgs */ self.handleUnplacedNavMenuInstances = function( navMenuInstanceArgs ) { var unplacedNavMenuInstances; unplacedNavMenuInstances = _.filter( _.values( self.data.navMenuInstanceArgs ), function( args ) { return ! api.selectiveRefresh.partial.has( 'nav_menu_instance[' + args.args_hmac + ']' ); } ); if ( _.findWhere( unplacedNavMenuInstances, navMenuInstanceArgs ) ) { api.selectiveRefresh.requestFullRefresh(); return true; } return false; }; /** * Add change listener for a nav_menu[], nav_menu_item[], or nav_menu_locations[] setting. * * @since 4.5.0 * * @param {wp.customize.Value} setting * @param {Object} [options] * @param {boolean} options.fire Whether to invoke the callback after binding. * This is used when a dynamic setting is added. * @return {boolean} Whether the setting was bound. */ self.bindSettingListener = function( setting, options ) { var matches; options = options || {}; matches = setting.id.match( /^nav_menu\[(-?\d+)]$/ ); if ( matches ) { setting._navMenuId = parseInt( matches[1], 10 ); setting.bind( this.onChangeNavMenuSetting ); if ( options.fire ) { this.onChangeNavMenuSetting.call( setting, setting(), false ); } return true; } matches = setting.id.match( /^nav_menu_item\[(-?\d+)]$/ ); if ( matches ) { setting._navMenuItemId = parseInt( matches[1], 10 ); setting.bind( this.onChangeNavMenuItemSetting ); if ( options.fire ) { this.onChangeNavMenuItemSetting.call( setting, setting(), false ); } return true; } matches = setting.id.match( /^nav_menu_locations\[(.+?)]/ ); if ( matches ) { setting._navMenuThemeLocation = matches[1]; setting.bind( this.onChangeNavMenuLocationsSetting ); if ( options.fire ) { this.onChangeNavMenuLocationsSetting.call( setting, setting(), false ); } return true; } return false; }; /** * Remove change listeners for nav_menu[], nav_menu_item[], or nav_menu_locations[] setting. * * @since 4.5.0 * * @param {wp.customize.Value} setting */ self.unbindSettingListener = function( setting ) { setting.unbind( this.onChangeNavMenuSetting ); setting.unbind( this.onChangeNavMenuItemSetting ); setting.unbind( this.onChangeNavMenuLocationsSetting ); }; /** * Handle change for nav_menu[] setting for nav menu instances lacking partials. * * @since 4.5.0 * * @this {wp.customize.Value} */ self.onChangeNavMenuSetting = function() { var setting = this; self.handleUnplacedNavMenuInstances( { menu: setting._navMenuId } ); // Ensure all nav menu instances with a theme_location assigned to this menu are handled. api.each( function( otherSetting ) { if ( ! otherSetting._navMenuThemeLocation ) { return; } if ( setting._navMenuId === otherSetting() ) { self.handleUnplacedNavMenuInstances( { theme_location: otherSetting._navMenuThemeLocation } ); } } ); }; /** * Handle change for nav_menu_item[] setting for nav menu instances lacking partials. * * @since 4.5.0 * * @param {Object} newItem New value for nav_menu_item[] setting. * @param {Object} oldItem Old value for nav_menu_item[] setting. * @this {wp.customize.Value} */ self.onChangeNavMenuItemSetting = function( newItem, oldItem ) { var item = newItem || oldItem, navMenuSetting; navMenuSetting = api( 'nav_menu[' + String( item.nav_menu_term_id ) + ']' ); if ( navMenuSetting ) { self.onChangeNavMenuSetting.call( navMenuSetting ); } }; /** * Handle change for nav_menu_locations[] setting for nav menu instances lacking partials. * * @since 4.5.0 * * @this {wp.customize.Value} */ self.onChangeNavMenuLocationsSetting = function() { var setting = this, hasNavMenuInstance; self.handleUnplacedNavMenuInstances( { theme_location: setting._navMenuThemeLocation } ); // If there are no wp_nav_menu() instances that refer to the theme location, do full refresh. hasNavMenuInstance = !! _.findWhere( _.values( self.data.navMenuInstanceArgs ), { theme_location: setting._navMenuThemeLocation } ); if ( ! hasNavMenuInstance ) { api.selectiveRefresh.requestFullRefresh(); } }; } /** * Connect nav menu items with their corresponding controls in the pane. * * Setup shift-click on nav menu items which are more granular than the nav menu partial itself. * Also this applies even if a nav menu is not partial-refreshable. * * @since 4.5.0 */ self.highlightControls = function() { var selector = '.menu-item'; // Skip adding highlights if not in the customizer preview iframe. if ( ! api.settings.channel ) { return; } // Focus on the menu item control when shift+clicking the menu item. $( document ).on( 'click', selector, function( e ) { var navMenuItemParts; if ( ! e.shiftKey ) { return; } navMenuItemParts = $( this ).attr( 'class' ).match( /(?:^|\s)menu-item-(-?\d+)(?:\s|$)/ ); if ( navMenuItemParts ) { e.preventDefault(); e.stopPropagation(); // Make sure a sub-nav menu item will get focused instead of parent items. api.preview.send( 'focus-nav-menu-item-control', parseInt( navMenuItemParts[1], 10 ) ); } }); }; api.bind( 'preview-ready', function() { self.init(); } ); return self; }( jQuery, _, wp, wp.customize ) );
Name | Type | Size | Permission | Actions |
---|---|---|---|---|
codemirror | Folder | 0555 |
|
|
crop | Folder | 0555 |
|
|
dist | Folder | 0555 |
|
|
imgareaselect | Folder | 0555 |
|
|
jcrop | Folder | 0555 |
|
|
jquery | Folder | 0555 |
|
|
mediaelement | Folder | 0555 |
|
|
plupload | Folder | 0555 |
|
|
swfupload | Folder | 0555 |
|
|
thickbox | Folder | 0555 |
|
|
tinymce | Folder | 0555 |
|
|
admin-bar.js | File | 10.3 KB | 0644 |
|
admin-bar.min.js | File | 3.41 KB | 0644 |
|
api-request.js | File | 3.25 KB | 0644 |
|
api-request.min.js | File | 1023 B | 0644 |
|
autosave.js | File | 21.95 KB | 0644 |
|
autosave.min.js | File | 5.67 KB | 0644 |
|
backbone.js | File | 77.87 KB | 0644 |
|
backbone.min.js | File | 23.57 KB | 0644 |
|
clipboard.js | File | 26.18 KB | 0644 |
|
clipboard.min.js | File | 8.8 KB | 0644 |
|
colorpicker.js | File | 28.4 KB | 0644 |
|
colorpicker.min.js | File | 16.13 KB | 0644 |
|
comment-reply.js | File | 12.17 KB | 0644 |
|
comment-reply.min.js | File | 2.91 KB | 0644 |
|
customize-base.js | File | 25.22 KB | 0644 |
|
customize-base.min.js | File | 7.67 KB | 0644 |
|
customize-loader.js | File | 7.72 KB | 0644 |
|
customize-loader.min.js | File | 3.47 KB | 0644 |
|
customize-models.js | File | 6.66 KB | 0644 |
|
customize-models.min.js | File | 3.59 KB | 0644 |
|
customize-preview-nav-menus.js | File | 14.67 KB | 0644 |
|
customize-preview-nav-menus.min.js | File | 4.92 KB | 0644 |
|
customize-preview-widgets.js | File | 22.71 KB | 0644 |
|
customize-preview-widgets.min.js | File | 7.64 KB | 0644 |
|
customize-preview.js | File | 27.3 KB | 0644 |
|
customize-preview.min.js | File | 10.45 KB | 0644 |
|
customize-selective-refresh.js | File | 32.55 KB | 0644 |
|
customize-selective-refresh.min.js | File | 10.44 KB | 0644 |
|
customize-views.js | File | 4.95 KB | 0644 |
|
customize-views.min.js | File | 2.39 KB | 0644 |
|
heartbeat.js | File | 23.39 KB | 0644 |
|
heartbeat.min.js | File | 5.87 KB | 0644 |
|
hoverIntent.js | File | 7.06 KB | 0644 |
|
hoverIntent.min.js | File | 1.46 KB | 0644 |
|
hoverintent-js.min.js | File | 1.68 KB | 0644 |
|
imagesloaded.min.js | File | 5.39 KB | 0644 |
|
json2.js | File | 17.99 KB | 0644 |
|
json2.min.js | File | 3.07 KB | 0644 |
|
masonry.min.js | File | 23.57 KB | 0644 |
|
mce-view.js | File | 25.24 KB | 0644 |
|
mce-view.min.js | File | 9.54 KB | 0644 |
|
media-audiovideo.js | File | 24.39 KB | 0644 |
|
media-audiovideo.min.js | File | 11.78 KB | 0644 |
|
media-editor.js | File | 28.44 KB | 0644 |
|
media-editor.min.js | File | 10.63 KB | 0644 |
|
media-grid.js | File | 26.18 KB | 0644 |
|
media-grid.min.js | File | 12.95 KB | 0644 |
|
media-models.js | File | 42.74 KB | 0644 |
|
media-models.min.js | File | 12.98 KB | 0644 |
|
media-views.js | File | 265.14 KB | 0644 |
|
media-views.min.js | File | 107.42 KB | 0644 |
|
quicktags.js | File | 22.07 KB | 0644 |
|
quicktags.min.js | File | 10.87 KB | 0644 |
|
shortcode.js | File | 10.51 KB | 0644 |
|
shortcode.min.js | File | 2.58 KB | 0644 |
|
swfobject.js | File | 9.99 KB | 0644 |
|
tw-sack.js | File | 4.85 KB | 0644 |
|
tw-sack.min.js | File | 3.21 KB | 0644 |
|
twemoji.js | File | 32.16 KB | 0644 |
|
twemoji.min.js | File | 15.42 KB | 0644 |
|
underscore.js | File | 66.77 KB | 0644 |
|
underscore.min.js | File | 18.39 KB | 0644 |
|
utils.js | File | 4.56 KB | 0644 |
|
utils.min.js | File | 1.82 KB | 0644 |
|
wp-ajax-response.js | File | 3.75 KB | 0644 |
|
wp-ajax-response.min.js | File | 2.45 KB | 0644 |
|
wp-api.js | File | 45.88 KB | 0644 |
|
wp-api.min.js | File | 14.34 KB | 0644 |
|
wp-auth-check.js | File | 4.11 KB | 0644 |
|
wp-auth-check.min.js | File | 1.62 KB | 0644 |
|
wp-backbone.js | File | 14.88 KB | 0644 |
|
wp-backbone.min.js | File | 2.97 KB | 0644 |
|
wp-custom-header.js | File | 10.22 KB | 0644 |
|
wp-custom-header.min.js | File | 4.34 KB | 0644 |
|
wp-embed-template.js | File | 6.62 KB | 0644 |
|
wp-embed-template.min.js | File | 3.1 KB | 0644 |
|
wp-embed.js | File | 3.14 KB | 0644 |
|
wp-embed.min.js | File | 1.22 KB | 0644 |
|
wp-emoji-loader.js | File | 11.92 KB | 0644 |
|
wp-emoji-loader.min.js | File | 2.86 KB | 0644 |
|
wp-emoji-release.min.js | File | 18.29 KB | 0644 |
|
wp-emoji.js | File | 8.76 KB | 0644 |
|
wp-emoji.min.js | File | 2.82 KB | 0644 |
|
wp-list-revisions.js | File | 970 B | 0644 |
|
wp-list-revisions.min.js | File | 597 B | 0644 |
|
wp-lists.js | File | 24.72 KB | 0644 |
|
wp-lists.min.js | File | 7.34 KB | 0644 |
|
wp-pointer.js | File | 9.99 KB | 0644 |
|
wp-pointer.min.js | File | 3.54 KB | 0644 |
|
wp-sanitize.js | File | 1.32 KB | 0644 |
|
wp-sanitize.min.js | File | 458 B | 0644 |
|
wp-util.js | File | 4.57 KB | 0644 |
|
wp-util.min.js | File | 1.39 KB | 0644 |
|
wpdialog.js | File | 569 B | 0644 |
|
wpdialog.min.js | File | 281 B | 0644 |
|
wplink.js | File | 20.76 KB | 0644 |
|
wplink.min.js | File | 11.07 KB | 0644 |
|
zxcvbn-async.js | File | 821 B | 0644 |
|
zxcvbn-async.min.js | File | 351 B | 0644 |
|
zxcvbn.min.js | File | 802.97 KB | 0644 |
|