Logging improvements to allow prettier displays in the server GUI.
[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 #include <dcp/smpte_subtitle_asset.h>
33 #include <boost/algorithm/string.hpp>
34
35 using std::string;
36 using std::list;
37 using boost::shared_ptr;
38
39 /** Create a Content object from an XML node.
40  *  @param film Film that the content will be in.
41  *  @param node XML description.
42  *  @param version XML state version.
43  *  @param notes A list to which is added descriptions of any non-critial warnings / messages.
44  *  @return Content object, or 0 if no content was recognised in the XML.
45  */
46 shared_ptr<Content>
47 content_factory (shared_ptr<const Film> film, cxml::NodePtr node, int version, list<string>& notes)
48 {
49         string const type = node->string_child ("Type");
50
51         boost::shared_ptr<Content> content;
52
53         if (type == "FFmpeg") {
54                 content.reset (new FFmpegContent (film, node, version, notes));
55         } else if (type == "Image") {
56                 content.reset (new ImageContent (film, node, version));
57         } else if (type == "Sndfile") {
58                 content.reset (new SndfileContent (film, node, version));
59         } else if (type == "SubRip") {
60                 content.reset (new SubRipContent (film, node, version));
61         } else if (type == "DCP") {
62                 content.reset (new DCPContent (film, node, version));
63         } else if (type == "DCPSubtitle") {
64                 content.reset (new DCPSubtitleContent (film, node, version));
65         }
66
67         return content;
68 }
69
70 /** Create a Content object from a file or directory.
71  *  @param film Film that the content will be in.
72  *  @param path File or directory.
73  *  @return Content object.
74  */
75 shared_ptr<Content>
76 content_factory (shared_ptr<const Film> film, boost::filesystem::path path)
77 {
78         shared_ptr<Content> content;
79
80         if (boost::filesystem::is_directory (path)) {
81
82                 if (boost::filesystem::is_empty (path)) {
83                         return shared_ptr<Content> ();
84                 }
85
86                 /* Guess if this is a DCP or a set of images: read the first ten filenames and if they
87                    are all valid image files we assume it is a set of images.
88                 */
89
90                 bool is_dcp = false;
91                 int read = 0;
92                 for (boost::filesystem::directory_iterator i(path); i != boost::filesystem::directory_iterator() && read < 10; ++i) {
93
94                         if (boost::starts_with (i->path().leaf().string(), "._")) {
95                                 /* We ignore these files */
96                                 continue;
97                         }
98
99                         if (!boost::filesystem::is_regular_file (i->path()) || !valid_image_file (i->path())) {
100                                 is_dcp = true;
101                         }
102
103                         ++read;
104                 }
105
106                 if (is_dcp) {
107                         content.reset (new DCPContent (film, path));
108                 } else {
109                         content.reset (new ImageContent (film, path));
110                 }
111
112         } else {
113
114                 string ext = path.extension().string ();
115                 transform (ext.begin(), ext.end(), ext.begin(), ::tolower);
116
117                 if (valid_image_file (path)) {
118                         content.reset (new ImageContent (film, path));
119                 } else if (SndfileContent::valid_file (path)) {
120                         content.reset (new SndfileContent (film, path));
121                 } else if (ext == ".srt") {
122                         content.reset (new SubRipContent (film, path));
123                 } else if (ext == ".xml") {
124                         content.reset (new DCPSubtitleContent (film, path));
125                 } else if (ext == ".mxf" && dcp::SMPTESubtitleAsset::valid_mxf (path)) {
126                         content.reset (new DCPSubtitleContent (film, path));
127                 }
128
129                 if (!content) {
130                         content.reset (new FFmpegContent (film, path));
131                 }
132         }
133
134         return content;
135 }