WPF& MVVM:传递事件参数不起作用

时间:2017-02-24 11:57:57

标签: c# wpf mvvm

我有WPF MVVM应用程序,我想将事件args传递给ViewModel,但它不起作用。我浏览了本网站上发现的一系列类似主题,但没有结果。

这是我在控件中的XAML:

<telerik:RadMap>
<i:Interaction.Triggers>
  <i:EventTrigger EventName="MouseMove" >
    <ei:CallMethodAction TargetObject="{Binding}" MethodName="VM.SomeMethod"/>
  </i:EventTrigger>
</i:Interaction.Triggers>
</telerik:RadMap>

在我的ViewModel中,我有方法:

public void SomeMethod(object sender, MouseEventArgs e)
{

}

当我将鼠标移到控件上时,会引发异常,说明该方法没有合适的签名。

然后我也尝试使用telerik的EventToCommandBehavior,如下所示:

<telerik:RadMap>
<telerik:EventToCommandBehavior ...>
...

它抱怨EventToCommandBehavior不存在。我google了,甚至在telerik文档中发现了这个EventToCommandBehavior。所以我猜想这可能是一个较新的功能,因为我的telerik是从2010年开始的。

无论如何,欢迎任何有关如何将事件参数传递给ViewModel的建议。

2 个答案:

答案 0 :(得分:2)

编辑:我应该更好地阅读这个问题。绑定表达式的默认根是绑定的UI元素的DataContext,因此绑定表达式将VM.SomeCommand读取为DataContext.VM.SomeCommand。 VM.SomeCommand应该更改为SomeCommand,因为视图的DataContext应该是您的ViewModel。

Expression Blend交互框架不支持将事件args传递给命令。我建议使用MVVM框架。

布拉德利·乌弗纳提到的MVVM Light,你可以这样做:

<i:Interaction.Triggers>
    <i:EventTrigger EventName="MouseMove">
        <cmd:EventToCommand Command="{Binding Mode=OneWay, Path=SomeCommand}"
                            PassEventArgsToCommand="True" />
    </i:EventTrigger>
</i:Interaction.Triggers>

但是,如果您出于某种原因反对这个想法,您可以尝试讨论here的解决方案。

来自链接的示例代码:

public sealed class InvokeDelegateCommandAction : TriggerAction<DependencyObject>
{
    /// <summary>
    /// 
    /// </summary>
    public static readonly DependencyProperty CommandParameterProperty =
        DependencyProperty.Register("CommandParameter", typeof(object), typeof(InvokeDelegateCommandAction), null);

    /// <summary>
    /// 
    /// </summary>
    public static readonly DependencyProperty CommandProperty = DependencyProperty.Register(
        "Command", typeof(ICommand), typeof(InvokeDelegateCommandAction), null);

    /// <summary>
    /// 
    /// </summary>
    public static readonly DependencyProperty InvokeParameterProperty = DependencyProperty.Register(
        "InvokeParameter", typeof(object), typeof(InvokeDelegateCommandAction), null);

    private string commandName;

    /// <summary>
    /// 
    /// </summary>
    public object InvokeParameter
    {
        get
        {
            return this.GetValue(InvokeParameterProperty);
        }
        set
        {
            this.SetValue(InvokeParameterProperty, value);
        }
    }

    /// <summary>
    /// 
    /// </summary>
    public ICommand Command
    {
        get
        {
            return (ICommand)this.GetValue(CommandProperty);
        }
        set
        {
            this.SetValue(CommandProperty, value);
        }
    }

    /// <summary>
    /// 
    /// </summary>
    public string CommandName
    {
        get
        {
            return this.commandName;
        }
        set
        {
            if (this.CommandName != value)
            {
                this.commandName = value;
            }
        }
    }

    /// <summary>
    /// 
    /// </summary>
    public object CommandParameter
    {
        get
        {
            return this.GetValue(CommandParameterProperty);
        }
        set
        {
            this.SetValue(CommandParameterProperty, value);
        }
    }

    /// <summary>
    /// 
    /// </summary>
    /// <param name="parameter"></param>
    protected override void Invoke(object parameter)
    {
        this.InvokeParameter = parameter;

        if (this.AssociatedObject != null)
        {
            ICommand command = this.ResolveCommand();
            if ((command != null) && command.CanExecute(this.CommandParameter))
            {
                command.Execute(this.CommandParameter);
            }
        }
    }

    private ICommand ResolveCommand()
    {
        ICommand command = null;
        if (this.Command != null)
        {
            return this.Command;
        }
        var frameworkElement = this.AssociatedObject as FrameworkElement;
        if (frameworkElement != null)
        {
            object dataContext = frameworkElement.DataContext;
            if (dataContext != null)
            {
                PropertyInfo commandPropertyInfo = dataContext
                    .GetType()
                    .GetProperties(BindingFlags.Public | BindingFlags.Instance)
                    .FirstOrDefault(
                        p =>
                        typeof(ICommand).IsAssignableFrom(p.PropertyType) &&
                        string.Equals(p.Name, this.CommandName, StringComparison.Ordinal)
                    );

                if (commandPropertyInfo != null)
                {
                    command = (ICommand)commandPropertyInfo.GetValue(dataContext, null);
                }
            }
        }
        return command;
    }
}

来自博客的示例用法:

<ComboBox>
    <ComboBoxItem Content="Foo option 1" />
    <ComboBoxItem Content="Foo option 2" />
    <ComboBoxItem Content="Foo option 3" />
    <Interactivity:Interaction.Triggers>
        <Interactivity:EventTrigger EventName="SelectionChanged" >
            <Presentation:InvokeDelegateCommandAction 
                Command="{Binding SubmitFormCommand}"
                CommandParameter="{Binding RelativeSource={RelativeSource Self}, Path=InvokeParameter}" />
        </Interactivity:EventTrigger>
    </Interactivity:Interaction.Triggers>                
</ComboBox>

注意:我没有在此博客中测试过该解决方案。

答案 1 :(得分:0)

this开始,我读了

  

被调用的方法必须是不带参数的公共方法,不返回其签名与事件处理程序匹配的值或公共方法。

如果您不需要MouseEventArgs信息,可以尝试不带参数,或者使用:

public void SomeMethod(object sender, EventArgs e)
{

}

我的2美分......