Ticket #33567: 33052.5.diff
File 33052.5.diff, 17.2 KB (added by , 9 years ago) |
---|
-
src/wp-admin/css/customize-controls.css
diff --git src/wp-admin/css/customize-controls.css src/wp-admin/css/customize-controls.css index b35bc3b..6b9d910 100644
body { 123 123 color: #0073aa; 124 124 } 125 125 126 #customize-controls .customize-info .customize-panel-description { 126 #customize-controls .customize-info .customize-panel-description, 127 #customize-controls .no-widget-areas-rendered-notice { 127 128 color: #555; 128 129 display: none; 129 130 background: #fff; 130 131 padding: 12px 15px; 131 132 border-top: 1px solid #ddd; 132 133 } 134 #customize-controls .customize-info .customize-panel-description.open + .no-widget-areas-rendered-notice { 135 border-top: none; 136 } 133 137 134 138 #customize-controls .customize-info .customize-panel-description p:first-child { 135 139 margin-top: 0; -
src/wp-admin/js/customize-controls.js
diff --git src/wp-admin/js/customize-controls.js src/wp-admin/js/customize-controls.js index e0e4a4d..b1e7ba6 100644
300 300 * @param {Object} args.completeCallback 301 301 */ 302 302 onChangeActive: function( active, args ) { 303 var duration, construct = this ;303 var duration, construct = this, expandedOtherPanel; 304 304 if ( args.unchanged ) { 305 305 if ( args.completeCallback ) { 306 306 args.completeCallback(); … … 309 309 } 310 310 311 311 duration = ( 'resolved' === api.previewer.deferred.active.state() ? args.duration : 0 ); 312 313 if ( construct.extended( api.Panel ) ) { 314 // If this is a panel is not currently expanded but another panel is expanded, do not animate. 315 api.panel.each(function ( panel ) { 316 if ( panel !== construct && panel.expanded() ) { 317 expandedOtherPanel = panel; 318 duration = 0; 319 } 320 }); 321 322 // Collapse any expanded sections inside of this panel first before deactivating. 323 if ( ! active ) { 324 _.each( construct.sections(), function( section ) { 325 section.collapse( { duration: 0 } ); 326 } ); 327 } 328 } 329 312 330 if ( ! $.contains( document, construct.container[0] ) ) { 313 331 // jQuery.fn.slideUp is not hiding an element if it is not in the DOM 314 332 construct.container.toggle( active ); … … 329 347 construct.container.stop( true, true ).slideUp( duration, args.completeCallback ); 330 348 } 331 349 } 350 351 // Recalculate the margin-top immediately, not waiting for debounced reflow, to prevent momentary (100ms) vertical jiggle. 352 if ( expandedOtherPanel ) { 353 expandedOtherPanel._recalculateTopMargin(); 354 } 332 355 }, 333 356 334 357 /** … … 378 401 }, 379 402 380 403 /** 381 * @param {Boolean} expanded 382 * @param {Object} [params] 383 * @returns {Boolean} false if state already applied 404 * Handle the toggle logic for expand/collapse. 405 * 406 * @param {Boolean} expanded - The new state to apply. 407 * @param {Object} [params] - Object containing options for expand/collapse. 408 * @param {Function} [params.completeCallback] - Function to call when expansion/collapse is complete. 409 * @returns {Boolean} false if state already applied or active state is false 384 410 */ 385 _toggleExpanded: function 386 var self = this;411 _toggleExpanded: function( expanded, params ) { 412 var instance = this, previousCompleteCallback; 387 413 params = params || {}; 388 var section = this, previousCompleteCallback = params.completeCallback; 389 params.completeCallback = function () { 414 previousCompleteCallback = params.completeCallback; 415 416 // Short-circuit expand() if the instance is not active. 417 if ( expanded && ! instance.active() ) { 418 return false; 419 } 420 421 params.completeCallback = function() { 390 422 if ( previousCompleteCallback ) { 391 previousCompleteCallback.apply( section, arguments );423 previousCompleteCallback.apply( instance, arguments ); 392 424 } 393 425 if ( expanded ) { 394 section.container.trigger( 'expanded' );426 instance.container.trigger( 'expanded' ); 395 427 } else { 396 section.container.trigger( 'collapsed' );428 instance.container.trigger( 'collapsed' ); 397 429 } 398 430 }; 399 if ( ( expanded && this.expanded.get() ) || ( ! expanded && ! this.expanded.get() ) ) {431 if ( ( expanded && instance.expanded.get() ) || ( ! expanded && ! instance.expanded.get() ) ) { 400 432 params.unchanged = true; 401 self.onChangeExpanded( self.expanded.get(), params );433 instance.onChangeExpanded( instance.expanded.get(), params ); 402 434 return false; 403 435 } else { 404 436 params.unchanged = false; 405 this.expandedArgumentsQueue.push( params );406 this.expanded.set( expanded );437 instance.expandedArgumentsQueue.push( params ); 438 instance.expanded.set( expanded ); 407 439 return true; 408 440 } 409 441 }, 410 442 411 443 /** 412 444 * @param {Object} [params] 413 * @returns {Boolean} false if already expanded 445 * @returns {Boolean} false if already expanded or if inactive. 414 446 */ 415 447 expand: function ( params ) { 416 448 return this._toggleExpanded( true, params ); … … 418 450 419 451 /** 420 452 * @param {Object} [params] 421 * @returns {Boolean} false if already collapsed 453 * @returns {Boolean} false if already collapsed. 422 454 */ 423 455 collapse: function ( params ) { 424 456 return this._toggleExpanded( false, params ); … … 539 571 }; 540 572 section.panel.bind( inject ); 541 573 inject( section.panel.get() ); // Since a section may never get a panel, assume that it won't ever get one 574 575 section.deferred.embedded.done(function() { 576 // Fix the top margin after reflow. 577 api.bind( 'pane-contents-reflowed', _.debounce( function() { 578 section._recalculateTopMargin(); 579 }, 100 ) ); 580 }); 542 581 }, 543 582 544 583 /** … … 646 685 // Fix the height after browser resize. 647 686 $( window ).on( 'resize.customizer-section', _.debounce( resizeContentHeight, 100 ) ); 648 687 649 // Fix the top margin after reflow. 650 api.bind( 'pane-contents-reflowed', _.debounce( function() { 651 var offset = ( content.offset().top - headerActionsHeight ); 652 if ( 0 < offset ) { 653 content.css( 'margin-top', ( parseInt( content.css( 'margin-top' ), 10 ) - offset ) ); 654 } 655 }, 100 ) ); 688 section._recalculateTopMargin(); 656 689 }; 657 690 } 658 691 … … 693 726 args.completeCallback(); 694 727 } 695 728 } 729 }, 730 731 /** 732 * Recalculate the top margin. 733 * 734 * @since 4.4.0 735 * @private 736 */ 737 _recalculateTopMargin: function() { 738 var section = this, content, offset, headerActionsHeight; 739 content = section.container.find( '.accordion-section-content' ); 740 if ( 0 === content.length ) { 741 return; 742 } 743 headerActionsHeight = $( '#customize-header-actions' ).height(); 744 offset = ( content.offset().top - headerActionsHeight ); 745 if ( 0 < offset ) { 746 content.css( 'margin-top', ( parseInt( content.css( 'margin-top' ), 10 ) - offset ) ); 747 } 696 748 } 697 749 }); 698 750 … … 1155 1207 parentContainer.append( panel.container ); 1156 1208 panel.renderContent(); 1157 1209 } 1210 1211 api.bind( 'pane-contents-reflowed', _.debounce( function() { 1212 panel._recalculateTopMargin(); 1213 }, 100 ) ); 1214 1158 1215 panel.deferred.embedded.resolve(); 1159 1216 }, 1160 1217 … … 1253 1310 * @param {Boolean} expanded 1254 1311 * @param {Object} args 1255 1312 * @param {Boolean} args.unchanged 1256 * @param { Callback} args.completeCallback1313 * @param {Function} args.completeCallback 1257 1314 */ 1258 1315 onChangeExpanded: function ( expanded, args ) { 1259 1316 … … 1268 1325 // Note: there is a second argument 'args' passed 1269 1326 var position, scroll, 1270 1327 panel = this, 1271 section = panel.container.closest( '.accordion-section' ), // This is actually the panel.1272 overlay = section.closest( '.wp-full-overlay' ),1273 container = section.closest( '.wp-full-overlay-sidebar-content' ),1328 accordionSection = panel.container.closest( '.accordion-section' ), 1329 overlay = accordionSection.closest( '.wp-full-overlay' ), 1330 container = accordionSection.closest( '.wp-full-overlay-sidebar-content' ), 1274 1331 siblings = container.find( '.open' ), 1275 1332 topPanel = overlay.find( '#customize-theme-controls > ul > .accordion-section > .accordion-section-title' ), 1276 backBtn = section.find( '.customize-panel-back' ),1277 panelTitle = section.find( '.accordion-section-title' ).first(),1278 content = section.find( '.control-panel-content' ),1333 backBtn = accordionSection.find( '.customize-panel-back' ), 1334 panelTitle = accordionSection.find( '.accordion-section-title' ).first(), 1335 content = accordionSection.find( '.control-panel-content' ), 1279 1336 headerActionsHeight = $( '#customize-header-actions' ).height(); 1280 1337 1281 1338 if ( expanded ) { … … 1297 1354 position = content.offset().top; 1298 1355 scroll = container.scrollTop(); 1299 1356 content.css( 'margin-top', ( headerActionsHeight - position - scroll ) ); 1300 section.addClass( 'current-panel' );1357 accordionSection.addClass( 'current-panel' ); 1301 1358 overlay.addClass( 'in-sub-panel' ); 1302 1359 container.scrollTop( 0 ); 1303 1360 if ( args.completeCallback ) { … … 1307 1364 topPanel.attr( 'tabindex', '-1' ); 1308 1365 backBtn.attr( 'tabindex', '0' ); 1309 1366 backBtn.focus(); 1310 1311 // Fix the top margin after reflow. 1312 api.bind( 'pane-contents-reflowed', _.debounce( function() { 1313 content.css( 'margin-top', ( parseInt( content.css( 'margin-top' ), 10 ) - ( content.offset().top - headerActionsHeight ) ) ); 1314 }, 100 ) ); 1367 panel._recalculateTopMargin(); 1315 1368 } else { 1316 1369 siblings.removeClass( 'open' ); 1317 section.removeClass( 'current-panel' );1370 accordionSection.removeClass( 'current-panel' ); 1318 1371 overlay.removeClass( 'in-sub-panel' ); 1319 1372 content.delay( 180 ).hide( 0, function() { 1320 1373 content.css( 'margin-top', 'inherit' ); // Reset … … 1330 1383 }, 1331 1384 1332 1385 /** 1386 * Recalculate the top margin. 1387 * 1388 * @since 4.4.0 1389 * @private 1390 */ 1391 _recalculateTopMargin: function() { 1392 var panel = this, headerActionsHeight, content, accordionSection; 1393 headerActionsHeight = $( '#customize-header-actions' ).height(); 1394 accordionSection = panel.container.closest( '.accordion-section' ); 1395 content = accordionSection.find( '.control-panel-content' ); 1396 content.css( 'margin-top', ( parseInt( content.css( 'margin-top' ), 10 ) - ( content.offset().top - headerActionsHeight ) ) ); 1397 }, 1398 1399 /** 1333 1400 * Render the panel from its JS template, if it exists. 1334 1401 * 1335 1402 * The panel's container must already exist in the DOM. -
src/wp-admin/js/customize-widgets.js
diff --git src/wp-admin/js/customize-widgets.js src/wp-admin/js/customize-widgets.js index bd25757..144365e 100644
1506 1506 } ); 1507 1507 1508 1508 /** 1509 * wp.customize.Widgets.WidgetsPanel 1510 * 1511 * Customizer panel containing the widget area sections. 1512 * 1513 * @since 4.4.0 1514 */ 1515 api.Widgets.WidgetsPanel = api.Panel.extend({ 1516 1517 /** 1518 * Add and manage the display of the no-rendered-areas notice. 1519 * 1520 * @since 4.4.0 1521 */ 1522 ready: function () { 1523 var panel = this; 1524 1525 api.Panel.prototype.ready.call( panel ); 1526 1527 panel.deferred.embedded.done(function() { 1528 var panelMetaContainer, noRenderedAreasNotice, shouldShowNotice; 1529 panelMetaContainer = panel.container.find( '.panel-meta' ); 1530 noRenderedAreasNotice = $( '<div></div>', { 1531 'class': 'no-widget-areas-rendered-notice' 1532 }); 1533 noRenderedAreasNotice.append( $( '<em></em>', { 1534 text: l10n.noAreasRendered 1535 } ) ); 1536 panelMetaContainer.append( noRenderedAreasNotice ); 1537 1538 shouldShowNotice = function() { 1539 return ( 0 === _.filter( panel.sections(), function( section ) { 1540 return section.active(); 1541 } ).length ); 1542 }; 1543 1544 /* 1545 * Defer setting visibility of no-rendered-areas-notice until 1546 * preview finishes loading since this is when the active 1547 * sections become available. 1548 */ 1549 api.previewer.deferred.active.done( function () { 1550 noRenderedAreasNotice.toggle( shouldShowNotice() ); 1551 1552 // Update the visibility of the notice whenever a reflow happens. 1553 api.bind( 'pane-contents-reflowed', function() { 1554 api.previewer.deferred.active.done( function () { 1555 if ( shouldShowNotice() ) { 1556 noRenderedAreasNotice.slideDown( 'fast' ); 1557 } else { 1558 noRenderedAreasNotice.slideUp( 'fast' ); 1559 } 1560 }); 1561 }); 1562 }); 1563 }); 1564 }, 1565 1566 /** 1567 * Allow an active widgets panel to be contextually active even when it has no active sections (widget areas). 1568 * 1569 * This ensures that the widgets panel appears even when there are no 1570 * sidebars displayed on the URL currently being previewed. 1571 * 1572 * @since 4.4.0 1573 * 1574 * @returns {boolean} 1575 */ 1576 isContextuallyActive: function() { 1577 var panel = this; 1578 return panel.active(); 1579 } 1580 }); 1581 1582 /** 1509 1583 * wp.customize.Widgets.SidebarSection 1510 1584 * 1511 1585 * Customizer section representing a widget area widget … … 1968 2042 } 1969 2043 } ); 1970 2044 1971 // Register models for custom section and control types 2045 // Register models for custom panel, section, and control types 2046 $.extend( api.panelConstructor, { 2047 widgets: api.Widgets.WidgetsPanel 2048 }); 1972 2049 $.extend( api.sectionConstructor, { 1973 2050 sidebar: api.Widgets.SidebarSection 1974 2051 }); -
src/wp-includes/class-wp-customize-widgets.php
diff --git src/wp-includes/class-wp-customize-widgets.php src/wp-includes/class-wp-customize-widgets.php index 7b73dc2..c1a8e5e 100644
final class WP_Customize_Widgets { 355 355 } 356 356 357 357 $this->manager->add_panel( 'widgets', array( 358 'title' => __( 'Widgets' ), 359 'description' => __( 'Widgets are independent sections of content that can be placed into widgetized areas provided by your theme (commonly called sidebars).' ), 360 'priority' => 110, 358 'type' => 'widgets', 359 'title' => __( 'Widgets' ), 360 'description' => __( 'Widgets are independent sections of content that can be placed into widgetized areas provided by your theme (commonly called sidebars).' ), 361 'priority' => 110, 362 'active_callback' => array( $this, 'is_panel_active' ), 361 363 ) ); 362 364 363 365 foreach ( $sidebars_widgets as $sidebar_id => $sidebar_widget_ids ) { … … final class WP_Customize_Widgets { 455 457 } 456 458 457 459 /** 460 * Return whether the widgets panel is active, based on whether there are sidebars registered. 461 * 462 * @since 4.4.0 463 * @access public 464 * 465 * @see WP_Customize_Panel::$active_callback 466 * 467 * @global array $wp_registered_sidebars 468 * @return bool Active. 469 */ 470 public function is_panel_active() { 471 global $wp_registered_sidebars; 472 return ! empty( $wp_registered_sidebars ); 473 } 474 475 /** 458 476 * Covert a widget_id into its corresponding Customizer setting ID (option name). 459 477 * 460 478 * @since 3.9.0 … … final class WP_Customize_Widgets { 655 673 'error' => __( 'An error has occurred. Please reload the page and try again.' ), 656 674 'widgetMovedUp' => __( 'Widget moved up' ), 657 675 'widgetMovedDown' => __( 'Widget moved down' ), 676 'noAreasRendered' => __( 'There are no widget areas currently rendered in the preview. Navigate in the preview to a template that makes use of a widget area in order to access its widgets here.' ), 658 677 ), 659 678 'tpl' => array( 660 679 'widgetReorderNav' => $widget_reorder_nav_tpl, -
tests/qunit/wp-admin/js/customize-controls.js
diff --git tests/qunit/wp-admin/js/customize-controls.js tests/qunit/wp-admin/js/customize-controls.js index 6ca4908..91a7d36 100644
jQuery( window ).load( function (){ 393 393 panelId = 'mockPanelId'; 394 394 panelTitle = 'Mock Panel Title'; 395 395 panelDescription = 'Mock panel description'; 396 panelContent = '<li id="accordion-panel- widgets" class="control-section control-panel accordion-section">';396 panelContent = '<li id="accordion-panel-mockPanelId" class="accordion-section control-section control-panel control-panel-default"> <h3 class="accordion-section-title" tabindex="0"> Fixture Panel <span class="screen-reader-text">Press return or enter to open this panel</span> </h3> <ul class="accordion-sub-container control-panel-content"> <li class="panel-meta customize-info accordion-section cannot-expand"> <button class="customize-panel-back" tabindex="-1"><span class="screen-reader-text">Back</span></button> <div class="accordion-section-title"> <span class="preview-notice">You are customizing <strong class="panel-title">Fixture Panel</strong></span> <button class="customize-help-toggle dashicons dashicons-editor-help" tabindex="0" aria-expanded="false"><span class="screen-reader-text">Help</span></button> </div> </li> </ul> </li>'; 397 397 panelData = { 398 398 content: panelContent, 399 399 title: panelTitle, -
tests/qunit/wp-admin/js/customize-widgets.js
diff --git tests/qunit/wp-admin/js/customize-widgets.js tests/qunit/wp-admin/js/customize-widgets.js index 3a8a3e3..f5c99e3 100644
jQuery( window ).load( function() { 34 34 ok( ! section.expanded() ); 35 35 ok( 0 === control.container.find( '> .widget' ).length ); 36 36 37 // Preview sets the active state. 38 section.active.set( true ); 39 control.active.set( true ); 40 api.control( 'sidebars_widgets[sidebar-1]' ).active.set( true ); 41 37 42 section.expand(); 38 ok( ! widgetAddedEvent );39 ok( 1 === control.container.find( '> .widget' ).length );43 ok( ! widgetAddedEvent, 'expected widget added event not fired' ); 44 ok( 1 === control.container.find( '> .widget' ).length, 'expected there to be one .widget element in the container' ); 40 45 ok( 0 === control.container.find( '.widget-content' ).children().length ); 41 46 42 47 control.expand();