Mercurial > dillo_port1.3
view src/imgbuf.cc @ 1918:55af06da5523
merge in recent changes from main repo
author | corvid <corvid@lavabit.com> |
---|---|
date | Fri, 25 Feb 2011 08:37:13 +0000 |
parents | e59742ef792a |
children |
line wrap: on
line source
/* * File: imgbuf.cc * * Copyright (C) 2008 Jorge Arellano Cid <jcid@dillo.org>, * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 3 of the License, or * (at your option) any later version. */ #include "msg.h" #include "imgbuf.hh" #include "dw/core.hh" #include "dw/image.hh" using namespace dw::core; /* * Local data */ static size_t linebuf_size = 0; static uchar_t *linebuf = NULL; /* * Decode 'buf' (an image line) into RGB format. */ static uchar_t *Imgbuf_rgb_line(const uchar_t *buf, DilloImgType type, uchar_t *cmap, uint_t width, uint_t y) { uint_t x; switch (type) { case DILLO_IMG_TYPE_INDEXED: if (cmap) { for (x = 0; x < width; x++) memcpy(linebuf + x * 3, cmap + buf[x] * 3, 3); } else { MSG_WARN("Gif:: image lacks a color map\n"); } break; case DILLO_IMG_TYPE_GRAY: for (x = 0; x < width; x++) memset(linebuf + x * 3, buf[x], 3); break; case DILLO_IMG_TYPE_CMYK_INV: /* * We treat CMYK as if it were "RGBW", and it works. Everyone who is * trying to handle CMYK jpegs is confused by this, and supposedly * the issue is that Adobe CMYK is "wrong" but ubiquitous. */ for (x = 0; x < width; x++) { uint_t white = buf[x * 4 + 3]; linebuf[x * 3] = buf[x * 4] * white / 0x100; linebuf[x * 3 + 1] = buf[x * 4 + 1] * white / 0x100; linebuf[x * 3 + 2] = buf[x * 4 + 2] * white / 0x100; } break; case DILLO_IMG_TYPE_RGB: /* avoid a memcpy here! --Jcid */ return (uchar_t *)buf; case DILLO_IMG_TYPE_NOTSET: MSG_ERR("Imgbuf_rgb_line: type not set...\n"); break; } return linebuf; } // Wrappers for Imgbuf ------------------------------------------------------- /* * Increment reference count for an Imgbuf */ void a_Imgbuf_ref(void *v_imgbuf) { ((Imgbuf*)v_imgbuf)->ref(); } /* * Decrement reference count for an Imgbuf */ void a_Imgbuf_unref(void *v_imgbuf) { if (v_imgbuf) ((Imgbuf*)v_imgbuf)->unref(); } /* * Create a new Imgbuf */ void *a_Imgbuf_new(void *v_dw, int img_type, uint_t width, uint_t height) { Layout *layout = ((Widget*)v_dw)->getLayout(); if (!layout) { MSG_ERR("a_Imgbuf_new: layout is NULL.\n"); exit(1); } // Assert linebuf is wide enough. if (3 * width > linebuf_size) { linebuf_size = 3 * width; linebuf = (uchar_t*) dRealloc(linebuf, linebuf_size); } return (void*)layout->createImgbuf(Imgbuf::RGB, width, height); } /* * Last reference for this Imgbuf? */ int a_Imgbuf_last_reference(void *v_imgbuf) { return ((Imgbuf*)v_imgbuf)->lastReference () ? 1 : 0; } /* * Update the root buffer of an imgbuf. */ void a_Imgbuf_update(void *v_imgbuf, const uchar_t *buf, DilloImgType type, uchar_t *cmap, uint_t width, uint_t height, uint_t y) { dReturn_if_fail ( y < height ); /* Decode 'buf' and copy it into the imgbuf */ uchar_t *newbuf = Imgbuf_rgb_line(buf, type, cmap, width, y); ((Imgbuf*)v_imgbuf)->copyRow(y, (byte *)newbuf); } /* * Reset for a new scan from a multiple-scan image. */ void a_Imgbuf_new_scan(void *v_imgbuf) { ((Imgbuf*)v_imgbuf)->newScan(); }