mirror of
https://github.com/scratchfoundation/scratch-vm.git
synced 2025-08-01 17:11:21 -04:00
MicroBit extension, Scratch Link first draft. (#1230)
* First microbit gui tests * Fixed JSONRPC inheritance. Renamed ScratchBLE/ScratchBT files. Removed ScratchBT test code from Microbit extension. Renamed addLine to log. * Fixed log comments. Removed addLine from Microbit. * Adding auto-connect to Microbit at extension loading. Adding hack for displayText block to Scratch-Link. * Resolved merge conflicts and brought in latest microbit extension example code. * Updated microbit write tests for displayText and displaySymbol blocks. Some linting. * Some linting and adding of BLE Characteristic consts. * Linting fixes. * Moving micro:bit device connection code all to the MicroBit class, decoupling Scratch3MicroBitBlocks from connection code. * Removing old disconenct handlers from MicroBit class. Moved service into new BLEUUID data structure. * Renamed _write to _send. Moved all BLE encoding concerns to the _send method. * Using the util log. Some linting. * Added _read method to MicroBit class. Renamed _send to _write. * Some linting and formatting comments. * First pass at peripheral chooser pattern for ScratchBLE. * Testing characteristicDidChange events, and some changes to ScratchBLE on ready events. * Refactoring work on PeripheralChooser and ScratchBLE. * Some variable renaming and method signature stubs. * Peripheral chooser method signatures. * Moved base64 encoding/decoding to util. Some method signature formatting. * Adding test stubs for new util and io classes. * Adding test stub for MicroBit extension. * Clean up for PR. * Clean up for PR. * Final cleanup for PR. * Removed logging to console. * Adding 'btoa' and 'atob' node modules and using them in Base64Util.
This commit is contained in:
parent
cda47fb3a7
commit
d09c3f0418
14 changed files with 568 additions and 151 deletions
src/extensions/scratch3_microbit
|
@ -1,6 +1,8 @@
|
|||
const ArgumentType = require('../../extension-support/argument-type');
|
||||
const BlockType = require('../../extension-support/block-type');
|
||||
const log = require('../../util/log');
|
||||
const ScratchBLE = require('../../io/scratchBLE');
|
||||
const Base64Util = require('../../util/base64-util');
|
||||
|
||||
/**
|
||||
* Icon svg to be displayed at the left edge of each extension block, encoded as a data URI.
|
||||
|
@ -17,38 +19,54 @@ const blockIconURI = '
|
|||
const menuIconURI = '';
|
||||
|
||||
/**
|
||||
* Manage communication with a MicroBit device over a Device Manager client socket.
|
||||
* Enum for micro:bit BLE command protocol.
|
||||
* https://github.com/LLK/scratch-microbit-firmware/blob/master/protocol.md
|
||||
* @readonly
|
||||
* @enum {number}
|
||||
*/
|
||||
const BLECommand = {
|
||||
CMD_PIN_CONFIG: 0x80,
|
||||
CMD_DISPLAY_TEXT: 0x81,
|
||||
CMD_DISPLAY_LED: 0x82
|
||||
};
|
||||
|
||||
/**
|
||||
* Enum for micro:bit protocol.
|
||||
* https://github.com/LLK/scratch-microbit-firmware/blob/master/protocol.md
|
||||
* @readonly
|
||||
* @enum {string}
|
||||
*/
|
||||
const BLEUUID = {
|
||||
service: 0xf005,
|
||||
rxChar: '5261da01-fa7e-42ab-850b-7c80220097cc',
|
||||
txChar: '5261da02-fa7e-42ab-850b-7c80220097cc'
|
||||
};
|
||||
|
||||
/**
|
||||
* Manage communication with a MicroBit device over a Scrath Link client socket.
|
||||
*/
|
||||
class MicroBit {
|
||||
|
||||
/**
|
||||
* @return {string} - the type of Device Manager device socket that this class will handle.
|
||||
*/
|
||||
static get DEVICE_TYPE () {
|
||||
return 'ble';
|
||||
}
|
||||
|
||||
/**
|
||||
* Construct a MicroBit communication object.
|
||||
* @param {Socket} socket - the socket for a MicroBit device, as provided by a Device Manager client.
|
||||
* @param {Runtime} runtime - the Scratch 3.0 runtime
|
||||
*/
|
||||
constructor (socket, runtime) {
|
||||
/**
|
||||
* The socket-IO socket used to communicate with the Device Manager about this device.
|
||||
* @type {Socket}
|
||||
* @private
|
||||
*/
|
||||
this._socket = socket;
|
||||
constructor (runtime) {
|
||||
|
||||
/**
|
||||
* The Scratch 3.0 runtime used to trigger the green flag button
|
||||
*
|
||||
* The Scratch 3.0 runtime used to trigger the green flag button.
|
||||
* @type {Runtime}
|
||||
* @private
|
||||
*/
|
||||
this._runtime = runtime;
|
||||
|
||||
/**
|
||||
* The ScratchBLE connection session for reading/writing device data.
|
||||
* @type {ScratchBLE}
|
||||
* @private
|
||||
*/
|
||||
this._ble = new ScratchBLE();
|
||||
|
||||
/**
|
||||
* The most recently received value for each sensor.
|
||||
* @type {Object.<string, number>}
|
||||
|
@ -64,6 +82,11 @@ class MicroBit {
|
|||
ledMatrixState: new Uint8Array(5)
|
||||
};
|
||||
|
||||
/**
|
||||
* The most recently received value for each gesture.
|
||||
* @type {Object.<string, Object>}
|
||||
* @private
|
||||
*/
|
||||
this._gestures = {
|
||||
moving: false,
|
||||
move: {
|
||||
|
@ -80,17 +103,32 @@ class MicroBit {
|
|||
}
|
||||
};
|
||||
|
||||
// this._onRxChar = this._onRxChar.bind(this);
|
||||
// this._onDisconnect = this._onDisconnect.bind(this);
|
||||
// TODO: Temporary until the gui requests a device connection
|
||||
this._ble.waitForSocket()
|
||||
// TODO: remove pinging once no longer needed
|
||||
.then(() => this._ble.sendRemoteRequest('pingMe'))
|
||||
.then(() => this._onBLEReady());
|
||||
|
||||
// TODO: Add ScratchBLE 'disconnect' handling
|
||||
|
||||
this._connectEvents();
|
||||
}
|
||||
|
||||
/**
|
||||
* Manually dispose of this object.
|
||||
* @param {string} text - the text to display.
|
||||
*/
|
||||
dispose () {
|
||||
this._disconnectEvents();
|
||||
displayText (text) {
|
||||
const output = new Uint8Array(text.length);
|
||||
for (let i = 0; i < text.length; i++) {
|
||||
output[i] = text.charCodeAt(i);
|
||||
}
|
||||
this._writeBLE(BLECommand.CMD_DISPLAY_TEXT, output);
|
||||
}
|
||||
|
||||
/**
|
||||
* @param {Uint8Array} matrix - the matrix to display.
|
||||
*/
|
||||
displayMatrix (matrix) {
|
||||
this._writeBLE(BLECommand.CMD_DISPLAY_LED, matrix);
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -144,31 +182,38 @@ class MicroBit {
|
|||
}
|
||||
|
||||
/**
|
||||
* Attach event handlers to the device socket.
|
||||
* @private
|
||||
* Requests connection to a device when BLE session is ready.
|
||||
*/
|
||||
_connectEvents () {
|
||||
// this._socket.on(BLE_UUIDs.rx, this._onRxChar);
|
||||
// this._socket.on('deviceWasClosed', this._onDisconnect);
|
||||
// this._socket.on('disconnect', this._onDisconnect);
|
||||
_onBLEReady () {
|
||||
this._ble.requestDevice({
|
||||
filters: [
|
||||
{services: [BLEUUID.service]}
|
||||
]
|
||||
}, this._onBLEConnect.bind(this), this._onBLEError);
|
||||
}
|
||||
|
||||
/**
|
||||
* Detach event handlers from the device socket.
|
||||
* @private
|
||||
* Starts reading data from device after BLE has connected to it.
|
||||
*/
|
||||
_disconnectEvents () {
|
||||
// this._socket.off(BLE_UUIDs.rx, this._onRxChar);
|
||||
// this._socket.off('deviceWasClosed', this._onDisconnect);
|
||||
// this._socket.off('disconnect', this._onDisconnect);
|
||||
_onBLEConnect () {
|
||||
const callback = this._processBLEData.bind(this);
|
||||
this._ble.read(BLEUUID.service, BLEUUID.rxChar, true, callback);
|
||||
}
|
||||
|
||||
/**
|
||||
* @param {string} e - Error from BLE session.
|
||||
*/
|
||||
_onBLEError (e) {
|
||||
log.error(`BLE error: ${e}`);
|
||||
}
|
||||
|
||||
/**
|
||||
* Process the sensor data from the incoming BLE characteristic.
|
||||
* @param {object} data - the incoming BLE data.
|
||||
* @param {object} base64 - the incoming BLE data.
|
||||
* @private
|
||||
*/
|
||||
_processData (data) {
|
||||
_processBLEData (base64) {
|
||||
const data = Base64Util.base64ToUint8Array(base64);
|
||||
|
||||
this._sensors.tiltX = data[1] | (data[0] << 8);
|
||||
if (this._sensors.tiltX > (1 << 15)) this._sensors.tiltX -= (1 << 16);
|
||||
|
@ -186,45 +231,22 @@ class MicroBit {
|
|||
}
|
||||
|
||||
/**
|
||||
* React to device disconnection. May be called more than once.
|
||||
* Write a message to the device BLE session.
|
||||
* @param {number} command - the BLE command hex.
|
||||
* @param {Uint8Array} message - the message to write.
|
||||
* @private
|
||||
*/
|
||||
_onDisconnect () {
|
||||
this._disconnectEvents();
|
||||
}
|
||||
|
||||
/**
|
||||
* Send a message to the device socket.
|
||||
* @param {string} message - the name of the message, such as 'playTone'.
|
||||
* @param {object} [details] - optional additional details for the message, such as tone duration and pitch.
|
||||
* @private
|
||||
*/
|
||||
_send (message, details) {
|
||||
this._socket.emit(message, details);
|
||||
_writeBLE (command, message) {
|
||||
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 b64enc = Base64Util.uint8ArrayToBase64(output);
|
||||
this._ble.write(BLEUUID.service, BLEUUID.txChar, b64enc, 'base64');
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
* const BLE_UUIDs = {
|
||||
* uuid: '4cdbbd87d6e646c29d0bdf87551e159a',
|
||||
* rx: '4cdb8702d6e646c29d0bdf87551e159a'
|
||||
* };
|
||||
*/
|
||||
|
||||
/*
|
||||
* const DEV_SPEC = {
|
||||
* info: {
|
||||
* uuid: [BLE_UUIDs.uuid],
|
||||
* read_characteristics: {
|
||||
* '4cdb8702d6e646c29d0bdf87551e159a': {
|
||||
* notify: true
|
||||
* }
|
||||
* }
|
||||
* },
|
||||
* type: 'ble'
|
||||
* };
|
||||
*/
|
||||
|
||||
/**
|
||||
* Enum for tilt sensor direction.
|
||||
* @readonly
|
||||
|
@ -258,17 +280,6 @@ const symbols2hex = {
|
|||
'?': 0xC91004
|
||||
};
|
||||
|
||||
/**
|
||||
* Enum for micro:bit BLE command protocol.
|
||||
* @readonly
|
||||
* @enum {number}
|
||||
*/
|
||||
const BLECommand = {
|
||||
CMD_PIN_CONFIG: 0x80,
|
||||
CMD_DISPLAY_TEXT: 0x81,
|
||||
CMD_DISPLAY_LED: 0x82
|
||||
};
|
||||
|
||||
/**
|
||||
* Scratch 3.0 blocks to interact with a MicroBit device.
|
||||
*/
|
||||
|
@ -306,7 +317,8 @@ class Scratch3MicroBitBlocks {
|
|||
*/
|
||||
this.runtime = runtime;
|
||||
|
||||
this.connect();
|
||||
// Create a new MicroBit device instance
|
||||
this._device = new MicroBit(this.runtime);
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -461,44 +473,6 @@ class Scratch3MicroBitBlocks {
|
|||
};
|
||||
}
|
||||
|
||||
/**
|
||||
* Use the Device Manager client to attempt to connect to a MicroBit device.
|
||||
*/
|
||||
connect () {
|
||||
this._device = new MicroBit(null, this.runtime);
|
||||
window.addEventListener('message', event => {
|
||||
if (event.data.type === 'data') {
|
||||
this._device._processData(new Uint8Array(event.data.buffer));
|
||||
}
|
||||
}, false);
|
||||
/*
|
||||
* if (this._device || this._finder) {
|
||||
* return;
|
||||
* }
|
||||
* const deviceManager = this.runtime.ioDevices.deviceManager;
|
||||
* const finder = this._finder =
|
||||
* deviceManager.searchAndConnect(Scratch3MicroBitBlocks.EXTENSION_NAME, MicroBit.DEVICE_TYPE, DEV_SPEC);
|
||||
*
|
||||
* this._finder.promise.then(
|
||||
* socket => {
|
||||
* if (this._finder === finder) {
|
||||
* this._finder = null;
|
||||
* this._device = new MicroBit(socket, this.runtime);
|
||||
* } else {
|
||||
* log.warn('Ignoring success from stale MicroBit connection attempt');
|
||||
* }
|
||||
* },
|
||||
* reason => {
|
||||
* if (this._finder === finder) {
|
||||
* this._finder = null;
|
||||
* log.warn(`MicroBit connection failed: ${reason}`);
|
||||
* } else {
|
||||
* log.warn('Ignoring failure from stale MicroBit connection attempt');
|
||||
* }
|
||||
* });
|
||||
*/
|
||||
}
|
||||
|
||||
/**
|
||||
* Test whether the A or B button is pressed
|
||||
* @param {object} args - the block's arguments.
|
||||
|
@ -546,12 +520,7 @@ class Scratch3MicroBitBlocks {
|
|||
*/
|
||||
displayText (args) {
|
||||
const text = String(args.TEXT).substring(0, 19);
|
||||
const output = new Uint8Array(text.length + 1);
|
||||
output[0] = BLECommand.CMD_DISPLAY_TEXT;
|
||||
for (let i = 0; i < text.length; i++) {
|
||||
output[i + 1] = text.charCodeAt(i);
|
||||
}
|
||||
window.postMessage({type: 'command', buffer: output}, '*');
|
||||
this._device.displayText(text);
|
||||
return;
|
||||
}
|
||||
|
||||
|
@ -562,14 +531,12 @@ class Scratch3MicroBitBlocks {
|
|||
displaySymbol (args) {
|
||||
const hex = symbols2hex[args.SYMBOL];
|
||||
if (!hex) return;
|
||||
const output = new Uint8Array(6);
|
||||
output[0] = BLECommand.CMD_DISPLAY_LED;
|
||||
output[1] = (hex >> 20) & 0x1F;
|
||||
output[2] = (hex >> 15) & 0x1F;
|
||||
output[3] = (hex >> 10) & 0x1F;
|
||||
output[4] = (hex >> 5) & 0x1F;
|
||||
output[5] = hex & 0x1F;
|
||||
window.postMessage({type: 'command', buffer: output}, '*');
|
||||
this._device.ledMatrixState[0] = (hex >> 20) & 0x1F;
|
||||
this._device.ledMatrixState[1] = (hex >> 15) & 0x1F;
|
||||
this._device.ledMatrixState[2] = (hex >> 10) & 0x1F;
|
||||
this._device.ledMatrixState[3] = (hex >> 5) & 0x1F;
|
||||
this._device.ledMatrixState[4] = hex & 0x1F;
|
||||
this._device.displayMatrix(this._device.ledMatrixState);
|
||||
return;
|
||||
}
|
||||
|
||||
|
@ -583,7 +550,7 @@ class Scratch3MicroBitBlocks {
|
|||
} else if (args.STATE === 'off') {
|
||||
this._device.ledMatrixState[args.Y - 1] &= ~(1 << 5 - args.X);
|
||||
} else return;
|
||||
this._displayLEDs(this._device.ledMatrixState);
|
||||
this._device.displayMatrix(this._device.ledMatrixState);
|
||||
return;
|
||||
}
|
||||
|
||||
|
@ -594,23 +561,10 @@ class Scratch3MicroBitBlocks {
|
|||
for (let i = 0; i < 5; i++) {
|
||||
this._device.ledMatrixState[i] = 0;
|
||||
}
|
||||
this._displayLEDs(this._device.ledMatrixState);
|
||||
this._device.displayMatrix(this._device.ledMatrixState);
|
||||
return;
|
||||
}
|
||||
|
||||
/**
|
||||
* Send value to the micro:bit LED matrix
|
||||
* @param {Uin8array} matrix - the value to send to the matrix.
|
||||
*/
|
||||
_displayLEDs (matrix) {
|
||||
const output = new Uint8Array(matrix.length + 1);
|
||||
output[0] = BLECommand.CMD_DISPLAY_LED;
|
||||
for (let i = 0; i < matrix.length; i++) {
|
||||
output[i + 1] = matrix[i];
|
||||
}
|
||||
window.postMessage({type: 'command', buffer: output}, '*');
|
||||
}
|
||||
|
||||
/**
|
||||
* Test whether the tilt sensor is currently tilted.
|
||||
* @param {object} args - the block's arguments.
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue