Skip to content

Instantly share code, notes, and snippets.

@chaidhat
Last active July 6, 2019 08:55
Show Gist options
  • Save chaidhat/47e72152c248570bb61da72c093a0234 to your computer and use it in GitHub Desktop.
Save chaidhat/47e72152c248570bb61da72c093a0234 to your computer and use it in GitHub Desktop.
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 nets = require('nets');
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 = '';
const blockIconURI = '';
//const docsURI: 'https://translate-service.scratch.mit.edu/';
/**
* 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 Scratch3SOOP {
constructor (runtime, util) {
this.runtime = runtime;
this.util = util;
this._a = 0;
this._funcs = {};
this._funcsparam = {};
this.debug = "all good";
this.debugerrorc = 0;
//[function_expecting, return value]
this._retFunctionExpecting = {};
this._retFunctionExpectingNum = 0;
this._retValue = "null";
this._structure = {};
this._giveDir = false;
this._altFunc = {};
this._testFunc = false;
this._scope = "";
this._tempscope = {};
this._tempscopenum = 0;
this._funcscope = "";
this._tempfuncscope = {};
this._tempfuncscopenum = 0;
}
/**
* 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: 'OOP',
description: 'Name of extension that adds translate blocks'
}),
color1: '#8B4200',
color2: '#7B3200',
color3: '#CCCCCC',
docsURI: 'https://translate-service.scratch.mit.edu/',
//blockIconURI: blockIconURI,
//menuIconURI: menuIconURI,
blocks: [
{
opcode: 'clas',
text: formatMessage({
id: 'translate.clas',
default: 'class [CLASSNAME]',
description: 'the languge of the project viewer'
}),
blockType: BlockType.REPORTER,
arguments: {
CLASSNAME: {
type: ArgumentType.STRING,
defaultValue: formatMessage({
id: 'translate.clas.name',
default: "Fruit",
description: 'hello: the default text to translate'
})
},
},
},
{
opcode: 'clasConstructor',
text: formatMessage({
id: 'translate.clasConstructor',
default: 'class [CLASSNAME]. [VALUE]',
description: 'the languge of the project viewer'
}),
blockType: BlockType.REPORTER,
arguments: {
CLASSNAME: {
type: ArgumentType.STRING,
defaultValue: formatMessage({
id: 'translate.clasConstructor.name',
default: "Fruit",
description: 'hello: the default text to translate'
})
},
VALUE: {
//type: ArgumentType.STRING,
defaultValue: formatMessage({
id: 'translate.clasConstructor.value',
default: 1,
description: 'hello: the default text to translate'
})
},
},
},
{
opcode: 'prop',
text: formatMessage({
id: 'translate.scopeableVar',
default: 'property [PROPNAME]',
description: 'the languge of the project viewer'
}),
blockType: BlockType.REPORTER,
arguments: {
PROPNAME: {
type: ArgumentType.STRING,
defaultValue: formatMessage({
id: 'translate.prop.name',
default: "apple",
description: 'hello: the default text to translate'
})
},
},
},
'---',
{
opcode: 'set',
text: formatMessage({
id: 'translate.set',
default: 'set object [NAME] to [VALUE]',
description: 'the languge of the project viewer'
}),
blockType: BlockType.COMMAND,
arguments: {
NAME: {
//type: ArgumentType.STRING,
defaultValue: formatMessage({
id: 'translate.set.name',
default: 1,
description: 'hello: the default text to translate'
})
},
VALUE: {
type: ArgumentType.STRING,
defaultValue: formatMessage({
id: 'translate.set.value',
default: 1,
description: 'hello: the default text to translate'
})
},
},
},
'---',
{
opcode: 'funcref',
text: formatMessage({
id: 'translate.funcref',
default: 'function [FUNCNAME]',
description: 'the languge of the project viewer'
}),
blockType: BlockType.REPORTER,
arguments: {
FUNCNAME: {
type: ArgumentType.STRING,
defaultValue: formatMessage({
id: 'translate.funcref.name',
default: "foo",
description: 'hello: the default text to translate'
})
},
},
},
{
opcode: 'callfunc',
text: formatMessage({
id: 'translate.callfunc',
default: 'call [NAME] <-- [PARAM]',
description: 'the languge of the project viewer'
}),
blockType: BlockType.REPORTER,
arguments: {
NAME: {
//type: ArgumentType.STRING,
defaultValue: formatMessage({
id: 'translate.callfunc.name',
default: "foo",
description: 'hello: the default text to translate'
})
},
PARAM: {
//type: ArgumentType.STRING,
defaultValue: formatMessage({
id: 'translate.callfunc.param',
default: 1,
description: 'hello: the default text to translate'
})
},
},
},
'---',
{
opcode: 'func',
text: formatMessage({
id: 'translate.func',
default: '[SCOPE] :: function [NAME] <-- [PARAM]',
description: 'the languge of the project viewer'
}),
blockType: BlockType.HAT,
arguments: {
SCOPE: {
//type: ArgumentType.STRING,
defaultValue: formatMessage({
id: 'translate.func.scope',
default: 1,
description: 'hello: the default text to translate'
})
},
NAME: {
type: ArgumentType.STRING,
defaultValue: formatMessage({
id: 'translate.func.name',
default: "foo",
description: 'hello: the default text to translate'
})
},
PARAM: {
//type: ArgumentType.STRING,
defaultValue: formatMessage({
id: 'translate.func.param',
default: 1,
description: 'hello: the default text to translate'
})
},
},
},
{
opcode: 'thi',
text: formatMessage({
id: 'translate.thi',
default: 'this class . [PARAM]',
description: 'the languge of the project viewer'
}),
blockType: BlockType.REPORTER,
arguments: {
PARAM: {
//type: ArgumentType.STRING,
defaultValue: formatMessage({
id: 'translate.thi.param',
default: 1,
description: 'hello: the default text to translate'
})
},
},
},
{
opcode: 'ret',
text: formatMessage({
id: 'translate.ret',
default: 'function return [VALUE]-->',
description: 'the languge of the project viewer'
}),
isTerminal: true,
blockType: BlockType.COMMAND,
arguments: {
VALUE: {
type: ArgumentType.STRING,
defaultValue: formatMessage({
id: 'translate.ret.value',
default: "bar",
description: 'hello: the default text to translate'
})
},
},
},
'---',
{
opcode: 'tuple',
text: formatMessage({
id: 'translate.tuple',
default: '[VALUE1] , [VALUE2]',
description: 'the languge of the project viewer'
}),
blockType: BlockType.REPORTER,
arguments: {
VALUE1: {
type: ArgumentType.STRING,
defaultValue: formatMessage({
id: 'translate.tuple.value1',
default: "",
description: 'hello: the default text to translate'
})
},
VALUE2: {
type: ArgumentType.STRING,
defaultValue: formatMessage({
id: 'translate.tuple.value2',
default: "",
description: 'hello: the default text to translate'
})
},
},
},
{
opcode: 'gettuple',
text: formatMessage({
id: 'translate.gettuple',
default: 'get item [NUMBER] of tuple [TUPLE]',
description: 'the languge of the project viewer'
}),
blockType: BlockType.REPORTER,
arguments: {
NUMBER: {
type: ArgumentType.NUMBER,
defaultValue: formatMessage({
id: 'translate.gettuple.number',
default: 1,
description: 'hello: the default text to translate'
})
},
TUPLE: {
//type: ArgumentType.STRING,
defaultValue: formatMessage({
id: 'translate.gettuple.tuple',
default: 1,
description: 'hello: the default text to translate'
})
},
},
},
'---',
{
opcode: 'scop',
text: formatMessage({
id: 'translate.scop',
default: 'using class [CLASSNAME]',
description: 'the languge of the project viewer'
}),
blockType: BlockType.CONDITIONAL,
arguments: {
CLASSNAME: {
defaultValue: formatMessage({
id: 'translate.scop.classname',
default: 1,
description: 'hello: the default text to translate'
})
},
},
},
'---',
{
opcode: 'c',
text: formatMessage({
id: 'translate.c',
default: 'debug.reset func [NAME]',
description: 'the languge of the project viewer'
}),
blockType: BlockType.COMMAND,
arguments: {
NAME: {
type: ArgumentType.STRING,
defaultValue: formatMessage({
id: 'translate.debug.name',
default: 1,
description: 'hello: the default text to translate'
})
},
},
},
{
opcode: 'dl',
text: formatMessage({
id: 'translate.dl',
default: 'debug.log',
description: 'the languge of the project viewer'
}),
blockType: BlockType.REPORTER,
arguments: {
},
},
{
opcode: 'de',
text: formatMessage({
id: 'translate.de',
default: 'debug.error?',
description: 'the languge of the project viewer'
}),
blockType: BlockType.BOOLEAN,
arguments: {
},
},
],
menus: {
languages: {
acceptReporters: true,
items: this._supportedLanguages
}
}
};
}
func({SCOPE,NAME,PARAM})
{
if (SCOPE)
NAME = SCOPE.substring(0,SCOPE.length - 1) + "," + NAME;
if (this._funcs[NAME] === "true")
{
this._tempscope[this._tempscopenum] = this._scope;
this._tempscopenum++;
this._tempfuncscope[this._tempfuncscopenum] = this._funcscope;
this._tempfuncscopenum++;
this._scope = "funcclass " + NAME + ",";
if (SCOPE)
this._funcscope = SCOPE.substring(0,SCOPE.length - 1) + ",";
this._funcs[NAME] = "standtrue";
if (this._testFunc)
{
return false;
}
return true;
}
else
{
if (this._funcs[NAME] === "standfalse")
{
this._funcs[NAME] = "false";
}
return false;
}
}
callfunc(args,util)
{
var a = args.NAME.substring(args.NAME.length - 1, args.NAME.length);
if (!(a === "]"))
{
this.debug = "ERROR NO" + this.debugerrorc +": in function " + args.NAME + ", " + "Code: 1. not a function! Proper usage: (call ( (function (FOO)) ) <-- ( )). Put only (function ( )) block into it!";
return this.debug;
}
else
{
args.NAME = args.NAME.substring(0, args.NAME.length - 1);
}
if (this._retFunctionExpectingNum < -10)
{
this.debug = "ERROR NO" + this.debugerrorc +": in function " + args.NAME + ", " + "Code: 2. this._retFunctionExpectingNum < 10, please reset errors.";
return this.debug;
}
if (util.stackTimerNeedsInit()) {
var j = 0;
for (var i = -10; i < this._retFunctionExpectingNum; i++)
{
if (this._retFunctionExpecting[i] === args.NAME)
{
j++;
}
}
if (j > 0)
{
this.debug = "ERROR NO" + this.debugerrorc + ": in function " + args.NAME + ", " + "Code: 3. no recursion allowed. I'm sorry.";
return this.debug;
}
if (this._funcs[args.NAME] != "true" && this._funcs[args.NAME] != "standfalse" && this._funcs[args.NAME] != "standtrue")
{
/*
/ Have we run before, starting threads?
if (!util.stackFrame.startedThreads) {
// No - start hats for this broadcast.
util.stackFrame.startedThreads = util.startHats(
'event_whenbroadcastreceived', {
BROADCAST_OPTION: broadcastOption
}
);
if (util.stackFrame.startedThreads.length === 0) {
// Nothing was started.
return;
}
}
// We've */
const efen = this._retFunctionExpectingNum;
this._funcs[args.NAME] = "true";
this._funcsparam[args.NAME] = args.PARAM;
this._retFunctionExpectingNum++;
this._retFunctionExpecting[efen] = args.NAME;
util.a = efen;
util.time = 0;
util.startStackTimer(100);
/*let efen = this._retFunctionExpectingNum;
this._retFunctionExpectingNum++;
for (var i = 0; i < 10; i++)
{
if (this._retFunctionExpectingNum === efen)
{
return this._retValue;
}
}*/
this.runtime.requestRedraw();
util.yield();
}
else
{
util.yield();
}
}
else
{
if (this._retFunctionExpectingNum === util.a) {
this._retFunctionExpecting[util.a] = "";
this._funcs[args.NAME] = "standfalse";
return this._retValue;
} else {
util.time++;
if (!(this._funcs[args.NAME] === "standtrue") && util.stackTimerFinished())
{
this._funcs[args.NAME] = "false";
this._retFunctionExpectingNum--;
this.debug = "ERROR NO" + this.debugerrorc + ": in function " + args.NAME + ", " + "Code: 4. Is there actually a function hat declaring this function?";
return this.debug;
}
util.yield();
}
}
}
ret({VALUE})
{
this._retValue = VALUE;
this._retFunctionExpectingNum--;
this._tempscopenum--;
this._scope = this._tempscope[this._tempscopenum];
return;
}
c(args,util)
{
for (var i = -10; i < this._retFunctionExpectingNum + 1; i++)
{
this._retFunctionExpecting[i] = "";
}
this._retFunctionExpectingNum = 0;
this._tempscopenum = 0;
this._funcs[args.NAME] = "false";
this._scope = "";
this.debug = "all good";
}
clas(args, util)
{
this._giveDir = ">" + args.CLASSNAME + "[";
//this.debug = (this._scope + this._giveDir);
return ">" + args.CLASSNAME + "[";
}
clasConstructor(args, util)
{
if (args.VALUE)
{
}
else
{
this._giveDir = ">" + args.CLASSNAME + "[";
//this.debug = (this._scope + this._giveDir);
return ">" + args.CLASSNAME + "[";
}
this._giveDir = ">" + args.CLASSNAME + "," + this._giveDir;
//this.debug = (this._scope + this._giveDir);
var a = (this._scope + this._giveDir).substring((this._scope + this._giveDir).length - 1, (this._scope + this._giveDir).length);
if (a === "[" || a === "]")
{
// class
return (this._scope + this._giveDir);
}
if (this._structure[(this._scope + this._giveDir)])
{
// property
if (this._structure[(this._scope + this._giveDir)].substring(0,1) === "*")
{
return this._structure[(this._scope + this._giveDir)].substring(1,this._structure[(this._scope + this._giveDir)].length);
}
else
{
return this._structure[(this._scope + this._giveDir)];
}
}
else
{
// nullptr
return "null";
}
}
prop(args, util)
{
this._giveDir = args.PROPNAME;
//this.debug = (this._scope + this._giveDir);
if (this._structure[(this._scope + this._giveDir)])
{
// property
if (this._structure[(this._scope + this._giveDir)].substring(0,1) === "*")
{
return this._structure[(this._scope + this._giveDir)].substring(1,this._structure[(this._scope + this._giveDir)].length);
}
else
{
return this._structure[(this._scope + this._giveDir)];
}
}
else
{
// nullptr
return "null";
}
}
thi(args)
{
if (args.VALUE)
{
}
else
{
this._giveDir = ">" + args.CLASSNAME + "[";
//this.debug = (this._scope + this._giveDir);
return ">" + args.CLASSNAME + "[";
}
this._giveDir = ">" + args.CLASSNAME + "," + this._giveDir;
//this.debug = (this._scope + this._giveDir);
var a = (this._scope + this._giveDir).substring((this._funcscope + this._giveDir).length - 1, (this._scope + this._giveDir).length);
if (a === "[" || a === "]")
{
// class
return (this._funcscope + this._giveDir);
}
if (this._structure[(this._funcscope + this._giveDir)])
{
// property
if (this._structure[(this._funcscope + this._giveDir)].substring(0,1) === "*")
{
return this._structure[(this._funcscope + this._giveDir)].substring(1,this._structure[(this._funcscope + this._giveDir)].length);
}
else
{
return this._structure[(this._funcscope + this._giveDir)];
}
}
else
{
// nullptr
return "null";
}
}
funcref(args, util)
{
this._giveDir = args.FUNCNAME + "]";
this.debug = (this._scope + this._giveDir);
return (this._scope + this._giveDir);
}
dl()
{
return this._funcscope;
}
de()
{
if (this.debug === "all good")
{
return false;
}
else
{
return true;
}
}
set(args, util)
{
//this.debug = "set " + (this._scope + this._giveDir) + ", ";
var de = ""
var a = getBranch((this._scope + this._giveDir), false);
var b = getBranch((this._scope + this._giveDir), true);
var c = a[b-1];
if (!this._structure[(this._scope + this._giveDir)])
{
//this.debug = "a";
for (var j = 1; j < b; j++)
{
var d = a[0];
for (var i = 1; i < j; i++)
{
d = d + "," + a[i];
}
var prefix = "";
// d is the indexer
if (j < b - 1)
{
//prefix = ">";
}
if (!this._structure[d])
{
this._structure[d] = prefix + a[j];
}
else
{
this._structure[d] = this._structure[d] + "," + prefix + a[j];
}
de = de + "::" + d + " to " + this._structure[d];
}
}
if ((this._scope + this._giveDir).substring((this._scope + this._giveDir).length - 1, (this._scope + this._giveDir).length) === "[")
{
// to make sure it exists, we add "*" to the beginning of it
//this._structure[(this._scope + this._giveDir)] = "*" + args.VALUE;
var name = args.NAME.substring(0,args.NAME.length - 1);
var val = args.VALUE.substring(0,args.VALUE.length - 1);
if (val)
{
// notice val and name are swapped in the CORRECT order
this.debug = val;
this._structure = dupeBranch(this._structure, val, name);
}
else
{
this.debug = "ERROR NO" + this.debugerrorc +": Code: 3. Value to be copied is missing.";
}
}
else if ((this._scope + this._giveDir).substring((this._scope + this._giveDir).length - 1, (this._scope + this._giveDir).length) === "]")
{
this._structure[(this._scope + this._giveDir)]
}
else
{
// to make sure it exists, we add "*" to the beginning of it
this._structure[(this._scope + this._giveDir)] = "*" + args.VALUE;
this.debug = de;
}
}
tuple(args)
{
return args.VALUE1 + "," + args.VALUE2;
}
gettuple(args)
{
var a = getBranch(args.TUPLE, false);
return a[args.NUMBER - 1];
}
scop(args, util)
{
if (typeof util.stackFrame.loopCounter === 'undefined') {
util.stackFrame.loopCounter = 1;
}
if (util.stackFrame.loopCounter > 0) {
if ((this._scope + this._giveDir).substring((this._scope + this._giveDir).length - 1, (this._scope + this._giveDir).length) === "[")
{
if (this._scope === "")
{
this._scope = args.CLASSNAME.substring(0,args.CLASSNAME.length - 1) + ",";
}
else
{
this._scope = this._scope + args.CLASSNAME.substring(0,args.CLASSNAME.length - 1) + ",";
}
util.stackFrame.loopCounter--;
util.startBranch(1, true);
}
else
{
this.debug = "ERROR NO" + this.debugerrorc +": Code: 4. Input must be a class.";
}
}
else
{
this._scope = this._scope.substring(0, this._scope.length - args.CLASSNAME.length)
;
}
}
/**
* 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();
}
}
//structure = dupeBranch(structure);
function getBranch (dir, getB)
{
var c = dir;
var a = {};
var b = 0;
var m = 0;
for (var i = 0; i < c.length - 1; i++)
{
if (c.substring(i,i+1) === ",")
{
a[b] = c.substring(m, i);
b++;
m = i + 1;
}
}
a[b] = c.substring(m, i + 1);
b++;
if (getB)
{
return b;
}
return a;
}
function dupeBranch (structure, dupeDir, newDir)
{
var a = getBranch(structure[dupeDir], false);
var b = getBranch(structure[dupeDir], true);
for (var i = 0; i < b; i++){
if (a[i].substring(0, 1) === ">")
{
structure = dupeBranch (structure, dupeDir + "," + a[i], newDir + "," + a[i])
}
if (i < 1)
{
d = a[i];
}
else
{
d = d + "," + a[i];
}
structure[newDir + "," + a[i]] = structure[dupeDir + "," + a[i]];
}
structure[newDir] = d;
/* var s = structure;
var c = s[dupeDir] = "iads:yewuio:cxcv";
var a = {};
var b = 0;
var s = 0;
for (var i = 0; i < c.length - 1; i++)
{
if (c.substring(i,i+1) === ",")
{
a[b] = c.substring(s, i);
b++;
s = i + 1;
}
}
a[b] = c.substring(s, i + 1);
b++;
structure[newDir] = c;
for (var i = 0; i < b; b++)
{
structure[newDir + "," + a[b]] = s[dupeDir + "," + a[b]];
}*/
/*this._structure["Fruit/apple/ripeness"] = 0;
this._structure["Fruit/apple/count"] = 0;
this._structure["Fruit/apple"] = "ripeness:count";*/
return structure;
}
module.exports = Scratch3SOOP;
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment