【问题标题】:UITableView cells containing a UISwitch update inconsistently包含 UISwitch 的 UITableView 单元格更新不一致
【发布时间】:2018-11-13 01:51:18
【问题描述】:

我有一个UITableView,其中每个单元格都包含一个UISwitch。该开关应该将单元格中命名的内容切换为活动或不活动,并且当用户切换开关时它可以完美地工作。

但是,我想到了在选择整个表格行时也进行切换的好主意。理论上很简单吧?就更新数据模型而言,确实如此。

但是,无论出于何种原因让我在过去 2 小时内感到困惑,只有单元格 #3 中的开关会更新以反映我重新加载表数据时的状态变化。无论我在表中有多少行,它总是只有第三个单元格有效。如果少于 3 个单元格,则没有单元格起作用。

我已经在其中转储了大量NSLog 调用,它们反映了我一直期望的正确数据状态,但它们只是没有反映在第一次加载之后的切换中。如果我离开视图控制器然后回来,开关都已正确设置...

我只知道这将是我在某个地方做过的一些愚蠢的事情,因为它只在一个实例中起作用,但对于我的生活我看不到它:(

你可以吗?

-(void)viewDidLoad
{
    [super viewDidLoad];

    [_tableAddPeople setDelegate:self];
    [_tableAddPeople setDataSource:self];
}

-(void)viewWillAppear:(BOOL)animated
{
    [_tableAddPeople reloadData];
}

-(void)listSwitchChanged:(id)sender
{
    UISwitch * switchControl = sender;

    Person * person = [SAVE_DATA getPersonWithIndex:(int)switchControl.tag];

    if (switchControl.on)
    {
        [SAVE_DATA activePeopleAdd:person.tag];
    }
    else
    {
        [SAVE_DATA activePeopleRemove:person.tag];
    }
}


#pragma mark UITableViewDataSource Delegate

-(NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
{
    return [SAVE_DATA peopleCount];
}

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

    UILabel * nameLabel = (UILabel *)[cell.contentView viewWithTag:1];
    UISwitch * activeSwitch = (UISwitch *)[cell.contentView viewWithTag:2];

    Person * person = [SAVE_DATA getPersonWithIndex:(int)[indexPath row]];

    [nameLabel setText:[person name]];

    BOOL active = [SAVE_DATA activePeopleContains:person.tag];
    NSLog(@"Row for %@ is %@", person.name, (active? @"ON" : @"OFF"));
    [activeSwitch setOn:active animated:YES];
    [activeSwitch addTarget:self action:@selector(listSwitchChanged:) forControlEvents:UIControlEventValueChanged];
    activeSwitch.tag = (int)[indexPath row];

    return cell;
}

-(BOOL)tableView:(UITableView *)tableView canEditRowAtIndexPath:(NSIndexPath *)indexPath
{
    return NO;
}

-(BOOL)tableView:(UITableView *)tableView canMoveRowAtIndexPath:(NSIndexPath *)indexPath
{
    return NO;
}


#pragma mark UITableView Delegate

-(void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath
{
    Person * person = [SAVE_DATA getPersonWithIndex:(int)[indexPath row]];

    if ([SAVE_DATA activePeopleContains:person.tag])
    {
        NSLog(@"Removing %@", person.name);
        [SAVE_DATA activePeopleRemove:person.tag];
    }
    else
    {
        NSLog(@"Adding %@", person.name);
        [SAVE_DATA activePeopleAdd:person.tag];
    }

    [_tableAddPeople reloadData];
    //dispatch_async(dispatch_get_main_queue(), ^{[tableView reloadData];});
}

【问题讨论】:

    标签: ios objective-c uitableview uiswitch


    【解决方案1】:

    您的问题是您将开关 tag 属性用于两件事:

    1. 在视图层次结构中识别它(您已分配值“2”)
    2. 确定在哪一行更改了开关

    因为您将indexPath.row 分配给tag,所以在索引路径不是2(即第三行)的情况下,[cell.contentView viewWithTag:2]; 将在重新使用单元格时返回nil

    您还遇到了单元格重复使用的问题,因为您要多次添加开关操作处理程序。

    我建议你采用以下方法:

    1. 使用属性公开您的单元格的子视图并将dequeueReusableCell 的结果转换为您的UITableViewCell 子类,以便您无需使用标签即可访问它们

    2. 建立您的UITableViewCell 子类作为开关操作处理程序。

    3. 在单元格和视图控制器之间实现委托模式,以便单元格可以通知视图控制器开关已更改。单元格可以将self 传递给委托方法。在委托方法中,您可以使用indexPathForCell 来确定受影响的行。

    4. 不要仅仅因为一行发生了变化就重新加载整个 tableview。仅重新加载受影响的行。

    【讨论】:

      猜你喜欢
      • 2011-04-15
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2015-05-11
      • 1970-01-01
      • 1970-01-01
      • 2012-12-19
      • 1970-01-01
      相关资源
      最近更新 更多