2 Copyright (C) 2012-2018 Carl Hetherington <cth@carlh.net>
4 This file is part of libdcp.
6 libdcp is free software; you can redistribute it and/or modify
7 it under the terms of the GNU General Public License as published by
8 the Free Software Foundation; either version 2 of the License, or
9 (at your option) any later version.
11 libdcp is distributed in the hope that it will be useful,
12 but WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 GNU General Public License for more details.
16 You should have received a copy of the GNU General Public License
17 along with libdcp. If not, see <http://www.gnu.org/licenses/>.
20 #include "interop_subtitle_asset.h"
21 #include "interop_load_font_node.h"
22 #include "subtitle_string.h"
23 #include <boost/test/unit_test.hpp>
27 using boost::shared_ptr;
28 using boost::dynamic_pointer_cast;
30 /** Load some subtitle content from Interop XML and check that it is read correctly */
31 BOOST_AUTO_TEST_CASE (read_interop_subtitle_test1)
33 dcp::InteropSubtitleAsset subs ("test/data/subs1.xml");
35 BOOST_CHECK_EQUAL (subs.id(), "cab5c268-222b-41d2-88ae-6d6999441b17");
36 BOOST_CHECK_EQUAL (subs.movie_title(), "Movie Title");
37 BOOST_CHECK_EQUAL (subs.reel_number(), "1");
38 BOOST_CHECK_EQUAL (subs.language(), "French");
40 list<shared_ptr<dcp::LoadFontNode> > lfn = subs.load_font_nodes ();
41 BOOST_REQUIRE_EQUAL (lfn.size(), 1);
42 shared_ptr<dcp::InteropLoadFontNode> interop_lfn = dynamic_pointer_cast<dcp::InteropLoadFontNode> (lfn.front ());
43 BOOST_REQUIRE (interop_lfn);
44 BOOST_CHECK_EQUAL (interop_lfn->id, "theFontId");
45 BOOST_CHECK_EQUAL (interop_lfn->uri, "arial.ttf");
47 list<shared_ptr<dcp::Subtitle> > s = subs.subtitles_during (dcp::Time (0, 0, 6, 1, 250), dcp::Time (0, 0, 6, 2, 250), false);
48 BOOST_REQUIRE_EQUAL (s.size(), 1);
49 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.front()));
50 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.front()), dcp::SubtitleString (
55 dcp::Colour (255, 255, 255),
58 dcp::Time (0, 0, 5, 198, 250),
59 dcp::Time (0, 0, 7, 115, 250),
65 "My jacket was Idi Amin's",
67 dcp::Colour (0, 0, 0),
68 dcp::Time (0, 0, 0, 1, 250),
69 dcp::Time (0, 0, 0, 1, 250)
72 s = subs.subtitles_during (dcp::Time (0, 0, 7, 190, 250), dcp::Time (0, 0, 7, 191, 250), false);
73 BOOST_REQUIRE_EQUAL (s.size(), 2);
74 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.front()));
75 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.front()), dcp::SubtitleString (
80 dcp::Colour (255, 255, 255),
83 dcp::Time (0, 0, 7, 177, 250),
84 dcp::Time (0, 0, 11, 31, 250),
90 "My corset was H.M. The Queen's",
92 dcp::Colour (0, 0, 0),
93 dcp::Time (0, 0, 0, 1, 250),
94 dcp::Time (0, 0, 0, 1, 250)
96 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.back()));
97 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.back()), dcp::SubtitleString (
102 dcp::Colour (255, 255, 255),
105 dcp::Time (0, 0, 7, 177, 250),
106 dcp::Time (0, 0, 11, 31, 250),
112 "My large wonderbra",
114 dcp::Colour (0, 0, 0),
115 dcp::Time (0, 0, 0, 1, 250),
116 dcp::Time (0, 0, 0, 1, 250)
119 s = subs.subtitles_during (dcp::Time (0, 0, 11, 95, 250), dcp::Time (0, 0, 11, 96, 250), false);
120 BOOST_REQUIRE_EQUAL (s.size(), 1);
121 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.back()));
122 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.back()), dcp::SubtitleString (
123 string ("theFontId"),
127 dcp::Colour (255, 255, 255),
130 dcp::Time (0, 0, 11, 94, 250),
131 dcp::Time (0, 0, 13, 63, 250),
137 "Once belonged to the Shah",
139 dcp::Colour (0, 0, 0),
140 dcp::Time (0, 0, 0, 1, 250),
141 dcp::Time (0, 0, 0, 1, 250)
144 s = subs.subtitles_during (dcp::Time (0, 0, 14, 42, 250), dcp::Time (0, 0, 14, 43, 250), false);
145 BOOST_REQUIRE_EQUAL (s.size(), 1);
146 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.back()));
147 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.back()), dcp::SubtitleString (
148 string ("theFontId"),
152 dcp::Colour (255, 255, 255),
155 dcp::Time (0, 0, 13, 104, 250),
156 dcp::Time (0, 0, 15, 177, 250),
162 "And these are Roy Hattersley's jeans",
164 dcp::Colour (0, 0, 0),
165 dcp::Time (0, 0, 0, 1, 250),
166 dcp::Time (0, 0, 0, 1, 250)
170 /** And similarly for another one */
171 BOOST_AUTO_TEST_CASE (read_interop_subtitle_test2)
173 dcp::InteropSubtitleAsset subs ("test/data/subs2.xml");
175 list<shared_ptr<dcp::Subtitle> > s = subs.subtitles_during (dcp::Time (0, 0, 42, 100, 250), dcp::Time (0, 0, 42, 101, 250), false);
176 BOOST_REQUIRE_EQUAL (s.size(), 2);
177 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.front()));
178 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.front()), dcp::SubtitleString (
183 dcp::Colour (255, 255, 255),
186 dcp::Time (0, 0, 41, 62, 250),
187 dcp::Time (0, 0, 43, 52, 250),
193 "At afternoon tea with John Peel",
195 dcp::Colour (0, 0, 0),
196 dcp::Time (0, 0, 0, 0, 250),
197 dcp::Time (0, 0, 0, 0, 250)
199 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.back()));
200 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.back()), dcp::SubtitleString (
205 dcp::Colour (255, 255, 255),
208 dcp::Time (0, 0, 41, 62, 250),
209 dcp::Time (0, 0, 43, 52, 250),
215 "I enquired if his accent was real",
217 dcp::Colour (0, 0, 0),
218 dcp::Time (0, 0, 0, 0, 250),
219 dcp::Time (0, 0, 0, 0, 250)
222 s = subs.subtitles_during (dcp::Time (0, 0, 50, 50, 250), dcp::Time (0, 0, 50, 51, 250), false);
223 BOOST_REQUIRE_EQUAL (s.size(), 2);
224 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.front()));
225 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.front()), dcp::SubtitleString (
230 dcp::Colour (255, 255, 255),
233 dcp::Time (0, 0, 50, 42, 250),
234 dcp::Time (0, 0, 52, 21, 250),
240 "He said \"out of the house",
242 dcp::Colour (0, 0, 0),
243 dcp::Time (0, 0, 0, 0, 250),
244 dcp::Time (0, 0, 0, 0, 250)
246 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.back()));
247 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.back()), dcp::SubtitleString (
252 dcp::Colour (255, 255, 255),
255 dcp::Time (0, 0, 50, 42, 250),
256 dcp::Time (0, 0, 52, 21, 250),
262 "I'm incredibly scouse",
264 dcp::Colour (0, 0, 0),
265 dcp::Time (0, 0, 0, 0, 250),
266 dcp::Time (0, 0, 0, 0, 250)
269 s = subs.subtitles_during (dcp::Time (0, 1, 2, 300, 250), dcp::Time (0, 1, 2, 301, 250), false);
270 BOOST_REQUIRE_EQUAL (s.size(), 2);
271 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.front()));
272 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.front()), dcp::SubtitleString (
277 dcp::Colour (255, 255, 255),
280 dcp::Time (0, 1, 2, 208, 250),
281 dcp::Time (0, 1, 4, 10, 250),
287 "At home it depends how I feel.\"",
289 dcp::Colour (0, 0, 0),
290 dcp::Time (0, 0, 0, 0, 250),
291 dcp::Time (0, 0, 0, 0, 250)
293 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.back()));
294 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.back()), dcp::SubtitleString (
299 dcp::Colour (255, 255, 255),
302 dcp::Time (0, 1, 2, 208, 250),
303 dcp::Time (0, 1, 4, 10, 250),
309 "I spent a long weekend in Brighton",
311 dcp::Colour (0, 0, 0),
312 dcp::Time (0, 0, 0, 0, 250),
313 dcp::Time (0, 0, 0, 0, 250)
316 s = subs.subtitles_during (dcp::Time (0, 1, 15, 50, 250), dcp::Time (0, 1, 15, 51, 250), false);
317 BOOST_REQUIRE_EQUAL (s.size(), 2);
318 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.front()));
319 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.front()), dcp::SubtitleString (
324 dcp::Colour (255, 255, 255),
327 dcp::Time (0, 1, 15, 42, 250),
328 dcp::Time (0, 1, 16, 42, 250),
334 "With the legendary Miss Enid Blyton",
336 dcp::Colour (0, 0, 0),
337 dcp::Time (0, 0, 0, 0, 250),
338 dcp::Time (0, 0, 0, 0, 250)
340 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.back()));
341 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.back()), dcp::SubtitleString (
346 dcp::Colour (255, 255, 255),
349 dcp::Time (0, 1, 15, 42, 250),
350 dcp::Time (0, 1, 16, 42, 250),
356 "She said \"you be Noddy",
358 dcp::Colour (0, 0, 0),
359 dcp::Time (0, 0, 0, 0, 250),
360 dcp::Time (0, 0, 0, 0, 250)
363 s = subs.subtitles_during (dcp::Time (0, 1, 27, 200, 250), dcp::Time (0, 1, 27, 201, 250), false);
364 BOOST_REQUIRE_EQUAL (s.size(), 2);
365 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.front()));
366 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.front()), dcp::SubtitleString (
371 dcp::Colour (255, 255, 255),
374 dcp::Time (0, 1, 27, 115, 250),
375 dcp::Time (0, 1, 28, 208, 250),
381 "That curious creature the Sphinx",
383 dcp::Colour (0, 0, 0),
384 dcp::Time (0, 0, 0, 0, 250),
385 dcp::Time (0, 0, 0, 0, 250)
387 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.back()));
388 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.back()), dcp::SubtitleString (
393 dcp::Colour (255, 255, 255),
396 dcp::Time (0, 1, 27, 115, 250),
397 dcp::Time (0, 1, 28, 208, 250),
403 "Is smarter than anyone thinks",
405 dcp::Colour (0, 0, 0),
406 dcp::Time (0, 0, 0, 0, 250),
407 dcp::Time (0, 0, 0, 0, 250)
410 s = subs.subtitles_during (dcp::Time (0, 1, 42, 300, 250), dcp::Time (0, 1, 42, 301, 250), false);
411 BOOST_REQUIRE_EQUAL (s.size(), 2);
412 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.front()));
413 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.front()), dcp::SubtitleString (
418 dcp::Colour (255, 255, 255),
421 dcp::Time (0, 1, 42, 229, 250),
422 dcp::Time (0, 1, 45, 62, 250),
428 "It sits there and smirks",
430 dcp::Colour (0, 0, 0),
431 dcp::Time (0, 0, 0, 0, 250),
432 dcp::Time (0, 0, 0, 0, 250)
434 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.back()));
435 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.back()), dcp::SubtitleString (
440 dcp::Colour (255, 255, 255),
443 dcp::Time (0, 1, 42, 229, 250),
444 dcp::Time (0, 1, 45, 62, 250),
450 "And you don't think it works",
452 dcp::Colour (0, 0, 0),
453 dcp::Time (0, 0, 0, 0, 250),
454 dcp::Time (0, 0, 0, 0, 250)
457 s = subs.subtitles_during (dcp::Time (0, 1, 45, 200, 250), dcp::Time (0, 1, 45, 201, 250), false);
458 BOOST_REQUIRE_EQUAL (s.size(), 2);
459 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.front()));
460 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.front()), dcp::SubtitleString (
465 dcp::Colour (255, 255, 255),
468 dcp::Time (0, 1, 45, 146, 250),
469 dcp::Time (0, 1, 47, 94, 250),
475 "Then when you're not looking, it winks.",
477 dcp::Colour (0, 0, 0),
478 dcp::Time (0, 0, 0, 0, 250),
479 dcp::Time (0, 0, 0, 0, 250)
481 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.back()));
482 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.back()), dcp::SubtitleString (
487 dcp::Colour (255, 255, 255),
490 dcp::Time (0, 1, 45, 146, 250),
491 dcp::Time (0, 1, 47, 94, 250),
497 "When it snows you will find Sister Sledge",
499 dcp::Colour (0, 0, 0),
500 dcp::Time (0, 0, 0, 0, 250),
501 dcp::Time (0, 0, 0, 0, 250)
504 s = subs.subtitles_during (dcp::Time (0, 1, 47, 249, 250), dcp::Time (0, 1, 47, 250, 250), false);
505 BOOST_REQUIRE_EQUAL (s.size(), 2);
506 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.front()));
507 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.front()), dcp::SubtitleString (
512 dcp::Colour (255, 255, 255),
515 dcp::Time (0, 1, 47, 146, 250),
516 dcp::Time (0, 1, 48, 167, 250),
522 "Out mooning, at night, on the ledge",
524 dcp::Colour (0, 0, 0),
525 dcp::Time (0, 0, 0, 0, 250),
526 dcp::Time (0, 0, 0, 0, 250)
528 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.back()));
529 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.back()), dcp::SubtitleString (
534 dcp::Colour (255, 255, 255),
537 dcp::Time (0, 1, 47, 146, 250),
538 dcp::Time (0, 1, 48, 167, 250),
546 dcp::Colour (0, 0, 0),
547 dcp::Time (0, 0, 0, 0, 250),
548 dcp::Time (0, 0, 0, 0, 250)
551 s = subs.subtitles_during (dcp::Time (0, 2, 6, 210, 250), dcp::Time (0, 2, 6, 211, 250), false);
552 BOOST_REQUIRE_EQUAL (s.size(), 2);
553 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.front()));
554 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.front()), dcp::SubtitleString (
559 dcp::Colour (255, 255, 255),
562 dcp::Time (0, 2, 5, 208, 250),
563 dcp::Time (0, 2, 7, 31, 250),
571 dcp::Colour (0, 0, 0),
572 dcp::Time (0, 0, 0, 0, 250),
573 dcp::Time (0, 0, 0, 0, 250)
575 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.back()));
576 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.back()), dcp::SubtitleString (
581 dcp::Colour (255, 255, 255),
584 dcp::Time (0, 2, 5, 208, 250),
585 dcp::Time (0, 2, 7, 31, 250),
593 dcp::Colour (0, 0, 0),
594 dcp::Time (0, 0, 0, 0, 250),
595 dcp::Time (0, 0, 0, 0, 250)