从 XAML 设置 ViewModel 属性值

Chr*_*ard 5 wpf xaml mef properties mvvm

我有一个在 XAML 中声明的视图(见下文)。关联的视图模型是使用 MEF 自动创建的。我希望能够做这样的事情:

<local:MyControl Owner={x:Static local:Owners.ProjectOwner} />
Run Code Online (Sandbox Code Playgroud)

所需的净效果是将某些视图模型属性设置为等于 Owners.ProjectOwner。

我可以使用 hacky 代码隐藏来实现所需的结果,但宁愿通过绑定或一些类似的方式来做到这一点。任何人都可以建议一种方法吗?

更新

我让自己去写一个行为。但是,我没有为一个特定的案例付出所有的努力,而是将我的解决方案通用化,并将其包含在下面以防万一有人感兴趣。这是一种混合行为 (System.Windows.Interactivity.dll),但也很容易成为传统的附加行为。

using System;
using System.Windows;
using System.Windows.Interactivity;

namespace BlendBehaviors
{
    public class SetViewModelPropertyBehavior : Behavior<FrameworkElement>
    {
        public static readonly DependencyProperty PropertyNameProperty =
            DependencyProperty.Register("PropertyName", typeof(string), typeof(SetViewModelPropertyBehavior));

        public static readonly DependencyProperty PropertyValueProperty =
            DependencyProperty.Register("PropertyValue", typeof(object), typeof(SetViewModelPropertyBehavior));

        public SetViewModelPropertyBehavior()
        { }

        public string PropertyName
        {
            get { return (string)GetValue(PropertyNameProperty); }
            set { SetValue(PropertyNameProperty, value); }
        }

        public object PropertyValue
        {
            get { return GetValue(PropertyValueProperty); }
            set { SetValue(PropertyValueProperty, value); }
        }

        protected override void OnAttached()
        {
            base.OnAttached();
            var ao = AssociatedObject;
            SetViewModel(ao.DataContext);
            ao.DataContextChanged += FrameworkElement_DataContextChanged;
        }

        protected override void OnDetaching()
        {
            base.OnDetaching();
            AssociatedObject.DataContextChanged -= FrameworkElement_DataContextChanged;
        }

        private void FrameworkElement_DataContextChanged(object sender, DependencyPropertyChangedEventArgs e)
        {
            SetViewModel(e.NewValue);
        }

        private void SetViewModel(object viewModel)
        {
            SetViewModelProperty(viewModel, PropertyName, PropertyValue);
        }

        private static void SetViewModelProperty(object viewModel, string propertyName, object propertyValue)
        {
            if (viewModel == null || propertyName == null) {
                return;
            }
            var info = viewModel.GetType().GetProperty(propertyName);
            if (info != null && CanAssignValue(propertyValue, info.PropertyType)) {
                info.SetValue(viewModel, propertyValue, null);
            }
        }

        private static bool CanAssignValue(object value, Type targetType)
        {
            if (value == null) {
                return !targetType.IsValueType || Nullable.GetUnderlyingType(targetType) != null;
            }
            return targetType.IsAssignableFrom(value.GetType());
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

然后像这样使用它:

<local:MyControl>
    <i:Interaction.Behaviors>
        <bb:SetViewModelPropertyBehavior PropertyName="Owner" PropertyValue="{x:Static local:Owners.ProjectOwner}" />
        <bb:SetViewModelPropertyBehavior PropertyName="AnotherProperty" PropertyValue="{StaticResource MyResourceKey}" />
    </i:Interaction.Behaviors>
</local:MyControl>
Run Code Online (Sandbox Code Playgroud)

Ken*_*art 3

任何 WPF 绑定的目标都必须是DependencyProperty. 源可以是DependencyProperty、实现 的 CLR 对象INotifyPropertyChanged,或者只是某个对象。可以通过更改属性来交换目标和源Binding.Mode

但在这种情况下,绑定中的一项是静态解析的属性 ( Owners.ProjectOwner)。因此,它不是一个DependencyProperty. 所以,它只能作为一个来源出现。因此,您将其绑定到的内容(目标)必须是DependencyProperty. 因此,它不能是视图模型上的属性(假设您没有创建DependencyObject基于视图模型,这将是一个错误)。

因此,您不能直接将虚拟机上的属性绑定到静态属性。不过,您可以编写一个附加行为来为您完成此操作。