Some more decode logging.
[dcpomatic.git] / src / lib / text_subtitle_decoder.cc
1 /*
2     Copyright (C) 2014-2016 Carl Hetherington <cth@carlh.net>
3
4     This file is part of DCP-o-matic.
5
6     DCP-o-matic 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.
10
11     DCP-o-matic 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.
15
16     You should have received a copy of the GNU General Public License
17     along with DCP-o-matic.  If not, see <http://www.gnu.org/licenses/>.
18
19 */
20
21 #include "text_subtitle_decoder.h"
22 #include "text_subtitle_content.h"
23 #include "subtitle_content.h"
24 #include <dcp/subtitle_string.h>
25 #include <boost/foreach.hpp>
26 #include <iostream>
27
28 using std::list;
29 using std::vector;
30 using std::string;
31 using std::cout;
32 using std::max;
33 using boost::shared_ptr;
34 using boost::optional;
35 using boost::dynamic_pointer_cast;
36
37 TextSubtitleDecoder::TextSubtitleDecoder (shared_ptr<const TextSubtitleContent> content, shared_ptr<Log> log)
38         : TextSubtitle (content)
39         , _next (0)
40 {
41         subtitle.reset (
42                 new SubtitleDecoder (
43                         this,
44                         content->subtitle,
45                         log,
46                         bind (&TextSubtitleDecoder::image_subtitles_during, this, _1, _2),
47                         bind (&TextSubtitleDecoder::text_subtitles_during, this, _1, _2)
48                         )
49                 );
50 }
51
52 void
53 TextSubtitleDecoder::seek (ContentTime time, bool accurate)
54 {
55         subtitle->seek (time, accurate);
56
57         _next = 0;
58         while (_next < _subtitles.size() && ContentTime::from_seconds (_subtitles[_next].from.all_as_seconds ()) < time) {
59                 ++_next;
60         }
61 }
62
63 bool
64 TextSubtitleDecoder::pass (PassReason, bool)
65 {
66         if (_next >= _subtitles.size ()) {
67                 return true;
68         }
69
70         ContentTimePeriod const p = content_time_period (_subtitles[_next]);
71         subtitle->give_text (p, _subtitles[_next]);
72         subtitle->set_position (p.from);
73
74         ++_next;
75         return false;
76 }
77
78 list<ContentTimePeriod>
79 TextSubtitleDecoder::image_subtitles_during (ContentTimePeriod, bool) const
80 {
81         return list<ContentTimePeriod> ();
82 }
83
84 list<ContentTimePeriod>
85 TextSubtitleDecoder::text_subtitles_during (ContentTimePeriod p, bool starting) const
86 {
87         /* XXX: inefficient */
88
89         list<ContentTimePeriod> d;
90
91         /* Only take `during' (not starting) subs if they overlap more than half the requested period;
92            here's the threshold for being significant.
93         */
94         ContentTime const significant (p.duration().get() / 2);
95
96         for (vector<sub::Subtitle>::const_iterator i = _subtitles.begin(); i != _subtitles.end(); ++i) {
97                 ContentTimePeriod t = content_time_period (*i);
98                 if (starting && p.contains(t.from)) {
99                         d.push_back (t);
100                 } else if (!starting) {
101                         optional<ContentTimePeriod> const o = p.overlap (t);
102                         if (o && o->duration() > significant) {
103                                 d.push_back (t);
104                         }
105                 }
106         }
107
108         return d;
109 }
110
111 ContentTimePeriod
112 TextSubtitleDecoder::content_time_period (sub::Subtitle s) const
113 {
114         return ContentTimePeriod (
115                 ContentTime::from_seconds (s.from.all_as_seconds()),
116                 ContentTime::from_seconds (s.to.all_as_seconds())
117                 );
118 }
119
120 void
121 TextSubtitleDecoder::reset ()
122 {
123         subtitle->reset ();
124 }