将数组的所有元素添加到除一个之外的datagridview行

时间:2012-12-05 21:52:13

标签: c# winforms datagridview

我正在逐行读取文本文件,并将其插入数组中。

然后我有一个名为custIndex的列表,它包含某些索引,我正在测试的items数组的索引,看它们是否是有效的代码。 (例如,custIndex [0] = 7,所以我检查项目[7-1]中的值,看看它是否有效,在我这里的两个词典中)。然后,如果代码无效,我将行(items数组)添加到dataGridView1。

问题是,dataGridView1中的一些列是组合框列,因此用户可以选择正确的值。当我尝试添加items数组时,我得到一个异常:“DataGridView中发生以下异常:System.ArgumentException:DataGridViewComboBoxCell值无效。”

我知道使用正确的数据源正确添加了组合框,因为如果我只是将items数组中的一些项添加到dataGridView1,就像items [0]一样,组合框显示正常,并且没有例外抛出。我想问题是当我尝试将items数组中的错误值添加到dataGridView1行时。

我不知道如何处理这件事。有没有办法可以添加除该值之外的所有项目?或者我可以从项目中添加值并将其显示在组合框单元格中,以及填充的下拉项目吗?

if(choosenFile.Contains("Cust"))
{
    var lines = File.ReadAllLines(path+"\\"+ choosenFile);

    foreach (string line in lines)
    {
        errorCounter = 0;
        string[] items = line.Split('\t').ToArray();

        for (int i = 0; i <custIndex.Count; i++)
        {
            int index = custIndex[i];
            /*Get the state and country codes from the files using the correct indices*/
            Globals.Code = items[index - 1].ToUpper();

            if (!CountryList.ContainsKey(Globals.Code) && !StateList.ContainsKey(Globals.Code))
            {
                errorCounter++;

                dataGridView1.Rows.Add(items);
            }
        }//inner for

        if (errorCounter == 0)
            dataGridView2.Rows.Add(items);

    }//inner for each

}//if file is a customer file

1 个答案:

答案 0 :(得分:6)

假设您的文本文件包含:

澳大利亚PNG,印度非洲
奥地利巴厘岛Indonisia
法国英格兰,苏格兰,爱尔兰格陵兰 德国巴哈马夏威夷
希腊哥伦比亚,墨西哥,秘鲁阿根廷
新西兰俄罗斯美国

并且假设您的DataGridView设置为3列,第2列是组合框。

enter image description here

当您填充网格并错误地填充组合框列时,您将收到错误。

解决问题的方法是“显式处理/声明”DataError事件,更重要的是正确填充组合框列。

private void dataGridView1_DataError(object sender, DataGridViewDataErrorEventArgs e)
{
    //Cancelling doesn't make a difference, specifying the event avoids the prompt 
    e.Cancel = true;
}

private void dataGridView2_DataError(object sender, DataGridViewDataErrorEventArgs e)
{
    e.Cancel = true;
}

所以想象第二列包含一个国家下拉列表和第一列和第一列。第3列包含文本字段。

对于第1列和第3列,它们只是字符串,因此我创建了一个表示每一行的类:

public class CountryData
{
    public string FirstCountry { get; set; }
    public string ThirdCountry { get; set; }
}

对于第二列“Countries”组合框单元格,我创建了一个单独的类,因为我将它绑定到第二列数据源。

public class MultiCountryData
{
    public string[] SeceondCountryOption { get; set; }
}

使用组合框列等填充网格,如下所示:https://stackoverflow.com/a/1292847/495455不是好习惯。您希望将业务逻辑与演示文稿分开,以获得更加封装,多态和抽象的方法,从而简化单元测试和维护。因此DataBinding。

以下是代码:

namespace BusLogic
{
public class ProcessFiles
{

internal List<CountryData> CountryDataList = new List<CountryData>();
internal List<MultiCountryData> MultiCountryDataList = new List<MultiCountryData>();

internal void foo(string path,string choosenFile)
{
    var custIndex = new List<int>();
    //if (choosenFile.Contains("Cust"))
    //{
        var lines = File.ReadAllLines(path + "\\" + choosenFile);
        foreach (string line in lines)
        {
            int errorCounter = 0;
            string[] items = line.Split('\t');

            //Put all your logic back here...

            if (errorCounter == 0)
            {
                var countryData = new CountryData()
                                      {
                                          FirstCountry = items[0],
                                          ThirdCountry = items[2]
                                      };
                countryDataList.Add(countryData);

                multiCountryDataList.Add( new MultiCountryData() { SeceondCountryOption = items[1].Split(',')});

            }
        //}
      }

}
}

在您的演示文稿项目中,按钮点击代码:

 imports BusLogic;
 private void button1_Click(object sender, EventArgs e)
 {
     var pf = new ProcessFiles();
     pf.foo(@"C:\temp","countries.txt"); 
     dataGridView2.AutoGenerateColumns = false;
     dataGridView2.DataSource = pf.CountryDataList;
     multiCountryDataBindingSource.DataSource = pf.MultiCountryDataList;      
 }

我设置dataGridView2.AutoGenerateColumns = false;因为我在设计时添加了3列;第1个文本列,第2个组合框列和第3个文本列。

绑定第二个组合框列的技巧是BindingSource。在设计时&gt;右键单击DataGridView&gt;选择编辑列&gt;选择第二列&gt;选择DataSource&gt;单击“添加项目数据源”&gt;选择对象&gt;然后勾选multiCountry类并单击Finish。

enter image description here

enter image description here

还将第1列的DataPropertyName设置为FirstCountry,将第3列的DataPropertyName设置为ThirdCountry,因此当您绑定数据时,映射会自动完成。

enter image description here

最后,不要忘记将BindingSource的DataMember属性设置为multiCountry类的SeceondCountryOption成员。

enter image description here

以下是代码演示http://temp-share.com/show/HKdPSzU1A