]> git.sesse.net Git - casparcg/blob - core/producer/scene/xml_scene_producer.cpp
* Implemented supports for timeline marks in scene_producer. A mark is set on a speci...
[casparcg] / core / producer / scene / xml_scene_producer.cpp
1 /*
2 * Copyright (c) 2011 Sveriges Television AB <info@casparcg.com>
3 *
4 * This file is part of CasparCG (www.casparcg.com).
5 *
6 * CasparCG 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 3 of the License, or
9 * (at your option) any later version.
10 *
11 * CasparCG 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.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with CasparCG. If not, see <http://www.gnu.org/licenses/>.
18 *
19 * Author: Helge Norberg, helge.norberg@svt.se
20 */
21
22 #include "../../StdAfx.h"
23
24 #include "xml_scene_producer.h"
25 #include "expression_parser.h"
26
27 #include <boost/filesystem.hpp>
28 #include <boost/filesystem/fstream.hpp>
29 #include <boost/property_tree/ptree.hpp>
30 #include <boost/property_tree/xml_parser.hpp>
31
32 #include <common/env.h>
33 #include <core/producer/frame_producer.h>
34
35 #include <future>
36
37 #include "scene_producer.h"
38 #include "scene_cg_proxy.h"
39
40 namespace caspar { namespace core { namespace scene {
41
42 void deduce_expression(variable& var, const variable_repository& repo)
43 {
44         auto expr_str = var.original_expr();
45         auto trimmed = boost::trim_copy(expr_str);
46
47         if (boost::starts_with(trimmed, L"${") && boost::ends_with(trimmed, L"}"))
48         {
49                 expr_str = trimmed.substr(2, expr_str.length() - 3);
50
51                 if (var.is<double>())
52                         var.as<double>().bind(parse_expression<double>(expr_str, repo));
53                 else if (var.is<bool>())
54                         var.as<bool>().bind(parse_expression<bool>(expr_str, repo));
55                 else if (var.is<std::wstring>())
56                         var.as<std::wstring>().bind(parse_expression<std::wstring>(expr_str, repo));
57         }
58         else if (!expr_str.empty())
59         {
60                 var.from_string(expr_str);
61         }
62 }
63
64 void init(module_dependencies dependencies)
65 {
66         dependencies.cg_registry->register_cg_producer(
67                         L"scene",
68                         { L".scene" },
69                         [](const std::wstring& filename) { return ""; },
70                         [](const spl::shared_ptr<core::frame_producer>& producer)
71                         {
72                                 return spl::make_shared<core::scene::scene_cg_proxy>(producer);
73                         },
74                         [](
75                         const spl::shared_ptr<core::frame_factory>& factory,
76                         const core::video_format_desc& format_desc,
77                         const std::wstring& filename)
78                         {
79                                 return create_xml_scene_producer(factory, format_desc, { filename });
80                         },
81                         false);
82 }
83
84 spl::shared_ptr<core::frame_producer> create_xml_scene_producer(
85                 const spl::shared_ptr<core::frame_factory>& frame_factory,
86                 const core::video_format_desc& format_desc,
87                 const std::vector<std::wstring>& params)
88 {
89         if (params.empty())
90                 return core::frame_producer::empty();
91
92         std::wstring filename = env::template_folder() + L"/" + params[0] + L".scene";
93         
94         if (!boost::filesystem::is_regular_file(boost::filesystem::path(filename)))
95                 return core::frame_producer::empty();
96
97         boost::property_tree::wptree root;
98         boost::filesystem::wifstream file(filename);
99         boost::property_tree::read_xml(
100                         file,
101                         root,
102                         boost::property_tree::xml_parser::trim_whitespace
103                                         | boost::property_tree::xml_parser::no_comments);
104
105         int width = root.get<int>(L"scene.<xmlattr>.width");
106         int height = root.get<int>(L"scene.<xmlattr>.height");
107
108         auto scene = spl::make_shared<scene_producer>(width, height, format_desc);
109
110         for (auto elem : root.get_child(L"scene.variables"))
111         {
112                 auto type = elem.second.get<std::wstring>(L"<xmlattr>.type");
113                 auto id = elem.second.get<std::wstring>(L"<xmlattr>.id");
114                 auto is_public = elem.second.get(L"<xmlattr>.public", false);
115                 auto expr = elem.second.get_value<std::wstring>();
116
117                 if (!is_public)
118                         id = L"variable." + id;
119
120                 if (type == L"number")
121                         scene->create_variable<double>(id, is_public, expr);
122                 else if (type == L"string")
123                         scene->create_variable<std::wstring>(id, is_public, expr);
124                 else if (type == L"bool")
125                         scene->create_variable<bool>(id, is_public, expr);
126         }
127
128         for (auto& elem : root.get_child(L"scene.layers"))
129         {
130                 auto id = elem.second.get<std::wstring>(L"<xmlattr>.id");
131                 auto producer = create_producer(frame_factory, format_desc, elem.second.get<std::wstring>(L"producer"));
132                 auto& layer = scene->create_layer(producer, 0, 0, id);
133                 auto variable_prefix = L"layer." + id + L".";
134
135                 layer.hidden = scene->create_variable<bool>(variable_prefix + L"hidden", false, elem.second.get(L"hidden", L"false"));
136                 layer.position.x = scene->create_variable<double>(variable_prefix + L"x", false, elem.second.get<std::wstring>(L"x"));
137                 layer.position.y = scene->create_variable<double>(variable_prefix + L"y", false, elem.second.get<std::wstring>(L"y"));
138                 layer.anchor.x = scene->create_variable<double>(variable_prefix + L"anchor_x", false, elem.second.get<std::wstring>(L"anchor_x", L"0.0"));
139                 layer.anchor.y = scene->create_variable<double>(variable_prefix + L"anchor_y", false, elem.second.get<std::wstring>(L"anchor_y", L"0.0"));
140                 layer.rotation = scene->create_variable<double>(variable_prefix + L"rotation", false, elem.second.get<std::wstring>(L"rotation", L"0.0"));
141                 layer.crop.upper_left.x = scene->create_variable<double>(variable_prefix + L"crop_upper_left_x", false, elem.second.get<std::wstring>(L"crop_upper_left_x", L"0.0"));
142                 layer.crop.upper_left.y = scene->create_variable<double>(variable_prefix + L"crop_upper_left_y", false, elem.second.get<std::wstring>(L"crop_upper_left_y", L"0.0"));
143                 layer.crop.lower_right.x = scene->create_variable<double>(variable_prefix + L"crop_lower_right_x", false, elem.second.get<std::wstring>(L"crop_lower_right_x", layer.producer.get()->pixel_constraints().width.as<std::wstring>().get()));
144                 layer.crop.lower_right.y = scene->create_variable<double>(variable_prefix + L"crop_lower_right_y", false, elem.second.get<std::wstring>(L"crop_lower_right_y", layer.producer.get()->pixel_constraints().height.as<std::wstring>().get()));
145                 layer.perspective.upper_left.x = scene->create_variable<double>(variable_prefix + L"perspective_upper_left_x", false, elem.second.get<std::wstring>(L"perspective_upper_left_x", L"0.0"));
146                 layer.perspective.upper_left.y = scene->create_variable<double>(variable_prefix + L"perspective_upper_left_y", false, elem.second.get<std::wstring>(L"perspective_upper_left_y", L"0.0"));
147                 layer.perspective.upper_right.x = scene->create_variable<double>(variable_prefix + L"perspective_upper_right_x", false, elem.second.get<std::wstring>(L"perspective_upper_right_x", layer.producer.get()->pixel_constraints().width.as<std::wstring>().get()));
148                 layer.perspective.upper_right.y = scene->create_variable<double>(variable_prefix + L"perspective_upper_right_y", false, elem.second.get<std::wstring>(L"perspective_upper_right_y", L"0.0"));
149                 layer.perspective.lower_right.x = scene->create_variable<double>(variable_prefix + L"perspective_lower_right_x", false, elem.second.get<std::wstring>(L"perspective_lower_right_x", layer.producer.get()->pixel_constraints().width.as<std::wstring>().get()));
150                 layer.perspective.lower_right.y = scene->create_variable<double>(variable_prefix + L"perspective_lower_right_y", false, elem.second.get<std::wstring>(L"perspective_lower_right_y", layer.producer.get()->pixel_constraints().height.as<std::wstring>().get()));
151                 layer.perspective.lower_left.x = scene->create_variable<double>(variable_prefix + L"perspective_lower_left_x", false, elem.second.get<std::wstring>(L"perspective_lower_left_x", L"0.0"));
152                 layer.perspective.lower_left.y = scene->create_variable<double>(variable_prefix + L"perspective_lower_left_y", false, elem.second.get<std::wstring>(L"perspective_lower_left_y", layer.producer.get()->pixel_constraints().height.as<std::wstring>().get()));
153
154                 layer.adjustments.opacity = scene->create_variable<double>(variable_prefix + L"adjustment.opacity", false, elem.second.get(L"adjustments.opacity", L"1.0"));
155                 layer.is_key = scene->create_variable<bool>(variable_prefix + L"is_key", false, elem.second.get(L"is_key", L"false"));
156                 layer.use_mipmap = scene->create_variable<bool>(variable_prefix + L"use_mipmap", false, elem.second.get(L"use_mipmap", L"false"));
157                 layer.blend_mode = scene->create_variable<std::wstring>(variable_prefix + L"blend_mode", false, elem.second.get(L"blend_mode", L"normal")).transformed([](const std::wstring& b) { return get_blend_mode(b); });
158                 layer.chroma_key.key = scene->create_variable<std::wstring>(variable_prefix + L"chroma_key.key", false, elem.second.get(L"chroma_key.key", L"none")).transformed([](const std::wstring& k) { return get_chroma_mode(k); });
159                 layer.chroma_key.threshold = scene->create_variable<double>(variable_prefix + L"chroma_key.threshold", false, elem.second.get(L"chroma_key.threshold", L"0.0"));
160                 layer.chroma_key.softness = scene->create_variable<double>(variable_prefix + L"chroma_key.softness", false, elem.second.get(L"chroma_key.softness", L"0.0"));
161                 layer.chroma_key.spill = scene->create_variable<double>(variable_prefix + L"chroma_key.spill", false, elem.second.get(L"chroma_key.spill", L"0.0"));
162
163                 scene->create_variable<double>(variable_prefix + L"width", false) = layer.producer.get()->pixel_constraints().width;
164                 scene->create_variable<double>(variable_prefix + L"height", false) = layer.producer.get()->pixel_constraints().height;
165
166                 for (auto& var_name : producer->get_variables())
167                 {
168                         auto& var = producer->get_variable(var_name);
169                         auto expr = elem.second.get<std::wstring>(L"parameters." + var_name, L"");
170
171                         if (var.is<double>())
172                                 scene->create_variable<double>(variable_prefix + L"parameter." + var_name, false, expr) = var.as<double>();
173                         else if (var.is<std::wstring>())
174                                 scene->create_variable<std::wstring>(variable_prefix + L"parameter." + var_name, false, expr) = var.as<std::wstring>();
175                         else if (var.is<bool>())
176                                 scene->create_variable<bool>(variable_prefix + L"parameter." + var_name, false, expr) = var.as<bool>();
177                 }
178         }
179
180         if (root.get_child_optional(L"scene.marks"))
181         {
182                 for (auto& mark : root.get_child(L"scene.marks"))
183                 {
184                         auto at = mark.second.get<int64_t>(L"<xmlattr>.at");
185                         auto type = get_mark_action(mark.second.get<std::wstring>(L"<xmlattr>.type"));
186                         auto label = mark.second.get(L"<xmlattr>.label", L"");
187
188                         scene->add_mark(at, type, label);
189                 }
190         }
191
192         for (auto& elem : root.get_child(L"scene.timelines"))
193         {
194                 auto& variable = scene->get_variable(elem.second.get<std::wstring>(L"<xmlattr>.variable"));
195
196                 for (auto& k : elem.second)
197                 {
198                         if (k.first == L"<xmlattr>")
199                                 continue;
200
201                         auto easing = k.second.get(L"<xmlattr>.easing", L"");
202                         auto at = k.second.get<int64_t>(L"<xmlattr>.at");
203
204                         if (variable.is<double>())
205                                 scene->add_keyframe(variable.as<double>(), k.second.get_value<double>(), at, easing);
206                         else if (variable.is<int>())
207                                 scene->add_keyframe(variable.as<int>(), k.second.get_value<int>(), at, easing);
208                 }
209         }
210
211         auto repo = [&scene](const std::wstring& name) -> variable&
212         {
213                 return scene->get_variable(name); 
214         };
215
216         for (auto& var_name : scene->get_variables())
217         {
218                 deduce_expression(scene->get_variable(var_name), repo);
219         }
220
221         auto params2 = params;
222         params2.erase(params2.begin());
223
224         scene->call(params2);
225
226         return scene;
227 }
228
229 }}}