Using new BTHome library

Posted on
  • Coming back to this for a bit. Great to see a BTHome library. Quick and easy device/sensors into HA.
    Works great with your example using a singlebutton press. I can see the event on the Home Assistant log.

    One thing I can't figure out is how to set up long press and double press events. This sends through both a double and a single press when I hit the puck button once

    1. ...
    2. function updateAdvertising() {
    3. NRF.setAdvertising(require("BTHome").getAdvertisement([
    4. {
    5. type : "battery",
    6. v : E.getBattery()
    7. },
    8. {
    9. type : "temperature",
    10. v : E.getTemperature()
    11. },
    12. {
    13. type: "button_event",
    14. v: buttonState ? "press" : "none"
    15. },
    16. {
    17. type: "button_event",
    18. v: buttonState ? "double_press" : "none"
    19. },
    20. ]), { name : "Puck" });
    21. ...

    Feeling stupid - Thanks

  • PS In an automation the device action can be single/double/long etc - but other than a single press nothing is recognized. Thanks.

  • What happens if you just have a single button_event and change what you put in for v?

    I just tried this and it seems to work ok for me - short presses are press and long presses are long_press:

    1. var slowTimeout; //< After 60s we revert to slow advertising
    2. // Update the data we're advertising here
    3. function updateAdvertising(buttonState) {
    4. NRF.setAdvertising(require("BTHome").getAdvertisement([
    5. {
    6. type : "battery",
    7. v : E.getBattery()
    8. },
    9. {
    10. type : "temperature",
    11. v : E.getTemperature()
    12. },
    13. {
    14. type: "button_event",
    15. v: buttonState
    16. },
    17. ]), {
    18. name : "Sensor",
    19. interval: (buttonState!="none")?20:2000, // fast when we have a button press, slow otherwise
    20. // not being connectable/scannable saves power (but you'll need to reboot to connect again with the IDE!)
    21. //connectable : false, scannable : false,
    22. });
    23. /* After 60s, call updateAdvertising again to update battery/temp
    24. and to ensure we're advertising slowly */
    25. if (slowTimeout) clearTimeout(slowTimeout);
    26. slowTimeout = setTimeout(function() {
    27. slowTimeout = undefined;
    28. updateAdvertising("none" /* no button pressed */);
    29. }, 60000);
    30. }
    31. // When a button is pressed, update advertising with the event
    32. setWatch(function(e) {
    33. var buttonState = ((e.time - e.lastTime) > 0.5) ? "long_press" : "press";
    34. updateAdvertising(buttonState);
    35. }, BTN, {edge:"falling", repeat:true})
    36. // Update advertising now
    37. updateAdvertising("none");
    38. // Enable highest power advertising (4 on nRF52, 8 on nRF52840)
    39. NRF.setTxPower(4);

    I'm struggling to get Home assistant logs at the moment but this is definitely advertising the correct data.

    You could extend the setWatch call so it stores the last time it was called, and then if it was called again within ~1 second you can send "double_press" instead?

  • this works with press and long_press. I can see the events hapenning under BTHome, device (Puck). I had to add var buttonState; at the top.

    will try double_press next.

    Thanks.

  • Rather than try to do the time calculations I am trying to use the SWButton.js module.
    This seems to work. Have I got it correct?

    1. var buttonState;
    2. var slowTimeout; //< After 60s we revert to slow advertising
    3. var btnData = {};
    4. var SWBtn = require("https://raw.githubusercontent.com/muet/EspruinoDocs/master/modules/SWButton.js");
    5. // these get created with a puck button press
    6. var mySWBtn = new SWBtn(function(k){
    7. if (k === "S" ) { // single button press
    8. btnData.btn = "press";
    9. }
    10. else if (k === "L" ) { // long button press
    11. btnData.btn = "long_press";
    12. }
    13. else if (k === "SS") { // double short press
    14. btnData.btn = "double_press";
    15. }
    16. updateBTHome(btnData);
    17. });
    18. function updateBTHome(btnData) {
    19. buttonState = btnData.btn;
    20. updateAdvertising(buttonState);
    21. }
    22. // Update the data we're advertising here
    23. function updateAdvertising(buttonState) {
    24. NRF.setAdvertising(require("BTHome").getAdvertisement([
    25. {
    26. type : "battery",
    27. v : E.getBattery()
    28. },
    29. {
    30. type : "temperature",
    31. v : E.getTemperature()
    32. },
    33. {
    34. type: "button_event",
    35. v: buttonState
    36. },
    37. ]), {
    38. name : "Sensor",
    39. interval: (buttonState!="none")?20:2000, // fast when we have a button press, slow otherwise
    40. });
    41. /* After 60s, call updateAdvertising again to update battery/temp and to ensure we're advertising slowly */
    42. if (slowTimeout) clearTimeout(slowTimeout);
    43. slowTimeout = setTimeout(function() {
    44. slowTimeout = undefined;
    45. updateAdvertising("none" /* no button pressed */);
    46. }, 60000);
    47. }
    48. // Update advertising now
    49. updateAdvertising("none");
    50. // Enable highest power advertising (4 on nRF52, 8 on nRF52840)
    51. NRF.setTxPower(4);

    Seems quite fast

    Thanks

  • Yes - that looks good to me! Glad you got it working!

    Just so you know I added a Bangle.js app that allows you to add custom icons : https://banglejs.com/apps/?id=bthome

  • Post a reply
    • Bold
    • Italics
    • Link
    • Image
    • List
    • Quote
    • code
    • Preview
    • $ Donate
About

Using new BTHome library

Posted by Avatar for user156756 @user156756

Actions