1
0
Fork 0
This commit is contained in:
iTNTPiston 2022-07-06 20:39:33 -07:00
commit 59a329b7bd
19 changed files with 1137 additions and 1222 deletions

View file

@ -24,6 +24,10 @@ h1 {
font-size: 10pt;
}
.CommandItemInvalid {
color: red;
}
.CommandItem {
padding: 2px 2px 2px 10px;
cursor: pointer;
@ -110,7 +114,7 @@ p.Reference {
padding-left: 20px;
}
p.Example {
.Example {
color:#eeee00;
}
@ -123,3 +127,65 @@ h3.Reference2 {
margin-top: 0;
margin-bottom: 0;
}
button.MainButton {
font-family: CalamitySans;
font-weight: bold;
display: inline-block;
padding: 5px;
width: 100px;
margin: 5px;
background-color: #00000099;
color: #ffffff;
border-radius: 0;
border: 2px solid #888888;
}
button.MainButton:hover {
border-color: #eeeeee;
box-shadow: 0 0 5px #eeeeee;
}
button.MainButton:active {
background-color: #888888;
}
.FullWidth {
width: 100% !important;
}
div.OtherPage {
height: 100%;
width: 100%;
color: #ffffff;
}
div.OtherPageContent{
padding: 10px;
}
.MainInput {
display: block;
width: 100%;
background-color: #00000099;
color: #ffffff;
border-radius: 0;
border: 2px solid #888888;
margin-bottom: 5px;
}
input.MainInput {
font-size: 14pt;
font-family: CalamitySans;
}
textarea {
resize: none;
height: 300px;
}
.MainInput:focus-visible {
outline: none;
border-color: #eeeeee;
box-shadow: 0 0 5px #eeeeee;
}

View file

@ -1,49 +1,46 @@
import { Command, CommandNothing } from "core/Command";
import { Inventory } from "core/Inventory";
import { Command, CommandNop } from "core/Command";
import React, { useCallback, useEffect, useMemo, useRef, useState } from "react";
import "./App.css";
import { CommandItem } from "./components/CommandItem";
import { DisplayPane } from "surfaces/DisplayPane";
import { Item } from "core/Item";
import { saveAs } from "data/FileSaver";
import { parseCommand } from "core/Parser";
import { ItemList } from "components/ItemList";
import { TitledList } from "components/TitledList";
import { createSimulationState, SimulationState } from "core/SimulationState";
import { ReferencePage } from "surfaces/ReferencePage";
import { GameData } from "core/GameData";
import { OptionPage } from "surfaces/OptionPage";
const getDefaultCommands = (): Command[]=>{
const encoded = localStorage.getItem("HDS.CurrentCommandsText");
if(encoded){
const lines = encoded.split("\n");
return lines.map(l=>parseCommand(l)).filter(c=>c) as Command[];
return lines.map(parseCommand);
}
return [
parseCommand("Get 5 Diamond 1 Slate 1 Glider 4 SpiritOrb"),
parseCommand("Save"),
parseCommand("# Magically break 4 slots"),
parseCommand("Break 4 Slots"),
parseCommand("Reload"),
parseCommand("Save"),
parseCommand("Reload"),
] as Command[];;
] as Command[];
};
export const App: React.FC = () => {
const [page, setPageInState] = useState<string>("#simulation");
const [overlaySave, setOverlaySave] = useState<boolean>(false);
// Option States
const [interlaceInventory, setInterlaceInventory] = useState<boolean>(false);
const [commands, setCommands] = useState<Command[]>(getDefaultCommands());
const [selectedSaveName, setSelectedSaveName] = useState<string>("");
const [displayIndex, setDisplayIndex] = useState<number>(0);
const [contextMenuX, setContextMenuX] = useState<number>(0);
const [contextMenuY, setContextMenuY] = useState<number>(0);
const [contextMenuShowing, setContextMenuShowing] = useState<boolean>(false);
const [contextIndex, setContextIndex] = useState<number>(-1);
const uploadRef = useRef<HTMLInputElement>(null);
const contextMenuRef = useRef<HTMLDivElement>(null);
// compute props
const simulationStates = useMemo(()=>{
@ -55,6 +52,9 @@ export const App: React.FC = () => {
});
return simulationStates;
}, [commands]);
const commandText = useMemo(()=>{
return commands.map(c=>c.getDisplayString()).join("\n");
}, [commands]);
const setPage = useCallback((hash: string)=>{
window.location.hash = hash;
@ -68,16 +68,25 @@ export const App: React.FC = () => {
useEffect(()=>{
window.onkeydown=(e)=>{
if(e.code==="ArrowDown"){
if(displayIndex===commands.length-1){
let nextCommandIndex = displayIndex+1;
while(nextCommandIndex<commands.length && !commands[nextCommandIndex].isValid()){
nextCommandIndex++;
}
if(nextCommandIndex===commands.length-1){
const arrCopy = [...commands];
arrCopy.push(new CommandNothing());
arrCopy.push(new CommandNop(""));
setCommands(arrCopy);
setDisplayIndex(arrCopy.length-1);
}else{
setDisplayIndex(Math.min(commands.length-1, displayIndex+1));
setDisplayIndex(Math.min(commands.length-1, nextCommandIndex));
}
}else if(e.code==="ArrowUp"){
setDisplayIndex(Math.max(0, displayIndex-1));
let nextCommandIndex = displayIndex-1;
while(nextCommandIndex>=0 && !commands[nextCommandIndex].isValid()){
nextCommandIndex--;
}
setDisplayIndex(Math.max(0, nextCommandIndex));
}
};
}, [commands, displayIndex]);
@ -92,40 +101,32 @@ export const App: React.FC = () => {
}, [commands]);
useEffect(()=>{
if(contextMenuRef.current && contextMenuShowing){
if(contextIndex < 0 || contextIndex >= commands.length){
setContextIndex(-1);
}else if(contextMenuRef.current){
const rect = contextMenuRef.current.getBoundingClientRect();
if (rect.bottom > window.innerHeight){
setContextMenuY(contextMenuY-rect.height);
}
}
}, [contextMenuRef, contextMenuShowing]);
}, [contextMenuRef, contextIndex, commands]);
return (
<div className='Calamity'>
<input ref={uploadRef} id="Upload" type="File" hidden onChange={(e)=>{
const files = e.target.files;
if(files?.length && files[0]){
const file = files[0];
file.text().then(text=>{
const lines = text.split("\n");
const parsedCommands: Command[] = lines.map(l=>parseCommand(l)).filter(c=>c) as Command[];
setDisplayIndex(0);
setContextIndex(-1);
setContextMenuShowing(false);
setCommands(parsedCommands);
});
}
}}/>
<div id="NavBar" style={{
backgroundColor: "#262626",
height: 40
}}>
<button onClick={()=>{
setPage("#simulation")
<button className="MainButton" onClick={()=>{
setPage("#simulation");
}}>Simulation</button>
<button onClick={()=>{
setPage("#reference")
<button className="MainButton" onClick={()=>{
setPage("#reference");
}}>Reference</button>
<button>Options</button>
<button className="MainButton" onClick={()=>{
setPage("#options");
}}>Options</button>
</div>
<div id="SidePane" style={{
@ -147,31 +148,33 @@ export const App: React.FC = () => {
{
!!simulationStates[displayIndex].getManualSave() &&
<CommandItem
onClick={()=>{
setSelectedSaveName("");
}}
comment={false}
isSelected={selectedSaveName===""}
onClick={()=>{
setSelectedSaveName("");
setPage("#simulation");
}}
useListItem
isSelected={selectedSaveName===""}
>
>
Manual Save
</CommandItem>
</CommandItem>
}
{
Object.entries(simulationStates[displayIndex].getNamedSaves()).map(([name, _gamedata])=>(
<CommandItem
onClick={()=>{
setSelectedSaveName(name);
}}
comment={false}
isSelected={selectedSaveName===name}
>
{name}
</CommandItem>
))
}
</ol>
{
Object.entries(simulationStates[displayIndex].getNamedSaves()).map(([name, _gamedata])=>
<CommandItem
onClick={()=>{
setSelectedSaveName(name);
setPage("#simulation");
}}
isSelected={selectedSaveName===name}
useListItem
>
{name}
</CommandItem>
)
}
</ol>
}
</TitledList>
@ -185,55 +188,46 @@ export const App: React.FC = () => {
}}>
<TitledList title="Instructions">
<ol style={{
}}>
{
commands.map((c,i)=>
<CommandItem
onClick={()=>{
setDisplayIndex(i);
const inputField = document.getElementById("CommandInputField");
if(inputField){
inputField.focus();
}
}}
onContextMenu={(x,y)=>{
setContextIndex(i);
setContextMenuX(x);
setContextMenuY(y);
setContextMenuShowing(true);
}}
key={i}
isSelected={displayIndex===i}
isContextSelected={contextIndex===i}
comment={c.getDisplayString().startsWith("#")}
>
{c.getDisplayString()}
</CommandItem>
)
}
<CommandItem onClick={()=>{
const arrCopy = [...commands];
arrCopy.push(new CommandNothing());
setCommands(arrCopy);
}} onContextMenu={()=>{
const arrCopy = [...commands];
arrCopy.push(new CommandNothing());
setCommands(arrCopy);
}}>(new)</CommandItem>
<CommandItem onClick={(x,y)=>{
setContextIndex(-1);
setContextMenuX(x);
setContextMenuY(y);
setContextMenuShowing(true);
}} onContextMenu={(x,y)=>{
setContextIndex(-1);
setContextMenuX(x);
setContextMenuY(y);
setContextMenuShowing(true);
}}>(options)</CommandItem>
<ol style={{
}}>
{
commands.map((c,i)=>
<CommandItem
onClick={()=>{
setDisplayIndex(i);
setPage("#simulation");
const inputField = document.getElementById("CommandInputField");
if(inputField){
inputField.focus();
}
}}
onContextMenu={(x,y)=>{
setContextIndex(i);
setContextMenuX(x);
setContextMenuY(y);
}}
key={i}
isSelected={displayIndex===i}
isContextSelected={contextIndex===i}
isComment={c.getDisplayString().startsWith("#")}
useListItem={!c.getDisplayString().startsWith("#")}
isInvalid={!c.isValid()}
>
{c.getDisplayString()}
</CommandItem>
)
}
<CommandItem onClick={()=>{
const arrCopy = [...commands];
arrCopy.push(new CommandNop(""));
setCommands(arrCopy);
}} onContextMenu={()=>{
const arrCopy = [...commands];
arrCopy.push(new CommandNop(""));
setCommands(arrCopy);
}}>(new)</CommandItem>
</ol>
</ol>
</TitledList>
</div>
@ -249,47 +243,51 @@ export const App: React.FC = () => {
}}>
{
page === "#simulation" && <>
<div style={{
maxHeight: 220,
height: "30vh",
border: "1px solid black",
boxSizing: "border-box",
overflowY: "hidden",
color: "white",
backgroundColor: "#262626"
} }>
{
(displayIndex >= 0 && displayIndex < commands.length) ?
<TitledList title="Save Data">
<div style={{
maxHeight: 220,
height: "30vh",
overflowY: "hidden",
color: "white",
backgroundColor: "#262626"
} }>
{
selectedSaveName === "" && !!simulationStates[displayIndex].getManualSave() &&
<ItemList slots={(simulationStates[displayIndex].getManualSave() as GameData).getDisplayedSlots()}/>
displayIndex >= 0 && displayIndex < commands.length ?
<TitledList title="Save Data">
{
(()=>{
if (selectedSaveName === ""){
const manualSave = simulationStates[displayIndex].getManualSave();
if(manualSave){
return <ItemList slots={manualSave.getDisplayedSlots()}/>;
}
}else if(selectedSaveName){
const namedSaves = simulationStates[displayIndex].getNamedSaves();
if(selectedSaveName in namedSaves){
const save = namedSaves[selectedSaveName];
return <ItemList slots={save.getDisplayedSlots()}/>;
}
}
return null;
})()
}
</TitledList>
:
<TitledList title="Select an instruction on the left to view it">
</TitledList>
}
{
selectedSaveName !== "" && !!simulationStates[displayIndex].getNamedSaves()[selectedSaveName] &&
<ItemList slots={(simulationStates[displayIndex].getNamedSaves()[selectedSaveName] as GameData).getDisplayedSlots()}/>
}
</TitledList>
:
<TitledList title="Select an instruction on the left to view it">
</TitledList>
}
</div>
<div style={{
minHeight: "calc( 70vh - 40px )",
height: "calc( 100vh - 40px - 220px )",
border: "1px solid black",
boxSizing: "border-box",
overflowY: "hidden"
}}>
{displayIndex >= 0 && displayIndex < commands.length &&
</div>
<div style={{
minHeight: "calc( 70vh - 40px )",
height: "calc( 100vh - 40px - 220px )",
border: "1px solid black",
boxSizing: "border-box",
overflowY: "hidden"
}}>
{displayIndex >= 0 && displayIndex < commands.length &&
<DisplayPane
overlaySave={overlaySave}
overlaySave={interlaceInventory}
displayIndex={displayIndex}
command={commands[displayIndex].getDisplayString()}
simulationState={simulationStates[displayIndex]}
@ -300,57 +298,39 @@ export const App: React.FC = () => {
}}
/>
}
</div>
</>
}
</div>
</>
}
{
page === "#reference" && <ReferencePage />
}
{
page === "#options" &&
<OptionPage
interlaceInventory={interlaceInventory}
setInterlaceInventory={setInterlaceInventory}
commandText={commandText}
setCommandText={(value)=>{
if(value !== commandText){
const commands = value.split("\n").map(parseCommand);
setCommands(commands);
}
}}
/>
}
</div>
{/* <div id="SavePane" style={{
height: "200px"
}}>
</div>
<div id="InstructionPane" style={{
}}> */}
{/* <div id="CommandList" style={{
width: "300px",
height: "calc( 60vh - 5px )",
overflowY: "auto",
border: "1px solid black",
boxSizing: "content-box"
} }>
</div> */}
{/* </div> */}
{
contextMenuShowing && <div style={{
contextIndex >= 0 && contextIndex < commands.length && <div style={{
position: "absolute",
top: 0,
left: 0,
width: "100vw",
height: "100vh",
}} onClick={()=>{
setContextMenuShowing(false);
setContextIndex(-1);
}} onContextMenu={(e)=>{
setContextMenuShowing(false);
setContextIndex(-1);
e.preventDefault();
}}>
@ -367,93 +347,33 @@ export const App: React.FC = () => {
listStyleType: "none",
paddingInlineStart: 0
}}>
{contextIndex >= 0 ? <>
<CommandItem onClick={()=>{
<CommandItem onClick={()=>{
const arrCopy = [...commands];
arrCopy.splice(contextIndex, 0, new CommandNop(""));
setCommands(arrCopy);
setContextIndex(-1);
}}>Insert Above</CommandItem>
<CommandItem onClick={()=>{
if(contextIndex > 0){
const arrCopy = [...commands];
arrCopy.splice(contextIndex, 0, new CommandNothing());
const temp = arrCopy[contextIndex];
arrCopy[contextIndex] = arrCopy[contextIndex-1];
arrCopy[contextIndex-1] = temp;
setCommands(arrCopy);
setContextMenuShowing(false);
setContextIndex(-1);
}}>Insert Above</CommandItem>
<CommandItem onClick={()=>{
if(contextIndex > 0){
const arrCopy = [...commands];
const temp = arrCopy[contextIndex];
arrCopy[contextIndex] = arrCopy[contextIndex-1];
arrCopy[contextIndex-1] = temp;
setCommands(arrCopy);
setContextMenuShowing(false);
setContextIndex(-1);
}
}
}}>Move Up</CommandItem>
<CommandItem onClick={()=>{
if(confirm("Delete?")){
setCommands(commands.filter((_,i)=>i!==contextIndex));
if(displayIndex >= commands.length){
setDisplayIndex(commands.length-1);
}
setContextMenuShowing(false);
setContextIndex(-1);
}}>Move Up</CommandItem>
<CommandItem onClick={()=>{
if(confirm("Delete?")){
setCommands(commands.filter((_,i)=>i!==contextIndex));
if(displayIndex >= commands.length){
setDisplayIndex(commands.length-1);
}
}}>Delete</CommandItem></> :
<>
<CommandItem onClick={()=>{
setOverlaySave(!overlaySave);
}}>Toggle Save Overlay</CommandItem>
<CommandItem onClick={()=>{
if(uploadRef.current){
uploadRef.current.click();
}
}}>Import</CommandItem>
<CommandItem onClick={()=>{
const lines = commands.map(c=>c.getDisplayString());
const text = lines.join("\n");
saveAs(text, "dupe.txt");
}}>Export</CommandItem>
<CommandItem onClick={()=>{
alert(`Available Commands:
Initialize X Item1 Y Item2 Z Item3 ...
Break X Slots - add X broken slots
Save
Reload
Sort Key/Material - sort key items or material
Get/Add/Cook/Pickup X ITEM
Remove/Drop/Sell X ITEM From Slot Y
Remove/Sell/Eat MEAL From Slot X
Limitations:
Inventory corruption is not implemented yet
`);
alert(`Available Items:
Slate
Glider
SpiritOrb
SpeedFood
Lotus
SilentPrincess
Honey
Acorn
FaroshScale
FaroshClaw
FaroshHorn
HeartyBass
Beetle
Opal
Diamond
Tail
Spring
Shaft
Core
Wood
Weapon
`);
}}>Reference</CommandItem>
</>
}
setContextIndex(-1);
}
}}>Delete</CommandItem>
</ul>
</div>

Binary file not shown.

After

Width:  |  Height:  |  Size: 32 KiB

View file

@ -1,29 +1,54 @@
import clsx from "clsx";
import { PropsWithChildren } from "react";
import React, { PropsWithChildren, useCallback } from "react";
type CommandItemProps = PropsWithChildren<{
useListItem?: boolean,
isSelected?: boolean,
comment?:boolean,
isComment?: boolean,
isInvalid?: boolean,
isContextSelected?: boolean,
onClick: (x: number, y: number)=>void,
onContextMenu?: (x: number, y: number)=>void
}>;
export const CommandItem: React.FC<CommandItemProps> = ({isSelected, isContextSelected, comment,children, onClick, onContextMenu}) => {
if(comment){
return <div className={clsx("CommandItem", isSelected && "CommandItemSelected", isContextSelected&& "CommandItemContextSelected",comment && "CommandItemComment")}>{children}</div>
export const CommandItem: React.FC<CommandItemProps> = ({
useListItem,
isSelected,
isContextSelected,
isComment,
isInvalid,
onClick,
onContextMenu,
children
}) => {
const className = clsx(
"CommandItem",
isSelected && "CommandItemSelected",
isContextSelected && "CommandItemContextSelected",
isComment && "CommandItemComment",
isInvalid && !isComment && "CommandItemInvalid",
);
const clickHandler = useCallback((e: React.MouseEvent)=>{
onClick(e.clientX, e.clientY);
}, [onClick]);
const contextMenuHandler = useCallback((e: React.MouseEvent)=>{
if(onContextMenu){
onContextMenu(e.clientX,e.clientY);
e.preventDefault();
}
}, [onContextMenu]);
if(!useListItem){
return (
<div className={className} onClick={clickHandler} onContextMenu={contextMenuHandler}>
{children}&nbsp;
</div>
);
}
return <li
className={clsx("CommandItem", isSelected && "CommandItemSelected", isContextSelected&& "CommandItemContextSelected",comment && "CommandItemComment")}
onClick={(e)=>{
onClick(e.clientX, e.clientY);
}}
onContextMenu={(e)=>{
if(onContextMenu){
onContextMenu(e.clientX,e.clientY);
e.preventDefault();
}
}}
>{children}&nbsp;</li>;
return (
<li className={className} onClick={clickHandler} onContextMenu={contextMenuHandler}>
{children}&nbsp;
</li>
);
};

View file

@ -2,7 +2,6 @@ import clsx from "clsx";
import { DisplayableSlot } from "core/DisplayableInventory";
import Background from "assets/Background.png";
type ItemSlotProps = {
slot: DisplayableSlot
};

View file

@ -1,25 +1,25 @@
import { PropsWithChildren } from "react"
import { PropsWithChildren } from "react";
type TitledListProps = PropsWithChildren<{
title: string
}>
export const TitledList: React.FC<TitledListProps> = ({title, children}) => {
return (
<>
<h3 className="ListHeader" style={{
height: 40,
borderBottom: "2px solid",
boxSizing: "border-box",
}}>
{title}
</h3>
<div style={{
height: "calc( 100% - 40px )",
overflowY: "auto"}}
>
{children}
</div>
</>
);
}
return (
<>
<h3 className="ListHeader" style={{
height: 40,
borderBottom: "2px solid",
boxSizing: "border-box",
}}>
{title}
</h3>
<div style={{
height: "calc( 100% - 40px )",
overflowY: "auto"}}
>
{children}
</div>
</>
);
};

View file

@ -1,27 +1,29 @@
import { Inventory } from "./Inventory";
import { Item, ItemStack, itemToArrowType } from "./Item";
import { Item, ItemStack } from "./Item";
import { SimulationState } from "./SimulationState";
export interface Command {
isValid(): boolean,
execute(state: SimulationState): void,
getDisplayString(): string,
}
export class CommandNothing implements Command {
execute(_state: Inventory): void {
class CommandImpl implements Command{
isValid(): boolean {
return true;
}
execute(_state: SimulationState): void {
// nothing
}
getDisplayString(): string {
return "";
throw new Error("Method not implemented.");
}
}
export class CommandInitialize implements Command {
export class CommandInitialize extends CommandImpl {
private stacks: ItemStack[];
constructor(stacks: ItemStack[]){
super();
this.stacks = stacks;
}
@ -34,7 +36,7 @@ export class CommandInitialize implements Command {
}
export class CommandSave implements Command {
export class CommandSave extends CommandImpl {
public execute(state: SimulationState): void {
state.save();
@ -44,9 +46,10 @@ export class CommandSave implements Command {
}
}
export class CommandSaveAs implements Command {
export class CommandSaveAs extends CommandImpl {
private name: string;
constructor(name: string){
super();
this.name = name;
}
public execute(state: SimulationState): void {
@ -57,9 +60,10 @@ export class CommandSaveAs implements Command {
}
}
export class CommandReload implements Command {
export class CommandReload extends CommandImpl {
private name?: string;
constructor(name?: string){
super();
this.name = name;
}
public execute(state: SimulationState): void {
@ -70,10 +74,10 @@ export class CommandReload implements Command {
}
}
export class CommandUse implements Command {
export class CommandUse extends CommandImpl{
private name: string;
constructor(name: string){
super();
this.name = name;
}
public execute(state: SimulationState): void {
@ -82,12 +86,16 @@ export class CommandUse implements Command {
public getDisplayString(): string {
return `Use ${this.name}`;
}
public isValid(): boolean {
return false; // this command is deprecated
}
}
export class CommandBreakSlots implements Command {
export class CommandBreakSlots extends CommandImpl {
private numToBreak: number;
constructor(numToBreak: number){
super();
this.numToBreak = numToBreak;
}
@ -99,11 +107,12 @@ export class CommandBreakSlots implements Command {
}
}
export class CommandAdd implements Command {
export class CommandAdd extends CommandImpl {
private verb: string;
private count: number;
private item: Item;
constructor(verb: string, count: number, item: Item){
super();
this.verb = verb;
this.count = count;
this.item = item;
@ -117,10 +126,11 @@ export class CommandAdd implements Command {
}
}
export class CommandAddWithoutCount implements Command {
export class CommandAddWithoutCount extends CommandImpl {
private verb: string;
private item: Item;
constructor(verb: string, item: Item){
super();
this.verb = verb;
this.item = item;
}
@ -133,10 +143,11 @@ export class CommandAddWithoutCount implements Command {
}
}
export class CommandAddMultiple implements Command {
export class CommandAddMultiple extends CommandImpl {
private verb: string;
private stacks: ItemStack[];
constructor(verb: string, stacks: ItemStack[]){
super();
this.verb = verb;
this.stacks = stacks;
}
@ -150,13 +161,14 @@ export class CommandAddMultiple implements Command {
}
}
export class CommandRemove implements Command {
export class CommandRemove extends CommandImpl {
private verb: string;
private count: number;
private item: Item;
private slot: number;
private noSlot: boolean;
constructor(verb: string, count: number, item: Item, slot: number, noSlot: boolean){
super();
this.verb = verb;
this.count = count;
this.item = item;
@ -172,12 +184,13 @@ export class CommandRemove implements Command {
}
}
export class CommandRemoveWithoutCount implements Command {
export class CommandRemoveWithoutCount extends CommandImpl {
private verb: string;
private item: Item;
private slot: number;
private noSlot: boolean;
constructor(verb: string, item: Item, slot: number, noSlot: boolean){
super();
this.verb = verb;
this.item = item;
this.slot = slot;
@ -192,10 +205,11 @@ export class CommandRemoveWithoutCount implements Command {
}
}
export class CommandRemoveMultiple implements Command {
export class CommandRemoveMultiple extends CommandImpl {
private verb: string;
private stacks: ItemStack[];
constructor(verb: string, stacks: ItemStack[]){
super();
this.verb = verb;
this.stacks = stacks;
}
@ -215,30 +229,33 @@ const joinItemStackString = (initial: string, stacks: ItemStack[]): string => {
parts.push(item);
});
return parts.join(" ");
}
};
export class CommandDaP implements Command {
private count: number;
private item: Item;
export class CommandDaP extends CommandImpl {
private stacks: ItemStack[];
constructor(count: number, item: Item,){
this.count = count;
this.item = item;
constructor(stacks: ItemStack[]){
super();
this.stacks = stacks;
}
public execute(state: SimulationState): void {
state.remove(this.item, this.count, 0);
state.obtain(this.item, this.count);
this.stacks.forEach(({item,count})=>{
state.remove(item, count, 0);
state.obtain(item, count);
});
}
public getDisplayString(): string {
return `D&P ${this.count} ${this.item}`;
return joinItemStackString("D&P", this.stacks);
}
}
export class CommandEquip implements Command {
export class CommandEquip extends CommandImpl {
private item: Item;
private slot: number;
private noSlot: boolean;
constructor(item: Item, slot: number, noSlot: boolean){
super();
this.item = item;
this.slot = slot;
this.noSlot = noSlot;
@ -253,11 +270,12 @@ export class CommandEquip implements Command {
}
}
export class CommandUnequip implements Command {
export class CommandUnequip extends CommandImpl {
private item: Item;
private slot: number;
private noSlot: boolean;
constructor(item: Item, slot: number, noSlot: boolean){
super();
this.item = item;
this.slot = slot;
this.noSlot = noSlot;
@ -272,9 +290,10 @@ export class CommandUnequip implements Command {
}
}
export class CommandShootArrow implements Command {
private count: number
export class CommandShootArrow extends CommandImpl {
private count: number;
constructor(count: number){
super();
this.count = count;
}
@ -286,85 +305,94 @@ export class CommandShootArrow implements Command {
}
}
export class CommandCloseGame extends CommandImpl {
public execute(state: SimulationState): void {
state.closeGame();
}
public getDisplayString(): string {
return "Close Game";
}
}
// export class CommandEquipArrow implements Command {
// private item: Item;
// private slot: number;
// private noSlot: boolean;
// constructor(item: Item, slot: number, noSlot: boolean){
// this.item = item;
// this.slot = slot;
// this.noSlot = noSlot;
// }
// public execute(inv: Inventory): void {
// inv.equipEquipmentOrArrow(this.item, this.slot);
// }
// public getDisplayString(): string {
// const slotString = this.noSlot ? "" : ` In Slot ${this.slot+1}`;
// return `Equip ${itemToArrowType(this.item)} Arrow${slotString}`;
// }
// }
export class CommandSync extends CommandImpl {
private actionString: string;
constructor(actionString: string){
super();
this.actionString = actionString;
}
public execute(state: SimulationState): void {
state.syncGameDataWithPouch();
}
public getDisplayString(): string {
return this.actionString;
}
}
export class CommandEventide extends CommandImpl {
private enter: boolean;
constructor(enter: boolean){
super();
this.enter = enter;
}
public execute(state: SimulationState): void {
state.setEventide(this.enter);
}
public getDisplayString(): string {
return `${this.enter? "Enter":"Exit"} Eventide`;
}
}
// export class CommandCloseGame implements Command {
// public execute(inv: Inventory): void {
// inv.closeGame();
// }
// public getDisplayString(): string {
// return "Close Game";
// }
// }
export class CommandComment implements Command {
private name: string;
constructor(name: string){
this.name = name;
export class CommandNop extends CommandImpl {
private text: string;
constructor(text: string){
super();
this.text = text;
}
public isValid(): boolean {
return false;
}
public execute(_state: SimulationState): void {
// nothing
}
public getDisplayString(): string {
return `# ${this.name}`;
return this.text;
}
}
// export class CommandSortKey implements Command {
// static Op = 0x5;
// // public fromBuffer(_buf: Buffer): number {
// // return 0;
// // }
// // public toBuffer(): Buffer {
// // const buf: Buffer = Buffer.alloc(1);
// // buf.writeInt8(CommandSortKey.Op);
// // return buf;
// // }
// public execute(inv: Inventory): void {
// inv.sortKey();
// }
// public getDisplayString(): string {
// return "Sort Key";
// }
// }
export class CommandSortKey extends CommandImpl {
static Op = 0x5;
// public fromBuffer(_buf: Buffer): number {
// return 0;
// }
// public toBuffer(): Buffer {
// const buf: Buffer = Buffer.alloc(1);
// buf.writeInt8(CommandSortKey.Op);
// return buf;
// }
public execute(_state: SimulationState): void {
// wip
}
public getDisplayString(): string {
return "Sort Key";
}
}
// export class CommandSortMaterial implements Command {
// static Op = 0x6;
// // public fromBuffer(_buf: Buffer): number {
// // return 0;
// // }
// // public toBuffer(): Buffer {
// // const buf: Buffer = Buffer.alloc(1);
// // buf.writeInt8(CommandSortMaterial.Op);
// // return buf;
// // }
// public execute(inv: Inventory): void {
// inv.sortMaterial();
// }
// public getDisplayString(): string {
// return "Sort Material";
// }
// }
export class CommandSortMaterial extends CommandImpl {
static Op = 0x6;
// public fromBuffer(_buf: Buffer): number {
// return 0;
// }
// public toBuffer(): Buffer {
// const buf: Buffer = Buffer.alloc(1);
// buf.writeInt8(CommandSortMaterial.Op);
// return buf;
// }
public execute(_state: SimulationState): void {
// wip
}
public getDisplayString(): string {
return "Sort Material";
}
}

View file

@ -1,4 +1,4 @@
import { Item, ItemStack, itemToItemData, ItemType } from "./Item"
import { ItemStack, itemToItemData, ItemType } from "./Item";
export type DisplayableSlot = {
image: string,
@ -13,12 +13,13 @@ export interface DisplayableInventory {
}
export const itemStackToDisplayableSlot = ({item, count, equipped}: ItemStack, isBrokenSlot: boolean): DisplayableSlot => {
const data = itemToItemData(item);
return {
image: data.image,
displayCount: data.stackable && (data.type === ItemType.Arrow || count > 0),
count,
isEquipped: equipped,
isBrokenSlot
}
}
const data = itemToItemData(item);
return {
image: data.image,
// for unstackable items (meal/key items) display count if count > 1, even if it's unstackable
displayCount: data.stackable ? data.type === ItemType.Arrow || count > 0 : count > 1,
count,
isEquipped: equipped,
isBrokenSlot
};
};

View file

@ -1,5 +1,4 @@
import { DisplayableInventory, DisplayableSlot, itemStackToDisplayableSlot } from "./DisplayableInventory";
import { Item, itemToItemData } from "./Item";
import { Slots } from "./Slots";
import { VisibleInventory } from "./VisibleInventory";
@ -8,28 +7,28 @@ import { VisibleInventory } from "./VisibleInventory";
*/
export class GameData implements DisplayableInventory {
private slots: Slots = new Slots([]);
constructor(slots: Slots){
this.slots = slots;
}
private slots: Slots = new Slots([]);
constructor(slots: Slots){
this.slots = slots;
}
public deepClone(): GameData {
return new GameData(this.slots.deepClone());
}
public deepClone(): GameData {
return new GameData(this.slots.deepClone());
}
public syncWith(pouch: VisibleInventory) {
this.slots = pouch.getSlots().deepClone();
}
public syncWith(pouch: VisibleInventory) {
this.slots = pouch.getSlots().deepClone();
}
public updateDurability(durability: number, slot: number){
this.slots.corrupt(durability, slot);
}
public updateDurability(durability: number, slot: number){
this.slots.corrupt(durability, slot);
}
public addAllToPouchOnReload(pouch: VisibleInventory) {
this.slots.getSlotsRef().forEach(stack=>pouch.addWhenReload(stack.item, stack.count, stack.equipped));
}
public addAllToPouchOnReload(pouch: VisibleInventory) {
this.slots.getSlotsRef().forEach(stack=>pouch.addWhenReload(stack.item, stack.count, stack.equipped));
}
public getDisplayedSlots(): DisplayableSlot[] {
return this.slots.getSlotsRef().map(stack=>itemStackToDisplayableSlot(stack, false));
}
public getDisplayedSlots(): DisplayableSlot[] {
return this.slots.getSlotsRef().map(stack=>itemStackToDisplayableSlot(stack, false));
}
}

View file

@ -1,215 +0,0 @@
import { Item, ItemStack, itemToItemData, ItemType, ItemTypes } from "./Item";
import { Slots } from "./Slots";
export class Inventory {
// private slots: Slots = new Slots([]);
// private savedSlots: Slots = new Slots([]);
// private namedSlots: {[name: string]: Slots} = {};
// private numBroken = 0;
// private isInitialSort = false;
// private isAltered = true;
// private inaccurate = false;
// private turnedInOrbs = 0;
// public deepClone(): Inventory {
// const other = new Inventory();
// other.slots = this.slots.deepClone();
// other.savedSlots = this.savedSlots.deepClone();
// other.numBroken = this.numBroken;
// other.isInitialSort = this.isInitialSort;
// other.isAltered = this.isAltered;
// other.inaccurate = this.inaccurate;
// other.turnedInOrbs = this.turnedInOrbs;
// other.namedSlots = {};
// for(const name in this.namedSlots){
// other.namedSlots[name] = this.namedSlots[name].deepClone();
// }
// return other;
// }
// public getSlots(): Slots {
// return this.slots;
// }
// public getSavedSlots(): Slots {
// return this.savedSlots;
// }
// public getNumBroken(): number {
// return this.numBroken;
// }
// public isInaccurate(): boolean {
// return this.inaccurate;
// }
// public getTurnedInOrbs(): number {
// return this.turnedInOrbs;
// }
// public rawInit(stacks: ItemStack[]) {
// this.slots = new Slots(stacks);
// }
// public init(stacks: ItemStack[]) {
// this.slots = new Slots([]);
// stacks.forEach(s=>{
// this.slots.add(s.item, s.count);
// });
// this.numBroken = 0;
// this.isInitialSort = false;
// this.isAltered = true;
// this.inaccurate = false;
// }
// public closeGame() {
// this.numBroken = 0;
// this.isInitialSort = false;
// this.isAltered = true;
// this.inaccurate = false;
// this.slots = new Slots([]);
// }
// public addBrokenSlots(num: number) {
// this.numBroken+=num;
// }
// public setTag(name: string){
// this.namedSlots[name] = this.savedSlots.deepClone();
// }
// public applyTag(name: string){
// if(name in this.namedSlots){
// this.savedSlots = this.namedSlots[name].deepClone();
// }else{
// this.savedSlots = new Slots([]);
// }
// }
// public save() {
// if(this.isAltered){
// this.savedSlots = this.slots.deepClone();
// }
// // Inventory Corruption
// // get durability transfer slots
// const durabilityTransferSlots: number[] = [];
// const equippedWeapon = this.slots.getFirstEquippedSlotIndex(ItemType.Weapon);
// if(equippedWeapon>=0){
// durabilityTransferSlots.push(equippedWeapon);
// }
// const equippedBow = this.slots.getFirstEquippedSlotIndex(ItemType.Bow);
// if(equippedBow>=0){
// durabilityTransferSlots.push(equippedBow);
// }
// const equippedShield = this.slots.getFirstEquippedSlotIndex(ItemType.Shield);
// if(equippedShield>=0){
// durabilityTransferSlots.push(equippedShield);
// }
// durabilityTransferSlots.forEach(s=>{
// if(s<this.savedSlots.length){
// // We ignore the case where durability transfer happens from equipment to equipment
// if(itemToItemData(this.savedSlots.get(s).item).stackable){
// this.savedSlots.get(s).count = 999;
// }
// }
// });
// }
// public reload() {
// // get things to dupe
// const dupeMap: {[k in ItemType]: Slots} = {
// [ItemType.Weapon]: new Slots([]),
// [ItemType.Bow]: new Slots([]),
// [ItemType.Arrow]: new Slots([]),
// [ItemType.Shield]: new Slots([]),
// [ItemType.Material]: new Slots([]),
// [ItemType.Meal]: new Slots([]),
// [ItemType.Key]: new Slots([])
// };
// for(let i=Math.max(0, this.slots.length-this.numBroken);i<this.slots.length;i++){
// const stack = this.slots.get(i);
// const itemData = itemToItemData(stack.item);
// dupeMap[itemData.type].addStackCopy(stack);
// }
// // apply dupe
// //console.log(dupeMap);
// this.slots = new Slots([]);
// // const dupeType = (type: ItemType) => {
// // }
// ItemTypes.forEach(type=>{
// this.slots.addSlotsToEnd(dupeMap[type]);
// this.slots.addSlotsToEnd(this.savedSlots.getByType(type).deepClone());
// });
// this.slots.sortArrows();
// this.isInitialSort = true;
// this.isAltered = false;
// }
// public sortKey() {
// const nonKeyItems = this.slots.getBeforeType(ItemType.Key);
// const keyItems = this.slots.getByType(ItemType.Key);
// keyItems.sort();
// nonKeyItems.addSlotsToEnd(keyItems);
// this.slots = nonKeyItems;
// this.isAltered=true;
// this.isInitialSort=false;
// }
// public sortMaterial() {
// const beforeMaterial = this.slots.getBeforeType(ItemType.Material);
// const afterMaterial = this.slots.getAfterType(ItemType.Material);
// const materials = this.slots.getByType(ItemType.Material);
// if(this.isInitialSort){
// // the materials in broken slots are not sorted
// const brokenSlots = Math.max(0, this.numBroken - afterMaterial.length);
// const noSortPart = materials.removeFromEnd(brokenSlots);
// materials.sort();
// beforeMaterial.addSlotsToEnd(materials);
// beforeMaterial.addSlotsToEnd(noSortPart);
// beforeMaterial.addSlotsToEnd(afterMaterial);
// }else{
// materials.sort();
// beforeMaterial.addSlotsToEnd(materials);
// beforeMaterial.addSlotsToEnd(afterMaterial);
// }
// this.slots = beforeMaterial;
// this.isInitialSort = false;
// this.isAltered=true;
// }
// public remove(item: Item, count: number, slot: number) {
// this.slots.remove(item, count, slot);
// if(item===Item.SpiritOrb){
// this.turnedInOrbs+=count;
// }
// this.isAltered=true;
// }
// public add(item: Item, count: number) {
// this.slots.add(item, count);
// if(itemToItemData(item).type===ItemType.Arrow){
// this.slots.sortArrows();
// }
// this.isAltered=true;
// }
// public equipEquipmentOrArrow(item: Item, slot: number) {
// this.slots.equip(item, slot);
// this.isAltered=true;
// }
// public unequipEquipment(item: Item, slot: number){
// this.slots.unequip(item, slot);
// this.isAltered=true;
// }
// public shootArrow(item: Item, count: number){
// this.slots.shoot(item, count);
// this.isAltered=true;
// }
}

View file

@ -76,6 +76,7 @@ export enum Item {
Fairy = "Fairy",
MasterSword = "MasterSword",
ZoraArmor = "ZoraArmor",
}
type ItemData = {
@ -167,7 +168,7 @@ register(0x50, Item.Weapon, ItemType.Weapon, {
});
register(0, Item.MasterSword, ItemType.Weapon, {
stackable: false,
})
});
register(0x60, Item.Bow, ItemType.Bow, {
image: Images.ForestDwellerBow,
@ -184,6 +185,10 @@ register(0x80, Item.Shield, ItemType.Shield, {
stackable: false
});
register(9, Item.ZoraArmor, ItemType.Armor, {
stackable: false
});
//export const idToItemData = (id: number): ItemData => IdToData[id];
export const itemToItemData = (item: Item): ItemData => ItemToData[item] as ItemData;
export const itemToArrowType = (item: Item): string => {
@ -193,3 +198,5 @@ export const itemToArrowType = (item: Item): string => {
}
return "";
};
export const getAllItems = (): string[] => Object.keys(ItemToData);

View file

@ -4,11 +4,12 @@ import {
CommandAddMultiple,
CommandAddWithoutCount,
CommandBreakSlots,
CommandComment,
CommandCloseGame,
CommandDaP,
CommandEquip,
CommandEventide,
CommandInitialize,
CommandNothing,
CommandNop,
CommandReload,
CommandRemove,
CommandRemoveMultiple,
@ -16,30 +17,32 @@ import {
CommandSave,
CommandSaveAs,
CommandShootArrow,
CommandSortKey,
CommandSortMaterial,
CommandSync,
CommandUnequip,
CommandUse
} from "./Command";
import { Item, ItemStack } from "./Item";
export const parseCommand = (cmdString: string): Command | undefined => {
export const parseCommand = (cmdString: string): Command => {
if(cmdString.startsWith("# ")){
return new CommandComment(cmdString.substring(2));
}
const tokens = cmdString.split(" ").filter(i=>i);
if(tokens.length===0){
return new CommandNothing();
return new CommandNop("");
}
// intialize
if(tokens.length>1 && tokens[0] === "Initialize"){
const stacks = parseItemStacks(tokens, 1);
return stacks ? new CommandInitialize(stacks) : undefined;
if(stacks){
return new CommandInitialize(stacks);
}
}
// Save/Reload
if(tokens.length===1 && tokens[0] === "Save"){
return new CommandSave();
}
// // Multi Save
// Multi Save
if (tokens.length === 3 && tokens[0] === "Save" && tokens[1] === "As"){
const name = tokens[2];
return new CommandSaveAs(name);
@ -54,7 +57,7 @@ export const parseCommand = (cmdString: string): Command | undefined => {
if(tokens.length===2 && tokens[0] === "Reload"){
return new CommandReload(tokens[1]);
}
// break
// break
if (tokens.length > 2 && tokens[0] === "Break" && tokens[2]=== "Slots" ){
const slots = parseInt(tokens[1]);
if(Number.isInteger(slots)){
@ -69,18 +72,18 @@ export const parseCommand = (cmdString: string): Command | undefined => {
if(Number.isInteger(count) && item in Item){
return new CommandAdd(tokens[0], count, Item[item as keyof typeof Item]);
}
return undefined;
}
if (tokens.length === 2 && isAddVerb(tokens[0])){
const item = tokens[1];
if(item in Item){
return new CommandAddWithoutCount(tokens[0], Item[item as keyof typeof Item]);
}
return undefined;
}
if(tokens.length>2 && isAddVerb(tokens[0])){
const stacks = parseItemStacks(tokens, 1);
return stacks ? new CommandAddMultiple(tokens[0], stacks) : undefined;
if(stacks){
return new CommandAddMultiple(tokens[0], stacks);
}
}
// remove X item From Slot Y
if (tokens.length === 6 && isRemoveVerb(tokens[0]) && tokens[3] === "From" && tokens[4] ==="Slot" ){
@ -90,7 +93,6 @@ export const parseCommand = (cmdString: string): Command | undefined => {
if(Number.isInteger(count) && Number.isInteger(slot) && item in Item){
return new CommandRemove(tokens[0], count, Item[item as keyof typeof Item], slot-1, false);
}
return undefined;
}
// remove X item
if (tokens.length === 3 && isRemoveVerb(tokens[0]) ){
@ -99,7 +101,6 @@ export const parseCommand = (cmdString: string): Command | undefined => {
if(Number.isInteger(count) && item in Item){
return new CommandRemove(tokens[0], count, Item[item as keyof typeof Item], 0, true);
}
return undefined;
}
// remove item From Slot Y
if (tokens.length === 5 && isRemoveVerb(tokens[0]) && tokens[2] === "From" && tokens[3] ==="Slot" ){
@ -108,7 +109,6 @@ export const parseCommand = (cmdString: string): Command | undefined => {
if(Number.isInteger(slot) && item in Item){
return new CommandRemoveWithoutCount(tokens[0], Item[item as keyof typeof Item], slot-1, false);
}
return undefined;
}
// remove item
if (tokens.length === 2 && isRemoveVerb(tokens[0]) ){
@ -116,21 +116,20 @@ export const parseCommand = (cmdString: string): Command | undefined => {
if(item in Item){
return new CommandRemoveWithoutCount(tokens[0], Item[item as keyof typeof Item], 0, true);
}
return undefined;
}
// remove multiple
if(tokens.length>2 && isRemoveVerb(tokens[0])){
const stacks = parseItemStacks(tokens, 1);
return stacks ? new CommandRemoveMultiple(tokens[0], stacks) : undefined;
if(stacks){
return new CommandRemoveMultiple(tokens[0], stacks);
}
}
//Shortcut for drop and pick up
if (tokens.length === 3 && tokens[0] === "D&P" ){
const count = parseInt(tokens[1]);
const item = tokens[2];
if(Number.isInteger(count) && item in Item){
return new CommandDaP(count, Item[item as keyof typeof Item]);
if (tokens.length >2 && tokens[0] === "D&P" ){
const stacks = parseItemStacks(tokens, 1);
if(stacks){
return new CommandDaP(stacks);
}
return undefined;
}
// Equip item In Slot X
@ -140,7 +139,6 @@ export const parseCommand = (cmdString: string): Command | undefined => {
if( Number.isInteger(slot) && item in Item){
return new CommandEquip(Item[item as keyof typeof Item], slot-1, false);
}
return undefined;
}
// Equip item
if (tokens.length === 2 && tokens[0] === "Equip"){
@ -148,7 +146,6 @@ export const parseCommand = (cmdString: string): Command | undefined => {
if( item in Item){
return new CommandEquip(Item[item as keyof typeof Item], 0, true);
}
return undefined;
}
// Unequip item in slot X
if (tokens.length === 5 && tokens[0] === "Unequip" && tokens[2] === "In" && tokens[3] ==="Slot" ){
@ -157,7 +154,6 @@ export const parseCommand = (cmdString: string): Command | undefined => {
if( Number.isInteger(slot) && item in Item){
return new CommandUnequip(Item[item as keyof typeof Item], slot-1, false);
}
return undefined;
}
// Unequip item
if (tokens.length === 2 && tokens[0] === "Unequip"){
@ -165,7 +161,6 @@ export const parseCommand = (cmdString: string): Command | undefined => {
if( item in Item){
return new CommandUnequip(Item[item as keyof typeof Item], -1, true);
}
return undefined;
}
// Shoot X Arrow
if (tokens.length === 3 && tokens[0] === "Shoot" && tokens[2] === "Arrow"){
@ -173,53 +168,34 @@ export const parseCommand = (cmdString: string): Command | undefined => {
if( Number.isInteger(count) ){
return new CommandShootArrow(count);
}
return undefined;
}
// if(tokens.length===2 && tokens[0] === "Sort" && tokens[1] === "Key"){
// return new CommandSortKey();
// }
// if(tokens.length===2 && tokens[0] === "Sort" && tokens[1] === "Material"){
// return new CommandSortMaterial();
// }
// if(tokens.length===2 && tokens[0] === "Close" && tokens[1] === "Game"){
// return new CommandCloseGame();
// }
if(tokens.length===2 && tokens[0] === "Sort" && tokens[1] === "Key"){
return new CommandSortKey();
}
if(tokens.length===2 && tokens[0] === "Sort" && tokens[1] === "Material"){
return new CommandSortMaterial();
}
if(tokens.length===2 && tokens[0] === "Close" && tokens[1] === "Game"){
return new CommandCloseGame();
}
if(tokens.length===2 && tokens[0] === "Sync" && tokens[1] === "GameData"){
return new CommandSync("Sync GameData");
}
if(tokens.length===2 && (tokens[0] === "Enter" || tokens[0] === "Exit") && tokens[1] === "Eventide"){
return new CommandEventide(tokens[0] === "Enter");
}
// return undefined;
// }
// // remove material
// // Equip Arrow
// if (tokens.length === 6 && tokens[0] === "Equip" && tokens[2] === "Arrow" && tokens[3] === "In" && tokens[4] ==="Slot" ){
// const item = tokens[1]+"Arrow";
// const slot = parseInt(tokens[5]);
// if( Number.isInteger(slot) && item in Item){
// return new CommandEquipArrow(Item[item as keyof typeof Item], slot-1, false);
// }
// return undefined;
// }
// if (tokens.length === 3 && tokens[0] === "Equip" && tokens[2] === "Arrow" ){
// const item = tokens[1]+"Arrow";
// if(item in Item){
// return new CommandEquipArrow(Item[item as keyof typeof Item], 0, true);
// }
// return undefined;
// }
return undefined;
return new CommandNop(cmdString);
};
const isAddVerb = (token: string): boolean => {
return token === "Get" || token === "Cook" || token === "Add" || token === "Pickup"
}
return token === "Get" || token === "Cook" || token === "Add" || token === "Pickup" || token === "Buy";
};
const isRemoveVerb = (token: string): boolean => {
return token === "Remove" || token === "Sell" || token === "Eat" || token === "Drop"
}
return token === "Remove" || token === "Sell" || token === "Eat" || token === "Drop" || token === "With";
};
const parseItemStacks = (tokens: string[], from: number): ItemStack[] | undefined => {
if((tokens.length-from)%2 !== 0){
@ -242,4 +218,4 @@ const parseItemStacks = (tokens: string[], from: number): ItemStack[] | undefine
}
}
return stacks;
}
};

View file

@ -5,160 +5,178 @@ import { Slots } from "./Slots";
import { VisibleInventory } from "./VisibleInventory";
export const createSimulationState = (): SimulationState => {
return new SimulationState(
new GameData(new Slots([])),
null,
{},
new VisibleInventory(new Slots([]), 0)
);
}
return new SimulationState(
new GameData(new Slots([])),
null,
{},
new VisibleInventory(new Slots([]), 0)
);
};
/*
* The state of simulation, including game data, visible inventory, and all save slots
*/
export class SimulationState {
private gameData: GameData;
private manualSave: GameData | null;
private namedSaves: {[name: string]: GameData} = {};
private pouch: VisibleInventory;
private nextReloadName?: string;
private isOnEventide: boolean = false;
private gameData: GameData;
private manualSave: GameData | null;
private namedSaves: {[name: string]: GameData} = {};
private pouch: VisibleInventory;
private nextReloadName?: string;
private isOnEventide = false;
constructor(gameData: GameData, manualSave: GameData | null, namedSaves: {[name: string]: GameData}, pouch: VisibleInventory){
this.gameData = gameData;
this.manualSave = manualSave;
this.namedSaves = namedSaves;
this.pouch = pouch;
}
constructor(gameData: GameData, manualSave: GameData | null, namedSaves: {[name: string]: GameData}, pouch: VisibleInventory){
this.gameData = gameData;
this.manualSave = manualSave;
this.namedSaves = namedSaves;
this.pouch = pouch;
}
public deepClone(): SimulationState {
const copyNamedSaves: {[name: string]: GameData} = {};
for(const name in this.namedSaves){
copyNamedSaves[name] = this.namedSaves[name].deepClone();
}
const newState = new SimulationState(
this.gameData.deepClone(),
this.manualSave ? this.manualSave.deepClone() : null,
copyNamedSaves,
this.pouch.deepClone()
);
newState.nextReloadName = this.nextReloadName;
newState.isOnEventide = this.isOnEventide;
public deepClone(): SimulationState {
const copyNamedSaves: {[name: string]: GameData} = {};
for(const name in this.namedSaves){
copyNamedSaves[name] = this.namedSaves[name].deepClone();
}
const newState = new SimulationState(
this.gameData.deepClone(),
this.manualSave ? this.manualSave.deepClone() : null,
copyNamedSaves,
this.pouch.deepClone()
);
newState.nextReloadName = this.nextReloadName;
newState.isOnEventide = this.isOnEventide;
return newState;
}
return newState;
}
public initialize(stacks: ItemStack[]) {
this.pouch = new VisibleInventory(new Slots([]), 0);
stacks.forEach((stack)=>this.pouch.addDirectly(stack));
this.gameData.syncWith(this.pouch);
}
public initialize(stacks: ItemStack[]) {
this.pouch = new VisibleInventory(new Slots([]), 0);
stacks.forEach((stack)=>this.pouch.addDirectly(stack));
this.gameData.syncWith(this.pouch);
}
public save(name?: string) {
if(name){
this.namedSaves[name] = this.gameData.deepClone();
}else{
this.manualSave = this.gameData.deepClone();
}
}
public save(name?: string) {
if(name){
this.namedSaves[name] = this.gameData.deepClone();
}else{
this.manualSave = this.gameData.deepClone();
}
}
public reload(name?: string) {
if(name){
if(name in this.namedSaves){
this.reloadFrom(this.namedSaves[name]);
}
}else{
if(this.nextReloadName){
if(this.nextReloadName in this.namedSaves){
this.reloadFrom(this.namedSaves[this.nextReloadName]);
}
}else{
const save = this.manualSave;
if(save){
this.reloadFrom(save);
}
}
}
}
public reload(name?: string) {
if(name){
if(name in this.namedSaves){
this.reloadFrom(this.namedSaves[name]);
}
}else{
if(this.nextReloadName){
if(this.nextReloadName in this.namedSaves){
this.reloadFrom(this.namedSaves[this.nextReloadName]);
}
}else{
const save = this.manualSave;
if(save){
this.reloadFrom(save);
}
}
}
}
private reloadFrom(data: GameData) {
this.gameData = data.deepClone();
this.pouch.clearForReload();
this.gameData.addAllToPouchOnReload(this.pouch);
this.pouch.updateEquipmentDurability(this.gameData);
}
private reloadFrom(data: GameData) {
this.gameData = data.deepClone();
this.pouch.clearForReload();
this.gameData.addAllToPouchOnReload(this.pouch);
this.pouch.updateEquipmentDurability(this.gameData);
this.isOnEventide = false;
}
public useSaveForNextReload(name: string){
this.nextReloadName = name;
}
public useSaveForNextReload(name: string){
this.nextReloadName = name;
}
public breakSlots(n: number) {
this.pouch.modifyCount(-n);
}
public breakSlots(n: number) {
this.pouch.modifyCount(-n);
}
public obtain(item: Item, count: number) {
this.pouch.addInGame(item, count);
this.syncGameDataWithPouch();
}
public obtain(item: Item, count: number) {
this.pouch.addInGame(item, count);
this.syncGameDataWithPouch();
}
public remove(item: Item, count: number, slot: number) {
this.pouch.remove(item, count, slot);
this.syncGameDataWithPouch();
}
public remove(item: Item, count: number, slot: number) {
this.pouch.remove(item, count, slot);
this.syncGameDataWithPouch();
}
public equip(item: Item, slot: number) {
this.pouch.equip(item, slot);
this.syncGameDataWithPouch();
}
public equip(item: Item, slot: number) {
this.pouch.equip(item, slot);
this.syncGameDataWithPouch();
}
public unequip(item: Item, slot: number){
this.pouch.unequip(item, slot);
this.syncGameDataWithPouch();
}
public unequip(item: Item, slot: number){
this.pouch.unequip(item, slot);
this.syncGameDataWithPouch();
}
public shootArrow(count: number){
this.pouch.shootArrow(count, this.gameData);
// does not sync
}
public shootArrow(count: number){
this.pouch.shootArrow(count, this.gameData);
// does not sync
}
public syncGameDataWithPouch() {
if(!this.isOnEventide){
this.gameData.syncWith(this.pouch);
}
}
public closeGame() {
this.pouch = new VisibleInventory(new Slots([]), 0);
this.gameData = new GameData(new Slots([]));
this.isOnEventide = false;
}
public get displayableGameData(): DisplayableInventory {
return this.gameData;
}
public setEventide(onEventide: boolean){
if(this.isOnEventide !== onEventide){
if(onEventide){
// clear everything except for key items
this.pouch.clearForEventide();
// game data is not updated (?)
}else{
// reload pouch from gamedata as if reloading a save
this.reloadFrom(this.gameData);
}
this.isOnEventide = onEventide;
}
}
public get displayablePouch(): DisplayableInventory {
return this.pouch;
}
public syncGameDataWithPouch() {
if(!this.isOnEventide){
this.gameData.syncWith(this.pouch);
}
}
public get inventoryMCount(): number {
return this.pouch.getCount();
}
public get displayableGameData(): DisplayableInventory {
return this.gameData;
}
public getManualSave(): GameData | null {
return this.manualSave;
}
public get displayablePouch(): DisplayableInventory {
return this.pouch;
}
public getNamedSaves(): {[name: string]: GameData} {
return this.namedSaves;
}
public get inventoryMCount(): number {
return this.pouch.getCount();
}
// public get displayableGameData(): DisplayableInventory {
// return this.gameData;
// }
public getManualSave(): GameData | null {
return this.manualSave;
}
public getNamedSaves(): {[name: string]: GameData} {
return this.namedSaves;
}
// public get displayableGameData(): DisplayableInventory {
// return this.gameData;
// }
}
// Shoot X Arrow, x can be ommited and default to 1
// Close Game
// Close Inventory, same as Resync GameData
// Enter Eventide / Leave Eventide
// Sort Key (In Tab X) - need more research on which tab is sorted. (might not be possible to select which tab to sort)

View file

@ -1,8 +1,6 @@
import { count } from "console";
import { stableSort } from "data/mergeSort";
import { Item, ItemStack, itemToItemData, ItemType } from "./Item";
/*
* This is the data model common to GameData and VisibleInventory
*/
@ -43,21 +41,7 @@ export class Slots {
public clearFirst(count: number) {
this.internalSlots.splice(0, count);
}
// public get(i: number): ItemStack{
// return this.internalSlots[i];
// }
// public getByType(type: ItemType): Slots {
// return new Slots(this.internalSlots.filter(s=>itemToItemData(s.item).type===type));
// }
// public getBeforeType(type: ItemType): Slots {
// return new Slots(this.internalSlots.filter(s=>itemToItemData(s.item).type<type));
// }
// public getAfterType(type: ItemType): Slots {
// return new Slots(this.internalSlots.filter(s=>itemToItemData(s.item).type>type));
// }
// public addSlotsToEnd(slots: Slots) {
// slots.internalSlots.forEach(s=>this.addStack(s));
// }
public addStackDirectly(stack: ItemStack): number {
const data = itemToItemData(stack.item);
if(data.stackable){
@ -73,18 +57,6 @@ export class Slots {
this.internalSlots.push({...stack});
this.sortItemType(mCount);
}
// public addStackCopy(stack: ItemStack) {
// this.addStack({...stack});
// }
// public sort() {
// this.internalSlots.sort((a,b)=>{
// return itemToItemData(a.item).sortOrder - itemToItemData(b.item).sortOrder;
// });
// }
// public removeFromEnd(count: number): Slots {
// const end = this.internalSlots.splice(-count, count);
// return new Slots(end);
// }
// remove item(s) start from slot
// return number of slots removed
@ -179,34 +151,29 @@ export class Slots {
}
if(reloading){
for(let i=0;i<count;i++){
this.addSlot({item,count:1,equipped: equippedDuringReload}, mCount+i+1);
this.addSlot({item,count,equipped: equippedDuringReload}, mCount+1);
return 1;
}
if(data.type===ItemType.Weapon || data.type===ItemType.Bow || data.type===ItemType.Shield){
//Check equip
const shouldEquipNew = this.internalSlots.filter(s=>{
const sData = itemToItemData(s.item);
return sData.type === data.type && s.equipped;
}).length === 0;
this.addSlot({item,count:1,equipped: shouldEquipNew}, mCount+1);
for(let i=1;i<count;i++){
this.addSlot({item,count:1,equipped: false}, mCount+i+1);
}
}else{
if(data.type===ItemType.Weapon || data.type===ItemType.Bow || data.type===ItemType.Shield){
//Check equip
const shouldEquipNew = this.internalSlots.filter(s=>{
const sData = itemToItemData(s.item);
return sData.type === data.type && s.equipped;
}).length === 0;
this.addSlot({item,count:1,equipped: shouldEquipNew}, mCount+1);
for(let i=1;i<count;i++){
this.addSlot({item,count:1,equipped: false}, mCount+i+1);
}
}else{
for(let i=0;i<count;i++){
this.addSlot({item,count:1,equipped: false}, mCount+i+1);
}
for(let i=0;i<count;i++){
this.addSlot({item,count:1,equipped: false}, mCount+i+1);
}
}
return count;
}
// this is for both equipments and arrows
public equip(item: Item, slot: number) {
let s = 0;
@ -259,8 +226,9 @@ export class Slots {
if(slot < 0 || slot >= this.internalSlots.length){
return;
}
const thisData = itemToItemData(this.internalSlots[slot].item);
if(thisData.stackable){
const thisData = itemToItemData(this.internalSlots[slot].item);
// Currently only supports corrupting arrows, material, meal and key items as durability values are not simulated on equipments
if(thisData.type >= ItemType.Material || thisData.stackable){
this.internalSlots[slot].count = durability;
}
}
@ -268,20 +236,22 @@ export class Slots {
// shoot count arrows. return the slot that was updated, or -1
public shootArrow(count: number): number {
// first find equipped arrow, search entire inventory
// this is the last equipped arrow before armor
let i=0;
let equippedArrow: Item | undefined = undefined;
for(;i<this.internalSlots.length;i++){
if(this.internalSlots[i].equipped){
const data = itemToItemData(this.internalSlots[i].item);
if(data.type === ItemType.Arrow){
break;
}
const data = itemToItemData(this.internalSlots[i].item);
if(data.type > ItemType.Shield){
break;
}
if(this.internalSlots[i].equipped && data.type === ItemType.Arrow){
equippedArrow = data.item;
}
}
if(i>=this.internalSlots.length){
//can't find equipped arrow
return -1;
}
const equippedArrow = this.internalSlots[i].item;
// now find the first slot of that arrow and update
for(let j=0;j<this.internalSlots.length;j++){
if(this.internalSlots[j].item === equippedArrow){
@ -293,34 +263,13 @@ export class Slots {
return -1;
}
// // Difference between shoot and remove:
// // 1. can only be from first (leftmost) slot
// // 2. empty slots not removed
// public shoot(item: Item, count: number) {
// for(let i = 0; i<this.internalSlots.length;i++){
// if(this.internalSlots[i].item === item){
// this.internalSlots[i].count-=count;
// }
// }
// }
// public sortArrows() {
// const after = this.removeFromEnd(this.getAfterType(ItemType.Arrow).length);
// const arrows = this.removeFromEnd(this.getByType(ItemType.Arrow).length);
// arrows.sort();
// this.addSlotsToEnd(arrows);
// this.addSlotsToEnd(after);
// }
// return how many slots are removed
public clearAllButKeyItems(): number {
const newslots = this.internalSlots.filter(stack=>itemToItemData(stack.item).type === ItemType.Key);
const removedCount = this.internalSlots.length - newslots.length;
this.internalSlots = newslots;
return removedCount;
}
// public getFirstEquippedSlotIndex(type: ItemType): number {
// for(let i = 0; i<this.internalSlots.length;i++){
// if(this.internalSlots[i].equipped){
// const data = itemToItemData(this.internalSlots[i].item);
// if(data.type === type){
// return i;
// }
// }
// }
// return -1;
// }
}

View file

@ -7,102 +7,106 @@ import { Slots } from "./Slots";
* Implementation of Visible Inventory (PauseMenuDataMgr) in botw
*/
export class VisibleInventory implements DisplayableInventory{
private slots: Slots = new Slots([]);
/* Implementation of mCount in botw */
private count: number = 0;
constructor(slots: Slots, count: number){
this.slots = slots;
this.count = count;
}
private slots: Slots = new Slots([]);
/* Implementation of mCount in botw */
private count = 0;
constructor(slots: Slots, count: number){
this.slots = slots;
this.count = count;
}
public deepClone(): VisibleInventory {
return new VisibleInventory(this.slots.deepClone(), this.count);
}
public deepClone(): VisibleInventory {
return new VisibleInventory(this.slots.deepClone(), this.count);
}
public getDisplayedSlots(): DisplayableSlot[] {
return this.slots.getSlotsRef().map((stack, i)=>itemStackToDisplayableSlot(stack, i>=this.count));
}
public getDisplayedSlots(): DisplayableSlot[] {
return this.slots.getSlotsRef().map((stack, i)=>itemStackToDisplayableSlot(stack, i>=this.count));
}
public getSlots(): Slots {
return this.slots;
}
public getSlots(): Slots {
return this.slots;
}
public addDirectly(stack: ItemStack){
this.count+=this.slots.addStackDirectly(stack);
}
public addDirectly(stack: ItemStack){
this.count+=this.slots.addStackDirectly(stack);
}
public addWhenReload(item: Item, count: number, equippedDuringReload: boolean) {
const slotsAdded = this.slots.add(item, count, equippedDuringReload, true, this.count);
this.count+=slotsAdded;
}
public addWhenReload(item: Item, count: number, equippedDuringReload: boolean) {
const slotsAdded = this.slots.add(item, count, equippedDuringReload, true, this.count);
this.count+=slotsAdded;
}
public addInGame(item: Item, count: number) {
const slotsAdded = this.slots.add(item, count, false, false, this.count);
this.count+=slotsAdded;
}
public addInGame(item: Item, count: number) {
const slotsAdded = this.slots.add(item, count, false, false, this.count);
this.count+=slotsAdded;
}
public remove(item: Item, count: number, slot: number) {
const slotsRemoved = this.slots.remove(item, count, slot);
this.count-=slotsRemoved;
}
public remove(item: Item, count: number, slot: number) {
const slotsRemoved = this.slots.remove(item, count, slot);
this.count-=slotsRemoved;
}
public equip(item: Item, slot: number) {
this.slots.equip(item, slot);
}
public equip(item: Item, slot: number) {
this.slots.equip(item, slot);
}
public unequip(item: Item, slot: number) {
this.slots.unequip(item, slot);
}
public unequip(item: Item, slot: number) {
this.slots.unequip(item, slot);
}
// Only clears first this.count
public clearForReload() {
if(this.count > 0){
this.slots.clearFirst(this.count);
this.count = 0;
}
}
// Only clears first this.count
public clearForReload() {
if(this.count > 0){
this.slots.clearFirst(this.count);
this.count = 0;
}
}
public updateEquipmentDurability(gameData: GameData) {
// find first weapon/bow/shield. this one searches entire inventory
let foundWeapon = false;
let foundBow = false;
let foundShield = false;
this.slots.getSlotsRef().forEach(({item, equipped}, i)=>{
if(equipped){
const type = itemToItemData(item).type;
if(type === ItemType.Weapon && !foundWeapon){
gameData.updateDurability(999, i);
foundWeapon = true;
}
if(type === ItemType.Bow && !foundBow){
gameData.updateDurability(999, i);
foundBow = true;
}
if(type === ItemType.Shield && !foundShield){
gameData.updateDurability(999, i);
foundShield = true;
}
}
})
}
public updateEquipmentDurability(gameData: GameData) {
// find first weapon/bow/shield. this one searches entire inventory
let foundWeapon = false;
let foundBow = false;
let foundShield = false;
this.slots.getSlotsRef().forEach(({item, equipped}, i)=>{
if(equipped){
const type = itemToItemData(item).type;
if(type === ItemType.Weapon && !foundWeapon){
gameData.updateDurability(999, i);
foundWeapon = true;
}
if(type === ItemType.Bow && !foundBow){
gameData.updateDurability(999, i);
foundBow = true;
}
if(type === ItemType.Shield && !foundShield){
gameData.updateDurability(999, i);
foundShield = true;
}
}
});
}
public shootArrow(count: number, gameData: GameData) {
const updatedSlot = this.slots.shootArrow(count);
if(updatedSlot>=0){
const durability = this.slots.getSlotsRef()[updatedSlot].count;
gameData.updateDurability(durability, updatedSlot);
}
}
public shootArrow(count: number, gameData: GameData) {
const updatedSlot = this.slots.shootArrow(count);
if(updatedSlot>=0){
const durability = this.slots.getSlotsRef()[updatedSlot].count;
gameData.updateDurability(durability, updatedSlot);
}
}
public getCount(): number {
return this.count;
}
public getCount(): number {
return this.count;
}
public modifyCount(delta: number): void {
this.count+=delta;
}
public modifyCount(delta: number): void {
this.count+=delta;
}
public resetCount(): void {
this.count = this.slots.length;
}
public resetCount(): void {
this.count = this.slots.length;
}
public clearForEventide(): void {
this.count-=this.slots.clearAllButKeyItems();
}
}

View file

@ -1,14 +1,14 @@
//https://medium.com/@fsufitch/is-javascript-array-sort-stable-46b90822543f
export const stableSort = <T>(array: T[], cmp: (a:T, b:T) => number): void => {
const stabilizedThis: [T, number][] = array.map((el, index) => [el, index]);
const stableCmp = (a: [T, number], b: [T, number]) => {
let order = cmp(a[0], b[0]);
if (order != 0) return order;
return a[1] - b[1];
}
stabilizedThis.sort(stableCmp);
const stabilizedThis: [T, number][] = array.map((el, index) => [el, index]);
const stableCmp = (a: [T, number], b: [T, number]) => {
const order = cmp(a[0], b[0]);
if (order != 0) {return order;}
return a[1] - b[1];
};
stabilizedThis.sort(stableCmp);
for (let i=0; i<array.length; i++) {
array[i] = stabilizedThis[i][0];
}
}
for (let i=0; i<array.length; i++) {
array[i] = stabilizedThis[i][0];
}
};

View file

@ -1,16 +1,14 @@
import clsx from "clsx";
import { ItemList, ItemListProps } from "components/ItemList";
import { DoubleItemSlot } from "components/ItemSlot";
import { TitledList } from "components/TitledList";
import { Command } from "core/Command";
import { ItemStack, itemToItemData } from "core/Item";
import { parseCommand } from "core/Parser";
import { SimulationState } from "core/SimulationState";
import { Slots } from "core/Slots";
import Background from "assets/Background.png";
import InGameBackground from "assets/InGame.png";
import React, { useEffect, useState } from "react";
import { ItemList } from "components/ItemList";
type DisplayPaneProps = {
command: string,
@ -20,23 +18,6 @@ type DisplayPaneProps = {
editCommand: (c: Command)=>void
}
// export const stacksToItemListProps = (slots: Slots, numBroken: number, isSave: boolean): ItemListProps => {
// return {
// items: stacksToItemProps(slots.getSlotsRef()),
// numBroken,
// isSave,
// };
// };
// export const stacksToItemProps = (stacks: ItemStack[]): ItemListItemProps[] => {
// return stacks.map(stackToItemProps);
// };
// export const stackToItemProps = ({item, count, equipped}: ItemStack): ItemListItemProps => {
// const data = itemToItemData(item);
// return {image: data.image, count: data.stackable ? count : 0, isEquipped:equipped};
// };
export const DisplayPane: React.FC<DisplayPaneProps> = ({command,editCommand,displayIndex,simulationState, overlaySave})=>{
const [commandString, setCommandString] = useState<string>("");
const [hasError, setHasError] = useState<boolean>(false);
@ -72,104 +53,96 @@ export const DisplayPane: React.FC<DisplayPaneProps> = ({command,editCommand,dis
outline: "none",
}}value={commandString}
placeholder="Type command here..."
spellCheck={false}
onChange={(e)=>{
const cmdString = e.target.value;
setCommandString(cmdString);
const parsedCommand = parseCommand(cmdString);
if(parsedCommand){
editCommand(parsedCommand);
setHasError(false);
}else{
setHasError(true);
}
editCommand(parsedCommand);
setHasError(cmdString!=="" &&!cmdString.startsWith("#") && !parsedCommand.isValid());
}}></input>
</div>
<div style={{
height: "calc( 100% - 40px )"
}}>
{overlaySave ?
<div style={{
borderTop: "1px solid black",
boxSizing: "border-box",
height: "100%",
overflowY: "auto",
background: `url(${InGameBackground})`,
backgroundPosition: "center",
backgroundSize: "auto 100%",
color: "white",
} }>
{overlaySave ?
<div style={{
borderTop: "1px solid black",
boxSizing: "border-box",
height: "100%",
overflowY: "auto",
background: `url(${InGameBackground})`,
backgroundPosition: "center",
backgroundSize: "auto 100%",
color: "white",
} }>
<TitledList title={`Game Data / Visible Inventory (Count=${simulationState.inventoryMCount})`}>
{
(()=>{
const doubleSlots: JSX.Element[] = [];
const gameDataSlots = simulationState.displayableGameData.getDisplayedSlots();
const inventorySlots = simulationState.displayablePouch.getDisplayedSlots();
console.log(inventorySlots);
for(let i=0;i<gameDataSlots.length && i<inventorySlots.length;i++){
doubleSlots.push(<DoubleItemSlot key={i}
first={{slot: gameDataSlots[i]}}
second={{slot: inventorySlots[i]}}
/>);
}
if(inventorySlots.length>gameDataSlots.length){
for(let i=inventorySlots.length;i<gameDataSlots.length;i++){
doubleSlots.push(<DoubleItemSlot key={i+inventorySlots.length}
{
(()=>{
const doubleSlots: JSX.Element[] = [];
const gameDataSlots = simulationState.displayableGameData.getDisplayedSlots();
const inventorySlots = simulationState.displayablePouch.getDisplayedSlots();
for(let i=0;i<gameDataSlots.length && i<inventorySlots.length;i++){
doubleSlots.push(<DoubleItemSlot key={i}
first={{slot: gameDataSlots[i]}}
/>);
}
}else if(inventorySlots.length > gameDataSlots.length){
for(let i=gameDataSlots.length;i<inventorySlots.length;i++){
doubleSlots.push(<DoubleItemSlot key={i + gameDataSlots.length}
second={{slot: inventorySlots[i]}}
/>);
}
}
return doubleSlots;
})()
}
if(gameDataSlots.length>inventorySlots.length){
for(let i=inventorySlots.length;i<gameDataSlots.length;i++){
doubleSlots.push(<DoubleItemSlot key={i+inventorySlots.length}
first={{slot: gameDataSlots[i]}}
/>);
}
}else if(inventorySlots.length > gameDataSlots.length){
for(let i=gameDataSlots.length;i<inventorySlots.length;i++){
doubleSlots.push(<DoubleItemSlot key={i + gameDataSlots.length}
second={{slot: inventorySlots[i]}}
/>);
}
}
return doubleSlots;
})()
}
</TitledList>
</div>
:<>
<div style={{
borderTop: "1px solid black",
background: `url(${Background})`,
color: "white",
borderBottom: "1px solid black",
boxSizing: "border-box",
height: "50%",
overflowY: "auto"
} }>
<TitledList title="Game Data">
<ItemList slots={simulationState.displayableGameData.getDisplayedSlots()}/>
</TitledList>
</div>
<div style={{
borderTop: "1px solid black",
background: `url(${InGameBackground})`,
backgroundPosition: "center",
backgroundSize: "100%",
boxSizing: "border-box",
height: "50%",
overflowY: "auto",
color: "white"
} }>
<TitledList title={`Visible Inventory (Count=${simulationState.inventoryMCount})`}>
<ItemList slots={simulationState.displayablePouch.getDisplayedSlots()}/>
</TitledList>
:<>
<div style={{
borderTop: "1px solid black",
background: `url(${Background})`,
color: "white",
borderBottom: "1px solid black",
boxSizing: "border-box",
height: "50%",
overflowY: "auto"
} }>
<TitledList title="Game Data">
<ItemList slots={simulationState.displayableGameData.getDisplayedSlots()}/>
</TitledList>
</div>
<div style={{
borderTop: "1px solid black",
background: `url(${InGameBackground})`,
backgroundPosition: "center",
backgroundSize: "100%",
boxSizing: "border-box",
height: "50%",
overflowY: "auto",
color: "white"
} }>
<TitledList title={`Visible Inventory (Count=${simulationState.inventoryMCount})`}>
<ItemList slots={simulationState.displayablePouch.getDisplayedSlots()}/>
</TitledList>
</div>
</>}
</div>
</>}
</div>
</div>;
};

106
src/surfaces/OptionPage.tsx Normal file
View file

@ -0,0 +1,106 @@
import { TitledList } from "components/TitledList";
import { saveAs } from "data/FileSaver";
import { useRef, useState } from "react";
type OptionPageProps = {
interlaceInventory: boolean,
setInterlaceInventory: (value: boolean)=>void,
commandText: string,
setCommandText: (value: string)=>void,
}
export const OptionPage: React.FC<OptionPageProps> = ({
interlaceInventory,
setInterlaceInventory,
commandText,
setCommandText
}) => {
const [currentText, setCurrentText] = useState<string>(commandText);
const [fileName, setFileName] = useState<string>("");
const uploadRef = useRef<HTMLInputElement>(null);
return (
<div className="OtherPage">
<input ref={uploadRef} id="Upload" type="File" hidden onChange={(e)=>{
const files = e.target.files;
if(files?.length && files[0]){
const file = files[0];
const fileName = file.name.endsWith(".txt") ? file.name.substring(0, file.name.length-4) : file.name;
setFileName(fileName);
file.text().then(text=>{
setCurrentText(text);
setCommandText(text);
});
}
}}/>
<TitledList title="Options">
<div className="OtherPageContent">
<h3 className="Reference">
Interlace Inventory with GameData
<button className="MainButton" onClick={()=>{
setInterlaceInventory(!interlaceInventory);
}}>
{interlaceInventory ? "ON" : "OFF"}
</button>
</h3>
<h4 className="Reference">
Toggle whether Visible Inventory should be displayed separetely from Game Data or interlaced.
</h4>
<h3 className="Reference">Import / Export</h3>
<h4 className="Reference">
You can also directly copy, paste, or edit the commands here
</h4>
<p className="Reference">
<button className="MainButton" onClick={()=>{
if(uploadRef.current){
uploadRef.current.click();
}
}}>
Import
</button>
<button className="MainButton" onClick={()=>{
saveAs(currentText, fileName+".txt" || "dupe.txt");
}}>
Export
</button>
<input
className="MainInput"
spellCheck={false}
value={fileName}
onChange={(e)=>{
setFileName(e.target.value);
}}
placeholder="File name"
/>
<textarea
className="MainInput"
spellCheck={false}
value={currentText}
onChange={(e)=>{
setCurrentText(e.target.value);
}}
/>
{
currentText !== commandText &&
<>
<button className="MainButton" onClick={()=>{
setCommandText(currentText);
}}>
Save
</button>
<span className="Example">Don't forget to save changes</span>
</>
}
</p>
</div>
</TitledList>
</div>
);
};

View file

@ -1,169 +1,228 @@
import { ItemList } from "components/ItemList";
import { TitledList } from "components/TitledList";
import { getAllItems } from "core/Item";
import React from "react";
export const ReferencePage: React.FC = React.memo(()=>{
return (
<div style={{height: "100%", width: "100%", color: "white"}}>
<TitledList title="Reference">
<div style={{padding: 10}}>
<h2>Commands</h2>
<h3 className="Reference">Initialize X item1 Y item2 Z item3 ...</h3>
<h4 className="Reference">Used for initializing inventory before simulation</h4>
<p className="Reference">
return (
<div className="OtherPage">
<TitledList title="Reference">
<div className="OtherPageContent">
<h2>Items</h2>
{
getAllItems().map((item, i)=><h4 key={i} className="Reference">{item}</h4>)
}
<h2>Commands</h2>
<p className="Reference">
This is a list of available commands. All commands and items are case-sensitive
</p>
<h3 className="Reference">Initialize X item1 Y item2 Z item3 ...</h3>
<h4 className="Reference">Used for initializing inventory before simulation</h4>
<p className="Reference">
Fully resets the inventory by clearing all items and set Count to 0, then forcefully write the item list to inventory.
This would reset any broken slot you already have, and any in-game checks that happen when adding items are disabled.
For example, the items will appear in the order you specify, not in the in-game tab order
</p>
<p className="Reference">
</p>
<p className="Reference">
If you specify count &gt; 1 for unstackable items like weapon or sheika slate, multiple of that item would be added.
Game Data will be synced with Visible Inventory after the reset
</p>
<p className="Reference">
</p>
<p className="Reference">
Note that this will not clear saves. You can use this command to initialize multiple saves
</p>
<p className="Reference Example">Example: Initialize 1 Apple 2 Axe 3 Slate 4 SpiritOrb</p>
</p>
<p className="Reference Example">Example: Initialize 1 Apple 2 Axe 3 Slate 4 SpiritOrb</p>
<h3 className="Reference">Save / Save As NAME</h3>
<h4 className="Reference">Simulates a hard save or auto save action</h4>
<p className="Reference">
<h3 className="Reference">Save</h3>
<h3 className="Reference2">Save As NAME</h3>
<h4 className="Reference">Simulates a hard save or auto save action</h4>
<p className="Reference">
Writes Game Data to the corresponding save slot. The auto saves are specified by NAME.
You can have as many auto saves as you want in the simulator.
</p>
</p>
<p className="Reference">
You cannot save on Eventide/ToTS. However, the simulator does not enforce that.
</p>
<p className="Reference Example">Example 1: Save</p>
<p className="Reference Example">Example 2: Save As MySave</p>
<p className="Reference">
<p className="Reference Example">Example 1: Save</p>
<p className="Reference Example">Example 2: Save As MySave</p>
<p className="Reference">
Example 1 will save to the manual save slot, while example 2 will save to the slot named "MySave".
There cannot be spaces in the name. If "MySave" doesn't exist, a new slot is created
</p>
</p>
<h3 className="Reference">Reload (NAME)</h3>
<h4 className="Reference">Simulates reloading a save</h4>
<p className="Reference">
<h3 className="Reference">Reload (NAME)</h3>
<h4 className="Reference">Simulates reloading a save</h4>
<p className="Reference">
First, reads Game Data from the corresponding save slot.
If NAME is not given, the manual save is used unless "Use" commands are used before this (see below).
If NAME is given, the corresponding save slot with that name is used
</p>
<p className="Reference">
</p>
<p className="Reference">
After that, the first Count items in the visible inventory is removed, and Count is decreased accordingly.
Then, each item slot in the Game Data is added to the inventory.
</p>
</p>
<p className="Reference Example">Example 1: Reload</p>
<p className="Reference Example">Example 2: Reload MySave</p>
<p className="Reference Example">Example 1: Reload</p>
<p className="Reference Example">Example 2: Reload MySave</p>
<h3 className="Reference">Use NAME</h3>
<h4 className="Reference">(Deprecated) Specify which save to load on the subsequent reload</h4>
<p className="Reference Example">
<h3 className="Reference">Use NAME</h3>
<h4 className="Reference">(Deprecated) Specify which save to load on the subsequent reload</h4>
<p className="Reference Example">
This command is only for backward compatibility. Use "Reload" instead
</p>
<p className="Reference">
</p>
<p className="Reference">
Specify the save named NAME to be reloaded on the next "Reload" command
</p>
</p>
<p className="Reference Example">Example: Use MySave</p>
<p className="Reference Example">Example: Use MySave</p>
<h3 className="Reference">Break X Slots</h3>
<h4 className="Reference">Simulate making X broken slots with hold smuggle glitch</h4>
<p className="Reference">
<h3 className="Reference">Break X Slots</h3>
<h4 className="Reference">Simulate making X broken slots with hold smuggle glitch</h4>
<p className="Reference">
Decrease inventory Count by X
</p>
<p className="Reference">
</p>
<p className="Reference">
This command does not automatically simulate the hold smuggle and sell process.
It just changes count (i.e. make broken slots) with magic.
</p>
</p>
<p className="Reference Example">Example: Break 4 Slots</p>
<p className="Reference Example">Example: Break 4 Slots</p>
<h3 className="Reference">Get/Add/Cook/Pickup item</h3>
<h3 className="Reference2">Get/Add/Cook/Pickup X item</h3>
<h3 className="Reference2">Get/Add/Cook/Pickup X item1 Y item2 Z item3 ...</h3>
<h4 className="Reference">Simulate obtaining items in game</h4>
<p className="Reference">
<h3 className="Reference">Get/Add/Cook/Pickup/Buy item</h3>
<h3 className="Reference2">Get/Add/Cook/Pickup/Buy X item</h3>
<h3 className="Reference2">Get/Add/Cook/Pickup/Buy X item1 Y item2 Z item3 ...</h3>
<h4 className="Reference">Simulate obtaining items in game</h4>
<p className="Reference">
Add the item(s) to visible inventory. Sync with Game Data unless you are on Eventide or inside TOTS
</p>
<p className="Reference">
</p>
<p className="Reference">
Like in game, you won't be able to obtain multiple unstackable key items, or multiple master sword in this way.
If a stackable item is at 999 or more when you invoke this command, the count is set to 999 (not fully accurate since you won't be able to pick up more items in game).
</p>
<p className="Reference">
</p>
<p className="Reference">
If you specify a count for unstackable items, they are added in different slots as if you pick them up in game, one after another.
</p>
<p className="Reference">
</p>
<p className="Reference">
Note that you must not enter plural forms for the item name.
</p>
</p>
<p className="Reference Example">Example 1: Add Apple</p>
<p className="Reference Example">Example 2: Get 10 Apple</p>
<p className="Reference Example">Example 3: Pickup 10 Apple 5 Diamond 1 Slate 5 MasterSword</p>
<p className="Reference Example">Example 1: Add Apple</p>
<p className="Reference Example">Example 2: Get 10 Apple</p>
<p className="Reference Example">Example 3: Pickup 10 Apple 5 Diamond 1 Slate 5 MasterSword</p>
<h3 className="Reference">Remove/Sell/Eat/Drop item</h3>
<h3 className="Reference2">Remove/Sell/Eat/Drop X item</h3>
<h3 className="Reference2">Remove/Sell/Eat/Drop item From Slot Y</h3>
<h3 className="Reference2">Remove/Sell/Eat/Drop X item From Slot Y</h3>
<h3 className="Reference2">Remove/Sell/Eat/Drop X item1 Y item2 Z item3 ...</h3>
<h4 className="Reference">Simulate removing items in game</h4>
<p className="Reference">
<h3 className="Reference">With/Remove/Sell/Eat/Drop item</h3>
<h3 className="Reference2">With/Remove/Sell/Eat/Drop X item</h3>
<h3 className="Reference2">With/Remove/Sell/Eat/Drop item From Slot Y</h3>
<h3 className="Reference2">With/Remove/Sell/Eat/Drop X item From Slot Y</h3>
<h3 className="Reference2">With/Remove/Sell/Eat/Drop X item1 Y item2 Z item3 ...</h3>
<h4 className="Reference">Simulate removing items in game</h4>
<p className="Reference">
Remove the item(s) to visible inventory. Sync with Game Data unless you are on Eventide or inside TOTS
</p>
<p className="Reference">
</p>
<p className="Reference">
When number of item is not specified, it defaults to 1. Up to X items will be removed from inventory, even when they span multiple slots.
If X &gt; total number of items in inventory, all of them will be removed.
</p>
<p className="Reference">
</p>
<p className="Reference">
When slot is specified, it starts removing from slot X (slot 1 is the leftmost slot with that item, slot 2 is the second leftmost slot with that item).
</p>
<p className="Reference">
</p>
<p className="Reference">
Note that you must not enter plural forms for the item name.
</p>
</p>
<p className="Reference Example">Example 1: Remove Apple</p>
<p className="Reference Example">Example 2: Drop 10 Diamond</p>
<p className="Reference Example">Example 3: Sell 10 Apple 5 Diamond</p>
<p className="Reference Example">Example 4: Sell 5 Apple From Slot 3</p>
<p className="Reference Example">Example 1: Remove Apple</p>
<p className="Reference Example">Example 2: Drop 10 Diamond</p>
<p className="Reference Example">Example 3: Sell 10 Apple 5 Diamond</p>
<p className="Reference Example">Example 4: Sell 5 Apple From Slot 3</p>
<h3 className="Reference">D&amp;P X item</h3>
<h4 className="Reference">Shortcut for drop and pick up, for sorting inventory</h4>
<p className="Reference">
This command drops X item from the first slot, then pick them up
</p>
<p className="Reference Example">Example: D&amp;P 5 Diamond</p>
<h3 className="Reference">D&amp;P X item1 Y item2 Z item3 ...</h3>
<h4 className="Reference">Shortcut for drop and pick up, for sorting inventory</h4>
<p className="Reference">
This command drops and pick up each item stack in the specified order.
You can also repeat items if you are combining more than 2 slots.
</p>
<p className="Reference">
You can only drop from slot 1 with this shortcut.
</p>
<p className="Reference Example">Example 1: D&amp;P 5 Diamond</p>
<p className="Reference Example">Example 2: D&amp;P 20 Shaft 5 Diamond</p>
<p className="Reference Example">Example 3: D&amp;P 5 Diamond 10 Diamond</p>
<h3 className="Reference">Equip item</h3>
<h3 className="Reference2">Equip item In Slot X</h3>
<h4 className="Reference">Simulates equipping something</h4>
<p className="Reference">
<h3 className="Reference">Equip item</h3>
<h3 className="Reference2">Equip item In Slot X</h3>
<h4 className="Reference">Simulates equipping something</h4>
<p className="Reference">
When equipping an item, all other item of the same type in the first tab is unequipped, then the item selected is equipped.
</p>
<p className="Reference">
</p>
<p className="Reference">
Slot can be used if you have multiple of the same item. When slot is not specified, the leftmost item will be equipped.
Note that you can use this command to equip something that is already equipped, which is not possible in game.
You can also equip unequippable items like materials, but it is not meaningful
</p>
<p className="Reference Example">Example 1: Equip Weapon</p>
<p className="Reference Example">Example 2: Equip Weapon In Slot 3</p>
</p>
<p className="Reference Example">Example 1: Equip Weapon</p>
<p className="Reference Example">Example 2: Equip Weapon In Slot 3</p>
<h3 className="Reference">Unequip item</h3>
<h3 className="Reference2">Unequip item In Slot X</h3>
<h4 className="Reference">Simulates unequipping something</h4>
<p className="Reference">
<h3 className="Reference">Unequip item</h3>
<h3 className="Reference2">Unequip item In Slot X</h3>
<h4 className="Reference">Simulates unequipping something</h4>
<p className="Reference">
When unequipping an item, only the selected item is unequipped.
</p>
<p className="Reference">
</p>
<p className="Reference">
Slot can be used if you have multiple of the same item. When slot is not specified, the leftmost equipped item will be unequipped.
Note that you can use this command to unequip something that is already unequipped, which is useless.
You cannot unequip arrows.
</p>
<p className="Reference Example">Example 1: Unequip Shield</p>
<p className="Reference Example">Example 2: Unequip Shield In Slot 5</p>
</div>
</p>
<p className="Reference Example">Example 1: Unequip Shield</p>
<p className="Reference Example">Example 2: Unequip Shield In Slot 5</p>
<h3 className="Reference">Close Game</h3>
<h4 className="Reference">Simulates closing the game and restarting</h4>
<p className="Reference">
When closing the game, Visible Inventory and Game Data are erased
</p>
<p className="Reference Example">Example: Close Game</p>
<h3 className="Reference">Sync GameData</h3>
<h4 className="Reference">Copy Visible Inventory to Game Data</h4>
<p className="Reference">
Usually done in game by opening and closing inventory.
</p>
<p className="Reference Example">Example: Sync GameData</p>
<h3 className="Reference">Sort Key/Material</h3>
<h4 className="Reference">Simulates press Y to sort tab</h4>
<p className="Reference Example">
This command is currently broken
</p>
<h3 className="Reference">Shoot X Arrow</h3>
<h4 className="Reference">Simulates shooting arrow without opening inventory</h4>
<p className="Reference">
When reloading a save with desynced game data, the equipped weapon/bow/shield are automatically corrupted, but not the arrows.
To corrupt the equipped arrow slot, you need to shoot an arrow.
</p>
<p className="Reference">
This command does not let you select which arrow to shoot.
When you reload a save, Link should have the last equipped arrow slot equipped in the overworld.
<span className="Example">[needs confirmation]</span>
</p>
<p className="Reference Example">Example: Shoot 1 Arrow</p>
<h3 className="Reference">Enter/Exit Eventide</h3>
<h4 className="Reference">Simulates entering/exiting Eventide or Trial of the Sword</h4>
<p className="Reference">
When entering Eventide or TotS, the entire inventory is cleared except for key items regardless of inventory count.
While the challenge is active, none of the inventory changes are synced to game data.
</p>
<p className="Reference">
When exiting the challenge, the game reloads the game data as if reloading a save
</p>
<p className="Reference Example">Example: Enter Eventide</p>
</div>
</TitledList>
</div>
)
</TitledList>
</div>
);
});