原文 WPF:DataTemplateSelector设置控件不同的样式

最近想实现这么个东西,一个ListBox, 里面的ListBoxItem可能是文本框、下拉框、日期选择控件等等。

很自然的想到了DataTemplateSelector,并且事先定义好各类DataTemplate以显示不同的控件。

先定义好各类资源


    <Window.Resources>

        <DataTemplate x:Key="textBox">

            <Border BorderBrush="Gray" BorderThickness="1">

                <TextBox Text="{Binding CombinedValue}"></TextBox>

            </Border>

        </DataTemplate>

        <DataTemplate x:Key="comboBox">

            <Border BorderBrush="Gray" BorderThickness="1">

                <ComboBox ItemsSource="{Binding CombinedValue}"></ComboBox>

            </Border>

        </DataTemplate>

        <DataTemplate x:Key="dateTime">

            <Border BorderBrush="Gray" BorderThickness="1">

                <DatePicker Text="{Binding CombinedValue}" ></DatePicker>

            </Border>

        </DataTemplate>

    </Window.Resources>

然后在ListBox中设置ItemDataTemplateSelector


<ListBox ItemsSource="{Binding}">

        <ListBox.ItemTemplateSelector>

            <local:DataTypeTemplateSelector TextBoxTemplate="{StaticResource textBox}"

                                            ComboBoxTemplate="{StaticResource comboBox}"

                                            DateTimeTemplate="{StaticResource dateTime}"></local:DataTypeTemplateSelector>

        </ListBox.ItemTemplateSelector>

    </ListBox>

新建一个类继承DataTemplateSelector


   public class DataTypeTemplateSelector:DataTemplateSelector

    {

        public DataTemplate TextBoxTemplate { get; set; }

        public DataTemplate ComboBoxTemplate { get; set; }

        public DataTemplate DateTimeTemplate { get; set; }

        public override DataTemplate SelectTemplate(object item, DependencyObject container)

        {

            CombinedEntity entity = item as CombinedEntity; //CombinedEnity为绑定数据对象

            string typeName = entity.TypeName;

            if (typeName == "TextBox")

            {

                return TextBoxTemplate;

            }

            if (typeName == "ComboBox")

            {

                return ComboBoxTemplate;

            }

            if (typeName == "DateTime")

            {

                return DateTimeTemplate;

            }

            return null;

        }

    }

设置好DataContext,即可运行


 public partial class CombinedControl : Window

    {

        public List<CombinedEntity> entities;

        public CombinedControl()

        {

            InitializeComponent();

            entities = new List<CombinedEntity>()

            {

                new CombinedEntity{ CombinedValue=new List<string>{"","",""}, TypeName="ComboBox"},

                new CombinedEntity{ CombinedValue ="Test", TypeName="TextBox"},

                new CombinedEntity{ CombinedValue=DateTime.Now, TypeName="DateTime"}

            };

            this.DataContext = entities;

        }

    }

    public class CombinedEntity

    {

        /// <summary>

        /// 绑定数据的值

        /// </summary>

        public object CombinedValue

        {

            get;

            set;

        }

        /// <summary>

        /// 数据的类型

        /// </summary>

        public string TypeName

        {

            get;

            set;

        }

    }

如果运行成功,我们可以看到一个下拉框,一个文本框,一个日期选择控件都做为ListBox的子项显示在窗口中。

但是,我发现,在DataTypeTemplateSelector对象的SelectTemplate 方法中,居然需要把item对象转换成我们的绑定数据对象

CombinedEntity entity = item as CombinedEntity; //CombinedEnity为绑定数据对象

这意味着前台需要引入后端的业务逻辑,代码的味道相当不好,不过没有关系,我们有强大的反射工具,重构下代码:


  public override DataTemplate SelectTemplate(object item, DependencyObject container)

        {

            Type t = item.GetType();

            string typeName = null;

            PropertyInfo[] properties = t.GetProperties();

            foreach (PropertyInfo pi in properties)

            {

                if (pi.Name == "TypeName")

                {

                    typeName = pi.GetValue(item, null).ToString();

                    break;

                }

            }

            if (typeName == "TextBox")

            {

                return TextBoxTemplate;

            }

            if (typeName == "ComboBox")

            {

                return ComboBoxTemplate;

            }

            if (typeName == "DateTime")

            {

                return DateTimeTemplate;

            }

            return null;

        }

这样,我们就无需引入后端的实体(Model)对象,保证了前端的干净。

运行起来,还是没有问题,仔细看DataTypeTemplateSelector对象的SelectTemplate
方法,还是有点丑陋,这里把CombinedEntity的TypeName属性硬编码,万一TypeName改成ControlName或其他名字,控
件则无法按照预期显示。

再次重构,首先修改绑定对象CombinedEntity


  public class CombinedEntity

    {

        /// <summary>

        /// 绑定数据的值

        /// </summary>

        public object CombinedValue

        {

            get;

            set;

        }

        /// <summary>

        /// 显示控件的类型

        /// </summary>

        public Type ControlType

        {

            get;

            set;

        }

    }

修改ListBox绑定数据源


 entities = new List<CombinedEntity>()

            {

                new CombinedEntity{ CombinedValue=new List<string>{"","",""}, ControlType = typeof(ComboBox)},

                new CombinedEntity{ CombinedValue ="Test", ControlType = typeof(TextBox)},

                new CombinedEntity{ CombinedValue=DateTime.Now, ControlType = typeof(DatePicker)}

            };

            this.DataContext = entities;

最后再次修改DataTypeTemplateSelector对象的SelectTemplate 方法


     public override DataTemplate SelectTemplate(object item, DependencyObject container)

        {

            Type t = item.GetType();

            Type controlType = null;

            PropertyInfo[] properties = t.GetProperties();

            foreach (PropertyInfo pi in properties)

            {

                if (pi.PropertyType == typeof(Type))

                {

                    controlType = (Type)pi.GetValue(item, null);

                    break;

                }

            }

            if (controlType == typeof(TextBox))

            {

                return TextBoxTemplate;

            }

            if (controlType == typeof(ComboBox))

            {

                return ComboBoxTemplate;

            }

            if (controlType == typeof(DatePicker))

            {

                return DateTimeTemplate;

            }

            return null;

        }

这样,要显示不同的控件,在ControlType里面定义即可,然后在XAML添加DataTemplate,在DataTemplateSelector对象中根据不同的ControlType返回不同的DataTemplate,而且实现的方式看上去比较优雅。

WPF:DataTemplateSelector设置控件不同的样式的更多相关文章

  1. WPF 4 DataGrid 控件(自定义样式篇)

    原文:WPF 4 DataGrid 控件(自定义样式篇)      在<WPF 4 DataGrid 控件(基本功能篇)>中我们已经学习了DataGrid 的基本功能及使用方法.本篇将继续 ...

  2. [转]设置控件全局显示样式appearance proxy

    转自:huifeidexin_1的专栏 appearance是apple在iOS5.0上加的一个协议,它让程序员可以很轻松地改变某控件的全局样式(背景) @selector(appearance) 支 ...

  3. 设置控件全局显示样式 appearance

    iOS5及其以后提供了一个比较强大的工具UIAppearance,我们通过UIAppearance设置一些UI的全局效果,这样就可以很方便的实现UI的自定义效果又能最简单的实现统一界面风格,它提供如下 ...

  4. WPF 定义Lookless控件的默认样式、 OnApplyTemplate 如何使用(实现方式、如何工作的)!

    写的非常详细: 作者地址:https://www.cnblogs.com/atskyline/archive/2012/11/16/2773806.html 参考资料: http://www.code ...

  5. WPF 4 DataGrid 控件(进阶篇一)

    原文:WPF 4 DataGrid 控件(进阶篇一)      上一篇<WPF 4 DataGrid 控件(自定义样式篇)>中,我们掌握了DataGrid 列表头.行表头.行.单元格相关的 ...

  6. WPF 4 DataGrid 控件(进阶篇二)

    原文:WPF 4 DataGrid 控件(进阶篇二)      上一篇<WPF 4 DataGrid 控件(进阶篇一)>中我们通过DataGridTemplateColumn 类自定义编辑 ...

  7. WPF设置控件获取键盘焦点时的样式FocusVisualStyle

    控件获取焦点除了用鼠标外,可以通过键盘来获取,比如Tab键或者方向键等,需要设置控件获取键盘焦点时的样式,可以通过设置FrameworkElemnt.FocusVisualStyle属性, 因为几乎所 ...

  8. WPF自定义分页控件,样式自定义,简单易用

    WPF自定义分页控件 做了许久伸手党,终于有机会贡献一波,搜索一下WPF分页控件,还是多,但是不太通用,主要就是样式问题,这个WPF很好解决,还有一个就是分页控件嘛,只关心几个数字的变动就行了,把页码 ...

  9. WPF Calendar 日历控件 样式自定义

    原文:WPF Calendar 日历控件 样式自定义 粗略的在代码上做了些注释 blend 生成出来的模版 有的时候 会生成 跟 vs ui界面不兼容的代码 会导致可视化设计界面 报错崩溃掉 但是确不 ...

随机推荐

  1. 用于展现图表的50种JavaScript库

    在很多项目中都会有在前端展现数据图表的需求,而在开发过程中,开发者往往会使用一些JavaScript库,从而更有效地达到想要的目标.最近,TechSlide上的一篇文章总结了50种用于展现图表的Jav ...

  2. Android 全屏方法

    我大概不想赘述什么其他方法,我就说一下我已知在用的方法QAQ requestWindowFeature(Window.FEATURE_NO_TITLE); 设置程序无标题栏 getWindow().s ...

  3. csv批量导入mysql命令

    今天把从Kaggle上下载下来的csv数据导入mysql,想做个统计分析,怎奈csv文件有些大.所以仅仅能用mysql 命令导入,现mark下,以备以后不时之需: 1. 导入: 基本的语法: load ...

  4. sql 时间和字符串 取到毫秒级

    (select replace(replace(replace(CONVERT(varchar, getdate(), 120 ),'-',''),' ',''),':','')+(Select ri ...

  5. SqlBulkCopy的使用

    1.问题:导入大数据量到数据库,用我们普通的SqlHelper来做是每插入一条都是打开连接关闭连接,这样太慢,因此我们会想到让SqlConnection一直打开直到所有数据都插入完成再关闭连接.但是根 ...

  6. Informatica 9.5.1 安装配置

    Informatica  结构 1个或多个资源库(Respository) PowerCenter数据整合引擎是基于元数据驱动的,提供了基于数据驱动的元数据知识库(Repository),该元数据知识 ...

  7. iOS开发 点击跳转到App Store 或者 点击按钮去评价

    //跳转到应用页面 NSString *str = [NSString stringWithFormat:@"http://itunes.apple.com/us/app/id%d" ...

  8. MAVEN入门(二)

    一.IDEA+MAVEN+Tomcat7 创建一个简单的Web app 1.用IDEA创建一个maven项目 注意: 红色部分一定要自己手选本地配置好的maven_home的地址,否则IDEA会选用内 ...

  9. python之高阶函数编程

    在这篇文章中我指出两点: 第一:系统函数可以被覆盖 比如: a=abs(-10) print a 10 但是,如果把函数本身赋值给变量呢? f = abs f <built-in functio ...

  10. BZOJ 1898: [Zjoi2004]Swamp 沼泽鳄鱼( dp + 矩阵快速幂 )

    ----------------------------------------------------------------------- #include<cstdio> #incl ...