C#ListView ItemSelectionChanged事件多选择仅获取最后选择的项目

时间:2013-09-28 02:11:00

标签: c# events listview multi-select

我在C#.NET 4.5中使用多选ListView 选择多个项目时会出现问题(例如Shift + End或Shift + Click等)。这些只是许多不同鼠标/键盘组合的几个例子,当然还有多选项。

这是我在列表中选择项目时的事件处理程序:

private void lvTitles_ItemSelectionChanged(object sender, ListViewItemSelectionChangedEventArgs e)
{
    MessageBox.Show(e.Item.Text.ToString());
    //MessageBox just for testing I am actually running a SQL query here
}

我的问题是,如果我选择了500项,则会触发500次事件。目的是获取用户选择的最后一项(通过上面提到的键盘/鼠标组合),并使用它做一些事情......在我的情况下,我需要针对它运行SQL查询。

如果我先点击列表视图中的第0项就可以运行查询,那么当你移动+结束时它会突出显示所有其余部分,我希望它只在最后选择的项目上运行查询。相反,它在两者之间的每个项目上运行。

编辑:另一方面,事件会因为取消选择而触发,在这种情况下,取消选择时它实际上不应该做任何事情。

1 个答案:

答案 0 :(得分:1)

您是否考虑过按下按钮执行操作?这样他们也可以使用Ctrl-Click选择他们想要的任何单个项目?

否则你需要做的就是在开始行动之前等待一段时间,称为去抖动,你可以在这里阅读更多关于去抖动的内容:https://stackoverflow.com/a/4517995/984780

我创建了一个可用于去抖动的课程:

public class Debounce {
    private Action _action;
    private bool _isThreadRunning;
    private Thread _thread;
    private DateTime _runAt;
    private double _waitSeconds;

    private Debounce(double waitSeconds, Action action) {
        _action = action;
        _waitSeconds = waitSeconds;
    }

    private void Invoke() {
        _runAt = DateTime.Now.AddSeconds(_waitSeconds);

        lock(this) {
            if(!_isThreadRunning) {
                _isThreadRunning = true;

                _thread = new Thread(() => {
                    while(true) {
                        Thread.Sleep(100);

                        lock(this) {
                            if(DateTime.Now > _runAt) {
                                _action();
                                _isThreadRunning = false;
                                _thread = null;
                                break;
                            }
                        }
                    }
                });

                _thread.Start();
            }
        }
    }

    private static Dictionary<Action, Debounce> __debounces;
    private static Dictionary<Action, Debounce> _debounces {
        get {
            if(__debounces == null) {
                __debounces = new Dictionary<Action, Debounce>();
            }

            return __debounces;
        }
    }

    public static void Run(double waitSeconds, Action action) {
        Debounce debounce;

        if(!_debounces.TryGetValue(action, out debounce)) {
            debounce = new Debounce(waitSeconds, action);
            _debounces.Add(action, debounce);
        }

        debounce._waitSeconds = waitSeconds;
        debounce.Invoke();
    }
}

然后您可以将代码更改为:

private void lvTitles_ItemSelectionChanged(object sender, ListViewItemSelectionChangedEventArgs e)
{
    Debounce.Run(5, () => MessageBox.Show(e.Item.Text.ToString()));
}

无论他们如何选择项目,这都应该有效,它将在最后一次选择操作后5秒运行您的代码。

我刚刚写了这门课并进行了快速测试,建议进行更彻底的测试。无论如何,希望它能够得到这个想法。