WPF:如何避免ListBox或ListView中已选中复选框的闪烁?

时间:2010-01-05 16:43:15

标签: wpf rendering

如何避免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;
        }
    }
}

3 个答案:

答案 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>