我知道这听起来很复杂,但请跟我来:
您有一个带有VirtualMode = true.的ListView
选择一个项目,向下滚动,使所选项目超出可视区域,然后尝试将另一个项目添加到ListView。
您将看到在瞬间它的行为异常,并看到某种闪烁。如果你升级情况并尝试非常快地添加很多项目(我每秒添加大约20次),你会发现小问题变得非常大。它是闪烁和无效项的组合。
我已经调查过这个问题,似乎ListView会为所选的项目生成很多RetrieveVirtualItem事件(尽管它显然不可见)。
似乎当我添加一个新项目(增加VirtualListSize)时,ListView首先尝试将焦点放在所选项目上,然后返回到前一个位置。
有没有人遇到过同样的问题?
发布于 2010-09-30 19:57:39
下面是一个派生类,它解决了这个问题。
使用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));
}
}
}
}发布于 2010-08-06 19:55:00
各种控件都具有受保护的DoubleBuffered属性。您可以尝试从ListView派生自己的DBListView,并在其构造函数中将其DoubleBuffered属性设置为true。
https://stackoverflow.com/questions/1596005
复制相似问题