WPF - 使用RelativeSource绑定为自定义标记扩展提供设计时值

时间:2014-03-09 23:16:44

标签: c# wpf xaml binding markup-extensions

N.B。:这不仅仅是关于定制标记扩展。请在标记为重复之前阅读。

我有一个带转换器的WPF标记扩展,其中两个如下:

  [ValueConversion(typeof(WindowState), typeof(object))]
  internal class WindowStateToObjectConverter : IValueConverter {
    public WindowStateToObjectConverter() { }

    public WindowStateToObjectConverter(object maximized, object normal) {
      this.maximized = maximized;
      this.normal = normal;
    }

    #region Properties
    #region Maximized Property
    private object maximized;

    public object Maximized {
      get { return maximized; }
      set { maximized = value; }
    }
    #endregion
    #region Normal Property
    private object normal;

    public object Normal {
      get { return normal; }
      set { normal = value; }
    }
    #endregion
    #endregion

    public object Convert(object value, Type targetType, object param, CultureInfo culture) {
      if((value as WindowState? ?? WindowState.Normal) == WindowState.Maximized) return maximized;
      else return normal;
    }

    public object ConvertBack(object value, Type targetType, object param, CultureInfo culture) {
      throw new InvalidOperationException("Cannot convert downwards to WindowState");
    }
  }

  [MarkupExtensionReturnType(typeof(Binding))]
  internal class WindowMaximizedSwitchExtension : MarkupExtension {
    object maximized, normal;

    public WindowMaximizedSwitchExtension(object maximized, object normal) {
      this.maximized = maximized;
      this.normal = normal;
    }

    public override object ProvideValue(IServiceProvider serviceProvider) {
      Binding ret = new Binding("WindowState");
      RelativeSource retRSource = new RelativeSource(RelativeSourceMode.FindAncestor);
      retRSource.AncestorType = typeof(Window);
      ret.RelativeSource = retRSource;
      ret.Converter = new WindowStateToObjectConverter(maximized, normal);
      return ret.ProvideValue(serviceProvider);
    }
  }

它们用于我正在设计的自定义窗口 - 当窗口最大化时,它们将用于切换某些值(边框宽度,边距等)。但是,在设计时,它们总是返回null,这是一个真正的痛苦,因为那时我的窗口看起来像这样:

Preview of the window in Visual Studio

......当它应该看起来像这样:

The window at runtime

(忽略第一张图片中缺少标题和图标,但是,如果您有解决方案,请随时回答 - 这基本上是同样的问题。)

由于显而易见的原因,用这个来设计是非常困难的。您在窗口预览中看到的唯一主要问题是我使用扩展程序设置Row / ColumnDefinition s的位置 - 当它返回null时,Height / {{1 }}设置为Width。所以,我的问题是,是否有办法在设计时选择默认值,可能代替绑定(例如非最大化值)。

1 个答案:

答案 0 :(得分:5)

嗯,我觉得自己像个白痴,但我很快找到了解决方案:

在表达式的ProvideValue方法中,我添加了以下行:

  ret.FallbackValue = normal;

其中normal是窗口未最大化时使用的值。

ProvideValue现在看起来像这样:

public override object ProvideValue(IServiceProvider serviceProvider) {
  Binding ret = new Binding("WindowState");
  RelativeSource retRSource = new RelativeSource(RelativeSourceMode.FindAncestor);
  retRSource.AncestorType = typeof(Window);
  ret.RelativeSource = retRSource;
  ret.Converter = new WindowStateToObjectConverter(maximized, normal);
  ret.FallbackValue = normal;
  return ret.ProvideValue(serviceProvider);
}

这会在设计时返回正常值。