0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看威廉希尔官方网站 视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

利用ROHM传感器评估套件实现UCLA AirMouse–第2部分

刘敏 来源:juanll5 作者:juanll5 2023-02-27 10:27 次阅读

这篇文章来源于DevicePlus.com英语网站的翻译稿。

在利用ROHM传感器评估套件实现UCLA AirMouse – 第1部分中,我们完成了项目的硬件。对于发射器部分,我们将AirMouse按钮以及Uno的GPIO引脚和RF模块之间的接口组装在一块面包板上。由于ROHM传感器开发板(SensorShield)在原型设计和DIY项目方面的应用非常方便,因此我们选择将其与加速度计模块接合在一起使用。正如您将在本教程中看到的那样,ROHM开发板内置Arduino与其外设之间的I2C通信功能,因此,用户通过一些简单代码就可以接收加速度计的数据,无需编写任何底层I2C函数来发送和接收来自设备地址的数据。对于接收器部分,我们为Teensy微控制器组装了类似的分线板,以便与RF模块对接。

本教程介绍和描述的代码将帮您将两个模块连接在一起,以完成该项目。我们将向您展示两个模块之间发送数据的基本代码以及处理加速度计数据以在计算机显示器上移动光标的基本代码。同时,我们希望您能创造一些更酷的补充和改进!

AirMouse快速概览

我们都用过电脑鼠标,但是它们只能在桌面之类的平面上工作。我们已经制作了一个“AirMouse”——一款能够在3D空间中运行的计算机鼠标。用户通过倾斜鼠标就可以让屏幕上的光标移动,从而可以进行大范围自定义动作。我们已经在第1部分中介绍了构建AirMouse的基础知识。AirMouse主要由两部分组成:戴在手上的鼠标发射器和连接用户计算机的接收器。发射器收集鼠标的方向和按钮状态信息;而接收器则负责转换这些信息,从而在计算机屏幕上执行相应操作。AirMouse由Arduino Uno和nRF24L01射频模块结合ROHM传感器开发板的加速度计构建而成。

第1部分的硬件:

发射器:

● 1 x Arduino Uno微控制器

1 x ROHM传感器评估套件

1 x ROHM加速度计 (KX022-1020)

1 x nRF24L01+ RF模块

排母

滑动开关

1 x 可焊接试验电路板,足够大,能焊接微控制器和所有电子器件

2 x 按键

1 x 1s LiPo电池

1 x 1kΩ电阻r

1 x 3.3kΩ电阻

接收器:

1 x Teensy 3.2微控制器

1 x NRF24L01+ RF模块

排母

1 x 可焊接试验电路板,足够大,能焊接微控制器和所有电子器件

1 x 红、黄、绿LED

1 x 150Ω电阻

按照第1部分连接硬件之后,您就可以利用以下程序运行发送器和接收器。

发射器:

#include 
#include "RF24.h"
 
#define byte uint8_t
#include 
#include 
 
KX022 accelerometer(KX022_DEVICE_ADDRESS_1E);
 
RF24 radio(9,10);
 
uint64_t pipes[2] = {0xF0F0F0F0F0, 0xF0F1F1F1F1}; //reading, writing
 
void initRadio()
{
 radio.setPALevel(RF24_PA_HIGH);
 //payload size default 32...
 radio.setChannel(10); //you can change the channel setting
 radio.setCRCLength(RF24_CRC_16); //2-byte CRC
 radio.setDataRate(RF24_1MBPS); //1Mbps data rate
 
 radio.openReadingPipe(0, pipes[0]);
 radio.openWritingPipe(pipes[1]);
}
 
#define buttonPinR 2 //change these accordingly
#define buttonPinL 3
 
void setup() {
 // put your setup code here, to run once:
 Serial.begin(9600);
 while (!Serial);
 
 pinMode(buttonPinR, INPUT);
 pinMode(buttonPinL, INPUT);
 
 radio.begin();
 initRadio();
 radio.stopListening();
 
 Wire.begin();
 accelerometer.init();
}
 
long lastDebounceTimeR = 0;  // the last time the output pin was toggled
long lastDebounceTimeL = 0;
long debounceDelay = 50;
int buttonStateR = LOW;             // the current reading from the input pin
int buttonStateL = LOW;
int lastReadingR = LOW;
int lastReadingL = LOW;
 
char readButtonR(){
 int reading = digitalRead(buttonPinR);//get what state the button is
 char out = 'a';//the value to return if nothing special happened
 
 if (reading != lastReadingR) {//We're reading a new state for button
   // reset the debouncing timer
   lastDebounceTimeR = millis();
 }
 
 if ((millis() - lastDebounceTimeR) > debounceDelay) {//We finally have a stable value
   if (reading != buttonStateR)//Compared to our previous state, we have a flip
   {
     out = 'r';//prepare to toggle the Mini
   }
   buttonStateR = reading;//Make the buttonState the same
 }
 
 lastReadingR = reading;//make the last state the "current" state
 return out;
}
 
char readButtonL(){
 int reading = digitalRead(buttonPinL);
 char out = 'a';
 
 if (reading != lastReadingL) {
   // reset the debouncing timer
   lastDebounceTimeL = millis();
 }
 
 if ((millis() - lastDebounceTimeL) > debounceDelay) {
   
   if (reading != buttonStateL)
   {
     out = 'l';
   }
   buttonStateL = reading;
 }
 
 lastReadingL = reading;
 return out;
}
 
struct data
{
 boolean isPushedR = false;
 boolean isPushedL = false;
 int8_t acceleration[3] = {0, 0, 0};
};
 
data packet;
boolean rState = false;//these states are used to represent the current state of the buttons
boolean lState = false;
 
void loop() {    
   
   if(readButtonR() == 'r'){ //toggle button state when button state change is detected
     rState = !rState;
   }
   
   if(readButtonL() == 'l'){ //toggle button state when button state change is detected
     lState=!lState;
   }
 
   packet.isPushedR = rState;
   packet.isPushedL = lState;
 
   uint8_t rc;
   float acc[3];
   rc = accelerometer.get_val(acc);
   
   if (rc == 0)
   {
     //we cast to drop the decimal, we don't need that high precision 
     packet.acceleration[0] = (int8_t)(acc[0]*100); //x
     //Serial.print(packet.acceleration[0]); Serial.print(" ");
     packet.acceleration[1] = (int8_t)(acc[1]*100); //y
     //Serial.print(packet.acceleration[1]); Serial.print(" ");
     packet.acceleration[2] = (int8_t)(acc[2]*100); //z
     //Serial.println(packet.acceleration[2]);
   }
 
   radio.write((char*) &packet, sizeof(packet));
}

接收器:

#include 
#include "RF24.h"
 
RF24 radio(9,10);
 
uint64_t pipes[2] = {0xF0F1F1F1F1, 0xF0F0F0F0F0}; //reading, writing
 
void initRadio()
{
 radio.setPALevel(RF24_PA_HIGH);
 //payload size default 32...
 radio.setChannel(10);
 radio.setCRCLength(RF24_CRC_16); //2-byte CRC
 radio.setDataRate(RF24_1MBPS); //1Mbps data rate
 
 radio.openReadingPipe(0, pipes[0]); //reading pipe
 radio.openWritingPipe(pipes[1]);
 
 radio.startListening();
 
}
#define R_PIN 6 //Red LED
#define G_PIN 7 //Green LED
#define Y_PIN 8 //Yellow LED
 
void setup() {
 Serial.begin(9600);
 while(!Serial); //wait until Serial is initialized...(we found that not including this line of code caused errors on the 
                 //Teensy because it started executing code without ensuring that Serial communication with the laptop was
                 //properly initialized...
 radio.begin();
 initRadio();
 
 Mouse.screenSize(1920, 1080);  // configure screen size
 
 randomSeed(analogRead(0));
 
 pinMode(R_PIN, OUTPUT);
 pinMode(G_PIN, OUTPUT);
 pinMode(Y_PIN, OUTPUT);
}
 
#define CALIX 6 //calibration for X
#define CALIY -1 //calibration for Y
 
#define scalingFactor 0.05
#define THRESHOLD 1
 
double moveVector[2] = {0, 0};
 
void tiltToVector(const int8_t* acceleration){
 moveVector[0] = 0;
 moveVector[1] = 0;
 if(abs(acceleration[0] - CALIX) > THRESHOLD){ //calculate move
   moveVector[1] = (double)(acceleration[0] * scalingFactor);
 }
 if(abs(acceleration[1] - CALIY) > THRESHOLD) {
   moveVector[0] = (double)(acceleration[1] * scalingFactor);
 }
}
 
struct data
{
 boolean isPushedR = false;
 boolean isPushedL = false;
 int8_t acceleration[3] = {0, 0, 0};
};
 
data packet;
 
void loop() {
 
 bool stillWaiting = true;
 //Serial.println("About to read");
   while(stillWaiting){
     if(radio.available(0)){ //You've got mail!!!
       radio.read((char*) &packet, sizeof(packet));
       stillWaiting = false;
     }
   }
 
 Mouse.move(moveVector[0], moveVector[1]);
 Mouse.move(moveVector[0], moveVector[1]); //call it twice within the loop for smoothness :) 
 
 //prints for debugging purposes
 Serial.println("Finished writing the pins");
 if (packet.isPushedR) {
   Serial.println("The right button has been clicked!!! (Did you mean to right click?!?!)");
 }
 
 if (packet.isPushedL) {
   Serial.println("The left button has been clicked!!! (Did you mean to left click?!?!)");
   //Mouse.click();
 }
 
 Serial.print("X: ");
 Serial.println(packet.acceleration[0]);
 
 Serial.print("Y: ");
 Serial.println(packet.acceleration[1]);
 
 Serial.print("Z: ");
 Serial.println(packet.acceleration[2]);
 
 tiltToVector(packet.acceleration); //re-calculate move vector coordinates
 
 
//  Mouse.move(moveVector[0], moveVector[1]);
 
}

将代码上传到Teensy与将代码上传到Arduino Uno略有不同。对于Uno,您只需按照通常的编译和上传步骤执行即可:

依次选择Tools > Board > Arduino/Genuino Uno

选择端口(Port)

点击Arduino IDE中的“Upload”(上传)按钮

而对于Teensy,请按照以下步骤上传接收器代码:

依次选择Tools > Board > Teensy 3.2 / 3.1

依次选择Tools > USB Type > Keyboard + Mouse + Joystick (我们将Teensy当作一个USB设备使用)

点击Arduino IDE中的“Upload”(上传)按钮

为了本教程的目的,我们不会详细涉及所使用的不同通信协议,也不会详细介绍RF模块的通信软件。要了解有关这些主题的更多信息,请查看我们的 通信协议 和 nRF24L01+ 模块 教程 相反,我们将简要介绍软件中主控制电路的工作原理

在AirMouse中,发射器负责收集数据,但大部分数据处理由接收器模块进行。系统的这种设计方式使得Arduino——比Teensy更弱的处理器——只需要收集数据即可,因此能够在决策和计算上花费更少的资源,并能够以更快的周期运行。通过这种实现方法,两台设备之间唯一发送的数据就是原始加速计数据和按钮数据。Teensy接收这些原始数据并进行处理,从而在计算机屏幕上执行相应操作。

为了检测AirMouse的方位,系统必须能够解析原始加速计数据。要做到这一点,首先必须确定每个坐标的“零值”。零值的定义如下:AirMouse保持平坦(平行于地面)时每个轴的加速度计输出。确定零值后,软件就能够将加速度计数据转换为方向和数量,通过分析每个轴的加速度(由于重力)并将其与零值进行比较以便在屏幕上移动光标。

现在,我们来看一下接收器模块与电脑之间的交互。Teensy被指定为USB人机界面设备(这里是指USB鼠标)。解析方位数据后,软件会计算光标移动的速度和方向。此外,该软件还将点击左键解析为左键单击,将点击右键解析为右键单击,调用适当的方法在计算机屏幕上显示左键单击或右键单击功能。以下才是最酷的部分:您完全可以只通过软件就能够修改或添加鼠标的屏幕功能!目前,鼠标只具有最基本的功能和特性,但是您只需对软件进行简单改动,就可以轻松添加诸如滚动、将光标移至屏幕上的某个点等功能!(请点击此处,了解Teensy USB鼠标的参考指南) 以下是您可以实现的一些很酷的硬件和软件想法:

将不同模式的按键点击用作不同的控制功能(比如,双击右键关闭窗口)

添加滚动功能!(同时点击两个按键变为“滚动模式”)

添加其他鼠标按键以执行滚动操作或为不同功能获取更多按键模式

我们希望您能够喜欢这个AirMouse项目,并且非常期待您可以对设计和功能进行修改和改进!

作为UCLA IEEE高级项目(Advanced Projects)计划的一部分,最初的AirMouse由Rahul Iyer、Aaron和Andrew Wilhelm研发。

审核编辑:汤梓红

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 传感器
    +关注

    关注

    2551

    文章

    51189

    浏览量

    754341
  • 发射器
    +关注

    关注

    6

    文章

    849

    浏览量

    53499
  • Arduino
    +关注

    关注

    188

    文章

    6471

    浏览量

    187258
收藏 人收藏

    评论

    相关推荐

    采用ROHM传感器套件的DIY Arduino家庭安防系统 2部分- Cayenne设置

    在DIY智能家居安防系统1部分中,我们将各种组件(例如PIR传感器、温度传感器和压力传感器)组合在一起,以创建家庭安防系统/探测
    的头像 发表于 09-19 10:20 4220次阅读

    传感器原理与应用(2版)

    全书部分共19章。1部分共两章介绍传感器的一般特性、分析方法,
    发表于 04-20 09:31 0次下载

    使用触摸传感器的设备的输入和控制,2部分:控制和开发包

    在本系列触摸感知的1部分中,我们研究了各种类型的触摸传感器威廉希尔官方网站 。在2部分中,我们将探讨触摸感
    发表于 06-21 09:28 5次下载
    使用触摸<b class='flag-5'>传感器</b>的设备的输入和控制,<b class='flag-5'>第</b><b class='flag-5'>2</b><b class='flag-5'>部分</b>:控制<b class='flag-5'>器</b>和开发包

    1部分:MAX32660的特点原理介绍

    该系列视频共包括2部分,这是1部分。本部分介绍MAX32660低功耗Arm® Cortex®-M4 FPU微控制
    的头像 发表于 10-08 01:46 7309次阅读

    DN236-1和2通道无延迟ΔΣ<sup>™</sup>24位ADC轻松实现多种传感器的数字化,1部分

    DN236-1和2通道无延迟ΔΣ™24位ADC轻松实现多种传感器的数字化,1部分
    发表于 04-19 20:11 1次下载
    DN236-1和<b class='flag-5'>2</b>通道无延迟ΔΣ<sup>™</sup>24位ADC轻松<b class='flag-5'>实现</b>多种<b class='flag-5'>传感器</b>的数字化,<b class='flag-5'>第</b>1<b class='flag-5'>部分</b>

    DN237-1通道和2通道无延迟ΔΣ<sup>™</sup>24位ADC轻松实现多种传感器的数字化,2部分

    DN237-1通道和2通道无延迟ΔΣ™24位ADC轻松实现多种传感器的数字化,2部分
    发表于 04-19 21:17 0次下载
    DN237-1通道和<b class='flag-5'>2</b>通道无延迟ΔΣ<sup>™</sup>24位ADC轻松<b class='flag-5'>实现</b>多种<b class='flag-5'>传感器</b>的数字化,<b class='flag-5'>第</b><b class='flag-5'>2</b><b class='flag-5'>部分</b>

    用一个Hercules™ LaunchPad™ 开发套件控制GaN功率级—2部分

    用一个Hercules™ LaunchPad™ 开发套件控制GaN功率级—2部分
    发表于 11-03 08:04 0次下载
    用一个Hercules™ LaunchPad™ 开发<b class='flag-5'>套件</b>控制GaN功率级—<b class='flag-5'>第</b><b class='flag-5'>2</b><b class='flag-5'>部分</b>

    用Hercules™ LaunchPad™ 开发套件控制GaN功率级—1部分

    用Hercules™ LaunchPad™ 开发套件控制GaN功率级—1部分
    发表于 11-03 08:04 0次下载
    用Hercules™ LaunchPad™ 开发<b class='flag-5'>套件</b>控制GaN功率级—<b class='flag-5'>第</b>1<b class='flag-5'>部分</b>

    带NTC的恒温-2部分

    电子发烧友网站提供《带NTC的恒温-2部分.zip》资料免费下载
    发表于 11-21 09:57 0次下载
    带NTC的恒温<b class='flag-5'>器</b>-<b class='flag-5'>第</b><b class='flag-5'>2</b><b class='flag-5'>部分</b>

    利用Arduino Create和ROHM传感器评估套件制作智能花园系统

    今天,我们将利用ROHM传感器评估套件打造一套智能花园系统。在此项目中,我们整合接近/环境光传感器
    的头像 发表于 02-23 17:58 1226次阅读
    <b class='flag-5'>利用</b>Arduino Create和<b class='flag-5'>ROHM</b><b class='flag-5'>传感器</b><b class='flag-5'>评估</b><b class='flag-5'>套件</b>制作智能花园系统

    ROHM Arduino传感器套件打造DIY万圣节僵尸面具

    今天,我们会利用ROHM传感器评估套件制作一款超级简单的万圣节僵尸面具!我们将使用ROHM接近
    的头像 发表于 02-23 17:45 943次阅读
    用<b class='flag-5'>ROHM</b> Arduino<b class='flag-5'>传感器</b><b class='flag-5'>套件</b>打造DIY万圣节僵尸面具

    采用ROHM传感器套件的DIY Arduino家庭安防系统 2部分- Cayenne设置

    在DIY智能家居安防系统1部分中,我们将各种组件(例如PIR传感器、温度传感器和压力传感器)组合在一起,以创建家庭安防系统/探测
    的头像 发表于 02-24 14:38 1055次阅读
    采用<b class='flag-5'>ROHM</b><b class='flag-5'>传感器</b><b class='flag-5'>套件</b>的DIY Arduino家庭安防系统 <b class='flag-5'>第</b><b class='flag-5'>2</b><b class='flag-5'>部分</b>- Cayenne设置

    采用ROHM传感器套件的DIY Arduino家庭安防系统 1部分 机制

    在该项目中,我们将使用Arduino Mega和ROHM传感器评估套件来制作一套DIY Arduino家庭安防系统。
    的头像 发表于 02-24 15:08 1058次阅读
    采用<b class='flag-5'>ROHM</b><b class='flag-5'>传感器</b><b class='flag-5'>套件</b>的DIY Arduino家庭安防系统 <b class='flag-5'>第</b>1<b class='flag-5'>部分</b> 机制

    ROHM传感器评估套件的3D外壳以及RohmMultiSensor库更新

    不久前,我创建了RohmMultiSensor - Arduino库,利用该库可以轻松控制ROHM传感器评估套件中的多个
    的头像 发表于 02-24 09:51 895次阅读
    <b class='flag-5'>ROHM</b><b class='flag-5'>传感器</b><b class='flag-5'>评估</b><b class='flag-5'>套件</b>的3D外壳以及RohmMultiSensor库更新

    利用ROHM传感器评估套件实现UCLA AirMouse1部分

    今天,我们介绍一款由洛加利福尼亚大学洛杉矶分校(UCLA)电气和电子工程师协会(IEEE)学生分会一群聪明的工程师研发的3D AirMouseAirMouse的功能与传统电脑鼠标非常类似,但是能够检测三维运动。
    的头像 发表于 02-27 10:27 366次阅读
    <b class='flag-5'>利用</b><b class='flag-5'>ROHM</b><b class='flag-5'>传感器</b><b class='flag-5'>评估</b><b class='flag-5'>套件</b><b class='flag-5'>实现</b><b class='flag-5'>UCLA</b> <b class='flag-5'>AirMouse</b>–<b class='flag-5'>第</b>1<b class='flag-5'>部分</b>