diff --git a/features/features.json b/features/features.json
index 9f0054a1..9588ccbe 100644
--- a/features/features.json
+++ b/features/features.json
@@ -1,4 +1,9 @@
[
+ {
+ "version": 2,
+ "id": "more-paint-functions",
+ "versionAdded": "v4.0.0"
+ },
{
"version": 2,
"id": "asset-size",
diff --git a/features/more-editor-fonts/script.js b/features/more-editor-fonts/script.js
index 791529dc..7efad503 100644
--- a/features/more-editor-fonts/script.js
+++ b/features/more-editor-fonts/script.js
@@ -9,6 +9,7 @@ export default async function ({ feature, console }) {
feature.page.waitForElements(
"div[class^='asset-panel_wrapper_'] div[class^='action-menu_more-buttons_']",
function (menu) {
+ if (feature.traps.gui().editorTab.activeTabIndex !== 1) return;
if (menu.querySelector(".ste-more-fonts")) return;
let div = document.createElement("div");
diff --git a/features/more-paint-functions/data.json b/features/more-paint-functions/data.json
new file mode 100644
index 00000000..11c00e90
--- /dev/null
+++ b/features/more-paint-functions/data.json
@@ -0,0 +1,22 @@
+{
+ "title": "More Paint Functions",
+ "description": "Adds new functions to the paint editor. The new functions include unite (combining them into one), subtract (removing one shape from another), exclude (removing the overlap of 2 items), and intersect (removing everything but the overlap of 2 items). Each function only works with 2 items.",
+ "credits": [
+ { "username": "rgantzos", "url": "https://scratch.mit.edu/users/rgantzos/" }
+ ],
+ "type": ["Editor"],
+ "tags": ["New", "Featured"],
+ "dynamic": true,
+ "scripts": [{ "file": "script.js", "runOn": "/projects/*" }],
+ "styles": [{ "file": "style.css", "runOn": "/projects/*" }],
+ "resources": [
+ { "name": "function-exclude", "path": "/icons/exclude.svg" },
+ { "name": "function-intersect", "path": "/icons/intersect.svg" },
+ { "name": "function-subtract", "path": "/icons/subtract.svg" },
+ { "name": "function-unite", "path": "/icons/unite.svg" }
+ ],
+ "components": [{
+ "type": "warning",
+ "content": "In order to avoid clutter in the paint editor, this feature replaces the Copy, Paste and Delete buttons. However, hotkeys still work."
+ }]
+}
diff --git a/features/more-paint-functions/icons/exclude.svg b/features/more-paint-functions/icons/exclude.svg
new file mode 100644
index 00000000..76c50759
--- /dev/null
+++ b/features/more-paint-functions/icons/exclude.svg
@@ -0,0 +1,13 @@
+
+
\ No newline at end of file
diff --git a/features/more-paint-functions/icons/intersect.svg b/features/more-paint-functions/icons/intersect.svg
new file mode 100644
index 00000000..399d3a7d
--- /dev/null
+++ b/features/more-paint-functions/icons/intersect.svg
@@ -0,0 +1,20 @@
+
+
\ No newline at end of file
diff --git a/features/more-paint-functions/icons/subtract.svg b/features/more-paint-functions/icons/subtract.svg
new file mode 100644
index 00000000..4d10c195
--- /dev/null
+++ b/features/more-paint-functions/icons/subtract.svg
@@ -0,0 +1,17 @@
+
+
\ No newline at end of file
diff --git a/features/more-paint-functions/icons/unite.svg b/features/more-paint-functions/icons/unite.svg
new file mode 100644
index 00000000..dee97002
--- /dev/null
+++ b/features/more-paint-functions/icons/unite.svg
@@ -0,0 +1,13 @@
+
+
\ No newline at end of file
diff --git a/features/more-paint-functions/script.js b/features/more-paint-functions/script.js
new file mode 100644
index 00000000..ddfd0849
--- /dev/null
+++ b/features/more-paint-functions/script.js
@@ -0,0 +1,161 @@
+export default async function ({ feature, console }) {
+ function unite() {
+ let paper = feature.traps.getPaper();
+ let items = paper.project.selectedItems;
+
+ if (items.length !== 2) return;
+
+ for (var i in items) {
+ if (i > 0) {
+ items[0].unite(items[i]);
+ }
+ }
+
+ for (var i in items) {
+ items[i].remove();
+ }
+
+ paper.tool.onUpdateImage();
+ }
+
+ function subtract() {
+ let paper = feature.traps.getPaper();
+ let items = paper.project.selectedItems;
+
+ if (items.length !== 2) return;
+
+ for (var i in items) {
+ if (i > 0) {
+ items[0].subtract(items[i]);
+ }
+ }
+
+ for (var i in items) {
+ items[i].remove();
+ }
+
+ paper.tool.onUpdateImage();
+ }
+
+ function exclude() {
+ let paper = feature.traps.getPaper();
+ let items = paper.project.selectedItems;
+
+ if (items.length !== 2) return;
+
+ for (var i in items) {
+ if (i > 0) {
+ items[0].exclude(items[i]);
+ }
+ }
+
+ for (var i in items) {
+ items[i].remove();
+ }
+
+ paper.tool.onUpdateImage();
+ }
+
+ function intersect() {
+ let paper = feature.traps.getPaper();
+ let items = paper.project.selectedItems;
+
+ if (items.length !== 2) return;
+
+ for (var i in items) {
+ if (i > 0) {
+ items[0].intersect(items[i]);
+ }
+ }
+
+ for (var i in items) {
+ items[i].remove();
+ }
+
+ paper.tool.onUpdateImage();
+ }
+
+ ScratchTools.waitForElements(
+ "div[class^='mode-tools_mod-labeled-icon-height_']",
+ async function (row) {
+ if (row.querySelector(".ste-more-functions")) return;
+
+ let functions = [
+ {
+ name: "Unite",
+ icon: "function-unite",
+ callback: unite,
+ },
+ {
+ name: "Subtract",
+ icon: "function-subtract",
+ callback: subtract,
+ },
+ {
+ name: "Exclude",
+ icon: "function-exclude",
+ callback: exclude,
+ },
+ {
+ name: "Intersect",
+ icon: "function-intersect",
+ callback: intersect,
+ },
+ ];
+
+ for (var i in functions) {
+ row.appendChild(makeButton(functions[i]));
+ }
+
+ let align = await ScratchTools.waitForElement(".ste-align-items");
+ row.appendChild(align);
+ }
+ );
+
+ feature.redux.subscribe(function () {
+ if (document.querySelector(".ste-more-functions")) {
+ let span = document.querySelector(".ste-more-functions");
+ if (
+ feature.traps.paint().format === "BITMAP" ||
+ feature.traps.paint().selectedItems?.length < 2
+ ) {
+ document.querySelectorAll(".ste-more-functions").forEach(function (el) {
+ el.classList.add("button_mod-disabled_1rf31");
+ });
+ } else {
+ document.querySelectorAll(".ste-more-functions").forEach(function (el) {
+ el.classList.remove("button_mod-disabled_1rf31");
+ });
+ }
+ }
+ });
+
+ function makeButton({ name, icon, callback }) {
+ let span = document.createElement("span");
+ span.className =
+ "button_button_u6SE2 labeled-icon-button_mod-edit-field_1bXYC ste-more-functions";
+ span.role = "button";
+
+ let img = document.createElement("img");
+ img.src = feature.self.getResource(icon);
+ img.className = "labeled-icon-button_edit-field-icon_3j-Pf";
+ img.alt = name;
+ img.title = name;
+ img.draggable = false;
+ span.appendChild(img);
+
+ let label = document.createElement("span");
+ label.textContent = name;
+ label.className = "labeled-icon-button_edit-field-title_1ZoEV";
+ span.appendChild(label);
+
+ span.addEventListener("click", function (e) {
+ if (span.className.includes("disabled")) return;
+ callback();
+ });
+
+ feature.self.hideOnDisable(span);
+
+ return span;
+ }
+}
diff --git a/features/more-paint-functions/style.css b/features/more-paint-functions/style.css
new file mode 100644
index 00000000..752ac1b7
--- /dev/null
+++ b/features/more-paint-functions/style.css
@@ -0,0 +1,7 @@
+div[class*='mode-tools_mode-tools_'] > div[class*='mode-tools_mod-dashed-border_']:nth-child(1), div[class*='mode-tools_mode-tools_'] > div[class*='mode-tools_mod-dashed-border_']:nth-child(2) {
+ display: none;
+}
+
+div[class*='mode-tools_mode-tools_'] > div[class*='mode-tools_mod-labeled-icon-height_']:nth-child(3) {
+ margin-left: 0px !important;
+}
\ No newline at end of file