我有两个列表框。每个都从 SQLight 查询中获取数据。用户可以通过 dbl-click 项目进行选择,它应该将其从 LB1 中删除并将其放置在 LB2 中。用户可以选择从 LB2 中删除,并且应该移回 LB1,直到他们对选择感到满意为止。我正在使用 LB2 中保存的数据来过滤掉在 LB1 中找到的任何项目(根据代码)。一旦用户对所选项目感到满意,LB2 数据就会保存回 DB。用户可以随时返回 UI 并根据需要添加/删除项目。
我正在使用 2 x ObservableCollections 来填充 LB1 和 LB2。我正在使用模板来控制两个 LB 的布局。IE 一旦项目在列中说 10 个项目,它就会“溢出”到 LB 中的新列中。
我可以让 LB1 正确填充,甚至将项目移动到 LB2,但是当我想再次将其移回时,我得到......
"Operation is not valid while ItemsSource is in use. Access and modify elements with ItemsControl.ItemsSource instead."
我是 WPF C# 的新手,需要有关如何解决此问题的明确说明。(现在已经努力解决这个问题了 4 周)。该解决方案需要适用于两个 ListBox。对此的任何帮助将不胜感激。
请查看附加的代码并发表评论...
CS...
using System;
using System.Windows;
using System.Windows.Input;
using System.Windows.Controls;
using System.Data;
using System.Data.SQLite;
using System.Collections;
using System.Collections.ObjectModel;
using System.Collections.Generic;
using PTWS.Class_Lib;
using System.ComponentModel;
using System.Windows.Data;
using System.Windows.Media;
namespace PTWS.MainPanelControls.FormControls.Permits
{
/// <summary>
/// Interaction logic for ColdWorkDraftUControl.xaml
/// </summary>
public partial class ColdWorkDraftUControl : UserControl
{
PTWDatabase db = new PTWDatabase();
ObservableCollection<CertificateLookup> CertificateLookupList = new ObservableCollection<CertificateLookup>();
ObservableCollection<CertificateSelected> CertificateSelectedList = new ObservableCollection<CertificateSelected>();
public ColdWorkDraftUControl()
{
InitializeComponent();
LoadDataCertificateLookup();
// all works great untill I uncomment the next line
//LoadDataCertificateSelected();
}
private void LoadDataCertificateLookup()
{
try
{
DataTable conditions;
String query = "select lc.Section \"Section\""
+ ", lc.Description \"Description\""
+ ", lc.SortOrder \"SortOrder\" "
+ "from LookupConditions lc "
+ "where not exists (select 1 from SelectedConditions sc where sc.Code = lc.Code and sc.PermitID = 'CWP-12-00001') "
+ "and lc.Section = 'Certificates'";
conditions = db.GetDataTable(query);
foreach (DataRow r in conditions.Rows)
{
CertificateLookupList.Add(new CertificateLookup()
{
Section = r["Section"].ToString(),
Description = r["Description"].ToString(),
SortOrder = r["SortOrder"].ToString()
});
}
listBoxCertificateLookup.ItemsSource = CertificateLookupList;
}
catch (Exception fail)
{
String error = "The following error has occurred:\n\n";
error += fail.Message.ToString() + "\n\n";
MessageBox.Show(error);
}
}
void LoadDataCertificateSelected()
{
try
{
DataTable conditions;
String query = "select Section \"Section\""
+ ", Description \"Description\""
+ ", SortOrder \"SortOrder\"";
query += "from SelectedConditions where PermitID = 'CWP-12-00001' and Section = 'Certificates'";
conditions = db.GetDataTable(query);
foreach (DataRow r in conditions.Rows)
{
CertificateSelectedList.Add(new CertificateSelected()
{
selectedSection = r["Section"].ToString(),
selectedDescription = r["Description"].ToString(),
selectedSortOrder = r["SortOrder"].ToString()
});
}
listBoxCertificateSelected.DataContext = CertificateSelectedList;
}
catch (Exception fail)
{
String error = "The following error has occurred:\n\n";
error += fail.Message.ToString() + "\n\n";
MessageBox.Show(error);
}
}
private void listBoxCertificateLookup_MouseDoubleClick(object sender, MouseButtonEventArgs e)
{
try
{
ListBoxItem myListBoxItem =
(ListBoxItem)(listBoxCertificateLookup.ItemContainerGenerator.ContainerFromItem(listBoxCertificateLookup.Items.CurrentItem));
// listBoxCertificateSelected.DataContext = null;
listBoxCertificateSelected.Items.Add(myListBoxItem);
}
catch (Exception fail)
{
String error = "The following error has occurred:\n\n";
error += fail.Message.ToString() + "\n\n";
MessageBox.Show(error);
}
}
private void listBoxCertificateSelected_MouseDoubleClick(object sender, MouseButtonEventArgs e)
{
try
{
ListBoxItem myListBoxItem =
(ListBoxItem)(listBoxCertificateSelected.ItemContainerGenerator.ContainerFromItem(listBoxCertificateSelected.Items.CurrentItem));
//listBoxCertificateLookup.DataContext = null;
listBoxCertificateLookup.Items.Add(myListBoxItem);
}
catch (Exception fail)
{
String error = "The following error has occurred:\n\n";
error += fail.Message.ToString() + "\n\n";
MessageBox.Show(error);
}
}
}
}
XAML...
<UserControl x:Class="PTWS.MainPanelControls.FormControls.Permits.ColdWorkDraftUControl"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
mc:Ignorable="d"
d:DesignHeight="300" d:DesignWidth="300">
<Grid>
<ListBox Name="listBoxCertificateLookup"
ItemsSource="{Binding CertificateSelectedList}"
MouseDoubleClick="listBoxCertificateLookup_MouseDoubleClick"
IsSynchronizedWithCurrentItem="True"
HorizontalAlignment="Left" Width="300" Height="75" VerticalAlignment="Top" >
<ListBox.ItemTemplate>
<DataTemplate>
<StackPanel Height="23" Orientation="Horizontal">
<TextBlock Text="{Binding Path=Description}" VerticalAlignment="Top" />
</StackPanel>
</DataTemplate>
</ListBox.ItemTemplate>
<ListBox.ItemsPanel>
<ItemsPanelTemplate>
<WrapPanel IsItemsHost="True" Orientation="Vertical" />
</ItemsPanelTemplate>
</ListBox.ItemsPanel>
</ListBox>
<ListBox Name="listBoxCertificateSelected"
ItemsSource="{Binding}"
MouseDoubleClick="listBoxCertificateSelected_MouseDoubleClick"
HorizontalAlignment="Left" Width="300" Height="75" VerticalAlignment="Top" Margin="0,153,0,0">
<ListBox.ItemTemplate>
<DataTemplate>
<StackPanel Height="23" Orientation="Horizontal">
<TextBlock Name="textBlock" Text="{Binding Path=selectedDescription}" VerticalAlignment="Top" />
</StackPanel>
</DataTemplate>
</ListBox.ItemTemplate>
<ListBox.ItemsPanel>
<ItemsPanelTemplate>
<WrapPanel IsItemsHost="True" Orientation="Vertical" />
</ItemsPanelTemplate>
</ListBox.ItemsPanel>
</ListBox>
</Grid>
</UserControl>
班级...
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace PTWS.Class_Lib
{
class CertificateLookup
{
public string Section { get; set; }
public string Description {get; set;}
public string SortOrder { get; set; }
public string selectedSection { get; set; }
public string selectedDescription { get; set; }
public string selectedSortOrder { get; set; }
}
}
已修改 CS 文件 mouseDoubleClick 事件(针对两个列表框进行了适当修改)...
CertificateLookup myListBoxItem = (CertificateLookup)((ListBox)sender).SelectedItem;
CertificateSelectedList.Add(new CertificateSelected()
{
selectedDescription = myListBoxItem.Description
});
CertificateLookupList.Remove(myListBoxItem);