diff options
| author | Carl Hetherington <cth@carlh.net> | 2022-07-06 00:55:37 +0200 |
|---|---|---|
| committer | Carl Hetherington <cth@carlh.net> | 2022-07-06 02:10:33 +0200 |
| commit | eba39d94cb28a3f80e850cdf75f475b23d330eb9 (patch) | |
| tree | ea3a8bd0d7c10561207a053a628e99444b5cb04b | |
| parent | 23945f433d9cac2e8b8461a574baa70e1acaae1e (diff) | |
(More) properly support MarginV in SSA/ASS.
| -rw-r--r-- | src/ssa_reader.cc | 54 | ||||
| -rw-r--r-- | test/ssa_reader_test.cc | 80 |
2 files changed, 76 insertions, 58 deletions
diff --git a/src/ssa_reader.cc b/src/ssa_reader.cc index 0379803..03d20dd 100644 --- a/src/ssa_reader.cc +++ b/src/ssa_reader.cc @@ -274,9 +274,12 @@ SSAReader::parse_line (RawSubtitle base, string line, int play_res_x, int play_r current.vertical_position.reference = BOTTOM_OF_SCREEN; } - if (!current.vertical_position.proportional) { - current.vertical_position.proportional = 0; - } + /* Any vertical_position that is set in base (and therefore current) is a margin, which + * we need to ignore if we end up vertically centering this subtitle. + * Clear out vertical_position from current; we'll re-add it from base later + * if required. + */ + current.vertical_position.proportional = 0; /* We must have a font size, as there could be a margin specified in pixels and in that case we must know how big the subtitle @@ -299,20 +302,6 @@ SSAReader::parse_line (RawSubtitle base, string line, int play_res_x, int play_r /* Imagine that the screen is 792 points (i.e. 11 inches) high (as with DCP) */ double const line_size = current.font_size.proportional(792) * 1.2; - /* Tweak vertical_position accordingly */ - switch (current.vertical_position.reference.get()) { - case TOP_OF_SCREEN: - case TOP_OF_SUBTITLE: - /* Nothing to do */ - break; - case VERTICAL_CENTRE_OF_SCREEN: - current.vertical_position.proportional = current.vertical_position.proportional.get() - ((line_breaks + 1) * line_size) / 2; - break; - case BOTTOM_OF_SCREEN: - current.vertical_position.proportional = current.vertical_position.proportional.get() + line_breaks * line_size; - break; - } - for (size_t i = 0; i < line.length(); ++i) { char const c = line[i]; switch (state) { @@ -363,6 +352,28 @@ SSAReader::parse_line (RawSubtitle base, string line, int play_res_x, int play_r subs.push_back (current); } + /* Now we definitely know the vertical position reference we can finish off the position */ + for (auto& sub: subs) { + switch (sub.vertical_position.reference.get()) { + case TOP_OF_SCREEN: + case TOP_OF_SUBTITLE: + /* Just re-add any margins we came in with */ + sub.vertical_position.proportional = sub.vertical_position.proportional.get() + base.vertical_position.proportional.get_value_or(0); + break; + case VERTICAL_CENTRE_OF_SCREEN: + /* Margins are ignored, but we need to centre */ + sub.vertical_position.proportional = sub.vertical_position.proportional.get() - ((line_breaks + 1) * line_size) / 2; + break; + case BOTTOM_OF_SCREEN: + /* Re-add margins and account for each line */ + sub.vertical_position.proportional = + sub.vertical_position.proportional.get() + + base.vertical_position.proportional.get_value_or(0) + + line_breaks * line_size; + break; + } + } + return subs; } @@ -477,9 +488,14 @@ SSAReader::read (function<optional<string> ()> get_line) sub.effect = style.effect; sub.horizontal_position.reference = style.horizontal_reference; sub.vertical_position.reference = style.vertical_reference; - sub.vertical_position.proportional = float(style.vertical_margin) / play_res_y; + if (sub.vertical_position.reference != sub::VERTICAL_CENTRE_OF_SCREEN) { + sub.vertical_position.proportional = float(style.vertical_margin) / play_res_y; + } } else if (event_format[i] == "MarginV") { - sub.vertical_position.proportional = raw_convert<float>(event[i]) / play_res_y; + if (event[i] != "0" && sub.vertical_position.reference != sub::VERTICAL_CENTRE_OF_SCREEN) { + /* Override the style if its non-zero */ + sub.vertical_position.proportional = raw_convert<float>(event[i]) / play_res_y; + } } else if (event_format[i] == "Text") { for (auto j: parse_line (sub, event[i], play_res_x, play_res_y)) { _subs.push_back (j); diff --git a/test/ssa_reader_test.cc b/test/ssa_reader_test.cc index d882718..59a161b 100644 --- a/test/ssa_reader_test.cc +++ b/test/ssa_reader_test.cc @@ -199,7 +199,7 @@ BOOST_AUTO_TEST_CASE (ssa_reader_test3) /* Hello world */ SUB_START (sub::Time::from_hms (0, 0, 1, 230), sub::Time::from_hms (0, 0, 4, 550)); - LINE (0, sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); + LINE ((10.0 / 1080), sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); BLOCK ("Hello world", "Arial", 20, false, false, false); SUB_END(); @@ -217,7 +217,7 @@ BOOST_AUTO_TEST_CASE (ssa_reader_test3) /* Some {\i1}italics{\i} are here. */ SUB_START (sub::Time::from_hms (0, 0, 7, 740), sub::Time::from_hms (0, 0, 9, 0)); - LINE(0, sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); + LINE((10.0 / 1080), sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); BLOCK("Some ", "Arial", 20, false, false, false); BLOCK("italics", "Arial", 20, false, true, false); BLOCK(" are here.", "Arial", 20, false, false, false); @@ -226,47 +226,48 @@ BOOST_AUTO_TEST_CASE (ssa_reader_test3) /* Alignments */ SUB_START (sub::Time::from_hms (0, 0, 9, 230), sub::Time::from_hms (0, 0, 11, 560)); - LINE (0, sub::BOTTOM_OF_SCREEN, 0, sub::LEFT_OF_SCREEN); + LINE ((10.0 / 1080), sub::BOTTOM_OF_SCREEN, 0, sub::LEFT_OF_SCREEN); BLOCK("bottom left", "Arial", 20, false, false, false); SUB_END (); SUB_START (sub::Time::from_hms (0, 0, 9, 240), sub::Time::from_hms (0, 0, 11, 560)); - LINE (0, sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); + LINE ((10.0 / 1080), sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); BLOCK("bottom centre", "Arial", 20, false, false, false); SUB_END (); SUB_START (sub::Time::from_hms (0, 0, 9, 250), sub::Time::from_hms (0, 0, 11, 560)); - LINE (0, sub::BOTTOM_OF_SCREEN, 0, sub::RIGHT_OF_SCREEN); + LINE ((10.0 / 1080), sub::BOTTOM_OF_SCREEN, 0, sub::RIGHT_OF_SCREEN); BLOCK("bottom right", "Arial", 20, false, false, false); SUB_END (); SUB_START (sub::Time::from_hms (0, 0, 9, 260), sub::Time::from_hms (0, 0, 11, 560)); - LINE (0, sub::VERTICAL_CENTRE_OF_SCREEN, 0, sub::LEFT_OF_SCREEN); + /* Position is half of a 20pt line (with line spacing) above vertical centre */ + LINE (-(10.0 * 1.2 / 792), sub::VERTICAL_CENTRE_OF_SCREEN, 0, sub::LEFT_OF_SCREEN); BLOCK("middle left", "Arial", 20, false, false, false); SUB_END (); SUB_START (sub::Time::from_hms (0, 0, 9, 270), sub::Time::from_hms (0, 0, 11, 560)); - LINE (0, sub::VERTICAL_CENTRE_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); + LINE (-(10.0 * 1.2 / 792), sub::VERTICAL_CENTRE_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); BLOCK("middle centre", "Arial", 20, false, false, false); SUB_END (); SUB_START (sub::Time::from_hms (0, 0, 9, 280), sub::Time::from_hms (0, 0, 11, 560)); - LINE (0, sub::VERTICAL_CENTRE_OF_SCREEN, 0, sub::RIGHT_OF_SCREEN); + LINE (-(10.0 * 1.2 / 792), sub::VERTICAL_CENTRE_OF_SCREEN, 0, sub::RIGHT_OF_SCREEN); BLOCK("middle right", "Arial", 20, false, false, false); SUB_END (); SUB_START (sub::Time::from_hms (0, 0, 9, 290), sub::Time::from_hms (0, 0, 11, 560)); - LINE (0, sub::TOP_OF_SCREEN, 0, sub::LEFT_OF_SCREEN); + LINE ((10.0 / 1080), sub::TOP_OF_SCREEN, 0, sub::LEFT_OF_SCREEN); BLOCK("top left", "Arial", 20, false, false, false); SUB_END (); SUB_START (sub::Time::from_hms (0, 0, 9, 300), sub::Time::from_hms (0, 0, 11, 560)); - LINE (0, sub::TOP_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); + LINE ((10.0 / 1080), sub::TOP_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); BLOCK("top centre", "Arial", 20, false, false, false); SUB_END (); SUB_START (sub::Time::from_hms (0, 0, 9, 310), sub::Time::from_hms (0, 0, 11, 560)); - LINE (0, sub::TOP_OF_SCREEN, 0, sub::RIGHT_OF_SCREEN); + LINE ((10.0 / 1080), sub::TOP_OF_SCREEN, 0, sub::RIGHT_OF_SCREEN); BLOCK("top right", "Arial", 20, false, false, false); SUB_END (); @@ -387,66 +388,67 @@ BOOST_AUTO_TEST_CASE (ssa_reader_test6) SUB_START (sub::Time::from_hms (0, 0, 0, 70), sub::Time::from_hms (0, 0, 1, 110)); /* The first line should be one line (30 points, 1.2 times - spaced, as a proportion of the total screen height 729 - points) up. + spaced, as a proportion of the total screen height 792 + points) up. There's also a 10 pixel (with respect to a + 288-pixel-high screen) margin. */ - LINE ((30.0 * 1.2 / 792), sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); + LINE (((30.0 * 1.2 / 792) + (10.0 / 288.0)), sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); BLOCK ("This line is normal", "Arial", 30, false, false, false); - LINE (0, sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); + LINE ((10.0 / 288.0), sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); BLOCK ("This line is bold", "Arial", 30, true, false, false); SUB_END (); SUB_START (sub::Time::from_hms (0, 0, 1, 200), sub::Time::from_hms (0, 0, 2, 240)); - LINE ((30.0 * 1.2 / 792), sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); + LINE (((30.0 * 1.2 / 792) + (10.0 / 288.0)), sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); BLOCK ("This line is bold", "Arial", 30, true, false, false); - LINE (0, sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); + LINE ((10.0 / 288.0), sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); BLOCK ("This line is normal", "Arial", 30, false, false, false); SUB_END (); SUB_START (sub::Time::from_hms (0, 0, 2, 300), sub::Time::from_hms (0, 0, 3, 380)); - LINE ((30.0 * 1.2 / 792), sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); + LINE (((30.0 * 1.2 / 792) + (10.0 / 288.0)), sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); BLOCK ("This line is bold", "Arial", 30, true, false, false); - LINE (0, sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); + LINE ((10.0 / 288.0), sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); BLOCK ("This line is italic", "Arial", 30, false, true, false); SUB_END (); SUB_START (sub::Time::from_hms (0, 0, 3, 400), sub::Time::from_hms (0, 0, 4, 480)); - LINE ((30.0 * 1.2 / 792), sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); + LINE (((30.0 * 1.2 / 792) + (10.0 / 288.0)), sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); BLOCK ("This line is italic", "Arial", 30, false, true, false); - LINE (0, sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); + LINE ((10.0 / 288.0), sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); BLOCK ("This line is bold", "Arial", 30, true, false, false); SUB_END (); SUB_START (sub::Time::from_hms (0, 0, 4, 510), sub::Time::from_hms (0, 0, 5, 600)); - LINE ((30.0 * 1.2 / 792), sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); + LINE (((30.0 * 1.2 / 792) + (10.0 / 288.0)), sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); BLOCK ("Last three words are ", "Arial", 30, false, false, false); BLOCK ("bold AND italic", "Arial", 30, true, true, false); - LINE (0, sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); + LINE ((10.0 / 288.0), sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); BLOCK ("Last three words are ", "Arial", 30, false, false, false); BLOCK ("italic AND bold", "Arial", 30, true, true, false); SUB_END (); SUB_START (sub::Time::from_hms (0, 0, 5, 620), sub::Time::from_hms (0, 0, 6, 710)); - LINE ((30.0 * 1.2 / 792), sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); + LINE (((30.0 * 1.2 / 792) + (10.0 / 288.0)), sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); BLOCK ("Last three words are ", "Arial", 30, false, false, false); BLOCK ("bold AND italic", "Arial", 30, true, true, false); - LINE (0, sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); + LINE ((10.0 / 288.0), sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); BLOCK ("First three words", "Arial", 30, true, true, false); BLOCK (" are italic AND bold", "Arial", 30, false, false, false); SUB_END (); SUB_START (sub::Time::from_hms (0, 0, 6, 730), sub::Time::from_hms (0, 0, 8, 30)); - LINE ((30.0 * 1.2 / 792), sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); + LINE (((30.0 * 1.2 / 792) + (10.0 / 288.0)), sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); BLOCK ("Last three words are ", "Arial", 30, false, false, false); BLOCK ("bold AND italic", "Arial", 30, true, true, false); - LINE (0, sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); + LINE ((10.0 / 288.0), sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); BLOCK ("This line is normal", "Arial", 30, false, false, false); SUB_END (); SUB_START (sub::Time::from_hms (0, 0, 8, 90), sub::Time::from_hms (0, 0, 9, 210)); - LINE ((30.0 * 1.2 / 792), sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); + LINE (((30.0 * 1.2 / 792) + (10.0 / 288.0)), sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); BLOCK ("Both lines are bold AND italic", "Arial", 30, true, true, false); - LINE (0, sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); + LINE ((10.0 / 288.0), sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); BLOCK ("Both lines are bold AND italic", "Arial", 30, true, true, false); SUB_END (); } @@ -468,30 +470,30 @@ BOOST_AUTO_TEST_CASE (ssa_reader_test7) BOOST_REQUIRE (i != subs.end()); SUB_START(sub::Time::from_hms(0, 0, 1, 0), sub::Time::from_hms(0, 0, 3, 0)); - LINE((60.0 * 1.2 / 792), sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); + LINE(((60.0 * 1.2 / 792) + (100.0 / 1080)), sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); BLOCK("Helvetica Neue 60pt - Default", "Helvetica Neue", 60, false, false, false); - LINE(0, sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); + LINE((100.0 / 1080), sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); BLOCK("Bottom 100 pt off edge", "Helvetica Neue", 60, false, false, false); SUB_END(); SUB_START(sub::Time::from_hms(0, 0, 4, 0), sub::Time::from_hms(0, 0, 6, 0)); - LINE((30.0 * 1.2 / 792), sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); + LINE((((30.0 * 1.2) / 792) + (100.0 / 1080)), sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); BLOCK("Helvetica Neue 30pt", "Helvetica Neue", 30, false, false, false); - LINE(0, sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); + LINE((100.0 / 1080), sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); BLOCK("Bottom 100pt off edge", "Helvetica Neue", 30, false, false, false); SUB_END(); SUB_START(sub::Time::from_hms(0, 0, 7, 0), sub::Time::from_hms(0, 0, 9, 0)); - LINE((120.0 * 1.2 / 792), sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); + LINE((((120.0 * 1.2) / 792) + (100.0 / 1080)), sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); BLOCK("Helvetica Neue 120pt", "Helvetica Neue", 120, false, false, false); - LINE(0, sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); + LINE((100.0 / 1080), sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); BLOCK("Bottom 100pt off edge", "Helvetica Neue", 120, false, false, false); SUB_END(); SUB_START(sub::Time::from_hms(0, 0, 10, 0), sub::Time::from_hms(0, 0, 12, 0)); - LINE(0, sub::TOP_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); + LINE((100.0 / 1080), sub::TOP_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); BLOCK("Helvetica Neue 60pt", "Helvetica Neue", 60, false, false, false); - LINE((60.0 * 1.2 / 792), sub::TOP_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); + LINE((((60.0) * 1.2 / 792) + (100.0 / 1080)), sub::TOP_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); BLOCK("Top Alignment 100pt off edge", "Helvetica Neue", 60, false, false, false); SUB_END(); @@ -519,9 +521,9 @@ BOOST_AUTO_TEST_CASE (ssa_reader_pos) /* Hello world */ SUB_START (sub::Time::from_hms (0, 0, 1, 230), sub::Time::from_hms (0, 0, 4, 550)); - LINE (0, sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); + LINE ((10.0 / 1080), sub::BOTTOM_OF_SCREEN, 0, sub::HORIZONTAL_CENTRE_OF_SCREEN); BLOCK ("Hello world this is ", "Arial", 20, false, false, false); - LINE (300.0 / 1080, sub::TOP_OF_SCREEN, 400.0 / 1920, sub::LEFT_OF_SCREEN); + LINE ((310.0 / 1080), sub::TOP_OF_SCREEN, 400.0 / 1920, sub::LEFT_OF_SCREEN); BLOCK ("positioning.", "Arial", 20, false, false, false); SUB_END(); } |
