2003-03-26 05:38:13 +00:00
|
|
|
#include <glib.h>
|
|
|
|
#include "../kernel/geom.h"
|
|
|
|
#include "image.h"
|
|
|
|
|
2003-04-27 17:08:40 +00:00
|
|
|
void image_draw(pixel32 *target, TextureRGBA *rgba, Rect *position,
|
|
|
|
Rect *surarea)
|
2003-03-26 05:38:13 +00:00
|
|
|
{
|
2003-05-18 23:06:11 +00:00
|
|
|
pixel32 *draw = rgba->data;
|
2003-05-09 16:57:17 +00:00
|
|
|
guint c, i, e, t, sfw, sfh;
|
|
|
|
sfw = position->width;
|
|
|
|
sfh = position->height;
|
2003-03-26 05:38:13 +00:00
|
|
|
|
2003-05-09 16:57:17 +00:00
|
|
|
/* it would be nice if this worked, but this function is well broken in
|
|
|
|
these circumstances. */
|
|
|
|
g_assert(position->width == surarea->width &&
|
|
|
|
position->height == surarea->height);
|
2003-04-27 17:08:40 +00:00
|
|
|
|
2003-05-09 16:57:17 +00:00
|
|
|
g_assert(rgba->data != NULL);
|
2003-03-26 05:38:13 +00:00
|
|
|
|
2003-05-09 16:57:17 +00:00
|
|
|
if ((rgba->width != sfw || rgba->height != sfh) &&
|
|
|
|
(rgba->width != rgba->cwidth || rgba->height != rgba->cheight)) {
|
|
|
|
double dx = rgba->width / (double)sfw;
|
|
|
|
double dy = rgba->height / (double)sfh;
|
|
|
|
double px = 0.0;
|
|
|
|
double py = 0.0;
|
|
|
|
int iy = 0;
|
2003-03-26 05:38:13 +00:00
|
|
|
|
2003-05-09 16:57:17 +00:00
|
|
|
/* scale it and cache it */
|
|
|
|
if (rgba->cache != NULL)
|
|
|
|
g_free(rgba->cache);
|
|
|
|
rgba->cache = g_new(unsigned long, sfw * sfh);
|
|
|
|
rgba->cwidth = sfw;
|
|
|
|
rgba->cheight = sfh;
|
|
|
|
for (i = 0, c = 0, e = sfw*sfh; i < e; ++i) {
|
|
|
|
rgba->cache[i] = rgba->data[(int)px + iy];
|
|
|
|
if (++c >= sfw) {
|
|
|
|
c = 0;
|
|
|
|
px = 0;
|
|
|
|
py += dy;
|
|
|
|
iy = (int)py * rgba->width;
|
|
|
|
} else
|
|
|
|
px += dx;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* do we use the cache we may have just created, or the original? */
|
|
|
|
if (rgba->width != sfw || rgba->height != sfh)
|
|
|
|
draw = rgba->cache;
|
|
|
|
|
|
|
|
/* apply the alpha channel */
|
|
|
|
for (i = 0, c = 0, t = position->x, e = sfw*sfh; i < e; ++i, ++t) {
|
|
|
|
guchar alpha, r, g, b, bgr, bgg, bgb;
|
2003-03-26 05:38:13 +00:00
|
|
|
|
2003-05-18 23:06:11 +00:00
|
|
|
alpha = draw[i] >> default_alpha_offset;
|
|
|
|
r = draw[i] >> default_red_offset;
|
|
|
|
g = draw[i] >> default_green_offset;
|
|
|
|
b = draw[i] >> default_blue_offset;
|
2003-03-26 05:38:13 +00:00
|
|
|
|
2003-05-09 16:57:17 +00:00
|
|
|
if (c >= sfw) {
|
|
|
|
c = 0;
|
|
|
|
t += surarea->width - sfw;
|
|
|
|
}
|
2003-03-26 05:38:13 +00:00
|
|
|
|
2003-05-09 16:57:17 +00:00
|
|
|
/* background color */
|
2003-05-18 19:23:26 +00:00
|
|
|
bgr = target[t] >> default_red_offset;
|
|
|
|
bgg = target[t] >> default_green_offset;
|
|
|
|
bgb = target[t] >> default_blue_offset;
|
2003-03-26 05:38:13 +00:00
|
|
|
|
2003-05-09 16:57:17 +00:00
|
|
|
r = bgr + (((r - bgr) * alpha) >> 8);
|
|
|
|
g = bgg + (((g - bgg) * alpha) >> 8);
|
|
|
|
b = bgb + (((b - bgb) * alpha) >> 8);
|
2003-03-26 05:38:13 +00:00
|
|
|
|
2003-05-18 19:23:26 +00:00
|
|
|
target[t] = (r << default_red_offset)
|
|
|
|
| (g << default_green_offset)
|
|
|
|
| (b << default_blue_offset);
|
2003-05-09 16:57:17 +00:00
|
|
|
}
|
2003-03-26 05:38:13 +00:00
|
|
|
}
|
|
|
|
}
|