Files correlati : pdflib Ricompilazione Demo : [ ] Commento : Aggiornata pdflib.dll alla versione 7.0.4 git-svn-id: svn://10.65.10.50/trunk@18580 c028cbd2-c16b-5b4b-a496-9718f37d4682
411 lines
10 KiB
C
Executable File
411 lines
10 KiB
C
Executable File
/*---------------------------------------------------------------------------*
|
||
| PDFlib - A library for generating PDF on the fly |
|
||
+---------------------------------------------------------------------------+
|
||
| Copyright (c) 1997-2006 Thomas Merz and PDFlib GmbH. All rights reserved. |
|
||
+---------------------------------------------------------------------------+
|
||
| |
|
||
| This software is subject to the PDFlib license. It is NOT in the |
|
||
| public domain. Extended versions and commercial licenses are |
|
||
| available, please check http://www.pdflib.com. |
|
||
| |
|
||
*---------------------------------------------------------------------------*/
|
||
|
||
/* $Id: p_draw.c,v 1.4 2009-03-23 08:51:17 guy Exp $
|
||
*
|
||
* PDFlib drawing routines
|
||
*
|
||
*/
|
||
|
||
#include "p_intern.h"
|
||
#include "p_layer.h"
|
||
#include "p_tagged.h"
|
||
|
||
/* Path segment operators */
|
||
|
||
static void
|
||
pdf_begin_path(PDF *p)
|
||
{
|
||
if (PDF_GET_STATE(p) == pdf_state_path)
|
||
return;
|
||
|
||
|
||
|
||
|
||
pdf_end_text(p);
|
||
PDF_PUSH_STATE(p, "pdf_begin_path", pdf_state_path);
|
||
}
|
||
|
||
static void
|
||
pdf_end_path(PDF *p)
|
||
{
|
||
pdf_ppt *ppt = p->curr_ppt;
|
||
|
||
PDF_POP_STATE(p, "pdf_end_path");
|
||
|
||
ppt->gstate[ppt->sl].x = 0;
|
||
ppt->gstate[ppt->sl].y = 0;
|
||
}
|
||
|
||
/* ----------------- Basic functions for API functions --------------*/
|
||
|
||
void
|
||
pdf__moveto(PDF *p, pdc_scalar x, pdc_scalar y)
|
||
{
|
||
pdf_ppt *ppt = p->curr_ppt;
|
||
|
||
pdc_check_number(p->pdc, "x", x);
|
||
pdc_check_number(p->pdc, "y", y);
|
||
|
||
ppt->gstate[ppt->sl].startx = ppt->gstate[ppt->sl].x = x;
|
||
ppt->gstate[ppt->sl].starty = ppt->gstate[ppt->sl].y = y;
|
||
|
||
pdf_begin_path(p);
|
||
pdc_printf(p->out, "%f %f m\n", x, y);
|
||
}
|
||
|
||
void
|
||
pdf__rmoveto(PDF *p, pdc_scalar x, pdc_scalar y)
|
||
{
|
||
pdf_ppt *ppt = p->curr_ppt;
|
||
|
||
pdc_scalar x_0 = ppt->gstate[ppt->sl].x;
|
||
pdc_scalar y_0 = ppt->gstate[ppt->sl].y;
|
||
|
||
pdc_check_number(p->pdc, "x", x);
|
||
pdc_check_number(p->pdc, "y", y);
|
||
|
||
pdf__moveto(p, x_0 + x, y_0 + y);
|
||
}
|
||
|
||
void
|
||
pdf__lineto(PDF *p, pdc_scalar x, pdc_scalar y)
|
||
{
|
||
pdf_ppt *ppt = p->curr_ppt;
|
||
|
||
pdc_check_number(p->pdc, "x", x);
|
||
pdc_check_number(p->pdc, "y", y);
|
||
|
||
pdc_printf(p->out, "%f %f l\n", x, y);
|
||
|
||
ppt->gstate[ppt->sl].x = x;
|
||
ppt->gstate[ppt->sl].y = y;
|
||
}
|
||
|
||
void
|
||
pdf__rlineto(PDF *p, pdc_scalar x, pdc_scalar y)
|
||
{
|
||
pdf_ppt *ppt = p->curr_ppt;
|
||
|
||
pdc_scalar x_0 = ppt->gstate[ppt->sl].x;
|
||
pdc_scalar y_0 = ppt->gstate[ppt->sl].y;
|
||
|
||
pdc_check_number(p->pdc, "x", x);
|
||
pdc_check_number(p->pdc, "y", y);
|
||
|
||
pdf__lineto(p, x_0 + x, y_0 + y);
|
||
}
|
||
|
||
void
|
||
pdf__curveto(PDF *p,
|
||
pdc_scalar x_1, pdc_scalar y_1,
|
||
pdc_scalar x_2, pdc_scalar y_2,
|
||
pdc_scalar x_3, pdc_scalar y_3)
|
||
{
|
||
pdf_ppt *ppt = p->curr_ppt;
|
||
|
||
pdc_check_number(p->pdc, "x_1", x_1);
|
||
pdc_check_number(p->pdc, "y_1", y_1);
|
||
pdc_check_number(p->pdc, "x_2", x_2);
|
||
pdc_check_number(p->pdc, "y_2", y_2);
|
||
pdc_check_number(p->pdc, "x_3", x_3);
|
||
pdc_check_number(p->pdc, "y_3", y_3);
|
||
|
||
/* second c.p. coincides with final point */
|
||
if (fabs(x_2 - x_3) < PDC_FLOAT_PREC &&
|
||
fabs(y_2 - y_3) < PDC_FLOAT_PREC)
|
||
pdc_printf(p->out, "%f %f %f %f y\n", x_1, y_1, x_3, y_3);
|
||
|
||
/* general case with four distinct points */
|
||
else
|
||
pdc_printf(p->out, "%f %f %f %f %f %f c\n",
|
||
x_1, y_1, x_2, y_2, x_3, y_3);
|
||
|
||
ppt->gstate[ppt->sl].x = x_3;
|
||
ppt->gstate[ppt->sl].y = y_3;
|
||
}
|
||
|
||
void
|
||
pdf__rcurveto(PDF *p,
|
||
pdc_scalar x_1, pdc_scalar y_1,
|
||
pdc_scalar x_2, pdc_scalar y_2,
|
||
pdc_scalar x_3, pdc_scalar y_3)
|
||
{
|
||
pdf_ppt *ppt = p->curr_ppt;
|
||
|
||
pdc_scalar x_0 = ppt->gstate[ppt->sl].x;
|
||
pdc_scalar y_0 = ppt->gstate[ppt->sl].y;
|
||
|
||
pdc_check_number(p->pdc, "x_1", x_1);
|
||
pdc_check_number(p->pdc, "y_1", y_1);
|
||
pdc_check_number(p->pdc, "x_2", x_2);
|
||
pdc_check_number(p->pdc, "y_2", y_2);
|
||
pdc_check_number(p->pdc, "x_3", x_3);
|
||
pdc_check_number(p->pdc, "y_3", y_3);
|
||
|
||
pdf__curveto(p, x_0 + x_1, y_0 + y_1,
|
||
x_0 + x_2, y_0 + y_2,
|
||
x_0 + x_3, y_0 + y_3);
|
||
}
|
||
|
||
void
|
||
pdf_rrcurveto(PDF *p,
|
||
pdc_scalar x_1, pdc_scalar y_1,
|
||
pdc_scalar x_2, pdc_scalar y_2,
|
||
pdc_scalar x_3, pdc_scalar y_3)
|
||
{
|
||
pdf__rcurveto(p, x_1, y_1,
|
||
x_1 + x_2, y_1 + y_2,
|
||
x_1 + x_2 + x_3, y_1 + y_2 + y_3);
|
||
}
|
||
|
||
void
|
||
pdf_hvcurveto(PDF *p, pdc_scalar x_1, pdc_scalar x_2,
|
||
pdc_scalar y_2, pdc_scalar y_3)
|
||
{
|
||
pdf_rrcurveto(p, x_1, 0, x_2, y_2, 0, y_3);
|
||
}
|
||
|
||
void
|
||
pdf_vhcurveto(PDF *p, pdc_scalar y_1, pdc_scalar x_2,
|
||
pdc_scalar y_2, pdc_scalar x_3)
|
||
{
|
||
pdf_rrcurveto(p, 0, y_1, x_2, y_2, x_3, 0);
|
||
}
|
||
|
||
void
|
||
pdf__rect(PDF *p, pdc_scalar x, pdc_scalar y,
|
||
pdc_scalar width, pdc_scalar height)
|
||
{
|
||
pdf_ppt *ppt = p->curr_ppt;
|
||
|
||
pdc_check_number(p->pdc, "x", x);
|
||
pdc_check_number(p->pdc, "y", y);
|
||
pdc_check_number(p->pdc, "width", width);
|
||
pdc_check_number(p->pdc, "height", height);
|
||
|
||
ppt->gstate[ppt->sl].startx = ppt->gstate[ppt->sl].x = x;
|
||
ppt->gstate[ppt->sl].starty = ppt->gstate[ppt->sl].y = y;
|
||
|
||
pdf_begin_path(p);
|
||
pdc_printf(p->out, "%f %f %f %f re\n", x, y, width, p->ydirection * height);
|
||
}
|
||
|
||
/* 4/3 * (1-cos 45<34>)/sin 45<34> = 4/3 * sqrt(2) - 1 */
|
||
#define ARC_MAGIC (0.552284749)
|
||
|
||
static void
|
||
pdf_short_arc(PDF *p, pdc_scalar x, pdc_scalar y, pdc_scalar r,
|
||
pdc_scalar alpha, pdc_scalar beta)
|
||
{
|
||
pdc_scalar bcp;
|
||
pdc_scalar cos_alpha, cos_beta, sin_alpha, sin_beta;
|
||
|
||
alpha = alpha * PDC_DEG2RAD;
|
||
beta = beta * PDC_DEG2RAD;
|
||
|
||
/* This formula yields ARC_MAGIC for alpha == 0, beta == 90 degrees */
|
||
bcp = (4.0/3 * (1 - cos((beta - alpha)/2)) / sin((beta - alpha)/2));
|
||
|
||
sin_alpha = sin(alpha);
|
||
sin_beta = sin(beta);
|
||
cos_alpha = cos(alpha);
|
||
cos_beta = cos(beta);
|
||
|
||
pdf__curveto(p,
|
||
x + r * (cos_alpha - bcp * sin_alpha), /* p1 */
|
||
y + r * (sin_alpha + bcp * cos_alpha),
|
||
x + r * (cos_beta + bcp * sin_beta), /* p2 */
|
||
y + r * (sin_beta - bcp * cos_beta),
|
||
x + r * cos_beta, /* p3 */
|
||
y + r * sin_beta);
|
||
}
|
||
|
||
static void
|
||
pdf_orient_arc(PDF *p, pdc_scalar x, pdc_scalar y, pdc_scalar r,
|
||
pdc_scalar alpha, pdc_scalar beta, pdc_scalar orient)
|
||
{
|
||
pdf_ppt *ppt = p->curr_ppt;
|
||
pdc_scalar rad_a = alpha * PDC_DEG2RAD;
|
||
pdc_scalar startx = (x + r * cos(rad_a));
|
||
pdc_scalar starty = (y + r * sin(rad_a));
|
||
|
||
if (PDF_GET_STATE(p) != pdf_state_path)
|
||
{
|
||
pdf__moveto(p, startx, starty); /* this enters pdf_state_path */
|
||
}
|
||
else if ((ppt->gstate[ppt->sl].x != startx
|
||
|| ppt->gstate[ppt->sl].y != starty))
|
||
{
|
||
pdf__lineto(p, startx, starty);
|
||
}
|
||
|
||
if (orient > 0)
|
||
{
|
||
while (beta < alpha)
|
||
beta += 360;
|
||
|
||
if (alpha == beta)
|
||
return;
|
||
|
||
while (beta - alpha > 90)
|
||
{
|
||
pdf_short_arc(p, x, y, r, alpha, alpha + 90);
|
||
alpha += 90;
|
||
}
|
||
}
|
||
else
|
||
{
|
||
while (alpha < beta)
|
||
alpha += 360;
|
||
|
||
if (alpha == beta)
|
||
return;
|
||
|
||
while (alpha - beta > 90)
|
||
{
|
||
pdf_short_arc(p, x, y, r, alpha, alpha - 90);
|
||
alpha -= 90;
|
||
}
|
||
}
|
||
|
||
if (alpha != beta)
|
||
pdf_short_arc(p, x, y, r, alpha, beta);
|
||
}
|
||
|
||
void
|
||
pdf__arc(PDF *p, pdc_scalar x, pdc_scalar y, pdc_scalar r,
|
||
pdc_scalar alpha, pdc_scalar beta)
|
||
{
|
||
pdc_check_number(p->pdc, "x", x);
|
||
pdc_check_number(p->pdc, "y", y);
|
||
pdc_check_number_limits(p->pdc, "r", r, PDC_FLOAT_PREC, PDC_FLOAT_MAX);
|
||
pdc_check_number(p->pdc, "alpha", alpha);
|
||
pdc_check_number(p->pdc, "beta", beta);
|
||
|
||
pdf_orient_arc(p, x, y, r,
|
||
p->ydirection * alpha, p->ydirection * beta, p->ydirection);
|
||
}
|
||
|
||
void
|
||
pdf__arcn(PDF *p, pdc_scalar x, pdc_scalar y, pdc_scalar r,
|
||
pdc_scalar alpha, pdc_scalar beta)
|
||
{
|
||
pdc_check_number(p->pdc, "x", x);
|
||
pdc_check_number(p->pdc, "y", y);
|
||
pdc_check_number_limits(p->pdc, "r", r, PDC_FLOAT_PREC, PDC_FLOAT_MAX);
|
||
pdc_check_number(p->pdc, "alpha", alpha);
|
||
pdc_check_number(p->pdc, "beta", beta);
|
||
|
||
pdf_orient_arc(p, x, y, r,
|
||
p->ydirection * alpha, p->ydirection * beta, -p->ydirection);
|
||
}
|
||
|
||
void
|
||
pdf__circle(PDF *p, pdc_scalar x, pdc_scalar y, pdc_scalar r)
|
||
{
|
||
pdc_check_number(p->pdc, "x", x);
|
||
pdc_check_number(p->pdc, "y", y);
|
||
pdc_check_number_limits(p->pdc, "r", r, PDC_FLOAT_PREC, PDC_FLOAT_MAX);
|
||
|
||
/*
|
||
* pdf_begin_path() not required since we descend to other
|
||
* path segment functions.
|
||
*/
|
||
|
||
/* draw four Bezier curves to approximate a circle */
|
||
pdf__moveto(p, x + r, y);
|
||
pdf__curveto(p, x + r, y + r*ARC_MAGIC, x + r*ARC_MAGIC, y + r, x, y + r);
|
||
pdf__curveto(p, x - r*ARC_MAGIC, y + r, x - r, y + r*ARC_MAGIC, x - r, y);
|
||
pdf__curveto(p, x - r, y - r*ARC_MAGIC, x - r*ARC_MAGIC, y - r, x, y - r);
|
||
pdf__curveto(p, x + r*ARC_MAGIC, y - r, x + r, y - r*ARC_MAGIC, x + r, y);
|
||
|
||
pdf__closepath(p);
|
||
}
|
||
|
||
void
|
||
pdf__closepath(PDF *p)
|
||
{
|
||
pdf_ppt *ppt = p->curr_ppt;
|
||
|
||
pdc_puts(p->out, "h\n");
|
||
|
||
ppt->gstate[ppt->sl].x = ppt->gstate[ppt->sl].startx;
|
||
ppt->gstate[ppt->sl].y = ppt->gstate[ppt->sl].starty;
|
||
}
|
||
|
||
void
|
||
pdf__endpath(PDF *p)
|
||
{
|
||
pdc_puts(p->out, "n\n");
|
||
pdf_end_path(p);
|
||
}
|
||
|
||
void
|
||
pdf__stroke(PDF *p)
|
||
{
|
||
pdc_puts(p->out, "S\n");
|
||
pdf_end_path(p);
|
||
}
|
||
|
||
void
|
||
pdf__closepath_stroke(PDF *p)
|
||
{
|
||
pdc_puts(p->out, "s\n");
|
||
pdf_end_path(p);
|
||
}
|
||
|
||
void
|
||
pdf__fill(PDF *p)
|
||
{
|
||
if (p->curr_ppt->fillrule == pdf_fill_winding)
|
||
pdc_puts(p->out, "f\n");
|
||
else if (p->curr_ppt->fillrule == pdf_fill_evenodd)
|
||
pdc_puts(p->out, "f*\n");
|
||
|
||
pdf_end_path(p);
|
||
}
|
||
|
||
void
|
||
pdf__fill_stroke(PDF *p)
|
||
{
|
||
if (p->curr_ppt->fillrule == pdf_fill_winding)
|
||
pdc_puts(p->out, "B\n");
|
||
else if (p->curr_ppt->fillrule == pdf_fill_evenodd)
|
||
pdc_puts(p->out, "B*\n");
|
||
|
||
pdf_end_path(p);
|
||
}
|
||
|
||
void
|
||
pdf__closepath_fill_stroke(PDF *p)
|
||
{
|
||
if (p->curr_ppt->fillrule == pdf_fill_winding)
|
||
pdc_puts(p->out, "b\n");
|
||
else if (p->curr_ppt->fillrule == pdf_fill_evenodd)
|
||
pdc_puts(p->out, "b*\n");
|
||
|
||
pdf_end_path(p);
|
||
}
|
||
|
||
void
|
||
pdf__clip(PDF *p)
|
||
{
|
||
if (p->curr_ppt->fillrule == pdf_fill_winding)
|
||
pdc_puts(p->out, "W n\n");
|
||
else if (p->curr_ppt->fillrule == pdf_fill_evenodd)
|
||
pdc_puts(p->out, "W* n\n");
|
||
|
||
pdf_end_path(p);
|
||
}
|
||
|