import { loadAliases, loadGlobalAliases, Aliases, GlobalAliases } from "./Alias"; import { Companies, loadCompanies } from "./Company/Companies"; import { CONSTANTS } from "./Constants"; import { Factions, loadFactions } from "./Faction/Factions"; import { loadAllGangs, AllGangs } from "./Gang/AllGangs"; import { loadMessages, initMessages, Messages } from "./Message/MessageHelpers"; import { Player, loadPlayer } from "./Player"; import { saveAllServers, loadAllServers, GetAllServers } from "./Server/AllServers"; import { Settings } from "./Settings/Settings"; import { SourceFileFlags } from "./SourceFile/SourceFileFlags"; import { loadStockMarket, StockMarket } from "./StockMarket/StockMarket"; import { staneksGift, loadStaneksGift } from "./CotMG/Helper"; import { SnackbarEvents } from "./ui/React/Snackbar"; import * as ExportBonus from "./ExportBonus"; import { dialogBoxCreate } from "./ui/React/DialogBox"; import { Reviver, Generic_toJSON, Generic_fromJSON } from "./utils/JSONReviver"; import { save } from "./db"; import { v1APIBreak } from "./utils/v1APIBreak"; import { AugmentationNames } from "./Augmentation/data/AugmentationNames"; import { PlayerOwnedAugmentation } from "./Augmentation/PlayerOwnedAugmentation"; import { LocationName } from "./Locations/data/LocationNames"; /* SaveObject.js * Defines the object used to save/load games */ class BitburnerSaveObject { PlayerSave = ""; AllServersSave = ""; CompaniesSave = ""; FactionsSave = ""; AliasesSave = ""; GlobalAliasesSave = ""; MessagesSave = ""; StockMarketSave = ""; SettingsSave = ""; VersionSave = ""; AllGangsSave = ""; LastExportBonus = ""; StaneksGiftSave = ""; SaveTimestamp = ""; getSaveString(): string { this.PlayerSave = JSON.stringify(Player); this.AllServersSave = saveAllServers(); this.CompaniesSave = JSON.stringify(Companies); this.FactionsSave = JSON.stringify(Factions); this.AliasesSave = JSON.stringify(Aliases); this.GlobalAliasesSave = JSON.stringify(GlobalAliases); this.MessagesSave = JSON.stringify(Messages); this.StockMarketSave = JSON.stringify(StockMarket); this.SettingsSave = JSON.stringify(Settings); this.VersionSave = JSON.stringify(CONSTANTS.VersionNumber); this.LastExportBonus = JSON.stringify(ExportBonus.LastExportBonus); this.StaneksGiftSave = JSON.stringify(staneksGift); this.SaveTimestamp = new Date().getTime().toString(); if (Player.inGang()) { this.AllGangsSave = JSON.stringify(AllGangs); } const saveString = btoa(unescape(encodeURIComponent(JSON.stringify(this)))); return saveString; } saveGame(emitToastEvent = true): void { const saveString = this.getSaveString(); save(saveString) .then(() => { if (emitToastEvent) { SnackbarEvents.emit("Game Saved!", "info", 2000); } }) .catch((err) => console.error(err)); } exportGame(): void { const saveString = this.getSaveString(); // Save file name is based on current timestamp and BitNode const epochTime = Math.round(Date.now() / 1000); const bn = Player.bitNodeN; const filename = `bitburnerSave_${epochTime}_BN${bn}x${SourceFileFlags[bn]}.json`; download(filename, saveString); } toJSON(): any { return Generic_toJSON("BitburnerSaveObject", this); } static fromJSON(value: { data: any }): BitburnerSaveObject { return Generic_fromJSON(BitburnerSaveObject, value.data); } } // Makes necessary changes to the loaded/imported data to ensure // the game stills works with new versions function evaluateVersionCompatibility(ver: string | number): void { // We have to do this because ts won't let us otherwise const anyPlayer = Player as any; if (typeof ver === "string") { // This version refactored the Company/job-related code if (ver <= "0.41.2") { // Player's company position is now a string if (anyPlayer.companyPosition != null && typeof anyPlayer.companyPosition !== "string") { anyPlayer.companyPosition = anyPlayer.companyPosition.data.positionName; if (anyPlayer.companyPosition == null) { anyPlayer.companyPosition = ""; } } // The "companyName" property of all Companies is renamed to "name" for (const companyName in Companies) { const company: any = Companies[companyName]; if (company.name == 0 && company.companyName != null) { company.name = company.companyName; } if (company.companyPositions instanceof Array) { const pos: any = {}; for (let i = 0; i < company.companyPositions.length; ++i) { pos[company.companyPositions[i]] = true; } company.companyPositions = pos; } } } // This version allowed players to hold multiple jobs if (ver < "0.43.0") { if (anyPlayer.companyName !== "" && anyPlayer.companyPosition != null && anyPlayer.companyPosition !== "") { anyPlayer.jobs[anyPlayer.companyName] = anyPlayer.companyPosition; } delete anyPlayer.companyPosition; } if (ver < "0.56.0") { for (const q of anyPlayer.queuedAugmentations) { if (q.name === "Graphene BranchiBlades Upgrade") { q.name = "Graphene BrachiBlades Upgrade"; } } for (const q of anyPlayer.augmentations) { if (q.name === "Graphene BranchiBlades Upgrade") { q.name = "Graphene BrachiBlades Upgrade"; } } } if (ver < "0.56.1") { if (anyPlayer.bladeburner === 0) { anyPlayer.bladeburner = null; } if (anyPlayer.gang === 0) { anyPlayer.gang = null; } if (anyPlayer.corporation === 0) { anyPlayer.corporation = null; } // convert all Messages to just filename to save space. const home = anyPlayer.getHomeComputer(); for (let i = 0; i < home.messages.length; i++) { if (home.messages[i].filename) { home.messages[i] = home.messages[i].filename; } } } if (ver < "0.58.0") { const changes: [RegExp, string][] = [ [/getStockSymbols/g, "stock.getSymbols"], [/getStockPrice/g, "stock.getPrice"], [/getStockAskPrice/g, "stock.getAskPrice"], [/getStockBidPrice/g, "stock.getBidPrice"], [/getStockPosition/g, "stock.getPosition"], [/getStockMaxShares/g, "stock.getMaxShares"], [/getStockPurchaseCost/g, "stock.getPurchaseCost"], [/getStockSaleGain/g, "stock.getSaleGain"], [/buyStock/g, "stock.buy"], [/sellStock/g, "stock.sell"], [/shortStock/g, "stock.short"], [/sellShort/g, "stock.sellShort"], [/placeOrder/g, "stock.placeOrder"], [/cancelOrder/g, "stock.cancelOrder"], [/getOrders/g, "stock.getOrders"], [/getStockVolatility/g, "stock.getVolatility"], [/getStockForecast/g, "stock.getForecast"], [/purchase4SMarketData/g, "stock.purchase4SMarketData"], [/purchase4SMarketDataTixApi/g, "stock.purchase4SMarketDataTixApi"], ]; function convert(code: string): string { for (const change of changes) { code = code.replace(change[0], change[1]); } return code; } for (const server of GetAllServers()) { for (const script of server.scripts) { script.code = convert(script.code); } } } v1APIBreak(); ver = 1; } if (typeof ver === "number") { if (ver < 2) { // Give 10 neuroflux because v1 API break. const nf = Player.augmentations.find((a) => a.name === AugmentationNames.NeuroFluxGovernor); if (nf) { nf.level += 10; } else { const nf = new PlayerOwnedAugmentation(AugmentationNames.NeuroFluxGovernor); nf.level = 10; Player.augmentations.push(nf); } Player.reapplyAllAugmentations(true); Player.reapplyAllSourceFiles(); } if (ver < 3) { anyPlayer.money = parseFloat(anyPlayer.money); if (anyPlayer.corporation) { anyPlayer.corporation.funds = parseFloat(anyPlayer.corporation.funds); anyPlayer.corporation.revenue = parseFloat(anyPlayer.corporation.revenue); anyPlayer.corporation.expenses = parseFloat(anyPlayer.corporation.expenses); for (let i = 0; i < anyPlayer.corporation.divisions.length; ++i) { const ind = anyPlayer.corporation.divisions[i]; ind.lastCycleRevenue = parseFloat(ind.lastCycleRevenue); ind.lastCycleExpenses = parseFloat(ind.lastCycleExpenses); ind.thisCycleRevenue = parseFloat(ind.thisCycleRevenue); ind.thisCycleExpenses = parseFloat(ind.thisCycleExpenses); } } } if (ver < 9) { if (StockMarket.hasOwnProperty("Joes Guns")) { const s = StockMarket["Joes Guns"]; delete StockMarket["Joes Guns"]; StockMarket[LocationName.Sector12JoesGuns] = s; } } } } function loadGame(saveString: string): boolean { if (!saveString) return false; saveString = decodeURIComponent(escape(atob(saveString))); const saveObj = JSON.parse(saveString, Reviver); loadPlayer(saveObj.PlayerSave); loadAllServers(saveObj.AllServersSave); loadCompanies(saveObj.CompaniesSave); loadFactions(saveObj.FactionsSave); if (saveObj.hasOwnProperty("StaneksGiftSave")) { loadStaneksGift(saveObj.StaneksGiftSave); } else { console.warn(`Could not load Staneks Gift from save`); loadStaneksGift(""); } if (saveObj.hasOwnProperty("AliasesSave")) { try { loadAliases(saveObj.AliasesSave); } catch (e) { console.warn(`Could not load Aliases from save`); loadAliases(""); } } else { console.warn(`Save file did not contain an Aliases property`); loadAliases(""); } if (saveObj.hasOwnProperty("GlobalAliasesSave")) { try { loadGlobalAliases(saveObj.GlobalAliasesSave); } catch (e) { console.warn(`Could not load GlobalAliases from save`); loadGlobalAliases(""); } } else { console.warn(`Save file did not contain a GlobalAliases property`); loadGlobalAliases(""); } if (saveObj.hasOwnProperty("MessagesSave")) { try { loadMessages(saveObj.MessagesSave); } catch (e) { console.warn(`Could not load Messages from save`); initMessages(); } } else { console.warn(`Save file did not contain a Messages property`); initMessages(); } if (saveObj.hasOwnProperty("StockMarketSave")) { try { loadStockMarket(saveObj.StockMarketSave); } catch (e) { loadStockMarket(""); } } else { loadStockMarket(""); } if (saveObj.hasOwnProperty("SettingsSave")) { try { Settings.load(saveObj.SettingsSave); } catch (e) { console.error("ERROR: Failed to parse Settings. Re-initing default values"); Settings.init(); } } else { Settings.init(); } if (saveObj.hasOwnProperty("LastExportBonus")) { try { ExportBonus.setLastExportBonus(JSON.parse(saveObj.LastExportBonus)); } catch (err) { ExportBonus.setLastExportBonus(new Date().getTime()); console.error("ERROR: Failed to parse last export bonus Settings " + err); } } if (Player.inGang() && saveObj.hasOwnProperty("AllGangsSave")) { try { loadAllGangs(saveObj.AllGangsSave); } catch (e) { console.error("ERROR: Failed to parse AllGangsSave: " + e); } } if (saveObj.hasOwnProperty("VersionSave")) { try { const ver = JSON.parse(saveObj.VersionSave, Reviver); evaluateVersionCompatibility(ver); if (window.location.href.toLowerCase().includes("bitburner-beta")) { // Beta branch, always show changes createBetaUpdateText(); } else if (ver !== CONSTANTS.VersionNumber) { createNewUpdateText(); } } catch (e) { createNewUpdateText(); } } else { createNewUpdateText(); } return true; } function createNewUpdateText(): void { setTimeout( () => dialogBoxCreate( "New update!
" + "Please report any bugs/issues through the github repository " + "or the Bitburner subreddit (reddit.com/r/bitburner).

" + CONSTANTS.LatestUpdate, ), 1000, ); } function createBetaUpdateText(): void { dialogBoxCreate( "You are playing on the beta environment! This branch of the game " + "features the latest developments in the game. This version may be unstable.
" + "Please report any bugs/issues through the github repository (https://github.com/danielyxie/bitburner/issues) " + "or the Bitburner subreddit (reddit.com/r/bitburner).

" + CONSTANTS.LatestUpdate, ); } function download(filename: string, content: string): void { const file = new Blob([content], { type: "text/plain" }); const navigator = window.navigator as any; if (navigator.msSaveOrOpenBlob) { // IE10+ navigator.msSaveOrOpenBlob(file, filename); } else { // Others const a = document.createElement("a"), url = URL.createObjectURL(file); a.href = url; a.download = filename; document.body.appendChild(a); a.click(); setTimeout(function () { document.body.removeChild(a); window.URL.revokeObjectURL(url); }, 0); } } Reviver.constructors.BitburnerSaveObject = BitburnerSaveObject; export { saveObject, loadGame, download }; const saveObject = new BitburnerSaveObject();