Various libdcp API changes.
[dcpomatic.git] / src / lib / render_subtitles.cc
1 /*
2     Copyright (C) 2014 Carl Hetherington <cth@carlh.net>
3
4     This program is free software; you can redistribute it and/or modify
5     it under the terms of the GNU General Public License as published by
6     the Free Software Foundation; either version 2 of the License, or
7     (at your option) any later version.
8
9     This program is distributed in the hope that it will be useful,
10     but WITHOUT ANY WARRANTY; without even the implied warranty of
11     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12     GNU General Public License for more details.
13
14     You should have received a copy of the GNU General Public License
15     along with this program; if not, write to the Free Software
16     Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
17
18 */
19
20 #include <cairomm/cairomm.h>
21 #include <pangomm.h>
22 #include "render_subtitles.h"
23 #include "types.h"
24 #include "image.h"
25
26 using std::list;
27 using std::cout;
28 using std::string;
29 using std::min;
30 using std::max;
31 using std::pair;
32 using boost::shared_ptr;
33 using boost::optional;
34
35 static int
36 calculate_position (dcp::VAlign v_align, double v_position, int target_height, int offset)
37 {
38         switch (v_align) {
39         case dcp::TOP:
40                 return v_position * target_height - offset;
41         case dcp::CENTER:
42                 return (0.5 + v_position) * target_height - offset;
43         case dcp::BOTTOM:
44                 return (1.0 - v_position) * target_height - offset;
45         }
46
47         return 0;
48 }
49
50 PositionImage
51 render_subtitles (list<dcp::SubtitleString> subtitles, dcp::Size target)
52 {
53         if (subtitles.empty ()) {
54                 return PositionImage ();
55         }
56
57         /* Estimate height that the subtitle image needs to be */
58         optional<int> top;
59         optional<int> bottom;
60         for (list<dcp::SubtitleString>::const_iterator i = subtitles.begin(); i != subtitles.end(); ++i) {
61                 int const b = calculate_position (i->v_align(), i->v_position(), target.height, 0);
62                 int const t = b - i->size() * target.height / (11 * 72);
63
64                 top = min (top.get_value_or (t), t);
65                 bottom = max (bottom.get_value_or (b), b);
66         }
67
68         top = top.get() - 32;
69         bottom = bottom.get() + 32;
70
71         shared_ptr<Image> image (new Image (PIX_FMT_RGBA, dcp::Size (target.width, bottom.get() - top.get ()), false));
72         image->make_black ();
73
74         Cairo::RefPtr<Cairo::ImageSurface> surface = Cairo::ImageSurface::create (
75                 image->data()[0],
76                 Cairo::FORMAT_ARGB32,
77                 image->size().width,
78                 image->size().height,
79                 Cairo::ImageSurface::format_stride_for_width (Cairo::FORMAT_ARGB32, image->size().width)
80                 );
81         
82         Cairo::RefPtr<Cairo::Context> context = Cairo::Context::create (surface);
83         Glib::RefPtr<Pango::Layout> layout = Pango::Layout::create (context);
84
85         layout->set_width (image->size().width * PANGO_SCALE);
86         layout->set_alignment (Pango::ALIGN_CENTER);
87
88         context->set_line_width (1);
89
90         for (list<dcp::SubtitleString>::const_iterator i = subtitles.begin(); i != subtitles.end(); ++i) {
91                 Pango::FontDescription font (i->font().get_value_or ("Arial"));
92                 font.set_absolute_size (i->size_in_pixels (target.height) * PANGO_SCALE);
93                 if (i->italic ()) {
94                         font.set_style (Pango::STYLE_ITALIC);
95                 }
96                 layout->set_font_description (font);
97                 layout->set_text (i->text ());
98
99                 /* Compute fade factor */
100                 /* XXX */
101                 float fade_factor = 1;
102 #if 0           
103                 dcp::Time now (time * 1000 / (4 * TIME_HZ));
104                 dcp::Time end_fade_up = i->in() + i->fade_up_time ();
105                 dcp::Time start_fade_down = i->out() - i->fade_down_time ();
106                 if (now < end_fade_up) {
107                         fade_factor = (now - i->in()) / i->fade_up_time();
108                 } else if (now > start_fade_down) {
109                         fade_factor = 1.0 - ((now - start_fade_down) / i->fade_down_time ());
110                 }
111 #endif          
112
113                 layout->update_from_cairo_context (context);
114                 
115                 /* Work out position */
116
117                 int const x = 0;
118                 int const y = calculate_position (i->v_align (), i->v_position (), target.height, (layout->get_baseline() / PANGO_SCALE) + top.get ());
119
120                 if (i->effect() == dcp::SHADOW) {
121                         /* Drop-shadow effect */
122                         dcp::Colour const ec = i->effect_colour ();
123                         context->set_source_rgba (float(ec.r) / 255, float(ec.g) / 255, float(ec.b) / 255, fade_factor);
124                         context->move_to (x + 4, y + 4);
125                         layout->add_to_cairo_context (context);
126                         context->fill ();
127                 }
128
129                 /* The actual subtitle */
130                 context->move_to (x, y);
131                 dcp::Colour const c = i->colour ();
132                 context->set_source_rgba (float(c.r) / 255, float(c.g) / 255, float(c.b) / 255, fade_factor);
133                 layout->add_to_cairo_context (context);
134                 context->fill ();
135
136                 if (i->effect() == dcp::BORDER) {
137                         /* Border effect */
138                         context->move_to (x, y);
139                         dcp::Colour ec = i->effect_colour ();
140                         context->set_source_rgba (float(ec.r) / 255, float(ec.g) / 255, float(ec.b) / 255, fade_factor);
141                         layout->add_to_cairo_context (context);
142                         context->stroke ();
143                 }
144         }
145
146         return PositionImage (image, Position<int> (0, top.get ()));
147 }
148