如何避免WPF ListBox或ListView中已选中复选框的闪烁?通过单击“刷新”按钮或滚动列表框,可以使用以下代码复制它。如果IsChecked为false,则不会闪烁。
Window1.xaml:
<Window x:Class="WpfApplication6.Window1"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
Title="Window1" Height="300" Width="300">
<Grid>
<Grid.ColumnDefinitions>
<ColumnDefinition/>
<ColumnDefinition/>
</Grid.ColumnDefinitions>
<ListBox Name="listBox">
<ListBox.ItemTemplate>
<DataTemplate>
<StackPanel Orientation="Horizontal">
<CheckBox IsChecked="True"
VerticalAlignment="Center"/>
<Label Padding="3"
Content="{Binding}"/>
</StackPanel>
</DataTemplate>
</ListBox.ItemTemplate>
</ListBox>
<Button Content="Refresh"
Grid.Column="1"
VerticalAlignment="Top"
Click="Button_Click"/>
</Grid>
</Window>
Window1.xaml.cs:
using System.Windows;
namespace WpfApplication6
{
partial class Window1 : Window
{
public Window1()
{
InitializeComponent();
Button_Click(null, null);
}
void Button_Click(object sender, RoutedEventArgs e)
{
var items = new int[10000];
for (int i = 0; i < items.Length; i++)
items[i] = i + 1;
listBox.ItemsSource = items;
}
}
}
答案 0 :(得分:4)
它闪烁,因为你扔掉了旧的ItemsSource并创建了一个新的ItemsSource。这需要重做所有绑定,并且需要重新创建显示每个项目的模板。要避免重新创建整个列表的性能开销,只需修改现有ItemsSource中的各个元素即可。然后,绑定到已更改的属性和/或项的DataTemplate部分将自动更新,而无需重新创建整个列表视图。这样做可以消除你所看到的“闪烁”。
尝试使用代码隐藏:
public partial class MainWindow : Window
{
private ObservableCollection<object> _items;
public MainWindow()
{
InitializeComponent();
_items = new ObservableCollection<object>();
for (int i = 0; i < 10000; i++)
_items.Add(i + 1);
listBox.ItemsSource = _items;
}
void Button_Click(object sender, RoutedEventArgs e)
{
for (int i = 0; i < _items.Count;i++)
{
if (!(_items[i] is int)) continue;
_items[i] = (int)_items[i] + 1;
}
}
}
答案 1 :(得分:2)
你的意思是选中复选框吗?我认为您需要在选中/设置复选框时更改动画。
它不会出现在Windows XP上(这就是我认为它是动画的原因),我还没有测试过Vista:)
祝你好运。答案 2 :(得分:1)
+1给Snake,他在这里有正确的答案。
添加到此:
CheckBox控件有一个带有故事板动画的控件模板,当检查状态发生变化时,它会打开/关闭选中的图标。当您绑定到ObservableCollection并重新创建ItemsSource时,Checked状态会发生变化,这会导致创建新的Checkbox(使用IsChecked = false)并绑定到ViewModel(可能导致IsChecked = True)。
要禁用此“功能”,您可以修改填充ObservableCollection的方式,或者如果不可能,则可以更改复选框的模板/样式。
只需对Checkbox的ControlTemplate进行反向工程(使用混合,或使用其中一个WPF主题)并找到这些行。你需要将两个动画的持续时间设置为零
<!-- Inside the CheckBoxTemplate ControlTemplate -->
<Storyboard x:Key="CheckedTrue">
<DoubleAnimationUsingKeyFrames Storyboard.TargetName="CheckIcon"
Storyboard.TargetProperty="(UIElement.Opacity)">
<SplineDoubleKeyFrame KeyTime="00:00:00.1000000" Value="1" />
</DoubleAnimationUsingKeyFrames>
</Storyboard>
<Storyboard x:Key="CheckedFalse">
<DoubleAnimationUsingKeyFrames Storyboard.TargetName="CheckIcon"
Storyboard.TargetProperty="(UIElement.Opacity)">
<SplineDoubleKeyFrame KeyTime="00:00:00.4000000" Value="0" />
</DoubleAnimationUsingKeyFrames>
</Storyboard>