gdi32: Simplify the Bresenham line drawing.

This commit is contained in:
Huw Davies 2011-11-21 10:12:51 +00:00 committed by Alexandre Julliard
parent 8cd1d5839f
commit f168f838f1
2 changed files with 97 additions and 58 deletions

View File

@ -147,6 +147,14 @@ static inline dibdrv_physdev *get_dibdrv_pdev( PHYSDEV dev )
return (dibdrv_physdev *)dev; return (dibdrv_physdev *)dev;
} }
struct line_params
{
int err_start, err_add_1, err_add_2, bias;
unsigned int length;
int x_inc, y_inc;
BOOL x_major;
};
struct stretch_params struct stretch_params
{ {
int err_start, err_add_1, err_add_2; int err_start, err_add_1, err_add_2;

View File

@ -545,44 +545,39 @@ int clip_line(const POINT *start, const POINT *end, const RECT *clip,
} }
} }
static void bres_line_with_bias(INT x1, INT y1, INT x2, INT y2, const bres_params *params, INT err, static void bres_line_with_bias(const POINT *start, const struct line_params *params,
BOOL last_pt, void (* callback)(dibdrv_physdev*,INT,INT), dibdrv_physdev *pdev) void (* callback)(dibdrv_physdev*,INT,INT), dibdrv_physdev *pdev)
{ {
const int xadd = is_x_increasing(params->octant) ? 1 : -1; POINT pt = *start;
const int yadd = is_y_increasing(params->octant) ? 1 : -1; int len = params->length, err = params->err_start;
INT erradd;
if (is_xmajor(params->octant)) /* line is "more horizontal" */ if (params->x_major)
{ {
erradd = 2*params->dy - 2*params->dx; while(len--)
while(x1 != x2)
{ {
callback(pdev, x1, y1); callback(pdev, pt.x, pt.y);
if (err + params->bias > 0) if (err + params->bias > 0)
{ {
y1 += yadd; pt.y += params->y_inc;
err += erradd; err += params->err_add_1;
} }
else err += 2*params->dy; else err += params->err_add_2;
x1 += xadd; pt.x += params->x_inc;
} }
if(last_pt) callback(pdev, x1, y1);
} }
else /* line is "more vertical" */ else
{ {
erradd = 2*params->dx - 2*params->dy; while(len--)
while(y1 != y2)
{ {
callback(pdev, x1, y1); callback(pdev, pt.x, pt.y);
if (err + params->bias > 0) if (err + params->bias > 0)
{ {
x1 += xadd; pt.x += params->x_inc;
err += erradd; err += params->err_add_1;
} }
else err += 2*params->dx; else err += params->err_add_2;
y1 += yadd; pt.y += params->y_inc;
} }
if(last_pt) callback(pdev, x1, y1);
} }
} }
@ -651,38 +646,58 @@ static BOOL solid_pen_line(dibdrv_physdev *pdev, POINT *start, POINT *end)
} }
else else
{ {
bres_params params; bres_params clip_params;
INT dx = end->x - start->x; struct line_params line_params;
INT dy = end->y - start->y; INT dx = end->x - start->x, dy = end->y - start->y;
INT abs_dx = abs(dx), abs_dy = abs(dy);
INT i; INT i;
params.dx = abs(dx); clip_params.dx = abs_dx;
params.dy = abs(dy); clip_params.dy = abs_dy;
params.octant = get_octant_mask(dx, dy); clip_params.octant = get_octant_mask(dx, dy);
params.bias = get_bias(params.octant); clip_params.bias = get_bias( clip_params.octant );
line_params.bias = clip_params.bias;
line_params.x_major = is_xmajor( clip_params.octant );
line_params.x_inc = is_x_increasing( clip_params.octant ) ? 1 : -1;
line_params.y_inc = is_y_increasing( clip_params.octant ) ? 1 : -1;
if (line_params.x_major)
{
line_params.err_add_1 = 2 * abs_dy - 2 * abs_dx;
line_params.err_add_2 = 2 * abs_dy;
}
else
{
line_params.err_add_1 = 2 * abs_dx - 2 * abs_dy;
line_params.err_add_2 = 2 * abs_dx;
}
for(i = 0; i < clip->numRects; i++) for(i = 0; i < clip->numRects; i++)
{ {
POINT clipped_start, clipped_end; POINT clipped_start, clipped_end;
int clip_status; int clip_status;
clip_status = clip_line(start, end, clip->rects + i, &params, &clipped_start, &clipped_end); clip_status = clip_line(start, end, clip->rects + i, &clip_params, &clipped_start, &clipped_end);
if(clip_status) if(clip_status)
{ {
int m = abs(clipped_start.x - start->x); int m = abs(clipped_start.x - start->x);
int n = abs(clipped_start.y - start->y); int n = abs(clipped_start.y - start->y);
int err;
BOOL last_pt = FALSE;
if(is_xmajor(params.octant)) if (line_params.x_major)
err = 2 * params.dy - params.dx + m * 2 * params.dy - n * 2 * params.dx; {
line_params.err_start = 2 * abs_dy - abs_dx + m * 2 * abs_dy - n * 2 * abs_dx;
line_params.length = abs( clipped_end.x - clipped_start.x ) + 1;
}
else else
err = 2 * params.dx - params.dy + n * 2 * params.dx - m * 2 * params.dy; {
line_params.err_start = 2 * abs_dx - abs_dy + n * 2 * abs_dx - m * 2 * abs_dy;
line_params.length = abs( clipped_end.y - clipped_start.y ) + 1;
}
if(clip_status == 1 && (end->x != clipped_end.x || end->y != clipped_end.y)) last_pt = TRUE; if (clipped_end.x == end->x && clipped_end.y == end->y) line_params.length--;
bres_line_with_bias(clipped_start.x, clipped_start.y, clipped_end.x, clipped_end.y, &params, bres_line_with_bias( &clipped_start, &line_params, solid_pen_line_callback, pdev );
err, last_pt, solid_pen_line_callback, pdev);
if(clip_status == 2) break; /* completely unclipped, so we can finish */ if(clip_status == 2) break; /* completely unclipped, so we can finish */
} }
@ -923,54 +938,70 @@ static BOOL dashed_pen_line(dibdrv_physdev *pdev, POINT *start, POINT *end)
} }
else else
{ {
bres_params params; bres_params clip_params;
INT dx = end->x - start->x; struct line_params line_params;
INT dy = end->y - start->y; INT dx = end->x - start->x, dy = end->y - start->y;
INT abs_dx = abs(dx), abs_dy = abs(dy);
INT i; INT i;
params.dx = abs(dx); clip_params.dx = abs_dx;
params.dy = abs(dy); clip_params.dy = abs_dy;
params.octant = get_octant_mask(dx, dy); clip_params.octant = get_octant_mask(dx, dy);
params.bias = get_bias(params.octant); clip_params.bias = get_bias( clip_params.octant );
line_params.bias = clip_params.bias;
line_params.x_major = is_xmajor( clip_params.octant );
line_params.x_inc = is_x_increasing( clip_params.octant ) ? 1 : -1;
line_params.y_inc = is_y_increasing( clip_params.octant ) ? 1 : -1;
if (line_params.x_major)
{
line_params.err_add_1 = 2 * abs_dy - 2 * abs_dx;
line_params.err_add_2 = 2 * abs_dy;
}
else
{
line_params.err_add_1 = 2 * abs_dx - 2 * abs_dy;
line_params.err_add_2 = 2 * abs_dx;
}
for(i = 0; i < clip->numRects; i++) for(i = 0; i < clip->numRects; i++)
{ {
POINT clipped_start, clipped_end; POINT clipped_start, clipped_end;
int clip_status; int clip_status;
clip_status = clip_line(start, end, clip->rects + i, &params, &clipped_start, &clipped_end); clip_status = clip_line(start, end, clip->rects + i, &clip_params, &clipped_start, &clipped_end);
if(clip_status) if(clip_status)
{ {
int m = abs(clipped_start.x - start->x); int m = abs(clipped_start.x - start->x);
int n = abs(clipped_start.y - start->y); int n = abs(clipped_start.y - start->y);
int err;
BOOL last_pt = FALSE;
pdev->dash_pos = start_pos; pdev->dash_pos = start_pos;
if(is_xmajor(params.octant)) if (line_params.x_major)
{ {
err = 2 * params.dy - params.dx + m * 2 * params.dy - n * 2 * params.dx; line_params.err_start = 2 * abs_dy - abs_dx + m * 2 * abs_dy - n * 2 * abs_dx;
line_params.length = abs( clipped_end.x - clipped_start.x ) + 1;
skip_dash(pdev, m); skip_dash(pdev, m);
} }
else else
{ {
err = 2 * params.dx - params.dy + n * 2 * params.dx - m * 2 * params.dy; line_params.err_start = 2 * abs_dx - abs_dy + n * 2 * abs_dx - m * 2 * abs_dy;
line_params.length = abs( clipped_end.y - clipped_start.y ) + 1;
skip_dash(pdev, n); skip_dash(pdev, n);
} }
if(clip_status == 1 && (end->x != clipped_end.x || end->y != clipped_end.y)) last_pt = TRUE; if (clipped_end.x == end->x && clipped_end.y == end->y) line_params.length--;
bres_line_with_bias(clipped_start.x, clipped_start.y, clipped_end.x, clipped_end.y, &params, bres_line_with_bias( &clipped_start, &line_params, dashed_pen_line_callback, pdev );
err, last_pt, dashed_pen_line_callback, pdev);
if(clip_status == 2) break; /* completely unclipped, so we can finish */ if(clip_status == 2) break; /* completely unclipped, so we can finish */
} }
} }
pdev->dash_pos = start_pos; pdev->dash_pos = start_pos;
if(is_xmajor(params.octant)) if(line_params.x_major)
skip_dash(pdev, params.dx); skip_dash(pdev, abs_dx);
else else
skip_dash(pdev, params.dy); skip_dash(pdev, abs_dy);
} }
release_wine_region(pdev->clip); release_wine_region(pdev->clip);