Clean up and all sensors working

This commit is contained in:
Valerie Young 2018-12-17 21:04:15 -05:00
parent d70fb96d01
commit 1471248657

View file

@ -1,11 +1,10 @@
const ArgumentType = require('../../extension-support/argument-type'); const ArgumentType = require('../../extension-support/argument-type');
const BlockType = require('../../extension-support/block-type'); const BlockType = require('../../extension-support/block-type');
// const log = require('../../util/log'); const log = require('../../util/log');
// const cast = require('../../util/cast'); // const cast = require('../../util/cast');
const formatMessage = require('format-message'); const formatMessage = require('format-message');
const BLE = require('../../io/ble'); const BLE = require('../../io/ble');
const Base64Util = require('../../util/base64-util'); const Base64Util = require('../../util/base64-util');
const GdxCommands = require('./gdx-commands');
//const ScratchLinkProxy = require('./scratch-link-proxy'); //const ScratchLinkProxy = require('./scratch-link-proxy');
const createDevice = require('@vernier/godirect/src/godirect').default.createDevice; const createDevice = require('@vernier/godirect/src/godirect').default.createDevice;
@ -16,19 +15,6 @@ const createDevice = require('@vernier/godirect/src/godirect').default.createDev
// eslint-disable-next-line max-len // eslint-disable-next-line max-len
const blockIconURI = ''; const blockIconURI = '';
/**
* Enum for GDX-FOR protocol.
* @readonly
* @enum {string}
*/
const BLEUUID = {
service: 'd91714ef-28b9-4f91-ba16-f0d9a604f112',
txChar: 'f4bf14a6-c7d5-4b6d-8aa8-df1a7c83adcb',
rxChar: 'b41e6675-a329-40e0-aa01-44d2f444babe'
};
const BLETimeout = 4500;
/** /**
* Manage communication with a GDX-FOR peripheral over a Scratch Link client socket. * Manage communication with a GDX-FOR peripheral over a Scratch Link client socket.
*/ */
@ -76,29 +62,8 @@ class GdxFor {
angularVelocityX: 0, angularVelocityX: 0,
}; };
/**
* Interval ID for data reading timeout.
* @type {number}
* @private
*/
this._timeoutID = null;
/**
* A flag that is true while we are busy sending data to the BLE socket.
* @type {boolean}
* @private
*/
this._busy = false;
/**
* ID for a timeout which is used to clear the busy flag if it has been
* true for a long time.
*/
this._busyTimeoutID = null;
this.disconnect = this.disconnect.bind(this); this.disconnect = this.disconnect.bind(this);
this._onConnect = this._onConnect.bind(this); this._onConnect = this._onConnect.bind(this);
this._onMessage = this._onMessage.bind(this);
this._connected = false; this._connected = false;
} }
@ -110,36 +75,19 @@ class GdxFor {
scan () { scan () {
if (this._ble) { if (this._ble) {
this._ble.close(); this._ble.close();
// this._ble.disconnect(); //scratch-link
} }
console.log("Requesting Device");
navigator.bluetooth.requestDevice({ navigator.bluetooth.requestDevice({
filters: [{ namePrefix: 'GDX' }], filters: [{ namePrefix: 'GDX' }],
optionalServices: ['d91714ef-28b9-4f91-ba16-f0d9a604f112'] optionalServices: ['d91714ef-28b9-4f91-ba16-f0d9a604f112']
}).then((bleWbtDevice) => { }).then((bleWbtDevice) => {
console.log("Got device: ", bleWbtDevice);
return createDevice(bleWbtDevice, {open: false}); return createDevice(bleWbtDevice, {open: false});
}).then((device) => { }).then((device) => {
console.log("Created BLE object: ", device);
this._ble = device; this._ble = device;
this._ble.open(false); // should emit a "device-opened" event when opened this._ble.open(false); // false prevents starting of measurements
this._ble.on('device-opened', this._onConnect); this._ble.on('device-opened', this._onConnect);
this._ble.on('device-closed', this._onClose); this._ble.on('device-closed', this._onClose);
}).catch((error) => {
console.log(error);
}); });
// scratch-link
// this._ble = new BLE(this._runtime, this._extensionId, {
// filters: [
// { namePrefix: 'GDX' }
// ],
// optionalServices: [
// BLEUUID.service
// ]
// }, this._onConnect);
} }
/** /**
@ -157,13 +105,8 @@ class GdxFor {
* Disconnect from the micro:bit. * Disconnect from the micro:bit.
*/ */
disconnect () { disconnect () {
// TODO: This funciton is called when you exist the connection popup.
// When using Web Bluetooth, we don't use the connection popup.
// When switching over from web bluetooth, make sure to figure out to handle
// this disconnection.
window.clearInterval(this._timeoutID);
if (this._ble) { if (this._ble) {
//this._ble.disconnect(); this._ble.close();
} }
} }
@ -174,55 +117,22 @@ class GdxFor {
isConnected () { isConnected () {
let connected = false; let connected = false;
if (this._ble) { if (this._ble) {
connected = this._connected; //this._ble.isConnected(); connected = this._connected;
} }
console.log("Is connected: ", connected);
return connected; return connected;
} }
/**
* Send a message to the peripheral BLE socket.
* @param {number} command - the BLE command hex.
* @param {Uint8Array} message - the message to write
*/
send (command, message) {
if (!this.isConnected()) return;
debugger;
const output = new Uint8Array(message.length + 1);
output[0] = command; // attach command to beginning of message
for (let i = 0; i < message.length; i++) {
output[i + 1] = message[i];
}
//const output = [0x58, 0x19, 0xfe, 0x3f, 0x1a, 0xa5, 0x4a, 0x06, 0x49, 0x07, 0x48, 0x08, 0x47, 0x09, 0x46, 0x0a, 0x45, 0x0b, 0x44, 0x0c, 0x43, 0x0d, 0x42, 0x0e, 0x41];
//const output = new Uint8Array([0x58, 0x19, 0xfe, 0x3f, 0x1a, 0xa5, 0x4a, 0x06, 0x49, 0x07, 0x48, 0x08, 0x47, 0x09, 0x46, 0x0a, 0x45, 0x0b, 0x44, 0x0c, 0x43, 0x0d, 0x42, 0x0e, 0x41]);
console.log(output);
const data = Base64Util.uint8ArrayToBase64(output);
console.log(data);
// WBn+PxqlSgZJB0gIRwlGCkULRAxDDUIOQQ==
this._ble.write(BLEUUID.service, BLEUUID.txChar, data, 'base64', true).then(
() => {
this._busy = false;
}
);
}
/** /**
* Starts reading data from peripheral after BLE has connected to it. * Starts reading data from peripheral after BLE has connected to it.
* @private * @private
*/ */
_onConnect () { _onConnect () {
console.log("CONNNECTED");
this._connected = true; this._connected = true;
this._ble.sensors.forEach(sensor => { this._ble.sensors.forEach(sensor => {
sensor.setEnabled(true); sensor.setEnabled(true);
// do some calculations? // do some calculations here?
sensor.on('value-changed', (sensor) => { sensor.on('value-changed', (sensor) => {
// We don't need this list, essentially. // We don't need this list, essentially.
@ -232,58 +142,52 @@ class GdxFor {
}); });
}); });
this._ble.start(); // Can set period here. this._ble.start(); // Can set period here if needed.
//this.send(GdxCommands.INIT, '');
//this._ble.read(BLEUUID.service, BLEUUID.rxChar, true, this._onMessage);
//this._timeoutID = window.setInterval(this.disconnect, BLETimeout);
} }
getForce () { getForce () {
return this._ble.getSensor(1).value; return this._ble.getSensor(1).value;
} }
getAccelerationX () {
return this._ble.getSensor(2).value;
}
getAccelerationY () {
return this._ble.getSensor(3).value;
}
getAccelerationZ () {
return this._ble.getSensor(4).value;
}
getAngularSpeedX () {
return this._ble.getSensor(5).value;
}
getAngularSpeedY () {
return this._ble.getSensor(6).value;
}
getAngularSpeedZ () {
return this._ble.getSensor(7).value;
}
_onClose () { _onClose () {
this._connected = false; this._connected = false;
} }
/**
* Process the sensor data from the incoming BLE characteristic.
* @param {object} base64 - the incoming BLE data.
* @private
*/
_onMessage (base64) {
console.log("In _onMessage");
// parse data
const data = Base64Util.base64ToUint8Array(base64);
console.log("data: ", data);
// this._sensors.tiltX = data[1] | (data[0] << 8);
// if (this._sensors.tiltX > (1 << 15)) this._sensors.tiltX -= (1 << 16);
// this._sensors.tiltY = data[3] | (data[2] << 8);
// if (this._sensors.tiltY > (1 << 15)) this._sensors.tiltY -= (1 << 16);
// this._sensors.buttonA = data[4];
// this._sensors.buttonB = data[5];
// this._sensors.touchPins[0] = data[6];
// this._sensors.touchPins[1] = data[7];
// this._sensors.touchPins[2] = data[8];
// this._sensors.gestureState = data[9];
// // cancel disconnect timeout and start a new one
// window.clearInterval(this._timeoutID);
// this._timeoutID = window.setInterval(this.disconnect, BLETimeout);
}
} }
/**
const CompareOptionsMenu = ['>', '<', '=']; * Enum for comparison operations.
* @readonly
const DirectionOptionsMenu = ['x', 'y', 'z']; * @enum {string}
*/
const ComparisonOptions = {
LESS_THAN: 'less_than',
GREATER_THAN: 'greater_than'
};
/** /**
@ -305,6 +209,37 @@ class Scratch3GdxForBlocks {
return 'gdxfor'; return 'gdxfor';
} }
get DIRECTIONS_MENU () {
return [
{
text: 'x',
value: 'x'
},
{
text: 'y',
value: 'y'
},
{
text: 'z',
value: 'z'
}
];
}
get COMPARE_MENU () {
return [
{
text: '<',
value: ComparisonOptions.LESS_THAN
},
{
text: '>',
value: ComparisonOptions.GREATER_THAN
}
];
}
/** /**
* Construct a set of GDX-FOR blocks. * Construct a set of GDX-FOR blocks.
* @param {Runtime} runtime - the Scratch 3.0 runtime. * @param {Runtime} runtime - the Scratch 3.0 runtime.
@ -342,12 +277,12 @@ class Scratch3GdxForBlocks {
DIRECTION: { DIRECTION: {
type: ArgumentType.STRING, type: ArgumentType.STRING,
menu: 'directionOptions', menu: 'directionOptions',
defaultValue: 0 defaultValue: 'x'
}, },
COMPARE: { COMPARE: {
type: ArgumentType.STRING, type: ArgumentType.STRING,
menu: 'compareOptions', menu: 'compareOptions',
defaultValue: 0 defaultValue: ComparisonOptions.GREATER_THAN
}, },
VALUE: { VALUE: {
type: ArgumentType.NUMBER, type: ArgumentType.NUMBER,
@ -367,12 +302,12 @@ class Scratch3GdxForBlocks {
DIRECTION: { DIRECTION: {
type: ArgumentType.STRING, type: ArgumentType.STRING,
menu: 'directionOptions', menu: 'directionOptions',
defaultValue: 0 defaultValue: 'x'
}, },
COMPARE: { COMPARE: {
type: ArgumentType.STRING, type: ArgumentType.STRING,
menu: 'compareOptions', menu: 'compareOptions',
defaultValue: 0 defaultValue: ComparisonOptions.GREATER_THAN
}, },
VALUE: { VALUE: {
type: ArgumentType.NUMBER, type: ArgumentType.NUMBER,
@ -392,7 +327,7 @@ class Scratch3GdxForBlocks {
COMPARE: { COMPARE: {
type: ArgumentType.STRING, type: ArgumentType.STRING,
menu: 'compareOptions', menu: 'compareOptions',
defaultValue: 0 defaultValue: 'x'
}, },
VALUE: { VALUE: {
type: ArgumentType.NUMBER, type: ArgumentType.NUMBER,
@ -412,7 +347,7 @@ class Scratch3GdxForBlocks {
DIRECTION: { DIRECTION: {
type: ArgumentType.STRING, type: ArgumentType.STRING,
menu: 'directionOptions', menu: 'directionOptions',
defaultValue: 0 defaultValue: 'x'
} }
} }
}, },
@ -421,14 +356,14 @@ class Scratch3GdxForBlocks {
text: formatMessage({ text: formatMessage({
id: 'gdxfor.getAngularSpeed', id: 'gdxfor.getAngularSpeed',
default: 'angular speed [DIRECTION]', default: 'angular speed [DIRECTION]',
description: 'gets tilt' description: 'gets angular speed'
}), }),
blockType: BlockType.REPORTER, blockType: BlockType.REPORTER,
arguments: { arguments: {
DIRECTION: { DIRECTION: {
type: ArgumentType.STRING, type: ArgumentType.STRING,
menu: 'directionOptions', menu: 'directionOptions',
defaultValue: 0 defaultValue: 'x'
} }
} }
}, },
@ -443,8 +378,8 @@ class Scratch3GdxForBlocks {
} }
], ],
menus: { menus: {
directionOptions: this._formatMenu(DirectionOptionsMenu), directionOptions: this.DIRECTIONS_MENU,
compareOptions: this._formatMenu(CompareOptionsMenu) compareOptions: this.COMPARE_MENU,
} }
}; };
} }
@ -458,45 +393,34 @@ class Scratch3GdxForBlocks {
whenForceCompare () { whenForceCompare () {
return Promise.resolve(); return Promise.resolve();
} }
getAcceleration () { getAcceleration (args) {
return Promise.resolve(); console.log(args);
switch (args.DIRECTION) {
case 'x':
return this._peripheral.getAccelerationX();
case 'y':
return this._peripheral.getAccelerationY();
case 'z':
return this._peripheral.getAccelerationZ();
default:
log.warn(`Unknown direction in getAcceleration: ${args.DIRECTION}`);
}
} }
getAngularSpeed () { getAngularSpeed (args) {
return Promise.resolve(); switch (args.DIRECTION) {
case 'x':
return this._peripheral.getAngularSpeedX();
case 'y':
return this._peripheral.getAngularSpeedY();
case 'z':
return this._peripheral.getAngularSpeedZ();
default:
log.warn(`Unknown direction in getAngularSpeed: ${args.DIRECTION}`);
}
} }
getForce () { getForce () {
return this._peripheral.getForce(); return this._peripheral.getForce();
} }
/**
* Formats menus into a format suitable for block menus, and loading previously
* saved projects:
* [
* {
* text: label,
* value: index
* },
* {
* text: label,
* value: index
* },
* etc...
* ]
*
* @param {array} menu - a menu to format.
* @return {object} - a formatted menu as an object.
* @private
*/
_formatMenu (menu) {
const m = [];
for (let i = 0; i < menu.length; i++) {
const obj = {};
obj.text = menu[i];
obj.value = i.toString();
m.push(obj);
}
return m;
}
} }
module.exports = Scratch3GdxForBlocks; module.exports = Scratch3GdxForBlocks;