2

我有一个 CGImage(核心图形,C/C++)。是灰度。嗯,原来是黑白的,但是CGImage可能是RGB的。那应该没关系。我想创建一个 CCITT-Group 4 TIFF。

我可以通过使用正确的字典创建目标并添加图像来创建 LZW TIFF(灰度或彩色)。没问题。

但是,似乎没有等效的 kCGImagePropertyTIFFCompression 值来表示 CCITT-4。它应该是 4,但这会产生未压缩。

我有一个手动 CCITT 压缩例程,所以如果我能得到二进制(每像素 1 位)数据,我就准备好了。但我似乎无法从 CGImage 中获取 1 个 BPP 数据。我有应该将 CGImage 放入 CGBitmapContext 然后给我数据的代码,但它似乎给了我全黑。

我今天问了几个问题试图解决这个问题,但我只是想,让我们问我真正想回答的问题,看看是否有人能回答。

必须有办法做到这一点。我必须错过一些愚蠢的东西。它是什么?

4

3 回答 3

3

这似乎工作并产生并非全黑的输出。可能有一种方法可以做到这一点,它不涉及首先手动转换为灰度,但至少它有效!

static void WriteCCITTTiffWithCGImage_URL_(CGImageRef im, CFURLRef url) {
    // produce grayscale image
    CGImageRef grayscaleImage;
    {
        CGColorSpaceRef colorSpace = CGColorSpaceCreateWithName(kCGColorSpaceGenericGray);
        CGContextRef bitmapCtx = CGBitmapContextCreate(NULL, CGImageGetWidth(im), CGImageGetHeight(im), 8, 0, colorSpace, kCGImageAlphaNone);
        CGContextDrawImage(bitmapCtx, CGRectMake(0,0,CGImageGetWidth(im), CGImageGetHeight(im)), im);
        grayscaleImage = CGBitmapContextCreateImage(bitmapCtx);
        CFRelease(bitmapCtx);
        CFRelease(colorSpace);
    }

    // generate options for ImageIO. Man this sucks in C.
    CFMutableDictionaryRef options = CFDictionaryCreateMutable(kCFAllocatorDefault, 2, &kCFTypeDictionaryKeyCallBacks, &kCFTypeDictionaryValueCallBacks);
    {
        {
            CFMutableDictionaryRef tiffOptions = CFDictionaryCreateMutable(kCFAllocatorDefault, 1, &kCFTypeDictionaryKeyCallBacks, &kCFTypeDictionaryValueCallBacks);
            int fourInt = 4;
            CFNumberRef fourNumber = CFNumberCreate(kCFAllocatorDefault, kCFNumberIntType, &fourInt);
            CFDictionarySetValue(tiffOptions, kCGImagePropertyTIFFCompression, fourNumber);
            CFRelease(fourNumber);

            CFDictionarySetValue(options, kCGImagePropertyTIFFDictionary, tiffOptions);

            CFRelease(tiffOptions);                
        }

        {
            int oneInt = 1;
            CFNumberRef oneNumber = CFNumberCreate(kCFAllocatorDefault, kCFNumberIntType, &oneInt);

            CFDictionarySetValue(options, kCGImagePropertyDepth, oneNumber);

            CFRelease(oneNumber);
        }
    }

    // write file
    CGImageDestinationRef idst = CGImageDestinationCreateWithURL(url, kUTTypeTIFF, 1, NULL);
    CGImageDestinationAddImage(idst, grayscaleImage, options);
    CGImageDestinationFinalize(idst);

    // clean up
    CFRelease(idst);
    CFRelease(options);
    CFRelease(grayscaleImage);
}


Nepheli:tmp ken$ tiffutil -info /tmp/output.tiff 
Directory at 0x1200
  Image Width: 842 Image Length: 562
  Bits/Sample: 1
  Sample Format: unsigned integer
  Compression Scheme: CCITT Group 4 facsimile encoding
  Photometric Interpretation: "min-is-black"
  Orientation: row 0 top, col 0 lhs
  Samples/Pixel: 1
  Number of Strips: 1
  Planar Configuration: Not planar
于 2009-04-27T10:22:16.687 回答
1

ImageMagick几乎可以转换为任何图像格式。由于它是开源的,您可以阅读源代码以找到您问题的答案。

如果您使用 C++,您甚至可以在您的应用程序中使用ImageMagick API 。

编辑:
如果您可以从 CGImage 以任何格式获取数据(听起来像您可以),您可以使用 ImageMagick 将其从您从 CGImage 获得的任何格式转换为 ImageMagick 支持的任何其他格式(您想要的 TIFF 格式)。

编辑: 技术问答 QA1509 从 CGImage 对象中获取像素数据状态:

在 Mac OS X 10.5 或更高版本上,添加了一个新调用,允许您从 CGImage 对象获取实际像素数据。此调用 CGDataProviderCopyData 返回一个 CFData 对象,其中包含来自相关图像的像素数据。

获得像素数据后,您可以使用 ImageMagick 对其进行转换。

于 2009-04-16T23:56:00.017 回答
1

NSBitmapImageRep 声称能够生成 CCITT FAX Group 4 压缩 TIFF。所以这样的事情可能会成功(未经测试):

CFDataRef tiffFaxG4DataForCGImage(CGImageRef cgImage) {
  NSBitmapImageRep *imageRep =
      [[[NSBitmapImageRep alloc] initWithCGImage:cgImage] autorelease];
  NSData *tiffData =
      [imageRep TIFFRepresentationUsingCompression:NSTIFFCompressionCCITTFAX4
                                            factor:0.0f];
  return (CFDataRef) tiffData;
}

此函数应返回您查找的数据。

于 2009-04-27T09:29:50.697 回答