home
***
CD-ROM
|
disk
|
FTP
|
other
***
search
/
Fractal Frenzy 1
/
WalnutCreekFractalFrenzy-1.iso
/
pc
/
viewers
/
x11
/
xloadimg.tz
/
xloadimg
/
reduce.c
< prev
next >
Wrap
C/C++ Source or Header
|
1991-05-20
|
16KB
|
626 lines
/* reduce.c:
*
* reduce an image's colormap usage to a set number of colors. this also
* translates a true color image to a TLA-style image of `n' colors.
*
* this uses an algorithm by Paul Heckbert discussed in `Color Image
* Quantization for Frame Buffer Display,' _Computer Graphics_ 16(3),
* pp 297-307. this implementation is based on one discussed in
* 'A Few Good Colors,' _Computer Language_, Aug. 1990, pp 32-41 by
* Dave Pomerantz.
*
* this function cannot reduce to any number of colors larger than 32768.
*
* jim frost 04.18.91
*
* Copyright 1991 Jim Frost.
* See included file "copyright.h" for complete copyright information.
*/
#include "copyright.h"
#include "image.h"
#define DIST(A, B) ((A) < (B) ? (B) - (A) : (A) - (B))
/* find the distance between two colors. we loose some accuracy here because
* a triple squared short may not fit in a long. we use a table lookup
* to help speed this up; it's an O(exp(n,2)) algorithm.
*/
unsigned int squareInit= 0;
unsigned long squareTable[32768];
void initSquareTable()
{ unsigned long a;
for (a= 0; a < 32768; a++)
squareTable[a]= a * a;
squareInit= 1;
}
unsigned long colorDistance(rgb, a, b)
RGBMap *rgb;
Pixel a, b;
{
return(squareTable[DIST(*(rgb->red + a), *(rgb->red + b)) >> 1] +
squareTable[DIST(*(rgb->green + a), *(rgb->green + b)) >> 1] +
squareTable[DIST(*(rgb->blue + a), *(rgb->blue + b)) >> 1]);
}
/* this converts a TLA-style pixel into a 15-bit true color pixel
*/
#define TLA_TO_15BIT(TABLE,PIXEL) \
((((TABLE).red[PIXEL] & 0xf800) >> 1) | \
(((TABLE).green[PIXEL] & 0xf800) >> 6) | \
(((TABLE).blue[PIXEL] & 0xf800) >> 11))
/* this converts a 24-bit true color pixel into a 15-bit true color pixel
*/
#define TRUE_TO_15BIT(PIXEL) \
((((PIXEL) & 0xf80000) >> 9) | \
(((PIXEL) & 0x00f800) >> 6) | \
(((PIXEL) & 0x0000f8) >> 3))
/* these macros extract color intensities from a 15-bit true color pixel
*/
#define RED_INTENSITY(P) (((P) & 0x7c00) >> 10)
#define GREEN_INTENSITY(P) (((P) & 0x03e0) >> 5)
#define BLUE_INTENSITY(P) ((P) & 0x001f)
/* this structure defines a color area which is made up of an array of pixel
* values and a count of the total number of image pixels represented by
* the area. color areas are kept in a list sorted by the number of image
* pixels they represent.
*/
struct color_area {
unsigned short *pixels; /* array of pixel values in this area */
unsigned short num_pixels; /* size of above array */
int (*sort_func)(); /* predicate func to sort with before
* splitting */
unsigned long pixel_count; /* # of image pixels we represent */
struct color_area *prev, *next;
};
/* predicate functions for qsort
*/
static sortRGB(p1, p2)
unsigned short *p1, *p2;
{ unsigned int red1, green1, blue1, red2, green2, blue2;
red1= RED_INTENSITY(*p1);
green1= GREEN_INTENSITY(*p1);
blue1= BLUE_INTENSITY(*p1);
red2= RED_INTENSITY(*p2);
green2= GREEN_INTENSITY(*p2);
blue2= BLUE_INTENSITY(*p2);
if (red1 == red2)
if (green1 == green2)
if (blue1 < blue2)
return(-1);
else
return(1);
else if (green1 < green2)
return(-1);
else
return(1);
else if (red1 < red2)
return(-1);
else
return(1);
}
static sortRBG(p1, p2)
unsigned short *p1, *p2;
{ unsigned int red1, green1, blue1, red2, green2, blue2;
red1= RED_INTENSITY(*p1);
green1= GREEN_INTENSITY(*p1);
blue1= BLUE_INTENSITY(*p1);
red2= RED_INTENSITY(*p2);
green2= GREEN_INTENSITY(*p2);
blue2= BLUE_INTENSITY(*p2);
if (red1 == red2)
if (blue1 == blue2)
if (green1 < green2)
return(-1);
else
return(1);
else if (blue1 < blue2)
return(-1);
else
return(1);
else if (red1 < red2)
return(-1);
else
return(1);
}
static sortGRB(p1, p2)
unsigned short *p1, *p2;
{ unsigned int red1, green1, blue1, red2, green2, blue2;
red1= RED_INTENSITY(*p1);
green1= GREEN_INTENSITY(*p1);
blue1= BLUE_INTENSITY(*p1);
red2= RED_INTENSITY(*p2);
green2= GREEN_INTENSITY(*p2);
blue2= BLUE_INTENSITY(*p2);
if (green1 == green2)
if (red1 == red2)
if (blue1 < blue2)
return(-1);
else
return(1);
else if (red1 < red2)
return(-1);
else
return(1);
else if (green1 < green2)
return(-1);
else
return(1);
}
static sortGBR(p1, p2)
unsigned short *p1, *p2;
{ unsigned int red1, green1, blue1, red2, green2, blue2;
red1= RED_INTENSITY(*p1);
green1= GREEN_INTENSITY(*p1);
blue1= BLUE_INTENSITY(*p1);
red2= RED_INTENSITY(*p2);
green2= GREEN_INTENSITY(*p2);
blue2= BLUE_INTENSITY(*p2);
if (green1 == green2)
if (blue1 == blue2)
if (red1 < red2)
return(-1);
else
return(1);
else if (blue1 < blue2)
return(-1);
else
return(1);
else if (green1 < green2)
return(-1);
else
return(1);
}
static sortBRG(p1, p2)
unsigned short *p1, *p2;
{ unsigned int red1, green1, blue1, red2, green2, blue2;
red1= RED_INTENSITY(*p1);
green1= GREEN_INTENSITY(*p1);
blue1= BLUE_INTENSITY(*p1);
red2= RED_INTENSITY(*p2);
green2= GREEN_INTENSITY(*p2);
blue2= BLUE_INTENSITY(*p2);
if (blue1 == blue2)
if (red1 == red2)
if (green1 < green2)
return(-1);
else
return(1);
else if (red1 < red2)
return(-1);
else
return(1);
else if (blue1 < blue2)
return(-1);
else
return(1);
}
static sortBGR(p1, p2)
unsigned short *p1, *p2;
{ unsigned int red1, green1, blue1, red2, green2, blue2;
red1= RED_INTENSITY(*p1);
green1= GREEN_INTENSITY(*p1);
blue1= BLUE_INTENSITY(*p1);
red2= RED_INTENSITY(*p2);
green2= GREEN_INTENSITY(*p2);
blue2= BLUE_INTENSITY(*p2);
if (blue1 == blue2)
if (green1 == green2)
if (red1 < red2)
return(-1);
else
return(1);
else if (green1 < green2)
return(-1);
else
return(1);
else if (blue1 < blue2)
return(-1);
else
return(1);
}
/* this does calculations on a color area following a split and inserts
* the color area in the list of color areas.
*/
static insertColorArea(pixel_counts, rlargest, rsmallest, area)
unsigned long *pixel_counts;
struct color_area **rlargest, **rsmallest, *area;
{ int a;
unsigned int red, green, blue;
unsigned int min_red, min_green, min_blue;
unsigned int max_red, max_green, max_blue= 0;
struct color_area *largest, *smallest, *tmp_area;
min_red= min_green= min_blue= 31;
max_red= max_green= max_blue= 0;
/* update pixel count for this area and find RGB intensity widths
*/
area->pixel_count= 0;
for (a= 0; a < area->num_pixels; a++) {
area->pixel_count += pixel_counts[area->pixels[a]];
red= RED_INTENSITY(area->pixels[a]);
green= GREEN_INTENSITY(area->pixels[a]);
blue= BLUE_INTENSITY(area->pixels[a]);
if (red < min_red)
min_red= red;
if (red > max_red)
max_red= red;
if (green < min_green)
min_green= green;
if (green > max_green)
max_green= green;
if (blue < min_blue)
min_blue= blue;
if (blue > max_blue)
max_blue= blue;
}
/* calculate widths and determine which predicate function to use based
* on the result
*/
red= max_red - min_red;
green= max_green - min_green;
blue= max_blue - min_blue;
if (red > green)
if (green > blue)
area->sort_func= sortRGB;
else if (red > blue)
area->sort_func= sortRBG;
else
area->sort_func= sortBRG;
else if (green > blue)
if (red > blue)
area->sort_func= sortGRB;
else
area->sort_func= sortGBR;
else
area->sort_func= sortBGR;
/* insert color area in color area list sorted by number of pixels that
* the area represents
*/
largest= *rlargest;
smallest= *rsmallest;
if (!largest) {
largest= smallest= area;
area->prev= area->next= (struct color_area *)NULL;
}
/* if we only have one element, our pixel count is immaterial so we get
* stuck on the end of the list.
*/
else if (area->num_pixels < 2) {
smallest->next= area;
area->prev= smallest;
area->next= (struct color_area *)NULL;
smallest= area;
}
/* insert node into list
*/
else {
for (tmp_area= largest; tmp_area; tmp_area= tmp_area->next)
if ((area->pixel_count > tmp_area->pixel_count) ||
(tmp_area->num_pixels < 2)) {
area->prev= tmp_area->prev;
area->next= tmp_area;
tmp_area->prev= area;
if (area->prev)
area->prev->next= area;
else
largest= area;
break;
}
if (!tmp_area) {
area->prev= smallest;
area->next= (struct color_area *)NULL;
smallest->next= area;
smallest= area;
}
}
*rlargest= largest;
*rsmallest= smallest;
}
Image *reduce(image, n, verbose)
Image *image;
unsigned int n, verbose;
{ unsigned long pixel_counts[32768]; /* pixel occurrance histogram */
unsigned short pixel_array[32768];
unsigned long count, midpoint;
int x, y, num_pixels, allocated, depth, ncolors;
byte *pixel, *dpixel;
struct color_area *areas, *largest_area, *smallest_area;
struct color_area *new_area, *old_area;
Image *new_image;
char buf[BUFSIZ];
goodImage(image, "reduce");
if (n > 32768) /* max # of colors we can handle */
n= 32768;
/* create a histogram of particular pixel occurrances
*/
bzero(pixel_counts, 32768 * sizeof(unsigned long));
switch (image->type) {
case IBITMAP:
return(image);
case IRGB:
if (image->rgb.used <= n)
return;
if (verbose) {
printf(" Reducing RGB image color usage to %d colors...", n);
fflush(stdout);
}
pixel= image->data;
for (y= 0; y < image->height; y++)
for (x= 0; x < image->width; x++) {
pixel_counts[TLA_TO_15BIT(image->rgb,
memToVal(pixel, image->pixlen))]++;
pixel += image->pixlen;
}
break;
case ITRUE:
if (image->pixlen != 3) {
fprintf(stderr, "reduce: true color image has strange pixel length?\n");
return(image);
}
if (verbose) {
printf(" Converting true color image to RGB image with %d colors...",
n);
fflush(stdout);
}
pixel= image->data;
for (y= 0; y < image->height; y++)
for (x= 0; x < image->width; x++) {
pixel_counts[TRUE_TO_15BIT(memToVal(pixel, 3))]++;
pixel += 3;
}
break;
default:
return(image); /* not something we can reduce, thank you anyway */
}
/* create array of 15-bit pixel values that actually occur in the image
*/
num_pixels= 0;
for (x= 0; x < 32768; x++)
if (pixel_counts[x] > 0)
pixel_array[num_pixels++]= (short)x;
if (verbose) {
printf("image uses %d colors...", num_pixels);
fflush(stdout);
}
/* create color area array and initialize first element
*/
areas= (struct color_area *)lmalloc(n * sizeof(struct color_area));
areas[0].pixels= pixel_array;
areas[0].num_pixels= num_pixels;
largest_area= smallest_area= (struct color_area *)NULL;
insertColorArea(pixel_counts, &largest_area, &smallest_area, areas);
allocated= 1;
/* keep splitting the color area until we have as many color areas as we
* need
*/
while (allocated < n) {
/* if our largest area can't be broken down, we can't even get the
* number of colors they asked us to
*/
if (largest_area->num_pixels < 2)
break;
/* find midpoint of largest area and do split
*/
qsort(largest_area->pixels, largest_area->num_pixels, sizeof(short),
largest_area->sort_func);
count= 0;
midpoint= largest_area->pixel_count / 2;
for (x= 0; x < largest_area->num_pixels; x++) {
count += pixel_counts[largest_area->pixels[x]];
if (count > midpoint)
break;
}
if (x == 0) /* degenerate case; divide in half */
x= 1;
new_area= areas + allocated;
new_area->pixels= largest_area->pixels + x;
new_area->num_pixels= largest_area->num_pixels - x;
largest_area->num_pixels= x;
old_area= largest_area;
largest_area= largest_area->next;
if (largest_area)
largest_area->prev= (struct color_area *)NULL;
else
smallest_area= (struct color_area *)NULL;
/* recalculate for each area of split and insert in the area list
*/
insertColorArea(pixel_counts, &largest_area, &smallest_area, old_area);
insertColorArea(pixel_counts, &largest_area, &smallest_area, new_area);
allocated++;
}
/* get destination image
*/
depth= colorsToDepth(n);
new_image= newRGBImage(image->width, image->height, depth);
sprintf(buf, "%s (%d colors)", image->title, n);
new_image->title= dupString(buf);
/* calculate RGB table from each color area. this should really calculate
* a new color by weighting the intensities by the number of pixels, but
* it's a pain to scale so this just averages all the intensities. it
* works pretty well regardless.
*/
for (x= 0; x < allocated; x++) {
long red, green, blue, count, pixel;
red= green= blue= 0;
count= areas[x].pixel_count;
for (y= 0; y < areas[x].num_pixels; y++) {
pixel= areas[x].pixels[y];
red += RED_INTENSITY(pixel);
green += GREEN_INTENSITY(pixel);
blue += BLUE_INTENSITY(pixel);
pixel_counts[pixel]= x;
}
red /= areas[x].num_pixels;
green /= areas[x].num_pixels;
blue /= areas[x].num_pixels;
new_image->rgb.red[x]= (unsigned short)(red << 11);
new_image->rgb.green[x]= (unsigned short)(green << 11);
new_image->rgb.blue[x]= (unsigned short)(blue << 11);
};
new_image->rgb.used= allocated;
new_image->rgb.compressed= 1;
lfree(areas);
/* copy old image into new image
*/
pixel= image->data;
dpixel= new_image->data;
switch(image->type) {
case IRGB:
for (y= 0; y < image->height; y++)
for (x= 0; x < image->width; x++) {
valToMem(pixel_counts[TLA_TO_15BIT(image->rgb,
memToVal(pixel, image->pixlen))],
dpixel, new_image->pixlen);
pixel += image->pixlen;
dpixel += new_image->pixlen;
}
break;
case ITRUE:
for (y= 0; y < image->height; y++)
for (x= 0; x < image->width; x++) {
valToMem(pixel_counts[TRUE_TO_15BIT(memToVal(pixel, 3))],
dpixel, new_image->pixlen);
pixel += 3;
dpixel += new_image->pixlen;
}
break;
}
if (verbose)
printf("done\n");
return(new_image);
}
/* expand an image into a true color image
*/
Image *expand(image)
Image *image;
{
Image *new_image;
int x, y;
Pixel spixval;
byte *spixel, *dpixel, *line;
unsigned int linelen;
byte mask;
goodImage(image, "expand");
if TRUEP(image)
return(image);
new_image= newTrueImage(image->width, image->height);
new_image->title= dupString(image->title);
switch (image->type) {
case IBITMAP:
line= image->data;
dpixel= new_image->data;
linelen= (image->width / 8) + (image->width % 8 ? 1 : 0);
for (y= 0; y < image->height; y++) {
spixel= line;
mask= 0x80;
for (x= 0; x < image->width; x++) {
valToMem((mask & *spixel ? 0L : 0xffffff), dpixel, 3);
mask >>= 1;
if (!mask) {
mask= 0x80;
spixel++;
}
dpixel += new_image->pixlen;
}
line += linelen;
}
break;
case IRGB:
spixel= image->data;
dpixel= new_image->data;
for (y= 0; y < image->height; y++)
for (x= 0; x < image->width; x++) {
spixval= memToVal(spixel, image->pixlen);
valToMem(RGB_TO_TRUE(image->rgb.red[spixval],
image->rgb.green[spixval],
image->rgb.blue[spixval]),
dpixel, new_image->pixlen);
spixel += image->pixlen;
dpixel += new_image->pixlen;
}
break;
}
return(new_image);
}