繁体   English   中英

为什么将CollectionView绑定到Window.Visibility在将相同的绑定属性绑定到ItemsControl时不起作用?

[英]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.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM