How to use hold/press interaction on the same button to trigger 2 different actions?

There is actually 2 different problems I want to solve, and I’m currently unable to solve either one, but the one I described in the title is the most important one.

First of all how do I trigger 2 different actions from the same button depending on the interaction (hold/press)? Think Dark Souls. In Dark Souls you sprint by holding down the B-button and you roll by pressing the B-button.

I’m currently not able to get this work. I know how I would make it work by just coding the logic myself with timers etc., but this has to be the sort of thing the new input system is meant to solve for you, right?

I came across this thread Hold and press on the same button/path without performing both actions when holding , but that deals with 2 different interactions for the same action, not 2 different actions like I have. And I wasn’t able to get that to work either way. Not to mention the image he posted doesn’t exist anymore, which I feel is embarrassing for Unity’s sake. That post is just over a year old and it’s not usable as support due to essential information missing from it.

And then there’s the second problem: In Dark Souls both actions are hardcoded as 1 bindable action with hold/press behavior. So you’re forced to bind both actions to the same key even when playing on keyboard. I want these actions to be freely bindable by the player. I was thinking I could allow my player to choose both the button/key for the action and also choose the interaction (hold/press) and that way let them freely configure it exactly how they want it and not have the limitation that Dark Souls has. But the actual rebinding is secondary, just getting this to work is more important. And like I said solving the problem in the topic title is the most important.

This is my input actions at the moment:

There you can see that I want to use Left Shift for Sprint and Space for Roll on keyboard, while I want to use the B-button for both actions on gamepad.

I’ve set the B-Button to Hold interaction for the Sprint action as shown and I’ve added a Press interaction for the Roll action. I’ve done the same for the keyboard keys.

So far in code I have this:

_inputActions.ActionMap.Sprint.performed += context => {
    if (context.interaction is HoldInteraction) {
        SprintInput(true);
    }
};
_inputActions.ActionMap.Sprint.canceled += context => {
    if (context.interaction is HoldInteraction) {
        SprintInput(false);
    }
};
_inputActions.ActionMap.Roll.performed += context => {
    if (context.interaction is PressInteraction) {
        RollInput();
    }
};

But this does not work at all:

  • If I tap the B-button then both RollInput() and SprintInput(false) are called. This is fine.
  • If I hold the B-Button then both SprintInput(true) and RollInput() are called. RollInput() should no be called here.
  • If I release the B-Button after a hold then nothing happens. After SprintInput(true) has been called I am unable to stop sprinting. SprintInput(false) is never called.
  • On Keyboard I obviously don’t have the problem that both sprint and roll are called at the same time as they are on different keys, but I’m still unable to stop sprinting.

I’ve looked at the samples included with the package and none of them are helpful from what I could tell.

Thanks to information in this thread New Input System : How to use the "Hold" interaction. I was able to solve this.

First of all I removed all the interactions from my sprint and roll actions in the input system because they didn’t work and then I changed my code to be like this:

using UnityEngine;
using UnityEngine.InputSystem;

namespace DarkSouls {

    public class Player : MonoBehaviour {

        public CharacterController controller;

        private InputActions _inputActions;

        [Tooltip("The time in seconds before sprint is started if the sprint button is held down. This is if both sprint and roll is bound to the same button.")]
        public float holdTimeBeforeSprintIsStarted = 0.2f;
        private bool _rollButtonPressed;
        private bool _sprintButtonHeld;
        private float _sprintButtonHoldTimer;

        private void OnEnable() {
            _inputActions.Enable();
        }

        private void OnDisable() {
            _inputActions.Disable();
        }

        private void Awake() {
            _inputActions = new InputActions();
            _inputActions.ActionMap.Sprint.started += SprintOrRollInput;
            _inputActions.ActionMap.Sprint.canceled += SprintOrRollInput;
            _inputActions.ActionMap.Roll.started += SprintOrRollInput;
            _inputActions.ActionMap.Roll.canceled += SprintOrRollInput;
        }

        private void Update() {
            // Continously read the hold state for the sprint action.
            // It returns 1 when held and 0 when not held so we convert that to a boolean for easier usability.
            _sprintButtonHeld = System.Convert.ToBoolean(_inputActions.ActionMap.Sprint.ReadValue<float>());
         
            if (_sprintButtonHeld) {
                _sprintButtonHoldTimer += Time.deltaTime;
                if (_sprintButtonHoldTimer > holdTimeBeforeSprintIsStarted) {
                    controller.SprintingStart();
                }
            }
        }

        private void SprintOrRollInput(InputAction.CallbackContext context) {
            if (context.started == true) {
                if (context.action == _inputActions.ActionMap.Sprint) {
                    _sprintButtonHoldTimer = 0;
                }
                if (context.action == _inputActions.ActionMap.Roll) {
                    _rollButtonPressed = true;
                    // If we're holding down the sprint button when we press the roll button
                    // then just roll immediately because I couldn't get it to work otherwise.
                    if (_sprintButtonHeld) {
                        controller.Roll();
                        _rollButtonPressed = false;
                    }
                }
            }

            if (context.canceled == true) {
                if (_sprintButtonHeld && _rollButtonPressed) {
                    if (_sprintButtonHoldTimer > holdTimeBeforeSprintIsStarted) {
                        controller.SprintingStop();
                    }
                    else {
                        controller.Roll();
                    }
                    _rollButtonPressed = false;
                }
                else if (_rollButtonPressed) {
                    controller.Roll();
                    _rollButtonPressed = false;
                }
                else {
                    controller.SprintingStop();
                }
            }
        }
    }
}

The code is convoluted, but it works, and it works better than Dark Souls.

I learned quite a bit from that other thread. For example I learned the syntax I used above for passing the context object into the method*. I also learned that the Hold interaction doesn’t work how I expected it to at all. The method you put in the callback is still only called once with Hold, same as a Press, not continuously as I thought. So you seemingly still have to implement the logic for detecting if the button is pressed, held or released yourself, same as with the old input system.

What I have now works better than Dark Souls because I’m able to have Roll and Sprint on different keys on keyboard while still having them on the same button on the gamepad. And of course they could be on the same key on keyboard or on different buttons on gamepad as I’ve done nothing to not make it device agnostic. It’s still only checking the actions, and not the bindings.

There’s still one thing that bothers me though and that is the fact that on keyboard where there’s currently separate keys for the actions you still only roll when you release the spacebar and it still takes 0.2 seconds for the sprint to initiate as if it was checking to see if it was a roll even though that’s impossible as they are on separate keys. It’s not a huge problem to begin with, and I’m sure it can be solved. I’m just happy to get this far. I know it could be solved by checking to see which binding triggered the action, but then it wouldn’t be device agnostic anymore.

* I have to say that the syntax for interacting with the new input system is completely foreign to me. I have no idea if this is native C# or some Unity gibberish, but what was wrong with Input.GetButton()? :stuck_out_tongue: I’m guessing this is some sort of event/delegate ordeal, but does doing it this way actually have any benefits other than to look advanced and feel convoluted?
Edit: Gah. Just found an obvious issue. Rolling with the gamepad button tied to both actions initiates the sprint after you roll. I’ll fix that later and update the code here.
Edit 2: While fixing the above issue I discovered two more issues with my initial approach. You would stop sprinting if you pressed the spacebar on the keyboard while still holding the shift key. And if you quickly tapped the left shift key and the spacebar together you wouldn’t roll, only if you held the shift key longer than 0.2 seconds would you roll. This code is no longer visible here because I edited the post so you won’t know and probably don’t care what it was, but at least so far my new code has no known issues. However it’s even more convoluted and I had to dive into the other thread to learn one more thing to get this to work, namely how to poll the input action directly from the Update method so that I could see if the sprint button was currently being held or not.
This can surely (hopefully) be simplified a lot, but it at least works now and does everything I want it to. For now at least. If I wanted to support rebindable controls then this would surely need to be expanded upon, but by then I would hopefully know how to write it simpler as well.
One more thing to note is that this new version of the code continously calls controller.SprintingStart() while the sprint button is held which looks and feels wrong, but all that method does at the moment is set an isSprinting bool to true if a couple of flags like isGrounded etc. are true so it doesn’t really matter.
Edit 3: I forgot the Awake method in the last code and I quickly refactored it a bit for my own future sake so I may as well post those changes here as well. I posted the entire script so you have the full context for how it’s used.

2 Likes