ダイソー Bluetooth モバイルシャッターに特化したライブラリ

2022/05/31 08:58
大バグを修正して再掲しました。


DaisoBleButton.hpp

#ifndef _DAISO_BLE_BUTTON_HPP_
#define _DAISO_BLE_BUTTON_HPP_

#include <NimBLEDevice.h>
#include <nvs.h>
#include <functional>
#include <map>

static const char * nvs_name = "DaisoBleButton";
static const char * allow_devices [] = {"AB Shutter3"};
static const NimBLEUUID uuid_service((uint16_t)0x1812);
static const NimBLEUUID uuid_characteristic((uint16_t)0x2a4d);

class DaisoBleButton {
  public:
    static int begin() {
      NimBLEDevice::init("");

      uint8_t devCount = 0;
      uint32_t nvs_handle;
      if(!nvs_open(nvs_name, NVS_READONLY, &nvs_handle)) {
        for(;;devCount++) {
          char key[3];  //, buf[20];
          sprintf(key, "%02u", devCount);
          uint64_t address;
          if(nvs_get_u64(nvs_handle, key, &address))
            break;
      
          auto pClient = NimBLEDevice::createClient(NimBLEAddress(address));
          if(pClient) {
            pClient->setConnectTimeout(1);
            buttons[pClient] = new DaisoBleButton(devCount, pClient);
          }
        }
      }
      nvs_close(nvs_handle);
      return devCount;
    }
    
    static int paring(const int period = 10) {
      auto pClients = NimBLEDevice::getClientList();
      for(auto pClient : *pClients)
        pClient->disconnect();
    
      auto pBLEScan = NimBLEDevice::getScan();
      pBLEScan->setActiveScan(true);
    
      Serial.println("paring mode. wait 10 secs..");
      auto pScanResults = pBLEScan->start(period);
    
      int devCount = 0;
      uint32_t nvs_handle;
      if(!nvs_open(nvs_name, NVS_READWRITE, &nvs_handle)) {
        for (int i = 0; i < pScanResults.getCount(); i++) {
          auto advertisedDevice = pScanResults.getDevice(i);
          Serial.print("Found Device ");
          Serial.println(advertisedDevice.getName().c_str());
          for(auto allow : allow_devices) {
            if (!strncmp(advertisedDevice.getName().c_str(), allow, strlen(allow)) && advertisedDevice.haveServiceUUID())  {
              auto pClient = NimBLEDevice::createClient(advertisedDevice.getAddress());
              if(pClient && pClient->connect()) {
                if(!devCount)
                  nvs_erase_all(nvs_handle);
      
                uint64_t address = advertisedDevice.getAddress();
                char key[3];
                sprintf(key, "%02d", devCount);
                nvs_set_u64(nvs_handle, key, address);
                Serial.printf("added to list[%d]: %llu %s", devCount, address, advertisedDevice.getAddress().toString().c_str());
                Serial.println();
                devCount ++;
                pClient->disconnect();
                break;
              }
            }
          }
        }
      }  
    
      nvs_close(nvs_handle);
      return devCount;
    }

    static void handle() {
      auto pClients = NimBLEDevice::getClientList();
      for (auto pClient : *pClients) {
        if(pClient && !pClient->isConnected() && pClient->connect()) {
          pClient->getServices(true);
          auto pService = pClient->getService(uuid_service);
          auto pCharacteristics = pService->getCharacteristics(true);
          for (auto pCharacteristic : *pCharacteristics) {
            if(uuid_characteristic.equals(pCharacteristic->getUUID())) {
              pCharacteristic->subscribe(false,
                [](BLERemoteCharacteristic* pRemoteCharacteristic, uint8_t* pData, size_t length, bool isNotify) {
                  buttons[pRemoteCharacteristic->getRemoteService()->getClient()]->subscribe(pRemoteCharacteristic, pData, length, isNotify);
                }
              );
            }
          }
        }
      }
    }

    static std::function<void(DaisoBleButton *)> onClick_A;
    static std::function<void(DaisoBleButton *)> onClick_B;
    
  private:
    static std::map<NimBLEClient *, DaisoBleButton *> buttons;

    
  public:
    uint8_t getId() {return id;}
    NimBLEAddress getAddress() {return bleClient->getPeerAddress();}

  private:
    DaisoBleButton(uint8_t _id, NimBLEClient * _bleClient) : id(_id), bleClient(_bleClient) {;}
    
    void subscribe(BLERemoteCharacteristic* pRemoteCharacteristic, uint8_t* pData, size_t length, bool isNotify) {
      if(length == 2 && (pData[0] || pData[1])) {
        if(pData[0] == 0x01 && pData[1] == 0x00) {
          if(last_value[0] == 0x00 && last_value[1] == 0x28) {
            if(onClick_B)
              onClick_B(this);
          }
          else {
            if(onClick_A)
              onClick_A(this);
          }
        }
        last_value[0] = pData[0];
        last_value[1] = pData[1];
      }
    }
  
  private:
    NimBLEClient * bleClient;
    uint8_t last_value[2] = {0x00, 0x00};
    uint8_t id;
};

std::function<void(DaisoBleButton *)> DaisoBleButton :: onClick_A = nullptr;
std::function<void(DaisoBleButton *)> DaisoBleButton :: onClick_B = nullptr;
std::map<NimBLEClient *, DaisoBleButton *> DaisoBleButton :: buttons;

#endif


使用例

/*
 * ダイソーモバイルシャッターリモコン
 *
 * IO0: HIGH→LOW/ペアリングモードに入り、10秒間の間に接続成功したデバイスを保存する
 */
 
#include "DaisoBleButton.hpp"

void setup() {
  Serial.begin(115200);
  pinMode(0, INPUT_PULLUP);

  DaisoBleButton::onClick_A = [] (DaisoBleButton * button) {
    Serial.println("button A");
  };

  DaisoBleButton::onClick_B = [] (DaisoBleButton * button) {
    Serial.println("button B");
  };
  
  if(!DaisoBleButton::begin()) {
    DaisoBleButton::paring();
    ESP.restart();
  }
}

void loop() {
  DaisoBleButton::handle();
  
  if(digitalRead(0) == LOW && DaisoBleButton::paring() > 0)
    ESP.restart();

  delay(1);
}

 ペアリングモード(デフォルト10秒間)の間に接続された ダイソーモバイルシャッター の情報を NVS に記録し、次回からはその接続のみを受け付けます。
 GPIO0 を押すと改めてペアリングモードに入り、ペアを登録し直すことが出来ます。
 ESP32-DevBoard 以外の、例えば M5Stack 等では GPIO0 ではなく備わってる物理スイッチに変更するといいでしょう。