diff --git a/src/lib/extensions.js b/src/lib/extensions.js index adb9602ce..10333e541 100644 --- a/src/lib/extensions.js +++ b/src/lib/extensions.js @@ -442,6 +442,14 @@ export default [ creatorAlias: "gaimerI17", note: "Extension thumbnail made by Dillon." }, + { + name: "Enumerations", + description: "Create and use enumerations in your project. Enums allow you to assign meaningful labels to arbitrary numbers, and even change them later without issue.", + code: "DogeisCut/Enumerations.js", + banner: "DogeisCut/Enumerations.svg", + creator: "DogeisCut", + isGitHub: true, + }, /* these extensions are completely dead as of now { name: "Online Captcha", diff --git a/static/extensions/DogeisCut/Enumerations.js b/static/extensions/DogeisCut/Enumerations.js new file mode 100644 index 000000000..880479e86 --- /dev/null +++ b/static/extensions/DogeisCut/Enumerations.js @@ -0,0 +1,406 @@ +// Name: Enumerations +// ID: dogeiscutenumerations +// Description: Create and use enumerations in your project. Enums allow you to assign meaningful labels to arbitrary numbers, and even change them later without issue. +// By: DogeisCut +// License: MIT + +// Version V.1.0.0 + +// TODO: +// - Use dynamic blocks instead of this opcode fuckery +// "being real" "you shouldnt create blocks using an opcode like that" "the better way is to just use dynamic blocks" +// "pretty simple" "you define it" "then" "when you make the block (using xml) you can set its blockInfo" "so" +// "you can change the text to literally anything" "and" "it still has the same opcode and everything" +// - Figure out more useful blocks to add +// - keys of [ENUM] +// - labels of [ENUM] + +(function(Scratch) { + 'use strict'; + + if (!Scratch.extensions.unsandboxed) { + throw new Error('\'Enumerations\' must run unsandboxed!'); + } + + const vm = Scratch.vm; + const runtime = vm.runtime; + + let hideEnumBlocks = true; + + let enums = {}; + + function createNewEnum(name, target, scope) { + const newUid = uid(); + const clones = target.sprite.clones; + + for (const clone of clones) { + if (!clone) { + + } + } + } + + // tiny patch for events to update the dropdown + if (Scratch.gui) Scratch.gui.getBlockly().then(SB => { + const { Events, mainWorkspace } = SB; + const workspaceEvents = (event) => { + if (mainWorkspace.id === event.workspaceId) { + if (event.type === Events.CHANGE) { + if (event.name == "ENUM") { + const block = mainWorkspace.getBlockById(event.blockId) + if (block.type == "dogeiscutenumerations_keyOfEnum") { + const chosenEnum = event.newValue + block.getInput(0).fieldRow[0].setValue(Object.keys(enums[chosenEnum])[0]) + } + } + } + } + }; + mainWorkspace.addChangeListener(workspaceEvents); + }); + + function openModal(titleName, promptTitle, addSelector, func) { + let enumData = {}; + ScratchBlocks.prompt( + titleName, + "", + (value) => { + enumData = getEnumData(); + func(value, enumData); + }, + promptTitle, + "broadcast_msg" + ); + + if (addSelector) { + const input = document.querySelector(`div[class="ReactModalPortal"] input`); + const newLabel = input.parentNode.previousSibling.cloneNode(true); + newLabel.textContent = "Labels and values:"; + + const container = document.createElement("div"); + container.setAttribute("class", "enum-container"); + + const addButton = document.createElement("button"); + addButton.textContent = "+"; + addButton.addEventListener("click", () => { + const enumEntry = document.createElement("div"); + enumEntry.setAttribute("class", "enum-entry"); + + const label = document.createElement("input"); + label.setAttribute("type", "text"); + label.setAttribute("placeholder", "Label"); + + const numberInput = document.createElement("input"); + numberInput.setAttribute("type", "number"); + numberInput.setAttribute("placeholder", "Value"); + + // Autofill with a unique number + const existingEntries = container.querySelectorAll('.enum-entry'); + numberInput.value = existingEntries.length + 1; + + const removeButton = document.createElement("button"); + removeButton.textContent = "-"; + removeButton.addEventListener("click", () => { + container.removeChild(enumEntry); + }); + + enumEntry.appendChild(label); + enumEntry.appendChild(numberInput); + enumEntry.appendChild(removeButton); + container.appendChild(enumEntry); + }); + + input.parentNode.append(newLabel, container, addButton); + } + } + + function getEnumData() { + const enumData = {}; + const entries = document.querySelectorAll('.enum-entry'); + entries.forEach((entry, index) => { + let label = entry.querySelector('input[type="text"]').value.trim(); + let value = parseFloat(entry.querySelector('input[type="number"]').value); + if (!label) { + label = `Label${index + 1}`; + } + if (isNaN(value)) { + value = index + 1; + } + let uniqueLabel = label; + let counter = 1; + while (enumData.hasOwnProperty(uniqueLabel)) { + uniqueLabel = `${label}${counter}`; + counter++; + } + if (uniqueLabel !== label) { + entry.querySelector('input[type="text"]').value = uniqueLabel; + } + enumData[uniqueLabel] = value; + }); + return enumData; + } + + function getCurrentBlockArgs() { + const ScratchBlocks = window.ScratchBlocks; + if (!ScratchBlocks) return {}; + const source = ScratchBlocks.selected; + if (!source) return {}; + + const args = {}; + for (const input of source.inputList) { + for (const field of input.fieldRow) { + if (field.isCurrentlyEditable()) args[field.name] = field.getValue(); + } + if (!input.connection) continue; + const block = input.connection.targetConnection.sourceBlock_; + if (!block || !block.isShadow()) continue; + for (const input2 of block.inputList) { + for (const field2 of input2.fieldRow) { + if (field2.isCurrentlyEditable()) args[input.name] = field2.getValue(); + } + } + } + return args; + } + + const icon = '' + + class Enumerations { + getInfo() { + return { + id: 'dogeiscutenumerations', + name: 'Enumerations', + color1: "#BE271A", + menuIconURI: icon, + blocks: [ + { + func: "makeAnEnum", + blockType: Scratch.BlockType.BUTTON, + text: "Make an Enum" + }, + { + func: "removeAnEnum", + blockType: Scratch.BlockType.BUTTON, + text: "Remove an Enum" + }, + '---', + { + opcode: "enum", + blockType: Scratch.BlockType.REPORTER, + text: "", + isDynamic: true, + hideFromPalette: true, + arguments: { + DICTIONARY: { + type: Scratch.ArgumentType.STRING, + defaultValue: "", + }, + }, + }, + { + blockType: Scratch.BlockType.XML, + xml: ` + + + + + ` + }, + '---', + { + opcode: "keyOfEnum", + text: "[KEY] of enum [ENUM]", + blockType: Scratch.BlockType.REPORTER, + hideFromPalette: hideEnumBlocks, + disableMonitor: true, + arguments: { + KEY: { + type: Scratch.ArgumentType.STRING, + menu: 'getKeysOf' + }, + ENUM: { + type: Scratch.ArgumentType.STRING, + menu: 'getEnums' + } + } + }, + { + opcode: "enumLength", + text: "length of [ENUM]", + blockType: Scratch.BlockType.REPORTER, + hideFromPalette: hideEnumBlocks, + disableMonitor: true, + arguments: { + ENUM: { + type: Scratch.ArgumentType.STRING, + menu: 'getEnums' + } + } + }, + { + opcode: "enumContains", + text: "[ENUM] contains [KEY]?", + blockType: Scratch.BlockType.BOOLEAN, + hideFromPalette: hideEnumBlocks, + disableMonitor: true, + arguments: { + ENUM: { + type: Scratch.ArgumentType.STRING, + menu: 'getEnums' + }, + KEY: { + type: Scratch.ArgumentType.STRING + } + } + } + ], + /*menus: enumMenus*/ + menus: { + getEnums: { + acceptReporters: false, + items: 'getEnums' + }, + getKeysOf: { + acceptReporters: false, + items: 'getKeysOf' + } + } + } + } + + serialize() { + return { dogeiscutenumerations: {enumBlocks, enums} } + } + + deserialize(data) { + if (data.dogeiscutenumerations) { + const { enumBlocks: savedEnumBlocks, enums: savedEnums } = data.dogeiscutenumerations; + enumBlocks = savedEnumBlocks || []; + enums = savedEnums || {}; + + enumBlocks.forEach(block => { + this.addBlock(block.opcode); + }); + + hideEnumBlocks = false; + } + } + + /* helper functions */ + + addBlock(opcode) { + Object.defineProperty(Enumerations.prototype, opcode, { + value: function (_, util, blockInfo) { + return this.thisEnum("", util, blockInfo); + }, + writable: true, + configurable: true, + }); + } + + addDropdownBlock(opcode) { + Object.defineProperty(Enumerations.prototype, opcode, { + value: function (args, util, blockInfo) { + return this.thisEnum(args, util, blockInfo); + }, + writable: true, + configurable: true, + }); + } + + getPrevBlock(util) { + const contain = util.thread.blockContainer; + const block = contain.getBlock(util.thread.isCompiled ? util.thread.peekStack() : util.thread.peekStackFrame().op?.id); + return contain.getBlock(block?.parent); + } + + /* menus */ + + getEnums() { + return Object.keys(enums); + } + + getKeysOf() { + const args = getCurrentBlockArgs(); + const enumName = args.ENUM; + if (enums[enumName]) { + const keys = Object.keys(enums[enumName]) + if (keys.length != 0) { + return keys; + } + } + return [""]; + } + + /* blocks */ + + enum({ mutator }) { + return enums[mutator.blockInfo.text]; + } + + thisEnum(_, util, blockInfo) { + const isInExtBlock = this.getPrevBlock(util)?.opcode.startsWith("dogeiscutenumerations_"); + const enumInfo = enums[blockInfo.text]; + return enumInfo ? isInExtBlock ? enumInfo : JSON.stringify(enumInfo) : ""; + } + + keyOfEnum(args) { + const enumName = args.ENUM; + const key = args.KEY; + if (enums[enumName] && enums[enumName].hasOwnProperty(key)) { + return enums[enumName][key]; + } + return ""; + } + + enumLength(args) { + const enumName = args.ENUM; + if (enums[enumName]) { + return Object.keys(enums[enumName]).length; + } + return 0; + } + + enumContains(args) { + const enumName = args.ENUM; + const key = args.KEY; + if (enums[enumName]) { + return enums[enumName].hasOwnProperty(key); + } + return false; + } + + /* buttons */ + + makeAnEnum() { + openModal("New enum name:", "New Enum", true, ((value, enumData) => { + if (!value) return; + if (Object.keys(enumData).length === 0) return; + const editingTarget = runtime.getEditingTarget(); + const stage = runtime.getTargetForStage(); + + + hideEnumBlocks = false; + vm.extensionManager.refreshBlocks("dogeiscutenumerations"); + this.serialize(); + enums[value] = enumData + })) + } + + removeAnEnum() { + openModal("Remove enum named:", "Remove Enum", false, (value) => { + const block = enumBlocks.find((i) => { return i.text == value }); + if (!block) return; + block.hideFromPalette = true; + delete enums[value]; + runtime.monitorBlocks.changeBlock({ id: `dogeiscutenumerations_enum_${value}`, element: "checkbox", value: false }, runtime); + + if (Object.keys(enums).length === 0) hideEnumBlocks = true; + this.serialize(); + vm.extensionManager.refreshBlocks("dogeiscutenumerations"); + }); + } + } + + Scratch.extensions.register(new Enumerations()); +})(Scratch); \ No newline at end of file diff --git a/static/images/DogeisCut/Enumerations.svg b/static/images/DogeisCut/Enumerations.svg new file mode 100644 index 000000000..9c2cf16b6 --- /dev/null +++ b/static/images/DogeisCut/Enumerations.svg @@ -0,0 +1,230 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +