chroma-key-filter.c 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289
  1. #include <obs-module.h>
  2. #include <graphics/matrix4.h>
  3. #include <graphics/vec2.h>
  4. #include <graphics/vec4.h>
  5. #define SETTING_OPACITY "opacity"
  6. #define SETTING_CONTRAST "contrast"
  7. #define SETTING_BRIGHTNESS "brightness"
  8. #define SETTING_GAMMA "gamma"
  9. #define SETTING_COLOR_TYPE "key_color_type"
  10. #define SETTING_KEY_COLOR "key_color"
  11. #define SETTING_SIMILARITY "similarity"
  12. #define SETTING_SMOOTHNESS "smoothness"
  13. #define SETTING_SPILL "spill"
  14. #define TEXT_OPACITY obs_module_text("Opacity")
  15. #define TEXT_CONTRAST obs_module_text("Contrast")
  16. #define TEXT_BRIGHTNESS obs_module_text("Brightness")
  17. #define TEXT_GAMMA obs_module_text("Gamma")
  18. #define TEXT_COLOR_TYPE obs_module_text("KeyColorType")
  19. #define TEXT_KEY_COLOR obs_module_text("KeyColor")
  20. #define TEXT_SIMILARITY obs_module_text("Similarity")
  21. #define TEXT_SMOOTHNESS obs_module_text("Smoothness")
  22. #define TEXT_SPILL obs_module_text("ColorSpillReduction")
  23. struct chroma_key_filter_data {
  24. obs_source_t *context;
  25. gs_effect_t *effect;
  26. gs_eparam_t *color_param;
  27. gs_eparam_t *contrast_param;
  28. gs_eparam_t *brightness_param;
  29. gs_eparam_t *gamma_param;
  30. gs_eparam_t *pixel_size_param;
  31. gs_eparam_t *chroma_param;
  32. gs_eparam_t *key_rgb_param;
  33. gs_eparam_t *similarity_param;
  34. gs_eparam_t *smoothness_param;
  35. gs_eparam_t *spill_param;
  36. struct vec4 color;
  37. float contrast;
  38. float brightness;
  39. float gamma;
  40. struct vec4 key_rgb;
  41. struct vec2 chroma;
  42. float similarity;
  43. float smoothness;
  44. float spill;
  45. };
  46. static const char *chroma_key_name(void *unused)
  47. {
  48. UNUSED_PARAMETER(unused);
  49. return obs_module_text("ChromaKeyFilter");
  50. }
  51. static const float yuv_mat[16] = {0.182586f, -0.100644f, 0.439216f, 0.0f,
  52. 0.614231f, -0.338572f, -0.398942f, 0.0f,
  53. 0.062007f, 0.439216f, -0.040274f, 0.0f,
  54. 0.062745f, 0.501961f, 0.501961f, 1.0f};
  55. static inline void color_settings_update(
  56. struct chroma_key_filter_data *filter, obs_data_t *settings)
  57. {
  58. uint32_t opacity = (uint32_t)obs_data_get_int(settings,
  59. SETTING_OPACITY);
  60. uint32_t color = 0xFFFFFF | (((opacity * 255) / 100) << 24);
  61. double contrast = obs_data_get_double(settings, SETTING_CONTRAST);
  62. double brightness = obs_data_get_double(settings, SETTING_BRIGHTNESS);
  63. double gamma = obs_data_get_double(settings, SETTING_GAMMA);
  64. contrast = (contrast < 0.0) ?
  65. (1.0 / (-contrast + 1.0)) : (contrast + 1.0);
  66. brightness *= 0.5;
  67. gamma = (gamma < 0.0) ? (-gamma + 1.0) : (1.0 / (gamma + 1.0));
  68. filter->contrast = (float)contrast;
  69. filter->brightness = (float)brightness;
  70. filter->gamma = (float)gamma;
  71. vec4_from_rgba(&filter->color, color);
  72. }
  73. static inline void chroma_settings_update(
  74. struct chroma_key_filter_data *filter, obs_data_t *settings)
  75. {
  76. int64_t similarity = obs_data_get_int(settings, SETTING_SIMILARITY);
  77. int64_t smoothness = obs_data_get_int(settings, SETTING_SMOOTHNESS);
  78. int64_t spill = obs_data_get_int(settings, SETTING_SPILL);
  79. uint32_t key_color = (uint32_t)obs_data_get_int(settings,
  80. SETTING_KEY_COLOR);
  81. const char *key_type = obs_data_get_string(settings,
  82. SETTING_COLOR_TYPE);
  83. struct vec4 key_color_v4;
  84. struct matrix4 yuv_mat_m4;
  85. if (strcmp(key_type, "green") == 0)
  86. key_color = 0x00FF00;
  87. else if (strcmp(key_type, "blue") == 0)
  88. key_color = 0xFF9900;
  89. else if (strcmp(key_type, "magenta") == 0)
  90. key_color = 0xFF00FF;
  91. vec4_from_rgba(&filter->key_rgb, key_color | 0xFF000000);
  92. memcpy(&yuv_mat_m4, yuv_mat, sizeof(yuv_mat));
  93. vec4_transform(&key_color_v4, &filter->key_rgb, &yuv_mat_m4);
  94. vec2_set(&filter->chroma, key_color_v4.y, key_color_v4.z);
  95. filter->similarity = (float)similarity / 1000.0f;
  96. filter->smoothness = (float)smoothness / 1000.0f;
  97. filter->spill = (float)spill / 1000.0f;
  98. }
  99. static void chroma_key_update(void *data, obs_data_t *settings)
  100. {
  101. struct chroma_key_filter_data *filter = data;
  102. color_settings_update(filter, settings);
  103. chroma_settings_update(filter, settings);
  104. }
  105. static void chroma_key_destroy(void *data)
  106. {
  107. struct chroma_key_filter_data *filter = data;
  108. if (filter->effect) {
  109. obs_enter_graphics();
  110. gs_effect_destroy(filter->effect);
  111. obs_leave_graphics();
  112. }
  113. bfree(data);
  114. }
  115. static void *chroma_key_create(obs_data_t *settings, obs_source_t *context)
  116. {
  117. struct chroma_key_filter_data *filter =
  118. bzalloc(sizeof(struct chroma_key_filter_data));
  119. char *effect_path = obs_module_file("chroma_key_filter.effect");
  120. filter->context = context;
  121. obs_enter_graphics();
  122. filter->effect = gs_effect_create_from_file(effect_path, NULL);
  123. if (filter) {
  124. filter->color_param = gs_effect_get_param_by_name(
  125. filter->effect, "color");
  126. filter->contrast_param = gs_effect_get_param_by_name(
  127. filter->effect, "contrast");
  128. filter->brightness_param = gs_effect_get_param_by_name(
  129. filter->effect, "brightness");
  130. filter->gamma_param = gs_effect_get_param_by_name(
  131. filter->effect, "gamma");
  132. filter->chroma_param = gs_effect_get_param_by_name(
  133. filter->effect, "chroma_key");
  134. filter->key_rgb_param = gs_effect_get_param_by_name(
  135. filter->effect, "key_rgb");
  136. filter->pixel_size_param = gs_effect_get_param_by_name(
  137. filter->effect, "pixel_size");
  138. filter->similarity_param = gs_effect_get_param_by_name(
  139. filter->effect, "similarity");
  140. filter->smoothness_param = gs_effect_get_param_by_name(
  141. filter->effect, "smoothness");
  142. filter->spill_param = gs_effect_get_param_by_name(
  143. filter->effect, "spill");
  144. }
  145. obs_leave_graphics();
  146. bfree(effect_path);
  147. if (!filter->effect) {
  148. chroma_key_destroy(filter);
  149. return NULL;
  150. }
  151. chroma_key_update(filter, settings);
  152. return filter;
  153. }
  154. static void chroma_key_render(void *data, gs_effect_t *effect)
  155. {
  156. struct chroma_key_filter_data *filter = data;
  157. obs_source_t *target = obs_filter_get_target(filter->context);
  158. uint32_t width = obs_source_get_base_width(target);
  159. uint32_t height = obs_source_get_base_height(target);
  160. struct vec2 pixel_size;
  161. obs_source_process_filter_begin(filter->context, GS_RGBA,
  162. OBS_ALLOW_DIRECT_RENDERING);
  163. vec2_set(&pixel_size, 1.0f / (float)width, 1.0f / (float)height);
  164. gs_effect_set_vec4(filter->color_param, &filter->color);
  165. gs_effect_set_float(filter->contrast_param, filter->contrast);
  166. gs_effect_set_float(filter->brightness_param, filter->brightness);
  167. gs_effect_set_float(filter->gamma_param, filter->gamma);
  168. gs_effect_set_vec2(filter->chroma_param, &filter->chroma);
  169. gs_effect_set_vec4(filter->key_rgb_param, &filter->key_rgb);
  170. gs_effect_set_vec2(filter->pixel_size_param, &pixel_size);
  171. gs_effect_set_float(filter->similarity_param, filter->similarity);
  172. gs_effect_set_float(filter->smoothness_param, filter->smoothness);
  173. gs_effect_set_float(filter->spill_param, filter->spill);
  174. obs_source_process_filter_end(filter->context, filter->effect, 0, 0);
  175. UNUSED_PARAMETER(effect);
  176. }
  177. static bool key_type_changed(obs_properties_t *props, obs_property_t *p,
  178. obs_data_t *settings)
  179. {
  180. const char *type = obs_data_get_string(settings, SETTING_COLOR_TYPE);
  181. bool custom = strcmp(type, "custom") == 0;
  182. obs_property_set_visible(obs_properties_get(props, SETTING_KEY_COLOR),
  183. custom);
  184. UNUSED_PARAMETER(p);
  185. return true;
  186. }
  187. static obs_properties_t *chroma_key_properties(void *data)
  188. {
  189. obs_properties_t *props = obs_properties_create();
  190. obs_property_t *p = obs_properties_add_list(props,
  191. SETTING_COLOR_TYPE, TEXT_COLOR_TYPE,
  192. OBS_COMBO_TYPE_LIST, OBS_COMBO_FORMAT_STRING);
  193. obs_property_list_add_string(p, obs_module_text("Green"), "green");
  194. obs_property_list_add_string(p, obs_module_text("Blue"), "blue");
  195. obs_property_list_add_string(p, obs_module_text("Magenta"), "magenta");
  196. obs_property_list_add_string(p, obs_module_text("Custom"), "custom");
  197. obs_property_set_modified_callback(p, key_type_changed);
  198. obs_properties_add_color(props, SETTING_KEY_COLOR, TEXT_KEY_COLOR);
  199. obs_properties_add_int_slider(props, SETTING_SIMILARITY,
  200. TEXT_SIMILARITY, 1, 1000, 1);
  201. obs_properties_add_int_slider(props, SETTING_SMOOTHNESS,
  202. TEXT_SMOOTHNESS, 1, 1000, 1);
  203. obs_properties_add_int_slider(props, SETTING_SPILL,
  204. TEXT_SPILL, 1, 1000, 1);
  205. obs_properties_add_int(props, SETTING_OPACITY, TEXT_OPACITY, 0, 100, 1);
  206. obs_properties_add_float_slider(props, SETTING_CONTRAST,
  207. TEXT_CONTRAST, -1.0, 1.0, 0.01);
  208. obs_properties_add_float_slider(props, SETTING_BRIGHTNESS,
  209. TEXT_BRIGHTNESS, -1.0, 1.0, 0.01);
  210. obs_properties_add_float_slider(props, SETTING_GAMMA,
  211. TEXT_GAMMA, -1.0, 1.0, 0.01);
  212. UNUSED_PARAMETER(data);
  213. return props;
  214. }
  215. static void chroma_key_defaults(obs_data_t *settings)
  216. {
  217. obs_data_set_default_int(settings, SETTING_OPACITY, 100);
  218. obs_data_set_default_double(settings, SETTING_CONTRAST, 0.0);
  219. obs_data_set_default_double(settings, SETTING_BRIGHTNESS, 0.0);
  220. obs_data_set_default_double(settings, SETTING_GAMMA, 0.0);
  221. obs_data_set_default_int(settings, SETTING_KEY_COLOR, 0x00FF00);
  222. obs_data_set_default_string(settings, SETTING_COLOR_TYPE, "green");
  223. obs_data_set_default_int(settings, SETTING_SIMILARITY, 400);
  224. obs_data_set_default_int(settings, SETTING_SMOOTHNESS, 80);
  225. obs_data_set_default_int(settings, SETTING_SPILL, 100);
  226. }
  227. struct obs_source_info chroma_key_filter = {
  228. .id = "chroma_key_filter",
  229. .type = OBS_SOURCE_TYPE_FILTER,
  230. .output_flags = OBS_SOURCE_VIDEO,
  231. .get_name = chroma_key_name,
  232. .create = chroma_key_create,
  233. .destroy = chroma_key_destroy,
  234. .video_render = chroma_key_render,
  235. .update = chroma_key_update,
  236. .get_properties = chroma_key_properties,
  237. .get_defaults = chroma_key_defaults
  238. };