gpt4 book ai didi

c# - WPF 组合框对更改不具有约束力

转载 作者:太空宇宙 更新时间:2023-11-03 21:22:50 25 4
gpt4 key购买 nike

我有3个组合框

<Grid>
<ComboBox Name="cbo1" SelectionChanged="OnComboBoxChanged" />
<ComboBox Name="cbo2" SelectionChanged="OnComboBoxChanged"/>
<ComboBox Name="cbo3" SelectionChanged="OnComboBoxChanged" />

组合框的列表是 { a,b,c,d}因此,如果在第一个框中选择了“b”,则下拉菜单不应包含 b,并且如果第二个设置为 a,则需要使用 {a,c,d} 进行更新,那么最后一个需要包含 {c, d}。如果他们回去更改任何内容,我们需要相应地更新列表。我添加了一个 oncomboboxchanged 事件,但当我将项目源设置为新列表时,它没有更新组合框。

private List<string> comboList = new List<string>();
string[] defaultParam = { "A", "B", "C", "D" };

public MainWindow()
{
InitializeComponent();
foreach(string s in defaultParam)
{
LoadCombo(s);
}

}
public void LoadCombo(string name)
{
comboList.Add(name);
cbo1.ItemsSource = comboList;
cbo2.ItemsSource = comboList;
cbo3.ItemsSource = comboList;
}
private void OnComboBoxChanged(object sender,SelectionChangedEventArgs e)
{
var combo = sender as ComboBox;
string oldval = combo.Text;
string id = combo.Name;
string itemSel = (sender as ComboBox).SelectedItem.ToString();
comboList.Remove(itemSel);
//add old value only if it is not empty
if (!string.IsNullOrEmpty(oldval))
{
comboList.Add(oldval);
}
combo.ItemsSource = comboList;
ComboBox[] comboNameLst = {cbo1,cbo2,cbo3 };
foreach (ComboBox cbo in comboNameLst)
{
if (id != cbo.Name)
{
if (cbo.SelectedItem == null)
{
cbo.ItemsSource = comboList;
}
else if (cbo.SelectedItem != null)
{
string tempitemsel = cbo.SelectedItem.ToString();
comboList.Add(tempitemsel);
cbo.ItemsSource = comboList;
comboList.Remove(tempitemsel);
}
}
}
}

所以 cbo.ItemSource 没有做任何事情,我是否需要做任何不同的事情才能看到更新。

最佳答案

  1. 您需要在 XAML 中使用绑定(bind),而不是在您的代码后面设置 ItemsSource。还有数据绑定(bind)SelectedItem:

     <Grid>
    <ComboBox ItemsSource="{Binding DefaultList}" SelectedItem="{Binding SelectedItem_Cob1}"/>
    <ComboBox ItemsSource="{Binding FilteredListA}" SelectedItem="{Binding SelectedItem_Cob2}"/>
    <ComboBox ItemsSource="{Binding FilteredListB}" SelectedItem="{Binding SelectedItem_Cob3}"/>
    </Grid>
  2. 在后面的代码中,您需要实现INotifyPropertyChanged;将相关的 ItemsSources 和 SlectedItems 定义为属性;并将 Windows 的 DataContext 设置为您的代码本身(您应该使用 MVVM 模式,但稍后您可能会担心):

    using System.ComponentModel;
    public partial class MainWindow: INotifyPropertyChanged
    {
    string[] defaultParam = { "A", "B", "C", "D" };
    private string _selecteditem_cob1;
    private string _selecteditem_cob2;
    private string _selecteditem_cob3;

    public List<string> DefaultList
    {
    get { return defaultParam.ToList(); }
    }

    public string SelectedItem_Cob1
    {
    get { return _selecteditem_cob1; }
    set
    {
    if (_selecteditem_cob1 != value)
    {
    _selecteditem_cob1 = value;
    RaisePropertyChanged("SelectedItem_Cob1");
    RaisePropertyChanged("FilteredListA");
    RaisePropertyChanged("FilteredListB");
    }
    }
    }

    public string SelectedItem_Cob2
    {
    get { return _selecteditem_cob2; }
    set
    {
    if (_selecteditem_cob2 != value)
    {
    _selecteditem_cob2 = value;
    RaisePropertyChanged("SelectedItem_Cob2");
    RaisePropertyChanged("FilteredListB");
    }
    }
    }

    public string SelectedItem_Cob3
    {
    get { return _selecteditem_cob3; }
    set
    {
    if (_selecteditem_cob3 != value)
    {
    _selecteditem_cob3 = value;
    RaisePropertyChanged("SelectedItem_Cob3");
    }
    }
    }

    public List<string> FilteredListA
    {
    get { return defaultParam.ToList().Where(a=>a!=SelectedItem_Cob1).ToList(); }
    }
    public List<string> FilteredListB
    {
    get { return FilteredListA.Where(a => a != SelectedItem_Cob2).ToList(); }
    }
    public MainWindow()
    {

    InitializeComponent();
    this.DataContext=this;
    }

    //Implementation for INotifyPropertyChanged
    public event PropertyChangedEventHandler PropertyChanged;

    protected virtual void OnPropertyChanged(PropertyChangedEventArgs e)
    {
    var handler = this.PropertyChanged;
    if (handler != null)
    {
    handler(this, e);
    }
    }

    protected void RaisePropertyChanged(String propertyName)
    {
    OnPropertyChanged(new PropertyChangedEventArgs(propertyName));
    }
    }

结果:

Three ComboBoxes will all show A,B,C,D at the initial stage. And then if user made selections cbo2 and cbo3 will only display filtered result dynamically.

I realized this is not 100% what you want (thanks to @TheodosiusVonRichthofen), but I feel you can still use this, and be able to easily modify it to suit your needs.

关于c# - WPF 组合框对更改不具有约束力,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29494456/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com