WPF之VirtualizingStackPanel.IsVirtualizing="False"

简介: 原文:WPF之VirtualizingStackPanel.IsVirtualizing="False" 相信从winform转到wpf的人都遇到过这样的困惑,在处理DataGrid时,我想让某一单元格根据格式校验的不同情况显示不同的颜色,以便于用户区分。
原文: WPF之VirtualizingStackPanel.IsVirtualizing="False"

相信从winform转到wpf的人都遇到过这样的困惑,在处理DataGrid时,我想让某一单元格根据格式校验的不同情况显示不同的颜色,以便于用户区分。

于是你去查找各种资料,wpf下DataGrid如何改变某一单元格颜色,如果运气不好,可能搜索到的结果会令你失望,运气不错的话你会搜到类似如下代码:

private void changeCell() 
        {
            this.sj_DataGrid.Dispatcher.Invoke(
            new Action(
                delegate
                {
                    foreach (LSErr l in lse)
                    {
                        for (int i = 0; i < this.sj_DataGrid.Items.Count; i++)
                        {
                            DataRowView drv = this.sj_DataGrid.Items[i] as DataRowView;
                            DataGridRow row = (DataGridRow)this.sj_DataGrid.ItemContainerGenerator.ContainerFromIndex(i);


                            if (drv["StockCode"].ToString()==l.stCode)
                            {
                                System.Windows.Controls.Primitives.DataGridCellsPresenter presenter = GetVisualChild<System.Windows.Controls.Primitives.DataGridCellsPresenter>(row);
                                if (presenter != null)
                                {
                                    DataGridCell cell = (DataGridCell)presenter.ItemContainerGenerator.ContainerFromIndex(l.ci);
                                    cell.Background = new SolidColorBrush(Colors.CadetBlue);
                                }
                                break;
                            }
                        }
                    }
                }
                )
                );
        }
//获取单元格
        public static T GetVisualChild<T>(Visual parent) where T : Visual
        {


            T childContent = default(T);
            if (parent != null)
            {
                int numVisuals = VisualTreeHelper.GetChildrenCount(parent);
                for (int i = 0; i < numVisuals; i++)
                {
                    Visual v = (Visual)VisualTreeHelper.GetChild(parent, i);
                    childContent = v as T;
                    if (childContent == null)
                    {
                        childContent = GetVisualChild<T>(v);
                    }
                    if (childContent != null)
                    {
                        break;
                    }
                }
                //Thread.Sleep(10);
            }
            return childContent;
        }



看起来很不错,而且运行也能成功。

不过很快你就会发现问题,那就是当你的窗口大小变化时,或者你拖动滚动条的时候奇葩的事情发生了,你选择变色的单元格变掉了,变色的可能是任何一个单元格,反正不是你想要的那一个。


于是你又再一次踏上了搜索之旅


终于让你找到了一个解决办法,就是我们标题中的VirtualizingStackPanel.IsVirtualizing="False",不错,效果很好,DataGrid应用这一属性之后我们可以准确的改变单元格颜色了。


但是,问题又来了...


当你的DataGrid加载的数据稍微多一些时,比如2000,3000,...10000...,再次运行你的程序你会发现主界面竟然卡住了好几秒,于是你设置断点查看,运行时间。你会发现由于ui虚拟化被禁用,数据加载的时间被无限拖长。用户界面卡顿,造成短时间内无响应,给用户死机的感受(PS:实际上此时程序正在玩儿命的绘制UI)


这种情况下,纵使你数据校验再好,数据校验整个过程中程序卡死的假象蒙蔽下,用户也会失去耐心,除非你的项目经理很强势,让用户耐心等待...


So,换一种实现策略吧


最终,还是暂时使用模板类中的触发器来实现吧,当不同的属性下生成不同的颜色

<Window.Resources>
        <!--<c:ErrorConverter x:Key="ErrorConverter"/>-->
        <c:SheetArr x:Key="SheetCols"/>
        <Style TargetType="{x:Type Button}" x:Key="highlightStyle">


            <Style.Triggers>


                <Trigger Property="IsEnabled" Value="false">


                    <Setter Property="Background" Value="#EEEEEE" />


                </Trigger>

                <MultiTrigger>


                    <MultiTrigger.Conditions>
                        <Condition Property="IsEnabled" Value="true" />


                    </MultiTrigger.Conditions>


                    <Setter Property="BorderBrush" Value="Red"/>


                    <Setter Property="FontSize" Value="14" />


                    <Setter Property="FontWeight" Value="Bold" />


                    <Setter Property="Foreground" Value="Red" />


                </MultiTrigger>


            </Style.Triggers>


        </Style>
        <Style TargetType="{x:Type DataGridRow}">
            <Style.Triggers>
                <DataTrigger Binding="{Binding State}" Value="格式错误">
                    <Setter Property="ToolTip">
                        <Setter.Value>
                            <ToolTip>
                                <TextBlock Text="{Binding State,StringFormat='错误:{0}'}"/>
                            </ToolTip>
                        </Setter.Value>
                    </Setter>
                    <Setter Property="Foreground" Value="Red" />
                </DataTrigger>
            </Style.Triggers>
        </Style>


    </Window.Resources>



这只是一种过度方案,当发现某单元格值达到某种条件时改变背景色,不够灵活,真正的解决这一问题还得研究wpf的ui visualization实现上下手


希望已经碰到这种问题的兄弟少走点儿弯路,或者如果有大神已经解决这种问题还请指教


目录
相关文章
|
C# 虚拟化 索引
【WPF】UI虚拟化之------自定义VirtualizingWrapPanel
原文:【WPF】UI虚拟化之------自定义VirtualizingWrapPanel 前言 前几天QA报了一个关于OOM的bug,在排查的过程中发现,ListBox控件中被塞入了过多的Item,而ListBox又定义了两种样式的ItemsPanelTemplate。
2023 0
|
8月前
|
C#
WPF-Binding问题-模板样式使用Binding TemplatedParent与TemplateBinding区别
WPF-Binding问题-模板样式使用Binding TemplatedParent与TemplateBinding区别
82 0
|
存储 数据可视化 程序员
Window Form 控件基础
在开始正式介绍如何开发自定义控件之前,有必要先了解一下控件开发的基础知识。下面从控件的概念、分类和开发模式上对控件做一个基本的概述。
819 0
Window Form 控件基础
|
C#
WPF: FishEyePanel/FanPanel - 自定义Panel
原文:WPF: FishEyePanel/FanPanel - 自定义Panel 原文来自CodeProject,主要介绍如何创建自定义的Panel,如同Grid和StackPanel。
1214 0
UWP FillRowViewPanel
原文:UWP FillRowViewPanel 最近有童鞋有这种需求,说实话我不知道这个Panel怎么起名字。 效果连接https://tuchong.com/tags/风光/ 下面是我做成的效果,可以规定每个Row的Items个数 2个 3个 4个 代码在:GitHub 下面我来说一下我的思路 其实很早之前就写过这种可变大小的控件,但这次的跟这个需求有点变化,这个每一行个数一定,大小根据图片的大小进行填充。
948 0
|
C# 数据可视化 Windows
[WPF]使用WindowChrome自定义Window Style
原文:[WPF]使用WindowChrome自定义Window Style 1. 前言 做了WPF开发多年,一直未曾自己实现一个自定义Window Style,无论是《WPF编程宝典》或是各种博客都建议使用WindowStyle="None" 和 AllowsTransparency="True",于是想当然以为这样就可以了。
2039 0
|
前端开发 Windows 容器
背水一战 Windows 10 (38) - 控件(布局类): Panel, Canvas, RelativePanel, StackPanel, Grid
原文:背水一战 Windows 10 (38) - 控件(布局类): Panel, Canvas, RelativePanel, StackPanel, Grid [源码下载] 背水一战 Windows 10 (38) - 控件(布局类): Panel, Canvas, RelativePanel...
978 0