3

我有一个奇怪的问题。

我正在使用 GalaSoft 的 MVVM Light 框架,到目前为止一切正常。我使用信使系统在 ViewModel 之间发送消息就好了,直到我尝试执行以下操作:

GateKeeper我有一个发送消息的单例类。

此类不是 ViewModel,因此不继承自ViewModelBase.

如果它发送一条消息,它不会在任何地方收到。

我尝试了以下方法:

  1. GateKeeper继承自ViewModeBase-> 没有成功。
  2. 注册GateKeeper以接收消息,从而查看它是否会捕获/接收实际从自身发送的消息 -> 不成功。
  3. 从单例更改GateKeeper为正常实例化 -> 没有成功
  4. 创建一个未连接到视图的 MVVM ViewModel,并让它发送消息,就像GateKeeper-> no Success

我所有连接到视图的视图模型都可以发送消息,并且它们将被接收。

在信使工作之前,似乎必须将视图模型“链接”到视图,但imo。这是一个主要限制。

以下是当前非常简化的设置。

在 GateKeeper 上调用 ApplicationInitialize 不会触发在 mainviewmodel 或 GateKeeper 类本身上接收到的消息。

我希望有人对这个问题提出建议。

谢谢..

示例设置: MainViewModel 构造函数:

 public MainViewModel()
    {
        Messenger.Default.Register<LoadViewMessage>(this, (message) =>
        {
            if (message.Sender is GateKeeper) CurrentView = message.View;
            else if (message.Sender is LoginViewModel) CurrentView = message.View;
            else if (message.Sender is MenuItemBarViewModel) CurrentView = message.View;
        });

看门人:

 public class GateKeeper : IGateKeeper
{
    private readonly IEmployeeService _employeeService;

    #region Implementation of IGateKeeper

    public void ApplicationInitialize()
    {
        Messenger.Default.Send<LoadViewMessage>(new LoadViewMessage(ObjectLocator.MainMapView), this);
    }

    public void LoginSucceeded(Employee employee)
    {
        //This is where we retrieve the available services for the current employee
        //TODO: add methods for retrieving service info from backend

        //Send a message that should make the mainview load the map into its currentview property
        Messenger.Default.Send(new LoadViewMessage(ObjectLocator.MainMapView), this);
    }

    #endregion

    public GateKeeper(IEmployeeService employeeService)
    {
        _employeeService = employeeService;

        //Test.. Is not triggered
        //Just used for debugging, thus nothing happens inhere.
        Messenger.Default.Register<LoadViewMessage>(this, (message) =>
        {
            if (message.Sender is GateKeeper) ;
            else if (message.Sender is LoginViewModel) ;
            else if (message.Sender is MenuItemBarViewModel);
        });
    }

消息类:LoadViewMessage

 public class LoadViewMessage : MessageBase
{
    public UserControl View { get; protected set; }

    public LoadViewMessage(UserControl view, object sender): base(sender)
    {
        View = view;
    }

    public LoadViewMessage(UserControl view):this(view, null){}
}

PS:ObjectLocator 是一个 NinJect 类,用于处理对象的所有实例化及其生命周期

@UPDATE LBugnion(MVVM Light 的创建者)指出问题出在 send 方法中,我实际上是在使用带有令牌的 Send 重载。

@这在我的情况下不起作用

Messenger.Default.Send(new LoadViewMessage(ObjectLocator.MainMapView), this);

@这将工作

Messenger.Default.Send(new LoadViewMessage(ObjectLocator.MainMapView, this));

应该作为令牌传递给loadViewMessage而不是 Send 方法

4

1 回答 1

4

您的问题出在 Send 方法上。您正在使用将令牌作为第二个参数的方法的重载。您将“this”作为令牌传递。这意味着您(可能是错误地)使用消息的发送者作为令牌。

如果您使用令牌发送,您还需要使用相同的令牌注册接收者(在这种情况下,与 Send 方法中使用的实例完全相同)。由于您没有注册令牌,因此消息永远不会由 Messenger 发送,这是一种优化机制。

我的猜测是您误解了 Send 方法中令牌的用法。如果您愿意,令牌只是作为构建“私人消息传递网络”的一种方式,其中两个对象可以使用相同的令牌注册/发送并建立私人通信。

在您的情况下,如果您想将 Sender 与消息一起发送,您需要将 Sender 保存在消息本身中,这就是 MessageBase 和派生类所做的。

希望这可以帮助

洛朗

于 2012-11-13T18:07:55.047 回答