用Arduino玩转掌控板(ESP32):蓝牙翻页笔(PPT 控制器)-程序员宅基地

技术标签: qt  arduino  物联网  蓝牙  单片机  

众所周知,掌控板在创客教育中用的非常广泛,它是一块基于 ESP32 的学习开发板。大家对掌控板编程,用的比较多的都是图形化编程的方式,比如 mPython、Mind+ 等。但是,既然掌控板是基于 ESP32 芯片的,所以我们也可以用 Arduino 软件对其编程。所以,有时间的话,我准备给大家分享一系列用 Arduino 代码对掌控板(ESP32)编程的教程:用 Arduino 玩转掌控板(ESP32)系列

本系列历史文章目录:

在前面几期中,有读者反馈,因为网络原因,在 Arduino 软件中配置 ESP32 和 ESP8266 开发环境,经常会失败。所以这次我给大家带来了已经配置好开发环境的 Arduino 绿色版软件,解压即用,可直达文末获取下载地址。

本期给大家带来的案例是:蓝牙翻页笔

先来看一下演示视频吧:

# 前言

关于翻页笔,其实我在之前就已经给大家分享过用好搭徽章板和 micro:bit 来实现翻页笔的案例,大家可以跳转下面的连接去查看:

萌宠体感翻页笔DIY教程

这次要设计的翻页笔,功能与之前做的类似,不同点在于:

  • 主控板采用了ESP32掌控板,而且只要一块板子就够了,不需要两块板子配合;

  • 程序采用了 Arduino 代码方式,当然在文末我也会提供图形化代码供参考;

  • 连接方式变成了 BLE 无线蓝牙,可以兼容 Mac、WIndows、iOS、Android 等。

下面开始进入正题。

# ESP32 蓝牙键盘库

编写蓝牙翻页笔程序之前,我们先来了解一下 ESP32-BLE-Keyboard 这个 Arduino 库。我们编写翻页笔程序主要是借助这个库来完成。这个库的功能,主要是将 ESP32 模拟成蓝牙键盘,然后向电脑发送按键命令,比如说我们翻页笔中会用到的:F5 按键(控制 PPT 开始播放)、ESC 按键(控制 PPT 退出播放)、还有上、下翻页按键 Page Up 和 Page Down(或者箭头 ← 和 →)。

使用这个库之前,首先我们要将这个库导入 Arduino 软件中,具体导入方法,这里不展开了,网上随便搜一下就有。

我们打开这个库的头文件 BleKeyboard.h,可以对这个库有一个大概的了解。这个库里面主要有以下几个功能模块:

## BleKeyboard 对象

BleKeyboard 用来声明蓝牙键盘对象,用在程序开头,用来初始化设定蓝牙键盘的名称 deviceName 、键盘生产厂商 deviceManufacturer、键盘电量 batteryLevel。当然你也可以不用设置,这个时候,程序就会用默认值来对键盘进行设置。

BleKeyboard(std::string deviceName = "ESP32 BLE Keyboard", std::string deviceManufacturer = "Espressif", uint8_t batteryLevel = 100);

## begin()  和 end() 函数

这两个函数的定义如下:

void begin(void);
void end(void);

主要是用来启动与结束键盘的。

## press() 函数

在头文件定义中,press() 函数有两个:

size_t press(uint8_t k);
size_t press(const MediaKeyReport k);

这两个 press() 函数,可以用来控制按下键盘上的所有按键,比如 0-9,A-Z,以及一些特殊的功能键等。

注意,这边我写的是按下键盘上的按键,表示并没有松开,这样的好处是,可以控制键盘发送一些组合键。

区别在于前一个是用来控制普通按键的,后一个是用来控制媒体键(MediaKey)的。什么是媒体键呢?比如说控制音乐播放的上一首、下一首、暂停、控制音量的加减等,这些都是媒体键。

实际上我们在使用时不用去区分,程序会根据我们输入的按键,自动去调用相应的函数。

比如我们要按下键盘上的 A 键,只要调用 press(‘a’) 就可以了;如果我们要按下键盘上的 Ctrl 按键,只要调用 press(KEY_LEFT_CTRL) 就可以了。普通按键,类似 A-Z 或者 0-9,我们只要正常输入就可以了,但是 Ctrl 这类特殊按键在程序中的对应名称是什么呢?在这个库的头文件 BleKeyboard.h中都有定义,读者可以自行去查看,这里由于篇幅原因不再赘述。

## release() 函数

release() 正好与 press() 函数相反,是用来控制松开键盘上的按键的,在定义中也有两个:

size_t release(uint8_t k);
size_t release(const MediaKeyReport k);

这两个函数的区别不再赘述,press() 函数和 release() 一般组合起来使用,可以用在需用使用组合键的场合,比如需要使用 Ctrl + C 按键时,可以先按下两个按键,再松开这两个按键。

## releaseAll() 函数

从名字就可以看出,是 release() 函数的加强版,可以一次性松开所有按键。比如我们同时按下 Ctrl + Alt + Delete 按键,可以通过 releaseAll() 函数,一次性松开 3 个按键。

## write() 函数

write() 函数可以简单理解为 press() 函数和 release() 函数的相加,也就是先按下按键,再松开按键,这个过程自动完成,适合用在发送单独按键的场合。

size_t write(uint8_t c);
size_t write(const MediaKeyReport c);
size_t write(const uint8_t *buffer, size_t size);

## isConnected() 函数

这个函数用来判断 ESP32 模拟的蓝牙键盘,是否与电脑已经配对连接。

## print() 函数

除了以上这些函数之外,还有一个 print() 函数,这个函数是继承自 Arduino 自带的库 Print 类的。

这个函数有什么用呢?试想一下,如果我们要一次性发送好几个按键,比如电脑上输入“Hello World!”这句话的时候,难道要调用好几次 write() 函数来把这句话发送完成么?其实不是的,只要通过 print("Hello World!") 这一句话就够了。

OK,有了这些讲解,模拟键盘控制电脑就很简单了,我们来看一下蓝牙翻页笔的程序怎么写吧。

# 翻页笔程序

## 完整程序

这里先把完整程序放上来:

## 引入头文件

我们现在程序开头引入了几个头文件,控制 OLED 显示屏的 U8g2lib 和 Wire 库,模拟键盘的 BleKeyboard 库:

#include <U8g2lib.h>
#include <Wire.h>
#include <BleKeyboard.h>

然后定义了一个键盘对象 bleKeyboard。这里设置键盘的名称为 mPython,键盘厂商为 Espressif,键盘电量为 95%。实际上这些设置都是随意的,主要是键盘名称,只要方便记忆就好了。

BleKeyboard bleKeyboard("mPython", "Espressif", 95);

接着定义了一个 OLED 屏对象 u8g2,OLED 屏幕的功能主要是提示我们控制 PPT 时按了哪个按键。这里需要根据你使用的 OLED 屏幕类型和连接方式来定义,先不展开讲,后面有时间再专门讲一讲屏幕显示的问题。掌控版自带的 OLED 屏是通过 I2C 连接,只要通过下面的语句定义即可。

U8G2_SH1106_128X64_NONAME_F_HW_I2C u8g2(U8G2_R0, U8X8_PIN_NONE);

## setup() 初始化

然后在 setup() 初始化中,初始化串口 Serial 方便后面可以打印调试信息;初始化 OLED 屏并进行相应的设置;初始化蓝牙键盘。

void setup() {
  Serial.begin(115200);


  // OLED 初始化
  u8g2.begin();
  u8g2.enableUTF8Print();
  u8g2.setFont(u8g2_font_wqy16_t_gb2312b);// 设置字体:宋体16号字
  u8g2.setFontPosTop();// 设置字体的定位点为左上角
  u8g2.clearDisplay();


  // 蓝牙键盘初始化
  Serial.println("Starting BLE Keyboard!");
  bleKeyboard.begin();
}

## loop() 主程序

在 loop() 循环中,通过 bleKeyboard.isConnected() 来判断我们的键盘有没有与电脑连接上了,如果没有连接,写在这里面的程序是不会执行的,只有在电脑上我们与这个蓝牙键盘配对连接之后,里面的程序才会执行。

void loop() {
  if (bleKeyboard.isConnected()) {
      // codes here...
  }
}

下面的程序,用来读取掌控版上的按键 A 和 B,以及触摸按键 P、Y、T、H、O、N 的值,通过读取他们的值,去判断按键有没有被按下触发。

// button A and B
int buttonA = !digitalRead(0);
int buttonB = !digitalRead(2);


// touchpad P Y T H O N
int touchpadP = touchRead(27);
int touchpadY = touchRead(14);
int touchpadT = touchRead(12);
int touchpadH = touchRead(13);
int touchpadO = touchRead(15);
int touchpadN = touchRead(4);

掌控板上的按键 A 和 B 是数字信号,而且默认情况下高电平信号,按下按键,通过 digitalRead() 函数会读取到低电平。所以前面加了个非(!),方便后面判断。

掌控板或 ESP32 自带了几个触摸按键,所以系统也自带了检测触摸按键的函数 touchRead() ,触摸按键读取的值是一个模拟量。

至于这几个按键的引脚编号为什么是这些呢?也就是 digitalRead() 和 touchRead() 函数里面的数字,为什么是这些呢?我们来看一下掌控板的引脚定义图。

掌控板引脚定义

先来看一下上图中的“引脚定义正面”这张图,我们看到触摸引脚 P 对饮的编号是 P23。再来看一下芯片引脚定义,P23 对应的芯片引脚是 IO27。在 Arduino 程序中,我们实际上编写的是芯片引脚,所以读取触摸引脚 P 的值时,我们在 touchRead() 函数中的数字是 27。其他几个触摸引脚对应的芯片引脚 IO 口,读者可以自己对应查一下。

读者也可以查询掌控板的官方文档,获取掌控板引脚定义:
https://mpython.readthedocs.io/zh/master/board/hardware.html
如果使用的是其他 ESP32 开发板的话,也可以查询相应的官方产品资料,获取对应的引脚编号。

然后通过判断按键 A 和 B 的值,控制 PPT 开始播放与结束播放。控制 PPT 播放,我们只要按 F5 键即可以,所以调用的程序是 bleKeyboard.write(KEY_F5);结束 PPT 播放,只要按 ESC 键即可以,所以调用的程序是bleKeyboard.write(KEY_ESC)。注意,由于是程序控制键盘,所以加了 delay(500) 程序,以防止按键按得太快。

// Start PPT
if (buttonA) {
  Serial.println("Button A pressed!");
  Serial.println("Sending key: F5 (start PPT");
  display("开始播放");


  bleKeyboard.write(KEY_F5);


  delay(500);
}


// Quit PPT
if (buttonB) {
  Serial.println("Button B pressed!");
  Serial.println("Sending key: ESC (quit PPT)");
  display("结束播放");


  bleKeyboard.write(KEY_ESC);


  delay(500);
}

里面的 display() 函数后面再讲,是用来控制掌控板的 OLED 显示屏上显示相应的提示语的,比如“开始播放”、“结束播放”等。

同样道理,我们可以通过检测触摸按键 P 和 N 的值,来控制 PPT 播放上一页和下一页,这里分别用 bleKeyboard.write(KEY_LEFT_ARROW)bleKeyboard.write(KEY_RIGHT_ARROW) 来控制。

// Previous page
if (touchpadP < 20 && touchpadP > 0) {
  Serial.println("Touchpad P touched!");
  Serial.println("Sending key: <-- (previous page)");
  display("上一页");


  bleKeyboard.write(KEY_LEFT_ARROW);


  delay(500);
}


// Next page
if (touchpadN < 20 && touchpadN > 0) {
  Serial.println("Touchpad N touched!");
  Serial.println("Sending key: --> (next page)");
  display("下一页");


  bleKeyboard.write(KEY_RIGHT_ARROW);


  delay(500);
}

至此,主程序也讲解完成了,逻辑是不是很简单,只要通过判断掌控板上哪个按键被按下了,模拟键盘触发相应的按键行为就好了。

## display() 函数

在程序中,我们还定义了另外一个函数 display(String tipsOnScreen),它的功能是根据掌控板上的按键,用来控制 OLED 显示相应的内容的,直接看代码就行。

void display(String tipsOnScreen) 
{
  u8g2.firstPage();
  do 
  {
   u8g2.clearDisplay();
   u8g2.setCursor(32, 24);
   u8g2.print(tipsOnScreen);
  } while (u8g2.nextPage());
}

至此,程序讲解完成,上传到掌控板即可。

# 连接蓝牙

上面我们通过编写程序已经将掌控板模拟成了蓝牙键盘,接下来只要与电脑进行配对,就可以愉快的使用啦。

这里我以 Windows 10 系统为例,其他系统类似。首先打开电脑的设置,找到添加蓝牙设备页面,点击最上面的“+”号,准备添加蓝牙设备。

打开蓝牙设置

选择“蓝牙”,开始搜寻电脑附近的蓝牙设备。

添加蓝牙设备

这个时候就会出现一个名称为“mPython”的蓝牙设备,“mPython”是我们在程序中给掌控板起的名字,这个边上有一个键盘的图标,说明掌控板已经被电脑识别为蓝牙键盘了,选择它与电脑进行配对连接即可。

选择蓝牙设备

连接成功后,我们就可以在蓝牙设置页面看到刚刚添加的蓝牙键盘“mPython”了,再看一看它的电量,95%,跟程序里设置的一模一样!

添加成功

这个时候,你就可以在电脑中打开一个 PPT,试试看你的翻页笔好不好使了~

# 彩蛋

看到这里的朋友,为了感谢你,所以特意留了一个彩蛋。

相信好多朋友看到代码就头大,所以我给大家做了一个蓝牙键盘的图形库,大家只要打开 Mind+ 软件 v1.6.2 及以上的版本,在用户库中加入我的蓝牙键盘库,就可以开始蓝牙键盘编程之旅啦。

蓝牙键盘库直达网址:

https://gitee.com/ironpanda/ext-BLEKeyboard

这上面有这个蓝牙键盘库的使用教程和简单案例。

将上述网址添加到 Mind+ 软件用户库中,如下图所示:

导入蓝牙键盘库

导入完成后,就可以在“用户库”分类中看到蓝牙键盘库相应的编程模块了。

蓝牙键盘库

本文对应的蓝牙键盘 Mind+ 图形程序如下:

蓝牙翻页笔Mind+程序

是不是很简单?你会了么?学会了的话,就点个“在看”吧~

# 代码下载

欢迎关注公众号:铁熊玩创客,不定期更新创客制作、技术教程、创客教育等相关内容。

  • 回复“蓝牙翻页笔”获取完整代码,包含 Arduino C 语言代码和 Mind+ 图形代码;

  • 回复“Arduino”获取绿色版 Arduino 软件下载地址,已为您配置好掌控板 ESP32 和 ESP8266 开发环境


*欢迎转发朋友圈。如需转载,请注明出处和原作者。

往期精彩内容

创客项目缺少高颜值电路图?看这里就对了

小白也能学会的激光切割创意盒子设计方法

还在羡慕大疆 S1?教你自制麦轮战车

萌宠Pando机器人,不但能卖萌,还能跳太空步

学生获奖作品:戒烟帽

学生获奖作品:体感转向安全帽

掌控+Mixly+Blynk,让你的麦轮战甲嗨炸全场

搞定掌控板Siri语音控制,只要半小时

我知道你在看

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/ironpandaaa/article/details/118081139

智能推荐

VUE{{}}中数据渲染出现undefined_vue undefined-程序员宅基地

文章浏览阅读2.5k次。VUE{{}}中数据渲染出现undefinedBackground在前端编码过程中,常常会出现在mustache表达式 {{}} 中undefine的某属性报错。原因是如果 {{}} 仅仅是个变量如 {{a}} 是不会报错的,a可以是undefine或者null。但是如果{{a.b}} 或者 {{a['b']}} ,那么当a是undefine的时候就会报错。并且因为Ajax是异步加载所以常常出现加载数据之前报出这个错误。Solution利用在标签上面添加v-if:例如: <div v-if_vue undefined

设计模式课程-高琪-专题视频课程-程序员宅基地

文章浏览阅读254次。设计模式是一套被反复使用、多数人知晓的、经过分类编目的、代码设计经验的总结!_设计模式 高琪

Java锁比较全的入门整理_java锁入门-程序员宅基地

文章浏览阅读485次。一、锁的7大分类需要首先指出的是,这些多种多样的分类,是评价一个事物的多种标准,比如评价一个城市,标准有人口多少、经济发达与否、城市面积大小等。而一个城市可能同时占据多个标准,以北京而言,人口多,经济发达,同时城市面积还很大。同理,对于 Java 中的锁而言,一把锁也有可能同时占有多个标准,符合多种分类,比如 ReentrantLock 既是可中断锁,又是可重入锁。根据分类标准我们把锁分为以下 7 大类别,分别是:偏向锁/轻量级锁/重量级锁;可重入锁/非可重入锁;共享锁/独占锁;公平锁/非公_java锁入门

golang,Python,我们该怎么选_golang python 数据分析-程序员宅基地

文章浏览阅读404次。在这些编程语言中,Go语言和Python语言都属于高级编程语言,它们在不同领域和场景下有着广泛的应用。本文将从以下几个方面对Go语言和Python进行优缺点对比:速度、并发处理、强类型、学习曲线、库支持、多用途性、社区支持、可移植性、性能调优等。综上所述,Golang和Python都有自己独特的优缺点,并且适用于不同的场景。相反,Python属于解释型语言,因此它的性能调优需要使用一些特殊的技巧和工具。在选择一种编程语言时,不仅要考虑它的优缺点,还要考虑其他因素,如可维护性、团队成员技能、部署效率等。_golang python 数据分析

字符串匹配之后缀树_后缀树 字符串匹配-程序员宅基地

文章浏览阅读1.8k次。引言试想有这样一个问题,有一个长度为N的字符串A(N值很大),还有一个模式串B,B的长度为M(N/M很大,说明B只是一个小片段),此时需要判断B是否是A的字串。如果我们使用KMP算法的话,那么复杂度为O(N),对A串进行K次模式匹配的话就是KO(N),此时为了降低复杂度,我们可以考虑预处理长字符串A,是的,如果我们预先处理好A的后缀树,那么搜索子串的复杂度就降为O(M),进行K次匹配为KO(M_后缀树 字符串匹配

ubuntu 16.04 + Qt 5.7.1 无法输入中文的解决方法(转)_qt5.7.1无法输入中文-程序员宅基地

文章浏览阅读186次。转自 https://blog.csdn.net/lql0716/article/details/54973160author@jason_ql http://blog.csdn.net/lql0716安装fcitx-frontend-qt5sudo apt-get install fcitx-frontend-qt5找到以下路径的文件/usr/lib/x86_64-linux-gnu/qt5/p..._qt5.7.1无法输入中文

随便推点

如何为微信小程序添加定位导航和地图标注功能_小程序地图导航-程序员宅基地

文章浏览阅读1.3k次。在JavaScript代码中,我们需要将获取到的用户位置信息设置为map组件的latitude和longitude属性,并创建一个标注点并设置在markers属性中。在获取到用户的地理位置信息后,我们需要将用户的位置在地图上进行标注。在JavaScript代码中,我们需要在数据中添加路线规划相关信息,并在路线规划成功的回调函数中更新数据。同样的,我们在实现地图标注功能时,也需要获取用户的地理位置信息。至此,我们已经成功地实现了获取用户地理位置信息并在地图上标注出用户位置的功能。二、地图标注功能的实现。_小程序地图导航

DML DDL DCL TCL_dml is used to: 数据操纵语言用于: a. add /modify/delete da-程序员宅基地

文章浏览阅读288次。接着上篇说起,数据库操作语句,看了网上说的,有很多种说法,我现在总结的为DML DDL DCL TCL.DML (Data Manipulation Language)数据操纵语言statements are used for managing data within schema objects.由DBMS提供,用于让用户或程序员使用,实现对数据库中数据的操作。DML分成交互型DML和_dml is used to: 数据操纵语言用于: a. add /modify/delete data in the table b.

Hive安装及启动异常解决_启动hive session id一直-程序员宅基地

文章浏览阅读5.8k次,点赞4次,收藏19次。前期准备1、关于Hive的安装包和文档可以从这里获取:Hive官方文档Hive下载地址Hive源码地址2、Hive依赖于Hadoop,关于Hadoop的安装可以查看这里:Hadoop安装——启动模式,基本配置,启动方式Hadoop集群搭建——环境配置3、安装mysql由于Hive需要在数据库中存储元数据信息,所以安装hive之前需要先安装mysql。hive有一个默认的存放在内存中的数据库,企业开发_启动hive session id一直

在Qt designer中实现按钮的圆角设置_qt designer border-radius:10px;-程序员宅基地

文章浏览阅读9.2k次,点赞12次,收藏40次。放入按钮按键,在StyleSheet中设置:QPushButton{background-color: rgb(170, 170, 255);#背景颜色 color: white; #字体颜色 border-radius: 10px; border: 2px groove gray;#圆角设置 border-style: outset;#..._qt designer border-radius:10px;

ssh远程连接出现 Permission denied, please try again问题_permission denied, please try again.-程序员宅基地

ssh远程连接出现Permission denied问题的解决方法是:检查是否安装了SSH服务,如果已安装,需要打开配置文件并将PermitRootLogin设置为yes,然后重启SSH服务。

LRNNet :上海交大最新提出0.68M超轻量实时语义分割模型,简化Non-local计算量-程序员宅基地

文章浏览阅读2.8k次。点击上方“AI算法修炼营”,选择“星标”公众号精选作品,第一时间送达论文地址:https://arxiv.org/pdf/2006.02706.pdf本文是上海交通大学团队提出的轻量级实..._lrnnet

推荐文章

热门文章

相关标签