UIGestureRecognizer on UIImageView UIGestureRecognizer on UIImageView ios ios

UIGestureRecognizer on UIImageView


Check that userInteractionEnabled is YES on the UIImageView. Then you can add a gesture recognizer.

imageView.userInteractionEnabled = YES;UIPinchGestureRecognizer *pgr = [[UIPinchGestureRecognizer alloc]     initWithTarget:self action:@selector(handlePinch:)];pgr.delegate = self;[imageView addGestureRecognizer:pgr];[pgr release];::- (void)handlePinch:(UIPinchGestureRecognizer *)pinchGestureRecognizer{  //handle pinch...}


Yes, a UIGestureRecognizer can be added to a UIImageView. As stated in the other answer, it is very important to remember to enable user interaction on the image view by setting its userInteractionEnabled property to YES. UIImageView inherits from UIView, whose user interaction property is set to YES by default, however, UIImageView's user interaction property is set to NO by default.

From the UIImageView docs:

New image view objects are configured to disregard user events by default. If you want to handle events in a custom subclass of UIImageView, you must explicitly change the value of the userInteractionEnabled property to YES after initializing the object.

Anyway, on the the bulk of the answer. Here's an example of how to create a UIImageView with a UIPinchGestureRecognizer, a UIRotationGestureRecognizer, and a UIPanGestureRecognizer.

First, in viewDidLoad, or another method of your choice, create an image view, give it an image, a frame, and enable its user interaction. Then create the three gestures as follows. Be sure to utilize their delegate property (most likely set to self). This will be required to use multiple gestures at the same time.

- (void)viewDidLoad{    [super viewDidLoad];    // set up the image view    UIImageView *imageView = [[UIImageView alloc] initWithImage:[UIImage imageNamed:@"someImage"]];    [imageView setBounds:CGRectMake(0.0, 0.0, 120.0, 120.0)];    [imageView setCenter:self.view.center];    [imageView setUserInteractionEnabled:YES]; // <--- This is very important    // create and configure the pinch gesture    UIPinchGestureRecognizer *pinchGestureRecognizer = [[UIPinchGestureRecognizer alloc] initWithTarget:self action:@selector(pinchGestureDetected:)];    [pinchGestureRecognizer setDelegate:self];    [imageView addGestureRecognizer:pinchGestureRecognizer];    // create and configure the rotation gesture    UIRotationGestureRecognizer *rotationGestureRecognizer = [[UIRotationGestureRecognizer alloc] initWithTarget:self action:@selector(rotationGestureDetected:)];    [rotationGestureRecognizer setDelegate:self];    [imageView addGestureRecognizer:rotationGestureRecognizer];    // creat and configure the pan gesture    UIPanGestureRecognizer *panGestureRecognizer = [[UIPanGestureRecognizer alloc] initWithTarget:self action:@selector(panGestureDetected:)];    [panGestureRecognizer setDelegate:self];    [imageView addGestureRecognizer:panGestureRecognizer];    [self.view addSubview:imageView]; // add the image view as a subview of the view controllers view}

Here are the three methods that will be called when the gestures on your view are detected. Inside them, we will check the current state of the gesture, and if it is in either the began or changed UIGestureRecognizerState we will read the gesture's scale/rotation/translation property, apply that data to an affine transform, apply the affine transform to the image view, and then reset the gestures scale/rotation/translation.

- (void)pinchGestureDetected:(UIPinchGestureRecognizer *)recognizer{    UIGestureRecognizerState state = [recognizer state];    if (state == UIGestureRecognizerStateBegan || state == UIGestureRecognizerStateChanged)    {        CGFloat scale = [recognizer scale];        [recognizer.view setTransform:CGAffineTransformScale(recognizer.view.transform, scale, scale)];        [recognizer setScale:1.0];    }}- (void)rotationGestureDetected:(UIRotationGestureRecognizer *)recognizer{    UIGestureRecognizerState state = [recognizer state];    if (state == UIGestureRecognizerStateBegan || state == UIGestureRecognizerStateChanged)    {        CGFloat rotation = [recognizer rotation];        [recognizer.view setTransform:CGAffineTransformRotate(recognizer.view.transform, rotation)];        [recognizer setRotation:0];    }}- (void)panGestureDetected:(UIPanGestureRecognizer *)recognizer{    UIGestureRecognizerState state = [recognizer state];    if (state == UIGestureRecognizerStateBegan || state == UIGestureRecognizerStateChanged)    {        CGPoint translation = [recognizer translationInView:recognizer.view];        [recognizer.view setTransform:CGAffineTransformTranslate(recognizer.view.transform, translation.x, translation.y)];        [recognizer setTranslation:CGPointZero inView:recognizer.view];    }}

Finally and very importantly, you'll need to utilize the UIGestureRecognizerDelegate method gestureRecognizer: shouldRecognizeSimultaneouslyWithGestureRecognizer to allow the gestures to work at the same time. If these three gestures are the only three gestures that have this class assigned as their delegate, then you can simply return YES as shown below. However, if you have additional gestures that have this class assigned as their delegate, you may need to add logic to this method to determine which gesture is which before allowing them to all work together.

- (BOOL)gestureRecognizer:(UIGestureRecognizer *)gestureRecognizer shouldRecognizeSimultaneouslyWithGestureRecognizer:(UIGestureRecognizer *)otherGestureRecognizer{    return YES;}

Don't forget to make sure that your class conforms to the UIGestureRecognizerDelegate protocol. To do so, make sure that your interface looks something like this:

@interface MyClass : MySuperClass <UIGestureRecognizerDelegate>

If you prefer to play with the code in a working sample project yourself, the sample project I've created containing this code can be found here.


Swift 4.2

myImageView.isUserInteractionEnabled = truelet tapGestureRecognizer = UITapGestureRecognizer(target: self, action: #selector(imageTapped))tapGestureRecognizer.numberOfTapsRequired = 1myImageView.addGestureRecognizer(tapGestureRecognizer)

and when tapped:

@objc func imageTapped(_ sender: UITapGestureRecognizer) {   // do something when image tapped   print("image tapped")}