repeat play, in/out <> shortcuts, append to proj wording, cleanup
[goodguy/history.git] / cinelerra-5.1 / cinelerra / appearanceprefs.C
1
2 /*
3  * CINELERRA
4  * Copyright (C) 2008 Adam Williams <broadcast at earthling dot net>
5  *
6  * This program is free software; you can redistribute it and/or modify
7  * it under the terms of the GNU General Public License as published by
8  * the Free Software Foundation; either version 2 of the License, or
9  * (at your option) any later version.
10  *
11  * This program is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14  * GNU General Public License for more details.
15  *
16  * You should have received a copy of the GNU General Public License
17  * along with this program; if not, write to the Free Software
18  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
19  *
20  */
21
22 #include "appearanceprefs.h"
23 #include "deleteallindexes.h"
24 #include "edl.h"
25 #include "edlsession.h"
26 #include "file.h"
27 #include "filesystem.h"
28 #include "language.h"
29 #include "mwindow.h"
30 #include "preferences.h"
31 #include "preferencesthread.h"
32 #include "shbtnprefs.h"
33 #include "theme.h"
34
35
36 AppearancePrefs::AppearancePrefs(MWindow *mwindow, PreferencesWindow *pwindow)
37  : PreferencesDialog(mwindow, pwindow)
38 {
39         hms = 0;
40         hmsf = 0;
41         samples = 0;
42         frames = 0;
43         hex = 0;
44         feet = 0;
45         thumbnails = 0;
46 }
47
48 AppearancePrefs::~AppearancePrefs()
49 {
50         delete hms;
51         delete hmsf;
52         delete samples;
53         delete frames;
54         delete hex;
55         delete feet;
56         delete thumbnails;
57 }
58
59
60 void AppearancePrefs::create_objects()
61 {
62         BC_Resources *resources = BC_WindowBase::get_resources();
63         int margin = mwindow->theme->widget_border;
64         char string[BCTEXTLEN];
65         int x0 = mwindow->theme->preferencesoptions_x;
66         int y0 = mwindow->theme->preferencesoptions_y;
67         int x = x0, y = y0, x1 = x + 100;
68
69         add_subwindow(new BC_Title(x, y, _("Layout:"), LARGEFONT,
70                 resources->text_default));
71         y += 35;
72
73         ViewTheme *theme;
74         add_subwindow(new BC_Title(x, y, _("Theme:")));
75         add_subwindow(theme = new ViewTheme(x1, y, pwindow));
76         theme->create_objects();
77         y += theme->get_h() + 5;
78
79         x = x0;
80         ViewPluginIcons *plugin_icons;
81         add_subwindow(new BC_Title(x, y, _("Plugin Icons:")));
82         add_subwindow(plugin_icons = new ViewPluginIcons(x1, y, pwindow));
83         plugin_icons->create_objects();
84         y += plugin_icons->get_h() + 5;
85
86         y += 10;
87         add_subwindow(new BC_Bar(5, y,  get_w() - 10));
88         y += 15;
89
90         add_subwindow(new BC_Title(x, y, _("Time Format:"), LARGEFONT,
91                 resources->text_default));
92
93         x1 = get_w()/2;
94         add_subwindow(new BC_Title(x1, y, _("Flags:"), LARGEFONT,
95                 resources->text_default));
96
97         y += get_text_height(LARGEFONT) + 5;
98         y += 10;
99         int y1 = y;
100
101         add_subwindow(hms = new TimeFormatHMS(pwindow, this,
102                 pwindow->thread->edl->session->time_format == TIME_HMS,
103                 x, y));
104         y += 20;
105         add_subwindow(hmsf = new TimeFormatHMSF(pwindow, this,
106                 pwindow->thread->edl->session->time_format == TIME_HMSF,
107                 x, y));
108         y += 20;
109         add_subwindow(samples = new TimeFormatSamples(pwindow, this,
110                 pwindow->thread->edl->session->time_format == TIME_SAMPLES,
111                 x, y));
112         y += 20;
113         add_subwindow(hex = new TimeFormatHex(pwindow, this,
114                 pwindow->thread->edl->session->time_format == TIME_SAMPLES_HEX,
115                 x, y));
116         y += 20;
117         add_subwindow(frames = new TimeFormatFrames(pwindow, this,
118                 pwindow->thread->edl->session->time_format == TIME_FRAMES,
119                 x, y));
120         y += 20;
121         add_subwindow(feet = new TimeFormatFeet(pwindow, this,
122                 pwindow->thread->edl->session->time_format == TIME_FEET_FRAMES,
123                 x, y));
124         x += feet->get_w() + 15;
125         BC_Title *title;
126         add_subwindow(title = new BC_Title(x, y, _("Frames per foot:")));
127         x += title->get_w() + margin;
128         sprintf(string, "%0.2f", pwindow->thread->edl->session->frames_per_foot);
129         add_subwindow(new TimeFormatFeetSetting(pwindow,
130                 x, y - 5,       string));
131         x = x0;
132         y += 20;
133         add_subwindow(seconds = new TimeFormatSeconds(pwindow, this,
134                 pwindow->thread->edl->session->time_format == TIME_SECONDS,
135                 x, y));
136         x = x0;
137         y += 35;
138         add_subwindow(new BC_Bar(5, y,  get_w()/2 - 30));
139         y += 15;
140
141         add_subwindow(new BC_Title(x, y, _("Color:"), LARGEFONT,
142                 resources->text_default));
143         y += 35;
144         add_subwindow(title = new BC_Title(x, y, _("Highlighting Inversion color:")));
145         x += title->get_w() + margin;
146         char hex_color[BCSTRLEN];
147         sprintf(hex_color, "%06x", preferences->highlight_inverse);
148         add_subwindow(new HighlightInverseColor(pwindow, x, y, hex_color));
149         y += 35;
150
151         x = x0;
152         add_subwindow(title = new BC_Title(x, y, _("YUV color space:")));
153         x += title->get_w() + margin;
154         add_subwindow(yuv_color_space = new YuvColorSpace(x, y, pwindow));
155         yuv_color_space->create_objects();
156         y += yuv_color_space->get_h() + 5;
157
158         x = x0;
159         add_subwindow(title = new BC_Title(x, y, _("YUV color range:")));
160         x += title->get_w() + margin;
161         add_subwindow(yuv_color_range = new YuvColorRange(x, y, pwindow));
162         yuv_color_range->create_objects();
163         y += yuv_color_range->get_h() + 5;
164
165         UseTipWindow *tip_win = new UseTipWindow(pwindow, x1, y1);
166         add_subwindow(tip_win);
167         y1 += tip_win->get_h() + 5;
168         UseWarnIndecies *idx_win = new UseWarnIndecies(pwindow, x1, y1);
169         add_subwindow(idx_win);
170         y1 += idx_win->get_h() + 5;
171         UseWarnVersion *ver_win = new UseWarnVersion(pwindow, x1, y1);
172         add_subwindow(ver_win);
173         y1 += ver_win->get_h() + 5;
174         BD_WarnRoot *bdwr_win = new BD_WarnRoot(pwindow, x1, y1);
175         add_subwindow(bdwr_win);
176         y1 += bdwr_win->get_h() + 5;
177         PopupMenuBtnup *pop_win = new PopupMenuBtnup(pwindow, x1, y1);
178         add_subwindow(pop_win);
179         y1 += pop_win->get_h() + 5;
180         GrabFocusPolicy *grab_input_focus = new GrabFocusPolicy(pwindow, x1, y1);
181         add_subwindow(grab_input_focus);
182         y1 += grab_input_focus->get_h() + 5;
183         ActivateFocusPolicy *focus_activate = new ActivateFocusPolicy(pwindow, x1, y1);
184         add_subwindow(focus_activate);
185         y1 += focus_activate->get_h() + 5;
186         DeactivateFocusPolicy *focus_deactivate = new DeactivateFocusPolicy(pwindow, x1, y1);
187         add_subwindow(focus_deactivate);
188         y1 += focus_deactivate->get_h() + 5;
189         ForwardRenderDisplacement *displacement = new ForwardRenderDisplacement(pwindow, x1, y1);
190         add_subwindow(displacement);
191         y1 += displacement->get_h() + 5;
192         add_subwindow(thumbnails = new ViewThumbnails(x1, y1, pwindow));
193         if( y < y1 ) y = y1;
194 }
195
196 int AppearancePrefs::update(int new_value)
197 {
198         pwindow->thread->redraw_times = 1;
199         pwindow->thread->edl->session->time_format = new_value;
200         hms->update(new_value == TIME_HMS);
201         hmsf->update(new_value == TIME_HMSF);
202         samples->update(new_value == TIME_SAMPLES);
203         hex->update(new_value == TIME_SAMPLES_HEX);
204         frames->update(new_value == TIME_FRAMES);
205         feet->update(new_value == TIME_FEET_FRAMES);
206         seconds->update(new_value == TIME_SECONDS);
207         return 0;
208 }
209
210
211 TimeFormatHMS::TimeFormatHMS(PreferencesWindow *pwindow, AppearancePrefs *tfwindow, int value, int x, int y)
212  : BC_Radial(x, y, value, TIME_HMS_TEXT)
213 { this->pwindow = pwindow; this->tfwindow = tfwindow; }
214
215 int TimeFormatHMS::handle_event()
216 {
217         tfwindow->update(TIME_HMS);
218         return 1;
219 }
220
221 TimeFormatHMSF::TimeFormatHMSF(PreferencesWindow *pwindow, AppearancePrefs *tfwindow, int value, int x, int y)
222  : BC_Radial(x, y, value, TIME_HMSF_TEXT)
223 { this->pwindow = pwindow; this->tfwindow = tfwindow; }
224
225 int TimeFormatHMSF::handle_event()
226 {
227         tfwindow->update(TIME_HMSF);
228         return 1;
229 }
230
231 TimeFormatSamples::TimeFormatSamples(PreferencesWindow *pwindow, AppearancePrefs *tfwindow, int value, int x, int y)
232  : BC_Radial(x, y, value, TIME_SAMPLES_TEXT)
233 { this->pwindow = pwindow; this->tfwindow = tfwindow; }
234
235 int TimeFormatSamples::handle_event()
236 {
237         tfwindow->update(TIME_SAMPLES);
238         return 1;
239 }
240
241 TimeFormatFrames::TimeFormatFrames(PreferencesWindow *pwindow, AppearancePrefs *tfwindow, int value, int x, int y)
242  : BC_Radial(x, y, value, TIME_FRAMES_TEXT)
243 { this->pwindow = pwindow; this->tfwindow = tfwindow; }
244
245 int TimeFormatFrames::handle_event()
246 {
247         tfwindow->update(TIME_FRAMES);
248         return 1;
249 }
250
251 TimeFormatHex::TimeFormatHex(PreferencesWindow *pwindow, AppearancePrefs *tfwindow, int value, int x, int y)
252  : BC_Radial(x, y, value, TIME_SAMPLES_HEX_TEXT)
253 { this->pwindow = pwindow; this->tfwindow = tfwindow; }
254
255 int TimeFormatHex::handle_event()
256 {
257         tfwindow->update(TIME_SAMPLES_HEX);
258         return 1;
259 }
260
261 TimeFormatSeconds::TimeFormatSeconds(PreferencesWindow *pwindow, AppearancePrefs *tfwindow, int value, int x, int y)
262  : BC_Radial(x, y, value, TIME_SECONDS_TEXT)
263 {
264         this->pwindow = pwindow;
265         this->tfwindow = tfwindow;
266 }
267
268 int TimeFormatSeconds::handle_event()
269 {
270         tfwindow->update(TIME_SECONDS);
271         return 1;
272 }
273
274 TimeFormatFeet::TimeFormatFeet(PreferencesWindow *pwindow, AppearancePrefs *tfwindow, int value, int x, int y)
275  : BC_Radial(x, y, value, TIME_FEET_FRAMES_TEXT)
276 { this->pwindow = pwindow; this->tfwindow = tfwindow; }
277
278 int TimeFormatFeet::handle_event()
279 {
280         tfwindow->update(TIME_FEET_FRAMES);
281         return 1;
282 }
283
284 TimeFormatFeetSetting::TimeFormatFeetSetting(PreferencesWindow *pwindow, int x, int y, char *string)
285  : BC_TextBox(x, y, 90, 1, string)
286 { this->pwindow = pwindow; }
287
288 int TimeFormatFeetSetting::handle_event()
289 {
290         pwindow->thread->edl->session->frames_per_foot = atof(get_text());
291         if(pwindow->thread->edl->session->frames_per_foot < 1) pwindow->thread->edl->session->frames_per_foot = 1;
292         return 0;
293 }
294
295
296 ViewTheme::ViewTheme(int x, int y, PreferencesWindow *pwindow)
297  : BC_PopupMenu(x, y, 200, pwindow->thread->preferences->theme, 1)
298 {
299         this->pwindow = pwindow;
300 }
301 ViewTheme::~ViewTheme()
302 {
303 }
304
305 void ViewTheme::create_objects()
306 {
307         ArrayList<PluginServer*> themes;
308         MWindow::search_plugindb(0, 0, 0, 0, 1, themes);
309
310         for(int i = 0; i < themes.total; i++) {
311                 add_item(new ViewThemeItem(this, themes.values[i]->title));
312         }
313 }
314
315 int ViewTheme::handle_event()
316 {
317         return 1;
318 }
319
320 ViewThemeItem::ViewThemeItem(ViewTheme *popup, const char *text)
321  : BC_MenuItem(text)
322 {
323         this->popup = popup;
324 }
325
326 int ViewThemeItem::handle_event()
327 {
328         popup->set_text(get_text());
329         strcpy(popup->pwindow->thread->preferences->theme, get_text());
330         popup->handle_event();
331         return 1;
332 }
333
334
335 ViewPluginIcons::ViewPluginIcons(int x, int y, PreferencesWindow *pwindow)
336  : BC_PopupMenu(x, y, 200, pwindow->thread->preferences->plugin_icons, 1)
337 {
338         this->pwindow = pwindow;
339 }
340 ViewPluginIcons::~ViewPluginIcons()
341 {
342 }
343
344 void ViewPluginIcons::create_objects()
345 {
346         add_item(new ViewPluginIconItem(this, DEFAULT_PICON));
347         FileSystem fs;
348         const char *plugin_path = File::get_plugin_path();
349         char picon_path[BCTEXTLEN];
350         snprintf(picon_path,sizeof(picon_path)-1,"%s/picon", plugin_path);
351         if( fs.update(picon_path) ) return;
352         for( int i=0; i<fs.dir_list.total; ++i ) {
353                 char *fs_path = fs.dir_list[i]->path;
354                 if( !fs.is_dir(fs_path) ) continue;
355                 char *cp = strrchr(fs_path,'/');
356                 cp = !cp ? fs_path : cp+1;
357                 if( !strcmp(cp,DEFAULT_PICON) ) continue;
358                 add_item(new ViewPluginIconItem(this, cp));
359         }
360 }
361
362 int ViewPluginIcons::handle_event()
363 {
364         return 1;
365 }
366
367 ViewPluginIconItem::ViewPluginIconItem(ViewPluginIcons *popup, const char *text)
368  : BC_MenuItem(text)
369 {
370         this->popup = popup;
371 }
372
373 int ViewPluginIconItem::handle_event()
374 {
375         popup->set_text(get_text());
376         strcpy(popup->pwindow->thread->preferences->plugin_icons, get_text());
377         popup->handle_event();
378         return 1;
379 }
380
381
382 ViewThumbnails::ViewThumbnails(int x,
383         int y,
384         PreferencesWindow *pwindow)
385  : BC_CheckBox(x,
386         y,
387         pwindow->thread->preferences->use_thumbnails, _("Use thumbnails in resource window"))
388 {
389         this->pwindow = pwindow;
390 }
391
392 int ViewThumbnails::handle_event()
393 {
394         pwindow->thread->preferences->use_thumbnails = get_value();
395         return 1;
396 }
397
398
399
400 UseTipWindow::UseTipWindow(PreferencesWindow *pwindow, int x, int y)
401  : BC_CheckBox(x,
402         y,
403         pwindow->thread->preferences->use_tipwindow,
404         _("Show tip of the day"))
405 {
406         this->pwindow = pwindow;
407 }
408 int UseTipWindow::handle_event()
409 {
410         pwindow->thread->preferences->use_tipwindow = get_value();
411         return 1;
412 }
413
414
415 UseWarnIndecies::UseWarnIndecies(PreferencesWindow *pwindow, int x, int y)
416  : BC_CheckBox(x, y, pwindow->thread->preferences->warn_indexes,
417         _("ffmpeg probe warns rebuild indexes"))
418 {
419         this->pwindow = pwindow;
420 }
421
422 int UseWarnIndecies::handle_event()
423 {
424         pwindow->thread->preferences->warn_indexes = get_value();
425         return 1;
426 }
427
428 UseWarnVersion::UseWarnVersion(PreferencesWindow *pwindow, int x, int y)
429  : BC_CheckBox(x, y, pwindow->thread->preferences->warn_version,
430         _("EDL version warns if mismatched"))
431 {
432         this->pwindow = pwindow;
433 }
434
435 int UseWarnVersion::handle_event()
436 {
437         pwindow->thread->preferences->warn_version = get_value();
438         return 1;
439 }
440
441 BD_WarnRoot::BD_WarnRoot(PreferencesWindow *pwindow, int x, int y)
442  : BC_CheckBox(x, y, pwindow->thread->preferences->bd_warn_root,
443         _("Create Bluray warns if not root"))
444 {
445         this->pwindow = pwindow;
446 }
447
448 int BD_WarnRoot::handle_event()
449 {
450         pwindow->thread->preferences->bd_warn_root = get_value();
451         return 1;
452 }
453
454 PopupMenuBtnup::PopupMenuBtnup(PreferencesWindow *pwindow, int x, int y)
455  : BC_CheckBox(x, y, pwindow->thread->preferences->popupmenu_btnup,
456         _("Popups activate on button up"))
457 {
458         this->pwindow = pwindow;
459 }
460
461 int PopupMenuBtnup::handle_event()
462 {
463         pwindow->thread->preferences->popupmenu_btnup = get_value();
464         return 1;
465 }
466
467 GrabFocusPolicy::GrabFocusPolicy(PreferencesWindow *pwindow, int x, int y)
468  : BC_CheckBox(x, y, (pwindow->thread->preferences->grab_input_focus) != 0,
469         _("Set Input Focus when window entered"))
470 {
471         this->pwindow = pwindow;
472 }
473
474 int GrabFocusPolicy::handle_event()
475 {
476         pwindow->thread->preferences->grab_input_focus = get_value();
477         return 1;
478 }
479
480 ActivateFocusPolicy::ActivateFocusPolicy(PreferencesWindow *pwindow, int x, int y)
481  : BC_CheckBox(x, y, (pwindow->thread->preferences->textbox_focus_policy & CLICK_ACTIVATE) != 0,
482         _("Click to activate text focus"))
483 {
484         this->pwindow = pwindow;
485 }
486
487 int ActivateFocusPolicy::handle_event()
488 {
489         if( get_value() )
490                 pwindow->thread->preferences->textbox_focus_policy |= CLICK_ACTIVATE;
491         else
492                 pwindow->thread->preferences->textbox_focus_policy &= ~CLICK_ACTIVATE;
493         return 1;
494 }
495
496 DeactivateFocusPolicy::DeactivateFocusPolicy(PreferencesWindow *pwindow, int x, int y)
497  : BC_CheckBox(x, y, (pwindow->thread->preferences->textbox_focus_policy & CLICK_DEACTIVATE) != 0,
498         _("Click to deactivate text focus"))
499 {
500         this->pwindow = pwindow;
501 }
502
503 int DeactivateFocusPolicy::handle_event()
504 {
505         if( get_value() )
506                 pwindow->thread->preferences->textbox_focus_policy |= CLICK_DEACTIVATE;
507         else
508                 pwindow->thread->preferences->textbox_focus_policy &= ~CLICK_DEACTIVATE;
509         return 1;
510 }
511
512 ForwardRenderDisplacement::ForwardRenderDisplacement(PreferencesWindow *pwindow, int x, int y)
513  : BC_CheckBox(x, y, pwindow->thread->preferences->forward_render_displacement,
514         _("Always show next frame"))
515 {
516         this->pwindow = pwindow;
517 }
518
519 int ForwardRenderDisplacement::handle_event()
520 {
521         pwindow->thread->preferences->forward_render_displacement = get_value();
522         return 1;
523 }
524
525 HighlightInverseColor::HighlightInverseColor(PreferencesWindow *pwindow, int x, int y, const char *hex)
526  : BC_TextBox(x, y, 80, 1, hex)
527 {
528         this->pwindow = pwindow;
529 }
530
531 int HighlightInverseColor::handle_event()
532 {
533         int inverse_color = strtoul(get_text(),0,16);
534         if( (inverse_color &= 0xffffff) == 0 ) {
535                 inverse_color = 0xffffff;
536                 char string[BCSTRLEN];
537                 sprintf(string,"%06x", inverse_color);
538                 update(string);
539         }
540         pwindow->thread->preferences->highlight_inverse = inverse_color;
541         return 1;
542 }
543
544
545 const char *YuvColorSpace::color_space[] = {
546         N_("BT601"),
547         N_("BT709"),
548         N_("BT2020"),
549 };
550
551 YuvColorSpace::YuvColorSpace(int x, int y, PreferencesWindow *pwindow)
552  : BC_PopupMenu(x, y, 100,
553         _(color_space[pwindow->thread->preferences->yuv_color_space]), 1)
554 {
555         this->pwindow = pwindow;
556 }
557 YuvColorSpace::~YuvColorSpace()
558 {
559 }
560
561 void YuvColorSpace::create_objects()
562 {
563         for( int id=0,nid=sizeof(color_space)/sizeof(color_space[0]); id<nid; ++id )
564                 add_item(new YuvColorSpaceItem(this, _(color_space[id]), id));
565         handle_event();
566 }
567
568 int YuvColorSpace::handle_event()
569 {
570         set_text(color_space[pwindow->thread->preferences->yuv_color_space]);
571         return 1;
572 }
573
574 YuvColorSpaceItem::YuvColorSpaceItem(YuvColorSpace *popup, const char *text, int id)
575  : BC_MenuItem(text)
576 {
577         this->popup = popup;
578         this->id = id;
579 }
580
581 int YuvColorSpaceItem::handle_event()
582 {
583         popup->set_text(get_text());
584         popup->pwindow->thread->preferences->yuv_color_space = id;
585         return popup->handle_event();
586 }
587
588
589 const char *YuvColorRange::color_range[] = {
590         N_("JPEG"),
591         N_("MPEG"),
592 };
593
594 YuvColorRange::YuvColorRange(int x, int y, PreferencesWindow *pwindow)
595  : BC_PopupMenu(x, y, 100,
596         _(color_range[pwindow->thread->preferences->yuv_color_range]), 1)
597 {
598         this->pwindow = pwindow;
599 }
600 YuvColorRange::~YuvColorRange()
601 {
602 }
603
604 void YuvColorRange::create_objects()
605 {
606         for( int id=0,nid=sizeof(color_range)/sizeof(color_range[0]); id<nid; ++id )
607                 add_item(new YuvColorRangeItem(this, _(color_range[id]), id));
608         handle_event();
609 }
610
611 int YuvColorRange::handle_event()
612 {
613         set_text(color_range[pwindow->thread->preferences->yuv_color_range]);
614         return 1;
615 }
616
617 YuvColorRangeItem::YuvColorRangeItem(YuvColorRange *popup, const char *text, int id)
618  : BC_MenuItem(text)
619 {
620         this->popup = popup;
621         this->id = id;
622 }
623
624 int YuvColorRangeItem::handle_event()
625 {
626         popup->set_text(get_text());
627         popup->pwindow->thread->preferences->yuv_color_range = id;
628         return popup->handle_event();
629 }
630