vector.js 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114
  1. /**
  2. * Collapsible tabs for Vector
  3. */
  4. /* eslint-disable no-jquery/no-global-selector */
  5. $( function () {
  6. var $cactions = $( '#p-cactions' ),
  7. $tabContainer = $( '#p-views ul' ),
  8. initialCactionsWidth = function () {
  9. // HACK: This depends on a discouraged feature of jQuery width().
  10. // The #p-cactions element is generally hidden by default, but
  11. // the consumers of this function need to know the width that the
  12. // "More" menu would consume if it were visible. This means it
  13. // must not return 0 if hidden, but rather virtually render it
  14. // and compute its width, then hide it again. jQuery width() does
  15. // all that for us.
  16. var width = $cactions.width();
  17. initialCactionsWidth = function () {
  18. return width;
  19. };
  20. return width;
  21. };
  22. // Bind callback functions to animate our drop down menu in and out
  23. // and then call the collapsibleTabs function on the menu
  24. $tabContainer
  25. .on( 'beforeTabCollapse', function () {
  26. var expandedWidth;
  27. // If the dropdown was hidden, show it
  28. if ( $cactions.hasClass( 'emptyPortlet' ) ) {
  29. $cactions.removeClass( 'emptyPortlet' );
  30. // Now that it is visible, force-render it virtually
  31. // to get its expanded width, then shrink it 1px before we
  32. // yield from JS (which means the expansion won't be visible).
  33. // Then animate from the 1px to the expanded width.
  34. expandedWidth = $cactions.width();
  35. // eslint-disable-next-line no-jquery/no-animate
  36. $cactions.find( 'h3' )
  37. .css( 'width', '1px' )
  38. .animate( { width: expandedWidth }, 'normal' );
  39. }
  40. } )
  41. .on( 'beforeTabExpand', function () {
  42. // If we're removing the last child node right now, hide the dropdown
  43. if ( $cactions.find( 'li' ).length === 1 ) {
  44. // eslint-disable-next-line no-jquery/no-animate
  45. $cactions.find( 'h3' ).animate( { width: '1px' }, 'normal', function () {
  46. $( this ).attr( 'style', '' )
  47. .parent().addClass( 'emptyPortlet' );
  48. } );
  49. }
  50. } )
  51. .collapsibleTabs( {
  52. expandCondition: function ( eleWidth ) {
  53. // This looks a bit awkward because we're doing expensive queries as late
  54. // as possible.
  55. var distance = $.collapsibleTabs.calculateTabDistance();
  56. // If there are at least eleWidth + 1 pixels of free space, expand.
  57. // We add 1 because .width() will truncate fractional values but .offset() will not.
  58. if ( distance >= eleWidth + 1 ) {
  59. return true;
  60. } else {
  61. // Maybe we can still expand? Account for the width of the "Actions" dropdown
  62. // if the expansion would hide it.
  63. if ( $cactions.find( 'li' ).length === 1 ) {
  64. return distance >= eleWidth + 1 - initialCactionsWidth();
  65. } else {
  66. return false;
  67. }
  68. }
  69. },
  70. collapseCondition: function () {
  71. var collapsibleWidth = 0,
  72. doCollapse = false;
  73. // This looks a bit awkward because we're doing expensive queries as late
  74. // as possible.
  75. // TODO: The dropdown itself should probably "fold" to just the down-arrow
  76. // (hiding the text) if it can't fit on the line?
  77. // Never collapse if there is no overlap.
  78. if ( $.collapsibleTabs.calculateTabDistance() >= 0 ) {
  79. return false;
  80. }
  81. // Always collapse if the "More" button is already shown.
  82. if ( !$cactions.hasClass( 'emptyPortlet' ) ) {
  83. return true;
  84. }
  85. // If we reach here, this means:
  86. // 1. #p-cactions is currently empty and invisible (e.g. when logged out),
  87. // 2. and, there is at least one li.collapsible link in #p-views, as asserted
  88. // by handleResize() before calling here. Such link exists e.g. as
  89. // "View history" on articles, but generally not on special pages.
  90. // 3. and, the left-navigation and right-navigation are overlapping
  91. // each other, e.g. when making the window very narrow, or if a gadget
  92. // added a lot of tabs.
  93. $tabContainer.children( 'li.collapsible' ).each( function ( index, element ) {
  94. collapsibleWidth += $( element ).width();
  95. if ( collapsibleWidth > initialCactionsWidth() ) {
  96. // We've found one or more collapsible links that are wider
  97. // than the "More" menu would be if it were made visible,
  98. // which means it is worth doing a collapsing.
  99. doCollapse = true;
  100. // Stop this possibly expensive loop the moment the condition is met once.
  101. return false;
  102. }
  103. } );
  104. return doCollapse;
  105. }
  106. } );
  107. } );