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 "subtitle_image.h"
24 #include <boost/test/unit_test.hpp>
29 using boost::shared_ptr;
30 using boost::dynamic_pointer_cast;
32 /** Load some subtitle content from Interop XML and check that it is read correctly */
33 BOOST_AUTO_TEST_CASE (read_interop_subtitle_test1)
35 dcp::InteropSubtitleAsset subs ("test/data/subs1.xml");
37 BOOST_CHECK_EQUAL (subs.id(), "cab5c268-222b-41d2-88ae-6d6999441b17");
38 BOOST_CHECK_EQUAL (subs.movie_title(), "Movie Title");
39 BOOST_CHECK_EQUAL (subs.reel_number(), "1");
40 BOOST_CHECK_EQUAL (subs.language(), "French");
42 list<shared_ptr<dcp::LoadFontNode> > lfn = subs.load_font_nodes ();
43 BOOST_REQUIRE_EQUAL (lfn.size(), 1);
44 shared_ptr<dcp::InteropLoadFontNode> interop_lfn = dynamic_pointer_cast<dcp::InteropLoadFontNode> (lfn.front ());
45 BOOST_REQUIRE (interop_lfn);
46 BOOST_CHECK_EQUAL (interop_lfn->id, "theFontId");
47 BOOST_CHECK_EQUAL (interop_lfn->uri, "arial.ttf");
49 list<shared_ptr<dcp::Subtitle> > s = subs.subtitles_during (dcp::Time (0, 0, 6, 1, 250), dcp::Time (0, 0, 6, 2, 250), false);
50 BOOST_REQUIRE_EQUAL (s.size(), 1);
51 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.front()));
52 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.front()), dcp::SubtitleString (
57 dcp::Colour (255, 255, 255),
60 dcp::Time (0, 0, 5, 198, 250),
61 dcp::Time (0, 0, 7, 115, 250),
67 "My jacket was Idi Amin's",
69 dcp::Colour (0, 0, 0),
70 dcp::Time (0, 0, 0, 1, 250),
71 dcp::Time (0, 0, 0, 1, 250)
74 s = subs.subtitles_during (dcp::Time (0, 0, 7, 190, 250), dcp::Time (0, 0, 7, 191, 250), false);
75 BOOST_REQUIRE_EQUAL (s.size(), 2);
76 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.front()));
77 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.front()), dcp::SubtitleString (
82 dcp::Colour (255, 255, 255),
85 dcp::Time (0, 0, 7, 177, 250),
86 dcp::Time (0, 0, 11, 31, 250),
92 "My corset was H.M. The Queen's",
94 dcp::Colour (0, 0, 0),
95 dcp::Time (0, 0, 0, 1, 250),
96 dcp::Time (0, 0, 0, 1, 250)
98 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.back()));
99 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.back()), dcp::SubtitleString (
100 string ("theFontId"),
104 dcp::Colour (255, 255, 255),
107 dcp::Time (0, 0, 7, 177, 250),
108 dcp::Time (0, 0, 11, 31, 250),
114 "My large wonderbra",
116 dcp::Colour (0, 0, 0),
117 dcp::Time (0, 0, 0, 1, 250),
118 dcp::Time (0, 0, 0, 1, 250)
121 s = subs.subtitles_during (dcp::Time (0, 0, 11, 95, 250), dcp::Time (0, 0, 11, 96, 250), false);
122 BOOST_REQUIRE_EQUAL (s.size(), 1);
123 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.back()));
124 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.back()), dcp::SubtitleString (
125 string ("theFontId"),
129 dcp::Colour (255, 255, 255),
132 dcp::Time (0, 0, 11, 94, 250),
133 dcp::Time (0, 0, 13, 63, 250),
139 "Once belonged to the Shah",
141 dcp::Colour (0, 0, 0),
142 dcp::Time (0, 0, 0, 1, 250),
143 dcp::Time (0, 0, 0, 1, 250)
146 s = subs.subtitles_during (dcp::Time (0, 0, 14, 42, 250), dcp::Time (0, 0, 14, 43, 250), false);
147 BOOST_REQUIRE_EQUAL (s.size(), 1);
148 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.back()));
149 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.back()), dcp::SubtitleString (
150 string ("theFontId"),
154 dcp::Colour (255, 255, 255),
157 dcp::Time (0, 0, 13, 104, 250),
158 dcp::Time (0, 0, 15, 177, 250),
164 "And these are Roy Hattersley's jeans",
166 dcp::Colour (0, 0, 0),
167 dcp::Time (0, 0, 0, 1, 250),
168 dcp::Time (0, 0, 0, 1, 250)
172 /** And similarly for another one */
173 BOOST_AUTO_TEST_CASE (read_interop_subtitle_test2)
175 dcp::InteropSubtitleAsset subs ("test/data/subs2.xml");
177 list<shared_ptr<dcp::Subtitle> > s = subs.subtitles_during (dcp::Time (0, 0, 42, 100, 250), dcp::Time (0, 0, 42, 101, 250), false);
178 BOOST_REQUIRE_EQUAL (s.size(), 2);
179 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.front()));
180 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.front()), dcp::SubtitleString (
185 dcp::Colour (255, 255, 255),
188 dcp::Time (0, 0, 41, 62, 250),
189 dcp::Time (0, 0, 43, 52, 250),
195 "At afternoon tea with John Peel",
197 dcp::Colour (0, 0, 0),
198 dcp::Time (0, 0, 0, 0, 250),
199 dcp::Time (0, 0, 0, 0, 250)
201 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.back()));
202 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.back()), dcp::SubtitleString (
207 dcp::Colour (255, 255, 255),
210 dcp::Time (0, 0, 41, 62, 250),
211 dcp::Time (0, 0, 43, 52, 250),
217 "I enquired if his accent was real",
219 dcp::Colour (0, 0, 0),
220 dcp::Time (0, 0, 0, 0, 250),
221 dcp::Time (0, 0, 0, 0, 250)
224 s = subs.subtitles_during (dcp::Time (0, 0, 50, 50, 250), dcp::Time (0, 0, 50, 51, 250), false);
225 BOOST_REQUIRE_EQUAL (s.size(), 2);
226 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.front()));
227 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.front()), dcp::SubtitleString (
232 dcp::Colour (255, 255, 255),
235 dcp::Time (0, 0, 50, 42, 250),
236 dcp::Time (0, 0, 52, 21, 250),
242 "He said \"out of the house",
244 dcp::Colour (0, 0, 0),
245 dcp::Time (0, 0, 0, 0, 250),
246 dcp::Time (0, 0, 0, 0, 250)
248 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.back()));
249 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.back()), dcp::SubtitleString (
254 dcp::Colour (255, 255, 255),
257 dcp::Time (0, 0, 50, 42, 250),
258 dcp::Time (0, 0, 52, 21, 250),
264 "I'm incredibly scouse",
266 dcp::Colour (0, 0, 0),
267 dcp::Time (0, 0, 0, 0, 250),
268 dcp::Time (0, 0, 0, 0, 250)
271 s = subs.subtitles_during (dcp::Time (0, 1, 2, 300, 250), dcp::Time (0, 1, 2, 301, 250), false);
272 BOOST_REQUIRE_EQUAL (s.size(), 2);
273 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.front()));
274 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.front()), dcp::SubtitleString (
279 dcp::Colour (255, 255, 255),
282 dcp::Time (0, 1, 2, 208, 250),
283 dcp::Time (0, 1, 4, 10, 250),
289 "At home it depends how I feel.\"",
291 dcp::Colour (0, 0, 0),
292 dcp::Time (0, 0, 0, 0, 250),
293 dcp::Time (0, 0, 0, 0, 250)
295 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.back()));
296 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.back()), dcp::SubtitleString (
301 dcp::Colour (255, 255, 255),
304 dcp::Time (0, 1, 2, 208, 250),
305 dcp::Time (0, 1, 4, 10, 250),
311 "I spent a long weekend in Brighton",
313 dcp::Colour (0, 0, 0),
314 dcp::Time (0, 0, 0, 0, 250),
315 dcp::Time (0, 0, 0, 0, 250)
318 s = subs.subtitles_during (dcp::Time (0, 1, 15, 50, 250), dcp::Time (0, 1, 15, 51, 250), false);
319 BOOST_REQUIRE_EQUAL (s.size(), 2);
320 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.front()));
321 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.front()), dcp::SubtitleString (
326 dcp::Colour (255, 255, 255),
329 dcp::Time (0, 1, 15, 42, 250),
330 dcp::Time (0, 1, 16, 42, 250),
336 "With the legendary Miss Enid Blyton",
338 dcp::Colour (0, 0, 0),
339 dcp::Time (0, 0, 0, 0, 250),
340 dcp::Time (0, 0, 0, 0, 250)
342 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.back()));
343 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.back()), dcp::SubtitleString (
348 dcp::Colour (255, 255, 255),
351 dcp::Time (0, 1, 15, 42, 250),
352 dcp::Time (0, 1, 16, 42, 250),
358 "She said \"you be Noddy",
360 dcp::Colour (0, 0, 0),
361 dcp::Time (0, 0, 0, 0, 250),
362 dcp::Time (0, 0, 0, 0, 250)
365 s = subs.subtitles_during (dcp::Time (0, 1, 27, 200, 250), dcp::Time (0, 1, 27, 201, 250), false);
366 BOOST_REQUIRE_EQUAL (s.size(), 2);
367 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.front()));
368 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.front()), dcp::SubtitleString (
373 dcp::Colour (255, 255, 255),
376 dcp::Time (0, 1, 27, 115, 250),
377 dcp::Time (0, 1, 28, 208, 250),
383 "That curious creature the Sphinx",
385 dcp::Colour (0, 0, 0),
386 dcp::Time (0, 0, 0, 0, 250),
387 dcp::Time (0, 0, 0, 0, 250)
389 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.back()));
390 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.back()), dcp::SubtitleString (
395 dcp::Colour (255, 255, 255),
398 dcp::Time (0, 1, 27, 115, 250),
399 dcp::Time (0, 1, 28, 208, 250),
405 "Is smarter than anyone thinks",
407 dcp::Colour (0, 0, 0),
408 dcp::Time (0, 0, 0, 0, 250),
409 dcp::Time (0, 0, 0, 0, 250)
412 s = subs.subtitles_during (dcp::Time (0, 1, 42, 300, 250), dcp::Time (0, 1, 42, 301, 250), false);
413 BOOST_REQUIRE_EQUAL (s.size(), 2);
414 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.front()));
415 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.front()), dcp::SubtitleString (
420 dcp::Colour (255, 255, 255),
423 dcp::Time (0, 1, 42, 229, 250),
424 dcp::Time (0, 1, 45, 62, 250),
430 "It sits there and smirks",
432 dcp::Colour (0, 0, 0),
433 dcp::Time (0, 0, 0, 0, 250),
434 dcp::Time (0, 0, 0, 0, 250)
436 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.back()));
437 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.back()), dcp::SubtitleString (
442 dcp::Colour (255, 255, 255),
445 dcp::Time (0, 1, 42, 229, 250),
446 dcp::Time (0, 1, 45, 62, 250),
452 "And you don't think it works",
454 dcp::Colour (0, 0, 0),
455 dcp::Time (0, 0, 0, 0, 250),
456 dcp::Time (0, 0, 0, 0, 250)
459 s = subs.subtitles_during (dcp::Time (0, 1, 45, 200, 250), dcp::Time (0, 1, 45, 201, 250), false);
460 BOOST_REQUIRE_EQUAL (s.size(), 2);
461 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.front()));
462 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.front()), dcp::SubtitleString (
467 dcp::Colour (255, 255, 255),
470 dcp::Time (0, 1, 45, 146, 250),
471 dcp::Time (0, 1, 47, 94, 250),
477 "Then when you're not looking, it winks.",
479 dcp::Colour (0, 0, 0),
480 dcp::Time (0, 0, 0, 0, 250),
481 dcp::Time (0, 0, 0, 0, 250)
483 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.back()));
484 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.back()), dcp::SubtitleString (
489 dcp::Colour (255, 255, 255),
492 dcp::Time (0, 1, 45, 146, 250),
493 dcp::Time (0, 1, 47, 94, 250),
499 "When it snows you will find Sister Sledge",
501 dcp::Colour (0, 0, 0),
502 dcp::Time (0, 0, 0, 0, 250),
503 dcp::Time (0, 0, 0, 0, 250)
506 s = subs.subtitles_during (dcp::Time (0, 1, 47, 249, 250), dcp::Time (0, 1, 47, 250, 250), false);
507 BOOST_REQUIRE_EQUAL (s.size(), 2);
508 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.front()));
509 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.front()), dcp::SubtitleString (
514 dcp::Colour (255, 255, 255),
517 dcp::Time (0, 1, 47, 146, 250),
518 dcp::Time (0, 1, 48, 167, 250),
524 "Out mooning, at night, on the ledge",
526 dcp::Colour (0, 0, 0),
527 dcp::Time (0, 0, 0, 0, 250),
528 dcp::Time (0, 0, 0, 0, 250)
530 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.back()));
531 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.back()), dcp::SubtitleString (
536 dcp::Colour (255, 255, 255),
539 dcp::Time (0, 1, 47, 146, 250),
540 dcp::Time (0, 1, 48, 167, 250),
548 dcp::Colour (0, 0, 0),
549 dcp::Time (0, 0, 0, 0, 250),
550 dcp::Time (0, 0, 0, 0, 250)
553 s = subs.subtitles_during (dcp::Time (0, 2, 6, 210, 250), dcp::Time (0, 2, 6, 211, 250), false);
554 BOOST_REQUIRE_EQUAL (s.size(), 2);
555 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.front()));
556 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.front()), dcp::SubtitleString (
561 dcp::Colour (255, 255, 255),
564 dcp::Time (0, 2, 5, 208, 250),
565 dcp::Time (0, 2, 7, 31, 250),
573 dcp::Colour (0, 0, 0),
574 dcp::Time (0, 0, 0, 0, 250),
575 dcp::Time (0, 0, 0, 0, 250)
577 BOOST_REQUIRE (dynamic_pointer_cast<dcp::SubtitleString>(s.back()));
578 BOOST_CHECK_EQUAL (*dynamic_pointer_cast<dcp::SubtitleString>(s.back()), dcp::SubtitleString (
583 dcp::Colour (255, 255, 255),
586 dcp::Time (0, 2, 5, 208, 250),
587 dcp::Time (0, 2, 7, 31, 250),
595 dcp::Colour (0, 0, 0),
596 dcp::Time (0, 0, 0, 0, 250),
597 dcp::Time (0, 0, 0, 0, 250)
601 /** And one with bitmap subtitles */
602 BOOST_AUTO_TEST_CASE (read_interop_subtitle_test3)
604 dcp::InteropSubtitleAsset subs ("test/data/subs3.xml");
606 BOOST_REQUIRE_EQUAL (subs.subtitles().size(), 1);
607 shared_ptr<dcp::SubtitleImage> si = dynamic_pointer_cast<dcp::SubtitleImage>(subs.subtitles().front());
609 BOOST_CHECK (si->png_image() == dcp::Data("test/data/sub.png"));