我知道这听起来很复杂,但请跟随我:
您有一个带有
VirtualMode = true.
的 ListView
选择一个项目,向下滚动,使所选项目超出可视区域,然后尝试将另一个项目添加到 ListView。
您会在一瞬间看到它表现异常并看到某种闪烁。 如果您升级情况并尝试快速添加大量项目(我每秒添加大约 20 次),您会发现小问题变得非常大。它是周围闪烁和无效项目的组合。
我已经调查了这个问题,似乎 ListView 为所选项目生成了很多
RetrieveVirtualItem
事件(即使它显然不可见)。
似乎当我添加一个新项目(增加 VirtualListSize)时,ListView 首先尝试将焦点集中在所选项目上,然后返回到之前的位置。
有人遇到同样的问题吗?
这里有一个派生类解决了这个问题。
使用
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));
}
}
}
}
各种控件都有受保护的 DoubleBuffered 属性。您可以尝试从 ListView 派生自己的 DBListView,并在其构造函数中将其 DoubleBuffered 属性设置为 true。
请注意,自从 .net 8 字段 virtualListSize 被重命名为 _virtualListSize
_internalVirtualListSizeField = typeof(ListView).GetField("_virtualListSize", System.Reflection.BindingFlags.NonPublic | BindingFlags.Instance);