10

我的问题基本上是这个。不过,我认为这将有助于提供更多信息和代码,以便更容易重现问题。

使用RibbonControlsLibrary中的 Microsoft.Windows.Controls.Ribbon.RibbonComboBox感觉就像走过一个充满错误的大沼泽,而不是如果您知道解决方法就可以做的事情。

随便。我遇到的最大问题是对我的 SelectedItem 进行数据绑定。

以下是我开始的(在我发现了RibbonGallery 之后?)。在 ComboBox 的子元素上拥有 ItemsSource 和 SelectedItem 甚至不在同一级别上已经给了我 heebie-jeebies,但这似乎是正确的。

在示例应用程序中,我在 ViewModel 的构造函数中设置 SelectedItem。但是,在运行应用程序时,不会显示 SelectedItem。甚至 VS 设计器也正确显示了“第二个选项”!

运行应用程序:运行应用VS 设计器:Visual Studio 设计器

在调试 SelectedItem 设置器时,您会注意到多次传递。在第一次将其设置为 ctor 中的“第二个选项”(1,请参见下面的调试日志)后,它将重置为 null (2)(通过外部代码,我认为是控件本身)。在 UI 中打开下拉菜单时,将再次设置为 null (3),然后在选择一个值时,两次设置为该值 (4,5)。我选择了“第二个选项”,然后用“第一个选项”(6-9)重复了这个过程。这产生了以下日志(忽略功能区控件中的一千零一个绑定异常......):

在此处输入图像描述

大问题显然是(2),它正在重置我的初始选择。看起来当控件第一次显示时,它被重置了。一个非常难看的解决方法是通过计时器设置值。在这个示例应用程序中,在用户控件的 Loaded 事件中设置它确实对我有用,但在我较重的现实应用程序中,它没有。无论如何,这一切都感觉不对。有谁知道更好的解决方案?

xml:

<UserControl x:Class="WpfApplication1.RibbonComboBoxDemo"
             xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
             xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
             xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" 
             xmlns:d="http://schemas.microsoft.com/expression/blend/2008" 
             xmlns:r="http://schemas.microsoft.com/winfx/2006/xaml/presentation/ribbon" 
             xmlns:local="clr-namespace:WpfApplication1"
             mc:Ignorable="d" 
             d:DesignHeight="300" d:DesignWidth="300">

    <UserControl.DataContext>
        <local:ViewModel />
    </UserControl.DataContext>

    <Grid>
        <r:Ribbon >
            <r:RibbonTab Header="First Tab">
                <r:RibbonGroup Header="Group">
                    <r:RibbonComboBox >
                        <r:RibbonGallery SelectedItem="{Binding SelectedItem, Mode=TwoWay}">
                            <r:RibbonGalleryCategory ItemsSource="{Binding Controls}" DisplayMemberPath="Caption" />
                        </r:RibbonGallery>
                    </r:RibbonComboBox>
                </r:RibbonGroup>
            </r:RibbonTab>
            <r:RibbonTab Header="Second Tab" />
        </r:Ribbon>
    </Grid>
</UserControl>

视图模型:

using System;
using System.Collections.Generic;
using System.Collections.ObjectModel;
using System.ComponentModel;
using System.Diagnostics;

namespace WpfApplication1
{
    public class ViewModel : INotifyPropertyChanged
    {
        public event PropertyChangedEventHandler PropertyChanged;
        private void OnPropertyChanged(string propertyName)
        {
            if (this.PropertyChanged != null)
                PropertyChanged(this, new PropertyChangedEventArgs(propertyName));
        }

        public ObservableCollection<ControlBaseModel> Controls { get; private set; }


        private ControlBaseModel _selectedItem;
        public ControlBaseModel SelectedItem { get { return _selectedItem; } set { LogSelectedItemChange(value); _selectedItem = value; OnPropertyChanged("SelectedItem"); } }

        public ViewModel()
        {
            this.Controls = new ObservableCollection<ControlBaseModel>();

            this.Controls.Add(new ControlBaseModel() { Caption = "first option" });
            this.Controls.Add(new ControlBaseModel() { Caption = "second option" });

            this.SelectedItem = this.Controls[1]; // set to second option
        }

        int i = 0;
        private void LogSelectedItemChange(ControlBaseModel value)
        {
            i++;
            string setObject = "null";
            if (value != null)
            {
                setObject = value.Caption;
            }
            Debug.WriteLine(string.Format("{0}: SelectedItem.set(): {1}", i, setObject));
        }

    }

    public class ControlBaseModel : INotifyPropertyChanged
    {
        public event PropertyChangedEventHandler PropertyChanged;
        private void OnPropertyChanged(string propertyName)
        {
            if (this.PropertyChanged != null)
                PropertyChanged(this, new PropertyChangedEventArgs(propertyName));
        }

        private string _name;
        public string Name { get { return _name; } set { _name = value; OnPropertyChanged("Name"); } }

        private string _caption;
        public string Caption { get { return _caption; } set { _caption = value; OnPropertyChanged("Caption"); } }
    }
}
4

2 回答 2

5

虽然在我的应用程序中将 ComboBox SelectedItem 重置为 null 之前发生了 View/UserControl 加载事件,但 ComboBox 加载事件实际上被触发了两次,第二次“迟到”就足够了。所以我目前的解决方案是:

<r:RibbonComboBox>
    <i:Interaction.Triggers>
        <i:EventTrigger EventName="Loaded">
            <i:InvokeCommandAction Command="{Binding LoadedCommand}" />
        </i:EventTrigger>
    </i:Interaction.Triggers>
    <r:RibbonGallery SelectedItem="{Binding SelectedItem, Mode=TwoWay}">
        <r:RibbonGalleryCategory ItemsSource="{Binding Controls}" DisplayMemberPath="Caption"/>
    </r:RibbonGallery>
</r:RibbonComboBox>

视图模型:

private ControlBaseModel _lastNonNullSelectedItem;

public ObservableCollection<ControlBaseModel> Controls { get; private set; }

private ControlBaseModel _selectedItem;
public ControlBaseModel SelectedItem 
{ 
    get { return _selectedItem; } 
    set 
    { 
        if (value != null) { _lastNonNullSelectedItem = value; } 
        _selectedItem = value; 
        OnPropertyChanged("SelectedItem"); 
    } 
}
public ICommand LoadedCommand { get; private set; }


public ViewModel()
{
    this.Controls = new ObservableCollection<ControlBaseModel>();
    this.LoadedCommand = new ActionCommand(OnLoaded); // ActionCommand: simple implementation of ICommand

    this.Controls.Add(new ControlBaseModel() { Caption = "first option" });
    this.Controls.Add(new ControlBaseModel() { Caption = "second option" });

    this.SelectedItem = this.Controls[1]; // set to second option
}

private void OnLoaded()
{
    this.SelectedItem = _lastNonNullSelectedItem;
}
于 2013-03-22T10:14:36.473 回答
3

我最终只使用了标准的组合框。

<ComboBox SelectedItem="{Binding Item}" ItemsSource="{Binding Items}"/>

如果您想要与 RibbonComboBox 相同(非常相似)的样式,请使用

<ComboBox SelectedItem="{Binding Item}" ItemsSource="{Binding Items}" IsEditable="True" IsReadOnly="True"/>
于 2013-05-22T15:03:16.950 回答