mirror of
https://github.com/scratchfoundation/scratch-vm.git
synced 2025-01-11 10:39:56 -05:00
Clean up and all sensors working
This commit is contained in:
parent
d70fb96d01
commit
1471248657
1 changed files with 104 additions and 180 deletions
|
@ -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;
|
||||
|
|
Loading…
Reference in a new issue