Сбой приложения при расширении ячейки таблицы

У меня есть система для добавления ячейки в раздел, когда уже существующая ячейка нажата. Приложение вылетает, когда я пытаюсь развернуть любой раздел после и включая indexpath 3. Он генерирует следующий отчет о сбое:

Terminating app due to uncaught exception 'NSRangeException', reason: '*** -[__NSArrayI objectAtIndex:]: index 4 beyond bounds [0 .. 2]'
*** First throw call stack:
(
    0   CoreFoundation                      0x0000000101983495 __exceptionPreprocess + 165
    1   libobjc.A.dylib                     0x00000001016e299e objc_exception_throw + 43
    2   CoreFoundation                      0x000000010193be3f -[__NSArrayI objectAtIndex:] + 175
    3   Quick Fit                           0x00000001000123d6 -[tuesdayVC tableView:cellForRowAtIndexPath:] + 1878
    4   UIKit                               0x0000000100363f8a -[UITableView _createPreparedCellForGlobalRow:withIndexPath:] + 348
    5   UIKit                               0x000000010054d5ed -[_UITableViewUpdateSupport(Private) _setupAnimationsForNewlyInsertedCells] + 7364
    6   UIKit                               0x0000000100556a81 -[_UITableViewUpdateSupport _setupAnimations] + 193
    7   UIKit                               0x000000010034c615 -[UITableView _updateWithItems:updateSupport:] + 1639
    8   UIKit                               0x0000000100348000 -[UITableView _endCellAnimationsWithContext:] + 11615
    9   Quick Fit                           0x0000000100012c3f -[tuesdayVC tableView:didSelectRowAtIndexPath:] + 879
    10  UIKit                               0x00000001003575c2 -[UITableView _selectRowAtIndexPath:animated:scrollPosition:notifyDelegate:] + 1312
    11  UIKit                               0x00000001003576eb -[UITableView _userSelectRowAtPendingSelectionIndexPath:] + 221
    12  UIKit                               0x00000001002a8100 _applyBlockToCFArrayCopiedToStack + 316
    13  UIKit                               0x00000001002a7f71 _afterCACommitHandler + 460
    14  CoreFoundation                      0x000000010194edc7 __CFRUNLOOP_IS_CALLING_OUT_TO_AN_OBSERVER_CALLBACK_FUNCTION__ + 23
    15  CoreFoundation                      0x000000010194ed37 __CFRunLoopDoObservers + 391
    16  CoreFoundation                      0x000000010192e522 __CFRunLoopRun + 946
    17  CoreFoundation                      0x000000010192dd83 CFRunLoopRunSpecific + 467
    18  GraphicsServices                    0x0000000103af9f04 GSEventRunModal + 161
    19  UIKit                               0x000000010028fe33 UIApplicationMain + 1010
    20  Quick Fit                           0x00000001000153d3 main + 115
    21  libdyld.dylib                       0x000000010201b5fd start + 1
    22  ???                                 0x0000000000000001 0x0 + 1
)
libc++abi.dylib: terminating with uncaught exception of type NSException
(lldb) 

Просматривая отчет о сбое, я думаю, что этот метод не работает:

- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath
{
    if ([self tableView:tableView canCollapseSection:indexPath.section])
    {
        if (!indexPath.row)
        {
            // only first row toggles exapand/collapse
            [tableView deselectRowAtIndexPath:indexPath animated:YES];

            NSInteger section = indexPath.section;
            BOOL currentlyExpanded = [expandedSections containsIndex:section];
            NSInteger rows;

            NSMutableArray *tmpArray = [NSMutableArray array];

            if (currentlyExpanded)
            {
                rows = [self tableView:tableView numberOfRowsInSection:section];
                [expandedSections removeIndex:section];

            }
            else
            {
                [expandedSections addIndex:section];
                rows = [self tableView:tableView numberOfRowsInSection:section];
            }

            for (int i=1; i<rows; i++)
            {
                NSIndexPath *tmpIndexPath = [NSIndexPath indexPathForRow:i
                                                               inSection:section];
                [tmpArray addObject:tmpIndexPath];
            }

            UITableViewCell *cell = [tableView cellForRowAtIndexPath:indexPath];

            if (currentlyExpanded)
            {
                [tableView deleteRowsAtIndexPaths:tmpArray
                                 withRowAnimation:UITableViewRowAnimationTop];

                cell.accessoryView = [DTCustomColoredAccessory accessoryWithColor:[UIColor orangeColor] type:DTCustomColoredAccessoryTypeDown];

            }
            else
            {
                [tableView insertRowsAtIndexPaths:tmpArray
                                 withRowAnimation:UITableViewRowAnimationTop];
                cell.accessoryView =  [DTCustomColoredAccessory accessoryWithColor:[UIColor orangeColor] type:DTCustomColoredAccessoryTypeUp];

            }
        }
    }
}

3 ответа

Решение

Попробуйте, оказывается, что в вашем массиве недостаточно данных, собранных табличным представлением при вызове. Попробуйте это: и поместите его в поле зрения.

self.techniques = [[NSArray alloc] initWithObjects:@"-",
                   @"-",
                   @"-",
                   @"-",
                   @"-", nil];

Я сталкивался с этой проблемой раньше при редактировании UITableView. Из того, что я могу сказать, вы обновляете свое табличное представление, но не свой источник данных.

didSelectRowAtIndexPath Метод делегата может быть вызван много раз.

Проверьте с помощью приведенного ниже кода, если он работает.

- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath
{
    static BOOL inprg = NO;
    // prevent reentry when the operation is in progress
    if (inprg) {
        NSLog(@"didSelectRowAtIndexPath in progress");
        return;
    }

    inprg = YES;

    .... << your original code >> ....

    inprg = NO;
}
Другие вопросы по тегам