2023-04-17 22:57:24 +01:00
|
|
|
import { FC, useState } from "react";
|
|
|
|
import styled from "styled-components";
|
|
|
|
|
|
|
|
const DefaultBindings = (): KeyBindings => {
|
|
|
|
return {
|
2023-07-05 10:06:06 +01:00
|
|
|
A: "Z",
|
|
|
|
B: "X",
|
|
|
|
L: "A",
|
|
|
|
R: "S",
|
|
|
|
Start: "ENTER",
|
|
|
|
Select: "SHIFT",
|
|
|
|
Up: "UP",
|
|
|
|
Down: "DOWN",
|
|
|
|
Left: "LEFT",
|
|
|
|
Right: "RIGHT",
|
2023-04-17 22:57:24 +01:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
export const DefaultBindingsSet = (): Bindings => ({
|
|
|
|
Actual: DefaultBindings(),
|
|
|
|
Displayed: DefaultBindings(),
|
|
|
|
});
|
|
|
|
|
|
|
|
export enum GbaKey {
|
|
|
|
A = "A",
|
|
|
|
B = "B",
|
|
|
|
L = "L",
|
|
|
|
R = "R",
|
|
|
|
Up = "Up",
|
|
|
|
Down = "Down",
|
|
|
|
Left = "Left",
|
|
|
|
Right = "Right",
|
|
|
|
Start = "Start",
|
|
|
|
Select = "Select",
|
|
|
|
}
|
|
|
|
|
|
|
|
const BindingsOrder = [
|
|
|
|
GbaKey.A,
|
|
|
|
GbaKey.B,
|
|
|
|
GbaKey.L,
|
|
|
|
GbaKey.R,
|
|
|
|
GbaKey.Up,
|
|
|
|
GbaKey.Down,
|
|
|
|
GbaKey.Left,
|
|
|
|
GbaKey.Right,
|
|
|
|
GbaKey.Start,
|
|
|
|
GbaKey.Select,
|
|
|
|
];
|
|
|
|
|
|
|
|
interface SelectButtonProps {
|
|
|
|
selected: boolean;
|
|
|
|
}
|
|
|
|
|
|
|
|
const SelectButton = styled.button<SelectButtonProps>`
|
|
|
|
grid-column: 1;
|
|
|
|
`;
|
|
|
|
|
|
|
|
const ButtonWrapper = styled.div`
|
|
|
|
display: grid;
|
|
|
|
`;
|
|
|
|
|
|
|
|
export type KeyBindings = {
|
|
|
|
[key in GbaKey]: string;
|
|
|
|
};
|
|
|
|
|
2023-07-05 11:23:59 +01:00
|
|
|
export interface Bindings {
|
2023-04-17 22:57:24 +01:00
|
|
|
Displayed: KeyBindings;
|
|
|
|
Actual: KeyBindings;
|
|
|
|
}
|
|
|
|
|
|
|
|
const toHumanName = (keyName: string) => {
|
|
|
|
return keyName.replace("Arrow", "");
|
|
|
|
};
|
|
|
|
|
|
|
|
export const BindingsControl: FC<{
|
|
|
|
bindings: Bindings;
|
|
|
|
setBindings: (a: Bindings) => void;
|
2023-04-25 21:45:41 +01:00
|
|
|
setPaused: (pause: boolean) => void;
|
|
|
|
}> = ({ bindings, setBindings, setPaused }) => {
|
2023-04-17 22:57:24 +01:00
|
|
|
const [buttonToChange, setButtonToChange] = useState<GbaKey | null>(null);
|
|
|
|
|
2023-04-25 21:30:55 +01:00
|
|
|
const setKey = (key: string) => {
|
2023-04-17 22:57:24 +01:00
|
|
|
if (buttonToChange === null) return;
|
|
|
|
|
|
|
|
const nextBindings = {
|
|
|
|
Displayed: { ...bindings.Displayed },
|
|
|
|
Actual: { ...bindings.Actual },
|
|
|
|
};
|
|
|
|
|
|
|
|
nextBindings.Displayed[buttonToChange] = toHumanName(key).toUpperCase();
|
2023-04-25 21:30:55 +01:00
|
|
|
nextBindings.Actual[buttonToChange] = key;
|
2023-04-17 22:57:24 +01:00
|
|
|
|
|
|
|
setButtonToChange(null);
|
|
|
|
setBindings(nextBindings);
|
2023-04-25 21:45:41 +01:00
|
|
|
setPaused(false);
|
|
|
|
};
|
|
|
|
|
|
|
|
const onSelectButtonClick = (key: GbaKey) => {
|
|
|
|
setPaused(true);
|
|
|
|
setButtonToChange(key);
|
2023-04-17 22:57:24 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
return (
|
2023-07-04 23:08:16 +01:00
|
|
|
<ButtonWrapper onKeyUp={(evt: React.KeyboardEvent) => setKey(evt.key)}>
|
2023-04-17 22:57:24 +01:00
|
|
|
{BindingsOrder.map((x) => (
|
|
|
|
<SelectButton
|
2023-04-25 21:45:41 +01:00
|
|
|
onClick={() => onSelectButtonClick(x)}
|
2023-04-17 22:57:24 +01:00
|
|
|
key={x}
|
|
|
|
selected={buttonToChange === x}
|
|
|
|
>
|
|
|
|
{buttonToChange === x
|
|
|
|
? `Change ${x}`
|
|
|
|
: `${x}: ${bindings.Displayed[x]}`}
|
|
|
|
</SelectButton>
|
|
|
|
))}
|
|
|
|
</ButtonWrapper>
|
|
|
|
);
|
|
|
|
};
|