3 * Copyright (C) 2008-2017 Adam Williams <broadcast at earthling dot net>
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 2 of the License, or
8 * (at your option) any later version.
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
15 * You should have received a copy of the GNU General Public License
16 * along with this program; if not, write to the Free Software
17 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
24 #include "automation.h"
28 #include "condition.h"
30 #include "cplayback.h"
32 #include "cwindowgui.h"
33 #include "cwindowtool.h"
35 #include "edlsession.h"
36 #include "floatauto.h"
37 #include "floatautos.h"
40 #include "localsession.h"
41 #include "mainsession.h"
44 #include "maskautos.h"
47 #include "mwindowgui.h"
51 #include "trackcanvas.h"
52 #include "transportque.h"
55 CWindowTool::CWindowTool(MWindow *mwindow, CWindowGUI *gui)
58 this->mwindow = mwindow;
62 current_tool = CWINDOW_NONE;
64 input_lock = new Condition(0, "CWindowTool::input_lock");
65 output_lock = new Condition(1, "CWindowTool::output_lock");
66 tool_gui_lock = new Mutex("CWindowTool::tool_gui_lock");
69 CWindowTool::~CWindowTool()
80 void CWindowTool::start_tool(int operation)
82 CWindowToolGUI *new_gui = 0;
85 //printf("CWindowTool::start_tool 1\n");
86 if(current_tool != operation)
88 int previous_tool = current_tool;
89 current_tool = operation;
93 new_gui = new CWindowEyedropGUI(mwindow, this);
96 new_gui = new CWindowCropGUI(mwindow, this);
99 new_gui = new CWindowCameraGUI(mwindow, this);
101 case CWINDOW_PROJECTOR:
102 new_gui = new CWindowProjectorGUI(mwindow, this);
105 new_gui = new CWindowMaskGUI(mwindow, this);
108 new_gui = new CWindowRulerGUI(mwindow, this);
110 case CWINDOW_PROTECT:
111 mwindow->edl->session->tool_window = 0;
112 gui->composite_panel->operation[CWINDOW_TOOL_WINDOW]->update(0);
120 //printf("CWindowTool::start_tool 1\n");
126 // Wait for previous tool GUI to finish
127 output_lock->lock("CWindowTool::start_tool");
128 this->tool_gui = new_gui;
129 tool_gui->create_objects();
130 if( previous_tool == CWINDOW_PROTECT || previous_tool == CWINDOW_NONE ) {
131 mwindow->edl->session->tool_window = 1;
132 gui->composite_panel->operation[CWINDOW_TOOL_WINDOW]->update(1);
134 update_show_window();
136 // Signal thread to run next tool GUI
137 input_lock->unlock();
139 //printf("CWindowTool::start_tool 1\n");
144 tool_gui->lock_window("CWindowTool::start_tool");
146 tool_gui->unlock_window();
149 //printf("CWindowTool::start_tool 2\n");
154 void CWindowTool::stop_tool()
158 tool_gui->lock_window("CWindowTool::stop_tool");
159 tool_gui->set_done(0);
160 tool_gui->unlock_window();
164 void CWindowTool::show_tool()
166 if(tool_gui && mwindow->edl->session->tool_window)
168 tool_gui->lock_window("CWindowTool::show_tool");
169 tool_gui->show_window();
170 tool_gui->unlock_window();
174 void CWindowTool::hide_tool()
176 if(tool_gui && mwindow->edl->session->tool_window)
178 tool_gui->lock_window("CWindowTool::show_tool");
179 tool_gui->hide_window();
180 tool_gui->unlock_window();
184 void CWindowTool::raise_tool()
186 if(tool_gui && mwindow->edl->session->tool_window)
188 tool_gui->lock_window("CWindowTool::show_tool");
189 tool_gui->raise_window();
190 tool_gui->unlock_window();
195 void CWindowTool::run()
199 input_lock->lock("CWindowTool::run");
202 tool_gui->run_window();
203 tool_gui_lock->lock("CWindowTool::run");
206 tool_gui_lock->unlock();
208 output_lock->unlock();
212 void CWindowTool::update_show_window()
216 tool_gui->lock_window("CWindowTool::update_show_window");
218 if(mwindow->edl->session->tool_window)
221 tool_gui->show_window();
224 tool_gui->hide_window();
227 tool_gui->unlock_window();
231 void CWindowTool::raise_window()
235 gui->unlock_window();
236 tool_gui->lock_window("CWindowTool::raise_window");
237 tool_gui->raise_window();
238 tool_gui->unlock_window();
239 gui->lock_window("CWindowTool::raise_window");
243 void CWindowTool::update_values()
245 tool_gui_lock->lock("CWindowTool::update_values");
248 tool_gui->lock_window("CWindowTool::update_values");
251 tool_gui->unlock_window();
253 tool_gui_lock->unlock();
262 CWindowToolGUI::CWindowToolGUI(MWindow *mwindow,
268 mwindow->session->ctool_x,
269 mwindow->session->ctool_y,
278 this->mwindow = mwindow;
279 this->thread = thread;
280 current_operation = 0;
283 CWindowToolGUI::~CWindowToolGUI()
287 int CWindowToolGUI::close_event()
291 mwindow->edl->session->tool_window = 0;
293 thread->gui->lock_window("CWindowToolGUI::close_event");
294 thread->gui->composite_panel->set_operation(mwindow->edl->session->cwindow_operation);
295 thread->gui->flush();
296 thread->gui->unlock_window();
297 lock_window("CWindowToolGUI::close_event");
301 int CWindowToolGUI::keypress_event()
305 switch( get_keypress() ) {
308 return close_event();
321 resend_event(thread->gui);
328 int CWindowToolGUI::translation_event()
330 mwindow->session->ctool_x = get_x();
331 mwindow->session->ctool_y = get_y();
336 void CWindowToolGUI::update_preview(int changed_edl)
339 draw_preview(changed_edl);
340 lock_window("CWindowToolGUI::update_preview");
343 void CWindowToolGUI::draw_preview(int changed_edl)
345 CWindowGUI *cgui = mwindow->cwindow->gui;
346 cgui->lock_window("CWindowToolGUI::draw_preview");
347 int change_type = !changed_edl ? CHANGE_PARAMS : CHANGE_EDL;
348 cgui->sync_parameters(change_type, 0, 1);
349 cgui->unlock_window();
353 CWindowCoord::CWindowCoord(CWindowToolGUI *gui, int x, int y, float value, int log_increment = 0)
354 : BC_TumbleTextBox(gui, (float)value, (float)-65536, (float)65536, x, y, 100, 3)
357 set_log_floatincrement(log_increment);
360 CWindowCoord::CWindowCoord(CWindowToolGUI *gui, int x, int y, int value)
361 : BC_TumbleTextBox(gui, (int64_t)value, (int64_t)-65536, (int64_t)65536, x, y, 100, 3)
365 int CWindowCoord::handle_event()
367 gui->event_caller = this;
373 CWindowCropOK::CWindowCropOK(MWindow *mwindow, CWindowToolGUI *gui, int x, int y)
374 : BC_GenericButton(x, y, _("Do it"))
376 this->mwindow = mwindow;
379 int CWindowCropOK::handle_event()
381 mwindow->crop_video();
386 int CWindowCropOK::keypress_event()
388 if(get_keypress() == 0xd)
402 CWindowCropGUI::CWindowCropGUI(MWindow *mwindow, CWindowTool *thread)
403 : CWindowToolGUI(mwindow,
405 _(PROGRAM_NAME ": Crop"),
412 CWindowCropGUI::~CWindowCropGUI()
416 void CWindowCropGUI::create_objects()
421 lock_window("CWindowCropGUI::create_objects");
423 int pad = MAX(BC_TextBox::calculate_h(this, MEDIUMFONT, 1, 1),
424 BC_Title::calculate_h(this, "X")) + 5;
425 add_subwindow(title = new BC_Title(x, y, "X1:"));
426 column1 = MAX(column1, title->get_w());
428 add_subwindow(title = new BC_Title(x, y, _("W:")));
429 column1 = MAX(column1, title->get_w());
431 add_subwindow(new CWindowCropOK(mwindow, thread->tool_gui, x, y));
435 x1 = new CWindowCoord(thread->tool_gui, x, y,
436 mwindow->edl->session->crop_x1);
437 x1->create_objects();
439 width = new CWindowCoord(thread->tool_gui, x, y,
440 mwindow->edl->session->crop_x2 - mwindow->edl->session->crop_x1);
441 width->create_objects();
444 x += x1->get_w() + 10;
447 add_subwindow(title = new BC_Title(x, y, "Y1:"));
448 column2 = MAX(column2, title->get_w());
450 add_subwindow(title = new BC_Title(x, y, _("H:")));
451 column2 = MAX(column2, title->get_w());
456 y1 = new CWindowCoord(thread->tool_gui, x, y,
457 mwindow->edl->session->crop_y1);
458 y1->create_objects();
460 height = new CWindowCoord(thread->tool_gui, x, y,
461 mwindow->edl->session->crop_y2 - mwindow->edl->session->crop_y1);
462 height->create_objects();
466 void CWindowCropGUI::handle_event()
469 new_x1 = atol(x1->get_text());
470 new_y1 = atol(y1->get_text());
471 if(new_x1 != mwindow->edl->session->crop_x1)
473 mwindow->edl->session->crop_x2 = new_x1 +
474 mwindow->edl->session->crop_x2 -
475 mwindow->edl->session->crop_x1;
476 mwindow->edl->session->crop_x1 = new_x1;
478 if(new_y1 != mwindow->edl->session->crop_y1)
480 mwindow->edl->session->crop_y2 = new_y1 +
481 mwindow->edl->session->crop_y2 -
482 mwindow->edl->session->crop_y1;
483 mwindow->edl->session->crop_y1 = atol(y1->get_text());
485 mwindow->edl->session->crop_x2 = atol(width->get_text()) +
486 mwindow->edl->session->crop_x1;
487 mwindow->edl->session->crop_y2 = atol(height->get_text()) +
488 mwindow->edl->session->crop_y1;
490 mwindow->cwindow->gui->canvas->redraw(1);
493 void CWindowCropGUI::update()
495 x1->update((int64_t)mwindow->edl->session->crop_x1);
496 y1->update((int64_t)mwindow->edl->session->crop_y1);
497 width->update((int64_t)mwindow->edl->session->crop_x2 -
498 mwindow->edl->session->crop_x1);
499 height->update((int64_t)mwindow->edl->session->crop_y2 -
500 mwindow->edl->session->crop_y1);
504 CWindowEyedropGUI::CWindowEyedropGUI(MWindow *mwindow, CWindowTool *thread)
505 : CWindowToolGUI(mwindow, thread, _(PROGRAM_NAME ": Color"), 220, 290)
509 CWindowEyedropGUI::~CWindowEyedropGUI()
513 void CWindowEyedropGUI::create_objects()
515 int margin = mwindow->theme->widget_border;
518 int x2 = 70, x3 = x2 + 60;
519 lock_window("CWindowEyedropGUI::create_objects");
520 BC_Title *title0, *title1, *title2, *title3, *title4, *title5, *title6, *title7;
521 add_subwindow(title0 = new BC_Title(x, y,_("X,Y:")));
522 y += title0->get_h() + margin;
523 add_subwindow(title7 = new BC_Title(x, y, _("Radius:")));
524 y += BC_TextBox::calculate_h(this, MEDIUMFONT, 1, 1) + margin;
526 add_subwindow(title1 = new BC_Title(x, y, _("Red:")));
527 y += title1->get_h() + margin;
528 add_subwindow(title2 = new BC_Title(x, y, _("Green:")));
529 y += title2->get_h() + margin;
530 add_subwindow(title3 = new BC_Title(x, y, _("Blue:")));
531 y += title3->get_h() + margin;
533 add_subwindow(title4 = new BC_Title(x, y, "Y:"));
534 y += title4->get_h() + margin;
535 add_subwindow(title5 = new BC_Title(x, y, "U:"));
536 y += title5->get_h() + margin;
537 add_subwindow(title6 = new BC_Title(x, y, "V:"));
539 add_subwindow(current = new BC_Title(x2, title0->get_y(), ""));
541 radius = new CWindowCoord(this, x2, title7->get_y(),
542 mwindow->edl->session->eyedrop_radius);
543 radius->create_objects();
544 radius->set_boundaries((int64_t)0, (int64_t)255);
546 add_subwindow(red = new BC_Title(x2, title1->get_y(), "0"));
547 add_subwindow(green = new BC_Title(x2, title2->get_y(), "0"));
548 add_subwindow(blue = new BC_Title(x2, title3->get_y(), "0"));
549 add_subwindow(rgb_hex = new BC_Title(x3, red->get_y(), "#000000"));
551 add_subwindow(this->y = new BC_Title(x2, title4->get_y(), "0"));
552 add_subwindow(this->u = new BC_Title(x2, title5->get_y(), "0"));
553 add_subwindow(this->v = new BC_Title(x2, title6->get_y(), "0"));
554 add_subwindow(yuv_hex = new BC_Title(x3, this->y->get_y(), "#000000"));
556 y = title6->get_y() + this->v->get_h() + 2*margin;
557 add_subwindow(sample = new BC_SubWindow(x, y, 50, 50));
558 y += sample->get_h() + margin;
559 add_subwindow(use_max = new CWindowEyedropCheckBox(mwindow, this, x, y));
564 void CWindowEyedropGUI::update()
566 char string[BCTEXTLEN];
567 sprintf(string, "%d, %d",
568 thread->gui->eyedrop_x,
569 thread->gui->eyedrop_y);
570 current->update(string);
572 radius->update((int64_t)mwindow->edl->session->eyedrop_radius);
574 LocalSession *local_session = mwindow->edl->local_session;
575 int use_max = local_session->use_max;
576 float r = use_max ? local_session->red_max : local_session->red;
577 float g = use_max ? local_session->green_max : local_session->green;
578 float b = use_max ? local_session->blue_max : local_session->blue;
579 this->red->update(r);
580 this->green->update(g);
581 this->blue->update(b);
583 int rx = 255*r + 0.5; bclamp(rx,0,255);
584 int gx = 255*g + 0.5; bclamp(gx,0,255);
585 int bx = 255*b + 0.5; bclamp(bx,0,255);
586 char rgb_text[BCSTRLEN];
587 sprintf(rgb_text, "#%02x%02x%02x", rx, gx, bx);
588 rgb_hex->update(rgb_text);
591 YUV::yuv.rgb_to_yuv_f(r, g, b, y, u, v);
593 this->u->update(u); u += 0.5;
594 this->v->update(v); v += 0.5;
596 int yx = 255*y + 0.5; bclamp(yx,0,255);
597 int ux = 255*u + 0.5; bclamp(ux,0,255);
598 int vx = 255*v + 0.5; bclamp(vx,0,255);
599 char yuv_text[BCSTRLEN];
600 sprintf(yuv_text, "#%02x%02x%02x", yx, ux, vx);
601 yuv_hex->update(yuv_text);
603 int rgb = (rx << 16) | (gx << 8) | (bx << 0);
604 sample->set_color(rgb);
605 sample->draw_box(0, 0, sample->get_w(), sample->get_h());
606 sample->set_color(BLACK);
607 sample->draw_rectangle(0, 0, sample->get_w(), sample->get_h());
611 void CWindowEyedropGUI::handle_event()
613 int new_radius = atoi(radius->get_text());
614 if(new_radius != mwindow->edl->session->eyedrop_radius)
616 CWindowGUI *gui = mwindow->cwindow->gui;
617 if(gui->eyedrop_visible)
619 gui->lock_window("CWindowEyedropGUI::handle_event");
622 gui->canvas->do_eyedrop(rerender, 0, 1);
625 mwindow->edl->session->eyedrop_radius = new_radius;
627 if(gui->eyedrop_visible)
631 gui->canvas->do_eyedrop(rerender, 0, 1);
632 gui->unlock_window();
639 /* Buttons to control Keyframe-Curve-Mode for Projector or Camera */
641 // Configuration for all possible Keyframe Curve Mode toggles
643 FloatAuto::t_mode mode;
649 const _CVD Camera_Crv_Smooth =
653 N_("\"smooth\" Curve on current Camera Keyframes")
655 const _CVD Camera_Crv_Linear =
659 N_("\"linear\" Curve on current Camera Keyframes")
661 const _CVD Projector_Crv_Smooth =
665 N_("\"smooth\" Curve on current Projector Keyframes")
667 const _CVD Projector_Crv_Linear =
671 N_("\"linear\" Curve on current Projector Keyframes")
674 // Implementation Class für Keyframe Curve Mode buttons
676 // This button reflects the state of the "current" keyframe
677 // (the nearest keyframe on the left) for all three automation
678 // lines together. Clicking on this button (re)sets the curve
679 // mode for the three "current" keyframes simultanously, but
680 // never creates a new keyframe.
682 class CWindowCurveToggle : public BC_Toggle
685 CWindowCurveToggle(_CVD mode, MWindow *mwindow, CWindowToolGUI *gui, int x, int y);
686 void check_toggle_state(FloatAuto *x, FloatAuto *y, FloatAuto *z);
695 CWindowCurveToggle::CWindowCurveToggle(_CVD mode, MWindow *mwindow, CWindowToolGUI *gui, int x, int y)
696 : BC_Toggle(x, y, mwindow->theme->get_image_set(mode.icon_id), false),
700 this->mwindow = mwindow;
701 set_tooltip(_(cfg.tooltip));
704 void CWindowCurveToggle::check_toggle_state(FloatAuto *x, FloatAuto *y, FloatAuto *z)
706 // the toggle state is only set to ON if all
707 // three automation lines have the same curve mode.
708 // For mixed states the toggle stays off.
709 set_value( x->curve_mode == this->cfg.mode &&
710 y->curve_mode == this->cfg.mode &&
711 z->curve_mode == this->cfg.mode
712 ,true // redraw to show new state
716 int CWindowCurveToggle::handle_event()
718 Track *track = mwindow->cwindow->calculate_affected_track();
720 FloatAuto *x=0, *y=0, *z=0;
721 mwindow->cwindow->calculate_affected_autos(track,
722 &x, &y, &z, cfg.use_camera, 0,0,0); // don't create new keyframe
723 if( x ) x->change_curve_mode(cfg.mode);
724 if( y ) y->change_curve_mode(cfg.mode);
725 if( z ) z->change_curve_mode(cfg.mode);
728 gui->update_preview();
735 CWindowEyedropCheckBox::CWindowEyedropCheckBox(MWindow *mwindow,
736 CWindowEyedropGUI *gui, int x, int y)
737 : BC_CheckBox(x, y, mwindow->edl->local_session->use_max, _("Use maximum"))
739 this->mwindow = mwindow;
743 int CWindowEyedropCheckBox::handle_event()
745 mwindow->edl->local_session->use_max = get_value();
752 CWindowCameraGUI::CWindowCameraGUI(MWindow *mwindow, CWindowTool *thread)
753 : CWindowToolGUI(mwindow,
755 _(PROGRAM_NAME ": Camera"),
760 CWindowCameraGUI::~CWindowCameraGUI()
764 void CWindowCameraGUI::create_objects()
766 int x = 10, y = 10, x1;
767 Track *track = mwindow->cwindow->calculate_affected_track();
768 FloatAuto *x_auto = 0, *y_auto = 0, *z_auto = 0;
772 lock_window("CWindowCameraGUI::create_objects");
774 mwindow->cwindow->calculate_affected_autos(track,
775 &x_auto, &y_auto, &z_auto, 1, 0, 0, 0);
778 add_subwindow(title = new BC_Title(x, y, "X:"));
780 this->x = new CWindowCoord(this, x, y,
781 x_auto ? x_auto->get_value() : (float)0);
782 this->x->create_objects();
787 add_subwindow(title = new BC_Title(x, y, "Y:"));
789 this->y = new CWindowCoord(this, x, y,
790 y_auto ? y_auto->get_value() : (float)0);
791 this->y->create_objects();
794 add_subwindow(title = new BC_Title(x, y, "Z:"));
796 this->z = new CWindowCoord(this, x, y,
797 z_auto ? z_auto->get_value() : (float)1);
798 this->z->create_objects();
799 this->z->set_increment(0.01);
803 add_subwindow(button = new CWindowCameraLeft(mwindow, this, x1, y));
804 x1 += button->get_w();
805 add_subwindow(button = new CWindowCameraCenter(mwindow, this, x1, y));
806 x1 += button->get_w();
807 add_subwindow(button = new CWindowCameraRight(mwindow, this, x1, y));
809 y += button->get_h();
811 add_subwindow(button = new CWindowCameraTop(mwindow, this, x1, y));
812 x1 += button->get_w();
813 add_subwindow(button = new CWindowCameraMiddle(mwindow, this, x1, y));
814 x1 += button->get_w();
815 add_subwindow(button = new CWindowCameraBottom(mwindow, this, x1, y));
816 // additional Buttons to control the curve mode of the "current" keyframe
817 x1 += button->get_w() + 15;
818 add_subwindow(this->t_smooth = new CWindowCurveToggle(Camera_Crv_Smooth, mwindow, this, x1, y));
819 x1 += button->get_w();
820 add_subwindow(this->t_linear = new CWindowCurveToggle(Camera_Crv_Linear, mwindow, this, x1, y));
822 // fill in current auto keyframe values, set toggle states.
827 void CWindowCameraGUI::handle_event()
829 FloatAuto *x_auto = 0;
830 FloatAuto *y_auto = 0;
831 FloatAuto *z_auto = 0;
832 Track *track = mwindow->cwindow->calculate_affected_track();
835 mwindow->undo->update_undo_before(_("camera"), this);
836 if(event_caller == x)
838 x_auto = (FloatAuto*)mwindow->cwindow->calculate_affected_auto(
839 track->automation->autos[AUTOMATION_CAMERA_X],
843 x_auto->set_value(atof(x->get_text()));
849 if(event_caller == y)
851 y_auto = (FloatAuto*)mwindow->cwindow->calculate_affected_auto(
852 track->automation->autos[AUTOMATION_CAMERA_Y],
856 y_auto->set_value(atof(y->get_text()));
862 if(event_caller == z)
864 z_auto = (FloatAuto*)mwindow->cwindow->calculate_affected_auto(
865 track->automation->autos[AUTOMATION_CAMERA_Z],
869 float zoom = atof(z->get_text());
870 if(zoom > 100.) zoom = 100.;
872 if(zoom < 0.01) zoom = 0.01;
873 // Doesn't allow user to enter from scratch
874 // if(zoom != atof(z->get_text()))
877 z_auto->set_value(zoom);
878 mwindow->gui->lock_window("CWindowCameraGUI::handle_event");
879 mwindow->gui->draw_overlays(1);
880 mwindow->gui->unlock_window();
886 mwindow->undo->update_undo_after(_("camera"), LOAD_ALL);
890 void CWindowCameraGUI::update()
892 FloatAuto *x_auto = 0;
893 FloatAuto *y_auto = 0;
894 FloatAuto *z_auto = 0;
895 Track *track = mwindow->cwindow->calculate_affected_track();
897 mwindow->cwindow->calculate_affected_autos(track,
898 &x_auto, &y_auto, &z_auto, 1, 0, 0, 0);
902 x->update(x_auto->get_value());
904 y->update(y_auto->get_value());
906 float value = z_auto->get_value();
908 thread->gui->lock_window("CWindowCameraGUI::update");
909 thread->gui->composite_panel->cpanel_zoom->update(value);
910 thread->gui->unlock_window();
913 if( x_auto && y_auto && z_auto )
915 t_smooth->check_toggle_state(x_auto, y_auto, z_auto);
916 t_linear->check_toggle_state(x_auto, y_auto, z_auto);
923 CWindowCameraLeft::CWindowCameraLeft(MWindow *mwindow, CWindowCameraGUI *gui, int x, int y)
924 : BC_Button(x, y, mwindow->theme->get_image_set("left_justify"))
927 this->mwindow = mwindow;
928 set_tooltip(_("Left justify"));
930 int CWindowCameraLeft::handle_event()
932 FloatAuto *x_auto = 0;
933 FloatAuto *z_auto = 0;
934 Track *track = mwindow->cwindow->calculate_affected_track();
936 mwindow->cwindow->calculate_affected_autos(track,
937 &x_auto, 0, &z_auto, 1, 1, 0, 0);
943 track->get_source_dimensions(
944 mwindow->edl->local_session->get_selectionstart(1),
950 mwindow->undo->update_undo_before(_("camera"), 0);
952 (double)track->track_w / z_auto->get_value() / 2 -
954 mwindow->undo->update_undo_after(_("camera"), LOAD_ALL);
956 gui->update_preview();
964 CWindowCameraCenter::CWindowCameraCenter(MWindow *mwindow, CWindowCameraGUI *gui, int x, int y)
965 : BC_Button(x, y, mwindow->theme->get_image_set("center_justify"))
968 this->mwindow = mwindow;
969 set_tooltip(_("Center horizontal"));
971 int CWindowCameraCenter::handle_event()
973 FloatAuto *x_auto = 0;
974 Track *track = mwindow->cwindow->calculate_affected_track();
976 x_auto = (FloatAuto*)mwindow->cwindow->calculate_affected_auto(
977 track->automation->autos[AUTOMATION_CAMERA_X],
982 mwindow->undo->update_undo_before(_("camera"), 0);
983 x_auto->set_value(0);
985 gui->update_preview();
986 mwindow->undo->update_undo_after(_("camera"), LOAD_ALL);
993 CWindowCameraRight::CWindowCameraRight(MWindow *mwindow, CWindowCameraGUI *gui, int x, int y)
994 : BC_Button(x, y, mwindow->theme->get_image_set("right_justify"))
997 this->mwindow = mwindow;
998 set_tooltip(_("Right justify"));
1000 int CWindowCameraRight::handle_event()
1002 FloatAuto *x_auto = 0;
1003 FloatAuto *z_auto = 0;
1004 Track *track = mwindow->cwindow->calculate_affected_track();
1006 mwindow->cwindow->calculate_affected_autos(track,
1007 &x_auto, 0, &z_auto, 1, 1, 0, 0);
1010 if(x_auto && z_auto)
1013 track->get_source_dimensions(
1014 mwindow->edl->local_session->get_selectionstart(1),
1020 mwindow->undo->update_undo_before(_("camera"), 0);
1021 x_auto->set_value( -((double)track->track_w / z_auto->get_value() / 2 -
1024 gui->update_preview();
1025 mwindow->undo->update_undo_after(_("camera"), LOAD_ALL);
1033 CWindowCameraTop::CWindowCameraTop(MWindow *mwindow, CWindowCameraGUI *gui, int x, int y)
1034 : BC_Button(x, y, mwindow->theme->get_image_set("top_justify"))
1037 this->mwindow = mwindow;
1038 set_tooltip(_("Top justify"));
1040 int CWindowCameraTop::handle_event()
1042 FloatAuto *y_auto = 0;
1043 FloatAuto *z_auto = 0;
1044 Track *track = mwindow->cwindow->calculate_affected_track();
1046 mwindow->cwindow->calculate_affected_autos(track,
1047 0, &y_auto, &z_auto, 1, 0, 1, 0);
1050 if(y_auto && z_auto)
1053 track->get_source_dimensions(
1054 mwindow->edl->local_session->get_selectionstart(1),
1060 mwindow->undo->update_undo_before(_("camera"), 0);
1061 y_auto->set_value((double)track->track_h / z_auto->get_value() / 2 -
1064 gui->update_preview();
1065 mwindow->undo->update_undo_after(_("camera"), LOAD_ALL);
1073 CWindowCameraMiddle::CWindowCameraMiddle(MWindow *mwindow, CWindowCameraGUI *gui, int x, int y)
1074 : BC_Button(x, y, mwindow->theme->get_image_set("middle_justify"))
1077 this->mwindow = mwindow;
1078 set_tooltip(_("Center vertical"));
1080 int CWindowCameraMiddle::handle_event()
1082 FloatAuto *y_auto = 0;
1083 Track *track = mwindow->cwindow->calculate_affected_track();
1085 y_auto = (FloatAuto*)mwindow->cwindow->calculate_affected_auto(
1086 track->automation->autos[AUTOMATION_CAMERA_Y], 1);
1090 mwindow->undo->update_undo_before(_("camera"), 0);
1091 y_auto->set_value(0);
1093 gui->update_preview();
1094 mwindow->undo->update_undo_after(_("camera"), LOAD_ALL);
1101 CWindowCameraBottom::CWindowCameraBottom(MWindow *mwindow, CWindowCameraGUI *gui, int x, int y)
1102 : BC_Button(x, y, mwindow->theme->get_image_set("bottom_justify"))
1105 this->mwindow = mwindow;
1106 set_tooltip(_("Bottom justify"));
1108 int CWindowCameraBottom::handle_event()
1110 FloatAuto *y_auto = 0;
1111 FloatAuto *z_auto = 0;
1112 Track *track = mwindow->cwindow->calculate_affected_track();
1114 mwindow->cwindow->calculate_affected_autos(track,
1115 0, &y_auto, &z_auto, 1, 0, 1, 0);
1118 if(y_auto && z_auto)
1121 track->get_source_dimensions(
1122 mwindow->edl->local_session->get_selectionstart(1),
1128 mwindow->undo->update_undo_before(_("camera"), 0);
1129 y_auto->set_value(-((double)track->track_h / z_auto->get_value() / 2 -
1132 gui->update_preview();
1133 mwindow->undo->update_undo_after(_("camera"), LOAD_ALL);
1141 CWindowProjectorGUI::CWindowProjectorGUI(MWindow *mwindow, CWindowTool *thread)
1142 : CWindowToolGUI(mwindow,
1144 _(PROGRAM_NAME ": Projector"),
1149 CWindowProjectorGUI::~CWindowProjectorGUI()
1152 void CWindowProjectorGUI::create_objects()
1154 int x = 10, y = 10, x1;
1155 Track *track = mwindow->cwindow->calculate_affected_track();
1156 FloatAuto *x_auto = 0;
1157 FloatAuto *y_auto = 0;
1158 FloatAuto *z_auto = 0;
1162 lock_window("CWindowProjectorGUI::create_objects");
1164 mwindow->cwindow->calculate_affected_autos(track,
1165 &x_auto, &y_auto, &z_auto, 0, 0, 0, 0);
1168 add_subwindow(title = new BC_Title(x, y, "X:"));
1169 x += title->get_w();
1170 this->x = new CWindowCoord(this, x, y,
1171 x_auto ? x_auto->get_value() : (float)0);
1172 this->x->create_objects();
1175 add_subwindow(title = new BC_Title(x, y, "Y:"));
1176 x += title->get_w();
1177 this->y = new CWindowCoord(this, x, y,
1178 y_auto ? y_auto->get_value() : (float)0);
1179 this->y->create_objects();
1182 add_subwindow(title = new BC_Title(x, y, "Z:"));
1183 x += title->get_w();
1184 this->z = new CWindowCoord(this, x, y,
1185 z_auto ? z_auto->get_value() : (float)1);
1186 this->z->create_objects();
1187 this->z->set_increment(0.01);
1191 add_subwindow(button = new CWindowProjectorLeft(mwindow, this, x1, y));
1192 x1 += button->get_w();
1193 add_subwindow(button = new CWindowProjectorCenter(mwindow, this, x1, y));
1194 x1 += button->get_w();
1195 add_subwindow(button = new CWindowProjectorRight(mwindow, this, x1, y));
1197 y += button->get_h();
1199 add_subwindow(button = new CWindowProjectorTop(mwindow, this, x1, y));
1200 x1 += button->get_w();
1201 add_subwindow(button = new CWindowProjectorMiddle(mwindow, this, x1, y));
1202 x1 += button->get_w();
1203 add_subwindow(button = new CWindowProjectorBottom(mwindow, this, x1, y));
1205 // additional Buttons to control the curve mode of the "current" keyframe
1206 x1 += button->get_w() + 15;
1207 add_subwindow(this->t_smooth = new CWindowCurveToggle(Projector_Crv_Smooth, mwindow, this, x1, y));
1208 x1 += button->get_w();
1209 add_subwindow(this->t_linear = new CWindowCurveToggle(Projector_Crv_Linear, mwindow, this, x1, y));
1211 // fill in current auto keyframe values, set toggle states.
1216 void CWindowProjectorGUI::handle_event()
1218 FloatAuto *x_auto = 0;
1219 FloatAuto *y_auto = 0;
1220 FloatAuto *z_auto = 0;
1221 Track *track = mwindow->cwindow->calculate_affected_track();
1225 mwindow->undo->update_undo_before(_("projector"), this);
1226 if(event_caller == x)
1228 x_auto = (FloatAuto*)mwindow->cwindow->calculate_affected_auto(
1229 track->automation->autos[AUTOMATION_PROJECTOR_X],
1233 x_auto->set_value(atof(x->get_text()));
1239 if(event_caller == y)
1241 y_auto = (FloatAuto*)mwindow->cwindow->calculate_affected_auto(
1242 track->automation->autos[AUTOMATION_PROJECTOR_Y],
1246 y_auto->set_value(atof(y->get_text()));
1252 if(event_caller == z)
1254 z_auto = (FloatAuto*)mwindow->cwindow->calculate_affected_auto(
1255 track->automation->autos[AUTOMATION_PROJECTOR_Z],
1259 float zoom = atof(z->get_text());
1260 if(zoom > 100.) zoom = 100.;
1261 else if(zoom < 0.01) zoom = 0.01;
1262 // if (zoom != atof(z->get_text()))
1264 z_auto->set_value(zoom);
1266 mwindow->gui->lock_window("CWindowProjectorGUI::handle_event");
1267 mwindow->gui->draw_overlays(1);
1268 mwindow->gui->unlock_window();
1274 mwindow->undo->update_undo_after(_("projector"), LOAD_ALL);
1278 void CWindowProjectorGUI::update()
1280 FloatAuto *x_auto = 0;
1281 FloatAuto *y_auto = 0;
1282 FloatAuto *z_auto = 0;
1283 Track *track = mwindow->cwindow->calculate_affected_track();
1285 mwindow->cwindow->calculate_affected_autos(track,
1286 &x_auto, &y_auto, &z_auto, 0, 0, 0, 0);
1290 x->update(x_auto->get_value());
1292 y->update(y_auto->get_value());
1294 float value = z_auto->get_value();
1296 thread->gui->lock_window("CWindowProjectorGUI::update");
1297 thread->gui->composite_panel->cpanel_zoom->update(value);
1298 thread->gui->unlock_window();
1301 if( x_auto && y_auto && z_auto )
1303 t_smooth->check_toggle_state(x_auto, y_auto, z_auto);
1304 t_linear->check_toggle_state(x_auto, y_auto, z_auto);
1308 CWindowProjectorLeft::CWindowProjectorLeft(MWindow *mwindow, CWindowProjectorGUI *gui, int x, int y)
1309 : BC_Button(x, y, mwindow->theme->get_image_set("left_justify"))
1312 this->mwindow = mwindow;
1313 set_tooltip(_("Left justify"));
1315 int CWindowProjectorLeft::handle_event()
1317 FloatAuto *x_auto = 0;
1318 FloatAuto *z_auto = 0;
1319 Track *track = mwindow->cwindow->calculate_affected_track();
1321 mwindow->cwindow->calculate_affected_autos(track,
1322 &x_auto, 0, &z_auto, 0, 1, 0, 0);
1324 if(x_auto && z_auto)
1326 mwindow->undo->update_undo_before(_("projector"), 0);
1327 x_auto->set_value( (double)track->track_w * z_auto->get_value() / 2 -
1328 (double)mwindow->edl->session->output_w / 2 );
1330 gui->update_preview();
1331 mwindow->undo->update_undo_after(_("projector"), LOAD_ALL);
1338 CWindowProjectorCenter::CWindowProjectorCenter(MWindow *mwindow, CWindowProjectorGUI *gui, int x, int y)
1339 : BC_Button(x, y, mwindow->theme->get_image_set("center_justify"))
1342 this->mwindow = mwindow;
1343 set_tooltip(_("Center horizontal"));
1345 int CWindowProjectorCenter::handle_event()
1347 FloatAuto *x_auto = 0;
1348 Track *track = mwindow->cwindow->calculate_affected_track();
1350 x_auto = (FloatAuto*)mwindow->cwindow->calculate_affected_auto(
1351 track->automation->autos[AUTOMATION_PROJECTOR_X],
1356 mwindow->undo->update_undo_before(_("projector"), 0);
1357 x_auto->set_value(0);
1359 gui->update_preview();
1360 mwindow->undo->update_undo_after(_("projector"), LOAD_ALL);
1367 CWindowProjectorRight::CWindowProjectorRight(MWindow *mwindow, CWindowProjectorGUI *gui, int x, int y)
1368 : BC_Button(x, y, mwindow->theme->get_image_set("right_justify"))
1371 this->mwindow = mwindow;
1372 set_tooltip(_("Right justify"));
1374 int CWindowProjectorRight::handle_event()
1376 FloatAuto *x_auto = 0;
1377 FloatAuto *z_auto = 0;
1378 Track *track = mwindow->cwindow->calculate_affected_track();
1380 mwindow->cwindow->calculate_affected_autos(track,
1381 &x_auto, 0, &z_auto, 0, 1, 0, 0);
1384 if(x_auto && z_auto)
1386 mwindow->undo->update_undo_before(_("projector"), 0);
1387 x_auto->set_value( -((double)track->track_w * z_auto->get_value() / 2 -
1388 (double)mwindow->edl->session->output_w / 2));
1390 gui->update_preview();
1391 mwindow->undo->update_undo_after(_("projector"), LOAD_ALL);
1398 CWindowProjectorTop::CWindowProjectorTop(MWindow *mwindow, CWindowProjectorGUI *gui, int x, int y)
1399 : BC_Button(x, y, mwindow->theme->get_image_set("top_justify"))
1402 this->mwindow = mwindow;
1403 set_tooltip(_("Top justify"));
1405 int CWindowProjectorTop::handle_event()
1407 FloatAuto *y_auto = 0;
1408 FloatAuto *z_auto = 0;
1409 Track *track = mwindow->cwindow->calculate_affected_track();
1411 mwindow->cwindow->calculate_affected_autos(track,
1412 0, &y_auto, &z_auto, 0, 0, 1, 0);
1415 if(y_auto && z_auto)
1417 mwindow->undo->update_undo_before(_("projector"), 0);
1418 y_auto->set_value( (double)track->track_h * z_auto->get_value() / 2 -
1419 (double)mwindow->edl->session->output_h / 2 );
1421 gui->update_preview();
1422 mwindow->undo->update_undo_after(_("projector"), LOAD_ALL);
1429 CWindowProjectorMiddle::CWindowProjectorMiddle(MWindow *mwindow, CWindowProjectorGUI *gui, int x, int y)
1430 : BC_Button(x, y, mwindow->theme->get_image_set("middle_justify"))
1433 this->mwindow = mwindow;
1434 set_tooltip(_("Center vertical"));
1436 int CWindowProjectorMiddle::handle_event()
1438 FloatAuto *y_auto = 0;
1439 Track *track = mwindow->cwindow->calculate_affected_track();
1441 y_auto = (FloatAuto*)mwindow->cwindow->calculate_affected_auto(
1442 track->automation->autos[AUTOMATION_PROJECTOR_Y], 1);
1446 mwindow->undo->update_undo_before(_("projector"), 0);
1447 y_auto->set_value(0);
1449 gui->update_preview();
1450 mwindow->undo->update_undo_after(_("projector"), LOAD_ALL);
1457 CWindowProjectorBottom::CWindowProjectorBottom(MWindow *mwindow, CWindowProjectorGUI *gui, int x, int y)
1458 : BC_Button(x, y, mwindow->theme->get_image_set("bottom_justify"))
1461 this->mwindow = mwindow;
1462 set_tooltip(_("Bottom justify"));
1464 int CWindowProjectorBottom::handle_event()
1466 FloatAuto *y_auto = 0;
1467 FloatAuto *z_auto = 0;
1468 Track *track = mwindow->cwindow->calculate_affected_track();
1470 mwindow->cwindow->calculate_affected_autos(track,
1471 0, &y_auto, &z_auto, 0, 0, 1, 0);
1474 if(y_auto && z_auto)
1476 mwindow->undo->update_undo_before(_("projector"), 0);
1477 y_auto->set_value( -((double)track->track_h * z_auto->get_value() / 2 -
1478 (double)mwindow->edl->session->output_h / 2));
1480 gui->update_preview();
1481 mwindow->undo->update_undo_after(_("projector"), LOAD_ALL);
1488 CWindowMaskOnTrack::CWindowMaskOnTrack(MWindow *mwindow, CWindowMaskGUI *gui,
1489 int x, int y, int w, const char *text)
1490 : BC_PopupTextBox(gui, 0, text, x, y, w, 120)
1492 this->mwindow = mwindow;
1496 CWindowMaskOnTrack::~CWindowMaskOnTrack()
1500 int CWindowMaskOnTrack::handle_event()
1502 int k = get_number();
1503 //printf("selected %d = %s\n", k, k<0 ? "()" : track_items[k]->get_text());
1504 CWindowMaskItem *track_item = k >= 0 ? (CWindowMaskItem *)track_items[k] : 0;
1505 Track *track = track_item ? mwindow->edl->tracks->get_track_by_id(track_item->id) : 0;
1506 int track_id = track_item && track && track->record ? track_item->id : -1;
1507 set_back_color(track_id >= 0 ?
1508 gui->get_resources()->text_background :
1509 gui->get_resources()->text_background_disarmed);
1510 gui->mask_on_track->update(track_item ? track_item->get_text() : "");
1511 mwindow->cwindow->mask_track_id = track_id;
1512 mwindow->edl->local_session->solo_track_id = -1;
1513 gui->mask_solo_track->update(0);
1514 gui->update_preview(1);
1518 void CWindowMaskOnTrack::update_items()
1520 track_items.remove_all_objects();
1521 for( Track *track=mwindow->edl->tracks->first; track; track=track->next ) {
1522 if( track->data_type != TRACK_VIDEO ) continue;
1523 int color = track->record ? -1 : RED;
1524 track_items.append(new CWindowMaskItem(track->title, track->get_id(), color));
1526 update_list(&track_items);
1529 CWindowMaskTrackTumbler::CWindowMaskTrackTumbler(MWindow *mwindow, CWindowMaskGUI *gui,
1533 this->mwindow = mwindow;
1536 CWindowMaskTrackTumbler::~CWindowMaskTrackTumbler()
1540 int CWindowMaskTrackTumbler::handle_up_event()
1545 int CWindowMaskTrackTumbler::handle_down_event()
1547 return do_event(-1);
1550 int CWindowMaskTrackTumbler::do_event(int dir)
1552 CWindowMaskItem *track_item = 0;
1553 CWindowMaskItem **items = (CWindowMaskItem**)&gui->mask_on_track->track_items[0];
1554 int n = gui->mask_on_track->track_items.size();
1555 int id = mwindow->cwindow->mask_track_id;
1558 while( --k >= 0 && items[k]->id != id );
1562 track_item = items[k];
1565 track_item = items[0];
1567 Track *track = track_item ? mwindow->edl->tracks->get_track_by_id(track_item->id) : 0;
1568 int track_id = track_item && track && track->record ? track_item->id : -1;
1569 gui->mask_on_track->set_back_color(track_id >= 0 ?
1570 gui->get_resources()->text_background :
1571 gui->get_resources()->text_background_disarmed);
1572 gui->mask_on_track->update(track_item ? track_item->get_text() : "");
1573 mwindow->cwindow->mask_track_id = track_item ? track_item->id : -1;
1574 mwindow->edl->local_session->solo_track_id = -1;
1575 gui->mask_solo_track->update(0);
1576 gui->update_preview(1);
1581 CWindowMaskName::CWindowMaskName(MWindow *mwindow, CWindowMaskGUI *gui,
1582 int x, int y, const char *text)
1583 : BC_PopupTextBox(gui, 0, text, x, y, 100, 160)
1585 this->mwindow = mwindow;
1589 CWindowMaskName::~CWindowMaskName()
1593 int CWindowMaskName::handle_event()
1600 //printf("CWindowMaskGUI::update 1\n");
1601 gui->get_keyframe(track, autos, keyframe, mask, point, 0);
1603 int k = get_number();
1604 if( k < 0 ) k = mwindow->edl->session->cwindow_mask;
1605 else mwindow->edl->session->cwindow_mask = k;
1606 if( k >= 0 && k < mask_items.size() ) {
1607 mask_items[k]->set_text(get_text());
1608 update_list(&mask_items);
1610 #ifdef USE_KEYFRAME_SPANNING
1611 MaskAuto temp_keyframe(mwindow->edl, autos);
1612 temp_keyframe.copy_data(keyframe);
1613 SubMask *submask = temp_keyframe.get_submask(mwindow->edl->session->cwindow_mask);
1614 memset(submask->name, 0, sizeof(submask->name));
1615 strncpy(submask->name, get_text(), sizeof(submask->name)-1);
1616 ((MaskAutos*)track->automation->autos[AUTOMATION_MASK])->update_parameter(&temp_keyframe);
1618 for(MaskAuto *current = (MaskAuto*)autos->default_auto; current; ) {
1619 SubMask *submask = current->get_submask(mwindow->edl->session->cwindow_mask);
1620 memset(submask->name, 0, sizeof(submask->name));
1621 strncpy(submask->name, get_text(), sizeof(submask->name));
1622 current = current == (MaskAuto*)autos->default_auto ?
1623 (MaskAuto*)autos->first : (MaskAuto*)NEXT;
1627 gui->update_preview();
1632 void CWindowMaskName::update_items(MaskAuto *keyframe)
1634 mask_items.remove_all_objects();
1635 int sz = !keyframe ? 0 : keyframe->masks.size();
1636 for( int i=0; i<SUBMASKS; ++i ) {
1637 char text[BCSTRLEN];
1639 SubMask *sub_mask = keyframe->masks.get(i);
1640 strncpy(text, sub_mask->name, sizeof(text));
1643 sprintf(text, "%d", i);
1644 mask_items.append(new CWindowMaskItem(text));
1646 update_list(&mask_items);
1650 CWindowMaskButton::CWindowMaskButton(MWindow *mwindow, CWindowMaskGUI *gui,
1651 int x, int y, int no, int v)
1652 : BC_CheckBox(x, y, v)
1654 this->mwindow = mwindow;
1659 CWindowMaskButton::~CWindowMaskButton()
1663 int CWindowMaskButton::handle_event()
1665 mwindow->edl->session->cwindow_mask = no;
1666 gui->mask_name->update(gui->mask_name->mask_items[no]->get_text());
1668 gui->update_preview();
1672 CWindowMaskThumbler::CWindowMaskThumbler(MWindow *mwindow, CWindowMaskGUI *gui,
1676 this->mwindow = mwindow;
1680 CWindowMaskThumbler::~CWindowMaskThumbler()
1684 int CWindowMaskThumbler::handle_up_event()
1689 int CWindowMaskThumbler::handle_down_event()
1691 return do_event(-1);
1694 int CWindowMaskThumbler::do_event(int dir)
1696 int k = mwindow->edl->session->cwindow_mask;
1697 if( (k+=dir) >= SUBMASKS ) k = 0;
1698 else if( k < 0 ) k = SUBMASKS-1;
1699 mwindow->edl->session->cwindow_mask = k;
1700 gui->mask_name->update(gui->mask_name->mask_items[k]->get_text());
1702 gui->update_preview();
1706 CWindowMaskEnable::CWindowMaskEnable(MWindow *mwindow, CWindowMaskGUI *gui,
1707 int x, int y, int no, int v)
1708 : BC_CheckBox(x, y, v)
1710 this->mwindow = mwindow;
1715 CWindowMaskEnable::~CWindowMaskEnable()
1719 int CWindowMaskEnable::handle_event()
1721 Track *track = mwindow->cwindow->calculate_mask_track();
1723 mwindow->undo->update_undo_before(_("mask enable"), this);
1726 track->masks |= bit;
1728 track->masks &= ~bit;
1730 gui->update_preview(1);
1731 mwindow->undo->update_undo_after(_("mask enable"), LOAD_PATCHES);
1736 CWindowMaskUnclear::CWindowMaskUnclear(MWindow *mwindow,
1737 CWindowMaskGUI *gui, int x, int y, int w)
1738 : BC_GenericButton(x, y, w, _("Enable"))
1740 this->mwindow = mwindow;
1742 set_tooltip(_("Show mask"));
1745 int CWindowMaskUnclear::handle_event()
1747 Track *track = mwindow->cwindow->calculate_mask_track();
1749 mwindow->undo->update_undo_before(_("mask enables"), this);
1750 int m = (1<<SUBMASKS)-1;
1751 if( track->masks == m )
1755 for( int i=0; i<SUBMASKS; ++i )
1756 gui->mask_enables[i]->update((track->masks>>i) & 1);
1757 gui->update_preview(1);
1758 mwindow->undo->update_undo_after(_("mask enables"), LOAD_PATCHES);
1763 CWindowMaskSoloTrack::CWindowMaskSoloTrack(MWindow *mwindow,
1764 CWindowMaskGUI *gui, int x, int y, int v)
1765 : BC_CheckBox(x, y, v, _("Solo"))
1767 this->mwindow = mwindow;
1769 set_tooltip(_("Solo video track"));
1772 int CWindowMaskSoloTrack::handle_event()
1774 mwindow->edl->local_session->solo_track_id =
1775 get_value() ? mwindow->cwindow->mask_track_id : -1;
1776 gui->update_preview(1);
1780 int CWindowMaskSoloTrack::calculate_w(BC_WindowBase *gui)
1783 calculate_extents(gui, &w, &h, _("Solo"));
1787 CWindowMaskDelMask::CWindowMaskDelMask(MWindow *mwindow,
1788 CWindowMaskGUI *gui, int x, int y)
1789 : BC_GenericButton(x, y, _("Delete"))
1791 this->mwindow = mwindow;
1793 set_tooltip(_("Delete mask"));
1796 int CWindowMaskDelMask::handle_event()
1805 // Get existing keyframe
1806 gui->get_keyframe(track, autos, keyframe, mask, point, 0);
1809 mwindow->undo->update_undo_before(_("mask delete"), 0);
1811 #ifdef USE_KEYFRAME_SPANNING
1812 // Create temp keyframe
1813 MaskAuto temp_keyframe(mwindow->edl, autos);
1814 temp_keyframe.copy_data(keyframe);
1815 SubMask *submask = temp_keyframe.get_submask(mwindow->edl->session->cwindow_mask);
1816 submask->points.remove_all_objects();
1818 // Commit change to span of keyframes
1819 ((MaskAutos*)track->automation->autos[AUTOMATION_MASK])->update_parameter(&temp_keyframe);
1821 for(MaskAuto *current = (MaskAuto*)autos->default_auto; current; ) {
1822 SubMask *submask = current->get_submask(mwindow->edl->session->cwindow_mask);
1823 submask->points.clear();
1824 current = current == (MaskAuto*)autos->default_auto ?
1825 (MaskAuto*)autos->first : (MaskAuto*)NEXT;
1829 if( mwindow->cwindow->gui->affected_point >= total_points )
1830 mwindow->cwindow->gui->affected_point =
1831 total_points > 0 ? total_points-1 : 0;
1834 gui->update_preview();
1835 mwindow->undo->update_undo_after(_("mask delete"), LOAD_AUTOMATION);
1841 CWindowMaskDelPoint::CWindowMaskDelPoint(MWindow *mwindow,
1842 CWindowMaskGUI *gui, int x, int y)
1843 : BC_GenericButton(x, y, _("Delete"))
1845 this->mwindow = mwindow;
1847 set_tooltip(_("Delete point"));
1850 int CWindowMaskDelPoint::handle_event()
1859 // Get existing keyframe
1860 gui->get_keyframe(track, autos, keyframe, mask, point, 0);
1862 mwindow->undo->update_undo_before(_("point delete"), 0);
1864 #ifdef USE_KEYFRAME_SPANNING
1865 // Create temp keyframe
1866 MaskAuto temp_keyframe(mwindow->edl, autos);
1867 temp_keyframe.copy_data(keyframe);
1869 SubMask *submask = temp_keyframe.get_submask(mwindow->edl->session->cwindow_mask);
1870 int i = mwindow->cwindow->gui->affected_point;
1871 for( ; i<submask->points.total-1; ++i )
1872 *submask->points.values[i] = *submask->points.values[i+1];
1873 if( submask->points.total > 0 ) {
1874 point = submask->points.values[submask->points.total-1];
1875 submask->points.remove_object(point);
1877 total_points = submask->points.total;
1879 // Commit change to span of keyframes
1880 ((MaskAutos*)track->automation->autos[AUTOMATION_MASK])->update_parameter(&temp_keyframe);
1882 MaskAuto *current = (MaskAuto*)autos->default_auto;
1884 SubMask *submask = current->get_submask(mwindow->edl->session->cwindow_mask);
1885 int i = mwindow->cwindow->gui->affected_point;
1886 for( ; i<submask->points.total-1; ++i ) {
1887 *submask->points.values[i] = *submask->points.values[i+1];
1888 if( submask->points.total > 0 ) {
1889 point = submask->points.values[submask->points.total-1];
1890 submask->points.remove_object(point);
1892 total_points = submask->points.total;
1893 current = current == (MaskAuto*)autos->default_auto ?
1894 (MaskAuto*)autos->first : (MaskAuto*)NEXT;
1897 if( mwindow->cwindow->gui->affected_point >= total_points )
1898 mwindow->cwindow->gui->affected_point =
1899 total_points > 0 ? total_points-1 : 0;
1902 gui->update_preview();
1903 mwindow->undo->update_undo_after(_("mask delete"), LOAD_AUTOMATION);
1909 int CWindowMaskDelPoint::keypress_event()
1911 if( get_keypress() == BACKSPACE ||
1912 get_keypress() == DELETE )
1913 return handle_event();
1920 CWindowMaskAffectedPoint::CWindowMaskAffectedPoint(MWindow *mwindow,
1921 CWindowMaskGUI *gui, int x, int y)
1922 : BC_TumbleTextBox(gui,
1923 (int64_t)mwindow->cwindow->gui->affected_point,
1924 (int64_t)0, INT64_MAX, x, y, 100)
1926 this->mwindow = mwindow;
1930 CWindowMaskAffectedPoint::~CWindowMaskAffectedPoint()
1934 int CWindowMaskAffectedPoint::handle_event()
1936 int total_points = 0;
1937 int affected_point = atol(get_text());
1938 Track *track = mwindow->cwindow->calculate_mask_track();
1940 MaskAutos *autos = (MaskAutos*)track->automation->autos[AUTOMATION_MASK];
1941 MaskAuto *keyframe = (MaskAuto*)mwindow->cwindow->calculate_affected_auto(autos, 0);
1943 SubMask *mask = keyframe->get_submask(mwindow->edl->session->cwindow_mask);
1944 total_points = mask->points.size();
1947 int active_point = affected_point;
1948 if( affected_point >= total_points )
1949 affected_point = total_points - 1;
1950 if( affected_point < 0 )
1952 if( active_point != affected_point )
1953 update((int64_t)affected_point);
1954 mwindow->cwindow->gui->affected_point = affected_point;
1956 gui->update_preview();
1961 CWindowMaskFocus::CWindowMaskFocus(MWindow *mwindow, CWindowMaskGUI *gui, int x, int y)
1962 : BC_CheckBox(x, y, gui->focused, _("Focus"))
1964 this->mwindow = mwindow;
1966 set_tooltip(_("Center for rotate/scale"));
1969 CWindowMaskFocus::~CWindowMaskFocus()
1973 int CWindowMaskFocus::handle_event()
1975 gui->focused = get_value();
1977 gui->update_preview();
1981 CWindowMaskDrawMarkers::CWindowMaskDrawMarkers(MWindow *mwindow, CWindowMaskGUI *gui, int x, int y)
1982 : BC_CheckBox(x, y, gui->markers, _("Markers"))
1984 this->mwindow = mwindow;
1986 set_tooltip("Display points");
1989 CWindowMaskDrawMarkers::~CWindowMaskDrawMarkers()
1993 int CWindowMaskDrawMarkers::handle_event()
1995 gui->markers = get_value();
1997 gui->update_preview();
2001 CWindowMaskDrawBoundary::CWindowMaskDrawBoundary(MWindow *mwindow, CWindowMaskGUI *gui, int x, int y)
2002 : BC_CheckBox(x, y, gui->boundary, _("Boundary"))
2004 this->mwindow = mwindow;
2006 set_tooltip("Display mask outline");
2009 CWindowMaskDrawBoundary::~CWindowMaskDrawBoundary()
2013 int CWindowMaskDrawBoundary::handle_event()
2015 gui->boundary = get_value();
2017 gui->update_preview();
2022 CWindowMaskFeather::CWindowMaskFeather(MWindow *mwindow, CWindowMaskGUI *gui, int x, int y)
2023 : BC_TumbleTextBox(gui, 0, -FEATHER_MAX, FEATHER_MAX, x, y, 64, 2)
2025 this->mwindow = mwindow;
2028 CWindowMaskFeather::~CWindowMaskFeather()
2032 int CWindowMaskFeather::update(float v)
2034 gui->feather_slider->update(v);
2035 return BC_TumbleTextBox::update(v);
2038 int CWindowMaskFeather::update_value(float v)
2045 #ifdef USE_KEYFRAME_SPANNING
2051 mwindow->undo->update_undo_before(_("mask feather"), this);
2053 // Get existing keyframe
2054 gui->get_keyframe(track, autos, keyframe,
2055 mask, point, create_it);
2057 int gang = gui->gang_feather->get_value();
2058 #ifdef USE_KEYFRAME_SPANNING
2059 MaskAuto temp_keyframe(mwindow->edl, autos);
2060 temp_keyframe.copy_data(keyframe);
2061 keyframe = &temp_keyframe;
2063 float change = v - mask->feather;
2064 int k = mwindow->edl->session->cwindow_mask;
2065 int n = gang ? keyframe->masks.size() : k+1;
2066 for( int i=gang? 0 : k; i<n; ++i ) {
2067 SubMask *sub_mask = keyframe->get_submask(i);
2068 float feather = sub_mask->feather + change;
2069 bclamp(feather, -FEATHER_MAX, FEATHER_MAX);
2070 sub_mask->feather = feather;
2072 #ifdef USE_KEYFRAME_SPANNING
2073 autos->update_parameter(keyframe);
2075 gui->update_preview();
2078 mwindow->undo->update_undo_after(_("mask feather"), LOAD_AUTOMATION);
2082 int CWindowMaskFeather::handle_event()
2084 float v = atof(get_text());
2085 gui->feather_slider->update(v);
2086 return gui->feather->update_value(v);
2089 CWindowMaskFeatherSlider::CWindowMaskFeatherSlider(MWindow *mwindow,
2090 CWindowMaskGUI *gui, int x, int y, int w, float v)
2091 : BC_FSlider(x, y, 0, w, w, -FEATHER_MAX, FEATHER_MAX, v)
2093 this->mwindow = mwindow;
2095 set_precision(0.01);
2096 timer = new Timer();
2101 CWindowMaskFeatherSlider::~CWindowMaskFeatherSlider()
2106 int CWindowMaskFeatherSlider::handle_event()
2108 float v = get_value();
2110 int64_t ms = timer->get_difference();
2111 if( ms < 250 && --stick > 0 ) {
2112 if( get_value() == 0 ) return 1;
2120 else if( (last_v>=0 && v<0) || (last_v<0 && v>=0) ) {
2127 gui->feather->BC_TumbleTextBox::update(v);
2128 return gui->feather->update_value(v);
2131 int CWindowMaskFeatherSlider::update(float v)
2133 return BC_FSlider::update(v);
2136 CWindowMaskFade::CWindowMaskFade(MWindow *mwindow, CWindowMaskGUI *gui, int x, int y)
2137 : BC_TumbleTextBox(gui, 0, -100.f, 100.f, x, y, 64, 2)
2139 this->mwindow = mwindow;
2142 CWindowMaskFade::~CWindowMaskFade()
2146 int CWindowMaskFade::update(float v)
2148 gui->fade_slider->update(v);
2149 return BC_TumbleTextBox::update(v);
2152 int CWindowMaskFade::update_value(float v)
2159 #ifdef USE_KEYFRAME_SPANNING
2165 mwindow->undo->update_undo_before(_("mask fade"), this);
2167 // Get existing keyframe
2168 gui->get_keyframe(track, autos, keyframe,
2169 mask, point, create_it);
2171 int gang = gui->gang_fader->get_value();
2172 #ifdef USE_KEYFRAME_SPANNING
2173 MaskAuto temp_keyframe(mwindow->edl, autos);
2174 temp_keyframe.copy_data(keyframe);
2175 keyframe = &temp_keyframe;
2177 float change = v - mask->fader;
2178 int k = mwindow->edl->session->cwindow_mask;
2179 int n = gang ? keyframe->masks.size() : k+1;
2180 for( int i=gang? 0 : k; i<n; ++i ) {
2181 SubMask *sub_mask = keyframe->get_submask(i);
2182 float fader = sub_mask->fader + change;
2183 bclamp(fader, -100.f, 100.f);
2184 sub_mask->fader = fader;
2186 #ifdef USE_KEYFRAME_SPANNING
2187 autos->update_parameter(keyframe);
2189 gui->update_preview();
2192 mwindow->undo->update_undo_after(_("mask fade"), LOAD_AUTOMATION);
2196 int CWindowMaskFade::handle_event()
2198 float v = atof(get_text());
2199 gui->fade_slider->update(v);
2200 return gui->fade->update_value(v);
2203 CWindowMaskFadeSlider::CWindowMaskFadeSlider(MWindow *mwindow, CWindowMaskGUI *gui,
2204 int x, int y, int w)
2205 : BC_ISlider(x, y, 0, w, w, -200, 200, 0)
2207 this->mwindow = mwindow;
2209 timer = new Timer();
2214 CWindowMaskFadeSlider::~CWindowMaskFadeSlider()
2219 int CWindowMaskFadeSlider::handle_event()
2221 float v = 100*get_value()/200;
2223 int64_t ms = timer->get_difference();
2224 if( ms < 250 && --stick > 0 ) {
2225 if( get_value() == 0 ) return 1;
2233 else if( (last_v>=0 && v<0) || (last_v<0 && v>=0) ) {
2240 gui->fade->BC_TumbleTextBox::update(v);
2241 return gui->fade->update_value(v);
2244 int CWindowMaskFadeSlider::update(int64_t v)
2246 return BC_ISlider::update(200*v/100);
2249 CWindowMaskGangFader::CWindowMaskGangFader(MWindow *mwindow,
2250 CWindowMaskGUI *gui, int x, int y)
2251 : BC_Toggle(x, y, mwindow->theme->get_image_set("gangpatch_data"), 0)
2253 this->mwindow = mwindow;
2255 set_tooltip(_("Gang fader"));
2258 CWindowMaskGangFader::~CWindowMaskGangFader()
2262 int CWindowMaskGangFader::handle_event()
2267 CWindowMaskBeforePlugins::CWindowMaskBeforePlugins(CWindowMaskGUI *gui, int x, int y)
2271 _("Apply mask before plugins"))
2276 int CWindowMaskBeforePlugins::handle_event()
2283 gui->get_keyframe(track, autos, keyframe, mask, point, 1);
2286 int v = get_value();
2287 #ifdef USE_KEYFRAME_SPANNING
2288 MaskAuto temp_keyframe(gui->mwindow->edl, autos);
2289 temp_keyframe.copy_data(keyframe);
2290 temp_keyframe.apply_before_plugins = v;
2291 autos->update_parameter(&temp_keyframe);
2293 keyframe->apply_before_plugins = v;
2295 gui->update_preview();
2301 CWindowDisableOpenGLMasking::CWindowDisableOpenGLMasking(CWindowMaskGUI *gui, int x, int y)
2302 : BC_CheckBox(x, y, 1, _("Disable OpenGL masking"))
2307 int CWindowDisableOpenGLMasking::handle_event()
2314 gui->get_keyframe(track, autos, keyframe, mask, point, 1);
2317 int v = get_value();
2318 #ifdef USE_KEYFRAME_SPANNING
2319 MaskAuto temp_keyframe(gui->mwindow->edl, autos);
2320 temp_keyframe.copy_data(keyframe);
2321 temp_keyframe.disable_opengl_masking = v;
2322 autos->update_parameter(&temp_keyframe);
2324 keyframe->disable_opengl_masking = v;
2326 gui->update_preview();
2332 CWindowMaskClrMask::CWindowMaskClrMask(MWindow *mwindow,
2333 CWindowMaskGUI *gui, int x, int y)
2334 : BC_Button(x, y, mwindow->theme->get_image_set("reset_button"))
2336 this->mwindow = mwindow;
2338 set_tooltip(_("Delete all masks"));
2341 CWindowMaskClrMask::~CWindowMaskClrMask()
2345 int CWindowMaskClrMask::calculate_w(MWindow *mwindow)
2347 VFrame *vfrm = *mwindow->theme->get_image_set("reset_button");
2348 return vfrm->get_w();
2351 int CWindowMaskClrMask::handle_event()
2359 // Get existing keyframe
2360 gui->get_keyframe(track, autos, keyframe, mask, point, 0);
2363 mwindow->undo->update_undo_before(_("del masks"), 0);
2364 ((MaskAutos*)track->automation->autos[AUTOMATION_MASK])->clear_all();
2366 gui->update_preview();
2367 mwindow->undo->update_undo_after(_("del masks"), LOAD_AUTOMATION);
2373 CWindowMaskGangFeather::CWindowMaskGangFeather(MWindow *mwindow,
2374 CWindowMaskGUI *gui, int x, int y)
2375 : BC_Toggle(x, y, mwindow->theme->get_image_set("gangpatch_data"), 0)
2377 this->mwindow = mwindow;
2379 set_tooltip(_("Gang feather"));
2382 CWindowMaskGangFeather::~CWindowMaskGangFeather()
2386 int CWindowMaskGangFeather::handle_event()
2391 CWindowMaskGUI::CWindowMaskGUI(MWindow *mwindow, CWindowTool *thread)
2392 : CWindowToolGUI(mwindow, thread,
2393 _(PROGRAM_NAME ": Mask"), 400, 660)
2395 this->mwindow = mwindow;
2396 this->thread = thread;
2404 CWindowMaskGUI::~CWindowMaskGUI()
2406 lock_window("CWindowMaskGUI::~CWindowMaskGUI");
2408 delete active_point;
2414 void CWindowMaskGUI::create_objects()
2416 int x = 10, y = 10, margin = mwindow->theme->widget_border;
2417 int clr_w = CWindowMaskClrMask::calculate_w(mwindow);
2418 int clr_x = get_w()-x - clr_w;
2419 int del_w = CWindowMaskDelMask::calculate_w(this,_("Delete"));
2420 int del_x = clr_x-2*margin - del_w;
2422 lock_window("CWindowMaskGUI::create_objects");
2423 BC_TitleBar *title_bar;
2424 add_subwindow(title_bar = new BC_TitleBar(x, y, get_w()-2*x, 20, 10, _("Masks on Track")));
2425 y += title_bar->get_h() + margin;
2427 add_subwindow(title = new BC_Title(x,y, _("Track:")));
2429 Track *track = mwindow->cwindow->calculate_affected_track();
2430 const char *text = track ? track->title : "";
2431 mwindow->cwindow->mask_track_id = track ? track->get_id() : -1;
2432 mask_on_track = new CWindowMaskOnTrack(mwindow, this, x1, y, 100, text);
2433 mask_on_track->create_objects();
2434 mask_on_track->set_tooltip(_("Video track"));
2435 int x2 = x1 + mask_on_track->get_w();
2436 add_subwindow(mask_track_tumbler = new CWindowMaskTrackTumbler(mwindow, this, x2, y));
2437 mwindow->edl->local_session->solo_track_id = -1;
2438 x2 = del_x + (del_w - CWindowMaskSoloTrack::calculate_w(this)) / 2;
2439 add_subwindow(mask_solo_track = new CWindowMaskSoloTrack(mwindow, this, x2, y, 0));
2440 y += mask_on_track->get_h() + margin;
2441 add_subwindow(title_bar = new BC_TitleBar(x, y, get_w()-2*x, 20, 10, _("Masks")));
2442 y += title_bar->get_h() + margin;
2443 add_subwindow(title = new BC_Title(x, y, _("Mask:")));
2444 mask_name = new CWindowMaskName(mwindow, this, x1, y, "");
2445 mask_name->create_objects();
2446 mask_name->set_tooltip(_("Mask name"));
2447 add_subwindow(clr_mask = new CWindowMaskClrMask(mwindow, this, clr_x, y));
2448 add_subwindow(del_mask = new CWindowMaskDelMask(mwindow, this, del_x, y));
2449 y += mask_name->get_h() + 2*margin;
2451 add_subwindow(title = new BC_Title(x, y, _("Select:")));
2453 BC_CheckBox::calculate_extents(this, &bw, &bh);
2454 int bdx = bw + margin;
2456 for( int i=0; i<SUBMASKS; x2+=bdx, ++i ) {
2457 int v = i == mwindow->edl->session->cwindow_mask ? 1 : 0;
2458 mask_buttons[i] = new CWindowMaskButton(mwindow, this, x2, y, i, v);
2459 add_subwindow(mask_buttons[i]);
2462 add_subwindow(mask_thumbler = new CWindowMaskThumbler(mwindow, this, x2, y));
2465 for( int i=0; i<SUBMASKS; x2+=bdx, ++i ) {
2466 char text[BCSTRLEN]; sprintf(text, "%d", i);
2467 int tx = (bw - get_text_width(MEDIUMFONT, text)) / 2;
2468 mask_blabels[i] = new BC_Title(x2+tx, y, text);
2469 add_subwindow(mask_blabels[i]);
2471 y += mask_blabels[0]->get_h() + margin;
2472 add_subwindow(unclr_mask = new CWindowMaskUnclear(mwindow, this, x, y, x1-x-2*margin));
2474 for( int i=0; i<SUBMASKS; x2+=bdx, ++i ) {
2475 mask_enables[i] = new CWindowMaskEnable(mwindow, this, x2, y, i, 1);
2476 add_subwindow(mask_enables[i]);
2478 y += mask_enables[0]->get_h() + 2*margin;
2479 add_subwindow(title_bar = new BC_TitleBar(x, y, get_w()-2*x, 20, 10, _("Fade & Feather")));
2480 y += title_bar->get_h() + margin;
2482 add_subwindow(title = new BC_Title(x, y, _("Fade:")));
2483 fade = new CWindowMaskFade(mwindow, this, x1, y);
2484 fade->create_objects();
2485 x2 = x1 + fade->get_w() + 2*margin;
2486 int w2 = clr_x-2*margin - x2;
2487 add_subwindow(fade_slider = new CWindowMaskFadeSlider(mwindow, this, x2, y, w2));
2488 add_subwindow(gang_fader = new CWindowMaskGangFader(mwindow, this, clr_x, y));
2489 y += fade->get_h() + margin;
2490 add_subwindow(title = new BC_Title(x, y, _("Feather:")));
2491 feather = new CWindowMaskFeather(mwindow, this, x1, y);
2492 feather->create_objects();
2493 w2 = clr_x - 2*margin - x2;
2494 feather_slider = new CWindowMaskFeatherSlider(mwindow, this, x2, y, w2, 0);
2495 add_subwindow(feather_slider);
2496 add_subwindow(gang_feather = new CWindowMaskGangFeather(mwindow, this, clr_x, y));
2497 y += feather->get_h() + 2*margin;
2498 add_subwindow(title_bar = new BC_TitleBar(x, y, get_w()-2*x, 20, 10, _("Mask Points")));
2499 y += title_bar->get_h() + margin;
2501 add_subwindow(title = new BC_Title(x, y, _("Point:")));
2502 active_point = new CWindowMaskAffectedPoint(mwindow, this, x1, y);
2503 active_point->create_objects();
2504 add_subwindow(del_point = new CWindowMaskDelPoint(mwindow, this, del_x, y));
2505 y += active_point->get_h() + margin;
2506 add_subwindow(title = new BC_Title(x, y, "X:"));
2507 this->x = new CWindowCoord(this, x1, y, (float)0.0);
2508 this->x->create_objects();
2509 add_subwindow(draw_markers = new CWindowMaskDrawMarkers(mwindow, this, del_x, y));
2510 y += this->x->get_h() + margin;
2511 add_subwindow(title = new BC_Title(x, y, "Y:"));
2512 this->y = new CWindowCoord(this, x1, y, (float)0.0);
2513 this->y->create_objects();
2514 add_subwindow(draw_boundary = new CWindowMaskDrawBoundary(mwindow, this, del_x, y));
2515 y += this->y->get_h() + 2*margin;
2516 add_subwindow(title_bar = new BC_TitleBar(x, y, get_w()-2*x, 20, 10, _("Pivot Point")));
2517 y += title_bar->get_h() + margin;
2519 add_subwindow(title = new BC_Title(x, y, "X:"));
2520 float cx = mwindow->edl->session->output_w / 2.f;
2521 focus_x = new CWindowCoord(this, x1, y, cx);
2522 focus_x->create_objects();
2523 add_subwindow(focus = new CWindowMaskFocus(mwindow, this, del_x, y));
2524 y += focus_x->get_h() + margin;
2525 add_subwindow(title = new BC_Title(x, y, "Y:"));
2526 float cy = mwindow->edl->session->output_h / 2.f;
2527 focus_y = new CWindowCoord(this, x1, y, cy);
2528 focus_y->create_objects();
2529 y += focus_x->get_h() + 2*margin;
2531 add_subwindow(bar = new BC_Bar(x, y, get_w()-2*x));
2532 y += bar->get_h() + margin;
2533 add_subwindow(this->apply_before_plugins = new CWindowMaskBeforePlugins(this, 10, y));
2534 y += this->apply_before_plugins->get_h();
2535 add_subwindow(this->disable_opengl_masking = new CWindowDisableOpenGLMasking(this, 10, y));
2536 y += this->disable_opengl_masking->get_h() + margin;
2537 add_subwindow(bar = new BC_Bar(x, y, get_w()-2*x));
2538 y += bar->get_h() + margin;
2541 add_subwindow(title = new BC_Title(x, y, _(
2542 "Shift+LMB: move an end point\n"
2543 "Ctrl+LMB: move a control point\n"
2544 "Alt+LMB: to drag translate the mask\n"
2545 "Shift+Key Delete to delete the point\n"
2546 "Wheel Up/Dn: rotate around pointer\n"
2547 "Shift+Wheel Up/Dn: scale around pointer\n"
2548 "Shift+MMB: Toggle focus center at pointer")));
2553 int CWindowMaskGUI::close_event()
2556 return CWindowToolGUI::close_event();
2559 void CWindowMaskGUI::done_event()
2561 if( mwindow->in_destructor ) return;
2562 int &solo_track_id = mwindow->edl->local_session->solo_track_id;
2563 if( solo_track_id >= 0 ) {
2569 void CWindowMaskGUI::get_keyframe(Track* &track,
2570 MaskAutos* &autos, MaskAuto* &keyframe,
2571 SubMask* &mask, MaskPoint* &point, int create_it)
2576 track = mwindow->cwindow->calculate_mask_track();
2578 track = mwindow->cwindow->calculate_affected_track();
2581 autos = (MaskAutos*)track->automation->autos[AUTOMATION_MASK];
2582 keyframe = (MaskAuto*)mwindow->cwindow->calculate_affected_auto(
2587 mask = !keyframe ? 0 :
2588 keyframe->get_submask(mwindow->edl->session->cwindow_mask);
2592 if( mwindow->cwindow->gui->affected_point < mask->points.total &&
2593 mwindow->cwindow->gui->affected_point >= 0 ) {
2594 point = mask->points.values[mwindow->cwindow->gui->affected_point];
2599 void CWindowMaskGUI::update()
2606 //printf("CWindowMaskGUI::update 1\n");
2607 get_keyframe(track, autos, keyframe, mask, point, 0);
2608 mask_on_track->set_back_color(!track || track->record ?
2609 get_resources()->text_background :
2610 get_resources()->text_background_disarmed);
2611 mask_on_track->update_items();
2612 mask_on_track->update(!track ? "" : track->title);
2613 mask_name->update_items(keyframe);
2614 const char *text = "";
2615 int sz = !keyframe ? 0 : keyframe->masks.size();
2616 int k = mwindow->edl->session->cwindow_mask;
2617 if( k >= 0 && k < sz )
2618 text = keyframe->masks[k]->name;
2620 k = mwindow->edl->session->cwindow_mask = 0;
2621 mask_name->update(text);
2622 update_buttons(keyframe, k);
2624 x->update(point->x);
2625 y->update(point->y);
2628 double position = mwindow->edl->local_session->get_selectionstart(1);
2629 int64_t position_i = track->to_units(position, 0);
2630 feather->update(autos->get_feather(position_i, k, PLAY_FORWARD));
2631 fade->update(autos->get_fader(position_i, k, PLAY_FORWARD));
2632 int show_mask = track->masks;
2633 for( int i=0; i<SUBMASKS; ++i )
2634 mask_enables[i]->update((show_mask>>i) & 1);
2637 apply_before_plugins->update(keyframe->apply_before_plugins);
2638 disable_opengl_masking->update(keyframe->disable_opengl_masking);
2640 active_point->update((int64_t)mwindow->cwindow->gui->affected_point);
2643 void CWindowMaskGUI::handle_event()
2650 get_keyframe(track, autos, keyframe, mask, point, 0);
2652 mwindow->undo->update_undo_before(_("mask point"), this);
2656 #ifdef USE_KEYFRAME_SPANNING
2657 // Create temp keyframe
2658 MaskAuto temp_keyframe(mwindow->edl, autos);
2659 temp_keyframe.copy_data(keyframe);
2660 // Get affected point in temp keyframe
2661 mask = temp_keyframe.get_submask(mwindow->edl->session->cwindow_mask);
2662 if(mwindow->cwindow->gui->affected_point < mask->points.total &&
2663 mwindow->cwindow->gui->affected_point >= 0)
2665 point = mask->points.values[mwindow->cwindow->gui->affected_point];
2670 point->x = atof(x->get_text());
2671 point->y = atof(y->get_text());
2672 // Commit to spanned keyframes
2673 autos->update_parameter(&temp_keyframe);
2676 point->x = atof(x->get_text());
2677 point->y = atof(y->get_text());
2682 mwindow->undo->update_undo_after(_("mask point"), LOAD_AUTOMATION);
2685 void CWindowMaskGUI::set_focused(int v, float cx, float cy)
2687 focus_x->update(cx);
2688 focus_y->update(cy);
2689 focus->update(focused = v);
2692 void CWindowMaskGUI::update_buttons(MaskAuto *keyframe, int k)
2694 int text_color = get_resources()->default_text_color;
2695 int high_color = get_resources()->button_highlighted;
2696 for( int i=0; i<SUBMASKS; ++i ) {
2697 int color = text_color;
2699 SubMask *submask = keyframe->get_submask(i);
2700 if( submask && submask->points.size() )
2703 mask_blabels[i]->set_color(color);
2704 mask_buttons[i]->update(i==k ? 1 : 0);
2708 CWindowRulerGUI::CWindowRulerGUI(MWindow *mwindow, CWindowTool *thread)
2709 : CWindowToolGUI(mwindow, thread, _(PROGRAM_NAME ": Ruler"), 320, 240)
2713 CWindowRulerGUI::~CWindowRulerGUI()
2717 void CWindowRulerGUI::create_objects()
2719 int x = 10, y = 10, x1 = 100;
2722 lock_window("CWindowRulerGUI::create_objects");
2723 add_subwindow(title = new BC_Title(x, y, _("Current:")));
2724 add_subwindow(current = new BC_TextBox(x1, y, 200, 1, ""));
2725 y += title->get_h() + 5;
2726 add_subwindow(title = new BC_Title(x, y, _("Point 1:")));
2727 add_subwindow(point1 = new BC_TextBox(x1, y, 200, 1, ""));
2728 y += title->get_h() + 5;
2729 add_subwindow(title = new BC_Title(x, y, _("Point 2:")));
2730 add_subwindow(point2 = new BC_TextBox(x1, y, 200, 1, ""));
2731 y += title->get_h() + 5;
2732 add_subwindow(title = new BC_Title(x, y, _("Deltas:")));
2733 add_subwindow(deltas = new BC_TextBox(x1, y, 200, 1, ""));
2734 y += title->get_h() + 5;
2735 add_subwindow(title = new BC_Title(x, y, _("Distance:")));
2736 add_subwindow(distance = new BC_TextBox(x1, y, 200, 1, ""));
2737 y += title->get_h() + 5;
2738 add_subwindow(title = new BC_Title(x, y, _("Angle:")));
2739 add_subwindow(angle = new BC_TextBox(x1, y, 200, 1, ""));
2740 y += title->get_h() + 10;
2741 char string[BCTEXTLEN];
2743 _("Press Ctrl to lock ruler to the\nnearest 45%c%c angle."),
2744 0xc2, 0xb0); // degrees utf
2745 add_subwindow(title = new BC_Title(x,
2748 y += title->get_h() + 10;
2749 sprintf(string, _("Press Alt to translate the ruler."));
2750 add_subwindow(title = new BC_Title(x,
2757 void CWindowRulerGUI::update()
2759 char string[BCTEXTLEN];
2760 int cx = mwindow->session->cwindow_output_x;
2761 int cy = mwindow->session->cwindow_output_y;
2762 sprintf(string, "%d, %d", cx, cy);
2763 current->update(string);
2764 double x1 = mwindow->edl->session->ruler_x1;
2765 double y1 = mwindow->edl->session->ruler_y1;
2766 sprintf(string, "%.0f, %.0f", x1, y1);
2767 point1->update(string);
2768 double x2 = mwindow->edl->session->ruler_x2;
2769 double y2 = mwindow->edl->session->ruler_y2;
2770 sprintf(string, "%.0f, %.0f", x2, y2);
2771 point2->update(string);
2772 double dx = x2 - x1, dy = y2 - y1;
2773 sprintf(string, "%s%.0f, %s%.0f", (dx>=0? "+":""), dx, (dy>=0? "+":""), dy);
2774 deltas->update(string);
2775 double d = sqrt(dx*dx + dy*dy);
2776 sprintf(string, _("%0.01f pixels"), d);
2777 distance->update(string);
2778 double a = d > 0 ? (atan2(-dy, dx) * 180/M_PI) : 0.;
2779 sprintf(string, "%0.02f %c%c", a, 0xc2, 0xb0);
2780 angle->update(string);
2783 void CWindowRulerGUI::handle_event()