繁体   English   中英

iOS-如何将png图像转换为4位位图灰度?

[英]iOS - How to convert a png image to 4-bits bitmap grayscale?

这样的话

我想使用iOS6 +将PNG图像(或JPEG)转换为4位位图灰度图像,即4位位图仅需要支持16种灰色。

我能怎么做 ?

提前致谢。

您可以将其转换为8位灰度图像(请参阅将UIImage转换为8位 ,或查看Spynet共享的链接,其中包括其他排列),但是我无法适应它来创建4位灰度位图( CGBitmapContextCreate抱怨说这不是有效的参数组合)。 从理论上讲,这似乎应该是可行的( CVPixelBuffer 像素格式类型下列出了4位格式),但是我无法使这种技术起作用。

我找到了将图像转换为4位灰度的解决方案,代码如下。

-(UIImage *)grayscaleImageAt4Bits:(UIImage *)_image
{
    CGImageRef imageRef = [_image CGImage];
    int width  = CGImageGetWidth(imageRef);
    int height = CGImageGetHeight(imageRef);
    NSInteger _bitsPerComponent = 8;
    NSInteger _bytesPerPixel    = 1;
    NSInteger _bytesPerRow      = _bytesPerPixel * width;
    CGColorSpaceRef colorSpace  = CGColorSpaceCreateDeviceGray();
    uint8_t *sourceData = malloc(height * width * _bytesPerPixel);
    CGContextRef context = CGBitmapContextCreate(sourceData,
                                                 width,
                                                 height,
                                                 _bitsPerComponent,
                                                 _bytesPerRow,
                                                 colorSpace,
                                                 kCGImageAlphaNone);
    CGContextDrawImage(context, CGRectMake(0, 0, width, height), imageRef);

    const int _redIndex   = 0;
    const int _greenIndex = 1;
    const int _blueIndex  = 2;
    int _byteIndex  = 0;
    int _bitIndex   = 0;
    uint8_t *_newImageBits = malloc( height * width * _bytesPerPixel / 2 );
    for(int j=0;j<height;j++)
    {
        for(int k=0;k<width;k++)
        {
            UInt8 _perPixel = _newImageBits[_byteIndex];
            UInt8 *rgbPixel = (UInt8 *) &sourceData[j * width + k];
            int _red   = rgbPixel[_redIndex];
            int _green = rgbPixel[_greenIndex];
            int _blue  = rgbPixel[_blueIndex];
            int _pixelGrayValue = (_green + _red + _blue) / 16;
            UInt8 _pixelByte = (UInt8)_pixelGrayValue;
            _perPixel |= (_pixelByte<<4);
            _newImageBits[_byteIndex] = _perPixel;
            _bitIndex += 4;
            if(_bitIndex > 7)
            {
                _byteIndex++;
                _bitIndex = 0;
            }
        }
    }
    free(sourceData);
    //Direct fetch Bytes of 4-Bits, then you can use this Bytes ( sourceData ) to do something.
    sourceData = _newImageBits;
    CGImageRef cgImage  = CGBitmapContextCreateImage(context);
    UIImage *_grayImage = [UIImage imageWithCGImage:cgImage];
    CGContextRelease(context);
    CGColorSpaceRelease(colorSpace);
    //Convert to NSData type, then you can use this NSData ( _sourceImageData ) to do something.
    NSData *_sourceImageData  = [NSData dataWithBytes:sourceData length:(width * height * _bytesPerPixel)];
    free(sourceData);
    return _grayImage;
}

而且iOS仅支持显示最少8位的颜色,因此4位数据仅传输到openGL的纹理以供使用。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM