[XR Input, Quest] How to get button inputs on Oculus Quest?

Hi,

How can I get button inputs on the oculus quest? I need to detect the A,B, X, Y buttons.
I found the CommonUsages class but it does not have usages for these cases. How can I detect inputs for these cases?

I suppose I have to use inputDevice.TryGetFeatureValue() somehow. Should I enter the string value for A, B and etc?

Thanks!

Okay, I managed to get the button inputs.

  1. First, get the input devices.
    Code
InputDevices.GetDevicesWithCharacteristics(InputDeviceCharacteristics.Controller &
InputDeviceCharacteristics.TrackedDevice, _inputDevices);
  1. Then check if it is the left or right controller (this could have already been filtered and cached when getting the devies).
    Code if (device.characteristics.HasFlag(InputDeviceCharacteristics.Left))

  2. Get the IsPressed value of InputHelpers.Button.PrimaryButton or the InputHelpers.Button.SecondaryButton. The primary button on the right hand controller is A and secondary button is B. For the left hand controller these are X and Y.

But this is where I have the next issue. Currently I am using inputDevice.IsPressed() (Code if (inputDevice.IsPressed(InputHelpers.Button.PrimaryButton, out bool isPressedPrimaryButton) && isPressedPrimaryButton))and then pass the buttons as arguments. However, this is true as long as the buttons are held down. How would I get the button down and button up events? I could implement this functionality on my own but from a complete input system I would expect to access these methods in a simple manner.

Also, the OVRInput has an Axis1D struct to get the amount of how much the button has been pressed down (from 0 to 1). How can I get the 1D Axis value of these buttons in XRInput?

Thanks!

By going through the new VR escape room example made by Unity, I found how they solved it:
Code

/// <summary>
/// InteractionState type to hold current state for a given interaction.
/// </summary>
internal struct InteractionState
{
    /// <summary>This field is true if it is is currently on.</summary>
    public bool active;
    /// <summary>This field is true if the interaction state was activated this frame.</summary>
    public bool activatedThisFrame;
    /// <summary>This field is true if the interaction state was de-activated this frame.</summary>
    public bool deActivatedThisFrame;
}

void HandleInteractionAction(XRNode node, InputHelpers.Button button, ref InteractionState interactionState)
{
    bool pressed = false;
    inputDevice.IsPressed(button, out pressed, m_AxisToPressThreshold);
             
    if (pressed)
    {
        if (!interactionState.active)
        {
            interactionState.activatedThisFrame = true;
            interactionState.active = true;
        }
    }
    else
    {
        if (interactionState.active)
        {
            interactionState.deActivatedThisFrame = true;
            interactionState.active = false;
        }
    }
}

This means that we would have to detect the ButtonDown and ButtonUp events ourselves if we wanted to create a custom controller. It would be really helpful if there was an extension method similar to inputDevice.IsPressed() such as .WasPressedDown or .WasReleased or .ButtonDown or .ButtonUp. Every other input system has these functionalities built in.

Yeah, we got really spoiled by the build in Down and Up events.

here is my code

//global
private float _gripStrength;
private bool _grabbingActive = false;

//OnUpdate
_device.TryGetFeatureValue(CommonUsages.grip, out _gripStrength);

if (_gripStrength > 0.2f)
{
    if (!_grabbingActive)
    {
        _grabbingActive = true;
        //press event
    }
 

} else if (_grabbingActive)
{
    _grabbingActive = false;
    //release event

}
2 Likes

Nice!

I wrote a little bit different solution.

Code

private bool _leftTriggerDown;
private bool _leftGripDown;
// and other left hand buttons

private bool _rightTriggerDown;
private bool _rightGripDown;
// and other right hand buttons

private void Update()
{
    foreach (InputDevice inputDevice in _inputDevices)
    {
        if (inputDevice.characteristic.HasFlag(InputDeviceCharacteristic.Left))
        {
            // Left hand, grip button
            ProcessInputDeviceButton(inputDevice, InputHelpers.Button.Grip, ref _leftTriggerDown,
            () => // On Button Down
            {
                Debug.Log("Left hand trigger down");
                // Your functionality
            },
            () => // On Button Up
            {
                Debug.Log("Left hand trigger up");
            });
            // Repeat ProcessInputDeviceButton for other buttons
        }
        // Repeat for right hand
    }
}

private void ProcessInputDeviceButton(InputDevice inputDevice, InputHelpers.Button button, ref bool _wasPressedDownPreviousFrame, Action onButtonDown = null, Action onButtonUp = null, Action onButtonHeld = null)
{
    if (inputDevice.IsPressed(button, out bool isPressed) && isPressed)
    {
        if (!_wasPressedDownPreviousFrame) // // this is button down
        {
            onButtonDown?.Invoke();
        }

        _wasPressedDownPreviousFrame = true;
        onButtonHeld?.Invoke();
    }
    else
    {
        if (_wasPressedDownPreviousFrame) // this is button up
        {
            onButtonUp?.Invoke();
        }

        _wasPressedDownPreviousFrame = false;
    }
}
1 Like

Ah yes, good solution by generalising with ProcessInputDeviceButton

You could also automate the creation of the booleans with a list or a dictionary. Would make it even cleaner.

Wattosan,

Thank you for contributing that method! I have been using it as the basis for my Input System, although it seems to work best when it is called in Update() without the Foreach loop.

I would really like to use the Foreach loop method, but when nested like this, the actions get called every frame resulting in the wrong effect. Are you having this experience as well? If not, how did you work around it?

Hey!

I am not having that issue. The actions are not getting called every frame. In the example code they get called only if the button is actually pressed down/held down or the button is let go. If the player is not pressing or holding down the controller buttons, the actions do not get called. This is assuming that by actions you refer to the 3 callback actions in the ProcessInputDeviceButton() method.

Ah, you’re correct. I tried reducing your method by passing in dictionary elements. (Mapping Hand–Button, to find the reference bool). But it turns out that c# doesn’t support accessing ref variables via indexing (such as through a dictionary).

So I tried changing the “ref bool” parameter to just “bool” in the ProcessInput method, but that breaks the method. Any idea how I could restructure the method to take a “bool” instead of a “ref bool”. If not, can you help explain why a “ref bool” is required?

Any help is appreciated! Thanks.

The problem is that the inputDevice API does not have a method to get ButtonDown or ButtonUp events. Only IsPressed(). So you have implement the button up and button down cases yourself. The ref bool is used to check if the button was held down the previous frame as well. If it wasn’t (line 36) but it is now (line 34) then it is the first frame the button is held down - meaning that we got the ButtonDown(). If it is held down (line 34) and it was held down the previous frame as well, we no longer get the button down case. Some logic applies to releasing the button. If the button is no longer pressed (line 44) but it was held down the previous frame (line 46) then we get the button up case.

Using ref bool is just one way to do it. You could, use an array instead. Perhaps even in combination with a struct.

An example to get rid of the ref bool, using an array and a struct:
Code

private InputDeviceButton[] inputDeviceButtons;

private void Awake()
{
    // Init array here
    // loop through input devices
    // check if left or right
    // add entries into the array (or use a list)
}

private void Update()
{
    foreach (InputDeviceButton inputDeviceButton in inputDeviceButtons)
    {
        ProcessInputDeviceButton(inputDeviceButton);
    }
}

private void ProcessInputDeviceButton(InputDeviceButton inputDeviceButton)
{
    if (inputDeviceButton.inputDevice.IsPressed(inputDeviceButton.button, out bool isPressed) && isPressed)
    {
        if (!inputDeviceButton.wasPressedDownPreviousFrame) // // this is button down
        {
            inputDeviceButton.onButtonDown?.Invoke();
        }

        inputDeviceButton.wasPressedDownPreviousFrame = true;
        inputDeviceButton.onButtonHeld?.Invoke();
    }
    else
    {
        if (inputDeviceButton.wasPressedDownPreviousFrame) // this is button up
        {
            inputDeviceButton.onButtonUp?.Invoke();
        }

        inputDeviceButton.wasPressedDownPreviousFrame = false;
    }
}

private struct InputDeviceButton
{
    public InputDevice inputDevice;
    public InputHelpers.Button button;
    public bool wasPressedDownPreviousFrame 
    public Action onButtonDown;
    public Action onButtonUp;
    public Action onButtonHeld;

    public InputDeviceButton(InputDevice inputDevice, InputHelpers.Button button, bool wasPressedDownPreviousFrame, Action onButtonDown, Action onButtonUp, Action onButtonHeld)
    {
        this.inputDevice = inputDevice;
        this.button = button;
        this.wasPressedDownPreviousFrame = wasPressedDownPreviousFrame;  
        this.onButtonDown = onButtonDown;
        this.onButtonUp = onButtonUp;
        this.onButtonHeld = onButtonHeld;
    }
}
1 Like

Hi Wattosan, thank you for all of your help! I tried implementing it with the struct method you suggested, but ultimately ran into the same problem of having my actions called repeatedly each frame. At the very least, I learned something from your code.

I ended up solving my ref bool issue by creating a Boolean wrapper class and then having ProcessInputDeviceButton() take a Boolean instead of a ref bool. Using this method allowed me to get a reference to the corresponding button booleans, while also being able to access them through dictionaries. (Unlike with ref bools).

//Boolean Wrapper Class
public class Boolean
{
        public bool Value;
        public Boolean(bool value)
        {
            Value = value;
        }
}


//Boolean construction and reference example:
public static Boolean right_grip;
public static Boolean left_thumbStickPress;

void Start()
{
    right_grip = new Boolean(false);
    left_thumbStickPress = new Boolean(false);
}
...etc for rest of the Booleans.


//Nested dictionaries to store the mappings (Takes the XR Node and Button to access the reference Boolean)
deviceButtonToBoolDict = new Dictionary<XRNode, Dictionary<InputHelpers.Button, Boolean>>()
{
    { RightHand, new Dictionary<InputHelpers.Button, Boolean>()
           { InputHelpers.Button.Primary2DAxisClick, right_thumbStickPress }
    },
     { LeftHand, new Dictionary<InputHelpers.Button, Boolean>()
           { InputHelpers.Button.Primary2DAxisClick, left_thumbStickPress }
     }
});
etc for rest of mappings...


//And then I call your method in Fixed Update like so:

foreach (XRNode node in InputManager.xrNodes)
{
    var triggerBool = InputManager.deviceButtonToBoolDict[node][InputHelpers.Button.Trigger];

    InputManager.ProcessInputDeviceButton(InputDevices.GetDeviceAtXRNode(node), InputHelpers.Button.Trigger, triggerBool, null, null,
() => //On Button Held
{
       //Do something
});

This method reduces unnecessary code and works as intended for both hands. I will leave this code block here for anyone who is running into the same problem that I did. Once again, thanks for all of your help!