How do I write a 1bpp tiff with libtiff on iOS?

前端 未结 2 1107
既然无缘
既然无缘 2021-02-11 02:10

I\'m trying to write a UIImage out as a tiff using libtiff. The problem is that even though I\'m writing it as 1 bit per pixel, the files are still coming out in the 2-5MB rang

2条回答
  •  Happy的楠姐
    2021-02-11 03:06

    I have the need to generate a TIFF image in the iPhone and send it to a remote server which is expecting TIFF files. I can't use the accepted answer which converts to 1bpp PNG and I have been working in a solution to convert to TIFF, 1bpp CCITT Group 4 format, using libTIFF.

    After debugging the method I have found where the errors are and I finally got the correct solution.

    The following block of code is the solution. Read after the code to found the explanation to the errors in the OP method.

    - (void) convertUIImage:(UIImage *)uiImage toTiff:(NSString *)file withThreshold:(float)threshold {
    
        CGImageRef srcCGImage = [uiImage CGImage];
        CFDataRef pixelData = CGDataProviderCopyData(CGImageGetDataProvider(srcCGImage));
        unsigned char *pixelDataPtr = (unsigned char *)CFDataGetBytePtr(pixelData);
    
        TIFF *tiff;
        if ((tiff = TIFFOpen([file UTF8String], "w")) == NULL) {
            [[[UIAlertView alloc] initWithTitle:@"Error" message:[NSString stringWithFormat:@"Unable to write to file %@.", file] delegate:nil cancelButtonTitle:nil otherButtonTitles:@"OK", nil] show];
            return;
        }
    
        size_t width = CGImageGetWidth(srcCGImage);
        size_t height = CGImageGetHeight(srcCGImage);
    
        TIFFSetField(tiff, TIFFTAG_IMAGEWIDTH, width);
        TIFFSetField(tiff, TIFFTAG_IMAGELENGTH, height);
        TIFFSetField(tiff, TIFFTAG_BITSPERSAMPLE, 1);
        TIFFSetField(tiff, TIFFTAG_SAMPLESPERPIXEL, 1);
        TIFFSetField(tiff, TIFFTAG_ROWSPERSTRIP, 1);
    
        TIFFSetField(tiff, TIFFTAG_COMPRESSION, COMPRESSION_CCITTFAX4);
        TIFFSetField(tiff, TIFFTAG_PHOTOMETRIC, PHOTOMETRIC_MINISWHITE);
        TIFFSetField(tiff, TIFFTAG_FILLORDER, FILLORDER_MSB2LSB);
        TIFFSetField(tiff, TIFFTAG_PLANARCONFIG, PLANARCONFIG_CONTIG);
    
        TIFFSetField(tiff, TIFFTAG_XRESOLUTION, 200.0);
        TIFFSetField(tiff, TIFFTAG_YRESOLUTION, 200.0);
        TIFFSetField(tiff, TIFFTAG_RESOLUTIONUNIT, RESUNIT_INCH);
    
        unsigned char *ptr = pixelDataPtr; // initialize pointer to the first byte of the image buffer 
        unsigned char red, green, blue, gray, eightPixels;
        tmsize_t bytesPerStrip = ceil(width/8.0);
        unsigned char *strip = (unsigned char *)_TIFFmalloc(bytesPerStrip);
    
        for (int y=0; y

    Here are the errors and the explanation of what is wrong.

    1) the allocation of memory for one scan line is 1 byte short if the width of the image is not a multiple of 8.

    unsigned char *line = (unsigned char *)_TIFFmalloc(width/8);

    should be replaced by

    tmsize_t bytesPerStrip = ceil(width/8.0); unsigned char *line = (unsigned char *)_TIFFmalloc(bytesPerStrip);

    The explanation is that we have to take the ceiling of the division by 8 in order to get the number of bytes for a strip. For example a strip of 83 pixels needs 11 bytes, not 10, or we could loose the 3 last pixels. Note also we have to divide by 8.0 in order to get a floating point number and pass it to the ceil function. Integer division in C looses the decimal part and rounds to the floor, which is wrong in our case.

    2) the last argument passed to the function TIFFWriteEncodedStrip is wrong. We can't pass the number of pixels in a strip, we have to pass the number of bytes per strip.

    So replace:

    TIFFWriteEncodedStrip(tiff, y, line, width);

    by

    TIFFWriteEncodedStrip(tiff, y, line, bytesPerStrip);

    3) A last error difficult to detect is related to the convention on whether a bit with 0 value represents white or black in the bi-tonal image. Thanks to the TIFF header TIFFTAG_PHOTOMETRIC we can safely indicate this. However I have found than some older software ignores this header. What happens if the header is not present or ignored is that a 0 bit gets interpreted as white and a 1 bit gets interpreted as black.

    For this reason I recommend to replace the line

    TIFFSetField(tiff, TIFFTAG_PHOTOMETRIC, PHOTOMETRIC_MINISBLACK);

    by

    TIFFSetField(tiff, TIFFTAG_PHOTOMETRIC, PHOTOMETRIC_MINISWHITE);

    and then invert the threshold comparison, replace line

    if (gray > threshold) bite = bite | 1;

    by

    if (gray < threshold) bite = bite | 1;

    In my method I use C-pointer arithmetic instead of an index to access the bitmap in memory.

    Finally, a couple of improvements:

    a) detect the encoding of the original UIImage (RGBA, ABGR, etc.) and get the correct RGB values for each pixel

    b) the algorithm to convert from a grayscale image to a bi-tonal image could be improved by using an adaptive-threshold algorithm instead of a pure binary conditional.

提交回复
热议问题