From 14712486579022038b331934b318c624e8f98461 Mon Sep 17 00:00:00 2001 From: Valerie Young Date: Mon, 17 Dec 2018 21:04:15 -0500 Subject: [PATCH] Clean up and all sensors working --- src/extensions/scratch3_gdx_for/index.js | 284 +++++++++-------------- 1 file changed, 104 insertions(+), 180 deletions(-) diff --git a/src/extensions/scratch3_gdx_for/index.js b/src/extensions/scratch3_gdx_for/index.js index bc244a227..ec29cb713 100644 --- a/src/extensions/scratch3_gdx_for/index.js +++ b/src/extensions/scratch3_gdx_for/index.js @@ -1,11 +1,10 @@ const ArgumentType = require('../../extension-support/argument-type'); const BlockType = require('../../extension-support/block-type'); -// const log = require('../../util/log'); +const log = require('../../util/log'); // const cast = require('../../util/cast'); const formatMessage = require('format-message'); const BLE = require('../../io/ble'); const Base64Util = require('../../util/base64-util'); -const GdxCommands = require('./gdx-commands'); //const ScratchLinkProxy = require('./scratch-link-proxy'); 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 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. */ @@ -76,29 +62,8 @@ class GdxFor { 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._onConnect = this._onConnect.bind(this); - this._onMessage = this._onMessage.bind(this); this._connected = false; } @@ -110,36 +75,19 @@ class GdxFor { scan () { if (this._ble) { this._ble.close(); - // this._ble.disconnect(); //scratch-link } - console.log("Requesting Device"); navigator.bluetooth.requestDevice({ filters: [{ namePrefix: 'GDX' }], optionalServices: ['d91714ef-28b9-4f91-ba16-f0d9a604f112'] }).then((bleWbtDevice) => { - console.log("Got device: ", bleWbtDevice); return createDevice(bleWbtDevice, {open: false}); }).then((device) => { - console.log("Created BLE object: ", 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-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 () { - // 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) { - //this._ble.disconnect(); + this._ble.close(); } } @@ -174,55 +117,22 @@ class GdxFor { isConnected () { let connected = false; if (this._ble) { - connected = this._connected; //this._ble.isConnected(); + connected = this._connected; } - console.log("Is connected: ", 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. * @private */ _onConnect () { - console.log("CONNNECTED"); this._connected = true; this._ble.sensors.forEach(sensor => { sensor.setEnabled(true); - // do some calculations? + // do some calculations here? sensor.on('value-changed', (sensor) => { // We don't need this list, essentially. @@ -232,58 +142,52 @@ class GdxFor { }); }); - this._ble.start(); // Can set period here. - //this.send(GdxCommands.INIT, ''); - - //this._ble.read(BLEUUID.service, BLEUUID.rxChar, true, this._onMessage); - - //this._timeoutID = window.setInterval(this.disconnect, BLETimeout); + this._ble.start(); // Can set period here if needed. } getForce () { 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 () { 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 = ['>', '<', '=']; - -const DirectionOptionsMenu = ['x', 'y', 'z']; +/** + * Enum for comparison operations. + * @readonly + * @enum {string} + */ +const ComparisonOptions = { + LESS_THAN: 'less_than', + GREATER_THAN: 'greater_than' +}; /** @@ -305,6 +209,37 @@ class Scratch3GdxForBlocks { 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. * @param {Runtime} runtime - the Scratch 3.0 runtime. @@ -342,12 +277,12 @@ class Scratch3GdxForBlocks { DIRECTION: { type: ArgumentType.STRING, menu: 'directionOptions', - defaultValue: 0 + defaultValue: 'x' }, COMPARE: { type: ArgumentType.STRING, menu: 'compareOptions', - defaultValue: 0 + defaultValue: ComparisonOptions.GREATER_THAN }, VALUE: { type: ArgumentType.NUMBER, @@ -367,12 +302,12 @@ class Scratch3GdxForBlocks { DIRECTION: { type: ArgumentType.STRING, menu: 'directionOptions', - defaultValue: 0 + defaultValue: 'x' }, COMPARE: { type: ArgumentType.STRING, menu: 'compareOptions', - defaultValue: 0 + defaultValue: ComparisonOptions.GREATER_THAN }, VALUE: { type: ArgumentType.NUMBER, @@ -392,7 +327,7 @@ class Scratch3GdxForBlocks { COMPARE: { type: ArgumentType.STRING, menu: 'compareOptions', - defaultValue: 0 + defaultValue: 'x' }, VALUE: { type: ArgumentType.NUMBER, @@ -412,7 +347,7 @@ class Scratch3GdxForBlocks { DIRECTION: { type: ArgumentType.STRING, menu: 'directionOptions', - defaultValue: 0 + defaultValue: 'x' } } }, @@ -421,14 +356,14 @@ class Scratch3GdxForBlocks { text: formatMessage({ id: 'gdxfor.getAngularSpeed', default: 'angular speed [DIRECTION]', - description: 'gets tilt' + description: 'gets angular speed' }), blockType: BlockType.REPORTER, arguments: { DIRECTION: { type: ArgumentType.STRING, menu: 'directionOptions', - defaultValue: 0 + defaultValue: 'x' } } }, @@ -443,8 +378,8 @@ class Scratch3GdxForBlocks { } ], menus: { - directionOptions: this._formatMenu(DirectionOptionsMenu), - compareOptions: this._formatMenu(CompareOptionsMenu) + directionOptions: this.DIRECTIONS_MENU, + compareOptions: this.COMPARE_MENU, } }; } @@ -458,45 +393,34 @@ class Scratch3GdxForBlocks { whenForceCompare () { return Promise.resolve(); } - getAcceleration () { - return Promise.resolve(); + getAcceleration (args) { + 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 () { - return Promise.resolve(); + getAngularSpeed (args) { + 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 () { 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;