web-dev-qa-db-ja.com

無効な更新:UICollectionViewのアイテムの数が無効です

これには困惑しています。これが私のシナリオです。私のAppdelegateでは、作成しています

  1. ユーザーから2つのデータを収集するためにモーダルで表示されるView Controllerのインスタンス
  2. TableViewコントローラー
  3. ルートビューとしてTableViewコントローラーを使用して初期化したNavigation Controller
    コントローラ
  4. CollectionviewController、ここでセルのクラスも登録していることに注意してください。
  5. Navigation Controllerを最初のビューとして追加するUITabBarControllerと
    collectionviewを2番目のビューとして.. 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
}

テーブルビューには、モデルビューコントローラーを表示するnavbarボタンがあります。モーダルView Controllerには、textFieldとdatepickerから値を取得し、UserInfoディクショナリの値と共にデフォルトの通知センターに投稿するボタンがあります。 tableViewコントローラーは通知をサブスクライブし、UserInfoディクショナリーをMutableArrayに入れてテーブルを更新します。それはうまくいきます。

問題はCollectionVIewにあります。 collectionVIewは通知を受け取り、ターゲットメソッドを呼び出して通知を処理します。通知からUserInfo辞書を取得し、collectionViewに新しいセルを追加しようとしています。

だが...

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

ここに何かを指摘できる人に感謝します。

31
Tommy Alexander

これは、空のinsertItemsAtIndexPathsUICollectionViewを使用した場合のバグです。これを行うだけです:

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

(これがiOS8でまだ壊れているとは信じられません。)

60
Timothy Moose

これはiOS 11でもUICollectionViewのバグですが、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)])

それがあなたの状況に役立つことを願っています。

19
Michael Su

1つのセクションのみを含む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];
12
vmeyer

私の場合、View Controllerのロードが完了する前にcollectionViewに何かを挿入しようとしたため、numberOfItemsInSectionself.performBatchUpdatesから2回呼び出されていました(BEFORE番号に対して1回、AFTER番号に対して1回) 。

解決策は、guard isViewLoaded else { return }を呼び出す前にself.performBatchUpdatesを追加することです。

4
p-sun

私に関しては、問題はインデックスセットに関連していました-セクションを挿入する場所。 1.セクション0 .. <10(10セクション)を挿入しました。2.セクションをIndexSet 9 .. <19に挿入しようとしました。ただし、新しいオブジェクトをdataSource配列の最後に追加します。正しいインデックスセットは(10 .. <20)でなければなりません

したがって、最後のセクションはまだセクション9からの行数を期待していました。しかし、データソースでは-インデックス19にありました。

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

0
Naloiko Eugene

私は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)] )
   }
}

このエラーを回避する必要があります。

Collectionviewが通知を受信すると、次のエラーが表示されます。

'無効な更新:セクション0のアイテムの数が無効です。更新後の既存のセクションに含まれるアイテムの数(1)は、更新前のセクションに含まれるアイテムの数(1)にプラスまたはマイナスする必要がありますそのセクションから挿入または削除されたアイテムの数(1が挿入、0が削除)およびそのセクションに出入りするアイテムの数にプラスまたはマイナス(0が移動、0が移動)。

0