Messenger在MVVM模式中的应用

Messenger在MVVM中应用的前提

我们知道在MVVM架构中,系统平台的Silverlight客户端界面开发和业务逻辑已经被分开,XAML是SL的主要部分,界面设计者只需要绑定ViewModel里的数据即可。但是在ViewModel里有些时候是需要界面发出响应的,在这种情况下,Messenger显示出它的用处。

Messenger的架构

Messager构件代码

定义Imessager接口

注册一个接收消息的类型,比如某一控件来接收消息

void Register<TMessage>(object recipient, Action<TMessage> action);

给最近注册的类型发送消息

void Send<TMessage, TTarget>(TMessage message);

取消最近注册的类型,当执行该句后,不再接收任何消息

void Unregister(object recipient);

实现Imessager接口

public class Messenger : IMessenger

{

private static Messenger _defaultInstance;

private Dictionary<Type, List<WeakActionAndToken>> _recipientsOfSubclassesAction;

private Dictionary<Type, List<WeakActionAndToken>> _recipientsStrictAction;

public static Messenger Default

{

get

{

if (_defaultInstance == null)

{

_defaultInstance = new Messenger();

}

return _defaultInstance;

}

}

public static void OverrideDefault(Messenger newMessenger)

{

_defaultInstance = newMessenger;

}

public static void Reset()

{

_defaultInstance = null;

}

public virtual void Register<TMessage>(object recipient, Action<TMessage> action)

{

Register(recipient, null, false, action);

}

public virtual void Register<TMessage>(object recipient, bool receiveDerivedMessagesToo, Action<TMessage> action)

{

Register(recipient, null, receiveDerivedMessagesToo, action);

}

public virtual void Register<TMessage>(object recipient, object token, Action<TMessage> action)

{

Register(recipient, token, false, action);

}

public virtual void Register<TMessage>(

object recipient,

object token,

bool receiveDerivedMessagesToo,

Action<TMessage> action)

{

var messageType = typeof(TMessage);

Dictionary<Type, List<WeakActionAndToken>> recipients;

if (receiveDerivedMessagesToo)

{

if (_recipientsOfSubclassesAction == null)

{

_recipientsOfSubclassesAction = new Dictionary<Type, List<WeakActionAndToken>>();

}

recipients = _recipientsOfSubclassesAction;

}

else

{

if (_recipientsStrictAction == null)

{

_recipientsStrictAction = new Dictionary<Type, List<WeakActionAndToken>>();

}

recipients = _recipientsStrictAction;

}

List<WeakActionAndToken> list;

if (!recipients.ContainsKey(messageType))

{

list = new List<WeakActionAndToken>();

recipients.Add(messageType, list);

}

else

{

list = recipients[messageType];

}

var weakAction = new WeakAction<TMessage>(recipient, action);

var item = new WeakActionAndToken

{

Action = weakAction,

Token = token

};

list.Add(item);

Cleanup();

}

public virtual void Send<TMessage>(TMessage message)

{

SendToTargetOrType(message, null, null);

}

[SuppressMessage(

"Microsoft.Design",

"CA1004:GenericMethodsShouldProvideTypeParameter",

Justification = "This syntax is more convenient than other alternatives.")]

public virtual void Send<TMessage, TTarget>(TMessage message)

{

SendToTargetOrType(message, typeof(TTarget), null);

}

public virtual void Send<TMessage>(TMessage message, object token)

{

SendToTargetOrType(message, null, token);

}

/// <summary>

public virtual void Unregister(object recipient)

{

UnregisterFromLists(recipient, _recipientsOfSubclassesAction);

UnregisterFromLists(recipient, _recipientsStrictAction);

}

[SuppressMessage(

"Microsoft.Design",

"CA1004:GenericMethodsShouldProvideTypeParameter",

Justification =

"The type parameter TMessage identifies the message type that the recipient wants to unregister for.")]

public virtual void Unregister<TMessage>(object recipient)

{

Unregister<TMessage>(recipient, null);

}

public virtual void Unregister<TMessage>(object recipient, Action<TMessage> action)

{

UnregisterFromLists(recipient, action, _recipientsStrictAction);

UnregisterFromLists(recipient, action, _recipientsOfSubclassesAction);

Cleanup();

}

private static void CleanupList(IDictionary<Type, List<WeakActionAndToken>> lists)

{

if (lists == null)

{

return;

}

var listsToRemove = new List<Type>();

foreach (var list in lists)

{

var recipientsToRemove = new List<WeakActionAndToken>();

foreach (var item in list.Value)

{

if (item.Action == null

|| !item.Action.IsAlive)

{

recipientsToRemove.Add(item);

}

}

foreach (var recipient in recipientsToRemove)

{

list.Value.Remove(recipient);

}

if (list.Value.Count == 0)

{

listsToRemove.Add(list.Key);

}

}

foreach (var key in listsToRemove)

{

lists.Remove(key);

}

}

private static bool Implements(Type instanceType, Type interfaceType)

{

if (interfaceType == null

|| instanceType == null)

{

return false;

}

var interfaces = instanceType.GetInterfaces();

foreach (var currentInterface in interfaces)

{

if (currentInterface == interfaceType)

{

return true;

}

}

return false;

}

private static void SendToList<TMessage>(

TMessage message,

IEnumerable<WeakActionAndToken> list,

Type messageTargetType,

object token)

{

if (list != null)

{

var listClone = list.Take(list.Count()).ToList();

foreach (var item in listClone)

{

var executeAction = item.Action as IExecuteWithObject;

if (executeAction != null

&& item.Action.IsAlive

&& item.Action.Target != null

&& (messageTargetType == null

|| item.Action.Target.GetType() == messageTargetType

|| Implements(item.Action.Target.GetType(), messageTargetType))

&& ((item.Token == null && token == null)

|| item.Token != null && item.Token.Equals(token)))

{

executeAction.ExecuteWithObject(message);

}

}

}

}

private static void UnregisterFromLists(object recipient, Dictionary<Type, List<WeakActionAndToken>> lists)

{

if (recipient == null

|| lists == null

|| lists.Count == 0)

{

return;

}

lock (lists)

{

foreach (var messageType in lists.Keys)

{

foreach (var item in lists[messageType])

{

var weakAction = item.Action;

if (weakAction != null

&& recipient == weakAction.Target)

{

weakAction.MarkForDeletion();

}

}

}

}

}

private static void UnregisterFromLists<TMessage>(

object recipient,

Action<TMessage> action,

Dictionary<Type, List<WeakActionAndToken>> lists)

{

var messageType = typeof(TMessage);

if (recipient == null

|| lists == null

|| lists.Count == 0

|| !lists.ContainsKey(messageType))

{

return;

}

lock (lists)

{

foreach (var item in lists[messageType])

{

var weakActionCasted = item.Action as WeakAction<TMessage>;

if (weakActionCasted != null

&& recipient == weakActionCasted.Target

&& (action == null

|| action == weakActionCasted.Action))

{

item.Action.MarkForDeletion();

}

}

}

}

private void Cleanup()

{

CleanupList(_recipientsOfSubclassesAction);

CleanupList(_recipientsStrictAction);

}

private void SendToTargetOrType<TMessage>(TMessage message, Type messageTargetType, object token)

{

var messageType = typeof(TMessage);

if (_recipientsOfSubclassesAction != null)

{

var listClone = _recipientsOfSubclassesAction.Keys.Take(_recipientsOfSubclassesAction.Count()).ToList();

foreach (var type in listClone)

{

List<WeakActionAndToken> list = null;

if (messageType == type

|| messageType.IsSubclassOf(type)

|| Implements(messageType, type))

{

list = _recipientsOfSubclassesAction[type];

}

SendToList(message, list, messageTargetType, token);

}

}

if (_recipientsStrictAction != null)

{

if (_recipientsStrictAction.ContainsKey(messageType))

{

var list = _recipientsStrictAction[messageType];

SendToList(message, list, messageTargetType, token);

}

}

Cleanup();

}

private struct WeakActionAndToken

{

public WeakAction Action;

public object Token;

}

}

Messager在MVVM的实例

在ViewModel端

public void UpdateJob(object sender, RoutedEventArgs v)

{

var focusrow = (JobMVVMClient.JobBL.View_T_SegmentJob)SelectRow;

if (focusrow != null)

{

JobMVVMClient.JobBL.JobBLClient Job = new          JobMVVMClient.JobBL.JobBLClient();

var row = (JobMVVMClient.JobBL.View_T_SegmentJob)selectRow;

if (row.班次号 == null || row.班组号 == null

|| row.计划结束时间 == null || row.计划开始时间 == null || row.投产量 == 0.0)

return;

Job.UpdateSegmentJobBySomeThingCompleted += (s, e) =>

{

if (e.Result)

{

MessageBox.Show(PromptString.UpdateJobYes);

foreach (var element in jobModel.Gradeinfo)

{

if (element.DetailID == row.班次号.ToString())

row.班次 = element.DetailName;

}

foreach (var element in jobModel.Groupinfo)

{

if (element.DetailID == row.班组号.ToString())

row.班组 = element.DetailName;

}

}

};

   Messenger.Default.Send<bool?,JobMVVMClinet.Views.JobPlanUpdateView>(true);

//给位于View层中的JobPlanUpdateView这个界面发送bool消息,这个消息是true.

Job.UpdateSegmentJobBySomeThingAsync(row.工单ID, row.投产量, (DateTime)row.计划开始时间, (DateTime)row.计划结束时间, (int)row.班次号, (int)row.班组号);

Job.CloseAsync();

}

}

在VIEW端

public partial class JobPlanUpdateView : ChildWindow

{

public JobPlanUpdateView(CreateJobViewModel model)

{

InitializeComponent();

DataContext = model;

Messenger.Default.Register<bool?>(this, m => this.DialogResult = m);

//注册接收消息,并且做出响应处理

}

}

Messager总结

消息在MVVM中广泛应用,更好的运用MESSAGER能更好的处理View与ViewModel的关系,为MVVM添加喝彩。

Messenger在MVVM模式中的应用的更多相关文章

  1. 在MVVM模式中,按钮Click事件的绑定方法

    在MVVM模式中,我们将Button的方法写到ViewModel中,然后绑定到前端界面.通常的做法是写一个类,继承ICommand接口,然而如果按钮比较多的话,就需要写很多的类,对于后期维护造成很大的 ...

  2. silverlighter下MVVM模式中利用Behavior和TargetedTriggerAction实现文本框的一些特效

    在silverlight一般开发模式中,给文本框添加一些事件是轻而易举的,然而MVVM开发模式中,想要给文本框添加一些事件并非那么容易,因为MVVM模式中,只有ICommand接口,而且也只有Butt ...

  3. “Win10 UAP 开发系列”之 在MVVM模式中控制ListView滚动位置

    这个扩展属性从WP8.1就开始用了,主要是为了解决MVVM模式中无法直接控制ListView滚动位置的问题.比如在VM中刷新了数据,需要将View中的ListView滚动到顶部,ListView只有一 ...

  4. WPF MVVM模式中,通过命令实现窗体拖动、跳转以及显隐控制

    原文:WPF MVVM模式中,通过命令实现窗体拖动.跳转以及显隐控制 在WPF中使用MVVM模式,可以让我们的程序实现界面与功能的分离,方便开发,易于维护.但是,很多初学者会在使用MVVM的过程中遇到 ...

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

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

  6. 深入理解MVVM模式中Silverlight的Trigger、Action和Behavior及Silverlight的继承机制

    接触Silverlight已经有两三个月了,开始一直感觉他和Winform很相似,拖拖控件就行了,所以一直把经历放在了研究后台和服务器交互和性能优化上面,很少去仔细研究Silverlight的页面.前 ...

  7. MVVM模式中WPF数据的完全绑定

    一:截图,描述:将后台代码的姓名.年龄绑定到文本框,单击”增加年龄“--年龄自+1,单击”显示年龄“--弹出年龄的显示对话框,实现了从文本框修改年龄和后台更改年龄并显示到文本框 运行结果和解决方案管理 ...

  8. MVVM模式中ViewModel和View、Model有什么区别

    Model:很简单,就是业务逻辑相关的数据对象,通常从数据库映射而来,我们可以说是与数据库对应的model. View:也很简单,就是展现出来的用户界面. 基本上,绝大多数软件所做的工作无非就是从数据 ...

  9. XAML: 在 MVVM 模式中,关于绑定的几处技巧

    以下会提到三个绑定的技巧,分别是 在 ListView 中为 ListViewItem 的 MenuFlyout 绑定 Command: 在 ListView 的 事件中绑定所选择项目,即其 Sele ...

随机推荐

  1. 多线程系列(四):Task

    目录: 为什么要使用任务 任务 一.为什么使用任务 线程池已经可以让我们简单地创建线程,并优化了性能. 但是,线程池的缺点在于,我不清楚我的操作什么时候完成,也不能收到返回值,因为委托是没有返回值的. ...

  2. Spring MVC知识

    f-sm-1. 讲下SpringMvc和Struts1,Struts2的比较的优势 性能上Struts1>SpringMvc>Struts2 开发速度上SpringMvc和Struts2差 ...

  3. [Abp 源码分析]七、仓储与 Entity Framework Core

    0.简介 Abp 框架在其内部实现了仓储模式,并且支持 EF Core 与 Dapper 来进行数据库连接与管理,你可以很方便地通过注入通用仓储来操作你的数据,而不需要你自己来为每一个实体定义单独的仓 ...

  4. 3.MySQL(三)

    索引类型 先创建表 mysql> CREATE TABLE test( -> id INT, -> username VARCHAR(16), -> city VARCHAR( ...

  5. BBS论坛(三十)

    30.显示评论和添加评论功能完成 (1)apps/models.py class CommentModel(db.Model): __tablename__='comment' id=db.Colum ...

  6. 一个简单的案例带你入门Dubbo分布式框架

    相信有很多小伙伴都知道,dubbo是一个分布式.高性能.透明化的RPC服务框架,提供服务自动注册.自动发现等高效服务治理方案,dubbo的中文文档也是非常全的,中文文档可以参考这里dubbo.io.由 ...

  7. 带着新人学springboot的应用11(springboot+Dubbo+Zookeeper 上)

    这次说个在大型项目比较常见的东西,就是分布式,分布式到底是个什么东西呢?概念太大,不好说,就像刚学javaee的人问你,什么是web啊,什么是spring啊等等,你可能觉得,这个东西我好像知道,但是用 ...

  8. Jexus~webApi程序的部署几个小坑

    今天上午写了Jexus~Linux环境下的部署,下午去实现了一个,出现了一些问题,下面总结一下. 一 首先要对jexus进行修改/usr/jexus/jws 删除下面的这行,解决csc文件未找到问题 ...

  9. 按行切割大文件(linux split 命令简版)

    按行切割大文件(linux split 命令简版) #-*- coding:utf-8 -*- __author__ = 'KnowLifeDeath' ''' Linux上Split命令可以方便对大 ...

  10. 数据结构之哈希(hash)表

    最近看PHP数组底层结构,用到了哈希表,所以还是老老实实回去看结构,在这里去总结一下. 1.哈希表的定义 这里先说一下哈希(hash)表的定义:哈希表是一种根据关键码去寻找值的数据映射结构,该结构通过 ...