Как я могу проверить, видна ли полоса прокрутки?

можно ли проверить overflow:auto div?

например:

HTML-код

<div id="my_div" style="width: 100px; height:100px; overflow:auto;" class="my_class"> 
  * content
</div>

в jQuery

$('.my_class').live('hover', function (event)
{
    if (event.type == 'mouseenter')
    {
         if( ...  if scrollbar visible ? ... )
         {
            alert('true'):
         }
         else
         {
            alert('false'):
         }
    }

});

иногда содержимое короткое (без полосы прокрутки), а иногда длинное (полоса прокрутки видна).

17 ответов


небольшой плагин для него.

(function($) {
    $.fn.hasScrollBar = function() {
        return this.get(0).scrollHeight > this.height();
    }
})(jQuery);

используйте его так,

$('#my_div1').hasScrollBar(); // returns true if there's a `vertical` scrollbar, false otherwise..

протестирована работа с Firefox, Chrome,IE6,7, 8

но не работает должным образом на body выбор тега

демо


редактировать

я узнал, что когда у вас есть горизонтальная полоса прокрутки, которая вызывает вертикальную полосу прокрутки, эта функция не работает....

я нашел другое решение... использовать clientHeight

return this.get(0).scrollHeight > this.get(0).clientHeight;

возможно, более простое решение.

if ($(document).height() > $(window).height()) {
    // scrollbar
}

(function($) {
    $.fn.hasScrollBar = function() {
        return this.get(0) ? this.get(0).scrollHeight > this.innerHeight() : false;
    }
})(jQuery);

innerHeight подсчитывает высоту элемента управления и его верхние и нижние прокладки


вы можете сделать это, используя комбинацию Element.scrollHeight и Element.clientHeight атрибуты.

согласно MDN:

на элемент.scrollHeight атрибут только для чтения-это измерение высоты содержимого элемента, включая содержимое, не видимое на экране из-за переполнения. значение scrollHeight равно минимальному clientHeight элемент будет требовать, чтобы соответствовать все содержимое в viewpoint без использования вертикальной полосы прокрутки. он включает элемент заполнения, но не его поля.

и:

на элемент.clientHeight свойство только для чтения возвращает внутреннюю высоту элемента в пикселях, включая заполнение, но не горизонтальную высоту полосы прокрутки, границу или поле.

clientHeight можно рассчитать как CSS height + CSS padding-высота горизонтальной полосы прокрутки (если настоящий.)

таким образом, элемент будет отображаться полосы прокрутки, если высота прокрутки больше высоты клиента, поэтому ответ на ваш вопрос:

function scrollbarVisible(element) {
  return element.scrollHeight > element.clientHeight;
}

это расширяет ответ @Reigel. Он вернет ответ для горизонтальных или вертикальных полос прокрутки.

(function($) {
    $.fn.hasScrollBar = function() {
        var e = this.get(0);
        return {
            vertical: e.scrollHeight > e.clientHeight,
            horizontal: e.scrollWidth > e.clientWidth
        };
    }
})(jQuery);

пример:

element.hasScrollBar()             // Returns { vertical: true/false, horizontal: true/false }
element.hasScrollBar().vertical    // Returns true/false
element.hasScrollBar().horizontal  // Returns true/false

вам нужно элемент.scrollHeight. Сравните это с $(element).height().


Я сделал новый пользовательский: псевдо-селектор для jQuery, чтобы проверить, имеет ли элемент одно из следующих свойств css:

  1. переполнение: [свиток|авто]
  2. переполнение-x: [прокрутка / авто]
  3. overflow-y: [прокрутка / авто]

Я хотел найти ближайшего прокручиваемого родителя другого элемента, поэтому я также написал еще один маленький плагин jQuery, чтобы найти ближайшего родителя с переполнением.

это решение, вероятно, не выполнив лучше, но, кажется, это работает. Я использовал его в сочетании с $.плагин scrollTo. Иногда мне нужно знать, находится ли элемент внутри другого прокручиваемого контейнера. В этом случае я хочу прокрутить Родительский прокручиваемый элемент против окна.

Я, вероятно, должен был завернуть это в один плагин и добавить селектор psuedo в качестве части плагина, а также предоставить "ближайший" метод для поиска ближайшего (родительского) прокручиваемого контейнера.

Anywho....здесь это.

$.isScrollable плагин jQuery:

$.fn.isScrollable = function(){
    var elem = $(this);
    return (
    elem.css('overflow') == 'scroll'
        || elem.css('overflow') == 'auto'
        || elem.css('overflow-x') == 'scroll'
        || elem.css('overflow-x') == 'auto'
        || elem.css('overflow-y') == 'scroll'
        || elem.css('overflow-y') == 'auto'
    );
};

$( ': scrollable') jQuery псевдо селектор:

$.expr[":"].scrollable = function(a) {
    var elem = $(a);
    return elem.isScrollable();
};

$.scrollableparent () плагин jQuery:

$.fn.scrollableparent = function(){
    return $(this).closest(':scrollable') || $(window); //default to $('html') instead?
};

реализация довольно проста

//does a specific element have overflow scroll?
var somedivIsScrollable = $(this).isScrollable();
//use :scrollable psuedo selector to find a collection of child scrollable elements
var scrollableChildren = $(this).find(':scrollable');
//use $.scrollableparent to find closest scrollable container
var scrollableparent = $(this).scrollableparent();

обновление: я обнаружил, что Роберт Коритник уже придумал гораздо более мощный: прокручиваемый псевдо-селектор, который идентифицирует прокручиваемые оси и высота прокручиваемых контейнеров, как часть его $.плагин jQuery scrollintoview (). плагин scrollintoview

вот его причудливый псевдо-селектор (реквизит):

    $.extend($.expr[":"], {

    scrollable: function (element, index, meta, stack) {

        var direction = converter[typeof (meta[3]) === "string" && meta[3].toLowerCase()] || converter.both;

        var styles = (document.defaultView && document.defaultView.getComputedStyle ? document.defaultView.getComputedStyle(element, null) : element.currentStyle);

        var overflow = {

            x: scrollValue[styles.overflowX.toLowerCase()] || false,

            y: scrollValue[styles.overflowY.toLowerCase()] || false,

            isRoot: rootrx.test(element.nodeName)

        };



        // check if completely unscrollable (exclude HTML element because it's special)

        if (!overflow.x && !overflow.y && !overflow.isRoot)

        {

            return false;

        }



        var size = {

            height: {

                scroll: element.scrollHeight,

                client: element.clientHeight

            },

            width: {

                scroll: element.scrollWidth,

                client: element.clientWidth

            },

            // check overflow.x/y because iPad (and possibly other tablets) don't dislay scrollbars

            scrollableX: function () {

                return (overflow.x || overflow.isRoot) && this.width.scroll > this.width.client;

            },

            scrollableY: function () {

                return (overflow.y || overflow.isRoot) && this.height.scroll > this.height.client;

            }

        };

        return direction.y && size.scrollableY() || direction.x && size.scrollableX();

    }

});

первое решение выше работает только в IE Второе решение выше работает только в FF

эта комбинация обеих функций работает в обоих браузерах:

//Firefox Only!!
if ($(document).height() > $(window).height()) {
    // has scrollbar
    $("#mtc").addClass("AdjustOverflowWidth");
    alert('scrollbar present - Firefox');
} else {
    $("#mtc").removeClass("AdjustOverflowWidth");
}

//Internet Explorer Only!!
(function($) {
    $.fn.hasScrollBar = function() {
        return this.get(0).scrollHeight > this.innerHeight();
    }
})(jQuery);
if ($('#monitorWidth1').hasScrollBar()) {
    // has scrollbar
    $("#mtc").addClass("AdjustOverflowWidth");
    alert('scrollbar present - Internet Exploder');
} else {
    $("#mtc").removeClass("AdjustOverflowWidth");
}​
  • завернуть в готовый документ
  • monitorWidth1: div, где переполнение установлено в auto
  • mtc: контейнер div внутри monitorWidth1
  • AdjustOverflowWidth: класс css, применяемый к # MTC div, когда полоса прокрутки активна *Используйте оповещение для проверки Креста браузер, а затем прокомментировать для окончательного производственного кода.

HTH


(scrollWidth / Height-clientWidth / Height) - хороший индикатор наличия полосы прокрутки, но во многих случаях он даст вам "ложноположительный" ответ. если вам нужно быть точным, я бы предложил использовать следующую функцию. вместо того, чтобы пытаться угадать, прокручивается ли элемент - вы можете прокрутить его...

function isScrollable( el ){
  var y1 = el.scrollTop;
  el.scrollTop  += 1;
  var y2 = el.scrollTop;
  el.scrollTop  -= 1;
  var y3 = el.scrollTop;
  el.scrollTop   = y1;
  var x1 = el.scrollLeft;
  el.scrollLeft += 1;
  var x2 = el.scrollLeft;
  el.scrollLeft -= 1;
  var x3 = el.scrollLeft;
  el.scrollLeft  = x1;
  return {
    horizontallyScrollable: x1 !== x2 || x2 !== x3,
    verticallyScrollable: y1 !== y2 || y2 !== y3
  }
}
function check( id ){
  alert( JSON.stringify( isScrollable( document.getElementById( id ))));
}
#outer1, #outer2, #outer3 {
  background-color: pink;
  overflow: auto;
  float: left;
}
#inner {
  width:  150px;
  height: 150px;
}
button {  margin: 2em 0 0 1em; }
<div id="outer1" style="width: 100px; height: 100px;">
  <div id="inner">
    <button onclick="check('outer1')">check if<br>scrollable</button>
  </div>
</div>
<div id="outer2" style="width: 200px; height: 100px;">
  <div id="inner">
    <button onclick="check('outer2')">check if<br>scrollable</button>
  </div>
</div>
<div id="outer3" style="width: 100px; height: 180px;">
  <div id="inner">
    <button onclick="check('outer3')">check if<br>scrollable</button>
  </div>
</div>

Ugh все ответы здесь неполные, и давайте прекратим использовать jquery в so answers уже пожалуйста. Проверьте документацию jquery, если вам нужна информация о jquery.

вот обобщенная функция pure-javascript для проверки того, имеет ли элемент полосы прокрутки в полном виде:

// dimension - Either 'y' or 'x'
// computedStyles - (Optional) Pass in the domNodes computed styles if you already have it (since I hear its somewhat expensive)
function hasScrollBars(domNode, dimension, computedStyles) {
    dimension = dimension.toUpperCase()
    if(dimension === 'Y') {
        var length = 'Height'
    } else {
        var length = 'Width'
    }

    var scrollLength = 'scroll'+length
    var clientLength = 'client'+length
    var overflowDimension = 'overflow'+dimension

    var hasVScroll = domNode[scrollLength] > domNode[clientLength]


    // Check the overflow and overflowY properties for "auto" and "visible" values
    var cStyle = computedStyles || getComputedStyle(domNode)
    return hasVScroll && (cStyle[overflowDimension] == "visible"
                         || cStyle[overflowDimension] == "auto"
                         )
          || cStyle[overflowDimension] == "scroll"
}

решения, приведенные выше, будут работать в большинстве случаев, но проверка высоты прокрутки и переполнения иногда недостаточно и сбой для элементов body и html: https://codepen.io/anon/pen/EvzXZw

это решение проверяет, прокручивается ли элемент:

function isScrollableY (element) {
  return !!(element.scrollTop || (++element.scrollTop && element.scrollTop--));
}

Примечание: элементы с overflow=hidden также рассматриваются как прокручиваемые (подробнее), поэтому вы можете добавить условие против этого, если нужны:

function hasVerticalScrollbar (element) {
    let style = window.getComputedStyle(element);
    return !!(element.scrollTop || (++element.scrollTop && element.scrollTop--)) 
           && style["overflow"] !== "hidden" && style["overflow-y"] !== "hidden";
}

объяснение: фокус в том, что попытка прокрутки вниз и возврата его не будет отображаться браузером. Самая верхняя функция также может быть записана следующим образом:

function isScrollableY (element) {
  // if scrollTop is not 0 / larger than 0, then the element is scrolled and therefore must be scrollable
  // -> true  
  if (element.scrollTop === 0) {
    // if the element is zero it may be scrollable  
    // -> try scrolling about 1 pixel
    element.scrollTop++;
    // if the element is zero then scrolling did not succeed and therefore it is not scrollable 
    // -> false  
    if (element.scrollTop === 0) return false;
    // else the element is scrollable; reset the scrollTop property
    // -> true
    element.scrollTop--;
  }
  return true;
}

Я собираюсь расширить это еще больше для тех бедных душ, которые, как и я, используют один из современные JS фреймворки, а не JQuery и были полностью оставлены людьми этой темы:

это было написано в Angular 6, но если вы напишете React 16, Vue 2, Polymer, Ionic, React-Native, вы будете знать, что делать, чтобы адаптировать его. И это весь компонент, так что это должно быть легко.

import {ElementRef, AfterViewInit} from '@angular/core';

@Component({
  selector: 'app',
  templateUrl: './app.html',
  styleUrls: ['./app.scss']
})
export class App implements AfterViewInit {
scrollAmount;

constructor(
  private fb: FormBuilder,
) {}

ngAfterViewInit(){
  this.scrollAmount = this.element.nativeElement.querySelector('.elem-list');
  this.scrollAmount.addEventListener('wheel', e => { //you can put () instead of e
  // but e is usefull if you require the deltaY amount.
    if(this.scrollAmount.scrollHeight > this.scrollAmount.offsetHeight){
       // there is a scroll bar, do something!
    }else{
       // there is NO scroll bar, do something!
    }
  });
}
}

в html будет div с классом " elem-list" который стилизован в css или scss, чтобы иметь height и overflow значение, которое не hidden. (так auto или sroll )

я запускаю этот eval при событии прокрутки, потому что моя конечная цель состояла в том, чтобы иметь "автоматические прокрутки фокуса", которые решают, прокручивают ли они весь набор компонентов по горизонтали, если указанные компоненты не имеют вертикальной прокрутки, доступной, а в противном случае прокручивают только внутренности одного из компонентов по вертикали.

но вы можете разместить eval в другом месте, чтобы он был вызван чем-то другим.

важно помнить здесь, что вы никогда не заставили вернуться к использованию JQuery, всегда есть способ получить доступ к тем же функциям, которые он имеет, не используя его.


вот мое улучшение: добавлена parseInt. по какой-то странной причине он не работал без него.

// usage: jQuery('#my_div1').hasVerticalScrollBar();
// Credit: http://stackoverflow.com/questions/4814398/how-can-i-check-if-a-scrollbar-is-visible
(function($) {
    $.fn.hasVerticalScrollBar = function() {
        return this.get(0) ? parseInt( this.get(0).scrollHeight ) > parseInt( this.innerHeight() ) : false;
    };
})(jQuery);

вот улучшенная версия ответа Эвана, которая, похоже, правильно учитывает логику переполнения.

            function element_scrollbars(node) {
                var element = $(node);
                var overflow_x = element.css("overflow-x");
                var overflow_y = element.css("overflow-y");
                var overflow = element.css("overflow");
                if (overflow_x == "undefined") overflow_x == "";
                if (overflow_y == "undefined") overflow_y == "";
                if (overflow == "undefined") overflow == "";
                if (overflow_x == "") overflow_x = overflow;
                if (overflow_y == "") overflow_y = overflow;
                var scrollbar_vertical = (
                    (overflow_y == "scroll")
                    || (
                        (
                            (overflow_y == "hidden")
                            || (overflow_y == "visible")
                        )
                        && (
                            (node.scrollHeight > node.clientHeight)
                        )
                    )
                );
                var scrollbar_horizontal = (
                    (overflow_x == "scroll")
                    || (
                        (
                            (overflow_x == "hidden")
                            || (overflow_x == "visible")
                        )
                        && (
                            (node.scrollWidth > node.clientWidth)
                        )
                    )
                );
                return {
                    vertical: scrollbar_vertical,
                    horizontal: scrollbar_horizontal
                };
            }

работает на Chrome, Edge, Firefox и Опера по крайней мере в новых версиях.

Использование JQuery...

настройки исправить футер:
function fixFooterCaller()
{
    const body = $('body');
    const footer = $('body footer');

    return function ()
    {
        // If the scroll bar is visible
        if ($(document).height() > $(window).height())
        {
            // Reset
            footer.css('position', 'inherit');
            // Erase the padding added in the above code
            body.css('padding-bottom', '0');
        }
        // If the scrollbar is NOT visible
        else
        {
            // Make it fixed at the bottom
            footer.css('position', 'fixed');
            // And put a padding to the body as the size of the footer
            // This makes the footer do not cover the content and when
            // it does, this event fix it
            body.css('padding-bottom', footer.outerHeight());
        }
    }
}

он возвращает функция. Сделал этот путь только для того, чтобы установить тело и нижний колонтитул один раз.

и затем установите это, когда документ будет готов.

$(document).ready(function ()
{
    const fixFooter = fixFooterCaller();

    // Put in a timeout call instead of just call the fixFooter function
    // to prevent the page elements needed don't be ready at this time
    setTimeout(fixFooter, 0);
    // The function must be called every time the window is resized
    $(window).resize(fixFooter);
});

добавьте это в свой нижний колонтитул css:

footer {
    bottom: 0;
}

большинство представленных ответов приблизили меня к тому, где я должен был быть, но не совсем там.

мы в основном хотели оценить, будут ли полосы прокрутки видны в нормальной ситуации, по этому определению, означающему, что размер элемента тела больше, чем порт просмотра. Это не было представленным решением, поэтому я его представляю.

надеюсь, это поможет кому-то!

(function($) {
    $.fn.hasScrollBar = function() {
        return this.get(0).scrollHeight > $(window).height();
    }
})(jQuery);

по сути, у нас есть


найдите родительский элемент текущего элемента с вертикальной прокруткой или телом.

$.fn.scrollableParent = function() {
    var $parents = this.parents();

    var $scrollable = $parents.filter(function(idx) {
        return this.scrollHeight > this.offsetHeight && this.offsetWidth !== this.clientWidth;
    }).first();

    if ($scrollable.length === 0) {
        $scrollable = $('html, body');
    }
    return $scrollable;
};

Он может использоваться для автопрокрутки к текущему элементу через:

var $scrollable = $elem.scrollableParent();
$scrollable.scrollTop($elem.position().top);