mirror of
https://github.com/scratchfoundation/scratch-vm.git
synced 2024-12-24 06:52:40 -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 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;
|
||||||
|
|
Loading…
Reference in a new issue