iOS 仿微信群頭像,簡單實用

前沿

最近公司也在搞一個群聊功能贪绘,再看到產(chǎn)品設計后就想到了肯定是要仿照微信進行群成員頭像拼接的方式了徘键。說時遲那時快练对,兩三下就搞定了哈!為了方便各位碼友吹害,特將源碼獻上螟凭。不用客氣。

先來一張示例圖
7人群聊.jpg

言歸正傳

1它呀、使用場景

一般情況都是在cell上面進行展示螺男,如果頻繁的進行圖片拼接操作肯定非常耗性能,因此肯定要進行性能優(yōu)化纵穿。

2下隧、使用邏輯

各位明白人肯定想到用緩存的,類似于微信再不存在點擊群聊詳情的時候我們沒有必要去重新生成新的群聊拼接圖片谓媒。我這邊采用了兩張方式獲取緩存的拼接圖片淆院。

/**
 類似于微信:如果不點擊圖片詳情則不會主動更新群圖片
 */
- (NSData*)avatarDataWithGroupId:(NSString*)groupId headImgUrls:(NSString*)imgUrls;
/**
 在非詳情時,根據(jù)群id獲取頭像
 */
- (NSData*)avatarDataWithGroupId:(NSString*)groupId;
實現(xiàn)方法 僅供參考
- (NSData*)avatarDataWithGroupId:(NSString*)groupId headImgUrls:(NSString*)imgUrls
{
    if(groupId == nil || imgUrls.length == 0)return nil;
    [_db open];
    NSData* resultData = nil;
    // 根據(jù)請求參數(shù)查詢數(shù)據(jù)
    FMResultSet *resultSet = nil;
    resultSet = [_db executeQuery:@"SELECT * FROM IM_GROUP_AVATAR_Table WHERE groupid = ? AND imgurls = ? ;",groupId,imgUrls];
    // 遍歷查詢結(jié)果
    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;
    // 根據(jù)請求參數(shù)查詢數(shù)據(jù)
    FMResultSet *resultSet = nil;
    resultSet = [_db executeQuery:@"SELECT * FROM IM_GROUP_AVATAR_Table WHERE groupid = ?;",groupId];
    // 遍歷查詢結(jié)果
    while (resultSet.next) {
        NSData *value = [resultSet objectForColumn:@"avatar"];
        if(value){
            resultData = value;
            break;
        }
    }
    [_db close];
    if(resultData == nil){
        return nil;
    }
    return resultData;
}

///當通過groupid和imageurls尋找頭像無法找到時土辩,則說明頭像更改了支救,則先將數(shù)據(jù)庫中的無效群頭像刪除
- (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.創(chuàng)建上下文尺寸
                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里面的元素沒有確保按位置匹配补君,這個一般應該不會要求吧引几,如果有要求的話,可以使用信號量挽铁,每次加載一張圖片完成后再加載第二張伟桅,這樣順序就不會亂了。

以上代碼有點長叽掘,但是呢已經(jīng)將整個流程代碼貼上了楣铁,不喜歡可以噴我哈,我只是想分享一下更扁,如果喜歡就點個贊吧盖腕!

?著作權歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市浓镜,隨后出現(xiàn)的幾起案子溃列,更是在濱河造成了極大的恐慌,老刑警劉巖膛薛,帶你破解...
    沈念sama閱讀 222,183評論 6 516
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件听隐,死亡現(xiàn)場離奇詭異,居然都是意外死亡哄啄,警方通過查閱死者的電腦和手機雅任,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,850評論 3 399
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來增淹,“玉大人椿访,你說我怎么就攤上這事乌企÷侨螅” “怎么了?”我有些...
    開封第一講書人閱讀 168,766評論 0 361
  • 文/不壞的土叔 我叫張陵加酵,是天一觀的道長拳喻。 經(jīng)常有香客問我哭当,道長,這世上最難降的妖魔是什么冗澈? 我笑而不...
    開封第一講書人閱讀 59,854評論 1 299
  • 正文 為了忘掉前任钦勘,我火速辦了婚禮,結(jié)果婚禮上亚亲,老公的妹妹穿的比我還像新娘彻采。我一直安慰自己,他們只是感情好捌归,可當我...
    茶點故事閱讀 68,871評論 6 398
  • 文/花漫 我一把揭開白布肛响。 她就那樣靜靜地躺著,像睡著了一般惜索。 火紅的嫁衣襯著肌膚如雪特笋。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 52,457評論 1 311
  • 那天巾兆,我揣著相機與錄音猎物,去河邊找鬼。 笑死角塑,一個胖子當著我的面吹牛蔫磨,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播圃伶,決...
    沈念sama閱讀 40,999評論 3 422
  • 文/蒼蘭香墨 我猛地睜開眼质帅,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了留攒?” 一聲冷哼從身側(cè)響起煤惩,我...
    開封第一講書人閱讀 39,914評論 0 277
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎炼邀,沒想到半個月后魄揉,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 46,465評論 1 319
  • 正文 獨居荒郊野嶺守林人離奇死亡拭宁,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,543評論 3 342
  • 正文 我和宋清朗相戀三年洛退,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片杰标。...
    茶點故事閱讀 40,675評論 1 353
  • 序言:一個原本活蹦亂跳的男人離奇死亡兵怯,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出腔剂,到底是詐尸還是另有隱情媒区,我是刑警寧澤,帶...
    沈念sama閱讀 36,354評論 5 351
  • 正文 年R本政府宣布,位于F島的核電站袜漩,受9級特大地震影響绪爸,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜宙攻,卻給世界環(huán)境...
    茶點故事閱讀 42,029評論 3 335
  • 文/蒙蒙 一奠货、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧座掘,春花似錦递惋、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,514評論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至嬉愧,卻和暖如春贩挣,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背没酣。 一陣腳步聲響...
    開封第一講書人閱讀 33,616評論 1 274
  • 我被黑心中介騙來泰國打工王财, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人裕便。 一個月前我還...
    沈念sama閱讀 49,091評論 3 378
  • 正文 我出身青樓绒净,卻偏偏與公主長得像,于是被迫代替她去往敵國和親偿衰。 傳聞我的和親對象是個殘疾皇子挂疆,可洞房花燭夜當晚...
    茶點故事閱讀 45,685評論 2 360

推薦閱讀更多精彩內(nèi)容

  • Swift1> Swift和OC的區(qū)別1.1> Swift沒有地址/指針的概念1.2> 泛型1.3> 類型嚴謹 對...
    cosWriter閱讀 11,113評論 1 32
  • 愛在流淌中。 好孩子祈禱文 OOO下翎,爸爸媽媽的心肝寶貝缤言。 很高興你來到這個家。 你是個多么可愛的寶貝视事。 我們很高興...
    牛牛簡書閱讀 611評論 0 0
  • 現(xiàn)在已經(jīng)不想想象讀完大學以后要干什么胆萧。 原先暢想的都是工作會賺到錢,然后過自己想過的生活俐东。但是工作又豈會比讀大學容...
    fari閱讀 359評論 0 0
  • 《終身成長》 這本書的作者是斯坦福大學心理學家 卡羅爾·德韋克 在對成功的數(shù)十年研究后跌穗,她在《終身成長》中表明: ...
    呂長勇閱讀 7,676評論 5 26