Delphi и Bluetooth. Часть 2 (исходники)Источник: mobileservicesoft Петриченко Михал
В первой части статьи мы научились получать список локальных радиомодулей Bluetooth и узнавать их свойства. Теперь пришло время получить список устройств Bluetooth, которые подключены к нашим локальным радиомодулям. Получение списка устройств Bluetooth Для получения списка устройств Bluetooth нам понадобятся следующие функции (они очень похожи на функции, используемые для получения списка локальных радиомодулей). BluetoothFindFirstDevice Начинает перечисление устройств Bluetooth. Объявление функции: function BluetoothFindFirstDevice( Параметры: pbtsp - указатель на структуру BLUETOOTH_DEVICE_SEARCH_PARAMS. Член dwSize этой структуры должен содержать размер структуры (устанавливается посредством SizeOf(BLUETOOTH_DEVICE_SEARCH_PARAMS)). Член hRadio должен содержать описатель локального радиомодуля, полученный вызовом функции BluetoothFindFirstRadio. Возвращаемые значения: в случае успешного выполнения функция вернет корректный описатель в качестве результата. В случае ошибки будет возвращен 0. Для получения кода ошибки используйте функцию GetLastError. BluetoothFindNextDevice Находит следующее устройство Bluetooth. Объявление функции: function BluetoothFindNextDevice( Параметры: hFind - описатель, который вернула функция BluetoothFindFirstDevice. Возвращаемые значения: вернет TRUE, если устройство найдено. Вернет FALSE в случае отсутствия устройства. phRadio содержит некорректный описатель. Используйте GetLastError для получения кода ошибки. BluetoothFindDeviceClose Закрывает описатель перечисления устройств Bluetooth. Объявление функции: function BluetoothFindDeviceClose( Параметры: hFind - описатель, который вернула функция BluetoothFindFirstDevice. Возвращаемые значения: вернет TRUE если описатель успешно закрыт. Вернет FALSE в случае ошибки. Для получения кода ошибки используйте GetLastError. BluetoothGetDeviceInfo Возвращает информацию об указанном устройстве Bluetooth. Объявление функции: function BluetoothGetDeviceInfo( Параметры: hRadio - описатель локального радиомодуля Bluetooth. Возвращаемые значения: вернет ERROR_SUCCESS если выполнено успешно и информация занесена в структуру pbtdi. Остальные значения - код ошибки. Обладая этими знаниями, можно написать процедуру получения информации об устройствах Bluetooth. procedure GetDevices(_hRadio: THandle); // Инициализация структуры BLUETOOTH_DEVICE_INFO // Начинаем поиск // Инициализация структуры BLUETOOTH_DEVICE_INFO // Закрываем поиск Нам осталось научиться получать список сервисов, предоставляемых удаленным устройством и управлять локальными радиомодулями. Так же, необходимо разобраться, как же все-таки передаются данные между различными устройствами Bluetooth. Мы создадим программу, которая поможет нам обобщить полученную информацию и покажет, как использовать новые функции, которые здесь будут описаны. Прежде, чем мы приступим, давайте определимся в терминах. Microsoft в своей документации вводит два термина: Radio и Device. Radio - это локальный радиомодуль Bluetooth (USB-брелок, интегрированное решение, в общем то, что установлено на вашем компьютере). Device - это то устройство Bluetooth с которым вы хотите обмениваться информацией. Будь то телефон, КПК, гарнитура или еще что-то. Важно понимать, что если мы пишем программу для PDA, то когда она работает на PDA - его модуль Bluetooth будет Radio, а компьютер - Device. Если же она работает на компьютере, то компьютерный модуль - Radio, а PDA - Device. Что мы знаем К сожалению, документация Microsoft по Bluetooth API и работе с Bluetooth настолько скудна (у меня получилось 50 страниц в Word с оформлением), а примеров они вообще не предоставляют, что из нее очень трудно понять, как же все-таки работает эта технология. Когда я только начинал изучать этот предмет, я перерыл весь Internet, но так ничего вразумительного не нашел. Поэтому, здесь мне хочется дать наиболее полную и подробную информацию об этом вопросе, что бы вы не столкнулись с той же проблемой отсутствия информации. И так, приступим. Создание проекта Давайте создадим в Delphi новый проект и сохраним его под именем BTWork, а модуль - под именем Main. Главную и пока единственную форму, назовем fmMain. Заголовок BTWork. Теперь нам понадобятся файл JwaBluetoothAPI.pas, JwaBtHDef.pas и JwaBthSdpDef.pas. Их можно найти в примерах из предыдущих частей или в библиотеке BTClasses. Для того, чтобы не тянуть с собой все остальные файлы из JWA, давайте эти чуть-чуть исправим. Найдите в них строку uses JwaWindows и замените JwaWindows на Windows. Далее удалить из них строки {$WEAKPACKAGEUNIT} И в файле JwaBluetoothAPI удалите все, что находится между {$IFDEF DYNAMIC_LINK} и {$ELSE} вместе с этими DEF. И в конце этого файле удалите {$ENDIF}. Далее, в JwaBluetoothAPI.pas после implementation uses Да простят нас ребята, которые эту библиотеку писали! Все эти действия я делал для того, что бы уменьшить архив примера. Да и не нужно тянуть за собой много лишнего. Хотя сама библиотека весьма полезна. Один модуль JwaWindows чего стоит. Там очень много интересного есть. Ну да ладно - что-то я отвлекся. После того, как мы кастрировали эти модули, давайте добавим их в наш проект. Готово? В этом приложении мы будем получать список локальных радиомодулей, устройств, к ним присоединенных, список сервисов, предоставляемых устройствами. Также мы должны управлять радиомодулями и научиться проходить авторизацию. Приступаем. Оформление главной формы
Далее поместим компонент TTreeView и установите свойства как в таблице:
Правее TTreeView поместим TSplitter и установим следующие его свойства:
И, наконец, помещаем компонент TListView еще правее TSplitter. Устанавливаем его свойства как в таблице:
На TPanel поместим кнопку TButton.
Теперь мы готовы писать программу. Пишем код При старте нашей программы, желательно чтобы сразу заполнялся TreeView. В нем будут показаны модули Bluetooth и устройства, которые к ним подключены. Для этого в обработчике OnCreate формы fmMain напишем такой код: procedure TfmMain.FormCreate(Sender: TObject); А в обработчике OnClick кнопки btRefresh напишем следующее: procedure TfmMain.btRefreshClick(Sender: TObject); // Очищаем дерево // Корневая ветвь в дереве // Начинаем поиск локальных модулей Bluetooth // Добавляем радио в дерево // Начинаем поиск устройств для найденного радиомодуля. New(DevInfo); // Ищем первое // Ищем следующее устройство // Поиск устройств закончен // Находим следующее радио // Поиск радиомодулей закончен EndUpdate; with TreeView do В uses нашего модуля, который относится к главной форме, допишем: implementation // Уже написано!!! Ниже добавим функцию: // Преобразует адрес из внутреннего формата (dword) в строку, Здесь хочу привести описание используемых структур, так как ранее я их не описывал: BLUETOOTH_DEVICE_SEARCH_PARAMS Объявление: BLUETOOTH_DEVICE_SEARCH_PARAMS = record Члены:
BLUETOOTH_DEVICE_INFO Объявление: BLUETOOTH_DEVICE_INFO = record Члены:
BLUETOOTH_RADIO_INFO Объявление: BLUETOOTH_RADIO_INFO = record Члены:
Далее напишем вот такой обработчик события OnChange для TreeView: procedure TfmMain.TreeViewChange(Sender: TObject; Node: TTreeNode); procedure ShowRadios; // Заполняем список CurNode := ASelected.GetFirstChild; while Assigned(CurNode) do begin // Перечитать информацию о радиомодуле with Add do CurNode := ASelected.GetNextChild(CurNode); EndUpdate; procedure ShowDevices; // Заполняем список CurNode := ASelected.GetFirstChild; while Assigned(CurNode) do // Перечитываем информацию об устройстве with Add do CurNode := ASelected.GetNextChild(CurNode); EndUpdate; procedure ShowServices; // Заполняем список // Получаем размер массива сервисов (hRadio, Info, ServiceCount, nil); // Выделяем память // Получаем список сервисов (hRadio, Info, ServiceCount, PGUID(Services)); // Рисуем их // Очищаем память EndUpdate; begin // Очищаем ListView with Items do // Заполняем информацией В этом коде появилось три новые функции. Вот они: BluetoothIsConnectable - определяет, возможно ли подключение к указанному радиомодулю. Объявление функции: function BluetoothIsConnectable(const hRadio : THandle): BOOL; stdcall; Параметры: hRadio - Handle радиомодуля, который мы проверяем. Если 0, то проверяются все радиомодули. Возвращаемые значения: вернет TRUE, если указанный радиомодуль разрешает входящие подключения. Если hRadio=0, то вернет TRUE, если хотя бы один радиомодуль разрешает входящие подключения. Если входящие подключения запрещены, то вернет FALSE. BluetoothIsDiscoverable - определяет, будет ли виден указанный радиомодуль другим при поиске. Если просматриваются все радиомодули, то вернет TRUE если хотя бы один разрешает обнаружение. Объявление функции: function BluetoothIsDiscoverable(const hRadio : THandle): BOOL; stdcall; Параметры: hRadio - Handle радиомодуля, который мы проверяем. Если 0, то проверяются все радиомодули. Возвращаемые значения: вернет TRUE, если указанный радиомодуль разрешает обнаружение. Если hRadio=0, то вернет TRUE, если хотя бы один радиомодуль разрешает обнаружение. Если обнаружение запрещено, то вернет FALSE. BluetoothEnumerateInstalledServices - получает список GUID сервисов, предоставляемых устройством. Если параметр hRadio=0, то просматривает все радиомодули. Объявление функции: function BluetoothEnumerateInstalledServices( Параметры: hRadio - Handle радиомодуля, который мы проверяем. Если 0, то проверяются все радиомодули. Возвращаемые значения: вернет ERROR_SUCCESS, если вызов успешен и количество сервисов в pcServices соответствует реальности. Вернет ERROR_MORE_DATA, если вызов успешен, но выделенное количество памяти (pcServices при вызове) меньше, чем количество предоставляемых сервисов. В случае ошибки - другие коды ошибок Win32. Примечания: Посмотрите на код получения списка сервисов: // Получаем размер массива сервисов // Выделяем память. // Получаем список сервисов Сначала мы вызываем функцию с pcServices=0 и pGuidServices=nil для того, чтобы получить количество сервисов, предоставляемых устройством. Потом выделяем память (SetLength()) и только затем вызываем функцию с реальными параметрами и получаем список сервисов. Еще важное замечание. В файле JwaBluetoothAPIs.pas параметр pbtdi имеет тип PBLUETOOTH_DEVICE_INFO, который раскрывается в BLUETOOTH_DEVICE_INFO. Заметьте, что это НЕ УКАЗАТЕЛЬ. Это не верно, так как в исходном виде функция требует именно указатель. Поэтому, я ввел тип type __PBLUETOOTH_DEVICE_INFO = ^PBLUETOOTH_DEVICE_INFO Так что ИСПОЛЬЗУЙТЕ файл из примера, а не из исходной библиотеки. Иначе получите нарушение доступа к памяти. Комментарий к коду: Мы перечитываем информацию об устройстве, так как за время, пока мы любуемся программой, могли произойти различные события: устройство отключили, отменили авторизацию и т. п. А мы хотим иметь самую свежую информацию об устройстве. В принципе то, что описано выше, мы уже знали, кроме двух указанных функций. Давайте расширим возможности нашего приложения. Добавим функции запрета/разрешения обнаружения радиомодуля и запрета/разрешения подключения к нему. |