]> granicus.if.org Git - libass/blob - libass/ass_drawing.c
64f7b29f4f68c9767f9196105e1c8787788a8f78
[libass] / libass / ass_drawing.c
1 /*
2  * Copyright (C) 2009 Grigori Goronzy <greg@geekmind.org>
3  *
4  * This file is part of libass.
5  *
6  * Permission to use, copy, modify, and distribute this software for any
7  * purpose with or without fee is hereby granted, provided that the above
8  * copyright notice and this permission notice appear in all copies.
9  *
10  * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
11  * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
12  * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
13  * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
14  * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
15  * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
16  * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
17  */
18
19 #include "config.h"
20 #include "ass_compat.h"
21
22 #include <ft2build.h>
23 #include FT_OUTLINE_H
24 #include <math.h>
25 #include <stdbool.h>
26 #include <limits.h>
27
28 #include "ass_utils.h"
29 #include "ass_drawing.h"
30 #include "ass_font.h"
31
32 #define GLYPH_INITIAL_POINTS 100
33 #define GLYPH_INITIAL_SEGMENTS 100
34
35 /*
36  * \brief Prepare drawing for parsing.  This just sets a few parameters.
37  */
38 static void drawing_prepare(ASS_Drawing *drawing)
39 {
40     // Scaling parameters
41     drawing->point_scale_x = drawing->scale_x / (1 << (drawing->scale - 1));
42     drawing->point_scale_y = drawing->scale_y / (1 << (drawing->scale - 1));
43 }
44
45 /*
46  * \brief Finish a drawing.  This only sets the horizontal advance according
47  * to the outline's bbox at the moment.
48  */
49 static void drawing_finish(ASS_Drawing *drawing, bool raw_mode)
50 {
51     ASS_Rect bbox = drawing->cbox;
52     ASS_Outline *ol = &drawing->outline;
53
54     if (drawing->library)
55         ass_msg(drawing->library, MSGL_V,
56                 "Parsed drawing with %d points and %d segments",
57                 ol->n_points, ol->n_segments);
58
59     if (raw_mode)
60         return;
61
62     drawing->advance.x = bbox.x_max - bbox.x_min;
63
64     double pbo = drawing->pbo / (1 << (drawing->scale - 1));
65     drawing->desc = double_to_d6(pbo * drawing->scale_y);
66     drawing->asc = bbox.y_max - bbox.y_min - drawing->desc;
67
68     // Place it onto the baseline
69     for (size_t i = 0; i < ol->n_points; i++)
70         ol->points[i].y -= drawing->asc;
71 }
72
73 /*
74  * \brief Check whether a number of items on the list is available
75  */
76 static int token_check_values(ASS_DrawingToken *token, int i, int type)
77 {
78     for (int j = 0; j < i; j++) {
79         if (!token || token->type != type) return 0;
80         token = token->next;
81     }
82
83     return 1;
84 }
85
86 /*
87  * \brief Tokenize a drawing string into a list of ASS_DrawingToken
88  * This also expands points for closing b-splines
89  */
90 static ASS_DrawingToken *drawing_tokenize(char *str)
91 {
92     char *p = str;
93     int type = -1, is_set = 0;
94     double val;
95     ASS_Vector point = {0, 0};
96
97     ASS_DrawingToken *root = NULL, *tail = NULL, *spline_start = NULL;
98
99     while (p && *p) {
100         int got_coord = 0;
101         if (*p == 'c' && spline_start) {
102             // Close b-splines: add the first three points of the b-spline
103             // back to the end
104             if (token_check_values(spline_start->next, 2, TOKEN_B_SPLINE)) {
105                 for (int i = 0; i < 3; i++) {
106                     tail->next = calloc(1, sizeof(ASS_DrawingToken));
107                     tail->next->prev = tail;
108                     tail = tail->next;
109                     tail->type = TOKEN_B_SPLINE;
110                     tail->point = spline_start->point;
111                     spline_start = spline_start->next;
112                 }
113                 spline_start = NULL;
114             }
115         } else if (!is_set && mystrtod(&p, &val)) {
116             point.x = double_to_d6(val);
117             is_set = 1;
118             got_coord = 1;
119             p--;
120         } else if (is_set == 1 && mystrtod(&p, &val)) {
121             point.y = double_to_d6(val);
122             is_set = 2;
123             got_coord = 1;
124             p--;
125         } else if (*p == 'm')
126             type = TOKEN_MOVE;
127         else if (*p == 'n')
128             type = TOKEN_MOVE_NC;
129         else if (*p == 'l')
130             type = TOKEN_LINE;
131         else if (*p == 'b')
132             type = TOKEN_CUBIC_BEZIER;
133         else if (*p == 'q')
134             type = TOKEN_CONIC_BEZIER;
135         else if (*p == 's')
136             type = TOKEN_B_SPLINE;
137         // We're simply ignoring TOKEN_EXTEND_B_SPLINE here.
138         // This is not harmful at all, since it can be ommitted with
139         // similar result (the spline is extended anyway).
140
141         // Ignore the odd extra value, it makes no sense.
142         if (!got_coord)
143             is_set = 0;
144
145         if (type != -1 && is_set == 2) {
146             if (root) {
147                 tail->next = calloc(1, sizeof(ASS_DrawingToken));
148                 tail->next->prev = tail;
149                 tail = tail->next;
150             } else
151                 root = tail = calloc(1, sizeof(ASS_DrawingToken));
152             tail->type = type;
153             tail->point = point;
154             is_set = 0;
155             if (type == TOKEN_B_SPLINE && !spline_start)
156                 spline_start = tail->prev;
157         }
158         p++;
159     }
160
161     return root;
162 }
163
164 /*
165  * \brief Free a list of tokens
166  */
167 static void drawing_free_tokens(ASS_DrawingToken *token)
168 {
169     while (token) {
170         ASS_DrawingToken *at = token;
171         token = token->next;
172         free(at);
173     }
174 }
175
176 /*
177  * \brief Update drawing cbox
178  */
179 static inline void update_cbox(ASS_Drawing *drawing, ASS_Vector *point)
180 {
181     ASS_Rect *box = &drawing->cbox;
182
183     box->x_min = FFMIN(box->x_min, point->x);
184     box->x_max = FFMAX(box->x_max, point->x);
185     box->y_min = FFMIN(box->y_min, point->y);
186     box->y_max = FFMAX(box->y_max, point->y);
187 }
188
189 /*
190  * \brief Translate and scale a point coordinate according to baseline
191  * offset and scale.
192  */
193 static inline void translate_point(ASS_Drawing *drawing, ASS_Vector *point)
194 {
195     point->x = drawing->point_scale_x * point->x;
196     point->y = drawing->point_scale_y * point->y;
197
198     update_cbox(drawing, point);
199 }
200
201 /*
202  * \brief Add curve to drawing
203  */
204 static bool drawing_add_curve(ASS_Drawing *drawing, ASS_DrawingToken *token,
205                               bool spline, int started)
206 {
207     ASS_Vector p[4];
208     for (int i = 0; i < 4; ++i) {
209         p[i] = token->point;
210         translate_point(drawing, &p[i]);
211         token = token->next;
212     }
213
214     if (spline) {
215         int x01 = (p[1].x - p[0].x) / 3;
216         int y01 = (p[1].y - p[0].y) / 3;
217         int x12 = (p[2].x - p[1].x) / 3;
218         int y12 = (p[2].y - p[1].y) / 3;
219         int x23 = (p[3].x - p[2].x) / 3;
220         int y23 = (p[3].y - p[2].y) / 3;
221
222         p[0].x = p[1].x + ((x12 - x01) >> 1);
223         p[0].y = p[1].y + ((y12 - y01) >> 1);
224         p[3].x = p[2].x + ((x23 - x12) >> 1);
225         p[3].y = p[2].y + ((y23 - y12) >> 1);
226         p[1].x += x12;
227         p[1].y += y12;
228         p[2].x -= x12;
229         p[2].y -= y12;
230     }
231
232     return (started ||
233         outline_add_point(&drawing->outline, p[0], 0)) &&
234         outline_add_point(&drawing->outline, p[1], 0) &&
235         outline_add_point(&drawing->outline, p[2], 0) &&
236         outline_add_point(&drawing->outline, p[3], OUTLINE_CUBIC_SPLINE);
237 }
238
239 /*
240  * \brief Create and initialize a new drawing and return it
241  */
242 ASS_Drawing *ass_drawing_new(ASS_Library *lib)
243 {
244     ASS_Drawing *drawing = calloc(1, sizeof(*drawing));
245     if (!drawing)
246         return NULL;
247     drawing->cbox.x_min = drawing->cbox.y_min = INT32_MAX;
248     drawing->cbox.x_max = drawing->cbox.y_max = INT32_MIN;
249     drawing->library = lib;
250     drawing->scale_x = 1.;
251     drawing->scale_y = 1.;
252
253     if (!outline_alloc(&drawing->outline, GLYPH_INITIAL_POINTS, GLYPH_INITIAL_SEGMENTS)) {
254         free(drawing);
255         return NULL;
256     }
257     return drawing;
258 }
259
260 /*
261  * \brief Free a drawing
262  */
263 void ass_drawing_free(ASS_Drawing *drawing)
264 {
265     if (drawing) {
266         free(drawing->text);
267         outline_free(&drawing->outline);
268     }
269     free(drawing);
270 }
271
272 /*
273  * \brief Copy an ASCII string to the drawing text buffer
274  */
275 void ass_drawing_set_text(ASS_Drawing *drawing, char *str, size_t len)
276 {
277     free(drawing->text);
278     drawing->text = strndup(str, len);
279 }
280
281 /*
282  * \brief Create a hashcode for the drawing
283  * XXX: To avoid collisions a better hash algorithm might be useful.
284  */
285 void ass_drawing_hash(ASS_Drawing *drawing)
286 {
287     if (!drawing->text)
288         return;
289     drawing->hash = fnv_32a_str(drawing->text, FNV1_32A_INIT);
290 }
291
292 /*
293  * \brief Convert token list to outline.  Calls the line and curve evaluators.
294  */
295 ASS_Outline *ass_drawing_parse(ASS_Drawing *drawing, bool raw_mode)
296 {
297     bool started = false;
298     ASS_DrawingToken *token;
299     ASS_Vector pen = {0, 0};
300
301     drawing->tokens = drawing_tokenize(drawing->text);
302     drawing_prepare(drawing);
303
304     token = drawing->tokens;
305     while (token) {
306         // Draw something according to current command
307         switch (token->type) {
308         case TOKEN_MOVE_NC:
309             pen = token->point;
310             translate_point(drawing, &pen);
311             token = token->next;
312             break;
313         case TOKEN_MOVE:
314             pen = token->point;
315             translate_point(drawing, &pen);
316             if (started) {
317                 if (!outline_add_segment(&drawing->outline, OUTLINE_LINE_SEGMENT))
318                     goto error;
319                 if (!outline_close_contour(&drawing->outline))
320                     goto error;
321                 started = false;
322             }
323             token = token->next;
324             break;
325         case TOKEN_LINE: {
326             ASS_Vector to = token->point;
327             translate_point(drawing, &to);
328             if (!started && !outline_add_point(&drawing->outline, pen, 0))
329                 goto error;
330             if (!outline_add_point(&drawing->outline, to, OUTLINE_LINE_SEGMENT))
331                 goto error;
332             started = true;
333             token = token->next;
334             break;
335         }
336         case TOKEN_CUBIC_BEZIER:
337             if (token_check_values(token, 3, TOKEN_CUBIC_BEZIER) &&
338                 token->prev) {
339                 if (!drawing_add_curve(drawing, token->prev, false, started))
340                     goto error;
341                 token = token->next;
342                 token = token->next;
343                 token = token->next;
344                 started = true;
345             } else
346                 token = token->next;
347             break;
348         case TOKEN_B_SPLINE:
349             if (token_check_values(token, 3, TOKEN_B_SPLINE) &&
350                 token->prev) {
351                 if (!drawing_add_curve(drawing, token->prev, true, started))
352                     goto error;
353                 token = token->next;
354                 started = true;
355             } else
356                 token = token->next;
357             break;
358         default:
359             token = token->next;
360             break;
361         }
362     }
363
364     // Close the last contour
365     if (started) {
366         if (!outline_add_segment(&drawing->outline, OUTLINE_LINE_SEGMENT))
367             goto error;
368         if (!outline_close_contour(&drawing->outline))
369             goto error;
370     }
371
372     drawing_finish(drawing, raw_mode);
373     drawing_free_tokens(drawing->tokens);
374     return &drawing->outline;
375
376 error:
377     drawing_free_tokens(drawing->tokens);
378     return NULL;
379 }