(资料图片)
背景业务开发过程中遇到一个日期范围选择的需求,和Element UI的DateTimePicker组件比较类似,由两个日历控件组成,联动选择起始时间和结束时间。
问题WPF中提供了一个DatePicker
的控件,主要由DatePickerTextBox
、Button
和一个Calendar
组成,其中Calendar
是后台代码动态添加的,因此不能直接通过自定义DatePicker
的控件模板实现需求。这里通过实现自定义DateTimePicker
控件来满足需求。
由于Calendar
结构比较复杂,本文通过控件组合的方式简单实现自定义DateTimePicker
。先来看下实现效果。
首先创建一个名为DateTimePicker
的UserControl,添加依赖属性HoverStart
和HoverEnd
用于控制日历中的开始日期和结束日期,添加依赖属性DateTimeRangeStart
和DateTimeRangeEnd
用于设置外部设置/获取起始时间和结束时间。
然后在XAML中添加两个WatermarkTextBox
用于输入起始时间和结束时间(增加校验规则验证时间的合法性,这里不再详细说明如何写校验规则,具体可参考ValidationRule实现参数绑定)。接着添加一个Popup
(默认关闭),并在其中添加两个Calendar
用于筛选日期,以及四个ComboBox
用于筛选小时和分钟。当WatermarkTextBox
捕获到鼠标时触发Popup
打开。
紧接着就是修改Calendar
的样式了。通常情况下,自定义控件模板只需要在Visual Studio的设计窗口或者Blend中选中控件,然后右键菜单中编辑模板即可。可能由于Calendar中的部分元素(CalendarButton
和CalendarDayButton
)是后台代码生成,这个方法编辑Calendar
模板副本生成的CalendarStyle
不包含完整的可视化树结构,无法对样式进一步修改。幸运的是微软官方文档公开了控件的默认样式和模板,在此基础上进行修改即可。通过官方文档可以发现Calendar
完整的可视化树中包含了四个类型控件Calendar
、CalendarItem
、CalendarButton
、CalendarDayButton
。其中CalendarDayButton
对应的就是日历中具体的“天”,管理着具体的“天”的状态,比如选中状态、不可选状态等,这也是我们主要修改的地方,接下来看下CalendarDayButton
的样式。(其他几个元素的样式和模板参照官方文档修改即可)
样式中用到一个MultiBinding
绑定CalendarDayButton
以及前边提到的两个依赖属性:HoverStart
和HoverEnd
,然后通过MultiValueConverter
转换器比较CalendarDayButton
是否处于选中的日期范围,根据不同的状态设置其背景色和字体颜色。
最后就是在后台代码中根据日历的SelectedDatesChanged
事件设置HoverStart
和HoverEnd
的值,以此来控制DateTimePicker
中选中日期的样式。
本文分享了一种简单实现自定义DateTimePicker
控件的方式,同时也介绍了另外一种查看原生控件默认样式和模板的方法:查看微软官方文档。这种方法虽然不如在Visual Studio的设计窗口或者Blend中编辑模板副本方便,但提供了完整的结构、每个元素的组成部分以及可视化状态,方便开发人员清晰的了解控件全貌,可以应对修改复杂的原生控件样式和模板的需求。
Copyright © 2015-2022 澳门网版权所有 备案号:京ICP备2022022245号-14 联系邮箱:435 226 40@qq.com