UIImagePickerController上传漫画滤镜图片失败

有用户反馈更换漫画滤镜照片作为头像,设置失败。
漫画滤镜照片是iOS12的新功能:漫画滤镜照片生成攻略

一、复现

(为了复现,我冒着生命危险把手机升级到了iOS12)

步骤:更换头像——选择漫画滤镜图片——卡住了+必现


上传漫画滤镜头像卡住了.jpeg

此时点击取消和选取两个按钮均无反应,只能右滑返回。
点击选取时控制台看到报错信息:

2018-11-19 15:37:01.512176+0800 NeiTui[4801:1041093] IIORecodeHEIF_to_JPEG:1831: *** FigPhotoDecompressionContainerJFIFTranscode failed err = kFigPhotoError_UnsupportedOperation [-16994]
2018-11-19 15:37:01.512364+0800 NeiTui[4801:1041093] *** Assertion failure in -[PUPhotoPickerExtensionHostContext _pathExtensionFromData:url:], /BuildRoot/Library/Caches/com.apple.xbs/Sources/MobileSlideShow/MobileSlideShow-3412.10.210/Projects/PhotosUI/PUPhotoPickerExtensionHostContext.m:442

第一条报错查询无果,第二条PUPhotoPickerExtensionHostContext看不到内部实现。

然后找别家App试了下,测试时间18年11月16日,测试结果如下:

  • 淘宝和美颜相机更换漫画图片成功,用的是非系统图片选择器
  • 知乎更换失败,用的系统UIImagePickerController(更新:18年11月21日 知乎换了非系统的图片选择器,更换成功)

所以,用自己封装的控件是有可能上传成功的。

二、自己封装的上传图片控件

恰好我们的App有其他地方使用了自己封装的控件。


自封控件测试.jpeg

测试结果:选取展示成功,点击发布提示失败
发布失败原因:文件为空

Debug调试发现,发布之前先把图片存储到Disk中:

//存储到Disk方法
 NSData *data = UIImageJPEGRepresentation(image, quaulity);
 size = data.length;
        NSError *error;
        [data writeToFile:imageFilePath options:NSDataWritingAtomic error:&error];
        if (error) {
            [[NTGlobal getInstance].errorLogger log:[NSString stringWithFormat:@"Publish SaveImage Failed, %@", imageFilePath] withError:error];
        }

然后上传Disk Path里的图片文件,最终文件为空。
打断点,发现
UIImageJPEGRepresentation(image, quaulity);
当image为漫画滤镜图片时,返回nil

同时报错:

2018-11-19 16:06:03.908717+0800 NeiTui[4919:1052863] IIORecodeHEIF_to_JPEG:1831: *** FigPhotoDecompressionContainerJFIFTranscode failed err = kFigPhotoError_UnsupportedOperation [-16994]
(lldb) 

该错误和UIImagePickerController报错相同,虽然我们看不到其内部实现,但是可以由此反推,UIImagePickerController在选取图片时也调用了UIImageJPEGRepresentation方法,从而选取失败。

三、解决

UIKIT_EXTERN  NSData * __nullable UIImageJPEGRepresentation(UIImage * __nonnull image, CGFloat compressionQuality);  // return image as JPEG. May return nil if image has no CGImageRef or invalid bitmap format. compression is 0(most)..1(least)

UIImageJPEGRepresentation官方说明,图片若是不支持的位图格式,会返回nil。

返回nil可以通过重绘image解决

+ (UIImage *)redrawImage:(UIImage *)image{
    //确定压缩后的size
    CGFloat redrawWidth = image.size.width;
    CGFloat redrawHeight = image.size.height;
    CGSize redrawSize = CGSizeMake(redrawWidth, redrawHeight);
    //开启图形上下文
    UIGraphicsBeginImageContext(redrawSize);
    //绘制图片
    [image drawInRect:CGRectMake(0, 0, redrawWidth, redrawHeight)];
    //从图形上下文获取图片
    UIImage *newImage = UIGraphicsGetImageFromCurrentImageContext();
    //关闭图形上下文
    UIGraphicsEndImageContext();
    return newImage;
}

+ (NSArray *)saveImageToDisk:(UIImage *)image quaulity:(CGFloat)quaulity singleThumb:(BOOL)singleThumb {
   ...
        NSData *data = UIImageJPEGRepresentation(image, quaulity);
        //处理data为nil
        if (!data) {
            UIImage *scaleImage = [self redrawImage:image];
            data = UIImageJPEGRepresentation(scaleImage, quaulity);
        }
        size = data.length; 
...
}

然后,自己封装的控件上传成功了。

重点来了,使用系统的UIImagePickerController,怎么解决?

我的第一个念头就是黑魔法,swizzle UIImageJPEGRepresentation的实现,这样的话一处改动,处处生效,非常之完美。

然而想法是美好的。

image.png

UIImageJPEGRepresentation是个Function,method_exchangeImplementations只能用于类方法和实例方法,C实现的函数的交换,真的做不到啊(如果有办法实现,请告诉我)。。

所以,不要用UIImagePickerController了,自己封装一套或者找个现成的库集成一下吧。

四、总结

问题描述:

iOS12,漫画滤镜的图片上传失败,不限机型

原因:

选取或者上传过程中使用了UIImageJPEGRepresentation
方法,该方法在无CGImageRef或者包含无效位图数据时返回nil,滤镜图片命中该情况,导致最终上传失败。

UIImagePickerController内部同样调用了UIImageJPEGRepresentation,触发同样的错误。

解决:

自己封装的控件,在UIImageJPEGRepresentation返回为nil时,重绘图片然后重新压缩,该方法已测有效。

UIImagePickerController报错无解,需要重新自封的控件。

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

推荐阅读更多精彩内容

  • 1、通过CocoaPods安装项目名称项目信息 AFNetworking网络请求组件 FMDB本地数据库组件 SD...
    阳明先生_x阅读 15,967评论 3 119
  • 喝点酒好啊多好晕晕乎乎的什么也不知道不会乱想不会失眠我想要你幸福快乐平安我想要你脸上的光一直在我羡慕嫉妒抱着你的那...
    小妮子诶阅读 178评论 0 0
  • 好久i老友阅读 143评论 0 0
  • 即将毕业 今天也是最后一次演讲 平时吧啦吧啦的说个不停歇 真要正儿八经的说时 却不知说些什么 这是病 得治 ! 昨...
    小小太阳000阅读 112评论 0 0
  • 二十岁在即 有过几次炽烈难忘的感情 也同样没有预期地溶解在时间的洪流里 把瞳孔曾溢满出的挚爱瓦解 然后再重组成为不...
    伊昂obey阅读 799评论 0 1