c# - 使用单独的数据库表在 DataGridComboBoxColumn 中设置 ItemSource

标签 c# wpf xaml combobox csla

我的第一个 StackExchange 问题,如果我做错了什么,请告诉我。

我正在将旧的 winforms 代码转换为 WPF,并且一直在使用它们。我大部分时间都取得了成功,但我正在处理我的第一个涉及数据绑定(bind)的数据表。这似乎是最难转换的事情,因为绑定(bind)主要是在 xaml 而不是代码中处理的。

作为附加说明,它使用 CSLA,据我了解,它是类似于 MVVM 的代码结构(仍在学习这些是什么)。

通过一些试验,我设法创建了一个正常工作的数据网格,但我需要将其中一列实现为组合框。它目前表示为一个数字(1-10),但我需要与数字相关的描述,该描述位于数据库的不同表中。
在切换到组合框之前最初显示的数字是无资格原因。我希望这个值成为组合框中当前选定的项目。

无法发布图片,所以这是我展示数据库表结构的尝试:

模组评论

  • 评论数
  • 因特卡诺
  • 新推荐
  • 屏幕日期
  • 符合条件
  • 无资格原因 (这是一个组合框)

  • 非理性 (这是与 nonligibilityreason 对齐的子表)
  • 非理性
  • noneligreasondesc

  • 代码包含这个函数,它似乎创建了一个集合,我认为应该是 ItemsSource:
    NERList.GetNameValueList();
    

    至于 ItemsSource,我想要所有的 nonligreasondesc 值。当然,还有一个问题是让 nonligibilityreason 与 nonligreasonid 相匹配。我找到了this example ,这似乎做同样的事情,但没有数据库,而是使用类:

    我试图复制它,因为我认为它应该在我的 XAML 中工作:
    <Window
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        xmlns:local="clr-namespace:WpfTestProject"       x:Class="WpfTestProject.MainWindow"
        Title="MainWindow" Height="350" Width="525" Loaded="Window_Loaded">
    <Window.Resources>
        <local:caseviewDataSet x:Key="caseviewDataSet"/>
        <CollectionViewSource x:Key="modreviewViewSource" 
                              Source="{Binding modreview, Source={StaticResource caseviewDataSet}}"/>
        <CollectionViewSource x:Key="noneligibilityViewSource"
                              Source="{Binding noneligreason, Source={StaticResource caseviewDataSet}}"/>
    </Window.Resources>
    <Grid DataContext="{StaticResource modreviewViewSource}">
        <Grid.RowDefinitions>
            <RowDefinition/>
        </Grid.RowDefinitions>
        <DataGrid x:Name="modreviewDataGrid" 
                  RowDetailsVisibilityMode="VisibleWhenSelected" 
                  ItemsSource="{Binding}" 
                  EnableRowVirtualization="True" 
                  AutoGenerateColumns="False">            
            <DataGrid.Columns>
                <DataGridTextColumn x:Name="reviewnumColumn" Width="SizeToHeader" Header="reviewnum" Binding="{Binding reviewnum}"/>
                <DataGridTextColumn x:Name="intcasenoColumn" Width="SizeToHeader" Header="intcaseno" Binding="{Binding intcaseno}"/>
                <DataGridCheckBoxColumn x:Name="newreferralColumn" Width="SizeToHeader" Header="newreferral" Binding="{Binding newreferral}"/>
                <DataGridTemplateColumn x:Name="screendateColumn" Width="Auto" Header="screendate">
                    <DataGridTemplateColumn.CellTemplate>
                        <DataTemplate>
                            <DatePicker SelectedDate="{Binding screendate, Mode=TwoWay, NotifyOnValidationError=true, ValidatesOnExceptions=true}"/>
                        </DataTemplate>
                    </DataGridTemplateColumn.CellTemplate>
                </DataGridTemplateColumn>
                <DataGridCheckBoxColumn x:Name="eligibleColumn" Width="SizeToHeader" Header="eligible" Binding="{Binding eligible}"/>
                <DataGridComboBoxColumn x:Name="noneligibilityreasonColumn" Width="SizeToHeader" Header="noneligibilityreason"
                                        ItemsSource="{Binding Source={StaticResource noneligibilityViewSource}}"
                                        SelectedValueBinding="{Binding noneligibilityreason}"
                                        DisplayMemberPath="Value"
                                        SelectedValuePath="Key"
                                            />
    
            </DataGrid.Columns>
        </DataGrid>
    </Grid>
    

    以及背后的代码:
    using CslaFactoryBusinessObjects;
    ...
    
    namespace WpfTestProject
    {
        public partial class MainWindow : Window
        {
            public MainWindow()
            {
                InitializeComponent();
            }
    
            private void Window_Loaded(object sender, RoutedEventArgs e)
            {
    
                WpfTestProject.caseviewDataSet caseviewDataSet = ((WpfTestProject.caseviewDataSet)(this.FindResource("caseviewDataSet")));
                // Load data into the table modreview. You can modify this code as needed.
                WpfTestProject.caseviewDataSetTableAdapters.modreviewTableAdapter caseviewDataSetmodreviewTableAdapter = new WpfTestProject.caseviewDataSetTableAdapters.modreviewTableAdapter();
                caseviewDataSetmodreviewTableAdapter.Fill(caseviewDataSet.modreview);
                System.Windows.Data.CollectionViewSource modreviewViewSource =     ((CollectionViewSource)(this.FindResource("modreviewViewSource")));
                modreviewViewSource.View.MoveCurrentToFirst();
    
                noneligibilityreasonColumn.ItemsSource = NERList.GetNameValueList();
        }
    }
    

    }

    大部分代码是在我从数据源拖放时生成的,但后面代码的最后一行是我认为应该添加 ItemsSource 的地方。不确定它是否属于 Loaded,但似乎在那里可能没问题。也不确定是否有办法在 XAML 中执行此操作。

    我意识到我在 XAML 和代码中重复了设置 ItemsSource,但没有一个可以正常工作,所以我将两者都包括在内以显示我尝试过的选项。

    最后,我想我应该展示我试图在 WPF 转换中模拟的旧 winforms 代码(不确定这是否足够的代码)。我认为它使用隐藏的组合框来设置绑定(bind),然后将其添加到表中:
    //from Program.cs used in setupModRvwGrdHdr()
    public static void ListControlBinding(ref UltraCombo comboBox, object lkupdataSource, string displayMember,
                                              string valueMember, object objDataSource, string objProp) {
            comboBox.DataSource = lkupdataSource;
            comboBox.DisplayMember = displayMember;
            if (!string.IsNullOrEmpty(valueMember))
                comboBox.ValueMember = valueMember;
            if (objDataSource != null)
                comboBox.DataBindings.Add("Value", objDataSource, objProp);
        }
    
    
    //from the code for the specific winform
    private void setupModRvwGrdHdr() {
            cbNonEligReason.DataBindings.Clear();
            grdModReviews.DataSource = bsModRvws;
    
            Program.ListControlBinding(ref cbNonEligReason, NERList.GetNameValueList(), "Value", "Key", bsModRvws,
                "NonEligibleReasonIDStr");
            cbNonEligReason.DisplayLayout.Bands[0].Columns["Key"].Hidden = true;
            cbNonEligReason.DisplayLayout.Bands[0].Columns["Value"].Header.Caption = "Noneligiblity Reason";
    
            grdModReviews.DisplayLayout.Bands[0].Columns["reviewnum"].CellActivation = Activation.NoEdit;
            grdModReviews.DisplayLayout.Bands[0].Columns["intcaseno"].Hidden = true;
            grdModReviews.DisplayLayout.Bands[0].Columns["noneligibilityreason"].Hidden = true;
    
            grdModReviews.DisplayLayout.Bands[0].Columns["screendate"].Hidden = false;
            grdModReviews.DisplayLayout.Bands[0].Columns["screendate"].Header.Caption = "Date of Screen";
            grdModReviews.DisplayLayout.Bands[0].Columns["screendate"].Width = 100;
            grdModReviews.DisplayLayout.Bands[0].Columns["screendate"].EditorComponent = dteModRvwDate;
    
            grdModReviews.DisplayLayout.Bands[0].Columns["eligible"].Hidden = false;
            grdModReviews.DisplayLayout.Bands[0].Columns["eligible"].Header.Caption = "Eligible";
            grdModReviews.DisplayLayout.Bands[0].Columns["eligible"].Width = 70;
    
            grdModReviews.DisplayLayout.Bands[0].Columns["NonEligibleReasonIDStr"].Hidden = false;
            grdModReviews.DisplayLayout.Bands[0].Columns["NonEligibleReasonIDStr"].Header.Caption =
                "Reason for Noneligibility";
            grdModReviews.DisplayLayout.Bands[0].Columns["NonEligibleReasonIDStr"].Width = 250;
            grdModReviews.DisplayLayout.Bands[0].Columns["NonEligibleReasonIDStr"].EditorComponent = cbNonEligReason;
            grdModReviews.DisplayLayout.Bands[0].Columns["NonEligibleReasonIDStr"].Nullable =
                Infragistics.Win.UltraWinGrid.Nullable.Nothing;
        }
    

    这是一个相当大的程序,CslaFactoryBusinessObjects 类中可能有一些有用的函数,但我认为我应该学习如何执行此操作以更好地理解 WPF 中的数据操作。

    几天来我一直在寻找解决方案,但没有找到与我的情况足够相似的案例。我只是不确定每个绑定(bind)属性是如何工作的,以及它们在这种特定情况下如何应用。最后只好认输,在这里做个账。抱歉,篇幅较长,但我想具体一点,并表明我已经有一段时间了。

    请帮忙!

    最佳答案

    这里有很多有用的信息,也是 SO 问题的良好开端。

    给定您的 XAML 绑定(bind):

    <DataGridComboBoxColumn x:Name="noneligibilityreasonColumn" Width="SizeToHeader" Header="noneligibilityreason"
                                        ItemsSource="{Binding Source={StaticResource noneligibilityViewSource}}"
                                        SelectedValueBinding="{Binding noneligibilityreason}"
                                        DisplayMemberPath="Value"
                                        SelectedValuePath="Key"
                                            />
    

    这就是说(对于来自 DataGrid 的每一行 - 这又是 DataSet 内的 DataTable 中的 modreview 对象/行)组合框控件应该使用 noneligibilityViewSource为其可选项目列表。

    (让我们暂时忽略 Window_Loaded 事件中项目来源的矛盾设置)。

    它还说,对于组合框的显示(在控件中直观地显示)应该来自 DisplayMemberPath 中指定的“值”属性。 .这将对应于 ItemSource 指定的集合中的 item 的相同命名属性。 .

    由于组合框的项目由 ItemSource 绑定(bind)提供,因此将来自 noneligibilityViewSource ,所以下一个问题是这个 noneligibilityViewSource 里面是什么?

    您已将其声明为此窗口的资源:
    <CollectionViewSource x:Key="noneligibilityViewSource" Source="{Binding noneligreason, Source={StaticResource caseviewDataSet}}" />
    

    以上说明 CollectionViewSource 实例(又名 noneligibilityViewSource)来自一个名为 noneligreason 的属性。在 caseviewDataSet .鉴于 DataSets 的工作方式,我期待一个名为 noneligreason 的 DataTable或带有为此添加的属性的自定义数据集。很可能是前者。

    现在,Window_Loaded 中有矛盾的代码。以编程方式将组合框的 ItemSource 设置为其他内容的事件。具体到调用 NERList.GetNameValueList(); 的结果.我说是矛盾的,因为 XAML 资源声明说这个非资格值列表来自 DataSet 上相应命名的属性,而事件中的代码说使用 CSLA 业务对象列表。

    你必须弄清楚你想要/应该使用哪一个。

    PS:很可能,如果 DataSet 中的非资格属性包含数据,则无需通过调用 NERList.GetNameValueList(); 来支付再次访问数据库的性能损失。因为你已经有了可用的数据。

    一旦您确定了哪个“来源”包含您的组合框的项目列表 - 即。 noneligreason caseviewDataSet 中的数据表或从调用 NERList.GetNameValueList(); 返回的 CSLA 业务对象列表中- 只有这样,您才能知道 DisplayMemberPath 应该使用哪个属性以及 SelectedValuePath 应该使用哪个属性。

    因此,如果组合框的 ItemSource 符合声明的 XAML 并且 caseviewDataSet 有另一个名为 noneligreason 的 DataTable ,然后您需要从中找出属性的名称。例如,它可以是 noneligreasonidnoneligreasondesc ,但它可能是其他东西,具体取决于表适配器的功能。您的装订DisplayMemberPath那么可能是 noneligreasondesc和你的SelectedValuePath那么可能是 noneligreasonid .

    如果组合框的 ItemSource 应该来自对 NERList.GetNameValueList(); 的调用然后,您需要确定返回的对象的属性名称是什么。根据命名约定,我猜它是“名称”和“值”,这意味着 DisplayMemberPath应该是 ValueSelectedValuePath应设置为 Name .猜测是不好的,所以去看看那个对象,或者使用调试器来检查值。
    SelectedValueBinding属性是指 modreview 上的属性应该包含所选组合框项的值的行对象,特别是 SelectedValuePath组合框项目的属性将应用于 SelectedValueBinding modreview 上的属性(property)实例。

    以下 MSDN 文档应该可以帮助您了解 DataGridComboBoxColumn 上的各种属性的用途。 https://msdn.microsoft.com/en-us/library/system.windows.controls.datagridcomboboxcolumn(v=vs.110).aspx

    因此,例如,您的 XAML 可以更改为以下两个声明之一:
    <DataGridComboBoxColumn x:Name="noneligibilityreasonColumn" Width="SizeToHeader" Header="noneligibilityreason"
                                            ItemsSource="{Binding Source={StaticResource noneligibilityViewSource}}"
                                            SelectedValueBinding="{Binding noneligibilityreason}"
                                            DisplayMemberPath="noneligreasondesc"
                                            SelectedValuePath="noneligreasonid"
                                                />
    

    或者
    <DataGridComboBoxColumn x:Name="noneligibilityreasonColumn" Width="SizeToHeader" Header="noneligibilityreason"
                                            ItemsSource="{Binding Source={StaticResource noneligibilityViewSource}}"
                                            SelectedValueBinding="{Binding noneligibilityreason}"
                                            DisplayMemberPath="Value"
                                            SelectedValuePath="Name"
                                                />
    

    希望有帮助。

    关于c# - 使用单独的数据库表在 DataGridComboBoxColumn 中设置 ItemSource,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31482083/

    相关文章:

    wpf - 将样式应用于 ListBoxItem 而不影响内部的 ComboboxItem

    xaml - 使用资源定义厚度

    wpf - 阴影出现在文本 block 本身而不是单个文本字符周围

    c# - 是否可以使用反射/TypeDescriptor 设置匿名类型属性的值?

    c# - 使用我无法修改的类的私有(private)构造函数模拟对象

    c# - 正则表达式获取字符串中的最后一个冒号及其后的所有内容

    c# - 应用程序无缘无故地停止响应

    c# - 在 Windows Phone 8 中选择联系人

    WPF:文本编辑器控件,可以通过拖放重新排列段落(如 OneNote)

    wpf - 如何使WPF行缩小以适合内容