soiz1's picture
Upload 811 files
30c32c8 verified
raw
history blame
16.5 kB
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