更改一个可重用单元格的UITableViewController会影响其他单元格

时间:2014-07-15 20:28:23

标签: ios uitableview didselectrowatindexpath reuseidentifier

我有一个非常简单的UITableViewController子类,旨在向用户显示单元格中字母表中的字符。当用户按下单元格时,将其附件类型设置为复选标记。

#import "MTGTableViewController.h"

@interface MTGTableViewController ()

@property (nonatomic, strong) NSArray *data;

@end

@implementation MTGTableViewController

- (void)viewDidLoad
{
    [super viewDidLoad];

    _data = @[@"A", @"B", @"C", @"D", @"E", @"F", @"G", @"H", @"I", @"J", @"K", @"L", @"M", @"N", @"O", @"P", @"Q", @"R", @"S", @"T", @"U", @"V", @"W", @"X", @"Y", @"Z"];
}

#pragma mark - Table view data source

- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView
{
    // Return the number of sections.
    return 1;
}

- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
{
    return _data.count;
}


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

    // Configure the cell...
    cell.textLabel.text = _data[indexPath.row];

    return cell;
}

- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath
{
    UITableViewCell *cell = [tableView cellForRowAtIndexPath:indexPath];
    cell.accessoryType = UITableViewCellAccessoryCheckmark;
}

@end

表视图工作正常。我在我的原型单元格的故事板中设置了reuseIdentifier属性,在开始选择单元格之前,它看起来都很好。

问题:当我选择任何一个单元格时,说“A”单元格时,当我向下滚动它们时,其他尚未显示的单元格也会给出复选标记。更糟糕的是,当我向上和向下滚动时,有时会删除单元格“A”上的复选标记并将其赋予单元格“B”。

4 个答案:

答案 0 :(得分:6)

这是因为表视图重用单元格的方式。调用dequeueReusableCellWithIdentifier后,您需要确保清除附件项。例如:

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

    // Configure the cell...
    cell.textLabel.text = _data[indexPath.row];
    cell.accessoryType = UITableViewCellAccessoryNone;

    return cell;
}

虽然 - 进行此更改后,您将遇到其他问题。由于此代码,单元格将“忘记”它被选中。因此,您需要实际更改设置了accessoryType的行,以检查该单元格是否被选中。

答案 1 :(得分:3)

我之前也遇到过这个问题,你需要添加另一个数组来跟踪标记的单元格。只需创建一个标记为NSIndexPath的数组。 有点像:

- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath
{
     UITableViewCell *cell = [tableView cellForRowAtIndexPath:indexPath];
     if ([arrayOfMarked containsObject:indexPath]) {
         cell.accessoryType = UITableViewCellAccessoryNone;
     } else {
         cell.accessoryType = UITableViewCellAccessoryCheckmark;
     }
}

答案 2 :(得分:1)

对于Swift 3:

我发现的一个简化版本是在" cellForRowAt"中使用swifts " .indexPathsForSelectedRows" 。功能

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

    //To keep from reusing checkmark in reusableCell
    if let selectedIndexPaths = tableView.indexPathsForSelectedRows {
        if selectedIndexPaths.contains(indexPath) {
            cell.accessoryType = .checkmark
        } else {
            cell.accessoryType = .none
        }
    }

    cell.contactsOutlet.text = namesOrganized[indexPath.section].names[indexPath.row]
    return cell
}

然后在didSelect和didDeselect:

func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {

    let cell = tableView.cellForRow(at: indexPath)
    cell?.accessoryType = UITableViewCellAccessoryType.checkmark
}

func tableView(_ tableView: UITableView, didDeselectRowAt indexPath: IndexPath) {

    let cell = tableView.cellForRow(at: indexPath)
    cell?.accessoryType = UITableViewCellAccessoryType.none
}

答案 3 :(得分:0)

在didselectRowAtIndex路径上,将索引路径附加到indexpath数组

override func tableView(tableView: UITableView, didSelectRowAtIndexPath indexPath: NSIndexPath) {


    if selectedIndexPaths.contains(indexPath) {

        let indexOfIndexPath = selectedIndexPaths.indexOf(indexPath)!
        selectedIndexPaths.removeAtIndex(indexOfIndexPath)
    } else {
        selectedIndexPaths += [indexPath]
    }

    tableView.reloadRowsAtIndexPaths([indexPath], withRowAnimation: .Automatic)

}

然后在cellForRowAtIndexPath上使用selectedIndexpaths数组来检查单元格是否被选中

 override func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {

    let mycell = tableView.dequeueReusableCellWithIdentifier("yourCell", forIndexPath: indexPath) as! YourCustomCell
 .
 .
 .
    if selectedIndexPaths.contains(indexPath){
 // do what you want to do if the cell is selected


    } else {

 // do what you want to do if the cell is not selected
    }