summaryrefslogtreecommitdiffstats
path: root/include/2geom/svg-path-parser.h
blob: e25316cf8cf58a120f1a169c0cb7c3454c68306d (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
/**
 * \file
 * \brief  parse SVG path specifications
 *
 * Copyright 2007 MenTaLguY <mental@rydia.net>
 * Copyright 2007 Aaron Spike <aaron@ekips.org>
 *
 * This library is free software; you can redistribute it and/or
 * modify it either under the terms of the GNU Lesser General Public
 * License version 2.1 as published by the Free Software Foundation
 * (the "LGPL") or, at your option, under the terms of the Mozilla
 * Public License Version 1.1 (the "MPL"). If you do not alter this
 * notice, a recipient may use your version of this file under either
 * the MPL or the LGPL.
 *
 * You should have received a copy of the LGPL along with this library
 * in the file COPYING-LGPL-2.1; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
 * You should have received a copy of the MPL along with this library
 * in the file COPYING-MPL-1.1
 *
 * The contents of this file are subject to the Mozilla Public License
 * Version 1.1 (the "License"); you may not use this file except in
 * compliance with the License. You may obtain a copy of the License at
 * http://www.mozilla.org/MPL/
 *
 * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY
 * OF ANY KIND, either express or implied. See the LGPL or the MPL for
 * the specific language governing rights and limitations.
 *
 */

#ifndef LIB2GEOM_SEEN_SVG_PATH_PARSER_H
#define LIB2GEOM_SEEN_SVG_PATH_PARSER_H

#include <iostream>
#include <iterator>
#include <stdexcept>
#include <vector>
#include <cstdio>
#include <2geom/exception.h>
#include <2geom/point.h>
#include <2geom/path-sink.h>
#include <2geom/forward.h>

namespace Geom {

/** @brief Read SVG path data and feed it to a PathSink
 *
 * This class provides an interface to an SVG path data parser written in Ragel.
 * It supports parsing the path data either at once or block-by-block.
 * Use the parse() functions to parse complete data and the feed() and finish()
 * functions to parse partial data.
 *
 * The parser will call the appropriate methods on the PathSink supplied
 * at construction. To store the path in memory as a PathVector, pass
 * a PathBuilder. You can also use one of the freestanding helper functions
 * if you don't need to parse data block-by-block.
 *
 * @ingroup Paths
 */
class SVGPathParser {
public:
    SVGPathParser(PathSink &sink);
    ~SVGPathParser();

    /** @brief Reset internal state.
     * Discards the internal state associated with partially parsed data,
     * letting you start from scratch. Note that any partial data written
     * to the path sink is not affected - you need to clear it yourself. */
    void reset();

    /** @brief Parse a C-style string.
     * The path sink is flushed and the internal state is reset after this call.
     * Note that the state is not reset before this method, so you can use it to
     * process the last block of partial data.
     * @param str String to parse
     * @param len Length of string or -1 if null-terminated */
    void parse(char const *str, int len = -1);
    /** @brief Parse an STL string. */
    void parse(std::string const &s);

    /** @brief Parse a part of path data stored in a C-style string.
     * This method does not reset internal state, so it can be called multiple
     * times to parse successive blocks of a longer SVG path data string.
     * To finish parsing, call finish() after the final block or call parse()
     * with the last block of data.
     * @param str String to parse
     * @param len Length of string or -1 if null-terminated */
    void feed(char const *str, int len = -1);
    /** @brief Parse a part of path data stored in an STL string. */
    void feed(std::string const &s);

    /** @brief Finalize parsing.
     * After the last block of data was submitted with feed(), call this method
     * to finalize parsing, flush the path sink and reset internal state.
     * You should not call this after parse(). */
    void finish();

    /** @brief Set the threshold for considering the closing segment degenerate.
     * When the current point was reached by a relative command, is closer
     * to the initial point of the path than the specified threshold
     * and a 'z' is encountered, the last segment will be adjusted instead so that
     * the closing segment has exactly zero length. This is useful when reading
     * SVG 1.1 paths that have non-linear final segments written in relative
     * coordinates, which always suffer from some loss of precision. SVG 2
     * allows alternate placement of 'z' which does not have this problem. */
    void setZSnapThreshold(Coord threshold) { _z_snap_threshold = threshold; }
    Coord zSnapThreshold() const { return _z_snap_threshold; }

private:
    bool _absolute;
    bool _moveto_was_absolute;
    Point _current;
    Point _initial;
    Point _cubic_tangent;
    Point _quad_tangent;
    std::vector<Coord> _params;
    PathSink &_sink;
    Coord _z_snap_threshold;
    Curve *_curve;

    int cs;
    std::string _number_part;

    void _push(Coord value);
    Coord _pop();
    bool _pop_flag();
    Coord _pop_coord(Geom::Dim2 axis);
    Point _pop_point();
    void _moveTo(Point const &p);
    void _lineTo(Point const &p);
    void _curveTo(Point const &c0, Point const &c1, Point const &p);
    void _quadTo(Point const &c, Point const &p);
    void _arcTo(double rx, double ry, double angle,
                bool large_arc, bool sweep, Point const &p);
    void _closePath();
    void _pushCurve(Curve *c);

    void _parse(char const *str, char const *strend, bool finish);
};

/** @brief Feed SVG path data to the specified sink
 * @ingroup Paths */
void parse_svg_path(char const *str, PathSink &sink);
/** @brief Feed SVG path data to the specified sink
 * @ingroup Paths */
inline void parse_svg_path(std::string const &str, PathSink &sink) {
    parse_svg_path(str.c_str(), sink);
}
/** Feed SVG path data from a C stream to the specified sink
 * @ingroup Paths */
void parse_svg_path_file(FILE *fi, PathSink &sink);

/** @brief Create path vector from SVG path data stored in a C string
 * @ingroup Paths */
inline PathVector parse_svg_path(char const *str) {
    PathVector ret;
    SubpathInserter iter(ret);
    PathIteratorSink<SubpathInserter> generator(iter);

    parse_svg_path(str, generator);
    return ret;
}

/** @brief Create path vector from a C stream with SVG path data
 * @ingroup Paths */
inline PathVector read_svgd_f(FILE * fi) {
    PathVector ret;
    SubpathInserter iter(ret);
    PathIteratorSink<SubpathInserter> generator(iter);

    parse_svg_path_file(fi, generator);
    return ret;
}

/** @brief Create path vector from SVG path data stored in a file
 * @ingroup Paths */
inline PathVector read_svgd(char const *filename) {
    FILE* fi = fopen(filename, "r");
    if(fi == NULL) throw(std::runtime_error("Error opening file"));
    PathVector out = read_svgd_f(fi);
    fclose(fi);
    return out;
}

} // end namespace Geom

#endif
/*
  Local Variables:
  mode:c++
  c-file-style:"stroustrup"
  c-file-offsets:((innamespace . 0)(inline-open . 0)(case-label . +))
  indent-tabs-mode:nil
  fill-column:99
  End:
*/
// vim: filetype=cpp:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:fileencoding=utf-8:textwidth=99 :