对于Silverlight和WPF应用程序,我有一个自定义控件,其中包含一个ObservableCollection作为依赖项属性。该控件的一个元素,边框,需要根据ObservableCollection中项的组合来更改颜色。
例如,假设收集的是动物、蔬菜和矿物质,称为ObjectList。如果至少有一种动物,我希望边界是红色的;如果没有动物,但至少有一种蔬菜,它是绿色的;否则,收集的只有矿物质,所以会出现蓝色。
我创建了一个转换器,它可以接收集合并确定颜色,所以有一个绑定,如:
<Border Background="{Binding ObjectList,
RelativeSource={RelativeSource Self},
Converter={StaticResource MyColorConverter}}" />挑战在于,随着项目从ObjectList中添加/删除,我需要触发背景色的重新评估;然而,ObjectList本身并没有改变。我想我有三个选择,但我不确定哪一个可能是最佳做法:
ObjectList被更改,从而触发后台更新。UpdateTarget的CollectionChanged回调中的背景属性调用ObjectList。由于UpdateTarget不适用于Silverlight,所以我只是删除并重新添加绑定--再次有点麻烦。INotifyPropertyChanged,并在CollectionChanged实现中在ObjectList上调用CollectionChanged我最喜欢3,但我有一个DependencyObject也实现了INPC,这似乎很奇怪。是吗?有没有更优雅的方法?
发布于 2014-06-10 09:35:46
有这是MSDN文档推荐的一种方法。 (向下滚动到使用VisualStateManager的最佳实践;它是为完整的.Net编写的,但本节也非常适合Silverlight )。每当您的VisualStates依赖于您的自定义Control的属性/状态时,建议为每个VisualState设置一个ChangedHandler --影响属性,并从那里调用一个私有的UpdateVisualStates方法。评估您的条件,并在此方法中以编程方式设置VisualStates。
即使您不使用VisualStates进行颜色更改,我也建议您遵循同样的模式。
为了简洁起见,下面的代码不完整:
public ObservableCollection ObjectList {...}
public static readonly DependencyProperty ObjectListProperty =
DependencyProperty.Register(...OnObjectListChanged...);
private static void OnObjectListChanged(...)
{ObjectList.CollectionChanged += OnObjectListCollectionChanged;}
private void OnObjectListCollectionChanged(...){ UpdateVisualStates(); }
private void UpdateVisualStates()
{
//actually you have to instatiate a SolidColorBrush here
if (ContainsAtLeastOneAnimal()) { m_border.Background = Colors.Red; }
else if (ContainsAtLeastOneVegetable()) {m_border.Background = Colors.Green;}
else { m_border.Background = Colors.Blue; }
}如果不希望引用边界,可以随意引入DependencyProperty BorderColor并从xaml绑定到它。这很好。有另一个移动的部分是没有问题的。这比模拟整个ObjectList实例改变要好得多。
https://stackoverflow.com/questions/24125128
复制相似问题