4 * Copyright (C) 1997-2020 Adam Williams <broadcast at earthling dot net>
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version.
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, write to the Free Software
18 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
24 #include "dragcheckbox.h"
28 #include "mainerror.h"
30 #include "pluginserver.h"
31 #include "pluginvclient.h"
40 class BoxBlurNumISlider;
41 class BoxBlurNumIText;
42 class BoxBlurNumClear;
60 void copy_from(BoxBlurConfig &that);
61 int equivalent(BoxBlurConfig &that);
62 void interpolate(BoxBlurConfig &prev, BoxBlurConfig &next,
63 int64_t prev_frame, int64_t next_frame, int64_t current_frame);
67 int horz_radius, vert_radius, power;
74 class BoxBlurNumISlider : public BC_ISlider
77 BoxBlurNumISlider(BoxBlurNum *num, int x, int y);
82 class BoxBlurNumIText : public BC_TumbleTextBox
85 BoxBlurNumIText(BoxBlurNum *num, int x, int y);
90 class BoxBlurNumClear : public BC_Button
93 BoxBlurNumClear(BoxBlurNum *num, int x, int y);
94 static int calculate_w(BoxBlurNum *num);
103 BoxBlurNum(BoxBlurWindow *gui, int x, int y, int w,
104 const char *name, int *iv, int imn, int imx);
106 void create_objects();
107 void update(int value);
114 int imn, imx, *ivalue;
115 int title_w, text_w, slider_w;
117 BoxBlurNumIText *text;
118 BoxBlurNumISlider *slider;
119 BoxBlurNumClear *clear;
123 class BoxBlurRadius : public BoxBlurNum
126 BoxBlurRadius(BoxBlurWindow *gui, int x, int y, int w,
127 const char *name, int *radius);
130 class BoxBlurPower : public BoxBlurNum
133 BoxBlurPower(BoxBlurWindow *gui, int x, int y, int w,
134 const char *name, int *power);
137 class BoxBlurX : public BC_TumbleTextBox
140 BoxBlurX(BoxBlurWindow *gui, int x, int y);
144 class BoxBlurY : public BC_TumbleTextBox
147 BoxBlurY(BoxBlurWindow *gui, int x, int y);
151 class BoxBlurW : public BC_TumbleTextBox
154 BoxBlurW(BoxBlurWindow *gui, int x, int y);
158 class BoxBlurH : public BC_TumbleTextBox
161 BoxBlurH(BoxBlurWindow *gui, int x, int y);
166 class BoxBlurDrag : public DragCheckBox
169 BoxBlurDrag(BoxBlurWindow *gui, BoxBlurEffect *plugin, int x, int y);
172 Track *get_drag_track();
173 int64_t get_drag_position();
174 static int calculate_w(BoxBlurWindow *gui);
177 BoxBlurEffect *plugin;
180 class BoxBlurReset : public BC_GenericButton
183 BoxBlurReset(BoxBlurWindow *gui, int x, int y);
185 static int calculate_w(BoxBlurWindow *gui);
190 class BoxBlurPreset : public BC_GenericButton
193 BoxBlurPreset(BoxBlurWindow *gui, int x, int y);
195 static int calculate_w(BoxBlurWindow *gui);
200 class BoxBlurWindow : public PluginClientWindow
203 BoxBlurWindow(BoxBlurEffect *plugin);
205 void create_objects();
209 BoxBlurEffect *plugin;
211 BoxBlurPreset *preset;
212 BoxBlurRadius *blur_horz;
213 BoxBlurRadius *blur_vert;
214 BoxBlurPower *blur_power;
223 class BoxBlurEffect : public PluginVClient
226 BoxBlurEffect(PluginServer *server);
229 PLUGIN_CLASS_MEMBERS(BoxBlurConfig)
230 void render_gui(void *data);
232 int process_realtime(VFrame *input, VFrame *output);
235 void save_data(KeyFrame *keyframe);
236 void read_data(KeyFrame *keyframe);
239 VFrame *input, *output;
245 void BoxBlurConfig::reset()
254 void BoxBlurConfig::preset()
264 BoxBlurConfig::BoxBlurConfig()
269 void BoxBlurConfig::copy_from(BoxBlurConfig &that)
271 horz_radius = that.horz_radius;
272 vert_radius = that.vert_radius;
274 box_x = that.box_x; box_y = that.box_y;
275 box_w = that.box_w; box_h = that.box_h;
278 int BoxBlurConfig::equivalent(BoxBlurConfig &that)
280 return horz_radius == that.horz_radius &&
281 vert_radius == that.vert_radius &&
282 power == that.power &&
283 EQUIV(box_x, that.box_x) && EQUIV(box_y, that.box_y) &&
284 box_w == that.box_w && box_h == that.box_h;
287 void BoxBlurConfig::interpolate(BoxBlurConfig &prev, BoxBlurConfig &next,
288 int64_t prev_frame, int64_t next_frame, int64_t current_frame)
290 double u = (double)(next_frame - current_frame) / (next_frame - prev_frame);
292 this->horz_radius = u*prev.horz_radius + v*next.horz_radius;
293 this->vert_radius = u*prev.vert_radius + v*next.vert_radius;
294 this->power = u*prev.power + v*next.power + 1e-6; // avoid truncation jitter
295 this->box_x = u*prev.box_x + v*next.box_x;
296 this->box_y = u*prev.box_y + v*next.box_y;
297 this->box_w = u*prev.box_w + v*next.box_w + 1e-6;
298 this->box_h = u*prev.box_h + v*next.box_h + 1e-6;
302 int BoxBlurNum::get_w() { return w; }
303 int BoxBlurNum::get_h() { return h; }
305 BoxBlurNumISlider::BoxBlurNumISlider(BoxBlurNum *num, int x, int y)
306 : BC_ISlider(x, y, 0, num->slider_w, num->slider_w,
307 num->imn, num->imx, *num->ivalue)
312 int BoxBlurNumISlider::handle_event()
314 int iv = get_value();
316 num->gui->update_drag();
320 BoxBlurNumIText::BoxBlurNumIText(BoxBlurNum *num, int x, int y)
321 : BC_TumbleTextBox(num->gui, *num->ivalue, num->imn, num->imx,
327 int BoxBlurNumIText::handle_event()
329 int iv = atoi(get_text());
331 num->gui->update_drag();
335 BoxBlurNumClear::BoxBlurNumClear(BoxBlurNum *num, int x, int y)
336 : BC_Button(x, y, num->gui->plugin->get_theme()->get_image_set("reset_button"))
341 int BoxBlurNumClear::calculate_w(BoxBlurNum *num)
343 VFrame **imgs = num->gui->plugin->get_theme()->get_image_set("reset_button");
344 return imgs[0]->get_w();
347 int BoxBlurNumClear::handle_event()
351 num->gui->update_drag();
355 BoxBlurNum::BoxBlurNum(BoxBlurWindow *gui, int x, int y, int w,
356 const char *name, int *iv, int imn, int imx)
367 int margin = gui->plugin->get_theme()->widget_border;
368 int clear_w = BoxBlurNumClear::calculate_w(this);
369 int tumble_w = BC_Tumbler::calculate_w();
370 int len = w - 2*margin - clear_w - tumble_w;
371 this->title_w = xS(60);
372 this->text_w = xS(60) - tumble_w;
373 this->slider_w = len - title_w - text_w - 2*margin;
381 BoxBlurNum::~BoxBlurNum()
386 void BoxBlurNum::create_objects()
389 gui->add_subwindow(title = new BC_Title(x1, y, name));
390 int margin = gui->plugin->get_theme()->widget_border;
391 x1 += title_w + margin;
392 text = new BoxBlurNumIText(this, x1, y);
393 text->create_objects();
394 x1 += text_w + BC_Tumbler::calculate_w() + margin;
395 gui->add_subwindow(slider = new BoxBlurNumISlider(this, x1, y));
396 x1 += slider_w + 2*margin;
397 gui->add_subwindow(clear = new BoxBlurNumClear(this, x1, y));
398 h = bmax(title->get_h(), bmax(text->get_h(),
399 bmax(slider->get_h(), clear->get_h())));
402 void BoxBlurNum::update(int value)
404 text->update((int64_t)value);
405 slider->update(value);
410 BoxBlurRadius::BoxBlurRadius(BoxBlurWindow *gui, int x, int y, int w,
411 const char *name, int *radius)
412 : BoxBlurNum(gui, x, y, w, name, radius, 0, 100)
416 BoxBlurPower::BoxBlurPower(BoxBlurWindow *gui, int x, int y, int w,
417 const char *name, int *power)
418 : BoxBlurNum(gui, x, y, w, name, power, 1, 10)
422 BoxBlurWindow::BoxBlurWindow(BoxBlurEffect *plugin)
423 : PluginClientWindow(plugin, xS(360), yS(260), xS(360), yS(260), 0)
425 this->plugin = plugin;
429 box_x = 0; box_y = 0;
430 box_w = 0; box_h = 0;
433 BoxBlurWindow::~BoxBlurWindow()
444 void BoxBlurWindow::create_objects()
446 int x = xS(10), y = yS(10);
447 int ys10 = yS(10), ys20 = yS(20), ys30 = yS(30), ys40 = yS(40);
448 int t1 = x, t2 = t1+xS(24), t3 = t2+xS(100), t4 = t3+xS(24);
449 int ww = get_w() - 2*x, bar_o = xS(30), bar_m = xS(15);
450 int margin = plugin->get_theme()->widget_border;
453 add_subwindow(tbar = new BC_TitleBar(x, y, ww, bar_o, bar_m, _("Position & Size")));
455 int x1 = ww - BoxBlurDrag::calculate_w(this) - margin;
456 if( plugin->drag && drag->drag_activate() ) {
457 eprintf("drag enabled, but compositor already grabbed\n");
460 add_subwindow(drag = new BoxBlurDrag(this, plugin, x1, y));
461 drag->create_objects();
464 add_subwindow(title = new BC_Title(t1, y, _("X:")));
465 box_x = new BoxBlurX(this, t2, y);
466 box_x->create_objects();
467 add_subwindow(title = new BC_Title(t3, y, _("W:")));
468 box_w = new BoxBlurW(this, t4, y);
469 box_w->create_objects();
471 add_subwindow(title = new BC_Title(t1, y, _("Y:")));
472 box_y = new BoxBlurY(this, t2, y);
473 box_y->create_objects();
474 add_subwindow(title = new BC_Title(t3, y, _("H:")));
475 box_h = new BoxBlurH(this, t4, y);
476 box_h->create_objects();
479 add_subwindow(tbar = new BC_TitleBar(x, y, ww, bar_o, bar_m, _("Blur")));
481 blur_horz = new BoxBlurRadius(this, x, y, ww, _("Horz:"),
482 &plugin->config.horz_radius);
483 blur_horz->create_objects();
485 blur_vert = new BoxBlurRadius(this, x, y, ww, _("Vert:"),
486 &plugin->config.vert_radius);
487 blur_vert->create_objects();
489 blur_power = new BoxBlurPower(this, x, y, ww, _("Power:"),
490 &plugin->config.power);
491 blur_power->create_objects();
494 add_subwindow(bar = new BC_Bar(x, y, ww));
497 add_subwindow(reset = new BoxBlurReset(this, x, y));
498 x1 = x + ww - BoxBlurPreset::calculate_w(this);
499 add_subwindow(preset = new BoxBlurPreset(this, x1, y));
500 y += bmax(title->get_h(), reset->get_h()) + 2*margin;
504 void BoxBlurWindow::update_gui()
506 BoxBlurConfig &config = plugin->config;
507 blur_horz->update(config.horz_radius);
508 blur_vert->update(config.vert_radius);
509 blur_power->update(config.power);
510 box_x->update(config.box_x);
511 box_y->update(config.box_y);
512 box_w->update((int64_t)config.box_w);
513 box_h->update((int64_t)config.box_h);
514 drag->drag_x = config.box_x;
515 drag->drag_y = config.box_y;
516 drag->drag_w = config.box_w;
517 drag->drag_h = config.box_h;
521 REGISTER_PLUGIN(BoxBlurEffect)
522 NEW_WINDOW_MACRO(BoxBlurEffect, BoxBlurWindow)
523 LOAD_CONFIGURATION_MACRO(BoxBlurEffect, BoxBlurConfig)
525 void BoxBlurEffect::render_gui(void *data)
527 BoxBlurEffect *box_blur = (BoxBlurEffect *)data;
528 box_blur->drag = drag;
531 int BoxBlurEffect::is_dragging()
534 send_render_gui(this);
539 BoxBlurEffect::BoxBlurEffect(PluginServer *server)
540 : PluginVClient(server)
546 BoxBlurEffect::~BoxBlurEffect()
551 const char* BoxBlurEffect::plugin_title() { return N_("BoxBlur"); }
552 int BoxBlurEffect::is_realtime() { return 1; }
555 void BoxBlurEffect::save_data(KeyFrame *keyframe)
558 output.set_shared_output(keyframe->xbuf);
559 output.tag.set_title("BOXBLUR");
560 output.tag.set_property("HORZ_RADIUS", config.horz_radius);
561 output.tag.set_property("VERT_RADIUS", config.vert_radius);
562 output.tag.set_property("POWER", config.power);
563 output.tag.set_property("BOX_X", config.box_x);
564 output.tag.set_property("BOX_Y", config.box_y);
565 output.tag.set_property("BOX_W", config.box_w);
566 output.tag.set_property("BOX_H", config.box_h);
568 output.tag.set_title("/BOXBLUR");
570 output.append_newline();
571 output.terminate_string();
574 void BoxBlurEffect::read_data(KeyFrame *keyframe)
577 input.set_shared_input(keyframe->xbuf);
580 while( !(result = input.read_tag()) ) {
581 if( input.tag.title_is("BOXBLUR") ) {
582 config.horz_radius = input.tag.get_property("HORZ_RADIUS", config.horz_radius);
583 config.vert_radius = input.tag.get_property("VERT_RADIUS", config.vert_radius);
584 config.power = input.tag.get_property("POWER", config.power);
585 config.box_x = input.tag.get_property("BOX_X", config.box_x);
586 config.box_y = input.tag.get_property("BOX_Y", config.box_y);
587 config.box_w = input.tag.get_property("BOX_W", config.box_w);
588 config.box_h = input.tag.get_property("BOX_H", config.box_h);
593 void BoxBlurEffect::draw_boundry()
595 if( !gui_open() ) return;
596 int box_x = config.box_x, box_y = config.box_y;
597 int box_w = config.box_w ? config.box_w : input->get_w();
598 int box_h = config.box_h ? config.box_h : input->get_h();
599 DragCheckBox::draw_boundary(output, box_x, box_y, box_w, box_h);
602 int BoxBlurEffect::process_realtime(VFrame *input, VFrame *output)
605 this->output = output;
606 load_configuration();
607 int out_w = output->get_w(), out_h = output->get_h();
610 int cpus = (out_w * out_h)/0x80000 + 1;
611 box_blur = new BoxBlur(cpus);
613 int x = config.box_x, y = config.box_y;
614 int ow = config.box_w ? config.box_w : out_w;
615 int oh = config.box_h ? config.box_h : out_h;
616 if( config.horz_radius ) {
617 box_blur->hblur(output, input, config.horz_radius, config.power,
621 if( config.vert_radius ) {
622 box_blur->vblur(output, input, config.vert_radius, config.power,
632 void BoxBlurEffect::update_gui()
634 if( !thread ) return;
635 load_configuration();
636 thread->window->lock_window("BoxBlurEffect::update_gui");
637 BoxBlurWindow *gui = (BoxBlurWindow *)thread->window;
639 thread->window->unlock_window();
643 BoxBlurX::BoxBlurX(BoxBlurWindow *gui, int x, int y)
644 : BC_TumbleTextBox(gui, gui->plugin->config.box_x,
645 -32767.f, 32767.f, x, y, xS(64))
650 int BoxBlurX::handle_event()
652 gui->plugin->config.box_x = atof(get_text());
657 BoxBlurY::BoxBlurY(BoxBlurWindow *gui, int x, int y)
658 : BC_TumbleTextBox(gui, gui->plugin->config.box_y,
659 -32767.f, 32767.f, x, y, xS(64))
664 int BoxBlurY::handle_event()
666 gui->plugin->config.box_y = atof(get_text());
671 BoxBlurW::BoxBlurW(BoxBlurWindow *gui, int x, int y)
672 : BC_TumbleTextBox(gui, gui->plugin->config.box_w,
673 0, 32767, x, y, xS(64))
677 int BoxBlurW::handle_event()
679 gui->plugin->config.box_w = atol(get_text());
684 BoxBlurH::BoxBlurH(BoxBlurWindow *gui, int x, int y)
685 : BC_TumbleTextBox(gui, gui->plugin->config.box_h,
686 0, 32767, x, y, xS(64))
690 int BoxBlurH::handle_event()
692 gui->plugin->config.box_h = atol(get_text());
697 BoxBlurDrag::BoxBlurDrag(BoxBlurWindow *gui, BoxBlurEffect *plugin, int x, int y)
698 : DragCheckBox(plugin->server->mwindow, x, y, _("Drag"), &plugin->drag,
699 plugin->config.box_x, plugin->config.box_y,
700 plugin->config.box_w, plugin->config.box_h)
702 this->plugin = plugin;
706 int BoxBlurDrag::calculate_w(BoxBlurWindow *gui)
709 calculate_extents(gui, &w, &h, _("Drag"));
713 Track *BoxBlurDrag::get_drag_track()
715 PluginServer *server = plugin->server;
716 int plugin_id = server->plugin_id;
717 Plugin *plugin = server->edl->tracks->plugin_exists(plugin_id);
718 return !plugin ? 0 : plugin->track;
720 int64_t BoxBlurDrag::get_drag_position()
722 return plugin->get_source_position();
725 void BoxBlurDrag::update_gui()
727 plugin->drag = get_value();
728 plugin->config.box_x = drag_x;
729 plugin->config.box_y = drag_y;
730 plugin->config.box_w = drag_w+0.5;
731 plugin->config.box_h = drag_h+0.5;
732 gui->box_x->update((float)plugin->config.box_x);
733 gui->box_y->update((float)plugin->config.box_y);
734 gui->box_w->update((int64_t)plugin->config.box_w);
735 gui->box_h->update((int64_t)plugin->config.box_h);
736 plugin->send_configure_change();
739 int BoxBlurDrag::handle_event()
741 int ret = DragCheckBox::handle_event();
742 plugin->drag = get_value();
743 plugin->send_configure_change();
747 void BoxBlurWindow::update_drag()
749 drag->drag_x = plugin->config.box_x;
750 drag->drag_y = plugin->config.box_y;
751 drag->drag_w = plugin->config.box_w;
752 drag->drag_h = plugin->config.box_h;
753 plugin->send_configure_change();
756 BoxBlurReset::BoxBlurReset(BoxBlurWindow *gui, int x, int y)
757 : BC_GenericButton(x, y, _("Reset"))
762 int BoxBlurReset::calculate_w(BoxBlurWindow *gui)
764 return BC_GenericButton::calculate_w(gui,_("Reset"));
767 int BoxBlurReset::handle_event()
769 BoxBlurEffect *plugin = gui->plugin;
770 plugin->config.reset();
771 gui->drag->update(0);
772 gui->drag->drag_deactivate();
778 BoxBlurPreset::BoxBlurPreset(BoxBlurWindow *gui, int x, int y)
779 : BC_GenericButton(x, y, _("Default"))
784 int BoxBlurPreset::calculate_w(BoxBlurWindow *gui)
786 return BC_GenericButton::calculate_w(gui,_("Default"));
789 int BoxBlurPreset::handle_event()
791 BoxBlurEffect *plugin = gui->plugin;
792 plugin->config.preset();
793 gui->drag->update(0);
794 gui->drag->drag_deactivate();