diff --git a/shapez-io/electron_wegame/README.md b/shapez-io/electron_wegame/README.md
new file mode 100644
index 00000000..70736caf
--- /dev/null
+++ b/shapez-io/electron_wegame/README.md
@@ -0,0 +1 @@
+To build, place the lib64 folder from the wegame sdk for electron 13 in `wegame_sdk` and run the `wegame.main.standalone` gulp task.
diff --git a/shapez-io/electron_wegame/electron_wegame.code-workspace b/shapez-io/electron_wegame/electron_wegame.code-workspace
new file mode 100644
index 00000000..fc9ab864
--- /dev/null
+++ b/shapez-io/electron_wegame/electron_wegame.code-workspace
@@ -0,0 +1,13 @@
+{
+ "folders": [
+ {
+ "path": "."
+ }
+ ],
+ "settings": {
+ "files.exclude": {
+ "**/node_modules": true,
+ "**/typedefs_gen": true
+ }
+ }
+}
\ No newline at end of file
diff --git a/shapez-io/electron_wegame/favicon.icns b/shapez-io/electron_wegame/favicon.icns
new file mode 100644
index 00000000..79e141a5
Binary files /dev/null and b/shapez-io/electron_wegame/favicon.icns differ
diff --git a/shapez-io/electron_wegame/favicon.ico b/shapez-io/electron_wegame/favicon.ico
new file mode 100644
index 00000000..81a9aa5c
Binary files /dev/null and b/shapez-io/electron_wegame/favicon.ico differ
diff --git a/shapez-io/electron_wegame/favicon.png b/shapez-io/electron_wegame/favicon.png
new file mode 100644
index 00000000..c837c787
Binary files /dev/null and b/shapez-io/electron_wegame/favicon.png differ
diff --git a/shapez-io/electron_wegame/index.js b/shapez-io/electron_wegame/index.js
new file mode 100644
index 00000000..2c183f15
--- /dev/null
+++ b/shapez-io/electron_wegame/index.js
@@ -0,0 +1,253 @@
+/* eslint-disable quotes,no-undef */
+
+const { app, BrowserWindow, Menu, MenuItem, ipcMain, shell } = require("electron");
+
+app.commandLine.appendSwitch("in-process-gpu");
+
+const path = require("path");
+const url = require("url");
+const fs = require("fs");
+const wegame = require("./wegame");
+const asyncLock = require("async-lock");
+
+const isDev = process.argv.indexOf("--dev") >= 0;
+const isLocal = process.argv.indexOf("--local") >= 0;
+
+const roamingFolder =
+ process.env.APPDATA ||
+ (process.platform == "darwin"
+ ? process.env.HOME + "/Library/Preferences"
+ : process.env.HOME + "/.local/share");
+let storePath = path.join(roamingFolder, "shapez.io", "saves");
+
+if (!fs.existsSync(storePath)) {
+ // No try-catch by design
+ fs.mkdirSync(storePath, { recursive: true });
+}
+
+/** @type {BrowserWindow} */
+let win = null;
+let menu = null;
+
+function createWindow() {
+ let faviconExtension = ".png";
+ if (process.platform === "win32") {
+ faviconExtension = ".ico";
+ }
+
+ win = new BrowserWindow({
+ width: 1280,
+ height: 800,
+ show: false,
+ backgroundColor: "#222428",
+ useContentSize: true,
+ minWidth: 800,
+ minHeight: 600,
+ title: "图形工厂",
+ transparent: false,
+ icon: path.join(__dirname, "favicon" + faviconExtension),
+ // fullscreen: true,
+ autoHideMenuBar: true,
+ webPreferences: {
+ nodeIntegration: false,
+ webSecurity: true,
+ sandbox: true,
+
+ contextIsolation: true,
+ preload: path.join(__dirname, "preload.js"),
+ },
+ allowRunningInsecureContent: false,
+ });
+
+ if (isLocal) {
+ win.loadURL("http://localhost:3005");
+ } else {
+ win.loadURL(
+ url.format({
+ pathname: path.join(__dirname, "index.html"),
+ protocol: "file:",
+ slashes: true,
+ })
+ );
+ }
+ win.webContents.session.clearCache(() => null);
+ win.webContents.session.clearStorageData();
+
+ win.webContents.on("new-window", (event, pth) => {
+ event.preventDefault();
+ shell.openExternal(pth);
+ });
+
+ win.on("closed", () => {
+ console.log("Window closed");
+ win = null;
+ });
+
+ if (isDev) {
+ menu = new Menu();
+
+ const mainItem = new MenuItem({
+ label: "Toggle Dev Tools",
+ click: () => win.webContents.toggleDevTools(),
+ accelerator: "F12",
+ });
+ menu.append(mainItem);
+
+ const reloadItem = new MenuItem({
+ label: "Restart",
+ click: () => win.reload(),
+ accelerator: "F5",
+ });
+ menu.append(reloadItem);
+
+ const fullscreenItem = new MenuItem({
+ label: "Fullscreen",
+ click: () => win.setFullScreen(!win.isFullScreen()),
+ accelerator: "F11",
+ });
+ menu.append(fullscreenItem);
+
+ Menu.setApplicationMenu(menu);
+ } else {
+ Menu.setApplicationMenu(null);
+ }
+
+ win.once("ready-to-show", () => {
+ win.show();
+ win.focus();
+ });
+}
+
+if (!app.requestSingleInstanceLock()) {
+ app.exit(0);
+} else {
+ app.on("second-instance", (event, commandLine, workingDirectory) => {
+ // Someone tried to run a second instance, we should focus
+ if (win) {
+ if (win.isMinimized()) {
+ win.restore();
+ }
+ win.focus();
+ }
+ });
+}
+
+app.on("ready", createWindow);
+
+app.on("window-all-closed", () => {
+ console.log("All windows closed");
+ app.quit();
+});
+
+ipcMain.on("set-fullscreen", (event, flag) => {
+ win.setFullScreen(flag);
+});
+
+ipcMain.on("exit-app", (event, flag) => {
+ win.close();
+ app.quit();
+});
+
+let renameCounter = 1;
+
+const fileLock = new asyncLock({
+ timeout: 30000,
+ maxPending: 1000,
+});
+
+function niceFileName(filename) {
+ return filename.replace(storePath, "@");
+}
+
+async function writeFileSafe(filename, contents) {
+ ++renameCounter;
+ const prefix = "[ " + renameCounter + ":" + niceFileName(filename) + " ] ";
+ const transactionId = String(new Date().getTime()) + "." + renameCounter;
+
+ if (fileLock.isBusy()) {
+ console.warn(prefix, "Concurrent write process on", filename);
+ }
+
+ fileLock.acquire(filename, async () => {
+ console.log(prefix, "Starting write on", niceFileName(filename), "in transaction", transactionId);
+
+ if (!fs.existsSync(filename)) {
+ // this one is easy
+ console.log(prefix, "Writing file instantly because it does not exist:", niceFileName(filename));
+ await fs.promises.writeFile(filename, contents, "utf8");
+ return;
+ }
+
+ // first, write a temporary file (.tmp-XXX)
+ const tempName = filename + ".tmp-" + transactionId;
+ console.log(prefix, "Writing temporary file", niceFileName(tempName));
+ await fs.promises.writeFile(tempName, contents, "utf8");
+
+ // now, rename the original file to (.backup-XXX)
+ const oldTemporaryName = filename + ".backup-" + transactionId;
+ console.log(
+ prefix,
+ "Renaming old file",
+ niceFileName(filename),
+ "to",
+ niceFileName(oldTemporaryName)
+ );
+ await fs.promises.rename(filename, oldTemporaryName);
+
+ // now, rename the temporary file (.tmp-XXX) to the target
+ console.log(
+ prefix,
+ "Renaming the temporary file",
+ niceFileName(tempName),
+ "to the original",
+ niceFileName(filename)
+ );
+ await fs.promises.rename(tempName, filename);
+
+ // we are done now, try to create a backup, but don't fail if the backup fails
+ try {
+ // check if there is an old backup file
+ const backupFileName = filename + ".backup";
+ if (fs.existsSync(backupFileName)) {
+ console.log(prefix, "Deleting old backup file", niceFileName(backupFileName));
+ // delete the old backup
+ await fs.promises.unlink(backupFileName);
+ }
+
+ // rename the old file to the new backup file
+ console.log(prefix, "Moving", niceFileName(oldTemporaryName), "to the backup file location");
+ await fs.promises.rename(oldTemporaryName, backupFileName);
+ } catch (ex) {
+ console.error(prefix, "Failed to switch backup files:", ex);
+ }
+ });
+}
+
+ipcMain.handle("fs-job", async (event, job) => {
+ const filenameSafe = job.filename.replace(/[^a-z\.\-_0-9]/i, "");
+ const fname = path.join(storePath, filenameSafe);
+ switch (job.type) {
+ case "read": {
+ if (!fs.existsSync(fname)) {
+ // Special FILE_NOT_FOUND error code
+ return { error: "file_not_found" };
+ }
+ return await fs.promises.readFile(fname, "utf8");
+ }
+ case "write": {
+ await writeFileSafe(fname, job.contents);
+ return job.contents;
+ }
+
+ case "delete": {
+ await fs.promises.unlink(fname);
+ return;
+ }
+
+ default:
+ throw new Error("Unknown fs job: " + job.type);
+ }
+});
+
+wegame.init(isDev);
+wegame.listen();
diff --git a/shapez-io/electron_wegame/package.json b/shapez-io/electron_wegame/package.json
new file mode 100644
index 00000000..aba5bb6a
--- /dev/null
+++ b/shapez-io/electron_wegame/package.json
@@ -0,0 +1,18 @@
+{
+ "name": "electron",
+ "version": "1.0.0",
+ "main": "index.js",
+ "license": "MIT",
+ "private": true,
+ "scripts": {
+ "startDev": "electron --disable-direct-composition --in-process-gpu . --dev --local",
+ "startDevGpu": "electron --enable-gpu-rasterization --enable-accelerated-2d-canvas --num-raster-threads=8 --enable-zero-copy . --dev --local",
+ "start": "electron --disable-direct-composition --in-process-gpu ."
+ },
+ "devDependencies": {
+ "electron": "^13.1.6"
+ },
+ "dependencies": {
+ "async-lock": "^1.2.8"
+ }
+}
diff --git a/shapez-io/electron_wegame/preloader/preloader.css b/shapez-io/electron_wegame/preloader/preloader.css
new file mode 100644
index 00000000..f6775f76
--- /dev/null
+++ b/shapez-io/electron_wegame/preloader/preloader.css
@@ -0,0 +1,262 @@
+* {
+ margin: 0;
+ padding: 0;
+ touch-action: pan-x pan-y !important;
+ pointer-events: none;
+ -webkit-tap-highlight-color: rgba(255, 255, 255, 0);
+}
+
+html {
+ position: fixed;
+ -ms-touch-action: pan-x, pan-y;
+ touch-action: pan-x, pan-y;
+ -ms-content-zooming: none;
+ top: 0;
+ left: 0;
+ bottom: 0;
+ right: 0;
+ background: #dee1ea;
+}
+
+body {
+ color: #555;
+ user-select: none;
+ -moz-user-select: none;
+ -ms-user-select: none;
+ background: inherit !important;
+ text-transform: none;
+ white-space: normal;
+ word-break: normal;
+ word-spacing: normal;
+ word-wrap: break-word;
+ font-style: normal;
+ line-break: auto;
+ font-stretch: 100%;
+ text-rendering: optimizeLegibility;
+ text-decoration: none;
+ text-size-adjust: 100%;
+ letter-spacing: normal;
+ scrollbar-width: 6px;
+ -webkit-font-smoothing: antialiased;
+ -webkit-touch-callout: none;
+ /* prevent callout to copy image, etc when tap to hold */
+ -webkit-text-size-adjust: none;
+ /* prevent webkit from resizing text to fit */
+ scrollbar-face-color: #888;
+ scrollbar-track-color: rgba(255, 255, 255, 0.1);
+}
+
+#ll_fp {
+ font-family: "GameFont", Arial, sans-serif;
+ font-size: 14px;
+ position: fixed;
+ z-index: -1;
+ top: 0;
+ left: 0;
+ opacity: 0.05;
+}
+
+#ll_p {
+ display: grid;
+ position: fixed;
+ z-index: 999;
+ top: 0;
+ left: 0;
+ right: 0;
+ bottom: 0;
+ justify-content: center;
+ justify-items: center;
+ align-items: center;
+ background: #d5d8de;
+ grid-template-rows: 1fr 200px;
+ grid-gap: 40px;
+ padding: 20px;
+ font-size: 14px;
+}
+
+#ll_p * {
+ line-height: 1em;
+}
+
+#ll_loader {
+ display: flex;
+ flex-direction: column;
+ align-items: center;
+ justify-self: end;
+ justify-content: center;
+}
+
+#ll_loader > .ll_text {
+ text-align: center;
+ color: #777a7f;
+ font-family: "GameFont", Arial, sans-serif;
+ font-size: 24px;
+ height: 30px;
+ line-height: 1.2em;
+}
+
+#ll_progressbar {
+ width: 80vw;
+ max-width: 800px;
+ margin-top: 40px;
+ height: 7px;
+ border-radius: 20px;
+ background: rgba(0, 10, 20, 0.08);
+
+ /* border: 5px solid transparent; */
+ display: flex;
+ position: relative;
+ align-items: flex-start;
+}
+
+@keyframes LL_LoadingAnimation {
+ 50% {
+ background-color: #34ae67;
+ }
+}
+
+#ll_progressbar > span {
+ border-radius: 20px;
+ position: absolute;
+ height: 190%;
+ width: 5%;
+ background: #fff;
+ transform: translateY(-50%);
+ top: 50%;
+ display: inline-flex;
+ background-color: #269fba;
+ animation: LL_LoadingAnimation 4s ease-in-out infinite;
+ position: relative;
+ z-index: 10;
+ border: 4px solid #d5d8de;
+ /* box-shadow: 0 2px 10px rgba(0, 0, 0, 0.2); */
+ transition: width 0.5s ease-in-out;
+ min-width: 4%;
+}
+
+#ll_progressbar > #ll_loadinglabel {
+ position: absolute;
+ z-index: 20;
+ top: 50%;
+ text-transform: uppercase;
+ border-radius: 7px;
+ left: 50%;
+ transform: translate(-50%, -50%);
+ font-size: 16px;
+ color: #33373f;
+}
+
+@keyframes ShowStandaloneBannerAfterDelay {
+ 0% {
+ opacity: 0;
+ }
+ 95% {
+ opacity: 0;
+ }
+ 100% {
+ opacity: 1;
+ }
+}
+
+#ll_standalone {
+ text-align: center;
+ color: #777a7f;
+ margin-top: 30px;
+ display: block;
+ font-size: 16px;
+ animation: ShowStandaloneBannerAfterDelay 60s linear;
+}
+
+#ll_standalone a {
+ color: #39f;
+ margin-left: 5px;
+ font-weight: bold;
+}
+
+#ll_logo {
+}
+
+#ll_logo > img {
+ width: 40vw;
+ max-width: 700px;
+ min-width: 150px;
+}
+
+#ll_loader > .ll_spinner {
+ width: 80px;
+ height: 80px;
+ display: inline-flex;
+ background: center center / contain no-repeat;
+ display: none;
+}
+
+#ll_preload_status {
+ position: absolute;
+ top: 40px;
+ left: 50%;
+ transform: translate(-50%, -50%);
+ z-index: 100;
+ opacity: 1 !important;
+ font-size: 18px;
+ color: rgba(0, 10, 20, 0.5);
+
+ font-family: "GameFont", Arial, sans-serif;
+ text-transform: uppercase;
+ text-align: center;
+}
+
+#ll_preload_error {
+ position: fixed;
+ top: 0;
+ right: 0;
+ bottom: 0;
+ left: 0;
+ z-index: 999999;
+ background: #d5d8de;
+ display: flex;
+ justify-content: center;
+ align-items: center;
+}
+
+#ll_preload_error > .inner {
+ color: #fff;
+ font-family: Arial, "sans-serif";
+ font-size: 15px;
+ padding: 0;
+ text-align: center;
+}
+
+#ll_preload_error > .inner > .heading {
+ color: #ef5072;
+ margin-bottom: 40px;
+ font-size: 45px;
+}
+
+#ll_preload_error > .inner > .content {
+ color: #55585f;
+ font-family: monospace;
+ text-align: left;
+ background-color: #fff;
+ padding: 20px;
+ border-radius: 10px;
+
+ box-shadow: 0 5px 10px rgba(0, 0, 0, 0.1);
+}
+
+#ll_preload_error > .inner .discordLink {
+ color: #333;
+ margin-top: 20px;
+ margin-bottom: 20px;
+ font-family: Arial;
+}
+
+#ll_preload_error > .inner .discordLink a {
+ color: #39f;
+}
+#ll_preload_error > .inner .discordLink strong {
+ font-weight: 900 !important;
+}
+
+#ll_preload_error > .inner .source {
+ color: #777;
+}
diff --git a/shapez-io/electron_wegame/preloader/preloader.html b/shapez-io/electron_wegame/preloader/preloader.html
new file mode 100644
index 00000000..48d3579c
--- /dev/null
+++ b/shapez-io/electron_wegame/preloader/preloader.html
@@ -0,0 +1,21 @@
+
_
+
+
+
+
+
+
diff --git a/shapez-io/electron_wegame/preloader/preloader.js b/shapez-io/electron_wegame/preloader/preloader.js
new file mode 100644
index 00000000..7e2fe518
--- /dev/null
+++ b/shapez-io/electron_wegame/preloader/preloader.js
@@ -0,0 +1,165 @@
+(function () {
+ var loadTimeout = null;
+ var callbackDone = false;
+
+ var searchString = window.location.search;
+ if (searchString.includes("steam_sso_auth_token=")) {
+ var pos = searchString.indexOf("steam_sso_auth_token");
+ const authToken = searchString.substring(pos + 21, pos + 57);
+ try {
+ window.localStorage.setItem("steam_sso_auth_token", authToken);
+ window.location.replace(window.location.protocol + "//" + window.location.host);
+ } catch (ex) {
+ alert("Failed to login via Steam SSO: " + ex);
+ window.location.replace("https://shapez.io");
+ }
+ return;
+ }
+
+ // Catch load errors
+
+ function errorHandler(event, source, lineno, colno, error) {
+ if (("" + event).indexOf("Script error.") >= 0) {
+ console.warn("Thirdparty script error:", event);
+ return;
+ }
+
+ if (("" + event).indexOf("NS_ERROR_FAILURE") >= 0) {
+ console.warn("Firefox NS_ERROR_FAILURE error:", event);
+ return;
+ }
+
+ if (("" + event).indexOf("Cannot read property 'postMessage' of null") >= 0) {
+ console.warn("Safari can not read post message error:", event);
+ return;
+ }
+
+ if (("" + event).indexOf("Possible side-effect in debug-evaluate") >= 0) {
+ console.warn("Chrome debug-evaluate error:", event);
+ return;
+ }
+
+ if (("" + source).indexOf("shapez.io") < 0) {
+ console.warn("Thirdparty error:", event);
+ return;
+ }
+
+ console.error("👀 App Error:", event, source, lineno, colno, error);
+ var element = document.createElement("div");
+ element.id = "ll_preload_error";
+
+ var inner = document.createElement("div");
+ inner.classList.add("inner");
+ element.appendChild(inner);
+
+ var heading = document.createElement("h3");
+ heading.classList.add("heading");
+ heading.innerText = "Fatal Error";
+ inner.appendChild(heading);
+
+ var content = document.createElement("p");
+ content.classList.add("content");
+ content.innerText = error || (event && event.message) || event || "Unknown Error";
+ inner.appendChild(content);
+
+ var discordLink = document.createElement("p");
+ discordLink.classList.add("discordLink");
+ discordLink.innerHTML =
+ "Please report this error in the #bugs channel of the official discord!";
+
+ inner.appendChild(discordLink);
+
+ if (source) {
+ var sourceElement = document.createElement("p");
+ sourceElement.classList.add("source");
+ sourceElement.innerText = source + ":" + lineno + ":" + colno;
+ inner.appendChild(sourceElement);
+ }
+
+ document.documentElement.appendChild(element);
+
+ window.APP_ERROR_OCCURED = true;
+ }
+
+ window.onerror = errorHandler;
+
+ function expectJsParsed() {
+ if (!callbackDone) {
+ console.error("👀 Got no core callback");
+ throw new Error("Core thread failed to respond within time.");
+ }
+ }
+
+ function onJsLoaded() {
+ console.log("👀 Core loaded at", Math.floor(performance.now()), "ms");
+ loadTimeout = setTimeout(expectJsParsed, 120000);
+ window.removeEventListener("unhandledrejection", errorHandler);
+ }
+
+ window.coreThreadLoadedCb = function () {
+ console.log("👀 Core responded at", Math.floor(performance.now()), "ms");
+ clearTimeout(loadTimeout);
+ loadTimeout = null;
+ callbackDone = true;
+ };
+
+ function progressHandler(progress) {
+ var progressElement = document.querySelector("#ll_preload_status");
+ if (progressElement) {
+ progressElement.innerText = "Downloading Bundle (" + Math.round(progress * 1200) + " / 1200 KB)";
+ }
+ var barElement = document.querySelector("#ll_progressbar span");
+ if (barElement) {
+ barElement.style.width = (5 + progress * 75.0).toFixed(2) + "%";
+ }
+ }
+
+ function startBundleDownload() {
+ var xhr = new XMLHttpRequest();
+ var notifiedNotComputable = false;
+
+ xhr.open("GET", bundleSrc, true);
+ xhr.responseType = "arraybuffer";
+ xhr.onprogress = function (ev) {
+ if (ev.lengthComputable) {
+ progressHandler(ev.loaded / ev.total);
+ } else {
+ // Hardcoded length
+ progressHandler(Math.min(1, ev.loaded / 2349009));
+ }
+ };
+
+ xhr.onloadend = function () {
+ if (!xhr.status.toString().match(/^2/)) {
+ throw new Error("Failed to load bundle: " + xhr.status + " " + xhr.statusText);
+ } else {
+ if (!notifiedNotComputable) {
+ progressHandler(1);
+ }
+
+ var options = {};
+ var headers = xhr.getAllResponseHeaders();
+ var m = headers.match(/^Content-Type\:\s*(.*?)$/im);
+
+ if (m && m[1]) {
+ options.type = m[1];
+ }
+
+ var blob = new Blob([this.response], options);
+ var script = document.createElement("script");
+ script.addEventListener("load", onJsLoaded);
+ script.src = window.URL.createObjectURL(blob);
+ script.type = "text/javascript";
+ script.charset = "utf-8";
+ if (bundleIntegrity) {
+ script.setAttribute("integrity", bundleIntegrity);
+ }
+ document.head.appendChild(script);
+ }
+ };
+ xhr.send();
+ }
+
+ console.log("Start bundle download ...");
+ window.addEventListener("load", startBundleDownload);
+})();
diff --git a/shapez-io/electron_wegame/steampipe/templates/app-darwin-demo.vdf b/shapez-io/electron_wegame/steampipe/templates/app-darwin-demo.vdf
new file mode 100644
index 00000000..4bdb46b1
--- /dev/null
+++ b/shapez-io/electron_wegame/steampipe/templates/app-darwin-demo.vdf
@@ -0,0 +1,14 @@
+"appbuild"
+{
+ "appid" "1930750"
+ "desc" "$BUILD_DESC$"
+ "buildoutput" "$TMP_DIR$"
+ "contentroot" ""
+ "setlive" ""
+ "preview" "0"
+ "local" ""
+ "depots"
+ {
+ "1930756" "$VDF_DIR$/demo-darwin.vdf"
+ }
+}
diff --git a/shapez-io/electron_wegame/steampipe/templates/app-darwin.vdf b/shapez-io/electron_wegame/steampipe/templates/app-darwin.vdf
new file mode 100644
index 00000000..fa63b846
--- /dev/null
+++ b/shapez-io/electron_wegame/steampipe/templates/app-darwin.vdf
@@ -0,0 +1,14 @@
+"appbuild"
+{
+ "appid" "1318690"
+ "desc" "$BUILD_DESC$"
+ "buildoutput" "$TMP_DIR$"
+ "contentroot" ""
+ "setlive" ""
+ "preview" "0"
+ "local" ""
+ "depots"
+ {
+ "1318693" "$VDF_DIR$/standalone-darwin.vdf"
+ }
+}
diff --git a/shapez-io/electron_wegame/steampipe/templates/app-winlinux-demo.vdf b/shapez-io/electron_wegame/steampipe/templates/app-winlinux-demo.vdf
new file mode 100644
index 00000000..b4859b8b
--- /dev/null
+++ b/shapez-io/electron_wegame/steampipe/templates/app-winlinux-demo.vdf
@@ -0,0 +1,17 @@
+"appbuild"
+{
+ "appid" "1930750"
+ "desc" "$BUILD_DESC$"
+ "buildoutput" "$TMP_DIR$"
+ "contentroot" ""
+ "setlive" ""
+ "preview" "0"
+ "local" ""
+ "depots"
+ {
+ "1930753" "$VDF_DIR$/demo-windows.vdf"
+ "1930754" "$VDF_DIR$/demo-china-windows.vdf"
+ "1930752" "$VDF_DIR$/demo-linux.vdf"
+ "1930755" "$VDF_DIR$/demo-china-linux.vdf"
+ }
+}
diff --git a/shapez-io/electron_wegame/steampipe/templates/app-winlinux.vdf b/shapez-io/electron_wegame/steampipe/templates/app-winlinux.vdf
new file mode 100644
index 00000000..9fd7f9df
--- /dev/null
+++ b/shapez-io/electron_wegame/steampipe/templates/app-winlinux.vdf
@@ -0,0 +1,17 @@
+"appbuild"
+{
+ "appid" "1318690"
+ "desc" "$BUILD_DESC$"
+ "buildoutput" "$TMP_DIR$"
+ "contentroot" ""
+ "setlive" ""
+ "preview" "0"
+ "local" ""
+ "depots"
+ {
+ "1318691" "$VDF_DIR$\standalone-windows.vdf"
+ "1318694" "$VDF_DIR$\standalone-china-windows.vdf"
+ "1318692" "$VDF_DIR$\standalone-linux.vdf"
+ "1318695" "$VDF_DIR$\standalone-china-linux.vdf"
+ }
+}
diff --git a/shapez-io/electron_wegame/steampipe/templates/demo-china-linux.vdf b/shapez-io/electron_wegame/steampipe/templates/demo-china-linux.vdf
new file mode 100644
index 00000000..2ec63419
--- /dev/null
+++ b/shapez-io/electron_wegame/steampipe/templates/demo-china-linux.vdf
@@ -0,0 +1,12 @@
+"DepotBuildConfig"
+{
+ "DepotID" "1930755"
+ "contentroot" "$BUNDLE_DIR$\standalone-steam-china-demo\shapez-linux-x64"
+ "FileMapping"
+ {
+ "LocalPath" "*"
+ "DepotPath" "."
+ "recursive" "1"
+ }
+ "FileExclusion" "*.pdb"
+}
\ No newline at end of file
diff --git a/shapez-io/electron_wegame/steampipe/templates/demo-china-windows.vdf b/shapez-io/electron_wegame/steampipe/templates/demo-china-windows.vdf
new file mode 100644
index 00000000..a06b4e9e
--- /dev/null
+++ b/shapez-io/electron_wegame/steampipe/templates/demo-china-windows.vdf
@@ -0,0 +1,12 @@
+"DepotBuildConfig"
+{
+ "DepotID" "1930754"
+ "contentroot" "$BUNDLE_DIR$\standalone-steam-china-demo\shapez-win32-x64"
+ "FileMapping"
+ {
+ "LocalPath" "*"
+ "DepotPath" "."
+ "recursive" "1"
+ }
+ "FileExclusion" "*.pdb"
+}
\ No newline at end of file
diff --git a/shapez-io/electron_wegame/steampipe/templates/demo-darwin.vdf b/shapez-io/electron_wegame/steampipe/templates/demo-darwin.vdf
new file mode 100644
index 00000000..d0e8f1e2
--- /dev/null
+++ b/shapez-io/electron_wegame/steampipe/templates/demo-darwin.vdf
@@ -0,0 +1,12 @@
+"DepotBuildConfig"
+{
+ "DepotID" "1930756"
+ "contentroot" "$BUNDLE_DIR$\standalone-steam-demo\shapez-darwin-x64"
+ "FileMapping"
+ {
+ "LocalPath" "*"
+ "DepotPath" "."
+ "recursive" "1"
+ }
+ "FileExclusion" "*.pdb"
+}
\ No newline at end of file
diff --git a/shapez-io/electron_wegame/steampipe/templates/demo-linux.vdf b/shapez-io/electron_wegame/steampipe/templates/demo-linux.vdf
new file mode 100644
index 00000000..4f2d274f
--- /dev/null
+++ b/shapez-io/electron_wegame/steampipe/templates/demo-linux.vdf
@@ -0,0 +1,12 @@
+"DepotBuildConfig"
+{
+ "DepotID" "1930752"
+ "contentroot" "$BUNDLE_DIR$\standalone-steam-demo\shapez-linux-x64"
+ "FileMapping"
+ {
+ "LocalPath" "*"
+ "DepotPath" "."
+ "recursive" "1"
+ }
+ "FileExclusion" "*.pdb"
+}
\ No newline at end of file
diff --git a/shapez-io/electron_wegame/steampipe/templates/demo-windows.vdf b/shapez-io/electron_wegame/steampipe/templates/demo-windows.vdf
new file mode 100644
index 00000000..1b6cdbc7
--- /dev/null
+++ b/shapez-io/electron_wegame/steampipe/templates/demo-windows.vdf
@@ -0,0 +1,12 @@
+"DepotBuildConfig"
+{
+ "DepotID" "1930753"
+ "contentroot" "$BUNDLE_DIR$\standalone-steam-demo\shapez-win32-x64"
+ "FileMapping"
+ {
+ "LocalPath" "*"
+ "DepotPath" "."
+ "recursive" "1"
+ }
+ "FileExclusion" "*.pdb"
+}
\ No newline at end of file
diff --git a/shapez-io/electron_wegame/steampipe/templates/standalone-china-linux.vdf b/shapez-io/electron_wegame/steampipe/templates/standalone-china-linux.vdf
new file mode 100644
index 00000000..56b9fe3c
--- /dev/null
+++ b/shapez-io/electron_wegame/steampipe/templates/standalone-china-linux.vdf
@@ -0,0 +1,12 @@
+"DepotBuildConfig"
+{
+ "DepotID" "1318695"
+ "contentroot" "$BUNDLE_DIR$\standalone-steam-china\shapez-linux-x64"
+ "FileMapping"
+ {
+ "LocalPath" "*"
+ "DepotPath" "."
+ "recursive" "1"
+ }
+ "FileExclusion" "*.pdb"
+}
\ No newline at end of file
diff --git a/shapez-io/electron_wegame/steampipe/templates/standalone-china-windows.vdf b/shapez-io/electron_wegame/steampipe/templates/standalone-china-windows.vdf
new file mode 100644
index 00000000..469158db
--- /dev/null
+++ b/shapez-io/electron_wegame/steampipe/templates/standalone-china-windows.vdf
@@ -0,0 +1,12 @@
+"DepotBuildConfig"
+{
+ "DepotID" "1318694"
+ "contentroot" "$BUNDLE_DIR$\standalone-steam-china\shapez-win32-x64"
+ "FileMapping"
+ {
+ "LocalPath" "*"
+ "DepotPath" "."
+ "recursive" "1"
+ }
+ "FileExclusion" "*.pdb"
+}
\ No newline at end of file
diff --git a/shapez-io/electron_wegame/steampipe/templates/standalone-darwin.vdf b/shapez-io/electron_wegame/steampipe/templates/standalone-darwin.vdf
new file mode 100644
index 00000000..026ab768
--- /dev/null
+++ b/shapez-io/electron_wegame/steampipe/templates/standalone-darwin.vdf
@@ -0,0 +1,12 @@
+"DepotBuildConfig"
+{
+ "DepotID" "1318693"
+ "contentroot" "$BUNDLE_DIR$\standalone-steam\shapez-darwin-x64"
+ "FileMapping"
+ {
+ "LocalPath" "*"
+ "DepotPath" "."
+ "recursive" "1"
+ }
+ "FileExclusion" "*.pdb"
+}
\ No newline at end of file
diff --git a/shapez-io/electron_wegame/steampipe/templates/standalone-linux.vdf b/shapez-io/electron_wegame/steampipe/templates/standalone-linux.vdf
new file mode 100644
index 00000000..9edb1963
--- /dev/null
+++ b/shapez-io/electron_wegame/steampipe/templates/standalone-linux.vdf
@@ -0,0 +1,12 @@
+"DepotBuildConfig"
+{
+ "DepotID" "1318692"
+ "contentroot" "$BUNDLE_DIR$\standalone-steam\shapez-linux-x64"
+ "FileMapping"
+ {
+ "LocalPath" "*"
+ "DepotPath" "."
+ "recursive" "1"
+ }
+ "FileExclusion" "*.pdb"
+}
\ No newline at end of file
diff --git a/shapez-io/electron_wegame/steampipe/templates/standalone-windows.vdf b/shapez-io/electron_wegame/steampipe/templates/standalone-windows.vdf
new file mode 100644
index 00000000..6f7cb408
--- /dev/null
+++ b/shapez-io/electron_wegame/steampipe/templates/standalone-windows.vdf
@@ -0,0 +1,12 @@
+"DepotBuildConfig"
+{
+ "DepotID" "1318691"
+ "contentroot" "$BUNDLE_DIR$\standalone-steam\shapez-win32-x64"
+ "FileMapping"
+ {
+ "LocalPath" "*"
+ "DepotPath" "."
+ "recursive" "1"
+ }
+ "FileExclusion" "*.pdb"
+}
\ No newline at end of file
diff --git a/shapez-io/electron_wegame/steampipe/upload-darwin-demo.sh b/shapez-io/electron_wegame/steampipe/upload-darwin-demo.sh
new file mode 100644
index 00000000..77bb29dc
--- /dev/null
+++ b/shapez-io/electron_wegame/steampipe/upload-darwin-demo.sh
@@ -0,0 +1,3 @@
+#!/bin/sh
+yarn gulp standalone.prepareVDF
+steamcmd.sh +login $STEAM_UPLOAD_SHAPEZ_ID $STEAM_UPLOAD_SHAPEZ_USER +run_app_build $PWD/built_vdfs/app-darwin-demo.vdf +quit
diff --git a/shapez-io/electron_wegame/steampipe/upload-darwin.sh b/shapez-io/electron_wegame/steampipe/upload-darwin.sh
new file mode 100644
index 00000000..06412dcd
--- /dev/null
+++ b/shapez-io/electron_wegame/steampipe/upload-darwin.sh
@@ -0,0 +1,3 @@
+#!/bin/sh
+yarn gulp standalone.prepareVDF
+steamcmd.sh +login $STEAM_UPLOAD_SHAPEZ_ID $STEAM_UPLOAD_SHAPEZ_USER +run_app_build $PWD/built_vdfs/app-darwin.vdf +quit
diff --git a/shapez-io/electron_wegame/steampipe/upload-winlinux-demo.bat b/shapez-io/electron_wegame/steampipe/upload-winlinux-demo.bat
new file mode 100644
index 00000000..e19f7f55
--- /dev/null
+++ b/shapez-io/electron_wegame/steampipe/upload-winlinux-demo.bat
@@ -0,0 +1,3 @@
+@echo off
+cmd /c yarn gulp standalone.prepareVDF
+steamcmd +login %STEAM_UPLOAD_SHAPEZ_ID% %STEAM_UPLOAD_SHAPEZ_USER% +run_app_build %cd%/built_vdfs/app-winlinux-demo.vdf +quit
diff --git a/shapez-io/electron_wegame/steampipe/upload-winlinux.bat b/shapez-io/electron_wegame/steampipe/upload-winlinux.bat
new file mode 100644
index 00000000..1c9bdfe7
--- /dev/null
+++ b/shapez-io/electron_wegame/steampipe/upload-winlinux.bat
@@ -0,0 +1,3 @@
+@echo off
+cmd /c yarn gulp standalone.prepareVDF
+steamcmd +login %STEAM_UPLOAD_SHAPEZ_ID% %STEAM_UPLOAD_SHAPEZ_USER% +run_app_build %cd%/built_vdfs/app-winlinux.vdf +quit
diff --git a/shapez-io/electron_wegame/wegame.js b/shapez-io/electron_wegame/wegame.js
new file mode 100644
index 00000000..05a0e186
--- /dev/null
+++ b/shapez-io/electron_wegame/wegame.js
@@ -0,0 +1,63 @@
+const railsdk = require("./wegame_sdk/railsdk.js");
+const { dialog, app, remote, ipcMain } = require("electron");
+
+function init(isDev) {
+ console.log("Step 1: wegame: init");
+
+ try {
+ console.log("Step 2: Calling need restart app");
+ const need_restart = railsdk.RailNeedRestartAppForCheckingEnvironment(
+ 2001639,
+ [`--rail_render_pid=${process.pid}`] //,"--rail_debug_mode",
+ );
+ console.log("Step 3: Needs restart =", need_restart);
+ if (need_restart) {
+ console.error("Step 4: Need restart");
+ dialog.showErrorBox("加载RailSDK失败", "请先运行WeGame开发者版本");
+ return;
+ }
+ } catch (err) {
+ console.error("Rail SDK error:", err);
+ dialog.showErrorBox("加载RailSDK失败", err);
+ return;
+ }
+
+ console.log("Step 5: starting rail sdk");
+ if (railsdk.RailInitialize() === false) {
+ console.error("RailInitialize() = false");
+ dialog.showErrorBox("RailInitialize调用失败", "请先运行WeGame开发者版本");
+ return;
+ }
+
+ console.log("Initialize RailSDK success!");
+
+ railsdk.RailRegisterEvent(railsdk.RailEventID.kRailEventSystemStateChanged, event => {
+ console.log(event);
+ if (event.result === railsdk.RailResult.kSuccess) {
+ if (
+ event.state === railsdk.RailSystemState.kSystemStatePlatformOffline ||
+ event.state === railsdk.RailSystemState.kSystemStatePlatformExit ||
+ event.state === railsdk.RailSystemState.kSystemStateGameExitByAntiAddiction
+ ) {
+ app.exit();
+ }
+ }
+ });
+}
+
+function listen() {
+ console.log("wegame: listen");
+ ipcMain.handle("profanity-check", async (event, data) => {
+ if (data.length === 0) {
+ return "";
+ }
+ const result = railsdk.RailUtils.DirtyWordsFilter(data, true);
+ if (result.check_result.dirty_type !== 0 /** kRailDirtyWordsTypeNormalAllowWords */) {
+ return result.check_result.replace_string;
+ }
+
+ return data;
+ });
+}
+
+module.exports = { init, listen };
diff --git a/shapez-io/electron_wegame/yarn.lock b/shapez-io/electron_wegame/yarn.lock
new file mode 100644
index 00000000..69c595ea
--- /dev/null
+++ b/shapez-io/electron_wegame/yarn.lock
@@ -0,0 +1,578 @@
+# THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY.
+# yarn lockfile v1
+
+
+"@electron/get@^1.0.1":
+ version "1.12.4"
+ resolved "https://registry.yarnpkg.com/@electron/get/-/get-1.12.4.tgz#a5971113fc1bf8fa12a8789dc20152a7359f06ab"
+ integrity sha512-6nr9DbJPUR9Xujw6zD3y+rS95TyItEVM0NVjt1EehY2vUWfIgPiIPVHxCvaTS0xr2B+DRxovYVKbuOWqC35kjg==
+ dependencies:
+ debug "^4.1.1"
+ env-paths "^2.2.0"
+ fs-extra "^8.1.0"
+ got "^9.6.0"
+ progress "^2.0.3"
+ semver "^6.2.0"
+ sumchecker "^3.0.1"
+ optionalDependencies:
+ global-agent "^2.0.2"
+ global-tunnel-ng "^2.7.1"
+
+"@sindresorhus/is@^0.14.0":
+ version "0.14.0"
+ resolved "https://registry.yarnpkg.com/@sindresorhus/is/-/is-0.14.0.tgz#9fb3a3cf3132328151f353de4632e01e52102bea"
+ integrity sha512-9NET910DNaIPngYnLLPeg+Ogzqsi9uM4mSboU5y6p8S5DzMTVEsJZrawi+BoDNUVBa2DhJqQYUFvMDfgU062LQ==
+
+"@szmarczak/http-timer@^1.1.2":
+ version "1.1.2"
+ resolved "https://registry.yarnpkg.com/@szmarczak/http-timer/-/http-timer-1.1.2.tgz#b1665e2c461a2cd92f4c1bbf50d5454de0d4b421"
+ integrity sha512-XIB2XbzHTN6ieIjfIMV9hlVcfPU26s2vafYWQcZHWXHOxiaRZYEDKEwdl129Zyg50+foYV2jCgtrqSA6qNuNSA==
+ dependencies:
+ defer-to-connect "^1.0.1"
+
+"@types/node@^14.6.2":
+ version "14.17.4"
+ resolved "https://registry.yarnpkg.com/@types/node/-/node-14.17.4.tgz#218712242446fc868d0e007af29a4408c7765bc0"
+ integrity sha512-8kQ3+wKGRNN0ghtEn7EGps/B8CzuBz1nXZEIGGLP2GnwbqYn4dbTs7k+VKLTq1HvZLRCIDtN3Snx1Ege8B7L5A==
+
+async-lock@^1.2.8:
+ version "1.2.8"
+ resolved "https://registry.yarnpkg.com/async-lock/-/async-lock-1.2.8.tgz#7b02bdfa2de603c0713acecd11184cf97bbc7c4c"
+ integrity sha512-G+26B2jc0Gw0EG/WN2M6IczuGepBsfR1+DtqLnyFSH4p2C668qkOCtEkGNVEaaNAVlYwEMazy1+/jnLxltBkIQ==
+
+boolean@^3.0.1:
+ version "3.1.2"
+ resolved "https://registry.yarnpkg.com/boolean/-/boolean-3.1.2.tgz#e30f210a26b02458482a8cc353ab06f262a780c2"
+ integrity sha512-YN6UmV0FfLlBVvRvNPx3pz5W/mUoYB24J4WSXOKP/OOJpi+Oq6WYqPaNTHzjI0QzwWtnvEd5CGYyQPgp1jFxnw==
+
+buffer-crc32@~0.2.3:
+ version "0.2.13"
+ resolved "https://registry.yarnpkg.com/buffer-crc32/-/buffer-crc32-0.2.13.tgz#0d333e3f00eac50aa1454abd30ef8c2a5d9a7242"
+ integrity sha1-DTM+PwDqxQqhRUq9MO+MKl2ackI=
+
+buffer-from@^1.0.0:
+ version "1.1.1"
+ resolved "https://registry.yarnpkg.com/buffer-from/-/buffer-from-1.1.1.tgz#32713bc028f75c02fdb710d7c7bcec1f2c6070ef"
+ integrity sha512-MQcXEUbCKtEo7bhqEs6560Hyd4XaovZlO/k9V3hjVUF/zwW7KBVdSK4gIt/bzwS9MbR5qob+F5jusZsb0YQK2A==
+
+cacheable-request@^6.0.0:
+ version "6.1.0"
+ resolved "https://registry.yarnpkg.com/cacheable-request/-/cacheable-request-6.1.0.tgz#20ffb8bd162ba4be11e9567d823db651052ca912"
+ integrity sha512-Oj3cAGPCqOZX7Rz64Uny2GYAZNliQSqfbePrgAQ1wKAihYmCUnraBtJtKcGR4xz7wF+LoJC+ssFZvv5BgF9Igg==
+ dependencies:
+ clone-response "^1.0.2"
+ get-stream "^5.1.0"
+ http-cache-semantics "^4.0.0"
+ keyv "^3.0.0"
+ lowercase-keys "^2.0.0"
+ normalize-url "^4.1.0"
+ responselike "^1.0.2"
+
+clone-response@^1.0.2:
+ version "1.0.2"
+ resolved "https://registry.yarnpkg.com/clone-response/-/clone-response-1.0.2.tgz#d1dc973920314df67fbeb94223b4ee350239e96b"
+ integrity sha1-0dyXOSAxTfZ/vrlCI7TuNQI56Ws=
+ dependencies:
+ mimic-response "^1.0.0"
+
+concat-stream@^1.6.2:
+ version "1.6.2"
+ resolved "https://registry.yarnpkg.com/concat-stream/-/concat-stream-1.6.2.tgz#904bdf194cd3122fc675c77fc4ac3d4ff0fd1a34"
+ integrity sha512-27HBghJxjiZtIk3Ycvn/4kbJk/1uZuJFfuPEns6LaEvpvG1f0hTea8lilrouyo9mVc2GWdcEZ8OLoGmSADlrCw==
+ dependencies:
+ buffer-from "^1.0.0"
+ inherits "^2.0.3"
+ readable-stream "^2.2.2"
+ typedarray "^0.0.6"
+
+config-chain@^1.1.11:
+ version "1.1.13"
+ resolved "https://registry.yarnpkg.com/config-chain/-/config-chain-1.1.13.tgz#fad0795aa6a6cdaff9ed1b68e9dff94372c232f4"
+ integrity sha512-qj+f8APARXHrM0hraqXYb2/bOVSV4PvJQlNZ/DVj0QrmNM2q2euizkeuVckQ57J+W0mRH6Hvi+k50M4Jul2VRQ==
+ dependencies:
+ ini "^1.3.4"
+ proto-list "~1.2.1"
+
+core-js@^3.6.5:
+ version "3.15.2"
+ resolved "https://registry.yarnpkg.com/core-js/-/core-js-3.15.2.tgz#740660d2ff55ef34ce664d7e2455119c5bdd3d61"
+ integrity sha512-tKs41J7NJVuaya8DxIOCnl8QuPHx5/ZVbFo1oKgVl1qHFBBrDctzQGtuLjPpRdNTWmKPH6oEvgN/MUID+l485Q==
+
+core-util-is@~1.0.0:
+ version "1.0.2"
+ resolved "https://registry.yarnpkg.com/core-util-is/-/core-util-is-1.0.2.tgz#b5fd54220aa2bc5ab57aab7140c940754503c1a7"
+ integrity sha1-tf1UIgqivFq1eqtxQMlAdUUDwac=
+
+debug@^2.6.9:
+ version "2.6.9"
+ resolved "https://registry.yarnpkg.com/debug/-/debug-2.6.9.tgz#5d128515df134ff327e90a4c93f4e077a536341f"
+ integrity sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==
+ dependencies:
+ ms "2.0.0"
+
+debug@^4.1.0, debug@^4.1.1:
+ version "4.3.2"
+ resolved "https://registry.yarnpkg.com/debug/-/debug-4.3.2.tgz#f0a49c18ac8779e31d4a0c6029dfb76873c7428b"
+ integrity sha512-mOp8wKcvj7XxC78zLgw/ZA+6TSgkoE2C/ienthhRD298T7UNwAg9diBpLRxC0mOezLl4B0xV7M0cCO6P/O0Xhw==
+ dependencies:
+ ms "2.1.2"
+
+decompress-response@^3.3.0:
+ version "3.3.0"
+ resolved "https://registry.yarnpkg.com/decompress-response/-/decompress-response-3.3.0.tgz#80a4dd323748384bfa248083622aedec982adff3"
+ integrity sha1-gKTdMjdIOEv6JICDYirt7Jgq3/M=
+ dependencies:
+ mimic-response "^1.0.0"
+
+defer-to-connect@^1.0.1:
+ version "1.1.3"
+ resolved "https://registry.yarnpkg.com/defer-to-connect/-/defer-to-connect-1.1.3.tgz#331ae050c08dcf789f8c83a7b81f0ed94f4ac591"
+ integrity sha512-0ISdNousHvZT2EiFlZeZAHBUvSxmKswVCEf8hW7KWgG4a8MVEu/3Vb6uWYozkjylyCxe0JBIiRB1jV45S70WVQ==
+
+define-properties@^1.1.3:
+ version "1.1.3"
+ resolved "https://registry.yarnpkg.com/define-properties/-/define-properties-1.1.3.tgz#cf88da6cbee26fe6db7094f61d870cbd84cee9f1"
+ integrity sha512-3MqfYKj2lLzdMSf8ZIZE/V+Zuy+BgD6f164e8K2w7dgnpKArBDerGYpM46IYYcjnkdPNMjPk9A6VFB8+3SKlXQ==
+ dependencies:
+ object-keys "^1.0.12"
+
+detect-node@^2.0.4:
+ version "2.1.0"
+ resolved "https://registry.yarnpkg.com/detect-node/-/detect-node-2.1.0.tgz#c9c70775a49c3d03bc2c06d9a73be550f978f8b1"
+ integrity sha512-T0NIuQpnTvFDATNuHN5roPwSBG83rFsuO+MXXH9/3N1eFbn4wcPjttvjMLEPWJ0RGUYgQE7cGgS3tNxbqCGM7g==
+
+duplexer3@^0.1.4:
+ version "0.1.4"
+ resolved "https://registry.yarnpkg.com/duplexer3/-/duplexer3-0.1.4.tgz#ee01dd1cac0ed3cbc7fdbea37dc0a8f1ce002ce2"
+ integrity sha1-7gHdHKwO08vH/b6jfcCo8c4ALOI=
+
+electron@^13.1.6:
+ version "13.1.6"
+ resolved "https://registry.yarnpkg.com/electron/-/electron-13.1.6.tgz#6ecaf969255d62ce82cc0b5c948bf26e7dfb489b"
+ integrity sha512-XiB55/JTaQpDFQrD9pulYnOGwaWeMyRIub5ispvoE2bWBvM5zVMLptwMLb0m3KTMrfSkzhedZvOu7fwYvR7L7Q==
+ dependencies:
+ "@electron/get" "^1.0.1"
+ "@types/node" "^14.6.2"
+ extract-zip "^1.0.3"
+
+encodeurl@^1.0.2:
+ version "1.0.2"
+ resolved "https://registry.yarnpkg.com/encodeurl/-/encodeurl-1.0.2.tgz#ad3ff4c86ec2d029322f5a02c3a9a606c95b3f59"
+ integrity sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=
+
+end-of-stream@^1.1.0:
+ version "1.4.4"
+ resolved "https://registry.yarnpkg.com/end-of-stream/-/end-of-stream-1.4.4.tgz#5ae64a5f45057baf3626ec14da0ca5e4b2431eb0"
+ integrity sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==
+ dependencies:
+ once "^1.4.0"
+
+env-paths@^2.2.0:
+ version "2.2.1"
+ resolved "https://registry.yarnpkg.com/env-paths/-/env-paths-2.2.1.tgz#420399d416ce1fbe9bc0a07c62fa68d67fd0f8f2"
+ integrity sha512-+h1lkLKhZMTYjog1VEpJNG7NZJWcuc2DDk/qsqSTRRCOXiLjeQ1d1/udrUGhqMxUgAlwKNZ0cf2uqan5GLuS2A==
+
+es6-error@^4.1.1:
+ version "4.1.1"
+ resolved "https://registry.yarnpkg.com/es6-error/-/es6-error-4.1.1.tgz#9e3af407459deed47e9a91f9b885a84eb05c561d"
+ integrity sha512-Um/+FxMr9CISWh0bi5Zv0iOD+4cFh5qLeks1qhAopKVAJw3drgKbKySikp7wGhDL0HPeaja0P5ULZrxLkniUVg==
+
+escape-string-regexp@^4.0.0:
+ version "4.0.0"
+ resolved "https://registry.yarnpkg.com/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz#14ba83a5d373e3d311e5afca29cf5bfad965bf34"
+ integrity sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==
+
+extract-zip@^1.0.3:
+ version "1.7.0"
+ resolved "https://registry.yarnpkg.com/extract-zip/-/extract-zip-1.7.0.tgz#556cc3ae9df7f452c493a0cfb51cc30277940927"
+ integrity sha512-xoh5G1W/PB0/27lXgMQyIhP5DSY/LhoCsOyZgb+6iMmRtCwVBo55uKaMoEYrDCKQhWvqEip5ZPKAc6eFNyf/MA==
+ dependencies:
+ concat-stream "^1.6.2"
+ debug "^2.6.9"
+ mkdirp "^0.5.4"
+ yauzl "^2.10.0"
+
+fd-slicer@~1.1.0:
+ version "1.1.0"
+ resolved "https://registry.yarnpkg.com/fd-slicer/-/fd-slicer-1.1.0.tgz#25c7c89cb1f9077f8891bbe61d8f390eae256f1e"
+ integrity sha1-JcfInLH5B3+IkbvmHY85Dq4lbx4=
+ dependencies:
+ pend "~1.2.0"
+
+fs-extra@^8.1.0:
+ version "8.1.0"
+ resolved "https://registry.yarnpkg.com/fs-extra/-/fs-extra-8.1.0.tgz#49d43c45a88cd9677668cb7be1b46efdb8d2e1c0"
+ integrity sha512-yhlQgA6mnOJUKOsRUFsgJdQCvkKhcz8tlZG5HBQfReYZy46OwLcY+Zia0mtdHsOo9y/hP+CxMN0TU9QxoOtG4g==
+ dependencies:
+ graceful-fs "^4.2.0"
+ jsonfile "^4.0.0"
+ universalify "^0.1.0"
+
+get-stream@^4.1.0:
+ version "4.1.0"
+ resolved "https://registry.yarnpkg.com/get-stream/-/get-stream-4.1.0.tgz#c1b255575f3dc21d59bfc79cd3d2b46b1c3a54b5"
+ integrity sha512-GMat4EJ5161kIy2HevLlr4luNjBgvmj413KaQA7jt4V8B4RDsfpHk7WQ9GVqfYyyx8OS/L66Kox+rJRNklLK7w==
+ dependencies:
+ pump "^3.0.0"
+
+get-stream@^5.1.0:
+ version "5.2.0"
+ resolved "https://registry.yarnpkg.com/get-stream/-/get-stream-5.2.0.tgz#4966a1795ee5ace65e706c4b7beb71257d6e22d3"
+ integrity sha512-nBF+F1rAZVCu/p7rjzgA+Yb4lfYXrpl7a6VmJrU8wF9I1CKvP/QwPNZHnOlwbTkY6dvtFIzFMSyQXbLoTQPRpA==
+ dependencies:
+ pump "^3.0.0"
+
+global-agent@^2.0.2:
+ version "2.2.0"
+ resolved "https://registry.yarnpkg.com/global-agent/-/global-agent-2.2.0.tgz#566331b0646e6bf79429a16877685c4a1fbf76dc"
+ integrity sha512-+20KpaW6DDLqhG7JDiJpD1JvNvb8ts+TNl7BPOYcURqCrXqnN1Vf+XVOrkKJAFPqfX+oEhsdzOj1hLWkBTdNJg==
+ dependencies:
+ boolean "^3.0.1"
+ core-js "^3.6.5"
+ es6-error "^4.1.1"
+ matcher "^3.0.0"
+ roarr "^2.15.3"
+ semver "^7.3.2"
+ serialize-error "^7.0.1"
+
+global-tunnel-ng@^2.7.1:
+ version "2.7.1"
+ resolved "https://registry.yarnpkg.com/global-tunnel-ng/-/global-tunnel-ng-2.7.1.tgz#d03b5102dfde3a69914f5ee7d86761ca35d57d8f"
+ integrity sha512-4s+DyciWBV0eK148wqXxcmVAbFVPqtc3sEtUE/GTQfuU80rySLcMhUmHKSHI7/LDj8q0gDYI1lIhRRB7ieRAqg==
+ dependencies:
+ encodeurl "^1.0.2"
+ lodash "^4.17.10"
+ npm-conf "^1.1.3"
+ tunnel "^0.0.6"
+
+globalthis@^1.0.1:
+ version "1.0.2"
+ resolved "https://registry.yarnpkg.com/globalthis/-/globalthis-1.0.2.tgz#2a235d34f4d8036219f7e34929b5de9e18166b8b"
+ integrity sha512-ZQnSFO1la8P7auIOQECnm0sSuoMeaSq0EEdXMBFF2QJO4uNcwbyhSgG3MruWNbFTqCLmxVwGOl7LZ9kASvHdeQ==
+ dependencies:
+ define-properties "^1.1.3"
+
+got@^9.6.0:
+ version "9.6.0"
+ resolved "https://registry.yarnpkg.com/got/-/got-9.6.0.tgz#edf45e7d67f99545705de1f7bbeeeb121765ed85"
+ integrity sha512-R7eWptXuGYxwijs0eV+v3o6+XH1IqVK8dJOEecQfTmkncw9AV4dcw/Dhxi8MdlqPthxxpZyizMzyg8RTmEsG+Q==
+ dependencies:
+ "@sindresorhus/is" "^0.14.0"
+ "@szmarczak/http-timer" "^1.1.2"
+ cacheable-request "^6.0.0"
+ decompress-response "^3.3.0"
+ duplexer3 "^0.1.4"
+ get-stream "^4.1.0"
+ lowercase-keys "^1.0.1"
+ mimic-response "^1.0.1"
+ p-cancelable "^1.0.0"
+ to-readable-stream "^1.0.0"
+ url-parse-lax "^3.0.0"
+
+graceful-fs@^4.1.6, graceful-fs@^4.2.0:
+ version "4.2.6"
+ resolved "https://registry.yarnpkg.com/graceful-fs/-/graceful-fs-4.2.6.tgz#ff040b2b0853b23c3d31027523706f1885d76bee"
+ integrity sha512-nTnJ528pbqxYanhpDYsi4Rd8MAeaBA67+RZ10CM1m3bTAVFEDcd5AuA4a6W5YkGZ1iNXHzZz8T6TBKLeBuNriQ==
+
+http-cache-semantics@^4.0.0:
+ version "4.1.0"
+ resolved "https://registry.yarnpkg.com/http-cache-semantics/-/http-cache-semantics-4.1.0.tgz#49e91c5cbf36c9b94bcfcd71c23d5249ec74e390"
+ integrity sha512-carPklcUh7ROWRK7Cv27RPtdhYhUsela/ue5/jKzjegVvXDqM2ILE9Q2BGn9JZJh1g87cp56su/FgQSzcWS8cQ==
+
+inherits@^2.0.3, inherits@~2.0.3:
+ version "2.0.4"
+ resolved "https://registry.yarnpkg.com/inherits/-/inherits-2.0.4.tgz#0fa2c64f932917c3433a0ded55363aae37416b7c"
+ integrity sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==
+
+ini@^1.3.4:
+ version "1.3.8"
+ resolved "https://registry.yarnpkg.com/ini/-/ini-1.3.8.tgz#a29da425b48806f34767a4efce397269af28432c"
+ integrity sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew==
+
+isarray@~1.0.0:
+ version "1.0.0"
+ resolved "https://registry.yarnpkg.com/isarray/-/isarray-1.0.0.tgz#bb935d48582cba168c06834957a54a3e07124f11"
+ integrity sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=
+
+json-buffer@3.0.0:
+ version "3.0.0"
+ resolved "https://registry.yarnpkg.com/json-buffer/-/json-buffer-3.0.0.tgz#5b1f397afc75d677bde8bcfc0e47e1f9a3d9a898"
+ integrity sha1-Wx85evx11ne96Lz8Dkfh+aPZqJg=
+
+json-stringify-safe@^5.0.1:
+ version "5.0.1"
+ resolved "https://registry.yarnpkg.com/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz#1296a2d58fd45f19a0f6ce01d65701e2c735b6eb"
+ integrity sha1-Epai1Y/UXxmg9s4B1lcB4sc1tus=
+
+jsonfile@^4.0.0:
+ version "4.0.0"
+ resolved "https://registry.yarnpkg.com/jsonfile/-/jsonfile-4.0.0.tgz#8771aae0799b64076b76640fca058f9c10e33ecb"
+ integrity sha1-h3Gq4HmbZAdrdmQPygWPnBDjPss=
+ optionalDependencies:
+ graceful-fs "^4.1.6"
+
+keyv@^3.0.0:
+ version "3.1.0"
+ resolved "https://registry.yarnpkg.com/keyv/-/keyv-3.1.0.tgz#ecc228486f69991e49e9476485a5be1e8fc5c4d9"
+ integrity sha512-9ykJ/46SN/9KPM/sichzQ7OvXyGDYKGTaDlKMGCAlg2UK8KRy4jb0d8sFc+0Tt0YYnThq8X2RZgCg74RPxgcVA==
+ dependencies:
+ json-buffer "3.0.0"
+
+lodash@^4.17.10:
+ version "4.17.21"
+ resolved "https://registry.yarnpkg.com/lodash/-/lodash-4.17.21.tgz#679591c564c3bffaae8454cf0b3df370c3d6911c"
+ integrity sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==
+
+lowercase-keys@^1.0.0, lowercase-keys@^1.0.1:
+ version "1.0.1"
+ resolved "https://registry.yarnpkg.com/lowercase-keys/-/lowercase-keys-1.0.1.tgz#6f9e30b47084d971a7c820ff15a6c5167b74c26f"
+ integrity sha512-G2Lj61tXDnVFFOi8VZds+SoQjtQC3dgokKdDG2mTm1tx4m50NUHBOZSBwQQHyy0V12A0JTG4icfZQH+xPyh8VA==
+
+lowercase-keys@^2.0.0:
+ version "2.0.0"
+ resolved "https://registry.yarnpkg.com/lowercase-keys/-/lowercase-keys-2.0.0.tgz#2603e78b7b4b0006cbca2fbcc8a3202558ac9479"
+ integrity sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA==
+
+lru-cache@^6.0.0:
+ version "6.0.0"
+ resolved "https://registry.yarnpkg.com/lru-cache/-/lru-cache-6.0.0.tgz#6d6fe6570ebd96aaf90fcad1dafa3b2566db3a94"
+ integrity sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==
+ dependencies:
+ yallist "^4.0.0"
+
+matcher@^3.0.0:
+ version "3.0.0"
+ resolved "https://registry.yarnpkg.com/matcher/-/matcher-3.0.0.tgz#bd9060f4c5b70aa8041ccc6f80368760994f30ca"
+ integrity sha512-OkeDaAZ/bQCxeFAozM55PKcKU0yJMPGifLwV4Qgjitu+5MoAfSQN4lsLJeXZ1b8w0x+/Emda6MZgXS1jvsapng==
+ dependencies:
+ escape-string-regexp "^4.0.0"
+
+mimic-response@^1.0.0, mimic-response@^1.0.1:
+ version "1.0.1"
+ resolved "https://registry.yarnpkg.com/mimic-response/-/mimic-response-1.0.1.tgz#4923538878eef42063cb8a3e3b0798781487ab1b"
+ integrity sha512-j5EctnkH7amfV/q5Hgmoal1g2QHFJRraOtmx0JpIqkxhBhI/lJSl1nMpQ45hVarwNETOoWEimndZ4QK0RHxuxQ==
+
+minimist@^1.2.5:
+ version "1.2.5"
+ resolved "https://registry.yarnpkg.com/minimist/-/minimist-1.2.5.tgz#67d66014b66a6a8aaa0c083c5fd58df4e4e97602"
+ integrity sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw==
+
+mkdirp@^0.5.4:
+ version "0.5.5"
+ resolved "https://registry.yarnpkg.com/mkdirp/-/mkdirp-0.5.5.tgz#d91cefd62d1436ca0f41620e251288d420099def"
+ integrity sha512-NKmAlESf6jMGym1++R0Ra7wvhV+wFW63FaSOFPwRahvea0gMUcGUhVeAg/0BC0wiv9ih5NYPB1Wn1UEI1/L+xQ==
+ dependencies:
+ minimist "^1.2.5"
+
+ms@2.0.0:
+ version "2.0.0"
+ resolved "https://registry.yarnpkg.com/ms/-/ms-2.0.0.tgz#5608aeadfc00be6c2901df5f9861788de0d597c8"
+ integrity sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=
+
+ms@2.1.2:
+ version "2.1.2"
+ resolved "https://registry.yarnpkg.com/ms/-/ms-2.1.2.tgz#d09d1f357b443f493382a8eb3ccd183872ae6009"
+ integrity sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==
+
+normalize-url@^4.1.0:
+ version "4.5.1"
+ resolved "https://registry.yarnpkg.com/normalize-url/-/normalize-url-4.5.1.tgz#0dd90cf1288ee1d1313b87081c9a5932ee48518a"
+ integrity sha512-9UZCFRHQdNrfTpGg8+1INIg93B6zE0aXMVFkw1WFwvO4SlZywU6aLg5Of0Ap/PgcbSw4LNxvMWXMeugwMCX0AA==
+
+npm-conf@^1.1.3:
+ version "1.1.3"
+ resolved "https://registry.yarnpkg.com/npm-conf/-/npm-conf-1.1.3.tgz#256cc47bd0e218c259c4e9550bf413bc2192aff9"
+ integrity sha512-Yic4bZHJOt9RCFbRP3GgpqhScOY4HH3V2P8yBj6CeYq118Qr+BLXqT2JvpJ00mryLESpgOxf5XlFv4ZjXxLScw==
+ dependencies:
+ config-chain "^1.1.11"
+ pify "^3.0.0"
+
+object-keys@^1.0.12:
+ version "1.1.1"
+ resolved "https://registry.yarnpkg.com/object-keys/-/object-keys-1.1.1.tgz#1c47f272df277f3b1daf061677d9c82e2322c60e"
+ integrity sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==
+
+once@^1.3.1, once@^1.4.0:
+ version "1.4.0"
+ resolved "https://registry.yarnpkg.com/once/-/once-1.4.0.tgz#583b1aa775961d4b113ac17d9c50baef9dd76bd1"
+ integrity sha1-WDsap3WWHUsROsF9nFC6753Xa9E=
+ dependencies:
+ wrappy "1"
+
+p-cancelable@^1.0.0:
+ version "1.1.0"
+ resolved "https://registry.yarnpkg.com/p-cancelable/-/p-cancelable-1.1.0.tgz#d078d15a3af409220c886f1d9a0ca2e441ab26cc"
+ integrity sha512-s73XxOZ4zpt1edZYZzvhqFa6uvQc1vwUa0K0BdtIZgQMAJj9IbebH+JkgKZc9h+B05PKHLOTl4ajG1BmNrVZlw==
+
+pend@~1.2.0:
+ version "1.2.0"
+ resolved "https://registry.yarnpkg.com/pend/-/pend-1.2.0.tgz#7a57eb550a6783f9115331fcf4663d5c8e007a50"
+ integrity sha1-elfrVQpng/kRUzH89GY9XI4AelA=
+
+pify@^3.0.0:
+ version "3.0.0"
+ resolved "https://registry.yarnpkg.com/pify/-/pify-3.0.0.tgz#e5a4acd2c101fdf3d9a4d07f0dbc4db49dd28176"
+ integrity sha1-5aSs0sEB/fPZpNB/DbxNtJ3SgXY=
+
+prepend-http@^2.0.0:
+ version "2.0.0"
+ resolved "https://registry.yarnpkg.com/prepend-http/-/prepend-http-2.0.0.tgz#e92434bfa5ea8c19f41cdfd401d741a3c819d897"
+ integrity sha1-6SQ0v6XqjBn0HN/UAddBo8gZ2Jc=
+
+process-nextick-args@~2.0.0:
+ version "2.0.1"
+ resolved "https://registry.yarnpkg.com/process-nextick-args/-/process-nextick-args-2.0.1.tgz#7820d9b16120cc55ca9ae7792680ae7dba6d7fe2"
+ integrity sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==
+
+progress@^2.0.3:
+ version "2.0.3"
+ resolved "https://registry.yarnpkg.com/progress/-/progress-2.0.3.tgz#7e8cf8d8f5b8f239c1bc68beb4eb78567d572ef8"
+ integrity sha512-7PiHtLll5LdnKIMw100I+8xJXR5gW2QwWYkT6iJva0bXitZKa/XMrSbdmg3r2Xnaidz9Qumd0VPaMrZlF9V9sA==
+
+proto-list@~1.2.1:
+ version "1.2.4"
+ resolved "https://registry.yarnpkg.com/proto-list/-/proto-list-1.2.4.tgz#212d5bfe1318306a420f6402b8e26ff39647a849"
+ integrity sha1-IS1b/hMYMGpCD2QCuOJv85ZHqEk=
+
+pump@^3.0.0:
+ version "3.0.0"
+ resolved "https://registry.yarnpkg.com/pump/-/pump-3.0.0.tgz#b4a2116815bde2f4e1ea602354e8c75565107a64"
+ integrity sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==
+ dependencies:
+ end-of-stream "^1.1.0"
+ once "^1.3.1"
+
+readable-stream@^2.2.2:
+ version "2.3.7"
+ resolved "https://registry.yarnpkg.com/readable-stream/-/readable-stream-2.3.7.tgz#1eca1cf711aef814c04f62252a36a62f6cb23b57"
+ integrity sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==
+ dependencies:
+ core-util-is "~1.0.0"
+ inherits "~2.0.3"
+ isarray "~1.0.0"
+ process-nextick-args "~2.0.0"
+ safe-buffer "~5.1.1"
+ string_decoder "~1.1.1"
+ util-deprecate "~1.0.1"
+
+responselike@^1.0.2:
+ version "1.0.2"
+ resolved "https://registry.yarnpkg.com/responselike/-/responselike-1.0.2.tgz#918720ef3b631c5642be068f15ade5a46f4ba1e7"
+ integrity sha1-kYcg7ztjHFZCvgaPFa3lpG9Loec=
+ dependencies:
+ lowercase-keys "^1.0.0"
+
+roarr@^2.15.3:
+ version "2.15.4"
+ resolved "https://registry.yarnpkg.com/roarr/-/roarr-2.15.4.tgz#f5fe795b7b838ccfe35dc608e0282b9eba2e7afd"
+ integrity sha512-CHhPh+UNHD2GTXNYhPWLnU8ONHdI+5DI+4EYIAOaiD63rHeYlZvyh8P+in5999TTSFgUYuKUAjzRI4mdh/p+2A==
+ dependencies:
+ boolean "^3.0.1"
+ detect-node "^2.0.4"
+ globalthis "^1.0.1"
+ json-stringify-safe "^5.0.1"
+ semver-compare "^1.0.0"
+ sprintf-js "^1.1.2"
+
+safe-buffer@~5.1.0, safe-buffer@~5.1.1:
+ version "5.1.2"
+ resolved "https://registry.yarnpkg.com/safe-buffer/-/safe-buffer-5.1.2.tgz#991ec69d296e0313747d59bdfd2b745c35f8828d"
+ integrity sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==
+
+semver-compare@^1.0.0:
+ version "1.0.0"
+ resolved "https://registry.yarnpkg.com/semver-compare/-/semver-compare-1.0.0.tgz#0dee216a1c941ab37e9efb1788f6afc5ff5537fc"
+ integrity sha1-De4hahyUGrN+nvsXiPavxf9VN/w=
+
+semver@^6.2.0:
+ version "6.3.0"
+ resolved "https://registry.yarnpkg.com/semver/-/semver-6.3.0.tgz#ee0a64c8af5e8ceea67687b133761e1becbd1d3d"
+ integrity sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==
+
+semver@^7.3.2:
+ version "7.3.5"
+ resolved "https://registry.yarnpkg.com/semver/-/semver-7.3.5.tgz#0b621c879348d8998e4b0e4be94b3f12e6018ef7"
+ integrity sha512-PoeGJYh8HK4BTO/a9Tf6ZG3veo/A7ZVsYrSA6J8ny9nb3B1VrpkuN+z9OE5wfE5p6H4LchYZsegiQgbJD94ZFQ==
+ dependencies:
+ lru-cache "^6.0.0"
+
+serialize-error@^7.0.1:
+ version "7.0.1"
+ resolved "https://registry.yarnpkg.com/serialize-error/-/serialize-error-7.0.1.tgz#f1360b0447f61ffb483ec4157c737fab7d778e18"
+ integrity sha512-8I8TjW5KMOKsZQTvoxjuSIa7foAwPWGOts+6o7sgjz41/qMD9VQHEDxi6PBvK2l0MXUmqZyNpUK+T2tQaaElvw==
+ dependencies:
+ type-fest "^0.13.1"
+
+sprintf-js@^1.1.2:
+ version "1.1.2"
+ resolved "https://registry.yarnpkg.com/sprintf-js/-/sprintf-js-1.1.2.tgz#da1765262bf8c0f571749f2ad6c26300207ae673"
+ integrity sha512-VE0SOVEHCk7Qc8ulkWw3ntAzXuqf7S2lvwQaDLRnUeIEaKNQJzV6BwmLKhOqT61aGhfUMrXeaBk+oDGCzvhcug==
+
+string_decoder@~1.1.1:
+ version "1.1.1"
+ resolved "https://registry.yarnpkg.com/string_decoder/-/string_decoder-1.1.1.tgz#9cf1611ba62685d7030ae9e4ba34149c3af03fc8"
+ integrity sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==
+ dependencies:
+ safe-buffer "~5.1.0"
+
+sumchecker@^3.0.1:
+ version "3.0.1"
+ resolved "https://registry.yarnpkg.com/sumchecker/-/sumchecker-3.0.1.tgz#6377e996795abb0b6d348e9b3e1dfb24345a8e42"
+ integrity sha512-MvjXzkz/BOfyVDkG0oFOtBxHX2u3gKbMHIF/dXblZsgD3BWOFLmHovIpZY7BykJdAjcqRCBi1WYBNdEC9yI7vg==
+ dependencies:
+ debug "^4.1.0"
+
+to-readable-stream@^1.0.0:
+ version "1.0.0"
+ resolved "https://registry.yarnpkg.com/to-readable-stream/-/to-readable-stream-1.0.0.tgz#ce0aa0c2f3df6adf852efb404a783e77c0475771"
+ integrity sha512-Iq25XBt6zD5npPhlLVXGFN3/gyR2/qODcKNNyTMd4vbm39HUaOiAM4PMq0eMVC/Tkxz+Zjdsc55g9yyz+Yq00Q==
+
+tunnel@^0.0.6:
+ version "0.0.6"
+ resolved "https://registry.yarnpkg.com/tunnel/-/tunnel-0.0.6.tgz#72f1314b34a5b192db012324df2cc587ca47f92c"
+ integrity sha512-1h/Lnq9yajKY2PEbBadPXj3VxsDDu844OnaAo52UVmIzIvwwtBPIuNvkjuzBlTWpfJyUbG3ez0KSBibQkj4ojg==
+
+type-fest@^0.13.1:
+ version "0.13.1"
+ resolved "https://registry.yarnpkg.com/type-fest/-/type-fest-0.13.1.tgz#0172cb5bce80b0bd542ea348db50c7e21834d934"
+ integrity sha512-34R7HTnG0XIJcBSn5XhDd7nNFPRcXYRZrBB2O2jdKqYODldSzBAqzsWoZYYvduky73toYS/ESqxPvkDf/F0XMg==
+
+typedarray@^0.0.6:
+ version "0.0.6"
+ resolved "https://registry.yarnpkg.com/typedarray/-/typedarray-0.0.6.tgz#867ac74e3864187b1d3d47d996a78ec5c8830777"
+ integrity sha1-hnrHTjhkGHsdPUfZlqeOxciDB3c=
+
+universalify@^0.1.0:
+ version "0.1.2"
+ resolved "https://registry.yarnpkg.com/universalify/-/universalify-0.1.2.tgz#b646f69be3942dabcecc9d6639c80dc105efaa66"
+ integrity sha512-rBJeI5CXAlmy1pV+617WB9J63U6XcazHHF2f2dbJix4XzpUF0RS3Zbj0FGIOCAva5P/d/GBOYaACQ1w+0azUkg==
+
+url-parse-lax@^3.0.0:
+ version "3.0.0"
+ resolved "https://registry.yarnpkg.com/url-parse-lax/-/url-parse-lax-3.0.0.tgz#16b5cafc07dbe3676c1b1999177823d6503acb0c"
+ integrity sha1-FrXK/Afb42dsGxmZF3gj1lA6yww=
+ dependencies:
+ prepend-http "^2.0.0"
+
+util-deprecate@~1.0.1:
+ version "1.0.2"
+ resolved "https://registry.yarnpkg.com/util-deprecate/-/util-deprecate-1.0.2.tgz#450d4dc9fa70de732762fbd2d4a28981419a0ccf"
+ integrity sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=
+
+wrappy@1:
+ version "1.0.2"
+ resolved "https://registry.yarnpkg.com/wrappy/-/wrappy-1.0.2.tgz#b5243d8f3ec1aa35f1364605bc0d1036e30ab69f"
+ integrity sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=
+
+yallist@^4.0.0:
+ version "4.0.0"
+ resolved "https://registry.yarnpkg.com/yallist/-/yallist-4.0.0.tgz#9bb92790d9c0effec63be73519e11a35019a3a72"
+ integrity sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==
+
+yauzl@^2.10.0:
+ version "2.10.0"
+ resolved "https://registry.yarnpkg.com/yauzl/-/yauzl-2.10.0.tgz#c7eb17c93e112cb1086fa6d8e51fb0667b79a5f9"
+ integrity sha1-x+sXyT4RLLEIb6bY5R+wZnt5pfk=
+ dependencies:
+ buffer-crc32 "~0.2.3"
+ fd-slicer "~1.1.0"
diff --git a/shapez-io/mod_examples/README.md b/shapez-io/mod_examples/README.md
new file mode 100644
index 00000000..5086bbdb
--- /dev/null
+++ b/shapez-io/mod_examples/README.md
@@ -0,0 +1,58 @@
+# shapez.io Modding
+
+## General Instructions
+
+Currently there are two options to develop mods for shapez.io:
+
+1. Writing single file mods, which doesn't require any additional tools and can be loaded directly in the game
+2. Using the [create-shapezio-mod](https://www.npmjs.com/package/create-shapezio-mod) package. This package is still in development but allows you to pack multiple files and images into a single mod file, so you don't have to base64 encode your images etc.
+
+## Mod Developer Discord
+
+A great place to get help with mod development is the official [shapez.io modloader discord](https://discord.gg/xq5v8uyMue).
+
+## Setting up your development environment
+
+The simplest way of developing mods is by just creating a `mymod.js` file and putting it in the `mods/` folder of the standalone (You can find the `mods/` folder by clicking "Open Mods Folder" in the shapez Standalone, be sure to select the 1.5.0-modloader branch on Steam).
+
+You can then add `--dev` to the launch options on Steam. This adds an application menu where you can click "Restart" to reload your mod, and will also show the developer console where you can see any potential errors.
+
+## Getting started
+
+To get into shapez.io modding, I highly recommend checking out all of the examples in this folder. Here's a list of examples and what features of the modloader they show:
+
+| Example | Description | Demonstrates |
+| ---------------------------------------------------------- | ------------------------------------------------------------------------------------------------- | ----------------------------------------------------------------------------------------------- |
+| [base.js](base.js) | The most basic mod | Base structure of a mod |
+| [class_extensions.js](class_extensions.js) | Shows how to extend multiple methods of one class at once, useful for overriding a lot of methods | Overriding and extending builtin methods |
+| [custom_css.js](custom_css.js) | Modifies the Main Menu State look | Modifying the UI styles with CSS |
+| [replace_builtin_sprites.js](replace_builtin_sprites.js) | Replaces all color sprites with icons | Replacing builtin sprites |
+| [translations.js](translations.js) | Shows how to replace and add new translations in multiple languages | Adding and replacing translations |
+| [add_building_basic.js](add_building_basic.js) | Shows how to add a new building | Registering a new building |
+| [add_building_flipper.js](add_building_flipper.js) | Adds a "flipper" building which mirrors shapes from top to bottom | Registering a new building, Adding a custom shape and item processing operation (flip) |
+| [custom_drawing.js](custom_drawing.js) | Displays a a small indicator on every item processing building whether it is currently working | Adding a new GameSystem and drawing overlays |
+| [custom_keybinding.js](custom_keybinding.js) | Adds a new customizable ingame keybinding (Shift+F) | Adding a new keybinding |
+| [custom_sub_shapes.js](custom_sub_shapes.js) | Adds a new type of sub-shape (Line) | Adding a new sub shape and drawing it, making it spawn on the map, modifying the builtin levels |
+| [modify_theme.js](modify_theme.js) | Modifies the default game themes | Modifying the builtin themes |
+| [custom_theme.js](custom_theme.js) | Adds a new UI and map theme | Adding a new game theme |
+| [mod_settings.js](mod_settings.js) | Shows a dialog counting how often the mod has been launched | Reading and storing mod settings |
+| [storing_data_in_savegame.js](storing_data_in_savegame.js) | Shows how to store custom (structured) data in the savegame | Storing custom data in savegame |
+| [modify_existing_building.js](modify_existing_building.js) | Makes the rotator building always unlocked and adds a new statistic to the building panel | Modifying a builtin building, replacing builtin methods |
+| [modify_ui.js](modify_ui.js) | Shows how to add custom UI elements to builtin game states (the Main Menu in this case) | Extending builtin UI states, Adding CSS |
+| [pasting.js](pasting.js) | Shows a dialog when pasting text in the game | Listening to paste events |
+| [smooth_zooming.js](smooth_zooming.js) | Allows to smoothly zoom in and out | Keybindings, overriding methods |
+| [sandbox.js](sandbox.js) | Makes blueprints free and always unlocked | Overriding builtin methods |
+
+### Advanced Examples
+
+| Example | Description | Demonstrates |
+| ------------------------------------------------ | ---------------------------------------------------------------------------------------------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
+| [notification_blocks.js](notification_blocks.js) | Adds a notification block building, which shows a user defined notification when receiving a truthy signal | Adding a new Component, Adding a new GameSystem, Working with wire networks, Adding a new building, Adding a new HUD part, Using Input Dialogs, Adding Translations |
+| [usage_statistics.js](usage_statistics.js) | Displays a percentage on every building showing its utilization | Adding a new component, Adding a new GameSystem, Drawing within a GameSystem, Modifying builtin buildings, Adding custom game logic |
+| [new_item_type.js](new_item_type.js) | Adds a new type of items to the map (fluids) | Adding a new item type, modifying map generation |
+| [buildings_have_cost.js](buildings_have_cost.js) | Adds a new currency, and belts cost 1 of that currency | Extending and replacing builtin methods, Adding CSS and custom sprites |
+| [mirrored_cutter.js](mirrored_cutter.js) | Adds a mirrored variant of the cutter | Adding a new variant to existing buildings |
+
+### Creating new sprites
+
+If you want to add new buildings and create sprites for them, you can download the original Photoshop PSD files here: https://static.shapez.io/building-psds.zip
diff --git a/shapez-io/mod_examples/add_building_basic.js b/shapez-io/mod_examples/add_building_basic.js
new file mode 100644
index 00000000..6b92e769
--- /dev/null
+++ b/shapez-io/mod_examples/add_building_basic.js
@@ -0,0 +1,67 @@
+// @ts-nocheck
+const METADATA = {
+ website: "https://tobspr.io",
+ author: "tobspr",
+ name: "Mod Example: Add new basic building",
+ version: "1",
+ id: "add-building-basic",
+ description: "Shows how to add a new basic building",
+ minimumGameVersion: ">=1.5.0",
+};
+
+class MetaDemoModBuilding extends shapez.ModMetaBuilding {
+ constructor() {
+ super("demoModBuilding");
+ }
+
+ static getAllVariantCombinations() {
+ return [
+ {
+ variant: shapez.defaultBuildingVariant,
+ name: "A test name",
+ description: "A test building",
+
+ regularImageBase64: RESOURCES["demoBuilding.png"],
+ blueprintImageBase64: RESOURCES["demoBuildingBlueprint.png"],
+ tutorialImageBase64: RESOURCES["demoBuildingBlueprint.png"],
+ },
+ ];
+ }
+
+ getSilhouetteColor() {
+ return "red";
+ }
+
+ setupEntityComponents(entity) {
+ // Here you can add components, for example an ItemProcessorComponent.
+ // To get an idea what you can do with the builtin components, have a look
+ // at the builtin buildings in
+ }
+}
+
+class Mod extends shapez.Mod {
+ init() {
+ // Register the new building
+ this.modInterface.registerNewBuilding({
+ metaClass: MetaDemoModBuilding,
+ buildingIconBase64: RESOURCES["demoBuilding.png"],
+ });
+
+ // Add it to the regular toolbar
+ this.modInterface.addNewBuildingToToolbar({
+ toolbar: "regular",
+ location: "primary",
+ metaClass: MetaDemoModBuilding,
+ });
+ }
+}
+
+////////////////////////////////////////////////////////////////////////
+
+const RESOURCES = {
+ "demoBuilding.png":
+ "",
+
+ "demoBuildingBlueprint.png":
+ "",
+};
diff --git a/shapez-io/mod_examples/add_building_flipper.js b/shapez-io/mod_examples/add_building_flipper.js
new file mode 100644
index 00000000..03442499
--- /dev/null
+++ b/shapez-io/mod_examples/add_building_flipper.js
@@ -0,0 +1,130 @@
+// @ts-nocheck
+const METADATA = {
+ website: "https://tobspr.io",
+ author: "tobspr",
+ name: "Mod Example: Add a flipper building",
+ version: "1",
+ id: "add-building-extended",
+ description:
+ "Shows how to add a new building with logic, in this case it flips/mirrors shapez from top to down",
+ minimumGameVersion: ">=1.5.0",
+};
+
+// Declare a new type of item processor
+shapez.enumItemProcessorTypes.flipper = "flipper";
+
+// For now, flipper always has the same speed
+shapez.MOD_ITEM_PROCESSOR_SPEEDS.flipper = () => 10;
+
+// Declare a handler for the processor so we define the "flip" operation
+shapez.MOD_ITEM_PROCESSOR_HANDLERS.flipper = function (payload) {
+ const shapeDefinition = payload.items.get(0).definition;
+
+ // Flip bottom with top on a new, cloned item (NEVER modify the incoming item!)
+ const newLayers = shapeDefinition.getClonedLayers();
+ newLayers.forEach(layer => {
+ const tr = layer[shapez.TOP_RIGHT];
+ const br = layer[shapez.BOTTOM_RIGHT];
+ const bl = layer[shapez.BOTTOM_LEFT];
+ const tl = layer[shapez.TOP_LEFT];
+
+ layer[shapez.BOTTOM_LEFT] = tl;
+ layer[shapez.BOTTOM_RIGHT] = tr;
+
+ layer[shapez.TOP_LEFT] = bl;
+ layer[shapez.TOP_RIGHT] = br;
+ });
+
+ const newDefinition = new shapez.ShapeDefinition({ layers: newLayers });
+ payload.outItems.push({
+ item: this.root.shapeDefinitionMgr.getShapeItemFromDefinition(newDefinition),
+ });
+};
+
+// Create the building
+class MetaModFlipperBuilding extends shapez.ModMetaBuilding {
+ constructor() {
+ super("modFlipperBuilding");
+ }
+
+ static getAllVariantCombinations() {
+ return [
+ {
+ name: "Flipper",
+ description: "Flipps/Mirrors shapez from top to bottom",
+ variant: shapez.defaultBuildingVariant,
+
+ regularImageBase64: RESOURCES["flipper.png"],
+ blueprintImageBase64: RESOURCES["flipper.png"],
+ tutorialImageBase64: RESOURCES["flipper.png"],
+ },
+ ];
+ }
+
+ getSilhouetteColor() {
+ return "red";
+ }
+
+ getAdditionalStatistics(root) {
+ const speed = root.hubGoals.getProcessorBaseSpeed(shapez.enumItemProcessorTypes.flipper);
+ return [[shapez.T.ingame.buildingPlacement.infoTexts.speed, shapez.formatItemsPerSecond(speed)]];
+ }
+
+ getIsUnlocked(root) {
+ return true;
+ }
+
+ setupEntityComponents(entity) {
+ // Accept shapes from the bottom
+ entity.addComponent(
+ new shapez.ItemAcceptorComponent({
+ slots: [
+ {
+ pos: new shapez.Vector(0, 0),
+ direction: shapez.enumDirection.bottom,
+ filter: "shape",
+ },
+ ],
+ })
+ );
+
+ // Process those shapes with tye processor type "flipper" (which we added above)
+ entity.addComponent(
+ new shapez.ItemProcessorComponent({
+ inputsPerCharge: 1,
+ processorType: shapez.enumItemProcessorTypes.flipper,
+ })
+ );
+
+ // Eject the result to the top
+ entity.addComponent(
+ new shapez.ItemEjectorComponent({
+ slots: [{ pos: new shapez.Vector(0, 0), direction: shapez.enumDirection.top }],
+ })
+ );
+ }
+}
+
+class Mod extends shapez.Mod {
+ init() {
+ // Register the new building
+ this.modInterface.registerNewBuilding({
+ metaClass: MetaModFlipperBuilding,
+ buildingIconBase64: RESOURCES["flipper.png"],
+ });
+
+ // Add it to the regular toolbar
+ this.modInterface.addNewBuildingToToolbar({
+ toolbar: "regular",
+ location: "primary",
+ metaClass: MetaModFlipperBuilding,
+ });
+ }
+}
+
+////////////////////////////////////////////////////////////////////////
+
+const RESOURCES = {
+ "flipper.png":
+ "",
+};
diff --git a/shapez-io/mod_examples/base.js b/shapez-io/mod_examples/base.js
new file mode 100644
index 00000000..f76ded9b
--- /dev/null
+++ b/shapez-io/mod_examples/base.js
@@ -0,0 +1,20 @@
+// @ts-nocheck
+const METADATA = {
+ website: "https://tobspr.io",
+ author: "tobspr",
+ name: "Mod Example: Base",
+ version: "1",
+ id: "base",
+ description: "The most basic mod",
+ minimumGameVersion: ">=1.5.0",
+
+ // You can specify this parameter if savegames will still work
+ // after your mod has been uninstalled
+ doesNotAffectSavegame: true,
+};
+
+class Mod extends shapez.Mod {
+ init() {
+ // Start the modding here
+ }
+}
diff --git a/shapez-io/mod_examples/buildings_have_cost.js b/shapez-io/mod_examples/buildings_have_cost.js
new file mode 100644
index 00000000..3dae84ae
--- /dev/null
+++ b/shapez-io/mod_examples/buildings_have_cost.js
@@ -0,0 +1,89 @@
+// @ts-nocheck
+const METADATA = {
+ website: "https://tobspr.io",
+ author: "tobspr",
+ name: "Mod Example: Patch Methods",
+ version: "1",
+ id: "patch-methods",
+ description: "Shows how to patch existing methods to change the game by making the belts cost shapes",
+ minimumGameVersion: ">=1.5.0",
+};
+
+class Mod extends shapez.Mod {
+ init() {
+ // Define our currency
+ const CURRENCY = "CyCyCyCy:--------:CuCuCuCu";
+
+ // Make sure the currency is always pinned
+ this.modInterface.runAfterMethod(shapez.HUDPinnedShapes, "rerenderFull", function () {
+ this.internalPinShape({
+ key: CURRENCY,
+ canUnpin: false,
+ className: "currency",
+ });
+ });
+
+ // Style it
+ this.modInterface.registerCss(`
+ #ingame_HUD_PinnedShapes .shape.currency::after {
+ content: " ";
+ position: absolute;
+ display: inline-block;
+ width: $scaled(8px);
+ height: $scaled(8px);
+ top: $scaled(4px);
+ left: $scaled(-7px);
+ background: url('${RESOURCES["currency.png"]}') center center / contain no-repeat;
+ }
+
+ .currencyIcon {
+ display: inline-block;
+ vertical-align: middle;
+ background: url('${RESOURCES["currency.png"]}') center center / contain no-repeat;
+ }
+
+ .currencyIcon.small {
+ width: $scaled(11px);
+ height: $scaled(11px);
+ }
+ `);
+
+ // Make the player start with some currency
+ this.modInterface.runAfterMethod(shapez.GameCore, "initNewGame", function () {
+ this.root.hubGoals.storedShapes[CURRENCY] = 100;
+ });
+
+ // Make belts have a cost
+ this.modInterface.replaceMethod(shapez.MetaBeltBuilding, "getAdditionalStatistics", function (
+ $original,
+ [root, variant]
+ ) {
+ const oldStats = $original(root, variant);
+ oldStats.push(["Cost", "1 x "]);
+ return oldStats;
+ });
+
+ // Only allow placing an entity when there is enough currency
+ this.modInterface.replaceMethod(shapez.GameLogic, "checkCanPlaceEntity", function (
+ $original,
+ [entity, options]
+ ) {
+ const storedCurrency = this.root.hubGoals.storedShapes[CURRENCY] || 0;
+ return storedCurrency > 0 && $original(entity, options);
+ });
+
+ // Take shapes when placing a building
+ this.modInterface.replaceMethod(shapez.GameLogic, "tryPlaceBuilding", function ($original, args) {
+ const result = $original(...args);
+ if (result && result.components.Belt) {
+ this.root.hubGoals.storedShapes[CURRENCY]--;
+ }
+ return result;
+ });
+ }
+}
+
+const RESOURCES = {
+ "currency.png":
+ "",
+};
diff --git a/shapez-io/mod_examples/class_extensions.js b/shapez-io/mod_examples/class_extensions.js
new file mode 100644
index 00000000..ace5aae9
--- /dev/null
+++ b/shapez-io/mod_examples/class_extensions.js
@@ -0,0 +1,32 @@
+// @ts-nocheck
+const METADATA = {
+ website: "https://tobspr.io",
+ author: "tobspr",
+ name: "Mod Example: Class Extensions",
+ version: "1",
+ id: "class-extensions",
+ description: "Shows how to extend builtin classes",
+ minimumGameVersion: ">=1.5.0",
+};
+
+const BeltExtension = ({ $super, $old }) => ({
+ getShowWiresLayerPreview() {
+ // Access the old method
+ return !$old.getShowWiresLayerPreview();
+ },
+
+ getIsReplaceable(variant, rotationVariant) {
+ // Instead of super, use $super
+ return $super.getIsReplaceable.call(this, variant, rotationVariant);
+ },
+
+ getIsRemoveable() {
+ return false;
+ },
+});
+
+class Mod extends shapez.Mod {
+ init() {
+ this.modInterface.extendClass(shapez.MetaBeltBuilding, BeltExtension);
+ }
+}
diff --git a/shapez-io/mod_examples/custom_css.js b/shapez-io/mod_examples/custom_css.js
new file mode 100644
index 00000000..0d28fda7
--- /dev/null
+++ b/shapez-io/mod_examples/custom_css.js
@@ -0,0 +1,44 @@
+// @ts-nocheck
+const METADATA = {
+ website: "https://tobspr.io",
+ author: "tobspr",
+ name: "Mod Example: Add custom CSS",
+ version: "1",
+ id: "custom-css",
+ description: "Shows how to add custom css",
+ minimumGameVersion: ">=1.5.0",
+
+ // You can specify this parameter if savegames will still work
+ // after your mod has been uninstalled
+ doesNotAffectSavegame: true,
+};
+
+class Mod extends shapez.Mod {
+ init() {
+ // Notice that, since the UI is scaled dynamically, every pixel value
+ // should be wrapped in '$scaled()' (see below)
+
+ this.modInterface.registerCss(`
+ * {
+ font-family: "Comic Sans", "Comic Sans MS", "ComicSans", Tahoma !important;
+ }
+
+ #state_MainMenuState {
+ background: #9dc499 url('${RESOURCES["cat.png"]}') top left repeat !important;
+ }
+
+ #state_MainMenuState .fullscreenBackgroundVideo {
+ display: none !important;
+ }
+
+ #state_MainMenuState .mainContainer, #state_MainMenuState .modsOverview {
+ border: $scaled(5px) solid #000 !important;
+ }
+ `);
+ }
+}
+
+const RESOURCES = {
+ "cat.png":
+ "",
+};
diff --git a/shapez-io/mod_examples/custom_drawing.js b/shapez-io/mod_examples/custom_drawing.js
new file mode 100644
index 00000000..2dccab2d
--- /dev/null
+++ b/shapez-io/mod_examples/custom_drawing.js
@@ -0,0 +1,63 @@
+// @ts-nocheck
+const METADATA = {
+ website: "https://tobspr.io",
+ author: "tobspr",
+ name: "Mod Example: custom drawing",
+ version: "1",
+ id: "base",
+ description: "Displays an indicator on every item processing building when its working",
+ minimumGameVersion: ">=1.5.0",
+
+ // You can specify this parameter if savegames will still work
+ // after your mod has been uninstalled
+ doesNotAffectSavegame: true,
+};
+
+class ItemProcessorStatusGameSystem extends shapez.GameSystem {
+ drawChunk(parameters, chunk) {
+ const contents = chunk.containedEntitiesByLayer.regular;
+ for (let i = 0; i < contents.length; ++i) {
+ const entity = contents[i];
+ const processorComp = entity.components.ItemProcessor;
+ if (!processorComp) {
+ continue;
+ }
+
+ const staticComp = entity.components.StaticMapEntity;
+
+ const context = parameters.context;
+ const center = staticComp.getTileSpaceBounds().getCenter().toWorldSpace();
+
+ // Culling for better performance
+ if (parameters.visibleRect.containsCircle(center.x, center.y, 40)) {
+ // Circle
+ context.fillStyle = processorComp.ongoingCharges.length === 0 ? "#aaa" : "#53cf47";
+ context.strokeStyle = "#000";
+ context.lineWidth = 1;
+
+ context.beginCircle(center.x + 5, center.y + 5, 4);
+ context.fill();
+ context.stroke();
+ }
+ }
+ }
+}
+
+class Mod extends shapez.Mod {
+ init() {
+ // Register our game system
+ this.modInterface.registerGameSystem({
+ id: "item_processor_status",
+ systemClass: ItemProcessorStatusGameSystem,
+
+ // Specify at which point the update method will be called,
+ // in this case directly before the belt system. You can use
+ // before: "end" to make it the last system
+ before: "belt",
+
+ // Specify where our drawChunk method should be called, check out
+ // map_chunk_view
+ drawHooks: ["staticAfter"],
+ });
+ }
+}
diff --git a/shapez-io/mod_examples/custom_keybinding.js b/shapez-io/mod_examples/custom_keybinding.js
new file mode 100644
index 00000000..0109833c
--- /dev/null
+++ b/shapez-io/mod_examples/custom_keybinding.js
@@ -0,0 +1,32 @@
+// @ts-nocheck
+const METADATA = {
+ website: "https://tobspr.io",
+ author: "tobspr",
+ name: "Mod Example: Custom Keybindings",
+ version: "1",
+ id: "base",
+ description: "Shows how to add a new keybinding",
+ minimumGameVersion: ">=1.5.0",
+
+ // You can specify this parameter if savegames will still work
+ // after your mod has been uninstalled
+ doesNotAffectSavegame: true,
+};
+
+class Mod extends shapez.Mod {
+ init() {
+ // Register keybinding
+ this.modInterface.registerIngameKeybinding({
+ id: "demo_mod_binding",
+ keyCode: shapez.keyToKeyCode("F"),
+ translation: "Do something (always with SHIFT)",
+ modifiers: {
+ shift: true,
+ },
+ handler: root => {
+ this.dialogs.showInfo("Mod Message", "It worked!");
+ return shapez.STOP_PROPAGATION;
+ },
+ });
+ }
+}
diff --git a/shapez-io/mod_examples/custom_sub_shapes.js b/shapez-io/mod_examples/custom_sub_shapes.js
new file mode 100644
index 00000000..3aea03cf
--- /dev/null
+++ b/shapez-io/mod_examples/custom_sub_shapes.js
@@ -0,0 +1,46 @@
+// @ts-nocheck
+const METADATA = {
+ website: "https://tobspr.io",
+ author: "tobspr",
+ name: "Mod Example: Custom Sub Shapes",
+ version: "1",
+ id: "custom-sub-shapes",
+ description: "Shows how to add custom sub shapes",
+ minimumGameVersion: ">=1.5.0",
+};
+
+class Mod extends shapez.Mod {
+ init() {
+ // Add a new type of sub shape ("Line", short code "L")
+ this.modInterface.registerSubShapeType({
+ id: "line",
+ shortCode: "L",
+
+ // Make it spawn on the map
+ weightComputation: distanceToOriginInChunks =>
+ Math.round(20 + Math.max(Math.min(distanceToOriginInChunks, 30), 0)),
+
+ // This defines how to draw it
+ draw: ({ context, quadrantSize, layerScale }) => {
+ const quadrantHalfSize = quadrantSize / 2;
+ context.beginPath();
+ context.moveTo(-quadrantHalfSize, quadrantHalfSize);
+ context.arc(
+ -quadrantHalfSize,
+ quadrantHalfSize,
+ quadrantSize * layerScale,
+ -Math.PI * 0.25,
+ 0
+ );
+ context.closePath();
+ context.fill();
+ context.stroke();
+ },
+ });
+
+ // Modify the goal of the first level to add our goal
+ this.signals.modifyLevelDefinitions.add(definitions => {
+ definitions[0].shape = "LuLuLuLu";
+ });
+ }
+}
diff --git a/shapez-io/mod_examples/custom_theme.js b/shapez-io/mod_examples/custom_theme.js
new file mode 100644
index 00000000..a70d949f
--- /dev/null
+++ b/shapez-io/mod_examples/custom_theme.js
@@ -0,0 +1,99 @@
+// @ts-nocheck
+const METADATA = {
+ website: "https://tobspr.io",
+ author: "tobspr",
+ name: "Mod Example: Custom Game Theme",
+ version: "1",
+ id: "custom-theme",
+ description: "Shows how to add a custom game theme",
+ minimumGameVersion: ">=1.5.0",
+
+ // You can specify this parameter if savegames will still work
+ // after your mod has been uninstalled
+ doesNotAffectSavegame: true,
+};
+
+class Mod extends shapez.Mod {
+ init() {
+ this.modInterface.registerGameTheme({
+ id: "my-theme",
+ name: "My fancy theme",
+ theme: RESOURCES["my-theme.json"],
+ });
+ }
+}
+
+const RESOURCES = {
+ "my-theme.json": {
+ map: {
+ background: "#abc",
+ grid: "#ccc",
+ gridLineWidth: 1,
+
+ selectionOverlay: "rgba(74, 163, 223, 0.7)",
+ selectionOutline: "rgba(74, 163, 223, 0.5)",
+ selectionBackground: "rgba(74, 163, 223, 0.2)",
+
+ chunkBorders: "rgba(0, 30, 50, 0.03)",
+
+ directionLock: {
+ regular: {
+ color: "rgb(74, 237, 134)",
+ background: "rgba(74, 237, 134, 0.2)",
+ },
+ wires: {
+ color: "rgb(74, 237, 134)",
+ background: "rgba(74, 237, 134, 0.2)",
+ },
+ error: {
+ color: "rgb(255, 137, 137)",
+ background: "rgba(255, 137, 137, 0.2)",
+ },
+ },
+
+ colorBlindPickerTile: "rgba(50, 50, 50, 0.4)",
+
+ resources: {
+ shape: "#eaebec",
+ red: "#ffbfc1",
+ green: "#cbffc4",
+ blue: "#bfdaff",
+ },
+
+ chunkOverview: {
+ empty: "#a6afbb",
+ filled: "#c5ccd6",
+ beltColor: "#777",
+ },
+
+ wires: {
+ overlayColor: "rgba(97, 161, 152, 0.75)",
+ previewColor: "rgb(97, 161, 152, 0.4)",
+ highlightColor: "rgba(72, 137, 255, 1)",
+ },
+
+ connectedMiners: {
+ overlay: "rgba(40, 50, 60, 0.5)",
+ textColor: "#fff",
+ textColorCapped: "#ef5072",
+ background: "rgba(40, 50, 60, 0.8)",
+ },
+
+ zone: {
+ borderSolid: "rgba(23, 192, 255, 1)",
+ outerColor: "rgba(240, 240, 255, 0.5)",
+ },
+ },
+
+ items: {
+ outline: "#55575a",
+ outlineWidth: 0.75,
+ circleBackground: "rgba(40, 50, 65, 0.1)",
+ },
+
+ shapeTooltip: {
+ background: "#dee1ea",
+ outline: "#54565e",
+ },
+ },
+};
diff --git a/shapez-io/mod_examples/mirrored_cutter.js b/shapez-io/mod_examples/mirrored_cutter.js
new file mode 100644
index 00000000..ae457a8c
--- /dev/null
+++ b/shapez-io/mod_examples/mirrored_cutter.js
@@ -0,0 +1,81 @@
+// @ts-nocheck
+const METADATA = {
+ website: "https://tobspr.io",
+ author: "tobspr",
+ name: "Mod Example: Mirrored Cutter Variant",
+ version: "1",
+ id: "mirrored-cutter",
+ description: "Shows how to add new variants to existing buildings",
+ minimumGameVersion: ">=1.5.0",
+};
+
+class Mod extends shapez.Mod {
+ init() {
+ shapez.enumCutterVariants.mirrored = "mirrored";
+
+ this.modInterface.addVariantToExistingBuilding(
+ shapez.MetaCutterBuilding,
+ shapez.enumCutterVariants.mirrored,
+ {
+ name: "Cutter (Mirrored)",
+ description: "A mirrored cutter",
+
+ tutorialImageBase64: RESOURCES["cutter-mirrored.png"],
+ regularSpriteBase64: RESOURCES["cutter-mirrored.png"],
+ blueprintSpriteBase64: RESOURCES["cutter-mirrored.png"],
+
+ dimensions: new shapez.Vector(2, 1),
+
+ additionalStatistics(root) {
+ const speed = root.hubGoals.getProcessorBaseSpeed(shapez.enumItemProcessorTypes.cutter);
+ return [
+ [
+ shapez.T.ingame.buildingPlacement.infoTexts.speed,
+ shapez.formatItemsPerSecond(speed),
+ ],
+ ];
+ },
+
+ isUnlocked(root) {
+ return true;
+ },
+ }
+ );
+
+ // Extend instance methods
+ this.modInterface.extendClass(shapez.MetaCutterBuilding, ({ $old }) => ({
+ updateVariants(entity, rotationVariant, variant) {
+ if (variant === shapez.enumCutterVariants.mirrored) {
+ entity.components.ItemEjector.setSlots([
+ { pos: new shapez.Vector(0, 0), direction: shapez.enumDirection.top },
+ { pos: new shapez.Vector(1, 0), direction: shapez.enumDirection.top },
+ ]);
+ entity.components.ItemProcessor.type = shapez.enumItemProcessorTypes.cutter;
+ entity.components.ItemAcceptor.setSlots([
+ {
+ pos: new shapez.Vector(1, 0),
+ direction: shapez.enumDirection.bottom,
+ filter: "shape",
+ },
+ ]);
+ } else {
+ // Since we are changing the ItemAcceptor slots, we should reset
+ // it to the regular slots when we are not using our mirrored variant
+ entity.components.ItemAcceptor.setSlots([
+ {
+ pos: new shapez.Vector(0, 0),
+ direction: shapez.enumDirection.bottom,
+ filter: "shape",
+ },
+ ]);
+ $old.updateVariants.bind(this)(entity, rotationVariant, variant);
+ }
+ },
+ }));
+ }
+}
+
+const RESOURCES = {
+ "cutter-mirrored.png":
+ "",
+};
diff --git a/shapez-io/mod_examples/mod_settings.js b/shapez-io/mod_examples/mod_settings.js
new file mode 100644
index 00000000..b87c138b
--- /dev/null
+++ b/shapez-io/mod_examples/mod_settings.js
@@ -0,0 +1,32 @@
+// @ts-nocheck
+const METADATA = {
+ website: "https://tobspr.io",
+ author: "tobspr",
+ name: "Mod Example: Mod Settings",
+ version: "1",
+ id: "mod-settings",
+ description: "Shows how to add settings to your mod",
+ minimumGameVersion: ">=1.5.0",
+
+ settings: {
+ timesLaunched: 0,
+ },
+};
+
+class Mod extends shapez.Mod {
+ init() {
+ // Increment the setting every time we launch the mod
+ this.settings.timesLaunched++;
+ this.saveSettings();
+
+ // Show a dialog in the main menu with the settings
+ this.signals.stateEntered.add(state => {
+ if (state instanceof shapez.MainMenuState) {
+ this.dialogs.showInfo(
+ "Welcome back",
+ `You have launched this mod ${this.settings.timesLaunched} times`
+ );
+ }
+ });
+ }
+}
diff --git a/shapez-io/mod_examples/modify_existing_building.js b/shapez-io/mod_examples/modify_existing_building.js
new file mode 100644
index 00000000..b09f5a20
--- /dev/null
+++ b/shapez-io/mod_examples/modify_existing_building.js
@@ -0,0 +1,27 @@
+// @ts-nocheck
+const METADATA = {
+ website: "https://tobspr.io",
+ author: "tobspr",
+ name: "Mod Example: Modify existing building",
+ version: "1",
+ id: "modify-existing-building",
+ description: "Shows how to modify an existing building",
+ minimumGameVersion: ">=1.5.0",
+};
+
+class Mod extends shapez.Mod {
+ init() {
+ // Make Rotator always unlocked
+ this.modInterface.replaceMethod(shapez.MetaRotaterBuilding, "getIsUnlocked", function () {
+ return true;
+ });
+
+ // Add some custom stats to the info panel when selecting the building
+ this.modInterface.replaceMethod(shapez.MetaRotaterBuilding, "getAdditionalStatistics", function (
+ root,
+ variant
+ ) {
+ return [["Awesomeness", 5]];
+ });
+ }
+}
diff --git a/shapez-io/mod_examples/modify_theme.js b/shapez-io/mod_examples/modify_theme.js
new file mode 100644
index 00000000..de7f0ad2
--- /dev/null
+++ b/shapez-io/mod_examples/modify_theme.js
@@ -0,0 +1,24 @@
+// @ts-nocheck
+const METADATA = {
+ website: "https://tobspr.io",
+ author: "tobspr",
+ name: "Mod Example: Modify Builtin Themes",
+ version: "1",
+ id: "modify-theme",
+ description: "Shows how to modify builtin themes",
+ minimumGameVersion: ">=1.5.0",
+
+ // You can specify this parameter if savegames will still work
+ // after your mod has been uninstalled
+ doesNotAffectSavegame: true,
+};
+
+class Mod extends shapez.Mod {
+ init() {
+ shapez.THEMES.light.map.background = "#eee";
+ shapez.THEMES.light.items.outline = "#000";
+
+ shapez.THEMES.dark.map.background = "#245";
+ shapez.THEMES.dark.items.outline = "#fff";
+ }
+}
diff --git a/shapez-io/mod_examples/modify_ui.js b/shapez-io/mod_examples/modify_ui.js
new file mode 100644
index 00000000..4beb403d
--- /dev/null
+++ b/shapez-io/mod_examples/modify_ui.js
@@ -0,0 +1,46 @@
+// @ts-nocheck
+const METADATA = {
+ website: "https://tobspr.io",
+ author: "tobspr",
+ name: "Mod Example: Modify UI",
+ version: "1",
+ id: "modify-ui",
+ description: "Shows how to modify a builtin game state, in this case the main menu",
+ minimumGameVersion: ">=1.5.0",
+
+ // You can specify this parameter if savegames will still work
+ // after your mod has been uninstalled
+ doesNotAffectSavegame: true,
+};
+
+class Mod extends shapez.Mod {
+ init() {
+ // Add fancy sign to main menu
+ this.signals.stateEntered.add(state => {
+ if (state.key === "MainMenuState") {
+ const element = document.createElement("div");
+ element.id = "demo_mod_hello_world_element";
+ document.body.appendChild(element);
+
+ const button = document.createElement("button");
+ button.classList.add("styledButton");
+ button.innerText = "Hello!";
+ button.addEventListener("click", () => {
+ this.dialogs.showInfo("Mod Message", "Button clicked!");
+ });
+ element.appendChild(button);
+ }
+ });
+
+ this.modInterface.registerCss(`
+ #demo_mod_hello_world_element {
+ position: absolute;
+ top: calc(10px * var(--ui-scale));
+ left: calc(10px * var(--ui-scale));
+ color: red;
+ z-index: 0;
+ }
+
+ `);
+ }
+}
diff --git a/shapez-io/mod_examples/new_item_type.js b/shapez-io/mod_examples/new_item_type.js
new file mode 100644
index 00000000..104ef0a0
--- /dev/null
+++ b/shapez-io/mod_examples/new_item_type.js
@@ -0,0 +1,149 @@
+// @ts-nocheck
+const METADATA = {
+ website: "https://tobspr.io",
+ author: "tobspr",
+ name: "Mod Example: New Item Type (Fluids)",
+ version: "1",
+ id: "new-item-type",
+ description: "Shows how to add a new item type (fluid)",
+ minimumGameVersion: ">=1.5.0",
+};
+
+// Define which fluid types there are
+const enumFluidType = {
+ water: "water",
+ oil: "oil",
+};
+
+// Define which color they should have on the map
+const fluidColors = {
+ [enumFluidType.water]: "#477be7",
+ [enumFluidType.oil]: "#bc483a",
+};
+
+// The fluid item class (also see ColorItem and ShapeItem)
+class FluidItem extends shapez.BaseItem {
+ static getId() {
+ return "fluid";
+ }
+
+ static getSchema() {
+ return shapez.types.enum(enumFluidType);
+ }
+
+ serialize() {
+ return this.fluidType;
+ }
+
+ deserialize(data) {
+ this.fluidType = data;
+ }
+
+ getItemType() {
+ return "fluid";
+ }
+
+ /**
+ * @returns {string}
+ */
+ getAsCopyableKey() {
+ return this.fluidType;
+ }
+
+ /**
+ * @param {BaseItem} other
+ */
+ equalsImpl(other) {
+ return this.fluidType === /** @type {FluidItem} */ (other).fluidType;
+ }
+
+ /**
+ * @param {enumFluidType} fluidType
+ */
+ constructor(fluidType) {
+ super();
+ this.fluidType = fluidType;
+ }
+
+ getBackgroundColorAsResource() {
+ return fluidColors[this.fluidType];
+ }
+
+ /**
+ * Draws the item to a canvas
+ * @param {CanvasRenderingContext2D} context
+ * @param {number} size
+ */
+ drawFullSizeOnCanvas(context, size) {
+ if (!this.cachedSprite) {
+ this.cachedSprite = shapez.Loader.getSprite(`sprites/fluids/${this.fluidType}.png`);
+ }
+ this.cachedSprite.drawCentered(context, size / 2, size / 2, size);
+ }
+
+ /**
+ * @param {number} x
+ * @param {number} y
+ * @param {number} diameter
+ * @param {DrawParameters} parameters
+ */
+ drawItemCenteredClipped(x, y, parameters, diameter = shapez.globalConfig.defaultItemDiameter) {
+ const realDiameter = diameter * 0.6;
+ if (!this.cachedSprite) {
+ this.cachedSprite = shapez.Loader.getSprite(`sprites/fluids/${this.fluidType}.png`);
+ }
+ this.cachedSprite.drawCachedCentered(parameters, x, y, realDiameter);
+ }
+}
+
+/**
+ * Singleton instances.
+ *
+ * NOTICE: The game tries to instantiate as few instances as possible.
+ * Which means that if you have two types of fluids in this case, there should
+ * ONLY be 2 instances of FluidItem at *any* time.
+ *
+ * This works by having a map from fluid type to the FluidItem singleton.
+ * Additionally, all items are and should be immutable.
+ * @type {Object}
+ */
+const FLUID_ITEM_SINGLETONS = {};
+
+for (const fluidType in enumFluidType) {
+ FLUID_ITEM_SINGLETONS[fluidType] = new FluidItem(fluidType);
+}
+
+class Mod extends shapez.Mod {
+ init() {
+ // Register the sprites
+ this.modInterface.registerSprite("sprites/fluids/oil.png", RESOURCES["oil.png"]);
+ this.modInterface.registerSprite("sprites/fluids/water.png", RESOURCES["water.png"]);
+
+ // Make the item spawn on the map
+ this.modInterface.runAfterMethod(shapez.MapChunk, "generatePatches", function ({
+ rng,
+ chunkCenter,
+ distanceToOriginInChunks,
+ }) {
+ // Generate a simple patch
+ // ALWAYS use rng and NEVER use Math.random() otherwise the map will look different
+ // every time you resume the game
+ if (rng.next() > 0.8) {
+ const fluidType = rng.choice(Array.from(Object.keys(enumFluidType)));
+ this.internalGeneratePatch(rng, 4, FLUID_ITEM_SINGLETONS[fluidType]);
+ }
+ });
+
+ this.modInterface.registerItem(FluidItem, itemData => FLUID_ITEM_SINGLETONS[itemData]);
+ }
+}
+
+///////////////////////////////////////
+
+const RESOURCES = {
+ "oil.png":
+ "",
+
+ "water.png":
+ "",
+};
diff --git a/shapez-io/mod_examples/notification_blocks.js b/shapez-io/mod_examples/notification_blocks.js
new file mode 100644
index 00000000..23f95943
--- /dev/null
+++ b/shapez-io/mod_examples/notification_blocks.js
@@ -0,0 +1,314 @@
+// @ts-nocheck
+const METADATA = {
+ website: "https://tobspr.io",
+ author: "tobspr",
+ name: "Mod Example: Notification Blocks",
+ version: "1",
+ id: "notification-blocks",
+ description:
+ "Adds a new building to the wires layer, 'Notification Blocks' which show a custom notification when they get a truthy signal.",
+
+ minimumGameVersion: ">=1.5.0",
+};
+
+////////////////////////////////////////////////////////////////////////
+// This is the component storing which text the block should show as
+// a notification.
+class NotificationBlockComponent extends shapez.Component {
+ static getId() {
+ return "NotificationBlock";
+ }
+
+ static getSchema() {
+ // Here you define which properties should be saved to the savegame
+ // and get automatically restored
+ return {
+ notificationText: shapez.types.string,
+ lastStoredInput: shapez.types.bool,
+ };
+ }
+
+ constructor() {
+ super();
+ this.notificationText = "Test";
+ this.lastStoredInput = false;
+ }
+}
+
+////////////////////////////////////////////////////////////////////////
+// The game system to trigger notifications when the signal changes
+class NotificationBlocksSystem extends shapez.GameSystemWithFilter {
+ constructor(root) {
+ // By specifying the list of components, `this.allEntities` will only
+ // contain entities which have *all* of the specified components
+ super(root, [NotificationBlockComponent]);
+
+ // Ask for a notification text once an entity is placed
+ this.root.signals.entityManuallyPlaced.add(entity => {
+ const editorHud = this.root.hud.parts.notificationBlockEdit;
+ if (editorHud) {
+ editorHud.editNotificationText(entity, { deleteOnCancel: true });
+ }
+ });
+ }
+
+ update() {
+ if (!this.root.gameInitialized) {
+ // Do not start updating before the wires network was
+ // computed to avoid dispatching all notifications
+ return;
+ }
+
+ // Go over all notification blocks and check if the signal changed
+ for (let i = 0; i < this.allEntities.length; ++i) {
+ const entity = this.allEntities[i];
+
+ // Compute if the bottom pin currently has a truthy input
+ const pinsComp = entity.components.WiredPins;
+ const network = pinsComp.slots[0].linkedNetwork;
+
+ let currentInput = false;
+
+ if (network && network.hasValue()) {
+ const value = network.currentValue;
+ if (value && shapez.isTruthyItem(value)) {
+ currentInput = true;
+ }
+ }
+
+ // If the value changed, show the notification if its truthy
+ const notificationComp = entity.components.NotificationBlock;
+ if (currentInput !== notificationComp.lastStoredInput) {
+ notificationComp.lastStoredInput = currentInput;
+ if (currentInput) {
+ this.root.hud.signals.notification.dispatch(
+ notificationComp.notificationText,
+ shapez.enumNotificationType.info
+ );
+ }
+ }
+ }
+ }
+}
+
+////////////////////////////////////////////////////////////////////////
+// The actual notification block building
+class MetaNotificationBlockBuilding extends shapez.ModMetaBuilding {
+ constructor() {
+ super("notification_block");
+ }
+
+ static getAllVariantCombinations() {
+ return [
+ {
+ variant: shapez.defaultBuildingVariant,
+ name: "Notification Block",
+ description: "Shows a predefined notification on screen when receiving a truthy signal",
+
+ regularImageBase64: RESOURCES["notification_block.png"],
+ blueprintImageBase64: RESOURCES["notification_block.png"],
+ tutorialImageBase64: RESOURCES["notification_block.png"],
+ },
+ ];
+ }
+
+ getSilhouetteColor() {
+ return "#daff89";
+ }
+
+ getIsUnlocked(root) {
+ return root.hubGoals.isRewardUnlocked(shapez.enumHubGoalRewards.reward_wires_painter_and_levers);
+ }
+
+ getLayer() {
+ return "wires";
+ }
+
+ getDimensions() {
+ return new shapez.Vector(1, 1);
+ }
+
+ getRenderPins() {
+ // Do not show pin overlays since it would hide our building icon
+ return false;
+ }
+
+ setupEntityComponents(entity) {
+ // Accept logical input from the bottom
+ entity.addComponent(
+ new shapez.WiredPinsComponent({
+ slots: [
+ {
+ pos: new shapez.Vector(0, 0),
+ direction: shapez.enumDirection.bottom,
+ type: shapez.enumPinSlotType.logicalAcceptor,
+ },
+ ],
+ })
+ );
+
+ // Add your notification component to identify the building as a notification block
+ entity.addComponent(new NotificationBlockComponent());
+ }
+}
+
+////////////////////////////////////////////////////////////////////////
+// HUD Component to be able to edit notification blocks by clicking them
+class HUDNotificationBlockEdit extends shapez.BaseHUDPart {
+ initialize() {
+ this.root.camera.downPreHandler.add(this.downPreHandler, this);
+ }
+
+ /**
+ * @param {Vector} pos
+ * @param {enumMouseButton} button
+ */
+ downPreHandler(pos, button) {
+ if (this.root.currentLayer !== "wires") {
+ return;
+ }
+
+ const tile = this.root.camera.screenToWorld(pos).toTileSpace();
+ const contents = this.root.map.getLayerContentXY(tile.x, tile.y, "wires");
+ if (contents) {
+ const notificationComp = contents.components.NotificationBlock;
+ if (notificationComp) {
+ if (button === shapez.enumMouseButton.left) {
+ this.editNotificationText(contents, {
+ deleteOnCancel: false,
+ });
+ return shapez.STOP_PROPAGATION;
+ }
+ }
+ }
+ }
+
+ /**
+ * Asks the player to enter a notification text
+ * @param {Entity} entity
+ * @param {object} param0
+ * @param {boolean=} param0.deleteOnCancel
+ */
+ editNotificationText(entity, { deleteOnCancel = true }) {
+ const notificationComp = entity.components.NotificationBlock;
+ if (!notificationComp) {
+ return;
+ }
+
+ // save the uid because it could get stale
+ const uid = entity.uid;
+
+ // create an input field to query the text
+ const textInput = new shapez.FormElementInput({
+ id: "notificationText",
+ placeholder: "",
+ defaultValue: notificationComp.notificationText,
+ validator: val => val.length > 0,
+ });
+
+ // create the dialog & show it
+ const dialog = new shapez.DialogWithForm({
+ app: this.root.app,
+ title: shapez.T.mods.notificationBlocks.dialogTitle,
+ desc: shapez.T.mods.notificationBlocks.enterNotificationText,
+ formElements: [textInput],
+ buttons: ["cancel:bad:escape", "ok:good:enter"],
+ closeButton: false,
+ });
+ this.root.hud.parts.dialogs.internalShowDialog(dialog);
+
+ // When confirmed, set the text
+ dialog.buttonSignals.ok.add(() => {
+ if (!this.root || !this.root.entityMgr) {
+ // Game got stopped
+ return;
+ }
+
+ const entityRef = this.root.entityMgr.findByUid(uid, false);
+ if (!entityRef) {
+ // outdated
+ return;
+ }
+
+ const notificationComp = entityRef.components.NotificationBlock;
+ if (!notificationComp) {
+ // no longer interesting
+ return;
+ }
+
+ // set the text
+ notificationComp.notificationText = textInput.getValue();
+ });
+
+ // When cancelled, destroy the entity again
+ if (deleteOnCancel) {
+ dialog.buttonSignals.cancel.add(() => {
+ if (!this.root || !this.root.entityMgr) {
+ // Game got stopped
+ return;
+ }
+
+ const entityRef = this.root.entityMgr.findByUid(uid, false);
+ if (!entityRef) {
+ // outdated
+ return;
+ }
+
+ const notificationComp = entityRef.components.NotificationBlock;
+ if (!notificationComp) {
+ // no longer interesting
+ return;
+ }
+
+ this.root.logic.tryDeleteBuilding(entityRef);
+ });
+ }
+ }
+}
+
+////////////////////////////////////////////////////////////////////////
+// The actual mod logic
+class Mod extends shapez.Mod {
+ init() {
+ // Register the component
+ this.modInterface.registerComponent(NotificationBlockComponent);
+
+ // Register the new building
+ this.modInterface.registerNewBuilding({
+ metaClass: MetaNotificationBlockBuilding,
+ buildingIconBase64: RESOURCES["notification_block.png"],
+ });
+
+ // Add it to the regular toolbar
+ this.modInterface.addNewBuildingToToolbar({
+ toolbar: "wires",
+ location: "secondary",
+ metaClass: MetaNotificationBlockBuilding,
+ });
+
+ // Register our game system so we can dispatch the notifications
+ this.modInterface.registerGameSystem({
+ id: "notificationBlocks",
+ systemClass: NotificationBlocksSystem,
+ before: "constantSignal",
+ });
+
+ // Register our hud element to be able to edit the notification texts
+ this.modInterface.registerHudElement("notificationBlockEdit", HUDNotificationBlockEdit);
+
+ // This mod also supports translations
+ this.modInterface.registerTranslations("en", {
+ mods: {
+ notificationBlocks: {
+ enterNotificationText:
+ "Enter the notification text to show once the signal switches from 0 to 1:",
+ },
+ },
+ });
+ }
+}
+
+const RESOURCES = {
+ "notification_block.png":
+ "",
+};
diff --git a/shapez-io/mod_examples/pasting.js b/shapez-io/mod_examples/pasting.js
new file mode 100644
index 00000000..698edeff
--- /dev/null
+++ b/shapez-io/mod_examples/pasting.js
@@ -0,0 +1,23 @@
+// @ts-nocheck
+const METADATA = {
+ website: "https://tobspr.io",
+ author: "tobspr",
+ name: "Mod Example: Pasting",
+ version: "1",
+ id: "pasting",
+ description: "Shows how to properly receive paste events ingame",
+ minimumGameVersion: ">=1.5.0",
+};
+
+class Mod extends shapez.Mod {
+ init() {
+ this.signals.gameInitialized.add(root => {
+ root.gameState.inputReciever.paste.add(event => {
+ event.preventDefault();
+
+ const data = event.clipboardData.getData("text");
+ this.dialogs.showInfo("Pasted", `You pasted: '${data}'`);
+ });
+ });
+ }
+}
diff --git a/shapez-io/mod_examples/replace_builtin_sprites.js b/shapez-io/mod_examples/replace_builtin_sprites.js
new file mode 100644
index 00000000..6e88b7fb
--- /dev/null
+++ b/shapez-io/mod_examples/replace_builtin_sprites.js
@@ -0,0 +1,48 @@
+// @ts-nocheck
+const METADATA = {
+ website: "https://tobspr.io",
+ author: "tobspr",
+ name: "Mod Example: Replace builtin sprites",
+ version: "1",
+ id: "replace-builtin-sprites",
+ description: "Shows how to replace builtin sprites",
+ minimumGameVersion: ">=1.5.0",
+
+ // You can specify this parameter if savegames will still work
+ // after your mod has been uninstalled
+ doesNotAffectSavegame: true,
+};
+
+class Mod extends shapez.Mod {
+ init() {
+ // Replace a builtin sprite
+ ["red", "green", "blue", "yellow", "purple", "cyan", "white"].forEach(color => {
+ this.modInterface.registerSprite(`sprites/colors/${color}.png`, RESOURCES[color + ".png"]);
+ });
+ }
+}
+
+////////////////////////////////////////////////////////////////////////
+
+const RESOURCES = {
+ "red.png":
+ "",
+
+ "green.png":
+ "",
+
+ "purple.png":
+ "",
+
+ "blue.png":
+ "",
+
+ "yellow.png":
+ "",
+
+ "cyan.png":
+ "",
+
+ "white.png":
+ "",
+};
diff --git a/shapez-io/mod_examples/sandbox.js b/shapez-io/mod_examples/sandbox.js
new file mode 100644
index 00000000..f405ab59
--- /dev/null
+++ b/shapez-io/mod_examples/sandbox.js
@@ -0,0 +1,21 @@
+// @ts-nocheck
+const METADATA = {
+ website: "https://tobspr.io",
+ author: "tobspr",
+ name: "Sandbox",
+ version: "1",
+ id: "sandbox",
+ description: "Blueprints are always unlocked and cost no money, also all buildings are unlocked",
+ minimumGameVersion: ">=1.5.0",
+};
+
+class Mod extends shapez.Mod {
+ init() {
+ this.modInterface.replaceMethod(shapez.Blueprint, "getCost", function () {
+ return 0;
+ });
+ this.modInterface.replaceMethod(shapez.HubGoals, "isRewardUnlocked", function () {
+ return true;
+ });
+ }
+}
diff --git a/shapez-io/mod_examples/smooth_zooming.js b/shapez-io/mod_examples/smooth_zooming.js
new file mode 100644
index 00000000..94254f2d
--- /dev/null
+++ b/shapez-io/mod_examples/smooth_zooming.js
@@ -0,0 +1,58 @@
+// @ts-nocheck
+const METADATA = {
+ website: "https://tobspr.io",
+ author: "tobspr",
+ name: "Smooth Zoom",
+ version: "1",
+ id: "smooth_zoom",
+ description: "Allows to zoom in and out smoothly, also disables map overview",
+ minimumGameVersion: ">=1.5.0",
+};
+
+class Mod extends shapez.Mod {
+ init() {
+ this.modInterface.registerIngameKeybinding({
+ id: "smooth_zoom_zoom_in",
+ keyCode: shapez.keyToKeyCode("1"),
+ translation: "Zoom In (use with SHIFT)",
+ modifiers: {
+ shift: true,
+ },
+ handler: root => {
+ root.camera.setDesiredZoom(5);
+ return shapez.STOP_PROPAGATION;
+ },
+ });
+ this.modInterface.registerIngameKeybinding({
+ id: "smooth_zoom_zoom_out",
+ keyCode: shapez.keyToKeyCode("0"),
+ translation: "Zoom Out (use with SHIFT)",
+ modifiers: {
+ shift: true,
+ },
+ handler: root => {
+ root.camera.setDesiredZoom(0.1);
+ return shapez.STOP_PROPAGATION;
+ },
+ });
+
+ this.modInterface.extendClass(shapez.Camera, ({ $old }) => ({
+ internalUpdateZooming(now, dt) {
+ if (!this.currentlyPinching && this.desiredZoom !== null) {
+ const diff = this.zoomLevel - this.desiredZoom;
+ if (Math.abs(diff) > 0.0001) {
+ const speed = 0.0005;
+ let step = Math.sign(diff) * Math.min(speed, Math.abs(diff));
+ const pow = 1 / 2;
+ this.zoomLevel = Math.pow(Math.pow(this.zoomLevel, pow) - step, 1 / pow);
+ } else {
+ this.zoomLevel = this.desiredZoom;
+ this.desiredZoom = null;
+ }
+ }
+ },
+ }));
+
+ shapez.globalConfig.mapChunkOverviewMinZoom = -1;
+ }
+}
diff --git a/shapez-io/mod_examples/storing_data_in_savegame.js b/shapez-io/mod_examples/storing_data_in_savegame.js
new file mode 100644
index 00000000..92f7733b
--- /dev/null
+++ b/shapez-io/mod_examples/storing_data_in_savegame.js
@@ -0,0 +1,78 @@
+// @ts-nocheck
+const METADATA = {
+ website: "https://tobspr.io",
+ author: "tobspr",
+ name: "Mod Example: Storing Data in Savegame",
+ version: "1",
+ id: "storing-savegame-data",
+ description: "Shows how to add custom data to a savegame",
+ minimumGameVersion: ">=1.5.0",
+};
+
+class Mod extends shapez.Mod {
+ init() {
+ ////////////////////////////////////////////////////////////////////
+ // Option 1: For simple data
+ this.signals.gameSerialized.add((root, data) => {
+ data.modExtraData["storing-savegame-data"] = Math.random();
+ });
+
+ this.signals.gameDeserialized.add((root, data) => {
+ alert("The value stored in the savegame was: " + data.modExtraData["storing-savegame-data"]);
+ });
+
+ ////////////////////////////////////////////////////////////////////
+ // Option 2: If you need a structured way of storing data
+
+ class SomeSerializableObject extends shapez.BasicSerializableObject {
+ static getId() {
+ return "SomeSerializableObject";
+ }
+
+ static getSchema() {
+ return {
+ someInt: shapez.types.int,
+ someString: shapez.types.string,
+ someVector: shapez.types.vector,
+
+ // this value is allowed to be null
+ nullableInt: shapez.types.nullable(shapez.types.int),
+
+ // There is a lot more .. be sure to checkout src/js/savegame/serialization.js
+ // You can have maps, classes, arrays etc..
+ // And if you need something specific you can always ask in the modding discord.
+ };
+ }
+
+ constructor() {
+ super();
+ this.someInt = 42;
+ this.someString = "Hello World";
+ this.someVector = new shapez.Vector(1, 2);
+
+ this.nullableInt = null;
+ }
+ }
+
+ // Store our object in the global game root
+ this.signals.gameInitialized.add(root => {
+ root.myObject = new SomeSerializableObject();
+ });
+
+ // Save it within the savegame
+ this.signals.gameSerialized.add((root, data) => {
+ data.modExtraData["storing-savegame-data-2"] = root.myObject.serialize();
+ });
+
+ // Restore it when the savegame is loaded
+ this.signals.gameDeserialized.add((root, data) => {
+ const errorText = root.myObject.deserialize(data.modExtraData["storing-savegame-data-2"]);
+ if (errorText) {
+ alert("Mod failed to deserialize from savegame: " + errorText);
+ }
+ alert("The other value stored in the savegame (option 2) was " + root.myObject.someInt);
+ });
+
+ ////////////////////////////////////////////////////////////////////
+ }
+}
diff --git a/shapez-io/mod_examples/translations.js b/shapez-io/mod_examples/translations.js
new file mode 100644
index 00000000..6b9c708e
--- /dev/null
+++ b/shapez-io/mod_examples/translations.js
@@ -0,0 +1,66 @@
+// @ts-nocheck
+const METADATA = {
+ website: "https://tobspr.io",
+ author: "tobspr",
+ name: "Mod Example: Translations",
+ version: "1",
+ id: "translations",
+ description: "Shows how to add and modify translations",
+ minimumGameVersion: ">=1.5.0",
+
+ // You can specify this parameter if savegames will still work
+ // after your mod has been uninstalled
+ doesNotAffectSavegame: true,
+};
+
+class Mod extends shapez.Mod {
+ init() {
+ // Replace an existing translation in the english language
+ this.modInterface.registerTranslations("en", {
+ ingame: {
+ interactiveTutorial: {
+ title: "Hello",
+ hints: {
+ "1_1_extractor": "World!",
+ },
+ },
+ },
+ });
+
+ // Replace an existing translation in german
+ this.modInterface.registerTranslations("de", {
+ ingame: {
+ interactiveTutorial: {
+ title: "Hallo",
+ hints: {
+ "1_1_extractor": "Welt!",
+ },
+ },
+ },
+ });
+
+ // Add an entirely new translation which is localized in german and english
+ this.modInterface.registerTranslations("en", {
+ mods: {
+ mymod: {
+ test: "Test Translation",
+ },
+ },
+ });
+ this.modInterface.registerTranslations("de", {
+ mods: {
+ mymod: {
+ test: "Test Übersetzung",
+ },
+ },
+ });
+
+ // Show a dialog in the main menu
+ this.signals.stateEntered.add(state => {
+ if (state instanceof shapez.MainMenuState) {
+ // Will show differently based on the selected language
+ this.dialogs.showInfo("My translation", shapez.T.mods.mymod.test);
+ }
+ });
+ }
+}
diff --git a/shapez-io/mod_examples/usage_statistics.js b/shapez-io/mod_examples/usage_statistics.js
new file mode 100644
index 00000000..80828d95
--- /dev/null
+++ b/shapez-io/mod_examples/usage_statistics.js
@@ -0,0 +1,148 @@
+// @ts-nocheck
+const METADATA = {
+ website: "https://tobspr.io",
+ author: "tobspr",
+ name: "Mod Example: Usage Statistics",
+ version: "1",
+ id: "usage-statistics",
+ description:
+ "Shows how to add a new component to the game, how to save additional data and how to add custom logic and drawings",
+
+ minimumGameVersion: ">=1.5.0",
+};
+
+/**
+ * Quick info on how this mod works:
+ *
+ * It tracks how many ticks a building was idle within X seconds to compute
+ * the usage percentage.
+ *
+ * Every tick the logic checks if the building is idle, if so, it increases aggregatedIdleTime.
+ * Once X seconds are over, the aggregatedIdleTime is copied to computedUsage which
+ * is displayed on screen via the UsageStatisticsSystem
+ */
+
+const MEASURE_INTERVAL_SECONDS = 5;
+
+class UsageStatisticsComponent extends shapez.Component {
+ static getId() {
+ return "UsageStatistics";
+ }
+
+ static getSchema() {
+ // Here you define which properties should be saved to the savegame
+ // and get automatically restored
+ return {
+ lastTimestamp: shapez.types.float,
+ computedUsage: shapez.types.float,
+ aggregatedIdleTime: shapez.types.float,
+ };
+ }
+
+ constructor() {
+ super();
+ this.lastTimestamp = 0;
+ this.computedUsage = 0;
+ this.aggregatedIdleTime = 0;
+ }
+}
+
+class UsageStatisticsSystem extends shapez.GameSystemWithFilter {
+ constructor(root) {
+ // By specifying the list of components, `this.allEntities` will only
+ // contain entities which have *all* of the specified components
+ super(root, [UsageStatisticsComponent, shapez.ItemProcessorComponent]);
+ }
+
+ update() {
+ const now = this.root.time.now();
+ for (let i = 0; i < this.allEntities.length; ++i) {
+ const entity = this.allEntities[i];
+
+ const processorComp = entity.components.ItemProcessor;
+ const usageComp = entity.components.UsageStatistics;
+
+ if (now - usageComp.lastTimestamp > MEASURE_INTERVAL_SECONDS) {
+ usageComp.computedUsage = shapez.clamp(
+ 1 - usageComp.aggregatedIdleTime / MEASURE_INTERVAL_SECONDS
+ );
+ usageComp.aggregatedIdleTime = 0;
+ usageComp.lastTimestamp = now;
+ }
+
+ if (processorComp.ongoingCharges.length === 0) {
+ usageComp.aggregatedIdleTime += this.root.dynamicTickrate.deltaSeconds;
+ }
+ }
+ }
+
+ drawChunk(parameters, chunk) {
+ const contents = chunk.containedEntitiesByLayer.regular;
+ for (let i = 0; i < contents.length; ++i) {
+ const entity = contents[i];
+ const usageComp = entity.components.UsageStatistics;
+ if (!usageComp) {
+ continue;
+ }
+
+ const staticComp = entity.components.StaticMapEntity;
+ const context = parameters.context;
+ const center = staticComp.getTileSpaceBounds().getCenter().toWorldSpace();
+
+ // Culling for better performance
+ if (parameters.visibleRect.containsCircle(center.x, center.y, 40)) {
+ // Background badge
+ context.fillStyle = "rgba(250, 250, 250, 0.8)";
+ context.beginRoundedRect(center.x - 10, center.y + 3, 20, 8, 2);
+
+ context.fill();
+
+ // Text
+ const usage = usageComp.computedUsage * 100.0;
+ if (usage > 99.99) {
+ context.fillStyle = "green";
+ } else if (usage > 70) {
+ context.fillStyle = "orange";
+ } else {
+ context.fillStyle = "red";
+ }
+
+ context.textAlign = "center";
+ context.font = "7px GameFont";
+ context.fillText(Math.round(usage) + "%", center.x, center.y + 10);
+ }
+ }
+ }
+}
+
+class Mod extends shapez.Mod {
+ init() {
+ // Register the component
+ this.modInterface.registerComponent(UsageStatisticsComponent);
+
+ // Add our new component to all item processor buildings so we can see how many items it processed.
+ // You can also inspect the entity with F8
+ const buildings = [
+ shapez.MetaBalancerBuilding,
+ shapez.MetaCutterBuilding,
+ shapez.MetaRotaterBuilding,
+ shapez.MetaStackerBuilding,
+ shapez.MetaMixerBuilding,
+ shapez.MetaPainterBuilding,
+ ];
+
+ buildings.forEach(metaClass => {
+ this.modInterface.runAfterMethod(metaClass, "setupEntityComponents", function (entity) {
+ entity.addComponent(new UsageStatisticsComponent());
+ });
+ });
+
+ // Register our game system so we can update and draw stuff
+ this.modInterface.registerGameSystem({
+ id: "demo_mod",
+ systemClass: UsageStatisticsSystem,
+ before: "belt",
+ drawHooks: ["staticAfter"],
+ });
+ }
+}
diff --git a/shapez-io/res/ui/interactive_tutorial.noinline/1_1_extractor.gif b/shapez-io/res/ui/interactive_tutorial.noinline/1_1_extractor.gif
new file mode 100644
index 00000000..c7208ac2
Binary files /dev/null and b/shapez-io/res/ui/interactive_tutorial.noinline/1_1_extractor.gif differ
diff --git a/shapez-io/res/ui/interactive_tutorial.noinline/1_2_conveyor.gif b/shapez-io/res/ui/interactive_tutorial.noinline/1_2_conveyor.gif
new file mode 100644
index 00000000..db59bfd4
Binary files /dev/null and b/shapez-io/res/ui/interactive_tutorial.noinline/1_2_conveyor.gif differ
diff --git a/shapez-io/res/ui/interactive_tutorial.noinline/1_3_expand.gif b/shapez-io/res/ui/interactive_tutorial.noinline/1_3_expand.gif
new file mode 100644
index 00000000..9c655ab1
Binary files /dev/null and b/shapez-io/res/ui/interactive_tutorial.noinline/1_3_expand.gif differ
diff --git a/shapez-io/res/ui/interactive_tutorial.noinline/21_1_place_quad_painter.gif b/shapez-io/res/ui/interactive_tutorial.noinline/21_1_place_quad_painter.gif
new file mode 100644
index 00000000..ea854cf2
Binary files /dev/null and b/shapez-io/res/ui/interactive_tutorial.noinline/21_1_place_quad_painter.gif differ
diff --git a/shapez-io/res/ui/interactive_tutorial.noinline/21_2_switch_to_wires.gif b/shapez-io/res/ui/interactive_tutorial.noinline/21_2_switch_to_wires.gif
new file mode 100644
index 00000000..78ab6fd2
Binary files /dev/null and b/shapez-io/res/ui/interactive_tutorial.noinline/21_2_switch_to_wires.gif differ
diff --git a/shapez-io/res/ui/interactive_tutorial.noinline/21_3_place_button.gif b/shapez-io/res/ui/interactive_tutorial.noinline/21_3_place_button.gif
new file mode 100644
index 00000000..52ffb076
Binary files /dev/null and b/shapez-io/res/ui/interactive_tutorial.noinline/21_3_place_button.gif differ
diff --git a/shapez-io/res/ui/interactive_tutorial.noinline/21_4_press_button.gif b/shapez-io/res/ui/interactive_tutorial.noinline/21_4_press_button.gif
new file mode 100644
index 00000000..5d79f1e3
Binary files /dev/null and b/shapez-io/res/ui/interactive_tutorial.noinline/21_4_press_button.gif differ
diff --git a/shapez-io/res/ui/interactive_tutorial.noinline/2_1_place_cutter.gif b/shapez-io/res/ui/interactive_tutorial.noinline/2_1_place_cutter.gif
new file mode 100644
index 00000000..1678c0b2
Binary files /dev/null and b/shapez-io/res/ui/interactive_tutorial.noinline/2_1_place_cutter.gif differ
diff --git a/shapez-io/res/ui/interactive_tutorial.noinline/2_2_place_trash.gif b/shapez-io/res/ui/interactive_tutorial.noinline/2_2_place_trash.gif
new file mode 100644
index 00000000..0d60fa9f
Binary files /dev/null and b/shapez-io/res/ui/interactive_tutorial.noinline/2_2_place_trash.gif differ
diff --git a/shapez-io/res/ui/interactive_tutorial.noinline/2_3_more_cutters.gif b/shapez-io/res/ui/interactive_tutorial.noinline/2_3_more_cutters.gif
new file mode 100644
index 00000000..50ce88f9
Binary files /dev/null and b/shapez-io/res/ui/interactive_tutorial.noinline/2_3_more_cutters.gif differ
diff --git a/shapez-io/res/ui/interactive_tutorial.noinline/3_1_rectangles.gif b/shapez-io/res/ui/interactive_tutorial.noinline/3_1_rectangles.gif
new file mode 100644
index 00000000..418d3123
Binary files /dev/null and b/shapez-io/res/ui/interactive_tutorial.noinline/3_1_rectangles.gif differ
diff --git a/shapez-io/res/ui/languages/ar.svg b/shapez-io/res/ui/languages/ar.svg
new file mode 100644
index 00000000..9b682077
--- /dev/null
+++ b/shapez-io/res/ui/languages/ar.svg
@@ -0,0 +1,62 @@
+
+
+
diff --git a/shapez-io/res/ui/languages/cs.svg b/shapez-io/res/ui/languages/cs.svg
new file mode 100644
index 00000000..653b342e
--- /dev/null
+++ b/shapez-io/res/ui/languages/cs.svg
@@ -0,0 +1,37 @@
+
+
+
diff --git a/shapez-io/res/ui/languages/da.svg b/shapez-io/res/ui/languages/da.svg
new file mode 100644
index 00000000..b6ace9ab
--- /dev/null
+++ b/shapez-io/res/ui/languages/da.svg
@@ -0,0 +1,38 @@
+
+
+
diff --git a/shapez-io/res/ui/languages/de.svg b/shapez-io/res/ui/languages/de.svg
new file mode 100644
index 00000000..4585512c
--- /dev/null
+++ b/shapez-io/res/ui/languages/de.svg
@@ -0,0 +1,39 @@
+
+
+
diff --git a/shapez-io/res/ui/languages/el.svg b/shapez-io/res/ui/languages/el.svg
new file mode 100644
index 00000000..90f83f5e
--- /dev/null
+++ b/shapez-io/res/ui/languages/el.svg
@@ -0,0 +1,46 @@
+
+
+
diff --git a/shapez-io/res/ui/languages/en.svg b/shapez-io/res/ui/languages/en.svg
new file mode 100644
index 00000000..a0805a7d
--- /dev/null
+++ b/shapez-io/res/ui/languages/en.svg
@@ -0,0 +1,59 @@
+
+
+
diff --git a/shapez-io/res/ui/languages/es-419.svg b/shapez-io/res/ui/languages/es-419.svg
new file mode 100644
index 00000000..5f671625
--- /dev/null
+++ b/shapez-io/res/ui/languages/es-419.svg
@@ -0,0 +1,107 @@
+
+
+
diff --git a/shapez-io/res/ui/languages/et.svg b/shapez-io/res/ui/languages/et.svg
new file mode 100644
index 00000000..63ae7dd8
--- /dev/null
+++ b/shapez-io/res/ui/languages/et.svg
@@ -0,0 +1,37 @@
+
+
+
diff --git a/shapez-io/res/ui/languages/fi.svg b/shapez-io/res/ui/languages/fi.svg
new file mode 100644
index 00000000..483fdde3
--- /dev/null
+++ b/shapez-io/res/ui/languages/fi.svg
@@ -0,0 +1,38 @@
+
+
+
diff --git a/shapez-io/res/ui/languages/fr.svg b/shapez-io/res/ui/languages/fr.svg
new file mode 100644
index 00000000..632ffc1d
--- /dev/null
+++ b/shapez-io/res/ui/languages/fr.svg
@@ -0,0 +1,37 @@
+
+
+
diff --git a/shapez-io/res/ui/languages/he.svg b/shapez-io/res/ui/languages/he.svg
new file mode 100644
index 00000000..aaa64e98
--- /dev/null
+++ b/shapez-io/res/ui/languages/he.svg
@@ -0,0 +1,42 @@
+
+
+
diff --git a/shapez-io/res/ui/languages/hu.svg b/shapez-io/res/ui/languages/hu.svg
new file mode 100644
index 00000000..39298040
--- /dev/null
+++ b/shapez-io/res/ui/languages/hu.svg
@@ -0,0 +1,37 @@
+
+
+
diff --git a/shapez-io/res/ui/languages/it.svg b/shapez-io/res/ui/languages/it.svg
new file mode 100644
index 00000000..a32bf996
--- /dev/null
+++ b/shapez-io/res/ui/languages/it.svg
@@ -0,0 +1,37 @@
+
+
+
diff --git a/shapez-io/res/ui/languages/ja.svg b/shapez-io/res/ui/languages/ja.svg
new file mode 100644
index 00000000..6657b409
--- /dev/null
+++ b/shapez-io/res/ui/languages/ja.svg
@@ -0,0 +1,36 @@
+
+
+
diff --git a/shapez-io/res/ui/languages/kor.svg b/shapez-io/res/ui/languages/kor.svg
new file mode 100644
index 00000000..6331281b
--- /dev/null
+++ b/shapez-io/res/ui/languages/kor.svg
@@ -0,0 +1,99 @@
+
+
+
diff --git a/shapez-io/res/ui/languages/mt-MT.svg b/shapez-io/res/ui/languages/mt-MT.svg
new file mode 100644
index 00000000..59e6165a
--- /dev/null
+++ b/shapez-io/res/ui/languages/mt-MT.svg
@@ -0,0 +1,50 @@
+
+
+
diff --git a/shapez-io/res/ui/languages/nb.svg b/shapez-io/res/ui/languages/nb.svg
new file mode 100644
index 00000000..64d2fa5e
--- /dev/null
+++ b/shapez-io/res/ui/languages/nb.svg
@@ -0,0 +1,37 @@
+
+
+
diff --git a/shapez-io/res/ui/languages/nl.svg b/shapez-io/res/ui/languages/nl.svg
new file mode 100644
index 00000000..edd8b343
--- /dev/null
+++ b/shapez-io/res/ui/languages/nl.svg
@@ -0,0 +1,37 @@
+
+
+
diff --git a/shapez-io/res/ui/languages/no.svg b/shapez-io/res/ui/languages/no.svg
new file mode 100644
index 00000000..5f5a4b6c
--- /dev/null
+++ b/shapez-io/res/ui/languages/no.svg
@@ -0,0 +1,41 @@
+
+
+
diff --git a/shapez-io/res/ui/languages/pl.svg b/shapez-io/res/ui/languages/pl.svg
new file mode 100644
index 00000000..b471827f
--- /dev/null
+++ b/shapez-io/res/ui/languages/pl.svg
@@ -0,0 +1,36 @@
+
+
+
diff --git a/shapez-io/res/ui/languages/pt-BR.svg b/shapez-io/res/ui/languages/pt-BR.svg
new file mode 100644
index 00000000..b1470d95
--- /dev/null
+++ b/shapez-io/res/ui/languages/pt-BR.svg
@@ -0,0 +1,51 @@
+
+
+
diff --git a/shapez-io/res/ui/languages/pt-PT.svg b/shapez-io/res/ui/languages/pt-PT.svg
new file mode 100644
index 00000000..4daa9aeb
--- /dev/null
+++ b/shapez-io/res/ui/languages/pt-PT.svg
@@ -0,0 +1,64 @@
+
+
+
diff --git a/shapez-io/res/ui/languages/ro.svg b/shapez-io/res/ui/languages/ro.svg
new file mode 100644
index 00000000..041ecc3f
--- /dev/null
+++ b/shapez-io/res/ui/languages/ro.svg
@@ -0,0 +1,38 @@
+
+
+
diff --git a/shapez-io/res/ui/languages/ru.svg b/shapez-io/res/ui/languages/ru.svg
new file mode 100644
index 00000000..ecd327ac
--- /dev/null
+++ b/shapez-io/res/ui/languages/ru.svg
@@ -0,0 +1,37 @@
+
+
+
diff --git a/shapez-io/res/ui/languages/sv.svg b/shapez-io/res/ui/languages/sv.svg
new file mode 100644
index 00000000..3754ad8c
--- /dev/null
+++ b/shapez-io/res/ui/languages/sv.svg
@@ -0,0 +1,36 @@
+
+
+
diff --git a/shapez-io/res/ui/languages/th.svg b/shapez-io/res/ui/languages/th.svg
new file mode 100644
index 00000000..93a280b1
--- /dev/null
+++ b/shapez-io/res/ui/languages/th.svg
@@ -0,0 +1,43 @@
+
+
+
diff --git a/shapez-io/res/ui/languages/tr.svg b/shapez-io/res/ui/languages/tr.svg
new file mode 100644
index 00000000..15d06a67
--- /dev/null
+++ b/shapez-io/res/ui/languages/tr.svg
@@ -0,0 +1,39 @@
+
+
+
diff --git a/shapez-io/res/ui/languages/uk.svg b/shapez-io/res/ui/languages/uk.svg
new file mode 100644
index 00000000..4d7db7f1
--- /dev/null
+++ b/shapez-io/res/ui/languages/uk.svg
@@ -0,0 +1,36 @@
+
+
+
diff --git a/shapez-io/res/ui/languages/vi.svg b/shapez-io/res/ui/languages/vi.svg
new file mode 100644
index 00000000..0aa76a0e
--- /dev/null
+++ b/shapez-io/res/ui/languages/vi.svg
@@ -0,0 +1,36 @@
+
+
+
diff --git a/shapez-io/res/ui/languages/zh-CN.svg b/shapez-io/res/ui/languages/zh-CN.svg
new file mode 100644
index 00000000..f89219a0
--- /dev/null
+++ b/shapez-io/res/ui/languages/zh-CN.svg
@@ -0,0 +1,42 @@
+
+
+
diff --git a/shapez-io/res/ui/languages/zh-TW.svg b/shapez-io/res/ui/languages/zh-TW.svg
new file mode 100644
index 00000000..c3dab661
--- /dev/null
+++ b/shapez-io/res/ui/languages/zh-TW.svg
@@ -0,0 +1,43 @@
+
+
+