home *** CD-ROM | disk | FTP | other *** search
/ Black Box 4 / BlackBox.cdr / progc / gs24src.arj / GXPATH.C < prev    next >
C/C++ Source or Header  |  1991-11-25  |  13KB  |  417 lines

  1. /* Copyright (C) 1989, 1990, 1991 Aladdin Enterprises.  All rights reserved.
  2.    Distributed by Free Software Foundation, Inc.
  3.  
  4. This file is part of Ghostscript.
  5.  
  6. Ghostscript is distributed in the hope that it will be useful, but
  7. WITHOUT ANY WARRANTY.  No author or distributor accepts responsibility
  8. to anyone for the consequences of using it or for whether it serves any
  9. particular purpose or works at all, unless he says so in writing.  Refer
  10. to the Ghostscript General Public License for full details.
  11.  
  12. Everyone is granted permission to copy, modify and redistribute
  13. Ghostscript, but only under the conditions described in the Ghostscript
  14. General Public License.  A copy of this license is supposed to have been
  15. given to you along with Ghostscript so you can know your rights and
  16. responsibilities.  It should be in a file named COPYING.  Among other
  17. things, the copyright notice and this notice must be preserved on all
  18. copies.  */
  19.  
  20. /* gxpath.c */
  21. /* Private path routines for Ghostscript library */
  22. #include "gx.h"
  23. #include "gserrors.h"
  24. #include "gxfixed.h"
  25. #include "gzpath.h"
  26.  
  27. /* These routines all assume that all points are */
  28. /* already in device coordinates, and in fixed representation. */
  29. /* As usual, they return either 0 or a (negative) error code. */
  30.  
  31. /* Forward references */
  32. private subpath *path_alloc_copy(P1(gx_path *));
  33. private int gx_path_new_subpath(P1(gx_path *));
  34. #ifdef DEBUG
  35. void gx_print_segment(P1(segment *));
  36. #endif
  37.  
  38. /* ------ Initialize/free paths ------ */
  39.  
  40. /* Initialize a path */
  41. void
  42. gx_path_init(register gx_path *ppath, gs_memory_procs *pprocs)
  43. {    ppath->memory_procs = *pprocs;
  44.     ppath->box_last = 0;
  45.     ppath->position_valid = 0;
  46.     ppath->first_subpath = ppath->current_subpath = 0;
  47.     ppath->subpath_count = 0;
  48.     ppath->curve_count = 0;
  49.     ppath->subpath_open = 0;
  50.     ppath->shares_segments = 0;
  51. }
  52.  
  53. /* Release the contents of a path.  We do this in reverse order */
  54. /* so as to maximize LIFO allocator behavior. */
  55. void
  56. gx_path_release(gx_path *ppath)
  57. {    segment *pseg;
  58.     if ( ppath->first_subpath == 0 ) return;    /* empty path */
  59.     if ( ppath->shares_segments ) return;    /* segments are shared */
  60.     pseg = (segment *)ppath->current_subpath->last;
  61.     while ( pseg )
  62.        {    segment *prev = pseg->prev;
  63.         static uint sizes[] = { segment_type_sizes };
  64. #ifdef DEBUG
  65. if ( gs_debug['p'] )
  66.         dprintf("[p]release"), gx_print_segment(pseg);
  67. #endif
  68.         (*ppath->memory_procs.free)((char *)pseg, 1, sizes[(int)pseg->type], "gx_path_release");
  69.         pseg = prev;
  70.        }
  71.     ppath->first_subpath = 0;    /* prevent re-release */
  72. }
  73.  
  74. /* Mark a path as shared */
  75. void
  76. gx_path_share(gx_path *ppath)
  77. {    if ( ppath->first_subpath ) ppath->shares_segments = 1;
  78. }
  79.  
  80. /* ------ Incremental path building ------ */
  81.  
  82. /* Macro for opening the current subpath. */
  83. /* ppath points to the path; psub has been set to ppath->current_subpath. */
  84. #define path_open()\
  85.     if ( !ppath->subpath_open )\
  86.        {    int code;\
  87.         if ( !ppath->position_valid )\
  88.           return_error(gs_error_nocurrentpoint);\
  89.         code = gx_path_new_subpath(ppath);\
  90.         if ( code < 0 ) return code;\
  91.         psub = ppath->current_subpath;\
  92.        }
  93.  
  94. /* Macros for allocating path segments. */
  95. /* Note that they assume that ppath points to the path, */
  96. /* and that psub points to the current subpath. */
  97. /* We have to split the macro into two because of limitations */
  98. /* on the size of a single statement (sigh). */
  99. #ifdef DEBUG
  100. #define p_alloc(pseg,size)\
  101.   if ( gs_debug['A'] ) dprintf2("[p]%lx<%u>\n", (ulong)pseg, size)
  102. #else
  103. #define p_alloc(pseg,size) 0
  104. #endif
  105. #define path_unshare()\
  106.   if(ppath->shares_segments)\
  107.     if(!(psub = path_alloc_copy(ppath)))return_error(gs_error_limitcheck)
  108. #define path_alloc_segment(pseg,ctype,stype,cname)\
  109.   path_unshare();\
  110.   if( !(pseg = (ctype *)(*ppath->memory_procs.alloc)(1, sizeof(ctype), cname)) )\
  111.     return_error(gs_error_limitcheck);\
  112.   p_alloc((char *)pseg, sizeof(ctype));\
  113.   pseg->type = stype, pseg->next = 0
  114. #define path_alloc_link(pseg)\
  115.   { segment *prev = psub->last;\
  116.     prev->next = (segment *)pseg;\
  117.     pseg->prev = prev;\
  118.     psub->last = (segment *)pseg;\
  119.   }
  120.  
  121. /* Open a new subpath */
  122. private int
  123. gx_path_new_subpath(gx_path *ppath)
  124. {    subpath *psub = ppath->current_subpath;
  125.     register subpath *spp;
  126.     path_alloc_segment(spp, subpath, s_start, "gx_path_new_subpath");
  127.     spp->last = (segment *)spp;
  128.     spp->curve_count = 0;
  129.     spp->closed = 0;
  130.     spp->pt = ppath->position;
  131.     ppath->subpath_open = 1;
  132.     if ( !psub )            /* first subpath */
  133.        {    ppath->first_subpath = spp;
  134.         spp->prev = 0;
  135.        }
  136.     else
  137.        {    segment *prev = psub->last;
  138.         prev->next = (segment *)spp;
  139.         spp->prev = prev;
  140.        }
  141.     ppath->current_subpath = spp;
  142.     ppath->subpath_count++;
  143. #ifdef DEBUG
  144. if ( gs_debug['p'] )
  145.     dprintf("[p]"), gx_print_segment((segment *)spp);
  146. #endif
  147.     return 0;
  148. }
  149.  
  150. /* Add a point to the current path (moveto). */
  151. int
  152. gx_path_add_point(register gx_path *ppath, fixed x, fixed y)
  153. {    ppath->subpath_open = 0;
  154.     ppath->position_valid = 1;
  155.     ppath->position.x = x;
  156.     ppath->position.y = y;
  157.     return 0;
  158. }
  159.  
  160. /* Add a relative point to the current path (rmoveto). */
  161. int
  162. gx_path_add_relative_point(register gx_path *ppath, fixed dx, fixed dy)
  163. {    if ( !ppath->position_valid )
  164.       return_error(gs_error_nocurrentpoint);
  165.     ppath->subpath_open = 0;
  166.     ppath->position.x += dx;
  167.     ppath->position.y += dy;
  168.     return 0;
  169. }
  170.  
  171. /* Set the segment point and the current point in the path. */
  172. /* Assumes ppath points to the path. */
  173. #define path_set_point(pseg, fx, fy)\
  174.     (pseg)->pt.x = ppath->position.x = (fx),\
  175.     (pseg)->pt.y = ppath->position.y = (fy)
  176.  
  177. /* Add a line to the current path (lineto). */
  178. int
  179. gx_path_add_line(gx_path *ppath, fixed x, fixed y)
  180. {    subpath *psub = ppath->current_subpath;
  181.     register line_segment *lp;
  182.     path_open();
  183.     path_alloc_segment(lp, line_segment, s_line, "gx_path_add_line");
  184.     path_alloc_link(lp);
  185.     path_set_point(lp, x, y);
  186. #ifdef DEBUG
  187. if ( gs_debug['p'] )
  188.     dprintf("[p]"), gx_print_segment((segment *)lp);
  189. #endif
  190.     return 0;
  191. }
  192.  
  193. /* Add a rectangle to the current path. */
  194. /* This is a special case of adding a parallelogram. */
  195. int
  196. gx_path_add_rectangle(gx_path *ppath, fixed x0, fixed y0, fixed x1, fixed y1)
  197. {    return gx_path_add_pgram(ppath, x0, y0, x0, y1, x1, y1);
  198. }
  199.  
  200. /* Add a parallelogram to the current path. */
  201. /* This is equivalent to an add_point, three add_lines, */
  202. /* and a close_subpath. */
  203. int
  204. gx_path_add_pgram(gx_path *ppath,
  205.   fixed x0, fixed y0, fixed x1, fixed y1, fixed x2, fixed y2)
  206. {    int code;
  207.      if (    (code = gx_path_add_point(ppath, x0, y0)) < 0 ||
  208.         (code = gx_path_add_line(ppath, x1, y1)) < 0 ||
  209.         (code = gx_path_add_line(ppath, x2, y2)) < 0 ||
  210.         (code = gx_path_add_line(ppath, x0 + x2 - x1, y0 + y2 - y1)) < 0 ||
  211.         (code = gx_path_close_subpath(ppath)) < 0
  212.        ) return code;
  213.     return 0;
  214. }
  215.  
  216. /* Add a curve to the current path (curveto). */
  217. int
  218. gx_path_add_curve(gx_path *ppath,
  219.   fixed x1, fixed y1, fixed x2, fixed y2, fixed x3, fixed y3)
  220. {    subpath *psub = ppath->current_subpath;
  221.     register curve_segment *lp;
  222.     path_open();
  223.     path_alloc_segment(lp, curve_segment, s_curve, "gx_path_add_curve");
  224.     path_alloc_link(lp);
  225.     lp->p1.x = x1;
  226.     lp->p1.y = y1;
  227.     lp->p2.x = x2;
  228.     lp->p2.y = y2;
  229.     path_set_point(lp, x3, y3);
  230.     psub->curve_count++;
  231.     ppath->curve_count++;
  232. #ifdef DEBUG
  233. if ( gs_debug['p'] )
  234.     dprintf("[p]"), gx_print_segment((segment *)lp);
  235. #endif
  236.     return 0;
  237. }
  238.  
  239. /*
  240.  * Add an approximation of an arc to the current path.
  241.  * Parameters are the initial and final points of the arc,
  242.  * and the point at which the extended tangents meet.
  243.  * We assume that the arc is less than a semicircle.
  244.  * The arc may go either clockwise or counterclockwise.
  245.  * The approximation is a very simple one: a single curve
  246.  * whose other two control points are a fraction F of the way
  247.  * to the intersection of the tangents, where
  248.  *    F = (4/3)(1 / (1 + sqrt(1+(d/r)^2)))
  249.  * where r is the radius and d is the distance from either tangent
  250.  * point to the intersection of the tangents.  This produces
  251.  * a curve whose center point, as well as its ends, lies on
  252.  * the desired arc.
  253.  *
  254.  * Because F has to be computed in user space, we let the client
  255.  * compute it and pass it in as an argument.
  256.  */
  257. int
  258. gx_path_add_arc(gx_path *ppath,
  259.   fixed x0, fixed y0, fixed x3, fixed y3, fixed xt, fixed yt, floatp fraction)
  260. {    return gx_path_add_curve(ppath,
  261.             x0 + (fixed)((xt - x0) * fraction),
  262.             y0 + (fixed)((yt - y0) * fraction),
  263.             x3 + (fixed)((xt - x3) * fraction),
  264.             y3 + (fixed)((yt - y3) * fraction),
  265.             x3, y3);
  266. }
  267.  
  268. /* Append a path to another path, and reset the first path. */
  269. /* Currently this is only used to append a path to its parent */
  270. /* (the path in the previous graphics context). */
  271. int
  272. gx_path_add_path(gx_path *ppath, gx_path *ppfrom)
  273. {    subpath *psub = ppath->current_subpath;
  274.     path_unshare();
  275.     if ( ppfrom->first_subpath )    /* i.e. ppfrom not empty */
  276.        {    if ( ppath->first_subpath )    /* i.e. ppath not empty */
  277.            {    segment *pseg = psub->last;
  278.             segment *pfseg = (segment *)ppfrom->first_subpath;
  279.             pseg->next = pfseg;
  280.             pfseg->prev = pseg;
  281.            }
  282.         else
  283.             ppath->first_subpath = ppfrom->first_subpath;
  284.         ppath->current_subpath = ppfrom->current_subpath;
  285.        }
  286.     /* Transfer the remaining state. */
  287.     ppath->subpath_count += ppfrom->subpath_count;
  288.     ppath->curve_count += ppfrom->curve_count;
  289.     ppath->position = ppfrom->position;
  290.     ppath->position_valid = ppfrom->position_valid;
  291.     ppath->subpath_open = ppfrom->subpath_open;
  292.     /* Reset the source path. */
  293.     ppfrom->first_subpath = 0;
  294.     return 0;
  295. }
  296.  
  297. /* Close the current subpath. */
  298. int
  299. gx_path_close_subpath(gx_path *ppath)
  300. {    subpath *psub = ppath->current_subpath;
  301.     register line_close_segment *lp;
  302.     if ( !ppath->subpath_open ) return 0;
  303.     path_alloc_segment(lp, line_close_segment, s_line_close,
  304.                "gx_path_close_subpath");
  305.     path_alloc_link(lp);
  306.     path_set_point(lp, psub->pt.x, psub->pt.y);
  307.     lp->sub = psub;
  308.     psub->closed = 1;
  309.     ppath->subpath_open = 0;
  310. #ifdef DEBUG
  311. if ( gs_debug['p'] )
  312.     if ( lp != 0 )
  313.       dprintf("[p]"), gx_print_segment((segment *)lp);
  314. #endif
  315.     return 0;
  316. }
  317.  
  318. /* ------ Internal routines ------ */
  319.  
  320. /* Copy the current path, because it was shared. */
  321. /* Return a pointer to the current subpath, or 0. */
  322. private subpath *
  323. path_alloc_copy(gx_path *ppath)
  324. {    gx_path path_new;
  325.     int code;
  326.     code = gx_path_copy(ppath, &path_new);
  327.     if ( code < 0 ) return 0;
  328.     *ppath = path_new;
  329.     ppath->shares_segments = 0;
  330.     return ppath->current_subpath;
  331. }
  332.  
  333. /* ------ Debugging printout ------ */
  334.  
  335. #ifdef DEBUG
  336.  
  337. /* Print out a path with a label */
  338. void
  339. gx_dump_path(gx_path *ppath, char *tag)
  340. {    dprintf2("[p]Path %lx %s:\n", (ulong)ppath, tag);
  341.     gx_path_print(ppath);
  342. }
  343.  
  344. /* Print a path */
  345. void
  346. gx_path_print(gx_path *ppath)
  347. {    segment *pseg = (segment *)ppath->first_subpath;
  348.     dprintf4("   subpaths=%d, curves=%d, point=(%f,%f)\n",
  349.          ppath->subpath_count, ppath->curve_count,
  350.          fixed2float(ppath->position.x),
  351.          fixed2float(ppath->position.y));
  352.     dprintf5("   box=(%f,%f),(%f,%f) last=%lx\n",
  353.          fixed2float(ppath->bbox.p.x), fixed2float(ppath->bbox.p.y),
  354.          fixed2float(ppath->bbox.q.x), fixed2float(ppath->bbox.q.y),
  355.          (ulong)ppath->box_last);
  356.     while ( pseg )
  357.        {    gx_print_segment(pseg);
  358.         pseg = pseg->next;
  359.        }
  360. }
  361. void
  362. gx_print_segment(segment *pseg)
  363. {    char out[80];
  364.     sprintf(out, "   %lx<%lx,%lx>: %%s (%6g,%6g) ",
  365.         (ulong)pseg, (ulong)pseg->prev, (ulong)pseg->next,
  366.         fixed2float(pseg->pt.x), fixed2float(pseg->pt.y));
  367.     switch ( pseg->type )
  368.        {
  369.     case s_start:
  370. #define psub ((subpath *)pseg)
  371.         dprintf1(out, "start");
  372.         dprintf2("#curves=%d last=%lx",
  373.              psub->curve_count, (ulong)psub->last);
  374. #undef psub
  375.         break;
  376.     case s_curve:
  377.         dprintf1(out, "curve");
  378. #define pcur ((curve_segment *)pseg)
  379.         dprintf4("\n\tp1=(%f,%f) p2=(%f,%f)",
  380.              fixed2float(pcur->p1.x), fixed2float(pcur->p1.y),
  381.              fixed2float(pcur->p2.x), fixed2float(pcur->p2.y));
  382. #undef pcur
  383.         break;
  384.     case s_line:
  385.         dprintf1(out, "line");
  386.         break;
  387.     case s_line_close:
  388. #define plc ((line_close_segment *)pseg)
  389.         dprintf1(out, "close");
  390.         dprintf1(" %lx", (ulong)(plc->sub));
  391. #undef plc
  392.         break;
  393.     default:
  394.        {    char t[20];
  395.         sprintf(t, "type 0x%x", pseg->type);
  396.         dprintf1(out, t);
  397.        }
  398.        }
  399.     dputc('\n');
  400. }
  401.  
  402. /* Print a clipping path */
  403. void
  404. gx_cpath_print(gx_clip_path *pcpath)
  405. {    if ( pcpath->segments_valid )
  406.         gx_path_print(&pcpath->path);
  407.     else
  408.         dputs("   (segments not valid)\n");
  409.     dprintf5("   cbox=(%f,%f),(%f,%f) count=%d\n",
  410.          fixed2float(pcpath->cbox.p.x), fixed2float(pcpath->cbox.p.y),
  411.          fixed2float(pcpath->cbox.q.x), fixed2float(pcpath->cbox.q.y),
  412.          pcpath->list.count);
  413.     /****** SHOULD PRINT CLIP LIST ******/
  414. }
  415.  
  416. #endif                    /* DEBUG */
  417.