Здравствуйте. Сегодня поговорим о веб-разработке. Несмотря на то, что популярность MVC растет очень быстро, Web Forms ещё никто не отменял. К тому же много приложений написано с использованием Web Forms, да и уже написанные приложения требуют поддержки. И если простейшие контролы (например, TextBox) у новичков вопросов, как правило, не вызывают, то сориентироваться в чем-то более сложном уже проблема.
В данном посте я расскажу в общих чертах, как использовать контрол GridView для работы со списком или специализированным источником данных. Предупреждаю сразу, тема не новая и статья ориентирована на начинающих разработчиков.
Поехали.
Начнем с теории. Класс GridView
Итак, задача: Имеется страница, на которой расположен GridView, и список, который нужно на этой странице формировать и иметь возможность его изменять (то есть добавлять/удалять элементы). Список хранится только на странице.
Порядок действий:
Теперь определимся со списком. Во-первых, так как список формируется и изменяется только в рамках этой страницы, я решил хранить его во ViewState. Я просто определил свойство на странице, которое автоматом себя сохраняет.
Добавим GridView в разметку. По пути также добавим пару текстовых полей и кнопку, для возможности добавления элемента в коллекцию.
В обработчике событий
Окей, теперь при первой загрузке страницы, у нас будет заполняться GridView. А при добавлении элемента в коллекцию, GridView будет обновляться. Вот как это выглядит:
Добавление работает, но у нас нет ни сортировки, ни редактирования, ни удаления, да ещё и включена автогенерация полей. Вот с последнего и начнем. Отключаем автогенерацию, указываем нужные нам поля, разрешаем сортировку (список возможных полей):
Теперь у нас есть кнопки, для того, чтобы сортировать, редактировать или удалять. Но только этого функционала пока нет. GridView не может самостоятельно править нашу коллекцию. Но это не беда, нам достаточно подписаться на нужные нам события и реализовать всё самим. Для редактирования нужны следующие события: OnRowEditing="GvEditing" OnRowUpdating="GvUpdating" OnRowCancelingEdit="GvCancelingEdit", для сортировки OnSorting="GvSorting", для удаления OnRowDeleting="GvDeleting".
Вот код для сортировки
Код для редактирования и удаления
В итоге имеем грид, элементы которого можем удалять, добавлять, редактировать и сортировать.
Это всё хорошо, но что делать, если мы работаем с базой данных? В таких случаях можно определять источники данных (например, наследники DataSourceControl) и связывать их с гридом. Я приведу пример с использованием ObjectDataSource. Я выбрал ObjectDataSource, так как он зависит только от вашего кода и ему всё равно, откуда берутся данные - из базы данных, сервиса, файла, да чего угодно.
Формализуем задачу. Необходимо реализовать такой же функционал, как и в предыдущем примере, но с использованием ObjectDataSource. Только на этот раз данные я буду хранить не на странице, а в статической переменной - чтобы эти данные были доступны в рамках всего приложения (ведь так и происходит, когда вы работаете, например, с базой данных).
Вот класс автомобиля, с которым будем работать
Начнем с класса-репозитория. Именно его методы будет вызывать ObjectDataSource для получения данных и манипуляции с ними. Хочу заметить, что я подобрал сигнатуру и набор методов репозитория специально для использования в ObjectDataSource. Вот сам класс:
Отлично. Теперь, как и в предыдущем примере, добавляем страницу в проект и кидаем на эту страницу 2 текстбокса и кнопку (для возможности добавления)
Теперь поработаем над самим контролом ObjectDataSource.
Поясню по каждому полю.
Теперь у нас всё готово для того, чтобы добавить на страницу грид. Приведу весь его код.
Собрав всё вместе, получаем результат:
Вот и всё. Данные можно изменять, сортировать, удалять.
Таким образом, используя GridView и различные источники данных, можно легко манипулировать данными и представлять их в удобном для вас виде.
Исходный код солюшена
P.S. Решил добавить немного информации о фильтрации наборов данных. Хотя это и выходит за рамки статьи, но это частый сценарий при работе с источниками данных.
Итак, давайте в наш последний пример добавим немного фильтров. Я просто хочу, чтобы можно было фильтровать список по Id используя параметры URL запроса, а также я добавлю 2 текстовых поля для фильтрации по производителю и модели.
Итак, наши текстовые поля:
Для чего там кнопка и почему она ничего не делает я поясню позже.
Далее необходимо указать источнику данных, что параметры для получения информации необходимо брать из текстовых полей и URL. Это можно сделать с помощью коллекции SelectParameters.
Собственно, названия QueryStringParameter и ControlParameter говорят сами за себя (Список типов параметров и как их использовать). Однако, так как мы имеем дело с ObjectDataSource, наш источник данных должен поддерживать эти параметры. Вот код всех обновленных функций класса-источника данных.
Теперь как это работает. При биндинге, GridView будет спрашивать данные у источника данных, который, в свою очередь, будет извлекать параметры и вызывать методы класса-источника данных, отправляя в него значения этих параметров. То есть это будет происходить само собой, при любом постбеке. Это означает, что если мы установим свойство у текстовых полей AutoPostBack=True или просто поставим кнопку (можно даже без обработчика), которая будет делать постбек, то этого достаточно, чтобы данные в гриде обновились с учетом заданных параметров. Как это выглядит:
Конечно, все вкусные плюшки в виде сортировки и добавления/изменения элементов остаются.
Новый исходный код солюшена
Всем спасибо.
Полезные ссылки:
В данном посте я расскажу в общих чертах, как использовать контрол GridView для работы со списком или специализированным источником данных. Предупреждаю сразу, тема не новая и статья ориентирована на начинающих разработчиков.
Поехали.
Начнем с теории. Класс GridView
Отображает значения источника данных в таблице, где каждый столбец представляет поле, а каждая строка — запись. Элемент управления GridView позволяет выбирать, сортировать и изменять эти записи.Его можно связывать с элементами-источниками данных (например, наследниками DataSourceControl), а можно просто отображать перечисляемые списки, указав значение свойства DataSource. Начнем с последнего.
Итак, задача: Имеется страница, на которой расположен GridView, и список, который нужно на этой странице формировать и иметь возможность его изменять (то есть добавлять/удалять элементы). Список хранится только на странице.
Порядок действий:
- Создаём веб-приложение
- Удаляем с домашней страницы всё лишнее (необязательно)
- Создаём страницу GridViewList.aspx, указываем для неё мастер страницу
- Добавляем в меню пункт со ссылкой на GridViewList.aspx
- [Serializable]
- public class People
- {
- [XmlAttribute]
- public int Id { get; set; }
- [XmlAttribute]
- public string FirstName { get; set; }
- [XmlAttribute]
- public string LastName { get; set; }
- }
Теперь определимся со списком. Во-первых, так как список формируется и изменяется только в рамках этой страницы, я решил хранить его во ViewState. Я просто определил свойство на странице, которое автоматом себя сохраняет.
- public List<People> Peoples
- {
- get
- {
- // Получение ключа для поиска во ViewState.
- // hfPeoplesViewState - HiddenField, которое хранит в себе этот ключ
- // Это всё сделано только для того, чтобы обеспечить уникальность ключа на странице.
- // По идее тут можно было просто написать var str = "Peoples_Key" и всё бы работало
- var str = hfPeoplesViewState.Value;
- if (string.IsNullOrEmpty(str))
- {
- // Если ещё ничего не хранит, то создать ключ
- hfPeoplesViewState.Value = string.Format("Peoples_{0}", Guid.NewGuid());
- str = hfPeoplesViewState.Value;
- }
- // Проверяю, если во ViewState ещё нет того списка, что мне нужен - создаю его
- if (ViewState[str] == null || !(ViewState[str] is List<People>))
- {
- var peoples = new List<People>
- {
- new People {Id = 1, LastName = "Мурадов", FirstName = "Артем"},
- new People {Id = 2, LastName = "Пупкин", FirstName = "Василий"},
- new People {Id = 3, LastName = "Елопанов", FirstName = "Инокентий"}
- };
- ViewState[str] = peoples;
- return peoples;
- }
- return ViewState[str] as List<People>;
- }
- set
- {
- // Аналогично методу get
- var str = hfPeoplesViewState.Value;
- if (string.IsNullOrEmpty(str))
- {
- hfPeoplesViewState.Value = string.Format("Peoples_{0}", Guid.NewGuid());
- str = hfPeoplesViewState.Value;
- }
- ViewState[str] = value;
- }
- }
Добавим GridView в разметку. По пути также добавим пару текстовых полей и кнопку, для возможности добавления элемента в коллекцию.
- <asp:TextBox runat="server" ID="tbFirstName">
- </asp:TextBox>
- <asp:TextBox runat="server" ID="tbLastName">
- </asp:TextBox>
- <asp:Button runat="server" OnClick="BtAddPeople" Text="+" />
- <hr />
- <asp:GridView runat="server" ID="gv" AutoGenerateColumns="True" Width="60%">
- <EmptyDataTemplate>
- Записей нет</EmptyDataTemplate>
- </asp:GridView>
В обработчике событий
- protected void Page_Load(object sender, EventArgs e)
- {
- if (!IsPostBack)
- {
- gv.DataSource = Peoples;
- gv.DataBind();
- }
- }
- protected void BtAddPeople(object sender, EventArgs e)
- {
- if (!string.IsNullOrEmpty(tbFirstName.Text) && !string.IsNullOrEmpty(tbLastName.Text))
- {
- var id = Peoples.Count > 0 ? Peoples.Max(x => x.Id) + 1 : 1;
- var p = new People
- {
- Id = id,
- FirstName = tbFirstName.Text,
- LastName = tbLastName.Text
- };
- Peoples.Add(p);
- UpdateGrid();
- }
- }
- private void UpdateGrid()
- {
- gv.DataSource = Peoples;
- gv.DataBind();
- }
Окей, теперь при первой загрузке страницы, у нас будет заполняться GridView. А при добавлении элемента в коллекцию, GridView будет обновляться. Вот как это выглядит:
Добавление работает, но у нас нет ни сортировки, ни редактирования, ни удаления, да ещё и включена автогенерация полей. Вот с последнего и начнем. Отключаем автогенерацию, указываем нужные нам поля, разрешаем сортировку (список возможных полей):
- <asp:GridView runat="server" ID="gv" AutoGenerateColumns="False" Width="60%" AllowSorting="True">
- <Columns>
- <asp:BoundField HeaderText="ИД" DataField="Id" SortExpression="Id" ReadOnly="True">
- <ItemStyle Width="5%"></ItemStyle>
- </asp:BoundField>
- <asp:BoundField HeaderText="Имя" DataField="FirstName" SortExpression="FirstName">
- <ItemStyle Width="40%"></ItemStyle>
- </asp:BoundField>
- <asp:BoundField HeaderText="Фамилия" DataField="LastName" SortExpression="LastName">
- <ItemStyle Width="40%"></ItemStyle>
- </asp:BoundField>
- <asp:CommandField ShowEditButton="True" ShowDeleteButton="True" ShowCancelButton="True">
- <ItemStyle Width="15%"></ItemStyle>
- </asp:CommandField>
- </Columns>
- <EmptyDataTemplate>
- Записей нет</EmptyDataTemplate>
- </asp:GridView>
- <asp:GridView runat="server" ID="gv" AllowSorting="True" OnSorting="GvSorting" OnRowEditing="GvEditing"
- OnRowUpdating="GvUpdating" OnRowCancelingEdit="GvCancelingEdit" OnRowDeleting="GvDeleting"
- DataKeyNames="Id" AutoGenerateColumns="False" Width="60%" >
- <Columns>
- <asp:BoundField HeaderText="ИД" DataField="Id" SortExpression="Id" ReadOnly="True">
- <ItemStyle Width="5%"></ItemStyle>
- </asp:BoundField>
- <asp:BoundField HeaderText="Имя" DataField="FirstName" SortExpression="FirstName">
- <ItemStyle Width="40%"></ItemStyle>
- </asp:BoundField>
- <asp:BoundField HeaderText="Фамилия" DataField="LastName" SortExpression="LastName">
- <ItemStyle Width="40%"></ItemStyle>
- </asp:BoundField>
- <asp:CommandField ShowEditButton="True" ShowDeleteButton="True" ShowCancelButton="True">
- <ItemStyle Width="15%"></ItemStyle>
- </asp:CommandField>
- </Columns>
- <EmptyDataTemplate>
- Записей нет</EmptyDataTemplate>
- </asp:GridView>
Вот код для сортировки
- /// <summary>
- /// Возникает при сортировке в гриде.
- /// </summary>
- protected void GvSorting(object sender, GridViewSortEventArgs e)
- {
- var p = Peoples;
- // Необходимо определить, по какому именно полю сортировать
- if (e.SortExpression == "LastName")
- {
- // используем вспомогательный метод Sort
- p = Sort("LastName", list => list.OrderBy(x => x.LastName).ToList(),
- list => list.OrderByDescending(x => x.LastName).ToList(), p);
- }
- if (e.SortExpression == "FirstName")
- {
- p = Sort("FirstName", list => list.OrderBy(x => x.FirstName).ToList(),
- list => list.OrderByDescending(x => x.FirstName).ToList(), p);
- }
- if (e.SortExpression == "Id")
- {
- p = Sort("Id", list => list.OrderBy(x => x.Id).ToList(),
- list => list.OrderByDescending(x => x.Id).ToList(), p);
- }
- Peoples = p;
- // обновление грида
- UpdateGrid();
- }
- /// <summary>
- /// Вспомогательный метод для сортировки
- /// </summary>
- private List<People> Sort(string column, Func<List<People>, List<People>> ascFunc, Func<List<People>, List<People>> descFunc, List<People> data)
- {
- List<People> result;
- // Если в прошлвй раз была сортировка по этому же полю
- if (hfLastSortFieldState.Value == column)
- {
- // смотрим, в каком направлении была сортировка в прошлвй раз и сортируем в другом
- result = hfLastSortDirectionState.Value != "ASC" ? ascFunc(data) : descFunc(data);
- // сохраняем колонку и направление сортировки
- hfLastSortDirectionState.Value = hfLastSortDirectionState.Value != "ASC" ? "ASC" : "DESC";
- hfLastSortFieldState.Value = column;
- }
- else
- {
- // Если это поле ещё не сортировано, сортируем по возрастанию
- result = ascFunc(data);
- // сохраняем колонку и направление сортировки
- hfLastSortDirectionState.Value = "ASC";
- hfLastSortFieldState.Value = column;
- }
- return result;
- }
Код для редактирования и удаления
- /// <summary>
- /// Событие удаления элемента. Поле Id попадает в набор ключей,
- /// так как в гриде указано DataKeyNames="Id"
- /// Нам осталось только определить ключ и убрать элемент с этим ключем из коллекции
- /// </summary>
- protected void GvDeleting(object sender, GridViewDeleteEventArgs e)
- {
- int id;
- // Определяем идентификатор
- if (int.TryParse(string.Format("{0}", e.Keys["Id"]), out id))
- {
- // Убираем из списка
- Peoples = Peoples.Where(x => x.Id != id).ToList();
- // обновляем грид
- UpdateGrid();
- }
- }
- /// <summary>
- /// Событие возникает, когда пользователь начинает редактировать строку.
- /// Необходимо просто указать гриду индекс редактируемой строки и обновить грид
- /// </summary>
- protected void GvEditing(object sender, GridViewEditEventArgs e)
- {
- gv.EditIndex = e.NewEditIndex;
- UpdateGrid();
- }
- /// <summary>
- /// Событие возникает, когда пользователь обновляет строку.
- /// Необходимо определить Id строки, получить элемент по этому Id, обновить поля и обновить грид
- /// </summary>
- protected void GvUpdating(object sender, GridViewUpdateEventArgs e)
- {
- int id;
- if (int.TryParse(string.Format("{0}", e.Keys["Id"]), out id))
- {
- Peoples
- .Where(x => x.Id == id)
- .ToList()
- .ForEach(x =>
- {
- // я использую string.Format, так как
- // e.NewValues["FirstName"] имеет тип object
- // и может быть равен null
- x.FirstName = string.Format("{0}", e.NewValues["FirstName"]);
- x.LastName = string.Format("{0}", e.NewValues["LastName"]);
- });
- gv.EditIndex = -1;
- UpdateGrid();
- }
- }
- /// <summary>
- /// Событие возникает, когда пользователь отменяет обновление строки
- /// Нужно установить gv.EditIndex = -1; и обновить грид
- /// </summary>
- protected void GvCancelingEdit(object sender, GridViewCancelEditEventArgs e)
- {
- gv.EditIndex = -1;
- UpdateGrid();
- }
В итоге имеем грид, элементы которого можем удалять, добавлять, редактировать и сортировать.
Это всё хорошо, но что делать, если мы работаем с базой данных? В таких случаях можно определять источники данных (например, наследники DataSourceControl) и связывать их с гридом. Я приведу пример с использованием ObjectDataSource. Я выбрал ObjectDataSource, так как он зависит только от вашего кода и ему всё равно, откуда берутся данные - из базы данных, сервиса, файла, да чего угодно.
Формализуем задачу. Необходимо реализовать такой же функционал, как и в предыдущем примере, но с использованием ObjectDataSource. Только на этот раз данные я буду хранить не на странице, а в статической переменной - чтобы эти данные были доступны в рамках всего приложения (ведь так и происходит, когда вы работаете, например, с базой данных).
Вот класс автомобиля, с которым будем работать
- public class Car
- {
- public int Id { get; set; }
- public string Manufacturer { get; set; }
- public string Model { get; set; }
- }
Начнем с класса-репозитория. Именно его методы будет вызывать ObjectDataSource для получения данных и манипуляции с ними. Хочу заметить, что я подобрал сигнатуру и набор методов репозитория специально для использования в ObjectDataSource. Вот сам класс:
- public class CarRepository
- {
- /// <summary>
- /// Статическая переменная. В ней будем хранить наши данные
- /// </summary>
- private static List<Car> _store;
- /// <summary>
- /// Статический конструктор. Внутри инициализируем наши данные
- /// </summary>
- static CarRepository()
- {
- _store = new List<Car>();
- for (var i = 0; i < 1000; i++)
- {
- _store.Add(new Car
- {
- Id = i,
- Manufacturer = "Manufacturer" + i,
- Model = "Model" + i
- });
- }
- }
- /// <summary>
- /// Получение данных. Принимает три параметра. Назначение первых двух очевидно,
- /// а вот в третьем приходит информация для сортировки. Например, у нас сортировка по полю
- /// "Id". При сортировке по возрастанию переметр sort будет равен "Id".
- /// При сортировке по убыванию - "Id DESC"
- /// </summary>
- public IEnumerable<Car> GetCars(int maximumRows, int startRowIndex, string sort)
- {
- IEnumerable<Car> temp = _store;
- // Сперва определяем направление сортировки, затем поле, по которому сортируем
- if (sort.Contains("DESC"))
- {
- if (sort.Contains("Id")) temp = _store.OrderByDescending(x => x.Id);
- if (sort.Contains("Manufacturer")) temp = _store.OrderByDescending(x => x.Manufacturer);
- if (sort.Contains("Model")) temp = _store.OrderByDescending(x => x.Model);
- }
- else
- {
- if (sort.Contains("Id")) temp = _store.OrderBy(x => x.Id);
- if (sort.Contains("Manufacturer")) temp = _store.OrderBy(x => x.Manufacturer);
- if (sort.Contains("Model")) temp = _store.OrderBy(x => x.Model);
- }
- return temp.Skip(startRowIndex).Take(maximumRows);
- }
- /// <summary>
- /// Необходим для подсчёта общего количества элеменов.
- /// Используется для постраничного вывода данных
- /// </summary>
- public int Count()
- {
- return _store.Count;
- }
- /// <summary>
- /// Для обновления в метод приходит экземпляр Car с
- /// идентификатором обновляемой машины и новыми значениями полей
- /// </summary>
- public void Update(Car car)
- {
- _store
- .Where(x => x.Id == car.Id)
- .ToList()
- .ForEach(x =>
- {
- x.Manufacturer = car.Manufacturer;
- x.Model = car.Model;
- });
- }
- /// <summary>
- /// Удаление. Код и так понятен.
- /// </summary>
- public void Delete(Car car)
- {
- _store = _store.Where(x => x.Id != car.Id).ToList();
- }
- /// <summary>
- /// Добавление
- /// </summary>
- public void Insert(Car car)
- {
- car.Id = _store.Count > 0 ? _store.Max(x => x.Id) + 1 : 1;
- _store.Add(car);
- }
- }
Отлично. Теперь, как и в предыдущем примере, добавляем страницу в проект и кидаем на эту страницу 2 текстбокса и кнопку (для возможности добавления)
- <asp:TextBox runat="server" ID="tbManufacturer">
- </asp:TextBox>
- <asp:TextBox runat="server" ID="tbModel">
- </asp:TextBox>
- <asp:Button runat="server" OnClick="BtAddCar" Text="+" />
- <hr />
- protected void BtAddCar(object sender, EventArgs e)
- {
- if (!string.IsNullOrEmpty(tbManufacturer.Text) && !string.IsNullOrEmpty(tbModel.Text))
- {
- var carRepository = new CarRepository();
- carRepository.Insert(new Car {Manufacturer = tbManufacturer.Text, Model = tbModel.Text});
- gv.DataBind();
- }
- }
Теперь поработаем над самим контролом ObjectDataSource.
- <asp:ObjectDataSource ID="ods" runat="server" TypeName="GridViewTest.CarRepository"
- DataObjectTypeName="GridViewTest.Car" EnablePaging="True" MaximumRowsParameterName="maximumRows"
- StartRowIndexParameterName="startRowIndex" SortParameterName="sort" InsertMethod="Insert"
- SelectCountMethod="Count" SelectMethod="GetCars" UpdateMethod="Update" DeleteMethod="Delete">
- </asp:ObjectDataSource>
Поясню по каждому полю.
- TypeName="GridViewTest.CarRepository" - указывает класс-источник данных
- DataObjectTypeName="GridViewTest.Car" - указывает тип объекта, которым будем оперировать. Благодаря этому мы можем в репозитории указывать методы, принимающие в качестве параметра экземпляр типа Car. Этот тип должен иметь конструктор без параметров и поля, доступные для записи.
- EnablePaging="True" указываем, что используется постраничный вывод
- MaximumRowsParameterName="maximumRows" StartRowIndexParameterName="startRowIndex" SortParameterName="sort" имена параметров в методах
- InsertMethod="Insert" SelectCountMethod="Count" SelectMethod="GetCars" UpdateMethod="Update" DeleteMethod="Delete" имена методов в репозитории
Теперь у нас всё готово для того, чтобы добавить на страницу грид. Приведу весь его код.
- <asp:GridView runat="server" ID="gv" AllowSorting="True" DataKeyNames="Id" AutoGenerateColumns="False"
- DataSourceID="ods" Width="60%" AllowPaging="True" PageSize="20">
- <Columns>
- <%--Тут, как и в предыдущем примере, я использую BoundField--%>
- <asp:BoundField HeaderText="ИД" DataField="Id" ReadOnly="True" SortExpression="Id">
- <ItemStyle Width="5%"></ItemStyle>
- </asp:BoundField>
- <%--Здесь я тоже мог бы использовать BoundField, но я хотел показать, как использовать
- TemplateField с односторонней (Eval) и двухсторонней (Bind) привязкой данных--%>
- <asp:TemplateField HeaderText="Производитель" SortExpression="Manufacturer">
- <ItemStyle Width="35%"></ItemStyle>
- <ItemTemplate>
- <%#Eval("Manufacturer")%>
- </ItemTemplate>
- <EditItemTemplate>
- <asp:TextBox runat="server" ID="tbMan" Text='<%#Bind("Manufacturer") %>'></asp:TextBox>
- </EditItemTemplate>
- </asp:TemplateField>
- <%--Тут всё аналогично предыдущему столбцу--%>
- <asp:TemplateField HeaderText="Модель" SortExpression="Model">
- <ItemStyle Width="35%"></ItemStyle>
- <ItemTemplate>
- <%#Eval("Model")%>
- </ItemTemplate>
- <EditItemTemplate>
- <asp:TextBox runat="server" ID="tbModel" Text='<%#Bind("Model") %>'></asp:TextBox>
- </EditItemTemplate>
- </asp:TemplateField>
- <%--Я хочу показать, что для того, чтобы использовать кнопки редактирования/удаления,
- вам не обязательно пользоваться CommandField. Достаточно указать нужное значение в поле
- CommandName любого кнопочного контрола--%>
- <asp:TemplateField>
- <ItemStyle Width="25%"></ItemStyle>
- <ItemTemplate>
- <asp:LinkButton runat="server" CommandName="Edit" Text="Изменить"></asp:LinkButton>
- <%--А раз это простые кнопки, то можно им добавлять нужное поведение. Например,
- OnClientClick отработает на клиенте, спросит у пользователя разрешение на действие,
- и постбек произойдет только в том случае, если пользователь ответит утвердительно --%>
- <asp:LinkButton runat="server" CommandName="Delete" Text="Удалить"
- OnClientClick="return confirm('Вы действительно хотите удалить запись?');"></asp:LinkButton>
- </ItemTemplate>
- <EditItemTemplate>
- <asp:LinkButton runat="server" CommandName="Update" Text="Обновить"></asp:LinkButton>
- <asp:LinkButton runat="server" CommandName="Cancel" Text="Отменить"></asp:LinkButton>
- </EditItemTemplate>
- </asp:TemplateField>
- </Columns>
- <%--Контент, который будет показан, если записей не будет вовсе--%>
- <EmptyDataTemplate>
- Записей нет</EmptyDataTemplate>
- </asp:GridView>
Собрав всё вместе, получаем результат:
Вот и всё. Данные можно изменять, сортировать, удалять.
Таким образом, используя GridView и различные источники данных, можно легко манипулировать данными и представлять их в удобном для вас виде.
Исходный код солюшена
P.S. Решил добавить немного информации о фильтрации наборов данных. Хотя это и выходит за рамки статьи, но это частый сценарий при работе с источниками данных.
Итак, давайте в наш последний пример добавим немного фильтров. Я просто хочу, чтобы можно было фильтровать список по Id используя параметры URL запроса, а также я добавлю 2 текстовых поля для фильтрации по производителю и модели.
Итак, наши текстовые поля:
- <asp:TextBox runat="server" ID="tbManufacturerFilter">
- </asp:TextBox>
- <asp:TextBox runat="server" ID="tbModelFilter">
- </asp:TextBox>
- <asp:Button runat="server" Text="Искать" />
Для чего там кнопка и почему она ничего не делает я поясню позже.
Далее необходимо указать источнику данных, что параметры для получения информации необходимо брать из текстовых полей и URL. Это можно сделать с помощью коллекции SelectParameters.
- <asp:ObjectDataSource ID="ods" runat="server" TypeName="GridViewTest.CarRepository"
- DataObjectTypeName="GridViewTest.Car" EnablePaging="True" MaximumRowsParameterName="maximumRows"
- StartRowIndexParameterName="startRowIndex" SortParameterName="sort" InsertMethod="Insert"
- SelectCountMethod="Count" SelectMethod="GetCars" UpdateMethod="Update" DeleteMethod="Delete">
- <SelectParameters>
- <asp:QueryStringParameter DefaultValue="-1" QueryStringField="id" Name="id" Type="Int32" />
- <asp:ControlParameter ControlID="tbManufacturerFilter" DefaultValue="" Name="manFilter" ConvertEmptyStringToNull="false"
- PropertyName="Text" Type="String" />
- <asp:ControlParameter ControlID="tbModelFilter" DefaultValue="" Name="modelFilter" ConvertEmptyStringToNull="false"
- PropertyName="Text" Type="String" />
- </SelectParameters>
- </asp:ObjectDataSource>
Собственно, названия QueryStringParameter и ControlParameter говорят сами за себя (Список типов параметров и как их использовать). Однако, так как мы имеем дело с ObjectDataSource, наш источник данных должен поддерживать эти параметры. Вот код всех обновленных функций класса-источника данных.
- /// <summary>
- /// Получение данных. Принимает шесть параметров. Первые три предназначены для фильтрации.
- /// Назначение следующих двух очевидно.
- /// А вот в шестом приходит информация для сортировки. Например, у нас сортировка по полю
- /// "Id". При сортировке по возрастанию переметр sort будет равен "Id".
- /// При сортировке по убыванию - "Id DESC"
- /// </summary>
- public IEnumerable<Car> GetCars(int id, string manFilter, string modelFilter, int maximumRows, int startRowIndex, string sort)
- {
- IEnumerable<Car> temp = _store.Where(x => CheckCar(x, id, manFilter, modelFilter));
- // Сперва определяем направление сортировки, затем поле, по которому сортируем
- if (sort.Contains("DESC"))
- {
- if (sort.Contains("Id")) temp = temp.OrderByDescending(x => x.Id);
- if (sort.Contains("Manufacturer")) temp = temp.OrderByDescending(x => x.Manufacturer);
- if (sort.Contains("Model")) temp = temp.OrderByDescending(x => x.Model);
- }
- else
- {
- if (sort.Contains("Id")) temp = temp.OrderBy(x => x.Id);
- if (sort.Contains("Manufacturer")) temp = temp.OrderBy(x => x.Manufacturer);
- if (sort.Contains("Model")) temp = temp.OrderBy(x => x.Model);
- }
- return temp.Skip(startRowIndex).Take(maximumRows);
- }
- /// <summary>
- /// Необходим для подсчёта общего количества элеменов.
- /// Используется для постраничного вывода данных
- /// </summary>
- public int Count(int id, string manFilter, string modelFilter)
- {
- return _store.Where(x => CheckCar(x, id, manFilter, modelFilter)).Count();
- }
- /// <summary>
- /// Проверка соответствия конкретного экземпляра Car входящим параметрам
- /// </summary>
- private static bool CheckCar(Car car, int id, string manFilter, string modelFilter)
- {
- return (id < 0 || car.Id == id) &&
- (string.IsNullOrEmpty(manFilter) || car.Manufacturer.Contains(manFilter)) &&
- (string.IsNullOrEmpty(modelFilter) || car.Model.Contains(modelFilter));
- }
Теперь как это работает. При биндинге, GridView будет спрашивать данные у источника данных, который, в свою очередь, будет извлекать параметры и вызывать методы класса-источника данных, отправляя в него значения этих параметров. То есть это будет происходить само собой, при любом постбеке. Это означает, что если мы установим свойство у текстовых полей AutoPostBack=True или просто поставим кнопку (можно даже без обработчика), которая будет делать постбек, то этого достаточно, чтобы данные в гриде обновились с учетом заданных параметров. Как это выглядит:
Конечно, все вкусные плюшки в виде сортировки и добавления/изменения элементов остаются.
Новый исходный код солюшена
Всем спасибо.
Полезные ссылки:
Спасибо, сейчас буду проходить туториал)
ОтветитьУдалитьСпасибо! Наконец эта магия с GridView раскрыта ;-)
ОтветитьУдалитьСтолкнулся с такой проблемой: Есть GridView связанный с таблицей в DataSet. Для того чтобы массово редактировать и обновлять строки в гриде определены ItemTemplate с TextBoxми привязаными через Bind(). Гриду разрешен постраничный просмотр. И все сохраняется и редактируется пока страница одна. Если к примеру изменил значение в строке находящейся на 3-ей странице, переключился на 1-ую и потом нажал на сохранение то цикл по строкам грида foreach (GridViewRow r in GV_fact0.Rows) проходит только по активной странице. И как следствие получить новые значения с 3-ей страницы нет никакой возможности. Подскажите пожалуста как добраться до всех страниц и строк грида? Или есть какой то обходной маневр? Буду очень признателен.
ОтветитьУдалитьДавайте разберемся:
Удалить1. У вас есть GridView, у которого в ItemTemplate указаны текстбоксы. То есть грид в процессе редактирования не участвует, а где то на форме есть кнопка, которая при нажатии проходит все строки грида и обновляет данные.
2. У грида настроен пейджинг. Что это означает. Что грид не хранит другие страницы, которые не отображает. То есть в гриде есть только те строки, что он показывает. Следовательно, когда пользователь переходит на другую страницу, все его изменения теряются.
Я тут вижу 4 варианта:
1. Обновлять данные как только их изменяют. (То есть пользователь изменил значение текстбокса и сразу обновить это значение в базе)
2. Сделать свой пейджинг, который применяет изменения при переходе на другую страницу (спрашивая предварительно разрешения у пользователя)
3. Хранить где то все изменения пользователя (то есть яваскриптом определять что и где пользователь меняет, куда то это сохранять и применять изменения если нажмется кнопка Сохранить)
4. Использовать какой то другой контрол или написать свой.
Спасибо за оперативный ответ.
УдалитьПо поводу первой части ответа: да так и есть, есть кнопка которая все сохраняет и грид получается не сохраняет никаких строк кроме строк на текущей странице(Я что то такое предполагал).
Придется до события GV_fact0_PageIndexChanging вешать клиентский confirm с вопросом типа «Сохранить изменения?». Или в самом GV_fact0_PageIndexChanging на стороне сервера пробежать по гриду и изменения сохранить в исходном DataSetе. А уж потом после нажатия кнопки сохранения заливать все скопом в базу.
В любом случае спасибо за подсказку.
Этот комментарий был удален автором.
ОтветитьУдалитьПожалуйста. Я для того и писал статью, чтобы "косить деревья" :)
УдалитьЭто не вы случаем удалили мой комент? или я случайно нажал. Перепишу:
УдалитьСпасибо, полезная статья , очень помогла))
Комментарий был удален автором, а автор комментария - Вы. Я Ваши комментарии не трогал :)
УдалитьИзвиняюсь за столь просто вопрос, но как Вы добавили файлы *.designer.cs в вебформы? И еще в двух словах, если не трудно, зачем нужны hidden поля в рассматриваемом примере? Я начинающий .net-чик, поэтому многое для меня пока что непонятно)
Удалить1. Когда вы добавляете веб форму в проект, то автоматически добавляются 3 файла:
УдалитьWebform.aspx - разметка формы
Webform.aspx.cs - обработчики событий и логика
Webform.aspx.designer.cs - содержимое автоматически генерируется, в зависимости от того, что написано в разметке
2. Поскольку веб приложения не имеют состояния, я не мог хранить нужные мне данные в полях классов. Поэтому я использовал скрытые поля. Можно также использовать куки, ViewState, сессию, приложение, профиль или базу данных для того, чтобы хранить какую-то информацию дольше одного постбека. Подробнее тут.
Таки допилил) сейчас через ado.net пытаюсь вытащить данные с базы. Еще раз спасибо, блог закину в закладки, вдруг еще что интересно будет :)
УдалитьЗдравствуйте!
ОтветитьУдалитьМожет вы мне ответите.
У меня в форме есть GridView id="GridView1", его DataSource установлен вручную, им служит некая таблица, кот. является результатом запроса. Есть также кнопка Button1, ее действие - постановка фильтра на эту таблицу (формирование предложения WHERE запроса). НО!-кнопка отрабатывает после того, как форма перезагрузится (у кнопки AutoPostBack="true")(я смотрел отладчиком), таким образом, таблица и GridView не фильтруются, т.е. последовательность событий такова Page_Load, Button1_Click(). Вопрос - как заставить кнопку отработать до загрузки GridView
Вам, насколько я понял, и не нужно заставлять отрабатывать кнопку раньше. В качестве быстрого, но не самого оптимального, решения, вы можете в обработчике кнопки Button1_Click() получить новые данные, заново установить DataSource у грида и вызвать DataBind().
УдалитьЗдравствуйте! Спасибо за быстрый ответ.
Удалить>установить DataSource у грида
Я должен сформировать и отработать новый запрос (на основании данных, полученных через кнопку)?
Собственно, да. Насколько я понял, раз вы сами устанавливаете датасурс у грида, то даные, что он отображает, не изменятся, пока вы сами их не измените или не привяжете грид к другому набору данных (в вашем случае, к фильтрованному). То есть как я вижу логику:
Удалить1. Вы закачиваете какие то данные в грид
2. Пользователь на странице указывает параметры фильтрации и жмет кнопку
3. На сервере в обработчике кнопки вы снова формируете набор данных, но уже фильтрованный, и привязываете к гриду
>На сервере в обработчике кнопки вы снова формируете набор данных, но уже фильтрованный, и привязываете к гриду
УдалитьЯ-то думал, для того и сделано событие перезагрузки страницы (в процедуре page_load), чтобы страницу отображать с новыми условиями и в этой процедуре менять DataSource.
На самом деле, вы вольны сами выбирать, как и где обновлять данные. Page_Load нужна только для отработки логики при загрузке страницы на сервере, есть ещё много других событий. Почитайте про жизненный цикл страницы asp.net
УдалитьЗдравствуйте! Извините, что пристаю к вам с "простыми" вопросами.
УдалитьДело в том, что в GridView, созданном при помощи мастера, есть возможности edit (update), insert и delete записей (записи берутся из таблицы в базе данных). Так вот delete - делается, а edit (update)и insert -нет! Я уже голову сломал, может вы подскажете? Я смотрел текст, UpdateCommand и там есть, но, видимо, чего-то не хватает.Я бы предоставил текст, но не знаю, как это сделать.
Ну я гадать не умею, покажите код страницы и разметки :).
Удалить>покажите код страницы и разметки :).
УдалитьА как это сделать?
Ну, грид у вас находится на какой то странице, верно? кроме дизайнера, там есть ещё разметка, а также файл для кода. Вот это и нужно показать (любым способом, хоть скриншот, хоть ссылку на файл, хоть оба файла сюда в комментарий поместите)
УдалитьПросто вставить код в комментарий?
УдалитьПочистил комментарии. Можете просто выслать мне на почту солюшен, tym32167@gmail.com. Ещё попробуйте в команде UPDATE указать просто UPDATE [$street] SET [ZIP] = @ZIP, [streetNAME] = @streetNAME, [streetSOCR] = @streetSOCR, [streetfull] = @streetfull WHERE [streetId] = @original_streetId. Если не заработает, будем дальше разбираться.
УдалитьСделал, как вы сказали - действительно помогло! Но почему? Я все-таки вышлю вам исходники.Как заставить работать надпись New ("Вставить").
УдалитьВам спасибо!
Скажите, а отменить операцию "Delete" можно только на стороне клиента, при помощи Javascript, или можно на это сделать на сервере (ASP.NET)?
там в гриде вроде есть события RowDeleted и RowDeleting (я точно не помню, как называются). Тот, который *ing, в этом событии можно отменить удаление на сервере.
УдалитьА как отменить? Сделать свойство IsPostBack=false?
ОтветитьУдалитьу меня ж в статье есть событие GvDeleting. Вот внутри обработчика этого события можно отменить удаление.
УдалитьЗдравствуйте!
ОтветитьУдалитьА как сделать скроллинг в GridView вместо разбиения на страницы и как можно выбрать строку в качестве текущей без AutoGenerateSelectButton
Здравствуйте.
Удалить1. Про скролинг. Если вам надо вывалить все данные сразу на клиента - просто не используйте пейджинг. Если вы имеете ввиду динамическую подгрузку данных во время скроллинга - это называется "бесконечный скролинг" и легко гуглится (например, вот).
2. Использовать свойство кнопки ButtonField.CommandName
Не подскажите, как задать стиль Confirm окну, у меня выскакивает серое, яваскриптовское.
ОтветитьУдалитьЯ в посте и использую яваскриптовское окно. Если хотите свое окно, то нужно менять обработчик нажатия кнопки удаления OnClientClick
УдалитьВот меня и смущает что в коде обычный confirm, а окошко на скрине без шапки, рамок и цвета явно другие. Вот над этой строкой "Вот и всё. Данные можно изменять, сортировать, удалять." скрин.
УдалитьПросто на скринах я использовал браузер Firefox, там такие конфирмы :)
УдалитьВот он ответ! сенкс)
ОтветитьУдалитьЗдравствуйте. Зарание прошу извининение за глупый вопрос.
ОтветитьУдалить1. У меня есть клас подключенный к ObjectDataSource, GridView к ObjectDataSource. В класе я обращаюсь к таблице SELECT(который работает, таблица SQL) и DELETE. Так вот, DELETE не работает, хотя DataKeyNames задан, и в процидуру заходит(когда поставить точку останова), и правильно опредиляет праметр(тоесть если usid=3 то передает 3 и т.д.), и запрос на DELETE выполняется без ошибок, а строка не удаляется
(OldValuesParameterFormatString ставлю как называется параметр, если удаляю то
ObjectDataSource 'ObjectDataSource1' не может найти не групповой метод 'DeleteQuery', который имеет параметры: p,
если ставлю {0}
ObjectDataSource 'ObjectDataSource1' не может найти не групповой метод 'DeleteQuery', который имеет параметры: p, usid..)?
2. У меня есть DataSet подключенный к ObjectDataSource, GridView к ObjectDataSource.
В DataSet грамотно составляю запрос на DELETE,OldValuesParameterFormatString ставлю как называется параметр, выдает ошибку Invalid columne name "имя параметра", тоесть параметр не передается хотя DELETE вызывается и DataKeyNames задан. Запрос в TableAdapter простой
DELETE FROM Table1
WHERE (usid = p)
параметр прописываю в Parametrs.
Пожалуйста помогите.
Зарание спасибо.
Если в метод, где происходит удаление, код заходит и параметры передаются правильные, то проблема не в гриде, а в коде этого метода. Возможно, где то не вызвали SaveChanges (если используете какую-либо ORM). Попробуйте этот метод вызвать, к примеру, в обработчике нажатия кнопки, подставив идентификатор существующей записи. Если удаления не будет - то проблема в вашем коде. Также возможно ваш метод содержит конструкцию WHERE, которая не возвращает записей. В любом случае, без взгляда на код я вряд ли смогу помочь. Можете прислать мне те фрагменты кода, что не работают, мой ящик tym32167[@]gmail.com. Правда, оперативность гарантировать не могу, постараюсь поглядеть на неделе.
УдалитьХотел еще сказать БОЛЬШОЕ СПАСАБО за статю!!!
ОтветитьУдалитьАртём Мурадов18 ноября 2012 г., 10:45
ОтветитьУдалитьБольшое спасибо что удилили мне внимание. Я выслал своих 2 проекта из адреса vasya-shara@ukr.net.
Буду очень рад если ответите.
Большое спасибо за ответ! Я поставил, так как вы сказали - "@" в запросе возле параметра и все нормально работает.
ОтветитьУдалитьЗдравствуйте! Очень полезная статья! Спасибо.
ОтветитьУдалитьХотелось узнать как можно выполнять поиск по одному из столбцов списка, например, "Имя" ?
Здравствуйте. Спасибо. GridView только отображает данные. Чтобы сделать поиск, нужно работать с источником данных. Например, в SqlDataSource это можно сделать с помощью QueryParameter или ControlParameter.
УдалитьСпасибо за оперативный ответ)
УдалитьВозник еще один вопрос: пытаюсь вставить поиск в вашем примере с автомобилями и не знаю к какому элементу применить QueryParameter или ControlParameter. Подскажите, пожалуйста?
УдалитьОбновил статью. Добавил информацию по фильтрации данных. Надеюсь, вам это поможет.
УдалитьСпасибо! Вы мне очень помогли.
УдалитьЗдравствуйте! У меня есть тоже вопрос по GridView!
ОтветитьУдалить1. GridView заполняется данными из БД с помощью SqlDataSource, DataKeyNames установлен значением Id
2. При первом заполнении GridView строки редактируются и удаляются без ошибок
3. На странице есть DropDownList с AutoPostBack который фильтрует записи по дате в зависимости от выбранного значения DropDownList (в обработчике DropDownList_OnSelectedIndexChanged я присваиваю SqlDataSource.SelectCommand новое значение и выполняю привязку)
4. После фильтрации данные отображаются правильно, но записи не редактируются и не удаляются. После дебага оказалось, что коллекция DataKeyNames пустая.
Если не сложно, подскажите что можно с этим сделать?
Возможно, когда Вы присваиваете новое значение для SqlDataSource.SelectCommand, в этой команде не делаете селект на поле Id и это поле просто не приходит? Попробуйте внутри грида поставить скрытое поле (asp:hiddenfield) и привязать к нему Value = <%#Eval("Id")%>
УдалитьСпасибо за быстрый ответ!!! И за помощь большое спасибо!!!
ОтветитьУдалитьВ GridView:
/>
В обработчике SqlDataSource_Updating:
int ridx = GridView1.EditIndex;
HiddenField h = (HiddenField)GridView1.Rows[ridx].FindControl("HiddenField1");
Int32 id = Convert.ToInt32(h.Value);
e.Command.Parameters["@IdAdditional"].Value = id;
Может у кого-то будет такая же проблема!!!
Еще раз большое спасибо))))
Не хочет печатать кусок кода, ну ладно(((
ОтветитьУдалитьПолезно, спасибо
ОтветитьУдалитьЗдравствуйте!
ОтветитьУдалитьТут описано, как сделать поиск в базе. А мне надо не отфильтровать запись (записи), а перейти к нужной записи (я знаю ее уникальный ключ) и отобразить ее, например, в GridView, так что соседние записи тоже были бы видны (если я поставлю фильтр по этому ключу, то соседние записи отсекаются и не видны ). В WinForms это делала функция Find объекта BindingSource, привязанного к нужной таблице. А здесь?
Здравствуйте. Тут вам необходимо сделать:
Удалить1. Запрос к источнику данных такой, чтобы он возвращал и ваш элемент, и ещё элементы. Это сделать нетрудно, достаточно правильно сортировать и фильтровать.
2. Для того, чтобы отметить в гриде строку как выбранную, необходимо указать индекс выбранной строки
Спасибо за ответ. Но сортировка в базе данных означает нарушение "естественного" порядка записей, а мне надо его сохранить.
УдалитьПо ходу дела у меня еще возник другой вопрос: есть источник данных для GridView, назовем его ds, он возвращает результаты некоего запроса, предложение SELECT этого запроса я могу узнать из свойства ds.SelectCommand, предложение WHERE из свойства ds.FilterExpression, а из какого свойства я узнаю каков порядок и поле сортировки, т.е. предложение ORDER BY?
Понятия "естественного порядка" в базах данных нет, данные всегда приходят в каком то прядке (например, по дате добавления или идентификатору). И я не совсем понимаю, зачем вам узнавать параметры запроса, если вы должны их задавать сами? По поводу параметров - у меня пока нет возможности подебажить и поглядеть, попробуйте это сделать самостоятельно. Ну, или есть же документация для этого контрола.
УдалитьЗдравствуйте Артём!Подскажите, как решить такую задачу для ASP.net
ОтветитьУдалитьВнутри GridView должны быть кнопки с надписями. При нажатии на кнопку внутри GridView я должен получать некое уникальное значение, позволяющее идентифицировать нажатую кнопку.
Эти кнопки - не управляющие кнопки типа "выбрать" или "редактировать" строку, а суть сами данные.
GridView формируется динамически, то есть в дизайне я ничего не создаю.
Внутри создаются ButtonField, потом присваиваю DataSet.
Пока мне удалось получить только номер строки, в которой было нажатие элемента.
заранее спасибо.
Здравствуйте. Вообще по номеру строки Вы можете узнать все, что Вам нужно, достаточно просто в какой нибудь невидимый столбец записать данные и потом, зная строку, поиском по контролам эти данные найти. Если нужно динамически прикрутить TemplateField, то это тоже можно (гуглим). Ну, и я накатал готовый примерчик, что можно кинуть на страничку и погонять (пример надуманный, просто как вариант)
УдалитьСпасибо за ответ, примеры посмотрю.
УдалитьТолько не понял, как "зная строку, поиском по контролам эти данные найти."
Возможно, ответ в примере.
Добрый день, Артём
ОтветитьУдалитьПо примерам получилось следующее:
1. не вызывается обработка события bt_Click у меня почему-то.
2. Использую поэтому обработку события в GridView:
GridView1_RowCommand(object sender, GridViewCommandEventArgs e)
Но оно возникает, если нажата в GridView кнопка типа ButtonField, а если использовать TemplateField то не срабатывает.
Ещё момент, который я не указал.
УдалитьДинамически строятся также и столбцы. Их имя и количество заранее неизвестно.
Их имена формируются динамически, например Name0, Name1 и т.д.
Причём, при нажатии кнопки внутри GridView формируется новый набор данных, со своим количеством строк и столбцов.
Прошу прощения, что задержал с ответом. Если ещё актуально, просто пришлите мне на мыло код, который не работает. Мыло есть в контактах.
УдалитьАртем, добрый день! Подскажите, пожалуйста как сделать сортировку GridView по определенным полям, если данные получены с помощью DataSet?
ОтветитьУдалитьСортировка должна происходить при нажатии на заголовки GridView.
ОтветитьУдалитьКак то так:
Удалитьprotected void gv_OnSorting(object sender, GridViewSortEventArgs e)
{
var ta = new Data.DataSet1TableAdapters.TestDataTableAdapter();
var dataTable = ta.GetData();
var prevSortExpression = ViewState["prevExpr"] as String;
var prevSortDirection = ViewState["prevDir"] as String;
var direction = string.Compare(prevSortExpression, e.SortExpression) == 0
? (string.Compare(prevSortDirection, "ASC") == 0 ? "DeSC" : "ASC")
: "ASC";
var dataview = new DataView(dataTable);
dataview.Sort = e.SortExpression + " " + direction;
ViewState["prevExpr"] = e.SortExpression;
ViewState["prevDir"] = direction;
gv.DataSource = dataview;
gv.DataBind();
}
Добрый день! Артем, у меня данные получены с помощью DataSet, а не через Adapter.
ОтветитьУдалитьНу через DataSet вы получаете DataTable? Если так, то логика то та ж самая. В противном случае, код в студию, или мне на мыло.
УдалитьВот ещё вариант. Без типизированных датасетов
Удалитьprotected void gv_OnSorting(object sender, GridViewSortEventArgs e)
{
using (var conn = new SqlConnection(ConfigurationManager.ConnectionStrings["DbTestConnectionString"].ConnectionString))
{
var command = "SELECT [ID],[Name] FROM [DbTest].[dbo].[TestData]";
var prevSortExpression = ViewState["prevExpr"] as String;
var prevSortDirection = ViewState["prevDir"] as String;
var direction = string.Compare(prevSortExpression, e.SortExpression) == 0
? (string.Compare(prevSortDirection, "ASC") == 0 ? "DeSC" : "ASC")
: "ASC";
var sortExpr = e.SortExpression + " " + direction;
ViewState["prevExpr"] = e.SortExpression;
ViewState["prevDir"] = direction;
if (!string.IsNullOrEmpty(direction)) command += " ORDER BY " + sortExpr;
var sqlCommand = new SqlCommand(command, conn);
var dataAdapter = new SqlDataAdapter(sqlCommand);
var ds = new DataSet();
dataAdapter.Fill(ds);
var table = ds.Tables[0];
gv.DataSource = table;
gv.DataBind();
}
Естественно, код надо будет допилить (например, избегать sql инъекций). Но суть должна быть ясна.
Заработало вот в каком виде:
ОтветитьУдалитьprivate const string ASCENDING = " ASC";
private const string DESCENDING = " DESC";
public SortDirection GridViewSortDirection
{
get
{
if (ViewState["sortDirection"] == null)
ViewState["sortDirection"] = SortDirection.Ascending;
return (SortDirection)ViewState["sortDirection"];
}
set { ViewState["sortDirection"] = value; }
}
protected void gvMain_Sorting1(object sender, GridViewSortEventArgs e)
{
string sortExpression = e.SortExpression;
if (GridViewSortDirection == SortDirection.Ascending)
{
GridViewSortDirection = SortDirection.Descending;
SortGridView(sortExpression, DESCENDING);
}
else
{
GridViewSortDirection = SortDirection.Ascending;
SortGridView(sortExpression, ASCENDING);
}
}
private void SortGridView(string sortExpression, string direction)
{
DataSet MyData = GetMainTable();
DataTable dt = MyData.Tables["Problems"];
DataView dv = new DataView(dt);
dv.Sort = sortExpression + direction;
gvMain.DataSource = dv;
gvMain.DataBind();
}
Будьте бдительны. Насколько я понял, в данном случае сортировка будет проходить уже после получения данных с сервера.
УдалитьЭтот комментарий был удален автором.
ОтветитьУдалитьНу можно сделать фиксированную высоту контейнеру грида, или я не понял в чем вопрос, собственно.
Удалитьпримеры смотрите в гугле, их полно.
УдалитьДобрый день! Добавил к таблице удалении строки. Первый раз удаление происходит без проблем, определяет ключ, но в последующие разы упорно ключи не определяет. Просто пишет пустую строку и все. Ну и собственно удаление не происходит. В чем тут проблема может быть?
ОтветитьУдалитьВсе, разобрался. Глюк связан с скрытым поле ключей. Его нужно "проявлять" перед каждой процедурой удаления и тогда все норм)
УдалитьЗдорово, что так быстро удалось разобраться :)
УдалитьОтлично написано! :)
ОтветитьУдалитьДобрый день! Подскажите как правильно построить работу с GridView для:
ОтветитьУдалить1. Перехода от страницы с DetailView с выбранными значения полей из GridView обратно на страницу с GridView.
2. При переходе обратно GridView должен отобразить туже страницу что и до первоначального перехода на страницу с DetailView.
Это может быть при просмотре детального содержимого со 2... N страницы GridView или при отображение GridView результатов поиска
С уважением Юрий Косенко
Так грид и DetailView можно же разместить на одной странице, и при выборе элемента в гриде отображать детализацию в Detail
Удалитьвопрос решен при помощи Session
Удалитьвот код
using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;
using System.Web.UI;
using System.Web.UI.WebControls;
using DSTableAdapters;
using System.IO;
using Udove;
using System.Data;
public partial class NEWS_Session : System.Web.UI.Page
{
UsersPlanZakupTA upzTA = new UsersPlanZakupTA();
protected void Page_Load(object sender, EventArgs e)
{
if (!IsPostBack)
{
gvPlanBind();
}
}
private void gvPlanBind()
{
if (Session["sPlan"] != null)
{
gvPlan.DataSource = upzTA.GetData();
gvPlan.PageIndex = Convert.ToInt32(Session["sPlan"]);
}
else
{
gvPlan.DataSource = upzTA.GetData();
}
gvPlan.DataBind();
}
protected void gvPlan_PageIndexChanging(object sender, GridViewPageEventArgs e)
{
gvPlan.PageIndex = e.NewPageIndex;
Session["sPlan"] = e.NewPageIndex;
gvPlanBind();
}
}
Мое мнение - для передачи целого числа между страницами, можно было бы обойтись параметром запроса, типа http://.../News.aspx?page=1 . Сессия - это уже тяжелая артиллерия, и копить там лишние параметры не очень хорошо. Но если уж делаете через сессию, то
Удалить1. Давайте ключам осмысленные значения, типа News_page_index
2. Сохраняйте эти значения ключей в конфигах или статических константах, чтобы можно было такими значениями централизованно управлять.
Артём спасибо за Ваше замечание - обязательно его учу при дальнейшей разработке.
ОтветитьУдалитьДобрый день!
ОтветитьУдалитьПодскажите, пожалуйста, у GridView есть возможность указать ShowInsertButton="true". После чего он отображает кнопку "вставить". Но не появляется никаких пустых колонок, для заполнения. Я для добавления использовал DetailsView у Вас в примере - текстовые поля. Грид не поддерживает вставку ? Для чего тогда кнопка ? Я нигде не могу найти информацию по этому поводу, к сожалению. Буду очень благодарен за разъяснение. Спасибо!
Я уже некоторое время не использую GridView (и вообще Web Forms), потому Вам придется слегка погуглить, благо информации об этом вроде достаточно
УдалитьСпасибо, не правильно гуглил. ShowInsertButton свойство CommandField (не GridView как такового), но, и GridView и DetailsView используют CommandFields - но GridView не поддерживает вставку.
УдалитьСпасибо, что отписали. Думаю, другим читателям это тоже будет полезно знать.
Удалить