import { cachebust } from "../core/cachebust"; import { globalConfig, openStandaloneLink, THIRDPARTY_URLS } from "../core/config"; import { GameState } from "../core/game_state"; import { DialogWithForm } from "../core/modal_dialog_elements"; import { FormElementInput } from "../core/modal_dialog_forms"; import { ReadWriteProxy } from "../core/read_write_proxy"; import { STOP_PROPAGATION } from "../core/signal"; import { WEB_STEAM_SSO_AUTHENTICATED } from "../core/steam_sso"; import { formatSecondsToTimeAgo, generateFileDownload, getLogoSprite, makeButton, makeDiv, makeDivElement, removeAllChildren, startFileChoose, waitNextFrame, } from "../core/utils"; import { HUDModalDialogs } from "../game/hud/parts/modal_dialogs"; import { MODS } from "../mods/modloader"; import { PlatformWrapperImplBrowser } from "../platform/browser/wrapper"; import { PlatformWrapperImplElectron } from "../platform/electron/wrapper"; import { Savegame } from "../savegame/savegame"; import { T } from "../translations"; const trim = require("trim"); /** * @typedef {import("../savegame/savegame_typedefs").SavegameMetadata} SavegameMetadata * @typedef {import("../profile/setting_types").EnumSetting} EnumSetting */ export class MainMenuState extends GameState { constructor() { super("MainMenuState"); this.refreshInterval = null; } getInnerHTML() { const showLanguageIcon = !G_CHINA_VERSION && !G_WEGAME_VERSION; const showExitAppButton = G_IS_STANDALONE; const showPuzzleDLC = !G_WEGAME_VERSION && (G_IS_STANDALONE || WEB_STEAM_SSO_AUTHENTICATED) && !G_IS_STEAM_DEMO && !G_GOG_VERSION; const showWegameFooter = G_WEGAME_VERSION; const hasMods = MODS.anyModsActive(); const hasSteamBridge = !G_GOG_VERSION && !G_IS_STEAM_DEMO; let showExternalLinks = true; if (G_IS_STANDALONE) { if (G_WEGAME_VERSION || G_CHINA_VERSION) { showExternalLinks = false; } } else { const wrapper = /** @type {PlatformWrapperImplBrowser} */ (this.app.platformWrapper); if (!wrapper.embedProvider.externalLinks) { showExternalLinks = false; } } let showDiscordLink = showExternalLinks; if (G_CHINA_VERSION) { showDiscordLink = true; } const showDemoAdvertisement = (showExternalLinks || G_CHINA_VERSION) && this.app.restrictionMgr.getIsStandaloneMarketingActive(); const ownsPuzzleDLC = WEB_STEAM_SSO_AUTHENTICATED || (G_IS_STANDALONE && !G_IS_STEAM_DEMO && /** @type { PlatformWrapperImplElectron}*/ (this.app.platformWrapper).dlcs.puzzle); const showShapez2 = showExternalLinks && MODS.mods.length === 0; const bannerHtml = `

${T.demoBanners.titleV2}

${Array.from(Object.entries(T.ingame.standaloneAdvantages.points)) .slice(0, 6) .map( ([key, trans]) => `
${trans.title}

${trans.desc}

` ) .join("")}
${ globalConfig.currentDiscount > 0 ? `${T.global.discount.replace( "", String(globalConfig.currentDiscount) )}` : "" } Play shapez on Steam ${!G_IS_STEAM_DEMO ? `
` : ""} `; return `
${ showLanguageIcon ? `` : "" } ${showExitAppButton ? `` : ""}
${ hasSteamBridge && (G_IS_STANDALONE || !WEB_STEAM_SSO_AUTHENTICATED) ? `
${ G_IS_STANDALONE ? T.mainMenu.playFullVersionStandalone : T.mainMenu.playFullVersionV2 } Sign in
` : "" } ${ hasSteamBridge && WEB_STEAM_SSO_AUTHENTICATED ? `
${T.mainMenu.playingFullVersion} ${T.mainMenu.logout}
` : "" }
${showDemoAdvertisement ? `
${bannerHtml}
` : ""} ${ showShapez2 ? `
We are currently prototyping Shapez 2!
` : "" } ${ showPuzzleDLC ? ` ${ ownsPuzzleDLC && !hasMods ? `
` : "" } ${ !ownsPuzzleDLC && !hasMods ? `

${T.mainMenu.puzzleDlcText}

` : "" } ` : "" } ${ hasMods ? `

${T.mods.title}

${MODS.mods .map(mod => { return `
${mod.metadata.name}
by ${mod.metadata.author}
`; }) .join("")}
${T.mainMenu.mods.warningPuzzleDLC}
` : "" }
${ showWegameFooter ? ` ` : ` ` } `; } /** * Asks the user to import a savegame */ requestImportSavegame() { if ( this.app.savegameMgr.getSavegamesMetaData().length > 0 && !this.app.restrictionMgr.getHasUnlimitedSavegames() ) { this.showSavegameSlotLimit(); return; } this.app.gameAnalytics.note("startimport"); // Create a 'fake' file-input to accept savegames startFileChoose(".bin").then(file => { if (file) { const closeLoader = this.dialogs.showLoadingDialog(); waitNextFrame().then(() => { const reader = new FileReader(); reader.addEventListener("load", event => { const contents = event.target.result; let realContent; try { realContent = ReadWriteProxy.deserializeObject(contents); } catch (err) { closeLoader(); this.dialogs.showWarning( T.dialogs.importSavegameError.title, T.dialogs.importSavegameError.text + "

" + err ); return; } this.app.savegameMgr.importSavegame(realContent).then( () => { closeLoader(); this.dialogs.showWarning( T.dialogs.importSavegameSuccess.title, T.dialogs.importSavegameSuccess.text ); this.renderMainMenu(); this.renderSavegames(); }, err => { closeLoader(); this.dialogs.showWarning( T.dialogs.importSavegameError.title, T.dialogs.importSavegameError.text + ":

" + err ); } ); }); reader.addEventListener("error", error => { this.dialogs.showWarning( T.dialogs.importSavegameError.title, T.dialogs.importSavegameError.text + ":

" + error ); }); reader.readAsText(file, "utf-8"); }); } }); } onBackButton() { this.app.platformWrapper.exitApp(); } onEnter(payload) { // Start loading already const app = this.app; setTimeout(() => app.backgroundResourceLoader.getIngamePromise(), 10); this.dialogs = new HUDModalDialogs(null, this.app); const dialogsElement = document.body.querySelector(".modalDialogParent"); this.dialogs.initializeToElement(dialogsElement); if (payload.loadError) { this.dialogs.showWarning( T.dialogs.gameLoadFailure.title, T.dialogs.gameLoadFailure.text + "

" + payload.loadError ); } if (G_IS_DEV && globalConfig.debug.testPuzzleMode) { this.onPuzzleModeButtonClicked(true); return; } if (G_IS_DEV && globalConfig.debug.fastGameEnter) { const games = this.app.savegameMgr.getSavegamesMetaData(); if (games.length > 0 && globalConfig.debug.resumeGameOnFastEnter) { this.resumeGame(games[0]); } else { this.onPlayButtonClicked(); } } // Initialize video this.videoElement = this.htmlElement.querySelector("video"); this.videoElement.playbackRate = 0.9; this.videoElement.addEventListener("canplay", () => { if (this.videoElement) { this.videoElement.classList.add("loaded"); } }); const clickHandling = { ".settingsButton": this.onSettingsButtonClicked, ".languageChoose": this.onLanguageChooseClicked, ".redditLink": this.onRedditClicked, ".twitterLink": this.onTwitterLinkClicked, ".patreonLink": this.onPatreonLinkClicked, ".changelog": this.onChangelogClicked, ".helpTranslate": this.onTranslationHelpLinkClicked, ".exitAppButton": this.onExitAppButtonClicked, ".steamLink": this.onSteamLinkClicked, ".steamLinkSocial": this.onSteamLinkClickedSocial, ".shapez2": this.onShapez2Clicked, ".discordLink": () => { this.app.platformWrapper.openExternalLink(THIRDPARTY_URLS.discord); }, ".githubLink": () => { this.app.platformWrapper.openExternalLink(THIRDPARTY_URLS.github); }, ".puzzleDlcPlayButton": this.onPuzzleModeButtonClicked, ".puzzleDlcGetButton": this.onPuzzleWishlistButtonClicked, ".wegameDisclaimer > .rating": this.onWegameRatingClicked, ".editMods": this.onModsClicked, }; for (const key in clickHandling) { const handler = clickHandling[key]; const element = this.htmlElement.querySelector(key); if (element) { this.trackClicks(element, handler, { preventClick: true }); } } this.renderMainMenu(); this.renderSavegames(); this.fetchPlayerCount(); this.refreshInterval = setInterval(() => this.fetchPlayerCount(), 10000); this.app.gameAnalytics.noteMinor("menu.enter"); } renderMainMenu() { const buttonContainer = this.htmlElement.querySelector(".mainContainer .buttons"); removeAllChildren(buttonContainer); const outerDiv = makeDivElement(null, ["outer"], null); // Import button this.trackClicks( makeButton(outerDiv, ["importButton", "styledButton"], T.mainMenu.importSavegame), this.requestImportSavegame ); if (this.savedGames.length > 0) { // Continue game this.trackClicks( makeButton(buttonContainer, ["continueButton", "styledButton"], T.mainMenu.continue), this.onContinueButtonClicked ); // New game this.trackClicks( makeButton(outerDiv, ["newGameButton", "styledButton"], T.mainMenu.newGame), this.onPlayButtonClicked ); } else { // New game this.trackClicks( makeButton(buttonContainer, ["playButton", "styledButton"], T.mainMenu.play), this.onPlayButtonClicked ); } this.htmlElement .querySelector(".mainContainer") .setAttribute("data-savegames", String(this.savedGames.length)); // Mods this.trackClicks( makeButton(outerDiv, ["modsButton", "styledButton"], T.mods.title), this.onModsClicked ); buttonContainer.appendChild(outerDiv); } fetchPlayerCount() { const element = this.htmlElement.querySelector(".onlinePlayerCount"); if (!element) { return; } fetch("https://analytics.shapez.io/v1/player-count", { cache: "no-cache", }) .then(res => res.json()) .then( count => { element.innerText = T.demoBanners.playerCount.replace("", String(count)); }, ex => { console.warn("Failed to get player count:", ex); } ); } onPuzzleModeButtonClicked(force = false) { const hasUnlockedBlueprints = this.app.savegameMgr.getSavegamesMetaData().some(s => s.level >= 12); if (!force && !hasUnlockedBlueprints) { const { ok } = this.dialogs.showWarning( T.dialogs.puzzlePlayRegularRecommendation.title, T.dialogs.puzzlePlayRegularRecommendation.desc, ["cancel:good", "ok:bad:timeout"] ); ok.add(() => this.onPuzzleModeButtonClicked(true)); return; } this.moveToState("LoginState", { nextStateId: "PuzzleMenuState", }); } onPuzzleWishlistButtonClicked() { this.app.platformWrapper.openExternalLink(THIRDPARTY_URLS.puzzleDlcStorePage); } onShapez2Clicked() { this.app.platformWrapper.openExternalLink("https://tobspr.io/shapez-2?utm_medium=shapez"); } onBackButtonClicked() { this.renderMainMenu(); this.renderSavegames(); } onSteamLinkClicked() { openStandaloneLink(this.app, "shapez_mainmenu"); return false; } onSteamLinkClickedSocial() { openStandaloneLink(this.app, "shapez_mainmenu_social"); return false; } onExitAppButtonClicked() { this.app.platformWrapper.exitApp(); } onChangelogClicked() { this.moveToState("ChangelogState"); } onRedditClicked() { this.app.platformWrapper.openExternalLink(THIRDPARTY_URLS.reddit); } onTwitterLinkClicked() { this.app.platformWrapper.openExternalLink(THIRDPARTY_URLS.twitter); } onPatreonLinkClicked() { this.app.platformWrapper.openExternalLink(THIRDPARTY_URLS.patreon); } onLanguageChooseClicked() { const setting = /** @type {EnumSetting} */ (this.app.settings.getSettingHandleById("language")); const { optionSelected } = this.dialogs.showOptionChooser(T.settings.labels.language.title, { active: this.app.settings.getLanguage(), options: setting.options.map(option => ({ value: setting.valueGetter(option), text: setting.textGetter(option), desc: setting.descGetter(option), iconPrefix: setting.iconPrefix, })), }); optionSelected.add(value => { this.app.settings.updateLanguage(value).then(() => { if (setting.restartRequired) { if (this.app.platformWrapper.getSupportsRestart()) { this.app.platformWrapper.performRestart(); } else { this.dialogs.showInfo( T.dialogs.restartRequired.title, T.dialogs.restartRequired.text, ["ok:good"] ); } } if (setting.changeCb) { setting.changeCb(this.app, value); } }); // Update current icon this.htmlElement.querySelector("button.languageChoose").setAttribute("data-languageIcon", value); }, this); } get savedGames() { return this.app.savegameMgr.getSavegamesMetaData(); } renderSavegames() { const oldContainer = this.htmlElement.querySelector(".mainContainer .savegames"); if (oldContainer) { oldContainer.remove(); } const games = this.savedGames; if (games.length > 0) { const parent = makeDiv(this.htmlElement.querySelector(".mainContainer .savegamesMount"), null, [ "savegames", ]); for (let i = 0; i < games.length; ++i) { const elem = makeDiv(parent, null, ["savegame"]); makeDiv( elem, null, ["playtime"], formatSecondsToTimeAgo((new Date().getTime() - games[i].lastUpdate) / 1000.0) ); makeDiv( elem, null, ["level"], games[i].level ? T.mainMenu.savegameLevel.replace("", "" + games[i].level) : T.mainMenu.savegameLevelUnknown ); const name = makeDiv( elem, null, ["name"], "" + (games[i].name ? games[i].name : T.mainMenu.savegameUnnamed) + "" ); const deleteButton = document.createElement("button"); deleteButton.classList.add("styledButton", "deleteGame"); deleteButton.setAttribute("aria-label", "Delete"); elem.appendChild(deleteButton); const downloadButton = document.createElement("button"); downloadButton.classList.add("styledButton", "downloadGame"); downloadButton.setAttribute("aria-label", "Download"); elem.appendChild(downloadButton); if (!G_WEGAME_VERSION) { const renameButton = document.createElement("button"); renameButton.classList.add("styledButton", "renameGame"); renameButton.setAttribute("aria-label", "Rename Savegame"); name.appendChild(renameButton); this.trackClicks(renameButton, () => this.requestRenameSavegame(games[i])); } const resumeButton = document.createElement("button"); resumeButton.classList.add("styledButton", "resumeGame"); resumeButton.setAttribute("aria-label", "Resumee"); elem.appendChild(resumeButton); this.trackClicks(deleteButton, () => this.deleteGame(games[i])); this.trackClicks(downloadButton, () => this.downloadGame(games[i])); this.trackClicks(resumeButton, () => this.resumeGame(games[i])); } } else { const parent = makeDiv( this.htmlElement.querySelector(".mainContainer .savegamesMount"), null, ["savegamesNone"], T.mainMenu.noActiveSavegames ); } } /** * @param {SavegameMetadata} game */ requestRenameSavegame(game) { const regex = /^[a-zA-Z0-9_\- ]{1,20}$/; const nameInput = new FormElementInput({ id: "nameInput", label: null, placeholder: "", defaultValue: game.name || "", validator: val => val.match(regex) && trim(val).length > 0, }); const dialog = new DialogWithForm({ app: this.app, title: T.dialogs.renameSavegame.title, desc: T.dialogs.renameSavegame.desc, formElements: [nameInput], buttons: ["cancel:bad:escape", "ok:good:enter"], }); this.dialogs.internalShowDialog(dialog); // When confirmed, save the name dialog.buttonSignals.ok.add(() => { game.name = trim(nameInput.getValue()); this.app.savegameMgr.writeAsync(); this.renderSavegames(); }); } /** * @param {SavegameMetadata} game */ resumeGame(game) { this.app.adProvider.showVideoAd().then(() => { const savegame = this.app.savegameMgr.getSavegameById(game.internalId); savegame .readAsync() .then(() => this.checkForModDifferences(savegame)) .then(() => { this.moveToState("InGameState", { savegame, }); }) .catch(err => { this.dialogs.showWarning( T.dialogs.gameLoadFailure.title, T.dialogs.gameLoadFailure.text + "

" + err ); }); }); } /** * @param {Savegame} savegame */ checkForModDifferences(savegame) { const difference = MODS.computeModDifference(savegame.currentData.mods); if (difference.missing.length === 0 && difference.extra.length === 0) { return Promise.resolve(); } let dialogHtml = T.dialogs.modsDifference.desc; /** * * @param {import("../savegame/savegame_typedefs").SavegameStoredMods[0]} mod */ function formatMod(mod) { return `
${mod.name}
${T.mods.version} ${mod.version}
`; } if (difference.missing.length > 0) { dialogHtml += "

" + T.dialogs.modsDifference.missingMods + "

"; dialogHtml += difference.missing.map(formatMod).join("
"); } if (difference.extra.length > 0) { dialogHtml += "

" + T.dialogs.modsDifference.newMods + "

"; dialogHtml += difference.extra.map(formatMod).join("
"); } const signals = this.dialogs.showWarning(T.dialogs.modsDifference.title, dialogHtml, [ "cancel:good", "continue:bad", ]); return new Promise(resolve => { signals.continue.add(resolve); }); } /** * @param {SavegameMetadata} game */ deleteGame(game) { const signals = this.dialogs.showWarning( T.dialogs.confirmSavegameDelete.title, T.dialogs.confirmSavegameDelete.text .replace("", game.name || T.mainMenu.savegameUnnamed) .replace("", String(game.level)), ["cancel:good", "delete:bad:timeout"] ); signals.delete.add(() => { this.app.savegameMgr.deleteSavegame(game).then( () => { this.renderSavegames(); if (this.savedGames.length <= 0) this.renderMainMenu(); }, err => { this.dialogs.showWarning( T.dialogs.savegameDeletionError.title, T.dialogs.savegameDeletionError.text + "

" + err ); } ); }); } /** * @param {SavegameMetadata} game */ downloadGame(game) { const savegame = this.app.savegameMgr.getSavegameById(game.internalId); savegame.readAsync().then(() => { const data = ReadWriteProxy.serializeObject(savegame.currentData); const filename = (game.name || "unnamed") + ".bin"; generateFileDownload(filename, data); }); } /** * Shows a hint that the slot limit has been reached */ showSavegameSlotLimit() { const { getStandalone } = this.dialogs.showWarning( T.dialogs.oneSavegameLimit.title, T.dialogs.oneSavegameLimit.desc, ["cancel:bad", "getStandalone:good"] ); getStandalone.add(() => { openStandaloneLink(this.app, "shapez_slotlimit"); }); this.app.gameAnalytics.note("slotlimit"); } onSettingsButtonClicked() { this.moveToState("SettingsState"); } onTranslationHelpLinkClicked() { this.app.platformWrapper.openExternalLink( "https://github.com/tobspr-games/shapez.io/blob/master/translations" ); } onPlayButtonClicked() { if ( this.app.savegameMgr.getSavegamesMetaData().length > 0 && !this.app.restrictionMgr.getHasUnlimitedSavegames() ) { this.app.gameAnalytics.noteMinor("menu.slotlimit"); this.showSavegameSlotLimit(); return; } this.app.adProvider.showVideoAd().then(() => { this.app.gameAnalytics.noteMinor("menu.play"); const savegame = this.app.savegameMgr.createNewSavegame(); this.moveToState("InGameState", { savegame, }); }); } onWegameRatingClicked() { this.dialogs.showInfo( "提示说明:", ` 1)本游戏是一款休闲建造类单机游戏,画面简洁而乐趣充足。适用于年满8周岁及以上的用户,建议未成年人在家长监护下使用游戏产品。
2)本游戏模拟简单的生产流水线,剧情简单且积极向上,没有基于真实历史和现实事件的改编内容。游戏玩法为摆放简单的部件,完成生产目标。游戏为单机作品,没有基于文字和语音的陌生人社交系统。
3)本游戏中有用户实名认证系统,认证为未成年人的用户将接受以下管理:未满8周岁的用户不能付费;8周岁以上未满16周岁的未成年人用户,单次充值金额不得超过50元人民币,每月充值金额累计不得超过200元人民币;16周岁以上的未成年人用户,单次充值金额不得超过100元人民币,每月充值金额累计不得超过400元人民币。未成年玩家,仅可在周五、周六、周日和法定节假日每日20时至21时进行游戏。
4)游戏功能说明:一款关于传送带自动化生产特定形状产品的工厂流水线模拟游戏,画面简洁而乐趣充足,可以让玩家在轻松愉快的氛围下获得各种游戏乐趣,体验完成目标的成就感。游戏没有失败功能,自动存档,不存在较强的挫折体验。 ` ); } onModsClicked() { this.app.gameAnalytics.noteMinor("menu.mods"); this.moveToState("ModsState", { backToStateId: "MainMenuState", }); } onContinueButtonClicked() { let latestLastUpdate = 0; let latestInternalId; this.app.savegameMgr.currentData.savegames.forEach(saveGame => { if (saveGame.lastUpdate > latestLastUpdate) { latestLastUpdate = saveGame.lastUpdate; latestInternalId = saveGame.internalId; } }); const savegame = this.app.savegameMgr.getSavegameById(latestInternalId); if (!savegame) { console.warn("No savegame to continue found:", this.app.savegameMgr.currentData.savegames); return; } this.app.gameAnalytics.noteMinor("menu.continue"); savegame .readAsync() .then(() => this.app.adProvider.showVideoAd()) .then(() => this.checkForModDifferences(savegame)) .then(() => { this.moveToState("InGameState", { savegame, }); }); } onLeave() { this.dialogs.cleanup(); clearInterval(this.refreshInterval); } }