mirror of
https://github.com/scratchfoundation/scratch-vm.git
synced 2025-07-07 19:43:57 -04:00
286 lines
17 KiB
JavaScript
286 lines
17 KiB
JavaScript
const ArgumentType = require('../../extension-support/argument-type');
|
|
const BlockType = require('../../extension-support/block-type');
|
|
const Cast = require('../../util/cast');
|
|
const log = require('../../util/log');
|
|
const {fetchWithTimeout} = require('../../util/fetch-with-timeout');
|
|
const languageNames = require('scratch-translate-extension-languages');
|
|
const formatMessage = require('format-message');
|
|
|
|
/**
|
|
* Icon svg to be displayed in the blocks category menu, encoded as a data URI.
|
|
* @type {string}
|
|
*/
|
|
// eslint-disable-next-line max-len
|
|
const menuIconURI = '';
|
|
|
|
/**
|
|
* 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 = '';
|
|
|
|
/**
|
|
* The url of the translate server.
|
|
* @type {string}
|
|
*/
|
|
const serverURL = 'https://translate-service.scratch.mit.edu/';
|
|
|
|
/**
|
|
* How long to wait in ms before timing out requests to translate server.
|
|
* @type {int}
|
|
*/
|
|
const serverTimeoutMs = 10000; // 10 seconds (chosen arbitrarily).
|
|
|
|
/**
|
|
* Class for the translate block in Scratch 3.0.
|
|
* @constructor
|
|
*/
|
|
class Scratch3TranslateBlocks {
|
|
constructor () {
|
|
/**
|
|
* Language code of the viewer, based on their locale.
|
|
* @type {string}
|
|
* @private
|
|
*/
|
|
this._viewerLanguageCode = this.getViewerLanguageCode();
|
|
|
|
/**
|
|
* List of supported language name and language code pairs, for use in the block menu.
|
|
* Filled in by getInfo so it is updated when the interface language changes.
|
|
* @type {Array.<object.<string, string>>}
|
|
* @private
|
|
*/
|
|
this._supportedLanguages = [];
|
|
|
|
/**
|
|
* A randomly selected language code, for use as the default value in the language menu.
|
|
* Properly filled in getInfo so it is updated when the interface languages changes.
|
|
* @type {string}
|
|
* @private
|
|
*/
|
|
this._randomLanguageCode = 'en';
|
|
|
|
|
|
/**
|
|
* The result from the most recent translation.
|
|
* @type {string}
|
|
* @private
|
|
*/
|
|
this._translateResult = '';
|
|
|
|
/**
|
|
* The language of the text most recently translated.
|
|
* @type {string}
|
|
* @private
|
|
*/
|
|
this._lastLangTranslated = '';
|
|
|
|
/**
|
|
* The text most recently translated.
|
|
* @type {string}
|
|
* @private
|
|
*/
|
|
this._lastTextTranslated = '';
|
|
}
|
|
|
|
/**
|
|
* The key to load & store a target's translate state.
|
|
* @return {string} The key.
|
|
*/
|
|
static get STATE_KEY () {
|
|
return 'Scratch.translate';
|
|
}
|
|
|
|
/**
|
|
* @returns {object} metadata for this extension and its blocks.
|
|
*/
|
|
getInfo () {
|
|
this._supportedLanguages = this._getSupportedLanguages(this.getViewerLanguageCode());
|
|
this._randomLanguageCode = this._supportedLanguages[
|
|
Math.floor(Math.random() * this._supportedLanguages.length)].value;
|
|
|
|
return {
|
|
id: 'translate',
|
|
name: formatMessage({
|
|
id: 'translate.categoryName',
|
|
default: 'Translate',
|
|
description: 'Name of extension that adds translate blocks'
|
|
}),
|
|
blockIconURI: blockIconURI,
|
|
menuIconURI: menuIconURI,
|
|
blocks: [
|
|
{
|
|
opcode: 'getTranslate',
|
|
text: formatMessage({
|
|
id: 'translate.translateBlock',
|
|
default: 'translate [WORDS] to [LANGUAGE]',
|
|
description: 'translate some text to a different language'
|
|
}),
|
|
blockType: BlockType.REPORTER,
|
|
arguments: {
|
|
WORDS: {
|
|
type: ArgumentType.STRING,
|
|
defaultValue: formatMessage({
|
|
id: 'translate.defaultTextToTranslate',
|
|
default: 'hello',
|
|
description: 'hello: the default text to translate'
|
|
})
|
|
},
|
|
LANGUAGE: {
|
|
type: ArgumentType.STRING,
|
|
menu: 'languages',
|
|
defaultValue: this._randomLanguageCode
|
|
}
|
|
}
|
|
},
|
|
{
|
|
opcode: 'getViewerLanguage',
|
|
text: formatMessage({
|
|
id: 'translate.viewerLanguage',
|
|
default: 'language',
|
|
description: 'the languge of the project viewer'
|
|
}),
|
|
blockType: BlockType.REPORTER,
|
|
arguments: {}
|
|
}
|
|
],
|
|
menus: {
|
|
languages: {
|
|
acceptReporters: true,
|
|
items: this._supportedLanguages
|
|
}
|
|
}
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Computes a list of language code and name pairs for the given language.
|
|
* @param {string} code The language code to get the list of language pairs
|
|
* @return {Array.<object.<string, string>>} An array of languge name and
|
|
* language code pairs.
|
|
* @private
|
|
*/
|
|
_getSupportedLanguages (code) {
|
|
return languageNames.menuMap[code].map(entry => {
|
|
const obj = {text: entry.name, value: entry.code};
|
|
return obj;
|
|
});
|
|
}
|
|
/**
|
|
* Get the human readable language value for the reporter block.
|
|
* @return {string} the language name of the project viewer.
|
|
*/
|
|
getViewerLanguage () {
|
|
this._viewerLanguageCode = this.getViewerLanguageCode();
|
|
const names = languageNames.menuMap[this._viewerLanguageCode];
|
|
let langNameObj = names.find(obj => obj.code === this._viewerLanguageCode);
|
|
|
|
// If we don't have a name entry yet, try looking it up via the Google langauge
|
|
// code instead of Scratch's (e.g. for es-419 we look up es to get espanol)
|
|
if (!langNameObj && languageNames.scratchToGoogleMap[this._viewerLanguageCode]) {
|
|
const lookupCode = languageNames.scratchToGoogleMap[this._viewerLanguageCode];
|
|
langNameObj = names.find(obj => obj.code === lookupCode);
|
|
}
|
|
|
|
let langName = this._viewerLanguageCode;
|
|
if (langNameObj) {
|
|
langName = langNameObj.name;
|
|
}
|
|
return langName;
|
|
}
|
|
|
|
/**
|
|
* Get the viewer's language code.
|
|
* @return {string} the language code.
|
|
*/
|
|
getViewerLanguageCode () {
|
|
const locale = formatMessage.setup().locale;
|
|
const viewerLanguages = [locale].concat(navigator.languages);
|
|
const languageKeys = Object.keys(languageNames.menuMap);
|
|
// Return the first entry in viewerLanguages that matches
|
|
// one of the available language keys.
|
|
const languageCode = viewerLanguages.reduce((acc, lang) => {
|
|
if (acc) {
|
|
return acc;
|
|
}
|
|
if (languageKeys.indexOf(lang.toLowerCase()) > -1) {
|
|
return lang;
|
|
}
|
|
return acc;
|
|
}, '') || 'en';
|
|
|
|
return languageCode.toLowerCase();
|
|
}
|
|
|
|
/**
|
|
* Get a language code from a block argument. The arg can be a language code
|
|
* or a language name, written in any language.
|
|
* @param {object} arg A block argument.
|
|
* @return {string} A language code.
|
|
*/
|
|
getLanguageCodeFromArg (arg) {
|
|
const languageArg = Cast.toString(arg).toLowerCase();
|
|
// Check if the arg matches a language code in the menu.
|
|
if (languageNames.menuMap.hasOwnProperty(languageArg)) {
|
|
return languageArg;
|
|
}
|
|
// Check for a dropped-in language name, and convert to a language code.
|
|
if (languageNames.nameMap.hasOwnProperty(languageArg)) {
|
|
return languageNames.nameMap[languageArg];
|
|
}
|
|
|
|
// There are some languages we launched in the language menu that Scratch did not
|
|
// end up launching in. In order to keep projects that may have had that menu item
|
|
// working, check for those language codes and let them through.
|
|
// Examples: 'ab', 'hi'.
|
|
if (languageNames.previouslySupported.indexOf(languageArg) !== -1) {
|
|
return languageArg;
|
|
}
|
|
// Default to English.
|
|
return 'en';
|
|
}
|
|
|
|
/**
|
|
* Translates the text in the translate block to the language specified in the menu.
|
|
* @param {object} args - the block arguments.
|
|
* @return {Promise} - a promise that resolves after the response from the translate server.
|
|
*/
|
|
getTranslate (args) {
|
|
// If the text contains only digits 0-9 and nothing else, return it without
|
|
// making a request.
|
|
if (/^\d+$/.test(args.WORDS)) return Promise.resolve(args.WORDS);
|
|
|
|
// Don't remake the request if we already have the value.
|
|
if (this._lastTextTranslated === args.WORDS &&
|
|
this._lastLangTranslated === args.LANGUAGE) {
|
|
return this._translateResult;
|
|
}
|
|
|
|
const lang = this.getLanguageCodeFromArg(args.LANGUAGE);
|
|
|
|
let urlBase = `${serverURL}translate?language=`;
|
|
urlBase += lang;
|
|
urlBase += '&text=';
|
|
urlBase += encodeURIComponent(args.WORDS);
|
|
|
|
const tempThis = this;
|
|
const translatePromise = fetchWithTimeout(urlBase, {}, serverTimeoutMs)
|
|
.then(response => response.text())
|
|
.then(responseText => {
|
|
const translated = JSON.parse(responseText).result;
|
|
tempThis._translateResult = translated;
|
|
// Cache what we just translated so we don't keep making the
|
|
// same call over and over.
|
|
tempThis._lastTextTranslated = args.WORDS;
|
|
tempThis._lastLangTranslated = args.LANGUAGE;
|
|
return translated;
|
|
})
|
|
.catch(err => {
|
|
log.warn(`error fetching translate result! ${err}`);
|
|
return '';
|
|
});
|
|
return translatePromise;
|
|
}
|
|
}
|
|
module.exports = Scratch3TranslateBlocks;
|