检测在 UITableView 中按下了哪个 UIButton

Detecting which UIButton was pressed in a UITableView

提问人:rein 提问时间:11/26/2009 最后编辑:Imanou Petitrein 更新时间:4/23/2019 访问量:136497

问:

我有一个 5 .每个单元格都包含一个,其设置如下:UITableViewUITableViewCellsUIButton

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
     NSString *identifier = @"identifier";
     UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:identifier];
     if (cell == nil) {
         cell = [[UITableView alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:identifier];
         [cell autorelelase];

         UIButton *button = [[UIButton alloc] initWithFrame:CGRectMake(10, 5, 40, 20)];
         [button addTarget:self action:@selector(buttonPressedAction:) forControlEvents:UIControlEventTouchUpInside];
         [button setTag:1];
         [cell.contentView addSubview:button];

         [button release];
     }

     UIButton *button = (UIButton *)[cell viewWithTag:1];
     [button setTitle:@"Edit" forState:UIControlStateNormal];

     return cell;
}

我的问题是:在方法中,我怎么知道按下了哪个按钮。我考虑过使用标签,但我不确定这是最好的路线。我希望能够以某种方式将控件标记到控件上。buttonPressedAction:indexPath

- (void)buttonPressedAction:(id)sender
{
    UIButton *button = (UIButton *)sender;
    // how do I know which button sent this message?
    // processing button press for this row requires an indexPath. 
}

执行此操作的标准方法是什么?

编辑:

我通过执行以下操作解决了它。我仍然想有一个意见,这是标准的方法还是有更好的方法?

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
     NSString *identifier = @"identifier";
     UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:identifier];
     if (cell == nil) {
         cell = [[UITableView alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:identifier];
         [cell autorelelase];

         UIButton *button = [[UIButton alloc] initWithFrame:CGRectMake(10, 5, 40, 20)];
         [button addTarget:self action:@selector(buttonPressedAction:) forControlEvents:UIControlEventTouchUpInside];
         [cell.contentView addSubview:button];

         [button release];
     }

     UIButton *button = (UIButton *)[cell.contentView.subviews objectAtIndex:0];
     [button setTag:indexPath.row];
     [button setTitle:@"Edit" forState:UIControlStateNormal];

     return cell;
}

- (void)buttonPressedAction:(id)sender
{
    UIButton *button = (UIButton *)sender;
    int row = button.tag;
}

需要注意的重要一点是,我无法在创建单元格时设置标记,因为单元格可能会被取消排队。感觉很脏。一定有更好的方法。

iOS iPhone UITableView UIBubutton

评论

0赞 Erik van der Neut 5/5/2015
我认为使用您的标签解决方案没有任何问题。单元格是重复使用的,因此将标记设置为行索引是有意义的,就像您在此处所做的那样。我发现这是一个比将触摸位置转换为行索引更优雅的解决方案,如下所述。

答:

0赞 Chris 11/26/2009 #1

我总是使用标签。

您需要从那里子类化并处理按钮按下。UITableviewCell

评论

0赞 rein 11/26/2009
我不太明白怎么回事。标记属性是在单元格创建期间设置的 - 此单元格可重复用于具有相同标识符的每一行。此标记特定于泛型可重用单元格中的控件。如何使用此标签来区分以通用方式创建的单元格中的按钮?你能发布一些代码吗?
3赞 ACBurk 11/26/2009 #2

我会像你说的那样使用标签属性,像这样设置标签:

[button setTag:indexPath.row];

然后在 buttonPressedAction 中获取标签,如下所示:

((UIButton *)sender).tag

UIButton *button = (UIButton *)sender; 
button.tag;

评论

5赞 ohhorob 5/14/2010
对于带有部分的表,这种方法是完全错误的。
0赞 ACBurk 5/14/2010
不,您也可以使用一些简单的函数将该部分放在标签中。
2赞 ohhorob 5/15/2010
tag是一个整数。将索引路径编码/解码到视图标签中似乎有点笨拙。
0赞 ACBurk 5/20/2010
这是正确的,但这是一个解决方案,尽管如果我有部分,我不会使用。我想说的是,它可以使用这种方法来完成,它没有被破坏。更好、更复杂的版本将从 UITableView 中的按钮位置确定索引路径。但是,由于 rein 说他只有五个单元格(没有部分),这可能会使该方法过于复杂,并且您的初始评论和整个评论线程毫无意义。
0赞 Nir Levy 11/26/2009 #3

您可以使用标记模式:

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
     NSString *identifier = @"identifier";
     UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:identifier];
     if (cell == nil) {
         cell = [[UITableView alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:identifier];
         [cell autorelelase];

         UIButton *button = [[UIButton alloc] initWithFrame:CGRectMake(10, 5, 40, 20)];
         [button addTarget:self action:@selector(buttonPressedAction:) forControlEvents:UIControlEventTouchUpInside];
         [button setTag:[indexPath row]]; //use the row as the current tag
         [cell.contentView addSubview:button];

         [button release];
     }

     UIButton *button = (UIButton *)[cell viewWithTag:[indexPath row]]; //use [indexPath row]
     [button setTitle:@"Edit" forState:UIControlStateNormal];

     return cell;
}

- (void)buttonPressedAction:(id)sender
{
    UIButton *button = (UIButton *)sender;
    //button.tag has the row number (you can convert it to indexPath)
}

评论

0赞 rein 11/26/2009
如果单个单元格上有多个控件,如何标记控件?
0赞 rein 11/26/2009
我不确定这是否有效 - 如果为行 #1 创建单元格,那么它将获得标签 1。如果它被取消了行 #3 的队列,那么它仍将具有 1 的标记,而不是 3。
0赞 Nir Levy 11/27/2009
猜猜你对第二条评论是对的。我的错。我认为您最好的解决方案是子类化 UIButton,添加您自己的另一个或两个属性,然后在适当的情况下设置/获取它们(坚持使用代码中的 tag:1)
401赞 Vladimir 11/26/2009 #4

在 Apple 的附件示例中,使用以下方法:

[button addTarget:self action:@selector(checkButtonTapped:) forControlEvents:UIControlEventTouchUpInside];

然后在触摸处理程序中检索触摸坐标,并从该坐标计算索引路径:

- (void)checkButtonTapped:(id)sender
{
    CGPoint buttonPosition = [sender convertPoint:CGPointZero toView:self.tableView];
    NSIndexPath *indexPath = [self.tableView indexPathForRowAtPoint:buttonPosition];
    if (indexPath != nil)
    {
     ...
    }
}

评论

0赞 rein 11/26/2009
是的,这就是我确定的(见我的编辑)。我同意你的看法,这不是最佳的。
2赞 Vladimir 5/14/2010
但是你自己将 UIButton 添加到 UITableViewCell,因此你必须与创建单元时执行的操作保持一致。虽然这种方法看起来并不优雅,但我不得不承认
1赞 raidfive 5/21/2010
对于第一个解决方案,您需要获取 [[button superview] superview],因为第一个 superview 调用将为您提供 contentView,最后第二个调用将为您提供 UITableViewCell。如果要添加/删除单元格,则第二种解决方案效果不佳,因为它会使行索引失效。因此,我按照概述选择了第一个解决方案,并且效果很好。
3赞 Jacob Lyles 8/12/2010
这将可靠地挑选出拥有按钮的单元格:UIView *view = button;而 (![视图 isKindOfClass:[UITableViewCell 类]]){ view = [view superview]}
1赞 bandw 12/1/2014
使用时有一个陷阱:[button addTarget:self action:@selector(checkButtonTapped:) forControlEvents:UIControlEventTouchUpInside];因为 addTarget:action:forControlEvents: 会在滚动表格时添加多个重复的目标和动作,所以不会删除之前的目标和动作,所以点击按钮时会多次调用 checkButtonTapped: 方法。最好先删除目标和操作,然后再添加它们
3赞 Eld 11/27/2009 #5

虽然我喜欢标签的方式......如果您出于某种原因不想使用标签, 您可以创建预制按钮的成员:NSArray

NSArray* buttons ;

然后在呈现 tableView 之前创建这些按钮,并将它们推送到数组中。

然后在函数内部,您可以执行以下操作:tableView:cellForRowAtIndexPath:

UIButton* button = [buttons objectAtIndex:[indexPath row] ] ;
[cell.contentView addSubview:button];

然后在函数中,你可以做buttonPressedAction:

- (void)buttonPressedAction:(id)sender {
   UIButton* button = (UIButton*)sender ;
   int row = [buttons indexOfObject:button] ;
   // Do magic
}
6赞 Alpinista 12/11/2009 #6

在其他地方找到了解决这个问题的好方法,没有弄乱按钮上的标签:

- (void)buttonPressedAction:(id)sender {

    NSSet *touches = [event allTouches];
    UITouch *touch = [touches anyObject];
    CGPoint currentTouchPosition = [touch locationInView:self.tableView];
    NSIndexPath *indexPath = [self.tableView indexPathForRowAtPoint: currentTouchPosition];

    // do stuff with the indexPath...
}

评论

5赞 Nick Ludlam 1/27/2010
在此示例中,不清楚从哪里获取“event”对象。
0赞 raidfive 5/21/2010
这是我选择的解决方案。添加/删除行时,使用标记是不可预测的,因为它们的索引会更改。也
0赞 KPM 10/23/2012
@NickLudlam:方法名称可能不是 but 。buttonPressedAction:buttonPressedAction:forEvent:
48赞 Cocoanut 1/6/2010 #7

我发现使用超级视图的超级视图来获取对单元格索引路径的引用的方法非常有效。感谢 iphonedevbook.com (macnsmith) 的提示链接文本

-(void)buttonPressed:(id)sender {
 UITableViewCell *clickedCell = (UITableViewCell *)[[sender superview] superview];
 NSIndexPath *clickedButtonPath = [self.tableView indexPathForCell:clickedCell];
...

}

评论

0赞 software evolved 5/10/2011
Cocoanut,你的代码片段为我指明了正确的方向,让我自己在这个问题上的变化。谢谢!如果其他人需要它,我的特殊情况是该按钮位于作为页脚的一部分显示的自定义单元格中。我将在下面添加代码
0赞 Jon Schneider 6/16/2013
如果您(Stackoverflow 读取器)尝试此操作,但它对您不起作用,请检查在您的实现中 UIButton 是否实际上是 UITableViewCell 的孙子。在我的实现中,我的 UIButton 是我的 UITableViewCell 的直接子级,所以我需要在 Cocoanut 的代码中取出一个“superview”,然后它就可以工作了。
29赞 Kenrik March 7/19/2013
这是非常非常错误的,并且在较新版本的操作系统中被破坏了。 不要走你不拥有的超级树。
2赞 Jon Schneider 10/5/2013
这在 iOS 6 下对我有用,但在 iOS 7 中被破坏了。看来@KenrikMarch是有道理的!
3赞 CW0007007 1/21/2014
在 iOS 7 中,它又提升了 1 个超级视图。例如:[[[sender superview] superview] superView];
0赞 Michael Morrison 4/23/2010 #8

我错过了什么吗?你不能只使用发送器来识别按钮吗?发件人将为您提供如下信息:

<UIButton: 0x4b95c10; frame = (246 26; 30 30); opaque = NO; tag = 104; layer = <CALayer: 0x4b95be0>>

然后,如果要更改按钮的属性,请说出您刚刚告诉发件人的背景图像:

[sender setBackgroundImage:[UIImage imageNamed:@"new-image.png"] forState:UIControlStateNormal];

如果你需要标签,那么 ACBurk 的方法就可以了。

评论

1赞 ohhorob 5/14/2010
他们正在寻找与按钮相关的“对象”
0赞 ohhorob 5/14/2010 #9
// how do I know which button sent this message?
// processing button press for this row requires an indexPath.

实际上很简单:

- (void)buttonPressedAction:(id)sender
{
    UIButton *button = (UIButton *)sender;
    CGPoint rowButtonCenterInTableView = [[rowButton superview] convertPoint:rowButton.center toView:self.tableView];
    NSIndexPath *indexPath = [self.tableView indexPathForRowAtPoint:rowButtonCenterInTableView];
    MyTableViewItem *rowItem = [self.itemsArray objectAtIndex:indexPath.row];
    // Now you're good to go.. do what the intention of the button is, but with
    // the context of the "row item" that the button belongs to
    [self performFooWithItem:rowItem];
}

对我来说效果很好:P

如果要调整目标操作设置,可以在方法中包含 Event 参数,然后使用该事件的触摸来解析触摸的坐标。坐标仍然需要在触摸视图边界中解析,但对于某些人来说,这似乎更容易。

0赞 rajesh 12/7/2010 #10

创建一个 nsmutable 数组并将所有按钮放在该数组中 usint[array addObject:yourButton];

在按钮按下方法中

-

 (void)buttonPressedAction:(id)sender
{
    UIButton *button = (UIButton *)sender;

for(int i=0;i<[yourArray count];i++){

if([buton isEqual:[yourArray objectAtIndex:i]]){

//here write wat u need to do

}
}
1赞 user366584 3/7/2011 #11

它也对我有用,谢谢@Cocoanut

我发现使用超级视图的超级视图来获取对单元格索引路径的引用的方法非常有效。感谢 iphonedevbook.com (macnsmith) 的提示链接文本

-(void)buttonPressed:(id)sender {
 UITableViewCell *clickedCell = (UITableViewCell *)[[sender superview] superview];
 NSIndexPath *clickedButtonPath = [self.tableView indexPathForCell:clickedCell];
...

}
0赞 software evolved 5/10/2011 #12

当按钮位于表格的页脚时,Cocoanuts 答案略有不同(这有助于我解决这个问题)(这会阻止您找到“点击的单元格”):

-(IBAction) buttonAction:(id)sender;
{
    id parent1 = [sender superview];   // UiTableViewCellContentView
    id parent2 = [parent1 superview];  // custom cell containing the content view
    id parent3 = [parent2 superview];  // UITableView containing the cell
    id parent4 = [parent3 superview];  // UIView containing the table

    UIView *myContentView = (UIView *)parent1;
    UITableViewCell *myTableCell = (UITableViewCell *)parent2;
    UITableView *myTable = (UITableView *)parent3;
    UIView *mainView = (UIView *)parent4;

    CGRect footerViewRect = myTableCell.frame;
    CGRect rect3 = [myTable convertRect:footerViewRect toView:mainView];    

    [cc doSomethingOnScreenAtY:rect3.origin.y];
}
2赞 brian.clear 9/11/2012 #13

处理部分 - 我将 NSIndexPath 存储在自定义 UITableViewCell 中

在 CLKIndexPricesHEADERTableViewCell.xib 中

IN IB 将 UIButton 添加到 XIB - 不要添加动作!

添加出口@property(保留,非原子) IBOutlet UIButton *buttonIndexSectionClose;

不要 CTRL+DRAG IB 中的操作(在下面的代码中完成)

@interface CLKIndexPricesHEADERTableViewCell : UITableViewCell
...
@property (retain, nonatomic) IBOutlet UIButton *buttonIndexSectionClose;
@property (nonatomic, retain) NSIndexPath * indexPathForCell;
@end

在 viewForHeaderInSection 中(也应该适用于 cellForRow...。等,如果你的表只有 1 个部分)

- viewForHeaderInSection is called for each section 1...2...3
- get the cell CLKIndexPricesHEADERTableViewCell 
- getTableRowHEADER just does the normal dequeueReusableCellWithIdentifier
- STORE the indexPath IN the UITableView cell
- indexPath.section = (NSInteger)section
- indexPath.row = 0 always (we are only interested in sections)

- (UIView *) tableView:(UITableView *)tableView1 viewForHeaderInSection:(NSInteger)section {


    //Standard method for getting a UITableViewCell
    CLKIndexPricesHEADERTableViewCell * cellHEADER = [self getTableRowHEADER];

...使用该部分获取单元格的数据

...填写

   indexName        = ffaIndex.routeCode;
   indexPrice       = ffaIndex.indexValue;

   //

   [cellHEADER.buttonIndexSectionClose addTarget:self
                                          action:@selector(buttonDELETEINDEXPressedAction:forEvent:)
                                forControlEvents:UIControlEventTouchUpInside];


   cellHEADER.indexPathForCell = [NSIndexPath indexPathForRow:0 inSection:section];


    return cellHEADER;
}

USER 按下 Section 标题上的 DELETE 按钮,这将调用

- (void)buttonDELETEINDEXPressedAction:(id)sender forEvent:(UIEvent *)event
{
    NSLog(@"%s", __PRETTY_FUNCTION__);


    UIView *  parent1 = [sender superview];   // UiTableViewCellContentView
    //UIView *myContentView = (UIView *)parent1;

    UIView *  parent2 = [parent1 superview];  // custom cell containing the content view
    //UIView *  parent3 = [parent2 superview];  // UITableView containing the cell
    //UIView *  parent4 = [parent3 superview];  // UIView containing the table


    if([parent2 isMemberOfClass:[CLKIndexPricesHEADERTableViewCell class]]){
        CLKIndexPricesHEADERTableViewCell *myTableCell = (CLKIndexPricesHEADERTableViewCell *)parent2;

        //UITableView *myTable = (UITableView *)parent3;
        //UIView *mainView = (UIView *)parent4;

        NSLog(@"%s indexPath.section,row[%d,%d]", __PRETTY_FUNCTION__, myTableCell.indexPathForCell.section,myTableCell.indexPathForCell.row);

        NSString *key = [self.sortedKeysArray objectAtIndex:myTableCell.indexPathForCell.section];
        if(key){
            NSLog(@"%s DELETE object at key:%@", __PRETTY_FUNCTION__,key);
            self.keyForSectionIndexToDelete = key;
            self.sectionIndexToDelete = myTableCell.indexPathForCell.section;

            UIAlertView *alertView = [[UIAlertView alloc] initWithTitle:@"Remove Index"
                                                                message:@"Are you sure"
                                                               delegate:self
                                                      cancelButtonTitle:@"No"
                                                      otherButtonTitles:@"Yes", nil];
            alertView.tag = kALERTVIEW_REMOVE_ONE_INDEX;
            [alertView show];
            [alertView release];
            //------
        }else{
            NSLog(@"ERROR: [%s] key is nil for section:%d", __PRETTY_FUNCTION__,myTableCell.indexPathForCell.section);
        }

    }else{
        NSLog(@"ERROR: [%s] CLKIndexPricesHEADERTableViewCell not found", __PRETTY_FUNCTION__);
    }
}

在此示例中,我添加了一个“删除”按钮,因此应显示 UIAlertView 以确认它

我将部分和键存储到字典中,在 VC 中以 ivar 形式存储有关该部分的信息

- (void)alertView:(UIAlertView *)alertView clickedButtonAtIndex:(NSInteger)buttonIndex {
   if(alertView.tag == kALERTVIEW_REMOVE_ONE_INDEX){
        if(buttonIndex==0){
            //NO
            NSLog(@"[%s] BUTTON:%d", __PRETTY_FUNCTION__,buttonIndex);
            //do nothing
        }
        else if(buttonIndex==1){
            //YES
            NSLog(@"[%s] BUTTON:%d", __PRETTY_FUNCTION__,buttonIndex);
            if(self.keyForSectionIndexToDelete != nil){

                //Remove the section by key
                [self.indexPricesDictionary removeObjectForKey:self.keyForSectionIndexToDelete];

                //sort the keys so sections appear alphabetically/numbericsearch (minus the one we just removed)
                [self updateTheSortedKeysArray];                

                //Delete the section from the table using animation
                [self.tableView beginUpdates];

                [self.tableView deleteSections:[NSIndexSet indexSetWithIndex:self.sectionIndexToDelete]
                              withRowAnimation:UITableViewRowAnimationAutomatic];
                [self.tableView endUpdates];

                //required to trigger refresh of myTableCell.indexPathForCell else old values in UITableViewCells
                [self.tableView reloadData];
            }else{
                NSLog(@"ERROR: [%s] OBJECT is nil", __PRETTY_FUNCTION__);
            }
        }
        else {
            NSLog(@"ERROR: [%s] UNHANDLED BUTTON:%d", __PRETTY_FUNCTION__,buttonIndex);
        }
    }else {
        NSLog(@"ERROR: [%s] unhandled ALERTVIEW TAG:%d", __PRETTY_FUNCTION__,alertView.tag);
    }
}
43赞 Chris Schwerdt 9/17/2012 #14

这是我是如何做到的。简单明了:

- (IBAction)buttonTappedAction:(id)sender
{
    CGPoint buttonPosition = [sender convertPoint:CGPointZero
                                           toView:self.tableView];
    NSIndexPath *indexPath = [self.tableView indexPathForRowAtPoint:buttonPosition];
    ...
}

评论

2赞 Jakob W 4/10/2013
更简单:用代替CGPointZeroCGPointMake(0, 0) ;-)
0赞 Francisco Romero 10/13/2016
易于使用。此外,很容易将其翻译成 Swift 3。你是最好的:)
0赞 Rutger Huijsmans 10/17/2016
在下面将其翻译成 Swift。我能找到的最简单的解决方案。谢谢克里斯!
0赞 piyush Bageria 10/5/2012 #15

这很简单;制作一个自定义单元格并取一个按钮的出口

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
    {
         NSString *identifier = @"identifier";
        customCell *cell = [tableView dequeueReusableCellWithIdentifier:identifier];

    cell.yourButton.tag = indexPath.Row;

- (void)buttonPressedAction:(id)sender

将上述方法中的 ID 更改为(UIButton *)

您可以通过执行 sender.tag 来获取正在点击的按钮的值。

5赞 magno cardona 12/10/2012 #16

如何像使用运行时注入一样发送信息。NSIndexPathUIButton

1) 导入时需要运行时

2)添加静态常数

3) 在运行时使用以下命令添加到您的按钮:NSIndexPath

(void)setMetaData:(id)target withObject:(id)newObj

4)按下按钮时,使用以下方法获取元数据:

(id)元数据:(id)目标

享受

    #import <objc/runtime.h>
    static char const * const kMetaDic = "kMetaDic";


    #pragma mark - Getters / Setters

- (id)metaData:(id)target {
    return objc_getAssociatedObject(target, kMetaDic);
}

- (void)setMetaData:(id)target withObject:(id)newObj {
    objc_setAssociatedObject(target, kMetaDic, newObj, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}



    #On the cell constructor
    - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
    {
    ....
    cell = [tableView dequeueReusableCellWithIdentifier:CellIdentifier];
    ....
    [btnSocial addTarget:self
                                   action:@selector(openComments:)
                         forControlEvents:UIControlEventTouchUpInside];

    #add the indexpath here or another object
    [self setMetaData:btnSocial withObject:indexPath];

    ....
    }



    #The action after button been press:

    - (IBAction)openComments:(UIButton*)sender{

        NSIndexPath *indexPath = [self metaData:sender];
        NSLog(@"indexPath: %d", indexPath.row);

        //Reuse your indexpath Now
    }

评论

1赞 Neil 9/20/2013
如果重新排列表格或删除了一行,则这将不起作用。
0赞 Jerome Chan Yeow Heong 10/6/2013 #17

子类化按钮以存储所需的值,也许创建一个协议(ControlWithData 或其他东西)。在将按钮添加到表视图单元格时设置该值。在 Touch up 事件中,查看发送方是否遵守协议并提取数据。我通常存储对表视图单元格上呈现的实际对象的引用。

2赞 mmmanishs 10/19/2013 #18
A better way would be to subclass your button and add a indexPath property to it.

//Implement a subclass for UIButton.

@interface NewButton:UIButton
@property(nonatomic, strong) NSIndexPath *indexPath;


Make your button of type NewButton in the XIB or in the code whereever you are initializing them.

Then in the cellForRowAtIndexPath put the following line of code.

button.indexPath = indexPath;

return cell; //As usual



Now in your IBAction

-(IBAction)buttonClicked:(id)sender{
   NewButton *button = (NewButton *)sender;

//Now access the indexPath by buttons property..

   NSIndexPath *indexPath = button.indexPath; //:)
}

评论

0赞 John Gibb 6/19/2014
这有点错误,因为如果调用 deleteRowsAtIndexPaths,单元格的 indexPath 可能会更改。
0赞 mmmanishs 10/3/2014
deleteRowsAtIndexPaths 将导致再次调用 cellForRowAtIndexPath。然后按钮将具有新的正确 indexPaths。
5赞 dennis 12/12/2014 #19

To do (@Vladimir) 的答案是 Swift:

var buttonPosition = sender.convertPoint(CGPointZero, toView: self.tableView)
var indexPath = self.tableView.indexPathForRowAtPoint(buttonPosition)!

虽然检查给了我手指...“NSIndexPath 不是 NSString 的子类型”indexPath != nil

0赞 Gaurav 2/6/2015 #20

请注意,我在这里使用自定义单元格,此代码非常适合我

 @IBAction func call(sender: UIButton)
    {
        var contentView = sender.superview;
        var cell = contentView?.superview as EmployeeListCustomCell
        if (!(cell.isKindOfClass(EmployeeListCustomCell)))
        {
            cell = (contentView?.superview)?.superview as EmployeeListCustomCell
        }

        let phone = cell.lblDescriptionText.text!
        //let phone = detailObject!.mobile!
        let url:NSURL = NSURL(string:"tel://"+phone)!;
        UIApplication.sharedApplication().openURL(url);
    }
5赞 Ankit Bansal 3/31/2015 #21
func buttonAction(sender:UIButton!)
    {
        var position: CGPoint = sender.convertPoint(CGPointZero, toView: self.tablevw)
        let indexPath = self.tablevw.indexPathForRowAtPoint(position)
        let cell: TableViewCell = tablevw.cellForRowAtIndexPath(indexPath!) as TableViewCell
        println(indexPath?.row)
        println("Button tapped")
    }
0赞 Lukesivi 11/7/2015 #22

SWIFT 2 更新

以下是如何找出点击了哪个按钮 + 从该按钮将数据发送到另一个 ViewController,因为我认为这是大多数人的重点!indexPath.row

@IBAction func yourButton(sender: AnyObject) {


     var position: CGPoint = sender.convertPoint(CGPointZero, toView: self.tableView)
        let indexPath = self.tableView.indexPathForRowAtPoint(position)
        let cell: UITableViewCell = tableView.cellForRowAtIndexPath(indexPath!)! as
        UITableViewCell
        print(indexPath?.row)
        print("Tap tap tap tap")

    }

对于那些使用 ViewController 类并添加了 tableView 的用户,我使用的是 ViewController 而不是 TableViewController,因此我手动添加了 tableView 以访问它。

以下是在点击该按钮并传递单元格的 indexPath.row

@IBAction func moreInfo(sender: AnyObject) {

    let yourOtherVC = self.storyboard!.instantiateViewControllerWithIdentifier("yourOtherVC") as! YourOtherVCVIewController



    var position: CGPoint = sender.convertPoint(CGPointZero, toView: self.tableView)
    let indexPath = self.tableView.indexPathForRowAtPoint(position)
    let cell: UITableViewCell = tableView.cellForRowAtIndexPath(indexPath!)! as
    UITableViewCell
    print(indexPath?.row)
    print("Button tapped")


    yourOtherVC.yourVarName = [self.otherVCVariable[indexPath!.row]]

    self.presentViewController(yourNewVC, animated: true, completion: nil)

}
0赞 Rutger Huijsmans 10/17/2016 #23

Chris Schwerdt 的解决方案,但在 Swift 中对我有用:

@IBAction func rateButtonTapped(sender: UIButton) {
    let buttonPosition : CGPoint = sender.convertPoint(CGPointZero, toView: self.ratingTableView)
    let indexPath : NSIndexPath = self.ratingTableView.indexPathForRowAtPoint(buttonPosition)!

    print(sender.tag)
    print(indexPath.row)
}
0赞 erkanyildiz 10/27/2016 #24

此问题分为两部分:

1) 获取包含按下的 UIButton 的 UITableViewCell 的索引路径

有一些建议,例如:

  • 使用索引路径的值更新 's in 方法。这不是一个好的解决方案,因为它需要不断更新,并且不适用于具有多个部分的表视图。UIButtontagcellForRowAtIndexPath:rowtag

  • 将属性添加到自定义单元格并更新它,而不是 in 方法。这解决了多个部分的问题,但仍然不好,因为它需要始终更新。NSIndexPathUIButtontagcellForRowAtIndexPath:

  • 在创建自定义单元格时,在自定义单元格中保留对父单元格的弱引用,并使用方法获取索引路径。似乎好一点,不需要更新方法中的任何内容,但在创建自定义单元格时仍然需要设置弱引用。UITableViewindexPathForCell:cellForRowAtIndexPath:

  • 使用 cell 的属性获取对父级的引用。无需向自定义单元格添加任何属性,也无需在创建/以后设置/更新任何内容。但 cell 取决于 iOS 实现细节。所以不能直接使用。superViewUITableViewsuperView

但这可以使用简单的循环来实现,因为我们确信有问题的单元格必须位于 UITableView 中:

UIView* view = self;
while (view && ![view isKindOfClass:UITableView.class])
    view = view.superview;
UITableView* parentTableView = (UITableView*)view;

因此,可以将这些建议组合成一个简单而安全的自定义单元格方法来获取索引路径:

- (NSIndexPath *)indexPath
{
    UIView* view = self;

    while (view && ![view isKindOfClass:UITableView.class])
        view = view.superview;

    return [(UITableView*)view indexPathForCell:self];
}

从现在开始,此方法可用于检测哪个被按下。UIButton

2) 通知其他方有关按钮按下事件的信息

在内部知道哪个自定义单元格中按下了具有确切索引路径的自定义单元格后,需要将此信息发送给其他方(很可能是处理 的视图控制器)。因此,此按钮单击事件可以在与 UITableView 委托方法类似的抽象和逻辑级别中进行处理。UIButtonUITableViewdidSelectRowAtIndexPath:

为此,可以使用两种方法:

a) 委派:自定义单元可以具有属性,并且可以定义协议。当按下按钮时,它只是在其属性上执行其委托方法。但是,在创建自定义单元格时,需要为每个自定义单元设置此属性。作为替代方案,自定义单元格也可以选择在其父表视图上执行其委托方法。delegatedelegatedelegatedelegate

b) 通知中心:自定义单元格可以定义自定义通知名称,并将此通知与对象中提供的索引路径和父表视图信息一起发布。无需为每个单元格设置任何内容,只需为自定义单元格的通知添加一个观察者就足够了。userInfo

0赞 Ben Ong 7/24/2017 #25

我使用了一个子类的解决方案,我想我应该在这里分享它,Swift 中的代码:UIButton

class ButtonWithIndexPath : UIButton {
    var indexPath:IndexPath?
}

然后记得更新它的 indexPathcellForRow(at:)

func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {

    let returnCell = tableView.dequeueReusableCell(withIdentifier: "cellWithButton", for: indexPath) as! cellWithButton
    ...
    returnCell.button.indexPath = IndexPath
    returnCell.button.addTarget(self, action:#selector(cellButtonPressed(_:)), for: .touchUpInside)

    return returnCell
}

因此,在响应按钮的事件时,您可以像这样使用它

func cellButtonPressed(_ sender:UIButton) {
    if sender is ButtonWithIndexPath {
        let button = sender as! ButtonWithIndexPath
        print(button.indexPath)
    }
}
8赞 Imanou Petit 8/8/2018 #26

使用 Swift 4.2 和 iOS 12,您可以选择以下 5 个完整示例之一来解决您的问题。


#1.使用 's 和 'sUIViewconvert(_:to:)UITableViewindexPathForRow(at:)

import UIKit

private class CustomCell: UITableViewCell {

    let button = UIButton(type: .system)

    override init(style: UITableViewCell.CellStyle, reuseIdentifier: String?) {
        super.init(style: style, reuseIdentifier: reuseIdentifier)

        button.setTitle("Tap", for: .normal)
        contentView.addSubview(button)

        button.translatesAutoresizingMaskIntoConstraints = false
        button.centerXAnchor.constraint(equalTo: contentView.centerXAnchor).isActive = true
        button.centerYAnchor.constraint(equalTo: contentView.centerYAnchor).isActive = true
        button.topAnchor.constraint(equalToSystemSpacingBelow: contentView.topAnchor, multiplier: 1).isActive = true
        button.leadingAnchor.constraint(greaterThanOrEqualToSystemSpacingAfter: contentView.leadingAnchor, multiplier: 1).isActive = true
    }

    required init?(coder aDecoder: NSCoder) {
        fatalError("init(coder:) has not been implemented")
    }

}
import UIKit

class TableViewController: UITableViewController {

    override func viewDidLoad() {
        super.viewDidLoad()
        tableView.register(CustomCell.self, forCellReuseIdentifier: "CustomCell")
    }

    override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        return 3
    }

    override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        let cell = tableView.dequeueReusableCell(withIdentifier: "CustomCell", for: indexPath) as! CustomCell
        cell.button.addTarget(self, action: #selector(customCellButtonTapped), for: .touchUpInside)
        return cell
    }

    @objc func customCellButtonTapped(_ sender: UIButton) {
        let point = sender.convert(CGPoint.zero, to: tableView)
        guard let indexPath = tableView.indexPathForRow(at: point) else { return }
        print(indexPath)
    }

}

#2.使用 's 和 's(替代)UIViewconvert(_:to:)UITableViewindexPathForRow(at:)

这是上一个示例的替代方法,在该示例中,我们传递给 中的参数。这样,如果第一个响应者没有实现该操作,它将被发送到响应者链中的下一个响应者,直到找到正确的实现。niltargetaddTarget(_:action:for:)

import UIKit

private class CustomCell: UITableViewCell {

    let button = UIButton(type: .system)

    override init(style: UITableViewCell.CellStyle, reuseIdentifier: String?) {
        super.init(style: style, reuseIdentifier: reuseIdentifier)

        button.setTitle("Tap", for: .normal)
        button.addTarget(nil, action: #selector(TableViewController.customCellButtonTapped), for: .touchUpInside)
        contentView.addSubview(button)

        button.translatesAutoresizingMaskIntoConstraints = false
        button.centerXAnchor.constraint(equalTo: contentView.centerXAnchor).isActive = true
        button.centerYAnchor.constraint(equalTo: contentView.centerYAnchor).isActive = true
        button.topAnchor.constraint(equalToSystemSpacingBelow: contentView.topAnchor, multiplier: 1).isActive = true
        button.leadingAnchor.constraint(greaterThanOrEqualToSystemSpacingAfter: contentView.leadingAnchor, multiplier: 1).isActive = true
    }

    required init?(coder aDecoder: NSCoder) {
        fatalError("init(coder:) has not been implemented")
    }

}
import UIKit

class TableViewController: UITableViewController {

    override func viewDidLoad() {
        super.viewDidLoad()
        tableView.register(CustomCell.self, forCellReuseIdentifier: "CustomCell")
    }

    override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        return 3
    }

    override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        let cell = tableView.dequeueReusableCell(withIdentifier: "CustomCell", for: indexPath) as! CustomCell
        return cell
    }

    @objc func customCellButtonTapped(_ sender: UIButton) {
        let point = sender.convert(CGPoint.zero, to: tableView)
        guard let indexPath = tableView.indexPathForRow(at: point) else { return }
        print(indexPath)
    }

}

#3.使用 's 和 delegate 模式UITableViewindexPath(for:)

在此示例中,我们将视图控制器设置为单元格的委托。当点击单元格的按钮时,它会触发对委托的相应方法的调用。

import UIKit

protocol CustomCellDelegate: AnyObject {
    func customCellButtonTapped(_ customCell: CustomCell)
}

class CustomCell: UITableViewCell {

    let button = UIButton(type: .system)
    weak var delegate: CustomCellDelegate?

    override init(style: UITableViewCell.CellStyle, reuseIdentifier: String?) {
        super.init(style: style, reuseIdentifier: reuseIdentifier)

        button.setTitle("Tap", for: .normal)
        button.addTarget(self, action: #selector(buttonTapped), for: .touchUpInside)
        contentView.addSubview(button)

        button.translatesAutoresizingMaskIntoConstraints = false
        button.centerXAnchor.constraint(equalTo: contentView.centerXAnchor).isActive = true
        button.centerYAnchor.constraint(equalTo: contentView.centerYAnchor).isActive = true
        button.topAnchor.constraint(equalToSystemSpacingBelow: contentView.topAnchor, multiplier: 1).isActive = true
        button.leadingAnchor.constraint(greaterThanOrEqualToSystemSpacingAfter: contentView.leadingAnchor, multiplier: 1).isActive = true
    }

    required init?(coder aDecoder: NSCoder) {
        fatalError("init(coder:) has not been implemented")
    }

    @objc func buttonTapped(sender: UIButton) {
        delegate?.customCellButtonTapped(self)
    }

}
import UIKit

class TableViewController: UITableViewController, CustomCellDelegate {

    override func viewDidLoad() {
        super.viewDidLoad()
        tableView.register(CustomCell.self, forCellReuseIdentifier: "CustomCell")
    }

    override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        return 3
    }

    override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        let cell = tableView.dequeueReusableCell(withIdentifier: "CustomCell", for: indexPath) as! CustomCell
        cell.delegate = self
        return cell
    }

    // MARK: - CustomCellDelegate

    func customCellButtonTapped(_ customCell: CustomCell) {
        guard let indexPath = tableView.indexPath(for: customCell) else { return }
        print(indexPath)
    }

}

#4.使用 ' 和闭包进行委派UITableViewindexPath(for:)

这是上一个示例的替代方法,在前面的示例中,我们使用闭包而不是协议委托声明来处理按钮点击。

import UIKit

class CustomCell: UITableViewCell {

    let button = UIButton(type: .system)
    var buttontappedClosure: ((CustomCell) -> Void)?

    override init(style: UITableViewCell.CellStyle, reuseIdentifier: String?) {
        super.init(style: style, reuseIdentifier: reuseIdentifier)

        button.setTitle("Tap", for: .normal)
        button.addTarget(self, action: #selector(buttonTapped), for: .touchUpInside)
        contentView.addSubview(button)

        button.translatesAutoresizingMaskIntoConstraints = false
        button.centerXAnchor.constraint(equalTo: contentView.centerXAnchor).isActive = true
        button.centerYAnchor.constraint(equalTo: contentView.centerYAnchor).isActive = true
        button.topAnchor.constraint(equalToSystemSpacingBelow: contentView.topAnchor, multiplier: 1).isActive = true
        button.leadingAnchor.constraint(greaterThanOrEqualToSystemSpacingAfter: contentView.leadingAnchor, multiplier: 1).isActive = true
    }

    required init?(coder aDecoder: NSCoder) {
        fatalError("init(coder:) has not been implemented")
    }

    @objc func buttonTapped(sender: UIButton) {
        buttontappedClosure?(self)
    }

}
import UIKit

class TableViewController: UITableViewController {

    override func viewDidLoad() {
        super.viewDidLoad()
        tableView.register(CustomCell.self, forCellReuseIdentifier: "CustomCell")
    }

    override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        return 3
    }

    override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        let cell = tableView.dequeueReusableCell(withIdentifier: "CustomCell", for: indexPath) as! CustomCell
        cell.buttontappedClosure = { [weak tableView] cell in
            guard let indexPath = tableView?.indexPath(for: cell) else { return }
            print(indexPath)
        }
        return cell
    }

}

#5.使用 's 和 'sUITableViewCellaccessoryTypeUITableViewDelegatetableView(_:accessoryButtonTappedForRowWith:)

如果你的按钮是 的标准配件控件,任何点击它都会触发对 的调用 ,从而允许您获取相关的索引路径。UITableViewCellUITableViewDelegatetableView(_:accessoryButtonTappedForRowWith:)

import UIKit

private class CustomCell: UITableViewCell {

    override init(style: UITableViewCell.CellStyle, reuseIdentifier: String?) {
        super.init(style: style, reuseIdentifier: reuseIdentifier)
        accessoryType = .detailButton
    }

    required init?(coder aDecoder: NSCoder) {
        fatalError("init(coder:) has not been implemented")
    }

}
import UIKit

class TableViewController: UITableViewController {

    override func viewDidLoad() {
        super.viewDidLoad()
        tableView.register(CustomCell.self, forCellReuseIdentifier: "CustomCell")
    }

    override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        return 3
    }

    override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        let cell = tableView.dequeueReusableCell(withIdentifier: "CustomCell", for: indexPath) as! CustomCell
        return cell
    }

    override func tableView(_ tableView: UITableView, accessoryButtonTappedForRowWith indexPath: IndexPath) {
        print(indexPath)
    }

}