Reputation: 5984
As the title implies, I am trying to asynchronously load tableview images via NSURLConnection. The problem here is that my tableview, as I scroll down, tends to load the same flag image for multiple countries in the tableview, and then a different flag image for multiple countries (as in multiple countries have the same flag image) as I scroll back up. I noticed that only if I scroll very slowly do I get different flag images for different countries, but the images are incorrect. I'm not quite sure what the issue is at the moment. Here is my code:
#import "ViewController.h"
@implementation ViewController
@synthesize tableView;
@synthesize countryNamesArray;
@synthesize receivedData;
@synthesize flagImage;
- (void)didReceiveMemoryWarning
{
[super didReceiveMemoryWarning];
// Release any cached data, images, etc that aren't in use.
}
#pragma mark - View lifecycle
- (void)viewDidLoad
{
countryNamesArray=[[NSArray alloc] initWithObjects:@"India",@"USA",@"Antarctica",@"Brazil",@"Canada",@"China",@"France",@"Germany",@"Italy",@"Japan",@"Kenya",@"Malaysia",@"Mexico",@"South Africa",@"United Kingdom",@"Vietnam",nil];
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
}
- (void)viewDidUnload
{
[super viewDidUnload];
// Release any retained subviews of the main view.
// e.g. self.myOutlet = nil;
}
- (void)viewWillAppear:(BOOL)animated
{
[super viewWillAppear:animated];
}
- (void)viewDidAppear:(BOOL)animated
{
[super viewDidAppear:animated];
}
- (void)viewWillDisappear:(BOOL)animated
{
[super viewWillDisappear:animated];
}
- (void)viewDidDisappear:(BOOL)animated
{
[super viewDidDisappear:animated];
}
- (BOOL)shouldAutorotateToInterfaceOrientation:(UIInterfaceOrientation)interfaceOrientation
{
// Return YES for supported orientations
if ([[UIDevice currentDevice] userInterfaceIdiom] == UIUserInterfaceIdiomPhone) {
return (interfaceOrientation != UIInterfaceOrientationPortraitUpsideDown);
} else {
return YES;
}
}
-(NSInteger)tableView:(UITableView *)tableview numberOfRowsInSection:(NSInteger)section{
return 16;
}
- (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response
{
// This method is called when the server has determined that it
// has enough information to create the NSURLResponse.
// It can be called multiple times, for example in the case of a
// redirect, so each time we reset the data.
// receivedData is an instance variable declared elsewhere.
[receivedData setLength:0];
}
- (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data
{
// Append the new data to receivedData.
// receivedData is an instance variable declared elsewhere.
[receivedData appendData:data];
}
- (void)connection:(NSURLConnection *)connection
didFailWithError:(NSError *)error
{
// release the connection, and the data object
// receivedData is declared as a method instance elsewhere
// inform the user
NSLog(@"Connection failed! Error - %@ %@",
[error localizedDescription],
[[error userInfo] objectForKey:NSURLErrorFailingURLStringErrorKey]);
}
- (void)connectionDidFinishLoading:(NSURLConnection *)connection
{
// do something with the data
// receivedData is declared as a method instance elsewhere
NSLog(@"Succeeded! Received %d bytes of data",[receivedData length]);
flagImage = [UIImage imageWithData: receivedData];
}
-(void)issueRequest:(NSString *)fullCountryImageURL{
NSURLRequest *theRequest=[NSURLRequest requestWithURL:[NSURL URLWithString:fullCountryImageURL]
cachePolicy:NSURLRequestUseProtocolCachePolicy
timeoutInterval:60.0];
// create the connection with the request
// and start loading the data
NSURLConnection *theConnection=[[NSURLConnection alloc] initWithRequest:theRequest delegate:self];
if (theConnection) {
// Create the NSMutableData to hold the received data.
// receivedData is an instance variable declared elsewhere.
receivedData = [NSMutableData data];
}
else {
// Inform the user that the connection failed.
}
}
-(UITableViewCell *) tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath{
static NSString *cellIdentifier = @"countryCell";
UITableViewCell *cell=(UITableViewCell *) [self.tableView dequeueReusableCellWithIdentifier:cellIdentifier];
if (cell==nil){
cell=[[UITableViewCell alloc] initWithStyle: UITableViewCellStyleSubtitle reuseIdentifier:cellIdentifier];
}
// Create the request.
NSString *flagImageURLPartOne = @"http://www.flagimage.com/";
NSString *countryNumber = [NSString stringWithFormat:@"%i", indexPath.row+1];
NSString *flagImageURLPartTwo = @".png";
NSString *fullCountryImageURL = [NSString stringWithFormat:@"%@%@%@", flagImageURLPartOne, countryNumber, flagImageURLPartTwo];
[self issueRequest:fullCountryImageURL];
cell.imageView.image = flagImage;
cell.textLabel.text= [countryNamesArray objectAtIndex:indexPath.row];
return cell;
}
-(void) tableView:(UITableView *) tableview didSelectRowAtIndexPath:(NSIndexPath *)indexPath{
}
@end
Anyone have any ideas?
Upvotes: 0
Views: 527
Reputation: 9185
Somehow, you have to associate the flagImage
with the country name through a data structure.
Take a look at what is happening in tableView:cellForRowAtIndexPath:
You fire off a request for the image asynchronously, then assign the current flagImage
to the cell's image view. But the NSURLConnection
delegate methods fire at some non-deterministic time in the future, reassigning flagImage
at that time.
What you are doing currently in tableView:cellForRowAtIndexPath:
would work for a synchronous NSURLConnection
but you are getting tripped-up by the asynchronous delegate call backs.
Upvotes: 1