Litvek - онлайн библиотека >> Кристиан Нагель и др. >> C, C++, C# и др. >> C# для профессионалов. Том II >> страница 200
txtLog и правым краем самой формы.

Наконец, мы добавляем кнопку в массив элементов управления и перемещаем у вниз в положение следующей кнопки:

  // добавление кнопки в форму ...

  this.Controls.Add(newButton);

  // следующая ...

  у += (height + spacing);

 }

}

В результате будет получено изображение:

C# для профессионалов. Том II. Иллюстрация № 192 Кнопки созданы, но пока они ничего не делают. Для того чтобы они работали, необходимо присоединить обработчики событий.

Присоединение обработчиков событий
В .NET существует небольшое различие в способах использования термина делегат. Делегат определяется с помощью ключевого слова Delegate, описывающего сигнатуру метода, который будет использоваться в делегате. Сигнатура метода включает комбинацию параметров и возвращаемый тип. Поэтому метод, который получает один параметр Object и возвращает String, будет иметь сигнатуру, отличную от сигнатуры метода, получающего параметр Hashtable и возвращающего String.

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

Для присоединения обработчика событий необходимо создать делегируемый метод со следующей сигнатурой:

void DelegateMethod(Object sender, System.EventArgs e)

Здесь важно то, что имя создаваемого метода является полностью произвольным, т.е. метод может иметь любое имя, какое мы захотим использовать. До тех пор пока создается метод, который не имеет возвращаемого значения, но получает System.Object, а затем объект System.EventArgs,— все будет нормально. Как можно понять, порядок параметров влияет на сигнатуру метода. Если изменить порядок параметров, метод не будет иметь требуемую сигнатуру для делегируемого метода.

Все обработчики событий в множестве элементов управления Windows.Forms соответствуют этой модели, значит, возможно создание достаточно мощных обработчиков событий. Можно было бы при желании сделать так, чтобы каждое событие каждого элемента управления на форме проходило через один и тот же делегируемый метод. Фактически делегируемый метод вызывается для любого объекта из любого другого объекта. Поэтому элемент управления кнопки объекта формы, называемого MyDialog, может вызывать метод для объекта формы, называемого MyDialogOwner.

Параметр sender предоставляет ссылку на элемент управления, который порождает событие. Чтобы добраться до свойства DynamicId, необходимо преобразовать его в тип DynamicButton. Параметр е определяет некоторые специальные аргументы события. Мы не собираемся это здесь рассматривать, но для некоторых событий в данных аргументах содержится дополнительная информация о том, что вызывает событие. Вот обработчик нажатия кнопки:

// DynamicButton_Click ...

protected void DynamicButton_Click(object sender, System.EventArgs e) {

 // преобразовать sender в button ...

 DynamicButton button = (DynamicButton)sender;

 // msgbox ...

 AddToLog("Clicked" + button.DynamicId);

}

В этом методе нет ничего необычного. Соединим его с событием Click нашей динамической кнопки. Непосредственно перед тем как вызывать Controls.Add, можно добавить обработчик:

// соединить обработчик ...

newButton.Click += new System.EventHandler(this.DynamicButton_Click);

// добавить кнопку к форме ...

this.Controls.Add(newButton);

Теперь, когда возникает связанное с кнопкой событие Click, рабочая среда Framework просматривает список делегированных методов, соединенных с этим событием, инициируя каждый из них по очереди.

Важная деталь — в .NET события могут иметь более одного обработчика. Это означает, что если создать другой делегированный метод, соединить его с помощью нового экземпляра System.EventHandler, то оба метода будут вызываться в момент порождения события. Конечно, методы не вызываются одновременно, так как фактически получится мультипоточное приложение, а это не то, что хотелось бы случайно получить.

Мы не определили, как выглядит наша функция AddToLog,— сделаем это сейчас:

// AddToLog — обновляет представление журнала ...

private void AddToLog(String buf) {

 // обновляет элемент управления журнала ...

 txtLog.Text = (buf + "\r\n" + txtLog.Text);

}

Теперь давайте попробуем выполнить приложение и понажимать на кнопки:

C# для профессионалов. Том II. Иллюстрация № 193
Другие обработчики
Обработчики, которые отвечают на другие события в DynamicButton, могут быть добавлены аналогичным образом. Прежде всего для каждого из них создается метод:

// DynamicButton_Enter ...

protected void DynamicButton_Enter(object sender, System.EventArgs e) {

 // преобразовать sender в button ...

 DynamicButton button = (DynamicButton)sender;

 // msgbox ...

 AddToLog("Enter " + button.DynamicId);

}


// DynamicButton_Leave ...

protected void DynamicButton_Leave(object sender, System.EventArgs e) {

 // преобразовать sender в button ...

 DynamicButton button = (DynamicButton)sender;

 // msgbox ...

 AddToLog("Left " + button.DynamicId);

}

Затем можно добавить обработчики для каждого из них:

// соединить обработчик

newButton.Click +=

 new System.EventHandler(this.DynamicButton_Click);

newButton.MouseEnter +=

 new System.EventHandler(this.DynamicButton_Enter);

newButton.MouseLeave +=

 new System.EventHandler(this.DynamicButton_Leave);

Теперь, если снова выполнить приложение, в журнал будут добавляться сообщения по мере перемещения по кнопкам.

Другой пример

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

Динамические элементы управления можно использовать для настройки интерфейса пользователя приложения в зависимости от некоторых данных среды выполнения. Классическим примером этого является добавление новых возможностей в панель инструментов, когда в каталог приложения вносятся новые дополнительные средства (plug-ins) или модули. Например, установка Adobe Acrobat на компьютере может автоматически добавлять в панель инструментов Word кнопку для создания документа Acrobat.

Другим примером может быть утилита администрирования базы данных. Когда утилита соединяется с базой данных, в интерфейс пользователя должна быть добавлена кнопка, представляющая каждую имеющуюся внутри базы данных таблицу. Или можно создать множество элементов управления текстовых полей с именами всех файлов XML, содержащихся в определенном каталоге, и т.д.

В этом примере мы собираемся создать приложение, загружающее с диска сборку, просматривающее сборку в поисках типов данных, которые наследуют от System.Windows.Forms.Control, и выводящее кнопку в форме для каждого найденного типа. Нажатие на кнопку будет вызывать экземпляр элемента управления и выводить его в форму.

Создание проекта
Создадим