类型列表的附加属性

Hap*_*mad 19 wpf attached-properties

我想创建一个可以使用此语法的附加属性:

<Button>
  <Image .../>
  <ui:ToolbarItem.DisplayFilter>
    <TabItem .../>
    <TabItem .../>
    <TabItem .../>
  </ui:ToolbarItem.DisplayFilter>
</Button> 
Run Code Online (Sandbox Code Playgroud)

这是我尝试这样做的:

public class ToolbarItem
{
  /// <summary>
  /// Identifies the DisplayFilter attached property. 
  /// </summary>
  public static readonly DependencyProperty DisplayFilterProperty =
    DependencyProperty.RegisterAttached(
     "DisplayFilter",
     typeof( IList ),
     typeof( ToolbarItem )
    );

  public static IList GetDisplayFilter( Control item ) {
    return (IList)item.GetValue( DisplayFilterProperty );
  }

  public static void SetDisplayFilter( Control item, IList value ) {
    item.SetValue( DisplayFilterProperty, value );
  }
}
Run Code Online (Sandbox Code Playgroud)

但是,这会在分析时导致异常 - System.ArgumentException:TabItem不是属性'DisplayFilter'的有效值.那么如何配置我的附加属性以便我可以使用所需的XAML语法?

gix*_*gix 35

请记住,XAML基本上只是对象创建的简写形式.因此,要创建一个集合/列表作为附加DisplayFilter属性的值,您必须将它们包含TabItems在另一个集合标记内.如果您不想这样做,这是可以理解的,您必须在第一次访问该属性时初始化该集合.

这只有一个问题:XAML阅读器跳过getter方法作为优化.您可以通过为调用的name参数选择其他名称来阻止此行为RegisterAttached:

DependencyProperty.RegisterAttached("DisplayFilterInternal", ...)
Run Code Online (Sandbox Code Playgroud)

然后将调用属性getter,您可以检查null.您可以在此博客文章中阅读更多相关信息.

编辑:似乎链接的博客帖子不是那么清楚.您更改传递给的字符串RegisterAttached的名称,而不是静态get/set方法的名称:

public static readonly DependencyProperty DisplayFilterProperty =
    DependencyProperty.RegisterAttached(
        "DisplayFilterInternal",
        typeof(IList),
        typeof(ToolbarItem));

public static TabItemCollection GetDisplayFilter(Control item)
{ ... }

public static void SetDisplayFilter(Control item, IList value)
{ ... }
Run Code Online (Sandbox Code Playgroud)

您必须在GetDisplayFilter方法中初始化集合:

public static TabItemCollection GetDisplayFilter(Control item)
{
    var collection = (IList)item.GetValue(DisplayFilterProperty);
    if (collection == null) {
        collection = new List<object>();
        item.SetValue(DisplayFilterProperty, collection);
    }
    return collection;
}
Run Code Online (Sandbox Code Playgroud)

您似乎只TabItem向该集合添加元素.然后,您可以使集合类型安全,但IList<T>由于XAML解析器无法调用泛型方法,因此使用不起作用Add(T).Collection<T>并且List<T>还实现非通用IList接口,并且可以在这种情况下使用.我建议创建一个新的集合类型,以防您希望将来对集合进行一些更改:

public class TabItemCollection : Collection<TabItem>
{
}
Run Code Online (Sandbox Code Playgroud)

如果您不关心如下显式设置集合:

<ui:ToolbarItem.DisplayFilter>
    <ui:TabItemCollection>
        <TabItem/>
    </ui:TabItemCollection>
</ui:ToolbarItem.DisplayFilter>
Run Code Online (Sandbox Code Playgroud)

你可以删除该SetDisplayFilter方法.

总结一下:

public class TabItemCollection : Collection<TabItem>
{
}

public class ToolbarItem
{
    public static readonly DependencyProperty DisplayFilterProperty =
        DependencyProperty.RegisterAttached(
            "DisplayFilterInternal", // Shadow the name so the parser does not skip GetDisplayFilter
            typeof(TabItemCollection),
            typeof(ToolbarItem));

    public static TabItemCollection GetDisplayFilter(Control item)
    {
        var collection = (TabItemCollection)item.GetValue(DisplayFilterProperty);
        if (collection == null) {
            collection = new TabItemCollection();
            item.SetValue(DisplayFilterProperty, collection);
        }
        return collection;
    }

    // Optional, see above note
    //public static void SetDisplayFilter(Control item, TabItemCollection value)
    //{
    //    item.SetValue(DisplayFilterProperty, value);
    //}
}
Run Code Online (Sandbox Code Playgroud)