chroma-key-filter.c 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536
  1. #include <obs-module.h>
  2. #include <graphics/matrix4.h>
  3. #include <graphics/vec2.h>
  4. #include <graphics/vec4.h>
  5. /* clang-format off */
  6. #define SETTING_OPACITY "opacity"
  7. #define SETTING_CONTRAST "contrast"
  8. #define SETTING_BRIGHTNESS "brightness"
  9. #define SETTING_GAMMA "gamma"
  10. #define SETTING_COLOR_TYPE "key_color_type"
  11. #define SETTING_KEY_COLOR "key_color"
  12. #define SETTING_SIMILARITY "similarity"
  13. #define SETTING_SMOOTHNESS "smoothness"
  14. #define SETTING_SPILL "spill"
  15. #define TEXT_OPACITY obs_module_text("Opacity")
  16. #define TEXT_CONTRAST obs_module_text("Contrast")
  17. #define TEXT_BRIGHTNESS obs_module_text("Brightness")
  18. #define TEXT_GAMMA obs_module_text("Gamma")
  19. #define TEXT_COLOR_TYPE obs_module_text("KeyColorType")
  20. #define TEXT_KEY_COLOR obs_module_text("KeyColor")
  21. #define TEXT_SIMILARITY obs_module_text("Similarity")
  22. #define TEXT_SMOOTHNESS obs_module_text("Smoothness")
  23. #define TEXT_SPILL obs_module_text("ColorSpillReduction")
  24. /* clang-format on */
  25. struct chroma_key_filter_data {
  26. obs_source_t *context;
  27. gs_effect_t *effect;
  28. gs_eparam_t *color_param;
  29. gs_eparam_t *contrast_param;
  30. gs_eparam_t *brightness_param;
  31. gs_eparam_t *gamma_param;
  32. gs_eparam_t *pixel_size_param;
  33. gs_eparam_t *chroma_param;
  34. gs_eparam_t *similarity_param;
  35. gs_eparam_t *smoothness_param;
  36. gs_eparam_t *spill_param;
  37. struct vec4 color;
  38. float contrast;
  39. float brightness;
  40. float gamma;
  41. struct vec2 chroma;
  42. float similarity;
  43. float smoothness;
  44. float spill;
  45. };
  46. struct chroma_key_filter_data_v2 {
  47. obs_source_t *context;
  48. gs_effect_t *effect;
  49. gs_eparam_t *opacity_param;
  50. gs_eparam_t *contrast_param;
  51. gs_eparam_t *brightness_param;
  52. gs_eparam_t *gamma_param;
  53. gs_eparam_t *pixel_size_param;
  54. gs_eparam_t *chroma_param;
  55. gs_eparam_t *similarity_param;
  56. gs_eparam_t *smoothness_param;
  57. gs_eparam_t *spill_param;
  58. float opacity;
  59. float contrast;
  60. float brightness;
  61. float gamma;
  62. struct vec2 chroma;
  63. float similarity;
  64. float smoothness;
  65. float spill;
  66. };
  67. static const char *chroma_key_name(void *unused)
  68. {
  69. UNUSED_PARAMETER(unused);
  70. return obs_module_text("ChromaKeyFilter");
  71. }
  72. static const float cb_vec[] = {-0.100644f, -0.338572f, 0.439216f, 0.501961f};
  73. static const float cr_vec[] = {0.439216f, -0.398942f, -0.040274f, 0.501961f};
  74. static inline void
  75. color_settings_update_v1(struct chroma_key_filter_data *filter,
  76. obs_data_t *settings)
  77. {
  78. uint32_t opacity =
  79. (uint32_t)obs_data_get_int(settings, SETTING_OPACITY);
  80. uint32_t color = 0xFFFFFF | (((opacity * 255) / 100) << 24);
  81. double contrast = obs_data_get_double(settings, SETTING_CONTRAST);
  82. double brightness = obs_data_get_double(settings, SETTING_BRIGHTNESS);
  83. double gamma = obs_data_get_double(settings, SETTING_GAMMA);
  84. contrast = (contrast < 0.0) ? (1.0 / (-contrast + 1.0))
  85. : (contrast + 1.0);
  86. brightness *= 0.5;
  87. gamma = (gamma < 0.0) ? (-gamma + 1.0) : (1.0 / (gamma + 1.0));
  88. filter->contrast = (float)contrast;
  89. filter->brightness = (float)brightness;
  90. filter->gamma = (float)gamma;
  91. vec4_from_rgba(&filter->color, color);
  92. }
  93. static inline void
  94. color_settings_update_v2(struct chroma_key_filter_data_v2 *filter,
  95. obs_data_t *settings)
  96. {
  97. filter->opacity = (float)obs_data_get_double(settings, SETTING_OPACITY);
  98. double contrast = obs_data_get_double(settings, SETTING_CONTRAST);
  99. contrast = (contrast < 0.0) ? (1.0 / (-contrast + 1.0))
  100. : (contrast + 1.0);
  101. filter->contrast = (float)contrast;
  102. filter->brightness =
  103. (float)obs_data_get_double(settings, SETTING_BRIGHTNESS);
  104. double gamma = obs_data_get_double(settings, SETTING_GAMMA);
  105. gamma = (gamma < 0.0) ? (-gamma + 1.0) : (1.0 / (gamma + 1.0));
  106. filter->gamma = (float)gamma;
  107. }
  108. static inline void
  109. chroma_settings_update_v1(struct chroma_key_filter_data *filter,
  110. obs_data_t *settings)
  111. {
  112. int64_t similarity = obs_data_get_int(settings, SETTING_SIMILARITY);
  113. int64_t smoothness = obs_data_get_int(settings, SETTING_SMOOTHNESS);
  114. int64_t spill = obs_data_get_int(settings, SETTING_SPILL);
  115. uint32_t key_color =
  116. (uint32_t)obs_data_get_int(settings, SETTING_KEY_COLOR);
  117. const char *key_type =
  118. obs_data_get_string(settings, SETTING_COLOR_TYPE);
  119. struct vec4 key_rgb;
  120. struct vec4 cb_v4;
  121. struct vec4 cr_v4;
  122. if (strcmp(key_type, "green") == 0)
  123. key_color = 0x00FF00;
  124. else if (strcmp(key_type, "blue") == 0)
  125. key_color = 0xFF9900;
  126. else if (strcmp(key_type, "magenta") == 0)
  127. key_color = 0xFF00FF;
  128. vec4_from_rgba(&key_rgb, key_color | 0xFF000000);
  129. memcpy(&cb_v4, cb_vec, sizeof(cb_v4));
  130. memcpy(&cr_v4, cr_vec, sizeof(cr_v4));
  131. filter->chroma.x = vec4_dot(&key_rgb, &cb_v4);
  132. filter->chroma.y = vec4_dot(&key_rgb, &cr_v4);
  133. filter->similarity = (float)similarity / 1000.0f;
  134. filter->smoothness = (float)smoothness / 1000.0f;
  135. filter->spill = (float)spill / 1000.0f;
  136. }
  137. static inline void
  138. chroma_settings_update_v2(struct chroma_key_filter_data_v2 *filter,
  139. obs_data_t *settings)
  140. {
  141. int64_t similarity = obs_data_get_int(settings, SETTING_SIMILARITY);
  142. int64_t smoothness = obs_data_get_int(settings, SETTING_SMOOTHNESS);
  143. int64_t spill = obs_data_get_int(settings, SETTING_SPILL);
  144. uint32_t key_color =
  145. (uint32_t)obs_data_get_int(settings, SETTING_KEY_COLOR);
  146. const char *key_type =
  147. obs_data_get_string(settings, SETTING_COLOR_TYPE);
  148. struct vec4 key_rgb;
  149. struct vec4 cb_v4;
  150. struct vec4 cr_v4;
  151. if (strcmp(key_type, "green") == 0)
  152. key_color = 0x00FF00;
  153. else if (strcmp(key_type, "blue") == 0)
  154. key_color = 0xFF9900;
  155. else if (strcmp(key_type, "magenta") == 0)
  156. key_color = 0xFF00FF;
  157. vec4_from_rgba(&key_rgb, key_color | 0xFF000000);
  158. memcpy(&cb_v4, cb_vec, sizeof(cb_v4));
  159. memcpy(&cr_v4, cr_vec, sizeof(cr_v4));
  160. filter->chroma.x = vec4_dot(&key_rgb, &cb_v4);
  161. filter->chroma.y = vec4_dot(&key_rgb, &cr_v4);
  162. filter->similarity = (float)similarity / 1000.0f;
  163. filter->smoothness = (float)smoothness / 1000.0f;
  164. filter->spill = (float)spill / 1000.0f;
  165. }
  166. static void chroma_key_update_v1(void *data, obs_data_t *settings)
  167. {
  168. struct chroma_key_filter_data *filter = data;
  169. color_settings_update_v1(filter, settings);
  170. chroma_settings_update_v1(filter, settings);
  171. }
  172. static void chroma_key_update_v2(void *data, obs_data_t *settings)
  173. {
  174. struct chroma_key_filter_data_v2 *filter = data;
  175. color_settings_update_v2(filter, settings);
  176. chroma_settings_update_v2(filter, settings);
  177. }
  178. static void chroma_key_destroy_v1(void *data)
  179. {
  180. struct chroma_key_filter_data *filter = data;
  181. if (filter->effect) {
  182. obs_enter_graphics();
  183. gs_effect_destroy(filter->effect);
  184. obs_leave_graphics();
  185. }
  186. bfree(data);
  187. }
  188. static void chroma_key_destroy_v2(void *data)
  189. {
  190. struct chroma_key_filter_data_v2 *filter = data;
  191. if (filter->effect) {
  192. obs_enter_graphics();
  193. gs_effect_destroy(filter->effect);
  194. obs_leave_graphics();
  195. }
  196. bfree(data);
  197. }
  198. static void *chroma_key_create_v1(obs_data_t *settings, obs_source_t *context)
  199. {
  200. struct chroma_key_filter_data *filter =
  201. bzalloc(sizeof(struct chroma_key_filter_data));
  202. char *effect_path = obs_module_file("chroma_key_filter.effect");
  203. filter->context = context;
  204. obs_enter_graphics();
  205. filter->effect = gs_effect_create_from_file(effect_path, NULL);
  206. if (filter->effect) {
  207. filter->color_param =
  208. gs_effect_get_param_by_name(filter->effect, "color");
  209. filter->contrast_param =
  210. gs_effect_get_param_by_name(filter->effect, "contrast");
  211. filter->brightness_param = gs_effect_get_param_by_name(
  212. filter->effect, "brightness");
  213. filter->gamma_param =
  214. gs_effect_get_param_by_name(filter->effect, "gamma");
  215. filter->chroma_param = gs_effect_get_param_by_name(
  216. filter->effect, "chroma_key");
  217. filter->pixel_size_param = gs_effect_get_param_by_name(
  218. filter->effect, "pixel_size");
  219. filter->similarity_param = gs_effect_get_param_by_name(
  220. filter->effect, "similarity");
  221. filter->smoothness_param = gs_effect_get_param_by_name(
  222. filter->effect, "smoothness");
  223. filter->spill_param =
  224. gs_effect_get_param_by_name(filter->effect, "spill");
  225. }
  226. obs_leave_graphics();
  227. bfree(effect_path);
  228. if (!filter->effect) {
  229. chroma_key_destroy_v1(filter);
  230. return NULL;
  231. }
  232. chroma_key_update_v1(filter, settings);
  233. return filter;
  234. }
  235. static void *chroma_key_create_v2(obs_data_t *settings, obs_source_t *context)
  236. {
  237. struct chroma_key_filter_data_v2 *filter =
  238. bzalloc(sizeof(struct chroma_key_filter_data_v2));
  239. char *effect_path = obs_module_file("chroma_key_filter_v2.effect");
  240. filter->context = context;
  241. obs_enter_graphics();
  242. filter->effect = gs_effect_create_from_file(effect_path, NULL);
  243. if (filter->effect) {
  244. filter->opacity_param =
  245. gs_effect_get_param_by_name(filter->effect, "opacity");
  246. filter->contrast_param =
  247. gs_effect_get_param_by_name(filter->effect, "contrast");
  248. filter->brightness_param = gs_effect_get_param_by_name(
  249. filter->effect, "brightness");
  250. filter->gamma_param =
  251. gs_effect_get_param_by_name(filter->effect, "gamma");
  252. filter->chroma_param = gs_effect_get_param_by_name(
  253. filter->effect, "chroma_key");
  254. filter->pixel_size_param = gs_effect_get_param_by_name(
  255. filter->effect, "pixel_size");
  256. filter->similarity_param = gs_effect_get_param_by_name(
  257. filter->effect, "similarity");
  258. filter->smoothness_param = gs_effect_get_param_by_name(
  259. filter->effect, "smoothness");
  260. filter->spill_param =
  261. gs_effect_get_param_by_name(filter->effect, "spill");
  262. }
  263. obs_leave_graphics();
  264. bfree(effect_path);
  265. if (!filter->effect) {
  266. chroma_key_destroy_v2(filter);
  267. return NULL;
  268. }
  269. chroma_key_update_v2(filter, settings);
  270. return filter;
  271. }
  272. static void chroma_key_render_v1(void *data, gs_effect_t *effect)
  273. {
  274. struct chroma_key_filter_data *filter = data;
  275. obs_source_t *target = obs_filter_get_target(filter->context);
  276. uint32_t width = obs_source_get_base_width(target);
  277. uint32_t height = obs_source_get_base_height(target);
  278. struct vec2 pixel_size;
  279. if (!obs_source_process_filter_begin(filter->context, GS_RGBA,
  280. OBS_ALLOW_DIRECT_RENDERING))
  281. return;
  282. vec2_set(&pixel_size, 1.0f / (float)width, 1.0f / (float)height);
  283. gs_effect_set_vec4(filter->color_param, &filter->color);
  284. gs_effect_set_float(filter->contrast_param, filter->contrast);
  285. gs_effect_set_float(filter->brightness_param, filter->brightness);
  286. gs_effect_set_float(filter->gamma_param, filter->gamma);
  287. gs_effect_set_vec2(filter->chroma_param, &filter->chroma);
  288. gs_effect_set_vec2(filter->pixel_size_param, &pixel_size);
  289. gs_effect_set_float(filter->similarity_param, filter->similarity);
  290. gs_effect_set_float(filter->smoothness_param, filter->smoothness);
  291. gs_effect_set_float(filter->spill_param, filter->spill);
  292. obs_source_process_filter_end(filter->context, filter->effect, 0, 0);
  293. UNUSED_PARAMETER(effect);
  294. }
  295. static void chroma_key_render_v2(void *data, gs_effect_t *effect)
  296. {
  297. struct chroma_key_filter_data_v2 *filter = data;
  298. obs_source_t *target = obs_filter_get_target(filter->context);
  299. uint32_t width = obs_source_get_base_width(target);
  300. uint32_t height = obs_source_get_base_height(target);
  301. struct vec2 pixel_size;
  302. if (!obs_source_process_filter_begin(filter->context, GS_RGBA,
  303. OBS_ALLOW_DIRECT_RENDERING))
  304. return;
  305. vec2_set(&pixel_size, 1.0f / (float)width, 1.0f / (float)height);
  306. gs_effect_set_float(filter->opacity_param, filter->opacity);
  307. gs_effect_set_float(filter->contrast_param, filter->contrast);
  308. gs_effect_set_float(filter->brightness_param, filter->brightness);
  309. gs_effect_set_float(filter->gamma_param, filter->gamma);
  310. gs_effect_set_vec2(filter->chroma_param, &filter->chroma);
  311. gs_effect_set_vec2(filter->pixel_size_param, &pixel_size);
  312. gs_effect_set_float(filter->similarity_param, filter->similarity);
  313. gs_effect_set_float(filter->smoothness_param, filter->smoothness);
  314. gs_effect_set_float(filter->spill_param, filter->spill);
  315. gs_blend_state_push();
  316. gs_blend_function(GS_BLEND_ONE, GS_BLEND_INVSRCALPHA);
  317. obs_source_process_filter_end(filter->context, filter->effect, 0, 0);
  318. gs_blend_state_pop();
  319. UNUSED_PARAMETER(effect);
  320. }
  321. static bool key_type_changed(obs_properties_t *props, obs_property_t *p,
  322. obs_data_t *settings)
  323. {
  324. const char *type = obs_data_get_string(settings, SETTING_COLOR_TYPE);
  325. bool custom = strcmp(type, "custom") == 0;
  326. obs_property_set_visible(obs_properties_get(props, SETTING_KEY_COLOR),
  327. custom);
  328. UNUSED_PARAMETER(p);
  329. return true;
  330. }
  331. static obs_properties_t *chroma_key_properties_v1(void *data)
  332. {
  333. obs_properties_t *props = obs_properties_create();
  334. obs_property_t *p = obs_properties_add_list(props, SETTING_COLOR_TYPE,
  335. TEXT_COLOR_TYPE,
  336. OBS_COMBO_TYPE_LIST,
  337. OBS_COMBO_FORMAT_STRING);
  338. obs_property_list_add_string(p, obs_module_text("Green"), "green");
  339. obs_property_list_add_string(p, obs_module_text("Blue"), "blue");
  340. obs_property_list_add_string(p, obs_module_text("Magenta"), "magenta");
  341. obs_property_list_add_string(p, obs_module_text("Custom"), "custom");
  342. obs_property_set_modified_callback(p, key_type_changed);
  343. obs_properties_add_color(props, SETTING_KEY_COLOR, TEXT_KEY_COLOR);
  344. obs_properties_add_int_slider(props, SETTING_SIMILARITY,
  345. TEXT_SIMILARITY, 1, 1000, 1);
  346. obs_properties_add_int_slider(props, SETTING_SMOOTHNESS,
  347. TEXT_SMOOTHNESS, 1, 1000, 1);
  348. obs_properties_add_int_slider(props, SETTING_SPILL, TEXT_SPILL, 1, 1000,
  349. 1);
  350. obs_properties_add_int_slider(props, SETTING_OPACITY, TEXT_OPACITY, 0,
  351. 100, 1);
  352. obs_properties_add_float_slider(props, SETTING_CONTRAST, TEXT_CONTRAST,
  353. -1.0, 1.0, 0.01);
  354. obs_properties_add_float_slider(props, SETTING_BRIGHTNESS,
  355. TEXT_BRIGHTNESS, -1.0, 1.0, 0.01);
  356. obs_properties_add_float_slider(props, SETTING_GAMMA, TEXT_GAMMA, -1.0,
  357. 1.0, 0.01);
  358. UNUSED_PARAMETER(data);
  359. return props;
  360. }
  361. static obs_properties_t *chroma_key_properties_v2(void *data)
  362. {
  363. obs_properties_t *props = obs_properties_create();
  364. obs_property_t *p = obs_properties_add_list(props, SETTING_COLOR_TYPE,
  365. TEXT_COLOR_TYPE,
  366. OBS_COMBO_TYPE_LIST,
  367. OBS_COMBO_FORMAT_STRING);
  368. obs_property_list_add_string(p, obs_module_text("Green"), "green");
  369. obs_property_list_add_string(p, obs_module_text("Blue"), "blue");
  370. obs_property_list_add_string(p, obs_module_text("Magenta"), "magenta");
  371. obs_property_list_add_string(p, obs_module_text("Custom"), "custom");
  372. obs_property_set_modified_callback(p, key_type_changed);
  373. obs_properties_add_color(props, SETTING_KEY_COLOR, TEXT_KEY_COLOR);
  374. obs_properties_add_int_slider(props, SETTING_SIMILARITY,
  375. TEXT_SIMILARITY, 1, 1000, 1);
  376. obs_properties_add_int_slider(props, SETTING_SMOOTHNESS,
  377. TEXT_SMOOTHNESS, 1, 1000, 1);
  378. obs_properties_add_int_slider(props, SETTING_SPILL, TEXT_SPILL, 1, 1000,
  379. 1);
  380. obs_properties_add_float_slider(props, SETTING_OPACITY, TEXT_OPACITY,
  381. 0.0, 1.0, 0.0001);
  382. obs_properties_add_float_slider(props, SETTING_CONTRAST, TEXT_CONTRAST,
  383. -4.0, 4.0, 0.01);
  384. obs_properties_add_float_slider(props, SETTING_BRIGHTNESS,
  385. TEXT_BRIGHTNESS, -1.0, 1.0, 0.0001);
  386. obs_properties_add_float_slider(props, SETTING_GAMMA, TEXT_GAMMA, -1.0,
  387. 1.0, 0.01);
  388. UNUSED_PARAMETER(data);
  389. return props;
  390. }
  391. static void chroma_key_defaults_v1(obs_data_t *settings)
  392. {
  393. obs_data_set_default_int(settings, SETTING_OPACITY, 100);
  394. obs_data_set_default_double(settings, SETTING_CONTRAST, 0.0);
  395. obs_data_set_default_double(settings, SETTING_BRIGHTNESS, 0.0);
  396. obs_data_set_default_double(settings, SETTING_GAMMA, 0.0);
  397. obs_data_set_default_int(settings, SETTING_KEY_COLOR, 0x00FF00);
  398. obs_data_set_default_string(settings, SETTING_COLOR_TYPE, "green");
  399. obs_data_set_default_int(settings, SETTING_SIMILARITY, 400);
  400. obs_data_set_default_int(settings, SETTING_SMOOTHNESS, 80);
  401. obs_data_set_default_int(settings, SETTING_SPILL, 100);
  402. }
  403. static void chroma_key_defaults_v2(obs_data_t *settings)
  404. {
  405. obs_data_set_default_double(settings, SETTING_OPACITY, 1.0);
  406. obs_data_set_default_double(settings, SETTING_CONTRAST, 0.0);
  407. obs_data_set_default_double(settings, SETTING_BRIGHTNESS, 0.0);
  408. obs_data_set_default_double(settings, SETTING_GAMMA, 0.0);
  409. obs_data_set_default_int(settings, SETTING_KEY_COLOR, 0x00FF00);
  410. obs_data_set_default_string(settings, SETTING_COLOR_TYPE, "green");
  411. obs_data_set_default_int(settings, SETTING_SIMILARITY, 400);
  412. obs_data_set_default_int(settings, SETTING_SMOOTHNESS, 80);
  413. obs_data_set_default_int(settings, SETTING_SPILL, 100);
  414. }
  415. struct obs_source_info chroma_key_filter = {
  416. .id = "chroma_key_filter",
  417. .type = OBS_SOURCE_TYPE_FILTER,
  418. .output_flags = OBS_SOURCE_VIDEO | OBS_SOURCE_CAP_OBSOLETE,
  419. .get_name = chroma_key_name,
  420. .create = chroma_key_create_v1,
  421. .destroy = chroma_key_destroy_v1,
  422. .video_render = chroma_key_render_v1,
  423. .update = chroma_key_update_v1,
  424. .get_properties = chroma_key_properties_v1,
  425. .get_defaults = chroma_key_defaults_v1,
  426. };
  427. struct obs_source_info chroma_key_filter_v2 = {
  428. .id = "chroma_key_filter",
  429. .version = 2,
  430. .type = OBS_SOURCE_TYPE_FILTER,
  431. .output_flags = OBS_SOURCE_VIDEO | OBS_SOURCE_SRGB,
  432. .get_name = chroma_key_name,
  433. .create = chroma_key_create_v2,
  434. .destroy = chroma_key_destroy_v2,
  435. .video_render = chroma_key_render_v2,
  436. .update = chroma_key_update_v2,
  437. .get_properties = chroma_key_properties_v2,
  438. .get_defaults = chroma_key_defaults_v2,
  439. };