javascript - 使用 chrome.bluetooth API 连接到设备

标签 javascript bluetooth bluetooth-lowenergy google-chrome-app

我一直在尝试创建一个使用 chrome.bluetooth API 的 Chrome 应用程序连接到 Texas Instruments CC2541 SensorTag 设备并与之通信。

这里的代码检测到SensorTag并获取设备信息,但是在设备上调用的'getProfiles'和'getServices'方法都返回空,'connect'方法报错'Profile not found: invalid uuid' .

我尝试了从 example SensorTag Android app 中获取的多种 UUID 变体(如代码中所示),但都给出相同的“无效 uuid”错误。

即使您无法解决这个特殊问题,也很高兴听到任何使用过 chrome.bluetooth API 的人的反馈。到目前为止,我的经验是它的移动目标太多而无法真正使用(是的,我知道它只是“开发”......),但如果可能的话我真的很想让它工作。

感谢您的关注 - 非常感谢任何帮助或想法!

编辑:附加平台信息
我首先尝试在带有 CSR 4.0 蓝牙加密狗的 Windows 7 上运行此程序,但事实证明这完全是徒劳的:使用通用的 Windows 7 BT 驱动程序,Chrome 可以看到适配器并检测 BT 设备,但该驱动程序不支持低功耗,所以无法检测到我想要的设备。使用支持 LE 并且我可以在 Windows“蓝牙设备”中使用它来连接到 LE 设备的 CSR 驱动程序,Chrome.bluetooth 根本无法检测到蓝牙适配器。

现在我正在使用 Acer C720 Chromebook,它看起来应该可以工作,但无论我尝试什么,我都收到“无效的 UUID”消息。

(尽管 Chrome OS 和 Win/Mac/Linux 的“Dev”版本的 Chrome 与更新不同步 - Chrome OS 落后于其他操作系统一段时间但现在已经 catch 了 - 所以有一段时间需要不同的格式' manifest.json' 文件以在不同平台上启动应用程序。)

ma​​in.js

// I have tried multiple variations of known UUIDS for device... all give "Profile not found: invalid uuid" 
var profiles = [
             // UUID_IRT_SERV from example Android app
                {uuid : 'f000aa00-0451-4000-b000-000000000000', name : 'SensorTag1-lc'},
                {uuid : 'F000AA00-0451-4000-B000-000000000000', name : 'SensorTag1-uc'},
                {uuid : 'f000aa00', name : 'SensorTag1-shortlc'},
                {uuid : 'F000AA00', name : 'SensorTag1-shortuc'},
             // UUID_IRT_DATA from example Android app
                {uuid : 'f000aa01-0451-4000-b000-000000000000', name : 'SensorTag2-lc'},
                {uuid : 'F000AA01-0451-4000-B000-000000000000', name : 'SensorTag2-uc'},
                {uuid : 'f000aa01', name : 'SensorTag2-shortlc'},
                {uuid : 'F000AA01', name : 'SensorTag2-shortuc'},
             // UUID_IRT_CONF from example Android app
                {uuid : 'f000aa02-0451-4000-b000-000000000000', name : 'SensorTag3-lc'},
                {uuid : 'F000AA02-0451-4000-B000-000000000000', name : 'SensorTag3-uc'},
                {uuid : 'f000aa02', name : 'SensorTag3-shortlc'},
                {uuid : 'F000AA02', name : 'SensorTag3-shortuc'},
             // UUID_IRT_PERI from example Android app
                {uuid : 'f000aa03-0451-4000-b000-000000000000', name : 'SensorTag4-lc'},
                {uuid : 'F000AA03-0451-4000-B000-000000000000', name : 'SensorTag4-uc'},
                {uuid : 'f000aa03', name : 'SensorTag4-shortlc'},
                {uuid : 'F000AA03', name : 'SensorTag4-shortuc'},
             // UUID_KEY_SERV from example Android app
                {uuid : '0000ffe0-0000-1000-8000-00805f9b34fb', name : 'SensorTag5-lc'},
                {uuid : '0000FFE0-0000-1000-8000-00805F9B34FB', name : 'SensorTag5-uc'},
                {uuid : '0000ffe0', name : 'SensorTag5-shortlc'},
                {uuid : '0000FFE0', name : 'SensorTag5-shortuc'},
             // UUID_KEY_DATA from example Android app
                {uuid : '0000ffe1-0000-1000-8000-00805f9b34fb', name : 'SensorTag6-lc'},
                {uuid : '0000FFE1-0000-1000-8000-00805F9B34FB', name : 'SensorTag6-uc'},
                {uuid : '0000ffe1', name : 'SensorTag6-shortlc'},
                {uuid : '0000FFE1', name : 'SensorTag6-shortuc'},
];

// Listener to deal with initial connection
chrome.bluetooth.onConnection.addListener(onConnected);

// onAdapterStateChanged callback - for debug only
chrome.bluetooth.onAdapterStateChanged.addListener(function(newStatus) {
    log('onAdapterStateChanged: ' + JSON.stringify(arguments));
});

// Logs debug messages to app window
function log(msg) {
  var msg_str = (typeof(msg) == 'object') ? JSON.stringify(msg) : msg;
  console.log(msg_str);
  var l = document.getElementById('log');
  if (l) {
    l.innerText += msg_str + '\n';
  }
}

// Function that is called on connection to device
var onConnected = function(socket) {
    log("Success - Connected to SensorTag!");
    log("Socket: " + JSON.stringify(socket));
}

function recordDevice(device) {
    log("Found BT Device: " + JSON.stringify(device));
    if (device.name == "SensorTag") {
        log("Got SensorTag...");
        // Stop discovery and then connect to SensorTag
        chrome.bluetooth.stopDiscovery(connectToSensorTag(device));
    }
}

function connectToSensorTag(device) {
    log("Getting profiles of SensorTag...");
    chrome.bluetooth.getProfiles({device: device}, function(profiles) { 
        log('Got profiles: ' + JSON.stringify(profiles));
    });
    chrome.bluetooth.getServices({deviceAddress: device.address}, function(services) { 
        log('Got services: ' + JSON.stringify(services));
    });
    for (var i = 0; i < profiles.length; i++) {
        chrome.bluetooth.connect({ device: device, profile: profiles[i] }, function() {
            log('Connect called: ' + JSON.stringify(arguments));
            if (chrome.runtime.lastError) {
                  log("Connection error: " + chrome.runtime.lastError.message);
            }
        });
    }
}

function findDevices() {
    log("Finding devices...");
    chrome.bluetooth.startDiscovery({deviceCallback: recordDevice});
}

// App execution begins here.
// Add all profiles to try connection later
for (var i = 0; i < profiles.length; i++) {
    log("Adding profile: " + profiles[i]);
    chrome.bluetooth.addProfile(profiles[i], function() {
        log("SensorTag profile added."));
    });
}

chrome.bluetooth.getAdapterState( function(result) {
      if (result.powered == false || result.available == false ) {
        log("Error: No bluetooth adapter available.");
      } else {
        log("Bluetooth adapter enabled.");
        findDevices();
      }   
});

index.html

<html>
<head></head>
<body>
    <div id="log"></div>
</body>
<script src="main.js"></script>
</html>

list .json

{
  "manifest_version": 2,
  "name": "Connect to SensorTag",
  "description": "Connects to TI SensorTag",
  "version": "1.0",
  "minimum_chrome_version": "30",
  "app": {
    "background": {
      "scripts": ["background.js"]
    }
  },
  "bluetooth": {}
}

background.js

chrome.app.runtime.onLaunched.addListener(function() {
  chrome.app.window.create('index.html', {
    id: "window1",
    bounds: {
      width: 640,
      height: 480
    }
  });
});

最佳答案

我遇到了同样的问题。经过相当多的挖掘后,我能够确定我使用的模式的两个问题。

第一个问题是,据我所知, list 中的空“蓝牙”对象实际上无效,尽管周围有使用它的示例并且 chrome 没有提示:

"bluetooth": {}

您的蓝牙 list 条目应该如下所示:

"bluetooth":{
"profiles":
    [
        "00001101-0000-1000-8000-00805f9b34fb" // array of uuids
    ]
},

可以在两周前的 chrome commit 中找到相关文档: https://codereview.chromium.org/145663004/patch/760001/770016

我在跟踪此问题时发现的此 API 的另一个特点是,不仅连接调用会在“chrome.runtime.lastError”中留下错误,这些调用中的任何一个都可能会为您留下消息。如果您在调用 addProfile 后检查 lastError,您可能会看到 chrome 提示权限:“添加配置文件的权限被拒绝。”

回去阅读 chrome.bluetooth api 文档,它实际上在靠近顶部的描述中说了这一点(尽管我和你一样错过了它)

Use the chrome.bluetooth API to connect to a Bluetooth device. All functions report failures via chrome.runtime.lastError.

(强调)http://developer.chrome.com/apps/bluetooth#type-Profile

希望这对您有所帮助。

关于javascript - 使用 chrome.bluetooth API 连接到设备,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21940200/

相关文章:

javascript - 有条件地在javascript代码中隐藏一段json

java - 如何使用 HandlerThread 类将单独的循环器传递给线程

android - 在应用程序退出时关闭 bt/gps/wifi 是否明智?

c# - Windows Phone 8.1 中的信标 - 没有可能性?

javascript - 处理为 P5 — 如何转换?

javascript - 通过基于 Javascript 中第一个数组的元素分组来合并两个数组

javascript - 从 wav 文件中读取样本

c# - 32feet蓝牙客户端连接失败

ios - 接收iOS系统(例如短信)通知

尝试编写特征时出现 Android BLE NullPointerException