6

使用绑定到绑定到LINQ to SQL类的 BindingSource 控件的 datagridview,我想知道如何将 bindingSource 定位到特定记录,也就是说,当我在文本框中键入产品名称时,绑定源应该移动到该特定产品。这是我的代码:

在我的表格 FrmFind 中:

    NorthwindDataContext dc;
    private void FrmFind_Load(object sender, EventArgs e)
    {
        dc = new NorthwindDataContext();

        var qry = (from p in dc.Products
                   select p).ToList();

        FindAbleBindingList<Product> list = new FindAbleBindingList<Product>(qry);

        productBindingSource.DataSource = list.OrderBy(o => o.ProductName);
    }

    private void textBox1_TextChanged(object sender, EventArgs e)
    {
        TextBox tb = sender as TextBox;

        int index = productBindingSource.Find("ProductName", tb.Text);

        if (index >= 0)
        {
            productBindingSource.Position = index;
        }
    }

在程序类中:

    public class FindAbleBindingList<T> : BindingList<T>
    {

        public FindAbleBindingList()
            : base()
        {
        }

        public FindAbleBindingList(List<T> list)
            : base(list)
        {
        }

        protected override int FindCore(PropertyDescriptor property, object key)
        {
            for (int i = 0; i < Count; i++)
            {
                T item = this[i];
                //if (property.GetValue(item).Equals(key))
                if (property.GetValue(item).ToString().StartsWith(key.ToString()))
                {
                    return i;
                }
            }
            return -1; // Not found
        }
    }

如何实现 find 方法以使其工作?

4

3 回答 3

16

您可以将BindingSource.Find()方法与Position属性结合起来。

例如,如果您的 TextBox changed 事件处理程序中有类似的内容:

private void textBox1_TextChanged(object sender, EventArgs e)
{
    TextBox tb = sender as TextBox;
    int index = bs.Find("Product", tb.Text);

    if (index >= 0)
    {
        bs.Position = index;
    }
}

这当然取决于很多事情,例如绑定源的数据源所具有的 Find 方法的特定实现。

在你刚才问的一个问题中,我给了你一个 Find 的实现,它适用于完全匹配。下面是一个稍微不同的实现,它将查看被检查属性的开始:

protected override int FindCore(PropertyDescriptor property, object key)
{
    // Simple iteration:
    for (int i = 0; i < Count; i++)
    {
        T item = this[i];
        if (property.GetValue(item).ToString().StartsWith(key.ToString()))
        {
            return i;
        }
    }
    return -1; // Not found
}

请注意,上述方法区分大小写 - 如果需要,您可以将 StartsWith 更改为不区分大小写。


关于 .Net 的工作方式需要注意的一个关键点是对象的实际类型总是不够的——声明的类型是消费代码所知道的。

这就是调用 Find 方法时出现异常的原因NotSupported,即使您的BindingList实现具有 Find 方法 - 接收此绑定列表的代码不知道 Find。

原因在于这些代码行:

dc = new NorthwindDataContext();

var qry = (from p in dc.Products
           select p).ToList();

FindAbleBindingList<Product> list = new FindAbleBindingList<Product>(qry);

productBindingSource.DataSource = list.OrderBy(o => o.ProductName);

当您为绑定源设置数据源时,您会包含扩展方法OrderBy- 选中此项表明它返回 IOrderedEnumerable,这是 MSDN 上此处描述的接口。请注意,该接口没有 Find 方法,因此即使底层FindableBindingList<T>支持 Find 绑定源也不知道它。

有几种解决方案(我认为最好的方法是扩展您的 FindableBindingList 以支持排序和排序列表),但对于您当前的代码,最快的方法是像这样提前排序:

dc = new NorthwindDataContext();

var qry = (from p in dc.Products
           select p).OrderBy(p => p.ProductName).ToList();

FindAbleBindingList<Product> list = new FindAbleBindingList<Product>(qry);

productBindingSource.DataSource = list;

在 WinForms 中,对于您正在尝试做的事情,没有完全开箱即用的解决方案 - 它们都需要一些自定义代码,您需要将这些代码组合在一起以满足您自己的需求。

于 2012-07-30T20:36:24.487 回答
2

我采取了不同的方法。我想,以编程方式,必须检查每条记录,直到找到匹配项,所以我只是使用 MoveNext 方法进行迭代,直到找到匹配项。不确定起始位置是否是第一条记录,所以我使用了 MoveFirst 方法来确保它是。

有一个假设,那就是您要搜索的内容在该列中是唯一的。就我而言,我希望匹配一个身份整数。

int seekID;        
this.EntityTableBindingSource.MoveFirst();
if (seekID > 0)
{
     foreach (EntityTable sd in EntityTableBindingSource)
     {
         if (sd.ID != seekID)
         {
             this.t_EntityTableBindingSource.MoveNext();
         }
         else
         {
             break;
         }
      }
 } 
于 2016-10-06T20:05:09.447 回答
1

我并不真正关心提供的任何一个答案。这是我为我的问题想出的:

// Create a list of items in the BindingSource and use labda to find your row:
var QuickAccessCode = customerListBindingSource.List.OfType<CustomerList>()
    .ToList().Find(f => f.QuickAccessCode == txtQAC.Text);

// Then use indexOf to find the object in your bindingSource:
var pos = customerListBindingSource.IndexOf(QuickAccessCode);
if (pos < 0)
{
    MessageBox.Show("Could not find " + txtQAC.Text);
}
else
{
    mainFrm.customerListBindingSource.Position = pos;
}
于 2020-10-14T04:38:36.433 回答