我有一个复杂的窗口,其中包含各种可见的控件或基于布尔值折叠的控件。我想添加一个自定义属性,以便在设计时显示所有这些控件。我的属性实现如下:
public static class CustomAttributes
{
private static bool? _inDesignMode;
public static readonly DependencyProperty Visibility = DependencyProperty.RegisterAttached(
"Visibility",
typeof(Visibility),
typeof(CustomAttributes),
new PropertyMetadata(VisibilityChanged));
private static bool InDesignMode
{
get
{
if (!_inDesignMode.HasValue)
{
var prop = DesignerProperties.IsInDesignModeProperty;
_inDesignMode =
(bool)DependencyPropertyDescriptor.FromProperty(prop, typeof(FrameworkElement)).Metadata.DefaultValue;
}
return _inDesignMode.Value;
}
}
public static Visibility GetVisibility(DependencyObject dependencyObject)
{
return (Visibility)dependencyObject.GetValue(Visibility);
}
public static void SetVisibility(DependencyObject dependencyObject, Visibility value)
{
dependencyObject.SetValue(Visibility, value);
}
private static void VisibilityChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
{
if (!InDesignMode)
return;
d.SetValue(Control.VisibilityProperty, e.NewValue);
}
}
在XAML中,我这样使用它:
<Button Visibility="{Binding SomeBoolValue, Converter={StaticResource BoolToVisibility}}"
helper:CustomAttributes.Visibility="Visible"
/>
然而,这似乎并不奏效。我使用了一些其他类似的自定义属性,它们完成了自己的工作,但可见性不会触发,它只是在设计视图中保持折叠状态。我错过了什么?
编辑:
谢谢你为我指明了正确的方向。我的问题的解决方案不需要像我最初认为的那样使用自定义属性。为了实现我想要的设计时行为,我按照下面接受的答案中的建议修改了转换器实现。
深入思考您创建的逻辑。
UI元素没有两个Visibility属性,它是唯一的一个
但是您希望同时以两种方式操作此属性:通过绑定和附加属性
因此,您在他们之间为这处房产制造了竞争
属性将采用最后分配给它的值。
当Button初始化时,附加的属性将只触发一次(来自示例(
当数据上下文和/或其SomeBoolValue
属性发生更改时,将触发绑定
但是窗口的数据上下文设置晚于此窗口的UI元素的初始化。
我看到了几种解决方案
如果需要始终在设计模式中显示元素,最简单的方法就是向转换器添加适当的逻辑
在最简单的形式中,这样一个转换器的例子是:
/// <summary>Bool to Visibility converter.</summary>
[ValueConversion(typeof(bool), typeof(Visibility))]
public class BooleanToVisibilityConverter : IValueConverter
{
public static bool IsDesignMode { get; } = DesignerProperties.GetIsInDesignMode(new DependencyObject());
public object Convert(object value, Type targetType, object parameter, CultureInfo culture)
{
if (value is bool val)
return IsDesignMode || val
? Visibility.Visible
: Visibility.Collapsed;
return DependencyProperty.UnsetValue;
}
public object ConvertBack(object value, Type targetType, object parameter, CultureInfo culture)
{
throw new NotImplementedException();
}
}