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报错无解,需要重新自封的控件。

对您有用的话顺手点个赞,转载请注明出处~

你可能感兴趣的:(UIImagePickerController上传漫画滤镜图片失败)