如何检测N个列表框末尾的滚动

本文关键字:列表 滚动 何检测 检测 | 更新日期: 2023-09-27 18:16:33

我发现这种方法是由slimcode创建的:

public static readonly DependencyProperty ListVerticalOffsetProperty = DependencyProperty.Register(
  "ListVerticalOffset",
  typeof( double ),
  typeof( SearchBusinessResultsPage ),
  new PropertyMetadata( new PropertyChangedCallback( OnListVerticalOffsetChanged ) ) );
public double ListVerticalOffset
{
  get { return ( double )this.GetValue( ListVerticalOffsetProperty ); }
  set { this.SetValue( ListVerticalOffsetProperty, value ); }
}
private static void OnListVerticalOffsetChanged( DependencyObject obj, DependencyPropertyChangedEventArgs e )
{
    // ...
}

它创建一个单一的readonly属性和一个单一的静态方法来处理它。但是我想让N个列表在滚动结束时处理一个事件。

我不知道如何使用这段代码来处理不同的列表…

有更好的方法吗?如何在不同的列表中使用相同的东西?

如何检测N个列表框末尾的滚动

您可以创建一个继承自ListBox的BusinessListBox类,并定义它的样式和模板,就像上面提到的示例一样。然后,您可以扩展DependencyProperty更改处理程序以拥有实例更改处理程序方法。

private static void OnListVerticalOffsetChanged(
    DependencyObject d, DependencyPropertyChangedEventArgs e)
{
    var target = (BusinessListBox)d;
    double oldListVerticalOffset = (double)e.OldValue;
    double newListVerticalOffset = target.ListVerticalOffset;
    target.OnListVerticalOffsetChanged(oldListVerticalOffset, newListVerticalOffset);
}
private void OnListVerticalOffsetChanged(
    double oldListVerticalOffset, double newListVerticalOffset)
{
}

理想情况下,你只需要实现一个行为或附加属性,允许你在普通的ListBox上做同样的事情。

你所需要做的就是为ScrollChangedEvent检测添加一个处理程序,并使用VerticalOffset属性来查找滚动是否已经到达滚动的末尾。

 listBox.AddHandler(ScrollViewer.ScrollChangedEvent, new ScrollChangedEventHandler((o , e) => {
            ListBox lb = (ListBox) o;
            ScrollViewer sv = GetDescendantByType(lb, typeof(ScrollViewer)) as ScrollViewer;
            if (sv.ScrollableHeight == sv.VerticalOffset) { 
                //End of the scroll reached
            }
        }));

在ListBox

中查找ScrollViewer的辅助方法
   /***
    * Helper method to traverse the root to find the the element of type `type`
    ***/
    public static Visual GetDescendantByType(Visual element, Type type) {
        if (element == null) {
            return null;
        }
        if (element.GetType() == type) {
            return element;
        }
        Visual foundElement = null;
        for (int i = 0; i < VisualTreeHelper.GetChildrenCount(element); i++) {
            Visual visual = VisualTreeHelper.GetChild(element, i) as Visual;
            foundElement = GetDescendantByType(visual, type);
            if (foundElement != null) {
                break;
            }
        }
        return foundElement;
    }