script>
"Knowledge itself is power" f.baconкарта сайта | помощщ | о проекте | ттх
Подземелье Магов Фильтр по датам Поиск Поиск по КС Поиск Яndex © Поиск книг Тематический каталог Все манускрипты Карта VCL ОШИБКИ Сообщения системы Форумы Круглый стол Вопросы за сегодня Базарная площадь Городская площадь Летопись Королевские Хроники Рыцарский Зал Глас народа! ТТХ КОНКУРСЫ Королевская клюква Разделы Hello, World! Лицей Сокровищница Подземелье МаговПодводные камни Свитки Арсенальная башня Фолианты Полигон Книга Песка Дальние земли АРХИВЫ Книги на сайте Книжная полка Библиотека
Создание приложений в Delphi (с CD-ROM)
Сейчас на сайте присутствуют: 19:55 Archi Shams19: 55 Werewolf19: 53 Snugforce19: 47 дмитрий мокрецов19: 38 алексей дорожкин19: 35 Chaos
[Войти] | [зарегистрироваться]
Borland Delphi / C Builder Toolsapi, или взгляд на borland IDE изнутри андрей семакдата публикации26-01-2004 11:59
Версия для печати
Введение Рано или поздно каждому профессиональному разработчику программного обеспечения становится тесной среда обитания. По мнению автора, читатель понял, что речь идет о среде разработки, в которой обитает разработчик. У кого это "дом", у кого "строительная площадка", а у кого и то, и другое. в этой статье мы рассмотрим возможность расширения нашего дома. Мы покажем, как именно можно наращивать мощность IDE своими силами, будь то среда разработки Borland Delphi или Borland C Builder. У многих из вас возникало желание добавить несколько полезных функций в любимое IDE. Разработчики фирмы Borland не оставили без внимания эту полезную возможность. Они разработали набор интерфейсов, позволяющих расширять возможности среды, хотя, к сожалению, отсутствие документации по данному вопросу мешает самостоятельному изучению. Жалкие комментарии в исходных текстах не могут воспроизвести всю мощь, которую предоставляет IDE разработчику расширений. автор статьи попытается передать читателю опыт с с воих изысканий и постарается своими советами уберечь читателя от многих подводных камней. Документ построен таким образом, чтобы мы переходили от этапа к этапу от простого к сложному. Насколько это удалось, судить вам. Вместе c вами мы рассмотрим на примерах все сервисы, предоставляемые IDE Borland Delphi / C Builder. Итак, начнем?
? Что такое ToolsAPI Многие спросят: "Что такое ToolsAPI?" На что смело можно ответить: ToolsAPI - это набор программных интерфейсов, позволяющих: получать информацию о происходящих в IDE событиях, вызывать функции IDE, создавать свои собственные функции, которые потом будут доступны в среде разработки. Этот набор позволяет нам создавать свои собственные меню, кнопки управления и так далее. Этот набор интерфейсов размещен в файле ToolsAPI.pas, находящийся в директории {IDEROOT} / Source / Toolsapi / вашего каталога Delphi / C Builder.Старый " стиль "и" новый стиль "ToolsAPI Существует две модели интерфейсов ToolsAPI, или как их называют -". старый "и" новый "стили старый стиль был реализован в средах Delphi 3-4 и C Builder 3.-4 Начиная с Delphi 5. и C Builder 5, в среду встроена реализация нового стиля В этом документе мы будем рассматривать только "новый стиль" "Почему?" -.. возможно, спросите вы А потому, -. ответит автор - что фирма Borland более не развивает интерфейсы "старого стиля . Это не значит, что расширения, написанные в старом стиле, не будут работать Разработчики Borland оставили API старого стиля для совместимости с предыдущими версиями IDE Интерфейсы "старого" ToolsAPI находятся в директории Source / Toolsapi / в следующих файлах..:
editintf.pas exptintf.pas fileintf.pas istreams.pas toolintf.pas vcsintf.pas Но они нам не понадобятся. Вы спросите, как их различать? Интерфейсы нового стиля имеют префикс "IOTA" и "INTA", а "старого" содержат префикс "Ti". Поэтому, их можно различить с первого взгляда. Запомним это.
Примечание:
Если читатель будет интересоваться "старым" стилем, автор рекомендует обратиться в Интернете к странице Сергея Орлика (в данный момент Сергей работает в российском представительстве Borland). К сожалению, с октября 1999 года страница автором более не поддерживается. Также на его сайте есть ссылка на PDF документ, демонстрирующий архитектуру ToolsAPI "нового" стиля, но для Delphi / C Builder 4 (http://www.geocities.com/SiliconValley/Way/9006/otapi.pdf)Особенности ToolsAPI в Delphi / C Builder различных версий на первый взгляд IDE Delphi и C Builder идентичны. Автор возьмет на себя смелость заявить, что и внутренних отличий так же практически нет. Во всяком случае, в разрезе ToolAPI. То есть, ToolsAPI Delphi5 тождественен ТoolsAPI С Builder5 и так далее . Расширения, собранные на Delphi без проблем работают в C Builder аналогичной версии. Согласитесь, это очень удобно. Создав полезную функцию для IDE Delphi, вы всегда сможете использовать её в С Builder той же версии. Многие спрашива ют: "? А как же столь популярный Borland Jbuilder" Jbuilder - это продукт полностью написанный на Java Он также имеет ToolsAPI, но кардинально отличается от рассматриваемого нами..
Примечание:
В вээо документе Borland JBuilder рассматриваться не удет.
Типы экспертов Немного терминологии. "Экспертом" мы называем набор функций, расширяющий возможности IDE и выполненный в виде загружаемого модуля. В "новом" стиле нет понятия "Expert" (пережиток "старого стиля"), но есть понятие "Wizard". На самом деле это одно и тоже Автор взял за основу слово "Эксперт" потому что, по его мнению, оно более точно характеризует предмет нашего с вами внимания.Существуют два типа исполнения экспертов:. В виде пакетов (расширение файла .BPL) В виде библиотек ( расширение файла .DLL) В этом документе мы рассмотрим оба типа, особенности их создания и регистрации. У каждого из этих двух типов есть преимущества и недостатки.
Пакеты Эксперты, выполненные в виде пакетов (BPL), регистрируются как обычные пакеты с компонентами прямо из среды разработки Для этого достаточно зайти в меню "Component" -.> "Install Packages" -> "Add" и выбрать BPL с экспертом.
рис. 1
Чтобы временно отключить эксперта, достаточно снять галочку напротив пакета в списке этого окна. Для того, чтобы удалить его из системы, следует выбрать пакет в списке и нажать кнопку "Remove".
Библиотеки У этого типа экспертов более сложная регистрация. Для регистрации нам необходимо запустить редактор реестра вашей операционной системы (regedit.exe) и открыть ключ HKEY_CURRENT_USER / Software / Borland / Delphi (или C Builder) / XX / Experts, где XX версия Delphi /C builder.пример показан на рис. 2.
рис. 2
Имя значения может иметь произвольное название, значение должно иметь тип REG_SZ (String) и содержать путь к библиотеке с экспертом. Добавляя или удаляя значения в этой ветке, мы подключаем или отключаем расширения.
Примечание:
В дальнейшем мы рассмотрим, как можно создать саморегистрирующийся эксперт, выполненный в виде библиотеки.Автор для удобства рекомендует использовать ExpertManager из набора Gexperts.Особенности отладки экспертов Отладка экспертов осуществляется так же, как и отладка 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.
Что такое интерфейсы и особенности их использования в ToolsAPI Автор рассчитывает на то, что читатель уже знаком с использованием COM-интерфейсов. В противном случае, перед прочтением статьи необходимо ознакомится с концепциями и архитектурой COM, поскольку эта часть в документе затронута не будет. Также рекомендуется обратить внимание читателя на то, как развиваются интерфейсы ToolsAPI от версии к версии. Если читатель решит создавать расширения, которые будут использоваться в нескольких версиях IDE, то можно рекомендовать использовать ToolsAPI от самой низкой версии используемого IDE. Это связано с тем, что развитие интерфейсов порождает правило о совместимости сверху вниз. От высшей версии к низшей. И ни в коем случае наоборот. Можно обойтись директивами компилятора, используя те или иные методы интерфейсов в высших версиях но, по мнению автора, это очень трудоемко и может привести к путанице.Основные сервисы Toolsapi итак, время заглянуть в файл Toolsapi.Pas. Этот файл содержит цел ый ряд сервисов, которые помогут получать информацию из IDE Знакомьтесь -.. IBorlandIDEServices Это самый "главный" интерфейс Из него мы получим все необходимые нам интерфейсы-сервисы Все интерфейсы, содержащие в себе слово "Services" являются производными от IBorlandIDEServices.Указатель на.. IborlandideServices можно получить двумя способами.
Через экспортную функцию регистрации эксперта путем присвоения указателя глобальной переменной BorlandIDEServices. (В случае с библиотекой) Через уже определенную глобальную переменную BorlandIDEServices (используя пакеты) Методы получения указателя на IBorlandIDEServices мы рассмотрим на примере в разделе 9. Например, нам необходимо получить указатель на INTAServices. Для этого достаточно написать следующую функцию: Function Ntaservices: intaservices
Begin
Result: = (BorlandIdeServices as intaservices);
END;
Аналогичные функции можно создать для всех остальных сервисов. Мы рассмотрим кажды сервис отдельно по мерериала.
Первый эксперт Первым продемонстрируем пример создания эксперта в пакете Для этого нам необходимо загрузить среду разработки Borland Delphi и создать новый пакет (New Package) Добавим в него новый модуль (Unit) В секции interface необходимо включить объявления ToolsAPI -.... Toolsapi.pas Теперь пришло время рассмотреть интерфейс IOTAWizard. Этот интерфейс и есть базовый для класса эксперта. Все его методы нуждаются в реализации, даже если не будут использоваться. Все объекты, которые будут взаимодействовать с ToolsAPI, порождаются от TNotifierObject. Его реализацию можно посмотреть в том же ToolsAPI. Pas. итак, декларируем расширение:
TYPE TFIRSTEXPERT = Class (TnotifierObject, Iotamenuwizard,
Iotawizard
public
Function GetIDSTRING: STRING;
Function GetName: String;
Function GetState: twizardstate;
Procedure execute;
END;
Теперь рассмотрим все методы по очереди.
. GetIDString Этот метод должен возвращать уникальную строку для идентификации эксперта внутри IDE Например:.. "MY.FIRST.EXPERT" Если IDE обнаружит два расширения с одинаковыми идентификаторами, то среда разработки выдаст сообщение об ошибке, и загрузится только первый эксперт Об этом нужно. .. помнить GetName Должен вернуть имя эксперта Например:.. "MyFirstExpert" GetState Имеет два состояния:..... wsEnabled, wsChecked В нашем примере метод будет возвращать wsEnabled Execute Метод выполнится, когда эксперт запустится В данном случае будет выводить MessageBox Итак, реализация: Implementation {tfirstexpert}
Procedure tfirstexpert.execute;
Begin
ShowMessage (Format ('% s запущен', [getname]));
END;
Function Tfirstexpert.getIDString: String;
Begin
Result: = 'my.first.expert';
END;
Function tfirstexpert.getname: string;
Begin
Result: = 'myfirstexpert';
END;
Function tfirstExpert.getState: twizardstate;
Begin
Result: = [wsenabled];
END;
Костяк готов, но эксперт требует регистрации. Поскольку рассматривается расширение в пакете, то будем использовать процедуру регистрации, принятую для пакетов. Объявим её в секции interface и реализуем в implementation.
...
PROCEDURE register;
IMPLEMENTATION
PROCEDURE register;
Begin
Register packagewizard (tfirstexpert.create);
END;
{Tfirstexpert}
...
Готовый код можно использовать как шаблон для создания экспертов на основе пакетов. Зарегистрируем его, как описано в разделе 5. Если все сделано правильно, и никаких ошибок регистрации не произошло, то эксперт установится в IDE. Но, увы, наш первенец пока не наделен никакими еункциями. его надо заставить выполнить метод EXECUTE. "как?" - спросите вы. чтоб ответить на этот вопрос автор поведет читателя дальше.примелание:
Рассматриваемый в этой главе пример находится в каталоге firstexpert / package в "приложении 1".
Методы интеграции со средой или создаем MenuWizard Ну что ж, пришло время рассмотреть MenuWizard Из файла ToolsAPI.pas видно, что интерфейс IOTAMenuWizard является наследником IOTAWizard В него добавлен новый метод GetMenuText Добавим этот интерфейс в объявление нашего эксперта...:
TYPE TFIRSTEXPERT = Class (TnotifierObject, Iotawizard,
Iotamenuwizard
public
Function GetIDSTRING: STRING;
Function GetName: String;
Function GetState: twizardstate;
Procedure execute;
Function GetMenutext: string;
END;
И реализуем метод, который добавит пункт деню в подменю "Help" IDE. Метод должен вернуть текст для свойства Caption добавляемого пункта:
Function Tfirstexpert.getMenutext: string;
Begin
Result: = 'Execute myfirstexpert';
END;
Соберем проект. Автор хотел бы обратить ваше внимание на то, что после компиляции нам не надо перерегистрировать расширение. Это произойдет автоматически. Теперь откроем главное меню IDE "Help". На рис. 5 видно, что в меню добавлен новый пункт.
рис. 5
Выполним его. Диалог Messagebox в методе Execute не заставил себя ждать. (Рис. 6)
рис. 6
Это простейший способ создать расширение, которое интегрируется в меню IDE. Эксперта на базе IOTAMenuWizard нельзя заставить подключиться к другим пунктам главного меню. Но есть другие способы, которые мы рассмотрим далее в этой статье. Пришла пора ознакомиться с методом и особенностями реализации экспертов в библиотеках. . Для начала создадим проект библиотеки (DLL) Для успешной регистрации эксперта в IDE необходимо реализовать две функции Это (см ToolsAPi.pas.):. InitWizard DoneWizard Прошу обратить внимание читателя на то, что данный тип регистрации не подразумевает динамическое отключение (выгрузку) расширения из среды, как это было с пакетами Эксперт регистрируется во время загрузки среды разработки и существует на протяжении всего времени, пока IDE загружено.Итак, рассмотрим более детально функцию регистрации: function InitWizard (const BorlandIDEServices:. IBorlandIDEServices;
RegisterProc: TwizardRegisterProc;
Var Terminate: twizardterminateproc): boolean; stdcall;
... BorlandIDEServices Константа с указателем IBorlandIDEServices RegisterProc Указатель на процедуру регистрации (Не будем использовать) Указатель на процедуру, которая выполнится по завершении работы расширения Реализуем эти функции в DPR-файле проекта библиотеки...:
Library fisrtexpertdll;
Uses Sysutils, Classes, Toolsapi, Forms,
Figstexpclassdll in 'firstxpclassdll.pas'; {$ r * .res}
Var fEXPERTINDEX: Integer;
Const InvalidIndex: integer = -1;
Procedure donewizard;
Var WizardServices: IotawizardServices;
Begin {если регистрация была удачной то ...}
IF fEXPERTINDEX <> invalidindex then
Begin
{Получаем указатель на IotawizardServices
для разрегистрации расширения}
WizardServices: = BorlandideServices as IotawizardServices; {даление расширения из среды}
WizardServices.Removewizard (fEXPERTINDEX);
{Сброс индекса расширения}
FEXPERTINDEX: = INVALIDINDEX;
END;
END;
Function INITWIZARD (Const BorlandIDeServices): IborlandIDeServices;
RegisterProc: TwizardRegisterProc;
Var Terminate: twizardterminateproc): boolean; stdcall;
Var WizardServices: IotawizardServices;
Begin
{Проверяем определена ли глобальная переменная BorlandideServices.
Если нет, присваиваем ей значение}
If Toolsapi.borlandideServices = NIL THEN
Toolsapi.borlandideServices: = BorlandIDeServices;
{Олучаем Handle приложения и ирисваиваем его себе,
для нормализации работы окон внутри расширения в контексте IDE}
Application.handle: = (BorlandIDeServices
as Iotaservices) .getparenthandle;
{Пределяем указатель на процедуру завершения}
Terminate: = DONEWIZARD;
{Получаем указатель на IotawizardServices
для регистраци расширения}
WizardServices: = BorlandideServices as IotawizardServices;
{Регистрируем расширение в IDE}
FEXPERTINDEX: = WizardServices.addwizard (tfirstexpertdll.create);
{Ернем истину, если регистрация удалась}
Result: = (FEXPERTINDEX <> InvalidIndex);
END;
{Экспорт точки входа для IDE}
Exports
INITWIZARD NAME WIZARDENTRYPOINT;
Begin
End.
Для регистрации такого расширения нам понадобится указатель на IOTAWizardServices. Регистрация и разрегистрация выполняется с помощью именно этого интерфейса. Это методы AddWizard и RemoveWizard соответственно. Метод AddWizard возвращает индекс расширения в системе, в случае удачной регистрации, или значение "-1" в случае неудачи. Метод RemoveWizard удаляет расширение из IDE, используя индекс, полученный при регистрации. Как это сделать, подробно с комментариями видно из листинга. Сохраним модуль с реализацией TfirstExpert под именем FirstExpClassDLL.pas в каталог с проектом библиотеки и удалим из него процедуру регистрации для пакетов. Расширение в DLL реализовано. Осталось подключить его в среду. Как это сделать мы рассматривали в пятом разделе данной статьи. Для упрощения установки библиотеки, мы можем воспользоваться функциями регистрации COM-объектов. Читателю известно, что регистрация COM-объектов в DLL осуществляется утилитой RegSvr32.exe , которая поставляется с юююой операцион . Ной системой Windows Для этого необходимо включить в исходный код проекта 2 экспортируемые функции: DllRegisterServer DllUnRegisterServer Эти функции описаны в модуле ComServ, который так же необходимо включить в uses секцию проекта Но для полноценной установки необходимо, чтобы библиотека определила свое местонахождение Воспользуемся для этого.. сервисной функцией, которая возвращает полный путь к киблиотеке.
Function GetdllFileName: String;
Var filename: array [0..max_path] of char;
Begin
GetModuleFileName (Hinstance, FileName, Max_Path - 1);
Result: = StrPas (FileName);
END;
Теперь очередь за реализацией установки:
Const
ConstructionKey = HKEY_CURRENT_USER;
ConstregKey = 'Software / Borland / Delphi / 7.0 / Experts'; constvaluename = 'myfirstexp';
Function DLLREGISTERSERVER: HRESULT; STDCALL;
Begin
With tregistry.create do
Try
Result: = E_FAIL;
RootKey: = ConstructionKey;
IF keyexists (constregKey) THEN
Begin
OpenKey (constregKey, true);
WriteString (constvaluename);
Closekey;
Result: = S_OK;
END;
Finally
FREE;
END;
END;
Function Dllunregister: hResult; stdcall;
Begin
With tregistry.create do
Try
Result: = E_FAIL;
RootKey: = ConstructionKey;
IF keyexists (constregKey) THEN
Begin
OpenKey (constregKey, true);
IF valueexists (constvaluename) THEN
Begin
DeleteValue (constvaluename);
Closekey;
Result: = S_OK;
END;
END;
Finally
FREE;
END;
END;
Exports
DllRegisterServer,
DllunregisterServer;
Из листинга видно, как происходит установка и удаление информации из системного реестра. Теперь необходимо собрать проект и установить расширение.Как это сделать, изображено на рис. 7
рис. 7
Результатом взполнения установки удет окно, изображенное на рис. 8
рис. 8
Удалить расширение из Registry ожно той те ттик с клююем - ожно проверить, прописан ли ходимый клюю в реестр. См. Рис. 9.
рис. 10
Запустим новый экземпляр IDE и убедимся, что расширение установилось. Правда, просто? Читатель сам должен решить, какой тип эксперта необходимо использовать в каждом конкретном случае.
Примечание:
Рассматриваемый в этой главе пример находится в каталоге firstexpert / library в "приложении 1."
. Работаем с сервисами Окно сообщений IDE Теперь рассмотрим один из самых простых и полезных сервисов IDE Автор расскажет, как управлять окном сообщений Работа с ним осуществляется с помощью сервиса IOTAMessageServices Обратимся в ToolAPI.pas за помощью Рассмотрим несколько методов, а именно....: {Добавить заголовочное соощщение}
Procedure AddtitleMessage (const message ";
{Добавить инструментальное соощщение}
Procedure AddToolMessage (Const Filename, MessageStr,
PrefixStr: String; LINENUMBER, ColumnNumber: Integer;
{Очистить окно соощщений}
Procedure ClearallMessages;
{Очисить соощщения компилятора / линкера}
Procedure ClearCompilerMessages;
{Очистить инструментальные соощщения от утилит или групп}
Procedure CleartoolMessages;
Для обращения к данному сервису нам будет необходима функция, возвращающая указатель на IOTAMessageServices. А поскольку этот интерфейс является сервисным, то получить его можно из глобальной переменной BorlandIDEServices.
Function Otamessageservices: IotamessageServices;
Begin
Result: = (BorlandIdeServices as Iotamessageservices);
END;
Включим эту функцию в модуль с экспертом в секцию implementation Осталось только изменить реализацию Execute Заменим MessageBox на вывод в окно сообщений IDE..:
Procedure tfirstexpert.execute;
Begin
OtamessageServices.addtitleMessage (Format ('% s запущен ",
[Getname]);
END;
Теперь установим и выполним расширение. Результат работы изображен на рис. 10
рис. 10
Примечание:
Рассматриваемый в этой главе пример находится в каталоге Messageservices в "приложении 1".
Понятие нотификаций Что такое интерфейс нотификаций? Нотификатором называют событийный интерфейс, позволяющий получать те или иные события от IDE. Существует несколько типов нотификаторов, но необходимо отметить, все они порождены от одного базового интерфейса IOTANotifier. Такая архитектура упрощает реализацию и понимание организации перехвата событий от среды разработки. в этой статье мы не будем рассматривать все нотификаторы, которые доступны в ToolsAPI, но обязательно остановимся на ключевых моментах. Итак, снова обратимся к ToolsAPI.pas. Какие методы имеет базовый интерфейс нотификатора IOTANotifier? type IOTANotifier = interface (IUnknown)
Procedure aftersave;
Procedure BeforeSave;
Procedure destroyed;
PROCEDURE MODIFIED;
END;
На методах BeforeSave, AfterSave и Modified мы остановимся позже, в разделе 18. Метод Destroyed вызывается перед разрушением нотификатора. Поскольку ToolsAPI основано на интерфейсной модели, то IDE сама следит за "нужностью" того или иного объекта. Если читатель заметил, то TnotifierObject является наследником TinterfacedObject и ему ни в коем случае нельзя вызывать метод Free принудительно из кода программы. Как только количество ссылок на созданный нами объект будет равно нулю, экземпляр объекта разрушится сам. Для более подробной информации необходимо обратиться к источникам, связанным с использованием COM-интерфейсов в разделы , посвященные подсчету количества ссылок на объекты. Нам позволено только создавать, подключать свои экземпляры к IDE и отключать их. После отключения, если экземпляр более никем не используется, он разрушится сам. Это важно, не будем забывать об этом.
Нотификации IDE Попробуем теорию на практике Для этого нам понадобится получить указатель на IOTAServices и создать нотификатор от IOTAIDENotifier Напишем функцию, возвращающую указатель на интерфейс необходимого нам сервиса: function OTAServices: IOTAServices;..
Begin
Result: = (BorlandIdeServices as Iotaservices);
END;
Пришло время оъъъвить класс нотификатора IDE:
Type
TIDENOTIER = Class (TnotifierObject, Iotanotifier,
IotaideNotifier
{From Iotanotifier}
Procedure aftersave;
Procedure BeforeSave;
Procedure destroyed;
PROCEDURE MODIFIED;
{From Iotaidenotifier}
PROCEDURE Filenotification (NotifyCode: Totafilenotification) (NOTAFILENOTIFIC)
Const filename: String;
Var Cancel: Boolean;
Procedure BeforeCompile (Const Project: IoTaproject;
VAR Cancel: Boolean; Overload;
Procedure afterCompile (succeeded); overload;
END;
Нас интересует метод FileNotification, который позволит получать сообщения об операциях с файлами в IDE Именно его мы и будем использовать в рассматриваемом примере Из ToolsAPI.pas видно, что параметр NotifyCode может принимать следующие значения..:
Значение параметра ОписаниеOfnFileOpening Перед открытием файла FileNameOfnFileOpened Файл FileName открыт в IDEOfnFileClosing Перед закрытием файла FileNameOfnDefaultDesktopLoad Загружены настройки для IDE по умолчаниюOfnDefaultDesktopSave Сохранены настройки IDE по умолчаниюOfnProjectDesktopLoad Загружены настройки для проекта FileNameOfnProjectDesktopSave Сохранены настройки для проекта FileNameOfnPackageInstalled Пакет FileName с компонентами или экспертами установлен в IDEOfnPackageUninstalled Пакет FileName с компонентами или экспертами удален из IDEOfnActiveProjectChanged Вызывается только при работе с группой проектов, если количество проектов более одного Важно:. данный тип отсутствует в IDE версии ниже 6. Параметр Cancel можно использовать для прерывания операции, которая породила нотификацию, если его установить в True В нашем. случае оставим его ез внимания реализуем метод FileNotification, который будет обрабатывать приходщщие события из Ide и сооощщать о них в окне Messa Gewindow Ide: Procedure Tidenotifier.FileNotification
NOTIFYCODE: TOTAFILENOTIFICATION;
Const filename: string; var calation: boolean;
Var stringType: string;
Begin
{Преобразовываем код нотификации в текстовое соощщение}
Case NotifyCode of
OFNFILEOPENING: STRINGTYPE: = 'OFNFILEOPENING';
OFNFILEOPENED: STRINGTYPE: = 'OFNFILEOPENED'
OFNFILECLOSING: STRINGTYPE: = 'OFNFILECLOSING'
FAULTDESKTOPLOAD: STRINGTYPE: = 'ofNDEFAULTDESKTOPLOAD'
FAULTDESKTOPSAVE: STRINGTYPE: = 'ofNDEFAULTDESKTOPSAVE'
OfnProjectDesktopload: StringType: = 'OFNPROJECTDESKTOPLOAD'
OfnProjectDesktopsave: StringType: = 'OFNPROJECTDESKTOPSAVE'
OFNPACKEINSTALLED: StringType: = 'OFNPACKAGEINSTALLED'; OFNPACKAGEUNSTALED: STRINGTYPE: = 'OFNPACKAGEUNSTALLED'
OFNACTIVEPROJECTCHANGED: STRINGTYPE: = 'OFNACTIVEPROJECTCHANGED'
END;
{Соощщаем о оришедшей нотификаци в окно MessageWindow IDE}
OtamessageServices.addtitleMessage (Format ('% s% s ",
[StringType, FileName]));
END;
Осталось рассмотреть добавление и удаление нотификатора Сервис IOTAServices для этого имеет соответствующие методы AddNotifier и RemoveNotifier Чтобы мы могли успешно удалить нотификатор из IDE, необходимо ввести для него значение индекса Введем индекс как переменную класса FNotifierIndex:... Integer в секции Private эксперта Мы будем осуществлять. добавление нотификатора при создании сксперта и его удаление в процессе разрушения. для отого введем оъъъввеиие конструктора и деструктора:
Type
TFIRSTEXPERT = Class (TnotifierObject, Iotawizard,
Iotamenuwizard
Private
FNotifierIndex: Integer;
public
Constructor crete;
DESTRUCTOR DESTROY; OVERRIDE;
Function GetIDSTRING: STRING;
Function GetName: String;
Function GetState: twizardstate;
Procedure execute;
Function GetMenutext: string;
END;
Добавление и удаление нотификатора имет следующую реализацию:
{Tfirstexpert}
Constructor tfirstexpert.create;
Begin
FNotifierIndex: = otaservices.addnotifier (TIDENOTIFIER.CREATE);
END;
DESTRUCTOR TFIRSTEXPERT.DESTROY;
Begin
IF FNotifierIndex <> -1 THEN
Otaservices.Removenotifier (FNotifierIndex);
inherited;
END;
В конструкторе создается экземпляр нотификатора объекта и его индекс сохраняется в переменной. Из деструктора видно, что если регистрация нотификатора произошла успешно, то он будет удален из IDE.Результат работы нотификатора изображен на рис. 11.рис. 11
Примечание:
Рассматриваемый в этой главе пример находится в каталоге IDServices в "приложении 1".
Управление напоминаниями ToDo Рассмотрим еще один сервис, который нам предоставляет IDE -. Сервис управления напоминаниями ToDo В ToolsAPi.pas он объявлен как IOTAToDoServices Немного расширим задачу и создадим упрощенный менеджер управления напоминаниями По сравнению с предыдущими примерами задача будет несколько сложнее, и автор постарается.. максимально разъяснить материал. Для решения этой задачи будет создан интерактивный эксперт, который позволит нам редактировать и удалять напоминания в текущем проекте.Итак, создадим функцию, которая обеспечит на доступ к IOTAToDoServices.
Function Otatodoservices: Iotatodoservices;
Begin
Result: = (BorlandIdeServices as Iotatodoservices);
END;
Создадим форму менеджера с основными дункциями и добавим ёё в проект (рис. 12).
рис. 12
В конструкторе создадим форму, веструкторе разрушим ёё и переопределим метод Execute у расширения для визуализации ормы менеджера.
Constructor tfirstexpert.create;
Begin
FRMTODOMANAGER: = tfrmtodomanager.create (application);
END;
DESTRUCTOR TFIRSTEXPERT.DESTROY;
Begin
Freeandnil (frmtodomanager);
inherited;
END;
Procedure tfirstexpert.execute;
Begin
IF assigned (frmtodomanager).
END;
Определик обработчик Onshow формы нашего менеджера. В нем мы удем сканировать весь спии записями, и переносить его в наш диалог:
Procedure tfrmtodomanager.formshow (sender: Tobject); var i: integer; item: tlistitem;
Begin
Listview1.items.beginupdate;
Try
Listview1.Items.clear;
Otatodoservices.updatelist;
For i: = 0 to otatodoservices.Itemcount - 1 DO
Begin
Item: = listview1.items.add;
Item.caption: = otatodoservices.Items [i] .gettext;
Item.SUBITEMS.ADD (Otatodoservices.Items [i] .getmoduLename);
Item.SUBITEMS.ADD (Otatodoservices.Items [i] .Getowner);
Item.Data: = Pointer (OtaTodoservices.Items [i]);
END;
Finally
ListView1.Items.Endupdate;
END;
END;
Автор просит обратить внимание на сохранение указателя на INTATODOItem в свойстве TListItem.Data Это нужно для того, чтобы можно было вызвать методы Edit и Delete интерфейса INTATODOItem для выбранной записи в списке Реализуем процедуры вызова редактирования и удаления, и обработчики Update для Actions..:
Procedure tfrmtodomanager.listview1dblclick (sender: TOBJECT);
Begin
IF aedit.enabled the aidit.execute;
END;
Procedure tfrmtodomanager.AEDITUPDATE (Sender: TOBJECT);
Begin
{TODO -OAUTHOR-CTEST: пробное напоминание 1}
AEDIT.ENABLED: = Assigned (listview1.selected);
Aremove.enabled: = AEDIT.ENABLED;
END;
Procedure tfrmtodomanager.AEDITEXECUTE (Sender: TOBJECT);
Var TodoItem: INTATODOITEM;
Begin
TodoItem: = INTATODOITEM (ListView1.Selected.data);
IF Assigned (TodoItem) THEN
Begin
TodoItem.edit;
FormShow (Self);
END;
END;
Procedure tfrmtodomanager.aremoveexecute (sender: Tobject);
Var TodoItem: INTATODOITEM;
Begin
TodoItem: = INTATODOITEM (ListView1.Selected.data);
IF Assigned (TodoItem) THEN
Begin
TodoItem.delete;
FormShow (Self);
END;
END;
Установим эксперт. Вызовем окно из главного меню IDE "Help" -> "My Todo Manager" рис. 13
Выполним редактирование или удаление как показано на рис. 14.
рис. 14
Сервис IOTATodoServices также поддерживает нотификации, но рассматривать мы их не будем, поскольку читатель уже ознакомился с концепциями нотификаций в предыдущей главе.На этом мы закончим рассматривать создание интерактивных расширений.
Примечание:
Продемонстрированный в этой главе пример находится в каталоге Todoservices в "приложении 1".
Хотим горячие клавиши IDE позволяет определить пользовательские клавиши вызова (ShortCuts) За эту часть ToolsAPI отвечает IOTAKeybordServices Этот сервис содержит много полезных методов, но мы рассмотрим именно работу с ShortCuts Чтобы изучить работу на примере, мы параллельно рассмотрим еще один сервис -!... INTAServices . Он позволит нам добавить пункты меню с иконками в любое место главного меню IDE, а также работать с главным ActionList среды разработки. мы уже создавали MenuWizard, и мы знаем об ограничениях такого метода интеграции с меню. Вы помните, что такой тип эксперта может поместить пункт меню только в меню Help IDE. сервис Intaservices поможет решить эту проблему.рассмотрим некоторые свойства сервиса:
Type
INTASERVICES = Interface (IUNKNOWN)
...
Property ActionList: tcustomactionList Read getActionList;
Property imagelist: tcustomimagelist read getimagelist;
Property Mainmenu: TmainMenu Read GetMainu;
...
END;
ActionList. Позволяет получить доступ к главному ActionList IDE ImageList. Позволяет получить доступ к списку с картинками 16x16 IDE MainMenu. Позволяет получить доступ к главному меню среды разработки. Допустим, мы хотим добавить новый пункт меню в главное меню IDE и потом назначить на него горячую клавишу . Автор обращает внимание читателя на то, что IDE не поддерживает стандартный механизм ShortCuts для TAction, как принято в VCL. Назначение "горячих кнопок" требует иного подхода. Но об этом позже. Для начала создадим новый пункт меню. Для этого необходимо получить ActionList и Опр пр у ункцию получения Intaservices: Function NTaservices: intaservices
Begin
Result: = (BorlandIdeServices as intaservices);
END;
После этого в конструкторе эксперта создадим пункт меню и добавим его в главное меню.
Constructor tfirstKeyExpert.create;
Var mainmenu: tMainMenu;
Const constcaption = 'keyexpert';
Begin
Mainmenu: = ntaservices.mainMenu;
{Если пункт меню щще не добавлен, то добавить}
IF Assigned (Mainmenu) and
Not assigned (mainmenu.Items.Find (constcaption) THEN
Begin
FMenuItem: = TMenuitem.create (Mainmenu);
Mainmenu.Items.Add (fMenuItem);
FMenuItem.caption: = consTcaption;
{Становить обработчик для пункта}
FMenuItem.onclick: = onMenuclick;
END;
END;
Destructor TfirstKeyExpert.destroy;
Begin
{Разрушить пункт меню}
IF assigned (fMenuItem) THEN
FMenuItem.free;
inherited;
END;
...
{оказать диалог, если выбран пункт меню}
Procedure TfirstKeyExpert.onMenuclick (Sender: TOBJECT);
Begin
ShowMessage ('Procedure Executed!')
END;
Установим эксперт. На рис. 15 видно, как эксперт интегрировался в главное меню среды разработки
рис. 15
Выполним пункт меню (рис. 16) рис. 16
Половина работы сделана, осталось разобраться с IOTAKeyBoardServices. Автором в свое время был написан специальный класс для назначения "горячей клавиши" любому пункту меню. Его вы найдете в "Приложении 1" под именем KeyBindingClasses.pas. Этот класс упростит нашу задачу. Чтобы создать ShortCut и назначить его пункту меню, необходимо подключить модуль в проект и создать экземпляр класса TKeyBind. После чего следует вызвать у экземпляра метод addKey. Параметрами являются созданный нами пункт меню и определяемая комбинация клавиш.
Примечание:
Более подробно можно узнать о Keybind, изучив содержимое файла KeybindingClasses.Pas, который находится в "приложении 1".
Итак, доработаем конструктор и деструктор:
{TfirstKeyExpert}
Constructor tfirstKeyExpert.create;
Var mainmenu: tMainMenu;
ActionList: tcustomactionList;
Const constcaption = 'keyexpert';
Begin
Mainmenu: = ntaservices.mainMenu;
ActionList: = ntaservices.ActionList;
{Если пункт меню щще не добавлен, то добавить}
IF assigned (mainmenu) and assigned (actionlist)
And not assigned (mainmenu.Items.Find (constcaption).
Begin
Faction: = TAction.create (ActionList);
FAction.ActionList: = ActionList;
Faction.caption: = constcaption;
{Становить обработчик для пункта}
Faction.onexecute: = onMenuclick;
FMenuItem: = TMenuitem.create (Mainmenu);
FMenuitem.Action: = FACTION;
Mainmenu.Items.Add (fMenuItem);
{Оздать класс назначения "горячих кнопок"}
Fkeybind: = tKeybind.create;
{Установить на созданный пункт меню комбинацию Shift Ctrl K}
Fkeybind.addkey (FMenuitem, TextToshortcut ('Shift Ctrl K');
END;
END;
Destructor TfirstKeyExpert.destroy; Begin
Freeandnil (FMenuItem);
FreeAndnil (FACTION);
If Assigned (fkeybind).
inherited;
END;
Листинг также демонстрирует, как добавить 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;
Var Services: IotamoduleServices;
Module: Iotamodule;
I: integer;
Begin
Result: = NIL;
Services: = BorlandideServices as IotamoduleServices;
For i: = 0 to Services.Modulecount - 1 DO
Begin
Module: = Services.Modules [i];
If Module.Queryinterface (IoTaprojectGroup, Result) = S_OK THEN
Break;
END;
END;
Получив этот указатель, мы всегда сможем запросить любой из проектов, находящихся в группе проектов Как это сделать Вот еще одна полезная функция, демонстрирующая, как получить по имени проекта из текущей группы конкретный проект.?:
Function getProjectByFileName (filename: string): Iotaproject;
VAR i: integer;
Begin
For i: = 0 to getCurrentProjectGroup.ProjectCount - 1 DO
IF Compare (getcurrentprojectgroup.projects [i] .filename,
FILENAME) THEN
Begin
Result: = getcurrentProjectGroup.Projects [i];
Break;
END;
END;
Свойства интерфейса IOTAModule мы не будем рассматривать в этой статье, поскольку она хорошо документирована в модуле ToolsAPI.pas Автор опять хотел бы обратить внимание читателя на то, что указатель на интерфейс IOTAModule можно запросить только для файлов открытых в IDE. "Что это значит?" -. спросите вы это значит, что этот интерфейс всегда можно запросить только для текущей рабочей группы и проектов в группе На файл в составе проекта это утверждение не распространяется до тех пор, пока вы не откроете его в редакторе IDE Поскольку IOTAModuleServices не может предоставить.. . нам информацию обо всех файлах проекта, рассмотрим интерфейс IOTAModuleInfo Чтобы получить указатель на этот интерфейс необходимо сначала получить указатель IOTAProject на любой из проектов, открытых в IDE.Рассмотрим абстрактную реализацию:
Procedure EnumcurrentProjectFiles (Project: IoTaproject);
Var i: integer; moduleinfo: IotamoduleInfo;
Begin
For J: = 0 to Project.getModulecount - 1 dobegin
ModuleInfo: = Project.getModule (j);
ShowMessage (ModuleInfo.FileName);
END;
END;
В отличие от IOTAModule, указатель на IOTAModuleInfo можно всегда получить на любую единицу группы проектов, отрытых в IDE. Фактически, эта информация отображена в Project Manager среды разработки. Интерфейс IOATModuleInfo может вернуть полное имя файла, тип модуля, имя формы, дизайн-класс формы . Также интерфейс содержит метод для открытия файла в редакторе IDE, что тоже важно.
Управление настройками проекта Borland IDE позволяет программно управлять всеми настройками проекта, которые доступны из среды разработки. Мы уже научились получать указатель на IOTAProject проекта, открытого в среде. Из него можно получить указатель на интерфейс IOTAProjectOptions, который позволит нам программно настроить параметры проекта.Рассмотрим декларацию интерфейса:
Type
Totaoptionname =
Record
Name: String;
Kind: TTYPEKIND;
END;
TotaoptionnameArray = array of totaoptionname
IotaOptions = Interface (IUNKNOWN)
...
Procedure editoptions;
Function Getoptionnames: TotaoptionnameArray;
Property Values [Const Valuename: String]: Variant;
...
END;
Метод GetOptionNames возвращает массив имен параметров. Метод EditOptions вызывает диалог настройки опций среды разработки (В данном случае для проекта. См. Рис. 3) С помощью свойства Values можно получать и присваивать значения опциям по именам из TOTAOptionNameArray. Следующая функция возвращает выходную директорию компилятора для текущего проекта:
Function GetcompilerPath: String;
Var Project: Iotaproject;
Options: IotaprojectOptions;
Begin
Result: = Emptystr;
Project: = getcurrentproject;
IF assigned (project) thenbegin
Options: = Project.ProjectOptions;
Result: = ExcludetrailingBackslash (options.values ['Outputdir']);
END;
END;
Юююой интерфейс, порожденный от iotaoptions, может выполнять подобные операции.
Понятие нотификатора модуля В этом разделе мы рассмотрим нотификаторы для IOTAModule. Пример, предлагаемый вашему вниманию, очень похож на пример, рассмотренный в тринадцатом разделе. Именно он и взят за основу упражнения. Реализуем эксперт, который будет отлеживать операции над файлами, открытыми в среде разработки , и сохранять результаты работы в MessageWindow IDE Интерфейс IOTAModule имеет два метода регистрации нотификаторов:. AddNotifier и RemoveNotifier соответственно Использование нотификатора для модуля идентично нотификатору IDE Рассмотрим IOTAModuleNotifier и реализуем его в нашем эксперте..:
Type
TModulenotifier = Class (TnotifierObject, IotAnotifier,
Iotamodulenotifier)
{From Iotanotifier}
Procedure aftersave;
Procedure BeforeSave;
Procedure destroyed;
PROCEDURE MODIFIED;
{From Iotamodulenotifier}
Function CheckoverWrite: boolean;
Procedure ModuleRenamed (const newname: string);
END;
Рассмотрим, какие из методов нам понадобится реализовать:
AfterSave. Вызывается после сохранения содержимого модуля на диск. BeforeSave. Вызывается перед сохранением модуля на диск Destroyed. Вызывается, если модуль был закрыт. Modified. Вызовется, если модуль был изменен. ModuleRenamed. Вызывается при выполнении операции переименования. NewName содержит новое имя модуля. Заставим нотификатор модуля узнать, к какому именно файлу он относится Введем переменную класса FfileName и переопределим конструктор нотификатора для того, чтобы во время посылки сообщения в MessageWindow мы могли точно знать, над которым из файлов происходит операция:. constructor TModuleNotifier.Create (FileName: string );
Begin
Inherited Create;
FFileName: = filename;
END;
Далее реализуем методы нотификатора для вывода информации в окно соощщений среды разработки:
{TModulenotifier}
Procedure tmodulenotifier.aftersave;
Begin
OtamessageServices.addtitleMessage ('AfTesave' FFileName);
END;
Procedure tmodulenotifier.beforesave;
Begin
OtamessageServices.addtitleMessage ('BeforeSave' FFileName);
END;
Function TModulenNotifier.CheckoverWrite: Boolean;
Begin
RESULT: = true;
END;
Constructor TModulenNotifier.create (filename: string);
Begin
Inherited Create;
FFileName: = filename;
END;
DESTRUCTOR TMODULENOTIFIER.DESTROY;
Begin
Otamessageservices.AddtitleMessage ('Closed' FFileName);
inherited;
END;
Procedure tmodulenotifier.destroyed;
Begin
END;
Procedure tmodulenotifier.modified;
Begin
Otamessageservices.addtitleMessage ('Modified' FFileName);
END;
Procedure TModulenotifier.ModuleRenamed (const newname: string);
Begin
OtamesSageServices.addtitleMessage ('file' ffilename
'renamed to' newfilename; ffilename: = newFileName
END;
Для дальнейшей работы нам понадобится изменить реализацию метода FileNotification нотификатора IDE. В нем будут использованы только значение NotifyCode = ofnFileOpened. Все остальные нотификации IDE будут проигнорированы. В этом методе мы будем получать указатель на IOTAModule по имени файла, и присоединять к нему нотификатор. Рассмотрим изменения :
...
Procedure tidenotifier.filenotification
NOTIFYCODE: TOTAFILENOTIFICATION;
Const filename: string; var calation: boolean;
Var module: iotamodule;
Begin
{айл открыт в редакторе IDE?}
if notifycode = OFNFILEOPENED THEN
Begin
{щем модуль по имени}
Module: = OtamoduleServices.FindModule (filename);
{если модуль найден - присоединить нотификатор}
IF assigned (module) THEN
Module.AddNotifier (TModunotifier.create (filename);
END;
END;
...
Автор хочет обратить внимание читателя на то, что при разрушении модуля (закрытии) нотификатор разрушится сам, поскольку количество ссылок, указывающих на него, будет рано нулю.
рис. 17
Результат работы нотификатора модуля изображен на рис. 17
Примечание:
Пример рассмотренный в данном разделе находится в каталоге ModulenNotifiers в "приложении 1".
Понятие редакторов В ToolsAPI есть возможность получать указатели на интерфейсы редактора кода, редактора формы и управлять ими Для этого в ToolsAPI.pas разработчиками Borland были любезно помещены объявления следующих интерфейсов.:
IOTAEDITOR = Interface (IUNKNOWN)
...
IotasourceEditor = Interface (IotaEditor)
...
OTAFORMEDITOR = Interface (IotaEditor)
Получить интерфейс редактора можно через интерфейс IOTAModule, который мы рассматривали выше. Для этого необходимо использовать метод GetModuleFileEditor. Поскольку метод возвращает указатель на базовый интерфейс IOTAEditor, нам необходимо применить функцию Supports для определения, какой из интерфейсов нам нужен. Рассмотрим две функции, которые демонстрируют, как получить из интерфейса IOTAModule указатели на интерфейсы редакторов исходного кода IOTASourceEditor и редактора формы IOTAFormEditor.function GetSourceEditor (Module: IOTAModule): IOTASourceEditor;
VAR i: integer;
Begin
Result: = NIL;
IF not assigned (module) THEN
EXIT;
For i: = 0 to module.getmoduleFileCount - 1 DO
IF supports (Module.getModuleFileEditor (i),
IotasourceEditor, Result
Then Break;
END;
Function getFormEditor (Module: Iotamodule): IotaFormeditor;
VAR i: integer;
Begin
Result: = NIL;
IF NOT ASSIGNED (Module).
For i: = 0 to module.getmoduleFileCount - 1 DO
IF supports (Module.getModuleFileEditor (i),
IotaFormeditor, Result
Then Break;
END;
Следующая функция демонстрирует, как можно получить интерфейс на otaeditbuffer, который позволяет управлять редактором исходного кода:
Function startitbuffer (module: iotamodule)
VAR Buffer: IotaEditbuffer: Boolean;
VAR i: integer;
Begin
Result: = FALSE;
IF NOT ASSIGNED (Module).
For i: = 0 to module.getmoduleFileCount - 1 DO
IF supports (Module.getModuleFileEditor (i),
IOTAEDITBUFFER, BUFFER) THEN
Begin
RESULT: = true;
Break;
END;
END;
Интерфейсы редакторов также имеют нотификаторы, для получения событий от них. В этом разделе мы не будем заострять на них свое внимание, поскольку работа с нотификаторами рассматривалась выше.
Представления редактора Интерфейс IOTAEditView предназначен для получения относительных координат курсора в редакторе кода, получения размеров окна редактора кода и так далее Получить указатель на этот интерфейс очень просто, достаточно обратиться к IOTAEditorServices и запросить у него указатель на IOTAEditView с помощью свойства TopView Например:.. Function OtaEditorServices: IotaEditorServices;
Begin
Result: = (BorlandIdeServices as IotaEditorServices);
END;
Function GettopView: IotaEditView;
Begin
Result: = OtaEditorServices.topView;
END;
Более подробно с методами и свойствами этого интерфейса можно ознакомиться в файле Toolsapi.PAS
Создаем RepositoryWizard Вспомним, как мы создаем новое приложение "File" -.> "New" -.> "Other" В появившемся диалоге видно, какие типы приложений мы можем создать ToolsAPI позволяет создать шаблоны для новых типов приложений Для этого существует интерфейс IOTARepositoryWizard.. Он интерфейса Iotawizard, поэтому за основу мы возьмем наш самый первый пример.рассмотрим декларацию интерфейса:
TfirstRepositoryExpert = Class (TnotifierObject, Iotawizard,
IoTarePositoryWizard, IotaFormWizard
public
{From Iotawizard}
Function GetIDSTRING: STRING;
Function GetName: String;
Function GetState: twizardstate;
Procedure execute;
{From iotarepositorywizard}
Function GetAuthor: string;
Function getcomment: String;
Function getpage: String;
Function getGlyph: cardinal
END;
GetAuthor должен возвращать имя автора GetComment - комментарий GetPage - имя страницы, на которой будет расположена иконка эксперта GetGlyph - дескриптор иконки В нашем случае вернем нуль Реализуем методы IOTARepositoryWizard..:
Function tfirstRepositoryExpert.getAuthor: string; begin
Result: = 'Andrew Semack';
END;
Function tfirstRepositoryExpert.getComment: string;
Begin
Result: = 'My Test Wizard';
END;
Function tfirstRepositoryExpert.getglyph: cardinal
Begin
Result: = 0;
END;
Function TfirstRepositoryExpert.getPage: String;
Begin
Result: = 'newmypage';
END;
Уставим эксперта в систему и вызовем диалог выбора из репозитария:
рис. 18
Эксперт зарегистрировался и выполнил возложенные на него функции. Теперь все зависит от того, как вы реализуете метод IOTAWizard.Execute. Чтобы определить закладку, используйте метод GetPage.
Примечание:
Пример рассмотренный в данном разделе находится в каталоге repository в "приложении 1".
Используем "HACK" Иногда возникает желание реализовать, казалось бы, нереализуемое. Например, добавить кнопочку куда-нибудь туда, где её добавить средствами ToolsAPI не представляется возможным. Для этого некоторые программисты идут на "крайние меры". В "простонародье" это называется HACKING . Чтобы использовать этот метод, человеку знакомому с RTTI достаточно внедриться в процесс приложения. Мы рассмотрим основные методы, которыми пользуются программисты. Начнем с того, что наши расширения функционируют в одном и том же адресном пространстве, что и IDE. То есть, экземпляр TApplication для нас общий! Это же замечательно! Но автор просит читателя быть осторожным с этой методикой, поскольку последствия могут быть самые непредсказуемые. Допустим, читатель желает получить экземпляр TPopupMenu редактора исходного кода IDE, для того чтобы добавить туда полезную функцию. Увы, ToolsAPI не предоставляет нам такой возможности. рассмотрим пару функций, которые помогут получить экземпляр меню, если окно редактора активно: Function GetitWindow: tcustom;
VAR i: integer;
Begin
Result: = NIL;
For i: = 0 to Application.comPonentcount - 1 DO
If Application.components [i] .name = ('editwindow_0') THEN
Begin
Result: = tcustom form (Application.Components [i]);
Break;
END;
END;
Function getTwindowPopup: tpopupmenu;
Var EditorForm: Tcustomform;
Begin
Result: = NIL;
EditorForm: = getEditwindow;
IF assigned (editorform) THEN
Result: =
TPopupMenu (EditorForm.findComponent ('EditorLocalmenu');
END;
Функция GetEditWindow методом перебора форм у экземпляра TApplication ищет окно редактора исходного текста по имени. А функция GetEditWindowPopup пытается обнаружить экземпляр выпадающего меню редактора. Как видно из примера поиск происходит по имени и типу компонента. "А как узнать имя и тип искомого элемента управления?" , - спросите вы Для этого существует два способа.Метод перебора элементов принадлежащих родителю (свойство Parent) через TComponent.Components Использовать DAP -.. Delphi Application Peeper Это разработка известного в определенных кругах донецкого разработчика Виктора Трунова Утилита доступна на сайте компании Devrace по адресу. http://www.devrace.com/dap. Это самый простой способ получить все необходимые данные. Она позволяет просматривать published свойства компонентов в RunTime, изучить установленные обработчики событий, иерархию наследования и так далее. Автор рекомендует эту утилиту всем, у кого возникает елание изучить IDE изнутри. изложенная автором методика, только теоретически демонстрирует основные приемы. читатель, руководствуясь этими принципами, может пойти намного дальше.
Известные ошибки в Toolsapi различных версий IDEизвестные ошибки в Delphi 7 Open Tools API (некоторые есть в Delphi / C Builder 6)
IOTAComponent.GetParent всегда возвращает nil. Вызов IOTAEditView.SetTempMsg убирает закладку диаграмм редактора кода при нажатии кнопки "назад" в редакторе исходного кода Некоторые из IOTAProjectOptions не работают, например IncludeVersionInfo и ModuleAttribs. Также некоторые полезные опции, такие как BreakOnException, отсутствуют. Некоторые опции , например LibraryPath, непостоянны в течение сессии. Параметр howMany в методе IOTAEditPosition.Delete игнорируется. В результате, метод всегда возвращает один символ. IOTASourceEditor.SetSyntaxHighlighter отменен и больше не может использоваться. Указание IOTAEditView.CursorPos не обновляет позицию курсора в окне редактора в строке . состояния среда не убирает экземпляры IOTACustomMessage из окна Message View до выгрузки эксперта Это может привести к аварийной ситуации, так как среда обращается к невыгруженной библиотеке Обходный путь -.. вызвать ClearToolMessages перед тем, как ваш эксперт выгружается, если он добавил свои сообщения IOTATo. DoManager.ProjectChanged никогда не вызывается. Вы не можете добавить такие горячие клавиши, как Ctrl / и Ctrl K. IOTAResourceEntry.DataSize должен делиться на 4 (выравниваться по 4-х байтной границе), иначе вы получите ошибку RLINK32 при компиляции. Известные ошибки в Delphi 6 Open Tools API (некоторые есть в Delphi / C Builder 5)
TIModuleInterface.GetFormInterface не реализован и всегда возвращает нулевое значение. Вместо него вы должны использовать IOTAFormEditor. Интерфейсы Open Tools назначения горячих клавиш иногда вызывают Aссess Violation при использовании IOTAKeyBoardServices.AddKeyboardBinding. Каждый раз при использовании IOTAEditView.PosToCharPos вызывает Aссess Violation в dfwedit.dll. IOTAEditorServices .TopView вызывает Aссess Violation в пакете coride, если он вызван, когда ни один файл не открыт. Известные ошибки в C Builder 5.01 Open Tools APIЕсли вызвать IOTAModule.GetModuleFileCount с открытым модулем без связанной формы, он возвращает 2, но если вызвать IOTAModule .GetModuleFileEditor с индексом 1, возникает Aссess Violation, и индекс 2 возвращает файл * .h. Установка опции проекта libDir при использовании OTAProjectOptions.Values выдает Aссess Violation. Известные ошибки в Delphi 5.01 Open Tools API
Вызов IOTAModuleServices.OpenProject в файле BPG приводит к аварийному закрытию среды. Вместо этого используйте IOTAModuleServices.OpenFile. При создании в IOTAProjectGroup.FileName, вы не получаете полное имя пути. Вместо него используйте IOTAModule, который реализует IOTAProjectGroup. Его свойство FileName выдаст полный путь. опции проекта MajorVersion, MinorVersion, Release, и Build не обновляют опции диалога проекта при их определении. Вы не можете использовать интерфейсы для назначения горячих клавиш типа Ctrl Enter, Shift Enter. при открытии файла BPG, IOTAIDENotifier отошлет параметр с пустым именем файла вместе с ofnFileOpened в метод FileNotification. В среде возникают Aссess Violation или I / O ошибки при задании имени файла без полного пути при использовании IOTAProjectCreator.GetFileName.
Примечание:
Список взят с сайта www.gexperts.org и составлен эриком ээрри (Erik Berry). Лт b b
Delphi and Kylix Cross-Platform Open Tools API. Copyright (C) 1995, 2001 Borland Software Corporation. Erik's Berry ToolsAPI FAQ Cyril's J. Jandia Open Tools Tutorial Open Tools Interfaces by Rappido Borland's OpenToolsApi newsgroups
Приложение 1 К документу приложен архив ToolsAPIExamples.zip, который находится в Интернете по адресу http://www.devrace.com/files/tools_api_examples.zip.В него вошли все примеры, рассмотренные в статье.
Смотрите также материалы по темам:
[Эксперты] обсуждение материала [28-01-2004 15:18] 4 соощщения
Если вы заметили орфографическую ошибку на этой странице, просто выделите ошибку мышью и нажмите Ctrl Enter.Функция может не работать в некоторых версиях броузеров.
© при использовании юююых материалов «королевства Delphi» необходимо указывать источник информации.
Все используемые на сайте торговые марки явлюются собственностюю их производителей.
Основная страница
|
Королевские роники
|
Карта сайта
|
Глас народа!
|
Дальние земли
|
Круглый стол
|
Книга песка
|
Свитки |
Фолианты |
Сокровищница |
Подземелье магов
|
Hello, World!
|
Арсенальная башня
|
Рыцарский зал |
Базарная площадь
|
Городская площадь
|
Почта
Тематический каталог сайта |
Все манускрипты
script>
script>
script>
script>
script>