// Name: Soup Objects // Author: soup // Description: JSON object handling with some helpful tools! Combine with Soup Arrays for the full experience. // ID: souputilsobjects // Version 4.2 --- 23 blocks // Part of the Soup Utils pack, which contains // Soup Utils, Soup Objects, Soup Arrays, and Soup Vectors // NOTES & CREDITS // // - Written by Soup (https://penguinmod.com/profile?user=soup) // - Made for PenguinMod; should also work in TurboWarp // - Used some code from SharkPool's "Sty-Lists" and "Variables Expanded" extensions involving use of variables // ############ ############ #### #### ############ ############ ######## ############ ############ ############ ############ ############ // ############ ############ #### #### ############ ############ ######## ############ ############ ############ ############ ############ // #### #### #### #### #### #### #### #### #### #### #### #### #### #### #### #### // #### #### #### #### #### #### #### #### #### #### #### #### #### #### #### #### // ############ #### #### #### #### ############ #### #### ######## #### ######## #### #### ############ // ############ #### #### #### #### ############ #### #### ######## #### ######## #### #### ############ // #### #### #### #### #### #### #### #### #### #### #### #### #### #### #### // #### #### #### #### #### #### #### #### #### #### #### #### #### #### #### // ############ ############ ############ #### ############ ######## ######## ############ ############ #### ############ // ############ ############ ############ #### ############ ######## ######## ############ ############ #### ############ (function(Scratch) { 'use strict'; if (!Scratch.extensions.unsandboxed) { throw new Error('Soup Objects must be unsandboxed to run!'); } let prev_extensions = Array.from(vm.extensionManager._loadedExtensions.keys()); if (prev_extensions.includes('souputilsobjects')) { return; // extension already loaded } // ###### ## ###### #### ###### ## ###### // ## ## ## ## ## ## ## ## ## ## // ## ## ## ## #### ###### ## ###### // ## ## ## ## ## ## ## ## ## ## ## // ###### ###### ###### #### ## ## ###### ###### // These globals are used across all extensions in the Soup Utils pack, and the entire "GLOBALS" section should be copy-pasted to every extension in the pack. // The one exception is that the extension's icon data (the next 2 lines) is specific to only that extension. const object_icon = ""; const object_menu_icon = ""; const goldenRatio = (1 + Math.sqrt(5)) / 2; const sqrt3 = Math.sqrt(3); const gammaConstant = 0.57721566490153286060651209008240243104215933593992; const runtime = vm.runtime; function cast_to_array(array) { let object = null; try { object = JSON.parse(array); if (!(object.constructor === [].constructor)) { object = []; } } catch { object = []; } return object; } function cast_to_object(str_object) { let object = null; try { object = JSON.parse(str_object); if (!(object.constructor === {}.constructor)) { object = {}; } } catch { object = {}; } return object; } function cast_to_vector(array,length=null) { array = cast_to_array(array); let vector = []; for (let i = 0; i < array.length; i++) { vector.push(Scratch.Cast.toNumber(stringify(array[i]))); } if (!(length === null)) { vector = vector.slice(0,length); for (let i = vector.length; i < length; i++) { vector.push(0); } } return vector; } function mod(a,b) { return ((a % b) + b) % b; } function json_stringify_censor(key, value) { if (value === Infinity) { return 'Infinity'; } else if (value === -Infinity) { return '-Infinity'; } else if (Number.isNaN(value)) { return 0; } else { return value; } } function stringify(json_object) { if (json_object === undefined) { return 'undefined'; } else if (json_object === null) { return 'null'; } else if (json_object.constructor === [].constructor || json_object.constructor === {}.constructor) { return JSON.stringify(json_object,json_stringify_censor); } else { return json_object.toString(); } } function convert_angle_units(angle,from_unit,to_unit,relative=false) { let degrees_units = ['Scratch degrees','trig degrees']; let radians_units = ['Scratch radians','trig radians']; let full_circle_units = ['Scratch full circle','trig full circle']; let scratch_units = ['Scratch degrees','Scratch radians','Scratch full circle']; // convert angle to degrees let degrees = null; if (degrees_units.includes(from_unit)) { degrees = angle; } else if (radians_units.includes(from_unit)) { degrees = angle * (180 / Math.PI); } else if (full_circle_units.includes(from_unit)) { degrees = angle * 360; } else { return 0; } // convert degrees into trig degrees if (scratch_units.includes(from_unit)) { if (relative) { degrees = -degrees; } else { degrees = 90 - degrees; } } // convert trig degrees to scratch/trig degrees depending on to_unit if (scratch_units.includes(to_unit)) { if (relative) { degrees = -degrees; } else { degrees = 90 - degrees; } } // mod degrees if (scratch_units.includes(to_unit)) { degrees = 180 - mod((180 - degrees),360); } else { degrees = mod(degrees,360); } // convert degrees to degrees/radians/full circle depending on to_unit let newAngle = null; if (degrees_units.includes(to_unit)) { newAngle = degrees; } else if (radians_units.includes(to_unit)) { newAngle = degrees * (Math.PI / 180); } else if (full_circle_units.includes(to_unit)) { newAngle = degrees / 360; } else { return 0; } return newAngle; } function stage_width() { return Scratch.renderer.canvas.width; } function stage_height() { return Scratch.renderer.canvas.height; } function convert_coordinate_units(xy,from_unit,to_unit) { let width = stage_width(); let height = stage_height(); if (from_unit == 'Scratch coords' && to_unit == 'JS coords') { return [xy[0] + width/2, height - (xy[1] + height/2)]; } else if (from_unit == 'JS coords' && to_unit == 'Scratch coords') { return [xy[0] - width/2, -((xy[1] - height) + height/2)]; } else if (from_unit == to_unit) { return xy; } else { return [0,0]; } } function var_id_from_name(variable_name,util) { // copied from SharkPool //support for all variable types (Cloud, Sprite-Only, Global) variable_name = Scratch.Cast.toString(variable_name); const cloudID = runtime.getTargetForStage().lookupVariableByNameAndType(Scratch.Cast.toString("☁ " + variable_name), ""); if (cloudID) return cloudID.id; let varFind = ""; for (const name of Object.getOwnPropertyNames(util.target.variables)) { varFind = util.target.variables[name].name; if (varFind === variable_name) return util.target.variables[name].id; } const ID = runtime.getTargetForStage().lookupVariableByNameAndType(variable_name, ""); if (!ID) return ""; return ID.id; } function var_name_exists(variable_name,util) { return Scratch.Cast.toBoolean(var_id_from_name(variable_name,util)); } function get_var(variable_name,util) { if (!var_name_exists(variable_name,util)) { return 'undefined'; } let variable = util.target.lookupOrCreateVariable(variable_name,variable_name); return variable.value; } function set_var(variable_name,value,util) { if (!var_name_exists(variable_name,util)) { return; } let variable = util.target.lookupOrCreateVariable(variable_name,variable_name); variable.value = value; if (variable.isCloud) { // added from TurboWarp's Scratch 3 code util.ioQuery('cloud', 'requestUpdateVariable', [variable.name, value]); } } // get list for variables menus; from SharkPool's extension (see top for credits) function get_variables_for_menu() { const globalVars = Object.values(vm.runtime.getTargetForStage().variables).filter((x) => x.type == ""); const localVars = Object.values(vm.editingTarget.variables).filter((x) => x.type == ""); const uniqueVars = [...new Set([...globalVars, ...localVars])]; if (uniqueVars.length === 0) return ["(choose a variable)"]; return uniqueVars.map((i) => (Scratch.Cast.toString(i.name))); } function bulk_math(array,operation) { if (operation == 'min') { return Math.min(...array); } else if (operation == 'max') { return Math.max(...array); } else if (operation == 'sum') { let sum = 0; for (let i = 0; i < array.length; i++) { sum += array[i]; } return sum; } else if (operation == 'product') { let product = 1; for (let i = 0; i < array.length; i++) { product *= array[i]; } return product; } else if (operation == 'average') { let sum = 0; for (let i = 0; i < array.length; i++) { sum += array[i]; } return sum / array.length; } else { return 0; } } function get_illion(n) { n = Math.round(n); // ensure integer // skip all logic and say "thousand" if n is 0 if (n <= 0) { return 'thousand'; } // convert n into a base-1000 number with its digits in "sections" from most significant to least significant (in other words splitting 1234567 into 1,234,567 where you would write commas) let sections = []; let place_value = 1; while (n >= place_value) { sections.push(mod(Math.floor(n / place_value), 1000)); place_value *= 1000; } // call "get_illion_short" for each base-1000 "digit" and combine them into one string let str = ''; for (let i = 0; i < sections.length; i++) { str += get_illion_short(sections[i]); } // add the final "-on" to finish "illion" str += 'on'; return str; } function get_illion_short(n) { // See this for more info about the algorithm: https://en.wikipedia.org/wiki/Names_of_large_numbers#Extensions_of_the_standard_dictionary_numbers // return one of the basic 10 illions if n < 10 if (n < 10) { let small_illions = ['nilli','milli','billi','trilli','quadrilli','quintilli','sextilli','septilli','octilli','nonilli']; return small_illions[n]; } // get each component of the illion let possible_ones = ['','un','duo','tre','quattuor','quin','se','septe','octo','nove']; let possible_tens = ['','deci','viginti','triginta','quadraginta','quinquaginta','sexaginta','septuaginta','octoginta','nonaginta']; let possible_hundreds = ['','centi','ducenti','trecenti','quadringenti','quingenti','sescenti','septingenti','octingenti','nongenti']; let ones_num = mod(n, 10); let ones = possible_ones[ones_num]; let tens_num = mod(Math.floor(n / 10), 10); let tens = possible_tens[tens_num]; let hundreds_num = Math.floor(n / 100); let hundreds = possible_hundreds[hundreds_num]; // apply marks to the ones component let markable = ['tre','se','septe','nove']; if (markable.includes(ones)) { // find marks that need to be applied let marks = { '':[],'un':[],'duo':[],'tre':[],'quattuor':[],'quin':[],'se':[],'septe':[],'octo':[],'nove':[], 'deci':['N'],'viginti':['M','S'],'triginta':['N','S'],'quadraginta':['N','S'],'quinquaginta':['N','S'],'sexaginta':['N'],'septuaginta':['N'],'octoginta':['M','X'],'nonaginta':[], 'centi':['NX'],'ducenti':['N'],'trecenti':['N','S'],'quadringenti':['N','S'],'quingenti':['N','S'],'sescenti':['N'],'septingenti':['N'],'octingenti':['M','X'],'nongenti':[] }; let applied_marks = marks[tens]; if (tens == '') { applied_marks = marks[hundreds]; } // apply them if (applied_marks.includes('S')) { if (ones == 'tre') { ones = 'tres'; } if (ones == 'se') { ones = 'ses'; } } if (applied_marks.includes('X')) { if (ones == 'tre') { ones = 'tres'; } if (ones == 'se') { ones = 'sex'; } } if (applied_marks.includes('M')) { if (ones == 'septe') { ones = 'septem'; } if (ones == 'nove') { ones = 'novem'; } } if (applied_marks.includes('N')) { if (ones == 'septe') { ones = 'septen'; } if (ones == 'nove') { ones = 'noven'; } } } // add "illi" at the end of names such as "quadraginta" that don't have it already // the only ones that would already have "illi" at the end are covered by "small_illions" at the start of the function let str = ones + tens + hundreds; str = str.slice(0, -1) + 'illi'; return str; } function get_abbreviated_illion(n) { let abbreviated_illions = ['k','m','b','t','qa','qi','sx','sp','o','n','d','ud','dd','td','qad','qid','sxd','spd','od','nd','v','uv','dv','tv','qav','qiv','sxv','spv','ov','nv','tg','utg','dtg','ttg','qatg','qitg','sxtg','sptg','otg','ntg','qag','uqag','dqag','tqag','qaqag','qiqag','sxqag','spqag','oqag','nqag','qig']; if (n >= abbreviated_illions.length) { return '×10^' + (3 * n + 3); } return abbreviated_illions[n]; } // ###### ## ## ###### ###### ## ## ###### ###### ###### ## ## ###### ## ###### ###### ###### // ## ## ## ## ## #### ## ## ## ## ## #### ## ## ## ## ## ## ## // #### ## ## #### ## #### ###### ## ## ## ## #### ## ## ###### ###### ###### // ## ## ## ## ## ## ## ## ## ## ## ## ## ## ## ## ## ## ## // ###### ## ## ## ###### ## ## ###### ###### ###### ## ## ###### ###### ## ## ###### ###### class SoupUtilsObjects { getInfo() { return { id: 'souputilsobjects', name: 'Soup Objects', color1: '#f55442', color2: '#b83527', color3: '#edbbad', menuIconURI: object_menu_icon, blockIconURI: object_icon, // BLOCKS ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- blocks: [ { opcode: 'isobjectvalid', blockType: Scratch.BlockType.BOOLEAN, text: 'is object [OBJECT] valid?', arguments: { OBJECT: { type: Scratch.ArgumentType.STRING, defaultValue: '{}' } } }, { opcode: 'toobject', blockType: Scratch.BlockType.REPORTER, text: 'object [OBJECT]', arguments: { OBJECT: { type: Scratch.ArgumentType.STRING, defaultValue: '{"key": "value"}' } } }, '---', { opcode: 'getfromobject', blockType: Scratch.BlockType.REPORTER, text: 'get key [KEY] from [OBJECT]', arguments: { OBJECT: { type: Scratch.ArgumentType.STRING, defaultValue: '{"key":"value"}' }, KEY: { type: Scratch.ArgumentType.STRING, defaultValue: 'key' } } }, { opcode: 'getfromobjectwithplaceholder', blockType: Scratch.BlockType.REPORTER, text: 'key [KEY] from [OBJECT], or [PLACEHOLDER] if it doesn\'t exist', arguments: { OBJECT: { type: Scratch.ArgumentType.STRING, defaultValue: '{"key":"value"}' }, KEY: { type: Scratch.ArgumentType.STRING, defaultValue: 'key' }, PLACEHOLDER: { type: Scratch.ArgumentType.STRING, defaultValue: 'placeholder' } } }, { opcode: 'setinobject', blockType: Scratch.BlockType.REPORTER, text: 'set key [KEY] to [VALUE] in [OBJECT]', arguments: { OBJECT: { type: Scratch.ArgumentType.STRING, defaultValue: '{}' }, KEY: { type: Scratch.ArgumentType.STRING, defaultValue: 'key' }, VALUE: { type: Scratch.ArgumentType.STRING, defaultValue: 'value' } } }, { opcode: 'deletefromobject', blockType: Scratch.BlockType.REPORTER, text: 'delete key [KEY] from [OBJECT]', arguments: { OBJECT: { type: Scratch.ArgumentType.STRING, defaultValue: '{"key":"value"}' }, KEY: { type: Scratch.ArgumentType.STRING, defaultValue: 'key' } } }, { opcode: 'objecthaskey', blockType: Scratch.BlockType.BOOLEAN, text: 'key [KEY] exists in [OBJECT]?', arguments: { OBJECT: { type: Scratch.ArgumentType.STRING, defaultValue: '{"key":"value"}' }, KEY: { type: Scratch.ArgumentType.STRING, defaultValue: 'key' } } }, '---', { opcode: 'objectkeys', blockType: Scratch.BlockType.REPORTER, text: 'all keys in [OBJECT]', arguments: { OBJECT: { type: Scratch.ArgumentType.STRING, defaultValue: '{"key":"value"}' } } }, { opcode: 'objectvalues', blockType: Scratch.BlockType.REPORTER, text: 'all values in [OBJECT]', arguments: { OBJECT: { type: Scratch.ArgumentType.STRING, defaultValue: '{"key":"value"}' } } }, { opcode: 'objectlength', blockType: Scratch.BlockType.REPORTER, text: 'number of keys in [OBJECT]', arguments: { OBJECT: { type: Scratch.ArgumentType.STRING, defaultValue: '{"key":"value"}' } } }, '---', { opcode: 'objectfromarrays', blockType: Scratch.BlockType.REPORTER, text: 'construct object from keys [KEYS] and values [VALUES]', arguments: { KEYS: { type: Scratch.ArgumentType.STRING, defaultValue: '["key"]' }, VALUES: { type: Scratch.ArgumentType.STRING, defaultValue: '["value"]' } } }, { opcode: 'transposeobject', blockType: Scratch.BlockType.REPORTER, text: 'transpose object [OBJECT]', arguments: { OBJECT: { type: Scratch.ArgumentType.STRING, defaultValue: '{"key":"value"}' } } }, { blockType: Scratch.BlockType.LABEL, text: "Soup Objects - Variables" }, { opcode: 'vartoobject', blockType: Scratch.BlockType.COMMAND, text: 'set [VARIABLE] to object [OBJECT]', arguments: { OBJECT: { type: Scratch.ArgumentType.STRING, defaultValue: '{"key": "value"}' }, VARIABLE: { type: Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' } } }, '---', { opcode: 'getfromobjectvar', blockType: Scratch.BlockType.REPORTER, text: 'get key [KEY] from var [VARIABLE]', arguments: { VARIABLE: { type: Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' }, KEY: { type: Scratch.ArgumentType.STRING, defaultValue: 'key' } } }, { opcode: 'getfromobjectvarwithplaceholder', blockType: Scratch.BlockType.REPORTER, text: 'key [KEY] from var [VARIABLE], or [PLACEHOLDER] if it doesn\'t exist', arguments: { VARIABLE: { type: Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' }, KEY: { type: Scratch.ArgumentType.STRING, defaultValue: 'key' }, PLACEHOLDER: { type: Scratch.ArgumentType.STRING, defaultValue: 'placeholder' } } }, { opcode: 'setinobjectvar', blockType: Scratch.BlockType.COMMAND, text: 'set key [KEY] to [VALUE] in var [VARIABLE]', arguments: { VARIABLE: { type: Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' }, KEY: { type: Scratch.ArgumentType.STRING, defaultValue: 'key' }, VALUE: { type: Scratch.ArgumentType.STRING, defaultValue: 'value' } } }, { opcode: 'deletefromobjectvar', blockType: Scratch.BlockType.COMMAND, text: 'delete key [KEY] from var [VARIABLE]', arguments: { VARIABLE: { type: Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' }, KEY: { type: Scratch.ArgumentType.STRING, defaultValue: 'key' } } }, { opcode: 'objectvarhaskey', blockType: Scratch.BlockType.BOOLEAN, text: 'key [KEY] exists in var [VARIABLE]?', arguments: { VARIABLE: { type: Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' }, KEY: { type: Scratch.ArgumentType.STRING, defaultValue: 'key' } } }, '---', { opcode: 'objectvarkeys', blockType: Scratch.BlockType.REPORTER, text: 'all keys in var [VARIABLE]', arguments: { VARIABLE: { type: Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' } } }, { opcode: 'objectvarvalues', blockType: Scratch.BlockType.REPORTER, text: 'all values in var [VARIABLE]', arguments: { VARIABLE: { type: Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' } } }, { opcode: 'objectvarlength', blockType: Scratch.BlockType.REPORTER, text: 'number of keys in var [VARIABLE]', arguments: { VARIABLE: { type: Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' } } }, '---', { opcode: 'objectvarfromarrays', blockType: Scratch.BlockType.COMMAND, text: 'set var [VARIABLE] to object from keys [KEYS] and values [VALUES]', arguments: { KEYS: { type: Scratch.ArgumentType.STRING, defaultValue: '["key"]' }, VALUES: { type: Scratch.ArgumentType.STRING, defaultValue: '["value"]' }, VARIABLE: { type: Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' } } }, { opcode: 'transposeobjectvar', blockType: Scratch.BlockType.COMMAND, text: 'transpose object var [VARIABLE]', arguments: { VARIABLE: { type: Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' } } }, ], menus: { VARIABLES_MENU: { acceptReporters: true, items: 'getvariables' } } }; } // CODE ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- getvariables() { return get_variables_for_menu(); } // variable category helper funcs run_in_place(func,args,util) { // run a reporter function against a variable and assign the new value set_var(args.VARIABLE, func({OBJECT: get_var(args.VARIABLE,util), ...args}), util); } run_against_var(func,args,util) { // run a reporter function against a variable and return the value return func({OBJECT: get_var(args.VARIABLE,util), ...args}); } isobjectvalid(args) { let object = null; try { object = JSON.parse(args.OBJECT); if (object.constructor === {}.constructor) { return true; } else { return false; } } catch { return false; } } toobject(args) { return stringify(cast_to_object(args.OBJECT)); } vartoobject(args,util) { set_var(args.VARIABLE, this.toobject(args), util); } getfromobject(args) { let object = cast_to_object(args.OBJECT); return stringify(object[args.KEY]); } getfromobjectvar(args,util) { return this.run_against_var(this.getfromobject,args,util); } getfromobjectwithplaceholder(args) { let object = cast_to_object(args.OBJECT); if (object[args.KEY] === undefined) { return args.PLACEHOLDER; } return stringify(object[args.KEY]); } getfromobjectvarwithplaceholder(args,util) { return this.run_against_var(this.getfromobjectwithplaceholder,args,util); } setinobject(args) { let object = cast_to_object(args.OBJECT); object[args.KEY] = args.VALUE; return stringify(object); } setinobjectvar(args,util) { this.run_in_place(this.setinobject,args,util); } deletefromobject(args) { let object = cast_to_object(args.OBJECT); delete object[args.KEY]; return stringify(object); } deletefromobjectvar(args,util) { this.run_in_place(this.deletefromobject,args,util); } objecthaskey(args) { let object = cast_to_object(args.OBJECT); return args.KEY in object; } objectvarhaskey(args,util) { return this.run_against_var(this.objecthaskey,args,util); } objectkeys(args) { let object = cast_to_object(args.OBJECT); return stringify(Object.keys(object)); } objectvarkeys(args,util) { return this.run_against_var(this.objectkeys,args,util); } objectvalues(args) { let object = cast_to_object(args.OBJECT); let keys = Object.keys(object); let values = []; for (let i = 0; i < keys.length; i++) { values.push(object[keys[i]]); } return stringify(values); } objectvarvalues(args,util) { return this.run_against_var(this.objectvalues,args,util); } objectlength(args) { let object = cast_to_object(args.OBJECT); return Object.keys(object).length; } objectvarlength(args,util) { return this.run_against_var(this.objectlength,args,util); } objectfromarrays(args) { let keys = cast_to_array(args.KEYS); let values = cast_to_array(args.VALUES); let length = Math.min(keys.length, values.length); let object = {}; for (let i = 0; i < length; i++) { object[stringify(keys[i])] = values[i]; } return stringify(object); } objectvarfromarrays(args,util) { set_var(args.VARIABLE, this.objectfromarrays(args), util); } transposeobject(args) { let object = cast_to_object(args.OBJECT); let new_object = {}; for (let key in object) { let value = stringify(object[key]); new_object[value] = key; } return stringify(new_object); } transposeobjectvar(args,util) { this.run_in_place(this.transposeobject,args,util); } } // load extension Scratch.extensions.register(new SoupUtilsObjects()); })(Scratch);