// Name: Soup Vectors // Author: soup // Description: Vector math tools so you'll never need to do an arctangent ever again. // ID: souputilsvectors // Version 5.1 --- 60 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('souputilsvectors')) { 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 vector_icon = ""; const vector_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 SoupUtilsVectors { getInfo() { return { id: 'souputilsvectors', name: 'Soup Vectors', color1: '#f55442', color2: '#b83527', color3: '#edbbad', menuIconURI: vector_menu_icon, blockIconURI: vector_icon, // BLOCKS ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- blocks: [ { opcode: 'tovector', blockType: Scratch.BlockType.REPORTER, text: 'vector [VECTOR]', arguments: { VECTOR: { type: Scratch.ArgumentType.STRING, defaultValue: '["0","0"]' } } }, { opcode: 'twovectorfromcoords', blockType: Scratch.BlockType.REPORTER, text: 'vector [X], [Y]', arguments: { X: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, Y: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' } } }, { opcode: 'threevectorfromcoords', blockType: Scratch.BlockType.REPORTER, text: 'vector [X], [Y], [Z]', arguments: { X: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, Y: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, Z: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' } } }, '---', { opcode: 'twovectorfromangle', blockType: Scratch.BlockType.REPORTER, text: 'unit vector with angle [ANGLE] [UNIT]', arguments: { UNIT: { type: Scratch.ArgumentType.STRING, menu: 'ANGLE_UNITS_MENU' }, ANGLE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '90' } } }, { opcode: 'twovectorfromangleandmagnitude', blockType: Scratch.BlockType.REPORTER, text: 'vector with angle [ANGLE] [UNIT] and magnitude [MAGNITUDE]', arguments: { UNIT: { type: Scratch.ArgumentType.STRING, menu: 'ANGLE_UNITS_MENU' }, ANGLE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '90' }, MAGNITUDE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '1' } } }, { opcode: 'twovectorfromslope', blockType: Scratch.BlockType.REPORTER, text: 'unit vector with slope [SLOPE]', arguments: { SLOPE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '1' } } }, { opcode: 'twovectorfromslopeandmagnitude', blockType: Scratch.BlockType.REPORTER, text: 'vector with slope [SLOPE] and magnitude [MAGNITUDE]', arguments: { SLOPE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '1' }, MAGNITUDE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '1' } } }, '---', { opcode: 'xofvector', blockType: Scratch.BlockType.REPORTER, text: 'x of [VECTOR]', arguments: { VECTOR: { type: Scratch.ArgumentType.STRING, defaultValue: '[0,0]' } } }, { opcode: 'yofvector', blockType: Scratch.BlockType.REPORTER, text: 'y of [VECTOR]', arguments: { VECTOR: { type: Scratch.ArgumentType.STRING, defaultValue: '[0,0]' } } }, { opcode: 'zofvector', blockType: Scratch.BlockType.REPORTER, text: 'z of [VECTOR]', hideFromPalette: true, arguments: { VECTOR: { type: Scratch.ArgumentType.STRING, defaultValue: '[0,0,0]' } } }, { opcode: 'coordfromvector', blockType: Scratch.BlockType.REPORTER, text: '[COORD] of [VECTOR]', arguments: { COORD: { type: Scratch.ArgumentType.STRING, menu: 'COORDS_MENU', defaultValue: 'z' }, VECTOR: { type: Scratch.ArgumentType.STRING, defaultValue: '[0,0,0]' } } }, { opcode: 'dirofvector', blockType: Scratch.BlockType.REPORTER, text: 'direction in [UNIT] of [VECTOR]', arguments: { VECTOR: { type: Scratch.ArgumentType.STRING, defaultValue: '[1,0]' }, UNIT: { type: Scratch.ArgumentType.STRING, menu: 'ANGLE_UNITS_MENU' } } }, { opcode: 'magnitudeofvector', blockType: Scratch.BlockType.REPORTER, text: 'magnitude of [VECTOR]', arguments: { VECTOR: { type: Scratch.ArgumentType.STRING, defaultValue: '[1,0]' } } }, { opcode: 'slopeofvector', blockType: Scratch.BlockType.REPORTER, text: 'slope of [VECTOR]', arguments: { VECTOR: { type: Scratch.ArgumentType.STRING, defaultValue: '[1,2]' } } }, '---', { opcode: 'setcoordofvector', blockType: Scratch.BlockType.REPORTER, text: 'set [COORD] of vector [VECTOR] to [VALUE]', arguments: { COORD: { type: Scratch.ArgumentType.STRING, menu: 'COORDS_MENU' }, VALUE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '1' }, VECTOR: { type: Scratch.ArgumentType.STRING, defaultValue: '[0,0]' } } }, { opcode: 'setdirofvector', blockType: Scratch.BlockType.REPORTER, text: 'set direction of vector [VECTOR] to [ANGLE] [UNIT]', arguments: { ANGLE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '90' }, UNIT: { type: Scratch.ArgumentType.STRING, menu: 'ANGLE_UNITS_MENU' }, VECTOR: { type: Scratch.ArgumentType.STRING, defaultValue: '[1,1]' } } }, { opcode: 'setmagnitudeofvector', blockType: Scratch.BlockType.REPORTER, text: 'set magnitude of vector [VECTOR] to [MAGNITUDE]', arguments: { MAGNITUDE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '1' }, VECTOR: { type: Scratch.ArgumentType.STRING, defaultValue: '[1,1]' } } }, { opcode: 'setslopeofvector', blockType: Scratch.BlockType.REPORTER, text: 'set slope of vector [VECTOR] to [SLOPE]', arguments: { SLOPE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '1' }, VECTOR: { type: Scratch.ArgumentType.STRING, defaultValue: '[1,0]' } } }, '---', { opcode: 'stagesizevector', blockType: Scratch.BlockType.REPORTER, text: 'stage [MODE]', disableMonitor: true, arguments: { MODE: { type: Scratch.ArgumentType.STRING, menu: 'STAGE_SIZE_MODES_MENU' } } }, { opcode: 'currentposvector', blockType: Scratch.BlockType.REPORTER, text: 'position on stage as vector [UNIT]', filter: [Scratch.TargetType.SPRITE], disableMonitor: true, arguments: { UNIT: { type: Scratch.ArgumentType.STRING, menu: 'COORDINATE_UNITS_MENU' } } }, { opcode: 'currentdirvector', blockType: Scratch.BlockType.REPORTER, text: 'direction as unit vector', filter: [Scratch.TargetType.SPRITE], disableMonitor: true, arguments: {} }, { opcode: 'gotovector', blockType: Scratch.BlockType.COMMAND, text: 'go to vector [VECTOR] [UNIT]', filter: [Scratch.TargetType.SPRITE], arguments: { UNIT: { type: Scratch.ArgumentType.STRING, menu: 'COORDINATE_UNITS_MENU' }, VECTOR: { type: Scratch.ArgumentType.STRING, defaultValue: '[0,0]' } } }, { opcode: 'setdirtovector', blockType: Scratch.BlockType.COMMAND, text: 'point in direction of vector [VECTOR]', filter: [Scratch.TargetType.SPRITE], arguments: { VECTOR: { type: Scratch.ArgumentType.STRING, defaultValue: '[1,0]' } } }, '---', { opcode: 'vectoroperations', blockType: Scratch.BlockType.REPORTER, text: '[OPERATION] vectors [VECTORA] and [VECTORB]', arguments: { VECTORA: { type: Scratch.ArgumentType.STRING, defaultValue: '[1,0]' }, VECTORB: { type: Scratch.ArgumentType.STRING, defaultValue: '[0,1]' }, OPERATION: { type: Scratch.ArgumentType.STRING, menu: 'VECTOR_OPERATIONS_MENU' } } }, '---', { opcode: 'rotatevector', blockType: Scratch.BlockType.REPORTER, text: 'rotate vector [VECTOR] by [ANGLE] [UNIT]', arguments: { VECTOR: { type: Scratch.ArgumentType.STRING, defaultValue: '[1,0]' }, UNIT: { type: Scratch.ArgumentType.STRING, menu: 'ANGLE_UNITS_MENU' }, ANGLE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '90' } } }, { opcode: 'scalemagnitudeofvector', blockType: Scratch.BlockType.REPORTER, text: 'scale vector [VECTOR] by [FACTOR]', arguments: { VECTOR: { type: Scratch.ArgumentType.STRING, defaultValue: '[1,0]' }, FACTOR: { type: Scratch.ArgumentType.NUMBER, defaultValue: '2' } } }, { opcode: 'normalizevector', blockType: Scratch.BlockType.REPORTER, text: 'normalize vector [VECTOR]', arguments: { VECTOR: { type: Scratch.ArgumentType.STRING, defaultValue: '[1,0]' } } }, '---', { opcode: 'convertangleunits', blockType: Scratch.BlockType.REPORTER, text: 'convert [ANGLE] from [FROMUNIT] to [TOUNIT]', arguments: { ANGLE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '60' }, FROMUNIT: { type: Scratch.ArgumentType.STRING, menu: 'ANGLE_UNITS_MENU' }, TOUNIT: { type: Scratch.ArgumentType.STRING, menu: 'ANGLE_UNITS_MENU', defaultValue: 'trig radians' } } }, { opcode: 'convertcoordinateunits', blockType: Scratch.BlockType.REPORTER, text: 'convert [VECTOR] from [FROMUNIT] to [TOUNIT]', arguments: { VECTOR: { type: Scratch.ArgumentType.STRING, defaultValue: '[0,0]' }, FROMUNIT: { type: Scratch.ArgumentType.STRING, menu: 'COORDINATE_UNITS_MENU' }, TOUNIT: { type: Scratch.ArgumentType.STRING, menu: 'COORDINATE_UNITS_MENU', defaultValue: 'JS coords' } } }, { blockType: Scratch.BlockType.LABEL, text: "Soup Vectors - Variables" }, { opcode: 'vartovector', blockType: Scratch.BlockType.COMMAND, text: 'set [VARIABLE] to vector [VECTOR]', arguments: { VECTOR: { type: Scratch.ArgumentType.STRING, defaultValue: '["0","0"]' }, VARIABLE: { type:Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' } } }, { opcode: 'twovectorvarfromcoords', blockType: Scratch.BlockType.COMMAND, text: 'set [VARIABLE] to vector [X], [Y]', arguments: { X: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, Y: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, VARIABLE: { type:Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' } } }, { opcode: 'threevectorvarfromcoords', blockType: Scratch.BlockType.COMMAND, text: 'set [VARIABLE] to vector [X], [Y], [Z]', arguments: { X: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, Y: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, Z: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, VARIABLE: { type:Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' } } }, '---', { opcode: 'twovectorvarfromangle', blockType: Scratch.BlockType.COMMAND, text: 'set [VARIABLE] to unit vector with angle [ANGLE] [UNIT]', arguments: { UNIT: { type: Scratch.ArgumentType.STRING, menu: 'ANGLE_UNITS_MENU' }, ANGLE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '90' }, VARIABLE: { type:Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' } } }, { opcode: 'twovectorvarfromangleandmagnitude', blockType: Scratch.BlockType.COMMAND, text: 'set [VARIABLE] to vector with angle [ANGLE] [UNIT] and magnitude [MAGNITUDE]', arguments: { UNIT: { type: Scratch.ArgumentType.STRING, menu: 'ANGLE_UNITS_MENU' }, ANGLE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '90' }, MAGNITUDE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '1' }, VARIABLE: { type:Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' } } }, { opcode: 'twovectorvarfromslope', blockType: Scratch.BlockType.COMMAND, text: 'set [VARIABLE] to unit vector with slope [SLOPE]', arguments: { SLOPE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '1' }, VARIABLE: { type:Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' } } }, { opcode: 'twovectorvarfromslopeandmagnitude', blockType: Scratch.BlockType.COMMAND, text: 'set [VARIABLE] to vector with slope [SLOPE] and magnitude [MAGNITUDE]', arguments: { SLOPE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '1' }, MAGNITUDE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '1' }, VARIABLE: { type:Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' } } }, '---', { opcode: 'xofvectorvar', blockType: Scratch.BlockType.REPORTER, text: 'x of var [VARIABLE]', arguments: { VARIABLE: { type: Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' } } }, { opcode: 'yofvectorvar', blockType: Scratch.BlockType.REPORTER, text: 'y of var [VARIABLE]', arguments: { VARIABLE: { type: Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' } } }, { opcode: 'zofvectorvar', blockType: Scratch.BlockType.REPORTER, text: 'z of var [VARIABLE]', hideFromPalette: true, arguments: { VARIABLE: { type: Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' } } }, { opcode: 'coordfromvectorvar', blockType: Scratch.BlockType.REPORTER, text: '[COORD] of var [VARIABLE]', arguments: { COORD: { type: Scratch.ArgumentType.STRING, menu: 'COORDS_MENU', defaultValue: 'z' }, VARIABLE: { type: Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' } } }, { opcode: 'dirofvectorvar', blockType: Scratch.BlockType.REPORTER, text: 'direction in [UNIT] of var [VARIABLE]', arguments: { VARIABLE: { type: Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' }, UNIT: { type: Scratch.ArgumentType.STRING, menu: 'ANGLE_UNITS_MENU' } } }, { opcode: 'magnitudeofvectorvar', blockType: Scratch.BlockType.REPORTER, text: 'magnitude of var [VARIABLE]', arguments: { VARIABLE: { type: Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' } } }, { opcode: 'slopeofvectorvar', blockType: Scratch.BlockType.REPORTER, text: 'slope of var [VARIABLE]', arguments: { VARIABLE: { type: Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' } } }, '---', { opcode: 'setcoordofvectorvar', blockType: Scratch.BlockType.COMMAND, text: 'set [COORD] of vector var [VARIABLE] to [VALUE]', arguments: { COORD: { type: Scratch.ArgumentType.STRING, menu: 'COORDS_MENU' }, VALUE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '1' }, VARIABLE: { type: Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' } } }, { opcode: 'setdirofvectorvar', blockType: Scratch.BlockType.COMMAND, text: 'set direction of vector var [VARIABLE] to [ANGLE] [UNIT]', arguments: { ANGLE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '90' }, UNIT: { type: Scratch.ArgumentType.STRING, menu: 'ANGLE_UNITS_MENU' }, VARIABLE: { type: Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' } } }, { opcode: 'setmagnitudeofvectorvar', blockType: Scratch.BlockType.COMMAND, text: 'set magnitude of vector var [VARIABLE] to [VALUE]', arguments: { VALUE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '1' }, VARIABLE: { type: Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' } } }, { opcode: 'setslopeofvectorvar', blockType: Scratch.BlockType.REPORTER, text: 'set slope of vector var [VARIABLE] to [SLOPE]', arguments: { SLOPE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '1' }, VARIABLE: { type: Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' } } }, '---', { opcode: 'vectorvaroperations', blockType: Scratch.BlockType.COMMAND, text: '[OPERATION] var [VARIABLE] and vector [VECTOR]', arguments: { VARIABLE: { type: Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' }, VECTOR: { type: Scratch.ArgumentType.STRING, defaultValue: '[0,1]' }, OPERATION: { type: Scratch.ArgumentType.STRING, menu: 'VECTOR_OPERATIONS_MENU' } } }, { opcode: 'twovectorvaroperations', blockType: Scratch.BlockType.COMMAND, text: '[OPERATION] vars [VARIABLEA] and [VARIABLEB]', arguments: { VARIABLEA: { type: Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' }, VARIABLEB: { type: Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' }, OPERATION: { type: Scratch.ArgumentType.STRING, menu: 'VECTOR_OPERATIONS_MENU' } } }, '---', { opcode: 'rotatevectorvar', blockType: Scratch.BlockType.COMMAND, text: 'rotate vector var [VARIABLE] by [ANGLE] [UNIT]', arguments: { VARIABLE: { type: Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' }, UNIT: { type: Scratch.ArgumentType.STRING, menu: 'ANGLE_UNITS_MENU' }, ANGLE: { type: Scratch.ArgumentType.NUMBER, defaultValue: '45' } } }, { opcode: 'scalemagnitudeofvectorvar', blockType: Scratch.BlockType.COMMAND, text: 'scale vector var [VARIABLE] by [FACTOR]', arguments: { VARIABLE: { type: Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' }, FACTOR: { type: Scratch.ArgumentType.NUMBER, defaultValue: '2' } } }, { opcode: 'normalizevectorvar', blockType: Scratch.BlockType.COMMAND, text: 'normalize vector var [VARIABLE]', arguments: { VARIABLE: { type: Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' } } }, { blockType: Scratch.BlockType.LABEL, text: "Soup Vectors - Higher Dimensions" }, { opcode: 'fourvectorfromcoords', blockType: Scratch.BlockType.REPORTER, text: 'vector [X], [Y], [Z], [W]', arguments: { X: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, Y: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, Z: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, W: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' } } }, { opcode: 'fivevectorfromcoords', blockType: Scratch.BlockType.REPORTER, text: 'vector [X], [Y], [Z], [W], [V]', arguments: { X: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, Y: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, Z: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, W: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, V: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' } } }, { opcode: 'sixvectorfromcoords', blockType: Scratch.BlockType.REPORTER, text: 'vector [X], [Y], [Z], [W], [V], [U]', arguments: { X: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, Y: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, Z: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, W: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, V: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, U: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' } } }, { opcode: 'sevenvectorfromcoords', blockType: Scratch.BlockType.REPORTER, text: 'vector [X], [Y], [Z], [W], [V], [U], [T]', arguments: { X: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, Y: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, Z: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, W: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, V: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, U: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, T: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' } } }, { opcode: 'eightvectorfromcoords', blockType: Scratch.BlockType.REPORTER, text: 'vector [X], [Y], [Z], [W], [V], [U], [T], [S]', arguments: { X: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, Y: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, Z: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, W: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, V: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, U: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, T: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, S: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' } } }, '---', { opcode: 'fourvectorvarfromcoords', blockType: Scratch.BlockType.COMMAND, text: 'set [VARIABLE] to vector [X], [Y], [Z], [W]', arguments: { X: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, Y: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, Z: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, W: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, VARIABLE: { type:Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' } } }, { opcode: 'fivevectorvarfromcoords', blockType: Scratch.BlockType.COMMAND, text: 'set [VARIABLE] to vector [X], [Y], [Z], [W], [V]', arguments: { X: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, Y: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, Z: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, W: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, V: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, VARIABLE: { type:Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' } } }, { opcode: 'sixvectorvarfromcoords', blockType: Scratch.BlockType.COMMAND, text: 'set [VARIABLE] to vector [X], [Y], [Z], [W], [V], [U]', arguments: { X: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, Y: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, Z: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, W: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, V: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, U: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, VARIABLE: { type:Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' } } }, { opcode: 'sevenvectorvarfromcoords', blockType: Scratch.BlockType.COMMAND, text: 'set [VARIABLE] to vector [X], [Y], [Z], [W], [V], [U], [T]', arguments: { X: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, Y: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, Z: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, W: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, V: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, U: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, T: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, VARIABLE: { type:Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' } } }, { opcode: 'eightvectorvarfromcoords', blockType: Scratch.BlockType.COMMAND, text: 'set [VARIABLE] to vector [X], [Y], [Z], [W], [V], [U], [T], [S]', arguments: { X: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, Y: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, Z: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, W: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, V: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, U: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, T: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, S: { type: Scratch.ArgumentType.NUMBER, defaultValue: '0' }, VARIABLE: { type:Scratch.ArgumentType.STRING, menu: 'VARIABLES_MENU' } } } ], menus: { ANGLE_UNITS_MENU: { acceptReporters: true, items: ['Scratch degrees','Scratch radians','Scratch full circle','trig degrees','trig radians','trig full circle'] }, COORDINATE_UNITS_MENU: { acceptReporters: true, items: ['Scratch coords','JS coords'] }, VECTOR_OPERATIONS_MENU: { acceptReporters: true, items: ['add','subtract','dot product of','cross product of','multiply','divide','add Scratch rotation of','subtract Scratch rotation of','add trig rotation of','subtract trig rotation of','add magnitude of','subtract magnitude of'] }, VARIABLES_MENU: { acceptReporters: true, items: 'getvariables' }, STAGE_SIZE_MODES_MENU: { acceptReporters: true, items: ['size','width','height'] }, COORDS_MENU: { acceptReporters: true, items: ['x', 'y', 'z', 'w', 'v', 'u', 't', 's'] } } }; } // 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({VECTOR: 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({VECTOR: get_var(args.VARIABLE,util), ...args}); } tovector(args) { return stringify(cast_to_vector(args.VECTOR)); } vartovector(args,util) { set_var(args.VARIABLE, this.tovector(args), util); } twovectorfromcoords(args) { let vector = []; vector.push(Scratch.Cast.toNumber(args.X)); vector.push(Scratch.Cast.toNumber(args.Y)); return stringify(vector); } twovectorvarfromcoords(args,util) { set_var(args.VARIABLE, this.twovectorfromcoords(args), util); } threevectorfromcoords(args) { let vector = []; vector.push(Scratch.Cast.toNumber(args.X)); vector.push(Scratch.Cast.toNumber(args.Y)); vector.push(Scratch.Cast.toNumber(args.Z)); return stringify(vector); } threevectorvarfromcoords(args,util) { set_var(args.VARIABLE, this.threevectorfromcoords(args), util); } twovectorfromangle(args) { let angle = Scratch.Cast.toNumber(args.ANGLE); angle = convert_angle_units(angle,args.UNIT,'trig radians'); let vector = [Math.cos(angle),Math.sin(angle)]; return stringify(vector); } twovectorvarfromangle(args,util) { set_var(args.VARIABLE, this.twovectorfromangle(args), util); } twovectorfromangleandmagnitude(args) { let angle = Scratch.Cast.toNumber(args.ANGLE); let magnitude = Scratch.Cast.toNumber(args.MAGNITUDE); angle = convert_angle_units(angle,args.UNIT,'trig radians'); let vector = [magnitude*Math.cos(angle),magnitude*Math.sin(angle)]; return stringify(vector); } twovectorvarfromangleandmagnitude(args,util) { set_var(args.VARIABLE, this.twovectorfromangleandmagnitude(args), util); } twovectorfromslope(args) { let slope = Scratch.Cast.toNumber(args.SLOPE); let normalization_factor = Math.sqrt(slope*slope+1); let vector = [1/normalization_factor,slope/normalization_factor]; return stringify(vector); } twovectorvarfromslope(args,util) { set_var(args.VARIABLE, this.twovectorfromslope(args), util); } twovectorfromslopeandmagnitude(args) { let slope = Scratch.Cast.toNumber(args.SLOPE); let magnitude = Scratch.Cast.toNumber(args.MAGNITUDE); let normalization_factor = Math.sqrt(slope*slope+1); let vector = [magnitude/normalization_factor,(magnitude*slope)/normalization_factor]; return stringify(vector); } twovectorvarfromslopeandmagnitude(args,util) { set_var(args.VARIABLE, this.twovectorfromslopeandmagnitude(args), util); } xofvector(args) { let vector = cast_to_vector(args.VECTOR,1); return vector[0]; } xofvectorvar(args,util) { return this.run_against_var(this.xofvector,args,util); } yofvector(args) { let vector = cast_to_vector(args.VECTOR,2); return vector[1]; } yofvectorvar(args,util) { return this.run_against_var(this.yofvector,args,util); } zofvector(args) { let vector = cast_to_vector(args.VECTOR,3); return vector[2]; } zofvectorvar(args,util) { return this.run_against_var(this.zofvector,args,util); } coordfromvector(args) { let index = Scratch.Cast.toNumber(args.COORD); if (args.COORD == 'x') { index = 1; } else if (args.COORD == 'y') { index = 2; } else if (args.COORD == 'z') { index = 3; } else if (args.COORD == 'w') { index = 4; } else if (args.COORD == 'v') { index = 5; } else if (args.COORD == 'u') { index = 6; } else if (args.COORD == 't') { index = 7; } else if (args.COORD == 's') { index = 8; } index = Math.round(index); if (index < 1) { return 0; } let vector = cast_to_vector(args.VECTOR); if (vector.length < index) { return 0; } return vector[index-1]; } coordfromvectorvar(args,util) { return this.run_against_var(this.coordfromvector,args,util); } setcoordofvector(args) { let vector = cast_to_vector(args.VECTOR); let value = Scratch.Cast.toNumber(args.VALUE); let index = Scratch.Cast.toNumber(args.COORD); if (args.COORD == 'x') { index = 1; } else if (args.COORD == 'y') { index = 2; } else if (args.COORD == 'z') { index = 3; } else if (args.COORD == 'w') { index = 4; } else if (args.COORD == 'v') { index = 5; } else if (args.COORD == 'u') { index = 6; } else if (args.COORD == 't') { index = 7; } else if (args.COORD == 's') { index = 8; } index = Math.round(index); if (index < 1) { return stringify(vector); } if (vector.length < index) { return stringify(vector); } vector[index-1] = value; return stringify(vector); } setcoordofvectorvar(args,util) { this.run_in_place(this.setcoordofvector,args,util); } setdirofvector(args) { let vector = cast_to_vector(args.VECTOR,2); let angle = Scratch.Cast.toNumber(args.ANGLE); angle = convert_angle_units(angle,args.UNIT,'trig radians'); let magnitude = Math.sqrt((vector[0] * vector[0]) + (vector[1] * vector[1])); let new_vector = [magnitude*Math.cos(angle),magnitude*Math.sin(angle)]; return stringify(new_vector); } setdirofvectorvar(args,util) { this.run_in_place(this.setdirofvector,args,util); } setmagnitudeofvector(args) { let vector = cast_to_vector(args.VECTOR); let new_magnitude = Scratch.Cast.toNumber(args.MAGNITUDE); let magnitude = 0; for (let i = 0; i < vector.length; i++) { magnitude += vector[i] * vector[i]; } magnitude = Math.sqrt(magnitude); if (magnitude == 0) { if (vector.length > 0) { vector[0] = new_magnitude; for (let i = 1; i < vector.length; i++) { vector[i] = 0; } } } else { for (let i = 0; i < vector.length; i++) { vector[i] /= magnitude; vector[i] *= new_magnitude; } } return stringify(vector); } setmagnitudeofvectorvar(args,util) { this.run_in_place(this.setmagnitudeofvector,args,util); } setslopeofvector(args) { let vector = cast_to_vector(args.VECTOR,2); let slope = Scratch.Cast.toNumber(args.SLOPE); let magnitude = Math.sqrt((vector[0] * vector[0]) + (vector[1] * vector[1])); let normalization_factor = Math.sqrt(slope*slope+1); let new_vector = [magnitude/normalization_factor,(magnitude*slope)/normalization_factor]; return stringify(new_vector); } setslopeofvectorvar(args,util) { this.run_in_place(this.setslopeofvector,args,util); } currentposvector(args,util) { let vector = [util.target.x,util.target.y]; return stringify(convert_coordinate_units(vector,'Scratch coords',args.UNIT)); } currentdirvector(args,util) { let angle = convert_angle_units(util.target.direction,'Scratch degrees','trig radians'); let vector = [Math.cos(angle),Math.sin(angle)]; return stringify(vector); } gotovector(args,util) { let vector = cast_to_vector(args.VECTOR,2); vector = convert_coordinate_units(vector,args.UNIT,'Scratch coords'); util.target.setXY(vector[0],vector[1]); } setdirtovector(args,util) { let vector = cast_to_vector(args.VECTOR,2); let angle = Math.atan2(vector[1],vector[0]); angle = convert_angle_units(angle,'trig radians','Scratch degrees'); util.target.setDirection(angle); } vectoroperations(args) { let vector1 = cast_to_vector(args.VECTORA); let vector2 = cast_to_vector(args.VECTORB,vector1.length); let vector = []; let operation = args.OPERATION; if (operation == 'add') { for (let i = 0; i < vector1.length; i++) { vector.push(vector1[i] + vector2[i]); } } else if (operation == 'subtract') { for (let i = 0; i < vector1.length; i++) { vector.push(vector1[i] - vector2[i]); } } else if (operation == 'multiply') { for (let i = 0; i < vector1.length; i++) { vector.push(vector1[i] * vector2[i]); } } else if (operation == 'divide') { for (let i = 0; i < vector1.length; i++) { vector.push(vector1[i] / vector2[i]); } } else if (operation == 'dot product of') { let sum = 0; for (let i = 0; i < vector1.length; i++) { sum += vector1[i] * vector2[i]; } return sum; } else if (operation == 'cross product of') { vector1 = cast_to_vector(args.VECTORA, 3); vector2 = cast_to_vector(args.VECTORB, 3); vector.push( vector1[1] * vector2[2] - vector1[2] * vector2[1], vector1[2] * vector2[0] - vector1[0] * vector2[2], vector1[0] * vector2[1] - vector1[1] * vector2[0] ); } else if (operation == 'add Scratch rotation of' || operation == 'subtract Scratch rotation of' || operation == 'add trig rotation of' || operation == 'subtract trig rotation of') { vector1 = cast_to_vector(args.VECTORA,2); // require both to be 2D vectors vector2 = cast_to_vector(args.VECTORB,2); let angle = null; if (operation == 'add Scratch rotation of') { angle = Math.atan2(vector1[1],vector1[0]) + (Math.atan2(vector2[1],vector2[0]) - Math.PI/2); } else if (operation == 'subtract Scratch rotation of') { angle = Math.atan2(vector1[1],vector1[0]) - (Math.atan2(vector2[1],vector2[0]) - Math.PI/2); } else if (operation == 'add trig rotation of') { angle = Math.atan2(vector1[1],vector1[0]) + Math.atan2(vector2[1],vector2[0]); } else { angle = Math.atan2(vector1[1],vector1[0]) - Math.atan2(vector2[1],vector2[0]); } let magnitude = Math.sqrt((vector1[0] * vector1[0]) + (vector1[1] * vector1[1])); vector.push(magnitude*Math.cos(angle)); vector.push(magnitude*Math.sin(angle)); } else if (operation == 'add magnitude of' || operation == 'subtract magnitude of') { vector2 = cast_to_vector(args.VECTORB); // remove requirement that both lengths must be the same let magnitude1 = 0; for (let i = 0; i < vector1.length; i++) { magnitude1 += vector1[i] * vector1[i]; } let magnitude2 = 0; for (let i = 0; i < vector2.length; i++) { magnitude2 += vector2[i] * vector2[i]; } magnitude1 = Math.sqrt(magnitude1); magnitude2 = Math.sqrt(magnitude2); let angle = Math.atan2(vector1[1],vector1[0]); let magnitude = null; if (operation == 'add magnitude of') { magnitude = magnitude1 + magnitude2; } else { magnitude = magnitude1 - magnitude2; } vector.push(magnitude*Math.cos(angle)); vector.push(magnitude*Math.sin(angle)); } return stringify(vector); } vectorvaroperations(args,util) { set_var(args.VARIABLE, this.vectoroperations({VECTORA: get_var(args.VARIABLE, util), VECTORB: args.VECTOR, OPERATION: args.OPERATION}), util); } twovectorvaroperations(args,util) { set_var(args.VARIABLEA, this.vectoroperations({VECTORA: get_var(args.VARIABLEA, util), VECTORB: get_var(args.VARIABLEB, util), OPERATION: args.OPERATION}), util); } dirofvector(args) { let vector = cast_to_vector(args.VECTOR,2); let angle = Math.atan2(vector[1],vector[0]); angle = convert_angle_units(angle,'trig radians',args.UNIT); return angle; } dirofvectorvar(args,util) { return this.run_against_var(this.dirofvector,args,util); } magnitudeofvector(args) { let vector = cast_to_vector(args.VECTOR); let sum = 0; for (let i = 0; i < vector.length; i++) { sum += vector[i] * vector[i]; } return Math.sqrt(sum); } magnitudeofvectorvar(args,util) { return this.run_against_var(this.magnitudeofvector,args,util); } slopeofvector(args) { let vector = cast_to_vector(args.VECTOR,2); if (vector[0] == 0 && vector[1] == 0) { return 0; } return vector[1] / vector[0]; } slopeofvectorvar(args,util) { return this.run_against_var(this.slopeofvector,args,util); } rotatevector(args) { let vector = cast_to_vector(args.VECTOR,2); let angle = Scratch.Cast.toNumber(args.ANGLE); angle = convert_angle_units(angle,args.UNIT,'trig radians',true); let old_angle = Math.atan2(vector[1],vector[0]); let magnitude = Math.sqrt((vector[0] * vector[0]) + (vector[1] * vector[1])); let new_angle = old_angle + angle; let new_vector = [magnitude*Math.cos(new_angle),magnitude*Math.sin(new_angle)]; return stringify(new_vector); } rotatevectorvar(args,util) { this.run_in_place(this.rotatevector,args,util); } scalemagnitudeofvector(args) { let vector = cast_to_vector(args.VECTOR); let factor = Scratch.Cast.toNumber(args.FACTOR); for (let i = 0; i < vector.length; i++) { vector[i] *= factor; } return stringify(vector); } scalemagnitudeofvectorvar(args,util) { this.run_in_place(this.scalemagnitudeofvector,args,util); } normalizevector(args) { let vector = cast_to_vector(args.VECTOR); let magnitude = 0; for (let i = 0; i < vector.length; i++) { magnitude += vector[i] * vector[i]; } magnitude = Math.sqrt(magnitude); for (let i = 0; i < vector.length; i++) { vector[i] /= magnitude; } return stringify(vector); } normalizevectorvar(args,util) { this.run_in_place(this.normalizevector,args,util); } convertangleunits(args) { let angle = Scratch.Cast.toNumber(args.ANGLE); let from_unit = args.FROMUNIT; let to_unit = args.TOUNIT; return convert_angle_units(angle,from_unit,to_unit); } convertcoordinateunits(args) { let vector = cast_to_vector(args.VECTOR,2); let from_unit = args.FROMUNIT; let to_unit = args.TOUNIT; return stringify(convert_coordinate_units(vector,from_unit,to_unit)); } stagesizevector(args) { if (args.MODE == 'size') { return stringify([stage_width(),stage_height()]); } else if (args.MODE == 'width') { return stage_width(); } else if (args.MODE == 'height') { return stage_height(); } else { return 0; } } fourvectorfromcoords(args) { let vector = []; vector.push(Scratch.Cast.toNumber(args.X)); vector.push(Scratch.Cast.toNumber(args.Y)); vector.push(Scratch.Cast.toNumber(args.Z)); vector.push(Scratch.Cast.toNumber(args.W)); return stringify(vector); } fourvectorvarfromcoords(args,util) { set_var(args.VARIABLE, this.fourvectorfromcoords(args), util); } fivevectorfromcoords(args) { let vector = []; vector.push(Scratch.Cast.toNumber(args.X)); vector.push(Scratch.Cast.toNumber(args.Y)); vector.push(Scratch.Cast.toNumber(args.Z)); vector.push(Scratch.Cast.toNumber(args.W)); vector.push(Scratch.Cast.toNumber(args.V)); return stringify(vector); } fivevectorvarfromcoords(args,util) { set_var(args.VARIABLE, this.fivevectorfromcoords(args), util); } sixvectorfromcoords(args) { let vector = []; vector.push(Scratch.Cast.toNumber(args.X)); vector.push(Scratch.Cast.toNumber(args.Y)); vector.push(Scratch.Cast.toNumber(args.Z)); vector.push(Scratch.Cast.toNumber(args.W)); vector.push(Scratch.Cast.toNumber(args.V)); vector.push(Scratch.Cast.toNumber(args.U)); return stringify(vector); } sixvectorvarfromcoords(args,util) { set_var(args.VARIABLE, this.sixvectorfromcoords(args), util); } sevenvectorfromcoords(args) { let vector = []; vector.push(Scratch.Cast.toNumber(args.X)); vector.push(Scratch.Cast.toNumber(args.Y)); vector.push(Scratch.Cast.toNumber(args.Z)); vector.push(Scratch.Cast.toNumber(args.W)); vector.push(Scratch.Cast.toNumber(args.V)); vector.push(Scratch.Cast.toNumber(args.U)); vector.push(Scratch.Cast.toNumber(args.T)); return stringify(vector); } sevenvectorvarfromcoords(args,util) { set_var(args.VARIABLE, this.sevenvectorfromcoords(args), util); } eightvectorfromcoords(args) { let vector = []; vector.push(Scratch.Cast.toNumber(args.X)); vector.push(Scratch.Cast.toNumber(args.Y)); vector.push(Scratch.Cast.toNumber(args.Z)); vector.push(Scratch.Cast.toNumber(args.W)); vector.push(Scratch.Cast.toNumber(args.V)); vector.push(Scratch.Cast.toNumber(args.U)); vector.push(Scratch.Cast.toNumber(args.T)); vector.push(Scratch.Cast.toNumber(args.S)); return stringify(vector); } eightvectorvarfromcoords(args,util) { set_var(args.VARIABLE, this.eightvectorfromcoords(args), util); } } // load extension Scratch.extensions.register(new SoupUtilsVectors()); })(Scratch);