![](/img/trans.png)
[英]why WPF Window.Visibility binding must have Mode=TwoWay for property's get to be called?
[英]why binding CollectionView to Window.Visibility does not work while binding to same bound property works for ItemsControl?
下面的ItemsControl绑定起作用,并且刷新collectionview时,控件中的工具栏也会相应刷新。 但是,可见性绑定不起作用(甚至一次),并且永远不会调用该转换器。 输出窗口中没有错误。
<Window.Resources>
<xpui:IEnumerableHasItemsToVisibilityConverter x:Key="IEnumerableHasItemsToVisibilityConverter" />
<ContextMenu x:Key="ToolbarContextMenu">
<MenuItem Header="Move to top" Click="MoveToTopClick" />
<MenuItem Header="Move to left" Click="MoveToLeftClick"/>
</ContextMenu>
<xpui:MenuItemToToolbarConverter x:Key="menutotoolbarconverter" />
</Window.Resources>
<ItemsControl Name="Toolbars" ItemsSource="{Binding GuiItemsInstance.FloatingToolbarsView}">
<ItemsControl.ItemTemplate>
<DataTemplate>
<ContentControl Content="{Binding Converter={StaticResource menutotoolbarconverter}, ConverterParameter={StaticResource ToolbarContextMenu}}"/>
</DataTemplate>
</ItemsControl.ItemTemplate>
</ItemsControl>
<Window.Visibility>
<Binding Path="GuiItemsInstance.FloatingToolbarsView" Converter="{StaticResource IEnumerableHasItemsToVisibilityConverter}"/>
</Window.Visibility>
在GuiItems-singleton(GuiItemsInstance是ViewModel的属性)中,实现INotifyPropertyChanged
public CollectionView FloatingToolbarsView
{
get;
private set;
}
//I also tried to bind directly to FloatVisibility:
private Visibility _floatVisibility = Visibility.Hidden;
public Visibility FloatVisibility
{
get { return _floatVisibility; }
set
{
_floatVisibility = value;
// Sets the property value. Raises the PropertyChanged event, if needed.
SetValue(ref _floatVisibility, value);
}
}
///Event of changes in underlying toolbar collection is handled like this
private void RefreshFloatingToolbars(int RoleId)
{
ActiveRoleId = RoleId;
FloatingToolbarsView.Refresh();
if (ToolbarItems.Any(i => i.ToolbarLocation == ToolbarLocation.Float && i.RoleId == RoleId))
FloatVisibility = Visibility.Visible;
else
FloatVisibility = Visibility.Hidden;
}
转换器:
public class IEnumerableHasItemsToVisibilityConverter : IValueConverter
{
public object Convert(object value, Type targetType, object parameter,
System.Globalization.CultureInfo culture)
{
if (value != null)
{
if (((IEnumerable)value).GetEnumerator().MoveNext())
return Visibility.Visible;
}
return Visibility.Hidden;
}
ViewModel:
//WindowViewModel implements INotifyPropertyChanged
public class FloatingToolbarWindowViewModel : WindowViewModel
{
public GuiItems GuiItemsInstance { get; set; }
public FloatingToolbarWindowViewModel(GuiItems guiItems)
{
GuiItemsInstance = guiItems;
GuiItemsInstance.Host = Host;
}
}
这根本不起作用,也尝试像这样绑定到ItemsControl本身:
<Window.Visibility>
<Binding Path="ItemsSource" ElementName="Toolbars" Converter="{StaticResource IEnumerableHasItemsToVisibilityConverter}"/>
</Window.Visibility>
这导致调用转换器的值为null。
发生这种情况是因为CollectionView
实现了ICollectionChanged
,当集合中的项目发生更改时会触发该事件。 ItemsControl
查找此事件并做出适当的反应。 一个“普通”控件侦听CollectionView
确实实现的IPropertyChanged
通知,但不会触发对基础集合的更改。 因此,永远不会将更改通知您的绑定。
您可以“钩住” CollectionChanged
事件,并通过绑定发送适当的PropertyChanged
事件,但是此时,自己更改可见属性可能更容易。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.