WPF:DataTemplateSelector设置控件不同的样式
原文 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设置控件不同的样式的更多相关文章
- WPF 4 DataGrid 控件(自定义样式篇)
原文:WPF 4 DataGrid 控件(自定义样式篇) 在<WPF 4 DataGrid 控件(基本功能篇)>中我们已经学习了DataGrid 的基本功能及使用方法.本篇将继续 ...
- [转]设置控件全局显示样式appearance proxy
转自:huifeidexin_1的专栏 appearance是apple在iOS5.0上加的一个协议,它让程序员可以很轻松地改变某控件的全局样式(背景) @selector(appearance) 支 ...
- 设置控件全局显示样式 appearance
iOS5及其以后提供了一个比较强大的工具UIAppearance,我们通过UIAppearance设置一些UI的全局效果,这样就可以很方便的实现UI的自定义效果又能最简单的实现统一界面风格,它提供如下 ...
- WPF 定义Lookless控件的默认样式、 OnApplyTemplate 如何使用(实现方式、如何工作的)!
写的非常详细: 作者地址:https://www.cnblogs.com/atskyline/archive/2012/11/16/2773806.html 参考资料: http://www.code ...
- WPF 4 DataGrid 控件(进阶篇一)
原文:WPF 4 DataGrid 控件(进阶篇一) 上一篇<WPF 4 DataGrid 控件(自定义样式篇)>中,我们掌握了DataGrid 列表头.行表头.行.单元格相关的 ...
- WPF 4 DataGrid 控件(进阶篇二)
原文:WPF 4 DataGrid 控件(进阶篇二) 上一篇<WPF 4 DataGrid 控件(进阶篇一)>中我们通过DataGridTemplateColumn 类自定义编辑 ...
- WPF设置控件获取键盘焦点时的样式FocusVisualStyle
控件获取焦点除了用鼠标外,可以通过键盘来获取,比如Tab键或者方向键等,需要设置控件获取键盘焦点时的样式,可以通过设置FrameworkElemnt.FocusVisualStyle属性, 因为几乎所 ...
- WPF自定义分页控件,样式自定义,简单易用
WPF自定义分页控件 做了许久伸手党,终于有机会贡献一波,搜索一下WPF分页控件,还是多,但是不太通用,主要就是样式问题,这个WPF很好解决,还有一个就是分页控件嘛,只关心几个数字的变动就行了,把页码 ...
- WPF Calendar 日历控件 样式自定义
原文:WPF Calendar 日历控件 样式自定义 粗略的在代码上做了些注释 blend 生成出来的模版 有的时候 会生成 跟 vs ui界面不兼容的代码 会导致可视化设计界面 报错崩溃掉 但是确不 ...
随机推荐
- php 前台数据显示
<pre name="code" class="html"> public function show(){ echo "访问了index ...
- CentOS 6.4下Squid代理服务器的安装与配置,反向代理
CentOS 6.4下Squid代理服务器的安装与配置 一.简介 代理服务器英文全称是Proxy Server,其功能就是代理网络用户去取得网络信息. Squid是一个缓存Internet 数据的软件 ...
- Noip2009提高组总结
Noip2009的题目还是有一定难度的,主要是搜索和最短路都是我的弱项,不检查第一遍下来只做了150分,还是这句话,素质和读题的仔细程度决定了分数.仔细想想,我们化学老师说的话没错,或许题目你都会做, ...
- mac 下安装和卸载 mysql
这里有一篇文章写得很详细: http://www.cnblogs.com/macro-cheng/archive/2011/10/25/mysql-001.html 关于卸载,我也百度了下.找了好几个 ...
- JAVA GUI学习 - JTable表格组件学习_A ***
public class JTableKnow_A extends JFrame { public JTableKnow_A() { this.setBounds(300, 100, 400, 300 ...
- 工程中.pch文件的作用 及使用方法
#ifdef __OBJC__ #define ABC 10#import "UIImage+Image.h"// 配置pch: buildSetting -> prefix ...
- FastReport配置打印预览button_C++
如需转载请标明出处:http://blog.csdn.net/itas109 FastReport採用C++方式.配置打印预览选项 //打印预览 //配置打印预览选项 pReport->Prev ...
- java中常见的单例模式详解
很多求职者在面试过程中都被问到了单例模式,最常见的问题,比如,每种单例模式的区别是什么?哪些模式是线程安全的?你们项目里用的哪种单例模式?原来没有注意这个问题,回来赶紧打开项目查看了一下代码,才发现我 ...
- 使用PowerShell 命令集进行SQL Server 2012 备份和还原
最近心相不错,所以打算翻译一些英文文档做福利,原文在此,翻译有不足的地方还请各位兄弟指点. 讨论什么是DBA最重要的工作的时候,你最常听到就是一条就是DBA只要做好备份和恢复.事实如此,如果你不做备份 ...
- lightOJ 1317 Throwing Balls into the Baskets
lightOJ 1317 Throwing Balls into the Baskets(期望) 解题报告 题目链接:http://acm.hust.edu.cn/vjudge/contest/ ...