subjective-c
subjective-c

Reputation: 1823

How to cancel a sequence of UITouch events?

I have a UIImage view that responds to touch events. I want to cancel the touch sequence, i.e., further calls to touchesMoved:, if the touch goes outside of certain bounds. How can I do that?

I know that in touchesMoved: I can inspect the coordinates of the touch object and ignore it, but what I don't know is how to cancel the sequence altogether. I don't see any method documented in the Apple Developer UIResponder Reference that I can call to cancel a touch sequence.

Upvotes: 22

Views: 18919

Answers (12)

Shimon Young
Shimon Young

Reputation: 11

I achieve this by removing the view from its superview and adding it straight back.

[view retain];
UIView *sv = view.superview;
[view removeFromSuperview];
[sv addSubview:view];
[view release];

This breaks the responder chain to the view so any remaining touches will not be received on the view. The next touch will still be received as normal.

Upvotes: 1

Indrajeet
Indrajeet

Reputation: 5666

This code will help you. I disable touch event in touchesMoved: method and I am enabling touch event in touchesCancelled: method

-(void)touchesMoved:(NSSet *)touches withEvent:(UIEvent *)event
{
     UITouch *touch = [touches anyObject];
     .......
     .......
     Your code
     .......
     .......

     //Condition when you want to disable touch event

     [[UIApplication sharedApplication] beginIgnoringInteractionEvents];

     .......
     .......
     Your code
     .......
     .......
 }


- (void)touchesCancelled:(NSSet *)touches withEvent:(UIEvent *)event
{
    [[UIApplication sharedApplication] endIgnoringInteractionEvents];
}

Upvotes: 0

user2292337
user2292337

Reputation: 1

You may create category to UITouch. You can to declare

@property (nonatomic,strong,readwrite) UIView *view;

And when you change touch.view to nil, for example, you can simulate end of dispatching touch events

Upvotes: -1

ceekay
ceekay

Reputation: 1157

"Brute force" solution depending if it suits your needs: Remove and re-initialize the subview that receives the touches or responds to them (of course take care of frame and content).

Upvotes: 1

James White
James White

Reputation: 784

I was just trying to solve something like this, and found that none of the solutions listed her worked for me. The best i could manage was temporarily ignoring the touches, but then they resumed when the touch re-entered the view.

Finally solved it.

  1. Set up a boolean flag called "disabled"
  2. In touchedMoved, first check that the touch is inside the desired view. If it isn't, set the disabled flag to YES.
  3. Still in in touchedMoved, check the flag before performing your action. So, at this point, nothing will happen if they're outside the view.
  4. In touchesEnded, set the disabled flag to NO. So the touch sequence is effectively halted until user lifts their finger.

I guess this will work for you unless you have a specific reason for literally needing the touch sequence cancelled.

Upvotes: 0

Zaky German
Zaky German

Reputation: 14334

Try temporary setting the UIImageView's userInteractionEnabled property to NO

Upvotes: 4

DanZimm
DanZimm

Reputation: 2568

On iOS5 there seems to be a private method in UITouch

-(void)setSentTouchesEnded:(BOOL)ended;

Depending on apple's implementation it could stop sending events

Another way of doing so would be using associative objects

- (void)touchesMoved:(NSSet *)touches withEvent:(UIEvent *)event
{
    if ([(NSNumber *)objc_getAssociatedObject(touch, &outKey) boolValue])
        return;
    if (sometouchisoutsideofview) {
        objc_setAssociatedObject(touch, &outKey, [NSNumber numberWithBool:YES], OBJC_ASSOCIATION_RETAIN);
    }
}

Upvotes: 0

Surjit Joshi
Surjit Joshi

Reputation: 3515

You can check that, the touch point locations are in CGRect (ie. points are in Rectangle of your imageview) or not. If they are not in that Rect, the touch will be cancelled.

-(void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event 
 {
UITouch *touch = [touches anyObject];
CGPoint touchLocation = [touch locationInView:self.view];

    if (CGRectContainsPoint(myImageView, touchLocation))
    {
       lastPoint = [touch locationInView: myImageView];
    }

    NSLog(@"Last :%.2f - %.2f",lastPoint.x, lastPoint.y);   
  }

 - (void)touchesMoved:(NSSet *)touches withEvent:(UIEvent *)event 
 {
  CGPoint currentPoint;
  UITouch *touch = [touches anyObject];
  CGPoint touchLocation = [touch locationInView:self.view];

     if (CGRectContainsPoint(myImageView.frame, touchLocation))
     {
       currentPoint = [touch locationInView: myImageView];
     }
 }

Upvotes: -2

ma11hew28
ma11hew28

Reputation: 126327

I don't think it's possible because I don't see it documented and none of these solutions work.

Upvotes: 2

David Beck
David Beck

Reputation: 10159

You need to call [super touchesMoved:withEvent:] in order to let the super view clean up from the event but more importantly, you need to not call [super touchesCancelled:withEvent:].

Here's what I used on a cell to keep it from getting selected when I detected a swipe:

- (void)touchesMoved:(NSSet *)touches withEvent:(UIEvent *)event
{
    if (!showingEdit) {
        if (IS_FAR_ENOUGH_TO_BE_A_SWIPE) {
            RESPOND_TO_SWIPE
            showingEdit = YES;
            [super touchesCancelled:touches withEvent:event];
        } else {
            [super touchesMoved:touches withEvent:event];
        }
    }
}

- (void)touchesEnded:(NSSet *)touches withEvent:(UIEvent *)event
{
    if (!showingEdit) {
        [super touchesEnded:touches withEvent:event];
    }
}

Upvotes: 3

user309122
user309122

Reputation: 189

I've faced the same problem recently and found a standard way to solve it. You can use [[UIApplication sharedApplication] beginIgnoringInteractionEvents] to stop delivering touchesMoved events to your whole app. Make sure to enable them using [[UIApplication sharedApplication] endIgnoringInteractionEvents] when you need to receive touches again.

Upvotes: 2

Michael Fey
Michael Fey

Reputation: 1239

This solution may be a bit kludgy, but you could implement and manually call

- (void) touchesCancelled:(NSSet *)touches withEvent:(UIEvent *)event;

I am basing this solution loosely on some tweaking I did to the MoveMe sample app on Apple's iPhone sample code site where I modified the touchesMoved method to look like this:

- (void) touchesMoved:(NSSet *)touches withEvent:(UIEvent *)event {
     UITouch *touch = [touches anyObject];
     if ([touch view] == placardView)
         CGPoint location = [touch locationInView:self];
         placardView.center = location;
         // I added the following line:
         [self touchesCancelled:touches withEvent:event];
         return;
}

Upvotes: 7

Related Questions