16

为自定义按钮处理按钮触摸事件的最佳做法是什么UITableViewCell

我的课: MyViewControllerMyCustomCell

我可以想到三个选项:

第一个选项- 将按钮作为 的属性MyCustomCell,然后在MyViewController.m 文件中添加一个目标MyViewController作为目标。

MyViewController.m 文件

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
    static NSString *CellIdentifier = @"customCell";

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

    [cell.theButton addTarget:self
                       action:@selector(theButtonTapped:)
             forControlEvents:UIControlEventTouchUpInside];
    }

    // Configure the cell...    
    [self configureCell:cell atIndexPath:indexPath];

    return cell;
}

- (void)theButtonTapped:(UIButton *)sender
{
    MyCustomCell *selectedCell = (MyCustomCell *)sender.superview;

    if (selectedCell) {
        NSIndexPath *indexPath = [self.tableView indexPathForCell:selectedCell];
        MyModel *selectedModel = [self.model objectAtIndex:indexPath.row]; 

        // do something with the model...
    }
}

第二个选项- 如果自定义单元格是在 IB 中制作的,请将 nib 文件的所有者设置为MyViewController,实现buttonTapped:方法 inMyViewController并将按钮的 Touch Up Inside 事件连接到该buttonTapped:方法。

第三个选项- 如果自定义单元格不是在 IB 中制作的,则将目标添加到MyCustomCell.m 文件中的按钮MyCustomCell作为目标。
定义一个MyCustomCellDelegateadd @property (nonatomic, assign) id<MyCustomCellDelegate> delegatetoMyCustomCell并在点击按钮时调用此委托。创建单元格并实现协议时
设置MyViewController为单元格的委托。MyCustomCellDelegate

MyCustomCell.h 文件

@class MyCustomCell;  

@protocol MyCustomCellDelegate <NSObject>
- (void)buttonTappedOnCell:(MyCustomCell *)cell;
@end

@interface MyCustomCell : UITableViewCell

@property (nonatomic, retain) UIButton *theButton;
@property (nonatomic, assign) id<MyCustomCellDelegate> delegate;

@end

MyCustomCell.m 文件

- (id)initWithStyle:(UITableViewCellStyle)style reuseIdentifier:(NSString *)reuseIdentifier
{
    self = [super initWithStyle:style reuseIdentifier:reuseIdentifier];
    if (self) {
        // Initialization code
        self.theButton = [UIButton buttonWithType:UIButtonTypeRoundedRect];
        self.theButton.frame = CGRectMake(10,10,50,30);
        [self addSubview:self.theButton];

        [self.theButton addTarget:self
                           action:@selector(theButtonTapped:)
                 forControlEvents:UIControlEventTouchUpInside];
    }
    return self;
}

- (void)theButtonTapped:(UIButton *)sender
{
    [self.delegate buttonTappedOnCell:self];
}

MyViewController.m 文件

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
    static NSString *CellIdentifier = @"customCell";

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

        cell.delegate = self;
    }

    // Configure the cell...    
    [self configureCell:cell atIndexPath:indexPath];

    return cell;
}

- (void)buttonTappedOnCell:(MyCustomCell *)selectedCell
{
    if (selectedCell) {
        NSIndexPath *indexPath = [self.tableView indexPathForCell:selectedCell];
        MyModel *selectedModel = [self.model objectAtIndex:indexPath.row];

        // do something with the model...
    }
}
4

5 回答 5

12

将单元格的行存储为tag自定义按钮的属性。

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
    // bla bla bla
    if (!cell)
    {
        //bla bla bla
        [cell.yourButton addTarget:self selector:@selector(yourButtonTapped:) forControlEvents:UIControlEventTouchUpInside];
    }
    // bla bla bla
    cell.yourButton.tag = indexPath.row;
}

-(void)yourButtonTapped:(id)sender
{
    int tag = [(UIButton *)sender tag];
    NSLog(@"tapped button in cell at row %i", tag);
}
于 2012-05-29T09:58:13.363 回答
1

从我的角度来看,使用标签会破坏代码的严格性。此外,当您有多个部分时,使用标签肯定会弄乱您的代码。

为避免此问题,您可以子类化UITableViewCell并使其拥有一个indexPath属性以让单元格知道其精确位置。

这里的另一个问题是,如果UITableView调用 API 到insertdelete行,您必须更新可见单元格的位置数据

我认为这不是最佳做法。

存在更好的方法。


当您必须在 Cell 中处理不同的触摸事件时,我强烈建议您使用 MVVM。

在这种模式下,您的 customUITableViewCell将持有一个 custom CellViewModel。此类将负责保存与单元格关联的所有数据,因此您可以检索数据并将事件处理逻辑放入单元格中。

于 2016-05-14T13:34:20.783 回答
0

Swift 3.0 解决方案

cell.btnRequest.tag = indexPath.row

cell.btnRequest.addTarget(self,action:#selector(buttonClicked(sender:)), for: .touchUpInside)

func buttonClicked(sender:UIButton) {

    let buttonRow = sender.tag
 }
于 2016-10-25T06:23:03.363 回答
0

我通过子类化 UIButton 实现了基于块的方法:

typedef void (^ActionBlock)(id sender);

@interface UIBlockButton : UIButton {
     ActionBlock _actionBlock;
 } 

-(void)handleControlEvent:(UIControlEvents)event withBlock:(ActionBlock) action;
​@end

@implementation UIBlockButton

 -(void) handleControlEvent:(UIControlEvents)event withBlock:(ActionBlock) action
    {
      _actionBlock = action;
      [self addTarget:self action:@selector(callActionBlock:) forControlEvents:event];
      }

  -(void) callActionBlock:(id)sender{
     _actionBlock(sender);
   }

@end

在 tableview 委托中:

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
    if (!cell)
    {
     cell.yourButton.tag = indexPath.row;// pass tag
     [cell.yourButton handleControlEvent:UIControlEventTouchUpInside withBlock:^(id sender) {
     // your selector action code
     NSLog(@"tapped button in cell at row %i",[(UIButton *)sender tag]);
     }];
   }  
}
于 2016-05-14T17:59:43.960 回答
0

在某些时候,您的按钮被点击,此时它是一个单元格的子视图,它是某个表格视图的子视图。

只需编写一个获取视图的方法,沿着 superview 链向上查找包含单元格,再向上查找 tableview,然后向 tableview 询问单元格的 indexPath。

这比存储包含行的标签要容易和可靠得多,因为在编辑 tableview 时不会遇到问题,而且在需要 indexPath 时,最好有代码找出它是哪个 indexPath,并且创建单元格时,不在某些完全不相关的代码中。

于 2016-05-14T18:52:41.000 回答