Fix loading of SMPTE subtitles that are not MXF-wrapped.
[dcpomatic.git] / src / lib / content_factory.cc
1 /*
2     Copyright (C) 2013-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 /** @file  src/lib/content_factory.cc
21  *  @brief Methods to create content objects.
22  */
23
24 #include "ffmpeg_content.h"
25 #include "image_content.h"
26 #include "sndfile_content.h"
27 #include "subrip_content.h"
28 #include "dcp_content.h"
29 #include "dcp_subtitle_content.h"
30 #include "util.h"
31 #include <libcxml/cxml.h>
32
33 using std::string;
34 using std::list;
35 using boost::shared_ptr;
36
37 /** Create a Content object from an XML node.
38  *  @param film Film that the content will be in.
39  *  @param node XML description.
40  *  @param version XML state version.
41  *  @param notes A list to which is added descriptions of any non-critial warnings / messages.
42  *  @return Content object, or 0 if no content was recognised in the XML.
43  */
44 shared_ptr<Content>
45 content_factory (shared_ptr<const Film> film, cxml::NodePtr node, int version, list<string>& notes)
46 {
47         string const type = node->string_child ("Type");
48
49         boost::shared_ptr<Content> content;
50         
51         if (type == "FFmpeg") {
52                 content.reset (new FFmpegContent (film, node, version, notes));
53         } else if (type == "Image") {
54                 content.reset (new ImageContent (film, node, version));
55         } else if (type == "Sndfile") {
56                 content.reset (new SndfileContent (film, node, version));
57         } else if (type == "SubRip") {
58                 content.reset (new SubRipContent (film, node, version));
59         } else if (type == "DCP") {
60                 content.reset (new DCPContent (film, node, version));
61         } else if (type == "DCPSubtitle") {
62                 content.reset (new DCPSubtitleContent (film, node, version));
63         }
64
65         return content;
66 }
67
68 /** Create a Content object from a file, depending on its extension.
69  *  @param film Film that the content will be in.
70  *  @param path File's path.
71  *  @return Content object.
72  */
73 shared_ptr<Content>
74 content_factory (shared_ptr<const Film> film, boost::filesystem::path path)
75 {
76         shared_ptr<Content> content;
77
78         string ext = path.extension().string ();
79         transform (ext.begin(), ext.end(), ext.begin(), ::tolower);
80
81         if (valid_image_file (path)) {
82                 content.reset (new ImageContent (film, path));
83         } else if (SndfileContent::valid_file (path)) {
84                 content.reset (new SndfileContent (film, path));
85         } else if (ext == ".srt") {
86                 content.reset (new SubRipContent (film, path));
87         } else if (ext == ".xml") {
88                 content.reset (new DCPSubtitleContent (film, path));
89         } else if (ext == ".mxf") {
90                 /* Try to read this .mxf as a subtitle file; if we fail, we fall back
91                    to using FFmpeg below.
92                 */
93                 try {
94                         content.reset (new DCPSubtitleContent (film, path));
95                 } catch (...) {
96
97                 }
98         }
99
100         if (!content) {
101                 content.reset (new FFmpegContent (film, path));
102         }
103
104         return content;
105 }