美文网首页
iOS 动态更换UITabBarController的图标

iOS 动态更换UITabBarController的图标

作者: 为了自由的白菜 | 来源:发表于2019-12-30 12:10 被阅读0次

    更换tabbar的图表并不难, 难在动态更换时的思路, 我们可以先看看设置tabbarItem代码找思路:

    UITabBarItem *item4 = [[UITabBarItem alloc] initWithTitle:@"我"
                                            image:[[UIImage imageNamed:@"tab_me_normal"]
                                     imageWithRenderingMode:UIImageRenderingModeAlwaysOriginal]
                                    selectedImage:[[UIImage imageNamed:@"tab_me_click"]
                           imageWithRenderingMode:UIImageRenderingModeAlwaysOriginal]];
    

    分析代码我们可以知道, 入口只有image和selectImage, 自己想出了两种方法, 如下:

    1. 及时更新法, 每次下载一套图片, 包括一个按钮的普通状态跟选中状态的图片, 下载完之后就及时更新, 下次再进入APP, 直接查找缓存, 不用再次下载 (此方法会出现的问题, 比如四个tabbarItem, 会出现图标不统一的问题, 即有的图标为新图标, 有的图标为老图标, 但是网络状态好的情况下, 一般不会出现这种问题, 再说更换tabbarItme的图标,不是经常性操作, 所以我目前就用的这种方法, 已经满足我的开发需要)
    2. 等待四套图标全部下载完, 再统一更新, 如果有部分图标下载不完成, 则不进行更换操作 (要与安卓商量好, 看看哪种方法适合, 我现在只写出上面那种方法, 这种方法请按上述方法进行扩展)

    我写了一个Tabbar下载图片的工具类, 下面是.h文件 以及.m文件

    #import <Foundation/Foundation.h>
    #import "PORouterModel.h"
    #import "SDImageCache.h"
    #import "SDWebImageDownloader.h"
    
    NS_ASSUME_NONNULL_BEGIN
    
    @interface POTabbarTool : NSObject
    
    
    /// 异步下载tabbar图片, (同时下载normal 跟 select图片, 两张图片都下载完成后 再触发最终的回调block)
    /// @param routerModel tabbar的图片 要从routerModel.image中获取url
    /// @param block 最终返回下载normal 跟 select图片
    - (void)downloadImageWithPORouterMode:(PORouterModel *)routerModel AndReturnImageBlock:(void(^)(UIImage *normalImage, UIImage *selectImage, PORouterModel *routerModel))block;
    
    @end
    
    NS_ASSUME_NONNULL_END
    
    #import "POTabbarTool.h"
    
    @implementation POTabbarTool
    
    /// 异步下载tabbar图片, (同时下载normal 跟 select图片, 两张图片都下载完成后 再触发最终的回调block)
    /// @param routerModel tabbar的图片 要从routerModel.image中获取url
    /// @param block 最终返回下载normal 跟 select图片
    - (void)downloadImageWithPORouterMode:(PORouterModel *)routerModel AndReturnImageBlock:(void(^)(UIImage *normalImage, UIImage *selectImage, PORouterModel *routerModel))block{
        __block UIImage *needNormalImage;
        __block UIImage *needSelectImage;
        dispatch_group_t group = dispatch_group_create();
        dispatch_queue_t queue = dispatch_get_global_queue(0, 0);
        
        dispatch_group_enter(group);//开启任务一
        dispatch_group_async(group, queue, ^{
            [self toSearchCacheWithUrlString:routerModel.image2 completed:^(UIImage *image) {
                needNormalImage = image;
                dispatch_group_leave(group);//任务一完成
            }];
        });
        
        dispatch_group_enter(group);//开启任务二
        dispatch_group_async(group, queue, ^{
            [self toSearchCacheWithUrlString:routerModel.image1 completed:^(UIImage *image) {
                needSelectImage = image;
                dispatch_group_leave(group);//任务二完成
            }];
        });
        
        dispatch_group_notify(group, dispatch_get_main_queue(), ^{
            block(needNormalImage,needSelectImage,routerModel);//返回任务一跟任务二的数据
        });
    }
    
    - (void)toSearchCacheWithUrlString:(NSString *)urlString completed:(void (^)(UIImage *image))block{///<用SDImageCache去查找或下载图片, 最终返回图片
        UIImage *needImage;
        SDImageCache *normalImageCache = [SDImageCache sharedImageCache];
        needImage = [normalImageCache imageFromMemoryCacheForKey:urlString];//从内存中获取图片
        if (!needImage) {
            needImage = [normalImageCache imageFromDiskCacheForKey:urlString];//从硬盘中获取图片
            if (!needImage) {//如果都没有则下载图片
                [SDWebImageDownloader.sharedDownloader downloadImageWithURL:[NSURL URLWithString:urlString] completed:^(UIImage * _Nullable image, NSData * _Nullable data, NSError * _Nullable error, BOOL finished) {
                    if (image && finished) {
                        [normalImageCache storeImage:image forKey:urlString toDisk:YES completion:nil];//执行缓存图片 (包括缓存到内存和本地)
                        [normalImageCache storeImage:image forKey:urlString toDisk:YES completion:^{//本来有想过在这里直接处理过的图片, 但是发现图片会出问题, 所以处理图片的逻辑就放最外层了
                            block(image);
                        }];
                    }
                }];
            }
            else{
                block(needImage);
            }
        }
        else{
            block(needImage);
        }
    }
    

    看懂上面的方法, 就可以任意扩展了, 在返回的block内更新tabbarItem的图标即可, 需要注意的是, 在返回的block内, 尽量少使用局部变量, 防止循环引用造成一些奇奇怪怪的问题

    再多补充一些block内的伪代码

        //声明, 不可在以下block内调用局部变量, 导致局部变量所对应的VC不释放, 会出现奇怪的tabbarUI问题, 现在以下block内写法没有问题
        WEAKSELF
        [self.tabbarTool downloadImageWithPORouterMode:routerModel AndReturnImageBlock:^(UIImage * _Nonnull normalImage, UIImage * _Nonnull selectImage, PORouterModel * _Nonnull routerModel) {
            UIImage *newNormalImage = [UIImage imageWithCGImage:normalImage.CGImage scale:2 orientation:normalImage.imageOrientation];
            newNormalImage = [newNormalImage sd_resizedImageWithSize:CGSizeMake(29, 29) scaleMode:SDImageScaleModeAspectFit];
            UIImage *newSelectImage = [UIImage imageWithCGImage:selectImage.CGImage scale:2 orientation:selectImage.imageOrientation];
            newSelectImage = [newSelectImage sd_resizedImageWithSize:CGSizeMake(29, 29) scaleMode:SDImageScaleModeAspectFit];
            weakSelf.needChangeVC.tabBarItem.image = [normalImage imageWithRenderingMode:UIImageRenderingModeAlwaysOriginal];
            weakSelf.needChangeVC.tabBarItem.selectedImage = [selectImage imageWithRenderingMode:UIImageRenderingModeAlwaysOriginal];
        }];
    

    查看文章:

    1. GCD多线程之多任务并发等待所有任务完成
    2. iOS Tabbar 实现自定义图片 放置用户头像
    3. iOS sdwebimage的独立缓存问题
    4. iOS-SDWebImage缓存图片

    相关文章

      网友评论

          本文标题:iOS 动态更换UITabBarController的图标

          本文链接:https://www.haomeiwen.com/subject/xvopoctx.html