/* * t2p: Create a PDF file from the contents of one or more TIFF * bilevel image files. The images in the resulting PDF file * will be compressed using ITU-T T.6 (G4) fax encoding. * * PDF routines * $Id: pdf_g4.c,v 1.5 2003/02/21 01:25:47 eric Exp $ * Copyright 2001, 2002, 2003 Eric Smith * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 as * published by the Free Software Foundation. Note that permission is * not granted to redistribute this program under the terms of any * other version of the General Public License. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111 USA */ #include #include #include #include #include "bitblt.h" #include "pdf.h" #include "pdf_util.h" #include "pdf_prim.h" #include "pdf_private.h" struct pdf_g4_image { double width, height; double x, y; double r, g, b; /* fill color, only for ImageMask */ unsigned long Columns; unsigned long Rows; bool ImageMask; bool BlackIs1; Bitmap *bitmap; char XObject_name [4]; }; char pdf_new_XObject (pdf_page_handle pdf_page, struct pdf_obj *ind_ref) { char XObject_name [4] = "Im "; XObject_name [2] = ++pdf_page->last_XObject_name; if (! pdf_page->XObject_dict) { pdf_page->XObject_dict = pdf_new_obj (PT_DICTIONARY); pdf_set_dict_entry (pdf_page->resources, "XObject", pdf_page->XObject_dict); } pdf_set_dict_entry (pdf_page->XObject_dict, & XObject_name [0], ind_ref); return (pdf_page->last_XObject_name); } void pdf_write_g4_content_callback (pdf_file_handle pdf_file, struct pdf_obj *stream, void *app_data) { struct pdf_g4_image *image = app_data; /* transformation matrix is: width 0 0 height x y cm */ pdf_stream_printf (pdf_file, stream, "q %g 0 0 %g %g %g cm ", image->width, image->height, image->x, image->y); if (image->ImageMask) pdf_stream_printf (pdf_file, stream, "%g %g %g rg ", image->r, image->g, image->b); pdf_stream_printf (pdf_file, stream, "/%s Do Q\r\n", image->XObject_name); } void pdf_write_g4_fax_image_callback (pdf_file_handle pdf_file, struct pdf_obj *stream, void *app_data) { struct pdf_g4_image *image = app_data; #if 0 pdf_stream_write_data (pdf_file, stream, image->data, image->len); #else unsigned long row = 0; word_type *ref; word_type *raw; ref = NULL; raw = image->bitmap->bits; while (row < image->Rows) { pdf_stream_write_data (pdf_file, stream, (uint8_t *) raw, image->bitmap->row_words * sizeof (word_type)); row++; ref = raw; raw += image->bitmap->row_words; } /* $$$ generate and write EOFB code */ /* $$$ flush any remaining buffered bits */ #endif } void pdf_write_g4_fax_image (pdf_page_handle pdf_page, double x, double y, double width, double height, Bitmap *bitmap, bool ImageMask, double r, /* RGB fill color, only for ImageMask */ double g, double b, bool BlackIs1) /* boolean, typ. false */ { struct pdf_g4_image *image; struct pdf_obj *stream; struct pdf_obj *stream_dict; struct pdf_obj *decode_parms; struct pdf_obj *content_stream; image = pdf_calloc (sizeof (struct pdf_g4_image)); image->width = width; image->height = height; image->x = x; image->y = y; image->r = r; image->g = g; image->b = b; image->bitmap = bitmap; image->Columns = bitmap->rect.max.x - bitmap->rect.min.x; image->Rows = bitmap->rect.max.y - bitmap->rect.min.y; image->ImageMask = ImageMask; image->BlackIs1 = BlackIs1; stream_dict = pdf_new_obj (PT_DICTIONARY); stream = pdf_new_ind_ref (pdf_page->pdf_file, pdf_new_stream (pdf_page->pdf_file, stream_dict, & pdf_write_g4_fax_image_callback, image)); strcpy (& image->XObject_name [0], "Im "); image->XObject_name [2] = pdf_new_XObject (pdf_page, stream); pdf_set_dict_entry (stream_dict, "Type", pdf_new_name ("XObject")); pdf_set_dict_entry (stream_dict, "Subtype", pdf_new_name ("Image")); pdf_set_dict_entry (stream_dict, "Name", pdf_new_name (& image->XObject_name [0])); pdf_set_dict_entry (stream_dict, "Width", pdf_new_integer (image->Columns)); pdf_set_dict_entry (stream_dict, "Height", pdf_new_integer (image->Rows)); pdf_set_dict_entry (stream_dict, "BitsPerComponent", pdf_new_integer (1)); if (ImageMask) pdf_set_dict_entry (stream_dict, "ImageMask", pdf_new_bool (ImageMask)); else pdf_set_dict_entry (stream_dict, "ColorSpace", pdf_new_name ("DeviceGray")); decode_parms = pdf_new_obj (PT_DICTIONARY); pdf_set_dict_entry (decode_parms, "K", pdf_new_integer (-1)); pdf_set_dict_entry (decode_parms, "Columns", pdf_new_integer (image->Columns)); pdf_set_dict_entry (decode_parms, "Rows", pdf_new_integer (image->Rows)); if (BlackIs1) pdf_set_dict_entry (decode_parms, "BlackIs1", pdf_new_bool (BlackIs1)); pdf_stream_add_filter (stream, "CCITTFaxDecode", decode_parms); /* the following will write the stream, using our callback function to get the actual data */ pdf_write_ind_obj (pdf_page->pdf_file, stream); content_stream = pdf_new_ind_ref (pdf_page->pdf_file, pdf_new_stream (pdf_page->pdf_file, pdf_new_obj (PT_DICTIONARY), & pdf_write_g4_content_callback, image)); pdf_set_dict_entry (pdf_page->page_dict, "Contents", content_stream); pdf_write_ind_obj (pdf_page->pdf_file, content_stream); }