87f7b51477f5b5edc4b0696a79f6e0f0770083f3
[libsub.git] / test / subrip_reader_test.cc
1 /*
2     Copyright (C) 2014-2020 Carl Hetherington <cth@carlh.net>
3
4     This program is free software; you can redistribute it and/or modify
5     it under the terms of the GNU General Public License as published by
6     the Free Software Foundation; either version 2 of the License, or
7     (at your option) any later version.
8
9     This program is distributed in the hope that it will be useful,
10     but WITHOUT ANY WARRANTY; without even the implied warranty of
11     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12     GNU General Public License for more details.
13
14     You should have received a copy of the GNU General Public License
15     along with this program; if not, write to the Free Software
16     Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
17
18 */
19
20 #include "subrip_reader.h"
21 #include "subtitle.h"
22 #include "test.h"
23 #include "exceptions.h"
24 #include "collect.h"
25 #include <boost/test/unit_test.hpp>
26 #include <boost/filesystem.hpp>
27 #include <cmath>
28 #include <iostream>
29 #include <cstdio>
30
31 using std::list;
32 using std::cerr;
33 using std::vector;
34 using std::fabs;
35
36 /* Test reading of a Subrip file */
37 BOOST_AUTO_TEST_CASE (subrip_reader_test)
38 {
39         FILE* f = fopen ("test/data/test.srt", "r");
40         sub::SubripReader reader (f);
41         fclose (f);
42         list<sub::Subtitle> subs = sub::collect<std::list<sub::Subtitle> > (reader.subtitles ());
43
44         list<sub::Subtitle>::iterator i = subs.begin ();
45
46
47         /* First subtitle */
48
49         BOOST_CHECK (i != subs.end ());
50         BOOST_CHECK_EQUAL (i->from, sub::Time::from_hms (0, 0, 41, 90));
51         BOOST_CHECK_EQUAL (i->to, sub::Time::from_hms (0, 0, 42, 210));
52
53         list<sub::Line>::iterator j = i->lines.begin ();
54         BOOST_CHECK (j != i->lines.end ());
55         BOOST_CHECK_EQUAL (j->blocks.size(), 1);
56         sub::Block b = j->blocks.front ();
57         BOOST_CHECK_EQUAL (b.text, "This is a subtitle");
58         /* No font is specified by subrip, so none should be seen here */
59         BOOST_CHECK (!b.font);
60         BOOST_CHECK (!b.font_size.specified());
61         BOOST_CHECK_EQUAL (b.bold, false);
62         BOOST_CHECK_EQUAL (b.italic, false);
63         BOOST_CHECK_EQUAL (b.underline, false);
64         BOOST_REQUIRE (j->vertical_position.line);
65         BOOST_CHECK_EQUAL (j->vertical_position.line.get(), 0);
66         BOOST_CHECK_EQUAL (j->vertical_position.reference.get(), sub::TOP_OF_SUBTITLE);
67         ++j;
68
69         BOOST_CHECK (j != i->lines.end ());
70         BOOST_CHECK_EQUAL (j->blocks.size(), 1);
71         b = j->blocks.front ();
72         BOOST_CHECK_EQUAL (b.text, "and that's a line break");
73         /* No font is specified by subrip, so none should be seen here */
74         BOOST_CHECK (!b.font);
75         BOOST_CHECK (!b.font_size.specified());
76         BOOST_CHECK_EQUAL (b.bold, false);
77         BOOST_CHECK_EQUAL (b.italic, false);
78         BOOST_CHECK_EQUAL (b.underline, false);
79         BOOST_REQUIRE (j->vertical_position.line);
80         BOOST_CHECK_EQUAL (j->vertical_position.line.get(), 1);
81         BOOST_CHECK_EQUAL (j->vertical_position.reference.get(), sub::TOP_OF_SUBTITLE);
82         ++i;
83
84
85         /* Second subtitle */
86
87         BOOST_CHECK (i != subs.end ());
88         BOOST_CHECK_EQUAL (i->from, sub::Time::from_hms (0, 1, 1, 10));
89         BOOST_CHECK_EQUAL (i->to, sub::Time::from_hms (0, 1, 2, 100));
90
91         BOOST_CHECK_EQUAL (i->lines.size(), 1);
92         sub::Line l = i->lines.front ();
93         BOOST_CHECK_EQUAL (l.blocks.size(), 7);
94         BOOST_CHECK_EQUAL (l.vertical_position.line.get(), 0);
95         BOOST_CHECK_EQUAL (l.vertical_position.reference.get(), sub::TOP_OF_SUBTITLE);
96
97         list<sub::Block>::iterator k = l.blocks.begin ();
98
99         BOOST_CHECK (k != l.blocks.end ());
100         BOOST_CHECK_EQUAL (k->text, "This is some ");
101         /* No font is specified by subrip, so none should be seen here */
102         BOOST_CHECK (!b.font);
103         BOOST_CHECK (!b.font_size.specified());
104         BOOST_CHECK_EQUAL (k->bold, false);
105         BOOST_CHECK_EQUAL (k->italic, false);
106         BOOST_CHECK_EQUAL (k->underline, false);
107         ++k;
108
109         BOOST_CHECK (k != l.blocks.end ());
110         BOOST_CHECK_EQUAL (k->text, "bold");
111         /* No font is specified by subrip, so none should be seen here */
112         BOOST_CHECK (!b.font);
113         BOOST_CHECK (!b.font_size.specified());
114         BOOST_CHECK_EQUAL (k->bold, true);
115         BOOST_CHECK_EQUAL (k->italic, false);
116         BOOST_CHECK_EQUAL (k->underline, false);
117         ++k;
118
119         BOOST_CHECK (k != l.blocks.end ());
120         BOOST_CHECK_EQUAL (k->text, " and some ");
121         /* No font is specified by subrip, so none should be seen here */
122         BOOST_CHECK (!b.font);
123         BOOST_CHECK (!b.font_size.specified());
124         BOOST_CHECK_EQUAL (k->bold, false);
125         BOOST_CHECK_EQUAL (k->italic, false);
126         BOOST_CHECK_EQUAL (k->underline, false);
127         ++k;
128
129         BOOST_CHECK (k != l.blocks.end ());
130         BOOST_CHECK_EQUAL (k->text, "bold italic");
131         /* No font is specified by subrip, so none should be seen here */
132         BOOST_CHECK (!b.font);
133         BOOST_CHECK (!b.font_size.specified());
134         BOOST_CHECK_EQUAL (k->bold, true);
135         BOOST_CHECK_EQUAL (k->italic, true);
136         BOOST_CHECK_EQUAL (k->underline, false);
137         ++k;
138
139         BOOST_CHECK (k != l.blocks.end ());
140         BOOST_CHECK_EQUAL (k->text, " and some ");
141         /* No font is specified by subrip, so none should be seen here */
142         BOOST_CHECK (!b.font);
143         BOOST_CHECK (!b.font_size.specified());
144         BOOST_CHECK_EQUAL (k->bold, false);
145         BOOST_CHECK_EQUAL (k->italic, false);
146         BOOST_CHECK_EQUAL (k->underline, false);
147         ++k;
148
149         BOOST_CHECK (k != l.blocks.end ());
150         BOOST_CHECK_EQUAL (k->text, "underlined");
151         /* No font is specified by subrip, so none should be seen here */
152         BOOST_CHECK (!b.font);
153         BOOST_CHECK (!b.font_size.specified());
154         BOOST_CHECK_EQUAL (k->bold, false);
155         BOOST_CHECK_EQUAL (k->italic, false);
156         BOOST_CHECK_EQUAL (k->underline, true);
157         ++k;
158
159         BOOST_CHECK (k != l.blocks.end ());
160         BOOST_CHECK_EQUAL (k->text, ".");
161         /* No font is specified by subrip, so none should be seen here */
162         BOOST_CHECK (!b.font);
163         BOOST_CHECK (!b.font_size.specified());
164         BOOST_CHECK_EQUAL (k->bold, false);
165         BOOST_CHECK_EQUAL (k->italic, false);
166         BOOST_CHECK_EQUAL (k->underline, false);
167         ++k;
168
169         BOOST_CHECK (k == l.blocks.end ());
170 }
171
172 /* Test reading of another Subrip file */
173 BOOST_AUTO_TEST_CASE (subrip_reader_test2)
174 {
175         FILE* f = fopen ("test/data/test2.srt", "r");
176         sub::SubripReader reader (f);
177         fclose (f);
178         list<sub::Subtitle> subs = sub::collect<list<sub::Subtitle> > (reader.subtitles ());
179
180         list<sub::Subtitle>::const_iterator i = subs.begin();
181
182         BOOST_CHECK (i != subs.end ());
183         BOOST_CHECK_EQUAL (i->from, sub::Time::from_hms (0, 1, 49, 200));
184         BOOST_CHECK_EQUAL (i->to, sub::Time::from_hms (0, 1, 52, 351));
185         BOOST_CHECK_EQUAL (i->lines.size(), 2);
186         BOOST_CHECK_EQUAL (i->lines.front().blocks.front().text, "This is a subtitle, and it goes ");
187         BOOST_CHECK_EQUAL (i->lines.back().blocks.front().text, "over two lines.");
188
189         ++i;
190         BOOST_CHECK (i != subs.end ());
191         BOOST_CHECK_EQUAL (i->from, sub::Time::from_hms (0, 1, 52, 440));
192         BOOST_CHECK_EQUAL (i->to, sub::Time::from_hms (0, 1, 54, 351));
193         BOOST_CHECK_EQUAL (i->lines.size(), 1);
194         BOOST_CHECK_EQUAL (i->lines.front().blocks.front().text, "We have emboldened this");
195         BOOST_CHECK_EQUAL (i->lines.front().blocks.front().bold, true);
196
197         ++i;
198         BOOST_CHECK (i != subs.end ());
199         BOOST_CHECK_EQUAL (i->from, sub::Time::from_hms (0, 1, 54, 440));
200         BOOST_CHECK_EQUAL (i->to, sub::Time::from_hms (0, 1, 56, 590));
201         BOOST_CHECK_EQUAL (i->lines.size(), 1);
202         BOOST_CHECK_EQUAL (i->lines.front().blocks.front().text, "And italicised this.");
203         BOOST_CHECK_EQUAL (i->lines.front().blocks.front().italic, true);
204
205         ++i;
206         BOOST_CHECK (i != subs.end ());
207         BOOST_CHECK_EQUAL (i->from, sub::Time::from_hms (0, 1, 56, 680));
208         BOOST_CHECK_EQUAL (i->to, sub::Time::from_hms (0, 1, 58, 955));
209         BOOST_CHECK_EQUAL (i->lines.size(), 1);
210         BOOST_CHECK_EQUAL (i->lines.front().blocks.front().text, "Shall I compare thee to a summers' day?");
211
212         ++i;
213         BOOST_CHECK (i != subs.end ());
214         BOOST_CHECK_EQUAL (i->from, sub::Time::from_hms (0, 2, 0, 840));
215         BOOST_CHECK_EQUAL (i->to, sub::Time::from_hms (0, 2, 3, 400));
216         BOOST_CHECK_EQUAL (i->lines.size(), 1);
217         BOOST_CHECK_EQUAL (i->lines.front().blocks.front().text, "Is this a dagger I see before me?");
218
219         ++i;
220         BOOST_CHECK (i != subs.end ());
221         BOOST_CHECK_EQUAL (i->from, sub::Time::from_hms (0, 3, 54, 560));
222         BOOST_CHECK_EQUAL (i->to, sub::Time::from_hms (0, 3, 56, 471));
223         BOOST_CHECK_EQUAL (i->lines.size(), 1);
224         BOOST_CHECK_EQUAL (i->lines.front().blocks.front().text, "Hello world.");
225
226         ++i;
227         BOOST_CHECK (i != subs.end ());
228         BOOST_CHECK_EQUAL (i->from, sub::Time::from_hms (0, 4, 50, 123));
229         BOOST_CHECK_EQUAL (i->to, sub::Time::from_hms (0, 4, 55, 23));
230         BOOST_CHECK_EQUAL (i->lines.size(), 2);
231         BOOST_CHECK_EQUAL (i->lines.front().blocks.front().text, "Some italics over");
232         BOOST_CHECK_EQUAL (i->lines.front().blocks.front().italic, true);
233         BOOST_CHECK_EQUAL (i->lines.back().blocks.front().text, "multiple lines");
234         BOOST_CHECK_EQUAL (i->lines.back().blocks.front().italic, true);
235
236         ++i;
237         BOOST_CHECK (i == subs.end ());
238 }
239
240 /** Test SubripReader::convert_line */
241 BOOST_AUTO_TEST_CASE (subrip_reader_convert_line_test)
242 {
243         sub::SubripReader r;
244
245         sub::RawSubtitle rs;
246         r.convert_line ("Hello world", rs);
247         BOOST_CHECK_EQUAL (r._subs.size(), 1);
248         BOOST_CHECK_EQUAL (r._subs.front().text, "Hello world");
249         r._subs.clear ();
250
251         rs = sub::RawSubtitle();
252         r.convert_line ("<b>Hello world</b>", rs);
253         BOOST_CHECK_EQUAL (r._subs.size(), 1);
254         BOOST_CHECK_EQUAL (r._subs.front().text, "Hello world");
255         BOOST_CHECK_EQUAL (r._subs.front().bold, true);
256         r._subs.clear ();
257
258         rs = sub::RawSubtitle();
259         r.convert_line ("<i>Hello world</i>", rs);
260         BOOST_CHECK_EQUAL (r._subs.size(), 1);
261         BOOST_CHECK_EQUAL (r._subs.front().text, "Hello world");
262         BOOST_CHECK_EQUAL (r._subs.front().italic, true);
263         r._subs.clear ();
264
265         rs = sub::RawSubtitle();
266         r.convert_line ("<u>Hello world</u>", rs);
267         BOOST_CHECK_EQUAL (r._subs.size(), 1);
268         BOOST_CHECK_EQUAL (r._subs.front().text, "Hello world");
269         BOOST_CHECK_EQUAL (r._subs.front().underline, true);
270         r._subs.clear ();
271
272         rs = sub::RawSubtitle();
273         r.convert_line ("{b}Hello world{/b}", rs);
274         BOOST_CHECK_EQUAL (r._subs.size(), 1);
275         BOOST_CHECK_EQUAL (r._subs.front().text, "Hello world");
276         BOOST_CHECK_EQUAL (r._subs.front().bold, true);
277         r._subs.clear ();
278
279         rs = sub::RawSubtitle();
280         r.convert_line ("{i}Hello world{/i}", rs);
281         BOOST_CHECK_EQUAL (r._subs.size(), 1);
282         BOOST_CHECK_EQUAL (r._subs.front().text, "Hello world");
283         BOOST_CHECK_EQUAL (r._subs.front().italic, true);
284         r._subs.clear ();
285
286         rs = sub::RawSubtitle();
287         r.convert_line ("{u}Hello world{/u}", rs);
288         BOOST_CHECK_EQUAL (r._subs.size(), 1);
289         BOOST_CHECK_EQUAL (r._subs.front().text, "Hello world");
290         BOOST_CHECK_EQUAL (r._subs.front().underline, true);
291         r._subs.clear ();
292
293         rs = sub::RawSubtitle();
294         r.convert_line ("<b>This is <i>nesting</i> of subtitles</b>", rs);
295         BOOST_CHECK_EQUAL (r._subs.size(), 3);
296         list<sub::RawSubtitle>::iterator i = r._subs.begin ();
297         BOOST_CHECK_EQUAL (i->text, "This is ");
298         BOOST_CHECK_EQUAL (i->bold, true);
299         BOOST_CHECK_EQUAL (i->italic, false);
300         ++i;
301         BOOST_CHECK_EQUAL (i->text, "nesting");
302         BOOST_CHECK_EQUAL (i->bold, true);
303         BOOST_CHECK_EQUAL (i->italic, true);
304         ++i;
305         BOOST_CHECK_EQUAL (i->text, " of subtitles");
306         BOOST_CHECK_EQUAL (i->bold, true);
307         BOOST_CHECK_EQUAL (i->italic, false);
308         ++i;
309         r._subs.clear ();
310
311         rs = sub::RawSubtitle();
312         r.convert_line ("<font color=\"#ff00ff\">simple color</font>", rs);
313         BOOST_CHECK_EQUAL (r._subs.size(), 1);
314         BOOST_CHECK_EQUAL (r._subs.front().text, "simple color");
315         BOOST_CHECK_EQUAL (r._subs.front().bold, false);
316         BOOST_CHECK_CLOSE (r._subs.front().colour.r, 1, 0.1);
317         BOOST_CHECK (fabs (r._subs.front().colour.g) < 0.01);
318         BOOST_CHECK_CLOSE (r._subs.front().colour.b, 1, 0.1);
319         r._subs.clear ();
320
321         rs = sub::RawSubtitle();
322         r.convert_line ("<font color=\"#ff0000\">some red text <b>in bold</b></font>", rs);
323         BOOST_CHECK_EQUAL (r._subs.size(), 2);
324         i = r._subs.begin ();
325         BOOST_CHECK_EQUAL (i->text, "some red text ");
326         BOOST_CHECK_EQUAL (i->bold, false);
327         BOOST_CHECK_CLOSE (i->colour.r, 1, 0.1);
328         BOOST_CHECK (fabs (i->colour.g) < 0.01);
329         BOOST_CHECK (fabs (i->colour.b) < 0.01);
330         ++i;
331         BOOST_CHECK_EQUAL (i->text, "in bold");
332         BOOST_CHECK_EQUAL (i->bold, true);
333         BOOST_CHECK_CLOSE (i->colour.r, 1, 0.1);
334         BOOST_CHECK (fabs (i->colour.g) < 0.01);
335         BOOST_CHECK (fabs (i->colour.b) < 0.01);
336         r._subs.clear ();
337
338         rs = sub::RawSubtitle();
339         r.convert_line ("<Font color=\"#ff0000\">some red text <b>in bold</b></font>", rs);
340         BOOST_CHECK_EQUAL (r._subs.size(), 2);
341         i = r._subs.begin ();
342         BOOST_CHECK_EQUAL (i->text, "some red text ");
343         BOOST_CHECK_EQUAL (i->bold, false);
344         BOOST_CHECK_CLOSE (i->colour.r, 1, 0.1);
345         BOOST_CHECK (fabs (i->colour.g) < 0.01);
346         BOOST_CHECK (fabs (i->colour.b) < 0.01);
347         ++i;
348         BOOST_CHECK_EQUAL (i->text, "in bold");
349         BOOST_CHECK_EQUAL (i->bold, true);
350         BOOST_CHECK_CLOSE (i->colour.r, 1, 0.1);
351         BOOST_CHECK (fabs (i->colour.g) < 0.01);
352         BOOST_CHECK (fabs (i->colour.b) < 0.01);
353         r._subs.clear ();
354
355         rs = sub::RawSubtitle();
356         r.convert_line ("<font color=\"#0000ff\">some blue text <b>in bold</b></font>", rs);
357         BOOST_CHECK_EQUAL (r._subs.size(), 2);
358         i = r._subs.begin ();
359         BOOST_CHECK_EQUAL (i->text, "some blue text ");
360         BOOST_CHECK_EQUAL (i->bold, false);
361         BOOST_CHECK (fabs (i->colour.r) < 0.01);
362         BOOST_CHECK (fabs (i->colour.g) < 0.01);
363         BOOST_CHECK_CLOSE (i->colour.b, 1, 0.1);
364         ++i;
365         BOOST_CHECK_EQUAL (i->text, "in bold");
366         BOOST_CHECK_EQUAL (i->bold, true);
367         BOOST_CHECK (fabs (i->colour.r) < 0.01);
368         BOOST_CHECK (fabs (i->colour.g) < 0.01);
369         BOOST_CHECK_CLOSE (i->colour.b, 1, 0.1);
370         r._subs.clear ();
371 }
372
373 /** Test SubripReader::convert_time */
374 BOOST_AUTO_TEST_CASE (subrip_reader_convert_time_test)
375 {
376         sub::SubripReader reader;
377         BOOST_CHECK_EQUAL (reader.convert_time ("00:03:10,500"), sub::Time::from_hms (0, 3, 10, 500));
378         BOOST_CHECK_EQUAL (reader.convert_time ("04:19:51,782"), sub::Time::from_hms (4, 19, 51, 782));
379 }
380
381 static void
382 test (boost::filesystem::path p)
383 {
384         p = private_test / p;
385         FILE* f = fopen (p.string().c_str(), "r");
386         BOOST_CHECK (f);
387         if (!f) {
388                 cerr << p << " not found.\n";
389                 return;
390         }
391         sub::SubripReader r (f);
392         fclose (f);
393 }
394
395 static void
396 test_throw (boost::filesystem::path p)
397 {
398         p = private_test / p;
399         FILE* f = fopen (p.string().c_str(), "r");
400         BOOST_CHECK (f);
401         if (!f) {
402                 cerr << p << " not found.\n";
403                 return;
404         }
405         BOOST_CHECK_THROW (sub::SubripReader r(f), sub::SubripError);
406         fclose (f);
407 }
408
409 /** Test of reading some typical .srt files */
410 BOOST_AUTO_TEST_CASE (subrip_read_test)
411 {
412         test ("sintel_en.srt");
413         test ("sintel_fr.srt");
414         test ("Fight.Club.1999.720p.BRRip.x264-x0r.srt");
415         test ("EU13.srt");
416         test ("Subtitulos_HURTO_eng.srt");
417         test_throw ("subtitulo1.srt");
418 }
419
420 #define SUB_START(f, t) \
421         BOOST_REQUIRE (i != subs.end ()); \
422         BOOST_CHECK_EQUAL (i->from, f); \
423         BOOST_CHECK_EQUAL (i->to, t); \
424         j = i->lines.begin ();
425
426 #define LINE(p)                                                 \
427         BOOST_REQUIRE (j != i->lines.end ()); \
428         BOOST_CHECK (j->vertical_position.line); \
429         BOOST_CHECK_EQUAL (j->vertical_position.line.get(), p); \
430         BOOST_CHECK (j->vertical_position.reference); \
431         BOOST_CHECK_EQUAL (j->vertical_position.reference.get(), sub::TOP_OF_SUBTITLE); \
432         k = j->blocks.begin (); \
433         ++j;
434
435 #define BLOCK(t, f, s, b, i, u) \
436         BOOST_REQUIRE (k != j->blocks.end ()); \
437         BOOST_CHECK_EQUAL (k->text, t); \
438         BOOST_CHECK_EQUAL (k->bold, b); \
439         BOOST_CHECK_EQUAL (k->italic, i); \
440         BOOST_CHECK_EQUAL (k->underline, u); \
441         ++k;
442
443 #define SUB_END() \
444         ++i;
445
446 /** Test reading of another .srt file */
447 BOOST_AUTO_TEST_CASE (subrip_reader_test3)
448 {
449         boost::filesystem::path p = private_test / "DCP-o-matic_test_subs_1.srt";
450         FILE* f = fopen (p.string().c_str(), "r");
451         sub::SubripReader reader (f);
452         fclose (f);
453         list<sub::Subtitle> subs = sub::collect<std::list<sub::Subtitle> > (reader.subtitles ());
454
455         list<sub::Subtitle>::iterator i = subs.begin ();
456         list<sub::Line>::iterator j;
457         list<sub::Block>::iterator k;
458
459         BOOST_REQUIRE (i != subs.end ());
460
461         SUB_START (sub::Time::from_hms (0, 0, 0, 76), sub::Time::from_hms (0, 0, 1, 116));
462         LINE (0);
463         BLOCK ("This line is normal", "Arial", 30, false, false, false);
464         LINE (1);
465         BLOCK ("This line is bold", "Arial", 30, true, false, false);
466         SUB_END ();
467
468         SUB_START (sub::Time::from_hms (0, 0, 1, 206), sub::Time::from_hms (0, 0, 2, 246));
469         LINE (0);
470         BLOCK ("This line is bold", "Arial", 30, true, false, false);
471         LINE (1);
472         BLOCK ("This line is normal", "Arial", 30, false, false, false);
473         SUB_END ();
474
475         SUB_START (sub::Time::from_hms (0, 0, 2, 308), sub::Time::from_hms (0, 0, 3, 380));
476         LINE (0);
477         BLOCK ("This line is bold", "Arial", 30, true, false, false);
478         LINE (1);
479         BLOCK ("This line is italic", "Arial", 30, false, true, false);
480         SUB_END ();
481
482         SUB_START (sub::Time::from_hms (0, 0, 3, 404), sub::Time::from_hms (0, 0, 4, 484));
483         LINE (0);
484         BLOCK ("This line is italic", "Arial", 30, false, true, false);
485         LINE (1);
486         BLOCK ("This line is bold", "Arial", 30, true, false, false);
487         SUB_END ();
488
489         SUB_START (sub::Time::from_hms (0, 0, 4, 519), sub::Time::from_hms (0, 0, 5, 604));
490         LINE (0);
491         BLOCK ("Last three words are ", "Arial", 30, false, false, false);
492         BLOCK ("bold AND italic", "Arial", 30, true, true, false);
493         LINE (1);
494         BLOCK ("Last three words are ", "Arial", 30, false, false, false);
495         BLOCK ("italic AND bold", "Arial", 30, true, true, false);
496         SUB_END ();
497
498         SUB_START (sub::Time::from_hms (0, 0, 5, 628), sub::Time::from_hms (0, 0, 6, 712));
499         LINE (0);
500         BLOCK ("Last three words are ", "Arial", 30, false, false, false);
501         BLOCK ("bold AND italic", "Arial", 30, true, true, false);
502         LINE (1);
503         BLOCK ("First three words", "Arial", 30, true, true, false);
504         BLOCK (" are italic AND bold", "Arial", 30, false, false, false);
505         SUB_END ();
506
507         SUB_START (sub::Time::from_hms (0, 0, 6, 736), sub::Time::from_hms (0, 0, 8, 31));
508         LINE (0);
509         BLOCK ("Last three words are ", "Arial", 30, false, false, false);
510         BLOCK ("bold AND italic", "Arial", 30, true, true, false);
511         LINE (1);
512         BLOCK ("This line is normal", "Arial", 30, false, false, false);
513         SUB_END ();
514
515         SUB_START (sub::Time::from_hms (0, 0, 8, 94), sub::Time::from_hms (0, 0, 9, 211));
516         LINE (0);
517         BLOCK ("Both lines are bold AND italic", "Arial", 30, true, true, false);
518         LINE (1);
519         BLOCK ("Both lines are bold AND italic", "Arial", 30, true, true, false);
520         SUB_END ();
521 }
522
523 /** Test reading of a .srt file with RTL text */
524 BOOST_AUTO_TEST_CASE (subrip_reader_test4)
525 {
526         boost::filesystem::path p = private_test / "rtl.srt";
527         FILE* f = fopen (p.string().c_str(), "r");
528         sub::SubripReader reader (f);
529         fclose (f);
530         list<sub::Subtitle> subs = sub::collect<std::list<sub::Subtitle> >(reader.subtitles());
531
532         list<sub::Subtitle>::iterator i = subs.begin ();
533         std::cout << i->lines.front().blocks.front().text << "\n";
534
535         std::string const t = i->lines.front().blocks.front().text;
536         for (size_t i = 0; i < t.length() - 2; ++i) {
537                 /* Check that unicode U+202B (right-to-left embedding) has been stripped */
538                 unsigned char const a = t[i];
539                 unsigned char const b = t[i+1];
540                 unsigned char const c = t[i+2];
541                 BOOST_CHECK ((a != 0xe2 || b != 0x80 || c != 0xab));
542         }
543
544         BOOST_CHECK (t == "- \"(دريه فابينار)\"");
545 }
546
547 /** Test <font color="rgba(255,255,255,255)"> */
548 BOOST_AUTO_TEST_CASE (subrip_reader_test5)
549 {
550         sub::RawSubtitle rs;
551         sub::SubripReader r;
552         r.convert_line ("<font color=\"rgba(255,128,64,15)\">Foo bar</font>", rs);
553         BOOST_REQUIRE_EQUAL (r._subs.size(), 1);
554         BOOST_CHECK_EQUAL (r._subs.front().text, "Foo bar");
555         BOOST_CHECK_CLOSE (r._subs.front().colour.r, 255.0 / 255, 0.1);
556         BOOST_CHECK_CLOSE (r._subs.front().colour.g, 128.0 / 255, 0.1);
557         BOOST_CHECK_CLOSE (r._subs.front().colour.b, 64.0 / 255, 0.1);
558         r._subs.clear ();
559
560         rs = sub::RawSubtitle ();
561         r.convert_line ("<font color=\"rgba(1, 2 , 3, 4)\">Foo bar</font>", rs);
562         BOOST_REQUIRE_EQUAL (r._subs.size(), 1);
563         BOOST_CHECK_EQUAL (r._subs.front().text, "Foo bar");
564         BOOST_CHECK_CLOSE (r._subs.front().colour.r, 1.0 / 255, 0.1);
565         BOOST_CHECK_CLOSE (r._subs.front().colour.g, 2.0 / 255, 0.1);
566         BOOST_CHECK_CLOSE (r._subs.front().colour.b, 3.0 / 255, 0.1);
567 }
568
569 /** Test alignment */
570 BOOST_AUTO_TEST_CASE (subrip_reader_test6)
571 {
572         sub::RawSubtitle rs;
573         rs.vertical_position.line = 0;
574         rs.vertical_position.reference = sub::TOP_OF_SUBTITLE;
575         sub::SubripReader r;
576         r.convert_line ("Hello world", rs);
577         BOOST_REQUIRE_EQUAL (r._subs.size(), 1);
578         BOOST_CHECK_EQUAL (r._subs.front().text, "Hello world");
579         BOOST_REQUIRE (r._subs.front().vertical_position.line);
580         BOOST_CHECK_EQUAL (r._subs.front().vertical_position.line.get(), 0);
581         BOOST_REQUIRE (r._subs.front().vertical_position.reference);
582         BOOST_CHECK_EQUAL (r._subs.front().vertical_position.reference.get(), sub::TOP_OF_SUBTITLE);
583         r._subs.clear ();
584
585         rs = sub::RawSubtitle ();
586         rs.vertical_position.line = 0;
587         r.convert_line ("{\\an1}Hello", rs);
588         BOOST_REQUIRE_EQUAL (r._subs.size(), 1);
589         BOOST_CHECK_EQUAL (r._subs.front().text, "Hello");
590         BOOST_REQUIRE (r._subs.front().vertical_position.line);
591         BOOST_CHECK_EQUAL (r._subs.front().vertical_position.line.get(), 0);
592         BOOST_REQUIRE (r._subs.front().vertical_position.reference);
593         BOOST_CHECK_EQUAL (r._subs.front().vertical_position.reference.get(), sub::BOTTOM_OF_SCREEN);
594         BOOST_CHECK_EQUAL (r._subs.front().horizontal_position.proportional, 0);
595         BOOST_CHECK_EQUAL (r._subs.front().horizontal_position.reference, sub::LEFT_OF_SCREEN);
596         r._subs.clear ();
597
598         rs = sub::RawSubtitle ();
599         rs.vertical_position.line = 0;
600         r.convert_line ("{\\an2}to", rs);
601         BOOST_REQUIRE_EQUAL (r._subs.size(), 1);
602         BOOST_CHECK_EQUAL (r._subs.front().text, "to");
603         BOOST_REQUIRE (r._subs.front().vertical_position.line);
604         BOOST_CHECK_EQUAL (r._subs.front().vertical_position.line.get(), 0);
605         BOOST_REQUIRE (r._subs.front().vertical_position.reference);
606         BOOST_CHECK_EQUAL (r._subs.front().vertical_position.reference.get(), sub::BOTTOM_OF_SCREEN);
607         BOOST_CHECK_EQUAL (r._subs.front().horizontal_position.proportional, 0);
608         BOOST_CHECK_EQUAL (r._subs.front().horizontal_position.reference, sub::HORIZONTAL_CENTRE_OF_SCREEN);
609         r._subs.clear ();
610
611         rs = sub::RawSubtitle ();
612         rs.vertical_position.line = 0;
613         r.convert_line ("{\\an3}you", rs);
614         BOOST_CHECK_EQUAL (r._subs.front().text, "you");
615         BOOST_REQUIRE (r._subs.front().vertical_position.line);
616         BOOST_CHECK_EQUAL (r._subs.front().vertical_position.line.get(), 0);
617         BOOST_REQUIRE (r._subs.front().vertical_position.reference);
618         BOOST_CHECK_EQUAL (r._subs.front().vertical_position.reference.get(), sub::BOTTOM_OF_SCREEN);
619         BOOST_CHECK_EQUAL (r._subs.front().horizontal_position.proportional, 0);
620         BOOST_CHECK_EQUAL (r._subs.front().horizontal_position.reference, sub::RIGHT_OF_SCREEN);
621         r._subs.clear ();
622 }
623