iOS 仿微信群头像,简单实用

前沿

最近公司也在搞一个群聊功能,再看到产品设计后就想到了肯定是要仿照微信进行群成员头像拼接的方式了。说时迟那时快,两三下就搞定了哈!为了方便各位码友,特将源码献上。不用客气。

先来一张示例图
7人群聊.jpg

言归正传

1、使用场景

一般情况都是在cell上面进行展示,如果频繁的进行图片拼接操作肯定非常耗性能,因此肯定要进行性能优化。

2、使用逻辑

各位明白人肯定想到用缓存的,类似于微信再不存在点击群聊详情的时候我们没有必要去重新生成新的群聊拼接图片。我这边采用了两张方式获取缓存的拼接图片。

/**
 类似于微信:如果不点击图片详情则不会主动更新群图片
 */
- (NSData*)avatarDataWithGroupId:(NSString*)groupId headImgUrls:(NSString*)imgUrls;
/**
 在非详情时,根据群id获取头像
 */
- (NSData*)avatarDataWithGroupId:(NSString*)groupId;
实现方法 仅供参考
- (NSData*)avatarDataWithGroupId:(NSString*)groupId headImgUrls:(NSString*)imgUrls
{
    if(groupId == nil || imgUrls.length == 0)return nil;
    [_db open];
    NSData* resultData = nil;
    // 根据请求参数查询数据
    FMResultSet *resultSet = nil;
    resultSet = [_db executeQuery:@"SELECT * FROM IM_GROUP_AVATAR_Table WHERE groupid = ? AND imgurls = ? ;",groupId,imgUrls];
    // 遍历查询结果
    while (resultSet.next) {
        NSData *value = [resultSet objectForColumn:@"avatar"];
        if(value){
            resultData = value;
            break;
        }
    }
    [_db close];
    if(resultData == nil){///这种情况下找不到对应的图片,则说明群聊的图片集合更新了
        [self deleteGroupAvatarDataWithGroupId:groupId];
        return nil;
    }
    return resultData;
}

- (NSData*)avatarDataWithGroupId:(NSString*)groupId
{
    if(groupId == nil)return nil;
    [_db open];
    NSData* resultData = nil;
    // 根据请求参数查询数据
    FMResultSet *resultSet = nil;
    resultSet = [_db executeQuery:@"SELECT * FROM IM_GROUP_AVATAR_Table WHERE groupid = ?;",groupId];
    // 遍历查询结果
    while (resultSet.next) {
        NSData *value = [resultSet objectForColumn:@"avatar"];
        if(value){
            resultData = value;
            break;
        }
    }
    [_db close];
    if(resultData == nil){
        return nil;
    }
    return resultData;
}

///当通过groupid和imageurls寻找头像无法找到时,则说明头像更改了,则先将数据库中的无效群头像删除
- (void)deleteGroupAvatarDataWithGroupId:(NSString*)groupId
{
    [_db open];
    BOOL isFlag = [_db executeUpdate:@"DELETE FROM IM_GROUP_AVATAR_Table WHERE groupid = ? ;",groupId];
    [_db close];
    if(isFlag){
        KFLog(@"已删除无效的群头像");
    }
}
图片的拼接

我这里的图片加载方式是大家都用的SDWebImage,相信你不会陌生!
这里就直接贴上拼接的源码,

///给一个宽度和高度一致的大图片,用来显示画布的大小
static const CGFloat nomalViewWidth = 200;

@implementation BDCreatGroupAvatar

+ (void)createGroupAvatar:(NSArray *)group completeBlock:(void (^)(NSData *groupAvatar))completeBlock {
    
    __block NSInteger loadCount = 0;
    NSInteger avatarCount = group.count > 9 ? 9 : group.count;
    CGFloat width = 0;//先默认宽带
    CGFloat space = 2;//图片的间距
    CGFloat x = 0;
    CGFloat y = 0;
    if(avatarCount == 1){
        width = nomalViewWidth-2*3;
    }else if (avatarCount >= 2 && avatarCount <= 4){
        width = (nomalViewWidth-3*space)*0.5;
    }else{
        width = 1.0*(nomalViewWidth-4*space)/3;
    }
    
    NSMutableArray *resultDataArray = [NSMutableArray arrayWithCapacity:group.count];
    NSMutableArray *resultFrameArray = [NSMutableArray arrayWithCapacity:group.count];
    for (int i = 0; i<group.count; i++) {
        NSString *avatarUrl = group[I];
        if(avatarCount == 1){
            x = 3;
            y = 3;
        }else if (avatarCount == 2){
            x = (i*width)+space;
            y = (nomalViewWidth-width)*0.5;
        }else if (avatarCount == 3){
            if(i == 0){
                x = (nomalViewWidth-width)*0.5;
                y = space;
            }else if (i == 1){
                x = space;
                y = width+2*space;
            }else{
                x = width+2*space;
                y = width+2*space;
            }
        }else if (avatarCount == 4){
            NSInteger maxLinePerCount = 2;
            x = (i%maxLinePerCount) * (width+space)+space;
            y = (i/maxLinePerCount) * (width+space)+space;
        }else if (avatarCount == 5){
            if(i == 0){
                x = (nomalViewWidth*0.5)-(space*0.5)-width;
                y = (nomalViewWidth-2*width-space)*0.5;
            }else if (i == 1){
                x = (nomalViewWidth*0.5)+(space*0.5);
                y = (nomalViewWidth-2*width-space)*0.5;
            }else{
                x = (i-2)*(width+space)+space;
                y = (nomalViewWidth-2*width-space)*0.5+width+space;
            }
        }else if (avatarCount == 6){
            NSInteger maxLinePerCount = 3;
            CGFloat ySpace = (nomalViewWidth-2*width-space)*0.5;
            x = (i%maxLinePerCount) * (width+space)+space;
            y = (i/maxLinePerCount) * (width+space)+ySpace;
        }else{
            NSInteger maxLinePerCount = 3;
            x = (i%maxLinePerCount) * (width+space)+space;
            y = (i/maxLinePerCount) * (width+space)+space;
        }
        CGRect imageFrame = CGRectMake(x, y, width, width);
        NSValue *frameValue = [NSValue valueWithCGRect:imageFrame];
        [resultFrameArray addObject:frameValue];
        
        [[SDWebImageManager sharedManager] loadImageWithURL:[NSURL URLWithString:avatarUrl] options:0 progress:nil completed:^(UIImage * _Nullable image, NSData * _Nullable data, NSError * _Nullable error, SDImageCacheType cacheType, BOOL finished, NSURL * _Nullable imageURL) {
            
            loadCount ++ ;
            UIImage *loadImage = nil;
            if(image == nil){
                loadImage = BD_PLACE_HEAD_IMAGE;
            }else{
                loadImage = image;
            }
            ///将压缩后的图片进行暂存
            NSData *resultData = [loadImage compressQualityWithMaxLength:1024*1024*0.5];
            [resultDataArray addObject:resultData];
            if (loadCount == avatarCount) {     //图片全部下载完成
                CGSize contentSize = CGSizeMake(nomalViewWidth, nomalViewWidth);
                //1.创建上下文尺寸
                UIGraphicsBeginImageContextWithOptions(contentSize, true, 0.0);
                CGContextRef ref = UIGraphicsGetCurrentContext();
                CGContextSetRGBFillColor(ref, 1.0*239/255, 1.0*239/255, 1.0*239/255, 1);
                CGContextFillRect(ref, CGRectMake(0, 0, nomalViewWidth, nomalViewWidth));
                for (int i = 0; i<resultDataArray.count; i++) {
                    NSData *imageData = resultDataArray[I];
                    NSValue *frameValue = resultFrameArray[I];
                    CGRect imageViewFrame = frameValue.CGRectValue;
                    UIImage *image = [[UIImage alloc]initWithData:imageData];
                    [image drawInRect:imageViewFrame];
                }
                UIImage *resultImg = UIGraphicsGetImageFromCurrentImageContext();//从当前上下文中获得最终图片
                UIGraphicsEndImageContext();//关闭上下文
                NSData *thetData = [resultImg compressQualityWithMaxLength:1024*1024*0.5];
                dispatch_async(dispatch_get_main_queue(), ^{
                    if (completeBlock) {
                        completeBlock(thetData);
                    }
                });
            }
        }];
    }
}
代码简单说明:

1、设置画布背景色

CGContextRef ref = UIGraphicsGetCurrentContext();
                CGContextSetRGBFillColor(ref, 1.0*239/255, 1.0*239/255, 1.0*239/255, 1);
                CGContextFillRect(ref, CGRectMake(0, 0, nomalViewWidth, nomalViewWidth));

2、以上针对1至6张图片时,是一个个判断的,如果各位有好的点子可以回复告知一下哈!

3、resultDataArray和resultFrameArray里面的元素没有确保按位置匹配,这个一般应该不会要求吧,如果有要求的话,可以使用信号量,每次加载一张图片完成后再加载第二张,这样顺序就不会乱了。

以上代码有点长,但是呢已经将整个流程代码贴上了,不喜欢可以喷我哈,我只是想分享一下,如果喜欢就点个赞吧!

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 218,941评论 6 508
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,397评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 165,345评论 0 356
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,851评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,868评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,688评论 1 305
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,414评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,319评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,775评论 1 315
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,945评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,096评论 1 350
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,789评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,437评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,993评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,107评论 1 271
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,308评论 3 372
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,037评论 2 355

推荐阅读更多精彩内容

  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,103评论 1 32
  • 爱在流淌中。 好孩子祈祷文 OOO,爸爸妈妈的心肝宝贝。 很高兴你来到这个家。 你是个多么可爱的宝贝。 我们很高兴...
    牛牛简书阅读 607评论 0 0
  • 现在已经不想想象读完大学以后要干什么。 原先畅想的都是工作会赚到钱,然后过自己想过的生活。但是工作又岂会比读大学容...
    fari阅读 358评论 0 0
  • 《终身成长》 这本书的作者是斯坦福大学心理学家 卡罗尔·德韦克 在对成功的数十年研究后,她在《终身成长》中表明: ...
    吕长勇阅读 7,670评论 5 26