iOS (cs193p): проблема с использованием делегата в MapViewController для получения изображения с использованием другого потока из другого контроллера
Я слежу за отличной лекцией CS193P по разработке под iOS 5.0:
http://www.stanford.edu/class/cs193p/cgi-bin/drupal/
Сейчас я на задании № 5, обязательное задание № 5б. Мне нужно отобразить выноску с миниатюрой, когда я нажимаю на аннотацию на карте.
Мне удалось сделать это без каких-либо проблем со следующим кодом (используя Flickr).
MapViewController.h:
@class MapViewController;
@protocol MapViewControllerDelegate <NSObject>
- (UIImage *)mapViewController:(MapViewController *)sender imageForAnnotation:(id <MKAnnotation>)annotation;
@end
@interface MapViewController : UIViewController
@property (nonatomic, strong) NSArray *annotations; // of id <MKAnnotation>
@property (nonatomic, weak) id <MapViewControllerDelegate> delegate;
@end
MapViewController.m:
- (void)mapView:(MKMapView *)mapView didSelectAnnotationView:(MKAnnotationView *)annotationView
{
if ([annotationView.leftCalloutAccessoryView isKindOfClass:[UIImageView class]]) {
UIImage *image = [self.delegate mapViewController:self imageForAnnotation:annotationView.annotation];
[(UIImageView *)annotationView.leftCalloutAccessoryView setImage:image];
}
}
PhotoListTableViewController.m:
@interface PhotoListTableViewController() <MapViewControllerDelegate>
- (UIImage *)mapViewController:(MapViewController *)sender imageForAnnotation:(id <MKAnnotation>)annotation
{
UIImage *image = nil;
if ([annotation isKindOfClass:[FlickrAnnotation class]]) { // make sure the annotation is a FlickrAnnotation
FlickrAnnotation *flickrAnnotation = (FlickrAnnotation *)annotation;
NSURL *photoUrl = [FlickrFetcher urlForPhoto:flickrAnnotation.photo format:FlickrPhotoFormatSquare];
image = [UIImage imageWithData:[NSData dataWithContentsOfURL:photoUrl]];
}
return image;
}
@end
Я использую делегата в mapView:didSelectAnnotationView: метод MapViewController для извлечения изображения из другого контроллера, PhotoListTableViewController (чтобы убедиться, что у меня есть универсальный MapViewController).
Он работает нормально... но мне нужно изменить этот код, чтобы вызвать Flickr в другом потоке. Поэтому я изменил mapViewController: imageForAnnotation: метод для этого:
- (UIImage *)mapViewController:(MapViewController *)sender imageForAnnotation:(id <MKAnnotation>)annotation
{
__block UIImage *image = nil;
if ([annotation isKindOfClass:[FlickrAnnotation class]]) { // make sure the annotation is a FlickrAnnotation
FlickrAnnotation *flickrAnnotation = (FlickrAnnotation *)annotation;
dispatch_queue_t downloadQueue = dispatch_queue_create("ThumbnailDownloader", NULL);
dispatch_async(downloadQueue, ^{
NSURL *photoUrl = [FlickrFetcher urlForPhoto:flickrAnnotation.photo format:FlickrPhotoFormatSquare];
dispatch_async(dispatch_get_main_queue(), ^{ // execute on the main thread
image = [UIImage imageWithData:[NSData dataWithContentsOfURL:photoUrl]];
});
});
dispatch_release(downloadQueue);
}
return image;
}
Однако с этим кодом миниатюра не отображается. Я знаю, что это потому, что когда метод "возврат изображения", изображение по-прежнему ноль, потому что поток downloadQueue не завершен. Я попытался "вернуть изображение" внутри блока следующим образом:
return [UIImage imageWithData:[NSData dataWithContentsOfURL:photoUrl]];
но компилятору это не нравится:
Несовместимые типы указателей блоков, передающие "UIImage *(^)(void)" параметру типа "disptach_block_t" (он же "void (^)(void)")
Я знаю, что ^{в начале блока означает void, но можно ли его изменить, чтобы вернуть UIImage? Или я делаю это неправильно с самого начала?
Я просто не знаю, как вернуть изображение из PhotoListTableViewController обратно в MapViewController. Как это должно быть сделано?
1 ответ
Я работаю над тем же заданием. Я переключал потоки, когда вызывался делегат контроллера табличного представления, вместо того, чтобы ждать, пока контроллер табличного представления извлечет изображение, и это работает для меня.
В коде:
- (void) mapView: (MKMapView *) mapView didSelectAnnotationView: (MKAnnotationView *) view {// NSLog (@ "Аннотация выбрана, получение эскиза");
dispatch_queue_t downloadQueue = dispatch_queue_create("flickr download", NULL);
dispatch_async(downloadQueue, ^{
UIImage *image = [self.delegate thumbNailImageForAnnotation:view.annotation sender: self];
dispatch_async(dispatch_get_main_queue(), ^{
[(UIImageView *)view.leftCalloutAccessoryView setImage:image];
});
});
}