iOS 滤镜之CIBlendWithAlphaMask在特殊系统渲染失败的解决方案

遇到一例案例,iOS 10.1.1 使用CIBlendWithAlphaMask 渲染生成马赛克涂抹效果时,没有渲染出来,改为GPU绘制后,倒就正常了。原因未知,暂且记录下解决方案。

已知问题机型:

iphone 7p   iOS 10.1.1
iphone 6sp  iOS 10.1.1
iphone 6        iOS 10.1.1
(iphone 5 iOS 10.1.1 没有问题)

原实现:

CIImage *resultImage = [[CIFilter filterWithName:@"CIBlendWithAlphaMask"
                                       keysAndValues:kCIInputImageKey, ciOutputImagePixellate,
                                                    kCIInputMaskImageKey, maskImage,
                                                    kCIInputBackgroundImageKey, ciInputImage, nil]
                            valueForKey:kCIOutputImageKey];

CIContext *ciContext = [CIContext contextWithOptions:nil];
CGImageRef cgImage = [ciContext createCGImage:resultImage fromRect:[resultImage extent]];

CGImageRef cgImage = [ciContext createCGImage:resultImage fromRect:[resultImage extent]];
    
// 获取UIImage
UIImage *filteredImage = [UIImage imageWithCGImage:cgImage];
CGImageRelease(cgImage);

修改后实现方案:

CIImage *resultImage = [[CIFilter filterWithName:@"CIBlendWithAlphaMask"
                                       keysAndValues:kCIInputImageKey, ciOutputImagePixellate,
                                                    kCIInputMaskImageKey, maskImage,
                                                    kCIInputBackgroundImageKey, ciInputImage, nil]
                            valueForKey:kCIOutputImageKey];
                            
EAGLContext *glContext = [[EAGLContext alloc] initWithAPI:kEAGLRenderingAPIOpenGLES2];

CIContext *ciContext = [CIContext contextWithEAGLContext:glContext
                                                     options:@{kCIContextWorkingColorSpace : [NSNull null]}];
    
CGImageRef cgImage = [ciContext createCGImage:resultImage fromRect:[resultImage extent]];
    
// 获取UIImage
UIImage *filteredImage = [UIImage imageWithCGImage:cgImage];
CGImageRelease(cgImage);

附上图片部分马赛克的源码:

- (UIImage *)demoCoreImageFilter {
    
    UIImage *originImage = [UIImage imageNamed:@"facetest"];
    // 导入CIImage
    CIImage *ciInputImage = [[CIImage alloc] initWithImage:originImage];
    
    // 创建CIFilter
    CIFilter *filterPixellate = [CIFilter filterWithName:@"CIPixellate"];
    [filterPixellate setValue:ciInputImage forKey:kCIInputImageKey];
    [filterPixellate setDefaults];
    [filterPixellate setValue:@32 forKey:@"inputScale"];
    NSLog(@"filterPixellate : %@", filterPixellate.attributes);
    
    // 获取滤镜效果之后的CIImage
    CIImage *ciOutputImagePixellate = [filterPixellate valueForKey:kCIOutputImageKey];
    
    
    // 人脸识别
    CIDetector *detector = [CIDetector detectorOfType:CIDetectorTypeFace
                                              context:nil
                                              options:nil];
    NSArray *faceArray = [detector featuresInImage:ciInputImage options:nil];
    
    // Create a green circle to cover the rects that are returned.
    
    CIImage *maskImage = nil;
    
    for (CIFeature *f in faceArray) {
        CGFloat centerX = f.bounds.origin.x + f.bounds.size.width / 2.0;
        CGFloat centerY = f.bounds.origin.y + f.bounds.size.height / 2.0;
        CGFloat radius = MIN(f.bounds.size.width, f.bounds.size.height) / 1.5;
        CIFilter *radialGradient = [CIFilter filterWithName:@"CIRadialGradient" withInputParameters:@{
                                                                                                      @"inputRadius0": @(radius),
                                                                                                      @"inputRadius1": @(radius + 1.0f),
                                                                                                      @"inputColor0": [CIColor colorWithRed:0.0 green:1.0 blue:0.0 alpha:1.0],
                                                                                                      @"inputColor1": [CIColor colorWithRed:0.0 green:0.0 blue:0.0 alpha:0.0],
                                                                                                      kCIInputCenterKey: [CIVector vectorWithX:centerX Y:centerY],
                                                                                                      }];
        CIImage *circleImage = [radialGradient valueForKey:kCIOutputImageKey];
        if (nil == maskImage)
            maskImage = circleImage;
        else
            maskImage = [[CIFilter filterWithName:@"CISourceOverCompositing" withInputParameters:@{
                                                                                                   kCIInputImageKey: circleImage,
                                                                                                   kCIInputBackgroundImageKey: maskImage,
                                                                                                   }] valueForKey:kCIOutputImageKey];
    }
    
    CIImage *resultImage = [[CIFilter filterWithName:@"CIBlendWithAlphaMask"
                                       keysAndValues:kCIInputImageKey, ciOutputImagePixellate,
                                                    kCIInputMaskImageKey, maskImage,
                                                    kCIInputBackgroundImageKey, ciInputImage, nil]
                            valueForKey:kCIOutputImageKey];
    
    EAGLContext *glContext = [[EAGLContext alloc] initWithAPI:kEAGLRenderingAPIOpenGLES2];
    CIContext *ciContext = [CIContext contextWithEAGLContext:glContext
                                                     options:@{kCIContextWorkingColorSpace : [NSNull null]}];
    CGImageRef cgImage = [ciContext createCGImage:resultImage fromRect:[resultImage extent]];
    
    // 获取UIImage
    UIImage *filteredImage = [UIImage imageWithCGImage:cgImage];
    CGImageRelease(cgImage);
    
    return filteredImage;
}

代码讲解:

  1. 将原图马赛克化,得到一张马赛克后的图片 ciOutputImagePixellate
  2. 对原图人脸识别,并取识别后的人脸区域 maskImage
  3. 使用 CIBlendWithAlphaMask 滤镜进行合成
  4. 指定GPU 渲染

参考链接

Anonymous Faces Filter Recipe

你可能感兴趣的:(iOS 滤镜之CIBlendWithAlphaMask在特殊系统渲染失败的解决方案)