UIGestureRecognizer and UITableViewCell issue
Asked Answered
C

4

44

I am attaching a UISwipeGestureRecognizer to a UITableViewCell in the cellForRowAtIndexPath: method like so:

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {

    static NSString *CellIdentifier = @"Cell";

    UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:CellIdentifier];
    if (cell == nil) {
        cell = [[[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:CellIdentifier] autorelease];

        UISwipeGestureRecognizer *gesture = [[UISwipeGestureRecognizer alloc] initWithTarget:self action:@selector(didSwipe:)];
        gesture.direction = UISwipeGestureRecognizerDirectionRight;
        [cell.contentView addGestureRecognizer:gesture];
        [gesture release];
    }
    return cell;
}

However, the didSwipe method is always getting called twice on successful swipe. I initially thought this was because the gesture starts and ends, but if I log out the gestureRecognizer itself, they are both in the "Ended" state:

-(void)didSwipe:(UIGestureRecognizer *)gestureRecognizer {

    NSLog(@"did swipe called %@", gestureRecognizer);
}

Console:

2011-01-05 12:57:43.478 App[20752:207] did swipe called <UISwipeGestureRecognizer: 0x5982fa0; state = Ended; view = <UITableViewCellContentView 0x5982c30>; target= <(action=didSwipe:, target=<RootViewController 0x5e3e080>)>; direction = right>
2011-01-05 12:57:43.480 App[20752:207] did swipe called <UISwipeGestureRecognizer: 0x5982fa0; state = Ended; view = <UITableViewCellContentView 0x5982c30>; target= <(action=didSwipe:, target=<RootViewController 0x5e3e080>)>; direction = right>

I really really don't know why. I tried obviously checking for the Ended state, but that is no help as they both come in as "Ended" anyway... Any ideas?

Cloverleaf answered 5/1, 2011 at 13:4 Comment(0)
M
109

Instead of adding the gesture recognizer to the cell directly, you can add it to the tableview in viewDidLoad.

In the didSwipe-Method you can determine the affected IndexPath and cell as follows:

-(void)didSwipe:(UIGestureRecognizer *)gestureRecognizer {

  if (gestureRecognizer.state == UIGestureRecognizerStateEnded) {
        CGPoint swipeLocation = [gestureRecognizer locationInView:self.tableView];
        NSIndexPath *swipedIndexPath = [self.tableView indexPathForRowAtPoint:swipeLocation];
        UITableViewCell* swipedCell = [self.tableView cellForRowAtIndexPath:swipedIndexPath];
        // ...
  }
}
Maurist answered 5/1, 2011 at 13:41 Comment(4)
Thanks! That stopped it from firing twice! :)Cloverleaf
Thanks, this helped, but I'm seeing something weird - after swiping I need to press twice if I want to select the row. The first time does nothing, and the second calls didSelectRow... Anybody seen this?Labio
@OdedBenDov I believe it is caused by gesture recogniser,I had the strange behaviour for table view cell selection when I added tap gesture recogniser and forgot to remove in did end editing method of text field(sub view of table view cell),may be this can act as a clue to solve your issue,more details hereAschim
You should probably check to see if swipedIndexPath is nil before calling cellForRowAtIndexPath. In my case, I was typing the third (non-existent) cell and it was returning a index path of nil, which then yielded an actual UITableViewCell (in that last line).Chilpancingo
E
0

It will work with app delegate

- (void)tableView:(UITableView*)tableView willBeginEditingRowAtIndexPath:(NSIndexPath *)indexPath
{

// code

}
Eyecup answered 18/2, 2012 at 10:8 Comment(0)
C
0

I had this same problem and solved it by ticking "Scrolling Enabled" in the table view attributes.

My table view doesn't need scrolling, so it doesn't affect the app in any other way, except now I don't get the first unresponsive tap after a swipe gesture.

Chronopher answered 22/3, 2014 at 11:59 Comment(0)
A
-1

Adding gesture in AwakeFromNib method works with no problems.

class TestCell: UITableViewCell {

    override func awakeFromNib() {
        super.awakeFromNib()

        let panGesture = UIPanGestureRecognizer(target: self,
                                            action: #selector(gestureAction))
        addGestureRecognizer(panGesture)
    }

    @objc func gestureAction() {
        print("gesture action")
    }
}
Ainslie answered 29/6, 2018 at 15:35 Comment(0)

© 2022 - 2024 — McMap. All rights reserved.