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 @@
+
+
+
+