在WPF C#中使用数据绑定更改组合框的选定索引

3

我正在尝试使用c#从Winforms转换到WPF。 我对MVVM数据绑定有些困惑。 我承认我没有完全掌握所有组件之间的联系,但我认为我只需要更多的练习。

我使用这篇文章在codeproject上来帮助我理解如何将我的本地通讯录应用程序从WinForms转换为WPF。

虽然我可以很好地运用数据绑定来绑定整数和字符串,但是我在ComboBox上遇到了一些问题。 数据存储在xml文件中,并在contacts类中以字符串和整数形式存储。 当从列表框中选择位置时,ComboBox应该自动更新。 对于允许用户自定义信息的属性,这已经全部工作正常。 但是,此特定设置需要下拉菜单以供预设值选择。

我该如何转换字符串并通过数据绑定将SelectedIndex[]属性传递给comboBox?

我省略了用于填充文本框的一些额外工作代码,以使其更加简洁。

XAML

<Window
    <Window.DataContext>
        <viewmodels:mainviewModel x:Name="viewModel1"/>
    </Window.DataContext>
    <Grid Background="#FF8B8686">
        <ListBox x:Name="listBox" HorizontalAlignment="Left" Height="174" Margin="44,77,0,0" VerticalAlignment="Top" Width="164" ItemsSource="{Binding Contacts}" DisplayMemberPath="Name" SelectedItem="{Binding SelectedContact}" SelectionChanged="listBox_SelectionChanged" />
        <TextBox x:Name="textBoxName" HorizontalAlignment="Left" Height="23" Margin="244,77,0,0" TextWrapping="Wrap" Text="{Binding SelectedContact.Name}" VerticalAlignment="Top" Width="120"/>
        <TextBox x:Name="textBoxAddress" HorizontalAlignment="Left" Height="23" Margin="244,120,0,0" TextWrapping="Wrap" Text="{Binding SelectedContact.Address}" VerticalAlignment="Top" Width="120"/>
        <TextBox x:Name="textBoxBandwidth" HorizontalAlignment="Left" Height="23" Margin="244,198,0,0" TextWrapping="Wrap" Text="{Binding SelectedContact.Bandwidth}" VerticalAlignment="Top" Width="120" DataContext="{Binding Mode=OneWay}"/>
        <Button x:Name="button1" Content="Write Data" HorizontalAlignment="Left" Margin="194,271,0,0" VerticalAlignment="Top" Width="75" Click="button1_Click"/>
        <ComboBox x:Name="comboBox" HorizontalAlignment="Left" Margin="267,229,0,0" VerticalAlignment="Top" Width="120" SelectedItem="{Binding SelectedContact.Combo}">
            <ComboBoxItem Content="1" />
            <ComboBoxItem Content="2" />
            <ComboBoxItem Content="3" />
            <ComboBoxItem Content="4" />
            </ComboBox>
    </Grid>
</Window>

Contact.cs

public class Contact : ViewModelEntity
{
    public Contact()
    {
    }

    protected int combo;
    public int Combo
    {
        get { return combo; }
        set
        {
            if (combo != value)
            {
                switch (value)
                {
                    case 1:
                        combo = ComboBox.SelectedIndex[0];
                        break;
                    case 2:
                        combo = ComboBox.SelectedIndex[1];
                        break;
                    case 3:
                        combo = ComboBox.SelectedIndex[2];
                        break;
                    case 4:
                        combo = ComboBox.SelectedIndex[3];
                        break;
                    default:
                        combo = ComboBox.SelectedIndex[0];
                        break;
                }
                // combo = value;
                NotifyPropertyChanged("Combo");
            }
        }
    }

mainviewModel.cs

using System.Collections.ObjectModel;

namespace WpfApplication1.viewmodels
{
    public class mainviewModel:ViewModelEntity
    {

       public mainviewModel()
        { }
        protected Contact selectedContact = null;

        protected ObservableCollection<Contact> contacts = new ObservableCollection<Contact>();

        public ObservableCollection<Contact> Contacts
        {
            get { return contacts; }
            set { contacts = value; }
        }


        public Contact SelectedContact
        {
            get { return selectedContact; }
            set {
                if (selectedContact != value)
                {
                    selectedContact = value;
                    NotifyPropertyChanged("SelectedContact");
                }


            }
        }
    }
}
1个回答

3
为什么不直接将它绑定到SelectedIndex
    <ComboBox x:Name="comboBox"
              Width="120"
              Margin="267,229,0,0"
              HorizontalAlignment="Left"
              VerticalAlignment="Top"
              SelectedIndex="{Binding SelectedContact.Combo}">
        <ComboBoxItem Content="1" />
        <ComboBoxItem Content="2" />
        <ComboBoxItem Content="3" />
        <ComboBoxItem Content="4" />
    </ComboBox>

在代码后端:
protected int combo;
public int Combo
{
    get { return combo; }
    set
    {
        combo = value;
        NotifyPropertyChanged("Combo");        
    }
}

这是一个如此简单的解决方案!感谢您指出这一点。我仍在努力习惯以这种方式思考控件。我从未想过只需将绑定移动到另一个属性即可。我一直认为selecteditem是唯一可以绑定的属性,但实际上并非如此!非常感谢您向我展示我的错误,并让我意识到我过于复杂化了它。 - eatumup
没关系。当你习惯于使用WPF时,你会像我一样思考。实际上,对于每个问题,都有几种解决方案。我们需要找出哪一个是最好的且耗时最少的。就这样。 - ViVi

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接