const BlockType = require('../../extension-support/block-type') const BlockShape = require('../../extension-support/block-shape') const ArgumentType = require('../../extension-support/argument-type') const Cast = require('../../util/cast') let arrayLimit = 2 ** 32 // credit to sharpool because i stole the for each code from his extension haha im soo evil /** * @param {number} x * @returns {string} */ function formatNumber(x) { if (x >= 1e6) { return x.toExponential(4) } else { x = Math.floor(x * 1000) / 1000 return x.toFixed(Math.min(3, (String(x).split('.')[1] || '').length)) } } function clampIndex(x) { return Math.min(Math.max(x, 0), arrayLimit) } function span(text) { let el = document.createElement('span') el.innerHTML = text el.style.display = 'hidden' el.style.whiteSpace = 'nowrap' el.style.width = '100%' el.style.textAlign = 'center' return el } class ArrayType { customId = "jwArray" array = [] constructor(array = []) { this.array = array } static toArray(x) { if (x instanceof ArrayType) return new ArrayType([...x.array]) if (x instanceof Array) return new ArrayType([...x]) if (x === "" || x === null || x === undefined) return new ArrayType() try { let parsed = JSON.parse(x) if (parsed instanceof Array) return new ArrayType(parsed) } catch {} return new ArrayType([x]) } static forArray(x) { if (x instanceof ArrayType) return new ArrayType([...x.array]) return x } static display(x) { try { switch (typeof x) { case "object": if (x === null) return "null" if (typeof x.jwArrayHandler == "function") { return x.jwArrayHandler() } return Cast.toString(x) case "undefined": return "null" case "number": return formatNumber(x) case "boolean": return x ? "true" : "false" case "string": return `"${Cast.toString(x)}"` } } catch {} return "?" } jwArrayHandler() { return `Array<${formatNumber(this.array.length)}>` } toString() { return JSON.stringify(this.array) } toMonitorContent = () => span(this.toString()) toReporterContent() { let root = document.createElement('div') root.style.display = 'flex' root.style.flexDirection = 'column' root.style.justifyContent = 'center' let arrayDisplay = span(`[${this.array.slice(0, 50).map(v => ArrayType.display(v)).join(', ')}]`) arrayDisplay.style.overflow = "hidden" arrayDisplay.style.whiteSpace = "nowrap" arrayDisplay.style.textOverflow = "ellipsis" arrayDisplay.style.maxWidth = "256px" root.appendChild(arrayDisplay) root.appendChild(span(`Length: ${this.array.length}`)) return root } get length() { return this.array.length } } const jwArray = { Type: ArrayType, Block: { blockType: BlockType.REPORTER, blockShape: BlockShape.SQUARE, forceOutputType: "Array", disableMonitor: true }, Argument: { shape: BlockShape.SQUARE, check: ["Array"] } } class Extension { constructor() { vm.jwArray = jwArray vm.runtime.registerSerializer( //this basically copies variable serialization "jwArray", v => v.array.map(w => { if (typeof w == "object" && w != null && w.customId) { return { customType: true, typeId: w.customId, serialized: vm.runtime.serializers[w.customId].serialize(w) }; } return w }), v => new jwArray.Type(v.map(w => { if (typeof w == "object" && w != null && w.customType) { return vm.runtime.serializers[w.typeId].deserialize(w.serialized) } return w })) ); } getInfo() { return { id: "jwArray", name: "Arrays", color1: "#ff513d", menuIconURI: "", blocks: [ { opcode: 'blank', text: 'blank array', ...jwArray.Block }, { opcode: 'blankLength', text: 'blank array of length [LENGTH]', arguments: { LENGTH: { type: ArgumentType.NUMBER, defaultValue: 1 } }, ...jwArray.Block }, { opcode: 'fromList', text: 'array from list [LIST]', arguments: { LIST: { menu: "list" } }, hideFromPalette: true, //doesn't work for some reason ...jwArray.Block }, { opcode: 'split', text: 'split [STRING] by [DIVIDER]', arguments: { STRING: { type: ArgumentType.STRING, defaultValue: "foo" }, DIVIDER: { type: ArgumentType.STRING } }, ...jwArray.Block }, "---", { opcode: 'get', text: 'get [INDEX] in [ARRAY]', blockType: BlockType.REPORTER, allowDropAnywhere: true, arguments: { ARRAY: jwArray.Argument, INDEX: { type: ArgumentType.NUMBER, defaultValue: 1 } } }, { opcode: 'index', text: 'index of [VALUE] in [ARRAY]', blockType: BlockType.REPORTER, arguments: { ARRAY: jwArray.Argument, VALUE: { type: ArgumentType.STRING, defaultValue: "foo", exemptFromNormalization: true } } }, { opcode: 'has', text: '[ARRAY] has [VALUE]', blockType: BlockType.BOOLEAN, arguments: { ARRAY: jwArray.Argument, VALUE: { type: ArgumentType.STRING, exemptFromNormalization: true } } }, { opcode: 'length', text: 'length of [ARRAY]', blockType: BlockType.REPORTER, arguments: { ARRAY: jwArray.Argument } }, "---", { opcode: 'set', text: 'set [INDEX] in [ARRAY] to [VALUE]', arguments: { ARRAY: jwArray.Argument, INDEX: { type: ArgumentType.NUMBER, defaultValue: 1 }, VALUE: { type: ArgumentType.STRING, defaultValue: "foo", exemptFromNormalization: true } }, ...jwArray.Block }, { opcode: 'append', text: 'append [VALUE] to [ARRAY]', arguments: { ARRAY: jwArray.Argument, VALUE: { type: ArgumentType.STRING, defaultValue: "foo", exemptFromNormalization: true } }, ...jwArray.Block }, { opcode: 'concat', text: 'merge [ONE] with [TWO]', arguments: { ONE: jwArray.Argument, TWO: jwArray.Argument }, ...jwArray.Block }, { opcode: 'fill', text: 'fill [ARRAY] with [VALUE]', arguments: { ARRAY: jwArray.Argument, VALUE: { type: ArgumentType.STRING, defaultValue: "foo", exemptFromNormalization: true } }, ...jwArray.Block }, { opcode: 'splice', text: 'splice [ARRAY] at [INDEX] with [ITEMS] items', arguments: { ARRAY: jwArray.Argument, INDEX: { type: ArgumentType.NUMBER, defaultValue: 1 }, ITEMS: { type: ArgumentType.NUMBER, defaultValue: 1 } }, ...jwArray.Block }, "---", { opcode: 'reverse', text: 'reverse [ARRAY]', arguments: { ARRAY: jwArray.Argument }, ...jwArray.Block }, "---", { opcode: 'forEachI', text: 'index', blockType: BlockType.REPORTER, hideFromPalette: true, canDragDuplicate: true }, { opcode: 'forEachV', text: 'value', blockType: BlockType.REPORTER, hideFromPalette: true, allowDropAnywhere: true, canDragDuplicate: true }, { opcode: 'forEach', text: 'for [I] [V] of [ARRAY]', blockType: BlockType.LOOP, arguments: { ARRAY: jwArray.Argument, I: { fillIn: 'forEachI' }, V: { fillIn: 'forEachV' } } }, /*{ opcode: 'forEachBreak', text: 'break', blockType: BlockType.COMMAND, isTerminal: true }*/ ], menus: { list: { acceptReporters: false, items: "getLists", }, } }; } getLists() { const globalLists = Object.values(vm.runtime.getTargetForStage().variables) .filter((x) => x.type == "list"); const localLists = Object.values(vm.editingTarget.variables) .filter((x) => x.type == "list"); const uniqueLists = [...new Set([...globalLists, ...localLists])]; if (uniqueLists.length === 0) return [{ text: "", value: "" }]; return uniqueLists.map((v) => ({ text: v.name, value: new jwArray.Type(v.value) })); } blank() { return new jwArray.Type() } blankLength({LENGTH}) { LENGTH = clampIndex(Cast.toNumber(LENGTH)) return new jwArray.Type(Array(LENGTH).fill(undefined)) } fromList({LIST}) { return jwArray.Type.toArray(LIST) } split({STRING, DIVIDER}) { STRING = Cast.toString(STRING) DIVIDER = Cast.toString(DIVIDER) return new jwArray.Type(STRING.split(DIVIDER)) } get({ARRAY, INDEX}) { ARRAY = jwArray.Type.toArray(ARRAY) return ARRAY.array[Cast.toNumber(INDEX)-1] || "" } index({ARRAY, VALUE}) { ARRAY = jwArray.Type.toArray(ARRAY) return ARRAY.array.indexOf(VALUE) + 1 } has({ARRAY, VALUE}) { ARRAY = jwArray.Type.toArray(ARRAY) return ARRAY.array.includes(VALUE) } length({ARRAY}) { ARRAY = jwArray.Type.toArray(ARRAY) return ARRAY.length } set({ARRAY, INDEX, VALUE}) { ARRAY = jwArray.Type.toArray(ARRAY) INDEX = Cast.toNumber(INDEX) ARRAY.array[clampIndex(Cast.toNumber(INDEX)-1)] = jwArray.Type.forArray(VALUE) return ARRAY } append({ARRAY, VALUE}) { ARRAY = jwArray.Type.toArray(ARRAY) ARRAY.array.push(jwArray.Type.forArray(VALUE)) return ARRAY } concat({ONE, TWO}) { ONE = jwArray.Type.toArray(ONE) TWO = jwArray.Type.toArray(TWO) return new jwArray.Type(ONE.array.concat(TWO.array)) } fill({ARRAY, VALUE}) { ARRAY = jwArray.Type.toArray(ARRAY) ARRAY.array.fill(jwArray.Type.forArray(VALUE)) return ARRAY } splice({ARRAY, INDEX, ITEMS}) { ARRAY = jwArray.Type.toArray(ARRAY) INDEX = Cast.toNumber(INDEX) ITEMS = Cast.toNumber(ITEMS) ARRAY.array.splice(INDEX - 1, ITEMS) return ARRAY } reverse({ARRAY}) { ARRAY = jwArray.Type.toArray(ARRAY) ARRAY.array.reverse() return ARRAY } forEachI({}, util) { let arr = util.thread.stackFrames[0].jwArray return arr ? Cast.toNumber(arr[0]) + 1 : 0 } forEachV({}, util) { let arr = util.thread.stackFrames[0].jwArray return arr ? arr[1] : "" } forEach({ARRAY}, util) { ARRAY = jwArray.Type.toArray(ARRAY) if (util.stackFrame.execute) { util.stackFrame.index++; const { index, entry } = util.stackFrame; if (index > entry.length - 1) return; util.thread.stackFrames[0].jwArray = entry[index]; } else { const entry = Object.entries(ARRAY.array); if (entry.length === 0) return; util.stackFrame.entry = entry; util.stackFrame.execute = true; util.stackFrame.index = 0; util.thread.stackFrames[0].jwArray = entry[0]; } util.startBranch(1, true); } forEachBreak({}, util) { util.stackFrame.entry = [] } } module.exports = Extension