37

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

  1. ユーザーから 2 つのデータを収集するためにモーダルに表示される View Controller のインスタンス
  2. tableView コントローラ
  3. テーブルビュー コントローラーをルートビュー コントローラーとして初期化するナビゲーション
    コントローラー
  4. collectionviewController、ここでもセルのクラスを登録していることに注意してください。
  5. ナビゲーション コントローラーを最初のビュー
    として、コレクション ビューを 2 番目のビューとして追加する UITabBarController..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 コントローラは通知にサブスクライブし、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

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

4

8 に答える 8

0

私の状況では、動画を読み込んでいるため実行collectionView.reloadData()できません。insertメソッドを使用して、各動画を cv の下部に追加する必要があります。

セクションが 1 つしかないという奇妙な問題に遭遇しました。現在のタブにいて、ビューが表示されている間、以下のコードは正常に機能しました。

datasource.append(post)

let lastIndex = datasource.count - 1
let indexPath = IndexPath(item: lastIndex, section: 0)

collectionView.performBatchUpdates({
    self.collectionView.insertItems(at: [indexPath])
})

しかし、タブを切り替えたためにビューが表示されなくなった場合、上記のコードが実行されると (別のタブにいても実行されます)、エラーが発生します。Invalid update: invalid number of items on UICollectionView

ここですべての答えを試しましたが、どれもうまくいきませんでした。使用すると言ったものでさえcv.reloadData()。うまくいったのは、nikita_gaydukov によるこのray wenderlich の回答に従ったときだけでした。彼は内部のすべてを行うように言いましたperformBatchUpdates。エラーがなくなったのでうまくいきました!

collectionView.performBatchUpdates({

    self.datasource.append(post)

    let lastIndex = self.datasource.count - 1
    let indexPath = IndexPath(item: lastIndex, section: 0)

    self.collectionView.insertItems(at: [indexPath])
})
于 2022-03-04T05:00:20.330 に答える