问题背景

某一天,我想做一个签到打卡的日历。基于 Calendar,想实现这个目标,于是找到了它的 SelectedDates 属性,用于标记签到过的日期。

问题来了。

基于MVVM模式,想将其在xaml中绑定到ViewModel中的一个ObservableCollection属性。但 SelectedDates只读CLR 属性。

解决思路:

给它搭个桥:创建一个相同Type的附加属性,用附加属性绑定到ViewModel中的ObservableCollection属性。并在附加属性的变更通知和集合变更通知中添加处理:进行操作的桥接,将集合对象、集合的变动传递到SelectedDates中即可。

主要代码如下:

public static class CalendarEx
{
#region SelectedDates 只读,不可在 Xaml 中绑定的问题解决方案:建立附加属性通道
public static ObservableCollection<DateTime> GetSelectedDatesSource(DependencyObject obj)
{
return (ObservableCollection<DateTime>)obj.GetValue(SelectedDatesSourceProperty);
}
public static void SetSelectedDatesSource(DependencyObject obj, ObservableCollection<DateTime> value)
{
obj.SetValue(SelectedDatesSourceProperty, value);
}
public static readonly DependencyProperty SelectedDatesSourceProperty =
DependencyProperty.RegisterAttached("SelectedDatesSource", typeof(ObservableCollection<DateTime>), typeof(CalendarEx), new PropertyMetadata(null, SetSelectedDatesSourcePropertyChangedCallback));
/// <summary>
/// REMARK 只读集合的 XAML 绑定(目前是单向绑定,双向绑定需定义 SelectedDates 的变更事件)
/// </summary>
/// <param name="d"></param>
/// <param name="args"></param>
private static void SetSelectedDatesSourcePropertyChangedCallback(DependencyObject d, DependencyPropertyChangedEventArgs args)
{
// 附加属性应用于 Calendar
Calendar calendar = d as Calendar;
if (calendar == null) return;
// 对象本身未变更,则不进行任何操作,避免相同对象多次添加变更事件
if (args.NewValue == args.OldValue) return;
// 变更 SelectedDatesSource 对象,先获取新集合对象的集合内容
ObservableCollection<DateTime> newValue = args.NewValue as ObservableCollection<DateTime>;
if (newValue == null) return; // 应用新内容
calendar.SelectedDates.Clear();
foreach (DateTime time in newValue)
{
calendar.SelectedDates.Add(time);
}
// 获取 SelectedDatesSource 对象,添加集合变更通知处理,在其中处理目标集合对象
ObservableCollection<DateTime> sourceCollection = GetSelectedDatesSource(d);
if (sourceCollection == null) return;
// _NOTE 这里使用 local 函数的目的是,集合变更通知函数(静态)需要访问 calendar,使用 local/匿名 函数方便访问,否则需要定义一个根据集合访问 calendar 的服务
// 另外,local 函数比匿名函数好的地方在于,可以取消事件订阅(经验证)
void SelectedDatesOnCollectionChanged(object sender, NotifyCollectionChangedEventArgs cArgs)
{
ObservableCollection<DateTime> collection = sender as ObservableCollection<DateTime>;
if (collection == null) return;
switch (cArgs.Action)
{
case NotifyCollectionChangedAction.Add:
foreach (DateTime item in cArgs.NewItems)
{
calendar.SelectedDates.Add(item);
}
break;
case NotifyCollectionChangedAction.Remove:
foreach (DateTime item in cArgs.NewItems)
{
calendar.SelectedDates.Remove(item);
}
break;
case NotifyCollectionChangedAction.Replace:
foreach (DateTime item in cArgs.OldItems)
{
calendar.SelectedDates.Remove(item);
}
foreach (DateTime item in cArgs.NewItems)
{
calendar.SelectedDates.Add(item);
}
break;
case NotifyCollectionChangedAction.Move:
// ignored
break;
case NotifyCollectionChangedAction.Reset:
// ignored
break;
default:
break;
}
}
// 这里
sourceCollection.CollectionChanged -= SelectedDatesOnCollectionChanged;
sourceCollection.CollectionChanged += SelectedDatesOnCollectionChanged;
}
#endregion
}

使用起来就简单了:

<Calender Grid.Column="0" x:Name="Calendar" Width=260" Heigh="270"
SelectionMode="MultipleRange" lui:CalendarEx.SelectedDatesSource="{Binding Checkins}"
lui:BehaviorRepo.ReleaseStylusCaptureCalendar="True"
/>

WPF 只读集合在 XAML 中的绑定(WPF:Binding for readonly collection in xaml)的更多相关文章

  1. WPF 的 ElementName 在 ContextMenu 中无法绑定成功?试试使用 x:Reference!

    在 Binding 中使用 ElementName 司空见惯,没见它出过什么事儿.不过当你预见 ContextMenu,或者类似 Grid.Row / Grid.Column 这样的属性中设置的时候, ...

  2. WPF绑定xaml中绑定对象需用属性表示,字段不可以绑定

    在练习WPF绑定时发现对象属性可以在XAML中绑定,但字段是不可以绑定: 比如: private Person person{get;set;}  可以绑定到XAML中,<TextBox Nam ...

  3. 整理:WPF中Xaml中绑定枚举的写法

    原文:整理:WPF中Xaml中绑定枚举的写法 目的:在Combobox.ListBox中直接绑定枚举对象的方式,比如:直接绑定字体类型.所有颜色等枚举类型非常方便 一.首先用ObjectDataPro ...

  4. ComboBox在WPF中的绑定示例:绑定项、集合、转换,及其源代码

    示例1.Selector(基类) 的示例Controls/SelectionControl/SelectorDemo.xaml <Page x:Class="Windows10.Con ...

  5. WPF Xaml中创建集合

    首先在xaml中创建集合是一个不可取的方法. 本方法仅作为xaml的学习. 本文略微无聊,主要是编写的东西都是老玩意. 首先是定义一个类,作为你要加载集合的模型. 结构如下 internal clas ...

  6. WPF 中双向绑定通知机制之ObservableCollection使用

    msdn中   ObservableCollection<T> 类    表示一个动态数据集合,在添加项.移除项或刷新整个列表时,此集合将提供通知. 在许多情况下,所使用的数据是对象的集合 ...

  7. WPF ContextMenu 在MVVM模式中绑定 Command及使用CommandParameter传参

    原文:WPF ContextMenu 在MVVM模式中绑定 Command及使用CommandParameter传参 ContextMenu无论定义在.cs或.xaml文件中,都不继承父级的DataC ...

  8. 封装:WPF中可以绑定的BindPassWord控件

    原文:封装:WPF中可以绑定的BindPassWord控件 一.目的:本身自带的PassWord不支持绑定 二.Xaml部分 <UserControl x:Class="HeBianG ...

  9. 使用异步方法在XAML中绑定系统时间

    最近的工作需要在程序界面上显示实时的系统时间,网上查了查大部分都是用Timer或者线程来实现. 个人非常不喜欢用Timer,感觉这东西有点太耗资源,然后思考了下觉得更好的方法应该是使用异步的方法在委托 ...

随机推荐

  1. 【数据库】MySQL & SQL 介绍

    文章目录 MySQL & SQL 介绍 1.MySQL的背景 2.MySQL的优点 3.MySQL的安装 4.MySQL服务的启动和停止 方式一 方式二 5.MySQL服务的登录和退出 方式一 ...

  2. 修改机器的hostname

    vi /etc/sysconfig/network hostname=你想设置的主机名 不重启器的情况下使显示名称变成 hostname  主机名

  3. Kioptix Level 1

    1. 简介 Vulnhub是一个提供各种漏洞环境的靶场平台. 个人学习目的:1,方便学习更多类型漏洞.2,为OSCP做打基础. 下载链接 https://www.vulnhub.com/entry/k ...

  4. spring boot项目问题汇总

    spring遇到的问题汇总 有关日志的打印和日志如何使用 在实际项目中,我们的程序都是运行在linux上,有错误时也不能在本地的控制台上直观看到,所有合理打印日志对于程序员迅速定位到错误. 打印日志时 ...

  5. POJ1629:picnic planning

    题目描述 矮人虽小却喜欢乘坐巨大的轿车,轿车大到可以装下无论多少矮人.某天,N(N≤20)个矮人打算到野外聚餐.为了 集中到聚餐地点,矮人A 有以下两种选择 1)开车到矮人B家中,留下自己的轿车在矮人 ...

  6. java 日期与时间操作

    我们先来了解一下基本的概念 日期 2020-11-21 2020-11-22 时间 15:36:43 2020-11-21 15:36:43 时区 北京时间 2020-11-21 15:36:43 东 ...

  7. websocket的应用---Django

    websocket的应用---Django 1.长轮询 轮询:在前端通过写js实现.缺点:有延迟.服务器压力大. 就是客户端通过一定的时间间隔以频繁请求的方式向服务器发送请求,来保持客户端和服务器端的 ...

  8. MySQL全面瓦解21(番外):一次深夜优化亿级数据分页的奇妙经历

    背景 1月22号晚上10点半,下班后愉快的坐在在回家的地铁上,心里想着周末的生活怎么安排. 突然电话响了起来,一看是我们的一个开发同学,顿时紧张了起来,本周的版本已经发布过了,这时候打电话一般来说是线 ...

  9. 使用cacti监控linux主机

    介绍:使用cacti监控linux主机,需要在linux主机上面安装snmp服务,并修改snmpd.conf文件,指定cacti服务器的地址,然后在cacti的前台界面添加此主机即可,此处以监控cen ...

  10. Redis 实战 —— 09. 实现任务队列、消息拉取和文件分发

    任务队列 P133 通过将待执行任务的相关信息放入队列里面,并在之后对队列进行处理,可以推迟执行那些耗时对操作,这种将工作交给任务处理器来执行对做法被称为任务队列 (task queue) . P13 ...