© 2003 Андрей Семак
Рано или поздно каждому профессиональному разработчику программного обеспечения становится тесной среда обитания. По мнению автора, читатель понял, что речь идет о среде разработки, в которой обитает разработчик. У кого это "дом", у кого "строительная площадка", а у кого и то, и другое. В этой статье мы рассмотрим возможность расширения нашего дома. Мы покажем, как именно можно наращивать мощность IDE своими силами, будь то среда разработки Borland Delphi или Borland C++ Builder.
У многих из вас возникало желание добавить несколько полезных функций в любимое IDE. Разработчики фирмы Borland не оставили без внимания эту полезную возможность. Они разработали набор интерфейсов, позволяющих расширять возможности среды, хотя, к сожалению, отсутствие документации по данному вопросу мешает самостоятельному изучению. Жалкие комментарии в исходных текстах не могут воспроизвести всю мощь, которую предоставляет IDE разработчику расширений. Автор статьи попытается передать читателю опыт своих изысканий и постарается своими советами уберечь читателя от многих подводных камней.
Документ построен таким образом, чтобы мы переходили от этапа к этапу от простого к сложному. Насколько это удалось, судить вам. Вместе c вами мы рассмотрим на примерах все сервисы, предоставляемые IDE Borland Delphi\C++Builder.
Итак, начнем?
Многие спросят: "Что такое ToolsAPI?" На что смело можно ответить: ToolsAPI – это набор программных интерфейсов, позволяющих: получать информацию о происходящих в IDE событиях, вызывать функции IDE, создавать свои собственные функции, которые потом будут доступны в среде разработки. Этот набор позволяет нам создавать свои собственные меню, кнопки управления и так далее.
Этот набор интерфейсов размещен в файле ToolsAPI.pas, находящийся в директории {IDEROOT}\Source\Toolsapi\ вашего каталога Delphi\C++Builder.
Существует две модели интерфейсов ToolsAPI, или как их называют – "старый" и "новый" стили. Старый стиль был реализован в средах Delphi 3-4 и C++ Builder 3.-4. Начиная с Delphi 5 и C++ Builder 5, в среду встроена реализация нового стиля. В этом документе мы будем рассматривать только "новый стиль". "Почему?" – возможно, спросите вы. А потому, - ответит автор - что фирма Borland более не развивает интерфейсы "старого" стиля. Это не значит, что расширения, написанные в старом стиле, не будут работать. Разработчики Borland оставили API старого стиля для совместимости с предыдущими версиями IDE. Интерфейсы "старого" ToolsAPI находятся в директории Source\Toolsapi\ в следующих файлах:
Но они нам не понадобятся. Вы спросите, как их различать? Интерфейсы нового стиля имеют префикс "IOTA" и "INTA", а "старого" содержат префикс "TI". Поэтому, их можно различить с первого взгляда. Запомним это.
* Если читатель будет интересоваться "старым" стилем, автор рекомендует обратиться в Интернете к странице Сергея Орлика (в данный момент Сергей работает в российском представительстве Borland) по адресу http://www.geocities.com/SiliconValley/Way/9006. К сожалению, с октября 1999 года страница автором более не поддерживается. Также на его сайте есть ссылка на PDF документ, демонстрирующий архитектуру ToolsAPI "нового" стиля, но для Delphi/C++Builder 4 (http://www.geocities.com/SiliconValley/Way/9006/otapi.pdf)
На первый взгляд IDE Delphi и C++Builder идентичны. Автор возьмет на себя смелость заявить, что и внутренних отличий так же практически нет*. Во всяком случае, в разрезе ToolAPI. То есть, ToolsAPI Delphi5 тождественен ТoolsAPI С++Builder5 и так далее.
* Особенности различий мы рассмотрим далее на примерах.
Расширения, собранные на Delphi без проблем работают в C++Builder аналогичной версии. Согласитесь, это очень удобно. Создав полезную функцию для IDE Delphi, вы всегда сможете использовать её в С++Builder той же версии.
Многие спрашивают: "А как же столь популярный Borland Jbuilder?". Jbuilder – это продукт полностью написанный на Java. Он также имеет ToolsAPI, но кардинально отличается от рассматриваемого нами.
* В этом документе BorlandJbuilder рассматриваться не будет.
Немного терминологии. "Экспертом" мы называем набор функций, расширяющий возможности IDE и выполненный в виде загружаемого модуля. В "новом" стиле нет понятия "Expert" (пережиток "старого стиля"), но есть понятие "Wizard". На самом деле это одно и тоже. Автор взял за основу слово "Эксперт" потому что, по его мнению, оно более точно характеризует предмет нашего с вами внимания. Существуют два типа исполнения экспертов:
В этом документе мы рассмотрим оба типа, особенности их создания и регистрации. У каждого из этих двух типов есть преимущества и недостатки.
Эксперты, выполненные в виде пакетов (BPL), регистрируются как обычные пакеты с компонентами прямо из среды разработки. Для этого достаточно зайти в меню "Component" -> "Install Packages" -> "Add" и выбрать BPL с экспертом.

Рис. 1
Чтобы временно отключить эксперта, достаточно снять галочку напротив пакета в списке этого окна. Для того, чтобы удалить его из системы, следует выбрать пакет в списке и нажать кнопку "Remove".
У этого типа экспертов более сложная регистрация. Для
регистрации нам необходимо запустить редактор реестра вашей
операционной системы (regedit.exe) и открыть ключ
HKEY_CURRENT_USER\Software\Borland\Delphi(или
C++Builder)\X.X\Experts,
где X.X версия
Delphi\C++Builder.
Пример показан на рис. 2.

Рис. 2
Имя значения может иметь произвольное название, значение должно иметь тип REG_SZ (String) и содержать путь к библиотеке с экспертом. Добавляя или удаляя значения в этой ветке, мы подключаем или отключаем расширения.
* В дальнейшем мы рассмотрим, как можно создать саморегистрирующийся эксперт, выполненный в виде библиотеки.
Примечание. Автор для удобства рекомендует использовать ExpertManager из набора Gexperts (http://www.gexperts.org)
Отладка экспертов осуществляется так же, как и отладка DLL и BPL. Для отладки необходимо в параметрах запуска указать приложение среды как Host Application (см рис. 3).

Рис. 3
В параметрах проекта эксперта необходимо выключить оптимизацию и включить Stack Frames, как показано на рис. 4. После установки описанных опций, необходимо сделать Build всему проекту эксперта.

Рис. 4
Под операционной системой Windows XP эксперты, разрабатываемые под Delphi5 и Delphi6, отлаживать проблематично, поскольку по не известным автору причинам отладчик отказывается загружать Debug Symbol Table. Это ведет к невозможности использования точек останова для отладчика (Break points), что приводит к невозможности осуществлять отладку.
Решение этой проблемы вы найдете в Интернете на сайте компании Devrace по адресу http://www.devrace.com/files/debug_xp.zip (или тут). Это небольшой эксперт, разработанный автором статьи, после установки которого принудительно загружается Symbol Table во время подключения проекта в текущий процесс отладчика. Данный эксперт лишний раз демонстрирует полезность и практичность расширений IDE.
Автор рассчитывает на то, что читатель уже знаком с использование COM-интерфейсов. В противном случае, перед прочтением статьи необходимо ознакомится с концепциями и архитектурой COM, поскольку эта часть в документе затронута не будет.
Также рекомендуется обратить внимание читателя на то, как развиваются интерфейсы ToolsAPI от версии к версии. Если читатель решит создавать расширения, которые будут использоваться в нескольких версиях IDE, то можно рекомендовать использовать ToolsAPI от самой низкой версии используемого IDE. Это связано с тем, что развитие интерфейсов порождает правило о совместимости сверху вниз. От высшей версии к низшей. И ни в коем случае наоборот.
Можно обойтись директивами компилятора, используя те или иные методы интерфейсов в высших версиях но, по мнению автора, это очень трудоемко и может привести к путанице.
Итак, время заглянуть в файл ToolsAPI.pas. Этот файл содержит целый ряд сервисов, которые помогут получать информацию из IDE.
Знакомьтесь – IBorlandIDEServices. Это самый "главный"
интерфейс. Из него мы получим все необходимые нам
интерфейсы-сервисы. Все интерфейсы, содержащие в себе слово
"Services" являются производными от
IBorlandIDEServices.
Указатель на IBorlandIDEServices можно
получить двумя способами.
Методы получения указателя на IBorlandIDEServices мы рассмотрим на примере в разделе 9.
Например, нам необходимо получить указатель на INTAServices. Для этого достаточно написать следующую функцию:
|
function NTAServices:
INTAServices; |
Аналогичные функции можно создать для всех остальных сервисов. Мы рассмотрим каждый сервис отдельно по мере изучения материала.
Первым продемонстрируем пример создания эксперта в пакете. Для этого нам необходимо загрузить среду разработки Borland Delphi и создать новый пакет (New Package). Добавим в него новый модуль (Unit). В секции interface необходимо включить объявления ToolsAPI – toolsapi.pas.
Теперь пришло время рассмотреть интерфейс IOTAWizard. Этот интерфейс и есть базовый для класса эксперта. Все его методы нуждаются в реализации, даже если не будут использоваться.
Все объекты, которые будут взаимодействовать с ToolsAPI,
порождаются от TNotifierObject. Его реализацию можно
посмотреть в том же ToolsAPI.pas.
Итак, декларируем
расширение:
|
type |
Теперь рассмотрим все методы по очереди.
Итак, реализация:
|
Implementation { TFirstExpert } procedure
TFirstExpert.Execute; function TFirstExpert.GetIDString:
string; function TFirstExpert.GetName:
string; function TFirstExpert.GetState:
TWizardState; |
Костяк готов, но эксперт требует регистрации. Поскольку рассматривается расширение в пакете, то будем использовать процедуру регистрации, принятую для пакетов. Объявим её в секции interface и реализуем в implementation.
|
… implementation procedure Register; { TFirstExpert } |
Готовый код можно использовать как шаблон для создания экспертов на основе пакетов.
Зарегистрируем его, как описано в разделе 5. Если все сделано правильно, и никаких ошибок регистрации не произошло, то эксперт установится в IDE. Но, увы, наш первенец пока не наделен никакими функциями. Его надо заставить выполнить метод Execute. "Как?" - спросите вы. Чтобы ответить на этот вопрос автор поведет читателя дальше.
* Рассматриваемый в этой главе пример находится в каталоге FirstExpert\Package в "Приложении 1".
Ну что ж, пришло время рассмотреть MenuWizard. Из файла ToolsAPI.pas видно, что интерфейс IOTAMenuWizard является наследником IOTAWizard. В него добавлен новый метод GetMenuText. Добавим этот интерфейс в объявление нашего эксперта:
|
Type |
И реализуем метод, который добавит пункт меню в подменю "Help" IDE. Метод должен вернуть текст для свойства Caption добавляемого пункта:
|
function TFirstExpert.GetMenuText:
string; |
Соберем проект. Автор хотел бы обратить ваше внимание на то, что после компиляции нам не надо перерегистрировать расширение. Это произойдет автоматически. Теперь откроем главное меню IDE "Help". На рис. 5 видно, что в меню добавлен новый пункт.

Рис. 5.
Выполним его. Диалог MessageBox в методе Execute не заставил себя ждать. (Рис. 6)

Рис. 6
Это простейший способ создать расширение, которое интегрируется в меню IDE.
Эксперта на базе IOTAMenuWizard нельзя заставить подключиться к другим пунктам главного меню. Но есть другие способы, которые мы рассмотрим далее в этой статье.
Пришла пора ознакомиться с методом и особенностями реализации экспертов в библиотеках. Для начала создадим проект библиотеки (DLL).
Для успешной регистрации эксперта в IDE необходимо реализовать две функции. Это (см. ToolsAPi.pas):
Прошу обратить внимание читателя на то, что данный тип регистрации не подразумевает динамическое отключение (выгрузку) расширения из среды, как это было с пакетами. Эксперт регистрируется во время загрузки среды разработки и существует на протяжении всего времени, пока IDE загружено.
Итак, рассмотрим более детально функцию регистрации:
|
function InitWizard(const
BorlandIDEServices: IBorlandIDEServices; |
Реализуем эти функции в DPR-файле проекта библиотеки:
|
library FisrtExpertDll; uses {$R *.res} var const procedure DoneWizard; function InitWizard(const
BorlandIDEServices: IBorlandIDEServices; {Экспорт точки входа для
IDE} begin end. |
Для регистрации такого расширения нам понадобится указатель на IOTAWizardServices. Регистрация и разрегистрация выполняется с помощью именно этого интерфейса. Это методы AddWizard и RemoveWizard соответственно.
Метод AddWizard возвращает индекс расширения в системе, в случае удачной регистрации, или значение "–1" в случае неудачи. Метод RemoveWizard удаляет расширение из IDE, используя индекс, полученный при регистрации. Как это сделать, подробно с комментариями видно из листинга.
Сохраним модуль с реализацией TfirstExpert под именем FirstExpClassDLL.pas в каталог с проектом библиотеки и удалим из него процедуру регистрации для пакетов. Расширение в DLL реализовано. Осталось подключить его в среду.
Как это сделать мы рассматривали в пятом разделе данной статьи.
Для упрощения установки библиотеки, мы можем воспользоваться функциями регистрации COM-объектов. Читателю известно, что регистрация COM-объектов в DLL осуществляется утилитой RegSvr32.exe, которая поставляется с любой операционной системой Windows. Для этого необходимо включить в исходный код проекта 2 экспортируемые функции:
Эти функции описаны в модуле ComServ, который так же необходимо включить в uses секцию проекта.
Но для полноценной установки необходимо, чтобы библиотека определила свое местонахождение. Воспользуемся для этого сервисной функцией, которая возвращает полный путь к библиотеке.
|
function GetDLLFileName:
string; |
Теперь очередь за реализацией установки:
|
const function DllRegisterServer: HResult;
stdcall; function DllUnregisterServer: HResult;
stdcall; exports |
Из листинга видно, как происходит установка и удаление информации из системного реестра. Теперь необходимо собрать проект и установить расширение.
Как это сделать, изображено на рис. 7

Рис. 7
Результатом выполнения установки будет окно, изображенное на рис. 8

Рис. 8
Удалить расширение из Registry можно той же утилитой, только с ключем –u.Можно проверить, прописан ли необходимый ключ в реестр. См. Рис. 9.

Рис. 9
Запустим новый экземпляр IDE и убедимся, что расширение установилось. Правда, просто?
Читатель сам должен решить, какой тип эксперта необходимо использовать в каждом конкретном случае.
* Рассматриваемый в этой главе пример находится в каталоге FirstExpert\Library в "Приложении 1."
Теперь рассмотрим один из самых простых и полезных сервисов IDE. Автор расскажет, как управлять окном сообщений. Работа с ним осуществляется с помощью сервиса IOTAMessageServices.
Обратимся в ToolAPI.pas за помощью. Рассмотрим несколько методов, а именно:
|
{ Добавить заголовочное
сообщение } |
Для обращения к данному сервису нам будет необходима функция, возвращающая указатель на IOTAMessageServices. А поскольку этот интерфейс является сервисным, то получить его можно из глобальной переменной BorlandIDEServices.
|
function OTAMessageServices:
IOTAMessageServices; |
Включим эту функцию в модуль с экспертом в секцию implementation.
Осталось только изменить реализацию Execute. Заменим MessageBox на вывод в окно сообщений IDE:
|
procedure
TFirstExpert.Execute; |
Теперь установим и выполним расширение. Результат работы изображен на рис. 10

Рис. 10
* Рассматриваемый в этой главе пример находится в каталоге MessageServices в "Приложении 1".
Что такое интерфейс нотификаций? Нотификатором называют событийный интерфейс, позволяющий получать те или иные события от IDE. Существует несколько типов нотификаторов, но необходимо отметить, все они порождены от одного базового интерфейса IOTANotifier. Такая архитектура упрощает реализацию и понимание организации перехвата событий от среды разработки.
В этой статье мы не будем рассматривать все нотификаторы, которые доступны в ToolsAPI, но обязательно остановимся на ключевых моментах.
Итак, снова обратимся к ToolsPAI.pas. Какие методы имеет базовый интерфейс нотификатора IOTANotifier?
|
type |
На методах BeforeSave, AfterSave и Modified мы остановимся позже, в разделе 18.
Метод Destroyed вызывается перед разрушением нотификатора.
Поскольку ToolsAPI основано на интерфейсной модели, то IDE сама следит за "нужностью" того или иного объекта. Если читатель заметил, то TnotifierObject является наследником TinterfacedObject и ему ни в коем случае нельзя вызывать метод Free принудительно из кода программы. Как только количество ссылок на созданный нами объект будет равно нулю, экземпляр объекта разрушится сам. Для более подробной информации необходимо обратиться к источникам, связанным с использованием COM–интерфейсов в разделы, посвященные подсчету количества ссылок на объекты.
Нам позволено только создавать, подключать свои экземпляры к IDE и отключать их. После отключения, если экземпляр более никем не используется, он разрушится сам. Это важно, не будем забывать об этом.
Попробуем теорию на практике. Для этого нам понадобится получить указатель на IOTAServices и создать нотификатор от IOTAIDENotifier.
Напишем функцию, возвращающую указатель на интерфейс необходимого нам сервиса:
|
function OTAServices:
IOTAServices; |
Пришло время объявить класс нотификатора IDE:
|
type |
Нас интересует метод FileNotification, который позволит получать сообщения об операциях с файлами в IDE. Именно его мы и будем использовать в рассматриваемом примере. Из ToolsAPI.pas видно, что параметр NotifyCode может принимать следующие значения:
Значение параметра |
Описание |
|
OfnFileOpening |
Перед открытием файла FileName |
|
OfnFileOpened |
Файл FileName открыт в IDE |
|
OfnFileClosing |
Перед закрытием файла FileName |
|
OfnDefaultDesktopLoad |
Загружены настройки для IDE по умолчанию |
|
OfnDefaultDesktopSave |
Сохранены настройки IDE по умолчанию |
|
OfnProjectDesktopLoad |
Загружены настройки для проекта FileName |
|
OfnProjectDesktopSave |
Сохранены настройки для проекта FileName |
|
OfnPackageInstalled |
Пакет FileName с компонентами или экспертами установлен в IDE |
|
OfnPackageUninstalled |
Пакет FileName с компонентами или экспертами удален из IDE |
|
OfnActiveProjectChanged |
Вызывается только при работе с группой проектов, если количество проектов более одного. Важно: данный тип отсутствует в IDE версии ниже 6. |
Параметр Cancel можно использовать для прерывания операции, которая породила нотификацию, если его установить в True. В нашем случае оставим его без внимания.
Реализуем метод FileNotification, который будет обрабатывать приходящие события из IDE и сообщать о них в окне MessageWindow IDE:
|
procedure
TIDENotifier.FileNotification(NotifyCode:
TOTAFileNotification; const FileName:
string; var Cancel:
Boolean); |
Осталось рассмотреть добавление и удаление нотификатора. Сервис IOTAServices для этого имеет соответствующие методы AddNotifier и RemoveNotifier.
Чтобы мы могли успешно удалить нотификатор из IDE, необходимо ввести для него значение индекса. Введем индекс как переменную класса FNotifierIndex : integer в секции Private эксперта. Мы будем осуществлять добавление нотификатора при создании эксперта и его удаление в процессе разрушения. Для этого введем объявление конструктора и деструктора:
|
type |
Добавление и удаление нотификатора имеет следующую реализацию:
|
… constructor
TFirstExpert.Create; destructor
TFirstExpert.Destroy; |
В конструкторе создается экземпляр нотификатора объекта и его индекс сохраняется в переменной. Из деструктора видно, что если регистрация нотификатора произошла успешно, то он будет удален из IDE.
Результат работы нотификатора изображен на рис. 11.

Рис. 11
* Рассматриваемый в этой главе пример находится в каталоге IDEServices в "Приложении 1".
Рассмотрим еще один сервис, который нам предоставляет IDE - сервис управления напоминаниями ToDo. В ToolsAPi.pas он объявлен как IOTAToDoServices.
Немного расширим задачу и создадим упрощенный менеджер управления напоминаниями. По сравнению с предыдущими примерами задача будет несколько сложнее, и автор постарается максимально разъяснить материал.
Для решения этой задачи будет создан интерактивный эксперт, который позволит нам редактировать и удалять напоминания в текущем проекте.
Итак, создадим функцию, которая обеспечит на доступ к IOTAToDoServices.
|
function OTATODOServices:
IOTATODOServices; |
Создадим форму менеджера с основными функциями и добавим её в проект (рис. 12).

Рис. 12
В конструкторе создадим форму, в деструкторе разрушим её и переопределим метод Execute у расширения для визуализации формы менеджера.
|
… destructor
TFirstExpert.Destroy; procedure
TFirstExpert.Execute; |
Определим обработчик OnShow формы нашего менеджера. В нем мы будем сканировать весь список с записями, и переносить его в наш диалог:
|
procedure TfrmToDoManager.FormShow(Sender:
TObject); |
Автор просит обратить внимание на сохранение указателя на INTATODOItem в свойстве TListItem.Data. Это нужно для того, чтобы можно было вызвать методы Edit и Delete интерфейса INTATODOItem для выбранной записи в списке.
Реализуем процедуры вызова редактирования и удаления, и обработчики Update для Actions:
|
procedure
TfrmToDoManager.ListView1DblClick(Sender:
TObject); procedure TfrmToDoManager.aEditUpdate(Sender:
TObject); procedure TfrmToDoManager.aEditExecute(Sender:
TObject); procedure
TfrmToDoManager.aRemoveExecute(Sender:
TObject); |
Установим эксперт. Вызовем окно из главного меню IDE "Help" -> "My TODO Manager" (Рис. 13)

Рис. 13
Выполним редактирование или удаление как показано на рис. 14.

Рис. 14
Сервис IOTATodoServices также поддерживает нотификации, но рассматривать мы их не будем, поскольку читатель уже ознакомился с концепциями нотификаций в предыдущей главе.
На этом мы закончим рассматривать создание интерактивных расширений.
* Продемонстрированный в этой главе пример находится в каталоге TodoServices в "Приложении 1".
IDE позволяет определить пользовательские клавиши вызова (ShortCuts). За эту часть ToolsAPI отвечает IOTAKeybordServices. Этот сервис содержит много полезных методов, но мы рассмотрим именно работу с ShortCuts.
Чтобы изучить работу на примере, мы параллельно рассмотрим еще один сервис - INTAServices. Он позволит нам добавить пункты меню с иконками в любое место главного меню IDE, а также работать с главным ActionList среды разработки. Мы уже создавали MenuWizard, и мы знаем об ограничениях такого метода интеграции с меню. Вы помните, что такой тип эксперта может поместить пункт меню только в меню Help IDE. Сервис INTAServices поможет решить эту проблему.
Рассмотрим некоторые свойства сервиса:
|
type |
Допустим, мы хотим добавить новый пункт меню в главное меню IDE и потом назначить на него горячую клавишу.
Автор обращает внимание читателя на то, что IDE не поддерживает стандартный механизм ShortCuts для TAction, как принято в VCL. Назначение "горячих кнопок" требует иного подхода. Но об этом позже.
Для начала создадим новый пункт меню. Для этого необходимо получить ActionList и MainMenu. Определим функцию получения INTAServices:
|
function NTAServices:
INTAServices; |
После этого в конструкторе эксперта создадим пункт меню и добавим его в главное меню.
|
constructor
TFirstKeyExpert.Create; destructor
TFirstKeyExpert.Destroy; … { показать диалог, если выбран пункт меню
} |
Установим эксперт. На рис. 15 видно, как эксперт интегрировался в главное меню среды разработки

Рис. 15
Выполним пункт меню (Рис. 16)

Рис. 16
Половина работы сделана, осталось разобраться с IOTAKeyBoardServices.
Автором в свое время был написан специальный класс для назначения "горячей клавиши" любому пункту меню. Его вы найдете в "Приложении 1" под именем KeyBindingClasses.pas. Этот класс упростит нашу задачу. Чтобы создать ShortCut и назначить его пункту меню, необходимо подключить модуль в проект и создать экземпляр класса TKeyBind. После чего следует вызвать у экземпляра метод AddKey.
Параметрами являются созданный нами пункт меню и определяемая комбинация клавиш.
* Более подробно можно узнать о KeyBind, изучив содержимое файла KeyBindingClasses.pas, который находится в "Приложении 1".
Итак, доработаем конструктор и деструктор:
|
{ TFirstKeyExpert } constructor
TFirstKeyExpert.Create; destructor
TFirstKeyExpert.Destroy; |
Листинг также демонстрирует, как добавить Action в главный ActionList IDE. По образу и подобию производится добавление картинок в ImageList среды разработки. Чтобы "подстегнуть" картинку к пункту меню, достаточно указать для Action индекс картинки в главном ImageList’е IDE.
Теперь после нажатия комбинации клавиш Shift+Ctrl+K на клавиатуре, появится MessageBox эксперта, изображенный на Рис. 16.
* Рассматриваемый в этой главе пример находится в каталоге KeyboardServices в "Приложении 1".
Пришло время рассмотреть один из интереснейших сервисов IDE – IOTAModuleServices. Этот сервис предоставляет методы для работы с нумератором файлов проектов. С его помощью можно получить информацию о файловой единице - модуле. Рассмотрим часть методов. Единицей, содержащей информацию о модуле, является указатель на интерфейс IOTAModule.
С помощью сервиса IOTAModuleServices мы можем получить указатель на IOTAProject проекта, который открыт в IDE, а также на IOTAProjectGroup.
Интерфейсы IOTAProject и IOTAProjectGroup являются наследуемыми от IOTAModule. Для начала рассмотрим, как получить указатель на интерфейс текущей рабочей группы проектов. Для этого нам понадобится метод QueryInterface модуля, который определит, является ли элемент IOTAModuleServices группой проектов.
Необходимо обойти все модули в нумераторе, пока не встретится интерфейс, возвращающий указатель на IOTAProjectGroup:
|
function GetCurrentProjectGroup:
IOTAProjectGroup; |
Получив этот указатель, мы всегда сможем запросить любой из проектов, находящихся в группе проектов. Как это сделать? Вот еще одна полезная функция, демонстрирующая, как получить по имени проекта из текущей группы конкретный проект:
|
function GetProjectByFileName(FileName:
string): IOTAProject; |
Свойства интерфейса IOTAModule мы не будем рассматривать в этой статье, поскольку она хорошо документирована в модуле ToolsAPI.pas
Автор опять хотел бы обратить внимание читателя на то, что указатель на интерфейс IOTAModule можно запросить только для файлов открытых в IDE. "Что это значит?" - спросите вы. Это значит, что этот интерфейс всегда можно запросить только для текущей рабочей группы и проектов в группе. На файл в составе проекта это утверждение не распространяется до тех пор, пока вы не откроете его в редакторе IDE. Поскольку IOTAModuleServices не может предоставить нам информацию обо всех файлах проекта, рассмотрим интерфейс IOTAModuleInfo.
Чтобы получить указатель на этот интерфейс необходимо сначала получить указатель IOTAProject на любой из проектов, открытых в IDE.
Рассмотрим абстрактную реализацию:
|
procedure EnumCurrentProjectFiles(Project :
IOTAProject); |
В отличие от IOTAModule, указатель на IOTAModuleInfo можно всегда получить на любую единицу группы проектов, отрытых в IDE. Фактически, эта информация отображена в Project Manager среды разработки.
Интерфейс IOATModuleInfo может вернуть полное имя файла, тип модуля, имя формы, дизайн-класс формы. Также интерфейс содержит метод для открытия файла в редакторе IDE, что тоже важно.
Borland IDE позволяет программно управлять всеми настройками проекта, которые доступны из среды разработки. Мы уже научились получать указатель на IOTAProject проекта, открытого в среде. Из него можно получить указатель на интерфейс IOTAProjectOptions, который позволит нам программно настроить параметры проекта.
Рассмотрим декларацию интерфейса:
|
type TOTAOptionNameArray = array of TOTAOptionName; IOTAOptions =
interface(IUnknown) |
Следующая функция возвращает выходную директорию компилятора для текущего проекта:
|
function GetOutCompilerPath:
string; |
Любой интерфейс, порожденный от IOTAOptions, может выполнять подобные операции.
В этом разделе мы рассмотрим нотификаторы для IOTAModule. Пример, предлагаемый вашему вниманию, очень похож на пример, рассмотренный в тринадцатом разделе. Именно он и взят за основу упражнения.
Реализуем эксперт, который будет отлеживать операции над файлами, открытыми в среде разработки, и сохранять результаты работы в MessageWindow IDE.
Интерфейс IOTAModule имеет два метода регистрации нотификаторов: AddNotifier и RemoveNotifier соответственно. Использование нотификатора для модуля идентично нотификатору IDE. Рассмотрим IOTAModuleNotifier и реализуем его в нашем эксперте:
|
type |
Рассмотрим, какие из методов нам понадобится реализовать:
Заставим нотификатор модуля узнать, к какому именно файлу он относится. Введем переменную класса FfileName и переопределим конструктор нотификатора для того, чтобы во время посылки сообщения в MessageWindow мы могли точно знать, над которым из файлов происходит операция:
|
… |
Далее реализуем методы нотификатора для вывода информации в окно сообщений среды разработки:
|
{ TModuleNotifier } procedure
TModuleNotifier.AfterSave; procedure
TModuleNotifier.BeforeSave; function TModuleNotifier.CheckOverwrite:
Boolean; constructor TModuleNotifier.Create(FileName:
string); destructor
TModuleNotifier.Destroy; procedure
TModuleNotifier.Destroyed; end; procedure
TModuleNotifier.Modified; procedure
TModuleNotifier.ModuleRenamed(const NewName:
string); |
Для дальнейшей работы нам понадобится изменить реализацию метода FileNotification нотификатора IDE. В нем будут использованы только значение NotifyCode = ofnFileOpened. Все остальные нотификации IDE будут проигнорированы.
В этом методе мы будем получать указатель на IOTAModule по имени файла, и присоединять к нему нотификатор. Рассмотрим изменения:
|
… |
Автор хочет обратить внимание читателя на то, что при разрушении модуля (закрытии) нотификатор разрушится сам, поскольку количество ссылок, указывающих на него, будет рано нулю.

Рис. 17
Результат работы нотификатора модуля изображен на рис. 17
* Пример рассмотренный в данном разделе находится в каталоге ModuleNotifiers в "Приложении 1".
В ToolsAPI есть возможность получать указатели на интерфейсы редактора кода, редактора формы и управлять ими. Для этого в ToolsAPI.pas разработчиками Borland были любезно помещены объявления следующих интерфейсов:
|
IOTAEditor =
interface(IUnknown) |
Получить интерфейс редактора можно через интерфейс IOTAModule, который мы рассматривали выше. Для этого необходимо использовать метод GetModuleFileEditor. Поскольку метод возвращает указатель на базовый интерфейс IOTAEditor, нам необходимо применить функцию Supports для определения, какой из интерфейсов нам нужен.
Рассмотрим две функции, которые демонстрируют, как получить из интерфейса IOTAModule указатели на интерфейсы редакторов исходного кода IOTASourceEditor и редактора формы IOTAFormEditor.
|
function GetSourceEditor(Module: IOTAModule):
IOTASourceEditor; function GetFormEditor(Module: IOTAModule):
IOTAFormEditor; |
Следующая функция демонстрирует, как можно получить интерфейс на IOTAEditBuffer, который позволяет управлять редактором исходного кода:
|
function GetEditBuffer(Module: IOTAModule;
var Buffer: IOTAEditBuffer):
boolean; |
Интерфейсы редакторов также имеют нотификаторы, для получения событий от них. В этом разделе мы не будем заострять на них свое внимание, поскольку работа с нотификаторами рассматривалась выше.
Интерфейс IOTAEditView предназначен для получения относительных координат курсора в редакторе кода, получения размеров окна редактора кода и так далее.
Получить указатель на этот интерфейс очень просто, достаточно обратиться к IOTAEditorServices и запросить у него указатель на IOTAEditView с помощью свойства TopView. Например:
|
function OTAEditorServices:
IOTAEditorServices; function GetTopView:
IOTAEditView; |
Более подробно с методами и свойствами этого интерфейса можно ознакомиться в файле ToolsAPI.pas
Вспомним, как мы создаем новое приложение. "File"-> "New" -> "Other". В появившемся диалоге видно, какие типы приложений мы можем создать.
ToolsAPI позволяет создать шаблоны для новых типов приложений. Для этого существует интерфейс IOTARepositoryWizard. Он порожден от интерфейса IOTAWizard, поэтому за основу мы возьмем наш самый первый пример.
Рассмотрим декларацию интерфейса:
|
TFirstRepositoryExpert =
class(TNotifierObject, IOTAWizard,
IOTARepositoryWizard,
|
Реализуем методы IOTARepositoryWizard:
|
function TFirstRepositoryExpert.GetAuthor:
string; function TFirstRepositoryExpert.GetComment:
string; function TFirstRepositoryExpert.GetGlyph:
Cardinal; function TFirstRepositoryExpert.GetPage:
string; |
Уставим эксперта в систему и вызовем диалог выбора из репозитария:

Рис. 18
Эксперт зарегистрировался и выполнил возложенные на него функции. Теперь все зависит от того, как вы реализуете метод IOTAWizard.Execute. Чтобы определить закладку, используйте метод GetPage.
* Пример рассмотренный в данном разделе находится в каталоге Repository в "Приложении 1".
Иногда возникает желание реализовать, казалось бы, нереализуемое. Например, добавить кнопочку куда-нибудь туда, где её добавить средствами ToolsAPI не представляется возможным. Для этого некоторые программисты идут на "крайние меры". В "простонародье" это называется HACKING.
Чтобы использовать этот метод, человеку знакомому с RTTI достаточно внедриться в процесс приложения.
Мы рассмотрим основные методы, которыми пользуются программисты. Начнем с того, что наши расширения функционируют в одном и том же адресном пространстве, что и IDE. То есть, экземпляр TApplication для нас общий! Это же замечательно!
Но автор просит читателя быть осторожным с этой методикой, поскольку последствия могут быть самые непредсказуемые.
Допустим, читатель желает получить экземпляр TPopupMenu редактора исходного кода IDE, для того чтобы добавить туда полезную функцию. Увы, ToolsAPI не предоставляет нам такой возможности. Рассмотрим пару функций, которые помогут получить экземпляр меню, если окно редактора активно:
|
function GetEditWindow:
TCustomForm; function GetEditWindowPopup:
TPopupMenu; |
Функция GetEditWindow методом перебора форм у экземпляра Tapplication ищет окно редактора исходного текста по имени. А функция GetEditWindowPopup пытается обнаружить экземпляр выпадающего меню редактора.
Как видно из примера поиск происходит по имени и типу компонента. "А как узнать имя и тип искомого элемента управления?", - спросите вы. Для этого существует два способа.
Изложенная автором методика, только теоретически демонстрирует основные приемы. Читатель, руководствуясь этими принципами, может пойти намного дальше.
В среде возникают Aссess Violation или I/O ошибки при задании имени файла без полного пути при использовании IOTAProjectCreator.GetFileName.
* Список взят с сайта http://www.gexperts.org и составлен Эриком Бэрри (Erik Berry).
К документу приложен архив ToolsAPIExamples.zip (37.5К; можно скачать с сайта Devrace), в который вошли все примеры, рассмотренные в статье.
Copyright© 2003 Андрей Семак
Статья стала победителем конкурса статей по технологиям Borland на сайте ITWare и получила приз Delphi 8 Professional Edition.