这世上,没人能一次性写出完美无缺的框架;因为,任何一个框架都需要项目的淬炼,然后才能升华,趋近完美。

所以,框架是个反复修改的东西,最终形成的东西。

如果你学了一点技术,觉得自己可以写出框架了,觉得自己有架构师的能力,然而自己总是怀才不遇——那一定是你的错觉。

因为,你框架没有经过项目淬炼;而淬炼过框架的人都了解,设计的再好的框架,最终会被业务需求打的细碎,然后被开发人员搅和再一起。

所以细节决定成败,没有细节的框架就是扯淡。

DataControl—数据控件

上文我们已经编写出来了WPF的MVVM基础框架,但为了让他更加强壮,为了让他多坚持一阵子再粉碎,我们要让ViewModel更强壮,所以我们要编写[数据控件]。

数据控件其实很好理解,它就是把UI控件中存储的数据提取出来,好让ViewModel可以通过修改数据来控制UI变化;当然,为了更好的控制UI变化,数据控件里还得包含一点管理UI的属性。

因为WPF里的控件大多继承自Control,所以我们先创建Control的数据控件。

  1. public class Control<T> : INotifyPropertyChanged
  2. {
  3. public event PropertyChangedEventHandler PropertyChanged;
  4.  
  5. public T _DataContent ;
  6. public T DataContent { get { return _DataContent; } set { _DataContent = value; OnPropertyChanged(); } }
  7.  
  8. public Visibility _Visibility;
  9. public Visibility Visibility { get { return _Visibility; } set { _Visibility = value; OnPropertyChanged(); } }
  10.  
  11. public bool _IsReadOnly;
  12. public bool IsReadOnly { get { return _IsReadOnly; } set { _IsReadOnly = value; OnPropertyChanged(); } }
  13.  
  14. public bool _IsEnabled;
  15. public bool IsEnabled { get { return _IsEnabled; } set { _IsEnabled = value; OnPropertyChanged(); } }
  16.  
  17. protected void OnPropertyChanged([CallerMemberName]string propertyName = "")
  18. {
  19. if (PropertyChanged != null)
  20. {
  21. PropertyChanged(this, new PropertyChangedEventArgs(propertyName));
  22. }
  23. }
  24. }

如上代码所示,我们创建了Control的数据控件。

可以看到,处理存贮数据的DataContent属性之外,还创建了一些管理UI的属性IsEnabled、IsReadOnly、Visibility。

父类数据控件创建完成后,我们开始创建子类的数据控件。[如果子类要管理的UI属性不在父类内,我们就需要额外创建一些]

TextBlock和TextBox

我们先创建最基础的,最常用的TextBlock和TextBox。

TextBlock代码如下:

  1. public class TextBlock<T> : Control<T>
  2. {
  3. public T _Text;
  4. public T Text
  5. {
  6. get { return _Text; }
  7. set
  8. {
  9. _Text = value;
  10. OnPropertyChanged();
  11. }
  12. }
  13. }

TextBox代码如下:

  1. public class TextBox<T> : Control<T>
  2. {
  3. public Action<T> TextChangeCallBack = null;
  4.  
  5. public T _Text;
  6. public T Text {
  7. get { return _Text; }
  8. set
  9. {
  10. _Text = value;
  11. if (TextChangeCallBack != null)
  12. {
  13. TextChangeCallBack(_Text);
  14. }
  15. OnPropertyChanged();
  16. }
  17. }
  18. }

可以看到TextBlock和TextBox都继承了Control,而他们的区别只是TextBox多了一个TextChangeCallBack。

有人会想到,那完全可以用TextBox替代TextBlock。

理论上,TextBlock是可以被替换,但为了程序清晰,还是区别开来更好。

控件定义好了,我们现在看一下如何应用。

TextBox应用

  1. xaml页面代码
  2. <TextBox Text="{Binding ChangeTextBox.Text,Mode=TwoWay}" Margin="5" FontSize="12"></TextBox>
  3. ----------------------------------
  4. ViewModel页面代码
  5. public TextBox<string> ChangeTextBox { get; set; }
  6. public VM_PageTextBox()
  7. {   ChangeTextBox = new TextBox<string>();
  8. ChangeTextBox.TextChangeCallBack = (text) => { MessageBox(text); };//声明TextChange
  9. }

如代码所示,我们在ViewModel中定义了ChangeTextBox属性,然后再Xaml中绑定了ChangeTextBox属性的Text到UI控件TextBox的Text属性上,这样我们就实现了数据联动。

并且代码中实例化了TextChangeCallBack委托,那么当Text数据变化时,该委托就会触发。

注意:TextChangeCallBack委托与TextChanged事件不同,并不是每次修改文字都会触发,而是当TextBox的Text内容真正被修改时,才会触发;我们可以简单的理解为TextBox失去焦点时才会触发。

这里只介绍TextBox应用,TextBlock应用就不介绍了,因为使用方式和TextBox一样。

如果想了解更多数据控件的应用,请去GitHub下载源码。

ComboBox

ComboBox稍微复杂一点,因为他多了一个ItemSource属性。

我们先看ComboBox的数据控件代码:

  1. public class ComboBox<T> : Control<T>
  2. {
  3. public Action<T> SelectCallBack = null;
  4. public ComboBox()
  5. {
  6.  
  7. }
  8. public ObservableCollection<T> _ItemsSource;
  9. public ObservableCollection<T> ItemsSource
  10. {
  11. get
  12. {
  13. return _ItemsSource;
  14. }
  15. set
  16. {
  17. _ItemsSource = value;
  18. if (_ItemsSource != null && _ItemsSource.Count > 0 && SelectedItem == null)
  19. {
  20. SelectedItem = _ItemsSource.First();
  21. }
  22. OnPropertyChanged();
  23. }
  24. }
  25. public T _SelectedItem;
  26. public T SelectedItem
  27. {
  28. get { return _SelectedItem; }
  29. set
  30. {
  31. _SelectedItem = value;
  32. if (SelectCallBack != null)
  33. {
  34. SelectCallBack(_SelectedItem);
  35. }
  36. OnPropertyChanged();
  37. }
  38. }
  39. private ICollectionView _ItemsSourceView;
  40. public ICollectionView ItemsSourceView
  41. {
  42. get
  43. {
  44. _ItemsSourceView = CollectionViewSource.GetDefaultView(_ItemsSource);
  45. return _ItemsSourceView;
  46. }
  47. set
  48. {
  49. _ItemsSourceView = value;
  50. OnPropertyChanged();
  51. }
  52. }
  53. public void SetItemsSource(List<T> itemSource)
  54. {
  55. ItemsSource = new ObservableCollection<T>(itemSource);
  56. }
  57. }

代码相对简单,SelectedItem和ItemsSource用来绑定UI控件ComboBox的同名属性。

ItemsSourceView:ItemsSourceView属性可能有些难理解,这里我们简单介绍一下。

因为WPF的UI控件被创建以后,要被添加到视觉树中,所以最终会被显示在屏幕上的是包裹着控件的视觉树;其中视觉树与控件是可以分离的;比如控件中绑定的数据是10行,而视觉树可以显示3行。

为了管理视觉树,我们创建了ItemsSourceView属性。

因为ItemsSourceView是ICollectionView类型,所以ItemsSourceView可以处理排序、筛选和分组。[有兴趣的同学可以自行了解下ICollectionView类型]

感觉这样描述还是很难理解,让我们一起在应用中慢慢理解吧。

ObservableCollection:我们可以看到ItemsSource是类型是ObservableCollection,而不是List。为什么要用ObservableCollection呢?

很简单,因为ObservableCollection继承了INotifyCollectionChanged,即,数据控件进行[行]的增删,也会让UI进行[行]的增删。

ComboBox应用

在应用之前,我们先在Proxy建立一个获取数据是代理。

创建获取数据的方法如下:

  1. public List<User> GetComboBoxData()
  2. {
  3. List<User> userList = new List<User>();
  4. User user1 = new User() { Id = 1, Name = "张三", Age = 11 };
  5. userList.Add(user1);
  6. return userList;
    }

Xaml页面代码如下:

  1. <ComboBox Margin="5" Width="200" FontSize="12" ItemsSource="{Binding TestComboBox.ItemsSource}" DisplayMemberPath="Name" SelectedValuePath="Id" SelectedItem="{Binding TestComboBox.SelectedItem}" ></ComboBox>

ViewModel代码如下:

  1. public ComboBox<User> TestComboBox { get; set; }
  2. TestDataProxy proxy = new TestDataProxy();
  3. public VM_PageComboBox()
  4. {
  5. TestComboBox = new ComboBox<User>();
  6. TestComboBox.SetItemsSource(proxy.GetComboBoxData());
  7. TestComboBox.SelectCallBack = (user) => {
  8. MessageBox(user.Name);
  9. };
  10. }

如上所示,我们已经实行了在ViewModel中管理ComboBox。

----------------------------------------------------------------------------------------------------

本篇文章就先讲到这了,下一篇文章我们将一起为框架编写DataGrid数据控件。

因为DataGrid数据控件是所有数据控件中最复杂的,而且代码量特别多;所以,我决定,单拿出一篇来介绍DataGrid。

框架代码已经传到Github上了,并且会持续更新。

相关文章:

【我们一起写框架】MVVM的WPF框架(一)—序篇

【我们一起写框架】MVVM的WPF框架(二)—绑定

To be continued——DataGrid

Github地址:https://github.com/kiba518/KibaFramework

----------------------------------------------------------------------------------------------------

注:此文章为原创,任何形式的转载都请联系作者获得授权并注明出处!
若您觉得这篇文章还不错,请点击下方的【推荐】,非常感谢!

【我们一起写框架】MVVM的WPF框架(三)—数据控件的更多相关文章

  1. CYQ.Data 支持WPF相关的数据控件绑定.Net获取iis版本

    CYQ.Data 支持WPF相关的数据控件绑定(2013-08-09) 事件的结果 经过多天的思考及忙碌的开发及测试,CYQ.Data 终于在UI上全面支持WPF,至此,CYQ.Data 已经可以方便 ...

  2. CYQ.Data 支持WPF相关的数据控件绑定(2013-08-09)

    事件的结果 经过多天的思考及忙碌的开发及测试,CYQ.Data 终于在UI上全面支持WPF,至此,CYQ.Data 已经可以方便支持wpf的开发,同时,框架仍保留最低.net framework2.0 ...

  3. Jquery如何序列化form表单数据为JSON对象 C# ADO.NET中设置Like模糊查询的参数 从客户端出现小于等于公式符号引发检测到有潜在危险的Request.Form 值 jquery调用iframe里面的方法 Js根据Ip地址自动判断是哪个城市 【我们一起写框架】MVVM的WPF框架(三)—数据控件 设计模式之简单工厂模式(C#语言描述)

    jquery提供的serialize方法能够实现. $("#searchForm").serialize();但是,观察输出的信息,发现serialize()方法做的是将表单中的数 ...

  4. 【我们一起写框架】MVVM的WPF框架(五)—完结篇

    前言 这篇文章是WPF框架系列的最后一篇,在这里我想阐述一下我对框架设计的理解. 我对框架设计的理解是这样的: 框架设计不应该局限于任何一种设计模式,我们在设计框架时,应该将设计模式揉碎,再重组:这样 ...

  5. 【我们一起写框架】MVVM的WPF框架(一)—序篇

    前言 我想,有一部分程序员应该是在二三线城市的,虽然不知道占比,但想来应该不在少数. 我是这部分人群中的一份子. 我们这群人,面对的客户,大多是国内中小企业,或者政府的小部门.这类客户的特点是,资金有 ...

  6. 【我们一起写框架】MVVM的WPF框架(二)—绑定

    MVVM的特点之一是实现数据同步,即,前台页面修改了数据,后台的数据会同步更新. 上一篇我们已经一起编写了框架的基础结构,并且实现了ViewModel反向控制Xaml窗体. 那么现在就要开始实现数据同 ...

  7. 【我们一起写框架】MVVM的WPF框架(四)—DataGrid

    前言 这个框架写到这里,应该有很多同学发现,框架很多地方的细节,其实是违背了MVVM的设计逻辑的. 没错,它的确是违背了. 但为什么明知道违背设计逻辑,还要这样编写框架呢? 那是因为,我们编写的是框架 ...

  8. 在DWZ框架中整合kindeditor复文本框控件

    今天上午在DWZ框架中整合kindeditor复文本框控件,发现上传图片是老是提示 “上传中,请稍候...”,上网查看别人说可能是文件路径问题,在想以前在其他项目中用这个控件一直没问题,到这里怎么会出 ...

  9. 【案例分享】在 React 框架中使用 SpreadJS 纯前端表格控件

    [案例分享]在 React 框架中使用 SpreadJS 纯前端表格控件 本期葡萄城公开课,将由国电联合动力技术有限公司,资深前端开发工程师——李林慧女士,与大家在线分享“在 React 框架中使用 ...

随机推荐

  1. 微信小程序分享到朋友圈方法与技巧

    小程序提供onShareAppMessage 函数,此函数只支持分享给我微信朋友.小程序如何分享到朋友圈呢? 我提供的方法是,使用canvas绘制一张图片,并用wx.previewImage预览图片, ...

  2. BZOJ_1485_[HNOI2009]有趣的数列_卡特兰数

    BZOJ_1485_[HNOI2009]有趣的数列_卡特兰数 Description 我们称一个长度为2n的数列是有趣的,当且仅当该数列满足以下三个条件: (1)它是从1到2n共2n个整数的一个排列{ ...

  3. BZOJ_3427_Poi2013 Bytecomputer_DP

    BZOJ_3427_Poi2013 Bytecomputer_DP Description 给定一个{-1,0,1}组成的序列,你可以进行x[i]=x[i]+x[i-1]这样的操作,求最少操作次数使其 ...

  4. LoadRunner接口测试方法

    实际上到目前为止,我所做过的几个关于性能测试的项目,都是在UI页面上能正常访问结束所有的前期功能测试而开始的性能测试.但loadrunner不仅仅只能靠录制回放修改脚本进行性能测试. 往往需要对发送报 ...

  5. visual studio2015中开发python

    之前下载了visual studio2017但是发现很不好用,不如使用matlab与visual studio混合编程就根本找不到visual studio,只有下了visual studio2015 ...

  6. CentOS7搭建本地YUM仓库,并定期同步阿里云源

    CentOS7同步阿里云镜像rpm包并自建本地yum仓库 系统环境 # cat /etc/centos-release CentOS Linux release 7.6.1810 (Core) # u ...

  7. 从壹开始 [ Nuxt.js ] 之一 || 为开源收录Bug之 TiBug项目 开篇讲

    缘起 哈喽大家周二好呀,刚刚经历过了几天火车抢票,整个人都不好了,不知道小伙伴对今年的春节是否还一如既往的期待呢,眼看都要春节了,本来也想写篇2018总结篇,但是怕不免会出现鸡汤文的窠臼嫌疑,想想还是 ...

  8. 前端笔记之NodeJS(四)MongoDB数据库&Mongoose&自制接口&MVC架构思想|实战

    一.MongoDB数据库 1.1 NoSQL简介 随着互联网web2.0网站的兴起,传统的SQL数据库(关系数据库)在应付web2.0网站,特别是超大规模和高并发的SNS(social network ...

  9. 《k8s-1.13版本源码分析》-调度预选

    本文大纲 预选流程 predicate的并发 一个node的predicate predicates的顺序 单个predicate执行过程 具体的predicate函数 本系列文章已经开源到githu ...

  10. Dynamics 365 CE在Pre Delete插件中应用Image

    微软动态CRM专家罗勇 ,回复327或者20190428可方便获取本文,同时可以在第一间得到我发布的最新博文信息,follow me! 在插件中限制记录的删除是常见的场景,比如根据statuscode ...