/* Copyright (C) 2015-2018 Carl Hetherington This file is part of libdcp. libdcp is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. libdcp is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with libdcp. If not, see . */ #include "interop_subtitle_asset.h" #include "smpte_subtitle_asset.h" #include "subtitle_string.h" #include "subtitle_image.h" #include "subtitle_asset_internal.h" #include "reel_subtitle_asset.h" #include "reel.h" #include "cpl.h" #include "dcp.h" #include "test.h" #include "util.h" #include using std::list; using std::string; using boost::shared_ptr; /** Test dcp::order::Font::take_intersection */ BOOST_AUTO_TEST_CASE (take_intersection_test) { dcp::order::Font A; A._values["foo"] = "bar"; A._values["fred"] = "jim"; dcp::order::Font B; B._values["foo"] = "bar"; B._values["sheila"] = "baz"; A.take_intersection (B); BOOST_REQUIRE_EQUAL (A._values.size(), 1); BOOST_CHECK_EQUAL (A._values["foo"], "bar"); A._values.clear (); B._values.clear (); A._values["foo"] = "bar"; A._values["fred"] = "jim"; B._values["foo"] = "hello"; B._values["sheila"] = "baz"; A.take_intersection (B); BOOST_CHECK_EQUAL (A._values.size(), 0); } /** Test dcp::order::Font::take_difference */ BOOST_AUTO_TEST_CASE (take_difference_test) { dcp::order::Font A; A._values["foo"] = "bar"; A._values["fred"] = "jim"; dcp::order::Font B; B._values["foo"] = "bar"; B._values["sheila"] = "baz"; A.take_difference (B); BOOST_REQUIRE_EQUAL (A._values.size(), 1); BOOST_CHECK_EQUAL (A._values["fred"], "jim"); } /** Test dcp::order::Subtitle::pull_fonts */ BOOST_AUTO_TEST_CASE (pull_fonts_test1) { shared_ptr root (new dcp::order::Part (shared_ptr ())); shared_ptr sub1 (new dcp::order::Subtitle (root, dcp::Time(), dcp::Time(), dcp::Time(), dcp::Time())); root->children.push_back (sub1); shared_ptr text1 (new dcp::order::Text (sub1, dcp::HALIGN_CENTER, 0, dcp::VALIGN_TOP, 0, dcp::DIRECTION_LTR)); sub1->children.push_back (text1); text1->font._values["font"] = "Inconsolata"; text1->font._values["size"] = "42"; dcp::SubtitleAsset::pull_fonts (root); BOOST_REQUIRE_EQUAL (sub1->font._values.size(), 2); BOOST_CHECK_EQUAL (sub1->font._values["font"], "Inconsolata"); BOOST_CHECK_EQUAL (sub1->font._values["size"], "42"); BOOST_CHECK_EQUAL (text1->font._values.size(), 0); } /** Test dcp::order::Subtitle::pull_fonts */ BOOST_AUTO_TEST_CASE (pull_fonts_test2) { shared_ptr root (new dcp::order::Part (shared_ptr ())); shared_ptr sub1 (new dcp::order::Subtitle (root, dcp::Time(), dcp::Time(), dcp::Time(), dcp::Time())); root->children.push_back (sub1); shared_ptr text1 (new dcp::order::Text (sub1, dcp::HALIGN_CENTER, 0, dcp::VALIGN_TOP, 0, dcp::DIRECTION_LTR)); sub1->children.push_back (text1); text1->font._values["font"] = "Inconsolata"; text1->font._values["size"] = "42"; shared_ptr text2 (new dcp::order::Text (sub1, dcp::HALIGN_CENTER, 0, dcp::VALIGN_TOP, 0, dcp::DIRECTION_LTR)); sub1->children.push_back (text2); text2->font._values["font"] = "Inconsolata"; text2->font._values["size"] = "48"; dcp::SubtitleAsset::pull_fonts (root); BOOST_REQUIRE_EQUAL (sub1->font._values.size(), 1); BOOST_CHECK_EQUAL (sub1->font._values["font"], "Inconsolata"); BOOST_REQUIRE_EQUAL (text1->font._values.size(), 1); BOOST_CHECK_EQUAL (text1->font._values["size"], "42"); BOOST_REQUIRE_EQUAL (text2->font._values.size(), 1); BOOST_CHECK_EQUAL (text2->font._values["size"], "48"); } /** Test dcp::order::Subtitle::pull_fonts */ BOOST_AUTO_TEST_CASE (pull_fonts_test3) { shared_ptr root (new dcp::order::Part (shared_ptr ())); shared_ptr sub1 (new dcp::order::Subtitle (root, dcp::Time(), dcp::Time(), dcp::Time(), dcp::Time())); root->children.push_back (sub1); shared_ptr text1 (new dcp::order::Text (sub1, dcp::HALIGN_CENTER, 0, dcp::VALIGN_TOP, 0, dcp::DIRECTION_LTR)); sub1->children.push_back (text1); dcp::order::Font font; font._values["font"] = "Inconsolata"; font._values["size"] = "42"; shared_ptr string1 (new dcp::order::String (text1, font, "Hello world")); text1->children.push_back (string1); dcp::SubtitleAsset::pull_fonts (root); BOOST_REQUIRE_EQUAL (sub1->font._values.size(), 2); BOOST_CHECK_EQUAL (sub1->font._values["font"], "Inconsolata"); BOOST_CHECK_EQUAL (sub1->font._values["size"], "42"); } /** Write some subtitle content as Interop XML and check that it is right */ BOOST_AUTO_TEST_CASE (write_interop_subtitle_test) { dcp::InteropSubtitleAsset c; c.set_reel_number ("1"); c.set_language ("EN"); c.set_movie_title ("Test"); c.add ( shared_ptr ( new dcp::SubtitleString ( string ("Frutiger"), false, false, false, dcp::Colour (255, 255, 255), 48, 1.0, dcp::Time (0, 4, 9, 22, 24), dcp::Time (0, 4, 11, 22, 24), 0, dcp::HALIGN_CENTER, 0.8, dcp::VALIGN_TOP, dcp::DIRECTION_LTR, "Hello world", dcp::NONE, dcp::Colour (0, 0, 0), dcp::Time (0, 0, 0, 0, 24), dcp::Time (0, 0, 0, 0, 24) ) ) ); c.add ( shared_ptr ( new dcp::SubtitleString ( boost::optional (), true, true, true, dcp::Colour (128, 0, 64), 91, 1.0, dcp::Time (5, 41, 0, 21, 24), dcp::Time (6, 12, 15, 21, 24), 0, dcp::HALIGN_CENTER, 0.4, dcp::VALIGN_BOTTOM, dcp::DIRECTION_LTR, "What's going on", dcp::BORDER, dcp::Colour (1, 2, 3), dcp::Time (1, 2, 3, 4, 24), dcp::Time (5, 6, 7, 8, 24) ) ) ); c._id = "a6c58cff-3e1e-4b38-acec-a42224475ef6"; check_xml ( "" "a6c58cff-3e1e-4b38-acec-a42224475ef6" "Test" "1" "EN" "" "" "Hello world" "" "" "" "" "What's going on" "" "" "", c.xml_as_string (), list () ); } /** Write some subtitle content as Interop XML and check that it is right. * This test includes some horizontal alignment. */ BOOST_AUTO_TEST_CASE (write_interop_subtitle_test2) { dcp::InteropSubtitleAsset c; c.set_reel_number ("1"); c.set_language ("EN"); c.set_movie_title ("Test"); c.add ( shared_ptr ( new dcp::SubtitleString ( string ("Frutiger"), false, false, false, dcp::Colour (255, 255, 255), 48, 1.0, dcp::Time (0, 4, 9, 22, 24), dcp::Time (0, 4, 11, 22, 24), -0.2, dcp::HALIGN_CENTER, 0.8, dcp::VALIGN_TOP, dcp::DIRECTION_LTR, "Hello world", dcp::NONE, dcp::Colour (0, 0, 0), dcp::Time (0, 0, 0, 0, 24), dcp::Time (0, 0, 0, 0, 24) ) ) ); c.add ( shared_ptr ( new dcp::SubtitleString ( boost::optional (), true, true, true, dcp::Colour (128, 0, 64), 91, 1.0, dcp::Time (5, 41, 0, 21, 24), dcp::Time (6, 12, 15, 21, 24), -0.2, dcp::HALIGN_CENTER, 0.4, dcp::VALIGN_BOTTOM, dcp::DIRECTION_LTR, "What's going on", dcp::BORDER, dcp::Colour (1, 2, 3), dcp::Time (1, 2, 3, 4, 24), dcp::Time (5, 6, 7, 8, 24) ) ) ); c._id = "a6c58cff-3e1e-4b38-acec-a42224475ef6"; check_xml ( "" "a6c58cff-3e1e-4b38-acec-a42224475ef6" "Test" "1" "EN" "" "" "Hello world" "" "" "" "" "What's going on" "" "" "", c.xml_as_string (), list () ); } /* Write some subtitle content as Interop XML using bitmaps and check that it is right */ BOOST_AUTO_TEST_CASE (write_interop_subtitle_test3) { shared_ptr c (new dcp::InteropSubtitleAsset()); c->set_reel_number ("1"); c->set_language ("EN"); c->set_movie_title ("Test"); c->add ( shared_ptr ( new dcp::SubtitleImage ( dcp::Data ("test/data/sub.png"), dcp::Time (0, 4, 9, 22, 24), dcp::Time (0, 4, 11, 22, 24), 0, dcp::HALIGN_CENTER, 0.8, dcp::VALIGN_TOP, dcp::Time (0, 0, 0, 0, 24), dcp::Time (0, 0, 0, 0, 24) ) ) ); c->_id = "a6c58cff-3e1e-4b38-acec-a42224475ef6"; boost::filesystem::create_directories ("build/test/write_interop_subtitle_test3"); c->write ("build/test/write_interop_subtitle_test3/subs.xml"); shared_ptr reel (new dcp::Reel()); reel->add(shared_ptr(new dcp::ReelSubtitleAsset(c, dcp::Fraction(24, 1), 6046, 0))); dcp::XMLMetadata xml_meta; xml_meta.issue_date = "2018-09-02T04:45:18+00:00"; shared_ptr cpl (new dcp::CPL ("My film", dcp::FEATURE)); cpl->add (reel); cpl->set_metadata (xml_meta); cpl->set_content_version_label_text ("foo"); dcp::DCP dcp ("build/test/write_interop_subtitle_test3"); dcp.add (cpl); dcp.write_xml (dcp::INTEROP, xml_meta); check_xml ( dcp::file_to_string("test/ref/write_interop_subtitle_test3/subs.xml"), dcp::file_to_string("build/test/write_interop_subtitle_test3/subs.xml"), list() ); check_file ("build/test/write_interop_subtitle_test3/822bd341-c751-45b1-94d2-410e4ffcff1b.png", "test/data/sub.png"); check_xml ( dcp::file_to_string("test/ref/write_interop_subtitle_test3/ASSETMAP"), dcp::file_to_string("build/test/write_interop_subtitle_test3/ASSETMAP"), list() ); check_xml ( dcp::file_to_string("test/ref/write_interop_subtitle_test3/pkl.xml"), dcp::file_to_string("build/test/write_interop_subtitle_test3/pkl_8d98d2e1-d2a1-458f-b96b-295e5b5d0860.xml"), list() ); } /* Write some subtitle content as SMPTE XML and check that it is right */ BOOST_AUTO_TEST_CASE (write_smpte_subtitle_test) { dcp::SMPTESubtitleAsset c; c.set_reel_number (1); c.set_language ("EN"); c.set_content_title_text ("Test"); c.set_issue_date (dcp::LocalTime ("2016-04-01T03:52:00+00:00")); c.add ( shared_ptr ( new dcp::SubtitleString ( string ("Frutiger"), false, false, false, dcp::Colour (255, 255, 255), 48, 1.0, dcp::Time (0, 4, 9, 22, 24), dcp::Time (0, 4, 11, 22, 24), 0, dcp::HALIGN_CENTER, 0.8, dcp::VALIGN_TOP, dcp::DIRECTION_LTR, "Hello world", dcp::NONE, dcp::Colour (0, 0, 0), dcp::Time (0, 0, 0, 0, 24), dcp::Time (0, 0, 0, 0, 24) ) ) ); c.add ( shared_ptr ( new dcp::SubtitleString ( boost::optional (), true, true, true, dcp::Colour (128, 0, 64), 91, 1.0, dcp::Time (5, 41, 0, 21, 24), dcp::Time (6, 12, 15, 21, 24), 0, dcp::HALIGN_CENTER, 0.4, dcp::VALIGN_BOTTOM, dcp::DIRECTION_RTL, "What's going on", dcp::BORDER, dcp::Colour (1, 2, 3), dcp::Time (1, 2, 3, 4, 24), dcp::Time (5, 6, 7, 8, 24) ) ) ); c._xml_id = "a6c58cff-3e1e-4b38-acec-a42224475ef6"; check_xml ( "" "" "urn:uuid:a6c58cff-3e1e-4b38-acec-a42224475ef6" "Test" "2016-04-01T03:52:00.000+00:00" "1" "EN" "24 1" "24" "" "" "" "Hello world" "" "" "" "" "What's going on" "" "" "" "", c.xml_as_string (), list () ); } /* Write some subtitle content as SMPTE XML and check that it is right. This includes in-line font changes. */ BOOST_AUTO_TEST_CASE (write_smpte_subtitle_test2) { dcp::SMPTESubtitleAsset c; c.set_reel_number (1); c.set_language ("EN"); c.set_content_title_text ("Test"); c.set_issue_date (dcp::LocalTime ("2016-04-01T03:52:00+00:00")); c.add ( shared_ptr ( new dcp::SubtitleString ( string ("Arial"), false, false, false, dcp::Colour (255, 255, 255), 48, 1.0, dcp::Time (0, 0, 1, 0, 24), dcp::Time (0, 0, 9, 0, 24), 0, dcp::HALIGN_CENTER, 0.8, dcp::VALIGN_TOP, dcp::DIRECTION_LTR, "Testing is ", dcp::NONE, dcp::Colour (0, 0, 0), dcp::Time (0, 0, 0, 0, 24), dcp::Time (0, 0, 0, 0, 24) ) ) ); c.add ( shared_ptr ( new dcp::SubtitleString ( string ("Arial"), true, false, false, dcp::Colour (255, 255, 255), 48, 1.0, dcp::Time (0, 0, 1, 0, 24), dcp::Time (0, 0, 9, 0, 24), 0, dcp::HALIGN_CENTER, 0.8, dcp::VALIGN_TOP, dcp::DIRECTION_LTR, "really", dcp::NONE, dcp::Colour (0, 0, 0), dcp::Time (0, 0, 0, 0, 24), dcp::Time (0, 0, 0, 0, 24) ) ) ); c.add ( shared_ptr ( new dcp::SubtitleString ( string ("Arial"), false, false, false, dcp::Colour (255, 255, 255), 48, 1.0, dcp::Time (0, 0, 1, 0, 24), dcp::Time (0, 0, 9, 0, 24), 0, dcp::HALIGN_CENTER, 0.8, dcp::VALIGN_TOP, dcp::DIRECTION_LTR, " fun", dcp::NONE, dcp::Colour (0, 0, 0), dcp::Time (0, 0, 0, 0, 24), dcp::Time (0, 0, 0, 0, 24) ) ) ); c.add ( shared_ptr ( new dcp::SubtitleString ( string ("Arial"), false, false, false, dcp::Colour (255, 255, 255), 48, 1.0, dcp::Time (0, 0, 1, 0, 24), dcp::Time (0, 0, 9, 0, 24), 0, dcp::HALIGN_CENTER, 0.9, dcp::VALIGN_TOP, dcp::DIRECTION_LTR, "This is the ", dcp::NONE, dcp::Colour (0, 0, 0), dcp::Time (0, 0, 0, 0, 24), dcp::Time (0, 0, 0, 0, 24) ) ) ); c.add ( shared_ptr ( new dcp::SubtitleString ( string ("Arial"), true, false, false, dcp::Colour (255, 255, 255), 48, 1.0, dcp::Time (0, 0, 1, 0, 24), dcp::Time (0, 0, 9, 0, 24), 0, dcp::HALIGN_CENTER, 0.9, dcp::VALIGN_TOP, dcp::DIRECTION_LTR, "second", dcp::NONE, dcp::Colour (0, 0, 0), dcp::Time (0, 0, 0, 0, 24), dcp::Time (0, 0, 0, 0, 24) ) ) ); c.add ( shared_ptr ( new dcp::SubtitleString ( string ("Arial"), false, false, false, dcp::Colour (255, 255, 255), 48, 1.0, dcp::Time (0, 0, 1, 0, 24), dcp::Time (0, 0, 9, 0, 24), 0, dcp::HALIGN_CENTER, 0.9, dcp::VALIGN_TOP, dcp::DIRECTION_LTR, " line", dcp::NONE, dcp::Colour (0, 0, 0), dcp::Time (0, 0, 0, 0, 24), dcp::Time (0, 0, 0, 0, 24) ) ) ); c._xml_id = "a6c58cff-3e1e-4b38-acec-a42224475ef6"; check_xml ( c.xml_as_string (), "" "" "urn:uuid:a6c58cff-3e1e-4b38-acec-a42224475ef6" "Test" "2016-04-01T03:52:00.000+00:00" "1" "EN" "24 1" "24" "" "" "" "" "Testing is " "really" " fun" "" "" "This is the " "second" " line" "" "" "" "" "", list () ); } /* Write some subtitle content as SMPTE using bitmaps and check that it is right */ BOOST_AUTO_TEST_CASE (write_smpte_subtitle_test3) { dcp::SMPTESubtitleAsset c; c.set_reel_number (1); c.set_language ("EN"); c.set_content_title_text ("Test"); c.add ( shared_ptr ( new dcp::SubtitleImage ( dcp::Data ("test/data/sub.png"), dcp::Time (0, 4, 9, 22, 24), dcp::Time (0, 4, 11, 22, 24), 0, dcp::HALIGN_CENTER, 0.8, dcp::VALIGN_TOP, dcp::Time (0, 0, 0, 0, 24), dcp::Time (0, 0, 0, 0, 24) ) ) ); c._id = "a6c58cff-3e1e-4b38-acec-a42224475ef6"; boost::filesystem::create_directories ("build/test/write_smpte_subtitle_test3"); c.write ("build/test/write_smpte_subtitle_test3/subs.mxf"); /* XXX: check this result when we can read them back in again */ }