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"
50 #include "trackcanvas.h"
51 #include "transportque.h"
54 CWindowTool::CWindowTool(MWindow *mwindow, CWindowGUI *gui)
57 this->mwindow = mwindow;
61 current_tool = CWINDOW_NONE;
63 input_lock = new Condition(0, "CWindowTool::input_lock");
64 output_lock = new Condition(1, "CWindowTool::output_lock");
65 tool_gui_lock = new Mutex("CWindowTool::tool_gui_lock");
68 CWindowTool::~CWindowTool()
79 void CWindowTool::start_tool(int operation)
81 CWindowToolGUI *new_gui = 0;
84 //printf("CWindowTool::start_tool 1\n");
85 if(current_tool != operation)
87 int previous_tool = current_tool;
88 current_tool = operation;
92 new_gui = new CWindowEyedropGUI(mwindow, this);
95 new_gui = new CWindowCropGUI(mwindow, this);
98 new_gui = new CWindowCameraGUI(mwindow, this);
100 case CWINDOW_PROJECTOR:
101 new_gui = new CWindowProjectorGUI(mwindow, this);
104 new_gui = new CWindowMaskGUI(mwindow, this);
107 new_gui = new CWindowRulerGUI(mwindow, this);
109 case CWINDOW_PROTECT:
110 mwindow->edl->session->tool_window = 0;
111 gui->composite_panel->operation[CWINDOW_TOOL_WINDOW]->update(0);
119 //printf("CWindowTool::start_tool 1\n");
125 // Wait for previous tool GUI to finish
126 output_lock->lock("CWindowTool::start_tool");
127 this->tool_gui = new_gui;
128 tool_gui->create_objects();
129 if( previous_tool == CWINDOW_PROTECT || previous_tool == CWINDOW_NONE ) {
130 mwindow->edl->session->tool_window = 1;
131 gui->composite_panel->operation[CWINDOW_TOOL_WINDOW]->update(1);
133 update_show_window();
135 // Signal thread to run next tool GUI
136 input_lock->unlock();
138 //printf("CWindowTool::start_tool 1\n");
143 tool_gui->lock_window("CWindowTool::start_tool");
145 tool_gui->unlock_window();
148 //printf("CWindowTool::start_tool 2\n");
153 void CWindowTool::stop_tool()
157 tool_gui->lock_window("CWindowTool::stop_tool");
158 tool_gui->set_done(0);
159 tool_gui->unlock_window();
163 void CWindowTool::show_tool()
165 if(tool_gui && mwindow->edl->session->tool_window)
167 tool_gui->lock_window("CWindowTool::show_tool");
168 tool_gui->show_window();
169 tool_gui->unlock_window();
173 void CWindowTool::hide_tool()
175 if(tool_gui && mwindow->edl->session->tool_window)
177 tool_gui->lock_window("CWindowTool::show_tool");
178 tool_gui->hide_window();
179 tool_gui->unlock_window();
184 void CWindowTool::run()
188 input_lock->lock("CWindowTool::run");
191 tool_gui->run_window();
192 tool_gui_lock->lock("CWindowTool::run");
195 tool_gui_lock->unlock();
197 output_lock->unlock();
201 void CWindowTool::update_show_window()
205 tool_gui->lock_window("CWindowTool::update_show_window");
207 if(mwindow->edl->session->tool_window)
210 tool_gui->show_window();
213 tool_gui->hide_window();
216 tool_gui->unlock_window();
220 void CWindowTool::raise_window()
224 gui->unlock_window();
225 tool_gui->lock_window("CWindowTool::raise_window");
226 tool_gui->raise_window();
227 tool_gui->unlock_window();
228 gui->lock_window("CWindowTool::raise_window");
232 void CWindowTool::update_values()
234 tool_gui_lock->lock("CWindowTool::update_values");
237 tool_gui->lock_window("CWindowTool::update_values");
240 tool_gui->unlock_window();
242 tool_gui_lock->unlock();
251 CWindowToolGUI::CWindowToolGUI(MWindow *mwindow,
257 mwindow->session->ctool_x,
258 mwindow->session->ctool_y,
267 this->mwindow = mwindow;
268 this->thread = thread;
269 current_operation = 0;
272 CWindowToolGUI::~CWindowToolGUI()
276 int CWindowToolGUI::close_event()
280 mwindow->edl->session->tool_window = 0;
285 thread->gui->lock_window("CWindowToolGUI::close_event");
286 thread->gui->composite_panel->set_operation(mwindow->edl->session->cwindow_operation);
287 thread->gui->flush();
288 thread->gui->unlock_window();
290 lock_window("CWindowToolGUI::close_event");
294 int CWindowToolGUI::keypress_event()
298 switch( get_keypress() ) {
301 return close_event();
314 resend_event(thread->gui);
321 int CWindowToolGUI::translation_event()
323 mwindow->session->ctool_x = get_x();
324 mwindow->session->ctool_y = get_y();
333 CWindowCoord::CWindowCoord(CWindowToolGUI *gui, int x, int y, float value, int log_increment = 0)
334 : BC_TumbleTextBox(gui, (float)value, (float)-65536, (float)65536, x, y, 100, 3)
337 set_log_floatincrement(log_increment);
340 CWindowCoord::CWindowCoord(CWindowToolGUI *gui, int x, int y, int value)
341 : BC_TumbleTextBox(gui, (int64_t)value, (int64_t)-65536, (int64_t)65536, x, y, 100, 3)
345 int CWindowCoord::handle_event()
347 gui->event_caller = this;
353 CWindowCropOK::CWindowCropOK(MWindow *mwindow, CWindowToolGUI *gui, int x, int y)
354 : BC_GenericButton(x, y, _("Do it"))
356 this->mwindow = mwindow;
359 int CWindowCropOK::handle_event()
361 mwindow->crop_video();
366 int CWindowCropOK::keypress_event()
368 if(get_keypress() == 0xd)
382 CWindowCropGUI::CWindowCropGUI(MWindow *mwindow, CWindowTool *thread)
383 : CWindowToolGUI(mwindow,
385 _(PROGRAM_NAME ": Crop"),
392 CWindowCropGUI::~CWindowCropGUI()
396 void CWindowCropGUI::create_objects()
401 lock_window("CWindowCropGUI::create_objects");
403 int pad = MAX(BC_TextBox::calculate_h(this, MEDIUMFONT, 1, 1),
404 BC_Title::calculate_h(this, "X")) + 5;
405 add_subwindow(title = new BC_Title(x, y, "X1:"));
406 column1 = MAX(column1, title->get_w());
408 add_subwindow(title = new BC_Title(x, y, _("W:")));
409 column1 = MAX(column1, title->get_w());
411 add_subwindow(new CWindowCropOK(mwindow, thread->tool_gui, x, y));
415 x1 = new CWindowCoord(thread->tool_gui, x, y,
416 mwindow->edl->session->crop_x1);
417 x1->create_objects();
419 width = new CWindowCoord(thread->tool_gui, x, y,
420 mwindow->edl->session->crop_x2 - mwindow->edl->session->crop_x1);
421 width->create_objects();
424 x += x1->get_w() + 10;
427 add_subwindow(title = new BC_Title(x, y, "Y1:"));
428 column2 = MAX(column2, title->get_w());
430 add_subwindow(title = new BC_Title(x, y, _("H:")));
431 column2 = MAX(column2, title->get_w());
436 y1 = new CWindowCoord(thread->tool_gui, x, y,
437 mwindow->edl->session->crop_y1);
438 y1->create_objects();
440 height = new CWindowCoord(thread->tool_gui, x, y,
441 mwindow->edl->session->crop_y2 - mwindow->edl->session->crop_y1);
442 height->create_objects();
446 void CWindowCropGUI::handle_event()
449 new_x1 = atol(x1->get_text());
450 new_y1 = atol(y1->get_text());
451 if(new_x1 != mwindow->edl->session->crop_x1)
453 mwindow->edl->session->crop_x2 = new_x1 +
454 mwindow->edl->session->crop_x2 -
455 mwindow->edl->session->crop_x1;
456 mwindow->edl->session->crop_x1 = new_x1;
458 if(new_y1 != mwindow->edl->session->crop_y1)
460 mwindow->edl->session->crop_y2 = new_y1 +
461 mwindow->edl->session->crop_y2 -
462 mwindow->edl->session->crop_y1;
463 mwindow->edl->session->crop_y1 = atol(y1->get_text());
465 mwindow->edl->session->crop_x2 = atol(width->get_text()) +
466 mwindow->edl->session->crop_x1;
467 mwindow->edl->session->crop_y2 = atol(height->get_text()) +
468 mwindow->edl->session->crop_y1;
470 mwindow->cwindow->gui->canvas->redraw(1);
473 void CWindowCropGUI::update()
475 x1->update((int64_t)mwindow->edl->session->crop_x1);
476 y1->update((int64_t)mwindow->edl->session->crop_y1);
477 width->update((int64_t)mwindow->edl->session->crop_x2 -
478 mwindow->edl->session->crop_x1);
479 height->update((int64_t)mwindow->edl->session->crop_y2 -
480 mwindow->edl->session->crop_y1);
484 CWindowEyedropGUI::CWindowEyedropGUI(MWindow *mwindow, CWindowTool *thread)
485 : CWindowToolGUI(mwindow, thread, _(PROGRAM_NAME ": Color"), 220, 290)
489 CWindowEyedropGUI::~CWindowEyedropGUI()
493 void CWindowEyedropGUI::create_objects()
495 int margin = mwindow->theme->widget_border;
498 int x2 = 70, x3 = x2 + 60;
499 lock_window("CWindowEyedropGUI::create_objects");
500 BC_Title *title0, *title1, *title2, *title3, *title4, *title5, *title6, *title7;
501 add_subwindow(title0 = new BC_Title(x, y,_("X,Y:")));
502 y += title0->get_h() + margin;
503 add_subwindow(title7 = new BC_Title(x, y, _("Radius:")));
504 y += BC_TextBox::calculate_h(this, MEDIUMFONT, 1, 1) + margin;
506 add_subwindow(title1 = new BC_Title(x, y, _("Red:")));
507 y += title1->get_h() + margin;
508 add_subwindow(title2 = new BC_Title(x, y, _("Green:")));
509 y += title2->get_h() + margin;
510 add_subwindow(title3 = new BC_Title(x, y, _("Blue:")));
511 y += title3->get_h() + margin;
513 add_subwindow(title4 = new BC_Title(x, y, "Y:"));
514 y += title4->get_h() + margin;
515 add_subwindow(title5 = new BC_Title(x, y, "U:"));
516 y += title5->get_h() + margin;
517 add_subwindow(title6 = new BC_Title(x, y, "V:"));
519 add_subwindow(current = new BC_Title(x2, title0->get_y(), ""));
521 radius = new CWindowCoord(this, x2, title7->get_y(),
522 mwindow->edl->session->eyedrop_radius);
523 radius->create_objects();
524 radius->set_boundaries((int64_t)0, (int64_t)255);
526 add_subwindow(red = new BC_Title(x2, title1->get_y(), "0"));
527 add_subwindow(green = new BC_Title(x2, title2->get_y(), "0"));
528 add_subwindow(blue = new BC_Title(x2, title3->get_y(), "0"));
529 add_subwindow(rgb_hex = new BC_Title(x3, red->get_y(), "#000000"));
531 add_subwindow(this->y = new BC_Title(x2, title4->get_y(), "0"));
532 add_subwindow(this->u = new BC_Title(x2, title5->get_y(), "0"));
533 add_subwindow(this->v = new BC_Title(x2, title6->get_y(), "0"));
534 add_subwindow(yuv_hex = new BC_Title(x3, this->y->get_y(), "#000000"));
536 y = title6->get_y() + this->v->get_h() + 2*margin;
537 add_subwindow(sample = new BC_SubWindow(x, y, 50, 50));
538 y += sample->get_h() + margin;
539 add_subwindow(use_max = new CWindowEyedropCheckBox(mwindow, this, x, y));
544 void CWindowEyedropGUI::update()
546 char string[BCTEXTLEN];
547 sprintf(string, "%d, %d",
548 thread->gui->eyedrop_x,
549 thread->gui->eyedrop_y);
550 current->update(string);
552 radius->update((int64_t)mwindow->edl->session->eyedrop_radius);
554 LocalSession *local_session = mwindow->edl->local_session;
555 int use_max = local_session->use_max;
556 float r = use_max ? local_session->red_max : local_session->red;
557 float g = use_max ? local_session->green_max : local_session->green;
558 float b = use_max ? local_session->blue_max : local_session->blue;
559 this->red->update(r);
560 this->green->update(g);
561 this->blue->update(b);
563 int rx = 255*r + 0.5; bclamp(rx,0,255);
564 int gx = 255*g + 0.5; bclamp(gx,0,255);
565 int bx = 255*b + 0.5; bclamp(bx,0,255);
566 char rgb_text[BCSTRLEN];
567 sprintf(rgb_text, "#%02x%02x%02x", rx, gx, bx);
568 rgb_hex->update(rgb_text);
571 YUV::yuv.rgb_to_yuv_f(r, g, b, y, u, v);
573 this->u->update(u); u += 0.5;
574 this->v->update(v); v += 0.5;
576 int yx = 255*y + 0.5; bclamp(yx,0,255);
577 int ux = 255*u + 0.5; bclamp(ux,0,255);
578 int vx = 255*v + 0.5; bclamp(vx,0,255);
579 char yuv_text[BCSTRLEN];
580 sprintf(yuv_text, "#%02x%02x%02x", yx, ux, vx);
581 yuv_hex->update(yuv_text);
583 int rgb = (rx << 16) | (gx << 8) | (bx << 0);
584 sample->set_color(rgb);
585 sample->draw_box(0, 0, sample->get_w(), sample->get_h());
586 sample->set_color(BLACK);
587 sample->draw_rectangle(0, 0, sample->get_w(), sample->get_h());
591 void CWindowEyedropGUI::handle_event()
593 int new_radius = atoi(radius->get_text());
594 if(new_radius != mwindow->edl->session->eyedrop_radius)
596 CWindowGUI *gui = mwindow->cwindow->gui;
597 if(gui->eyedrop_visible)
599 gui->lock_window("CWindowEyedropGUI::handle_event");
602 gui->canvas->do_eyedrop(rerender, 0, 1);
605 mwindow->edl->session->eyedrop_radius = new_radius;
607 if(gui->eyedrop_visible)
611 gui->canvas->do_eyedrop(rerender, 0, 1);
612 gui->unlock_window();
619 /* Buttons to control Keyframe-Curve-Mode for Projector or Camera */
621 // Configuration for all possible Keyframe Curve Mode toggles
623 FloatAuto::t_mode mode;
629 const _CVD Camera_Crv_Smooth =
633 N_("\"smooth\" Curve on current Camera Keyframes")
635 const _CVD Camera_Crv_Linear =
639 N_("\"linear\" Curve on current Camera Keyframes")
641 const _CVD Projector_Crv_Smooth =
645 N_("\"smooth\" Curve on current Projector Keyframes")
647 const _CVD Projector_Crv_Linear =
651 N_("\"linear\" Curve on current Projector Keyframes")
654 // Implementation Class für Keyframe Curve Mode buttons
656 // This button reflects the state of the "current" keyframe
657 // (the nearest keyframe on the left) for all three automation
658 // lines together. Clicking on this button (re)sets the curve
659 // mode for the three "current" keyframes simultanously, but
660 // never creates a new keyframe.
662 class CWindowCurveToggle : public BC_Toggle
665 CWindowCurveToggle(_CVD mode, MWindow *mwindow, CWindowToolGUI *gui, int x, int y);
666 void check_toggle_state(FloatAuto *x, FloatAuto *y, FloatAuto *z);
675 CWindowCurveToggle::CWindowCurveToggle(_CVD mode, MWindow *mwindow, CWindowToolGUI *gui, int x, int y)
676 : BC_Toggle(x, y, mwindow->theme->get_image_set(mode.icon_id), false),
680 this->mwindow = mwindow;
681 set_tooltip(_(cfg.tooltip));
684 void CWindowCurveToggle::check_toggle_state(FloatAuto *x, FloatAuto *y, FloatAuto *z)
686 // the toggle state is only set to ON if all
687 // three automation lines have the same curve mode.
688 // For mixed states the toggle stays off.
689 set_value( x->curve_mode == this->cfg.mode &&
690 y->curve_mode == this->cfg.mode &&
691 z->curve_mode == this->cfg.mode
692 ,true // redraw to show new state
696 int CWindowCurveToggle::handle_event()
698 Track *track = mwindow->cwindow->calculate_affected_track();
700 FloatAuto *x=0, *y=0, *z=0;
701 mwindow->cwindow->calculate_affected_autos(track,
702 &x, &y, &z, cfg.use_camera, 0,0,0); // don't create new keyframe
703 if( x ) x->change_curve_mode(cfg.mode);
704 if( y ) y->change_curve_mode(cfg.mode);
705 if( z ) z->change_curve_mode(cfg.mode);
708 gui->update_preview();
715 CWindowEyedropCheckBox::CWindowEyedropCheckBox(MWindow *mwindow,
716 CWindowEyedropGUI *gui, int x, int y)
717 : BC_CheckBox(x, y, mwindow->edl->local_session->use_max, _("Use maximum"))
719 this->mwindow = mwindow;
723 int CWindowEyedropCheckBox::handle_event()
725 mwindow->edl->local_session->use_max = get_value();
732 CWindowCameraGUI::CWindowCameraGUI(MWindow *mwindow, CWindowTool *thread)
733 : CWindowToolGUI(mwindow,
735 _(PROGRAM_NAME ": Camera"),
740 CWindowCameraGUI::~CWindowCameraGUI()
744 void CWindowCameraGUI::create_objects()
746 int x = 10, y = 10, x1;
747 Track *track = mwindow->cwindow->calculate_affected_track();
748 FloatAuto *x_auto = 0, *y_auto = 0, *z_auto = 0;
752 lock_window("CWindowCameraGUI::create_objects");
754 mwindow->cwindow->calculate_affected_autos(track,
755 &x_auto, &y_auto, &z_auto, 1, 0, 0, 0);
758 add_subwindow(title = new BC_Title(x, y, "X:"));
760 this->x = new CWindowCoord(this, x, y,
761 x_auto ? x_auto->get_value() : (float)0);
762 this->x->create_objects();
767 add_subwindow(title = new BC_Title(x, y, "Y:"));
769 this->y = new CWindowCoord(this, x, y,
770 y_auto ? y_auto->get_value() : (float)0);
771 this->y->create_objects();
774 add_subwindow(title = new BC_Title(x, y, "Z:"));
776 this->z = new CWindowCoord(this, x, y,
777 z_auto ? z_auto->get_value() : (float)1);
778 this->z->create_objects();
779 this->z->set_increment(0.01);
783 add_subwindow(button = new CWindowCameraLeft(mwindow, this, x1, y));
784 x1 += button->get_w();
785 add_subwindow(button = new CWindowCameraCenter(mwindow, this, x1, y));
786 x1 += button->get_w();
787 add_subwindow(button = new CWindowCameraRight(mwindow, this, x1, y));
789 y += button->get_h();
791 add_subwindow(button = new CWindowCameraTop(mwindow, this, x1, y));
792 x1 += button->get_w();
793 add_subwindow(button = new CWindowCameraMiddle(mwindow, this, x1, y));
794 x1 += button->get_w();
795 add_subwindow(button = new CWindowCameraBottom(mwindow, this, x1, y));
796 // additional Buttons to control the curve mode of the "current" keyframe
797 x1 += button->get_w() + 15;
798 add_subwindow(this->t_smooth = new CWindowCurveToggle(Camera_Crv_Smooth, mwindow, this, x1, y));
799 x1 += button->get_w();
800 add_subwindow(this->t_linear = new CWindowCurveToggle(Camera_Crv_Linear, mwindow, this, x1, y));
802 // fill in current auto keyframe values, set toggle states.
807 void CWindowCameraGUI::update_preview()
809 CWindowGUI *cgui = mwindow->cwindow->gui;
810 cgui->lock_window("CWindowCameraGUI::update_preview");
811 cgui->sync_parameters(CHANGE_PARAMS, 0, 1);
812 cgui->unlock_window();
816 void CWindowCameraGUI::handle_event()
818 FloatAuto *x_auto = 0;
819 FloatAuto *y_auto = 0;
820 FloatAuto *z_auto = 0;
821 Track *track = mwindow->cwindow->calculate_affected_track();
824 mwindow->undo->update_undo_before(_("camera"), this);
825 if(event_caller == x)
827 x_auto = (FloatAuto*)mwindow->cwindow->calculate_affected_auto(
828 track->automation->autos[AUTOMATION_CAMERA_X],
832 x_auto->set_value(atof(x->get_text()));
838 if(event_caller == y)
840 y_auto = (FloatAuto*)mwindow->cwindow->calculate_affected_auto(
841 track->automation->autos[AUTOMATION_CAMERA_Y],
845 y_auto->set_value(atof(y->get_text()));
851 if(event_caller == z)
853 z_auto = (FloatAuto*)mwindow->cwindow->calculate_affected_auto(
854 track->automation->autos[AUTOMATION_CAMERA_Z],
858 float zoom = atof(z->get_text());
859 if(zoom > 100.) zoom = 100.;
861 if(zoom < 0.01) zoom = 0.01;
862 // Doesn't allow user to enter from scratch
863 // if(zoom != atof(z->get_text()))
866 z_auto->set_value(zoom);
867 mwindow->gui->lock_window("CWindowCameraGUI::handle_event");
868 mwindow->gui->draw_overlays(1);
869 mwindow->gui->unlock_window();
875 mwindow->undo->update_undo_after(_("camera"), LOAD_ALL);
879 void CWindowCameraGUI::update()
881 FloatAuto *x_auto = 0;
882 FloatAuto *y_auto = 0;
883 FloatAuto *z_auto = 0;
884 Track *track = mwindow->cwindow->calculate_affected_track();
886 mwindow->cwindow->calculate_affected_autos(track,
887 &x_auto, &y_auto, &z_auto, 1, 0, 0, 0);
891 x->update(x_auto->get_value());
893 y->update(y_auto->get_value());
895 float value = z_auto->get_value();
897 thread->gui->lock_window("CWindowCameraGUI::update");
898 thread->gui->composite_panel->cpanel_zoom->update(value);
899 thread->gui->unlock_window();
902 if( x_auto && y_auto && z_auto )
904 t_smooth->check_toggle_state(x_auto, y_auto, z_auto);
905 t_linear->check_toggle_state(x_auto, y_auto, z_auto);
912 CWindowCameraLeft::CWindowCameraLeft(MWindow *mwindow, CWindowCameraGUI *gui, int x, int y)
913 : BC_Button(x, y, mwindow->theme->get_image_set("left_justify"))
916 this->mwindow = mwindow;
917 set_tooltip(_("Left justify"));
919 int CWindowCameraLeft::handle_event()
921 FloatAuto *x_auto = 0;
922 FloatAuto *z_auto = 0;
923 Track *track = mwindow->cwindow->calculate_affected_track();
925 mwindow->cwindow->calculate_affected_autos(track,
926 &x_auto, 0, &z_auto, 1, 1, 0, 0);
932 track->get_source_dimensions(
933 mwindow->edl->local_session->get_selectionstart(1),
939 mwindow->undo->update_undo_before(_("camera"), 0);
941 (double)track->track_w / z_auto->get_value() / 2 -
943 mwindow->undo->update_undo_after(_("camera"), LOAD_ALL);
945 gui->update_preview();
953 CWindowCameraCenter::CWindowCameraCenter(MWindow *mwindow, CWindowCameraGUI *gui, int x, int y)
954 : BC_Button(x, y, mwindow->theme->get_image_set("center_justify"))
957 this->mwindow = mwindow;
958 set_tooltip(_("Center horizontal"));
960 int CWindowCameraCenter::handle_event()
962 FloatAuto *x_auto = 0;
963 Track *track = mwindow->cwindow->calculate_affected_track();
965 x_auto = (FloatAuto*)mwindow->cwindow->calculate_affected_auto(
966 track->automation->autos[AUTOMATION_CAMERA_X],
971 mwindow->undo->update_undo_before(_("camera"), 0);
972 x_auto->set_value(0);
974 gui->update_preview();
975 mwindow->undo->update_undo_after(_("camera"), LOAD_ALL);
982 CWindowCameraRight::CWindowCameraRight(MWindow *mwindow, CWindowCameraGUI *gui, int x, int y)
983 : BC_Button(x, y, mwindow->theme->get_image_set("right_justify"))
986 this->mwindow = mwindow;
987 set_tooltip(_("Right justify"));
989 int CWindowCameraRight::handle_event()
991 FloatAuto *x_auto = 0;
992 FloatAuto *z_auto = 0;
993 Track *track = mwindow->cwindow->calculate_affected_track();
995 mwindow->cwindow->calculate_affected_autos(track,
996 &x_auto, 0, &z_auto, 1, 1, 0, 0);
1002 track->get_source_dimensions(
1003 mwindow->edl->local_session->get_selectionstart(1),
1009 mwindow->undo->update_undo_before(_("camera"), 0);
1010 x_auto->set_value( -((double)track->track_w / z_auto->get_value() / 2 -
1013 gui->update_preview();
1014 mwindow->undo->update_undo_after(_("camera"), LOAD_ALL);
1022 CWindowCameraTop::CWindowCameraTop(MWindow *mwindow, CWindowCameraGUI *gui, int x, int y)
1023 : BC_Button(x, y, mwindow->theme->get_image_set("top_justify"))
1026 this->mwindow = mwindow;
1027 set_tooltip(_("Top justify"));
1029 int CWindowCameraTop::handle_event()
1031 FloatAuto *y_auto = 0;
1032 FloatAuto *z_auto = 0;
1033 Track *track = mwindow->cwindow->calculate_affected_track();
1035 mwindow->cwindow->calculate_affected_autos(track,
1036 0, &y_auto, &z_auto, 1, 0, 1, 0);
1039 if(y_auto && z_auto)
1042 track->get_source_dimensions(
1043 mwindow->edl->local_session->get_selectionstart(1),
1049 mwindow->undo->update_undo_before(_("camera"), 0);
1050 y_auto->set_value((double)track->track_h / z_auto->get_value() / 2 -
1053 gui->update_preview();
1054 mwindow->undo->update_undo_after(_("camera"), LOAD_ALL);
1062 CWindowCameraMiddle::CWindowCameraMiddle(MWindow *mwindow, CWindowCameraGUI *gui, int x, int y)
1063 : BC_Button(x, y, mwindow->theme->get_image_set("middle_justify"))
1066 this->mwindow = mwindow;
1067 set_tooltip(_("Center vertical"));
1069 int CWindowCameraMiddle::handle_event()
1071 FloatAuto *y_auto = 0;
1072 Track *track = mwindow->cwindow->calculate_affected_track();
1074 y_auto = (FloatAuto*)mwindow->cwindow->calculate_affected_auto(
1075 track->automation->autos[AUTOMATION_CAMERA_Y], 1);
1079 mwindow->undo->update_undo_before(_("camera"), 0);
1080 y_auto->set_value(0);
1082 gui->update_preview();
1083 mwindow->undo->update_undo_after(_("camera"), LOAD_ALL);
1090 CWindowCameraBottom::CWindowCameraBottom(MWindow *mwindow, CWindowCameraGUI *gui, int x, int y)
1091 : BC_Button(x, y, mwindow->theme->get_image_set("bottom_justify"))
1094 this->mwindow = mwindow;
1095 set_tooltip(_("Bottom justify"));
1097 int CWindowCameraBottom::handle_event()
1099 FloatAuto *y_auto = 0;
1100 FloatAuto *z_auto = 0;
1101 Track *track = mwindow->cwindow->calculate_affected_track();
1103 mwindow->cwindow->calculate_affected_autos(track,
1104 0, &y_auto, &z_auto, 1, 0, 1, 0);
1107 if(y_auto && z_auto)
1110 track->get_source_dimensions(
1111 mwindow->edl->local_session->get_selectionstart(1),
1117 mwindow->undo->update_undo_before(_("camera"), 0);
1118 y_auto->set_value(-((double)track->track_h / z_auto->get_value() / 2 -
1121 gui->update_preview();
1122 mwindow->undo->update_undo_after(_("camera"), LOAD_ALL);
1130 CWindowProjectorGUI::CWindowProjectorGUI(MWindow *mwindow, CWindowTool *thread)
1131 : CWindowToolGUI(mwindow,
1133 _(PROGRAM_NAME ": Projector"),
1138 CWindowProjectorGUI::~CWindowProjectorGUI()
1141 void CWindowProjectorGUI::create_objects()
1143 int x = 10, y = 10, x1;
1144 Track *track = mwindow->cwindow->calculate_affected_track();
1145 FloatAuto *x_auto = 0;
1146 FloatAuto *y_auto = 0;
1147 FloatAuto *z_auto = 0;
1151 lock_window("CWindowProjectorGUI::create_objects");
1153 mwindow->cwindow->calculate_affected_autos(track,
1154 &x_auto, &y_auto, &z_auto, 0, 0, 0, 0);
1157 add_subwindow(title = new BC_Title(x, y, "X:"));
1158 x += title->get_w();
1159 this->x = new CWindowCoord(this, x, y,
1160 x_auto ? x_auto->get_value() : (float)0);
1161 this->x->create_objects();
1164 add_subwindow(title = new BC_Title(x, y, "Y:"));
1165 x += title->get_w();
1166 this->y = new CWindowCoord(this, x, y,
1167 y_auto ? y_auto->get_value() : (float)0);
1168 this->y->create_objects();
1171 add_subwindow(title = new BC_Title(x, y, "Z:"));
1172 x += title->get_w();
1173 this->z = new CWindowCoord(this, x, y,
1174 z_auto ? z_auto->get_value() : (float)1);
1175 this->z->create_objects();
1176 this->z->set_increment(0.01);
1180 add_subwindow(button = new CWindowProjectorLeft(mwindow, this, x1, y));
1181 x1 += button->get_w();
1182 add_subwindow(button = new CWindowProjectorCenter(mwindow, this, x1, y));
1183 x1 += button->get_w();
1184 add_subwindow(button = new CWindowProjectorRight(mwindow, this, x1, y));
1186 y += button->get_h();
1188 add_subwindow(button = new CWindowProjectorTop(mwindow, this, x1, y));
1189 x1 += button->get_w();
1190 add_subwindow(button = new CWindowProjectorMiddle(mwindow, this, x1, y));
1191 x1 += button->get_w();
1192 add_subwindow(button = new CWindowProjectorBottom(mwindow, this, x1, y));
1194 // additional Buttons to control the curve mode of the "current" keyframe
1195 x1 += button->get_w() + 15;
1196 add_subwindow(this->t_smooth = new CWindowCurveToggle(Projector_Crv_Smooth, mwindow, this, x1, y));
1197 x1 += button->get_w();
1198 add_subwindow(this->t_linear = new CWindowCurveToggle(Projector_Crv_Linear, mwindow, this, x1, y));
1200 // fill in current auto keyframe values, set toggle states.
1205 void CWindowProjectorGUI::update_preview()
1207 CWindowGUI *cgui = mwindow->cwindow->gui;
1208 cgui->lock_window("CWindowProjectorGUI::update_preview");
1209 cgui->sync_parameters(CHANGE_PARAMS, 0, 1);
1210 cgui->unlock_window();
1213 void CWindowProjectorGUI::handle_event()
1215 FloatAuto *x_auto = 0;
1216 FloatAuto *y_auto = 0;
1217 FloatAuto *z_auto = 0;
1218 Track *track = mwindow->cwindow->calculate_affected_track();
1222 mwindow->undo->update_undo_before(_("projector"), this);
1223 if(event_caller == x)
1225 x_auto = (FloatAuto*)mwindow->cwindow->calculate_affected_auto(
1226 track->automation->autos[AUTOMATION_PROJECTOR_X],
1230 x_auto->set_value(atof(x->get_text()));
1236 if(event_caller == y)
1238 y_auto = (FloatAuto*)mwindow->cwindow->calculate_affected_auto(
1239 track->automation->autos[AUTOMATION_PROJECTOR_Y],
1243 y_auto->set_value(atof(y->get_text()));
1249 if(event_caller == z)
1251 z_auto = (FloatAuto*)mwindow->cwindow->calculate_affected_auto(
1252 track->automation->autos[AUTOMATION_PROJECTOR_Z],
1256 float zoom = atof(z->get_text());
1257 if(zoom > 100.) zoom = 100.;
1258 else if(zoom < 0.01) zoom = 0.01;
1259 // if (zoom != atof(z->get_text()))
1261 z_auto->set_value(zoom);
1263 mwindow->gui->lock_window("CWindowProjectorGUI::handle_event");
1264 mwindow->gui->draw_overlays(1);
1265 mwindow->gui->unlock_window();
1271 mwindow->undo->update_undo_after(_("projector"), LOAD_ALL);
1275 void CWindowProjectorGUI::update()
1277 FloatAuto *x_auto = 0;
1278 FloatAuto *y_auto = 0;
1279 FloatAuto *z_auto = 0;
1280 Track *track = mwindow->cwindow->calculate_affected_track();
1282 mwindow->cwindow->calculate_affected_autos(track,
1283 &x_auto, &y_auto, &z_auto, 0, 0, 0, 0);
1287 x->update(x_auto->get_value());
1289 y->update(y_auto->get_value());
1291 float value = z_auto->get_value();
1293 thread->gui->lock_window("CWindowProjectorGUI::update");
1294 thread->gui->composite_panel->cpanel_zoom->update(value);
1295 thread->gui->unlock_window();
1298 if( x_auto && y_auto && z_auto )
1300 t_smooth->check_toggle_state(x_auto, y_auto, z_auto);
1301 t_linear->check_toggle_state(x_auto, y_auto, z_auto);
1305 CWindowProjectorLeft::CWindowProjectorLeft(MWindow *mwindow, CWindowProjectorGUI *gui, int x, int y)
1306 : BC_Button(x, y, mwindow->theme->get_image_set("left_justify"))
1309 this->mwindow = mwindow;
1310 set_tooltip(_("Left justify"));
1312 int CWindowProjectorLeft::handle_event()
1314 FloatAuto *x_auto = 0;
1315 FloatAuto *z_auto = 0;
1316 Track *track = mwindow->cwindow->calculate_affected_track();
1318 mwindow->cwindow->calculate_affected_autos(track,
1319 &x_auto, 0, &z_auto, 0, 1, 0, 0);
1321 if(x_auto && z_auto)
1323 mwindow->undo->update_undo_before(_("projector"), 0);
1324 x_auto->set_value( (double)track->track_w * z_auto->get_value() / 2 -
1325 (double)mwindow->edl->session->output_w / 2 );
1327 gui->update_preview();
1328 mwindow->undo->update_undo_after(_("projector"), LOAD_ALL);
1335 CWindowProjectorCenter::CWindowProjectorCenter(MWindow *mwindow, CWindowProjectorGUI *gui, int x, int y)
1336 : BC_Button(x, y, mwindow->theme->get_image_set("center_justify"))
1339 this->mwindow = mwindow;
1340 set_tooltip(_("Center horizontal"));
1342 int CWindowProjectorCenter::handle_event()
1344 FloatAuto *x_auto = 0;
1345 Track *track = mwindow->cwindow->calculate_affected_track();
1347 x_auto = (FloatAuto*)mwindow->cwindow->calculate_affected_auto(
1348 track->automation->autos[AUTOMATION_PROJECTOR_X],
1353 mwindow->undo->update_undo_before(_("projector"), 0);
1354 x_auto->set_value(0);
1356 gui->update_preview();
1357 mwindow->undo->update_undo_after(_("projector"), LOAD_ALL);
1364 CWindowProjectorRight::CWindowProjectorRight(MWindow *mwindow, CWindowProjectorGUI *gui, int x, int y)
1365 : BC_Button(x, y, mwindow->theme->get_image_set("right_justify"))
1368 this->mwindow = mwindow;
1369 set_tooltip(_("Right justify"));
1371 int CWindowProjectorRight::handle_event()
1373 FloatAuto *x_auto = 0;
1374 FloatAuto *z_auto = 0;
1375 Track *track = mwindow->cwindow->calculate_affected_track();
1377 mwindow->cwindow->calculate_affected_autos(track,
1378 &x_auto, 0, &z_auto, 0, 1, 0, 0);
1381 if(x_auto && z_auto)
1383 mwindow->undo->update_undo_before(_("projector"), 0);
1384 x_auto->set_value( -((double)track->track_w * z_auto->get_value() / 2 -
1385 (double)mwindow->edl->session->output_w / 2));
1387 gui->update_preview();
1388 mwindow->undo->update_undo_after(_("projector"), LOAD_ALL);
1395 CWindowProjectorTop::CWindowProjectorTop(MWindow *mwindow, CWindowProjectorGUI *gui, int x, int y)
1396 : BC_Button(x, y, mwindow->theme->get_image_set("top_justify"))
1399 this->mwindow = mwindow;
1400 set_tooltip(_("Top justify"));
1402 int CWindowProjectorTop::handle_event()
1404 FloatAuto *y_auto = 0;
1405 FloatAuto *z_auto = 0;
1406 Track *track = mwindow->cwindow->calculate_affected_track();
1408 mwindow->cwindow->calculate_affected_autos(track,
1409 0, &y_auto, &z_auto, 0, 0, 1, 0);
1412 if(y_auto && z_auto)
1414 mwindow->undo->update_undo_before(_("projector"), 0);
1415 y_auto->set_value( (double)track->track_h * z_auto->get_value() / 2 -
1416 (double)mwindow->edl->session->output_h / 2 );
1418 gui->update_preview();
1419 mwindow->undo->update_undo_after(_("projector"), LOAD_ALL);
1426 CWindowProjectorMiddle::CWindowProjectorMiddle(MWindow *mwindow, CWindowProjectorGUI *gui, int x, int y)
1427 : BC_Button(x, y, mwindow->theme->get_image_set("middle_justify"))
1430 this->mwindow = mwindow;
1431 set_tooltip(_("Center vertical"));
1433 int CWindowProjectorMiddle::handle_event()
1435 FloatAuto *y_auto = 0;
1436 Track *track = mwindow->cwindow->calculate_affected_track();
1438 y_auto = (FloatAuto*)mwindow->cwindow->calculate_affected_auto(
1439 track->automation->autos[AUTOMATION_PROJECTOR_Y], 1);
1443 mwindow->undo->update_undo_before(_("projector"), 0);
1444 y_auto->set_value(0);
1446 gui->update_preview();
1447 mwindow->undo->update_undo_after(_("projector"), LOAD_ALL);
1454 CWindowProjectorBottom::CWindowProjectorBottom(MWindow *mwindow, CWindowProjectorGUI *gui, int x, int y)
1455 : BC_Button(x, y, mwindow->theme->get_image_set("bottom_justify"))
1458 this->mwindow = mwindow;
1459 set_tooltip(_("Bottom justify"));
1461 int CWindowProjectorBottom::handle_event()
1463 FloatAuto *y_auto = 0;
1464 FloatAuto *z_auto = 0;
1465 Track *track = mwindow->cwindow->calculate_affected_track();
1467 mwindow->cwindow->calculate_affected_autos(track,
1468 0, &y_auto, &z_auto, 0, 0, 1, 0);
1471 if(y_auto && z_auto)
1473 mwindow->undo->update_undo_before(_("projector"), 0);
1474 y_auto->set_value( -((double)track->track_h * z_auto->get_value() / 2 -
1475 (double)mwindow->edl->session->output_h / 2));
1477 gui->update_preview();
1478 mwindow->undo->update_undo_after(_("projector"), LOAD_ALL);
1485 CWindowMaskName::CWindowMaskName(MWindow *mwindow,
1486 CWindowToolGUI *gui, int x, int y, const char *text)
1487 : BC_PopupTextBox(gui, 0, text, x, y, 100, 160)
1489 this->mwindow = mwindow;
1493 CWindowMaskName::~CWindowMaskName()
1497 int CWindowMaskName::handle_event()
1504 //printf("CWindowMaskGUI::update 1\n");
1505 ((CWindowMaskGUI*)gui)->get_keyframe(track, autos, keyframe, mask, point, 0);
1507 int k = get_number();
1508 if( k < 0 ) k = mwindow->edl->session->cwindow_mask;
1509 else mwindow->edl->session->cwindow_mask = k;
1510 if( k >= 0 && k < mask_items.size() ) {
1511 mask_items[k]->set_text(get_text());
1512 update_list(&mask_items);
1514 #ifdef USE_KEYFRAME_SPANNING
1515 MaskAuto temp_keyframe(mwindow->edl, autos);
1516 temp_keyframe.copy_data(keyframe);
1517 SubMask *submask = temp_keyframe.get_submask(mwindow->edl->session->cwindow_mask);
1518 memset(submask->name, 0, sizeof(submask->name));
1519 strncpy(submask->name, get_text(), sizeof(submask->name)-1);
1520 ((MaskAutos*)track->automation->autos[AUTOMATION_MASK])->update_parameter(&temp_keyframe);
1522 for(MaskAuto *current = (MaskAuto*)autos->default_auto; current; ) {
1523 SubMask *submask = current->get_submask(mwindow->edl->session->cwindow_mask);
1524 memset(submask->name, 0, sizeof(submask->name));
1525 strncpy(submask->name, get_text(), sizeof(submask->name));
1526 current = current == (MaskAuto*)autos->default_auto ?
1527 (MaskAuto*)autos->first : (MaskAuto*)NEXT;
1531 gui->update_preview();
1536 void CWindowMaskName::update_items(MaskAuto *keyframe)
1538 mask_items.remove_all_objects();
1539 int sz = keyframe->masks.size();
1540 for( int i=0; i<sz; ++i ) {
1541 SubMask *sub_mask = keyframe->masks.get(i);
1542 char *text = sub_mask->name;
1543 mask_items.append(new BC_ListBoxItem(text));
1545 update_list(&mask_items);
1549 CWindowMaskDelMask::CWindowMaskDelMask(MWindow *mwindow,
1550 CWindowToolGUI *gui, int x, int y)
1551 : BC_GenericButton(x, y, _("Delete"))
1553 this->mwindow = mwindow;
1555 set_tooltip(_("Delete mask"));
1558 int CWindowMaskDelMask::handle_event()
1567 // Get existing keyframe
1568 ((CWindowMaskGUI*)gui)->get_keyframe(track, autos, keyframe, mask, point, 0);
1571 mwindow->undo->update_undo_before(_("mask delete"), 0);
1573 #ifdef USE_KEYFRAME_SPANNING
1574 // Create temp keyframe
1575 MaskAuto temp_keyframe(mwindow->edl, autos);
1576 temp_keyframe.copy_data(keyframe);
1577 SubMask *submask = temp_keyframe.get_submask(mwindow->edl->session->cwindow_mask);
1578 submask->points.remove_all_objects();
1580 // Commit change to span of keyframes
1581 ((MaskAutos*)track->automation->autos[AUTOMATION_MASK])->update_parameter(&temp_keyframe);
1583 for(MaskAuto *current = (MaskAuto*)autos->default_auto; current; ) {
1584 SubMask *submask = current->get_submask(mwindow->edl->session->cwindow_mask);
1585 submask->points.clear();
1586 current = current == (MaskAuto*)autos->default_auto ?
1587 (MaskAuto*)autos->first : (MaskAuto*)NEXT;
1591 if( mwindow->cwindow->gui->affected_point >= total_points )
1592 mwindow->cwindow->gui->affected_point =
1593 total_points > 0 ? total_points-1 : 0;
1596 gui->update_preview();
1597 mwindow->undo->update_undo_after(_("mask delete"), LOAD_AUTOMATION);
1603 CWindowMaskDelPoint::CWindowMaskDelPoint(MWindow *mwindow,
1604 CWindowToolGUI *gui, int x, int y)
1605 : BC_GenericButton(x, y, _("Delete"))
1607 this->mwindow = mwindow;
1609 set_tooltip(_("Delete point"));
1612 int CWindowMaskDelPoint::handle_event()
1621 // Get existing keyframe
1622 ((CWindowMaskGUI*)gui)->get_keyframe(track, autos, keyframe, mask, point, 0);
1624 mwindow->undo->update_undo_before(_("point delete"), 0);
1626 #ifdef USE_KEYFRAME_SPANNING
1627 // Create temp keyframe
1628 MaskAuto temp_keyframe(mwindow->edl, autos);
1629 temp_keyframe.copy_data(keyframe);
1631 SubMask *submask = temp_keyframe.get_submask(mwindow->edl->session->cwindow_mask);
1632 int i = mwindow->cwindow->gui->affected_point;
1633 for( ; i<submask->points.total-1; ++i )
1634 *submask->points.values[i] = *submask->points.values[i+1];
1635 if( submask->points.total > 0 ) {
1636 point = submask->points.values[submask->points.total-1];
1637 submask->points.remove_object(point);
1639 total_points = submask->points.total;
1641 // Commit change to span of keyframes
1642 ((MaskAutos*)track->automation->autos[AUTOMATION_MASK])->update_parameter(&temp_keyframe);
1644 MaskAuto *current = (MaskAuto*)autos->default_auto;
1646 SubMask *submask = current->get_submask(mwindow->edl->session->cwindow_mask);
1647 int i = mwindow->cwindow->gui->affected_point;
1648 for( ; i<submask->points.total-1; ++i ) {
1649 *submask->points.values[i] = *submask->points.values[i+1];
1650 if( submask->points.total > 0 ) {
1651 point = submask->points.values[submask->points.total-1];
1652 submask->points.remove_object(point);
1654 total_points = submask->points.total;
1655 current = current == (MaskAuto*)autos->default_auto ?
1656 (MaskAuto*)autos->first : (MaskAuto*)NEXT;
1659 if( mwindow->cwindow->gui->affected_point >= total_points )
1660 mwindow->cwindow->gui->affected_point =
1661 total_points > 0 ? total_points-1 : 0;
1664 gui->update_preview();
1665 mwindow->undo->update_undo_after(_("mask delete"), LOAD_AUTOMATION);
1671 int CWindowMaskDelPoint::keypress_event()
1673 if( get_keypress() == BACKSPACE ||
1674 get_keypress() == DELETE )
1675 return handle_event();
1682 CWindowMaskAffectedPoint::CWindowMaskAffectedPoint(MWindow *mwindow,
1683 CWindowToolGUI *gui, int x, int y)
1684 : BC_TumbleTextBox(gui,
1685 (int64_t)mwindow->cwindow->gui->affected_point,
1686 (int64_t)0, INT64_MAX, x, y, 100)
1688 this->mwindow = mwindow;
1692 CWindowMaskAffectedPoint::~CWindowMaskAffectedPoint()
1696 int CWindowMaskAffectedPoint::handle_event()
1698 int total_points = 0;
1699 int affected_point = atol(get_text());
1700 Track *track = mwindow->cwindow->calculate_affected_track();
1702 MaskAutos *autos = (MaskAutos*)track->automation->autos[AUTOMATION_MASK];
1703 MaskAuto *keyframe = (MaskAuto*)mwindow->cwindow->calculate_affected_auto(autos, 0);
1705 SubMask *mask = keyframe->get_submask(mwindow->edl->session->cwindow_mask);
1706 total_points = mask->points.size();
1709 int active_point = affected_point;
1710 if( affected_point >= total_points )
1711 affected_point = total_points - 1;
1712 else if( affected_point < 0 )
1714 if( active_point != affected_point )
1715 update((int64_t)affected_point);
1716 mwindow->cwindow->gui->affected_point = affected_point;
1718 gui->update_preview();
1723 CWindowMaskFocus::CWindowMaskFocus(MWindow *mwindow, CWindowToolGUI *gui, int x, int y)
1724 : BC_CheckBox(x, y, ((CWindowMaskGUI*)gui)->focused, _("Focus"))
1726 this->mwindow = mwindow;
1730 CWindowMaskFocus::~CWindowMaskFocus()
1734 int CWindowMaskFocus::handle_event()
1736 ((CWindowMaskGUI*)gui)->focused = get_value();
1738 gui->update_preview();
1742 CWindowMaskDrawMarkers::CWindowMaskDrawMarkers(MWindow *mwindow, CWindowToolGUI *gui, int x, int y)
1743 : BC_CheckBox(x, y, ((CWindowMaskGUI*)gui)->markers, _("Markers"))
1745 this->mwindow = mwindow;
1749 CWindowMaskDrawMarkers::~CWindowMaskDrawMarkers()
1753 int CWindowMaskDrawMarkers::handle_event()
1755 ((CWindowMaskGUI*)gui)->markers = get_value();
1757 gui->update_preview();
1761 CWindowMaskDrawBoundary::CWindowMaskDrawBoundary(MWindow *mwindow, CWindowToolGUI *gui, int x, int y)
1762 : BC_CheckBox(x, y, ((CWindowMaskGUI*)gui)->boundary, _("Boundary"))
1764 this->mwindow = mwindow;
1768 CWindowMaskDrawBoundary::~CWindowMaskDrawBoundary()
1772 int CWindowMaskDrawBoundary::handle_event()
1774 ((CWindowMaskGUI*)gui)->boundary = get_value();
1776 gui->update_preview();
1781 CWindowMaskFeather::CWindowMaskFeather(MWindow *mwindow, CWindowToolGUI *gui, int x, int y)
1782 : BC_TumbleTextBox(gui, 0, -FEATHER_MAX, FEATHER_MAX, x, y, 64, 2)
1784 this->mwindow = mwindow;
1787 CWindowMaskFeather::~CWindowMaskFeather()
1791 int CWindowMaskFeather::update(float v)
1793 CWindowMaskGUI *mask_gui = (CWindowMaskGUI*)gui;
1794 mask_gui->feather_slider->update(v);
1795 return BC_TumbleTextBox::update(v);
1798 int CWindowMaskFeather::update_value(float v)
1805 #ifdef USE_KEYFRAME_SPANNING
1811 mwindow->undo->update_undo_before(_("mask feather"), this);
1813 // Get existing keyframe
1814 ((CWindowMaskGUI*)gui)->get_keyframe(track, autos, keyframe,
1815 mask, point, create_it);
1817 int gang = ((CWindowMaskGUI*)gui)->gang_feather->get_value();
1818 #ifdef USE_KEYFRAME_SPANNING
1819 MaskAuto temp_keyframe(mwindow->edl, autos);
1820 temp_keyframe.copy_data(keyframe);
1821 keyframe = &temp_keyframe;
1823 float change = v - mask->feather;
1824 int k = mwindow->edl->session->cwindow_mask;
1825 int n = gang ? keyframe->masks.size() : k+1;
1826 for( int i=gang? 0 : k; i<n; ++i ) {
1827 SubMask *sub_mask = keyframe->get_submask(i);
1828 float feather = sub_mask->feather + change;
1829 bclamp(feather, -FEATHER_MAX, FEATHER_MAX);
1830 sub_mask->feather = feather;
1832 #ifdef USE_KEYFRAME_SPANNING
1833 autos->update_parameter(keyframe);
1835 gui->update_preview();
1838 mwindow->undo->update_undo_after(_("mask feather"), LOAD_AUTOMATION);
1842 int CWindowMaskFeather::handle_event()
1844 float v = atof(get_text());
1845 CWindowMaskGUI * mask_gui = (CWindowMaskGUI*)gui;
1846 mask_gui->feather_slider->update(v);
1847 return mask_gui->feather->update_value(v);
1850 CWindowMaskFeatherSlider::CWindowMaskFeatherSlider(MWindow *mwindow,
1851 CWindowToolGUI *gui, int x, int y, int w, float v)
1852 : BC_FSlider(x, y, 0, w, w, -FEATHER_MAX, FEATHER_MAX, v)
1854 this->mwindow = mwindow;
1856 set_precision(0.01);
1857 timer = new Timer();
1862 CWindowMaskFeatherSlider::~CWindowMaskFeatherSlider()
1867 int CWindowMaskFeatherSlider::handle_event()
1869 float v = get_value();
1871 int64_t ms = timer->get_difference();
1872 if( ms < 250 && --stick > 0 ) {
1873 if( get_value() == 0 ) return 1;
1881 else if( (last_v>=0 && v<0) || (last_v<0 && v>=0) ) {
1888 CWindowMaskGUI * mask_gui = (CWindowMaskGUI*)gui;
1889 mask_gui->feather->BC_TumbleTextBox::update(v);
1890 return mask_gui->feather->update_value(v);
1893 int CWindowMaskFeatherSlider::update(float v)
1895 return BC_FSlider::update(v);
1898 CWindowMaskFade::CWindowMaskFade(MWindow *mwindow, CWindowToolGUI *gui, int x, int y)
1899 : BC_TumbleTextBox(gui, 0, -100.f, 100.f, x, y, 64, 2)
1901 this->mwindow = mwindow;
1904 CWindowMaskFade::~CWindowMaskFade()
1908 int CWindowMaskFade::update(float v)
1910 CWindowMaskGUI *mask_gui = (CWindowMaskGUI*)gui;
1911 mask_gui->fade_slider->update(v);
1912 return BC_TumbleTextBox::update(v);
1915 int CWindowMaskFade::update_value(float v)
1922 #ifdef USE_KEYFRAME_SPANNING
1928 mwindow->undo->update_undo_before(_("mask fade"), this);
1930 // Get existing keyframe
1931 ((CWindowMaskGUI*)gui)->get_keyframe(track, autos, keyframe,
1932 mask, point, create_it);
1934 int gang = ((CWindowMaskGUI*)gui)->gang_fader->get_value();
1935 #ifdef USE_KEYFRAME_SPANNING
1936 MaskAuto temp_keyframe(mwindow->edl, autos);
1937 temp_keyframe.copy_data(keyframe);
1938 keyframe = &temp_keyframe;
1940 float change = v - mask->fader;
1941 int k = mwindow->edl->session->cwindow_mask;
1942 int n = gang ? keyframe->masks.size() : k+1;
1943 for( int i=gang? 0 : k; i<n; ++i ) {
1944 SubMask *sub_mask = keyframe->get_submask(i);
1945 float fader = sub_mask->fader + change;
1946 bclamp(fader, -100.f, 100.f);
1947 sub_mask->fader = fader;
1949 #ifdef USE_KEYFRAME_SPANNING
1950 autos->update_parameter(keyframe);
1952 gui->update_preview();
1955 mwindow->undo->update_undo_after(_("mask fade"), LOAD_AUTOMATION);
1959 int CWindowMaskFade::handle_event()
1961 float v = atof(get_text());
1962 CWindowMaskGUI * mask_gui = (CWindowMaskGUI*)gui;
1963 mask_gui->fade_slider->update(v);
1964 return mask_gui->fade->update_value(v);
1967 CWindowMaskFadeSlider::CWindowMaskFadeSlider(MWindow *mwindow, CWindowToolGUI *gui,
1968 int x, int y, int w)
1969 : BC_ISlider(x, y, 0, w, w, -200, 200, 0)
1971 this->mwindow = mwindow;
1973 timer = new Timer();
1978 CWindowMaskFadeSlider::~CWindowMaskFadeSlider()
1983 int CWindowMaskFadeSlider::handle_event()
1985 float v = 100*get_value()/200;
1987 int64_t ms = timer->get_difference();
1988 if( ms < 250 && --stick > 0 ) {
1989 if( get_value() == 0 ) return 1;
1997 else if( (last_v>=0 && v<0) || (last_v<0 && v>=0) ) {
2004 CWindowMaskGUI *mask_gui = (CWindowMaskGUI*)gui;
2005 mask_gui->fade->BC_TumbleTextBox::update(v);
2006 return mask_gui->fade->update_value(v);
2009 int CWindowMaskFadeSlider::update(int64_t v)
2011 return BC_ISlider::update(200*v/100);
2014 CWindowMaskGangFader::CWindowMaskGangFader(MWindow *mwindow,
2015 CWindowToolGUI *gui, int x, int y)
2016 : BC_Toggle(x, y, mwindow->theme->get_image_set("gangpatch_data"), 0)
2018 this->mwindow = mwindow;
2020 set_tooltip(_("Gang fader"));
2023 CWindowMaskGangFader::~CWindowMaskGangFader()
2027 int CWindowMaskGangFader::handle_event()
2032 CWindowMaskBeforePlugins::CWindowMaskBeforePlugins(CWindowToolGUI *gui, int x, int y)
2036 _("Apply mask before plugins"))
2041 int CWindowMaskBeforePlugins::handle_event()
2048 ((CWindowMaskGUI*)gui)->get_keyframe(track, autos, keyframe, mask, point, 1);
2051 int v = get_value();
2052 #ifdef USE_KEYFRAME_SPANNING
2053 MaskAuto temp_keyframe(gui->mwindow->edl, autos);
2054 temp_keyframe.copy_data(keyframe);
2055 temp_keyframe.apply_before_plugins = v;
2056 autos->update_parameter(&temp_keyframe);
2058 keyframe->apply_before_plugins = v;
2060 gui->update_preview();
2066 CWindowDisableOpenGLMasking::CWindowDisableOpenGLMasking(CWindowToolGUI *gui, int x, int y)
2067 : BC_CheckBox(x, y, 1, _("Disable OpenGL masking"))
2072 int CWindowDisableOpenGLMasking::handle_event()
2079 ((CWindowMaskGUI*)gui)->get_keyframe(track, autos, keyframe, mask, point, 1);
2082 int v = get_value();
2083 #ifdef USE_KEYFRAME_SPANNING
2084 MaskAuto temp_keyframe(gui->mwindow->edl, autos);
2085 temp_keyframe.copy_data(keyframe);
2086 temp_keyframe.disable_opengl_masking = v;
2087 autos->update_parameter(&temp_keyframe);
2089 keyframe->disable_opengl_masking = v;
2091 gui->update_preview();
2097 CWindowMaskClrMask::CWindowMaskClrMask(MWindow *mwindow,
2098 CWindowMaskGUI *gui, int x, int y)
2099 : BC_Button(x, y, mwindow->theme->get_image_set("reset_button"))
2101 this->mwindow = mwindow;
2103 set_tooltip(_("Delete all masks"));
2106 CWindowMaskClrMask::~CWindowMaskClrMask()
2110 int CWindowMaskClrMask::calculate_w(MWindow *mwindow)
2112 VFrame *vfrm = *mwindow->theme->get_image_set("reset_button");
2113 return vfrm->get_w();
2116 int CWindowMaskClrMask::handle_event()
2124 // Get existing keyframe
2125 ((CWindowMaskGUI*)gui)->get_keyframe(track, autos, keyframe, mask, point, 0);
2128 mwindow->undo->update_undo_before(_("del masks"), 0);
2129 ((MaskAutos*)track->automation->autos[AUTOMATION_MASK])->clear_all();
2131 gui->update_preview();
2132 mwindow->undo->update_undo_after(_("del masks"), LOAD_AUTOMATION);
2138 CWindowMaskGangFeather::CWindowMaskGangFeather(MWindow *mwindow,
2139 CWindowToolGUI *gui, int x, int y)
2140 : BC_Toggle(x, y, mwindow->theme->get_image_set("gangpatch_data"), 0)
2142 this->mwindow = mwindow;
2144 set_tooltip(_("Gang feather"));
2147 CWindowMaskGangFeather::~CWindowMaskGangFeather()
2151 int CWindowMaskGangFeather::handle_event()
2156 CWindowMaskGUI::CWindowMaskGUI(MWindow *mwindow, CWindowTool *thread)
2157 : CWindowToolGUI(mwindow, thread,
2158 _(PROGRAM_NAME ": Mask"), 360, 440)
2160 this->mwindow = mwindow;
2161 this->thread = thread;
2169 CWindowMaskGUI::~CWindowMaskGUI()
2171 lock_window("CWindowMaskGUI::~CWindowMaskGUI");
2172 delete active_point;
2178 void CWindowMaskGUI::create_objects()
2180 int x = 10, y = 10, margin = mwindow->theme->widget_border;
2181 int clr_x = get_w()-x - CWindowMaskClrMask::calculate_w(mwindow);
2182 int del_x = clr_x-margin - CWindowMaskDelMask::calculate_w(this,_("Delete"));
2183 //MaskAuto *keyframe = 0;
2184 //Track *track = mwindow->cwindow->calculate_affected_track();
2186 // keyframe = (MaskAuto*)mwindow->cwindow->calculate_affected_auto(track->automation->autos[AUTOMATION_MASK], 0);
2188 lock_window("CWindowMaskGUI::create_objects");
2190 add_subwindow(title = new BC_Title(x, y, _("Mask:")));
2192 name = new CWindowMaskName(mwindow, this, x1, y, "");
2193 name->create_objects();
2194 add_subwindow(clr_mask = new CWindowMaskClrMask(mwindow, this, clr_x, y));
2195 add_subwindow(del_mask = new CWindowMaskDelMask(mwindow, this, del_x, y));
2196 y += name->get_h() + margin;
2197 add_subwindow(title = new BC_Title(x, y, _("Fade:")));
2198 fade = new CWindowMaskFade(mwindow, this, x1, y);
2199 fade->create_objects();
2200 int x2 = x1 + fade->get_w() + 2*margin;
2201 int w2 = clr_x-2*margin - x2;
2202 add_subwindow(fade_slider = new CWindowMaskFadeSlider(mwindow, this, x2, y, w2));
2203 add_subwindow(gang_fader = new CWindowMaskGangFader(mwindow, this, clr_x, y));
2204 y += fade->get_h() + margin;
2205 add_subwindow(title = new BC_Title(x, y, _("Feather:")));
2206 feather = new CWindowMaskFeather(mwindow, this, x1, y);
2207 feather->create_objects();
2208 w2 = clr_x - 2*margin - x2;
2209 feather_slider = new CWindowMaskFeatherSlider(mwindow, this, x2, y, w2, 0);
2210 add_subwindow(feather_slider);
2211 add_subwindow(gang_feather = new CWindowMaskGangFeather(mwindow, this, clr_x, y));
2212 y += feather->get_h() + 3*margin;
2214 add_subwindow(title = new BC_Title(x, y, _("Point:")));
2215 active_point = new CWindowMaskAffectedPoint(mwindow, this, x1, y);
2216 active_point->create_objects();
2217 add_subwindow(del_point = new CWindowMaskDelPoint(mwindow, this, del_x, y));
2218 y += active_point->get_h() + margin;
2219 add_subwindow(title = new BC_Title(x, y, "X:"));
2220 this->x = new CWindowCoord(this, x1, y, (float)0.0);
2221 this->x->create_objects();
2222 add_subwindow(draw_markers = new CWindowMaskDrawMarkers(mwindow, this, del_x, y));
2223 y += this->x->get_h() + margin;
2224 add_subwindow(title = new BC_Title(x, y, "Y:"));
2225 this->y = new CWindowCoord(this, x1, y, (float)0.0);
2226 this->y->create_objects();
2227 add_subwindow(draw_boundary = new CWindowMaskDrawBoundary(mwindow, this, del_x, y));
2228 y += this->y->get_h() + margin;
2230 add_subwindow(bar = new BC_Bar(x, y, get_w()-2*x));
2231 y += bar->get_h() + margin;
2233 add_subwindow(title = new BC_Title(x, y, "X:"));
2234 focus_x = new CWindowCoord(this, x1, y, (float)0.0);
2235 focus_x->create_objects();
2236 add_subwindow(focus = new CWindowMaskFocus(mwindow, this, del_x, y));
2237 y += focus_x->get_h() + margin;
2238 add_subwindow(title = new BC_Title(x, y, "Y:"));
2239 focus_y = new CWindowCoord(this, x1, y, (float)0.0);
2240 focus_y->create_objects();
2241 y += focus_x->get_h() + margin;
2242 add_subwindow(this->apply_before_plugins = new CWindowMaskBeforePlugins(this, 10, y));
2243 y += this->apply_before_plugins->get_h();
2244 add_subwindow(this->disable_opengl_masking = new CWindowDisableOpenGLMasking(this, 10, y));
2245 y += this->disable_opengl_masking->get_h() + margin;
2246 add_subwindow(bar = new BC_Bar(x, y, get_w()-2*x));
2247 y += bar->get_h() + margin;
2250 add_subwindow(title = new BC_Title(x, y, _(
2251 "Shift+LMB: move an end point\n"
2252 "Ctrl+LMB: move a control point\n"
2253 "Alt+LMB: to drag translate the mask\n"
2254 "Shift+Key Delete to delete the mask\n"
2255 "Wheel Up/Dn: rotate around pointer\n"
2256 "Shift+Wheel Up/Dn: scale around pointer\n"
2257 "Shift+MMB: Toggle focus center at pointer")));
2262 void CWindowMaskGUI::get_keyframe(Track* &track,
2264 MaskAuto* &keyframe,
2272 track = mwindow->cwindow->calculate_affected_track();
2275 autos = (MaskAutos*)track->automation->autos[AUTOMATION_MASK];
2276 keyframe = (MaskAuto*)mwindow->cwindow->calculate_affected_auto(
2282 mask = keyframe->get_submask(mwindow->edl->session->cwindow_mask);
2289 if(mwindow->cwindow->gui->affected_point < mask->points.total &&
2290 mwindow->cwindow->gui->affected_point >= 0)
2292 point = mask->points.values[mwindow->cwindow->gui->affected_point];
2297 void CWindowMaskGUI::update()
2304 //printf("CWindowMaskGUI::update 1\n");
2305 get_keyframe(track, autos, keyframe, mask, point, 0);
2307 double position = mwindow->edl->local_session->get_selectionstart(1);
2308 position = mwindow->edl->align_to_frame(position, 0);
2311 int64_t position_i = track->to_units(position, 0);
2314 x->update(point->x);
2315 y->update(point->y);
2319 int k = mwindow->edl->session->cwindow_mask;
2320 feather->update(autos->get_feather(position_i, k, PLAY_FORWARD));
2321 fade->update(autos->get_fader(position_i, k, PLAY_FORWARD));
2322 apply_before_plugins->update(keyframe->apply_before_plugins);
2323 disable_opengl_masking->update(keyframe->disable_opengl_masking);
2327 active_point->update((int64_t)mwindow->cwindow->gui->affected_point);
2328 const char *text = "";
2330 name->update_items(keyframe);
2331 int k = mwindow->edl->session->cwindow_mask;
2332 if( k >= 0 && k < keyframe->masks.size() )
2333 text = keyframe->masks[k]->name;
2338 void CWindowMaskGUI::handle_event()
2345 get_keyframe(track, autos, keyframe, mask, point, 0);
2347 mwindow->undo->update_undo_before(_("mask point"), this);
2351 #ifdef USE_KEYFRAME_SPANNING
2352 // Create temp keyframe
2353 MaskAuto temp_keyframe(mwindow->edl, autos);
2354 temp_keyframe.copy_data(keyframe);
2355 // Get affected point in temp keyframe
2356 mask = temp_keyframe.get_submask(mwindow->edl->session->cwindow_mask);
2357 if(mwindow->cwindow->gui->affected_point < mask->points.total &&
2358 mwindow->cwindow->gui->affected_point >= 0)
2360 point = mask->points.values[mwindow->cwindow->gui->affected_point];
2365 point->x = atof(x->get_text());
2366 point->y = atof(y->get_text());
2367 // Commit to spanned keyframes
2368 autos->update_parameter(&temp_keyframe);
2371 point->x = atof(x->get_text());
2372 point->y = atof(y->get_text());
2377 mwindow->undo->update_undo_after(_("mask point"), LOAD_AUTOMATION);
2380 void CWindowMaskGUI::update_preview()
2383 CWindowGUI *cgui = mwindow->cwindow->gui;
2384 cgui->lock_window("CWindowMaskGUI::update_preview");
2385 cgui->sync_parameters(CHANGE_PARAMS, 0, 1);
2386 cgui->unlock_window();
2387 lock_window("CWindowMaskGUI::update_preview");
2390 void CWindowMaskGUI::set_focused(int v, float cx, float cy)
2392 focus_x->update(cx);
2393 focus_y->update(cy);
2394 focus->update(focused = v);
2397 CWindowRulerGUI::CWindowRulerGUI(MWindow *mwindow, CWindowTool *thread)
2398 : CWindowToolGUI(mwindow,
2400 _(PROGRAM_NAME ": Ruler"),
2406 CWindowRulerGUI::~CWindowRulerGUI()
2410 void CWindowRulerGUI::create_objects()
2412 int x = 10, y = 10, x1 = 100;
2415 lock_window("CWindowRulerGUI::create_objects");
2416 add_subwindow(title = new BC_Title(x, y, _("Current:")));
2417 add_subwindow(current = new BC_TextBox(x1, y, 200, 1, ""));
2418 y += title->get_h() + 5;
2419 add_subwindow(title = new BC_Title(x, y, _("Point 1:")));
2420 add_subwindow(point1 = new BC_TextBox(x1, y, 200, 1, ""));
2421 y += title->get_h() + 5;
2422 add_subwindow(title = new BC_Title(x, y, _("Point 2:")));
2423 add_subwindow(point2 = new BC_TextBox(x1, y, 200, 1, ""));
2424 y += title->get_h() + 5;
2425 add_subwindow(title = new BC_Title(x, y, _("Deltas:")));
2426 add_subwindow(deltas = new BC_TextBox(x1, y, 200, 1, ""));
2427 y += title->get_h() + 5;
2428 add_subwindow(title = new BC_Title(x, y, _("Distance:")));
2429 add_subwindow(distance = new BC_TextBox(x1, y, 200, 1, ""));
2430 y += title->get_h() + 5;
2431 add_subwindow(title = new BC_Title(x, y, _("Angle:")));
2432 add_subwindow(angle = new BC_TextBox(x1, y, 200, 1, ""));
2433 y += title->get_h() + 10;
2434 char string[BCTEXTLEN];
2436 _("Press Ctrl to lock ruler to the\nnearest 45%c%c angle."),
2437 0xc2, 0xb0); // degrees utf
2438 add_subwindow(title = new BC_Title(x,
2441 y += title->get_h() + 10;
2442 sprintf(string, _("Press Alt to translate the ruler."));
2443 add_subwindow(title = new BC_Title(x,
2450 void CWindowRulerGUI::update()
2452 char string[BCTEXTLEN];
2453 int cx = mwindow->session->cwindow_output_x;
2454 int cy = mwindow->session->cwindow_output_y;
2455 sprintf(string, "%d, %d", cx, cy);
2456 current->update(string);
2457 double x1 = mwindow->edl->session->ruler_x1;
2458 double y1 = mwindow->edl->session->ruler_y1;
2459 sprintf(string, "%.0f, %.0f", x1, y1);
2460 point1->update(string);
2461 double x2 = mwindow->edl->session->ruler_x2;
2462 double y2 = mwindow->edl->session->ruler_y2;
2463 sprintf(string, "%.0f, %.0f", x2, y2);
2464 point2->update(string);
2465 double dx = x2 - x1, dy = y2 - y1;
2466 sprintf(string, "%s%.0f, %s%.0f", (dx>=0? "+":""), dx, (dy>=0? "+":""), dy);
2467 deltas->update(string);
2468 double d = sqrt(dx*dx + dy*dy);
2469 sprintf(string, _("%0.01f pixels"), d);
2470 distance->update(string);
2471 double a = d > 0 ? (atan2(-dy, dx) * 180/M_PI) : 0.;
2472 sprintf(string, "%0.02f %c%c", a, 0xc2, 0xb0);
2473 angle->update(string);
2476 void CWindowRulerGUI::handle_event()