PpHd Le 17/06/2002 à 13:10 Extrait d'Allegro ?
/*
* Scanline drawers.
*/
#define SCANLINE_DRAWER_GENERIC(name, INIT, PUTPIXEL)
static void draw_scanline_##name(BITMAP *bmp, BITMAP *spr,
fixed l_bmp_x, int bmp_y_i,
fixed r_bmp_x,
fixed l_spr_x, fixed l_spr_y,
fixed spr_dx, fixed spr_dy)
{
int c;
int mask_color;
INIT;
mask_color = bmp->vtable->mask_color;
r_bmp_x >>= 16;
l_bmp_x >>= 16;
for (; l_bmp_x <= r_bmp_x; l_bmp_x++) {
c = getpixel(spr, l_spr_x>>16, l_spr_y>>16);
if (c != mask_color)
PUTPIXEL;
l_spr_x += spr_dx;
l_spr_y += spr_dy;
}
}
SCANLINE_DRAWER_GENERIC(generic_convert,
int bmp_depth;
int spr_depth;
bmp_depth = bitmap_color_depth(bmp);
spr_depth = bitmap_color_depth(spr);
,
putpixel(bmp, l_bmp_x, bmp_y_i,
makecol_depth(bmp_depth,
getr_depth(spr_depth, c),
getg_depth(spr_depth, c),
getb_depth(spr_depth, c))))
SCANLINE_DRAWER_GENERIC(generic,
;
,
putpixel(bmp, l_bmp_x, bmp_y_i, c))
#define SCANLINE_DRAWER(bits_pp, GETPIXEL)
static void draw_scanline_##bits_pp(BITMAP *bmp, BITMAP *spr,
fixed l_bmp_x, int bmp_y_i,
fixed r_bmp_x,
fixed l_spr_x, fixed l_spr_y,
fixed spr_dx, fixed spr_dy)
{
int c;
unsigned long addr, end_addr;
unsigned char **spr_line = spr->line;
r_bmp_x >>= 16;
l_bmp_x >>= 16;
bmp_select(bmp);
addr = bmp_write_line(bmp, bmp_y_i);
end_addr = addr + r_bmp_x * ((bits_pp + 7) / 8);
addr += l_bmp_x * ((bits_pp + 7) / 8);
for (; addr <= end_addr; addr += ((bits_pp + 7) / 8)) {
GETPIXEL;
if (c != MASK_COLOR_##bits_pp)
bmp_write##bits_pp(addr, c);
l_spr_x += spr_dx;
l_spr_y += spr_dy;
}
}
#ifdef ALLEGRO_COLOR8
SCANLINE_DRAWER(8, c = spr_line[l_spr_y>>16][l_spr_x>>16]);
#endif
#ifdef ALLEGRO_COLOR16
SCANLINE_DRAWER(15, c = ((unsigned short *)spr_line[l_spr_y>>16])
[l_spr_x>>16])
SCANLINE_DRAWER(16, c = ((unsigned short *)spr_line[l_spr_y>>16])
[l_spr_x>>16])
#endif
#ifdef ALLEGRO_COLOR24
#ifdef ALLEGRO_LITTLE_ENDIAN
SCANLINE_DRAWER(24,
{
unsigned char *p = spr_line[l_spr_y>>16] +
(l_spr_x>>16) * 3;
c = p[0];
c |= (int)p[1] << 8;
c |= (int)p[2] << 16;
})
#else
SCANLINE_DRAWER(24,
{
unsigned char *p = spr_line[l_spr_y>>16] +
(l_spr_x>>16) * 3;
c = (int)p[0] << 16;
c |= (int)p[1] << 8;
c |= p[2];
})
#endif
#endif
#ifdef ALLEGRO_COLOR32
SCANLINE_DRAWER(32,
c = ((unsigned long *)spr_line[l_spr_y>>16])
[l_spr_x>>16])
#endif
#ifdef GFX_MODEX
static void draw_scanline_modex(
BITMAP *bmp, BITMAP *spr, fixed l_bmp_x, int bmp_y_i, fixed r_bmp_x,
fixed l_spr_x, fixed l_spr_y, fixed spr_dx, fixed spr_dy)
{
int c;
unsigned long start_addr, addr, end_addr;
unsigned char **spr_line = spr->line;
int plane;
fixed spr_x, spr_y;
r_bmp_x >>= 16;
l_bmp_x >>= 16;
start_addr = (unsigned long)bmp->line[bmp_y_i];
spr_dx <<= 2;
spr_dy <<= 2;
for (plane = 0; plane < 4; plane++) {
addr = start_addr + ((l_bmp_x + plane) >> 2);
end_addr = addr + ((r_bmp_x - l_bmp_x - plane) >> 2);
outportw(0x3C4, (0x100 << ((l_bmp_x + plane) & 3)) | 2);
spr_x = l_spr_x;
spr_y = l_spr_y;
for (; addr < end_addr; addr++) {
c = spr_line[spr_y >> 16][spr_x >> 16];
if (c != MASK_COLOR_8)
_farnspokeb(addr, c);
spr_x += spr_dx;
spr_y += spr_dy;
}
l_spr_x += spr_dx >> 2;
l_spr_y += spr_dy >> 2;
}
}
#endif
/* _parallelogram_map:
* Worker routine for drawing rotated and/or scaled and/or flipped sprites:
* It actually maps the sprite to any parallelogram-shaped area of the
* bitmap. The top left corner is mapped to (xs[0], ys[0]), the top right to
* (xs[1], ys[1]), the bottom right to x (xs[2], ys[2]), and the bottom left
* to (xs[3], ys[3]). The corners are assumed to form a perfect
* parallelogram, i.e. xs[0]+xs[2] = xs[1]+xs[3]. The corners are given in
* fixed point format, so xs[] and ys[] are coordinates of the outer corners
* of corner pixels in clockwise order beginning with top left.
* All coordinates begin with 0 in top left corner of pixel (0, 0). So a
* rotation by 0 degrees of a sprite to the top left of a bitmap can be
* specified with coordinates (0, 0) for the top left pixel in source
* bitmap. With the default scanline drawer, a pixel in the destination
* bitmap is drawn if and only if its center is covered by any pixel in the
* sprite. The color of this covering sprite pixel is used to draw.
* If sub_pixel_accuracy=FALSE, then the scanline drawer will be called with
* *_bmp_x being a fixed point representation of the integers representing
* the x coordinate of the first and last point in bmp whose centre is
* covered by the sprite. If sub_pixel_accuracy=TRUE, then the scanline
* drawer will be called with the exact fixed point position of the first
* and last point in which the horizontal line passing through the centre is
* at least partly covered by the sprite. This is useful for doing
* anti-aliased blending.
*/
void _parallelogram_map(BITMAP *bmp, BITMAP *spr, fixed xs[4], fixed ys[4],
void (*draw_scanline)(BITMAP *bmp, BITMAP *spr,
fixed l_bmp_x, int bmp_y,
fixed r_bmp_x,
fixed l_spr_x, fixed l_spr_y,
fixed spr_dx, fixed spr_dy),
int sub_pixel_accuracy)
{
/* Index in xs[] and ys[] to topmost point. */
int top_index;
/* Rightmost point has index (top_index+right_index) int xs[] and ys[]. */
int right_index;
/* Loop variables. */
int index, i;
/* Coordinates in bmp ordered as top-right-bottom-left. */
fixed corner_bmp_x[4], corner_bmp_y[4];
/* Coordinates in spr ordered as top-right-bottom-left. */
fixed corner_spr_x[4], corner_spr_y[4];
/* y coordinate of bottom point, left point and right point. */
int clip_bottom_i, l_bmp_y_bottom_i, r_bmp_y_bottom_i;
/* Left and right clipping. */
fixed clip_left, clip_right;
/* Temporary variable. */
fixed extra_scanline_fraction;
/*
* Variables used in the loop
*/
/* Coordinates of sprite and bmp points in beginning of scanline. */
fixed l_spr_x, l_spr_y, l_bmp_x, l_bmp_dx;
/* Increment of left sprite point as we move a scanline down. */
fixed l_spr_dx, l_spr_dy;
/* Coordinates of sprite and bmp points in end of scanline. */
fixed r_bmp_x, r_bmp_dx;
#ifdef KEEP_TRACK_OF_RIGHT_SPRITE_SCANLINE
fixed r_spr_x, r_spr_y;
/* Increment of right sprite point as we move a scanline down. */
fixed r_spr_dx, r_spr_dy;
#endif
/* Increment of sprite point as we move right inside a scanline. */
fixed spr_dx, spr_dy;
/* Positions of beginning of scanline after rounding to integer coordinate
in bmp. */
fixed l_spr_x_rounded, l_spr_y_rounded, l_bmp_x_rounded;
fixed r_bmp_x_rounded;
/* Current scanline. */
int bmp_y_i;
/* Right edge of scanline. */
int right_edge_test;
/* Get index of topmost point. */
top_index = 0;
if (ys[1] < ys[0])
top_index = 1;
if (ys[2] < ys[top_index])
top_index = 2;
if (ys[3] < ys[top_index])
top_index = 3;
/* Get direction of points: clockwise or anti-clockwise. */
if (fixmul(xs[(top_index+1) & 3] - xs[top_index],
ys[(top_index-1) & 3] - ys[top_index]) >
fixmul(xs[(top_index-1) & 3] - xs[top_index],
ys[(top_index+1) & 3] - ys[top_index]))
right_index = 1;
else
right_index = -1;
/*
* Get coordinates of the corners.
*/
/* corner_*[0] is top, [1] is right, [2] is bottom, [3] is left. */
index = top_index;
for (i = 0; i < 4; i++) {
corner_bmp_x[i] = xs[index];
corner_bmp_y[i] = ys[index];
if (index < 2)
corner_spr_y[i] = 0;
else
/* Need `- 1' since otherwise it would be outside sprite. */
corner_spr_y[i] = (spr->h << 16) - 1;
if ((index == 0) || (index == 3))
corner_spr_x[i] = 0;
else
corner_spr_x[i] = (spr->w << 16) - 1;
index = (index + right_index) & 3;
}
/*
* Get scanline starts, ends and deltas, and clipping coordinates.
*/
#define top_bmp_y corner_bmp_y[0]
#define right_bmp_y corner_bmp_y[1]
#define bottom_bmp_y corner_bmp_y[2]
#define left_bmp_y corner_bmp_y[3]
#define top_bmp_x corner_bmp_x[0]
#define right_bmp_x corner_bmp_x[1]
#define bottom_bmp_x corner_bmp_x[2]
#define left_bmp_x corner_bmp_x[3]
#define top_spr_y corner_spr_y[0]
#define right_spr_y corner_spr_y[1]
#define bottom_spr_y corner_spr_y[2]
#define left_spr_y corner_spr_y[3]
#define top_spr_x corner_spr_x[0]
#define right_spr_x corner_spr_x[1]
#define bottom_spr_x corner_spr_x[2]
#define left_spr_x corner_spr_x[3]
/* Calculate left and right clipping. */
if (bmp->clip) {
clip_left = bmp->cl << 16;
clip_right = (bmp->cr << 16) - 1;
}
else {
clip_left = 0;
clip_right = bmp->w << 16;
}
/* Quit if we're totally outside. */
if ((left_bmp_x > clip_right) &&
(top_bmp_x > clip_right) &&
(bottom_bmp_x > clip_right))
return;
if ((right_bmp_x < clip_left) &&
(top_bmp_x < clip_left) &&
(bottom_bmp_x < clip_left))
return;
/* Bottom clipping. */
if (sub_pixel_accuracy)
clip_bottom_i = (bottom_bmp_y + 0xffff) >> 16;
else
clip_bottom_i = (bottom_bmp_y + 0x8000) >> 16;
if (bmp->clip)
if (clip_bottom_i > bmp->cb)
clip_bottom_i = bmp->cb;
/* Calculate y coordinate of first scanline. */
if (sub_pixel_accuracy)
bmp_y_i = top_bmp_y >> 16;
else
bmp_y_i = (top_bmp_y + 0x8000) >> 16;
if (bmp->clip)
if (bmp_y_i < bmp->ct)
bmp_y_i = bmp->ct;
/* Sprite is above or below bottom clipping area. */
if (bmp_y_i >= clip_bottom_i)
return;
/* Vertical gap between top corner and centre of topmost scanline. */
extra_scanline_fraction = (bmp_y_i << 16) + 0x8000 - top_bmp_y;
/* Calculate x coordinate of beginning of scanline in bmp. */
l_bmp_dx = fixdiv(left_bmp_x - top_bmp_x,
left_bmp_y - top_bmp_y);
l_bmp_x = top_bmp_x + fixmul(extra_scanline_fraction, l_bmp_dx);
/* Calculate x coordinate of beginning of scanline in spr. */
/* note: all these are rounded down which is probably a Good Thing (tm) */
l_spr_dx = fixdiv(left_spr_x - top_spr_x,
left_bmp_y - top_bmp_y);
l_spr_x = top_spr_x + fixmul(extra_scanline_fraction, l_spr_dx);
/* Calculate y coordinate of beginning of scanline in spr. */
l_spr_dy = fixdiv(left_spr_y - top_spr_y,
left_bmp_y - top_bmp_y);
l_spr_y = top_spr_y + fixmul(extra_scanline_fraction, l_spr_dy);
/* Calculate left loop bound. */
l_bmp_y_bottom_i = (left_bmp_y + 0x8000) >> 16;
if (l_bmp_y_bottom_i > clip_bottom_i)
l_bmp_y_bottom_i = clip_bottom_i;
/* Calculate x coordinate of end of scanline in bmp. */
r_bmp_dx = fixdiv(right_bmp_x - top_bmp_x,
right_bmp_y - top_bmp_y);
r_bmp_x = top_bmp_x + fixmul(extra_scanline_fraction, r_bmp_dx);
#ifdef KEEP_TRACK_OF_RIGHT_SPRITE_SCANLINE
/* Calculate x coordinate of end of scanline in spr. */
r_spr_dx = fixdiv(right_spr_x - top_spr_x,
right_bmp_y - top_bmp_y);
r_spr_x = top_spr_x + fixmul(extra_scanline_fraction, r_spr_dx);
/* Calculate y coordinate of end of scanline in spr. */
r_spr_dy = fixdiv(right_spr_y - top_spr_y,
right_bmp_y - top_bmp_y);
r_spr_y = top_spr_y + fixmul(extra_scanline_fraction, r_spr_dy);
#endif
/* Calculate right loop bound. */
r_bmp_y_bottom_i = (right_bmp_y + 0x8000) >> 16;
/* Get dx and dy, the offsets to add to the source coordinates as we move
one pixel rightwards along a scanline. This formula can be derived by
considering the 2x2 matrix that transforms the sprite to the
parallelogram.
We'd better use double to get this as exact as possible, since any
errors will be accumulated along the scanline.
*/
spr_dx = (fixed)((ys[3] - ys[0]) * 65536.0 * (65536.0 * spr->w) /
((xs[1] - xs[0]) * (double)(ys[3] - ys[0]) -
(xs[3] - xs[0]) * (double)(ys[1] - ys[0])));
spr_dy = (fixed)((ys[1] - ys[0]) * 65536.0 * (65536.0 * spr->h) /
((xs[3] - xs[0]) * (double)(ys[1] - ys[0]) -
(xs[1] - xs[0]) * (double)(ys[3] - ys[0])));
/*
* Loop through scanlines.
*/
while (1) {
/* Has beginning of scanline passed a corner? */
if (bmp_y_i >= l_bmp_y_bottom_i) {
/* Are we done? */
if (bmp_y_i >= clip_bottom_i)
break;
/* Vertical gap between left corner and centre of scanline. */
extra_scanline_fraction = (bmp_y_i << 16) + 0x8000 - left_bmp_y;
/* Update x coordinate of beginning of scanline in bmp. */
l_bmp_dx = fixdiv(bottom_bmp_x - left_bmp_x,
bottom_bmp_y - left_bmp_y);
l_bmp_x = left_bmp_x + fixmul(extra_scanline_fraction, l_bmp_dx);
/* Update x coordinate of beginning of scanline in spr. */
l_spr_dx = fixdiv(bottom_spr_x - left_spr_x,
bottom_bmp_y - left_bmp_y);
l_spr_x = left_spr_x + fixmul(extra_scanline_fraction, l_spr_dx);
/* Update y coordinate of beginning of scanline in spr. */
l_spr_dy = fixdiv(bottom_spr_y - left_spr_y,
bottom_bmp_y - left_bmp_y);
l_spr_y = left_spr_y + fixmul(extra_scanline_fraction, l_spr_dy);
/* Update loop bound. */
if (sub_pixel_accuracy)
l_bmp_y_bottom_i = (bottom_bmp_y + 0xffff) >> 16;
else
l_bmp_y_bottom_i = (bottom_bmp_y + 0x8000) >> 16;
if (l_bmp_y_bottom_i > clip_bottom_i)
l_bmp_y_bottom_i = clip_bottom_i;
}
/* Has end of scanline passed a corner? */
if (bmp_y_i >= r_bmp_y_bottom_i) {
/* Vertical gap between right corner and centre of scanline. */
extra_scanline_fraction = (bmp_y_i << 16) + 0x8000 - right_bmp_y;
/* Update x coordinate of end of scanline in bmp. */
r_bmp_dx = fixdiv(bottom_bmp_x - right_bmp_x,
bottom_bmp_y - right_bmp_y);
r_bmp_x = right_bmp_x + fixmul(extra_scanline_fraction, r_bmp_dx);
#ifdef KEEP_TRACK_OF_RIGHT_SPRITE_SCANLINE
/* Update x coordinate of beginning of scanline in spr. */
r_spr_dx = fixdiv(bottom_spr_x - right_spr_x,
bottom_bmp_y - right_bmp_y);
r_spr_x = right_spr_x + fixmul(extra_scanline_fraction, r_spr_dx);
/* Update y coordinate of beginning of scanline in spr. */
r_spr_dy = fixdiv(bottom_spr_y - right_spr_y,
bottom_bmp_y - right_bmp_y);
r_spr_y = right_spr_y + fixmul(extra_scanline_fraction, r_spr_dy);
#endif
/* Update loop bound: We aren't supposed to use this any more, so
just set it to some big enough value. */
r_bmp_y_bottom_i = clip_bottom_i;
}
/* Make left bmp coordinate be an integer and clip it. */
if (sub_pixel_accuracy)
l_bmp_x_rounded = l_bmp_x;
else
l_bmp_x_rounded = (l_bmp_x + 0x8000) & ~0xffff;
if (l_bmp_x_rounded < clip_left)
l_bmp_x_rounded = clip_left;
/* ... and move starting point in sprite accordingly. */
if (sub_pixel_accuracy) {
l_spr_x_rounded = l_spr_x +
fixmul((l_bmp_x_rounded - l_bmp_x), spr_dx);
l_spr_y_rounded = l_spr_y +
fixmul((l_bmp_x_rounded - l_bmp_x), spr_dy);
}
else {
l_spr_x_rounded = l_spr_x +
fixmul(l_bmp_x_rounded + 0x7fff - l_bmp_x, spr_dx);
l_spr_y_rounded = l_spr_y +
fixmul(l_bmp_x_rounded + 0x7fff - l_bmp_x, spr_dy);
}
/* Make right bmp coordinate be an integer and clip it. */
if (sub_pixel_accuracy)
r_bmp_x_rounded = r_bmp_x;
else
r_bmp_x_rounded = (r_bmp_x - 0x8000) & ~0xffff;
if (r_bmp_x_rounded > clip_right)
r_bmp_x_rounded = clip_right;
/* Draw! */
if (l_bmp_x_rounded <= r_bmp_x_rounded) {
if (!sub_pixel_accuracy) {
/* The bodies of these ifs are only reached extremely seldom,
it's an ugly hack to avoid reading outside the sprite when
the rounding errors are accumulated the wrong way. It would
be nicer if we could ensure that this never happens by making
all multiplications and divisions be rounded up or down at
the correct places.
I did try another approach: recalculate the edges of the
scanline from scratch each scanline rather than incrementally.
Drawing a sprite with that routine took about 25% longer time
though.
*/
if ((unsigned)(l_spr_x_rounded >> 16) >= (unsigned)spr->w) {
if ((l_spr_x_rounded < 0) == (spr_dx < 0)) {
/* This can happen. */
goto skip_draw;
}
else {
/* I don't think this can happen, but I can't prove it. */
while ((unsigned)(l_spr_x_rounded >> 16) >=
(unsigned)spr->w) {
l_spr_x_rounded += spr_dx;
l_bmp_x_rounded += 65536;
}
if (l_bmp_x_rounded > r_bmp_x_rounded)
goto skip_draw;
}
}
right_edge_test = l_spr_x_rounded +
((r_bmp_x_rounded - l_bmp_x_rounded) >> 16) *
spr_dx;
if ((unsigned)(right_edge_test >> 16) >= (unsigned)spr->w) {
if ((right_edge_test < 0) == (spr_dx < 0)) {
/* This can happen. */
while ((unsigned)(right_edge_test >> 16) >=
(unsigned)spr->w) {
r_bmp_x_rounded -= 65536;
right_edge_test -= spr_dx;
}
if (l_bmp_x_rounded > r_bmp_x_rounded)
goto skip_draw;
}
else {
/* I don't think this can happen, but I can't prove it. */
goto skip_draw;
}
}
if ((unsigned)(l_spr_y_rounded >> 16) >= (unsigned)spr->h) {
if ((l_spr_y_rounded < 0) == (spr_dy < 0)) {
/* This can happen. */
goto skip_draw;
}
else {
/* I don't think this can happen, but I can't prove it. */
while (((unsigned)l_spr_y_rounded >> 16) >=
(unsigned)spr->h) {
l_spr_y_rounded += spr_dy;
l_bmp_x_rounded += 65536;
}
if (l_bmp_x_rounded > r_bmp_x_rounded)
goto skip_draw;
}
}
right_edge_test = l_spr_y_rounded +
((r_bmp_x_rounded - l_bmp_x_rounded) >> 16) *
spr_dy;
if ((unsigned)(right_edge_test >> 16) >= (unsigned)spr->h) {
if ((right_edge_test < 0) == (spr_dy < 0)) {
/* This can happen. */
while ((unsigned)(right_edge_test >> 16) >=
(unsigned)spr->h) {
r_bmp_x_rounded -= 65536;
right_edge_test -= spr_dy;
}
if (l_bmp_x_rounded > r_bmp_x_rounded)
goto skip_draw;
}
else {
/* I don't think this can happen, but I can't prove it. */
goto skip_draw;
}
}
}
draw_scanline(bmp, spr,
l_bmp_x_rounded, bmp_y_i, r_bmp_x_rounded,
l_spr_x_rounded, l_spr_y_rounded,
spr_dx, spr_dy);
}
/* I'm not going to apoligize for this label and its gotos: to get
rid of it would just make the code look worse. */
skip_draw:
/* Jump to next scanline. */
bmp_y_i++;
/* Update beginning of scanline. */
l_bmp_x += l_bmp_dx;
l_spr_x += l_spr_dx;
l_spr_y += l_spr_dy;
/* Update end of scanline. */
r_bmp_x += r_bmp_dx;
#ifdef KEEP_TRACK_OF_RIGHT_SPRITE_SCANLINE
r_spr_x += r_spr_dx;
r_spr_y += r_spr_dy;
#endif
}
bmp_unwrite_line(bmp);
}
/* _parallelogram_map_standard:
* Helper function for calling _parallelogram_map() with the appropriate
* scanline drawer. I didn't want to include this in the
* _parallelogram_map() function since then you can bypass it and define
* your own scanline drawer, eg. for anti-aliased rotations.
*/
void _parallelogram_map_standard(BITMAP *bmp, BITMAP *sprite,
fixed xs[4], fixed ys[4])
{
int old_drawing_mode;
if (bitmap_color_depth(bmp) != bitmap_color_depth(sprite)) {
/* These scanline drawers use putpixel() so we must set solid mode. */
old_drawing_mode = _drawing_mode;
drawing_mode(DRAW_MODE_SOLID, _drawing_pattern,
_drawing_x_anchor, _drawing_y_anchor);
_parallelogram_map(bmp, sprite, xs, ys,
draw_scanline_generic_convert, FALSE);
drawing_mode(old_drawing_mode, _drawing_pattern,
_drawing_x_anchor, _drawing_y_anchor);
}
else if (!is_memory_bitmap(sprite)) {
old_drawing_mode = _drawing_mode;
drawing_mode(DRAW_MODE_SOLID, _drawing_pattern,
_drawing_x_anchor, _drawing_y_anchor);
_parallelogram_map(bmp, sprite, xs, ys,
draw_scanline_generic, FALSE);
drawing_mode(old_drawing_mode, _drawing_pattern,
_drawing_x_anchor, _drawing_y_anchor);
}
else if (is_linear_bitmap(bmp)) {
switch (bitmap_color_depth(bmp)) {
#ifdef ALLEGRO_COLOR8
case 8:
_parallelogram_map(bmp, sprite, xs, ys,
draw_scanline_8, FALSE);
break;
#endif
#ifdef ALLEGRO_COLOR16
case 15:
_parallelogram_map(bmp, sprite, xs, ys,
draw_scanline_15, FALSE);
break;
case 16:
_parallelogram_map(bmp, sprite, xs, ys,
draw_scanline_16, FALSE);
break;
#endif
#ifdef ALLEGRO_COLOR24
case 24:
_parallelogram_map(bmp, sprite, xs, ys,
draw_scanline_24, FALSE);
break;
#endif
#ifdef ALLEGRO_COLOR32
case 32:
_parallelogram_map(bmp, sprite, xs, ys,
draw_scanline_32, FALSE);
break;
#endif
default:
/* NOTREACHED */
ASSERT(0);
}
}
#ifdef GFX_MODEX
else {
_parallelogram_map(bmp, sprite, xs, ys,
draw_scanline_modex, FALSE);
}
#endif
}
/* _rotate_scale_flip_coordinates:
* Calculates the coordinates for the rotated, scaled and flipped sprite,
* and passes them on to the given function.
*/
void _rotate_scale_flip_coordinates(fixed w, fixed h,
fixed x, fixed y, fixed cx, fixed cy,
fixed angle,
fixed scale_x, fixed scale_y,
int h_flip, int v_flip,
fixed xs[4], fixed ys[4])
{
fixed fix_cos, fix_sin;
int tl = 0, tr = 1, bl = 3, br = 2;
int tmp;
double cos_angle, sin_angle;
fixed xofs, yofs;
/* Setting angle to the range -180...180 degrees makes sin & cos
more numerically stable. (Yes, this does have an effect for big
angles!) Note that using "real" sin() and cos() gives much better
precision than fixsin() and fixcos(). */
angle = angle & 0xffffff;
if (angle >= 0x800000)
angle -= 0x1000000;
#if (defined ALLEGRO_DJGPP) && (DJGPP >= 2) && (DJGPP_MINOR >= 3)
/* Faster way to obtain both sin and cos with one call under djgpp.
Todo: Do other platforms support this? (not watcom, mingw or msvc)
*/
sincos(&cos_angle, &sin_angle, angle * (M_PI / (double)0x800000));
#else
cos_angle = cos(angle * (M_PI / (double)0x800000));
sin_angle = sin(angle * (M_PI / (double)0x800000));
#endif
if (cos_angle >= 0)
fix_cos = (int)(cos_angle * 0x10000 + 0.5);
else
fix_cos = (int)(cos_angle * 0x10000 - 0.5);
if (sin_angle >= 0)
fix_sin = (int)(sin_angle * 0x10000 + 0.5);
else
fix_sin = (int)(sin_angle * 0x10000 - 0.5);
/* Decide what order to take corners in. */
if (v_flip) {
tl = 3;
tr = 2;
bl = 0;
br = 1;
}
else {
tl = 0;
tr = 1;
bl = 3;
br = 2;
}
if (h_flip) {
tmp = tl;
tl = tr;
tr = tmp;
tmp = bl;
bl = br;
br = tmp;
}
/* Calculate new coordinates of all corners. */
w = fixmul(w, scale_x);
h = fixmul(h, scale_y);
cx = fixmul(cx, scale_x);
cy = fixmul(cy, scale_y);
xofs = x - fixmul(cx, fix_cos) + fixmul(cy, fix_sin);
yofs = y - fixmul(cx, fix_sin) - fixmul(cy, fix_cos);
xs[tl] = xofs;
ys[tl] = yofs;
xs[tr] = xofs + fixmul(w, fix_cos);
ys[tr] = yofs + fixmul(w, fix_sin);
xs[bl] = xofs - fixmul(h, fix_sin);
ys[bl] = yofs + fixmul(h, fix_cos);
xs[br] = xs[tr] + xs[bl] - xs[tl];
ys[br] = ys[tr] + ys[bl] - ys[tl];
}
/* _pivot_scaled_sprite_flip:
* The most generic routine to which you specify the position with angles,
* scales, etc.
*/
void _pivot_scaled_sprite_flip(BITMAP *bmp, BITMAP *sprite,
fixed x, fixed y, fixed cx, fixed cy,
fixed angle, fixed scale, int v_flip)
{
fixed xs[4], ys[4];
_rotate_scale_flip_coordinates(sprite->w << 16, sprite->h << 16,
x, y, cx, cy, angle, scale, scale,
FALSE, v_flip, xs, ys);
_parallelogram_map_standard(bmp, sprite, xs, ys);
}
Je suis un boieuf, je sais.