blob: f63b2450029f70836e680418b7223410b13361b9 [file] [log] [blame] [edit]
//========================================================================
//
// TiffWriter.cc
//
// This file is licensed under the GPLv2 or later
//
// Copyright (C) 2010 William Bader <williambader@hotmail.com>
//
//========================================================================
#include "TiffWriter.h"
#if ENABLE_LIBTIFF
#include <string.h>
TiffWriter::~TiffWriter()
{
// no cleanup needed
}
TiffWriter::TiffWriter()
{
f = NULL;
numRows = 0;
curRow = 0;
compressionString = NULL;
splashMode = splashModeRGB8;
}
// Set the compression type
void TiffWriter::setCompressionString(const char *compressionStringArg)
{
compressionString = compressionStringArg;
}
// Set the bitmap mode
void TiffWriter::setSplashMode(SplashColorMode splashModeArg)
{
splashMode = splashModeArg;
}
// Write a TIFF file.
bool TiffWriter::init(FILE *openedFile, int width, int height, int hDPI, int vDPI)
{
unsigned int compression;
uint16 photometric;
uint32 rowsperstrip = (uint32) -1;
int bitspersample;
uint16 samplesperpixel;
const struct compression_name_tag {
const char *compressionName; // name of the compression option from the command line
unsigned int compressionCode; // internal libtiff code
const char *compressionDescription; // descriptive name
} compressionList[] = {
{ "none", COMPRESSION_NONE, "no compression" },
{ "ccittrle", COMPRESSION_CCITTRLE, "CCITT modified Huffman RLE" },
{ "ccittfax3", COMPRESSION_CCITTFAX3,"CCITT Group 3 fax encoding" },
{ "ccittt4", COMPRESSION_CCITT_T4, "CCITT T.4 (TIFF 6 name)" },
{ "ccittfax4", COMPRESSION_CCITTFAX4, "CCITT Group 4 fax encoding" },
{ "ccittt6", COMPRESSION_CCITT_T6, "CCITT T.6 (TIFF 6 name)" },
{ "lzw", COMPRESSION_LZW, "Lempel-Ziv & Welch" },
{ "ojpeg", COMPRESSION_OJPEG, "!6.0 JPEG" },
{ "jpeg", COMPRESSION_JPEG, "%JPEG DCT compression" },
{ "next", COMPRESSION_NEXT, "NeXT 2-bit RLE" },
{ "packbits", COMPRESSION_PACKBITS, "Macintosh RLE" },
{ "ccittrlew", COMPRESSION_CCITTRLEW, "CCITT modified Huffman RLE w/ word alignment" },
{ "deflate", COMPRESSION_DEFLATE, "Deflate compression" },
{ "adeflate", COMPRESSION_ADOBE_DEFLATE, "Deflate compression, as recognized by Adobe" },
{ "dcs", COMPRESSION_DCS, "Kodak DCS encoding" },
{ "jbig", COMPRESSION_JBIG, "ISO JBIG" },
{ "jp2000", COMPRESSION_JP2000, "Leadtools JPEG2000" },
{ NULL, 0, NULL }
};
// Initialize
f = NULL;
curRow = 0;
// Store the number of rows
numRows = height;
// Set the compression
compression = COMPRESSION_NONE;
if (compressionString == NULL || strcmp(compressionString, "") == 0) {
compression = COMPRESSION_NONE;
} else {
int i;
for (i = 0; compressionList[i].compressionName != NULL; i++) {
if (strcmp(compressionString, compressionList[i].compressionName) == 0) {
compression = compressionList[i].compressionCode;
break;
}
}
if (compressionList[i].compressionName == NULL) {
fprintf(stderr, "TiffWriter: Unknown compression type '%.10s', using 'none'.\n", compressionString);
fprintf(stderr, "Known compression types (the tiff library might not support every type)\n");
for (i = 0; compressionList[i].compressionName != NULL; i++) {
fprintf(stderr, "%10s %s\n", compressionList[i].compressionName, compressionList[i].compressionDescription);
}
}
}
// Set bits per sample, samples per pixel, and photometric type from the splash mode
bitspersample = (splashMode == splashModeMono1? 1: 8);
switch (splashMode) {
case splashModeMono1:
case splashModeMono8:
samplesperpixel = 1;
photometric = PHOTOMETRIC_MINISBLACK;
break;
case splashModeRGB8:
case splashModeBGR8:
samplesperpixel = 3;
photometric = PHOTOMETRIC_RGB;
break;
default:
fprintf(stderr, "TiffWriter: Mode %d not supported\n", splashMode);
return false;
}
// Open the file
if (openedFile == NULL) {
fprintf(stderr, "TiffWriter: No output file given.\n");
return false;
}
f = TIFFFdOpen(fileno(openedFile), "-", "w");
if (!f) {
return false;
}
// Set TIFF tags
TIFFSetField(f, TIFFTAG_IMAGEWIDTH, width);
TIFFSetField(f, TIFFTAG_IMAGELENGTH, height);
TIFFSetField(f, TIFFTAG_ORIENTATION, ORIENTATION_TOPLEFT);
TIFFSetField(f, TIFFTAG_SAMPLESPERPIXEL, samplesperpixel);
TIFFSetField(f, TIFFTAG_BITSPERSAMPLE, bitspersample);
TIFFSetField(f, TIFFTAG_PLANARCONFIG, PLANARCONFIG_CONTIG);
TIFFSetField(f, TIFFTAG_PHOTOMETRIC, photometric);
TIFFSetField(f, TIFFTAG_COMPRESSION, (uint16) compression);
TIFFSetField(f, TIFFTAG_ROWSPERSTRIP, TIFFDefaultStripSize(f, rowsperstrip));
TIFFSetField(f, TIFFTAG_XRESOLUTION, (double) hDPI);
TIFFSetField(f, TIFFTAG_YRESOLUTION, (double) vDPI);
TIFFSetField(f, TIFFTAG_RESOLUTIONUNIT, RESUNIT_INCH);
return true;
}
bool TiffWriter::writePointers(unsigned char **rowPointers, int rowCount)
{
// Write all rows to the file
for (int row = 0; row < rowCount; row++) {
if (TIFFWriteScanline(f, rowPointers[row], row, 0) < 0) {
fprintf(stderr, "TiffWriter: Error writing tiff row %d\n", row);
return false;
}
}
return true;
}
bool TiffWriter::writeRow(unsigned char **rowData)
{
// Add a single row
if (TIFFWriteScanline(f, *rowData, curRow, 0) < 0) {
fprintf(stderr, "TiffWriter: Error writing tiff row %d\n", curRow);
return false;
}
curRow++;
return true;
}
bool TiffWriter::close()
{
// Close the file
TIFFClose(f);
return true;
}
#endif