Skip to content

Commit b813e13

Browse files
Unit Tests & Documentation (#1095)
2 parents ca0f6d0 + 21f318d commit b813e13

19 files changed

+445
-132
lines changed

fission/package.json

+3
Original file line numberDiff line numberDiff line change
@@ -46,6 +46,9 @@
4646
"@emotion/react": "^11.11.3",
4747
"@emotion/styled": "^11.11.0",
4848
"@mui/material": "^5.15.6",
49+
"@testing-library/dom": "^10.4.0",
50+
"@testing-library/react": "^16.0.0",
51+
"@testing-library/user-event": "^14.5.2",
4952
"@types/node": "^20.4.4",
5053
"@types/pako": "^2.0.3",
5154
"@types/react": "^18.2.47",

fission/src/Synthesis.tsx

-2
Original file line numberDiff line numberDiff line change
@@ -60,7 +60,6 @@ import NewInputSchemeModal from "./ui/modals/configuring/theme-editor/NewInputSc
6060
import AssignNewSchemeModal from "./ui/modals/configuring/theme-editor/AssignNewSchemeModal.tsx"
6161
import AnalyticsConsent from "./ui/components/AnalyticsConsent.tsx"
6262
import PreferencesSystem from "./systems/preferences/PreferencesSystem.ts"
63-
import ResetAllInputsModal from "./ui/modals/configuring/inputs/ResetAllInputsModal.tsx"
6463
import APSManagementModal from "./ui/modals/APSManagementModal.tsx"
6564
import ConfigurePanel from "./ui/panels/configuring/assembly-config/ConfigurePanel.tsx"
6665

@@ -217,7 +216,6 @@ const initialModals = [
217216
<MatchModeModal key="match-mode" modalId="match-mode" />,
218217
<ConfigMotorModal key="config-motor" modalId="config-motor" />,
219218
<ImportLocalMirabufModal key="import-local-mirabuf" modalId="import-local-mirabuf" />,
220-
<ResetAllInputsModal key="reset-inputs" modalId="reset-inputs" />,
221219
<APSManagementModal key="aps-management" modalId="aps-management" />,
222220
]
223221

fission/src/systems/input/DefaultInputs.ts

+5-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
import { InputScheme } from "./InputSchemeManager"
22
import { AxisInput, ButtonInput, EmptyModifierState } from "./InputSystem"
33

4+
/** The purpose of this class is to store any defaults related to the input system. */
45
class DefaultInputs {
56
static ernie = () => {
67
return {
@@ -133,6 +134,7 @@ class DefaultInputs {
133134
}
134135
}
135136

137+
/** We like this guy */
136138
public static hunter = () => {
137139
return {
138140
schemeName: "Hunter",
@@ -187,7 +189,8 @@ class DefaultInputs {
187189
}
188190
}
189191

190-
public static get defaultInputCopies() {
192+
/** @returns {InputScheme[]} New copies of the default input schemes without reference to any others. */
193+
public static get defaultInputCopies(): InputScheme[] {
191194
return [
192195
DefaultInputs.ernie(),
193196
DefaultInputs.luna(),
@@ -197,6 +200,7 @@ class DefaultInputs {
197200
]
198201
}
199202

203+
/** @returns {InputScheme} A new blank input scheme with no control bound. */
200204
public static get newBlankScheme(): InputScheme {
201205
return {
202206
schemeName: "",

fission/src/systems/input/InputSchemeManager.ts

-14
Original file line numberDiff line numberDiff line change
@@ -129,20 +129,6 @@ class InputSchemeManager {
129129
PreferencesSystem.setGlobalPreference("InputSchemes", customizedSchemes)
130130
PreferencesSystem.savePreferences()
131131
}
132-
133-
/** Returns a copy of a scheme without references to the original in any way */
134-
public static copyScheme(scheme: InputScheme): InputScheme {
135-
const copiedInputs: Input[] = []
136-
scheme.inputs.forEach(i => copiedInputs.push(i.getCopy()))
137-
138-
return {
139-
schemeName: scheme.schemeName,
140-
descriptiveName: scheme.descriptiveName,
141-
customized: scheme.customized,
142-
usesGamepad: scheme.usesGamepad,
143-
inputs: copiedInputs,
144-
}
145-
}
146132
}
147133

148134
export default InputSchemeManager

fission/src/systems/input/InputSystem.ts

+76-44
Original file line numberDiff line numberDiff line change
@@ -9,36 +9,45 @@ export type ModifierState = {
99
}
1010
export const EmptyModifierState: ModifierState = { ctrl: false, alt: false, shift: false, meta: false }
1111

12-
// Represents any input
12+
/** Represents any user input */
1313
abstract class Input {
1414
public inputName: string
1515

16+
/** @param {string} inputName - The name given to this input to identify it's function. */
1617
constructor(inputName: string) {
1718
this.inputName = inputName
1819
}
1920

20-
// Returns the current value of the input. Range depends on input type
21+
/** @returns {number} a number between -1 and 1 for this input. */
2122
abstract getValue(useGamepad: boolean): number
22-
23-
// Creates a copy to avoid modifying the default inputs by reference
24-
abstract getCopy(): Input
2523
}
2624

27-
// A single button
25+
/** Represents any user input that is a single true/false button. */
2826
class ButtonInput extends Input {
2927
public keyCode: string
3028
public keyModifiers: ModifierState
3129

3230
public gamepadButton: number
3331

32+
/**
33+
* All optional params will remain unassigned if not value is given. This can be assigned later by the user through the configuration panel.
34+
*
35+
* @param {string} inputName - The name given to this input to identify it's function.
36+
* @param {string} [keyCode] - The keyboard button for this input if a gamepad is not used.
37+
* @param {number} [gamepadButton] - The gamepad button for this input if a gamepad is used.
38+
* @param {ModifierState} [keyModifiers] - The key modifier state for the keyboard input.
39+
*/
3440
public constructor(inputName: string, keyCode?: string, gamepadButton?: number, keyModifiers?: ModifierState) {
3541
super(inputName)
3642
this.keyCode = keyCode ?? ""
3743
this.keyModifiers = keyModifiers ?? EmptyModifierState
3844
this.gamepadButton = gamepadButton ?? -1
3945
}
4046

41-
// Returns 1 if pressed and 0 if not pressed
47+
/**
48+
* @param useGamepad Looks at the gamepad if true and the keyboard if false.
49+
* @returns 1 if pressed, 0 if not pressed or not found.
50+
*/
4251
getValue(useGamepad: boolean): number {
4352
// Gamepad button input
4453
if (useGamepad) {
@@ -48,13 +57,9 @@ class ButtonInput extends Input {
4857
// Keyboard button input
4958
return InputSystem.isKeyPressed(this.keyCode, this.keyModifiers) ? 1 : 0
5059
}
51-
52-
getCopy(): Input {
53-
return new ButtonInput(this.inputName, this.keyCode, this.gamepadButton, this.keyModifiers)
54-
}
5560
}
5661

57-
// An axis between two buttons (-1 to 1)
62+
/** Represents any user input that is an axis between -1 and 1. Can be a gamepad axis, two gamepad buttons, or two keyboard buttons. */
5863
class AxisInput extends Input {
5964
public posKeyCode: string
6065
public posKeyModifiers: ModifierState
@@ -67,6 +72,20 @@ class AxisInput extends Input {
6772
public posGamepadButton: number
6873
public negGamepadButton: number
6974

75+
/**
76+
* All optional params will remain unassigned if not value is given. This can be assigned later by the user through the configuration panel.
77+
*
78+
* @param {string} inputName - The name given to this input to identify it's function.
79+
* @param {string} [posKeyCode] - The keyboard input that corresponds to a positive input value (1).
80+
* @param {string} [negKeyCode] - The keyboard input that corresponds to a negative input value (-1).
81+
* @param {number} [gamepadAxisNumber] - The gamepad axis that this input looks at if the scheme is set to use a gamepad.
82+
* @param {boolean} [joystickInverted] - Inverts the input if a gamepad axis is used.
83+
* @param {boolean} [useGamepadButtons] - If this is true and the scheme is set to use a gamepad, this axis will be between two buttons on the controller.
84+
* @param {number} [posGamepadButton] - The gamepad button that corresponds to a positive input value (1).
85+
* @param {number} [negGamepadButton] - The gamepad button that corresponds to a negative input value (-1).
86+
* @param {ModifierState} [posKeyModifiers] - The key modifier state for the positive keyboard input.
87+
* @param {ModifierState} [negKeyModifiers] - The key modifier state for the negative keyboard input.
88+
*/
7089
public constructor(
7190
inputName: string,
7291
posKeyCode?: string,
@@ -94,11 +113,14 @@ class AxisInput extends Input {
94113
this.negGamepadButton = negGamepadButton ?? -1
95114
}
96115

97-
// For keyboard: returns 1 if positive pressed, -1 if negative pressed, or 0 if none or both are pressed
98-
// For gamepad axis: returns a range between -1 and 1 with a deadband in the middle
116+
/**
117+
* @param useGamepad Looks at the gamepad if true and the keyboard if false.
118+
* @returns {number} KEYBOARD: 1 if positive pressed, -1 if negative pressed, or 0 if none or both are pressed.
119+
* @returns {number} GAMEPAD: a number between -1 and 1 with a deadband in the middle.
120+
*/
99121
getValue(useGamepad: boolean): number {
100-
// Gamepad joystick axis
101122
if (useGamepad) {
123+
// Gamepad joystick axis
102124
if (!this.useGamepadButtons)
103125
return InputSystem.getGamepadAxis(this.gamepadAxisNumber) * (this.joystickInverted ? -1 : 1)
104126

@@ -115,38 +137,28 @@ class AxisInput extends Input {
115137
(InputSystem.isKeyPressed(this.negKeyCode, this.negKeyModifiers) ? 1 : 0)
116138
)
117139
}
118-
119-
getCopy(): Input {
120-
return new AxisInput(
121-
this.inputName,
122-
this.posKeyCode,
123-
this.negKeyCode,
124-
this.gamepadAxisNumber,
125-
this.joystickInverted,
126-
this.useGamepadButtons,
127-
this.posGamepadButton,
128-
this.negGamepadButton,
129-
this.posKeyModifiers,
130-
this.negKeyModifiers
131-
)
132-
}
133140
}
134141

142+
/**
143+
* The input system listens for and records key presses and joystick positions to be used by robots.
144+
* It also maps robot behaviors (such as an arcade drivetrain or an arm) to specific keys through customizable input schemes.
145+
*/
135146
class InputSystem extends WorldSystem {
136147
public static currentModifierState: ModifierState
137148

138-
// A list of keys currently being pressed
149+
/** The keys currently being pressed. */
139150
private static _keysPressed: { [key: string]: boolean } = {}
140151

141152
private static _gpIndex: number | null
142153
public static gamepad: Gamepad | null
143154

144-
// Maps a brain index to a certain input scheme
155+
/** Maps a brain index to an input scheme. */
145156
public static brainIndexSchemeMap: Map<number, InputScheme> = new Map()
146157

147158
constructor() {
148159
super()
149160

161+
// Initialize input events
150162
this.handleKeyDown = this.handleKeyDown.bind(this)
151163
document.addEventListener("keydown", this.handleKeyDown)
152164

@@ -159,7 +171,7 @@ class InputSystem extends WorldSystem {
159171
this.gamepadDisconnected = this.gamepadDisconnected.bind(this)
160172
window.addEventListener("gamepaddisconnected", this.gamepadDisconnected)
161173

162-
// Detect when the user leaves the page to clear inputs
174+
// Initialize an event that's triggered when the user exits/enters the page
163175
document.addEventListener("visibilitychange", () => {
164176
if (document.hidden) this.clearKeyData()
165177
})
@@ -177,13 +189,13 @@ class InputSystem extends WorldSystem {
177189
}
178190

179191
public Update(_: number): void {
180-
InputSystem
181192
// Fetch current gamepad information
182193
if (InputSystem._gpIndex == null) InputSystem.gamepad = null
183194
else InputSystem.gamepad = navigator.getGamepads()[InputSystem._gpIndex]
184195

185196
if (!document.hasFocus()) this.clearKeyData()
186197

198+
// Update the current modifier state to be checked against target stats when getting input values
187199
InputSystem.currentModifierState = {
188200
ctrl: InputSystem.isKeyPressed("ControlLeft") || InputSystem.isKeyPressed("ControlRight"),
189201
alt: InputSystem.isKeyPressed("AltLeft") || InputSystem.isKeyPressed("AltRight"),
@@ -199,21 +211,22 @@ class InputSystem extends WorldSystem {
199211
window.removeEventListener("gamepaddisconnected", this.gamepadDisconnected)
200212
}
201213

202-
// Called when any key is first pressed
214+
/** Called when any key is first pressed */
203215
private handleKeyDown(event: KeyboardEvent) {
204216
InputSystem._keysPressed[event.code] = true
205217
}
206218

207-
// Called when any key is released
219+
/* Called when any key is released */
208220
private handleKeyUp(event: KeyboardEvent) {
209221
InputSystem._keysPressed[event.code] = false
210222
}
211223

224+
/** Clears all stored key data when the user leaves the page. */
212225
private clearKeyData() {
213226
for (const keyCode in InputSystem._keysPressed) delete InputSystem._keysPressed[keyCode]
214227
}
215228

216-
// Called once when a gamepad is first connected
229+
/* Called once when a gamepad is first connected */
217230
private gamepadConnected(event: GamepadEvent) {
218231
console.log(
219232
"Gamepad connected at index %d: %s. %d buttons, %d axes.",
@@ -226,22 +239,30 @@ class InputSystem extends WorldSystem {
226239
InputSystem._gpIndex = event.gamepad.index
227240
}
228241

229-
// Called once when a gamepad is first disconnected
242+
/* Called once when a gamepad is first disconnected */
230243
private gamepadDisconnected(event: GamepadEvent) {
231244
console.log("Gamepad disconnected from index %d: %s", event.gamepad.index, event.gamepad.id)
232245

233246
InputSystem._gpIndex = null
234247
}
235248

236-
// Returns true if the given key is currently down
249+
/**
250+
* @param {string} key - The keycode of the target key.
251+
* @param {ModifierState} modifiers - The target modifier state. Assumed to be no modifiers if undefined.
252+
* @returns {boolean} True if the key is pressed or false otherwise.
253+
*/
237254
public static isKeyPressed(key: string, modifiers?: ModifierState): boolean {
238255
if (modifiers != null && !InputSystem.compareModifiers(InputSystem.currentModifierState, modifiers))
239256
return false
240257

241258
return !!InputSystem._keysPressed[key]
242259
}
243260

244-
// If an input exists, return it's value
261+
/**
262+
* @param {string} inputName The name of the function of the input.
263+
* @param {number} brainIndex The robot brain index for this input. Used to map to a control scheme.
264+
* @returns {number} A number between -1 and 1 based on the current state of the input.
265+
*/
245266
public static getInput(inputName: string, brainIndex: number): number {
246267
const targetScheme = InputSystem.brainIndexSchemeMap.get(brainIndex)
247268

@@ -252,8 +273,12 @@ class InputSystem extends WorldSystem {
252273
return targetInput.getValue(targetScheme.usesGamepad)
253274
}
254275

255-
// Returns true if two modifier states are identical
256-
private static compareModifiers(state1: ModifierState, state2: ModifierState): boolean {
276+
/**
277+
* @param {ModifierState} state1 Any key modifier state.
278+
* @param {ModifierState} state2 Any key modifier state.
279+
* @returns {boolean} True if the modifier states are identical and false otherwise.
280+
*/
281+
public static compareModifiers(state1: ModifierState, state2: ModifierState): boolean {
257282
if (!state1 || !state2) return false
258283

259284
return (
@@ -264,7 +289,10 @@ class InputSystem extends WorldSystem {
264289
)
265290
}
266291

267-
// Returns a number between -1 and 1 with a deadband
292+
/**
293+
* @param {number} axisNumber The joystick axis index. Must be an integer.
294+
* @returns {number} A number between -1 and 1 based on the position of this axis or 0 if no gamepad is connected or the axis is not found.
295+
*/
268296
public static getGamepadAxis(axisNumber: number): number {
269297
if (InputSystem.gamepad == null) return 0
270298

@@ -276,7 +304,11 @@ class InputSystem extends WorldSystem {
276304
return Math.abs(value) < 0.15 ? 0 : value
277305
}
278306

279-
// Returns true if a gamepad is connected and a certain button is pressed
307+
/**
308+
*
309+
* @param {number} buttonNumber - The gamepad button index. Must be an integer.
310+
* @returns {boolean} True if the button is pressed, false if not, a gamepad isn't connected, or the button can't be found.
311+
*/
280312
public static isGamepadButtonPressed(buttonNumber: number): boolean {
281313
if (InputSystem.gamepad == null) return false
282314

0 commit comments

Comments
 (0)