| 
									
										
										
										
											2020-05-21 11:05:21 +00:00
										 |  |  | /* eslint-disable quotes,no-undef */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const { app, BrowserWindow, Menu, MenuItem, session } = require("electron"); | 
					
						
							|  |  |  | const path = require("path"); | 
					
						
							|  |  |  | const url = require("url"); | 
					
						
							|  |  |  | const childProcess = require("child_process"); | 
					
						
							| 
									
										
										
										
											2020-10-31 11:21:39 +00:00
										 |  |  | const { ipcMain, shell } = require("electron"); | 
					
						
							| 
									
										
										
										
											2020-05-21 11:05:21 +00:00
										 |  |  | const fs = require("fs"); | 
					
						
							| 
									
										
										
										
											2021-03-12 11:44:40 +00:00
										 |  |  | const steam = require("./steam"); | 
					
						
							| 
									
										
										
										
											2020-05-21 11:05:21 +00:00
										 |  |  | 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: "shapez.io Standalone", | 
					
						
							|  |  |  |         transparent: false, | 
					
						
							|  |  |  |         icon: path.join(__dirname, "favicon" + faviconExtension), | 
					
						
							|  |  |  |         // fullscreen: true,
 | 
					
						
							|  |  |  |         autoHideMenuBar: true, | 
					
						
							|  |  |  |         webPreferences: { | 
					
						
							|  |  |  |             nodeIntegration: true, | 
					
						
							|  |  |  |             webSecurity: false, | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         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(); | 
					
						
							|  |  |  |     win.webContents.session.clearStorageData(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     win.webContents.on("new-window", (event, pth) => { | 
					
						
							|  |  |  |         event.preventDefault(); | 
					
						
							| 
									
										
										
										
											2020-10-31 11:21:39 +00:00
										 |  |  |         shell.openExternal(pth); | 
					
						
							| 
									
										
										
										
											2020-05-21 11:05:21 +00:00
										 |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     win.on("closed", () => { | 
					
						
							|  |  |  |         console.log("Window closed"); | 
					
						
							|  |  |  |         win = null; | 
					
						
							|  |  |  |         app.quit(); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function handleWindowBeforeunload(event) { | 
					
						
							|  |  |  |         const confirmed = dialog.showMessageBox(remote.getCurrentWindow(), options) === 1; | 
					
						
							|  |  |  |         if (confirmed) { | 
					
						
							|  |  |  |             remote.getCurrentWindow().close(); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             event.returnValue = false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     win.on("", handleWindowBeforeunload); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (isDev) { | 
					
						
							|  |  |  |         menu = new Menu(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const mainItem = new MenuItem({ | 
					
						
							|  |  |  |             label: "Toggle Dev Tools", | 
					
						
							|  |  |  |             click: () => win.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(); | 
					
						
							|  |  |  | }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function performFsJob(job) { | 
					
						
							|  |  |  |     const fname = path.join(storePath, job.filename); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     switch (job.type) { | 
					
						
							|  |  |  |         case "read": { | 
					
						
							|  |  |  |             if (!fs.existsSync(fname)) { | 
					
						
							|  |  |  |                 return { | 
					
						
							|  |  |  |                     // Special FILE_NOT_FOUND error code
 | 
					
						
							|  |  |  |                     error: "file_not_found", | 
					
						
							|  |  |  |                 }; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             let contents = ""; | 
					
						
							|  |  |  |             try { | 
					
						
							|  |  |  |                 contents = fs.readFileSync(fname, { encoding: "utf8" }); | 
					
						
							|  |  |  |             } catch (ex) { | 
					
						
							|  |  |  |                 return { | 
					
						
							|  |  |  |                     error: ex, | 
					
						
							|  |  |  |                 }; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return { | 
					
						
							|  |  |  |                 success: true, | 
					
						
							|  |  |  |                 data: contents, | 
					
						
							|  |  |  |             }; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         case "write": { | 
					
						
							|  |  |  |             try { | 
					
						
							|  |  |  |                 fs.writeFileSync(fname, job.contents); | 
					
						
							|  |  |  |             } catch (ex) { | 
					
						
							|  |  |  |                 return { | 
					
						
							|  |  |  |                     error: ex, | 
					
						
							|  |  |  |                 }; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return { | 
					
						
							|  |  |  |                 success: true, | 
					
						
							|  |  |  |                 data: job.contents, | 
					
						
							|  |  |  |             }; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         case "delete": { | 
					
						
							|  |  |  |             try { | 
					
						
							|  |  |  |                 fs.unlinkSync(fname); | 
					
						
							|  |  |  |             } catch (ex) { | 
					
						
							|  |  |  |                 return { | 
					
						
							|  |  |  |                     error: ex, | 
					
						
							|  |  |  |                 }; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return { | 
					
						
							|  |  |  |                 success: true, | 
					
						
							|  |  |  |                 data: null, | 
					
						
							|  |  |  |             }; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             throw new Error("Unkown fs job: " + job.type); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ipcMain.on("fs-job", (event, arg) => { | 
					
						
							|  |  |  |     const result = performFsJob(arg); | 
					
						
							|  |  |  |     event.reply("fs-response", { id: arg.id, result }); | 
					
						
							|  |  |  | }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ipcMain.on("fs-sync-job", (event, arg) => { | 
					
						
							|  |  |  |     const result = performFsJob(arg); | 
					
						
							|  |  |  |     event.returnValue = result; | 
					
						
							|  |  |  | }); | 
					
						
							| 
									
										
											  
											
												Achievements (#1087)
* [WIP] Add boilerplate for achievement implementation
* Add config.local.template.js and rm cached copy of config.local.js
* [WIP] Implement painting, cutting, rotating achievements (to log only)
* [WIP] Refactor achievements, jsdoc fixes, add npm script
- Refactor achievements to make use of Signals
- Move implemented achievement interfaces to appropriate
platform folders (SteamAchievements in currently in use
in browser wrapper for testing)
- Fix invalid jsdocs
- Add dev-standalone script to package.json scripts
* Add steam/greenworks IPC calls and optional private-artifact dependency
* Include private artifacts in standalone builds
* Uncomment appid include
* [WIP] Add steam overlay fix, add hash to artifact dependency
* Update electron, greenworks. Add task to add local config if not present
* Add more achievements, refactor achievement code
* Add receiver flexibility and more achievements
- Add check to see if necessary to create achievement and add receiver
- Add remove receiver functionality when achievement is unlocked
* Add achievements and accommodations for switching states
- Fix startup code to avoid clobbering achievements on state switch
- Add a few more achievements
* Add achievements, ids. Update names, keys for consistency
* Add play time achievements
* [WIP] Add more achievements
* Add more achievements. Add bulk achievement check signal
* [WIP] Add achievements. Start savefile migration
* Add achievements. Add savefile migration
* Remove superfluous achievement stat
* Update lock files, fix merge conflict
											
										 
											2021-03-10 06:33:39 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | steam.init(isDev); | 
					
						
							|  |  |  | steam.listen(); |