Опубликован: 13.07.2010 | Уровень: специалист | Доступ: платный
Самостоятельная работа 32:

Пользовательские элементы управления

Добавление событий

Событие - это встроенный механизм C#, с помощью которого автоматически обеспечивается уведомление зарегистрированных объектов о необходимости выполнения ими какого-то действия. Объект предоставляет свои методы-обработчики для реагирования на возникновение определенного события. В этом случае говорят, что объект подписался на некоторое событие. Такой механизм позволяет отдельным объектам обмениваться сообщениями и реагировать на них.

Технически событие представляет собой некую адресную переменную, ссылающуюся на экземпляр класса, поддерживающего событие. Если значение этой ссылки равно null, то событие не произошло, поскольку нет соответствующего объекта. Один объект генерирует событие (создает экземпляр) и посылает уведомление через канал сообщений операционной системы. Другие, подписанные на это событие объекты приложения, получают это сообщение вместе с дополнительной информацией и запускают соответствующие обработчики, предварительно передав им дополнительную информацию в качестве аргументов.

Для объявления адресной переменной события используются делегаты. Делегаты представляются специальным классом, определяющим сигнатуру (типы, количество и порядок следования аргументов) передаваемой с событием информации отправляющей стороной, и сигнатуру обработчиков события принимающей стороны. Делегат объявляется в глобальной области видимости уровня класса, чтобы сделать событие видимым во всех классах приложения. Объект, порожденный классом-делегатом, и является событием.

С помощью конструктора делегата при создании его нового экземпляра в событие добавляется ссылка на обработчик и устанавливается контроль за сигнатурой этого обработчика. Этот процесс называется регистрацией события в классе или подпиской на событие. События поддерживают многоадресатную передачу (multicasting). Это значит, что одно и тоже событие могут принимать многие объекты, но обрабатывать это событие каждый объект будет своим методом, который он указал в конструкторе делегата при подписке на это событие.

На предыдущем шаге мы с помощью вспомогательного класса LinkTableItem могли добавлять в классе пользовательского элемента управления LinkTableTest новые гиперссылки типа asp:HyperLink к элементу asp:DataList. По щелчку клиента на одной из гиперссылок он безусловно направлялся по указанному адресу. Теперь же мы хотим перехватывать и распознавать щелчок на любой гиперссылке для контроля за действиями пользователя. Сейчас мы не будем оценивать полномочия пользователя, просто распознаем гиперссылку и выведем ее на экран.

Ранее в дескрипторном представлении пользовательского элемента управления мы применяли стандартный элемент asp:HyperLink. Если посмотреть на класс этого элемента System.Web.UI.WebControls. HyperLink через панель Object Browser, то можно увидеть, что он не имеет никаких событий


С другой стороны, класс LinkButton имеет два события, которые генерирует элемент при выборе пользователя


Эти события мы и будем перехватывать в нашем элементе управления.

  • Измените в шаблоне <ItemTemplate> дескрипторного представления файла LinkTable.ascx имя элемента HyperLink на LinkButton и имя атрибута NavigateUrl на CommandArgument, после чего файл должен стать таким
<%@ Control Language="C#" AutoEventWireup="true" 
    CodeFile="LinkTable.ascx.cs" Inherits="LinkTable" %>
    
<table cellpadding="2">
    <tr>
        <td>
            <asp:Label ID="lblTitle" runat="server" Font-Bold="true" 
                       Font-Names="Vernada" Font-Size="Small"
                       ForeColor="#C00000">
                <%--Здесь будет заголовок--%>
            </asp:Label>
        </td>
    </tr>
    <tr>
        <td>
            <asp:DataList ID="listContent" runat="server">
                <ItemTemplate>
                    <img alt="маркер" height="13" src="32_27.gif" width="25">
                    <asp:LinkButton ID="HyperLink1" runat="server" 
                        CommandArgument='<%# DataBinder.Eval(Container.DataItem, "Url") %>'
                        Text='<%# DataBinder.Eval(Container.DataItem, "Text") %>'>
                    </asp:LinkButton>
                </ItemTemplate>
            </asp:DataList>
        </td>
    </tr>
</table>
Листинг 32.16. Смена типа элемента в шаблоне файла LinkTable.ascx

Мы хотим создать событие с именем LinkClicked. Для распознавания выбранного пункта списка нам вместе с будущим событием LinkCliked нужно будет передавать информацию о значениях свойств Text и Url, которые мы определили ранее во вспомогательном классе LinkTableItem.cs. Затем в обработчике события LinkClicked мы распознаем эту информацию и выведем ее на экран.

Напрямую экземпляр класса LinkTableItem.cs в аргументах события передавать нельзя - не тот тип, поэтому упакуем его в класс LinkTableEventArgs, который будет наследником библиотечного класса System.EventArgs. Объявим также делегат события в глобальной области видимости.

  • Откройте файл LinkTable.ascx.cs пользовательского элемента управления и добавьте в него объявление нового класса и делегата, после чего общий код должен быть таким
using System;
using System.Data;
using System.Configuration;
using System.Collections;
using System.Web;
using System.Web.Security;
using System.Web.UI;
using System.Web.UI.WebControls;
using System.Web.UI.WebControls.WebParts;
using System.Web.UI.HtmlControls;
    
public partial class LinkTable : System.Web.UI.UserControl
{
    protected void Page_Load(object sender, EventArgs e)
    {
    
    }
    
    public string Title
    {
        get { return lblTitle.Text; }
        set { lblTitle.Text = value; }
    }
    
    private LinkTableItem[] items;
    public LinkTableItem[] Items
    {
        get { return items; }
        set
        {
            items = value;
    
            // Обновляем сетку
            listContent.DataSource = items;
            listContent.DataBind();
        }
    }
}
    
// Объявляем делегат будущего события LinkClicked, 
// который определит сигнатуру события и обработчика
public delegate void LinkClickedEventHandler(object sender, LinkTableEventArgs e);
    
// Класс-упаковка экземпляра LinkTableItem для 
// передачи с событием в качестве аргумента
public class LinkTableEventArgs : EventArgs
{
    // Конструктор
    public LinkTableEventArgs(LinkTableItem item)
    {
        selectedItem = item;
    }
    
    // Внутреннее поле будет хранить объект гиперссылки
    // и контролируется свойством "только для чтения"
    private LinkTableItem selectedItem;
    public LinkTableItem SelectedItem
    {
        get { return selectedItem; }
    }
    
    // Внутренне поле хранит флаг отмены, переданный сгенерировавшим 
    // событие классом-источником для обработчика класса-приемника
    private bool cancel = false;// Инициализируется при создании
    public bool Cancel
    {
        get { return cancel; }
        set { cancel = value; }
    }
}
Листинг 32.17. Добавление нового класса-упаковки в файл LinkTable.ascx.cs

Таким образом, мы объявили класс для упаковки передаваемой в событии информации. мы также объявили делегата, с помощью которого будет создано событие. Теперь нужно добавить код, который перехватит пользовательский щелчок на определенной гиперссылке, распознает эту гиперссылку и через созданное событие LinkClicked отправит эту информацию в канал сообщений Windows. Конечно же, этим должен заниматься класс LinkTable поддержки пользовательского элемента управления.

  • Откройте на редактирование файл LinkTable.ascx в режиме Design
  • Выделите пользовательский элемент управления listContent и в панели Properties двойным щелчком на поле события ItemCommand создайте обработчик с именем по умолчанию


  • Добавьте в класс LinkTable и созданный обработчик стандартного события ItemCommand следующий код
public partial class LinkTable : System.Web.UI.UserControl
{
    protected void Page_Load(object sender, EventArgs e)
    {
    
    }
    
    public string Title
    {
        get { return lblTitle.Text; }
        set { lblTitle.Text = value; }
    }
    
    private LinkTableItem[] items;
    public LinkTableItem[] Items
    {
        get { return items; }
        set
        {
            items = value;
    
            // Обновляем сетку
            listContent.DataSource = items;
            listContent.DataBind();
        }
    }
    
    // Объявляем в классе наше событие LinkClicked 
    // с помощью ранее созданного глобального делегата
    public event LinkClickedEventHandler LinkClicked;
    
    protected void listContent_ItemCommand(object source, DataListCommandEventArgs e)
    {
        if (LinkClicked != null)
        {
            // Выделяем объект ссылки, на которой был выполнен щелчок
            LinkButton link = (LinkButton)e.Item.Controls[1];
    
            // Выделяем нужные свойства Text и Url через атрибуты
            // дескриптора элемента LinkButton и сохраняем их
            // в экземпляре вспомогательного класса
            LinkTableItem item = new LinkTableItem(link.Text, link.CommandArgument);
    
            // Передаем ссылку на сохраненные свойства в класс упаковки аргументов
            LinkTableEventArgs args = new LinkTableEventArgs(item);
    
            // Генерируем событие
            LinkClicked(this, args);
    
            // Пускаем клиента по ссылке, если класс-получатель не запретил
            if (args.Cancel != true)
            {
                Response.Redirect(item.Url);
            }
        }
    }
}
Листинг 32.18. Добавленный код в класс LinkTable файла LinkTable.ascx.cs

Теперь событие LinkClicked будет возникать всякий раз, когда пользователь щелкнет на гиперссылке пользовательского элемента управления. Классом-получателем этого события определим нашу тестовую страницу. Именно в ней мы создадим функцию-обработчик, которая распакует переданные событием аргументы и выведет полученную информацию на экран. Проверку существования события мы выполняем на случай, чтобы наверняка предотвратить генерацию системного исключения, если не существует ни одного обработчика.

Учитывая, что в режиме проектирования оболочка поддерживает автоматическое создание обработчиков через панель Properties только для событий библиотечных классов, создадим заготовку обработчика вручную. При этом нужно соблюсти сигнатуру, установленную объявленным ранее делегатом. Но и в этом случае подсказчик IntelliSense оболочки нам помогает.

  • Откройте на редактирование файл LinkTableTest.aspx.cs и в конце обработчика события Page_Load() класса LinkTableTest начните набирать имя объекта LinkTable1, под которым на тестовой странице у нас числится экземпляр зарегистрированного на ней пользовательского элемента управления. Вы сразу увидите реакцию подсказчика кода IntelliSense

Нажав два раза клавишу Tab мы получим и регистрацию события в классе-приемнике через глобального делегата, и заготовку обработчика. После генерации оболочкой таким способом заготовки обработчика далее можно поступить одним из следующих способов:

  1. Либо оставить все как есть и начать заполнять обработчик
  2. Либо оставить заготовку обработчика, а код регистрации события уничтожить
LinkTable1.LinkClicked += new LinkClickedEventHandler(LinkTable1_LinkClicked);

Если выбрать второй способ, тогда нужно в дескрипторном представлении пользовательского элемента добавить атрибут OnLinkClicked="LinkTable1_LinkClicked", чтобы привязать к событию созданный обработчик (обратите внимание, что имя атрибута формируется из имени события и префикса On ). Но и в этом случае подсказчик кода нам тоже помогает

Выберем второй способ регистрации обработчика для события LinkClicked.

  • Уничтожте регистрацию события в классе LinkTableTest, скопируйте имя сгенерированного обработчика и добавьте его в атрибут OnLinkClicked="LinkTable1_LinkClicked" дескриптора <uc1:LinkTable ID="LinkTable1" runat="server" /> файла тестовой страницы LinkTableTest.aspx
  • В режиме Design тестовой страницы разместите последней текстовую метку asp:Label из вкладки Standard, задайте ей имя lblInfo и очистите поле ее свойства Text (либо удалите одноименный атрибут в дескрипторе). В ней мы будем отображать информацию о нажатой гиперссылке
  • Вернитесь в файл LinkTest.aspx.cs и заполните обработчик так
protected void LinkTable1_LinkClicked(object sender, LinkTableEventArgs e)
    {
        // Ищем подстроку в вызывающей строке
        if (e.SelectedItem.Text.IndexOf("Диалог-Сервис") > 0)
        {
            e.Cancel = false;// Пропустить
        }
        else
        {
            e.Cancel = true; // Не пускаем дальше
            lblInfo.Text = "Вы щелкнули " +
                "\"" + e.SelectedItem.Text + "\".";
            lblInfo.Text += " Извините, но Вам не разрешено идти на " +
                e.SelectedItem.Url + "!!!";
        }
    }
Листинг 32.19. Обработчик добавленного события LinkClicked
  • Запустите тестовую страницу LinkTableTest.aspx и полюбуйтесь на свою работу


Анатолий Федоров
Анатолий Федоров
Россия, Москва, Московский государственный университет им. М. В. Ломоносова, 1989