c# – ListView中的“无限滚动”/避免可重入滚动事件

我的目标是在 WPF ListView中模拟“无限滚动”.我用一些不太理想的方法完成了这项任务,我确信有更好的方法来做到这一点.

通过“无限滚动”我的意思是:

假设ListView有20个项目(订购1,2,3,4,…… 17,18,19,20).当用户向下滚动一个项目时,ListView顶部的项目将被删除并放置在ListView的末尾,因此项目的顺序为2,5,… 18,20,1.现在,如果用户向下滚动两个项目,则前两个项目被移除并放置在最后,因此项目的顺序是4,6,7,… 20,1,3.现在,类似地,如果用户向上滚动一个项目,则ListView底部的项目将被删除并放置在开头,因此项目的顺序为3,… 19,2 .

我已经完成了这个任务,并将以下函数分配给ScrollViewer的ScrollChanged事件,我希望它是“无限的”:

// sv - the ScrollViewer to which this event handler is listening
// lv - the ListView associated with "sv"
bool handle_scroll = true;
private void inf_scroll(object sender,ScrollChangedEventArgs e) {
    if (handle_scroll) {
        for (int i = 0; i < e.VerticalChange; i++) {
            object tmp = lv.Items[0];
            lv.Items.RemoveAt(0);
            lv.Items.Add(tmp);
            handle_scroll = false;
        }
        for (int i = 0; i > e.VerticalChange; i--) {
            object tmp = lv.Items[lv.Items.Count - 1];
            lv.Items.RemoveAt(lv.Items.Count - 1);
            lv.Items.Insert(0,tmp);
            handle_scroll = false;
        }
        if(!handle_scroll){
            sv.ScrollToVerticalOffset(sv.VerticalOffset - e.VerticalChange);
        }
    }
    else {
        handle_scroll = true;
    }
}

注意变量handle_scroll.我把它放到位是因为对sv.ScrollToVerticalOffset的调用将导致整个inf_scroll函数被递归调用(如果不存在的话).

我知道在ScrollChanged事件处理程序中滚动ScrollViewer是不好的做法,这就是我要问的原因:有更好的方法吗?如何阻止对inf_scroll的递归调用?有没有更好的方法来模拟“无限滚动”?

解决方法

虽然它没有明确地避免在ScrollChanged处理程序中滚动,但您可以尝试以下操作:
// sv - the ScrollViewer to which this event handler is listening
// lv - the ListView associated with "sv"
private void inf_scroll(object sender,ScrollChangedEventArgs e) {
    for (int i = 0; i < e.VerticalChange; i++) {
        object tmp = lv.Items[0];
        lv.Items.RemoveAt(0);
        lv.Items.Add(tmp);
    }
    for (int i = 0; i > e.VerticalChange; i--) {
        object tmp = lv.Items[lv.Items.Count - 1];
        lv.Items.RemoveAt(lv.Items.Count - 1);
        lv.Items.Insert(0,tmp);
    }
    lv.ScrollChanged -= inf_scroll;        // remove the handler temporarily
    sv.ScrollToVerticalOffset(sv.VerticalOffset - e.VerticalChange);
    dispatcher.BeginInvoke(dispatcherPriority.Send,new Action(() =>{
        sv.ScrollChanged += inf_scroll;    // add the handler back after the scrolling has occurred to avoid recursive scrolling
    }));
}

相关文章

在要实现单例模式的类当中添加如下代码:实例化的时候:frmC...
1、如果制作圆角窗体,窗体先继承DOTNETBAR的:public parti...
根据网上资料,自己很粗略的实现了一个winform搜索提示,但是...
近期在做DSOFramer这个控件,打算自己弄一个自定义控件来封装...
今天玩了一把WMI,查询了一下电脑的硬件信息,感觉很多代码都...
最近在研究WinWordControl这个控件,因为上级要求在系统里,...