Justin Cabral
Justin Cabral

Reputation: 585

UICollectionView datasource methods not getting called, but are being set in the init

Here is my source code

- (id)initWithCollectionView:(UICollectionView *)collectionView
{
self = [super init];

if (self)
{
    self.collectionView = collectionView;
    self.collectionView.dataSource = self;
    self.collectionView.delegate = self;
    [self.collectionView registerClass:[TWNTweetCell class] forCellWithReuseIdentifier:kCell];
    self.collectionViewLayout = self.collectionView.collectionViewLayout;



    self.tweetArray = @[];
    self.tweetTextArray = @[];

    self.twitter = [STTwitterAPI twitterAPIOSWithFirstAccount];
}

return self;
}

#pragma mark - CollectionView
#pragma mark DataSource

-(NSInteger)numberOfSectionsInCollectionView:(UICollectionView *)collectionView
{
return 1;
}

- (NSInteger)collectionView:(UICollectionView *)collectionView numberOfItemsInSection:      (NSInteger)section
{
return [self.tweetArray count];
}

- (UICollectionViewCell *)collectionView:(UICollectionView *)collectionView cellForItemAtIndexPath:(NSIndexPath *)indexPath
{
TWNTweetCell *cell = (TWNTweetCell *)[collectionView dequeueReusableCellWithReuseIdentifier:kCell forIndexPath:indexPath];

NSDictionary *status = [self.tweetArray objectAtIndex:indexPath.row];

NSString *text = [status valueForKey:@"text"];

cell.usernameLabel.text = screenName;
//    cell.createdAtLabel.text = dateString;

cell.autoresizingMask = UIViewAutoresizingFlexibleWidth;

UITextView *textView = [self.tweetTextArray objectAtIndex:indexPath.row];
[cell setTweet:text withTweetTextView:textView];

return cell;
}

All the methods don't get interupted at all by breakpoints. The tweets are getting loaded in the log so I know everything else is ok, its just not recognizing the collection view. And yes i've set the

Anyone have any idea whats going on?

Upvotes: 12

Views: 33379

Answers (7)

user2403271
user2403271

Reputation: 91

for swift do this , set a property

//MARK: props
let dataSource = MyDataSource()

and in

viewDidLoad(){
// your other code 
..
..
collectionView.dataSource = dataSource // it is a strong reference 
}

apart form these other general pitfall are

  • not returning the count or the data source
  • not populating the data source

Upvotes: 2

Ali Awais
Ali Awais

Reputation: 113

I created collection view in storyboard and linked datasource and delegate but they were not being called in Xcode 8.0 with Swift 3.0. Tried multiple things but the solution was to declare the delegate and datasource in class declaration line:

class ViewController: UIViewController, UICollectionViewDelegate, UICollectionViewDataSource { ... }

Previously when we linked delegate and datasource through storyboard it was not required, may be a bug :)

Upvotes: 0

Julian
Julian

Reputation: 9337

It is not your case, it might be helpful for others who will came here having problem with data source methods not being called. It could be assigning data source like:

collectionView.dataSource = MyDataSource()

which is wrong as dataSource is a weak reference so it needs to be stored by some strong reference to be alive after creating it. Added a private property in a ViewController to keep the strong reference, initialising and then assigning it fixes the issue.

Upvotes: 50

user3378170
user3378170

Reputation: 2387

Add the collectionView to a view hierarchy.

In the init method you set the property (self.collectionView) but you do not add the collectionView to a view hierarchy. So the collectionView won't call any dataSource or delegate method.

Upvotes: 1

Tim
Tim

Reputation: 9042

A few suggestions:

  • Do all your UICollectionView setup and configuration in viewDidLoad.
  • Ensure you calling the create init method from your other class
  • Your tweetArray is also empty, so if the number of items method is called, it will return nothing and the other methods will not be called

Upvotes: 4

Timothy Moose
Timothy Moose

Reputation: 9915

Call [collectionView reloadData] at the end of your init method. The collection view needs to be told to populate itself. I assume UICollectionViewController does this internally, but you don't seem to be using UICollectionViewController (or at least not in the usual way).

Upvotes: -1

Michael Dautermann
Michael Dautermann

Reputation: 89509

A couple things:

1) in (and only in) your "init" method, use the underlying instance variable for your @property. That is,

_collectionView = collectionView;
_collectionView.dataSource = self;
_collectionView.delegate = self;

This is called "direct access", and more information can be seen in this related question.

2) in your .h file, make certain to declare that your object conforms to the data source & delegate protocols. E.G.

@interface JustinViewController : UIViewController <UICollectionViewDelegate, UICollectionViewDataSource>

Upvotes: 3

Related Questions