UICollectionView indexPathForItemAtPoint returning

2019-04-06 12:58发布

What is the correct syntax for using UICollectionView's indexPathForItemAtPoint?

I'm using the following code to try to identify the cell that is at the centre of the UICollectionView as I scroll through it.

- (void)scrollViewDidScroll:(UIScrollView *)scrollView {

    //locate the scrollview which is in the centre
    CGPoint centerPoint = CGPointMake(self.collectionView.frame.size.width / 2, self.collectionView.frame.size.height /2);
    NSIndexPath *indexPathOfCentralCell = [self.collectionView indexPathForItemAtPoint:centerPoint];
}

However, this always gives me the same indexPath back, no matter how I scroll.

What am I doing wrong?

4条回答
爱情/是我丢掉的垃圾
2楼-- · 2019-04-06 13:19

Here is how to do it in Swift if you want to retrieve a specific UICollectionViewCell with a UILongPressGesture

@IBAction func cellAtLongPress(sender:UILongPressGestureRecognizer) {

    var location = sender.locationInView(self.view)

    // adjust the location to account for scrolling
    location.x += self.collectionView.contentOffset.x
    location.y += self.collectionView.contentOffset.y

    let indexPath = self.collectionView.indexPathForItemAtPoint(location)

    if (indexPath == nil) {
        print("not over cell")
    } else {
        // get the cell where the longPress occured
        let cell = self.collectionView.cellForItemAtIndexPath(indexPath!)!

        // you now have the the cell you pressed
    }
}
查看更多
Root(大扎)
3楼-- · 2019-04-06 13:26

Duncan, this is what it works for me in Swift:

   let location = CGPointMake(self.clvGroups.frame.size.width / 2 +
                self.clvGroups.contentOffset.x, self.clvGroups.frame.size.height / 2 + self.clvGroups.contentOffset.y);
   if let ip = self.clvGroups.indexPathForItemAtPoint(location){
                 groupToUpdate.icon = self.dataSource![ip.row].icon
            }

You have to keep in mind the scrolling view of collection view.

查看更多
Fickle 薄情
4楼-- · 2019-04-06 13:33

An example of where you'd want to do this is when you want to capture a gesture on the UICollectionView, and then do something with the specific item in the UICollectionView that falls under the location of the swipe gesture. The code below as a method in your UICollectionViewController will return the correct cell, and accounts for scrolling within the Collection View.

- (UICollectionViewCell *) cellForGesture:(id)sender
{
    UISwipeGestureRecognizer * gesture = sender;
    CGPoint point = [gesture locationInView:self.view];
    NSLog(@"Swipe location: %f, %f", point.x, point.y, nil);

    CGPoint pointInCollection = CGPointMake(point.x + self.collectionView.contentOffset.x, point.y + self.collectionView.contentOffset.y);

    NSIndexPath * indexPath = [self.collectionView indexPathForItemAtPoint:pointInCollection];
    UICollectionViewCell * cell = [self.collectionView cellForItemAtIndexPath:indexPath];
    return cell;
}

You might call the above method in a context like this (where revealSupplementalControls is a custom method on your UICollectionViewCell class):

- (IBAction) swipeLeft:(id)sender {
    UICollectionViewCell * cell = [self cellForGesture:sender];
    [cell revealSupplementalControls];
}
查看更多
爷的心禁止访问
5楼-- · 2019-04-06 13:35

You are looking up a static point in a static field of cells. Yes you will always get the same cell.

In order to deal with scrolling you need to add the scrollview's position into your check. Something like this:

- (void)scrollViewDidScroll:(UIScrollView *)scrollView {

    //locate the scrollview which is in the centre
    CGPoint centerPoint = CGPointMake(self.collectionView.frame.size.width / 2 + scrollView.contentOffset.x, self.collectionView.frame.size.height /2 + scrollView.contentOffset.y);
    NSIndexPath *indexPathOfCentralCell = [self.collectionView indexPathForItemAtPoint:centerPoint];
}
查看更多
登录 后发表回答