forked from sugarlabs/musicblocks
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathgenerate.js
384 lines (346 loc) · 14.1 KB
/
generate.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
/* eslint-disable no-console */
/**
* @file This contains the utilities for generating code from block stacks for JS Editor widget.
* @author Anindya Kundu
*
* @copyright 2020 Anindya Kundu
*
* @license
* This program is free software; you can redistribute it and/or modify it under the terms of the
* The GNU Affero General Public License as published by the Free Software Foundation; either
* version 3 of the License, or (at your option) any later version.
*
* You should have received a copy of the GNU Affero General Public License along with this
* library; if not, write to the Free Software Foundation, 51 Franklin Street, Suite 500 Boston,
* MA 02110-1335 USA.
*/
/* global globalActivity, JSInterface, last, ASTUtils, astring */
/* exported JSGenerate */
/**
* @class
* @classdesc
* Contains data structures that store the list of start blocks and actions blocks.
* Conatins utility methods to generate block stack trees of start and action blocks.
* Contains utility method that generates AST for the corresponding code for the trees.
*
* Code is generated from the Abstract Syntax Tree using a library called "Astring".
*
* * Internal functions' names are in PascalCase.
*/
class JSGenerate {
/** list of the Block index numbers of all start blocks */
static startBlocks = [];
/** list of the Block index numbers of all action blocks */
static actionBlocks = [];
/** list of all the generated trees of start blocks */
static startTrees = [];
/** list of all the generated trees of action blocks */
static actionTrees = [];
/** list of all the action names corresponding to the generated trees of action blocks */
static actionNames = [];
/** Abstract Syntax Tree for the corresponding code to be generated from the stack trees */
static AST = {};
/** final output code generated from the AST of the block stack trees */
static code = "";
/** whether code generation has failed */
static generateFailed = false;
/**
* Generates a tree representation of the "start" and "action" block stacks.
*
* @static
* @returns {void}
*/
static generateStacksTree() {
JSGenerate.startBlocks = [];
JSGenerate.actionBlocks = [];
JSGenerate.startTrees = [];
JSGenerate.actionTrees = [];
JSGenerate.actionNames = [];
globalActivity.blocks.findStacks();
for (const blk of globalActivity.blocks.stackList) {
if (
globalActivity.blocks.blockList[blk].name === "start" &&
!globalActivity.blocks.blockList[blk].trash
) {
JSGenerate.startBlocks.push(blk);
} else if (
globalActivity.blocks.blockList[blk].name === "action" &&
!globalActivity.blocks.blockList[blk].trash
) {
// does the action stack have a name?
const c = globalActivity.blocks.blockList[blk].connections[1];
// is there a block in the action clamp?
const b = globalActivity.blocks.blockList[blk].connections[2];
if (c !== null && b !== null) {
JSGenerate.actionBlocks.push(blk);
}
}
}
/**
* Recursively generates a tree representation of the block stack starting with blk.
*
* @param {Number} blk - Block representation in blocks.blockList
* @param {Object} tree - list structure to store tree from blk
* @param {Number} level - for clamp/doubleclamp blocks: 0/undefined for first flow, 1 for second flow
* @returns {[*]} tree from blk
*/
function GenerateStackTree(blk, tree, level) {
/**
* Recursively generates a tree representation of the arguments starting with blk.
*
* @param {Object} blk - Block object
* @returns {[*]} tree representation
*/
function ParseArg(blk) {
let argLen = blk.protoblock.args;
if (blk.protoblock.style === "clamp") {
argLen -= 1;
} else if (blk.protoblock.style === "doubleclamp") {
argLen -= 2;
}
const args = [];
for (let i = 1; i <= argLen; i++) {
let arg = globalActivity.blocks.blockList[blk.connections[i]];
if (arg === undefined) {
args.push(null);
continue;
}
if (
arg.protoblock.style === "clamp" &&
arg.protoblock._style.flows.left === true
) {
console.warn(`CANNOT PROCESS "${arg.name}" BLOCK`);
args.push(null);
continue;
}
// ignore horizontal spacers
while (arg.name === "hspace") {
arg = globalActivity.blocks.blockList[arg.connections[1]];
}
if (arg.protoblock.style === "value") {
if (JSInterface.isGetter(arg.name)) {
args.push([arg.name, null]);
} else if (
arg.protoblock.__proto__.__proto__.constructor.name === "BooleanBlock"
) {
if (arg.name === "boolean") {
args.push("bool_" + arg.value);
} else {
args.push([arg.name, ParseArg(arg)]);
}
} else {
if (arg.name === "namedbox") {
args.push("box_" + arg.privateData);
} else {
args.push(arg.value);
}
}
} else if (arg.protoblock.style === "arg") {
args.push([arg.name, ParseArg(arg)]);
}
}
return args;
}
if (level === undefined) level = 0;
let nextBlk = blk.connections[blk.connections.length - 2 - level];
nextBlk = globalActivity.blocks.blockList[nextBlk];
while (nextBlk !== undefined) {
// ignore vertical spacers and hidden blocks
if (nextBlk.name !== "hidden" && nextBlk.name !== "vspace") {
if (["storein2", "nameddo"].includes(nextBlk.name)) {
tree.push([nextBlk.name + "_" + nextBlk.privateData]);
} else {
tree.push([nextBlk.name]);
}
const args = ParseArg(nextBlk);
last(tree).push(args.length === 0 ? null : args);
if (nextBlk.protoblock.style === "clamp") {
last(tree).push(GenerateStackTree(nextBlk, []));
} else if (nextBlk.protoblock.style === "doubleclamp") {
last(tree).push(GenerateStackTree(nextBlk, [], 1));
last(tree).push(GenerateStackTree(nextBlk, [], 0));
} else {
last(tree).push(null);
}
}
if (nextBlk.connections.length > 0) {
nextBlk = globalActivity.blocks.blockList[last(nextBlk.connections)];
if (nextBlk === undefined) break;
} else {
break;
}
}
return tree;
}
for (const blk of JSGenerate.startBlocks) {
JSGenerate.startTrees.push(GenerateStackTree(globalActivity.blocks.blockList[blk], []));
}
for (const blk of JSGenerate.actionBlocks) {
const actionName =
globalActivity.blocks.blockList[globalActivity.blocks.blockList[blk].connections[1]]
.value;
if (actionName === null || actionName === undefined) continue;
JSGenerate.actionNames.push(actionName);
JSGenerate.actionTrees.push(
GenerateStackTree(globalActivity.blocks.blockList[blk], [])
);
}
}
/**
* Prints all the tree representations of start and action block stacks.
*
* @static
* @returns {void}
*/
static printStacksTree() {
/**
* Recursively prints a tree starting from level.
*
* @param {[*]} tree - block stack tree
* @param {Number} level - nesting level
*/
function PrintTree(tree, level) {
/**
* Recursively generates the string of arguments.
*
* @param {[*]} args - arguments list
* @returns {String} serialized arguments
*/
function PrintArgs(args) {
if (args === null || args.length === 0) return "none";
let str = "(";
for (const arg of args) {
if (arg === null) {
str += "null";
} else if (typeof arg === "object") {
str += arg[0] + PrintArgs(arg[1]);
} else {
str += arg;
}
str += ", ";
}
str = str.substring(0, str.length - 2) + ")";
return str;
}
if (level === undefined) level = 0;
for (const i of tree) {
let spaces = "";
for (let j = 0; j < 4 * level; j++) spaces += " ";
console.log(
"%c" + spaces + "%c" + i[0] + " : " + "%c" + PrintArgs(i[1]),
"background: mediumslateblue",
"background; none",
"color: dodgerblue"
);
if (i[2] !== null) {
PrintTree(i[2], level + 1);
if (i.length === 4 && i[3] !== null) {
console.log("%c" + spaces + "** NEXTFLOW **", "color: green");
PrintTree(i[3], level + 1);
}
}
}
}
if (JSGenerate.startTrees.length === 0) {
console.log("%cno start trees generated", "color: tomato");
} else {
for (const tree of JSGenerate.startTrees) {
console.log(
"\n " + "%c START ",
"background: navy; color: white; font-weight: bold"
);
PrintTree(tree);
console.log("\n");
}
}
if (JSGenerate.startTrees.length === 0) {
console.log("%cno action trees generated", "color: tomato");
} else {
for (const tree of JSGenerate.actionTrees) {
console.log(
"\n " + "%c ACTION ",
"background: green; color: white; font-weight: bold"
);
PrintTree(tree);
console.log("\n");
}
}
}
/**
* Generates the corresponding code to the generated from the stack trees by structuring the
* different Abstract Syntax Trees.
*
* @static
* @returns {void}
*/
static generateCode() {
JSGenerate.generateFailed = false;
JSGenerate.AST = JSON.parse(JSON.stringify(ASTUtils.BAREBONE_AST));
try {
for (let i = 0; i < JSGenerate.actionTrees.length; i++) {
JSGenerate.AST["body"].splice(
i,
0,
ASTUtils.getMethodAST(JSGenerate.actionNames[i], JSGenerate.actionTrees[i])
);
}
const offset = JSGenerate.actionTrees.length;
for (let i = 0; i < JSGenerate.startTrees.length; i++) {
JSGenerate.AST["body"].splice(
i + offset,
0,
ASTUtils.getMouseAST(JSGenerate.startTrees[i])
);
}
} catch (e) {
JSGenerate.generateFailed = true;
console.error("CANNOT GENERATE ABSTRACT SYNTAX TREE\nError:", e);
}
if (!JSGenerate.generateFailed) {
try {
JSGenerate.code = astring.generate(JSGenerate.AST, { indent: " " });
} catch (e) {
JSGenerate.generateFailed = true;
console.error("CANNOT GENERATE CODE\nError: INVALID ABSTRACT SYNTAX TREE");
}
}
if (JSGenerate.generateFailed) {
const AST = JSON.parse(JSON.stringify(ASTUtils.BAREBONE_AST));
AST["body"].splice(0, 0, ASTUtils.getMouseAST([]));
JSGenerate.code = astring.generate(AST);
}
}
/**
* Runs the code generator
* * Runs the blocks stacks tree generator
* * Generates Abstract Syntax Trees from the blocks stacks trees
* * Serializes the Abstract Syntax Tree into JavaScript code
*
* @static
* @param {Object} activity - the Activity object
* @param {Boolean} printStacksTree - whether to print the stacks tree in the browser console
* @param {Boolean} printCode - whether to print the generated code in the browser console
* @returns {void}
*/
static run(printStacksTree, printCode) {
JSGenerate.generateStacksTree();
if (printStacksTree) {
console.log(
"\n " + "%c STACK TREES ",
"background: greenyellow; color: midnightblue; font-weight: bold"
);
JSGenerate.printStacksTree();
}
JSGenerate.generateCode();
if (printStacksTree & printCode) {
console.log("%c _______________________________________", "color: darkorange");
}
if (printCode) {
console.log(
"\n " + "%c CODE ",
"background: greenyellow; color: midnightblue; font-weight: bold"
);
console.log(JSGenerate.code);
}
}
}