0

我有一个绑定到 ObservableCollection ProductsFound 的 DataGrid,它在我的 ViewModel 中作为属性公开。通过在 TextBox 中键入文本,模型中包含的具有 Code 属性(包含插入到 TextBox 中的文本)的产品将添加到 ProductsFound。

我发现如果 DataGrid 包含在 StackPanel 或 TabItem 等任何控件中,当我尝试在 TextBox 中键入文本时,Window(程序)会停止响应;而如果 DataGrid 不包含在任何控件中,则一切正常运行。

这是窗口的代码:

public class MainWindow : Window
{
    public MainWindow()
    {
        InitializeComponent();

        // This method just fill the dataset I pass to the model's contructor in the next line.
        Init();
        ProductsModel model = new ProductsModel(dataSet);
        searchViewModel = new ProductsSearchViewModel(model);

        DataContext = searchViewModel;           
    }

    private ProductsSearchViewModel searchViewModel;

   // This handler supports the binding between the TextBox and the MatchText property of the View Model.
    private void TextBox_TextChanged(object sender, TextChangedEventArgs e)
    {
        var binding = ((TextBox)sender).GetBindingExpression(TextBox.TextProperty);
        binding.UpdateSource();
    }

}

这是我的 ViewModel:

public class ProductsSearchViewModel : Notifier, IProductsSearchViewModel
{
    public ProductsSearchViewModel(IProductsModel inModel)
    {
        model = inModel;
        productsFound = new ObservableCollection<ProductViewModel>();       
    }

    private string matchText;
    private IProductsModel model;
    private ObservableCollection<ProductViewModel> productsFound;

    // This is a helper method that search for the products in the model and adds them to ProductsFound.
    private void Search(string text)
    {
        Results.Clear();

        foreach (Product product in model.Products)
        {                
            if (product.Code.ToLower().Contains(text.ToLower()))
                Results.Add(new ProductViewModel(product));
        }
    }

    public string MatchText
    {
        get { return matchText; }

        // This setter is meant to be executed every time the Text property of the TextBox is changed.
        set
        {
            if ((value != matchText) && (value != ""))
            {
                matchText = value;

                // This raises INotifyPropertyChanged.PropertyChaged.
                NotifyPropertyChanged("MatchText");
                Search(value);                  
            }
        }
    }

    public ObservableCollection<ProductViewModel> ProductsFound
    {
        get
        {
            return productsFound;
        }
        set
        {
            productsFound = value;
            NotifyPropertyChanged("Results");
        }
    }
}

这是 XAML:

<Window x:Class="MyNameSpace.UI.MainWindow"
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
    Title="MainWindow" Height="350" Width="525">
<Grid>
    <StackPanel>
        <TextBox Text="{Binding MatchText, Mode=TwoWay}" TextChanged="TextBox_TextChanged"  />
        <DataGrid x:Name="grid1" ItemsSource="{Binding Results}" >
    </StackPanel>
</Grid>

使用 StackPanel,当我尝试在文本框中键入文本并且没有项目添加到 DataGrid 时,程序停止响应;但如果我删除它一切运行正常。问题可能是什么?我是否遗漏了 WPF 绑定系统的工作原理?我的视图模型编码错误吗?

提前致谢。

4

1 回答 1

2

把它放在StackPanel那里会阻止DataGrid获取特定的Height,因此它只会扩展到无穷大,这会破坏 UI 虚拟化。

从那里删除StackPanel并使用非无限容器,例如Gridor DockPanel

于 2013-06-14T18:16:22.490 回答