You know what we don't have? A good input abstraction crate.
icefoxen opened this issue · 5 comments
I'm NOT talking about something that actually PRODUCES input events, such as winit, SDL2, gilrs, etc. I am rather saying that we should have a simple but general-purpose crate that lets us configure physical<->logical input bindings and such, so it's easy to do sensible things like allow users to remap keys and load the mappings from config files.
Examples of what I want, from non-Rust game engines:
- https://docs.unity3d.com/Manual/ConventionalGameInput.html
- https://docs.godotengine.org/en/3.1/tutorials/inputs/input_examples.html#inputmap
Current state of the art as far as I'm aware:
- https://crates.io/crates/amethyst_input -- Basically what I want but idk how easy it is to decouple from the rest of Amethyst. I think it's designed to be possible, so, it might be exactly what we want? It does depend on
amethyst_core
andamethyst_window
though. - https://crates.io/crates/ggez-goodies -- The
input
module is intended to be exactly this but I've never bothered going back and making it suck less.
Thoughts?
The SDL2 "mapping string" is industry standard here. Even when not using SDL2 itself, the mapping string format should be supported. End users expect your programs to support this format. For example, Steam will place any alternate mapping you should use in an environment variable that you're supposed to read at startup and respect if it's set. This allows steam to set "system wide" remappings when people have odd controllers.
The format is a textual description of how things work, and assumes that you're trying to take some sort of "thing with buttons and axes" and make it act more like a "thing that's like an xbox 360 controller". An example string is in their wiki:
"341a3608000000000000504944564944,Afterglow PS3 Controller,a:b1,b:b2,y:b3,x:b0,start:b9,guide:b12,back:b8,dpup:h0.1,dpleft:h0.8,dpdown:h0.4,dpright:h0.2,leftshoulder:b4,rightshoulder:b5,leftstick:b10,rightstick:b11,leftx:a0,lefty:a1,rightx:a2,righty:a3,lefttrigger:b6,righttrigger:b7"
So since we have "a thing that's kinda like an XBox 360 controllers", we can use that as one starting point for what sort of abstract inputs we'd have and would want to be able to map to game inputs.
I didn't quite mean something like that, which afaik is there mostly to deal with the zoo of mutually-incompatible controller layouts. (gilrs
does use this format and load its definitions from SDL2, I believe.) More something that goes atop of it, and says "Right stick X and Y axis: camera" and "Left stick X and Y axis: motion".
That sounds like it quickly gets very game specific.
Can you expand your example idea some more?
Now that I'm at a keyboard:
The OpenXR action API works as follows:
- An application declares at startup all the typed logical actions that the user can take
- Optionally, the application may suggest default physical bindings for those actions on one or more abstract input device archetypes (e.g. xbox controller).
- As the application undergoes state changes, it activates different subsets of the declared actions
- The state of active actions is sampled by game logic as needed
The OpenXR runtime maps physical inputs to logical actions in the spirit @icefoxen describes. In lieu of user-customized bindings, the application's suggested bindings can be mapped directly to hardware that closely corresponds to the archetype, loosely to hardware that is analogous, or ignored entirely if there's no reasonable mapping to available hardware. Mappings can transform input; for example, mapping an analog axis to a boolean action could result in a boolean that is true if the axis is above a certain threshold.
I believe Khronos made a mistake in requiring applications to consume input by sampling states. For the proposed library, I suggest instead exposing a lossless event queue.
In the spirit of the above, the proposed library could have three mostly-independent APIs: a hardware-facing one that allows input from winit/gilrs/... to be dumped in, an application-facing API to declare, suggest bindings for, and consume events for actions, and an application-facing API suitable for implementation of user-facing binding UIs. Additionally, a standardized human-friendly and aggressively forwards-compatible config format for representing mappings would be useful. Ideally winit and co (or helper crates) could implement support for the hardware-facing API, allowing application developers to focus on the logical aspects.
It would be interesting but likely difficult to attempt to build an interface with first-class support for driving OpenXR underneath rather than the built-in processing that would be required for conventional applications.