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

问题描述 投票:0回答:2

我在 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,则可以运行查询,然后当您 shift+end 时,它会突出显示其余所有项目,并且我希望它仅对所选的最后一个项目运行查询。 相反,它在中间的每个项目上运行。

编辑:另一方面,事件也会在取消选择时触发,在这种情况下,取消选择时它实际上不应该执行任何操作。

c# events listview multi-select
2个回答
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 秒运行您的代码。

我刚刚写了这门课并做了一个快速测试,建议进行更彻底的测试。无论如何,希望这足以让你明白这个想法。


0
投票

这样你就不必担心 NULL 值。 所以你可以使用多项选择:

private void listView1_ItemSelectionChanged(object sender, ListViewItemSelectionChangedEventArgs e)
{
 ListViewItem lvi = e.Item;
 lvi.Checked = !lvi.Checked;
}
© www.soinside.com 2019 - 2024. All rights reserved.