Mercurial > MadButterfly
diff src/shape_path.c @ 97:9453e68092b5
Fix bug of translation relative to absolute points
author | Thinker K.F. Li <thinker@branda.to> |
---|---|
date | Thu, 04 Sep 2008 08:21:39 +0800 |
parents | ca94493b75bb |
children | 688f76b8e71c |
line wrap: on
line diff
--- a/src/shape_path.c Sat Aug 30 00:32:05 2008 +0800 +++ b/src/shape_path.c Thu Sep 04 08:21:39 2008 +0800 @@ -104,6 +104,8 @@ if(p == old) return ERR; arg_cnt++; + + cmd_cnt++; } break; case 's': @@ -139,6 +141,8 @@ if(p == old) return ERR; arg_cnt++; + + cmd_cnt++; } break; case 'm': @@ -162,6 +166,8 @@ if(p == old) return ERR; arg_cnt++; + + cmd_cnt++; } break; case 'h': @@ -175,6 +181,8 @@ if(p == old) break; arg_cnt += 2; + + cmd_cnt++; } break; case 'A': @@ -195,16 +203,18 @@ } arg_cnt += 6; + + cmd_cnt++; } break; case 'z': case 'Z': + cmd_cnt++; break; default: return ERR; } /*! \todo cmd_cnt should be increased for each implicit repeating. */ - cmd_cnt++; SKIP_SPACE(p); } @@ -213,27 +223,86 @@ return OK; } +#include <math.h> +/*! \brief Calculate center of the ellipse of an arc. + * + * - ux0 = x0 / rx + * - uy0 = y0 / ry + * - ux = x / rx + * - uy = y / rx + * ux0, uy0, ux, yu are got by transforming (x0, y0) and (x, y) into points + * on the unit circle. The center of unit circle are (ucx, ucy): + * - umx = (ux0 + ux) / 2 + * - umy = (uy0 + uy) / 2 + * - udcx = ucx - umx + * - udcy = ucy - umy + * - udx = ux - umx + * - udy = uy - umy + * - udcx * udx + udcy * udy = 0 + * - udcy = - udcx * udx / udy + * - udcx ** 2 + udcy ** 2 + udx ** 2 + udy ** 2 = 1 + * - udcx ** 2 + (udcx * udx / udy) ** 2 = 1 - udx ** 2 - udy ** 2 + * - udcx ** 2 = (1 - udx ** 2 - udy ** 2) / (1 + (udx/udy) ** 2) + * + * - cx = rx * ucx + * - cx = rx * (udcx + umx) + * - cy = ry * ucy + * - cy = ry * (udcy + umy) + */ static int calc_center_and_x_aix(co_aix x0, co_aix y0, co_aix x, co_aix y, + co_aix rx, co_aix ry, co_aix x_rotate, int large, int sweep, co_aix *cx, co_aix *cy, co_aix *xx, co_aix *xy) { + co_aix nrx, nry, nrx0, nry0; + co_aix udx, udy, udx2, udy2; + co_aix umx, umy; + co_aix udcx, udcy; + co_aix nrcx, nrcy; + float _sin = sinf(x_rotate); + float _cos = cosf(x_rotate); + nrx = x * _cos + y * _sin; + nry = x * -_sin + y * _cos; + nrx0 = x0 * _cos + y0 * _sin; + nry0 = x0 * -_sin + y0 * _cos; + + udx = (nrx - nrx0) / 2 / rx; + udy = (nry - nry0) / 2 / ry; + umx = (nrx + nrx0) / 2 / rx; + umy = (nry + nry0) / 2 / ry; + udx2 = udx * udx; + udy2 = udy * udy; + udcx = sqrtf((1 - udx2 - udy2) / (1 + udx2 / udy2)); + udcy = - udcx * udx / udy; + nrcx = rx * (udcx + umx); + nrcy = ry * (udcy + umy); + + *cx = nrcx * _cos - nrcy * _sin; + *cy = nrcx * _sin + nrcy * _cos; + + *xx = rx * _cos + *cx; + *xy = rx * _sin + *cy; + + return OK; } -static int sh_path_arc_cmd_arg_fill(char cmd, const char *data, - co_aix **args_p, const char **old_p) { +static int sh_path_arc_cmd_arg_fill(char cmd, char **cmds_p, + const char **data_p, co_aix **args_p) { co_aix rx, ry; co_aix x_rotate; int large, sweep; co_aix x, y, x0, y0, cx, cy, xx, xy; co_aix *args = *args_p; - const char *old = *old_p; + const char *old; const char *p; + char *cmds; - p = data; + p = *data_p; + cmds = *cmds_p; while(*p) { SKIP_SPACE(p); old = p; @@ -293,6 +362,7 @@ } calc_center_and_x_aix(x0, y0, x, y, + rx, ry, x_rotate, large, sweep, &cx, &cy, &xx, &xy); @@ -302,20 +372,31 @@ *(args++) = xy; *(args++) = x; *(args++) = y; + + *cmds++ = toupper(cmd); } - *old_p = old; + *data_p = p; + *args_p = args; + *cmds_p = cmds; return OK; } -#define TO_ABS islower(cmd)? *(args - 2) + atof(old): atof(old) +void sh_path_arc_path(cairo_t *cr, const co_aix **args) { + +} + +#define TO_ABSX islower(cmd)? x + atof(old): atof(old) +#define TO_ABSY islower(cmd)? y + atof(old): atof(old) static int sh_path_cmd_arg_fill(char *data, sh_path_t *path) { char *p, *old; char *cmds; char cmd; co_aix *args; + co_aix x, y; + int r; cmds = path->user_data; args = (co_aix *)(cmds + path->cmd_len); @@ -323,7 +404,9 @@ SKIP_SPACE(p); while(*p) { /* Transform all relative to absolute, */ - *cmds++ = toupper(*p); + x = *(args - 2); + y = *(args - 1); + switch((cmd = *p++)) { case 'c': case 'C': @@ -334,7 +417,7 @@ SKIP_NUM(p); if(p == old) break; - *args = TO_ABS; + *args = TO_ABSX; args++; SKIP_SPACE(p); @@ -342,7 +425,7 @@ SKIP_NUM(p); if(p == old) return ERR; - *args = TO_ABS; + *args = TO_ABSY; args++; SKIP_SPACE(p); @@ -350,7 +433,7 @@ SKIP_NUM(p); if(p == old) return ERR; - *args = TO_ABS; + *args = TO_ABSX; args++; SKIP_SPACE(p); @@ -358,14 +441,14 @@ SKIP_NUM(p); if(p == old) return ERR; - *args = TO_ABS; + *args = TO_ABSY; args++; SKIP_SPACE(p); old = p; SKIP_NUM(p); if(p == old) return ERR; - *args = TO_ABS; + *args = TO_ABSX; args++; SKIP_SPACE(p); @@ -373,8 +456,10 @@ SKIP_NUM(p); if(p == old) return ERR; - *args = TO_ABS; + *args = TO_ABSY; args++; + + *cmds++ = toupper(cmd); } break; case 's': @@ -388,7 +473,7 @@ SKIP_NUM(p); if(p == old) break; - *args = TO_ABS; + *args = TO_ABSX; args++; SKIP_SPACE(p); @@ -396,7 +481,7 @@ SKIP_NUM(p); if(p == old) return ERR; - *args = TO_ABS; + *args = TO_ABSY; args++; SKIP_SPACE(p); @@ -404,7 +489,7 @@ SKIP_NUM(p); if(p == old) return ERR; - *args = TO_ABS; + *args = TO_ABSX; args++; SKIP_SPACE(p); @@ -412,8 +497,10 @@ SKIP_NUM(p); if(p == old) return ERR; - *args = TO_ABS; + *args = TO_ABSY; args++; + + *cmds++ = toupper(cmd); } break; case 'm': @@ -429,7 +516,7 @@ SKIP_NUM(p); if(p == old) break; - *args = TO_ABS; + *args = TO_ABSX; args++; SKIP_SPACE(p); @@ -437,8 +524,10 @@ SKIP_NUM(p); if(p == old) return ERR; - *args = TO_ABS; + *args = TO_ABSY; args++; + + *cmds++ = toupper(cmd); } break; case 'h': @@ -450,11 +539,14 @@ case 'A': case 'a': - + r = sh_path_arc_cmd_arg_fill(cmd, &cmds, (const char **)&p, &args); + if(r != OK) + return ERR; break; case 'z': case 'Z': + *cmds++ = toupper(cmd); break; default: return ERR; @@ -557,7 +649,7 @@ sh_path_t *path; int cmd_len; char *cmds, cmd; - co_aix *args; + const co_aix *args; co_aix x, y, x1, y1, x2, y2; int i; @@ -569,37 +661,21 @@ args = (co_aix *)(cmds + cmd_len); x = y = x1 = y1 = x2 = y2 = 0; for(i = 0; i < cmd_len; i++) { + /* All path commands and arguments are transformed + * to absoluted form. + */ cmd = *cmds++; switch(cmd) { - case 'm': - x = x + *args++; - y = y + *args++; - cairo_move_to(cr, x, y); - break; case 'M': x = *args++; y = *args++; cairo_move_to(cr, x, y); break; - case 'l': - x = x + *args++; - y = y + *args++; - cairo_line_to(cr, x, y); - break; case 'L': x = *args++; y = *args++; cairo_line_to(cr, x, y); break; - case 'c': - x1 = x + *args++; - y1 = y + *args++; - x2 = x + *args++; - y2 = y + *args++; - x = x + *args++; - y = y + *args++; - cairo_curve_to(cr, x1, y1, x2, y2, x, y); - break; case 'C': x1 = *args++; y1 = *args++; @@ -609,15 +685,6 @@ y = *args++; cairo_curve_to(cr, x1, y1, x2, y2, x, y); break; - case 's': - x1 = x + x - x2; - y1 = y + y - y2; - x2 = x + *args++; - y2 = y + *args++; - x = x + *args++; - y = y + *args++; - cairo_curve_to(cr, x1, y1, x2, y2, x, y); - break; case 'S': x1 = x + x - x2; y1 = y + y - y2; @@ -627,15 +694,6 @@ y = *args++; cairo_curve_to(cr, x1, y1, x2, y2, x, y); break; - case 'q': - x1 = x + *args++; - y1 = y + *args++; - x2 = x1; - y2 = y1; - x = x + *args++; - y = y + *args++; - cairo_curve_to(cr, x1, y1, x2, y2, x, y); - break; case 'Q': x1 = *args++; y1 = *args++; @@ -645,15 +703,6 @@ y = *args++; cairo_curve_to(cr, x1, y1, x2, y2, x, y); break; - case 't': - x1 = x + x - x2; - y1 = y + y - y2; - x2 = x1; - y2 = y1; - x = x + *args++; - y = y + *args++; - cairo_curve_to(cr, x1, y1, x2, y2, x, y); - break; case 'T': x1 = x + x - x2; y1 = y + y - y2; @@ -663,8 +712,10 @@ y = *args++; cairo_curve_to(cr, x1, y1, x2, y2, x, y); break; + case 'A': + sh_path_arc_path(cr, &args); + break; case 'Z': - case 'z': cairo_close_path(cr); break; case '\x0': @@ -686,7 +737,7 @@ sh_path_t *path; co_aix *args; - path = (sh_path_t *)sh_path_new("M 33 25l33 55C 33 87 44 22 55 99L33 77z"); + path = (sh_path_t *)sh_path_new("M 33 25l33 55c 33 87 44 22 55 99L33 77z"); CU_ASSERT(path != NULL); CU_ASSERT(path->cmd_len == ((5 + RESERVED_AIXS + 3) & ~0x3)); CU_ASSERT(path->arg_len == 12); @@ -698,12 +749,12 @@ CU_ASSERT(args[1] == 25); CU_ASSERT(args[2] == 66); CU_ASSERT(args[3] == 80); - CU_ASSERT(args[4] == 33); - CU_ASSERT(args[5] == 87); - CU_ASSERT(args[6] == 44); - CU_ASSERT(args[7] == 22); - CU_ASSERT(args[8] == 55); - CU_ASSERT(args[9] == 99); + CU_ASSERT(args[4] == 99); + CU_ASSERT(args[5] == 167); + CU_ASSERT(args[6] == 110); + CU_ASSERT(args[7] == 102); + CU_ASSERT(args[8] == 121); + CU_ASSERT(args[9] == 179); CU_ASSERT(args[10] == 33); CU_ASSERT(args[11] == 77); sh_path_free((shape_t *)path);