I have a problem with drop-down buttons inside tables when are responsive and scroll active because the drop-down is not visible because of overflow: auto;
property. How can I fix that in order to show drop-down option of button when this is collapsed? I can use some jQuery but after I have problems with scroll left-right so I decided to find another solution.
I have attached a photo to understand better.
This question is related to
javascript
jquery
css
twitter-bootstrap
overflow
I'd took a different approach, I had detached the element from the parent and set it with position absolute by jQuery
Working JS fidle: http://jsfiddle.net/s270Lyrd/
The JS solution I am using.
//fix menu overflow under the responsive table
// hide menu on click... (This is a must because when we open a menu )
$(document).click(function (event) {
//hide all our dropdowns
$('.dropdown-menu[data-parent]').hide();
});
$(document).on('click', '.table-responsive [data-toggle="dropdown"]', function () {
// if the button is inside a modal
if ($('body').hasClass('modal-open')) {
throw new Error("This solution is not working inside a responsive table inside a modal, you need to find out a way to calculate the modal Z-index and add it to the element")
return true;
}
$buttonGroup = $(this).parent();
if (!$buttonGroup.attr('data-attachedUl')) {
var ts = +new Date;
$ul = $(this).siblings('ul');
$ul.attr('data-parent', ts);
$buttonGroup.attr('data-attachedUl', ts);
$(window).resize(function () {
$ul.css('display', 'none').data('top');
});
} else {
$ul = $('[data-parent=' + $buttonGroup.attr('data-attachedUl') + ']');
}
if (!$buttonGroup.hasClass('open')) {
$ul.css('display', 'none');
return;
}
dropDownFixPosition($(this).parent(), $ul);
function dropDownFixPosition(button, dropdown) {
var dropDownTop = button.offset().top + button.outerHeight();
dropdown.css('top', dropDownTop + "px");
dropdown.css('left', button.offset().left + "px");
dropdown.css('position', "absolute");
dropdown.css('width', dropdown.width());
dropdown.css('heigt', dropdown.height());
dropdown.css('display', 'block');
dropdown.appendTo('body');
}
});
For reference, it's 2018 and I'm using BS4.1
Try adding data-boundary="viewport"
to the button that toggles the dropdown (the one with the class dropdown-toggle
). See https://getbootstrap.com/docs/4.1/components/dropdowns/#options
We solved this issue here at work by applying a .dropup class to the dropdown when the dropdown is close to the bottom of a table.enter image description here
This worked for me in Bootstrap 4 since it has different breakpoints than v3:
@media (min-width: 992px) {
.table-responsive {
overflow: inherit;
}
}
Simply Use This
.table-responsive {
overflow: inherit;
}
It works on Chrome, but not IE10 or Edge because inherit property is not supported
Well, reading the top answer, i saw that it really dont works when you are seeing the scroll bar and the toggle button was on last column (in my case) or other column that is unseen
But, if you change 'inherit' for 'hidden' it will work.
$('.table-responsive').on('show.bs.dropdown', function () {
$('.table-responsive').css( "overflow", "hidden" );
}).on('hide.bs.dropdown', function () {
$('.table-responsive').css( "overflow", "auto" );
})
Try to do that way.
This could be useful for someone else. I'm using DatatablesJS. I add 500px to the current height of the table. I do this because Datatables allow you to use 10, 20, etc pages in your table. So I need to calculate dinamically the table's height.
When dropdown is shown, I add extra height.
When dropdown is hiden, I reset original table's height.
$(document).ready(function() {
$('.table-responsive .dropdown').on('shown.bs.dropdown', function () {
console.log($('#table-responsive-cliente').height() + 500)
$("#table-responsive-cliente").css("height",$('#table-responsive-cliente').height() + 500 );
})
$('.table-responsive .dropdown').on('hide.bs.dropdown', function () {
$("#table-responsive-cliente").css("height","auto");
})
})
And the HTML
<div class="table-responsive" id="table-responsive-cliente">
<table class="table-striped table-hover">
....
....
</table>
</div>
my 2ยข quick global fix:
// drop down in responsive table
(function () {
$('.table-responsive').on('shown.bs.dropdown', function (e) {
var $table = $(this),
$menu = $(e.target).find('.dropdown-menu'),
tableOffsetHeight = $table.offset().top + $table.height(),
menuOffsetHeight = $menu.offset().top + $menu.outerHeight(true);
if (menuOffsetHeight > tableOffsetHeight)
$table.css("padding-bottom", menuOffsetHeight - tableOffsetHeight);
});
$('.table-responsive').on('hide.bs.dropdown', function () {
$(this).css("padding-bottom", 0);
})
})();
Explications: When a dropdown-menu inside a '.table-responsive' is shown, it calculate the height of the table and expand it (with padding) to match the height required to display the menu. The menu can be any size.
In my case, this is not the table that has the '.table-responsive' class, it's a wrapping div:
<div class="table-responsive" style="overflow:auto;">
<table class="table table-hover table-bordered table-condensed server-sort">
So the $table var in the script is actually a div! (just to be clear... or not) :)
Note: I wrap it in a function so my IDE can collapse function ;) but it's not mandatory!
Based on the accepted answer and the answer of @LeoCaseiro here is what I ended up using in my case :
@media (max-width: 767px) {
.table-responsive{
overflow-x: auto;
overflow-y: auto;
}
}
@media (min-width: 767px) {
.table-responsive{
overflow: inherit !important; /* Sometimes needs !important */
}
}
on big screens the dropdown won't be hidden behind the reponsive-table and in small screen it will be hidden but it's ok because there is scrolls bar in mobile anyway.
Hope this help someone.
I have a solution using only CSS, just use position relative for dropdowns inside the table-responsive:
@media (max-width: 767px) {
.table-responsive .dropdown-menu {
position: relative; /* Sometimes needs !important */
}
}
Define this properties. Good Luck!
data-toggle="dropdown" data-boundary="window"
Try it once. after 1 hour of research on net I found Best Solution for this Problem.
Solution:- just add script
(function () {
// hold onto the drop down menu
var dropdownMenu;
// and when you show it, move it to the body
$(window).on('show.bs.dropdown', function (e) {
// grab the menu
dropdownMenu = $(e.target).find('.dropdown-menu');
// detach it and append it to the body
$('body').append(dropdownMenu.detach());
// grab the new offset position
var eOffset = $(e.target).offset();
// make sure to place it where it would normally go (this could be improved)
dropdownMenu.css({
'display': 'block',
'top': eOffset.top + $(e.target).outerHeight(),
'left': eOffset.left
});
});
// and when you hide it, reattach the drop down, and hide it normally
$(window).on('hide.bs.dropdown', function (e) {
$(e.target).append(dropdownMenu.detach());
dropdownMenu.hide();
});
})();
This solution worked great for me :
@media (max-width: 767px) {
.table-responsive .dropdown-menu {
position: static !important;
}
}
@media (min-width: 768px) {
.table-responsive {
overflow: visible;
}
}
More detail: https://github.com/twbs/bootstrap/issues/15374
This has been fixed in Bootstrap v4.1 and above by adding data-boundary="viewport"
(Bootstrap Dropdowns Docs)
But for earlier versions (v4.0 and below), I found this javascript snippet that works perfectly. It works for small tables and scrolling tables:
$('.table-responsive').on('shown.bs.dropdown', function (e) {
var t = $(this),
m = $(e.target).find('.dropdown-menu'),
tb = t.offset().top + t.height(),
mb = m.offset().top + m.outerHeight(true),
d = 20; // Space for shadow + scrollbar.
if (t[0].scrollWidth > t.innerWidth()) {
if (mb + d > tb) {
t.css('padding-bottom', ((mb + d) - tb));
}
}
else {
t.css('overflow', 'visible');
}
}).on('hidden.bs.dropdown', function () {
$(this).css({'padding-bottom': '', 'overflow': ''});
});
Inside bootstrap.css search the next code:
.fixed-table-body {
overflow-x: auto;
overflow-y: auto;
height: 100%;
}
...and update with this:
.fixed-table-body {
overflow-x: visible;
overflow-y: visible;
height: 100%;
}
The recommended and chosen solution, is not always the best solution. Unfortunately its the solution linkedin recently used and it creates multiple scrollbars on the page based on the situation.
My method was slightly different.
I contained the table-responsive div in another div. Then I applied height 100%, width:100%, display block and position absolute so the height and width is based on the page size, and set overflow to hidden.
Then on the table responsive div I added a min-height of 100%
<div class="table_container"
style="height: 100%; width: 100%; display: block;position: absolute;overflow: hidden;">
<div class="table-responsive" style="min-height:100%;">
As you can see in the working example below, no added scroll bars, no funny behavior, and practically as its using percentages - it should work regardless of screen size. I have not testing this for that however. If that fails for some reason, one can replace 100% with 100vh and 100vw respectively.
<!-- Latest compiled and minified CSS -->_x000D_
<link rel="stylesheet" href="//maxcdn.bootstrapcdn.com/bootstrap/3.3.5/css/bootstrap.min.css">_x000D_
_x000D_
<!-- Optional theme -->_x000D_
<link rel="stylesheet" href="//maxcdn.bootstrapcdn.com/bootstrap/3.3.5/css/bootstrap-theme.min.css">_x000D_
_x000D_
<script src="https://code.jquery.com/jquery-1.12.4.min.js"></script>_x000D_
<script src="https://maxcdn.bootstrapcdn.com/bootstrap/3.3.5/js/bootstrap.min.js"></script>_x000D_
_x000D_
_x000D_
<div class="table_container" style="height: 100%; width: 100%; display: block;position: absolute;overflow: hidden;">_x000D_
<div class="table-responsive" style="min-height:100%;">_x000D_
<table class="table">_x000D_
<thead>_x000D_
<tr>_x000D_
<th>Value1</th>_x000D_
<th>Value2</th>_x000D_
<th>Value3</th>_x000D_
<th>Value4</th>_x000D_
</tr>_x000D_
</thead>_x000D_
<tbody>_x000D_
<tr>_x000D_
<td>_x000D_
DATA_x000D_
<div class="btn-group btn-group-rounded">_x000D_
<button type="button" class="btn btn-default btn-xs" data-toggle="dropdown" aria-haspopup="true" aria-expanded="false" style="border-radius:3px;">_x000D_
<span class="caret"></span>_x000D_
</button>_x000D_
<ul class="dropdown-menu">_x000D_
<li><a href="#">One</a></li>_x000D_
<li><a href="#">Two</a></li>_x000D_
<li><a href="#">Three</a></li>_x000D_
<li role="seperator" class="divider"></li>_x000D_
<li><a href="#">Four</a></li>_x000D_
</ul>_x000D_
</div>_x000D_
</td>_x000D_
_x000D_
<td>_x000D_
DATA_x000D_
<div class="btn-group btn-group-rounded">_x000D_
<button type="button" class="btn btn-default btn-xs" data-toggle="dropdown" aria-haspopup="true" aria-expanded="false" style="border-radius:3px;">_x000D_
<span class="caret"></span>_x000D_
</button>_x000D_
<ul class="dropdown-menu">_x000D_
<li><a href="#">One</a></li>_x000D_
<li><a href="#">Two</a></li>_x000D_
<li><a href="#">Three</a></li>_x000D_
<li role="seperator" class="divider"></li>_x000D_
<li><a href="#">Four</a></li>_x000D_
</ul>_x000D_
</div>_x000D_
</td>_x000D_
<td>_x000D_
DATA_x000D_
<div class="btn-group btn-group-rounded">_x000D_
<button type="button" class="btn btn-default btn-xs" data-toggle="dropdown" aria-haspopup="true" aria-expanded="false" style="border-radius:3px;">_x000D_
<span class="caret"></span>_x000D_
</button>_x000D_
<ul class="dropdown-menu">_x000D_
<li><a href="#">One</a></li>_x000D_
<li><a href="#">Two</a></li>_x000D_
<li><a href="#">Three</a></li>_x000D_
<li role="seperator" class="divider"></li>_x000D_
<li><a href="#">Four</a></li>_x000D_
</ul>_x000D_
</div>_x000D_
</td>_x000D_
<td>DATA</td>_x000D_
</tr>_x000D_
<tr>_x000D_
<td>_x000D_
DATA_x000D_
<div class="btn-group btn-group-rounded">_x000D_
<button type="button" class="btn btn-default btn-xs" data-toggle="dropdown" aria-haspopup="true" aria-expanded="false" style="border-radius:3px;">_x000D_
<span class="caret"></span>_x000D_
</button>_x000D_
<ul class="dropdown-menu">_x000D_
<li><a href="#">One</a></li>_x000D_
<li><a href="#">Two</a></li>_x000D_
<li><a href="#">Three</a></li>_x000D_
<li role="seperator" class="divider"></li>_x000D_
<li><a href="#">Four</a></li> </ul>_x000D_
</div>_x000D_
</td>_x000D_
_x000D_
<td>_x000D_
DATA_x000D_
<div class="btn-group btn-group-rounded">_x000D_
<button type="button" class="btn btn-default btn-xs" data-toggle="dropdown" aria-haspopup="true" aria-expanded="false" style="border-radius:3px;">_x000D_
<span class="caret"></span>_x000D_
</button>_x000D_
<ul class="dropdown-menu">_x000D_
<li><a href="#">One</a></li>_x000D_
<li><a href="#">Two</a></li>_x000D_
<li><a href="#">Three</a></li>_x000D_
<li role="seperator" class="divider"></li>_x000D_
<li><a href="#">Four</a></li>_x000D_
</ul>_x000D_
</div>_x000D_
</td>_x000D_
<td>_x000D_
DATA_x000D_
<div class="btn-group btn-group-rounded">_x000D_
<button type="button" class="btn btn-default btn-xs" data-toggle="dropdown" aria-haspopup="true" aria-expanded="false" style="border-radius:3px;">_x000D_
<span class="caret"></span>_x000D_
</button>_x000D_
<ul class="dropdown-menu">_x000D_
<li><a href="#">One</a></li>_x000D_
<li><a href="#">Two</a></li>_x000D_
<li><a href="#">Three</a></li>_x000D_
<li role="seperator" class="divider"></li>_x000D_
<li><a href="#">Four</a></li>_x000D_
</ul>_x000D_
</div>_x000D_
</td>_x000D_
<td>DATA</td>_x000D_
</tr>_x000D_
</tbody>_x000D_
</table>_x000D_
</div>_x000D_
</div>
_x000D_
A CSS only solution is to allow the y-axis to overflow.
http://www.bootply.com/YvePJTDzI0
.table-responsive {
overflow-y: visible !important;
}
EDIT
Another CSS only solution is to responsively apply the overflow based on viewport width:
@media (max-width: 767px) {
.table-responsive .dropdown-menu {
position: static !important;
}
}
@media (min-width: 768px) {
.table-responsive {
overflow: inherit;
}
}
Burebistaruler response works ok for me on ios8 (iphone4s) but doen't woks on android that before was working. What i've donne that Works for me on ios8 (iphone4s) and andoir is:
$('.table-responsive').on('show.bs.dropdown', function () {
$('.table-responsive').css( "min-height", "400px" );
});
$('.table-responsive').on('hide.bs.dropdown', function () {
$('.table-responsive').css( "min-height", "none" );
})
In my case, this works fine:
.table-responsive {
overflow-y: visible !important;
}
As long as people still stuck in this issue and we are in 2020 already. I get a pure CSS solution by giving the drop down menu a flex display
this snippet works great with datatable-scroll-wrap
class
.datatable-scroll-wrap .dropdown.dropup.open .dropdown-menu {
display: flex;
}
.datatable-scroll-wrap .dropdown.dropup.open .dropdown-menu li a {
display: flex;
}
Cleaned up @Wazime solution a little. Works great as a general solution.
$(document).on('shown.bs.dropdown', '.table-responsive', function (e) {
// The .dropdown container
var $container = $(e.target);
// Find the actual .dropdown-menu
var $dropdown = $container.find('.dropdown-menu');
if ($dropdown.length) {
// Save a reference to it, so we can find it after we've attached it to the body
$container.data('dropdown-menu', $dropdown);
} else {
$dropdown = $container.data('dropdown-menu');
}
$dropdown.css('top', ($container.offset().top + $container.outerHeight()) + 'px');
$dropdown.css('left', $container.offset().left + 'px');
$dropdown.css('position', 'absolute');
$dropdown.css('display', 'block');
$dropdown.appendTo('body');
});
$(document).on('hide.bs.dropdown', '.table-responsive', function (e) {
// Hide the dropdown menu bound to this button
$(e.target).data('dropdown-menu').css('display', 'none');
});
Source: Stackoverflow.com