diff --git a/src/extension-support/extension-manager.js b/src/extension-support/extension-manager.js index 7cb556c5d0d..989f3fcb20b 100644 --- a/src/extension-support/extension-manager.js +++ b/src/extension-support/extension-manager.js @@ -17,6 +17,9 @@ const builtinExtensions = { wedo2: () => require('../extensions/scratch3_wedo2'), music: () => require('../extensions/scratch3_music'), microbit: () => require('../extensions/scratch3_microbit'), + jdcode: () => require('../extensions/scratch3_jdcode'), + jcboard: () => require('../extensions/scratch3_jcboard'), + uglybot: () => require('../extensions/scratch3_uglybot'), text2speech: () => require('../extensions/scratch3_text2speech'), translate: () => require('../extensions/scratch3_translate'), videoSensing: () => require('../extensions/scratch3_video_sensing'), diff --git a/src/extensions/scratch3_jcboard/index.js b/src/extensions/scratch3_jcboard/index.js new file mode 100644 index 00000000000..a8ee24e0232 --- /dev/null +++ b/src/extensions/scratch3_jcboard/index.js @@ -0,0 +1,470 @@ +const ArgumentType = require('../../extension-support/argument-type'); +const BlockType = require('../../extension-support/block-type'); +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'); + + +// eslint-disable-next-line max-len +const blockIconURI = ''; + + + +const BLETimeout = 4500; +const BLESendInterval = 20; +const BLEDataStoppedError = 'JCBoard extension stopped receiving data'; + + +const BLEUUID = { + service: 0x2262, + rxChar: '00000227-0000-1000-8000-00805f9b34fb', + txChar: '00000227-0000-1000-8000-00805f9b34fb' +}; + + +class JCBoard { + constructor (runtime, extensionId) { + this._runtime = runtime; + this._ble = null; + this._runtime.registerPeripheralExtension(extensionId, this); + this._extensionId = extensionId; + this._runtime.on('PROJECT_STOP_ALL', this.stopAll.bind(this)); + this._timeoutID = null; + this._busy = false; + this._busyTimeoutID = null; + this.reset = this.reset.bind(this); + this._onConnect = this._onConnect.bind(this); + this._onMessage = this._onMessage.bind(this); + this.rxData = [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0]; + this.txData = new Uint8Array(20); + this._sensors = { + button: 0, + ir: [0, 0, 0], + ultrasonic: 0, + joystick: [0, 0], + tilt: [0, 0], + sound: 0, + illum:0 + }; + } + + stopAll () { + for (let i = 0; i < 20; i++) + this.txData[i] = 0; + this.send(this.txData); + } + + scan () { + if (this._ble) { + this._ble.disconnect(); + } + this._ble = new BLE(this._runtime, this._extensionId, { + filters: [ + {services: [BLEUUID.service]} + ] + }, this._onConnect, this.reset); + } + + connect (id) { + if (this._ble) { + this._ble.connectPeripheral(id); + } + } + + disconnect () { + if (this._ble) { + this._ble.disconnect(); + } + this.reset(); + } + + reset () { + if (this._timeoutID) { + window.clearTimeout(this._timeoutID); + this._timeoutID = null; + } + } + + isConnected () { + + let connected = false; + if (this._ble) { + connected = this._ble.isConnected(); + } + return connected; + } + + send (message) { + if (!this.isConnected()) return; + + console.log(message); + this._ble.write(BLEUUID.service, BLEUUID.txChar, message, 'ASCII', true).then( + () => { + this._busy = false; + window.clearTimeout(this._busyTimeoutID); + } + ); + } + + _onConnect () { + this._ble.read(BLEUUID.service, BLEUUID.rxChar, true, this._onMessage); + this._timeoutID = window.setTimeout( + () => this._ble.handleDisconnectError(BLEDataStoppedError), + BLETimeout + ); + } + + _onMessage (strData) { + this.rxData = strData.split(','); + window.clearTimeout(this._timeoutID); + this._timeoutID = window.setTimeout( + () => this._ble.handleDisconnectError(BLEDataStoppedError), + BLETimeout + ); + } + + get getRxData() { + return this.rxData; + } + + get getTxData() { + return this.rxData; + } +} + +class Scratch3JCBoardBlocks { + static get EXTENSION_NAME () { + return 'JCBoard'; + } + static get EXTENSION_ID () { + return 'jcboard'; + } + + constructor (runtime) { + + this.runtime = runtime; + this._peripheral = new JCBoard(this.runtime, Scratch3JCBoardBlocks.EXTENSION_ID); + this.message = this._peripheral.txData; + this.tuneID = 0; + this.moveID = 0; + this.rotID = 0; + } + + + getInfo () { + return { + id: Scratch3JCBoardBlocks.EXTENSION_ID, + name: Scratch3JCBoardBlocks.EXTENSION_NAME, + blockIconURI: blockIconURI, + showStatusButton: true, + blocks: [ + { + opcode: 'ultrasonic', + text: formatMessage({ + id: 'ultrasonic', + default: '[ONEFIVE] \uD3EC\uD2B8\uB97C \uCD08\uC74C\uD30C\uC13C\uC11C\uB85C \uC0AC\uC6A9', + }), + blockType: BlockType.COMMAND, + arguments: { + ONEFIVE: { + type: ArgumentType.STRING, + menu: 'onefive', + defaultValue: '1\uBC88' + } + } + }, + { + opcode: 'led', + text: formatMessage({ + id: 'led', + default: '[ONETWO] LED [ONOFF]', + }), + blockType: BlockType.COMMAND, + arguments: { + ONETWO: { + type: ArgumentType.STRING, + menu: 'onetwo', + defaultValue: '1\uBC88' + }, + ONOFF: { + type: ArgumentType.STRING, + menu: 'onoff', + defaultValue: '\uCF1C\uAE30' + } + } + }, + { + opcode: 'buzzer', + text: formatMessage({ + id: 'buzzer', + default: '[TUNE] \uC74C\uC744 [DELAY] \uCD08\uB3D9\uC548 \uC18C\uB9AC\uB0B4\uAE30', + }), + blockType: BlockType.COMMAND, + arguments: { + TUNE: { + type: ArgumentType.STRING, + menu: 'tune', + defaultValue: '\uB3C4' + }, + DELAY: { + type: ArgumentType.STRING, + menu: 'delay', + defaultValue: '1' + } + } + }, + { + opcode: 'motor', + text: formatMessage({ + id: 'motor', + default: '[ONETWO] DC\uBAA8\uD130\uB97C [TEXT] \uC138\uAE30\uB85C \uD68C\uC804', + }), + blockType: BlockType.COMMAND, + arguments: { + ONETWO: { + type: ArgumentType.STRING, + menu: 'onetwo', + defaultValue: '1\uBC88' + }, + TEXT: { + type: ArgumentType.STRING, + defaultValue: '0' + } + } + }, + { + opcode: 'servo', + text: formatMessage({ + id: 'servo', + default: '[ONEFOUR] \uC11C\uBCF4\uBAA8\uD130\uB97C [TEXT] \uB3C4 \uD68C\uC804', + }), + blockType: BlockType.COMMAND, + arguments: { + ONEFOUR: { + type: ArgumentType.STRING, + menu: 'onefour', + defaultValue: '1\uBC88' + }, + TEXT: { + type: ArgumentType.STRING, + defaultValue: '0' + } + } + }, + { + opcode: 'digitalpin', + text: formatMessage({ + id: 'digitalpin', + default: '[ONEFIVE] \uB514\uC9C0\uD138\uD540\uC744 [HIGHLOW] \uB85C \uC124\uC815', + }), + blockType: BlockType.COMMAND, + arguments: { + ONEFIVE: { + type: ArgumentType.STRING, + menu: 'onefive', + defaultValue: '1\uBC88' + }, + HIGHLOW: { + type: ArgumentType.STRING, + menu: 'highlow', + defaultValue: 'HIGH' + }, + } + }, + + '---', + + { + opcode: 'getbutton', + text: formatMessage({ + id: 'getbutton', + default: '[ONETWO] \uBC84\uD2BC \uAC12', + }), + blockType: BlockType.REPORTER, + arguments: { + ONETWO: { + type: ArgumentType.STRING, + menu: 'onetwo', + defaultValue: '1\uBC88' + }, + } + }, + { + opcode: 'getanalog', + text: formatMessage({ + id: 'getanalog', + default: '[ONEFIVE] \uC544\uB0A0\uB85C\uADF8 \uAC12', + }), + blockType: BlockType.REPORTER, + arguments: { + ONEFIVE: { + type: ArgumentType.STRING, + menu: 'onefive', + defaultValue: '1\uBC88' + }, + } + }, + ], + menus: { + onoff: { + acceptReporters: true, + items: ['\uCF1C\uAE30', '\uB044\uAE30'] //Äѱ⠲ô±â + }, + tune: { + acceptReporters: true, + items: ['\uB3C4', '\uB808', '\uBBF8', '\uD30C', '\uC194', '\uB77C', '\uC2DC'] //µµ·¹¹ÌÆļֶó½Ã + }, + delay: { + acceptReporters: true, + items: ['0.5', '0.8', '1', '2', '3', '4', '5'] + }, + onetwo: { + acceptReporters: true, + items: ['1\uBC88', '2\uBC88'] + }, + onefour: { + acceptReporters: true, + items: ['1\uBC88', '2\uBC88', '3\uBC88', '4\uBC88'] + }, + onefive: { + acceptReporters: true, + items: ['1\uBC88', '2\uBC88', '3\uBC88', '4\uBC88', '5\uBC88'] + }, + highlow: { + acceptReporters: true, + items: ['HIGH', 'LOW'] + }, + } + }; + } + led (args) { + if(args.ONOFF == '\uCF1C\uAE30'){ + if(args.ONETWO == '1\uBC88') + this.message[0] |= 0x11; + else + this.message[0] |= 0x12; + } + else{ + if(args.ONETWO == '1\uBC88') + this.message[0] &= ~0x01; + else + this.message[0] &= ~0x02; + } + this._peripheral.send(this.message); + return new Promise(resolve => { + setTimeout(() => { + resolve(); + }, BLESendInterval); + }); + } + digitalpin(args){ + if(args.HIGHLOW == 'HIGH'){ + if(args.ONEFIVE == '1\uBC88') this.message[1] |= 0x01; + if(args.ONEFIVE == '2\uBC88') this.message[1] |= 0x02; + if(args.ONEFIVE == '3\uBC88') this.message[1] |= 0x04; + if(args.ONEFIVE == '4\uBC88') this.message[1] |= 0x08; + if(args.ONEFIVE == '5\uBC88') this.message[1] |= 0x10; + } + else{ + if(args.ONEFIVE == '1\uBC88') this.message[1] &= ~0x01; + if(args.ONEFIVE == '2\uBC88') this.message[1] &= ~0x02; + if(args.ONEFIVE == '3\uBC88') this.message[1] &= ~0x04; + if(args.ONEFIVE == '4\uBC88') this.message[1] &= ~0x08; + if(args.ONEFIVE == '5\uBC88') this.message[1] &= ~0x10; + } + + this._peripheral.send(this.message); + return new Promise(resolve => { + setTimeout(() => { + resolve(); + }, BLESendInterval); + }); + } + buzzer(args){ + this.tuneID = (this.tuneID+1)%16; + if(args.TUNE == '\uB3C4') this.message[2] = 1; + if(args.TUNE == '\uB808') this.message[2] = 2; + if(args.TUNE == '\uBBF8') this.message[2] = 3; + if(args.TUNE == '\uD30C') this.message[2] = 4; + if(args.TUNE == '\uC194') this.message[2] = 5; + if(args.TUNE == '\uB77C') this.message[2] = 6; + if(args.TUNE == '\uC2DC') this.message[2] = 7; + this.message[2] |= (this.tuneID<<4); + + if(args.DELAY == '0.5') this.message[3] = 5; + if(args.DELAY == '0.8') this.message[3] = 8; + if(args.DELAY == '1') this.message[3] = 10; + if(args.DELAY == '2') this.message[3] = 20; + if(args.DELAY == '3') this.message[3] = 30; + if(args.DELAY == '4') this.message[3] = 40; + if(args.DELAY == '5') this.message[3] = 50; + + this._peripheral.send(this.message); + return new Promise(resolve => { + setTimeout(() => { + resolve(); + }, BLESendInterval); + }); + } + + motor (args){ + if(args.ONETWO == '1\uBC88') + this.message[4] = args.TEXT; + else + this.message[5] = args.TEXT; + + this._peripheral.send(this.message); + return new Promise(resolve => { + setTimeout(() => { + resolve(); + }, BLESendInterval); + }); + } + + servo (args){ + if(args.ONEFOUR == '1\uBC88') this.message[6] = args.TEXT; + if(args.ONEFOUR == '2\uBC88') this.message[7] = args.TEXT; + if(args.ONEFOUR == '3\uBC88') this.message[8] = args.TEXT; + if(args.ONEFOUR == '4\uBC88') this.message[9] = args.TEXT; + + this._peripheral.send(this.message); + return new Promise(resolve => { + setTimeout(() => { + resolve(); + }, BLESendInterval); + }); + } + ultrasonic (args) { + if(args.ONEFIVE == '1\uBC88') this.message[10] = 0x01; + if(args.ONEFIVE == '2\uBC88') this.message[10] = 0x02; + if(args.ONEFIVE == '3\uBC88') this.message[10] = 0x04; + if(args.ONEFIVE == '4\uBC88') this.message[10] = 0x08; + if(args.ONEFIVE == '5\uBC88') this.message[10] = 0x10; + + this._peripheral.send(this.message); + return new Promise(resolve => { + setTimeout(() => { + resolve(); + }, BLESendInterval); + }); + } + + getbutton(args){ + var rxData = this._peripheral.rxData; + if(args.ONETWO == '1\uBC88') return rxData[0]&0x01; + if(args.ONETWO == '2\uBC88') return (rxData[0]>>1)&0x01; + } + + getanalog(args){ + var rxData = this._peripheral.rxData; + if(args.ONEFIVE == '1\uBC88') return rxData[1]; + if(args.ONEFIVE == '2\uBC88') return rxData[2]; + if(args.ONEFIVE == '3\uBC88') return rxData[3]; + if(args.ONEFIVE == '4\uBC88') return rxData[4]; + if(args.ONEFIVE == '5\uBC88') return rxData[5]; + } +} + +module.exports = Scratch3JCBoardBlocks; diff --git a/src/extensions/scratch3_jdcode/index.js b/src/extensions/scratch3_jdcode/index.js new file mode 100644 index 00000000000..100b5855285 --- /dev/null +++ b/src/extensions/scratch3_jdcode/index.js @@ -0,0 +1,560 @@ +const ArgumentType = require('../../extension-support/argument-type'); +const BlockType = require('../../extension-support/block-type'); +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'); + + +// eslint-disable-next-line max-len +const blockIconURI = ''; + + + +const BLETimeout = 4500; +const BLESendInterval = 20; +const BLEDataStoppedError = 'JDCode extension stopped receiving data'; + + +const BLEUUID = { + service: 0x2261, + rxChar: '00000227-0000-1000-8000-00805f9b34fb', + txChar: '00000227-0000-1000-8000-00805f9b34fb' +}; + + +class JDCode { + constructor (runtime, extensionId) { + this._runtime = runtime; + this._ble = null; + this._runtime.registerPeripheralExtension(extensionId, this); + this._extensionId = extensionId; + this._runtime.on('PROJECT_STOP_ALL', this.stopAll.bind(this)); + this._timeoutID = null; + this._busy = false; + this._busyTimeoutID = null; + this.reset = this.reset.bind(this); + this._onConnect = this._onConnect.bind(this); + this._onMessage = this._onMessage.bind(this); + this.rxData = [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0]; + this.txData = new Int16Array(7); + this.txArray = new Uint16Array(14); + this.moveX = 0; + this.moveY = 0; + this.rotation = 0; + this._sensors = { + button: 0, + ir: [0, 0, 0], + ultrasonic: 0, + joystick: [0, 0], + tilt: [0, 0], + sound: 0, + illum:0 + }; + this.txData[5] = 100; + this.txData[6] = 100; + } + + stopAll () { + for (let i = 0; i < 7; i++) + this.txData[i] = 0; + this.txData[5] = 100; + this.txData[6] = 100; + this.send(this.txData); + this.moveX = 0; + this.moveY = 0; + this.rotation = 0; + } + + scan () { + if (this._ble) { + this._ble.disconnect(); + } + this._ble = new BLE(this._runtime, this._extensionId, { + filters: [ + {services: [BLEUUID.service]} + ] + }, this._onConnect, this.reset); + } + + connect (id) { + if (this._ble) { + this._ble.connectPeripheral(id); + } + } + + disconnect () { + if (this._ble) { + this._ble.disconnect(); + } + this.reset(); + } + + reset () { + if (this._timeoutID) { + window.clearTimeout(this._timeoutID); + this._timeoutID = null; + } + } + + isConnected () { + + let connected = false; + if (this._ble) { + connected = this._ble.isConnected(); + } + return connected; + } + + send (message) { + if (!this.isConnected()) return; + for(n=0;n<7;n++){ + this.txArray[n*2] = message[n]&0xFF; + this.txArray[n*2+1] = (message[n]>>8)&0xFF; + } + console.log(this.txArray); + this._ble.write(BLEUUID.service, BLEUUID.txChar, this.txArray, 'ASCII', true).then( + () => { + this._busy = false; + window.clearTimeout(this._busyTimeoutID); + } + ); + } + + _onConnect () { + this._ble.read(BLEUUID.service, BLEUUID.rxChar, true, this._onMessage); + this._timeoutID = window.setTimeout( + () => this._ble.handleDisconnectError(BLEDataStoppedError), + BLETimeout + ); + } + + _onMessage (strData) { + this.rxData = strData.split(','); + window.clearTimeout(this._timeoutID); + this._timeoutID = window.setTimeout( + () => this._ble.handleDisconnectError(BLEDataStoppedError), + BLETimeout + ); + } + + get getRxData() { + return this.rxData; + } + + get getTxData() { + return this.rxData; + } +} + +class Scratch3JDCodeBlocks { + static get EXTENSION_NAME () { + return 'JDCode'; + } + static get EXTENSION_ID () { + return 'jdcode'; + } + + constructor (runtime) { + + this.runtime = runtime; + this._peripheral = new JDCode(this.runtime, Scratch3JDCodeBlocks.EXTENSION_ID); + this.message = this._peripheral.txData; + this.tuneID = 0; + this.moveID = 0; + this.rotID = 0; + } + + + getInfo () { + return { + id: Scratch3JDCodeBlocks.EXTENSION_ID, + name: Scratch3JDCodeBlocks.EXTENSION_NAME, + blockIconURI: blockIconURI, + showStatusButton: true, + blocks: [ + { + opcode: 'takeoff', + text: formatMessage({ + id: 'takeoff', + default: '\uB4DC\uB860 \uC774\uB959\uD558\uAE30', + }), + }, + { + opcode: 'landing', + text: formatMessage({ + id: 'landing', + default: '\uB4DC\uB860 \uCC29\uB959\uD558\uAE30', + }), + }, + { + opcode: 'alt', + text: formatMessage({ + id: 'alt', + default: '[TEXT] cm \uB192\uC774\uB85C \uBE44\uD589', + }), + blockType: BlockType.COMMAND, + arguments: { + TEXT: { + type: ArgumentType.STRING, + defaultValue: '70' + } + } + }, + + { + opcode: 'velocity', + text: formatMessage({ + id: 'velocity', + default: '[FBRL] (\uC73C)\uB85C [TEXT] \uC18D\uB3C4(cm/s)\uB85C \uBE44\uD589', + }), + blockType: BlockType.COMMAND, + arguments: { + FBRL: { + type: ArgumentType.STRING, + menu: 'fbrl', + defaultValue: '\uC55E' + }, + TEXT: { + type: ArgumentType.STRING, + defaultValue: '70' + } + } + }, + + { + opcode: 'move', + text: formatMessage({ + id: 'move', + default: '[FBRL](\uC73C)\uB85C [TEXT1]cm \uAC70\uB9AC\uB97C [TEXT2] \uC18D\uB3C4(cm/s)\uB85C \uBE44\uD589', + }), + blockType: BlockType.COMMAND, + arguments: { + FBRL: { + type: ArgumentType.STRING, + menu: 'fbrl', + defaultValue: '\uC55E' + }, + TEXT1: { + type: ArgumentType.STRING, + defaultValue: '100' + }, + TEXT2: { + type: ArgumentType.STRING, + defaultValue: '70' + } + } + }, + + { + opcode: 'rotation', + text: formatMessage({ + id: 'rotation', + default: '[ROTDIR]\uC73C\uB85C [TEXT1] \uB3C4\uB97C [TEXT2]\uAC01\uC18D\uB3C4(deg/s)\uB85C \uD68C\uC804', + }), + blockType: BlockType.COMMAND, + arguments: { + ROTDIR: { + type: ArgumentType.STRING, + menu: 'rotdir', + defaultValue: '\uC2DC\uACC4\uBC29\uD5A5' + }, + TEXT1: { + type: ArgumentType.STRING, + defaultValue: '90' + }, + TEXT2: { + type: ArgumentType.STRING, + defaultValue: '70' + } + } + }, + { + opcode: 'proprot', + text: formatMessage({ + id: 'proprot', + default: '\uD504\uB85C\uD3A0\uB7EC\uB97C [TEXT]\uC138\uAE30\uB85C \uB3CC\uB9AC\uAE30', + }), + blockType: BlockType.COMMAND, + arguments: { + TEXT: { + type: ArgumentType.STRING, + defaultValue: '0' + } + } + }, + { + opcode: 'motorot', + text: formatMessage({ + id: 'motorot', + default: '[LTRB]\uBAA8\uD130\uB97C [TEXT]\uC138\uAE30\uB85C \uB3CC\uB9AC\uAE30', + }), + blockType: BlockType.COMMAND, + arguments: { + LTRB: { + type: ArgumentType.STRING, + menu: 'ltrb', + defaultValue: '\uC67C\uCABD\uC544\uB798' + }, + TEXT: { + type: ArgumentType.STRING, + defaultValue: '0' + } + } + }, + { + opcode: 'emergency', + text: formatMessage({ + id: 'emergency', + default: '\uB4DC\uB860\uBE44\uD589\uC744 \uC989\uC2DC \uBA48\uCDA4', + }), + }, + '---', + { + opcode: 'getready', + text: formatMessage({ + id: 'getready', + default: '\uB4DC\uB860 \uBE44\uD589 \uC900\uBE44 \uC0C1\uD0DC', + }), + blockType: BlockType.REPORTER, + }, + { + opcode: 'getbattery', + text: formatMessage({ + id: 'getbattery', + default: '\uBC30\uD130\uB9AC(%)', + }), + blockType: BlockType.REPORTER, + }, + { + opcode: 'getalt', + text: formatMessage({ + id: 'getalt', + default: '\uB4DC\uB860 \uB192\uC774', + }), + blockType: BlockType.REPORTER, + }, + { + opcode: 'gettilt', + text: formatMessage({ + id: 'gettilt', + default: '\uB4DC\uB860[FBLR] \uAE30\uC6B8\uAE30', + }), + blockType: BlockType.REPORTER, + arguments: { + FBLR: { + type: ArgumentType.STRING, + menu: 'fblr', + defaultValue: '\uC55E\uB4A4' + }, + } + }, + { + opcode: 'getmove', + text: formatMessage({ + id: 'getmove', + default: '\uB4DC\uB860[FBLR] \uC774\uB3D9', + }), + blockType: BlockType.REPORTER, + arguments: { + FBLR: { + type: ArgumentType.STRING, + menu: 'fblr', + defaultValue: '\uC55E\uB4A4' + }, + } + }, + ], + menus: { + fbrl: { + acceptReporters: true, + items: ['\uC55E', '\uB4A4','\uC624\uB978\uCABD', '\uC67C\uCABD'] //¾Õ µÚ ¿À¸¥ÂÊ ¿ÞÂÊ + }, + rotdir: { + acceptReporters: true, + items: ['\uC2DC\uACC4\uBC29\uD5A5', '\uBC18\uC2DC\uACC4\uBC29\uD5A5'] //½Ã°è¹æÇ⠹ݽðè¹æÇâ + }, + fblr: { + acceptReporters: true, + items: ['\uC55E\uB4A4', '\uC88C\uC6B0'] // ¾ÕµÚ Á¿ì + }, + ltrb: { + acceptReporters: true, + items: ['\uC67C\uCABD\uC544\uB798', '\uC67C\uCABD\uC704', '\uC624\uB978\uCABD\uC544\uB798', '\uC624\uB978\uCABD\uC704'] + //¿ÞÂʾƷ¡, ¿ÞÂÊ À§, ¿À¸¥ÂʾƷ¡, ¿À¸¥ÂÊÀ§ + }, + } + }; + } + takeoff () { + this.message[3] = 70; + this.message[4] = 0x2F; + this._peripheral.send(this.message); + return new Promise(resolve => { + setTimeout(() => { + resolve(); + }, BLESendInterval); + }); + } + landing () { + this.message[3] = 0x0; + this._peripheral.send(this.message); + return new Promise(resolve => { + setTimeout(() => { + resolve(); + }, BLESendInterval); + }); + } + + alt(args){ + var value = args.TEXT>150? 150 : args.Text<0? 0 : args.TEXT; + this.message[3] = value; + this._peripheral.send(this.message); + return new Promise(resolve => { + setTimeout(() => { + resolve(); + }, BLESendInterval); + }); + } + + velocity(args){ + var vel = args.TEXT>200? 200 : args.TEXT<0? 0 : args.TEXT; + if(args.FBRL == '\uC55E') this.message[1] = vel; + if(args.FBRL == '\uB4A4') this.message[1] = vel*-1; + if(args.FBRL == '\uC624\uB978\uCABD') this.message[0] = vel; + if(args.FBRL == '\uC67C\uCABD') this.message[0] = vel*-1; + this.message[4] = 0x0F; + + this._peripheral.send(this.message); + return new Promise(resolve => { + setTimeout(() => { + resolve(); + }, BLESendInterval); + }); + } + + move (args){ + var dist = args.TEXT1>2000? 2000 : args.TEXT1<0? 0 : args.TEXT1; + var vel = args.TEXT2>200? 200 : args.TEXT2<0? 0 : args.TEXT2; + if(args.FBRL == '\uC55E') this._peripheral.moveY += dist; //this.message[1] = dist; + if(args.FBRL == '\uB4A4') this._peripheral.moveY += (-1*dist); //this.message[1] = dist*-1; + if(args.FBRL == '\uC624\uB978\uCABD') this._peripheral.moveX += dist; //this.message[0] = dist; + if(args.FBRL == '\uC67C\uCABD') this._peripheral.moveX += (-1*dist); //this.message[0] = dist*-1; + this.message[0] = this._peripheral.moveX; + this.message[1] = this._peripheral.moveY; + this.message[4] = 0x2F; + this.message[5] = vel; + + this._peripheral.send(this.message); + return new Promise(resolve => { + setTimeout(() => { + resolve(); + }, BLESendInterval); + }); + } + + rotation (args){ + var degree = args.TEXT1>179? 179 : args.TEXT1<0? 0 : args.TEXT1; + var vel = args.TEXT2>200? 200 : args.TEXT2<0? 0 : args.TEXT2; + if(args.ROTDIR == '\uC2DC\uACC4\uBC29\uD5A5') this._peripheral.rotation += degree; //this.message[2] = degree; + else this.message[2] = this._peripheral.rotation += (-1*degree); //degree*-1; + this.message[2] = this._peripheral.rotation; + this.message[6] = vel; + + this._peripheral.send(this.message); + return new Promise(resolve => { + setTimeout(() => { + resolve(); + }, BLESendInterval); + }); + } + + proprot (args){ + var speed = args.TEXT>1000? 1000 : args.TEXT<0? 0 : args.TEXT; + this.message[3] = speed; + this.message[4] = 0x01; + + this._peripheral.send(this.message); + return new Promise(resolve => { + setTimeout(() => { + resolve(); + }, BLESendInterval); + }); + } + + motorot (args){ + var speed = args.TEXT>100? 100 : args.TEXT<0? 0 : args.TEXT; + if(args.LTRB == '\uC67C\uCABD\uC544\uB798') this.message[2] = speed; + if(args.LTRB == '\uC67C\uCABD\uC704') this.message[1] = speed; + if(args.LTRB == '\uC624\uB978\uCABD\uC544\uB798') this.message[3] = speed; + if(args.LTRB == '\uC624\uB978\uCABD\uC704') this.message[0] = speed; + this.message[4] = 0x8000; + + this._peripheral.send(this.message); + return new Promise(resolve => { + setTimeout(() => { + resolve(); + }, BLESendInterval); + }); + } + + emergency (args){ + this._peripheral.moveX = 0; + this._peripheral.moveY = 0; + this._peripheral.rotation = 0; + for (let i = 0; i < 7; i++) + this.message[i] = 0; + this.message[5] = 100; + this.message[6] = 100; + + this._peripheral.send(this.message); + return new Promise(resolve => { + setTimeout(() => { + resolve(); + }, BLESendInterval); + }); + } + + + getready(){ + var rxData = this._peripheral.rxData; + if((rxData[0]&0x03) == 0) + return 1; + else + return 0; + } + + getbattery(){ + var rxData = this._peripheral.rxData; + return rxData[1]; + } + + getalt(){ + var rxData = this._peripheral.rxData; + return rxData[4]; + } + + gettilt(args){ + var rxData = this._peripheral.rxData; + if(args.FBLR == '\uC88C\uC6B0') + return rxData[2]>127? rxData[2]-256 : rxData[2]; + else + return rxData[3]>127? rxData[3]-256 : rxData[3]; + } + + getmove(args){ + var rxData = this._peripheral.rxData; + if(args.FBLR == '\uC88C\uC6B0'){ + var dist = (rxData[6]<<8) | rxData[5]; + return dist>0x7FFF? dist-0x10000 : dist; + + } + else{ + var dist = (rxData[8]<<8) | rxData[7]; + return dist>0x7FFF? dist-0x10000 : dist; + } + } +} + +module.exports = Scratch3JDCodeBlocks; diff --git a/src/extensions/scratch3_uglybot/index.js b/src/extensions/scratch3_uglybot/index.js new file mode 100644 index 00000000000..cd3494b003e --- /dev/null +++ b/src/extensions/scratch3_uglybot/index.js @@ -0,0 +1,614 @@ +const ArgumentType = require('../../extension-support/argument-type'); +const BlockType = require('../../extension-support/block-type'); +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'); + + +// eslint-disable-next-line max-len +const blockIconURI = ''; + + + +const BLETimeout = 4500; +const BLESendInterval = 20; +const BLEDataStoppedError = 'UglyBot extension stopped receiving data'; + + +const BLEUUID = { + service: 0x2263, + rxChar: '00000227-0000-1000-8000-00805f9b34fb', + txChar: '00000227-0000-1000-8000-00805f9b34fb' +}; + + +class UglyBot { + constructor (runtime, extensionId) { + this._runtime = runtime; + this._ble = null; + this._runtime.registerPeripheralExtension(extensionId, this); + this._extensionId = extensionId; + this._runtime.on('PROJECT_STOP_ALL', this.stopAll.bind(this)); + this._timeoutID = null; + this._busy = false; + this._busyTimeoutID = null; + this.reset = this.reset.bind(this); + this._onConnect = this._onConnect.bind(this); + this._onMessage = this._onMessage.bind(this); + this.rxData = [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0]; + this.txData = new Uint8Array(12); + this._sensors = { + button: 0, + ir: [0, 0, 0], + ultrasonic: 0, + joystick: [0, 0], + tilt: [0, 0], + sound: 0, + illum:0 + }; + } + + stopAll () { + for (let i = 0; i < 12; i++) + this.txData[i] = 0; + this.send(this.txData); + } + + scan () { + if (this._ble) { + this._ble.disconnect(); + } + this._ble = new BLE(this._runtime, this._extensionId, { + filters: [ + {services: [BLEUUID.service]} + ] + }, this._onConnect, this.reset); + } + + connect (id) { + if (this._ble) { + this._ble.connectPeripheral(id); + } + } + + disconnect () { + if (this._ble) { + this._ble.disconnect(); + } + this.reset(); + } + + reset () { + if (this._timeoutID) { + window.clearTimeout(this._timeoutID); + this._timeoutID = null; + } + } + + isConnected () { + + let connected = false; + if (this._ble) { + connected = this._ble.isConnected(); + } + return connected; + } + + send (message) { + if (!this.isConnected()) return; + + //console.log(message); + this._ble.write(BLEUUID.service, BLEUUID.txChar, message, 'ASCII', true).then( + () => { + this._busy = false; + window.clearTimeout(this._busyTimeoutID); + } + ); + } + + _onConnect () { + this._ble.read(BLEUUID.service, BLEUUID.rxChar, true, this._onMessage); + this._timeoutID = window.setTimeout( + () => this._ble.handleDisconnectError(BLEDataStoppedError), + BLETimeout + ); + } + + _onMessage (strData) { + this.rxData = strData.split(','); + window.clearTimeout(this._timeoutID); + this._timeoutID = window.setTimeout( + () => this._ble.handleDisconnectError(BLEDataStoppedError), + BLETimeout + ); + } + + get getRxData() { + return this.rxData; + } + + get getTxData() { + return this.rxData; + } +} + +class Scratch3UglyBotBlocks { + static get EXTENSION_NAME () { + return 'UglyBot'; + } + static get EXTENSION_ID () { + return 'uglybot'; + } + + constructor (runtime) { + + this.runtime = runtime; + this._peripheral = new UglyBot(this.runtime, Scratch3UglyBotBlocks.EXTENSION_ID); + this.message = this._peripheral.txData; + this.tuneID = 0; + this.moveID = 0; + this.rotID = 0; + } + + + getInfo () { + return { + id: Scratch3UglyBotBlocks.EXTENSION_ID, + name: Scratch3UglyBotBlocks.EXTENSION_NAME, + blockIconURI: blockIconURI, + showStatusButton: true, + blocks: [ + { + opcode: 'led', + text: formatMessage({ + id: 'led', + default: '[RIGHTLEFT] LED [ONOFF]', + }), + blockType: BlockType.COMMAND, + arguments: { + RIGHTLEFT: { + type: ArgumentType.STRING, + menu: 'rightleft', + defaultValue: '\uC624\uB978\uCABD' + }, + ONOFF: { + type: ArgumentType.STRING, + menu: 'onoff', + defaultValue: '\uCF1C\uAE30' + } + } + }, + { + opcode: 'buzzer', + text: formatMessage({ + id: 'buzzer', + default: '[TUNE] \uC74C\uC744 [DELAY] \uCD08\uB3D9\uC548 \uC18C\uB9AC\uB0B4\uAE30', + }), + blockType: BlockType.COMMAND, + arguments: { + TUNE: { + type: ArgumentType.STRING, + menu: 'tune', + defaultValue: '\uB3C4' + }, + DELAY: { + type: ArgumentType.STRING, + menu: 'delay', + defaultValue: '1' + } + } + }, + { + opcode: 'motor', + text: formatMessage({ + id: 'motor', + default: '[RIGHTLEFT] \uBAA8\uD130\uB97C [TEXT] \uC138\uAE30\uB85C \uD68C\uC804', + }), + blockType: BlockType.COMMAND, + arguments: { + RIGHTLEFT: { + type: ArgumentType.STRING, + menu: 'rightleft', + defaultValue: '\uC624\uB978\uCABD' + }, + TEXT: { + type: ArgumentType.STRING, + defaultValue: '0' + } + } + }, + + { + opcode: 'move', + text: formatMessage({ + id: 'move', + default: '[FRONTBACK] (\uC73C)\uB85C [TEXT] cm \uC774\uB3D9', + }), + blockType: BlockType.COMMAND, + arguments: { + FRONTBACK: { + type: ArgumentType.STRING, + menu: 'frontback', + defaultValue: '\uC55E' + }, + TEXT: { + type: ArgumentType.STRING, + defaultValue: '0' + } + } + }, + + { + opcode: 'rotate', + text: formatMessage({ + id: 'rotate', + default: '[ROTDIR] \uC73C\uB85C [TEXT] \uB3C4 \uD68C\uC804', + }), + blockType: BlockType.COMMAND, + arguments: { + ROTDIR: { + type: ArgumentType.STRING, + menu: 'rotdir', + defaultValue: '\uC2DC\uACC4\uBC29\uD5A5' + }, + TEXT: { + type: ArgumentType.STRING, + defaultValue: '0' + } + } + }, + + { + opcode: 'servo', + text: formatMessage({ + id: 'servo', + default: '[ONETWO] \uC11C\uBCF4\uBAA8\uD130\uB97C [TEXT] \uB3C4 \uD68C\uC804', + }), + blockType: BlockType.COMMAND, + arguments: { + ONETWO: { + type: ArgumentType.STRING, + menu: 'onetwo', + defaultValue: '1\uBC88' + }, + TEXT: { + type: ArgumentType.STRING, + defaultValue: '0' + } + } + }, + { + opcode: 'irsensor', + text: formatMessage({ + id: 'irsensor', + default: '[LMR] \uC801\uC678\uC120 \uC13C\uC11C\uB97C [ONOFF]', + }), + blockType: BlockType.COMMAND, + arguments: { + LMR: { + type: ArgumentType.STRING, + menu: 'lmr', + defaultValue: '\uC67C\uCABD' + }, + ONOFF: { + type: ArgumentType.STRING, + menu: 'onoff', + defaultValue: '\uCF1C\uAE30' + } + } + }, + + '---', + + { + opcode: 'getbutton', + text: formatMessage({ + id: 'getbutton', + default: '\uBC84\uD2BC \uAC12', + }), + blockType: BlockType.REPORTER, + }, + { + opcode: 'getirsensor', + text: formatMessage({ + id: 'getirsensor', + default: '[LMR] \uC801\uC678\uC120\uC13C\uC11C \uAC12', + }), + blockType: BlockType.REPORTER, + arguments: { + LMR: { + type: ArgumentType.STRING, + menu: 'lmr', + defaultValue: '\uC67C\uCABD' + }, + } + }, + { + opcode: 'getultrasonic', + text: formatMessage({ + id: 'getultrasonic', + default: '\uCD08\uC74C\uD30C\uC13C\uC11C \uAC12', + }), + blockType: BlockType.REPORTER, + }, + { + opcode: 'getjoystic', + text: formatMessage({ + id: 'getjoystic', + default: '[FBLR] \uC870\uC774\uC2A4\uD2F1 \uAC12', + }), + blockType: BlockType.REPORTER, + arguments: { + FBLR: { + type: ArgumentType.STRING, + menu: 'fblr', + defaultValue: '\uC55E\uB4A4' + }, + } + }, + { + opcode: 'gettilt', + text: formatMessage({ + id: 'gettilt', + default: '[FBLR] \uAE30\uC6B8\uAE30 \uAC12', + }), + blockType: BlockType.REPORTER, + arguments: { + FBLR: { + type: ArgumentType.STRING, + menu: 'fblr', + defaultValue: '\uC55E\uB4A4' + }, + } + }, + { + opcode: 'getsound', + text: formatMessage({ + id: 'getsound', + default: '\uC18C\uB9AC\uC13C\uC11C \uAC12', + }), + blockType: BlockType.REPORTER, + }, + { + opcode: 'getillum', + text: formatMessage({ + id: 'getillum', + default: '\uC870\uB3C4\uC13C\uC11C \uAC12', + }), + blockType: BlockType.REPORTER, + } + ], + menus: { + rightleft: { + acceptReporters: true, + items: ['\uC624\uB978\uCABD', '\uC67C\uCABD'] //¿À¸¥ÂÊ ¿ÞÂÊ + }, + lmr: { + acceptReporters: true, + items: ['\uC67C\uCABD', '\uC911\uAC04', '\uC624\uB978\uCABD'] //¿ÞÂÊ Áß°£ ¿À¸¥ÂÊ + }, + frontback: { + acceptReporters: true, + items: ['\uC55E', '\uB4A4'] //¾Õ µÚ + }, + onoff: { + acceptReporters: true, + items: ['\uCF1C\uAE30', '\uB044\uAE30'] //Äѱ⠲ô±â + }, + tune: { + acceptReporters: true, + items: ['\uB3C4', '\uB808', '\uBBF8', '\uD30C', '\uC194', '\uB77C', '\uC2DC'] //µµ·¹¹ÌÆļֶó½Ã + }, + delay: { + acceptReporters: true, + items: ['0.5', '0.8', '1', '2', '3', '4', '5'] + }, + rotdir: { + acceptReporters: true, + items: ['\uC2DC\uACC4\uBC29\uD5A5', '\uBC18\uC2DC\uACC4\uBC29\uD5A5'] //½Ã°è¹æÇ⠹ݽðè¹æÇâ + }, + onetwo: { + acceptReporters: true, + items: ['1\uBC88', '2\uBC88'] // 1¹ø 2 + }, + fblr: { + acceptReporters: true, + items: ['\uC55E\uB4A4', '\uC88C\uC6B0'] // ¾ÕµÚ Á¿ì + }, + } + }; + } + led (args) { + if(args.ONOFF == '\uCF1C\uAE30'){ + if(args.RIGHTLEFT == '\uC624\uB978\uCABD') + this.message[0] |= 0x11; + else + this.message[0] |= 0x12; + } + else{ + if(args.RIGHTLEFT == '\uC624\uB978\uCABD') + this.message[0] &= ~0x01; + else + this.message[0] &= ~0x02; + } + this._peripheral.send(this.message); + return new Promise(resolve => { + setTimeout(() => { + resolve(); + }, BLESendInterval); + }); + } + + buzzer(args){ + this.tuneID = (this.tuneID+1)%16; + if(args.TUNE == '\uB3C4') this.message[1] = 1; + if(args.TUNE == '\uB808') this.message[1] = 2; + if(args.TUNE == '\uBBF8') this.message[1] = 3; + if(args.TUNE == '\uD30C') this.message[1] = 4; + if(args.TUNE == '\uC194') this.message[1] = 5; + if(args.TUNE == '\uB77C') this.message[1] = 6; + if(args.TUNE == '\uC2DC') this.message[1] = 7; + this.message[1] |= (this.tuneID<<4); + + if(args.DELAY == '0.5') this.message[2] = 5; + if(args.DELAY == '0.8') this.message[2] = 8; + if(args.DELAY == '1') this.message[2] = 10; + if(args.DELAY == '2') this.message[2] = 20; + if(args.DELAY == '3') this.message[2] = 30; + if(args.DELAY == '4') this.message[2] = 40; + if(args.DELAY == '5') this.message[2] = 50; + + this._peripheral.send(this.message); + return new Promise(resolve => { + setTimeout(() => { + resolve(); + }, BLESendInterval); + }); + } + + motor (args){ + if(args.RIGHTLEFT == '\uC624\uB978\uCABD') + this.message[3] = args.TEXT; + else + this.message[4] = args.TEXT; + + this._peripheral.send(this.message); + return new Promise(resolve => { + setTimeout(() => { + resolve(); + }, BLESendInterval); + }); + } + + move (args){ + var value = args.TEXT; + if(args.FRONTBACK == '\uB4A4') + value *= -1; + + this.moveID = (this.moveID+1)%16; + value = value&0xFFF; + value |= (this.moveID<<12); + + this.message[6] = (value>>8); + this.message[5] = value&0xFF; + + this._peripheral.send(this.message); + return new Promise(resolve => { + setTimeout(() => { + resolve(); + }, BLESendInterval); + }); + } + + rotate (args){ + var value = args.TEXT; + if(args.ROTDIR == '\uBC18\uC2DC\uACC4\uBC29\uD5A5') + value *= -1; + + this.rotID = (this.rotID+1)%16; + value = value&0xFFF; + value |= (this.rotID<<12); + + this.message[8] = (value>>8); + this.message[7] = value&0xFF; + + this._peripheral.send(this.message); + return new Promise(resolve => { + setTimeout(() => { + resolve(); + }, BLESendInterval); + }); + } + + servo (args){ + if(args.ONETWO == '1\uBC88') + this.message[9] = args.TEXT; + else + this.message[10] = args.TEXT; + + this._peripheral.send(this.message); + return new Promise(resolve => { + setTimeout(() => { + resolve(); + }, BLESendInterval); + }); + } + + irsensor (args){ + if(args.ONOFF == '\uCF1C\uAE30'){ + if(args.LMR == '\uC67C\uCABD') + this.message[11] |= 0x01; + else if(args.LMR == '\uC911\uAC04') + this.message[11] |= 0x02; + else + this.message[11] |= 0x04; + } + else{ + if(args.LMR == '\uC67C\uCABD') + this.message[11] &= ~0x01; + else if(args.LMR == '\uC911\uAC04') + this.message[11] &= ~0x02; + else + this.message[11] &= ~0x04; + } + this._peripheral.send(this.message); + return new Promise(resolve => { + setTimeout(() => { + resolve(); + }, BLESendInterval); + }); + } + + + + + getbutton(){ + var rxData = this._peripheral.rxData; + return rxData[0]; + } + + getirsensor(args){ + var rxData = this._peripheral.rxData; + if(args.LMR == '\uC67C\uCABD') + return rxData[1]; + if(args.LMR == '\uC911\uAC04') + return rxData[2]; + else + return rxData[3]; + } + + getultrasonic(){ + var rxData = this._peripheral.rxData; + return rxData[4]; + } + + getjoystic(args){ + var rxData = this._peripheral.rxData; + if(args.FBLR == '\uC55E\uB4A4') + return rxData[5]; + else + return rxData[6]; + } + + gettilt(args){ + var rxData = this._peripheral.rxData; + if(args.FBLR == '\uC55E\uB4A4') + return rxData[7]; + else + return rxData[8]; + } + + getsound(){ + var rxData = this._peripheral.rxData; + return rxData[9]; + } + + getillum(){ + var rxData = this._peripheral.rxData; + return rxData[10]; + } +} + +module.exports = Scratch3UglyBotBlocks; diff --git a/src/util/scratch-link-websocket.js b/src/util/scratch-link-websocket.js index 035cea566de..b35e349dc31 100644 --- a/src/util/scratch-link-websocket.js +++ b/src/util/scratch-link-websocket.js @@ -25,7 +25,7 @@ class ScratchLinkWebSocket { open () { switch (this._type) { case 'BLE': - this._ws = new WebSocket('wss://device-manager.scratch.mit.edu:20110/scratch/ble'); + this._ws = new WebSocket('ws://device-manager.scratch.mit.edu:20110/scratch/ble'); break; case 'BT': this._ws = new WebSocket('wss://device-manager.scratch.mit.edu:20110/scratch/bt');