【发布时间】:2023-03-22 15:40:01
【问题描述】:
我正在开发一个自定义敲除绑定,该绑定确定是否正在滚动特定元素,并使用元素相对于视口的顶部更新绑定的 observable。目前,绑定似乎有效,但我担心是否在某些情况下无效。
HTML:
Scroll position: <span data-bind="text: scrollPosition"></span>
<div class="longdiv">
<p data-bind="scroll: scrollPosition">This is some text.</p>
<div class="shim"></div>
</div>
CSS:
.longdiv {
width: 200px;
height: 200px;
overflow: scroll;
border: 1px solid black;
}
JS:
ko.bindingHandlers.scroll = {
init: function(element, valueAccessor, allBindings, viewModel, bindingContext) {
var firstScrollableContainer = null;
var binding = allBindings.get('scroll');
$(element).parents().each(function (i, parent) {
if ($(parent).css('overflow')=='scroll') {
firstScrollableContainer = parent;
return false;
}
});
firstScrollableContainer = firstScrollableContainer || window;
binding(element.getBoundingClientRect().top);
$(firstScrollableContainer).scroll(function() {
binding(element.getBoundingClientRect().top);
});
}
};
var ViewModel = function() {
var self = this;
self.scrollPosition = ko.observable(0);
};
ko.applyBindings(new ViewModel());
绑定获取元素并使用 jQuery 遍历父链查看父元素是否有溢出:滚动集。如果它找到一个带有溢出的 div:滚动,它会将一个事件处理程序绑定到该元素的滚动事件。如果它没有找到带有溢出:滚动的父级,则它会绑定到窗口的滚动事件。
所以我在找什么,给定一个结构如下的文档:
body > div > div > div > p
是最接近可滚动的 p 的包含元素,因此我可以将事件处理程序附加到它。
我的问题是:正在查看溢出:滚动一个足够的测试以查看是否可以滚动父元素?如果没有,我应该看什么?
编辑:根据您有用的 cmets 和答案,这是我想出的解决方案:
function scrollable(element) {
var vertically_scrollable, horizontally_scrollable;
var e = $(element);
if ( e.css('overflow') == 'scroll'
|| e.css('overflow') == 'auto'
|| e.css('overflowY') == 'scroll'
|| e.css('overflowY') == 'auto'
|| e.css('height') != 'none'
|| e.css('max-height') != 'none'
) {
return true;
} else {
return false;
}
}
【问题讨论】:
-
我倾向于认为验证
overflow、overflow-x和overflow-y就足够了,但我不确定。检查这个:jsperf.com/jquery-scrollable -
注释:如果需要,可以使用 jQuery 的
scrollTop()来获取滚动位置,而不是getBoundingClientRect。
标签: jquery html knockout.js