在动态 tableView 中重新加载包含 textFields 的单元格时出现奇怪的行为

Weird behavior when reloading cells containing textFields in dynamic tableView

我有一个动态 tableView,其中包含自定义单元格,每个单元格都包含一个文本字段和标签。我想要以下行为:


用户点击 "Add" 按钮,table 在索引 0 处插入一个单元格,单元格的 textField 成为第一响应者。

当用户完成输入数据时,单行文本字段退出第一响应者并隐藏自己。

然后单元格的标签变为可见,填充有用户输入,并自动调整其高度以适应输入。单元格也在此时调整大小。


我有代码在第一次添加单元格时完全执行此操作,但是当添加更多单元格时它会变得很奇怪。第二个添加的单元格 s textField 不会成为第一响应者,而是填充上一个 textField 中的文本。随后添加的单元格显示为空白,除了每 6 个空白单元格,第一个单元格的文本显示。

这是怎么回事,我该如何解决?

编辑:感谢@Rayfleck,我已经解决了第二个添加的单元格被前一个文本字段中的文本填充的问题,但是第二个单元格被添加(以及随后添加的单元格)仍然没有成为第一响应者。

这是显示此行为的演示:

这是我的代码:

#define FONT_SIZE 16.0f
#define CELL_CONTENT_WIDTH 300.0f // This will change for different size!
#define CELL_CONTENT_MARGIN_WIDTH 10.0f
#define CELL_CONTENT_MARGIN_HEIGHT 20.0f

@interface ViewController ()

@property (strong) IBOutlet UITableView *tableView;
@property (strong) NSMutableArray *notesArray;

@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];

    self.notesArray = [NSMutableArray arrayWithObjects:@"Testo 1", @"This is a long string of text to show how the cell resizes to fit a variable height label.", @"Testo 3", @"Testo 4", @"Testo 5", nil];
}

- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView {

    // Return the number of sections.
    return 1;
}

- (IBAction)addButtonTapped {

    NSMutableArray *indexPaths = [NSMutableArray array];
    [indexPaths addObject:[NSIndexPath indexPathForRow:0 inSection:0]]; // insert at first index

    [self.tableView setEditing:YES animated:NO]; // go into edit mode so cell has label hidden and textField visible

    // do the insertion
    [self.notesArray insertObject:@"" atIndex:0]; // placeholder for when user starts typing
    [self.tableView beginUpdates];
    [self.tableView insertRowsAtIndexPaths:indexPaths withRowAnimation:UITableViewRowAnimationTop];
    [self.tableView endUpdates];

    [self.tableView setEditing:NO animated:NO];
}

-(BOOL)textFieldShouldReturn:(UITextField *)textField {

    NSIndexPath *path = [NSIndexPath indexPathForRow:0 inSection:0]; // first table cell
    HC_NoteCell *cell = (HC_NoteCell *)[self.tableView cellForRowAtIndexPath:path];

    cell.noteTextField.hidden = YES;
    cell.noteLabel.text = cell.noteTextField.text;
    cell.noteLabel.hidden = NO;

    [self.notesArray replaceObjectAtIndex:0 withObject:cell.noteTextField.text];

    [textField resignFirstResponder];

    // Need to reload rows so cell height can be recalculated
    [self.tableView beginUpdates];
    [self.tableView reloadRowsAtIndexPaths:@[path] withRowAnimation:UITableViewRowAnimationAutomatic];
    [self.tableView endUpdates];

    return YES;
}

- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section {

    return self.notesArray.count;
}

- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath {

    NSString *text;
    text = [self.notesArray objectAtIndex:indexPath.row];

    CGSize constraint = CGSizeMake(CELL_CONTENT_WIDTH - (CELL_CONTENT_MARGIN_WIDTH * 2), 20000.0f);

    UIFont *font = [UIFont fontWithName:@"Helvetica Neue" size:FONT_SIZE];
    NSDictionary *attrsDictionary = [NSDictionary dictionaryWithObject:font
                                                                forKey:NSFontAttributeName];
    NSAttributedString *attributedString = [[NSAttributedString alloc] initWithString:text attributes:attrsDictionary];

    CGRect paragraphRect = [attributedString boundingRectWithSize:constraint options:(NSStringDrawingUsesLineFragmentOrigin|NSStringDrawingUsesFontLeading) context:nil];

    CGFloat height = MAX(paragraphRect.size.height, 22.0f);

    return height + (CELL_CONTENT_MARGIN_HEIGHT * 2);

}

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

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

    if (self.tableView.isEditing) {

        cell.noteLabel.hidden = YES;
        cell.noteTextField.hidden = NO;
        cell.noteTextField.delegate = self;

        [cell.noteTextField becomeFirstResponder];

    } else {

        cell.noteLabel.hidden = NO;
        cell.noteTextField.hidden = YES;
        cell.noteTextField.delegate = self;

        cell.noteLabel.text = [self.notesArray objectAtIndex:indexPath.row];
    }

    return cell;
}

@end

如果您是 reusing/dequeing 个单元格,那么在 cellForRowAtIndexPath 中,您必须为单元格的每个子视图设置新值,因为它可能包含前一个单元格的内容。

所以 cell.noteLabel.text = nil; 在所有情况下,然后在你的其他情况下,从数组中设置它的值。