@@ -9,36 +9,45 @@ export type ModifierState = {
9
9
}
10
10
export const EmptyModifierState : ModifierState = { ctrl : false , alt : false , shift : false , meta : false }
11
11
12
- // Represents any input
12
+ /** Represents any user input */
13
13
abstract class Input {
14
14
public inputName : string
15
15
16
+ /** @param {string } inputName - The name given to this input to identify it's function. */
16
17
constructor ( inputName : string ) {
17
18
this . inputName = inputName
18
19
}
19
20
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. */
21
22
abstract getValue ( useGamepad : boolean ) : number
22
-
23
- // Creates a copy to avoid modifying the default inputs by reference
24
- abstract getCopy ( ) : Input
25
23
}
26
24
27
- // A single button
25
+ /** Represents any user input that is a single true/false button. */
28
26
class ButtonInput extends Input {
29
27
public keyCode : string
30
28
public keyModifiers : ModifierState
31
29
32
30
public gamepadButton : number
33
31
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
+ */
34
40
public constructor ( inputName : string , keyCode ?: string , gamepadButton ?: number , keyModifiers ?: ModifierState ) {
35
41
super ( inputName )
36
42
this . keyCode = keyCode ?? ""
37
43
this . keyModifiers = keyModifiers ?? EmptyModifierState
38
44
this . gamepadButton = gamepadButton ?? - 1
39
45
}
40
46
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
+ */
42
51
getValue ( useGamepad : boolean ) : number {
43
52
// Gamepad button input
44
53
if ( useGamepad ) {
@@ -48,13 +57,9 @@ class ButtonInput extends Input {
48
57
// Keyboard button input
49
58
return InputSystem . isKeyPressed ( this . keyCode , this . keyModifiers ) ? 1 : 0
50
59
}
51
-
52
- getCopy ( ) : Input {
53
- return new ButtonInput ( this . inputName , this . keyCode , this . gamepadButton , this . keyModifiers )
54
- }
55
60
}
56
61
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. */
58
63
class AxisInput extends Input {
59
64
public posKeyCode : string
60
65
public posKeyModifiers : ModifierState
@@ -67,6 +72,20 @@ class AxisInput extends Input {
67
72
public posGamepadButton : number
68
73
public negGamepadButton : number
69
74
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
+ */
70
89
public constructor (
71
90
inputName : string ,
72
91
posKeyCode ?: string ,
@@ -94,11 +113,14 @@ class AxisInput extends Input {
94
113
this . negGamepadButton = negGamepadButton ?? - 1
95
114
}
96
115
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
+ */
99
121
getValue ( useGamepad : boolean ) : number {
100
- // Gamepad joystick axis
101
122
if ( useGamepad ) {
123
+ // Gamepad joystick axis
102
124
if ( ! this . useGamepadButtons )
103
125
return InputSystem . getGamepadAxis ( this . gamepadAxisNumber ) * ( this . joystickInverted ? - 1 : 1 )
104
126
@@ -115,38 +137,28 @@ class AxisInput extends Input {
115
137
( InputSystem . isKeyPressed ( this . negKeyCode , this . negKeyModifiers ) ? 1 : 0 )
116
138
)
117
139
}
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
- }
133
140
}
134
141
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
+ */
135
146
class InputSystem extends WorldSystem {
136
147
public static currentModifierState : ModifierState
137
148
138
- // A list of keys currently being pressed
149
+ /** The keys currently being pressed. */
139
150
private static _keysPressed : { [ key : string ] : boolean } = { }
140
151
141
152
private static _gpIndex : number | null
142
153
public static gamepad : Gamepad | null
143
154
144
- // Maps a brain index to a certain input scheme
155
+ /** Maps a brain index to an input scheme. */
145
156
public static brainIndexSchemeMap : Map < number , InputScheme > = new Map ( )
146
157
147
158
constructor ( ) {
148
159
super ( )
149
160
161
+ // Initialize input events
150
162
this . handleKeyDown = this . handleKeyDown . bind ( this )
151
163
document . addEventListener ( "keydown" , this . handleKeyDown )
152
164
@@ -159,7 +171,7 @@ class InputSystem extends WorldSystem {
159
171
this . gamepadDisconnected = this . gamepadDisconnected . bind ( this )
160
172
window . addEventListener ( "gamepaddisconnected" , this . gamepadDisconnected )
161
173
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
163
175
document . addEventListener ( "visibilitychange" , ( ) => {
164
176
if ( document . hidden ) this . clearKeyData ( )
165
177
} )
@@ -177,13 +189,13 @@ class InputSystem extends WorldSystem {
177
189
}
178
190
179
191
public Update ( _ : number ) : void {
180
- InputSystem
181
192
// Fetch current gamepad information
182
193
if ( InputSystem . _gpIndex == null ) InputSystem . gamepad = null
183
194
else InputSystem . gamepad = navigator . getGamepads ( ) [ InputSystem . _gpIndex ]
184
195
185
196
if ( ! document . hasFocus ( ) ) this . clearKeyData ( )
186
197
198
+ // Update the current modifier state to be checked against target stats when getting input values
187
199
InputSystem . currentModifierState = {
188
200
ctrl : InputSystem . isKeyPressed ( "ControlLeft" ) || InputSystem . isKeyPressed ( "ControlRight" ) ,
189
201
alt : InputSystem . isKeyPressed ( "AltLeft" ) || InputSystem . isKeyPressed ( "AltRight" ) ,
@@ -199,21 +211,22 @@ class InputSystem extends WorldSystem {
199
211
window . removeEventListener ( "gamepaddisconnected" , this . gamepadDisconnected )
200
212
}
201
213
202
- // Called when any key is first pressed
214
+ /** Called when any key is first pressed */
203
215
private handleKeyDown ( event : KeyboardEvent ) {
204
216
InputSystem . _keysPressed [ event . code ] = true
205
217
}
206
218
207
- // Called when any key is released
219
+ /* Called when any key is released */
208
220
private handleKeyUp ( event : KeyboardEvent ) {
209
221
InputSystem . _keysPressed [ event . code ] = false
210
222
}
211
223
224
+ /** Clears all stored key data when the user leaves the page. */
212
225
private clearKeyData ( ) {
213
226
for ( const keyCode in InputSystem . _keysPressed ) delete InputSystem . _keysPressed [ keyCode ]
214
227
}
215
228
216
- // Called once when a gamepad is first connected
229
+ /* Called once when a gamepad is first connected */
217
230
private gamepadConnected ( event : GamepadEvent ) {
218
231
console . log (
219
232
"Gamepad connected at index %d: %s. %d buttons, %d axes." ,
@@ -226,22 +239,30 @@ class InputSystem extends WorldSystem {
226
239
InputSystem . _gpIndex = event . gamepad . index
227
240
}
228
241
229
- // Called once when a gamepad is first disconnected
242
+ /* Called once when a gamepad is first disconnected */
230
243
private gamepadDisconnected ( event : GamepadEvent ) {
231
244
console . log ( "Gamepad disconnected from index %d: %s" , event . gamepad . index , event . gamepad . id )
232
245
233
246
InputSystem . _gpIndex = null
234
247
}
235
248
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
+ */
237
254
public static isKeyPressed ( key : string , modifiers ?: ModifierState ) : boolean {
238
255
if ( modifiers != null && ! InputSystem . compareModifiers ( InputSystem . currentModifierState , modifiers ) )
239
256
return false
240
257
241
258
return ! ! InputSystem . _keysPressed [ key ]
242
259
}
243
260
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
+ */
245
266
public static getInput ( inputName : string , brainIndex : number ) : number {
246
267
const targetScheme = InputSystem . brainIndexSchemeMap . get ( brainIndex )
247
268
@@ -252,8 +273,12 @@ class InputSystem extends WorldSystem {
252
273
return targetInput . getValue ( targetScheme . usesGamepad )
253
274
}
254
275
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 {
257
282
if ( ! state1 || ! state2 ) return false
258
283
259
284
return (
@@ -264,7 +289,10 @@ class InputSystem extends WorldSystem {
264
289
)
265
290
}
266
291
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
+ */
268
296
public static getGamepadAxis ( axisNumber : number ) : number {
269
297
if ( InputSystem . gamepad == null ) return 0
270
298
@@ -276,7 +304,11 @@ class InputSystem extends WorldSystem {
276
304
return Math . abs ( value ) < 0.15 ? 0 : value
277
305
}
278
306
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
+ */
280
312
public static isGamepadButtonPressed ( buttonNumber : number ) : boolean {
281
313
if ( InputSystem . gamepad == null ) return false
282
314
0 commit comments