bitburner-src/src/GameOptions/ui/GameOptionsSidebar.tsx

284 lines
10 KiB
TypeScript
Raw Normal View History

2022-04-17 22:08:40 +02:00
import {
BugReport,
Chat,
Download,
LibraryBooks,
Palette,
Reddit,
Save,
SystemUpdateAlt,
Upload,
} from "@mui/icons-material";
import { Box, Button, List, ListItemButton, Paper, Tooltip, Typography } from "@mui/material";
2022-04-17 19:51:14 +02:00
import { default as React, useRef, useState } from "react";
import { FileDiagnosticModal } from "../../Diagnostic/FileDiagnosticModal";
import { ImportData, saveObject } from "../../SaveObject";
import { Settings } from "../../Settings/Settings";
import { StyleEditorButton } from "../../Themes/ui/StyleEditorButton";
import { ThemeEditorButton } from "../../Themes/ui/ThemeEditorButton";
import { ConfirmationModal } from "../../ui/React/ConfirmationModal";
import { DeleteGameButton } from "../../ui/React/DeleteGameButton";
import { SnackbarEvents, ToastVariant } from "../../ui/React/Snackbar";
import { SoftResetButton } from "../../ui/React/SoftResetButton";
2022-09-06 15:07:12 +02:00
import { Router } from "../../ui/GameRoot";
2022-12-04 09:14:06 +01:00
import { Page } from "../../ui/Router";
2022-04-17 19:51:14 +02:00
import { convertTimeMsToTimeElapsedString } from "../../utils/StringHelperFunctions";
2022-04-17 22:09:47 +02:00
import { GameOptionsTab } from "../GameOptionsTab";
2022-04-17 19:51:14 +02:00
interface IProps {
2022-04-17 22:09:47 +02:00
tab: GameOptionsTab;
setTab: (tab: GameOptionsTab) => void;
2022-04-17 19:51:14 +02:00
save: () => void;
export: () => void;
forceKill: () => void;
softReset: () => void;
}
interface ITabProps {
sideBarProps: IProps;
2022-04-17 22:09:47 +02:00
tab: GameOptionsTab;
2022-04-17 19:51:14 +02:00
tabName: string;
}
const SideBarTab = (props: ITabProps): React.ReactElement => {
return (
<ListItemButton
selected={props.sideBarProps.tab === props.tab}
onClick={() => props.sideBarProps.setTab(props.tab)}
>
<Typography>{props.tabName}</Typography>
</ListItemButton>
);
};
export const GameOptionsSidebar = (props: IProps): React.ReactElement => {
const importInput = useRef<HTMLInputElement>(null);
const [diagnosticOpen, setDiagnosticOpen] = useState(false);
const [importSaveOpen, setImportSaveOpen] = useState(false);
const [importData, setImportData] = useState<ImportData | null>(null);
function startImport(): void {
if (!window.File || !window.FileReader || !window.FileList || !window.Blob) return;
const ii = importInput.current;
if (ii === null) throw new Error("import input should not be null");
ii.click();
}
async function onImport(event: React.ChangeEvent<HTMLInputElement>): Promise<void> {
try {
const base64Save = await saveObject.getImportStringFromFile(event.target.files);
const data = await saveObject.getImportDataFromString(base64Save);
setImportData(data);
setImportSaveOpen(true);
2022-07-15 05:03:54 +02:00
} catch (e: unknown) {
2023-01-05 19:04:26 +01:00
console.error(e);
2022-07-15 05:03:54 +02:00
SnackbarEvents.emit(String(e), ToastVariant.ERROR, 5000);
2022-04-17 19:51:14 +02:00
}
}
async function confirmedImportGame(): Promise<void> {
if (!importData) return;
try {
await saveObject.importGame(importData.base64);
2022-07-15 05:03:54 +02:00
} catch (e: unknown) {
SnackbarEvents.emit(String(e), ToastVariant.ERROR, 5000);
2022-04-17 19:51:14 +02:00
}
setImportSaveOpen(false);
setImportData(null);
}
function compareSaveGame(): void {
if (!importData) return;
2022-09-06 15:07:12 +02:00
Router.toImportSave(importData.base64);
2022-04-17 19:51:14 +02:00
setImportSaveOpen(false);
setImportData(null);
}
return (
<Box>
<Paper sx={{ height: "fit-content", mb: 1 }}>
<List>
2022-04-17 22:09:47 +02:00
<SideBarTab sideBarProps={props} tab={GameOptionsTab.SYSTEM} tabName="System" />
<SideBarTab sideBarProps={props} tab={GameOptionsTab.GAMEPLAY} tabName="Gameplay" />
<SideBarTab sideBarProps={props} tab={GameOptionsTab.INTERFACE} tabName="Interface" />
<SideBarTab sideBarProps={props} tab={GameOptionsTab.MISC} tabName="Misc" />
<SideBarTab sideBarProps={props} tab={GameOptionsTab.REMOTE_API} tabName="Remote API" />
2022-04-17 19:51:14 +02:00
</List>
</Paper>
<Box
sx={{
display: "grid",
2022-04-17 22:08:40 +02:00
width: "100%",
2022-04-17 19:51:14 +02:00
height: "fit-content",
gridTemplateAreas: `"save delete"
"export import"
"kill kill"
"reset diagnose"
"browse browse"
"theme style"
2022-04-17 22:08:40 +02:00
"links links"
"devs devs"`,
2022-04-17 19:51:14 +02:00
gridTemplateColumns: "1fr 1fr",
}}
>
<Button onClick={() => props.save()} startIcon={<Save />} sx={{ gridArea: "save" }}>
Save Game
</Button>
2022-04-17 22:08:40 +02:00
<Box sx={{ gridArea: "delete", "& .MuiButton-root": { height: "100%", width: "100%" } }}>
2022-04-17 19:51:14 +02:00
<DeleteGameButton />
</Box>
<Tooltip title={<Typography>Export your game to a text file.</Typography>}>
<Button onClick={() => props.export()} startIcon={<Download />} sx={{ gridArea: "export" }}>
Export Game
</Button>
</Tooltip>
<Tooltip
title={
<Typography>
Import your game from a text file.
<br />
This will <strong>overwrite</strong> your current game. Back it up first!
</Typography>
}
>
<Button onClick={startImport} startIcon={<Upload />} sx={{ gridArea: "import" }}>
Import Game
<input ref={importInput} id="import-game-file-selector" type="file" hidden onChange={onImport} />
</Button>
</Tooltip>
<ConfirmationModal
open={importSaveOpen}
onClose={() => setImportSaveOpen(false)}
onConfirm={() => confirmedImportGame()}
additionalButton={<Button onClick={compareSaveGame}>Compare Save</Button>}
confirmationText={
<>
Importing a new game will <strong>completely wipe</strong> the current data!
<br />
<br />
Make sure to have a backup of your current save file before importing.
<br />
The file you are attempting to import seems valid.
{(importData?.playerData?.lastSave ?? 0) > 0 && (
<>
<br />
<br />
The export date of the save file is{" "}
<strong>{new Date(importData?.playerData?.lastSave ?? 0).toLocaleString()}</strong>
</>
)}
{(importData?.playerData?.totalPlaytime ?? 0) > 0 && (
<>
<br />
<br />
Total play time of imported game:{" "}
{convertTimeMsToTimeElapsedString(importData?.playerData?.totalPlaytime ?? 0)}
</>
)}
<br />
<br />
</>
}
/>
<Tooltip
title={
<Typography>
Forcefully kill all active running scripts, in case there is a bug or some unexpected issue with the game.
After using this, save the game and then reload the page. This is different than normal kill in that
2022-04-17 19:51:14 +02:00
normal kill will tell the script to shut down while force kill just removes the references to it (and it
should crash on its own). This will not remove the files on your computer, just forcefully kill all
2022-04-17 22:36:19 +02:00
running instances of all scripts.
2022-04-17 19:51:14 +02:00
</Typography>
}
>
<Button onClick={() => props.forceKill()} sx={{ gridArea: "kill" }}>
Force kill all active scripts
</Button>
</Tooltip>
2022-04-17 22:08:40 +02:00
<Box sx={{ gridArea: "reset", "& .MuiButton-root": { height: "100%", width: "100%" } }}>
<SoftResetButton
noConfirmation={Settings.SuppressBuyAugmentationConfirmation}
onTriggered={props.softReset}
/>
</Box>
2022-04-17 19:51:14 +02:00
<Tooltip
title={
<Typography>
If your save file is extremely big you can use this button to view a map of all the files on every server.
Be careful: there might be spoilers.
2022-04-17 19:51:14 +02:00
</Typography>
}
>
<Button onClick={() => setDiagnosticOpen(true)} sx={{ gridArea: "diagnose" }}>
Diagnose files
</Button>
</Tooltip>
<Tooltip title="Head to the theme browser to see a collection of prebuilt themes.">
2022-12-04 09:14:06 +01:00
<Button startIcon={<Palette />} onClick={() => Router.toPage(Page.ThemeBrowser)} sx={{ gridArea: "browse" }}>
2022-04-17 19:51:14 +02:00
Theme Browser
</Button>
</Tooltip>
2022-04-17 22:08:40 +02:00
<Box sx={{ gridArea: "theme", "& .MuiButton-root": { height: "100%", width: "100%" } }}>
2022-09-06 15:07:12 +02:00
<ThemeEditorButton />
2022-04-17 19:51:14 +02:00
</Box>
2022-04-17 22:08:40 +02:00
<Box sx={{ gridArea: "style", "& .MuiButton-root": { height: "100%", width: "100%" } }}>
2022-04-17 19:51:14 +02:00
<StyleEditorButton />
</Box>
2022-04-17 22:08:40 +02:00
<Box
sx={{
gridArea: "links",
display: "grid",
gridTemplateAreas: `"bug changelog"
"docs docs"
"discord reddit"
"plaza plaza"`,
gridTemplateColumns: "1fr 1fr",
my: 1,
}}
>
<Button
startIcon={<BugReport />}
href="https://github.com/bitburner-official/bitburner-src/issues/new"
2022-04-17 22:08:40 +02:00
target="_blank"
sx={{ gridArea: "bug" }}
>
Report Bug
</Button>
<Button
startIcon={<SystemUpdateAlt />}
2023-01-06 14:24:10 +01:00
href="https://bitburner-official.readthedocs.io/en/latest/changelog.html"
2022-04-17 22:08:40 +02:00
target="_blank"
sx={{ gridArea: " changelog" }}
>
Changelog
</Button>
<Button
startIcon={<LibraryBooks />}
2023-01-06 14:24:10 +01:00
href="https://bitburner-official.readthedocs.io/en/latest/index.html"
2022-04-17 22:08:40 +02:00
target="_blank"
sx={{ gridArea: "docs" }}
>
Documentation
</Button>
<Button startIcon={<Chat />} href="https://discord.gg/TFc3hKD" target="_blank" sx={{ gridArea: "discord" }}>
Discord
</Button>
<Button
startIcon={<Reddit />}
href="https://www.reddit.com/r/bitburner"
target="_blank"
sx={{ gridArea: "reddit" }}
>
Reddit
</Button>
2022-04-17 19:51:14 +02:00
</Box>
</Box>
<FileDiagnosticModal open={diagnosticOpen} onClose={() => setDiagnosticOpen(false)} />
</Box>
);
};