带CheckBoxes的WPF ComboBox显示有关已检查项目的信息?

Eri*_*k83 4 c# wpf combobox

我试图制作一个包含复选框作为项目的ComboBox,并根据检查的内容在组合框"关闭"时显示不同的内容.

在图像中可以看到我尝试实现的外观.

在此输入图像描述

理想情况下,我不希望用户能够选择顶部的文本(在图像中).

有一个简单的解决方案吗?我已经看到了一些解决方案,当使用DataTriggers隐藏不同的嵌套控件时,可以显示所有项目时显示更多信息,但这并不是我想要的.

有任何想法吗?

/埃里克

Xav*_*ier 8

这是一种使用a实现大多数所需内容的方法ComboBox,除了仍然可以选择文本(使用自定义文本仅在IsEditabletrue 时才有效).它是不可编辑的,因为IsReadOnly="true".

视图

<ComboBox
    IsEditable="True"
    IsReadOnly="True"
    ItemsSource="{Binding Items}"
    Text="{Binding Text}">
    <ComboBox.ItemTemplate>
        <DataTemplate
            DataType="{x:Type local:Item}">
            <CheckBox
                Content="{Binding Name}"
                IsChecked="{Binding IsChecked}" />
        </DataTemplate>
    </ComboBox.ItemTemplate>
</ComboBox>
Run Code Online (Sandbox Code Playgroud)

视图模型

// ObservableObject is a custom base class that implements INotifyPropertyChanged
internal class MainWindowVM : ObservableObject
{
    private ObservableCollection<Item> mItems;
    private HashSet<Item> mCheckedItems;

    public IEnumerable<Item> Items { get { return mItems; } }

    public string Text
    {
        get { return _text; }
        set { Set(ref _text, value); }
    }
    private string _text;

    public MainWindowVM()
    {
        mItems = new ObservableCollection<Item>();
        mCheckedItems = new HashSet<Item>();
        mItems.CollectionChanged += Items_CollectionChanged;

        // Adding test data
        for (int i = 0; i < 10; ++i)
        {
            mItems.Add(new Item(string.Format("Item {0}", i.ToString("00"))));
        }
    }

    private void Items_CollectionChanged(object sender, NotifyCollectionChangedEventArgs e)
    {
        if (e.OldItems != null)
        {
            foreach (Item item in e.OldItems)
            {
                item.PropertyChanged -= Item_PropertyChanged;
                mCheckedItems.Remove(item);
            }
        }
        if (e.NewItems != null)
        {
            foreach (Item item in e.NewItems)
            {
                item.PropertyChanged += Item_PropertyChanged;
                if (item.IsChecked) mCheckedItems.Add(item);
            }
        }
        UpdateText();
    }

    private void Item_PropertyChanged(object sender, PropertyChangedEventArgs e)
    {
        if (e.PropertyName == "IsChecked")
        {
            Item item = (Item)sender;
            if (item.IsChecked)
            {
                mCheckedItems.Add(item);
            }
            else
            {
                mCheckedItems.Remove(item);
            }
            UpdateText();
        }
    }

    private void UpdateText()
    {
        switch (mCheckedItems.Count)
        {
            case 0:
                Text = "<none>";
                break;
            case 1:
                Text = mCheckedItems.First().Name;
                break;
            default:
                Text = "<multiple>";
                break;
        }
    }
}

// Test item class
// Test item class
internal class Item : ObservableObject
{
    public string Name { get; private set; }

    public bool IsChecked
    {
        get { return _isChecked; }
        set { Set(ref _isChecked, value); }
    }
    private bool _isChecked;

    public Item(string name)
    {
        Name = name;
    }

    public override string ToString()
    {
        return Name;
    }
}
Run Code Online (Sandbox Code Playgroud)

如果可选文本存在问题,您可能需要创建自定义ComboBox控件模板(此处为默认示例).或者,您可以使用其他东西而不是a ComboBox,并使其看起来像ComboBox.

示例屏幕截图:

截图

  • 太棒了!Id已经进行了一些测试,但是IsReadonly = true与IsEditable = true的结合是一个非常聪明的技巧! (3认同)