原文:An Introduction To WebBluetooth
作者:Niels 发表时间:february 13, 2019
译者:西楼听雨 发表时间: 2019/02/24 (转载请注明出处)

这里省略一段开篇介绍,太长,不是什么干货,直接跳过不翻译了,想看的读者可以前往原文查看

WebBluetooth is a new specification that has been implemented in Chrome and Samsung Internet that allows us to communicate directly to Bluetooth Low Energy devices from the browser. Progressive Web Apps in combination with WebBluetooth offer the security and convenience of a web application with the power to directly talk to devices.

WebBlutetooth(Web 蓝牙)是一项已经在 Chrome 和 Samsung Internet (三星浏览器) 中被实现的新规范,它可以让我们直接在浏览器中与低功耗蓝牙设备进行通讯。渐进式网页应用配合 WebBluetooth 为可以直接与设备进行通讯的网页应用提供了安全保障和便利。

Bluetooth has a pretty bad name due to limited range, bad audio quality, and pairing problems. But, pretty much all those problems are a thing of the past. Bluetooth Low Energy is a modern specification that has little to do with the old Bluetooth specifications, apart from using the same frequency spectrum. More than 10 million devices ship with Bluetooth support every single day. That includes computers and phones, but also a variety of devices like heart rate and glucose monitors, IoT devices like light bulbs and toys like remote controllable cars and drones.

蓝牙由于它有限的输入距离、较差的音频质量以及配对问题,背负了一个不好的名声。但其实,所有这些问题都已经成为了过去。Bluetooth Low Energy (低功耗蓝牙) 是一项与以往的蓝牙规范没有什么关系的现代化的规范——除了都使用了同样的频段以外。每天会有超过千万的设备配备了蓝牙,这些设备不仅包括了手机和电脑,还包括了各种各样的如心率、血糖监视器,还有物联网设备如灯泡,玩具如遥控车、飞行器等。

枯燥的理论部分

Since Bluetooth itself is not a web technology, it uses some vocabulary that may seem unfamiliar to us. So let’s go over how Bluetooth works and some of the terminology.

由于蓝牙本身并不是一项 Web 技术,它会用到一些对我们来说可能并不熟悉的词汇。所以接下来我们就来看一下它是怎么工作的以及它的一些术语。

Every Bluetooth device is either a ‘Central device’ or a ‘Peripheral’. Only central devices can initiate communication and can only talk to peripherals. An example of a central device would be a computer or a mobile phone.

每个蓝牙设备,要么是“中心设备”,要么是“外围设备”。只有中心设备才可以发起通讯,而且只能与外围设备进行通讯。电脑和手机就是中心设备的一个例子。

A peripheral cannot initiate communication and can only talk to a central device. Furthermore, a peripheral can only talk to one central device at the same time. A peripheral cannot talk to another peripheral.

外围设备是不能发起通讯的,也只能与中心设备进行通讯;而且,外围设备在同一时间只能与一个中心设备通讯。外围设备不能与另一个外围设备进行通讯。

A central device can talk to multiple peripherals at the same time and could relay messages if it wanted to. So a heart rate monitor could not talk to your lightbulbs, however, you could write a program that runs on a central device that receives your heart rate and turns the lights red if the heart rate gets above a certain threshold.

中心设备可以与多个外围设备同时通讯,也可以对消息进行中继。所以,虽然心率监控器不能与你的灯泡通讯,但是,你可以编写一个运行在中心设备上的程序,让他接收你的心率并在心率达到特定阈值时将灯光变红。

When we talk about WebBluetooth, we are talking about a specific part of the Bluetooth specification called Generic Attribute Profile, which has the very obvious abbreviation GATT. (Apparently, GAP was already taken.)

当我们在谈论 WebBluetooth 时,其实我们谈论的是蓝牙规范中的一个特定的叫做 Generic Attribute Profile (通用属性协议——译注) 的部分,简称 GATT (貌似是因为 GAP 已经被占用而这样简称)

In the context of GATT, we are no longer talking about central devices and peripherals, but clients and servers. Your light bulbs are servers. That may seem counter-intuitive, but it actually makes sense if you think about it. The light bulb offers a service, i.e. light. Just like when the browser connects to a server on the Internet, your phone or computer is a client that connects to the GATT server in the light bulb.

在 GATT 的语境下,我们不再称中心设备和外围设备了,而是改称为客户端和服务端。你的灯泡就是服务端,这看上去有点反直觉,但如果你认真思考一下就会发现这实际上是有其意义的。灯泡提供了一项服务,即“光”,就像浏览器连接到服务器一样,你的手机或电脑就是一个连接到了这个灯泡里的 GATT 服务端的客户端。

Each server offers one or more services. Some of those services are officially part of the standard, but you can also define your own. In the case of the heart rate monitor, there is an official service defined in the specification. In case of the light bulb, there is not, and pretty much every manufacturer tries to re-invent the wheel. Every service has one or more characteristics. Each characteristic has a value that can be read or written. For now, it would be best to think of it as an array of objects, with each object having properties that have values.

每个服务端可以提供一项或多项服务。这些服务,有些是属于官方标准的一部分,但你也可以定义属于你自己的服务。对于心率监视器来说,已经有一项官方的服务在规范中存在了;而对于灯泡来说,还没有,所以几乎所有厂商都会尝试“重复造轮子”。每项服务又有一个或多个特性(characteristic)。每项特性都有一个可以被读写的值。在现在来看,把它想象成一个对象数组最好理解,每个对象都有自己的属性和值。

Unlike properties of objects, the services and characteristics are not identified by a string. Each service and characteristic has a unique UUID which can be 16 or 128 bits long. Officially, the 16 bit UUID is reserved for official standards, but pretty much nobody follows that rule. Finally, every value is an array of bytes. There are no fancy data types in Bluetooth.

和对象的属性不一样,服务项和特性不是用字符串来标识的。每项服务和每个特性都有一个 16 或 128 位比特长的唯一的 UUID。官方规定,16 比特的 UUID 用来保留在各项官方标准上,但几乎没有人遵守这项规定。最后要说的就是,每个特性值都是一个字节数组——在蓝牙中没有所谓的什么数据类型。

近距离观察一个蓝牙灯泡

So let’s look at an actual Bluetooth device: a Mipow Playbulb Sphere. You can use an app like BLE Scanner, or nRF Connect to connect to the device and see all the services and characteristics. In this case, I am using the BLE Scanner app for iOS.

下面我们来看一下一个真实的蓝牙设备:一台 Mipow 牌的灯光球。你可以使用 BLE Scanner 或者 nRF Connect 这类 APP 来连接这台设备并查看它的所有服务项和特性。这里我使用的是 BLE Scanner 应用的 iOS 版。

视频演示地址(需越墙):vimeo.com/303046505

The first thing you see when you connect to the light bulb is a list of services. There are some standardized ones like the device information service and the battery service. But there are also some custom services. I am particularly interested in the service with the 16 bit UUID of 0xff0f. If you open this service, you can see a long list of characteristics. I have no idea what most of these characteristics do, as they are only identified by a UUID and because they are unfortunately a part of a custom service; they are not standardized, and the manufacturer did not provide any documentation.

当你连接到这个灯泡时,第一眼看到的是一个服务项清单。里面有一些是标准化的服务项,如设备信息(device Information)服务项和电池信息服务项;不过也有一些是自定义的服务项。我特别感兴趣的是那项 16 比特长的 UUID 的值为 0xff0f 的服务项。如果你点开这项服务项的话,你会看到一个长长的特性清单;这些特性的大部分我都不知道是什么,因为他们只有 UUID,而且更加遗憾的他们归属于自定义服务项;他们没有被标准化,厂商也没有提供任何文档。

The first characteristic with the UUID of 0xfffc seems particularly interesting. It has a value of four bytes. If we change the value of these bytes from 0x00000000 to 0x00ff0000, the light bulb turns red. Changing it to 0x0000ff00 turns the light bulb green, and 0x000000ff blue. These are RGB colors and correspond exactly to the hex colors we use in HTML and CSS.

第一个特性的 UUID 为 0xfffc,看起来特别有趣,它的值是4个字节,如果我们把这些字节从 0x00000000 改为 0x00ff0000,灯泡就会变红;改为 0x0000ff00 则会变绿,0x000000ff 变蓝。这些都是 RGB 颜色,刚好与我们在 HTML 和 CSS 中使用的十六进制的颜色对应。

What does that first byte do? Well, if we change the value to 0xff000000, the lightbulb turns white. The lightbulb contains four different LEDs, and by changing the value of each of the four bytes, we can create every single color we want.

那么第一个字节是用来干嘛的呢?嗯,如果我们把值改为 0xff000000,灯泡就会变白。灯泡里有四个不同的 LED,通过改变这四个字节的每个的值,我们就可以制作出我们想要的所有颜色。

WebBluetooth API

It is fantastic that we can use a native app to change the color of a light bulb, but how do we do this from the browser? It turns out that with the knowledge about Bluetooth and GATT we just learned, this is relatively simple thanks to the WebBluetooth API. It only takes a couple of lines of JavaScript to change the color of a light bulb.

用本地应用来改变灯泡的颜色是极其可行的,但如果是放在浏览器里面来做,我们该怎么做呢?刚刚我们已经学习了蓝牙和 GATT 相关的知识,借助于 WebBluetooth API 。只需要几行 JS 代码就可以改变灯泡的颜色。

Let’s go over the WebBluetooth API.

下面我们就来看下 WebBluetooth API。

连接到一个设备

The first thing we need to do is to connect from the browser to the device. We call the function navigator.bluetooth.requestDevice()and provide the function with a configuration object. That object contains information about which device we want to use and which services should be available to our API.

我们需要做的第一件事就是,在浏览器中与那台设备进行连接。调用函数 navigator.bluetooth.requestDevice() ,并传入一个配置对象,这个对象包含了关于我们想要使用的设备和服务的信息。

In the following example, we are filtering on the name of the device, as we only want to see devices that contain the prefix PLAYBULB in the name. We are also specifying 0xff0f as a service we want to use. Since the requestDevice() function returns a promise, we can await the result.

在下面这个例子中,我们基于设备的名字进行了筛选,因为我们只希望看到名字中包含了 PLAYBULB 前缀的设备;我们还用 0xff0f 来指定了我们想使用的服务项。由于 requestDevice() 函数返回的是一个 promise,所以我们可以 await 它的结果。

let device = await navigator.bluetooth.requestDevice({filters: [ { namePrefix: 'PLAYBULB' } ],optionalServices: [ 0xff0f ]
});
复制代码

When we call this function, a window pops up with the list of devices that conform to the filters we’ve specified. Now we have to select the device we want to connect to manually. That is an essential step for security and privacy and gives control to the user. The user decides whether the web app is allowed to connect, and of course, to which device it is allowed to connect. The web app cannot get a list of devices or connect without the user manually selecting a device.

当我们调用这个函数时,会弹出一个窗口,里面是一个满足我们所指定的过滤条件的设备清单。然后,我们必须从中选择我们想要连接的设备。这一步骤对于安全和隐私来说是不可或缺的,它把控制权交给了用户。用户决定了网页应用是否可以进行连接,当然,也决定了它所允许进行连接的是哪个设备。没有用户的手动选择,网页应用是不能获取到设备清单的,同样也是无法连接的。

After we get access to the device, we can connect to the GATT server by calling the connect() function on the gatt property of the device and await the result.

在我们获取到这台设备后,我让就可以通过调用这个设备的 gatt 属性上的 connect()` 函数来连接到 GATT 服务端上,并 await 它的结果。

let server = await device.gatt.connect();
复制代码

Once we have the server, we can call getPrimaryService() on the server with the UUID of the service we want to use as a parameter and await the result.

获得服务端后,我们就可以用我们想要使用的服务项的 UUID 作为参数来调用它的 getPrimaryService() ,并 await 其结果。

let service = await server.getPrimaryService(0xff0f);
复制代码

Then call getCharacteristic() on the service with the UUID of the characteristic as a parameter and again await the result.

然后再在服务项上用特性的 UUID 作为参数来调用 getCharacteristic() ,然后继续 await 其结果。

We now have our characteristics which we can use to write and read data:

然后得到了我们的特性之后,我们就可以用它来读写数据了:

let characteristic = await service.getCharacteristic(0xfffc);
复制代码

写入数据

To write data, we can call the function writeValue() on the characteristic with the value we want to write as an ArrayBuffer, which is a storage method for binary data. The reason we cannot use a regular array is that regular arrays can contain data of various types and can even have empty holes.

想要写入数据,我们可以把我们想要写入的值作为一个 ArrayBuffer 来在特性上调用 writeValue() 函数——ArrayBuffer 是一种二进制数据的存储方式。我们不使用常规数组的原因是数组可以包含任意类型的数据,而且甚至可能存在“空洞”。

Since we cannot create or modify an ArrayBuffer directly, we are using a ‘typed array’ instead. Every element of a typed array is always the same type, and it does not have any holes. In our case, we are going to use a Uint8Array, which is unsigned so it cannot contain any negative numbers; an integer, so it cannot contain fractions; and it is 8 bits and can contain only values from 0 to 255. In other words: an array of bytes.

由于我们不能直接创建和修改 ArrayBuffer,我们需要改用“typed array” (类型化数组) 来实现——Typed Array 中的所有元素都是相同的类型,也没有任何“空洞”。在我们的这个例子中,我们将使用的是 Unit8Array ,它是无符号的整型,所以不会包含任何负数和小数部分;同时他还是 8 比特长的,所以只能包含 0~255。换言之:它就是一个字节数组。

characteristic.writeValue(new Uint8Array([ 0, r, g, b  ])
);
复制代码

We already know how this particular light bulb works. We have to provide four bytes, one for each LED. Each byte has a value between 0 and 255, and in this case, we only want to use the red, green and blue LEDs, so we leave the white LED off, by using the value 0.

我们已经知道这个灯泡是如何工作的了。我们需要提供四个字节,对应到各个 LED。每个字节的值,范围为 0~255,在这个例子中,我们想要使用到的只有红、绿、蓝 LED ,所以我们通过使用 0 来保持白色 LED 关闭。

读取数据

To read the current color of the light bulb, we can use the readValue() function and await the result.

我们可以使用 readValue() 函数来读取灯泡当前的颜色,并 await 它的结果。

let value = await characteristic.readValue();let r = value.getUint8(1);
let g = value.getUint8(2);
let b = value.getUint8(3);
复制代码

The value we get back is a DataView of an ArrayBuffer, and it offers a way to get the data out of the ArrayBuffer. In our case, we can use the getUint8() function with an index as a parameter to pull out the individual bytes from the array.

我们取回来的值是一个 ArrayBuffer 的 DataView (数据视图),它提供了一种从 ArrayBuffer 取出数据的方式。在我们的例子中,我们可以通过将一个下标作为参数来使用 getUint8() 函数拉取单个字节。

监听变动

Finally, there is also a way to get notified when the value of a device changes. That isn’t really useful for a lightbulb, but for our heart rate monitor we have constantly changing values, and we don’t want to poll the current value manually every single second.

最后,还有一种方式是在设备的值发生变动了获得通知。对于灯泡来说,这个其实真的没什么用,但对于我们的心率监视器来说,它的值是持续不断变化的,我们不希望手动每秒来获取当前的值。

characteristic.addEventListener('characteristicvaluechanged', e => {let r = e.target.value.getUint8(1); let g = e.target.value.getUint8(2);let b = e.target.value.getUint8(3);}
);characteristic.startNotifications();
复制代码

To get a callback whenever a value changes, we have to call the addEventListener() function on the characteristic with the parameter characteristicvaluechanged and a callback function. Whenever the value changes, the callback function will be called with an event object as a parameter, and we can get the data from the value property of the target of the event. And, finally extract the individual bytes again from the DataView of the ArrayBuffer.

要想在值发生变动时获得回调,我们需要在特性上调用 addEventListener() 函数——使用 characteristicvaluechanged 和一个回调函数作为参数。这样,在值发生变动时,回调函数就会被调用,并接受到一个 event 对象,我们可以从这个 event 的 target 属性的 value 属性上获得数据,然后再通过 ArrayBuffer 的 DataView 提取各个字节。

Because the bandwidth on the Bluetooth network is limited, we have to manually start this notification mechanism by calling startNotifications() on the characteristic. Otherwise, the network is going to be flooded by unnecessary data. Furthermore, because these devices typically use a battery, every single byte that we do not have to send will definitively improve the battery life of the device because the internal radio does not need to be turned on as often.

由于蓝牙网络的带宽有限,我们必须手动调用 startNotifications() 来启动通知机制;否则,网络中就会充斥着没必要的数据。然后,由于这些设备通常会用到一个电池,所以每节省一个没必要发送的字节,都可以提升设备的电池续航,因为没必要经常性地打开内部的射频信号。

总结

We’ve now gone over 90% of the WebBluetooth API. With just a few function calls and sending 4 bytes, you can create a web app that controls the colors of your light bulbs. If you add a few more lines, you can even control a toy car or fly a drone. With more and more Bluetooth devices making their way on to the market, the possibilities are endless.

我们已经对 WebBluetooth API 做了 90% 的讲解了。只需调用几个函数,发送4个字节,你就可以创建一个能控制你灯泡颜色的网页应用。如果再多写几行代码,你甚至可以控制一台玩具车或者飞起一台飞行器。随着越来越多的蓝牙设备不断地进入市场,未来将有无限的可能。

视频演示地址(需越墙):vimeo.com/303045191

(这个视频里演示了通过网页来控制彩灯、LED 面板、玩具车、飞行器等——译注)

扩展资源

  • Bluetooth.rocks! Demos (以上项目的示例程序)| (Source code on GitHub) (及其源代码)

  • “Web Bluetooth Specification,” Web Bluetooth Community Group 蓝牙规范文档

  • Open GATT Registry 开放性 GATT 登记表 An unofficial collection of documentation for Generic Attribute services for Bluetooth Low Energy devices.

    一份非官方性质的针对微型蓝牙设备的 Generic Attribute 服务文档集合。

[译] 如何在浏览器中编写一款蓝牙应用相关推荐

  1. server vscode中的live_太方便了!这款神器能在浏览器中运行 VS Code,随时随地写代码...

    最近看到 iPadOS 出来了,各种牛逼的操作真的很有吸引力,于是咬咬牙买了 iPad air.买来之后,当然是研究怎么提高效率了,于是就寻找.研究各种高效的软件.折腾了一段时间,各种 APP 都找得 ...

  2. 推荐一款神器:在浏览器中运行 vscode,随时随地写代码

    目录 瞎比比 什么都别说,先上图 需要什么配置条件? 如何配置? 关于 vscode 的插件 配置一个 python 开发环境 遇到的错误 瞎比比 最近看到 iPadOS 出来了,各种牛逼的操作真的很 ...

  3. 怎么保存在界面输入的内容_还在担心忘记密码?使用这款软件轻松找回浏览器中保存的密码...

    今天给大家介绍的是一个浏览器密码查看器---webbrowserpassview,这里分享的是简体中文版本的,适合于Windows端,这里测试用的操作系统是win10. 我想大家在使用浏览器的时候经常 ...

  4. 推荐一款神器:在浏览器中运行 VS Code,随时随地写代码

    点击上方蓝色小字,关注"涛哥聊Python" 重磅干货,第一时间送达 目录 发现 什么都别说,先上图 需要什么配置条件? 如何配置? 关于 vscode 的插件 配置一个 pyth ...

  5. 「每周译Go」如何在 Go 中编写 Switch 语句

    目录 在 Go 中导入包 理解 Go 中包的可见性 如何在 Go 中编写条件语句 如何在 Go 中编写 Switch 语句 如何在 Go 中构造 for 循环 在循环中使用 Break 和 Conti ...

  6. 如何使用功能性JavaScript编写经典游戏Snake并在浏览器中播放-完整的代码示例教程

    Remember the game Snake that came pre-installed on every Nokia phone back in the 1990s? You steered ...

  7. notepad++ 编写html代码快捷键切换到浏览器查看,notepad++在chrome浏览器中打开查看网页效果...

    notepad++在chrome浏览器中打开查看网页效果,操作设置:运行--在chrome浏览器中打开,这是notepad默认的设置方式 但是notepad++如果没有配置chrome浏览器打开,可以 ...

  8. JavaScript 编程精解 中文第三版 十三、浏览器中的 JavaScript

    十三.浏览器中的 JavaScript 原文:JavaScript and the Browser 译者:飞龙 协议:CC BY-NC-SA 4.0 自豪地采用谷歌翻译 部分参考了<JavaSc ...

  9. 【译】.NET 7 中的性能改进(十一)

    ▲ 点击上方"DotNet NB"关注公众号 回复"1"获取开发者路线图 学习分享 丨作者 / 郑 子 铭 这是DotNet NB 公众号的第215篇原创文章 ...

最新文章

  1. 8款审核AWS帐户安全性的免费工具,你值得拥有
  2. 查看DLL 及LIB 库导出函数方法
  3. 系统单据号生成规则推荐
  4. java asc码_Java中ASC码与字符互相转化
  5. [转载]SYSCALL_DEFINE宏定义
  6. STM32CubeMX的安装
  7. Git笔记(29) 搜索
  8. 如何评估 Serverless 服务能力?这份报告给出了 40 条标准
  9. Ubuntu 11.10 快捷键 gnome gFTP 服务器 vsftpd 程序 面板
  10. iView学习笔记(四):Form表单操作
  11. node 压缩图片_设计神器!图片在线工具–Online Image Tool
  12. memcached+magent实现负载
  13. javascript类式继承函数最优版
  14. java6可以玩儿我的世界吗_我的世界Java1.16预发行版6下载_我的世界Java1.16预发行版6官方游戏下载 v1.17.30.94571-菜鸟下载...
  15. 主流数据库优缺点以及性能分析
  16. 3DMAX解决Vray渲染材质溢色问题的三种方法
  17. u盘中毒后文件夹被病毒隐藏
  18. 偶极子天线的优缺点_一种双面印刷偶极子天线解析
  19. 微信聊天功能软件测试用例,软件测试用例实例之常见功能测试点
  20. Tomcat-线程模型及设计精髓

热门文章

  1. android os.access,Establishing mandatory access control on Android OS
  2. 利用python构建分子碎片库
  3. (附源码)springboot苔藓植物科普网站 毕业设计 345641
  4. 专题总纲目录 Linux总纲
  5. 什么是PCM?它和.wav文件是什么关系?
  6. Python中列表截取(Slice,即冒号 : )的用法详解
  7. PHP科大讯飞翻译API接口接入
  8. 错误使用 xlswrite 无法激活 Excel 工作表。
  9. xargs使用教程:在实战中学习xargs
  10. i5 12400和i5 11400差距大不大