Использование CascadingDropDown с базой данных (C#)

Кристиан Венц (Christian Wenz)

Загрузить PDF-файл

Элемент управления CascadingDropDown в наборе элементов управления AJAX расширяет элемент управления DropDownList, чтобы изменения в одном DropDownList загружали связанные значения в другой DropDownList. Для этого необходимо создать специальную веб-службу.

Общие сведения

Элемент управления CascadingDropDown в наборе элементов управления AJAX расширяет элемент управления DropDownList, чтобы изменения в одном DropDownList загружали связанные значения в другой DropDownList. (Например, один список содержит список штатов США, а следующий список заполняется крупными городами в этом штате.) Для этого необходимо создать специальную веб-службу.

Этапы

Прежде всего, требуется источник данных. В этом примере используется база данных AdventureWorks и SQL Server 2005, экспресс-выпуск Майкрософт. База данных является необязательной частью установки Visual Studio (включая экспресс-выпуск) и также доступна для отдельной загрузки в разделе https://go.microsoft.com/fwlink/?LinkId=64064. База данных AdventureWorks является частью SQL Server примеров и примеров баз данных 2005 года (скачать по адресу https://www.microsoft.com/download/details.aspx?id=10679). Самый простой способ настроить базу данных — использовать SQL Server Management Studio Майкрософт (/sql/ssms/download-sql-server-management-studio-ssms) и присоединить AdventureWorks.mdf файл базы данных.

В этом примере предполагается, что экземпляр SQL Server 2005, экспресс-выпуск вызывается SQLEXPRESS и находится на том же компьютере, что и веб-сервер. Это также настройка по умолчанию. Если конфигурация отличается, необходимо адаптировать сведения о подключении к базе данных.

Чтобы активировать функциональные возможности ASP.NET AJAX и control Toolkit, ScriptManager элемент управления должен размещаться в любом месте страницы (но в элементеform<> ):

<asp:ScriptManager ID="asm" runat="server" />

На следующем шаге требуются два элемента управления DropDownList. В этом примере мы используем сведения о поставщике и контактные данные из AdventureWorks, поэтому создадим один список для доступных поставщиков и один для доступных контактов:

<div>
 Vendor: <asp:DropDownList ID="VendorsList" runat="server"/><br />
 Contacts: <asp:DropDownList ID="ContactsList" runat="server"/><br />
</div>

Затем на страницу необходимо добавить два расширителя CascadingDropDown. Один заполняет первый список (поставщиков), а другой — второй (список контактов). Необходимо задать следующие атрибуты:

  • ServicePath: URL-адрес веб-службы, доставляющего записи списка.
  • ServiceMethod: веб-метод доставки записей списка
  • TargetControlID: идентификатор раскрывающегося списка.
  • Category: сведения о категории, которые передаются в веб-метод при вызове.
  • PromptText: текст отображается при асинхронной загрузке данных списка с сервера.
  • ParentControlID: (необязательно) родительский раскрывающийся список, который запускает загрузку текущего списка.

В зависимости от используемого языка программирования имя рассматриваемой веб-службы меняется, но все остальные значения атрибутов совпадают. Ниже приведен элемент CascadingDropDown для первого раскрывающегося списка:

<ajaxToolkit:CascadingDropDown ID="ccd1" runat="server"
 ServicePath="CascadingDropdown1.cs.asmx" ServiceMethod="GetVendors"
 TargetControlID="VendorsList" Category="Vendor"
 PromptText="Select Vendor" />

Расширители элементов управления для второго списка должны задать ParentControlID атрибут , чтобы при выборе записи в списке поставщиков активировался загрузка связанных элементов в списке контактов.

<ajaxToolkit:CascadingDropDown ID="ccd2" runat="server"
 ServicePath="CascadingDropdown1.cs.asmx" ServiceMethod="GetContactsForVendor"
 TargetControlID="ContactsList" ParentControlID="VendorsList"
 Category="Contact"
 PromptText="Select Contact" />

Затем фактические действия выполняются в веб-службе, которая настраивается следующим образом. Обратите внимание, что [ScriptService] атрибут используется, в противном случае ASP.NET AJAX не сможет создать прокси-сервер JavaScript для доступа к веб-методам из кода клиентского скрипта.

<%@ WebService Language="C#" Class="CascadingDropdown1" %>
using System.Web.Script.Services;
using AjaxControlToolkit;
using System;
using System.Web;
using System.Web.Services;
using System.Web.Services.Protocols;
using System.Collections.Generic;
using System.Collections.Specialized;
using System.Data.SqlClient;
[ScriptService]
public class CascadingDropdown1 : System.Web.Services.WebService
{
 // ...
}

Сигнатура веб-методов, вызываемых CascadingDropDown, выглядит следующим образом:

public CascadingDropDownNameValue[] MethodNameHere(string knownCategoryValues, 
    string category)

Таким образом, возвращаемое значение должно быть массивом типа CascadingDropDownNameValue , определенного набором средств управления. Этот GetVendors() метод довольно прост в реализации: код подключается к базе данных AdventureWorks и запрашивает первые 25 поставщиков. Первый параметр в конструкторе CascadingDropDownNameValue — это подпись записи списка, второй — его значение (атрибут value в элементе <option> HTML). Ниже приведен код:

[WebMethod]
public CascadingDropDownNameValue[] GetVendors(string knownCategoryValues, string category)
{
    SqlConnection conn = new SqlConnection("server=(local)\\SQLEXPRESS; 
    Integrated Security=true; Initial Catalog=AdventureWorks");
    conn.Open();
    SqlCommand comm = new SqlCommand("SELECT TOP 25 VendorID, Name 
    FROM Purchasing.Vendor",conn);
    SqlDataReader dr = comm.ExecuteReader();
    List<CascadingDropDownNameValue> l = new List<CascadingDropDownNameValue>();
    while (dr.Read())
    {
        l.Add(new CascadingDropDownNameValue(dr["Name"].ToString(),
        dr["VendorID"].ToString()));
    }
    conn.Close();
    return l.ToArray();
}

Получить связанные контакты для поставщика (имя метода) GetContactsForVendor()немного сложнее. Прежде всего, необходимо определить поставщика, который был выбран в первом раскрывающемся списке. Набор средств управления определяет вспомогательный метод для этой задачи. Метод ParseKnownCategoryValuesString() возвращает StringDictionary элемент с данными раскрывающегося списка:

[WebMethod]
public CascadingDropDownNameValue[] GetContactsForVendor(string knownCategoryValues, 
    string category)
{
    int VendorID;
    CascadingDropDown.ParseKnownCategoryValuesString(knownCategoryValues);

По соображениям безопасности эти данные необходимо сначала проверить. Таким образом, если имеется запись Vendor (так как Category свойство первого элемента CascadingDropDown имеет значение "Vendor"), можно получить идентификатор выбранного поставщика:

if (!kv.ContainsKey("Vendor") || !Int32.TryParse(kv["Vendor"],out VendorID)) 
{
    throw new ArgumentException("Couldn't find vendor.");
};

Остальная часть метода довольно прямолинейна, то. Идентификатор поставщика используется в качестве параметра для SQL-запроса, который получает все связанные контакты для этого поставщика. Еще раз метод возвращает массив типа CascadingDropDownNameValue.

SqlConnection conn = new SqlConnection("server=(local)\\SQLEXPRESS; 
 Integrated Security=true; Initial Catalog=AdventureWorks");
 conn.Open();
 SqlCommand comm = new SqlCommand("SELECT Person.Contact.ContactID, FirstName, LastName 
 FROM Person.Contact,Purchasing.VendorContact 
 WHERE VendorID=@VendorID 
 AND Person.Contact.ContactID=Purchasing.VendorContact.ContactID",conn);
 comm.Parameters.AddWithValue("@VendorID", VendorID);
 SqlDataReader dr = comm.ExecuteReader();
 List<CascadingDropDownNameValue> l = new List<CascadingDropDownNameValue>();
 while (dr.Read())
 {
 l.Add(new CascadingDropDownNameValue(
 dr["FirstName"].ToString() + " " + dr["LastName"].ToString(),
 dr["ContactID"].ToString()));
 }
 conn.Close();
 return l.ToArray();
}

Загрузите страницу ASP.NET и через некоторое время список поставщиков будет заполнен 25 записями. Выберите одну запись и обратите внимание на то, как второй раскрывающийся список заполняется данными.

Первый список заполняется автоматически

Первый список заполняется автоматически (щелкните для просмотра полноразмерного изображения)

Второй список заполняется в соответствии с выбором в первом списке

Второй список заполняется в соответствии с выбранным в первом списке (щелкните для просмотра полноразмерного изображения)