]> git.sesse.net Git - casparcg/blobdiff - modules/image/producer/image_producer.cpp
Changed default log level to info and moved logging statements that we always want...
[casparcg] / modules / image / producer / image_producer.cpp
index 059decfa8b96f9425172f8dd8c3ec3299d68d666..92f805145d4d47f2d6146838a9d96447b23b1b55 100644 (file)
 #include <core/video_format.h>
 
 #include <core/producer/frame_producer.h>
+#include <core/producer/scene/const_producer.h>
 #include <core/frame/frame.h>
 #include <core/frame/draw_frame.h>
 #include <core/frame/frame_factory.h>
 #include <core/frame/pixel_format.h>
+#include <core/frame/audio_channel_layout.h>
 #include <core/monitor/monitor.h>
+#include <core/help/help_sink.h>
+#include <core/help/help_repository.h>
 
 #include <common/env.h>
 #include <common/log.h>
 #include <common/array.h>
+#include <common/base64.h>
+#include <common/os/filesystem.h>
 
-#include <boost/assign.hpp>
 #include <boost/filesystem.hpp>
 #include <boost/property_tree/ptree.hpp>
+#include <boost/algorithm/string.hpp>
 
 #include <algorithm>
-
-using namespace boost::assign;
+#include <set>
 
 namespace caspar { namespace image {
 
+std::pair<core::draw_frame, core::constraints> load_image(
+               const spl::shared_ptr<core::frame_factory>& frame_factory,
+               const std::wstring& filename)
+{
+       auto bitmap = load_image(filename);
+       FreeImage_FlipVertical(bitmap.get());
+               
+       core::pixel_format_desc desc = core::pixel_format::bgra;
+       auto width = FreeImage_GetWidth(bitmap.get());
+       auto height = FreeImage_GetHeight(bitmap.get());
+       desc.planes.push_back(core::pixel_format_desc::plane(width, height, 4));
+       auto frame = frame_factory->create_frame(bitmap.get(), desc, core::audio_channel_layout::invalid());
+
+       std::copy_n(
+                       FreeImage_GetBits(bitmap.get()),
+                       frame.image_data(0).size(),
+                       frame.image_data(0).begin());
+       
+       return std::make_pair(
+                       core::draw_frame(std::move(frame)),
+                       core::constraints(width, height));
+}
+
 struct image_producer : public core::frame_producer_base
 {      
-       monitor::basic_subject  event_subject_;
-       const std::wstring              filename_;
-       core::draw_frame                frame_;
+       core::monitor::subject                                          monitor_subject_;
+       const std::wstring                                                      description_;
+       const spl::shared_ptr<core::frame_factory>      frame_factory_;
+       core::draw_frame                                                        frame_                          = core::draw_frame::empty();
+       core::constraints                                                       constraints_;
        
-       explicit image_producer(const spl::shared_ptr<core::frame_factory>& frame_factory, const std::wstring& filename) 
-               : filename_(filename)
-               , frame_(core::draw_frame::empty())     
+       image_producer(const spl::shared_ptr<core::frame_factory>& frame_factory, const std::wstring& description, bool thumbnail_mode) 
+               : description_(description)
+               , frame_factory_(frame_factory)
        {
-               auto bitmap = load_image(filename_);
-               FreeImage_FlipVertical(bitmap.get());
-               
-               core::pixel_format_desc desc = core::pixel_format::bgra;
-               desc.planes.push_back(core::pixel_format_desc::plane(FreeImage_GetWidth(bitmap.get()), FreeImage_GetHeight(bitmap.get()), 4));
-               auto frame = frame_factory->create_frame(this, desc);
+               load(load_image(description_));
 
-               std::copy_n(FreeImage_GetBits(bitmap.get()), frame.image_data(0).size(), frame.image_data(0).begin());
-               frame_ = core::draw_frame(std::move(frame));
+               if (thumbnail_mode)
+                       CASPAR_LOG(debug) << print() << L" Initialized";
+               else
+                       CASPAR_LOG(info) << print() << L" Initialized";
+       }
+
+       image_producer(const spl::shared_ptr<core::frame_factory>& frame_factory, const void* png_data, size_t size) 
+               : description_(L"png from memory")
+               , frame_factory_(frame_factory)
+       {
+               load(load_png_from_memory(png_data, size));
 
                CASPAR_LOG(info) << print() << L" Initialized";
        }
+
+       void load(const std::shared_ptr<FIBITMAP>& bitmap)
+       {
+               FreeImage_FlipVertical(bitmap.get());
+               core::pixel_format_desc desc;
+               desc.format = core::pixel_format::bgra;
+               desc.planes.push_back(core::pixel_format_desc::plane(FreeImage_GetWidth(bitmap.get()), FreeImage_GetHeight(bitmap.get()), 4));
+               auto frame = frame_factory_->create_frame(this, desc, core::audio_channel_layout::invalid());
+               std::copy_n(FreeImage_GetBits(bitmap.get()), frame.image_data().size(), frame.image_data().begin());
+               frame_ = core::draw_frame(std::move(frame));
+               constraints_.width.set(FreeImage_GetWidth(bitmap.get()));
+               constraints_.height.set(FreeImage_GetHeight(bitmap.get()));
+       }
        
        // frame_producer
 
        core::draw_frame receive_impl() override
        {
-               event_subject_ << monitor::event("file/path") % filename_;
+               monitor_subject_ << core::monitor::message("/file/path") % description_;
 
                return frame_;
        }
+
+       core::draw_frame create_thumbnail_frame() override
+       {
+               return frame_;
+       }
+
+       core::constraints& pixel_constraints() override
+       {
+               return constraints_;
+       }
                        
        std::wstring print() const override
        {
-               return L"image_producer[" + filename_ + L"]";
+               return L"image_producer[" + description_ + L"]";
        }
 
        std::wstring name() const override
@@ -92,36 +150,148 @@ struct image_producer : public core::frame_producer_base
        {
                boost::property_tree::wptree info;
                info.add(L"type", L"image");
-               info.add(L"filename", filename_);
+               info.add(L"location", description_);
                return info;
        }
 
-       void subscribe(const monitor::observable::observer_ptr& o) override                                                                                                                     
+       core::monitor::subject& monitor_output() 
+       {
+               return monitor_subject_;
+       }
+};
+
+class ieq
+{
+       std::wstring test_;
+public:
+       ieq(const std::wstring& test)
+               : test_(test)
        {
-               return event_subject_.subscribe(o);
        }
 
-       void unsubscribe(const monitor::observable::observer_ptr& o) override           
+       bool operator()(const std::wstring& elem) const
        {
-               return event_subject_.unsubscribe(o);
+               return boost::iequals(elem, test_);
        }
 };
 
-spl::shared_ptr<core::frame_producer> create_producer(const spl::shared_ptr<core::frame_factory>& frame_factory, const core::video_format_desc& format_desc, const std::vector<std::wstring>& params)
+void describe_producer(core::help_sink& sink, const core::help_repository& repo)
 {
-       static const std::vector<std::wstring> extensions = list_of(L".png")(L".tga")(L".bmp")(L".jpg")(L".jpeg")(L".gif")(L".tiff")(L".tif")(L".jp2")(L".jpx")(L".j2k")(L".j2c");
-       std::wstring filename = env::media_folder() + L"\\" + params[0];
-       
-       auto ext = std::find_if(extensions.begin(), extensions.end(), [&](const std::wstring& ex) -> bool
-               {                       
-                       return boost::filesystem::is_regular_file(boost::filesystem::path(filename).replace_extension(ex));
-               });
+       sink.short_description(L"Loads a still image.");
+       sink.syntax(L"{[image_file:string]},{[PNG_BASE64] [encoded:string]}");
+       sink.para()->text(L"Loads a still image, either from disk or via a base64 encoded image submitted via AMCP.");
+       sink.para()->text(L"Examples:");
+       sink.example(L">> PLAY 1-10 image_file", L"Plays an image from the media folder.");
+       sink.example(L">> PLAY 1-10 [PNG_BASE64] data...", L"Plays a PNG image transferred as a base64 encoded string.");
+}
+
+static const auto g_extensions = {
+       L".png",
+       L".tga",
+       L".bmp",
+       L".jpg",
+       L".jpeg",
+       L".gif",
+       L".tiff",
+       L".tif",
+       L".jp2",
+       L".jpx",
+       L".j2k",
+       L".j2c"
+};
+
+spl::shared_ptr<core::frame_producer> create_producer(const core::frame_producer_dependencies& dependencies, const std::vector<std::wstring>& params)
+{
+
+       if (boost::iequals(params.at(0), L"[IMG_SEQUENCE]"))
+       {
+               if (params.size() != 2)
+                       return core::frame_producer::empty();
+
+               auto dir = boost::filesystem::path(env::media_folder() + params.at(1)).parent_path();
+               auto basename = boost::filesystem::basename(params.at(1));
+               std::set<std::wstring> files;
+               boost::filesystem::directory_iterator end;
+
+               for (boost::filesystem::directory_iterator it(dir); it != end; ++it)
+               {
+                       auto name = it->path().filename().wstring();
+
+                       if (!boost::algorithm::istarts_with(name, basename))
+                               continue;
+
+                       auto extension = it->path().extension().wstring();
 
-       if(ext == extensions.end())
+                       if (std::find_if(g_extensions.begin(), g_extensions.end(), ieq(extension)) == g_extensions.end())
+                               continue;
+
+                       files.insert(it->path().wstring());
+               }
+
+               if (files.empty())
+                       return core::frame_producer::empty();
+
+               int width = -1;
+               int height = -1;
+               std::vector<core::draw_frame> frames;
+               frames.reserve(files.size());
+
+               for (auto& file : files)
+               {
+                       auto frame = load_image(dependencies.frame_factory, file);
+
+                       if (width == -1)
+                       {
+                               width = static_cast<int>(frame.second.width.get());
+                               height = static_cast<int>(frame.second.height.get());
+                       }
+
+                       frames.push_back(std::move(frame.first));
+               }
+
+               return core::create_const_producer(std::move(frames), width, height);
+       }
+       else if(boost::iequals(params.at(0), L"[PNG_BASE64]"))
+       {
+               if (params.size() < 2)
+                       return core::frame_producer::empty();
+
+               auto png_data = from_base64(std::string(params.at(1).begin(), params.at(1).end()));
+
+               return spl::make_shared<image_producer>(dependencies.frame_factory, png_data.data(), png_data.size());
+       }
+
+       std::wstring filename = env::media_folder() + params.at(0);
+
+       auto ext = std::find_if(g_extensions.begin(), g_extensions.end(), [&](const std::wstring& ex) -> bool
+       {
+               auto file = caspar::find_case_insensitive(boost::filesystem::path(filename).replace_extension(ex).wstring());
+
+               return static_cast<bool>(file);
+       });
+
+       if(ext == g_extensions.end())
                return core::frame_producer::empty();
 
-       return spl::make_shared<image_producer>(frame_factory, filename + *ext);
+       return spl::make_shared<image_producer>(dependencies.frame_factory, *caspar::find_case_insensitive(filename + *ext), false);
 }
 
 
-}}
\ No newline at end of file
+spl::shared_ptr<core::frame_producer> create_thumbnail_producer(const core::frame_producer_dependencies& dependencies, const std::vector<std::wstring>& params)
+{
+       std::wstring filename = env::media_folder() + params.at(0);
+
+       auto ext = std::find_if(g_extensions.begin(), g_extensions.end(), [&](const std::wstring& ex) -> bool
+       {
+               auto file = caspar::find_case_insensitive(boost::filesystem::path(filename).replace_extension(ex).wstring());
+
+               return static_cast<bool>(file);
+       });
+
+       if (ext == g_extensions.end())
+               return core::frame_producer::empty();
+
+       return spl::make_shared<image_producer>(dependencies.frame_factory, *caspar::find_case_insensitive(filename + *ext), true);
+}
+
+}}