123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459 |
- #include "sys-defines.h"
- #include "extern.h"
- static void write_svg_transform (plOutbuf *outbuf, const double m[6]);
- bool
- _pl_s_end_page (S___(Plotter *_plotter))
- {
- plOutbuf *svg_header, *svg_trailer;
-
-
- if (_plotter->data->page_number != 1)
- return true;
-
- svg_header = _new_outbuf ();
-
-
- sprintf (svg_header->point, "\
- <?xml version=\"1.0\" encoding=\"ISO-8859-1\" standalone=\"no\"?>\n\
- <!DOCTYPE svg PUBLIC \"-//W3C//DTD SVG 1.1//EN\" \"http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd\">\n");
- _update_buffer (svg_header);
-
- if (_plotter->data->page_data->metric)
- sprintf (svg_header->point,
- "<svg version=\"1.1\" baseProfile=\"full\" id=\"body\" width=\"%.5gcm\" height=\"%.5gcm\" ",
- 2.54 * FABS(_plotter->data->viewport_xsize),
- 2.54 * FABS(_plotter->data->viewport_ysize));
- else
- sprintf (svg_header->point,
- "<svg version=\"1.1\" baseProfile=\"full\" id=\"body\" width=\"%.5gin\" height=\"%.5gin\" ",
- FABS(_plotter->data->viewport_xsize),
- FABS(_plotter->data->viewport_ysize));
- _update_buffer (svg_header);
- sprintf (svg_header->point,
- "%s %s %s %s %s>\n",
- "viewBox=\"0 0 1 1\"",
- "preserveAspectRatio=\"none\"",
-
- "xmlns=\"http://www.w3.org/2000/svg\"",
-
- "xmlns:xlink=\"http://www.w3.org/1999/xlink\"",
- "xmlns:ev=\"http://www.w3.org/2001/xml-events\"");
- _update_buffer (svg_header);
- sprintf (svg_header->point, "<title>SVG drawing</title>\n");
- _update_buffer (svg_header);
- sprintf (svg_header->point, "<desc>This was produced by version %s of GNU libplot, a free library for exporting 2-D vector graphics.</desc>\n",
- PL_LIBPLOT_VER_STRING);
- _update_buffer (svg_header);
- if (_plotter->s_bgcolor_suppressed == false)
-
- {
- char color_buf[8];
- sprintf (svg_header->point,
- "<rect id=\"background\" x=\"0\" y=\"0\" width=\"1\" height=\"1\" stroke=\"none\" fill=\"%s\"/>\n",
- _libplot_color_to_svg_color (_plotter->s_bgcolor, color_buf));
- _update_buffer (svg_header);
- }
-
- sprintf (svg_header->point, "<g id=\"content\" ");
- _update_buffer (svg_header);
-
- if (_plotter->s_matrix_is_unknown == false
- && _plotter->s_matrix_is_bogus == false)
-
- {
- double product[6];
- _matrix_product (_plotter->s_matrix, _plotter->data->m_ndc_to_device,
- product);
- write_svg_transform (svg_header, product);
- }
-
- sprintf (svg_header->point, "xml:space=\"preserve\" ");
- _update_buffer (svg_header);
-
- sprintf (svg_header->point, "stroke=\"%s\" ",
- "black");
- _update_buffer (svg_header);
- sprintf (svg_header->point, "stroke-linecap=\"%s\" ",
- "butt");
- _update_buffer (svg_header);
- sprintf (svg_header->point, "stroke-linejoin=\"%s\" ",
- "miter");
- _update_buffer (svg_header);
- sprintf (svg_header->point, "stroke-miterlimit=\"%.5g\" ",
- PL_DEFAULT_MITER_LIMIT);
- _update_buffer (svg_header);
- sprintf (svg_header->point, "stroke-dasharray=\"%s\" ",
- "none");
- _update_buffer (svg_header);
-
- sprintf (svg_header->point, "stroke-dashoffset=\"%.5g\" ",
- 0.0);
- _update_buffer (svg_header);
- sprintf (svg_header->point, "stroke-opacity=\"%.5g\" ",
- 1.0);
- _update_buffer (svg_header);
- sprintf (svg_header->point, "fill=\"%s\" ",
- "none");
- _update_buffer (svg_header);
- sprintf (svg_header->point, "fill-rule=\"%s\" ",
- "evenodd");
- _update_buffer (svg_header);
- sprintf (svg_header->point, "fill-opacity=\"%.5g\" ",
- 1.0);
- _update_buffer (svg_header);
- sprintf (svg_header->point, "font-style=\"%s\" ",
- "normal");
- _update_buffer (svg_header);
- sprintf (svg_header->point, "font-variant=\"%s\" ",
- "normal");
- _update_buffer (svg_header);
- sprintf (svg_header->point, "font-weight=\"%s\" ",
- "normal");
- _update_buffer (svg_header);
- sprintf (svg_header->point, "font-stretch=\"%s\" ",
- "normal");
- _update_buffer (svg_header);
- sprintf (svg_header->point, "font-size-adjust=\"%s\" ",
- "none");
- _update_buffer (svg_header);
- sprintf (svg_header->point, "letter-spacing=\"%s\" ",
- "normal");
- _update_buffer (svg_header);
- sprintf (svg_header->point, "word-spacing=\"%s\" ",
- "normal");
- _update_buffer (svg_header);
- sprintf (svg_header->point, "text-anchor=\"%s\"",
- "start");
- _update_buffer (svg_header);
- sprintf (svg_header->point, ">\n");
- _update_buffer (svg_header);
-
- _plotter->data->page->header = svg_header;
-
- svg_trailer = _new_outbuf ();
-
- sprintf (svg_trailer->point, "</g>\n");
- _update_buffer (svg_trailer);
- sprintf (svg_trailer->point, "</svg>\n");
- _update_buffer (svg_trailer);
-
-
- _plotter->data->page->trailer = svg_trailer;
- return true;
- }
- void
- _pl_s_set_matrix (R___(Plotter *_plotter) const double m_local[6])
- {
- double m_base[6], m[6];
- const double *m_emitted = (const double *)NULL;
- bool need_transform_attribute = false;
- int i;
-
- for (i = 0; i < 6; i++)
- m_base[i] = _plotter->drawstate->transform.m_user_to_ndc[i];
-
- if (_plotter->s_matrix_is_unknown)
- {
- for (i = 0; i < 6; i++)
- _plotter->s_matrix[i] = m_base[i];
- _plotter->s_matrix_is_unknown = false;
- if (m_base[0] * m_base[3] - m_base[1] * m_base[2] == 0.0)
-
- _plotter->s_matrix_is_bogus = true;
- }
-
- _matrix_product (m_local, m_base, m);
-
- if (_plotter->s_matrix_is_bogus == false)
-
- {
- for (i = 0; i < 6; i++)
- {
- if (m[i] != _plotter->s_matrix[i])
-
- {
- need_transform_attribute = true;
- break;
- }
- }
- if (need_transform_attribute)
- {
- double inverse_of_global[6], product[6];
- _matrix_inverse (_plotter->s_matrix, inverse_of_global);
-
- _matrix_product (m, inverse_of_global, product);
- m_emitted = product;
- }
- }
- else
-
- {
- need_transform_attribute = true;
- m_emitted = m;
- }
-
-
- if (need_transform_attribute)
- write_svg_transform (_plotter->data->page, m_emitted);
- }
- static void
- write_svg_transform (plOutbuf *outbuf, const double m[6])
- {
- double mm[6];
- double max_value = 0.0;
- int i;
- int type = 0;
-
-
- #define VERY_SMALL_FACTOR 1e-10
-
- for (i = 0; i < 4; i++)
- max_value = DMAX(max_value, FABS(m[i]));
- for (i = 0; i < 6; i++)
- if (i < 4 && FABS(m[i]) < VERY_SMALL_FACTOR * max_value)
- mm[i] = 0;
- else
- mm[i] = m[i];
- if (mm[0] == 1.0 && mm[1] == 0.0 && mm[2] == 0.0 && mm[3] == 1.0
- && mm[4] == 0.0 && mm[5] == 0.0)
-
- return;
-
- if (mm[1] == 0.0 && mm[2] == 0.0)
- type = 1;
- else if (mm[0] == 0.0 && mm[1] == 1.0 && mm[2] == -1.0 && mm[3] == 0.0)
- type = 2;
- else if (mm[0] == 0.0 && mm[1] == -1.0 && mm[2] == 1.0 && mm[3] == 0.0)
- type = 3;
- else if (mm[0] == 0.0 && mm[1] == 1.0 && mm[2] == 1.0 && mm[3] == 0.0)
- type = 4;
- else if (mm[0] == 0.0 && mm[1] == -1.0 && mm[2] == -1.0 && mm[3] == 0.0)
- type = 5;
-
- sprintf (outbuf->point, "transform=\"");
- _update_buffer (outbuf);
-
- if (type != 0)
- {
-
- if (mm[4] != 0.0 || mm[5] != 0.0)
- {
- if (mm[5] == 0.0)
- sprintf (outbuf->point, "translate(%.5g) ",
- mm[4]);
- else
- sprintf (outbuf->point, "translate(%.5g,%.5g) ",
- mm[4], mm[5]);
- _update_buffer (outbuf);
- }
- switch (type)
- {
- case 1:
- if (mm[0] != 1.0 || mm[3] != 1.0)
- {
- if (mm[3] == mm[0])
- sprintf (outbuf->point, "scale(%.5g) ",
- mm[0]);
- else if (mm[3] == -mm[0])
- {
- if (mm[0] != 1.0)
- sprintf (outbuf->point, "scale(1,-1) scale(%.5g) ",
- mm[0]);
- else
- sprintf (outbuf->point, "scale(1,-1) ");
- }
- else
- sprintf (outbuf->point, "scale(%.5g,%.5g) ",
- mm[0], mm[3]);
- _update_buffer (outbuf);
- }
- break;
- case 2:
- sprintf (outbuf->point, "rotate(90) ");
- _update_buffer (outbuf);
- break;
- case 3:
- sprintf (outbuf->point, "rotate(270) ");
- _update_buffer (outbuf);
- break;
- case 4:
- sprintf (outbuf->point, "rotate(90) scale(1,-1) ");
- _update_buffer (outbuf);
- break;
- case 5:
- sprintf (outbuf->point, "rotate(270) scale(1,-1) ");
- _update_buffer (outbuf);
- break;
- default:
- break;
- }
- }
- else
-
- {
- sprintf (outbuf->point, "matrix(%.5g %.5g %.5g %.5g %.5g %.5g) ",
- mm[0], mm[1], mm[2], mm[3], mm[4], mm[5]);
- _update_buffer (outbuf);
- }
- sprintf (outbuf->point, "\" ");
- _update_buffer (outbuf);
- }
|