Найти строку в dataGridView на основе столбца и значения
У меня есть dataGridView, который имеет 3 столбца: SystemId, FirstName, LastName, связанное с использованием информации о базе данных. Я хотел бы выделить определенную строку, которую я бы использовал, используя:
dataGridView1.Rows[????].Selected = true;
Идентификатор строки, который я, однако, не знаю, и связующий источник продолжает меняться, поэтому строка 10 может быть "Джон Смит" в одном экземпляре, но даже не существует в другом (у меня есть фильтр, который отфильтровывает исходный код на основе того, что пользователь вводит, поэтому ввод "joh" приведет к появлению всех строк, в которых у первого или последнего имени есть "joh", поэтому мой список может перейти от 50 имен до 3 щелчком).
Я хочу найти способ выбора строки на основе SystemId и соответствующего номера. Я могу получить идентификатор системы, используя следующий метод:
systemId = dataGridView1.Rows[dataGridView1.CurrentRow.Index].Cells["SystemId"].Value.ToString();
Теперь мне просто нужно применить его к селектору строк. Что-то вроде dataGridView1.Columns [ "SystemId" ]. IndexOf (systemId}, но это не работает (и такой метод не существует). Любая помощь приветствуется.
Ответы
Ответ 1
Это даст вам индекс строки gridview для значения:
String searchValue = "somestring";
int rowIndex = -1;
foreach(DataGridViewRow row in DataGridView1.Rows)
{
if(row.Cells[1].Value.ToString().Equals(searchValue))
{
rowIndex = row.Index;
break;
}
}
Или запрос LINQ
int rowIndex = -1;
DataGridViewRow row = dgv.Rows
.Cast<DataGridViewRow>()
.Where(r => r.Cells["SystemId"].Value.ToString().Equals(searchValue))
.First();
rowIndex = row.Index;
то вы можете сделать:
dataGridView1.Rows[rowIndex].Selected = true;
Ответ 2
Вышеуказанные ответы работают только в том случае, если для параметра AllowUserToAddRows
установлено значение false
. Если для этого свойства установлено значение true
, вы получите NullReferenceException
, когда запрос цикла или Linq попытается согласовать новую строку. Я изменил два принятых ответа выше, чтобы обрабатывать AllowUserToAddRows = true
.
Ответ на петлю:
String searchValue = "somestring";
int rowIndex = -1;
foreach(DataGridViewRow row in DataGridView1.Rows)
{
if (row.Cells["SystemId"].Value != null) // Need to check for null if new row is exposed
{
if(row.Cells["SystemId"].Value.ToString().Equals(searchValue))
{
rowIndex = row.Index;
break;
}
}
}
Ответ LINQ:
int rowIndex = -1;
bool tempAllowUserToAddRows = dgv.AllowUserToAddRows;
dgv.AllowUserToAddRows = false; // Turn off or .Value below will throw null exception
DataGridViewRow row = dgv.Rows
.Cast<DataGridViewRow>()
.Where(r => r.Cells["SystemId"].Value.ToString().Equals(searchValue))
.First();
rowIndex = row.Index;
dgv.AllowUserToAddRows = tempAllowUserToAddRows;
Ответ 3
Или вы можете так использовать. Это может быть быстрее.
int iFindNo = 14;
int j = dataGridView1.Rows.Count-1;
int iRowIndex = -1;
for (int i = 0; i < Convert.ToInt32(dataGridView1.Rows.Count/2) +1; i++)
{
if (Convert.ToInt32(dataGridView1.Rows[i].Cells[0].Value) == iFindNo)
{
iRowIndex = i;
break;
}
if (Convert.ToInt32(dataGridView1.Rows[j].Cells[0].Value) == iFindNo)
{
iRowIndex = j;
break;
}
j--;
}
if (iRowIndex != -1)
MessageBox.Show("Index is " + iRowIndex.ToString());
else
MessageBox.Show("Index not found." );
Ответ 4
Попробуйте следующее:
string searchValue = textBox3.Text;
int rowIndex = -1;
dataGridView1.SelectionMode = DataGridViewSelectionMode.FullRowSelect;
try
{
foreach (DataGridViewRow row in dataGridView1.Rows)
{
if (row.Cells["peseneli"].Value.ToString().Equals(searchValue))
{
rowIndex = row.Index;
dataGridView1.CurrentCell = dataGridView1.Rows[rowIndex].Cells[0];
dataGridView1.Rows[dataGridView1.CurrentCell.RowIndex].Selected = true;
break;
}
}
}
catch (Exception exc)
{
MessageBox.Show(exc.Message);
}
Ответ 5
Если вы просто хотите проверить, существует ли этот элемент:
IEnumerable<DataGridViewRow> rows = grdPdfs.Rows
.Cast<DataGridViewRow>()
.Where(r => r.Cells["SystemId"].Value.ToString().Equals(searchValue));
if (rows.Count() == 0)
{
// Not Found
}
else
{
// Found
}
Ответ 6
Это основывается на вышеупомянутом ответе Гордона - не все это моя оригинальная работа.
То, что я сделал, это добавить более общий метод для моего статического класса утилиты.
public static int MatchingRowIndex(DataGridView dgv, string columnName, string searchValue)
{
int rowIndex = -1;
bool tempAllowUserToAddRows = dgv.AllowUserToAddRows;
dgv.AllowUserToAddRows = false; // Turn off or .Value below will throw null exception
if (dgv.Rows.Count > 0 && dgv.Columns.Count > 0 && dgv.Columns[columnName] != null)
{
DataGridViewRow row = dgv.Rows
.Cast<DataGridViewRow>()
.FirstOrDefault(r => r.Cells[columnName].Value.ToString().Equals(searchValue));
rowIndex = row.Index;
}
dgv.AllowUserToAddRows = tempAllowUserToAddRows;
return rowIndex;
}
Затем в любой форме, которую я хочу использовать, я вызываю метод, передающий DataGridView, имя столбца и значение поиска. Для простоты я преобразовываю все в строки для поиска, хотя было бы достаточно просто добавить перегрузки для указания типов данных.
private void UndeleteSectionInGrid(string sectionLetter)
{
int sectionRowIndex = UtilityMethods.MatchingRowIndex(dgvSections, "SectionLetter", sectionLetter);
dgvSections.Rows[sectionRowIndex].Cells["DeleteSection"].Value = false;
}
Ответ 7
Те, кто использует WPF
for (int i = 0; i < dataGridName.Items.Count; i++)
{
string cellValue= ((DataRowView)dataGridName.Items[i]).Row["columnName"].ToString();
if (cellValue.Equals("Search_string")) // check the search_string is present in the row of ColumnName
{
object item = dataGridName.Items[i];
dataGridName.SelectedItem = item; // selecting the row of dataGridName
dataGridName.ScrollIntoView(item);
break;
}
}
если вы хотите получить выбранные элементы строки после этого, полезно использовать фрагмент кода.
DataRowView drv = dataGridName.SelectedItem as DataRowView;
DataRow dr = drv.Row;
string item1= Convert.ToString(dr.ItemArray[0]);// get the first column value from selected row
string item2= Convert.ToString(dr.ItemArray[1]);// get the second column value from selected row