// Name: Soup Utils // Author: soup // Description: Math, logic, and variable utilities galore! // ID: souputils // Version 4.4 --- 32 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 Utils must be unsandboxed to run!'); } let prev_extensions = Array.from(vm.extensionManager._loadedExtensions.keys()); if (prev_extensions.includes('souputils')) { 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 util_icon = ""; const util_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.unshift(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 SoupUtils { getInfo() { return { id: 'souputils', name: 'Soup Utils', color1: '#f55442', color2: '#b83527', color3: '#edbbad', menuIconURI: util_menu_icon, blockIconURI: util_icon, blocks: [ // BLOCKS ----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- { opcode: 'lerp', blockType: Scratch.BlockType.REPORTER, text: 'lerp from [A] to [B] by [N]', arguments: { A: { type: Scratch.ArgumentType.NUMBER, defaultValue: '5' }, B: { type: Scratch.ArgumentType.NUMBER, defaultValue: '10' }, N: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0.4' } } }, { opcode: 'inverselerp', blockType: Scratch.BlockType.REPORTER, text: 'inverse lerp of [N] range [A] to [B]', arguments: { A: { type: Scratch.ArgumentType.NUMBER, defaultValue: '5' }, B: { type: Scratch.ArgumentType.NUMBER, defaultValue: '10' }, N: { type: Scratch.ArgumentType.NUMBER, defaultValue: '7' } } }, { opcode: 'doublelerp', blockType: Scratch.BlockType.REPORTER, text: 'lerp from [A] to [B] by [N]\'s position between [C] and [D]', arguments: { A: { type: Scratch.ArgumentType.NUMBER, defaultValue: '5' }, B: { type: Scratch.ArgumentType.NUMBER, defaultValue: '10' }, C: { type: Scratch.ArgumentType.NUMBER, defaultValue: '20' }, D: { type: Scratch.ArgumentType.NUMBER, defaultValue: '30' }, N: { type: Scratch.ArgumentType.NUMBER, defaultValue: '25' } } }, '---', { opcode: 'customround', blockType: Scratch.BlockType.REPORTER, text: '[TYPE] [A] to the nearest [B]', arguments: { A: { type: Scratch.ArgumentType.NUMBER, defaultValue: '3.46' }, B: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0.1' }, TYPE: { type: Scratch.ArgumentType.STRING, menu: 'ROUND_TYPES_MENU' } } }, { opcode: 'customroundreciprocal', blockType: Scratch.BlockType.REPORTER, text: '[TYPE] [A] to the nearest 1/[B]', arguments: { A: { type: Scratch.ArgumentType.NUMBER, defaultValue: '3.46' }, B: { type: Scratch.ArgumentType.NUMBER, defaultValue: '10' }, TYPE: { type: Scratch.ArgumentType.STRING, menu: 'ROUND_TYPES_MENU' } } }, '---', { opcode: 'extramath', blockType: Scratch.BlockType.REPORTER, text: '[A] [MATH] [B]', arguments: { A: { type: Scratch.ArgumentType.NUMBER, defaultValue: '12' }, B: { type: Scratch.ArgumentType.NUMBER, defaultValue: '5' }, MATH: { type: Scratch.ArgumentType.STRING, menu: 'EXTRA_MATH_MENU' } } }, { opcode: 'extramathoneinput', blockType: Scratch.BlockType.REPORTER, text: '[MATH] [A]', arguments: { A: { type: Scratch.ArgumentType.NUMBER, defaultValue: '6' }, MATH: { type: Scratch.ArgumentType.STRING, menu: 'EXTRA_MATH_ONE_INPUT_MENU' } } }, { opcode: 'mathconstants', blockType: Scratch.BlockType.REPORTER, text: '[CONSTANT]', disableMonitor: true, arguments: { CONSTANT: { type: Scratch.ArgumentType.STRING, menu: 'MATH_CONSTANTS' } } }, '---', { opcode: 'extralogic', blockType: Scratch.BlockType.BOOLEAN, text: '[A] [LOGIC] [B]', arguments: { A: { type: Scratch.ArgumentType.BOOLEAN }, B: { type: Scratch.ArgumentType.BOOLEAN }, LOGIC: { type: Scratch.ArgumentType.STRING, menu: 'EXTRA_LOGIC_MENU' } } }, { opcode: 'extracomparators', blockType: Scratch.BlockType.BOOLEAN, text: '[A] [OPERATOR] [B]', arguments: { A: { type: Scratch.ArgumentType.STRING, defaultValue: '4' }, B: { type: Scratch.ArgumentType.STRING, defaultValue: '4.6' }, OPERATOR: { type: Scratch.ArgumentType.STRING, menu: 'EXTRA_COMPARATORS_MENU' } } }, '---', { opcode: 'resetsprite', blockType: Scratch.BlockType.COMMAND, text: 'reset all sprite settings, go to [VECTOR] [COORDINATEUNIT], and [SHOWORHIDE]', arguments: { VECTOR: { type: Scratch.ArgumentType.STRING, defaultValue: '[0,0]' }, COORDINATEUNIT: { type: Scratch.ArgumentType.STRING, menu: 'COORDINATE_UNITS_MENU' }, SHOWORHIDE: { type: Scratch.ArgumentType.STRING, menu: 'SHOW_OR_HIDE_MENU' } } }, '---', { opcode: 'getvar', blockType: Scratch.BlockType.REPORTER, text: 'var [VARIABLE]', arguments: { VARIABLE: { type: Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' } } }, { opcode: 'setvar', blockType: Scratch.BlockType.COMMAND, text: 'set var [VARIABLE] to [VALUE]', arguments: { VARIABLE: { type: Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' }, VALUE: { type: Scratch.ArgumentType.STRING, defaultValue: '0' } } }, { opcode: 'modifyvar', blockType: Scratch.BlockType.COMMAND, text: '[VARIABLE] [OPERATOR] [VALUE]', arguments: { VARIABLE: { type: Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' }, VALUE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '1' }, OPERATOR: { type: Scratch.ArgumentType.STRING, menu: 'ASSIGNMENT_OPERATORS_MENU' } } }, '---', { opcode: 'jsalert', blockType: Scratch.BlockType.COMMAND, text: 'alert [MESSAGE]', arguments: { MESSAGE: { type: Scratch.ArgumentType.STRING, defaultValue: 'Hello World!' } } }, { opcode: 'jsconfirm', blockType: Scratch.BlockType.BOOLEAN, text: 'confirm [MESSAGE]', arguments: { MESSAGE: { type: Scratch.ArgumentType.STRING, defaultValue: 'Are you sure?' } } }, { opcode: 'jsprompt', blockType: Scratch.BlockType.REPORTER, text: 'prompt [MESSAGE] default [PLACEHOLDER]', arguments: { MESSAGE: { type: Scratch.ArgumentType.STRING, defaultValue: 'What is your name?' }, PLACEHOLDER: { type: Scratch.ArgumentType.STRING, defaultValue: '' } } }, { opcode: 'jspromptwithdefault', blockType: Scratch.BlockType.REPORTER, text: 'prompt [MESSAGE] default [PLACEHOLDER], or [DEFAULT] if they cancel', arguments: { MESSAGE: { type: Scratch.ArgumentType.STRING, defaultValue: 'What is your name?' }, PLACEHOLDER: { type: Scratch.ArgumentType.STRING, defaultValue: '' }, DEFAULT: { type: Scratch.ArgumentType.STRING, defaultValue: 'null' } } }, '---', { opcode: 'booleanconstants', blockType: Scratch.BlockType.BOOLEAN, text: '[BOOLEAN]', arguments: { BOOLEAN: { type: Scratch.ArgumentType.STRING, menu: 'BOOLEANS_MENU' } } }, { opcode: 'tonumber', blockType: Scratch.BlockType.REPORTER, text: 'number [NUMBER]', arguments: { NUMBER: { type: Scratch.ArgumentType.STRING, defaultValue: '' } } }, { opcode: 'valueif', blockType: Scratch.BlockType.REPORTER, text: 'if [BOOLEAN] then [VALUEIFTRUE] else [VALUEIFFALSE]', arguments: { BOOLEAN: { type: Scratch.ArgumentType.BOOLEAN }, VALUEIFTRUE: { type: Scratch.ArgumentType.STRING, defaultValue: 'true' }, VALUEIFFALSE: { type: Scratch.ArgumentType.STRING, defaultValue: 'false' } } }, '---', { opcode: 'randomchancepercent', blockType: Scratch.BlockType.BOOLEAN, text: '[CHANCE]% chance', arguments: { CHANCE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '50' } } }, { opcode: 'randomchanceratio', blockType: Scratch.BlockType.BOOLEAN, text: '[NUMERATOR] out of [DENOMINATOR] chance', arguments: { NUMERATOR: { type: Scratch.ArgumentType.NUMBER, defaultValue: '1' }, DENOMINATOR: { type: Scratch.ArgumentType.NUMBER, defaultValue: '3' } } }, { opcode: 'randomvalue', blockType: Scratch.BlockType.REPORTER, text: 'random', arguments: {} }, '---', { opcode: 'userfriendlynumber', blockType: Scratch.BlockType.REPORTER, text: 'user-friendly number [NUMBER] with format [FORMAT]', arguments: { NUMBER: { type: Scratch.ArgumentType.NUMBER, defaultValue: '1000000' }, FORMAT: { type: Scratch.ArgumentType.STRING, menu: 'USER_FRIENDLY_NUMBERS_MENU' } } }, '---', { opcode: 'bulkmathtwo', blockType: Scratch.BlockType.REPORTER, text: '[OPERATION] [NUMONE] [NUMTWO]', arguments: { OPERATION: { type: Scratch.ArgumentType.STRING, menu: 'BULK_MATH_OPERATIONS_MENU' }, NUMONE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '1' }, NUMTWO: { type: Scratch.ArgumentType.NUMBER, defaultValue: '2' } } }, { opcode: 'bulkmaththree', blockType: Scratch.BlockType.REPORTER, text: '[OPERATION] [NUMONE] [NUMTWO] [NUMTHREE]', arguments: { OPERATION: { type: Scratch.ArgumentType.STRING, menu: 'BULK_MATH_OPERATIONS_MENU' }, NUMONE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '1' }, NUMTWO: { type: Scratch.ArgumentType.NUMBER, defaultValue: '2' }, NUMTHREE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '3' } } }, { opcode: 'bulkmathfour', blockType: Scratch.BlockType.REPORTER, text: '[OPERATION] [NUMONE] [NUMTWO] [NUMTHREE] [NUMFOUR]', arguments: { OPERATION: { type: Scratch.ArgumentType.STRING, menu: 'BULK_MATH_OPERATIONS_MENU' }, NUMONE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '1' }, NUMTWO: { type: Scratch.ArgumentType.NUMBER, defaultValue: '2' }, NUMTHREE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '3' }, NUMFOUR: { type: Scratch.ArgumentType.NUMBER, defaultValue: '4' } } }, { opcode: 'bulkmathfive', blockType: Scratch.BlockType.REPORTER, text: '[OPERATION] [NUMONE] [NUMTWO] [NUMTHREE] [NUMFOUR] [NUMFIVE]', arguments: { OPERATION: { type: Scratch.ArgumentType.STRING, menu: 'BULK_MATH_OPERATIONS_MENU' }, NUMONE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '1' }, NUMTWO: { type: Scratch.ArgumentType.NUMBER, defaultValue: '2' }, NUMTHREE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '3' }, NUMFOUR: { type: Scratch.ArgumentType.NUMBER, defaultValue: '4' }, NUMFIVE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '5' } } }, { opcode: 'bulkmathsix', blockType: Scratch.BlockType.REPORTER, text: '[OPERATION] [NUMONE] [NUMTWO] [NUMTHREE] [NUMFOUR] [NUMFIVE] [NUMSIX]', arguments: { OPERATION: { type: Scratch.ArgumentType.STRING, menu: 'BULK_MATH_OPERATIONS_MENU' }, NUMONE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '1' }, NUMTWO: { type: Scratch.ArgumentType.NUMBER, defaultValue: '2' }, NUMTHREE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '3' }, NUMFOUR: { type: Scratch.ArgumentType.NUMBER, defaultValue: '4' }, NUMFIVE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '5' }, NUMSIX: { type: Scratch.ArgumentType.NUMBER, defaultValue: '6' } } }, { opcode: 'bulkmathseven', blockType: Scratch.BlockType.REPORTER, text: '[OPERATION] [NUMONE] [NUMTWO] [NUMTHREE] [NUMFOUR] [NUMFIVE] [NUMSIX] [NUMSEVEN]', arguments: { OPERATION: { type: Scratch.ArgumentType.STRING, menu: 'BULK_MATH_OPERATIONS_MENU' }, NUMONE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '1' }, NUMTWO: { type: Scratch.ArgumentType.NUMBER, defaultValue: '2' }, NUMTHREE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '3' }, NUMFOUR: { type: Scratch.ArgumentType.NUMBER, defaultValue: '4' }, NUMFIVE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '5' }, NUMSIX: { type: Scratch.ArgumentType.NUMBER, defaultValue: '6' }, NUMSEVEN: { type: Scratch.ArgumentType.NUMBER, defaultValue: '7' } } }, { opcode: 'bulkmatheight', blockType: Scratch.BlockType.REPORTER, text: '[OPERATION] [NUMONE] [NUMTWO] [NUMTHREE] [NUMFOUR] [NUMFIVE] [NUMSIX] [NUMSEVEN] [NUMEIGHT]', arguments: { OPERATION: { type: Scratch.ArgumentType.STRING, menu: 'BULK_MATH_OPERATIONS_MENU' }, NUMONE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '1' }, NUMTWO: { type: Scratch.ArgumentType.NUMBER, defaultValue: '2' }, NUMTHREE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '3' }, NUMFOUR: { type: Scratch.ArgumentType.NUMBER, defaultValue: '4' }, NUMFIVE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '5' }, NUMSIX: { type: Scratch.ArgumentType.NUMBER, defaultValue: '6' }, NUMSEVEN: { type: Scratch.ArgumentType.NUMBER, defaultValue: '7' }, NUMEIGHT: { type: Scratch.ArgumentType.NUMBER, defaultValue: '8' } } } ], menus: { EXTRA_MATH_MENU: { acceptReporters: true, items: ['//','^','-base log of','-th root of','abs diff','+','-','*','/','%','bitwise AND','bitwise XOR','bitwise OR','L-shift','R-shift (signed)','R-shift (unsigned)'] }, EXTRA_MATH_ONE_INPUT_MENU: { acceptReporters: true, items: ['-','abs of','sign of','√','∛','∜','10^','e^','log10 of','ln of','round','floor of','ceil of','negative abs of','bitwise NOT of','sin (trig radians) of','cos (trig radians) of','tan (trig radians) of','asin (trig radians) of','acos (trig radians) of','atan (trig radians) of','sin (Scratch degrees) of','cos (Scratch degrees) of','tan (Scratch degrees) of','asin (Scratch degrees) of','acos (Scratch degrees) of','atan (Scratch degrees) of'] }, MATH_CONSTANTS: { acceptReporters: true, items: ['π','τ','φ','√2','√3','γ','ln(10)','ln(2)','log10(e)','log2(e)','√½','∞','-∞'] }, ROUND_TYPES_MENU: { acceptReporters: true, items: ['round','floor','ceiling'] }, EXTRA_LOGIC_MENU: { acceptReporters: true, items: ['and','or','nand','nor','xor (≠)','xnor (=)','but not'] }, EXTRA_COMPARATORS_MENU: { acceptReporters: true, items: ['=','≠','<','>','≤','≥','≈','≉','===','=≠='] }, SHOW_OR_HIDE_MENU: { acceptReporters: true, items: ['show','hide'] }, COORDINATE_UNITS_MENU: { acceptReporters: true, items: ['Scratch coords','JS coords'] }, VARIABLES_MENU: { acceptReporters: true, items: 'getvariables' }, ASSIGNMENT_OPERATORS_MENU: { acceptReporters: true, items: ['+=','-=','*=','/=','//=','%='] }, BOOLEANS_MENU: { acceptReporters: true, items: ['true','false'] }, BULK_MATH_OPERATIONS_MENU: { acceptReporters: true, items: ['min','max','sum','product','average'] }, USER_FRIENDLY_NUMBERS_MENU: { acceptReporters: true, items: ['1,234,567.89 (truncated)','1,234,567.89012 (not truncated)','1.23 million','1.23m','1,234,567.890 (always 3 decimals)','1,234,567.890120 (always 6 decimals)','1.234×10⁶','1d 23h 45m 37s','01:23:45, 00:00:01','1:23:45, 0:01','01:23, 00:01', 'n\'th -illion number'] } } }; } // CODE ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- getvariables() { return get_variables_for_menu(); } // blocks lerp(args) { let a = Scratch.Cast.toNumber(args.A); let b = Scratch.Cast.toNumber(args.B); let n = Scratch.Cast.toNumber(args.N); return (n * (b - a)) + a; } inverselerp(args) { let a = Scratch.Cast.toNumber(args.A); let b = Scratch.Cast.toNumber(args.B); let n = Scratch.Cast.toNumber(args.N); return (n - a) / (b - a); } doublelerp(args) { let a = Scratch.Cast.toNumber(args.A); let b = Scratch.Cast.toNumber(args.B); let c = Scratch.Cast.toNumber(args.C); let d = Scratch.Cast.toNumber(args.D); let n = Scratch.Cast.toNumber(args.N); return (((n - c) / (d - c)) * (b - a)) + a; } customround(args) { let a = Scratch.Cast.toNumber(args.A); let b = Scratch.Cast.toNumber(args.B); if (args.TYPE == 'round') { return Math.round(a / b) * b; } else if (args.TYPE == 'floor') { return Math.floor(a / b) * b; } else if (args.TYPE == 'ceiling') { return Math.ceil(a / b) * b; } else { return 0; } } customroundreciprocal(args) { let a = Scratch.Cast.toNumber(args.A); let b = Scratch.Cast.toNumber(args.B); if (args.TYPE == 'round') { return Math.round(a * b) / b; } else if (args.TYPE == 'floor') { return Math.floor(a * b) / b; } else if (args.TYPE == 'ceiling') { return Math.ceil(a * b) / b; } else { return 0; } } extramath(args) { let a = Scratch.Cast.toNumber(args.A); let b = Scratch.Cast.toNumber(args.B); if (args.MATH == '//') { return Math.floor(a / b); } else if (args.MATH == '^') { return a ** b; } else if (args.MATH == '-base log of') { return Math.log(b) / Math.log(a); } else if (args.MATH == '-th root of') { return b ** (1/a); } else if (args.MATH == 'abs diff') { return Math.abs(a - b); } else if (args.MATH == '+') { return a + b; } else if (args.MATH == '-') { return a - b; } else if (args.MATH == '*') { return a * b; } else if (args.MATH == '/') { return a / b; } else if (args.MATH == '%') { return mod(a,b); } else if (args.MATH == 'bitwise AND') { return a & b; } else if (args.MATH == 'bitwise XOR') { return a ^ b; } else if (args.MATH == 'bitwise OR') { return a | b; } else if (args.MATH == 'L-shift') { return a << b; } else if (args.MATH == 'R-shift (signed)') { return a >> b; } else if (args.MATH == 'R-shift (unsigned)') { return a >>> b; } else { return 0; } } extramathoneinput(args) { let a = Scratch.Cast.toNumber(args.A); if (args.MATH == '-') { return 0 - a; } else if (args.MATH == 'abs of') { return Math.abs(a); } else if (args.MATH == 'sign of') { if (a > 0) { return 1; } else if (a < 0) { return -1; } else { return 0; } } else if (args.MATH == '10^') { return 10 ** a; } else if (args.MATH == 'e^') { return Math.exp(a); } else if (args.MATH == '√') { return Math.sqrt(a); } else if (args.MATH == '∛') { return Math.cbrt(a); } else if (args.MATH == '∜') { return a ** 0.25; } else if (args.MATH == 'log10 of') { return Math.log10(a); } else if (args.MATH == 'ln of') { return Math.log(a); } else if (args.MATH == 'sin (trig radians) of') { return Math.sin(a); } else if (args.MATH == 'cos (trig radians) of') { return Math.cos(a); } else if (args.MATH == 'tan (trig radians) of') { return Math.tan(a); } else if (args.MATH == 'asin (trig radians) of') { return Math.asin(a); } else if (args.MATH == 'acos (trig radians) of') { return Math.acos(a); } else if (args.MATH == 'atan (trig radians) of') { return Math.atan(a); } else if (args.MATH == 'sin (Scratch degrees) of') { return Math.sin((Math.PI / 180) * (90 - a)); } else if (args.MATH == 'cos (Scratch degrees) of') { return Math.cos((Math.PI / 180) * (90 - a)); } else if (args.MATH == 'tan (Scratch degrees) of') { return Math.tan((Math.PI / 180) * (90 - a)); } else if (args.MATH == 'asin (Scratch degrees) of') { return 90 - (Math.asin(a) * (180 / Math.PI)); } else if (args.MATH == 'acos (Scratch degrees) of') { return 90 - (Math.acos(a) * (180 / Math.PI)); } else if (args.MATH == 'atan (Scratch degrees) of') { return 90 - (Math.atan(a) * (180 / Math.PI)); } else if (args.MATH == 'round') { return Math.round(a); } else if (args.MATH == 'ceil of') { return Math.ceil(a); } else if (args.MATH == 'floor of') { return Math.floor(a); } else if (args.MATH == 'negative abs of') { return 0 - Math.abs(a); } else if (args.MATH == 'bitwise NOT of') { return ~a; } else { return 0; } } mathconstants(args) { if (args.CONSTANT == 'π') { return Math.PI; } else if (args.CONSTANT == 'τ') { return 2 * Math.PI; } else if (args.CONSTANT == 'φ') { return goldenRatio; } else if (args.CONSTANT == '√2') { return Math.SQRT2; } else if (args.CONSTANT == '√3') { return sqrt3; } else if (args.CONSTANT == 'γ') { return gammaConstant; } else if (args.CONSTANT == '∞') { return 'Infinity'; } else if (args.CONSTANT == '-∞') { return '-Infinity'; } else if (args.CONSTANT == '√½') { return Math.SQRT1_2; } else if (args.CONSTANT == 'ln(10)') { return Math.LN10; } else if (args.CONSTANT == 'ln(2)') { return Math.LN2; } else if (args.CONSTANT == 'log10(e)') { return Math.LOG10E; } else if (args.CONSTANT == 'log2(e)') { return Math.LOG2E; } else { return 0; } } extralogic(args) { let a = Scratch.Cast.toBoolean(args.A); let b = Scratch.Cast.toBoolean(args.B); if (args.LOGIC == 'and') { return a && b; } else if (args.LOGIC == 'or') { return a || b; } else if (args.LOGIC == 'nand') { return !(a && b); } else if (args.LOGIC == 'nor') { return !(a || b); } else if (args.LOGIC == 'xor (≠)') { return a != b; } else if (args.LOGIC == 'xnor (=)') { return a == b; } else if (args.LOGIC == 'but not') { return a && (!b); } else { return false; } } extracomparators(args) { let comparison = Scratch.Cast.compare(args.A,args.B); let a = Scratch.Cast.toNumber(args.A); let b = Scratch.Cast.toNumber(args.B); if (args.OPERATOR == '=') { return comparison == 0; } else if (args.OPERATOR == '≠') { return comparison != 0; } else if (args.OPERATOR == '<') { return comparison < 0; } else if (args.OPERATOR == '>') { return comparison > 0; } else if (args.OPERATOR == '≤') { return comparison <= 0; } else if (args.OPERATOR == '≥') { return comparison >= 0; } else if (args.OPERATOR == '≈') { return Math.abs(a - b) < 1; } else if (args.OPERATOR == '≉') { return Math.abs(a - b) >= 1; } else if (args.OPERATOR == '===') { return args.A === args.B; } else if (args.OPERATOR == '=≠=') { return args.A !== args.B; } else { return false; } } resetsprite(args,util) { util.target.setDirection(90); util.target.setRotationStyle('all around'); util.target.clearEffects(); util.target.setSize(100); let vector = cast_to_vector(args.VECTOR,2); vector = convert_coordinate_units(vector,args.COORDINATEUNIT,'Scratch coords'); util.target.setXY(vector[0],vector[1]); if (args.SHOWORHIDE == 'show') { util.target.setVisible(true); } else { util.target.setVisible(false); } } getvar(args,util) { return get_var(args.VARIABLE,util); } setvar(args,util) { set_var(args.VARIABLE,args.VALUE,util); } modifyvar(args,util) { let a = Scratch.Cast.toNumber(get_var(args.VARIABLE,util)); let b = Scratch.Cast.toNumber(args.VALUE); let newValue = 0; if (args.OPERATOR == '+=') { newValue = a + b; } else if (args.OPERATOR == '-=') { newValue = a - b; } else if (args.OPERATOR == '-=') { newValue = a - b; } else if (args.OPERATOR == '*=') { newValue = a * b; } else if (args.OPERATOR == '/=') { newValue = a / b; } else if (args.OPERATOR == '//=') { newValue = Math.floor(a / b); } else if (args.OPERATOR == '%=') { newValue = mod(a,b); } set_var(args.VARIABLE,newValue.toString(),util); } jsalert(args) { return new Promise(function(resolve, reject) { alert(args.MESSAGE); resolve(); }); } jsconfirm(args) { return new Promise(function(resolve, reject) { let result = confirm(args.MESSAGE); resolve(result); }); } jsprompt(args) { return new Promise(function(resolve, reject) { let result = prompt(args.MESSAGE,args.PLACEHOLDER); if (result === null) { resolve('null'); } else { resolve(result); } }); } jspromptwithdefault(args) { return new Promise(function(resolve, reject) { let result = prompt(args.MESSAGE,args.PLACEHOLDER); if (result === null) { resolve(args.DEFAULT); } else { resolve(result); } }); } booleanconstants(args) { let boolean = Scratch.Cast.toBoolean(args.BOOLEAN); return boolean; } tonumber(args) { let number = Scratch.Cast.toNumber(args.NUMBER); return number; } bulkmathtwo(args) { return bulk_math([Scratch.Cast.toNumber(args.NUMONE),Scratch.Cast.toNumber(args.NUMTWO)], args.OPERATION); } bulkmaththree(args) { return bulk_math([Scratch.Cast.toNumber(args.NUMONE),Scratch.Cast.toNumber(args.NUMTWO),Scratch.Cast.toNumber(args.NUMTHREE)], args.OPERATION); } bulkmathfour(args) { return bulk_math([Scratch.Cast.toNumber(args.NUMONE),Scratch.Cast.toNumber(args.NUMTWO),Scratch.Cast.toNumber(args.NUMTHREE),Scratch.Cast.toNumber(args.NUMFOUR)], args.OPERATION); } bulkmathfive(args) { return bulk_math([Scratch.Cast.toNumber(args.NUMONE),Scratch.Cast.toNumber(args.NUMTWO),Scratch.Cast.toNumber(args.NUMTHREE),Scratch.Cast.toNumber(args.NUMFOUR),Scratch.Cast.toNumber(args.NUMFIVE)], args.OPERATION); } bulkmathsix(args) { return bulk_math([Scratch.Cast.toNumber(args.NUMONE),Scratch.Cast.toNumber(args.NUMTWO),Scratch.Cast.toNumber(args.NUMTHREE),Scratch.Cast.toNumber(args.NUMFOUR),Scratch.Cast.toNumber(args.NUMFIVE),Scratch.Cast.toNumber(args.NUMSIX)], args.OPERATION); } bulkmathseven(args) { return bulk_math([Scratch.Cast.toNumber(args.NUMONE),Scratch.Cast.toNumber(args.NUMTWO),Scratch.Cast.toNumber(args.NUMTHREE),Scratch.Cast.toNumber(args.NUMFOUR),Scratch.Cast.toNumber(args.NUMFIVE),Scratch.Cast.toNumber(args.NUMSIX),Scratch.Cast.toNumber(args.NUMSEVEN)], args.OPERATION); } bulkmatheight(args) { return bulk_math([Scratch.Cast.toNumber(args.NUMONE),Scratch.Cast.toNumber(args.NUMTWO),Scratch.Cast.toNumber(args.NUMTHREE),Scratch.Cast.toNumber(args.NUMFOUR),Scratch.Cast.toNumber(args.NUMFIVE),Scratch.Cast.toNumber(args.NUMSIX),Scratch.Cast.toNumber(args.NUMSEVEN),Scratch.Cast.toNumber(args.NUMEIGHT)], args.OPERATION); } valueif(args) { let boolean = Scratch.Cast.toBoolean(args.BOOLEAN); if (boolean) { return args.VALUEIFTRUE; } else { return args.VALUEIFFALSE; } } randomchancepercent(args) { let chance = Scratch.Cast.toNumber(args.CHANCE) / 100; let rand = Math.random(); return rand < chance; } randomchanceratio(args) { let chance = Scratch.Cast.toNumber(args.NUMERATOR) / Scratch.Cast.toNumber(args.DENOMINATOR); let rand = Math.random(); return rand < chance; } randomvalue(args) { return Math.random(); } userfriendlynumber(args) { let num = Scratch.Cast.toNumber(args.NUMBER); let format = args.FORMAT; let neg_sign = ''; if (num < 0) { neg_sign = '-'; num = Math.abs(num); } if (num === Infinity) { return neg_sign + 'Infinity'; } num = Math.round(num*10000000)/10000000; // any more precision risks floating point errors let comma_formats = ['1,234,567.89 (truncated)','1,234,567.89012 (not truncated)','1,234,567.890 (always 3 decimals)','1,234,567.890120 (always 6 decimals)']; let whole_part_with_commas = null; if (comma_formats.includes(format) || (format == '1.23 million' && num < 1000000) || (format == '1.23m' && num < 1000)) { whole_part_with_commas = ''; let place_value = 1; let next_comma = 3; let digit = null; while (num >= place_value) { if (next_comma <= 0) { next_comma = 3; whole_part_with_commas = ',' + whole_part_with_commas; } digit = mod(Math.floor(num / place_value), 10); whole_part_with_commas = digit + whole_part_with_commas; place_value *= 10; next_comma -= 1; } if (whole_part_with_commas == '') { whole_part_with_commas = '0'; } whole_part_with_commas = neg_sign + whole_part_with_commas; } if (format == '1,234,567.89 (truncated)' || (format == '1.23 million' && num < 1000000) || (format == '1.23m' && num < 1000)) { let str = whole_part_with_commas; let decimal_part = Math.floor(Math.round(mod(num, 1) * 10000000) / 100000).toString(); if (decimal_part > 0) { str += '.'; if (decimal_part < 10) { str += '0'; } if (decimal_part.endsWith('0')) { decimal_part = decimal_part.slice(0,-1); } str += decimal_part; } return str; } else if (format == '1,234,567.89012 (not truncated)') { let str = whole_part_with_commas; let place_value = 0.1; let decimal_part = '.'; num = mod(num, 1); let digit = null; while (num > 0.00000001) { digit = mod(Math.floor(Math.round((num / place_value) * 10000000) / 10000000), 10); num -= digit * place_value; decimal_part += digit; place_value /= 10; } if (decimal_part == '.') { decimal_part = ''; } str += decimal_part; return str; } else if (format == '1.23 million' || format == '1.23m') { let illion_index = Math.floor(Math.log10(num) / 3) - 1; let illion = null; if (format == '1.23m') { illion = get_abbreviated_illion(illion_index); } else { illion = ' ' + get_illion(illion_index); } let number_part = num / (10 ** ((illion_index + 1) * 3)); number_part = Math.floor(number_part * 100) / 100; return neg_sign + number_part.toString() + illion; } else if (format == '1,234,567.890 (always 3 decimals)' || format == '1,234,567.890120 (always 6 decimals)') { let digits = 3; if (format == '1,234,567.890120 (always 6 decimals)') { digits = 6; } let str = whole_part_with_commas; let place_value = 0.1; let decimal_part = '.'; num = mod(num, 1); let digit = null; for (let i = 0; i < digits; i++) { digit = mod(Math.floor(Math.round((num / place_value) * 10000000) / 10000000), 10); decimal_part += digit; place_value /= 10; } if (decimal_part == '.') { decimal_part = ''; } str += decimal_part; return str; } else if (format == '1.234×10⁶') { if (num == 0) { return '0'; } let power = Math.floor(Math.log10(num)); let number_part = num / (10 ** power); number_part = Math.floor(number_part * 1000) / 1000; power = power.toString(); let superscript_numbers = {'0':'⁰','1':'¹','2':'²','3':'³','4':'⁴','5':'⁵','6':'⁶','7':'⁷','8':'⁸','9':'⁹','-':'⁻'}; let superscript_power = ''; for (let i = 0; i < power.length; i++) { superscript_power += superscript_numbers[power[i]]; } return neg_sign + number_part.toString() + '×10' + superscript_power; } else if (format == '1d 23h 45m 37s' || format == '01:23:45, 00:00:01' || format == '1:23:45, 0:01' || format == '01:23, 00:01') { let years = null; let days = null; if (format == '1d 23h 45m 37s') { years = Math.floor(num / (365 * 24 * 60 * 60)); days = mod(Math.floor(num / (24 * 60 * 60)), 365); } else { years = 0; days = Math.floor(num / (24 * 60 * 60)); } let hours = mod(Math.floor(num / (60 * 60)), 24); let minutes = mod(Math.floor(num / 60), 60); let seconds = mod(Math.floor(num), 60); let sections = []; if (format == '1d 23h 45m 37s') { if (years > 0) { sections.push(neg_sign + years + 'y'); } if (days > 0) { sections.push(neg_sign + days + 'd'); } if (hours > 0) { sections.push(neg_sign + hours + 'h'); } if (minutes > 0) { sections.push(neg_sign + minutes + 'm'); } if (seconds > 0) { sections.push(neg_sign + seconds + 's'); } } else if (format == '01:23:45, 00:00:01') { if (days > 0) { sections.push(days.toString()); } sections.push(hours.toString().padStart(2,'0')); sections.push(minutes.toString().padStart(2,'0')); sections.push(seconds.toString().padStart(2,'0')); } else if (format == '1:23:45, 0:01') { if (days > 0) { sections.push(days.toString()); } if (hours > 0 || sections.length > 0) { if (sections.length > 0) { sections.push(hours.toString().padStart(2,'0')); } else { sections.push(hours.toString()); } /* leading zero only if others come before */ } if (sections.length > 0) { sections.push(minutes.toString().padStart(2,'0')); } else { sections.push(minutes.toString()); } // leading zero only if others came before sections.push(seconds.toString().padStart(2,'0')); } else if (format == '01:23, 00:01') { if (days > 0) { sections.push(days.toString()); } if (hours > 0 || sections.length > 0) { if (sections.length > 0) { sections.push(hours.toString().padStart(2,'0')); } else { sections.push(hours.toString()); } /* leading zero only if others come before */ } sections.push(minutes.toString().padStart(2,'0')); sections.push(seconds.toString().padStart(2,'0')); } if (format == '1d 23h 45m 37s') { return sections.join(' '); } return neg_sign + sections.join(':'); } else if (format == 'n\'th -illion number') { return neg_sign + get_illion(num); } return '0'; } } // load extension Scratch.extensions.register(new SoupUtils()); })(Scratch);