WPF Data-Bound Datagrid Фильтрация / Поиск
У меня есть приложение WPF, которое использует (в настоящее время) локальную базу данных в качестве источника привязки. Используя инструменты Visual Studio 2010, у меня есть модель LINQ-SQL, которая выступает в качестве Datacontext для большинства форм.
У меня есть UserControl с TextBox и Datagrid. Сетка данных ItemSource устанавливается на событие UserControl.Loaded с таблицей. TextBox имеет событие, назначенное так, что запрос выполняется к базе данных, когда текст изменяется, а ItemSource обновляется на сетке данных.
Проблема в том, что для запроса базы данных требуется время. Поскольку я переназначаю источник данных DataGrid для каждого поиска.
Должен ли я загружать все записи при загрузке UserControl - есть ли способ загрузить записи асинхронно в BackgroundWorker или подобном?
Нужно ли переназначать DataGrid ItemsSource после каждого поиска или это более эффективный способ фильтрации данных?
Благодарю. Liam
<UserControl x:Class="Tracker.DocumentsView"
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>
<DataGrid AutoGenerateColumns="False" Margin="12,34,12,50" Name="dataGrid1">
<DataGrid.Columns>
<DataGridTextColumn Binding="{Binding Path=ID}" Header="ID" />
<DataGridTextColumn Binding="{Binding Path=Reference}" Header="Reference" />
<DataGridTextColumn Binding="{Binding Path=Subject}" Header="Subject" />
</DataGrid.Columns>
</DataGrid>
<TextBox HorizontalAlignment="Left" Margin="64,8,0,0" Name="txtSearchBox" VerticalAlignment="Top" Width="224" TextChanged="txtSearchBox_TextChanged" />
<TextBlock Text="Search" HorizontalAlignment="Left" Margin="11,12,0,0" Name="label1" VerticalAlignment="Top" Height="23" />
</Grid>
</UserControl>
Код:
using System.Windows.Data;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Imaging;
using System.Windows.Navigation;
using System.Windows.Shapes;
using Tracker.Model;
namespace Tracker
{
/// <summary>
/// Interaction logic for DocumentsView.xaml
/// </summary>
public partial class DocumentsView : UserControl
{
private TrackerDataContext db;
public DocumentsView()
{
InitializeComponent();
this.Loaded += new RoutedEventHandler(DocumentsView_Loaded);
}
void DocumentsView_Loaded(object sender, RoutedEventArgs e)
{
db = new TrackerDataContext();
dataGrid1.ItemsSource = db.Documents;
}
private void txtSearchBox_TextChanged(object sender, TextChangedEventArgs e)
{
TextBox textbox = sender as TextBox;
if (textbox != null)
{
string searchstr = textbox.Text;
if (!string.IsNullOrEmpty(searchstr))
{
var filtered = from document in db.Documents
where document.Subject.Contains(searchstr)
|| document.Reference.Contains(searchstr)
select document;
dataGrid1.ItemsSource = filtered;
}
else
{
dataGrid1.ItemsSource = db.Documents;
}
}
}
}
}
1 ответ
Я думаю, что вы должны загрузить все записи из базы данных в начале, а затем использовать ICollectionView.Filter на ItemsSource. Так что тогда вам не придется делать транзакции базы данных
Вы должны написать что-то подобное
private void txtSearchBox_TextChanged(object sender, TextChangedEventArgs e)
{
TextBox textbox = sender as TextBox;
if (textbox != null)
{
_searchstr = textbox.Text;
if (!string.IsNullOrEmpty(_searchstr))
{
ICollectionView view = CollectionViewSource.GetDefaultView(ItemsSource);
view.Filter = new Predicate<object>(filter);
}
}
}
private bool filter(object item)
{
if(item.Subject.Contains(_searchstr) || item.Reference.Contains(searchstr))
{
return true;
}
return false;
}
Надеюсь, это поможет, Нидал.