インフラ・デバイス

SwitchBotの気温と湿度を
BLEでアプリに連携する

この記事は約21分で読めます。
記事内に広告が含まれています。
スポンサーリンク

この記事は Androidスマホ用のアプリ開発の中で、
今後の開発で再使用性が高いと思われるコーディングをまとめたものです。
Java での開発経験、XML構文規則、Android のアプリ開発経験がある方を対象としています。
Android のアプリ開発でお役にたててれば、嬉しいです。
(これから Android のアプリ開発や Java での開発を始めたい方への案内は、
記事の最後で紹介します)

この記事のテーマ


SwitchBotから気温と湿度をBLE通信でアプリに連携する機能を実装する

意外かも知れませんが、スマホには温度と湿度のセンサーをもっていません。
アプリで温度と湿度を扱う必要がある場合、BLEで通信できるSwitchBotをおススメします♪

ポイント

SwitchBot温湿度計には高精度の温度センサーと湿度センサーを搭載しています。
本体の液晶画面に表示するほか、BLE(BluetoothLowEnergy)を使って、気温と湿度をインタフェースすることが可能です。
また、温湿度計は、現在の気温と湿度をBLEアドバタイズに含めてデータ送信しています。
温湿度計から送信されているBLEアドバタイズにある気温と湿度をアプリに連携する機能を、Bluetoothライブラリで実装する方法を紹介します。

GPS走行記録アプリ(Archive)は、SwitchBot温湿度計から気温と湿度を取得しています

BLEコントローラ

Bluetoothデバイスと通信する場合、マニフェストファイルへの権限指定が必要です。
さらに Android12以降では、アプリ側でユーザー承認をリクエストする必要があります。
マニフェストファイルへの権限指定と権限チェックの実装については、以下の参考記事を参照ください。

BLEコントローラクラス

BLEコントローラクラスでは、BLE MACBLEアドバタイズのタイプを指定して、受信するBLEアドバタイズのフィルタ処理を行い、目的のサービスデータを取得します。
スキャン(受信待ち状態)処理では、
Handlerを使用して、タイムアウト処理を実装します。
また、インタフェースを実装して、受信したサービスデータの受け渡しを行います。

:
public class BluetoothLowEnergyController {
    private static final long           SCAN_PERIOD = 30000;    // スキャンタイムアウト(30秒)
    private final Handler               handler;
    private final BluetoothAdapter      bluetoothAdapter;
    private final BleScanCallback       bleScanCallback;
    private final BluetoothLeScanner    bluetoothLeScanner;
    private OnChangeListener            onChangeListener;
    private final Set<ScanResult>       results = new HashSet<>();
    private final List<ScanResult>      batchScanResults = new ArrayList<>();
    private byte[]                      scanData = new byte[]{0};
    private Runnable                    runnable;
    private String                      macAddress;
    private Long                        updated;
    private long                        timer;
    private Byte                        type = null;
    private final Context               context;
    // コンストラクタ
    public BluetoothLowEnergyController(Context context) {
        this.context = context;
        this.handler = new Handler(Looper.getMainLooper());
        this.bluetoothAdapter = BluetoothAdapter.getDefaultAdapter();
        this.bluetoothLeScanner = bluetoothAdapter.getBluetoothLeScanner();
        this.bleScanCallback = new BleScanCallback();
    }
    // インタフェース
    public interface OnChangeListener {
        void onValueChanged(byte[] value);
    }
    public void setOnChangeListener(final BluetoothLowEnergyController.OnChangeListener onChangeListener) {
        this.onChangeListener = onChangeListener;
    }
    @SuppressLint("MissingPermission")
    public void scan() {
        timer = System.currentTimeMillis();
        runnable = new Runnable() {
            @Override
            public void run() {
                if (results.size() == 0 && System.currentTimeMillis() > timer + SCAN_PERIOD ) {
                    handler.removeCallbacks(runnable);
                    // BLEスキャン停止
                    bluetoothLeScanner.stopScan(bleScanCallback);
                }
                handler.postDelayed(this, SCAN_PERIOD);
            }
        };
        bleScanCallback.clear();
        // BLEスキャン開始
        bluetoothLeScanner.startScan(buildScanFilters(), buildScanSettings(), bleScanCallback);
        handler.post(runnable);
    }
    @SuppressLint("MissingPermission")
    public void pause() {
        handler.removeCallbacks(runnable);
        bluetoothLeScanner.stopScan(bleScanCallback);
        bleScanCallback.clear();
    }
    private List<ScanFilter> buildScanFilters() {
        List<ScanFilter> scanFilters = new ArrayList<>();
        ScanFilter.Builder builder = new ScanFilter.Builder();
        builder.setDeviceAddress(macAddress);
        scanFilters.add(builder.build());
        return scanFilters;
    }
    private ScanSettings buildScanSettings() {
        ScanSettings.Builder builder = new ScanSettings.Builder();
        builder.setScanMode(ScanSettings.SCAN_MODE_LOW_POWER);
        return builder.build();
    }
    private class BleScanCallback extends ScanCallback {
        @SuppressLint("MissingPermission")
        @Override
        public void onScanResult(int callbackType, ScanResult result) {
            if (callbackType == ScanSettings.CALLBACK_TYPE_ALL_MATCHES) {
                results.add(result);
                BluetoothDevice bluetoothDevice = result.getDevice();
                scanData = result.getScanRecord().getBytes();
                updated = System.currentTimeMillis();
                if (type != null) {
                    int length;
                    int position = 0;
                    byte newType;
                    while (position < scanData.length) {
                        length = scanData[position];
                        if (length == 0) { break; }
                        position++;
                        newType = scanData[position];
                        if (newType == type) {
                            byte[] value = Arrays.copyOfRange(scanData, position + 1, position + length);
                            // 受信データ通知
                            onChangeListener.onValueChanged(value);
                        }
                        position = position + length;
                    }
                } else {
                    handler.removeCallbacks(runnable);
                    //BLEスキャン停止
                    bluetoothLeScanner.stopScan(bleScanCallback);
                }
            }
        }
        @Override
        public void onBatchScanResults(List<ScanResult> results) {
            batchScanResults.addAll(results);
        }
        @Override
        public void onScanFailed(int errorCode) {
        }
        synchronized void clear() {
            results.clear();
            batchScanResults.clear();
            Arrays.fill(scanData,(byte)0);
        }
    }
    // Android端末のBluetooth機能の有効化判定
    public boolean requestBluetoothFeature() {
        return bluetoothAdapter.isEnabled();
    }
    // デバイスのMACアドレス設定 //
    public void setMacAddress(String mac) { macAddress = (mac.length() > 0 && !mac.equals("none") ? mac : "00:00:00:00:00:00"); }
    // BLEアドバタイズパケット取得 //
    public byte[] getScanData() { return scanData; }
    // BLEアドバタイズパケット取得時刻 //
    public String getUpdated() {
        SimpleDateFormat HHmm = new SimpleDateFormat("HH:mm", Locale.JAPAN);
        Date date = new Date(updated);
        return HHmm.format(date);
    }
    // タイプ設定 //
    public void setType(Byte newType) { type = newType; }
    // 値取得 //
    public Set<ScanResult> getScanResult() { return results; }
    public List<ScanResult> getBatchScanResults() { return batchScanResults; }
}

SwichBotインタフェース

SwitchBotインタフェースでは、BLEコントローラにSwitchBotデバイスのBLE MACBLEアドバタイズのタイプを指定して、BLEスキャンを開始します。
SwitchBotから取得した気温と湿度はバイナリデータのため、数値変換と、気温と湿度それぞれのTextViewに数値変換した値を出力します。
バイナリデータの構造(フォーマット)は、以下の参考記事を参照ください。

また、インスタンス化したBLEコントローラを操作するためのgetterメソッドを実装します。

    :
    private static final String         DEVICE_MAC = "CA:D4:47:66:1E:BB";
    private static final byte           SWITCHBOT = 0x16;
    :
    public BluetoothLowEnergyController getBluetoothLowEnergyController() {
        return bluetoothLowEnergyController;
    }
    public void setBluetoothLowEnergyController(BluetoothLowEnergyController newBluetoothLowEnergyController, TextView temp, TextView hum) {
        if (bluetoothLowEnergyController == null) {
            bluetoothLowEnergyController = newBluetoothLowEnergyController;
            bluetoothLowEnergyController.setOnChangeListener(value -> {
                if (value.length > 6) {
                    int sign = value[6] & 0x80;
                    float decimals = (value[5] & 0x0f);
                    temperature = (sign == 0x80) ? (value[6] & 0x7f) + (decimals / 10) : (value[6] & 0x7f) + (decimals / 10) * -1;
                    temp.setText(String.format(context.getString(R.string.format_temperature), temperature));
                    if (value.length > 7) {
                        humidity = value[7] & 0x7f;
                        hum.setText(String.format(context.getString(R.string.format_humidity), humidity));
                    }
                }
            });
            bluetoothLowEnergyController.setMacAddress(DEVICE_MAC);
            bluetoothLowEnergyController.setType(SWITCHBOT);
            bluetoothLowEnergyController.scan();
        }
    }
    :

Fragment

Fragement(または、Activity)では、気温と湿度を表示するためのTextViewをセットし、SwitchBotインタフェースにインスタンス化したBLEコントローラとTextViewを引き渡します。
BLEコントローラは、
SwitchBotから受信した気温と湿度をTextViewに一定間隔で更新します。
BLEコントローラの停止はonPauseで行い、再開はonResumeで行います。

    : 
    @Override
    public View onCreateView(@NonNull LayoutInflater inflater, ViewGroup container,
                             Bundle savedInstanceState)