有用戶反饋更換漫畫濾鏡照片作為頭像,設置失敗。
漫畫濾鏡照片是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日,測試結果如下:
- 淘寶和美顏相機更換漫畫圖片成功,用的是非系統(tǒng)圖片選擇器
- 知乎更換失敗,用的系統(tǒng)UIImagePickerController(更新:18年11月21日 知乎換了非系統(tǒng)的圖片選擇器,更換成功)
所以,用自己封裝的控件是有可能上傳成功的。
二、自己封裝的上傳圖片控件
恰好我們的App有其他地方使用了自己封裝的控件。

測試結果:選取展示成功,點擊發(fā)布提示失敗
發(fā)布失敗原因:文件為空
Debug調試發(fā)現,發(fā)布之前先把圖片存儲到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里的圖片文件,最終文件為空。
打斷點,發(fā)現
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;
...
}
然后,自己封裝的控件上傳成功了。
重點來了,使用系統(tǒng)的UIImagePickerController,怎么解決?
我的第一個念頭就是黑魔法,swizzle UIImageJPEGRepresentation的實現,這樣的話一處改動,處處生效,非常之完美。
然而想法是美好的。

UIImageJPEGRepresentation是個Function,method_exchangeImplementations只能用于類方法和實例方法,C實現的函數的交換,真的做不到啊(如果有辦法實現,請告訴我)。。
所以,不要用UIImagePickerController了,自己封裝一套或者找個現成的庫集成一下吧。
四、總結
問題描述:
iOS12,漫畫濾鏡的圖片上傳失敗,不限機型
原因:
選取或者上傳過程中使用了UIImageJPEGRepresentation
方法,該方法在無CGImageRef或者包含無效位圖數據時返回nil,濾鏡圖片命中該情況,導致最終上傳失敗。
UIImagePickerController內部同樣調用了UIImageJPEGRepresentation,觸發(fā)同樣的錯誤。
解決:
自己封裝的控件,在UIImageJPEGRepresentation返回為nil時,重繪圖片然后重新壓縮,該方法已測有效。
UIImagePickerController報錯無解,需要重新自封的控件。