From 42d2bb422cc57116ef3ff88638747b1084909722 Mon Sep 17 00:00:00 2001 From: DogeisCut Date: Sat, 15 Mar 2025 23:54:18 -0600 Subject: [PATCH 1/5] Add Enumerations Extension --- src/lib/extensions.js | 8 + static/extensions/DogeisCut/Enumerations.js | 376 ++++++++++++++++++++ static/images/DogeisCut/Enumerations.svg | 230 ++++++++++++ 3 files changed, 614 insertions(+) create mode 100644 static/extensions/DogeisCut/Enumerations.js create mode 100644 static/images/DogeisCut/Enumerations.svg diff --git a/src/lib/extensions.js b/src/lib/extensions.js index adad79b5..6912ac95 100644 --- a/src/lib/extensions.js +++ b/src/lib/extensions.js @@ -527,4 +527,12 @@ export default [ unstable: true, unstableReason: "WebGPU is still experimental and not supported by all browsers and does not work when packaged to electron. Check compatibility at webgpu.io." }, + { + name: "Enumerations", + description: "Create and use enumerations in your project.", + code: "DogeisCut/Enumerations.js", + banner: "DogeisCut/Enumerations.svg", + creator: "DogeisCut", + isGitHub: true, + } ]; diff --git a/static/extensions/DogeisCut/Enumerations.js b/static/extensions/DogeisCut/Enumerations.js new file mode 100644 index 00000000..58fb6450 --- /dev/null +++ b/static/extensions/DogeisCut/Enumerations.js @@ -0,0 +1,376 @@ +// Name: Enumerations +// ID: dogeiscutenumerations +// Description: No description provided. +// By: DogeisCut +// License: MIT + +(function(Scratch) { + 'use strict'; + + if (!Scratch.extensions.unsandboxed) { + throw new Error('\'Enumerations\' must run unsandboxed!'); + } + + const vm = Scratch.vm; + const runtime = vm.runtime; + + let enumBlocks = []; + const defaultEnumBlock = { + opcode: "enum...", text: "...", + blockType: Scratch.BlockType.REPORTER + }; + + let hideEnumBlocks = true; + + let enums = {}; + + // 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" + }, + '---', + ...enumBlocks, + '---', + /* + ...enumDropdownBlocks, + '---', + */ + { + 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 */ + + 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 newBlock = { + ...defaultEnumBlock, + opcode: "enum_" + value, text: value + }; + this.addBlock(newBlock.opcode); + + const block = enumBlocks.find((i) => { return i.text == value }); + if (block) block.hideFromPalette = false; + else enumBlocks.push(newBlock); + + 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 00000000..9c2cf16b --- /dev/null +++ b/static/images/DogeisCut/Enumerations.svg @@ -0,0 +1,230 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + From 994105547939d3fa956d157fec8f2fe620fab89d Mon Sep 17 00:00:00 2001 From: DogeisCut Date: Sat, 15 Mar 2025 23:57:47 -0600 Subject: [PATCH 2/5] Update extensions.js --- src/lib/extensions.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/lib/extensions.js b/src/lib/extensions.js index 6912ac95..6c09a3d7 100644 --- a/src/lib/extensions.js +++ b/src/lib/extensions.js @@ -534,5 +534,5 @@ export default [ banner: "DogeisCut/Enumerations.svg", creator: "DogeisCut", isGitHub: true, - } + }, ]; From 65864e80815f4877be0467e8d02f0ece4ba8cf1a Mon Sep 17 00:00:00 2001 From: DogeisCut Date: Sun, 16 Mar 2025 06:38:25 -0600 Subject: [PATCH 3/5] Todo List --- static/extensions/DogeisCut/Enumerations.js | 11 +++++++++++ 1 file changed, 11 insertions(+) diff --git a/static/extensions/DogeisCut/Enumerations.js b/static/extensions/DogeisCut/Enumerations.js index 58fb6450..ba7658cc 100644 --- a/static/extensions/DogeisCut/Enumerations.js +++ b/static/extensions/DogeisCut/Enumerations.js @@ -4,6 +4,17 @@ // 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'; From f94c4ab75a76129eb43621c09acf187da9ace0ed Mon Sep 17 00:00:00 2001 From: DogeisCut Date: Thu, 20 Mar 2025 15:12:22 -0600 Subject: [PATCH 4/5] Update extensions.js --- src/lib/extensions.js | 18 +++++++++--------- 1 file changed, 9 insertions(+), 9 deletions(-) diff --git a/src/lib/extensions.js b/src/lib/extensions.js index 2ed29943..21efe114 100644 --- a/src/lib/extensions.js +++ b/src/lib/extensions.js @@ -550,14 +550,6 @@ export default [ unstable: true, unstableReason: "WebGPU is still experimental and not supported by all browsers and does not work when packaged to electron. Check compatibility at webgpu.io." }, - { - name: "Enumerations", - description: "Create and use enumerations in your project.", - code: "DogeisCut/Enumerations.js", - banner: "DogeisCut/Enumerations.svg", - creator: "DogeisCut", - isGitHub: true, - }, { name: "Format Numbers", description: "Format large numbers into AD standard, fixed decimal, comma separated, or scientific notation.", @@ -567,4 +559,12 @@ export default [ isGitHub: true, notes: "Gallery banner by Dillon", }, -]; + { + name: "Enumerations", + description: "Create and use enumerations in your project.", + code: "DogeisCut/Enumerations.js", + banner: "DogeisCut/Enumerations.svg", + creator: "DogeisCut", + isGitHub: true, + }, +]; \ No newline at end of file From 1b23020f29bff7351ce5721eb0511e59d6662da7 Mon Sep 17 00:00:00 2001 From: DogeisCut Date: Thu, 20 Mar 2025 15:13:16 -0600 Subject: [PATCH 5/5] trying to figure out dynamic blocks --- src/lib/extensions.js | 2 +- static/extensions/DogeisCut/Enumerations.js | 50 ++++++++++++--------- 2 files changed, 29 insertions(+), 23 deletions(-) diff --git a/src/lib/extensions.js b/src/lib/extensions.js index 21efe114..96f4c21b 100644 --- a/src/lib/extensions.js +++ b/src/lib/extensions.js @@ -561,7 +561,7 @@ export default [ }, { name: "Enumerations", - description: "Create and use enumerations in your project.", + 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", diff --git a/static/extensions/DogeisCut/Enumerations.js b/static/extensions/DogeisCut/Enumerations.js index ba7658cc..d2557fde 100644 --- a/static/extensions/DogeisCut/Enumerations.js +++ b/static/extensions/DogeisCut/Enumerations.js @@ -1,6 +1,6 @@ // Name: Enumerations // ID: dogeiscutenumerations -// Description: No description provided. +// 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 @@ -25,16 +25,21 @@ const vm = Scratch.vm; const runtime = vm.runtime; - let enumBlocks = []; - const defaultEnumBlock = { - opcode: "enum...", text: "...", - blockType: Scratch.BlockType.REPORTER - }; - 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; @@ -179,12 +184,20 @@ text: "Remove an Enum" }, '---', - ...enumBlocks, - '---', - /* - ...enumDropdownBlocks, + { + opcode: "getEunm", + blockType: Scratch.BlockType.REPORTER, + text: "MyEnum", + isDynamic: true, + hideFromPalette: true, + arguments: { + DICTIONARY: { + type: Scratch.ArgumentType.STRING, + defaultValue: "", + }, + }, + }, '---', - */ { opcode: "keyOfEnum", text: "[KEY] of enum [ENUM]", @@ -350,18 +363,11 @@ openModal("New enum name:", "New Enum", true, ((value, enumData) => { if (!value) return; if (Object.keys(enumData).length === 0) return; - const newBlock = { - ...defaultEnumBlock, - opcode: "enum_" + value, text: value - }; - this.addBlock(newBlock.opcode); - - const block = enumBlocks.find((i) => { return i.text == value }); - if (block) block.hideFromPalette = false; - else enumBlocks.push(newBlock); + const editingTarget = runtime.getEditingTarget(); + const stage = runtime.getTargetForStage(); + hideEnumBlocks = false; - vm.extensionManager.refreshBlocks("dogeiscutenumerations"); this.serialize(); enums[value] = enumData