繁体   English   中英

如何解决MarkupExtension中数据绑定的值?

[英]How do I resolve the value of a databinding inside a MarkupExtension?

我已经为基于密钥的字符串翻译做了标记扩展。

<TextBlock Text="{Translate myKey}" />

现在我希望能够使用嵌套绑定来提供我的密钥。 例:

<TextBlock Text="{Translate {Binding KeyFromDataContext}}" />

当我这样做时,我得到一个System.Windows.Data.Binding对象。 通过调用ProvideValue并传递ServiceProvider,我可以得到一个BindingExpression:

var binding = Key as Binding;
if (binding == null) {
    return null;
}
var bindingExpression = binding.ProvideValue(_serviceProvider) as BindingExpression;
if (bindingExpression == null) {
    return null;
}
var bindingKey = bindingExpression.DataItem;

我可以得到这个bindingExpression,但DataItem属性为null。 我已经像这样测试了我的绑定

<TextBlock Text="{Binding KeyFromDataContext}" />

它工作正常。

有任何想法吗?

这是不可能获得一个绑定的值。 你不应该尝试这样做。 WPF使用一些花哨的反射来解决绑定并相信我 - 你不要自己开始尝试。

无论如何,这是我最终做的,这实际上是一个很好的解决方案:

我制作了一个负责TranslateConverter

public object Convert(object value, Type targetType, object parameter, CultureInfo culture)
{
    var key = value as string ?? parameter as string;

    if (key != null)
    {
    // Do translation based on the key

    }
    return null;
}

然后在我的TranslateExtension我只需这样做:

var binding = Key as Binding ?? new Binding{Mode = BindingMode.OneWay};
binding.Converter = new TranslateConverter(_targetObject, _targetProperty, Dictionary, Converter);
binding.ConverterParameter = Key is Binding ? null : Key as string;

return binding.ProvideValue(serviceProvider);

这样,WPF解析绑定并将其作为值传递给转换器,而将简单的文本键作为参数传递给转换器。

_targetObject_targetProperty是从ServiceProvider获得的。

toxvaerd的答案并不普遍。 如果原始绑定已经有转换器,它会中断。 或者在无法编写转换器时。

有一个更好的解决方案。 我们可以声明两个构造函数。 当使用绑定时,接受BindingBase的第二个将由XAML调用。 要解析绑定的值,我们可以声明一个私有附加属性。 为此,我们需要知道标记扩展的目标元素。

有一个问题:当在模板中使用标记扩展时,没有目标元素(显然)。 在这种情况下,您应该ProvideValue()使用return this - 这样在应用模板时将再次调用扩展。

public class TranslateExtension : MarkupExtension
{
    private readonly BindingBase _binding;

    public TranslateExtension(BindingBase binding)
    {
        _binding = binding;
    }

    public TranslateExtension(string key)
    {
        Key = key;
    }

    [ConstructorArgument("key")]
    public string Key { get; set; }

    public override object ProvideValue(IServiceProvider serviceProvider)
    {
        if (_binding != null)
        {
            var pvt = (IProvideValueTarget)serviceProvider.GetService(typeof(IProvideValueTarget));
            var target = pvt.TargetObject as DependencyObject;

            // if we are inside a template, WPF will call us again when it is applied
            if (target == null)
                return this; 

            BindingOperations.SetBinding(target, ValueProperty, _binding);
            Key = (string)target.GetValue(ValueProperty);
            BindingOperations.ClearBinding(target, ValueProperty);
        }

        // now do the translation using Key
        return ...;
    }

    private static readonly DependencyProperty ValueProperty = 
        DependencyProperty.RegisterAttached("Value", typeof(string), typeof(TranslateExtension));
}

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM