From 55b1a794ce55d0749b8105e10cfbe76d051fc7bd Mon Sep 17 00:00:00 2001 From: Andrew Sliwinski Date: Tue, 5 Jun 2018 17:51:55 -0400 Subject: [PATCH 1/2] Add initial working version of the Amazon Polly extension --- src/extension-support/extension-manager.js | 2 + src/extensions/scratch3_speak/index.js | 152 +++++++++++++++++++++ 2 files changed, 154 insertions(+) create mode 100644 src/extensions/scratch3_speak/index.js diff --git a/src/extension-support/extension-manager.js b/src/extension-support/extension-manager.js index 147d57305..cff9f4f0e 100644 --- a/src/extension-support/extension-manager.js +++ b/src/extension-support/extension-manager.js @@ -11,6 +11,7 @@ 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 Scratch3SpeakBlocks = require('../extensions/scratch3_speak'); const Scratch3TranslateBlocks = require('../extensions/scratch3_translate'); const Scratch3VideoSensingBlocks = require('../extensions/scratch3_video_sensing'); const Scratch3SpeechBlocks = require('../extensions/scratch3_speech'); @@ -20,6 +21,7 @@ const builtinExtensions = { wedo2: Scratch3WeDo2Blocks, music: Scratch3MusicBlocks, microbit: Scratch3MicroBitBlocks, + speak: Scratch3SpeakBlocks, translate: Scratch3TranslateBlocks, videoSensing: Scratch3VideoSensingBlocks, speech: Scratch3SpeechBlocks diff --git a/src/extensions/scratch3_speak/index.js b/src/extensions/scratch3_speak/index.js new file mode 100644 index 000000000..1558409cd --- /dev/null +++ b/src/extensions/scratch3_speak/index.js @@ -0,0 +1,152 @@ +const formatMessage = require('format-message'); +const nets = require('nets'); + +const ArgumentType = require('../../extension-support/argument-type'); +const BlockType = require('../../extension-support/block-type'); +const Cast = require('../../util/cast'); +const log = require('../../util/log'); + +/** + * The url of the translate server. + * @type {string} + */ +const SERVER_HOST = 'https://synthesis-service.scratch.mit.edu'; + +/** + * How long to wait in ms before timing out requests to translate server. + * @type {int} + */ +const SERVER_TIMEOUT = 10000; // 10 seconds (chosen arbitrarily). + +/** + * Class for the translate block in Scratch 3.0. + * @constructor + */ +class Scratch3SpeakBlocks { + constructor () { + /** + * Locale code of the viewer + * @type {string} + * @private + */ + this._language = this.getViewerLanguageCode(); + } + + /** + * The key to load & store a target's translate state. + * @return {string} The key. + */ + static get STATE_KEY () { + return 'Scratch.speak'; + } + + /** + * @returns {object} metadata for this extension and its blocks. + */ + getInfo () { + return { + id: 'speak', + name: 'Amazon Polly', + menuIconURI: '', // @todo Add the final icons. + blockIconURI: '', + blocks: [ + { + opcode: 'speak', + text: formatMessage({ + id: 'speak.speakBlock', + default: 'speak [WORDS]', + description: 'speak some words' + }), + blockType: BlockType.COMMAND, + arguments: { + WORDS: { + type: ArgumentType.STRING, + defaultValue: formatMessage({ + id: 'speak.defaultTextToSpeak', + default: 'hello', + description: 'hello: the default text to speak' + }) + } + } + } + ], + menus: { + voices: this.supportedVoices + } + }; + } + + /** + * Get the viewer's language code. + * @return {string} the language code. + */ + getViewerLanguageCode () { + // @todo This should be the language code of the project *creator* + // rather than the project viewer. + return navigator.language || navigator.userLanguage || 'en-US'; + } + + /** + * Return the supported voices for the extension. + * @return {Array} Supported voices + */ + supportedVoices () { + return [ + 'Quinn', + 'Max', + 'Squeak', + 'Monster', + 'Puppy' + ]; + } + + /** + * Convert the provided text into a sound file and then play the file. + * @param {object} args Block arguments + * @return {Promise} A promise that resolves after playing the sound + */ + speak (args) { + // Cast input to string + args.WORDS = Cast.toString(args.WORDS); + + // Build up URL + let path = `${SERVER_HOST}/synth`; + path += `?locale=${this._language}`; + path += `&gender=male`; // @todo + path += `&text=${encodeURI(args.WORDS)}`; + + // Perform HTTP request to get audio file + return new Promise(resolve => { + nets({ + url: path, + timeout: SERVER_TIMEOUT + }, (err, res, body) => { + if (err) { + log.warn(err); + return resolve(); + } + + if (res.statusCode !== 200) { + log.warn(res.statusCode); + return resolve(); + } + + // Play the sound + const ctx = new (window.AudioContext || window.webkitAudioContext)(); + const source = ctx.createBufferSource(); + ctx.decodeAudioData(body.buffer, buffer => { + source.buffer = buffer; + source.connect(ctx.destination); + source.loop = false; + }, e => { + log.warn(e.err); + }); + source.addEventListener('ended', () => { + resolve(); + }); + source.start(0); + }); + }); + } +} +module.exports = Scratch3SpeakBlocks; From bccdef06746cdf56bcf52c8d7fa9da9832369a17 Mon Sep 17 00:00:00 2001 From: Andrew Sliwinski Date: Tue, 5 Jun 2018 18:32:02 -0400 Subject: [PATCH 2/2] Future-proof opcode and fix issues with copied comments --- src/extensions/scratch3_speak/index.js | 14 +++++++------- 1 file changed, 7 insertions(+), 7 deletions(-) diff --git a/src/extensions/scratch3_speak/index.js b/src/extensions/scratch3_speak/index.js index 1558409cd..9a64228ff 100644 --- a/src/extensions/scratch3_speak/index.js +++ b/src/extensions/scratch3_speak/index.js @@ -7,19 +7,19 @@ const Cast = require('../../util/cast'); const log = require('../../util/log'); /** - * The url of the translate server. + * The url of the synthesis server. * @type {string} */ const SERVER_HOST = 'https://synthesis-service.scratch.mit.edu'; /** - * How long to wait in ms before timing out requests to translate server. + * How long to wait in ms before timing out requests to synthesis server. * @type {int} */ const SERVER_TIMEOUT = 10000; // 10 seconds (chosen arbitrarily). /** - * Class for the translate block in Scratch 3.0. + * Class for the synthesis block in Scratch 3.0. * @constructor */ class Scratch3SpeakBlocks { @@ -33,7 +33,7 @@ class Scratch3SpeakBlocks { } /** - * The key to load & store a target's translate state. + * The key to load & store a target's synthesis state. * @return {string} The key. */ static get STATE_KEY () { @@ -51,9 +51,9 @@ class Scratch3SpeakBlocks { blockIconURI: '', blocks: [ { - opcode: 'speak', + opcode: 'speakAndWait', text: formatMessage({ - id: 'speak.speakBlock', + id: 'speak.speakAndWaitBlock', default: 'speak [WORDS]', description: 'speak some words' }), @@ -105,7 +105,7 @@ class Scratch3SpeakBlocks { * @param {object} args Block arguments * @return {Promise} A promise that resolves after playing the sound */ - speak (args) { + speakAndWait (args) { // Cast input to string args.WORDS = Cast.toString(args.WORDS);