92

What I'm using the tap event for is very time-sensitive, so I'm curious if it's possible to make UITapGestureRecognizer activate when the user simply touches down, rather than requiring them to touch up as well?

Rob Caraway
  • 3,856
  • 3
  • 30
  • 37
user212541
  • 1,878
  • 1
  • 21
  • 30
  • 1
    If it helps, UITouch has a touchesStarted method. But that isn't using gesture recognizers, as you asked. – vqdave Mar 26 '13 at 02:14

6 Answers6

201

Use a UILongPressGestureRecognizer and set its minimumPressDuration to 0. It will act like a touch down during the UIGestureRecognizerStateBegan state.

For Swift 4+

func setupTap() {
    let touchDown = UILongPressGestureRecognizer(target:self, action: #selector(didTouchDown))
    touchDown.minimumPressDuration = 0
    view.addGestureRecognizer(touchDown)
}

@objc func didTouchDown(gesture: UILongPressGestureRecognizer) {
    if gesture.state == .began {
        doSomething()
    }
}

For Objective-C

-(void)setupLongPress
{
   self.longPress = [[UILongPressGestureRecognizer alloc] initWithTarget:self action:@selector(didLongPress:)];
   self.longPress.minimumPressDuration = 0;
   [self.view addGestureRecognizer:self.longPress];
}

-(void)didLongPress:(UILongPressGestureRecognizer *)gesture
{
   if (gesture.state == UIGestureRecognizerStateBegan){
      [self doSomething];
   }
}
Rob Caraway
  • 3,856
  • 3
  • 30
  • 37
142

Create your custom TouchDownGestureRecognizer subclass and implement gesture in touchesBegan:

TouchDownGestureRecognizer.h

#import <UIKit/UIKit.h>

@interface TouchDownGestureRecognizer : UIGestureRecognizer

@end

TouchDownGestureRecognizer.m

#import "TouchDownGestureRecognizer.h"
#import <UIKit/UIGestureRecognizerSubclass.h>

@implementation TouchDownGestureRecognizer
-(void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event{
    if (self.state == UIGestureRecognizerStatePossible) {
        self.state = UIGestureRecognizerStateRecognized;
    }
}

-(void)touchesMoved:(NSSet *)touches withEvent:(UIEvent *)event{
    self.state = UIGestureRecognizerStateFailed;
}

-(void)touchesEnded:(NSSet *)touches withEvent:(UIEvent *)event{
    self.state = UIGestureRecognizerStateFailed;
}


@end

implementation:

#import "TouchDownGestureRecognizer.h"
    TouchDownGestureRecognizer *touchDown = [[TouchDownGestureRecognizer alloc] initWithTarget:self action:@selector(handleTouchDown:)];
    [yourView addGestureRecognizer:touchDown];

-(void)handleTouchDown:(TouchDownGestureRecognizer *)touchDown{
    NSLog(@"Down");
}

Swift implementation:

import UIKit
import UIKit.UIGestureRecognizerSubclass

class TouchDownGestureRecognizer: UIGestureRecognizer
{
    override func touchesBegan(touches: Set<UITouch>, withEvent event: UIEvent)
    {
        if self.state == .Possible
        {
            self.state = .Recognized
        }
    }

    override func touchesMoved(touches: Set<UITouch>, withEvent event: UIEvent)
    {
        self.state = .Failed
    }

    override func touchesEnded(touches: Set<UITouch>, withEvent event: UIEvent)
    {
        self.state = .Failed
    }
}

Here is the Swift syntax for 2017 to paste:

import UIKit.UIGestureRecognizerSubclass

class SingleTouchDownGestureRecognizer: UIGestureRecognizer {
    override func touchesBegan(_ touches: Set<UITouch>, with event: UIEvent) {
        if self.state == .possible {
            self.state = .recognized
        }
    }
    override func touchesMoved(_ touches: Set<UITouch>, with event: UIEvent) {
        self.state = .failed
    }
    override func touchesEnded(_ touches: Set<UITouch>, with event: UIEvent) {
        self.state = .failed
    }
}

Note that this is a drop-in replacement for UITap. So in code like...

func add(tap v:UIView, _ action:Selector) {
    let t = UITapGestureRecognizer(target: self, action: action)
    v.addGestureRecognizer(t)
}

you can safely swap to....

func add(hairtriggerTap v:UIView, _ action:Selector) {
    let t = SingleTouchDownGestureRecognizer(target: self, action: action)
    v.addGestureRecognizer(t)
}

Testing shows it will not be called more than once. It works as a drop-in replacement; you can just swap between the two calls.

amleszk
  • 6,192
  • 5
  • 38
  • 43
LE SANG
  • 10,955
  • 7
  • 59
  • 78
  • 10
    extra +1 for the "UIGestureRecognizerSubclass.h" import. Nice. – Sebastian Hojas Jul 26 '13 at 12:18
  • 3
    Shouldn't super be invoked in the touchesBegin,touchesMoved,touchesEnded methods? – neoneye Feb 25 '14 at 06:21
  • 3
    @JasonSilberman you forget #import – LE SANG Jul 18 '14 at 02:20
  • How would you implement double tap with this? – Babiker May 24 '15 at 09:07
  • Wait a minute, but how do you get Up and Down callbacks for the same tap? I want to know both the tap down and tap up events for the same tap. Does this just replace Up with Down? – etayluz Jul 01 '15 at 18:31
  • 1
    @etayluz you have assign state to begin and end. Then check it's state in handle. Custom recognizer mean you can control everything. – LE SANG Jul 02 '15 at 00:55
  • Thanks @Bum, I've been playing with that and I've noticed that while begin is always fired, end is fickle and not dependable - sometimes will fire and sometimes will NOT fire – etayluz Jul 02 '15 at 03:14
  • @swipekeys.com hi, I found that this subclass will block my long press gesture recogniser, that never called after I add this recogniser to my code. I have one thought about it. I work on Swift, and there is no `Recognized` state, so I used `Ended` state. I wonder, could this be the case? Can you show code for Swift? Thanks. – Dima Deplov Sep 24 '15 at 10:02
  • Found the answer! Forget to added `UIGestureRecognizerDelegate` protocol and set delegate = self. Method `- gestureRecognizer:shouldRecognizeSimultaneouslyWithGestureRecognizer: ` is what you looking for – Dima Deplov Sep 24 '15 at 10:58
  • The -state property is not writeable. Please consider updating the code above. – osxdirk Oct 20 '15 at 12:43
  • @osxdirk you forget import – LE SANG Oct 20 '15 at 12:57
  • touches ended not called. Do u know why ? – ZeeroCool77 Jan 21 '16 at 21:08
  • bjour @LESANG, thanks for the great answer. I did paste in the latest Swift3 code. Feel free to edit, unwind or use the edit. Thanks again – Fattie Feb 27 '17 at 11:40
  • @LESANG this answer works but I lost the ability to swipe because as soon as I put my finger on the object to swipe it it thinks it's a touch. Why can't it differentiate between them both? – Lance Samaria Jan 10 '20 at 13:34
26

Swift (without subclassing)

Here is a Swift version similar to Rob Caraway's Objective-C answer.

The idea is to use a long press gesture recognizer with the minimumPressDuration set to zero rather than using a tap gesture recognizer. This is because the long press gesture recognizer reports touch began events while the tap gesture does not.

import UIKit
class ViewController: UIViewController {

    @IBOutlet weak var myView: UIView!

    override func viewDidLoad() {
        super.viewDidLoad()

        // Add "long" press gesture recognizer
        let tap = UILongPressGestureRecognizer(target: self, action: #selector(tapHandler))
        tap.minimumPressDuration = 0
        myView.addGestureRecognizer(tap)
    }

    // called by gesture recognizer
    @objc func tapHandler(gesture: UITapGestureRecognizer) {

        // handle touch down and touch up events separately
        if gesture.state == .began {
            // do something...
            print("tap down")
        } else if gesture.state == .ended { // optional for touch up event catching
            // do something else...
            print("tap up")
        }
    }
}
Suragch
  • 484,302
  • 314
  • 1,365
  • 1,393
  • 1
    @richy, It felt a bit like a hack to me, too, since the name is **long** press gesture recognizer, but this method is a lot easier than [subclassing the view](http://stackoverflow.com/a/34764356/3681880) and like you said, it works great. – Suragch Jul 06 '16 at 21:05
  • This solution cause scrolling issue in UIscrollView subclasses – SPatel Jan 09 '18 at 06:11
  • I think bater approach is Custom Gesture recognizer – SPatel Jan 09 '18 at 06:12
  • @Suragch I ran into the same issue with this that I ran into with lessing's answer. This answer works but I lost the ability to swipe because as soon as I put my finger on the object to swipe it it thinks it's a touch. How can it differentiate between them both? – Lance Samaria Jan 10 '20 at 13:58
  • 1
    @LanceSamaria, If you need more complex touch recognition than just tap I would use a custom gesture recognizer. – Suragch Jan 11 '20 at 08:10
  • @Suragch thanks for getting back to me. I’ll ask a new question regarding it. Cheers! – Lance Samaria Jan 11 '20 at 11:10
1

This is another solution. Create subclass of UIControl. You can use it like UIView even in Storyboard because UIControl is subclass of UIView.

class TouchHandlingView: UIControl {
}

And addTarget to it:

@IBOutlet weak var mainView: TouchHandlingView!
...

mainView.addTarget(self, action: "startAction:", forControlEvents: .TouchDown)
...

Then the designated action will be called like UIButton:

func startAction(sender: AnyObject) {
    print("start")
}
morizotter
  • 1,926
  • 2
  • 24
  • 34
1

I needed the ability for my view to have a hair trigger so as soon as it's tapped it responds. Using both @LESANG answer worked and so did using @RobCaraway answer. The problem I ran into with both answers was I lost the ability to recognize swipes. I needed my view to rotate when swiped but as soon as my finger touched the view only the tap was recognized. The tapRecognizer was too sensitive and couldn't differentiate between a tap and a swipe.

This is what I came up with based off of @LESANG answer combined with this answer and this answer.

I put 6 comments in each event.

import UIKit.UIGestureRecognizerSubclass

class SingleTouchDownGestureRecognizer: UIGestureRecognizer {


    var wasSwiped = false

    override func touchesBegan(_ touches: Set<UITouch>, with event: UIEvent) {

        guard let view = self.view else { return }
        guard let touches = event.touches(for: view) else { return } // 1. compare that event in touchesBegan has touches for the view that is the same as the view to which your gesture recognizer was assigned

        if touches.first != nil {
            print("Finger touched!") // 2. this is when the user's finger first touches the view and is at locationA
            wasSwiped = false // 3. it would seem that I didn't have to set this to false because the property was already set to false but for some reason when I didn't add this it wasn't responding correctly. Basically set this to false
        }
    }

    override func touchesMoved(_ touches: Set<UITouch>, with event: UIEvent) {

        guard let touch = touches.first else { return }

        let newLocation = touch.location(in: self.view)
        let previousLocation = touch.previousLocation(in: self.view)

        if (newLocation.x > previousLocation.x) || (newLocation.x < previousLocation.x) {
            print("finger touch went right or left") // 4. when the user's finger first touches it's at locationA. If the the user moves their finger to either the left or the right then the finger is no longer at locationA. That means it moved which means a swipe occurred so set the "wasSwiped" property to true

            wasSwiped = true // 5. set the property to true because the user moved their finger
        }
    }

    override func touchesEnded(_ touches: Set<UITouch>, with event: UIEvent) {
        print("finger is no longer touching.") // 6. the user has lifted their finger off of the view. If "wasSwiped" is true then ".fail" but if it wasn't swiped then ".recognize"

        if wasSwiped {
            self.state = .failed
        } else {
            self.state = .recognized
        }
    }
}

And to use it so that view that uses it gets the hair trigger response and left and right swipe gestures.:

let tapGesture = SingleTouchDownGestureRecognizer(target: self, action: #selector(viewWasTapped(_:)))
myView.addGestureRecognizer(tapGesture)

let rightGesture = UISwipeGestureRecognizer(target: self, action: #selector(respondToSwipeGesture(recognizer:)))
rightGesture.direction = .right
myView.addGestureRecognizer(rightGesture)

let leftGesture = UISwipeGestureRecognizer(target: self, action: #selector(respondToSwipeGesture(recognizer:)))
leftGesture.direction = .left
myView.addGestureRecognizer(leftGesture)
Lance Samaria
  • 17,576
  • 18
  • 108
  • 256
0

An alternative approach: Add an transparent button inside the UIView and assign touchUp and touchDown actions accordingly.

Billy
  • 437
  • 1
  • 6
  • 13