/* Part of XPCE --- The SWI-Prolog GUI toolkit Author: Jan Wielemaker and Anjo Anjewierden E-mail: jan@swi.psy.uva.nl WWW: http://www.swi.psy.uva.nl/projects/xpce/ Copyright (c) 2000-2013, University of Amsterdam All rights reserved. Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: 1. Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. 2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include "include.h" /* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - Actually, this module does both medium-level JPEG and GIF writing. The low-level routines are in the img directory. These routines are called by msimage.c, which in turn implements the OS specific version of gra/image.c implementing class image. - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */ #if defined(__MINGW32__) #define XMD_H #endif #ifdef HAVE_LIBJPEG #ifdef __RPCNDR_H__ #define HAVE_BOOLEAN /* prevent jmorecfg.h from redefining it */ #endif #undef GLOBAL /* conflict */ #include #include #include extern void jpeg_iostream_dest(j_compress_ptr cinfo, IOSTREAM *outfile); extern void jpeg_iostream_src(j_decompress_ptr cinfo, IOSTREAM* infile); extern void attach_dib_image(Image image, BITMAPINFO *bmi, BYTE *bits); int write_jpeg_file(IOSTREAM *fd, Image image, HBITMAP bm) { BITMAP bitmap; int width, height; int y; HDC hdc; HBITMAP obm; struct jpeg_compress_struct cinfo; struct jpeg_error_mgr jerr; JSAMPLE *row; DisplayObj d = image->display; HPALETTE ohpal=0, hpal; if ( !GetObject(bm, sizeof(BITMAP), &bitmap) ) { Cprintf("write_jpeg_file(): GetObject() failed\n"); return -1; } if ( isNil(d) ) d = CurrentDisplay(image); if ( instanceOfObject(d->colour_map, ClassColourMap) ) hpal = getPaletteColourMap(d->colour_map); else hpal = NULL; width = bitmap.bmWidth; height = bitmap.bmHeight; /*depth = bitmap.bmPlanes * bitmap.bmBitsPixel;*/ hdc = CreateCompatibleDC(NULL); if ( hpal ) { ohpal = SelectPalette(hdc, hpal, FALSE); RealizePalette(hdc); } obm = ZSelectObject(hdc, bm); row = pceMalloc(sizeof(JSAMPLE)*3*width); cinfo.err = jpeg_std_error(&jerr); jpeg_create_compress(&cinfo); jpeg_iostream_dest(&cinfo, fd); cinfo.image_width = width; cinfo.image_height = height; cinfo.input_components = 3; cinfo.in_color_space = JCS_RGB; jpeg_set_defaults(&cinfo); jpeg_start_compress(&cinfo, TRUE); for(y=0; yallocated = size; map->size = 0; map->colours[0] = alloc(sizeof(JSAMPLE)*size); map->colours[1] = alloc(sizeof(JSAMPLE)*size); map->colours[2] = alloc(sizeof(JSAMPLE)*size); map->dib_colours = alloc(sizeof(RGBQUAD)*size); return map; } void free_jpeg_cmap(JpegColourMap map) { unalloc(sizeof(JSAMPLE)*map->allocated, map->colours[0]); unalloc(sizeof(JSAMPLE)*map->allocated, map->colours[1]); unalloc(sizeof(JSAMPLE)*map->allocated, map->colours[2]); unalloc(sizeof(RGBQUAD)*map->allocated, map->dib_colours); unalloc(sizeof(*map), map); } JpegColourMap jpeg_cmap_from_colour_map(ColourMap cm, DisplayObj d) { WsCmdata data = getWsCmdata(cm); Vector colours; if ( data->jpeg_cmap ) return data->jpeg_cmap; if ( (colours = get(cm, NAME_colours, EAV)) ) { int ncolors = valInt(colours->size); int i=0; Colour e; JpegColourMap map = alloc_jpeg_cmap(ncolors); for_vector(colours, e, { if ( notNil(e) ) { int r; int g; int b; if ( isDefault(e->red) ) getXrefObject(e, d); /* Open the colour */ r = valInt(e->red)>>8; g = valInt(e->green)>>8; b = valInt(e->blue)>>8; map->colours[0][i] = r; map->colours[1][i] = g; map->colours[2][i] = b; map->dib_colours[i].rgbRed = r; map->dib_colours[i].rgbGreen = g; map->dib_colours[i].rgbBlue = b; i++; } }); map->size = i; data->jpeg_cmap = map; return map; } return NULL; } /******************************* * ERRORS * *******************************/ struct my_jpeg_error_mgr { struct jpeg_error_mgr jerr; jmp_buf jmp_context; }; static void my_exit(j_common_ptr cl) { struct jpeg_decompress_struct *cinfo = (struct jpeg_decompress_struct *)cl; struct my_jpeg_error_mgr *jerr = (struct my_jpeg_error_mgr *)cinfo->err; longjmp(jerr->jmp_context, 1); } status read_jpeg_file(IOSTREAM *fd, Image image) { struct jpeg_decompress_struct cinfo; struct my_jpeg_error_mgr jerr; long here = Stell(fd); long row_stride; int width, height, bwidth, image_size; JSAMPLE **buff; BYTE *data; BITMAPINFO *dib = NULL; BITMAPINFOHEADER *header = NULL; /* silence compiler */ DisplayObj d = image->display; int outline; JpegColourMap cmap = NULL; cinfo.err = jpeg_std_error((struct jpeg_error_mgr *)&jerr); if ( setjmp(jerr.jmp_context) ) { switch(jerr.jerr.msg_code) { case JERR_OUT_OF_MEMORY: return sysPce("Not enough memory"); case JERR_NO_SOI: break; /* invalid */ default: DEBUG(NAME_image, { char buf[1024]; (*jerr.jerr.format_message)((j_common_ptr)&cinfo, buf); Cprintf("JPEG: %s\n", buf); }); break; /* also invalid */ } jpeg_destroy_decompress(&cinfo); Sseek(fd, here, SEEK_SET); fail; } jerr.jerr.error_exit = my_exit; jpeg_create_decompress(&cinfo); jpeg_iostream_src(&cinfo, fd); jpeg_save_markers(&cinfo, JPEG_COM, 0xffff); jpeg_read_header(&cinfo, TRUE); /* colourmap handling */ if ( cinfo.output_components == 3 ) { if ( isNil(d) ) d = CurrentDisplay(image); openDisplay(d); if ( ws_depth_display(d) < 16 && instanceOfObject(d->colour_map, ClassColourMap) && (cmap = jpeg_cmap_from_colour_map(d->colour_map, d)) ) { dib = pceMalloc(sizeof(dib->bmiHeader)+cmap->size*sizeof(RGBQUAD)); header = &dib->bmiHeader; memset(header, 0, sizeof(*header)); memcpy(&dib->bmiColors[0], cmap->dib_colours, cmap->size*sizeof(RGBQUAD)); header->biBitCount = 8; header->biClrUsed = cmap->size; cinfo.colormap = cmap->colours; cinfo.actual_number_of_colors = cmap->size; cinfo.quantize_colors = TRUE; } } jpeg_start_decompress(&cinfo); row_stride = cinfo.output_width * cinfo.output_components; buff = (*cinfo.mem->alloc_sarray)((j_common_ptr)&cinfo, JPOOL_IMAGE, row_stride, 1); width = cinfo.image_width; height = cinfo.image_height; if ( cmap ) bwidth = width; else bwidth = ((width*3+3)&0xfffc); /* why is this? */ image_size = bwidth*height; data = pceMalloc(image_size); for(outline = height-1; cinfo.output_scanline < cinfo.output_height; outline--) { int i; BYTE *src, *dest; dest = data + bwidth*outline; jpeg_read_scanlines(&cinfo, buff, 1); i = width; src = buff[0]; if ( cmap ) /* colour-mapped */ { while(i--) { *dest++ = *src++; } } else { switch( cinfo.output_components ) { case 1: /* grayscale JPEG */ while(i--) { *dest++ = src[0]; *dest++ = src[0]; *dest++ = src[0]; src++; } break; case 3: /* RGB JPEG */ while(i--) { *dest++ = src[2]; *dest++ = src[1]; *dest++ = src[0]; src += 3; } break; default: /* We don't have this */ Sseek(fd, here, SEEK_SET); Cprintf("JPeg with %d output_components??\n"); fail; } memset(dest, 0, bwidth - width*3); } } if ( cinfo.marker_list ) { jpeg_saved_marker_ptr m; Chain ch; attributeObject(image, NAME_comment, (ch=newObject(ClassChain, EAV))); for(m = cinfo.marker_list; m; m = m->next ) { if ( m->marker == JPEG_COM ) { string s; if ( str_set_n_ascii(&s, m->data_length, (char*)m->data) ) appendChain(ch, StringToString(&s)); } } } jpeg_finish_decompress(&cinfo); jpeg_destroy_decompress(&cinfo); if ( !dib ) { dib = pceMalloc(sizeof(*dib)); header = &dib->bmiHeader; memset(dib, 0, sizeof(*dib)); header->biBitCount = 24; } header->biSize = sizeof(BITMAPINFOHEADER); header->biWidth = width; header->biHeight = height; header->biPlanes = 1; header->biCompression = BI_RGB; header->biSizeImage = image_size; attach_dib_image(image, dib, data); succeed; } #endif /*HAVE_LIBJPEG*/ /******************************* * GIF (SHOULD MOVE) * *******************************/ #ifdef O_GIFWRITE #include typedef unsigned char GSAMPLE; static GSAMPLE *mask_bits(HBITMAP mask); /* forwards */ #define ROUND(p, n) ((((p) + (n) - 1) & ~((n) - 1))) /* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - Old versions used GetPixel(). Now it uses GetDIBits(), which probably gives a big performance boost. Unfortutanely though we need packed RGB and GetDIBits() returns a word-aligned array of RGB triples Billy nicely orders as BGR. Hence the shifting and swapping ... Note that the height field of the structure is set to the negative height, the bits are extracted top-to-bottom rather than MS native bottom-to-top. - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */ int write_gif_file(IOSTREAM *fd, Image image, HBITMAP bm, HBITMAP mask) { int width = valInt(image->size->w); int height = valInt(image->size->h); int rval, sl; HDC hdc; GSAMPLE *data, *maskdata = NULL; DisplayObj d = image->display; HPALETTE ohpal=0, hpal; BITMAPINFO info; if ( isNil(d) ) d = CurrentDisplay(image); if ( instanceOfObject(d->colour_map, ClassColourMap) ) hpal = getPaletteColourMap(d->colour_map); else hpal = NULL; hdc = CreateCompatibleDC(NULL); if ( hpal ) { ohpal = SelectPalette(hdc, hpal, FALSE); RealizePalette(hdc); } memset(&info, 0, sizeof(info)); info.bmiHeader.biSize = sizeof(info.bmiHeader); info.bmiHeader.biWidth = width; info.bmiHeader.biHeight = -height; /* work top-down */ info.bmiHeader.biPlanes = 1; info.bmiHeader.biBitCount = 24; info.bmiHeader.biCompression = BI_RGB; /* from WINGDI, this is 0 */ if ( !(sl = GetDIBits(hdc, bm, 0, height, NULL, &info, DIB_RGB_COLORS)) ) { Cprintf("%s: GetDIBits() returned %d", pp(image), sl); return FALSE; } else { DEBUG(NAME_image, { Cprintf("%s: GetDIBits() returned %d; ", pp(image), sl); Cprintf("Image = %dx%dx%d (%ld bytes)\n", info.bmiHeader.biWidth, info.bmiHeader.biHeight, info.bmiHeader.biBitCount, info.bmiHeader.biSizeImage); }); data = pceMalloc(info.bmiHeader.biSizeImage); height = abs(info.bmiHeader.biHeight); width = info.bmiHeader.biWidth; if ( !GetDIBits(hdc, bm, 0, height, data, &info, DIB_RGB_COLORS) ) { Cprintf("%s: GetDIBits() failed to get bits\n", pp(image)); return FALSE; } /* adjust alignment and colours */ { int outlensl = width*3; int inlensl = ROUND(outlensl, sizeof(DWORD)); int y; for(y=0; ybmiHeader.biSize = sizeof(info->bmiHeader); info->bmiHeader.biWidth = bitmap.bmWidth; info->bmiHeader.biHeight = -bitmap.bmHeight; info->bmiHeader.biPlanes = 1; info->bmiHeader.biBitCount = 1; info->bmiHeader.biCompression = BI_RGB; /* get info */ if ( !GetDIBits(hdc, mask, 0, bitmap.bmHeight, NULL, info, DIB_RGB_COLORS) ) { Cprintf("%d: mask_bits(): GetDIBits() failed\n", __LINE__); goto out; } /*Cprintf("Mask: %d bytes\n", info->bmiHeader.biSizeImage);*/ /* get the bits */ data = pceMalloc(info->bmiHeader.biSizeImage); height = abs(info->bmiHeader.biHeight); width = info->bmiHeader.biWidth; if ( !GetDIBits(hdc, mask, 0, height, data, info, DIB_RGB_COLORS) ) { Cprintf("Mask: GetDIBits() failed to get mask bits\n"); return FALSE; } inlensl = ROUND(width, sizeof(DWORD)*8)/8; outlensl = (width+7)/8; for(y=0; ybmiHeader.biWidth; x++) { Cprintf("%c", p[0]&m ? '.' : '*'); m>>=1; if ( !m ) { m = 0x80; p++; } } Cprintf("\n"); } #endif } out: DeleteDC(hdc); return data; } #endif /*HAVE_LIBJPEG*/