Profectus-Demo/src/components/SavesManager.vue

325 lines
10 KiB
Vue
Raw Normal View History

2021-06-21 04:29:55 +00:00
<template>
2022-01-25 04:25:34 +00:00
<Modal v-model="isOpen" ref="modal">
<template v-slot:header>
<h2>Saves Manager</h2>
</template>
<template #body="{ shown }">
2022-01-25 04:25:34 +00:00
<Draggable
:list="settings.saves"
handle=".handle"
v-if="shown"
2022-01-25 04:25:34 +00:00
:itemKey="(save: string) => save"
>
<template #item="{ element }">
<Save
:save="saves[element]"
@open="openSave(element)"
@export="exportSave(element)"
@editName="name => editSave(element, name)"
@duplicate="duplicateSave(element)"
@delete="deleteSave(element)"
/>
</template>
</Draggable>
</template>
<template v-slot:footer>
<div class="modal-footer">
2022-01-14 04:25:47 +00:00
<Text
v-model="saveToImport"
title="Import Save"
placeholder="Paste your save here!"
:class="{ importingFailed }"
/>
<div class="field">
<span class="field-title">Create Save</span>
<div class="field-buttons">
2022-04-30 23:41:04 +00:00
<button class="button" @click="openSave(newSave().id)">New Game</button>
<Select
v-if="Object.keys(bank).length > 0"
:options="bank"
:modelValue="undefined"
2022-01-14 04:25:47 +00:00
@update:modelValue="preset => newFromPreset(preset as string)"
closeOnSelect
placeholder="Select preset"
class="presets"
/>
</div>
</div>
<div class="footer">
<div style="flex-grow: 1"></div>
2022-01-14 04:25:47 +00:00
<button class="button modal-default-button" @click="isOpen = false">
Close
</button>
</div>
</div>
</template>
</Modal>
2021-06-21 04:29:55 +00:00
</template>
2022-01-14 04:25:47 +00:00
<script setup lang="ts">
import projInfo from "data/projInfo.json";
2022-03-04 03:39:48 +00:00
import Modal from "components/Modal.vue";
import player, { PlayerData, stringifySave } from "game/player";
2022-03-04 03:39:48 +00:00
import settings from "game/settings";
import { getUniqueID, loadSave, save, newSave } from "util/save";
import { ComponentPublicInstance, computed, nextTick, ref, shallowReactive, watch } from "vue";
2022-02-27 22:04:56 +00:00
import Select from "./fields/Select.vue";
import Text from "./fields/Text.vue";
2022-01-14 04:25:47 +00:00
import Save from "./Save.vue";
2022-01-25 04:25:34 +00:00
import Draggable from "vuedraggable";
import LZString from "lz-string";
import { ProxyState } from "util/proxies";
2021-06-21 04:29:55 +00:00
2022-01-14 04:25:47 +00:00
export type LoadablePlayerData = Omit<Partial<PlayerData>, "id"> & { id: string; error?: unknown };
2021-06-21 04:29:55 +00:00
2022-01-14 04:25:47 +00:00
const isOpen = ref(false);
2022-01-25 04:25:34 +00:00
const modal = ref<ComponentPublicInstance<typeof Modal> | null>(null);
2021-06-21 04:29:55 +00:00
2022-01-14 04:25:47 +00:00
defineExpose({
open() {
isOpen.value = true;
}
});
2021-06-21 04:29:55 +00:00
2022-01-14 04:25:47 +00:00
const importingFailed = ref(false);
const saveToImport = ref("");
2021-06-21 04:29:55 +00:00
watch(saveToImport, importedSave => {
if (importedSave) {
2022-01-14 04:25:47 +00:00
nextTick(() => {
try {
if (importedSave[0] === "{") {
// plaintext. No processing needed
} else if (importedSave[0] === "e") {
// Assumed to be base64, which starts with e
importedSave = decodeURIComponent(escape(atob(importedSave)));
} else if (importedSave[0] === "ᯡ") {
// Assumed to be lz, which starts with ᯡ
// eslint-disable-next-line @typescript-eslint/no-non-null-assertion
importedSave = LZString.decompressFromUTF16(importedSave)!;
} else {
console.warn("Unable to determine preset encoding", importedSave);
importingFailed.value = true;
return;
}
const playerData = JSON.parse(importedSave);
2022-01-14 04:25:47 +00:00
if (typeof playerData !== "object") {
importingFailed.value = true;
return;
}
const id = getUniqueID();
playerData.id = id;
save(playerData);
2022-01-14 04:25:47 +00:00
saveToImport.value = "";
importingFailed.value = false;
settings.saves.push(id);
} catch (e) {
importingFailed.value = true;
}
2022-01-14 04:25:47 +00:00
});
} else {
importingFailed.value = false;
}
});
2022-02-27 22:04:56 +00:00
let bankContext = require.context("raw-loader!../../saves", true, /\.txt$/);
2022-01-14 04:25:47 +00:00
let bank = ref(
bankContext.keys().reduce((acc: Array<{ label: string; value: string }>, curr) => {
// .slice(2, -4) strips the leading ./ and the trailing .txt
acc.push({
label: curr.slice(2, -4),
value: bankContext(curr).default
});
return acc;
}, [])
);
2021-06-21 04:29:55 +00:00
const cachedSaves = shallowReactive<Record<string, LoadablePlayerData | undefined>>({});
2022-01-25 04:25:34 +00:00
function getCachedSave(id: string) {
if (cachedSaves[id] == null) {
let save = localStorage.getItem(id);
2022-01-25 04:25:34 +00:00
if (save == null) {
cachedSaves[id] = { error: `Save doesn't exist in localStorage`, id };
} else if (save === "dW5kZWZpbmVk") {
cachedSaves[id] = { error: `Save is undefined`, id };
2022-01-25 04:25:34 +00:00
} else {
try {
if (save[0] === "{") {
// plaintext. No processing needed
} else if (save[0] === "e") {
// Assumed to be base64, which starts with e
save = decodeURIComponent(escape(atob(save)));
} else if (save[0] === "ᯡ") {
// Assumed to be lz, which starts with ᯡ
// eslint-disable-next-line @typescript-eslint/no-non-null-assertion
save = LZString.decompressFromUTF16(save)!;
} else {
console.warn("Unable to determine preset encoding", save);
importingFailed.value = true;
cachedSaves[id] = { error: "Unable to determine preset encoding", id };
// eslint-disable-next-line @typescript-eslint/no-non-null-assertion
return cachedSaves[id]!;
}
cachedSaves[id] = { ...JSON.parse(save), id };
2022-01-25 04:25:34 +00:00
} catch (error) {
cachedSaves[id] = { error, id };
console.warn(
`SavesManager: Failed to load info about save with id ${id}:\n${error}\n${save}`
);
}
}
2022-01-25 04:25:34 +00:00
}
// eslint-disable-next-line @typescript-eslint/no-non-null-assertion
return cachedSaves[id]!;
2022-01-14 04:25:47 +00:00
}
2022-01-25 04:25:34 +00:00
// Wipe cache whenever the modal is opened
watch(isOpen, isOpen => {
if (isOpen) {
Object.keys(cachedSaves).forEach(key => delete cachedSaves[key]);
}
});
const saves = computed(() =>
settings.saves.reduce((acc: Record<string, LoadablePlayerData>, curr: string) => {
acc[curr] = getCachedSave(curr);
return acc;
}, {})
);
2022-01-14 04:25:47 +00:00
function exportSave(id: string) {
let saveToExport;
if (player.id === id) {
saveToExport = stringifySave(player[ProxyState]);
2022-01-14 04:25:47 +00:00
} else {
saveToExport = JSON.stringify(saves.value[id]);
}
switch (projInfo.exportEncoding) {
default:
console.warn(`Unknown save encoding: ${projInfo.exportEncoding}. Defaulting to lz`);
case "lz":
saveToExport = LZString.compressToUTF16(saveToExport);
break;
case "base64":
saveToExport = btoa(unescape(encodeURIComponent(saveToExport)));
break;
case "plain":
break;
}
2022-01-14 04:25:47 +00:00
// Put on clipboard. Using the clipboard API asks for permissions and stuff
const el = document.createElement("textarea");
el.value = saveToExport;
document.body.appendChild(el);
el.select();
el.setSelectionRange(0, 99999);
document.execCommand("copy");
document.body.removeChild(el);
}
function duplicateSave(id: string) {
if (player.id === id) {
save();
}
const playerData = { ...saves.value[id], id: getUniqueID() };
save(playerData as PlayerData);
2022-01-14 04:25:47 +00:00
settings.saves.push(playerData.id);
}
function deleteSave(id: string) {
settings.saves = settings.saves.filter((save: string) => save !== id);
localStorage.removeItem(id);
cachedSaves[id] = undefined;
2022-01-14 04:25:47 +00:00
}
function openSave(id: string) {
2022-01-25 04:23:30 +00:00
// eslint-disable-next-line @typescript-eslint/no-non-null-assertion
saves.value[player.id]!.time = player.time;
2022-01-14 04:25:47 +00:00
save();
cachedSaves[player.id] = undefined;
2022-01-25 04:23:30 +00:00
// eslint-disable-next-line @typescript-eslint/no-non-null-assertion
loadSave(saves.value[id]!);
// Delete cached version in case of opening it again
cachedSaves[id] = undefined;
2022-01-14 04:25:47 +00:00
}
function newFromPreset(preset: string) {
if (preset[0] === "{") {
// plaintext. No processing needed
} else if (preset[0] === "e") {
// Assumed to be base64, which starts with e
preset = decodeURIComponent(escape(atob(preset)));
} else if (preset[0] === "ᯡ") {
// Assumed to be lz, which starts with ᯡ
// eslint-disable-next-line @typescript-eslint/no-non-null-assertion
preset = LZString.decompressFromUTF16(preset)!;
} else {
console.warn("Unable to determine preset encoding", preset);
return;
}
const playerData = JSON.parse(preset);
2022-01-14 04:25:47 +00:00
playerData.id = getUniqueID();
save(playerData as PlayerData);
2022-01-14 04:25:47 +00:00
settings.saves.push(playerData.id);
2022-04-30 23:41:04 +00:00
openSave(playerData.id);
2022-01-14 04:25:47 +00:00
}
function editSave(id: string, newName: string) {
2022-01-25 04:25:34 +00:00
const currSave = saves.value[id];
if (currSave) {
currSave.name = newName;
if (player.id === id) {
player.name = newName;
save();
} else {
save(currSave as PlayerData);
cachedSaves[id] = undefined;
2022-01-25 04:25:34 +00:00
}
2022-01-14 04:25:47 +00:00
}
}
2021-06-21 04:29:55 +00:00
</script>
<style scoped>
.field form,
.field .field-title,
.field .field-buttons {
margin: 0;
2021-06-21 04:29:55 +00:00
}
.field-buttons {
display: flex;
2021-06-21 04:29:55 +00:00
}
.field-buttons .field {
margin: 0;
margin-left: 8px;
2021-06-21 04:29:55 +00:00
}
.modal-footer {
margin-top: -20px;
}
.footer {
display: flex;
margin-top: 20px;
2021-06-21 04:29:55 +00:00
}
</style>
<style>
.importingFailed input {
color: red;
2021-06-21 04:29:55 +00:00
}
.field-buttons .v-select {
width: 220px;
2021-07-24 22:08:52 +00:00
}
.presets .vue-select[aria-expanded="true"] vue-dropdown {
visibility: hidden;
2021-07-24 22:08:52 +00:00
}
2021-06-21 04:29:55 +00:00
</style>