2 Copyright (C) 2013-2019 Carl Hetherington <cth@carlh.net>
4 This file is part of DCP-o-matic.
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.
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.
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/>.
22 #include "compose.hpp"
23 #include "dcpomatic_assert.h"
25 #include <dcp/raw_convert.h>
28 #include <dcp/reel_mxf.h>
29 #include <dcp/reel_asset.h>
30 DCPOMATIC_DISABLE_WARNINGS
31 #include <libxml++/libxml++.h>
32 DCPOMATIC_ENABLE_WARNINGS
33 #include <libcxml/cxml.h>
41 using std::shared_ptr;
42 using dcp::raw_convert;
44 bool operator== (Crop const & a, Crop const & b)
46 return (a.left == b.left && a.right == b.right && a.top == b.top && a.bottom == b.bottom);
49 bool operator!= (Crop const & a, Crop const & b)
54 /** @param r Resolution.
55 * @return Untranslated string representation.
58 resolution_to_string (Resolution r)
67 DCPOMATIC_ASSERT (false);
73 string_to_resolution (string s)
83 DCPOMATIC_ASSERT (false);
87 Crop::Crop (shared_ptr<cxml::Node> node)
89 left = node->number_child<int> ("LeftCrop");
90 right = node->number_child<int> ("RightCrop");
91 top = node->number_child<int> ("TopCrop");
92 bottom = node->number_child<int> ("BottomCrop");
96 Crop::as_xml (xmlpp::Node* node) const
98 node->add_child("LeftCrop")->add_child_text (raw_convert<string> (left));
99 node->add_child("RightCrop")->add_child_text (raw_convert<string> (right));
100 node->add_child("TopCrop")->add_child_text (raw_convert<string> (top));
101 node->add_child("BottomCrop")->add_child_text (raw_convert<string> (bottom));
105 string_to_text_type (string s)
107 if (s == "unknown") {
109 } else if (s == "open-subtitle") {
110 return TEXT_OPEN_SUBTITLE;
111 } else if (s == "closed-caption") {
112 return TEXT_CLOSED_CAPTION;
114 throw MetadataError (String::compose ("Unknown text type %1", s));
119 text_type_to_string (TextType t)
124 case TEXT_OPEN_SUBTITLE:
125 return "open-subtitle";
126 case TEXT_CLOSED_CAPTION:
127 return "closed-caption";
129 DCPOMATIC_ASSERT (false);
134 text_type_to_name (TextType t)
138 return _("Timed text");
139 case TEXT_OPEN_SUBTITLE:
140 return _("Open subtitles");
141 case TEXT_CLOSED_CAPTION:
142 return _("Closed captions");
144 DCPOMATIC_ASSERT (false);
149 video_frame_type_to_string (VideoFrameType t)
152 case VIDEO_FRAME_TYPE_2D:
154 case VIDEO_FRAME_TYPE_3D:
156 case VIDEO_FRAME_TYPE_3D_LEFT_RIGHT:
157 return "3d-left-right";
158 case VIDEO_FRAME_TYPE_3D_TOP_BOTTOM:
159 return "3d-top-bottom";
160 case VIDEO_FRAME_TYPE_3D_ALTERNATE:
161 return "3d-alternate";
162 case VIDEO_FRAME_TYPE_3D_LEFT:
164 case VIDEO_FRAME_TYPE_3D_RIGHT:
167 DCPOMATIC_ASSERT (false);
170 DCPOMATIC_ASSERT (false);
174 string_to_video_frame_type (string s)
177 return VIDEO_FRAME_TYPE_2D;
178 } else if (s == "3d") {
179 return VIDEO_FRAME_TYPE_3D;
180 } else if (s == "3d-left-right") {
181 return VIDEO_FRAME_TYPE_3D_LEFT_RIGHT;
182 } else if (s == "3d-top-bottom") {
183 return VIDEO_FRAME_TYPE_3D_TOP_BOTTOM;
184 } else if (s == "3d-alternate") {
185 return VIDEO_FRAME_TYPE_3D_ALTERNATE;
186 } else if (s == "3d-left") {
187 return VIDEO_FRAME_TYPE_3D_LEFT;
188 } else if (s == "3d-right") {
189 return VIDEO_FRAME_TYPE_3D_RIGHT;
192 DCPOMATIC_ASSERT (false);
195 CPLSummary::CPLSummary (boost::filesystem::path p)
196 : dcp_directory (p.leaf().string())
200 list<dcp::VerificationNote> notes;
202 for (auto i: notes) {
203 if (i.code() != dcp::VerificationNote::EXTERNAL_ASSET) {
204 /* It's not just a warning about this DCP being a VF */
205 throw dcp::ReadError(dcp::note_to_string(i));
209 cpl_id = dcp.cpls().front()->id();
210 cpl_annotation_text = dcp.cpls().front()->annotation_text();
211 cpl_file = dcp.cpls().front()->file().get();
214 for (auto j: dcp.cpls()) {
215 for (auto k: j->reel_mxfs()) {
222 last_write_time = boost::filesystem::last_write_time (p);
226 bool operator== (NamedChannel const& a, NamedChannel const& b)
228 return a.name == b.name && a.index == b.index;
233 video_range_to_string (VideoRange r)
236 case VIDEO_RANGE_FULL:
238 case VIDEO_RANGE_VIDEO:
241 DCPOMATIC_ASSERT (false);
247 string_to_video_range (string s)
250 return VIDEO_RANGE_FULL;
251 } else if (s == "video") {
252 return VIDEO_RANGE_VIDEO;
255 DCPOMATIC_ASSERT (false);
256 return VIDEO_RANGE_FULL;