Подтвердить что ты не робот

Недопустимое обновление: недопустимое количество элементов в UICollectionView

Я в тупике. Вот мой сценарий. В моем Appdelegate я создаю

  • Экземпляр контроллера представления, который будет представлен в виде монеты для сбора двух частей данных от пользователя
  • Контроллер tableView
  • Контроллер навигации, который я запускаю с контроллером tableview в качестве корневого каталога
    контроллер
  • CollectionviewController, обратите внимание, что я также регистрирую класс для ячейки.
  • UITabBarController, что я добавляю контроллер навигации в качестве первого вида и collectionview в качестве второго представления. TabBarCOntorller устанавливается как корневое окно окна.

Вот код:

 - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:   
(NSDictionary *)launchOptions
{
self.window = [[UIWindow alloc] initWithFrame:[[UIScreen mainScreen] bounds]];
self.evc = [[WTAEntryControllerViewController alloc] init];
WTATableView *tv = [[WTATableView alloc] initWithNibName:@"WTATableView" bundle:nil];
UINavigationController *nav = [[UINavigationController alloc]    
initWithRootViewController:tv];
nav.tabBarItem.title = NSLocalizedString(@"Birthday List", nil);
nav.tabBarItem.image = [UIImage imageNamed:@"birthdaycake"];
WTACollectionViewController *cv = [[WTACollectionViewController alloc] 
initWithCollectionViewLayout:[[UICollectionViewFlowLayout alloc] init]];
[cv.collectionView registerClass:[UICollectionViewCell class] 
forCellWithReuseIdentifier:CellIdentifier];
cv.collectionView.backgroundColor = [UIColor whiteColor];
NSLog(@"cv instance %@", cv);
UITabBarController *tabController = [[UITabBarController alloc] init];
tabController.viewControllers = @[nav, cv];
self.window.rootViewController = tabController;
[self.window makeKeyAndVisible];
return YES
}

В представлении таблицы есть кнопка навигации, которая представляет контроллер представления модели. Контроллер модального представления имеет кнопку, которая принимает значения из textField и datepicker и отправляет их в центр уведомлений по умолчанию со значениями в словаре UserInfo. TableView contorller подписывается на уведомление, помещает словарь UserInfo в MutableArray и обновляет таблицу. Это прекрасно работает.

Проблема с CollectionVIew. CollectionVIew получает уведомление и вызывает целевой метод для обработки уведомления. Я пытаюсь взять словарь UserInfo из уведомления и добавить новую ячейку в collectionView.

НО...

Когда получение коллекции получает уведомление, я получаю сообщение об ошибке:

'Неверное обновление: недопустимое количество элементов в разделе 0. Количество элементов, содержащихся в существующем разделе после обновления (1), должно быть равно количеству элементов, содержащихся в этом разделе, перед обновлением (1), плюс или минус количество элементов, вставленных или удаленных из этого раздела (1 вставлено, 0 удалено) и плюс или минус количество элементов, перемещенных в или из этого раздела (0 перемещен, 0 перемещен).

Вот код для CollectionView:

#import "WTACollectionViewController.h"
#import "UIColor+WTAExtensions.h"
#import "WTAAppDelegate.h"
#import "WTAEntryControllerViewController.h"

@interface WTACollectionViewController () <UICollectionViewDelegateFlowLayout>
@property (strong, nonatomic) NSMutableArray *birthdays;
@end

@implementation WTACollectionViewController

-(id)initWithCollectionViewLayout:(UICollectionViewLayout *)layout{

NSLog(@"Calling init on Collection");
if(!(self = [super initWithCollectionViewLayout:layout])){
    return nil;
}

self.birthdays = [NSMutableArray array];

NSLog(@"Count of Birthdays at init %ld", (long)[self.birthdays count] );
self.title = NSLocalizedString(@"Birthday Grid", nil);
self.tabBarItem.image = [UIImage imageNamed:@"package"];
NSLog(@"cv instance getting notif %@", self);
[[NSNotificationCenter defaultCenter]
 addObserver:self
 selector:@selector(handleNotification:)
 name:@"BirthdayAdded"
 object:nil];
 return self;

}


-(void)handleNotification:(NSNotification *)notification
{
[self.birthdays addObject:[notification userInfo]];
NSLog(@"Count of birthdays when the notification is received: %ld", (long)         
[self.birthdays count]);
[self.collectionView insertItemsAtIndexPaths:@[[NSIndexPath indexPathForItem:   
(self.birthdays.count -1) inSection:0]]];
}

- (void)viewDidLoad
{
[super viewDidLoad];

NSLog(@"Calling View Did Load on Collection");
}

- (NSInteger)numberOfSectionsInCollectionView:(UICollectionView *)collectionView {

return 1;

}

- (NSInteger)collectionView:(UICollectionView *)collectionView numberOfItemsInSection:   
(NSInteger)section{

NSLog(@"Count of birthdays in  the number of items in section: %ld", (long)  
[self.birthdays count]);
return [self.birthdays count];
}

- (UICollectionViewCell *)collectionView:(UICollectionView *)collectionView    
cellForItemAtIndexPath:(NSIndexPath *)indexPath {

NSLog(@"Calling the cell for index path method");

NSDictionary *dict = [[NSMutableDictionary alloc] init];
dict = self.birthdays[indexPath.item];

UICollectionViewCell *cell = [collectionView   
dequeueReusableCellWithReuseIdentifier:CellIdentifier forIndexPath:indexPath];
NSAssert(cell != nil, @"Expected a Cell");

cell.contentView.backgroundColor = [UIColor randomColor];
return cell;

}

#define GAP (1.0f)

-(CGSize)collectionView:(UICollectionView *)collectionView layout:   
(UICollectionViewLayout *)collectionViewLayout sizeForItemAtIndexPath:(NSIndexPath   
*)indexPath {


CGFloat edgeLength = self.collectionView.frame.size.width / 4.0f - GAP;
return (CGSize) {.width = edgeLength, .height = edgeLength};
}

-(CGFloat)collectionView:(UICollectionView *)collectionView layout:   
(UICollectionViewLayout *)collectionViewLayout minimumLineSpacingForSectionAtIndex:    
(NSInteger)section{

return GAP;
}

-(CGFloat)collectionView:(UICollectionView *)collectionView layout:   
(UICollectionViewLayout *)collectionViewLayout minimumInteritemSpacingForSectionAtIndex:  
(NSInteger)section{

return GAP;
}
@end

Я бы очень признателен всем, кто может что-то указать здесь.

4b9b3361

Ответ 1

Это ошибка с использованием insertItemsAtIndexPaths на пустой UICollectionView. Просто сделайте следующее:

if (self.birthdays.count == 1) {
    [self.collectionView reloadData];
} else {
    [self.collectionView insertItemsAtIndexPaths:@[[NSIndexPath indexPathForItem:   (self.birthdays.count -1) inSection:0]]];
}

(Не могу поверить, что это все еще нарушено в iOS8.)

Ответ 2

Это ошибка в UICollectionView даже в iOS 11, но не существует в UITableView, очень легко воспроизводится: Предположим, что работает следующий SWIFT-код:

addItemInDataSource()
collectionView!.insertItems(at: [IndexPath(item: position, section: 0)])

Измените его на:

collectionView!.reloadData()    // <-- This new line will make UICollection crash...

addItemInDataSource()
collectionView!.insertItems(at: [IndexPath(item: position, section: 0)])

Он выйдет из строя...

UICollectionView потеряет исходное количество элементов в некотором состоянии, просто добавьте фиктивную строку до того, как команда insertItems избежит такого сбоя:

collectionView!.reloadData()

collectionView!.numberOfItems(inSection: 0) //<-- This code is no used, but it will let UICollectionView synchronize number of items, so it will not crash in following code.
addItemInDataSource()
collectionView!.insertItems(at: [IndexPath(item: position, section: 0)])

Надеюсь, это поможет в вашей ситуации.

Ответ 3

У меня была такая же ошибка, когда я вставил элементы в collectionView только с одним разделом.

Я сделал исправление Тимоти, но этого было недостаточно. Фактически, collectionView уже обновил свои данные, когда я попытался вставить новые элементы. Использование collectionView numberOfItemsInSection решает проблему.

[self.collectionView performBatchUpdates:^{
        NSMutableArray *arrayWithIndexPaths = [NSMutableArray array];
        for (NSUInteger i = [self.collectionView numberOfItemsInSection:0]; i < self.elements.count ; i++)
        {
            [arrayWithIndexPaths addObject:[NSIndexPath indexPathForRow:i inSection:0]];
        }

        [self.collectionView insertItemsAtIndexPaths:arrayWithIndexPaths];
    } completion:nil];

Ответ 4

В моем случае numberOfItemsInSection вызывается дважды из self.performBatchUpdates (один раз для номера BEFORE, один раз для номера AFTER), потому что я пытался вставить что-то в collectionView до того, как контроллер представления закончил загрузку.

Решение состоит в том, чтобы добавить guard isViewLoaded else { return } перед вызовом self.performBatchUpdates.

Ответ 5

Что касается меня, то проблема была связана с Индексом Set - куда вставлять разделы. 1. Я вставил разделы 0.. <10 (10 разделов) 2. Я пытался вставить разделы в IndexSet 9.. <19. Но добавление новых объектов в конец массива dataSource. Правильный индексный индекс должен быть (10.. <20)

Таким образом, последний раздел все еще ожидал количества строк из раздела 9. Но в источнике данных - это был индекс 19.

collectionView?.insertSections(IndexSet (integersIn: startIndex.. <(startIndex + series.count)))

Ответ 6

Я не практикуюсь с Obj-c, поэтому я пишу свой ответ на Swift.

предположим, что дни рождения - это массив дат в виде строки

birthdays = [String]()

func addDate() {
   birthdays.append("your_new_date")
   collectionView.insertItems(at: [IndexPath(item: birthdays.count - 1, section: 0)] ) 
}

func removeDate() {
   if birthdays.count > 0 {
      birthdays.removeLast() // use remove(at: Int) instead if u want
      collectionView.deleteItems(at: [IndexPath(item: birthdays.count, section: 0)] )
   }
}

Вам следует избегать этой ошибки:

Когда представление коллекции получает уведомление, я получаю сообщение об ошибке:

'Недопустимое обновление: недопустимое количество элементов в разделе 0. Количество элементов, содержащихся в существующем разделе после обновления (1), должно быть равно количеству элементов, содержащихся в этом разделе до обновления (1), плюс или минус количество элементов, вставленных или удаленных из этого раздела (1 вставлено, 0 удалено) и плюс или минус количество элементов, перемещенных в или из этого раздела (0 перемещено, 0 перемещено). '

Ответ 7

Принятый ответ вводит в заблуждение. Причина такого поведения задокументирована, как указано в fooobar.com/info/1206756/...