有用户反馈更换漫画滤镜照片作为头像,设置失败。
漫画滤镜照片是iOS12的新功能:漫画滤镜照片生成攻略
一、复现
(为了复现,我冒着生命危险把手机升级到了iOS12)
步骤:更换头像——选择漫画滤镜图片——卡住了+必现
此时点击取消和选取两个按钮均无反应,只能右滑返回。
点击选取时控制台看到报错信息:
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有其他地方使用了自己封装的控件。
测试结果:选取展示成功,点击发布提示失败
发布失败原因:文件为空
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的实现,这样的话一处改动,处处生效,非常之完美。
然而想法是美好的。
UIImageJPEGRepresentation是个Function,method_exchangeImplementations只能用于类方法和实例方法,C实现的函数的交换,真的做不到啊(如果有办法实现,请告诉我)。。
所以,不要用UIImagePickerController了,自己封装一套或者找个现成的库集成一下吧。
四、总结
问题描述:
iOS12,漫画滤镜的图片上传失败,不限机型
原因:
选取或者上传过程中使用了UIImageJPEGRepresentation
方法,该方法在无CGImageRef或者包含无效位图数据时返回nil,滤镜图片命中该情况,导致最终上传失败。
UIImagePickerController内部同样调用了UIImageJPEGRepresentation,触发同样的错误。
解决:
自己封装的控件,在UIImageJPEGRepresentation返回为nil时,重绘图片然后重新压缩,该方法已测有效。
UIImagePickerController报错无解,需要重新自封的控件。