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?
-
1If 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 Answers
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];
}
}

- 3,856
- 3
- 30
- 37
-
5for folks that like interface builder (me), minimumPressDuration can also be set in IB (thank you Rob for great solution) – tmr Nov 08 '14 at 04:07
-
-
1@CalZone Yes, check for the gestures state `UIGestureRecognizerStateEnded` – Rob Caraway Nov 17 '14 at 11:31
-
2
-
-
-
@Anish웃 Setting `gestureRecognizer.cancelsTouchesInView = false` will allow for other GR's to act on the event as well. – Clay Ellis Oct 27 '16 at 05:13
-
1***DANGER*** it can often be called more than once, with a typical touch. – Fattie Feb 26 '17 at 21:29
-
-
If it's called more than once be sure to add the if statement in the didTouchDown function – Carioni Feb 13 '19 at 09:42
-
This also helps mask the touch delay in a UITableView due to its UIScrollView (see `delaysContentTouches`). – Jeff May 07 '19 at 09:42
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.
-
10extra +1 for the "UIGestureRecognizerSubclass.h" import. Nice. – Sebastian Hojas Jul 26 '13 at 12:18
-
3Shouldn't super be invoked in the touchesBegin,touchesMoved,touchesEnded methods? – neoneye Feb 25 '14 at 06:21
-
3
-
-
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
-
-
-
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
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")
}
}
}

- 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
-
-
-
@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
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")
}

- 1,926
- 2
- 24
- 34
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)

- 17,576
- 18
- 108
- 256
An alternative approach: Add an transparent button inside the UIView and assign touchUp and touchDown actions accordingly.

- 437
- 1
- 6
- 13