【问题标题】:UICollectionView loading image only on ScrollUICollectionView 仅在 Scroll 上加载图像
【发布时间】:2014-03-30 17:39:44
【问题描述】:

我很难在数据被解析后加载 UICollectionView 图像。

这是来自 Apple 的 LazyTable 解决方案,一切都在 tableview 上工作,但对于

CollectionView 图像仅在我滚动时出现。其他数据,如标签或其他显示在

只有图片没有问题的单元格给我带来了麻烦。

这是我的委托中的重新加载方法:

    __block ParseOperation *weakParser = parser;

    parser.completionBlock = ^(void) {
        if (weakParser.appRecordList) {
            dispatch_async(dispatch_get_main_queue(), ^{
                CategroyScreenViewController *rootViewController = (CategroyScreenViewController*)[(UINavigationController*)self.window.rootViewController topViewController];

                rootViewController.entries = weakParser.appRecordList;
               [rootViewController.collectionView reloadData];
            });
        }
        self.queue = nil;
    };

    [self.queue addOperation:parser]; 
    self.appListData = nil;
}

这是我的包含 collectionview 的 CategroyScreenViewController :

#define kCustomRowCount     7

@interface CategroyScreemViewController () <UIScrollViewDelegate>
@property (nonatomic, strong) NSMutableDictionary *imageDownloadsInProgress;
@property (nonatomic) NSMutableArray* numbers;
@end
int num = 0;
@implementation CategroyScreemViewController

- (id)initWithNibName:(NSString *)nibNameOrNil bundle:(NSBundle *)nibBundleOrNil
{
    self = [super initWithNibName:nibNameOrNil bundle:nibBundleOrNil];
    if (self) {
        // Custom initialization
    }
    return self;
}

- (void)viewDidLoad
{
     NSLog(@"view did load");
    [super viewDidLoad];
    self.imageDownloadsInProgress = [NSMutableDictionary dictionary];
    [self.collectionView reloadData];


}

- (CGSize) blockSizeForItemAtIndexPath:(NSIndexPath *)indexPath {


     NSLog(@"Asking for index paths of non-existant cells!! %d", indexPath.row);

    if (indexPath.row == 0) return CGSizeMake(2, 2);    
    return CGSizeMake(1, 1);
}
- (void) viewDidAppear:(BOOL)animated {
     NSLog(@"viewDidAppear load");
    [self.collectionView reloadData];
}
- (UIEdgeInsets)insetsForItemAtIndexPath:(NSIndexPath *)indexPath {
      NSLog(@"insetsForItemAtIndexPath load");
    return UIEdgeInsetsMake(2, 2, 2, 2);
}
- (void)didReceiveMemoryWarning
{
    [super didReceiveMemoryWarning];
    NSArray *allDownloads = [self.imageDownloadsInProgress allValues];
    [allDownloads makeObjectsPerformSelector:@selector(cancelDownload)];

    [self.imageDownloadsInProgress removeAllObjects];
}
-(NSInteger)numberOfSectionsInCollectionView:(UICollectionView *)collectionView
{
    NSLog(@"numberOfSectionsInCollectionView load");
    return 1;
}

-(NSInteger)collectionView:(UICollectionView *)collectionView numberOfItemsInSection:(NSInteger)section
{
    NSUInteger count = [self.entries count];
      NSLog(@"numberOfItemsInSection load");
    if (count == 0)
    {
        return kCustomRowCount;
    }
    return count;
}
-(UICollectionViewCell *)collectionView:(UICollectionView *)collectionView cellForItemAtIndexPath:(NSIndexPath *)indexPath
{
    CategoryScreenCell *myCell = [collectionView dequeueReusableCellWithReuseIdentifier:@"MyCell" forIndexPath:indexPath];
    UILabel *label = (UILabel *)[myCell.contentView viewWithTag:10];
    NSUInteger nodeCount = [self.entries count];
    if (nodeCount > 0)
    {
        AppRecord *appRecord = [self.entries objectAtIndex:indexPath.row];
        [label setText:appRecord.appName];
        if (!appRecord.appIcon)
        {

            if (self.collectionView.dragging == NO && self.collectionView.decelerating == NO)
            {

                [self startIconDownload:appRecord forIndexPath:indexPath];
            }
            myCell.imageView.image = [UIImage imageNamed:@"Placeholder.png"];
        }
        else
        {

            myCell.imageView.image = appRecord.appIcon;
        }

    }

    return myCell;
}
- (UIColor*) colorForNumber:(NSNumber*)num {
    return [UIColor colorWithHue:((19 * num.intValue) % 255)/255.f saturation:1.f brightness:1.f alpha:1.f];
}
- (void)startIconDownload:(AppRecord *)appRecord forIndexPath:(NSIndexPath *)indexPath
{
    IconDownloader *iconDownloader = [self.imageDownloadsInProgress objectForKey:indexPath];
    if (iconDownloader == nil)
    {
        iconDownloader = [[IconDownloader alloc] init];
        iconDownloader.appRecord = appRecord;
        [iconDownloader setCompletionHandler:^{

            CategoryScreenCell *myCell = [self.collectionView dequeueReusableCellWithReuseIdentifier:@"MyCell" forIndexPath:indexPath];
            myCell.imageView.image = appRecord.appIcon;
            myCell.imageView.layer.masksToBounds = YES;
            [self.imageDownloadsInProgress removeObjectForKey:indexPath];

        }];
        [self.imageDownloadsInProgress setObject:iconDownloader forKey:indexPath];
        [iconDownloader startDownload];
    }
}
- (void)loadImagesForOnscreenRows
{
     NSLog(@"loadImagesForOnscreenRows");
    if ([self.entries count] > 0)
    {
        NSArray *visiblePaths = [self.collectionView indexPathsForVisibleItems];
        for (NSIndexPath *indexPath in visiblePaths)
        {
            AppRecord *appRecord = [self.entries objectAtIndex:indexPath.item];

            if (!appRecord.appIcon)
                // Avoid the app icon download if the app already has an icon
            {
                [self startIconDownload:appRecord forIndexPath:indexPath];
            }
        }
    }
}

- (void)scrollViewDidEndDragging:(UIScrollView *)scrollView willDecelerate:(BOOL)decelerate
{
    NSLog(@"scrollViewDidEndDragging");

    if (!decelerate)
    {
        NSLog(@"decelerate");
        [self loadImagesForOnscreenRows];
    }
}


- (void)scrollViewDidEndDecelerating:(UIScrollView *)scrollView
{
    NSLog(@"scrollViewDidEndDecelerating");
    [self loadImagesForOnscreenRows];
}

@end

在我改变这个之后好的:

        CategoryScreenCell *myCell = [self.collectionView dequeueReusableCellWithReuseIdentifier:@"MyCell" forIndexPath:indexPath];

到这个:

    CategoryScreenCell *myCell = [self.collectionView cellForItemAtIndexPath:indexPath];

programs 运行良好,但它给了我这个警告: “使用类型表达式初始化 'CategoryScreenCell *' 的不兼容指针类型 'UICollectionViewCell *'

【问题讨论】:

    标签: ios objective-c uicollectionview


    【解决方案1】:

    我看到的这种方法的问题是,在您的IconDownloader 的完成处理程序中,您总是将一个单元格出列并填充它,即使它不可见并且可能不会显示。我不确定这是多少开销 - 这取决于您拥有多少个单元格以及人们可能滚动的速度。

    另一种方法是在完成处理程序中发布通知,所有单元格都将注册该通知。在通知发生时调用的方法中,单元格会检查通知的userInfo 中的AppRecord 是否为当前单元格,并更新其图像。

    另一个问题是您为滚动经过的每个单元格排队下载图像,因此如果用户快速滚动到集合视图的底部,他们必须等待上面的所有图像首先加载。您可以通过以下几种方式解决此问题:

    1) 出列时为单元格提供对IconDownloader 的引用。然后在单元格的prepareForReuse 中,如果仍然挂起,请取消下载。

    2) 如果您不想取消挂起的下载,但想确保尽快更新可见单元格,您可以将下载放入优先级队列,并在每个新 @ 上增加优先级987654326@实例。

    【讨论】:

      【解决方案2】:

      假设您知道它将是所需的类型,您只需转换单元格类型即可消除警告:

      CategoryScreenCell *myCell = (CategoryScreenCell *)[self.collectionView cellForItemAtIndexPath:indexPath];
      

      【讨论】:

        猜你喜欢
        • 2018-12-09
        • 1970-01-01
        • 2015-12-29
        • 1970-01-01
        • 2013-08-22
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        相关资源
        最近更新 更多