1 #include "../stdafx.h"
\r
4 #include "frame_producer.h"
\r
6 #include "../video_format.h"
\r
8 #include <common/concurrency/executor.h>
\r
9 #include <common/utility/assert.h>
\r
10 #include <common/utility/printable.h>
\r
12 #include <mixer/frame/draw_frame.h>
\r
13 #include <mixer/image/image_mixer.h>
\r
14 #include <mixer/audio/audio_mixer.h>
\r
15 #include <mixer/audio/audio_transform.h>
\r
17 #include <tbb/spin_mutex.h>
\r
19 namespace caspar { namespace core {
\r
21 class frame_producer_remover
\r
24 tbb::atomic<int> count_;
\r
26 void do_remove(safe_ptr<frame_producer>& producer)
\r
28 auto name = producer->print();
\r
29 producer = frame_producer::empty();
\r
30 CASPAR_LOG(info) << name << L" Removed.";
\r
34 frame_producer_remover()
\r
40 void remove(safe_ptr<frame_producer>&& producer)
\r
42 CASPAR_ASSERT(producer.unique());
\r
43 executor_.begin_invoke(std::bind(&frame_producer_remover::do_remove, this, std::move(producer)));
\r
47 frame_producer_remover g_remover;
\r
50 struct layer::implementation : boost::noncopyable
\r
52 printer parent_printer_;
\r
55 safe_ptr<frame_producer> foreground_;
\r
56 safe_ptr<frame_producer> background_;
\r
57 safe_ptr<draw_frame> last_frame_;
\r
60 implementation(int index, const printer& parent_printer)
\r
61 : parent_printer_(parent_printer)
\r
63 , foreground_(frame_producer::empty())
\r
64 , background_(frame_producer::empty())
\r
65 , last_frame_(draw_frame::empty())
\r
66 , is_paused_(false){}
\r
68 void load(const safe_ptr<frame_producer>& frame_producer, bool play_on_load)
\r
70 background_ = frame_producer;
\r
76 void preview(const safe_ptr<frame_producer>& frame_producer)
\r
78 load(frame_producer, true);
\r
87 background_->set_leading_producer(foreground_);
\r
88 foreground_ = background_;
\r
89 CASPAR_LOG(info) << foreground_->print() << L" Added.";
\r
90 background_ = frame_producer::empty();
\r
103 last_frame_ = draw_frame::empty();
\r
104 foreground_ = frame_producer::empty();
\r
109 foreground_ = frame_producer::empty();
\r
110 background_ = frame_producer::empty();
\r
111 last_frame_ = draw_frame::empty();
\r
112 is_paused_ = false;
\r
115 safe_ptr<draw_frame> receive()
\r
119 last_frame_->get_audio_transform().set_gain(0.0);
\r
120 return last_frame_;
\r
125 last_frame_ = foreground_->receive();
\r
126 if(last_frame_ == draw_frame::eof())
\r
128 CASPAR_ASSERT(foreground_ != frame_producer::empty());
\r
130 auto following = foreground_->get_following_producer();
\r
131 following->set_leading_producer(foreground_);
\r
132 following->set_parent_printer(boost::bind(&implementation::print, this));
\r
133 g_remover.remove(std::move(foreground_));
\r
134 foreground_ = following;
\r
135 CASPAR_LOG(info) << foreground_->print() << L" Added.";
\r
137 last_frame_ = receive();
\r
142 CASPAR_LOG(error) << print() << L" Unhandled Exception: ";
\r
143 CASPAR_LOG_CURRENT_EXCEPTION();
\r
147 return last_frame_;
\r
150 std::wstring print() const
\r
152 return (parent_printer_ ? parent_printer_() + L"/" : L"") + L"layer[" + boost::lexical_cast<std::wstring>(index_) + L"]";
\r
155 void swap(implementation& other)
\r
157 std::swap(foreground_, other.foreground_);
\r
158 std::swap(background_, other.background_);
\r
159 std::swap(last_frame_, other.last_frame_);
\r
160 std::swap(is_paused_, other.is_paused_);
\r
164 layer::layer(int index, const printer& parent_printer) : impl_(new implementation(index, parent_printer)){}
\r
165 layer::layer(layer&& other) : impl_(std::move(other.impl_)){}
\r
166 layer& layer::operator=(layer&& other)
\r
168 impl_ = std::move(other.impl_);
\r
171 void layer::swap(layer& other)
\r
173 impl_->swap(*other.impl_);
\r
175 void layer::load(const safe_ptr<frame_producer>& frame_producer, bool play_on_load){return impl_->load(frame_producer, play_on_load);}
\r
176 void layer::preview(const safe_ptr<frame_producer>& frame_producer){return impl_->preview(frame_producer);}
\r
177 void layer::play(){impl_->play();}
\r
178 void layer::pause(){impl_->pause();}
\r
179 void layer::stop(){impl_->stop();}
\r
180 void layer::clear(){impl_->clear();}
\r
181 safe_ptr<draw_frame> layer::receive() {return impl_->receive();}
\r
182 safe_ptr<frame_producer> layer::foreground() const { return impl_->foreground_;}
\r
183 safe_ptr<frame_producer> layer::background() const { return impl_->background_;}
\r
184 std::wstring layer::print() const { return impl_->print();}
\r