我的第一个 StackExchange 问题,如果我做错了什么,请告诉我。
我正在将旧的 winforms 代码转换为 WPF,并且一直在使用它们。我大部分时间都取得了成功,但我正在处理我的第一个涉及数据绑定(bind)的数据表。这似乎是最难转换的事情,因为绑定(bind)主要是在 xaml 而不是代码中处理的。
作为附加说明,它使用 CSLA,据我了解,它是类似于 MVVM 的代码结构(仍在学习这些是什么)。
通过一些试验,我设法创建了一个正常工作的数据网格,但我需要将其中一列实现为组合框。它目前表示为一个数字(1-10),但我需要与数字相关的描述,该描述位于数据库的不同表中。
在切换到组合框之前最初显示的数字是无资格原因。我希望这个值成为组合框中当前选定的项目。
无法发布图片,所以这是我展示数据库表结构的尝试:
模组评论
非理性 (这是与 nonligibilityreason 对齐的子表)
代码包含这个函数,它似乎创建了一个集合,我认为应该是 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 ,然后您需要从中找出属性的名称。例如,它可以是 noneligreasonid
和 noneligreasondesc
,但它可能是其他东西,具体取决于表适配器的功能。您的装订DisplayMemberPath
那么可能是 noneligreasondesc
和你的SelectedValuePath
那么可能是 noneligreasonid
.如果组合框的 ItemSource 应该来自对
NERList.GetNameValueList();
的调用然后,您需要确定返回的对象的属性名称是什么。根据命名约定,我猜它是“名称”和“值”,这意味着 DisplayMemberPath
应该是 Value
和 SelectedValuePath
应设置为 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/