diff --git a/src/extension-support/extension-manager.js b/src/extension-support/extension-manager.js index 5ea965b21..0f7cc2a84 100644 --- a/src/extension-support/extension-manager.js +++ b/src/extension-support/extension-manager.js @@ -10,6 +10,7 @@ const BlockType = require('./block-type'); const Scratch3PenBlocks = require('../extensions/scratch3_pen'); const Scratch3WeDo2Blocks = require('../extensions/scratch3_wedo2'); const Scratch3MusicBlocks = require('../extensions/scratch3_music'); +const Scratch3MicroBitBlocks = require('../extensions/scratch3_microbit'); const Scratch3TranslateBlocks = require('../extensions/scratch3_translate'); const Scratch3VideoSensingBlocks = require('../extensions/scratch3_video_sensing'); @@ -17,6 +18,7 @@ const builtinExtensions = { pen: Scratch3PenBlocks, wedo2: Scratch3WeDo2Blocks, music: Scratch3MusicBlocks, + microbit: Scratch3MicroBitBlocks, translate: Scratch3TranslateBlocks, videoSensing: Scratch3VideoSensingBlocks }; diff --git a/src/extensions/scratch3_microbit/index.js b/src/extensions/scratch3_microbit/index.js new file mode 100644 index 000000000..46b5ddf49 --- /dev/null +++ b/src/extensions/scratch3_microbit/index.js @@ -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.} + * @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;