Add micro:bit Scratch 3.0 extension (#1113)

* Add micro:bit Scratch 3.0 extension

* Fix lint errors in micro:bit extension

* Fix doc. Incorrect return type

* Check for valid pin in when pin connected block

* Drop mapping function

* Drop question mark from tilt hat block

* Generate list of symbols from object keys

* Trim display text block to max 20 characters
This commit is contained in:
Kreg Hanning 2018-05-14 13:52:49 -04:00 committed by Eric Rosenbaum
parent 846f212110
commit 82fd6f0d2f
2 changed files with 664 additions and 0 deletions

View file

@ -10,6 +10,7 @@ const BlockType = require('./block-type');
const Scratch3PenBlocks = require('../extensions/scratch3_pen'); const Scratch3PenBlocks = require('../extensions/scratch3_pen');
const Scratch3WeDo2Blocks = require('../extensions/scratch3_wedo2'); const Scratch3WeDo2Blocks = require('../extensions/scratch3_wedo2');
const Scratch3MusicBlocks = require('../extensions/scratch3_music'); const Scratch3MusicBlocks = require('../extensions/scratch3_music');
const Scratch3MicroBitBlocks = require('../extensions/scratch3_microbit');
const Scratch3TranslateBlocks = require('../extensions/scratch3_translate'); const Scratch3TranslateBlocks = require('../extensions/scratch3_translate');
const Scratch3VideoSensingBlocks = require('../extensions/scratch3_video_sensing'); const Scratch3VideoSensingBlocks = require('../extensions/scratch3_video_sensing');
@ -17,6 +18,7 @@ const builtinExtensions = {
pen: Scratch3PenBlocks, pen: Scratch3PenBlocks,
wedo2: Scratch3WeDo2Blocks, wedo2: Scratch3WeDo2Blocks,
music: Scratch3MusicBlocks, music: Scratch3MusicBlocks,
microbit: Scratch3MicroBitBlocks,
translate: Scratch3TranslateBlocks, translate: Scratch3TranslateBlocks,
videoSensing: Scratch3VideoSensingBlocks videoSensing: Scratch3VideoSensingBlocks
}; };

View file

@ -0,0 +1,662 @@
const ArgumentType = require('../../extension-support/argument-type');
const BlockType = require('../../extension-support/block-type');
const log = require('../../util/log');
/**
* Icon svg to be displayed at the left edge of each extension block, encoded as a data URI.
* @type {string}
*/
// eslint-disable-next-line max-len
const blockIconURI = '';
/**
* Icon svg to be displayed in the menu encoded as a data URI.
* @type {string}
*/
// eslint-disable-next-line max-len
const menuIconURI = '';
/**
* Manage communication with a MicroBit device over a Device Manager 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;
/**
* The Scratch 3.0 runtime used to trigger the green flag button
*
* @type {Runtime}
* @private
*/
this._runtime = runtime;
/**
* The most recently received value for each sensor.
* @type {Object.<string, number>}
* @private
*/
this._sensors = {
tiltX: 0,
tiltY: 0,
buttonA: 0,
buttonB: 0,
touchPins: [0, 0, 0],
gestureState: 0,
ledMatrixState: new Uint8Array(5)
};
this._gestures = {
moving: false,
move: {
active: false,
timeout: false
},
shake: {
active: false,
timeout: false
},
jump: {
active: false,
timeout: false
}
};
// this._onRxChar = this._onRxChar.bind(this);
// this._onDisconnect = this._onDisconnect.bind(this);
this._connectEvents();
}
/**
* Manually dispose of this object.
*/
dispose () {
this._disconnectEvents();
}
/**
* @return {number} - the latest value received for the tilt sensor's tilt about the X axis.
*/
get tiltX () {
return this._sensors.tiltX;
}
/**
* @return {number} - the latest value received for the tilt sensor's tilt about the Y axis.
*/
get tiltY () {
return this._sensors.tiltY;
}
/**
* @return {boolean} - the latest value received for the A button.
*/
get buttonA () {
return this._sensors.buttonA;
}
/**
* @return {boolean} - the latest value received for the B button.
*/
get buttonB () {
return this._sensors.buttonB;
}
/**
* @return {number} - the latest value received for the motion gesture states.
*/
get gestureState () {
return this._sensors.gestureState;
}
/**
* @return {Uint8Array} - the current state of the 5x5 LED matrix.
*/
get ledMatrixState () {
return this._sensors.ledMatrixState;
}
/**
* @param {number} pin - the pin to check touch state.
* @return {number} - the latest value received for the touch pin states.
*/
_checkPinState (pin) {
return this._sensors.touchPins[pin];
}
/**
* Attach event handlers to the device socket.
* @private
*/
_connectEvents () {
// this._socket.on(BLE_UUIDs.rx, this._onRxChar);
// this._socket.on('deviceWasClosed', this._onDisconnect);
// this._socket.on('disconnect', this._onDisconnect);
}
/**
* Detach event handlers from the device socket.
* @private
*/
_disconnectEvents () {
// this._socket.off(BLE_UUIDs.rx, this._onRxChar);
// this._socket.off('deviceWasClosed', this._onDisconnect);
// this._socket.off('disconnect', this._onDisconnect);
}
/**
* Process the sensor data from the incoming BLE characteristic.
* @param {object} data - the incoming BLE data.
* @private
*/
_processData (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];
}
/**
* React to device disconnection. May be called more than once.
* @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);
}
}
/*
* 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
* @enum {string}
*/
const TiltDirection = {
FRONT: 'front',
BACK: 'back',
LEFT: 'left',
RIGHT: 'right',
ANY: 'any'
};
/**
* Converting symbols to hex values
* @readonly
*/
const symbols2hex = {
'❤': 0xAAC544,
'♫': 0xF4AF78,
'☓': 0x1151151,
'✓': 0x8A88,
'↑': 0x477C84,
'↓': 0x427DC4,
'←': 0x467D84,
'→': 0x437CC4,
'◯': 0xE8C62E,
'☀': 0x1577DD5,
'☺': 0x5022E,
'!': 0x421004,
'?': 0xC91004
};
/**
* Scratch 3.0 blocks to interact with a MicroBit device.
*/
class Scratch3MicroBitBlocks {
/**
* @return {string} - the name of this extension.
*/
static get EXTENSION_NAME () {
return 'MicroBit';
}
/**
* @return {string} - the ID of this extension.
*/
static get EXTENSION_ID () {
return 'microbit';
}
/**
* @return {number} - the tilt sensor counts as "tilted" if its tilt angle meets or exceeds this threshold.
*/
static get TILT_THRESHOLD () {
return 15;
}
/**
* Construct a set of MicroBit blocks.
* @param {Runtime} runtime - the Scratch 3.0 runtime.
*/
constructor (runtime) {
/**
* The Scratch 3.0 runtime.
* @type {Runtime}
*/
this.runtime = runtime;
this.connect();
}
/**
* @returns {object} metadata for this extension and its blocks.
*/
getInfo () {
return {
id: Scratch3MicroBitBlocks.EXTENSION_ID,
name: Scratch3MicroBitBlocks.EXTENSION_NAME,
menuIconURI: menuIconURI,
blockIconURI: blockIconURI,
blocks: [
{
opcode: 'whenButtonPressed',
text: 'when [BTN] button pressed',
blockType: BlockType.HAT,
arguments: {
BTN: {
type: ArgumentType.STRING,
menu: 'buttons',
defaultValue: 'A'
}
}
},
{
opcode: 'whenMoved',
text: 'when moved',
blockType: BlockType.HAT
},
{
opcode: 'whenShaken',
text: 'when shaken',
blockType: BlockType.HAT
},
{
opcode: 'whenJumped',
text: 'when jumped',
blockType: BlockType.HAT
},
{
opcode: 'displayText',
text: 'display [TEXT]',
blockType: BlockType.COMMAND,
arguments: {
TEXT: {
type: ArgumentType.STRING,
defaultValue: 'Hello!'
}
}
},
{
opcode: 'displaySymbol',
text: 'display [SYMBOL]',
blockType: BlockType.COMMAND,
arguments: {
SYMBOL: {
type: ArgumentType.STRING,
menu: 'symbols',
defaultValue: '❤'
}
}
},
{
opcode: 'displayMatrix',
text: 'set light x:[X] y:[Y] [STATE]',
blockType: BlockType.COMMAND,
arguments: {
X: {
type: ArgumentType.STRING,
menu: 'rowcol',
defaultValue: '1'
},
Y: {
type: ArgumentType.STRING,
menu: 'rowcol',
defaultValue: '1'
},
STATE: {
type: ArgumentType.STRING,
menu: 'pinState',
defaultValue: 'on'
}
}
},
{
opcode: 'displayClear',
text: 'set all lights off',
blockType: BlockType.COMMAND
},
{
opcode: 'whenTilted',
text: 'when tilted [DIRECTION]',
blockType: BlockType.HAT,
arguments: {
DIRECTION: {
type: ArgumentType.STRING,
menu: 'tiltDirectionAny',
defaultValue: TiltDirection.ANY
}
}
},
{
opcode: 'isTilted',
text: 'tilted [DIRECTION]?',
blockType: BlockType.BOOLEAN,
arguments: {
DIRECTION: {
type: ArgumentType.STRING,
menu: 'tiltDirectionAny',
defaultValue: TiltDirection.ANY
}
}
},
{
opcode: 'getTiltAngle',
text: 'tilt angle [DIRECTION]',
blockType: BlockType.REPORTER,
arguments: {
DIRECTION: {
type: ArgumentType.STRING,
menu: 'tiltDirection',
defaultValue: TiltDirection.FRONT
}
}
},
{
opcode: 'whenPinConnected',
text: 'when pin [PIN] connected',
blockType: BlockType.HAT,
arguments: {
PIN: {
type: ArgumentType.STRING,
menu: 'touchPins',
defaultValue: '0'
}
}
}
],
menus: {
buttons: ['A', 'B', 'any'],
rowcol: ['1', '2', '3', '4', '5'],
pinState: ['on', 'off'],
symbols: Object.keys(symbols2hex),
tiltDirection: [TiltDirection.FRONT, TiltDirection.BACK, TiltDirection.LEFT, TiltDirection.RIGHT],
tiltDirectionAny: [
TiltDirection.FRONT, TiltDirection.BACK, TiltDirection.LEFT,
TiltDirection.RIGHT, TiltDirection.ANY
],
touchPins: ['0', '1', '2']
}
};
}
/**
* 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.
* @return {boolean} - true if the button is pressed.
*/
whenButtonPressed (args) {
if (args.BTN === 'any') {
return this._device.buttonA | this._device.buttonB;
} else if (args.BTN === 'A') {
return this._device.buttonA;
} else if (args.BTN === 'B') {
return this._device.buttonB;
}
return false;
}
/**
* Test whether the micro:bit is moving
* @return {boolean} - true if the micro:bit is moving.
*/
whenMoved () {
return (this._device.gestureState >> 2) & 1;
}
/**
* Test whether the micro:bit is shaken
* @return {boolean} - true if the micro:bit is shaken.
*/
whenShaken () {
return this._device.gestureState & 1;
}
/**
* Test whether the micro:bit is free falling
* @return {boolean} - true if the micro:bit is free falling.
*/
whenJumped () {
return (this._device.gestureState >> 1) & 1;
}
/**
* Display text on the 5x5 LED matrix.
* @param {object} args - the block's arguments.
* Note the limit is 20 characters
*/
displayText (args) {
const text = String(args.TEXT).substring(0, 20);
window.postMessage({type: 'command', uuid: 'text', buffer: text}, '*');
return;
}
/**
* Display a predefined symbol on the 5x5 LED matrix.
* @param {object} args - the block's arguments.
*/
displaySymbol (args) {
const hex = symbols2hex[args.SYMBOL];
const output = new Uint8Array(5);
output[0] = (hex >> 20) & 0x1F;
output[1] = (hex >> 15) & 0x1F;
output[2] = (hex >> 10) & 0x1F;
output[3] = (hex >> 5) & 0x1F;
output[4] = hex & 0x1F;
window.postMessage({type: 'command', uuid: 'matrix', buffer: output}, '*');
return;
}
/**
* Control individual LEDs on the 5x5 matrix.
* @param {object} args - the block's arguments.
*/
displayMatrix (args) {
if (args.STATE === 'on') {
this._device.ledMatrixState[args.Y - 1] |= 1 << 5 - args.X;
} else if (args.STATE === 'off') {
this._device.ledMatrixState[args.Y - 1] &= ~(1 << 5 - args.X);
} else return;
window.postMessage({type: 'command', uuid: 'matrix', buffer: this._device.ledMatrixState}, '*');
return;
}
/**
* Turn all 5x5 matrix LEDs off.
*/
displayClear () {
for (let i = 0; i < 5; i++) {
this._device.ledMatrixState[i] = 0;
}
window.postMessage({type: 'command', uuid: 'matrix', buffer: this._device.ledMatrixState}, '*');
return;
}
/**
* Test whether the tilt sensor is currently tilted.
* @param {object} args - the block's arguments.
* @property {TiltDirection} DIRECTION - the tilt direction to test (front, back, left, right, or any).
* @return {boolean} - true if the tilt sensor is tilted past a threshold in the specified direction.
*/
whenTilted (args) {
return this._isTilted(args.DIRECTION);
}
/**
* Test whether the tilt sensor is currently tilted.
* @param {object} args - the block's arguments.
* @property {TiltDirection} DIRECTION - the tilt direction to test (front, back, left, right, or any).
* @return {boolean} - true if the tilt sensor is tilted past a threshold in the specified direction.
*/
isTilted (args) {
return this._isTilted(args.DIRECTION);
}
/**
* @param {object} args - the block's arguments.
* @property {TiltDirection} DIRECTION - the direction (front, back, left, right) to check.
* @return {number} - the tilt sensor's angle in the specified direction.
* Note that getTiltAngle(front) = -getTiltAngle(back) and getTiltAngle(left) = -getTiltAngle(right).
*/
getTiltAngle (args) {
return this._getTiltAngle(args.DIRECTION);
}
/**
* Test whether the tilt sensor is currently tilted.
* @param {TiltDirection} direction - the tilt direction to test (front, back, left, right, or any).
* @return {boolean} - true if the tilt sensor is tilted past a threshold in the specified direction.
* @private
*/
_isTilted (direction) {
switch (direction) {
case TiltDirection.ANY:
return (Math.abs(this._device.tiltX / 10) >= Scratch3MicroBitBlocks.TILT_THRESHOLD) ||
(Math.abs(this._device.tiltY / 10) >= Scratch3MicroBitBlocks.TILT_THRESHOLD);
default:
return this._getTiltAngle(direction) >= Scratch3MicroBitBlocks.TILT_THRESHOLD;
}
}
/**
* @param {TiltDirection} direction - the direction (front, back, left, right) to check.
* @return {number} - the tilt sensor's angle in the specified direction.
* Note that getTiltAngle(front) = -getTiltAngle(back) and getTiltAngle(left) = -getTiltAngle(right).
* @private
*/
_getTiltAngle (direction) {
switch (direction) {
case TiltDirection.FRONT:
return Math.round(this._device.tiltY / -10);
case TiltDirection.BACK:
return Math.round(this._device.tiltY / 10);
case TiltDirection.LEFT:
return Math.round(this._device.tiltX / -10);
case TiltDirection.RIGHT:
return Math.round(this._device.tiltX / 10);
default:
log.warn(`Unknown tilt direction in _getTiltAngle: ${direction}`);
}
}
/**
* @param {object} args - the block's arguments.
* @return {boolean} - the touch pin state.
* @private
*/
whenPinConnected (args) {
const pin = parseInt(args.PIN, 10);
if (isNaN(pin)) return;
if (pin < 0 || pin > 2) return false;
return this._device._checkPinState(pin);
}
}
module.exports = Scratch3MicroBitBlocks;