X-Git-Url: https://git.sesse.net/?p=nageru;a=blobdiff_plain;f=theme.h;h=bbef35ef350d1513afc92aa8216836b13b36b937;hp=f58681d080666dfe8fb3861512dc44b4de9bdd68;hb=327534a3031a332423411c9599c741f2f81657df;hpb=1e4d9085b0fd09607c8da23008cbee3be8ff119d diff --git a/theme.h b/theme.h index f58681d..bbef35e 100644 --- a/theme.h +++ b/theme.h @@ -1,66 +1,186 @@ #ifndef _THEME_H #define _THEME_H 1 -#include -#include -#include - +#include +#include +#include +#include #include +#include #include -#include +#include +#include +#include -#include -#include +#include "bmusb/bmusb.h" +#include "ref_counted_frame.h" +#include "tweaked_inputs.h" + +class CEFCapture; +class FFmpegCapture; +class LiveInputWrapper; +struct InputState; + +namespace movit { +class Effect; +class EffectChain; +class ResourcePool; +} // namespace movit class Theme { public: - Theme(const char *filename, movit::ResourcePool *resource_pool); - void register_class(const char *class_name, const luaL_Reg *funcs); + Theme(const std::string &filename, const std::vector &search_dirs, movit::ResourcePool *resource_pool, unsigned num_cards); + ~Theme(); - std::pair> - get_chain(unsigned num, float t, unsigned width, unsigned height); + struct Chain { + movit::EffectChain *chain; + std::function setup_chain; - void set_input_textures(int signal_num, GLuint tex_y, GLuint tex_cbcr) { - input_textures[signal_num].tex_y = tex_y; - input_textures[signal_num].tex_cbcr = tex_cbcr; - } + // FRAME_HISTORY frames for each input, in order. Will contain duplicates + // for non-interlaced inputs. + std::vector input_frames; + }; + + Chain get_chain(unsigned num, float t, unsigned width, unsigned height, InputState input_state); + + int get_num_channels() const { return num_channels; } + int map_signal(int signal_num); + void set_signal_mapping(int signal_num, int card_num); + std::string get_channel_name(unsigned channel); + int get_channel_signal(unsigned channel); + bool get_supports_set_wb(unsigned channel); + void set_wb(unsigned channel, double r, double g, double b); + std::string get_channel_color(unsigned channel); + + std::vector get_transition_names(float t); - void connect_signal(movit::YCbCrInput *input, int signal_num); void transition_clicked(int transition_num, float t); + void channel_clicked(int preview_num); -private: - std::mutex m; - lua_State *L; - movit::ResourcePool *resource_pool; - struct { - GLuint tex_y = 0, tex_cbcr = 0; - } input_textures[16]; // FIXME -}; + movit::ResourcePool *get_resource_pool() const { return resource_pool; } -class LiveInputWrapper { -public: - LiveInputWrapper(Theme *theme, movit::EffectChain *chain); + // Should be called as part of VideoInput.new() only. + void register_video_input(FFmpegCapture *capture) + { + video_inputs.push_back(capture); + } + + std::vector get_video_inputs() const + { + return video_inputs; + } - void connect_signal(int signal_num); -#if 0 +#ifdef HAVE_CEF + // Should be called as part of HTMLInput.new() only. + void register_html_input(CEFCapture *capture) { - connected_signal_num = signal_num; + html_inputs.push_back(capture); } - int get_connected_signal_num() const { - return connected_signal_num; + std::vector get_html_inputs() const + { + return html_inputs; } #endif - movit::YCbCrInput *get_input() const + void register_video_signal_connection(movit::EffectChain *chain, LiveInputWrapper *live_input, FFmpegCapture *capture) + { + video_signal_connections[chain].emplace_back(VideoSignalConnection { live_input, capture }); + } + +#ifdef HAVE_CEF + void register_html_signal_connection(movit::EffectChain *chain, LiveInputWrapper *live_input, CEFCapture *capture) + { + html_signal_connections[chain].emplace_back(CEFSignalConnection { live_input, capture }); + } +#endif + + struct MenuEntry { + std::string text; + int lua_ref; + }; + std::vector get_theme_menu() { return theme_menu; } // Can be empty for no menu. + void theme_menu_entry_clicked(int lua_ref); + + // Will be invoked every time the theme sets a new menu. + // Is not invoked for a menu that exists at the time of the callback. + void set_theme_menu_callback(std::function callback) + { + theme_menu_callback = callback; + } + +private: + void register_constants(); + void register_class(const char *class_name, const luaL_Reg *funcs); + int set_theme_menu(lua_State *L); + + std::string theme_path; + + std::mutex m; + lua_State *L; // Protected by . + const InputState *input_state = nullptr; // Protected by . Only set temporarily, during chain setup. + movit::ResourcePool *resource_pool; + int num_channels; + unsigned num_cards; + + std::mutex map_m; + std::map signal_to_card_mapping; // Protected by . + + std::vector video_inputs; + struct VideoSignalConnection { + LiveInputWrapper *wrapper; + FFmpegCapture *source; + }; + std::unordered_map> + video_signal_connections; +#ifdef HAVE_CEF + std::vector html_inputs; + struct CEFSignalConnection { + LiveInputWrapper *wrapper; + CEFCapture *source; + }; + std::unordered_map> + html_signal_connections; +#endif + + std::vector theme_menu; + std::function theme_menu_callback; + + friend class LiveInputWrapper; + friend int ThemeMenu_set(lua_State *L); +}; + +// LiveInputWrapper is a facade on top of an YCbCrInput, exposed to +// the Lua code. It contains a function (connect_signal()) intended +// to be called during chain setup, that picks out the current frame +// (in the form of a set of textures) from the input state given by +// the mixer, and communicates that state over to the actual YCbCrInput. +class LiveInputWrapper { +public: + // Note: is irrelevant for PixelFormat_8BitBGRA. + LiveInputWrapper(Theme *theme, movit::EffectChain *chain, bmusb::PixelFormat pixel_format, bool override_bounce, bool deinterlace); + + void connect_signal(int signal_num); // Must be called with the theme's lock held, since it accesses theme->input_state. + void connect_signal_raw(int signal_num, const InputState &input_state); + movit::Effect *get_effect() const { - return input; + if (deinterlace) { + return deinterlace_effect; + } else if (pixel_format == bmusb::PixelFormat_8BitBGRA) { + return rgba_inputs[0]; + } else { + return ycbcr_inputs[0]; + } } private: Theme *theme; // Not owned by us. - movit::YCbCrInput *input; // Owned by the chain. - int connected_signal_num = 0; + bmusb::PixelFormat pixel_format; + movit::YCbCrFormat input_ycbcr_format; + std::vector ycbcr_inputs; // Multiple ones if deinterlacing. Owned by the chain. + std::vector rgba_inputs; // Multiple ones if deinterlacing. Owned by the chain. + movit::Effect *deinterlace_effect = nullptr; // Owned by the chain. + bool deinterlace; }; #endif // !defined(_THEME_H)