Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Passing through touches to UIViews underneath

I have a UIView with 4 buttons on it and another UIView on top of the buttons view. The top most view contains a UIImageView with a UITapGestureRecognizer on it.

The behavoir I am trying to create is that when the user taps the UIImageView it toggles between being small in the bottom right hand corner of the screen and animating to become larger. When it is large I want the buttons on the bottom view to be disabled and when it is small and in the bottom right hand corner I want the touches to be passed through to the buttons and for them to work as normal. I am almost there but I cannot get the touches to pass through to the buttons unless I disable the UserInteractions of the top view.

I have this in my initWithFrame: of the top view:

// Add a gesture recognizer to the image view UITapGestureRecognizer *tapGestureRecognizer = [[UITapGestureRecognizer alloc] initWithTarget:self action:@selector(imageTapped:)]; tapGestureRecognizer.cancelsTouchesInView = NO; [imageView addGestureRecognizer:tapGestureRecognizer]; [tapGestureRecognizer release]; 

and I this is my imageTapped: method:

- (void) imageTapped:(UITapGestureRecognizer *) gestureRecognizer {     // Toggle between expanding and contracting the image     if (expanded) {         [self contractAnimated:YES];         expanded = NO;         gestureRecognizer.cancelsTouchesInView = NO;         self.userInteractionEnabled = NO;         self.exclusiveTouch = NO;     }     else {         [self expandAnimated:YES];         expanded = YES;         gestureRecognizer.cancelsTouchesInView = NO;         self.userInteractionEnabled = YES;         self.exclusiveTouch = YES;     } } 

With the above code, when the image is large the buttons are inactive, when I touch the image it shrinks and the buttons become active. However, the small image doesn't receive the touches and therefore wont expand.

If I set self.userInteractionEnabled = YES in both cases, then the image expands and contracts when touched but the buttons never receive touches and act as though disabled.

Is there away to get the image to expand and contract when touched but for the buttons underneath to only receive touches if the image is in its contracted state? Am I doing something stupid here and missing something obvious?

I am going absolutely mad trying to get this to work so any help would be appreciated,

Dave

UPDATE:

For further testing I overrode the touchesBegan: and touchesCancelled: methods and called their super implementations on my view containing the UIImageView. With the code above, the touchesCancelled: is never called and the touchesBegan: is always called.

So it would appear that the view is getting the touches, they are just not passed to the view underneath.

UPDATE

Is this because of the way the responder chain works? My view hierarchy looks like this:

VC - View1      -View2       -imageView1 (has tapGestureRecogniser)       -imageView2      -View3       -button1       -button2 

I think the OS first does a hitTest as says View2 is in front so should get all the touches and these are never passed on to View3 unless userInteractions is set to NO for View2, in which case the imageView1 is also prevented from receiving touches. Is this how it works and is there a way for View2 to pass through it's touches to View3?

like image 432
Magic Bullet Dave Avatar asked Jan 26 '12 22:01

Magic Bullet Dave


People also ask

What is Cancelstouchesinview?

A Boolean value affecting whether touches are delivered to a view when a gesture is recognized.

How do I use UITapGestureRecognizer?

The iOS UITapGestureRecognizer class has a built-in way to detect a double tap on any view. All you need to do is create the recognizer, set its numberOfTapsRequired property to 2, then add it to the view you want to monitor.

How do I add tap gestures in swift 5?

Adding a Tap Gesture Recognizer to an Image View in Interface Builder. Open Main. storyboard and drag a tap gesture recognizer from the Object Library and drop it onto the image view we added earlier. The tap gesture recognizer appears in the Document Outline on the left.


1 Answers

The UIGestureRecognizer is a red herring I think. In the end to solve this I overrode the pointInside:withEvent: method of my UIView:

- (BOOL)pointInside:(CGPoint)point withEvent:(UIEvent *)event {     BOOL pointInside = NO;      if (CGRectContainsPoint(imageView.frame, point) || expanded) pointInside = YES;      return pointInside; } 

This causes the view to trap all touches if you touch either the imageView or if its expanded flag is set. If it is not expanded then only trap the touches if they are on the imageView.

By returning NO, the top level VC's View queries the rest of its view hierarchy looking for a hit.

like image 175
Magic Bullet Dave Avatar answered Sep 22 '22 17:09

Magic Bullet Dave