标记扩展和对象创建

时间:2013-01-22 01:45:59

标签: c# wpf markup-extensions

目前我有这样的标记

<TextBlock>
  <TextBlock.Text>
    <Binding Path="Value" ElementName="window" ConverterParameter="M">
      <Binding.Converter>
        <local:DatePartValueConverter />
      </Binding.Converter>
     </Binding>
  </TextBlock.Text>
</TextBlock>

我想将它缩短为类似的东西

<TextBlock Text="{Binding Path=Value,ElementName=window,
  ConverterParameter=M,Converter={local:DatePartValueConverter}}" />

但是编译器barf因为DatePartValueConverter不是从MarkupExtension派生的。有没有其他方法可以使用短格式标记来创建DatePartValueConverter的实例?

顺便说一句,我尝试从MarkupExtension派生,它确实解决了这个问题。我对ProvideValue的实现看起来像这样

public override object ProvideValue(IServiceProvider serviceProvider)
{
  return new DatePartValueConverter();
}

并且它有效,但我仍然对serviceProvider的起源和本质以及人们可能会对它做什么感到朦胧。

有趣的是,当我在使用此标记扩展的绑定上使用Visual Studio 2012绑定编辑器时,它会立即再次扩展它,使整个标记扩展支持的东西有点无意义。

我应该提一下,我需要为每个绑定提供一个私有实例,因为它维护内部状态 - 它需要知道整个DateTime值来设置某些方面,比如

public object ConvertBack(object value, Type targetType, 
  object parameter, System.Globalization.CultureInfo culture)
{
  culture = System.Globalization.CultureInfo.CurrentCulture;
  var strValue = value as string;
  int y = _value.Year, M = _value.Month, d = _value.Day,
    H = _value.Hour, m = _value.Minute, s = _value.Second;
  if (strValue == null)
    return null;
  else
  {
    string p = parameter as string;
    switch (p)
    {
      case "yyyy":
        y = int.Parse(strValue); break;
      case "yy":
        y = (strValue.Length == 4) ?
          int.Parse(strValue) :
          int.Parse(DateTime.Now.Year.ToString().Substring(0, 2) + strValue);
        break;
      case "M":
      case "MM":
        M = int.Parse(strValue); break;
      ...
    }
  }
  return new DateTime(y, M, d, H, m, s);
}

2 个答案:

答案 0 :(得分:3)

您可以创建DatePartValueConverter的实例作为资源,然后使用它。

<Window.Resources>
  <local:DatePartValueConverter x:Key="datePartValueConverter" />
</Window.Resources>

<TextBlock Text="{Binding Path=Value, ElementName=window,
  ConverterParameter=M, Converter={StaticResource datePartValueConverter}}" />

编辑如果您确实想要将转换器设为MarkupExtension,那么您可以这样做。您可以在ProvideValue中返回新实例,也可以使用return this;

返回当前实例

通过返回当前实例,您可以在转换器中拥有属性,并允许您执行此类操作。

public class DatePartValueConverter : MarkupExtension, IValueConverter {
    public string ParseType { get; set; }
    // other methods
}

<TextBlock Text="{Binding Path=Value, ElementName=window,
  Converter={local:DatePartValueConverter ParseType=M}}" />

至于IServiceProvider,请参阅MarkupExtension.ProvideValue — Is the IServiceProvider actually used?

答案 1 :(得分:1)

除了从我知道的MarkupExtension派生转换器之外,唯一的方法是创建自己的自定义绑定类(请查看DelayBinding)。