📢 Webサイト閉鎖と移転のお知らせ
このWebサイトは2026年9月に閉鎖いたします。
新しい記事は移転先で追加しております。(旧サイトでは記事を追加しておりません)
| (同じ利用者による、間の5版が非表示) | |||
| 144行目: | 144行目: | ||
* AD Data (可変長) | * AD Data (可変長) | ||
<br> | <br> | ||
[[ファイル:Qt BLE 3.png|フレームなし|中央]] | |||
<br> | |||
==== その他 (通信特性 / セキュリティ等) ==== | ==== その他 (通信特性 / セキュリティ等) ==== | ||
* 通信特性 | * 通信特性 | ||
| 526行目: | 529行目: | ||
<br> | <br> | ||
==== 使用例 ==== | ==== 使用例 ==== | ||
===== QLowEnergyControllerクラスのインスタンスの生成 ===== | |||
まず、<code>QBluetoothDeviceDiscoveryAgent</code>クラスを使用して、スキャンを実行する。<br> | |||
<br> | |||
次に、接続するデバイスの<code>QBluetoothDeviceInfo</code>クラスのデバイス情報を取得する。<br> | |||
取得したデバイス情報を元に、<code>QLowEnergyController</code>クラスのインスタンスを生成する。<br> | |||
<br> | |||
===== コントローラのシグナル / スロット接続 ===== | |||
最低限必要なシグナルを示す。<br> | |||
* connected | |||
*: デバイスへの接続完了を通知 | |||
* disconnected | |||
*: デバイスとの切断を通知 | |||
* serviceDiscovered | |||
*: 新しいサービスの発見を通知 | |||
* discoveryFinished | |||
*: サービス探索の完了を通知 | |||
<br> | |||
===== BLEデバイスの接続 ===== | |||
connectToDeviceメソッドを実行してBLEデバイスに接続する。<br> | |||
connectedシグナルの受信を待つ。<br> | |||
<br> | |||
===== サービスの探索 ===== | |||
BLEデバイスへ接続後、discoverServicesメソッドを実行してサービスの探索を開始する。<br> | |||
<br> | |||
* serviceDiscoveredシグナルで個々のサービスが見つかる度に通知される。 | |||
* discoveryFinishedシグナルで探索完了が通知される。 | |||
<br> | |||
===== サービスの取得 ===== | |||
createServiceObjectメソッドをを実行して、探索で発見したサービスのQLowEnergyServiceオブジェクトを生成する。<br> | |||
このオブジェクトを使用して、特性 (Characteristic) や ディスクリプタにアクセスできる。<br> | |||
<br> | |||
===== 組み合わせ ===== | |||
<syntaxhighlight lang="c++"> | <syntaxhighlight lang="c++"> | ||
// BLEのサービス探索を管理するクラス | |||
// BLEデバイスへの接続, サービスの探索と監視 | |||
#include <QObject> | #include <QObject> | ||
#include <QLowEnergyController> | #include <QLowEnergyController> | ||
#include <QLowEnergyService> | #include <QLowEnergyService> | ||
#include <QTimer> | #include <QTimer> | ||
#include <QDebug> | #include <QDebug> | ||
| 539行目: | 576行目: | ||
private: | private: | ||
QLowEnergyController controller; // BLE接続とサービス探索を制御するコントローラ | |||
QTimer discoveryTimeout; // サービス探索のタイムアウトを管理するタイマ | |||
// コントローラの各種シグナルを接続 | |||
// 接続 / 切断, サービス探索の進行状況, 状態変更, エラー | |||
void connectControllerSignals() | void connectControllerSignals() | ||
{ | { | ||
connect(controller | connect(&controller, &QLowEnergyController::connected, this, &BLEServiceDiscovery::onConnected); | ||
connect(controller | connect(&controller, &QLowEnergyController::disconnected, this, &BLEServiceDiscovery::onDisconnected); | ||
connect(controller | connect(&controller, &QLowEnergyController::serviceDiscovered, this, &BLEServiceDiscovery::onServiceDiscovered); | ||
connect(controller | connect(&controller, &QLowEnergyController::discoveryFinished, this, &BLEServiceDiscovery::onDiscoveryFinished); | ||
connect(controller | connect(&controller, static_cast<void(QLowEnergyController::*)(QLowEnergyController::Error)>(&QLowEnergyController::error), this, &BLEServiceDiscovery::onError); | ||
connect(controller | connect(&controller, &QLowEnergyController::stateChanged, this, &BLEServiceDiscovery::onStateChanged); | ||
} | } | ||
// 個別のBLEサービスに対するシグナル | |||
// 状態変更とエラーイベントを監視してログを出力する | |||
void connectServiceSignals(QLowEnergyService *service) | void connectServiceSignals(QLowEnergyService *service) | ||
{ | { | ||
| 563行目: | 604行目: | ||
} | } | ||
// エラーコードの変換 | |||
QString getErrorMessage(QLowEnergyController::Error error) | QString getErrorMessage(QLowEnergyController::Error error) | ||
{ | { | ||
| 619行目: | 661行目: | ||
{ | { | ||
// タイムアウトタイマの初期化 | // タイムアウトタイマの初期化 | ||
discoveryTimeout | discoveryTimeout.setInterval(10000); // 10秒のタイムアウト | ||
discoveryTimeout.setSingleShot(true); | |||
discoveryTimeout | |||
connect(discoveryTimeout | connect(&discoveryTimeout, &QTimer::timeout, this, &BLEServiceDiscovery::onDiscoveryTimeout); | ||
} | } | ||
// デバイスへの接続とサービス探索の開始 | // デバイスへの接続とサービス探索の開始 | ||
void startDiscovery(const QBluetoothDeviceInfo& device) | // コントローラの初期化, シグナルの接続, デバイスへの接続開始, タイムアウトタイマの開始 | ||
void startDiscovery(const QBluetoothDeviceInfo &device) | |||
{ | { | ||
try { | try { | ||
| 633行目: | 675行目: | ||
// コントローラの初期化 | // コントローラの初期化 | ||
connectControllerSignals(); | connectControllerSignals(); | ||
// 接続開始 | // 接続開始 | ||
controller | controller.connectToDevice(); | ||
discoveryTimeout | discoveryTimeout.start(); | ||
} | } | ||
catch (const std::exception &e) { | catch (const std::exception &e) { | ||
| 647行目: | 688行目: | ||
} | } | ||
// 探索を停止 | // 探索を停止 (タイムアウトタイマも停止) | ||
void stopDiscovery() | void stopDiscovery() | ||
{ | { | ||
try { | try { | ||
if (controller) { | if (controller) { | ||
controller | controller.disconnectFromDevice(); | ||
discoveryTimeout | discoveryTimeout.stop(); | ||
} | } | ||
} | } | ||
| 664行目: | 705行目: | ||
signals: | signals: | ||
void serviceDiscovered(QLowEnergyService *service); | void serviceDiscovered(QLowEnergyService *service); // 新しいサービスが発見された時に発行 | ||
void discoveryComplete(); | void discoveryComplete(); // 全てのサービス探索が完了した時に発行 | ||
void errorOccurred(const QString &error); | void errorOccurred(const QString &error); // エラーが発生した時に発行 | ||
void connectionStateChanged(QLowEnergyController::ControllerState state); | void connectionStateChanged(QLowEnergyController::ControllerState state); // 接続状態が変更された時に発行 | ||
private slots: | private slots: | ||
// デバイスへの接続が完了した時 (タイムアウトタイマを停止して、サービス探索を開始) | |||
void onConnected() | void onConnected() | ||
{ | { | ||
qDebug() << "デバイスに接続"; | qDebug() << "デバイスに接続"; | ||
discoveryTimeout | discoveryTimeout.stop(); | ||
// サービス探索の開始 | // サービス探索の開始 | ||
controller | controller.discoverServices(); | ||
} | } | ||
// デバイスから切断された時 (タイムアウトタイマを停止) | |||
void onDisconnected() | void onDisconnected() | ||
{ | { | ||
qDebug() << "デバイスから切断"; | qDebug() << "デバイスから切断"; | ||
discoveryTimeout | discoveryTimeout.stop(); | ||
} | } | ||
// 新しいサービスが発見された時 | |||
void onServiceDiscovered(const QBluetoothUuid& uuid) | void onServiceDiscovered(const QBluetoothUuid& uuid) | ||
{ | { | ||
| 690行目: | 734行目: | ||
// サービスオブジェクトの作成 | // サービスオブジェクトの作成 | ||
QLowEnergyService* service = controller | QLowEnergyService* service = controller.createServiceObject(uuid, this); | ||
if (service) { | if (service) { | ||
connectServiceSignals(service); | connectServiceSignals(service); | ||
| 697行目: | 741行目: | ||
} | } | ||
// サービス探索が完了した時 | |||
void onDiscoveryFinished() | void onDiscoveryFinished() | ||
{ | { | ||
qDebug() << "サービス探索が完了"; | qDebug() << "サービス探索が完了"; | ||
discoveryTimeout | discoveryTimeout.stop(); | ||
emit discoveryComplete(); | emit discoveryComplete(); | ||
} | } | ||
// エラーが発生した時 | |||
void onError(QLowEnergyController::Error error) | void onError(QLowEnergyController::Error error) | ||
{ | { | ||
| 711行目: | 757行目: | ||
} | } | ||
// タイムアウトが発生した時 (探索を停止して、エラーとして通知) | |||
void onDiscoveryTimeout() | void onDiscoveryTimeout() | ||
{ | { | ||
| 718行目: | 765行目: | ||
} | } | ||
// 接続状態が変更された時 (状態の変更をログ出力して、通知) | |||
void onStateChanged(QLowEnergyController::ControllerState state) | void onStateChanged(QLowEnergyController::ControllerState state) | ||
{ | { | ||
| 1,059行目: | 1,107行目: | ||
#include <QLowEnergyCharacteristic> | #include <QLowEnergyCharacteristic> | ||
#include <QTimer> | #include <QTimer> | ||
#include <QDebug> | #include <QDebug> | ||
| 1,067行目: | 1,114行目: | ||
private: | private: | ||
QLowEnergyController controller; | |||
QTimer reconnectTimer; | |||
QBluetoothDeviceInfo currentDevice; | QBluetoothDeviceInfo currentDevice; | ||
bool autoReconnect = false; | bool autoReconnect = false; | ||
| 1,074行目: | 1,121行目: | ||
void connectControllerSignals() | void connectControllerSignals() | ||
{ | { | ||
connect(controller | connect(&controller, &QLowEnergyController::connected, this, &BLEConnection::onConnected); | ||
connect(controller | connect(&controller, &QLowEnergyController::disconnected, this, &BLEConnection::onDisconnected); | ||
connect(controller | connect(&controller, &QLowEnergyController::serviceDiscovered, this, &BLEConnection::onServiceDiscovered); | ||
connect(controller | connect(&controller, static_cast<void(QLowEnergyController::*)(QLowEnergyController::Error)>(&QLowEnergyController::error), this, &BLEConnection::onError); | ||
connect(controller | connect(&controller, &QLowEnergyController::stateChanged, this, &BLEConnection::onStateChanged); | ||
} | } | ||
| 1,189行目: | 1,236行目: | ||
{ | { | ||
// 再接続タイマの初期化 | // 再接続タイマの初期化 | ||
reconnectTimer | reconnectTimer.setInterval(5000); // 5秒間隔で再接続 | ||
reconnectTimer.setSingleShot(true); | |||
reconnectTimer | |||
connect(reconnectTimer | connect(&reconnectTimer &QTimer::timeout, this, &BLEConnection::onReconnectTimeout); | ||
} | } | ||
| 1,203行目: | 1,249行目: | ||
// コントローラの初期化 | // コントローラの初期化 | ||
connectControllerSignals(); | connectControllerSignals(); | ||
// 接続パラメータの設定 | // 接続パラメータの設定 | ||
controller | controller.setRemoteAddressType(QLowEnergyController::PublicAddress); | ||
// 接続開始 | // 接続開始 | ||
controller | controller.connectToDevice(); | ||
currentDevice = device; | currentDevice = device; | ||
} | } | ||
| 1,225行目: | 1,270行目: | ||
try { | try { | ||
if (controller) { | if (controller) { | ||
controller | controller.disconnectFromDevice(); | ||
reconnectTimer | reconnectTimer.stop(); | ||
} | } | ||
} | } | ||
| 1,241行目: | 1,286行目: | ||
autoReconnect = enable; | autoReconnect = enable; | ||
if (!enable) { | if (!enable) { | ||
reconnectTimer | reconnectTimer.stop(); | ||
} | } | ||
} | } | ||
| 1,256行目: | 1,301行目: | ||
{ | { | ||
qDebug() << "デバイスに接続"; | qDebug() << "デバイスに接続"; | ||
reconnectTimer | reconnectTimer.stop(); | ||
emit connected(); | emit connected(); | ||
// サービスの探索を開始 | // サービスの探索を開始 | ||
controller | controller.discoverServices(); | ||
} | } | ||
| 1,271行目: | 1,316行目: | ||
if (autoReconnect) { | if (autoReconnect) { | ||
qDebug() << "再接続を試行..."; | qDebug() << "再接続を試行..."; | ||
reconnectTimer | reconnectTimer.start(); | ||
} | } | ||
} | } | ||
| 1,279行目: | 1,324行目: | ||
qDebug() << "サービスを発見: " << uuid.toString(); | qDebug() << "サービスを発見: " << uuid.toString(); | ||
QLowEnergyService* service = controller | QLowEnergyService* service = controller.createServiceObject(uuid, this); | ||
if (service) { | if (service) { | ||
connectServiceSignals(service); | connectServiceSignals(service); | ||
| 1,292行目: | 1,337行目: | ||
emit errorOccurred(errorMessage); | emit errorOccurred(errorMessage); | ||
if (autoReconnect && error != QLowEnergyController::InvalidBluetoothAdapterError) reconnectTimer | if (autoReconnect && error != QLowEnergyController::InvalidBluetoothAdapterError) reconnectTimer.start(); | ||
} | } | ||
| 1,305行目: | 1,350行目: | ||
if (autoReconnect && currentDevice.isValid()) { | if (autoReconnect && currentDevice.isValid()) { | ||
qDebug() << "再接続を試行..."; | qDebug() << "再接続を試行..."; | ||
controller | controller.connectToDevice(); | ||
} | } | ||
} | } | ||