C# VirtualMode 下的 ListView 当所选项目不可见时闪烁

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

我知道这听起来很复杂,但请跟随我:

您有一个带有

VirtualMode = true.

的 ListView

选择一个项目,向下滚动,使所选项目超出可视区域,然后尝试将另一个项目添加到 ListView。

您会在一瞬间看到它表现异常并看到某种闪烁。 如果您升级情况并尝试快速添加大量项目(我每秒添加大约 20 次),您会发现小问题变得非常大。它是周围闪烁和无效项目的组合。

我已经调查了这个问题,似乎 ListView 为所选项目生成了很多

RetrieveVirtualItem
事件(即使它显然不可见)。

似乎当我添加一个新项目(增加 VirtualListSize)时,ListView 首先尝试将焦点集中在所选项目上,然后返回到之前的位置。

有人遇到同样的问题吗?

c# listview flicker virtualmode
3个回答
5
投票

这里有一个派生类解决了这个问题。

使用

SetVirtualListSize()
方法代替常规的
VirtualListSize

public class FlickerFreeListView : ListView
{
    #region Static Functionality

    private static FieldInfo _internalVirtualListSizeField;

    static FlickerFreeListView()
    {
        _internalVirtualListSizeField = typeof(ListView).GetField("virtualListSize", System.Reflection.BindingFlags.NonPublic | BindingFlags.Instance);

        if (_internalVirtualListSizeField == null)
        {
            string msg = "Private field virtualListSize in type System.Windows.Forms.ListView is not found. Workaround is incompatible with installed .NET Framework version, running without workaround.";
            Trace.WriteLine(msg);
        }
    }

    #endregion


    [DllImport("user32.dll", CharSet = CharSet.Auto)]
    private static extern IntPtr SendMessage(HandleRef hWnd, int msg, IntPtr wParam, IntPtr lParam);

    internal IntPtr SendMessage(int msg, IntPtr wparam, IntPtr lparam)
    {
        return SendMessage(new HandleRef(this, this.Handle), msg, wparam, lparam);
    }

    public void SetVirtualListSize(int size)
    {
        // if workaround incompatible with current framework version (usually MONO)
        if (_internalVirtualListSizeField == null)
        {
            VirtualListSize = size;
        }
        else
        {
            if (size < 0)
            {
                throw new ArgumentException("ListViewVirtualListSizeInvalidArgument");
            }

            _internalVirtualListSizeField.SetValue(this, size);
            if ((base.IsHandleCreated && this.VirtualMode) && !base.DesignMode)
            {
                SendMessage(0x102f, new IntPtr(size), new IntPtr(2));
            }
        }
    }
}

1
投票

各种控件都有受保护的 DoubleBuffered 属性。您可以尝试从 ListView 派生自己的 DBListView,并在其构造函数中将其 DoubleBuffered 属性设置为 true。


0
投票

请注意,自从 .net 8 字段 virtualListSize 被重命名为 _virtualListSize

_internalVirtualListSizeField = typeof(ListView).GetField("_virtualListSize", System.Reflection.BindingFlags.NonPublic | BindingFlags.Instance);

© www.soinside.com 2019 - 2024. All rights reserved.