2 Copyright (C) 2013-2015 Carl Hetherington <cth@carlh.net>
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.
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.
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.
20 #include "video_content_scale.h"
21 #include "video_content.h"
23 #include "safe_stringstream.h"
25 #include <libcxml/cxml.h>
26 #include <libxml++/libxml++.h>
27 #include <boost/optional.hpp>
36 using boost::shared_ptr;
37 using boost::optional;
39 vector<VideoContentScale> VideoContentScale::_scales;
41 VideoContentScale::VideoContentScale (Ratio const * r)
48 VideoContentScale::VideoContentScale ()
55 VideoContentScale::VideoContentScale (bool scale)
62 VideoContentScale::VideoContentScale (shared_ptr<cxml::Node> node)
66 optional<string> r = node->optional_string_child ("Ratio");
68 _ratio = Ratio::from_id (r.get ());
70 _scale = node->bool_child ("Scale");
75 VideoContentScale::as_xml (xmlpp::Node* node) const
78 node->add_child("Ratio")->add_child_text (_ratio->id ());
80 node->add_child("Scale")->add_child_text (_scale ? "1" : "0");
85 VideoContentScale::id () const
92 s << (_scale ? "S1" : "S0");
99 VideoContentScale::name () const
102 return _ratio->nickname ();
106 return _("No stretch");
109 return _("No scale");
113 VideoContentScale::from_id (string id)
115 Ratio const * r = Ratio::from_id (id);
117 return VideoContentScale (r);
121 return VideoContentScale (false);
124 return VideoContentScale (true);
127 /** @param display_container Size of the container that we are displaying this content in.
128 * @param film_container The size of the film's image.
131 VideoContentScale::size (shared_ptr<const VideoContent> c, dcp::Size display_container, dcp::Size film_container) const
133 /* Work out the size of the content if it were put inside film_container */
135 dcp::Size const video_size_after_crop = c->video_size_after_crop ();
140 /* Stretch to fit the requested ratio */
141 size = fit_ratio_within (_ratio->ratio (), film_container);
142 } else if (_scale || video_size_after_crop.width > film_container.width || video_size_after_crop.height > film_container.height) {
143 /* Scale, preserving aspect ratio; this is either if we have been asked to scale with no stretch
144 or if the unscaled content is too big for film_container.
146 size = fit_ratio_within (video_size_after_crop.ratio(), film_container);
148 /* No stretch nor scale */
149 size = video_size_after_crop;
152 /* Now scale it down if the display container is smaller than the film container */
153 if (display_container != film_container) {
154 float const scale = min (
155 float (display_container.width) / film_container.width,
156 float (display_container.height) / film_container.height
159 size.width = lrintf (size.width * scale);
160 size.height = lrintf (size.height * scale);
167 VideoContentScale::setup_scales ()
169 vector<Ratio const *> ratios = Ratio::all ();
170 for (vector<Ratio const *>::const_iterator i = ratios.begin(); i != ratios.end(); ++i) {
171 _scales.push_back (VideoContentScale (*i));
174 _scales.push_back (VideoContentScale (true));
175 _scales.push_back (VideoContentScale (false));
179 operator== (VideoContentScale const & a, VideoContentScale const & b)
181 return (a.ratio() == b.ratio() && a.scale() == b.scale());
185 operator!= (VideoContentScale const & a, VideoContentScale const & b)
187 return (a.ratio() != b.ratio() || a.scale() != b.scale());