Merge remote-tracking branch 'origin/master' into 2.0
[dcpomatic.git] / src / wx / dcp_panel.cc
1 /*
2     Copyright (C) 2012-2014 Carl Hetherington <cth@carlh.net>
3
4     This program is free software; you can redistribute it and/or modify
5     it under the terms of the GNU General Public License as published by
6     the Free Software Foundation; either version 2 of the License, or
7     (at your option) any later version.
8
9     This program is distributed in the hope that it will be useful,
10     but WITHOUT ANY WARRANTY; without even the implied warranty of
11     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12     GNU General Public License for more details.
13
14     You should have received a copy of the GNU General Public License
15     along with this program; if not, write to the Free Software
16     Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
17
18 */
19
20 #include "dcp_panel.h"
21 #include "wx_util.h"
22 #include "isdcf_metadata_dialog.h"
23 #include "lib/ratio.h"
24 #include "lib/scaler.h"
25 #include "lib/config.h"
26 #include "lib/dcp_content_type.h"
27 #include "lib/util.h"
28 #include "lib/film.h"
29 #include "lib/ffmpeg_content.h"
30 #include <wx/wx.h>
31 #include <wx/notebook.h>
32 #include <wx/gbsizer.h>
33 #include <wx/spinctrl.h>
34 #include <boost/lexical_cast.hpp>
35
36 using std::cout;
37 using std::list;
38 using std::string;
39 using std::vector;
40 using boost::lexical_cast;
41 using boost::shared_ptr;
42
43 DCPPanel::DCPPanel (wxNotebook* n, boost::shared_ptr<Film> f)
44         : _film (f)
45         , _generally_sensitive (true)
46 {
47         _panel = new wxPanel (n);
48         _sizer = new wxBoxSizer (wxVERTICAL);
49         _panel->SetSizer (_sizer);
50
51         wxGridBagSizer* grid = new wxGridBagSizer (DCPOMATIC_SIZER_X_GAP, DCPOMATIC_SIZER_Y_GAP);
52         _sizer->Add (grid, 0, wxEXPAND | wxALL, 8);
53
54         int r = 0;
55         
56         add_label_to_grid_bag_sizer (grid, _panel, _("Name"), true, wxGBPosition (r, 0));
57         _name = new wxTextCtrl (_panel, wxID_ANY);
58         grid->Add (_name, wxGBPosition(r, 1), wxDefaultSpan, wxEXPAND | wxLEFT | wxRIGHT);
59         ++r;
60         
61         add_label_to_grid_bag_sizer (grid, _panel, _("DCP Name"), true, wxGBPosition (r, 0));
62         _dcp_name = new wxStaticText (_panel, wxID_ANY, wxT (""), wxDefaultPosition, wxDefaultSize, wxST_ELLIPSIZE_END);
63         grid->Add (_dcp_name, wxGBPosition(r, 1), wxDefaultSpan, wxALIGN_CENTER_VERTICAL | wxEXPAND);
64         ++r;
65
66         int flags = wxALIGN_CENTER_VERTICAL;
67 #ifdef __WXOSX__
68         flags |= wxALIGN_RIGHT;
69 #endif  
70
71         _use_isdcf_name = new wxCheckBox (_panel, wxID_ANY, _("Use ISDCF name"));
72         grid->Add (_use_isdcf_name, wxGBPosition (r, 0), wxDefaultSpan, flags);
73         _edit_isdcf_button = new wxButton (_panel, wxID_ANY, _("Details..."));
74         grid->Add (_edit_isdcf_button, wxGBPosition (r, 1));
75         ++r;
76
77         add_label_to_grid_bag_sizer (grid, _panel, _("Content Type"), true, wxGBPosition (r, 0));
78         _dcp_content_type = new wxChoice (_panel, wxID_ANY);
79         grid->Add (_dcp_content_type, wxGBPosition (r, 1));
80         ++r;
81
82         _notebook = new wxNotebook (_panel, wxID_ANY);
83         _sizer->Add (_notebook, 1, wxEXPAND | wxTOP, 6);
84
85         _notebook->AddPage (make_video_panel (), _("Video"), false);
86         _notebook->AddPage (make_audio_panel (), _("Audio"), false);
87         
88         _signed = new wxCheckBox (_panel, wxID_ANY, _("Signed"));
89         grid->Add (_signed, wxGBPosition (r, 0), wxGBSpan (1, 2));
90         ++r;
91         
92         _encrypted = new wxCheckBox (_panel, wxID_ANY, _("Encrypted"));
93         grid->Add (_encrypted, wxGBPosition (r, 0), wxGBSpan (1, 2));
94         ++r;
95
96         add_label_to_grid_bag_sizer (grid, _panel, _("Standard"), true, wxGBPosition (r, 0));
97         _standard = new wxChoice (_panel, wxID_ANY);
98         grid->Add (_standard, wxGBPosition (r, 1), wxDefaultSpan, wxALIGN_CENTER_VERTICAL);
99         ++r;
100
101         _name->Bind             (wxEVT_COMMAND_TEXT_UPDATED,          boost::bind (&DCPPanel::name_changed, this));
102         _use_isdcf_name->Bind   (wxEVT_COMMAND_CHECKBOX_CLICKED,      boost::bind (&DCPPanel::use_isdcf_name_toggled, this));
103         _edit_isdcf_button->Bind(wxEVT_COMMAND_BUTTON_CLICKED,        boost::bind (&DCPPanel::edit_isdcf_button_clicked, this));
104         _dcp_content_type->Bind (wxEVT_COMMAND_CHOICE_SELECTED,       boost::bind (&DCPPanel::dcp_content_type_changed, this));
105         _signed->Bind           (wxEVT_COMMAND_CHECKBOX_CLICKED,      boost::bind (&DCPPanel::signed_toggled, this));
106         _encrypted->Bind        (wxEVT_COMMAND_CHECKBOX_CLICKED,      boost::bind (&DCPPanel::encrypted_toggled, this));
107         _standard->Bind         (wxEVT_COMMAND_CHOICE_SELECTED,       boost::bind (&DCPPanel::standard_changed, this));
108
109         vector<DCPContentType const *> const ct = DCPContentType::all ();
110         for (vector<DCPContentType const *>::const_iterator i = ct.begin(); i != ct.end(); ++i) {
111                 _dcp_content_type->Append (std_to_wx ((*i)->pretty_name ()));
112         }
113
114         _standard->Append (_("SMPTE"));
115         _standard->Append (_("Interop"));
116
117         Config::instance()->Changed.connect (boost::bind (&DCPPanel::config_changed, this));
118 }
119
120 void
121 DCPPanel::name_changed ()
122 {
123         if (!_film) {
124                 return;
125         }
126
127         _film->set_name (string (_name->GetValue().mb_str()));
128 }
129
130 void
131 DCPPanel::j2k_bandwidth_changed ()
132 {
133         if (!_film) {
134                 return;
135         }
136         
137         _film->set_j2k_bandwidth (_j2k_bandwidth->GetValue() * 1000000);
138 }
139
140 void
141 DCPPanel::signed_toggled ()
142 {
143         if (!_film) {
144                 return;
145         }
146
147         _film->set_signed (_signed->GetValue ());
148 }
149
150 void
151 DCPPanel::burn_subtitles_toggled ()
152 {
153         if (!_film) {
154                 return;
155         }
156
157         _film->set_burn_subtitles (_burn_subtitles->GetValue ());
158 }
159
160 void
161 DCPPanel::encrypted_toggled ()
162 {
163         if (!_film) {
164                 return;
165         }
166
167         _film->set_encrypted (_encrypted->GetValue ());
168 }
169                                
170 /** Called when the frame rate choice widget has been changed */
171 void
172 DCPPanel::frame_rate_choice_changed ()
173 {
174         if (!_film) {
175                 return;
176         }
177
178         _film->set_video_frame_rate (
179                 boost::lexical_cast<int> (
180                         wx_to_std (_frame_rate_choice->GetString (_frame_rate_choice->GetSelection ()))
181                         )
182                 );
183 }
184
185 /** Called when the frame rate spin widget has been changed */
186 void
187 DCPPanel::frame_rate_spin_changed ()
188 {
189         if (!_film) {
190                 return;
191         }
192
193         _film->set_video_frame_rate (_frame_rate_spin->GetValue ());
194 }
195
196 void
197 DCPPanel::audio_channels_changed ()
198 {
199         if (!_film) {
200                 return;
201         }
202
203         _film->set_audio_channels (_audio_channels->GetValue ());
204 }
205
206 void
207 DCPPanel::resolution_changed ()
208 {
209         if (!_film) {
210                 return;
211         }
212
213         _film->set_resolution (_resolution->GetSelection() == 0 ? RESOLUTION_2K : RESOLUTION_4K);
214 }
215
216 void
217 DCPPanel::standard_changed ()
218 {
219         if (!_film) {
220                 return;
221         }
222
223         _film->set_interop (_standard->GetSelection() == 1);
224 }
225
226 void
227 DCPPanel::film_changed (int p)
228 {
229         switch (p) {
230         case Film::NONE:
231                 break;
232         case Film::CONTAINER:
233                 setup_container ();
234                 break;
235         case Film::NAME:
236                 checked_set (_name, _film->name());
237                 setup_dcp_name ();
238                 break;
239         case Film::DCP_CONTENT_TYPE:
240                 checked_set (_dcp_content_type, DCPContentType::as_index (_film->dcp_content_type ()));
241                 setup_dcp_name ();
242                 break;
243         case Film::SCALER:
244                 checked_set (_scaler, Scaler::as_index (_film->scaler ()));
245                 break;
246         case Film::BURN_SUBTITLES:
247                 checked_set (_burn_subtitles, _film->burn_subtitles ());
248                 break;
249         case Film::SIGNED:
250                 checked_set (_signed, _film->is_signed ());
251                 break;
252         case Film::ENCRYPTED:
253                 checked_set (_encrypted, _film->encrypted ());
254                 if (_film->encrypted ()) {
255                         _film->set_signed (true);
256                         _signed->Enable (false);
257                 } else {
258                         _signed->Enable (_generally_sensitive);
259                 }
260                 break;
261         case Film::RESOLUTION:
262                 checked_set (_resolution, _film->resolution() == RESOLUTION_2K ? 0 : 1);
263                 setup_dcp_name ();
264                 break;
265         case Film::J2K_BANDWIDTH:
266                 checked_set (_j2k_bandwidth, _film->j2k_bandwidth() / 1000000);
267                 break;
268         case Film::USE_ISDCF_NAME:
269                 checked_set (_use_isdcf_name, _film->use_isdcf_name ());
270                 setup_dcp_name ();
271                 break;
272         case Film::ISDCF_METADATA:
273                 setup_dcp_name ();
274                 break;
275         case Film::VIDEO_FRAME_RATE:
276         {
277                 bool done = false;
278                 for (unsigned int i = 0; i < _frame_rate_choice->GetCount(); ++i) {
279                         if (wx_to_std (_frame_rate_choice->GetString(i)) == boost::lexical_cast<string> (_film->video_frame_rate())) {
280                                 checked_set (_frame_rate_choice, i);
281                                 done = true;
282                                 break;
283                         }
284                 }
285
286                 if (!done) {
287                         checked_set (_frame_rate_choice, -1);
288                 }
289
290                 _frame_rate_spin->SetValue (_film->video_frame_rate ());
291
292                 _best_frame_rate->Enable (_film->best_video_frame_rate () != _film->video_frame_rate ());
293                 break;
294         }
295         case Film::AUDIO_CHANNELS:
296                 checked_set (_audio_channels, _film->audio_channels ());
297                 setup_dcp_name ();
298                 break;
299         case Film::THREE_D:
300                 checked_set (_three_d, _film->three_d ());
301                 setup_dcp_name ();
302                 break;
303         case Film::INTEROP:
304                 checked_set (_standard, _film->interop() ? 1 : 0);
305                 break;
306         default:
307                 break;
308         }
309 }
310
311 void
312 DCPPanel::film_content_changed (int property)
313 {
314         if (property == FFmpegContentProperty::AUDIO_STREAM ||
315             property == SubtitleContentProperty::USE_SUBTITLES ||
316             property == VideoContentProperty::VIDEO_SCALE) {
317                 setup_dcp_name ();
318         }
319 }
320
321
322 void
323 DCPPanel::setup_container ()
324 {
325         int n = 0;
326         vector<Ratio const *> ratios = Ratio::all ();
327         vector<Ratio const *>::iterator i = ratios.begin ();
328         while (i != ratios.end() && *i != _film->container ()) {
329                 ++i;
330                 ++n;
331         }
332         
333         if (i == ratios.end()) {
334                 checked_set (_container, -1);
335         } else {
336                 checked_set (_container, n);
337         }
338         
339         setup_dcp_name ();
340 }       
341
342 /** Called when the container widget has been changed */
343 void
344 DCPPanel::container_changed ()
345 {
346         if (!_film) {
347                 return;
348         }
349
350         int const n = _container->GetSelection ();
351         if (n >= 0) {
352                 vector<Ratio const *> ratios = Ratio::all ();
353                 assert (n < int (ratios.size()));
354                 _film->set_container (ratios[n]);
355         }
356 }
357
358 /** Called when the DCP content type widget has been changed */
359 void
360 DCPPanel::dcp_content_type_changed ()
361 {
362         if (!_film) {
363                 return;
364         }
365
366         int const n = _dcp_content_type->GetSelection ();
367         if (n != wxNOT_FOUND) {
368                 _film->set_dcp_content_type (DCPContentType::from_index (n));
369         }
370 }
371
372 void
373 DCPPanel::set_film (shared_ptr<Film> film)
374 {
375         _film = film;
376         
377         film_changed (Film::NAME);
378         film_changed (Film::USE_ISDCF_NAME);
379         film_changed (Film::CONTENT);
380         film_changed (Film::DCP_CONTENT_TYPE);
381         film_changed (Film::CONTAINER);
382         film_changed (Film::RESOLUTION);
383         film_changed (Film::SCALER);
384         film_changed (Film::SIGNED);
385         film_changed (Film::BURN_SUBTITLES);
386         film_changed (Film::ENCRYPTED);
387         film_changed (Film::J2K_BANDWIDTH);
388         film_changed (Film::ISDCF_METADATA);
389         film_changed (Film::VIDEO_FRAME_RATE);
390         film_changed (Film::AUDIO_CHANNELS);
391         film_changed (Film::SEQUENCE_VIDEO);
392         film_changed (Film::THREE_D);
393         film_changed (Film::INTEROP);
394 }
395
396 void
397 DCPPanel::set_general_sensitivity (bool s)
398 {
399         _name->Enable (s);
400         _use_isdcf_name->Enable (s);
401         _edit_isdcf_button->Enable (s);
402         _dcp_content_type->Enable (s);
403
404         bool si = s;
405         if (_film && _film->encrypted ()) {
406                 si = false;
407         }
408         _burn_subtitles->Enable (s);
409         _signed->Enable (si);
410         
411         _encrypted->Enable (s);
412         _frame_rate_choice->Enable (s);
413         _frame_rate_spin->Enable (s);
414         _audio_channels->Enable (s);
415         _j2k_bandwidth->Enable (s);
416         _container->Enable (s);
417         _best_frame_rate->Enable (s && _film && _film->best_video_frame_rate () != _film->video_frame_rate ());
418         _resolution->Enable (s);
419         _scaler->Enable (s);
420         _three_d->Enable (s);
421         _standard->Enable (s);
422 }
423
424 /** Called when the scaler widget has been changed */
425 void
426 DCPPanel::scaler_changed ()
427 {
428         if (!_film) {
429                 return;
430         }
431
432         int const n = _scaler->GetSelection ();
433         if (n >= 0) {
434                 _film->set_scaler (Scaler::from_index (n));
435         }
436 }
437
438 void
439 DCPPanel::use_isdcf_name_toggled ()
440 {
441         if (!_film) {
442                 return;
443         }
444
445         _film->set_use_isdcf_name (_use_isdcf_name->GetValue ());
446 }
447
448 void
449 DCPPanel::edit_isdcf_button_clicked ()
450 {
451         if (!_film) {
452                 return;
453         }
454
455         ISDCFMetadataDialog* d = new ISDCFMetadataDialog (_panel, _film->isdcf_metadata ());
456         d->ShowModal ();
457         _film->set_isdcf_metadata (d->isdcf_metadata ());
458         d->Destroy ();
459 }
460
461 void
462 DCPPanel::setup_dcp_name ()
463 {
464         string s = _film->dcp_name (true);
465         if (s.length() > 28) {
466                 _dcp_name->SetLabel (std_to_wx (s.substr (0, 28)) + N_("..."));
467                 _dcp_name->SetToolTip (std_to_wx (s));
468         } else {
469                 _dcp_name->SetLabel (std_to_wx (s));
470         }
471 }
472
473 void
474 DCPPanel::best_frame_rate_clicked ()
475 {
476         if (!_film) {
477                 return;
478         }
479         
480         _film->set_video_frame_rate (_film->best_video_frame_rate ());
481 }
482
483 void
484 DCPPanel::three_d_changed ()
485 {
486         if (!_film) {
487                 return;
488         }
489
490         _film->set_three_d (_three_d->GetValue ());
491 }
492
493 void
494 DCPPanel::config_changed ()
495 {
496         _j2k_bandwidth->SetRange (1, Config::instance()->maximum_j2k_bandwidth() / 1000000);
497         setup_frame_rate_widget ();
498 }
499
500 void
501 DCPPanel::setup_frame_rate_widget ()
502 {
503         if (Config::instance()->allow_any_dcp_frame_rate ()) {
504                 _frame_rate_choice->Hide ();
505                 _frame_rate_spin->Show ();
506         } else {
507                 _frame_rate_choice->Show ();
508                 _frame_rate_spin->Hide ();
509         }
510
511         _frame_rate_sizer->Layout ();
512 }
513
514 wxPanel *
515 DCPPanel::make_video_panel ()
516 {
517         wxPanel* panel = new wxPanel (_notebook);
518         wxSizer* sizer = new wxBoxSizer (wxVERTICAL);
519         wxGridBagSizer* grid = new wxGridBagSizer (DCPOMATIC_SIZER_X_GAP, DCPOMATIC_SIZER_Y_GAP);
520         sizer->Add (grid, 0, wxALL, 8);
521         panel->SetSizer (sizer);
522
523         int r = 0;
524         
525         add_label_to_grid_bag_sizer (grid, panel, _("Container"), true, wxGBPosition (r, 0));
526         _container = new wxChoice (panel, wxID_ANY);
527         grid->Add (_container, wxGBPosition (r, 1));
528         ++r;
529
530         {
531                 add_label_to_grid_bag_sizer (grid, panel, _("Frame Rate"), true, wxGBPosition (r, 0));
532                 _frame_rate_sizer = new wxBoxSizer (wxHORIZONTAL);
533                 _frame_rate_choice = new wxChoice (panel, wxID_ANY);
534                 _frame_rate_sizer->Add (_frame_rate_choice, 1, wxALIGN_CENTER_VERTICAL);
535                 _frame_rate_spin = new wxSpinCtrl (panel, wxID_ANY);
536                 _frame_rate_sizer->Add (_frame_rate_spin, 1, wxALIGN_CENTER_VERTICAL);
537                 setup_frame_rate_widget ();
538                 _best_frame_rate = new wxButton (panel, wxID_ANY, _("Use best"));
539                 _frame_rate_sizer->Add (_best_frame_rate, 1, wxALIGN_CENTER_VERTICAL | wxEXPAND);
540                 grid->Add (_frame_rate_sizer, wxGBPosition (r, 1));
541         }
542         ++r;
543
544         _burn_subtitles = new wxCheckBox (panel, wxID_ANY, _("Burn subtitles into image"));
545         grid->Add (_burn_subtitles, wxGBPosition (r, 0), wxGBSpan (1, 2));
546         ++r;
547
548         _three_d = new wxCheckBox (panel, wxID_ANY, _("3D"));
549         grid->Add (_three_d, wxGBPosition (r, 0), wxGBSpan (1, 2));
550         ++r;
551
552         add_label_to_grid_bag_sizer (grid, panel, _("Resolution"), true, wxGBPosition (r, 0));
553         _resolution = new wxChoice (panel, wxID_ANY);
554         grid->Add (_resolution, wxGBPosition (r, 1));
555         ++r;
556
557         {
558                 add_label_to_grid_bag_sizer (grid, panel, _("JPEG2000 bandwidth"), true, wxGBPosition (r, 0));
559                 wxSizer* s = new wxBoxSizer (wxHORIZONTAL);
560                 _j2k_bandwidth = new wxSpinCtrl (panel, wxID_ANY);
561                 s->Add (_j2k_bandwidth, 1);
562                 add_label_to_sizer (s, panel, _("Mbit/s"), false);
563                 grid->Add (s, wxGBPosition (r, 1));
564         }
565         ++r;
566
567         add_label_to_grid_bag_sizer (grid, panel, _("Scaler"), true, wxGBPosition (r, 0));
568         _scaler = new wxChoice (panel, wxID_ANY);
569         grid->Add (_scaler, wxGBPosition (r, 1), wxDefaultSpan, wxALIGN_CENTER_VERTICAL);
570         ++r;
571
572         _container->Bind        (wxEVT_COMMAND_CHOICE_SELECTED,       boost::bind (&DCPPanel::container_changed, this));
573         _scaler->Bind           (wxEVT_COMMAND_CHOICE_SELECTED,       boost::bind (&DCPPanel::scaler_changed, this));
574         _frame_rate_choice->Bind(wxEVT_COMMAND_CHOICE_SELECTED,       boost::bind (&DCPPanel::frame_rate_choice_changed, this));
575         _frame_rate_spin->Bind  (wxEVT_COMMAND_SPINCTRL_UPDATED,      boost::bind (&DCPPanel::frame_rate_spin_changed, this));
576         _best_frame_rate->Bind  (wxEVT_COMMAND_BUTTON_CLICKED,        boost::bind (&DCPPanel::best_frame_rate_clicked, this));
577         _burn_subtitles->Bind   (wxEVT_COMMAND_CHECKBOX_CLICKED,      boost::bind (&DCPPanel::burn_subtitles_toggled, this));
578         _j2k_bandwidth->Bind    (wxEVT_COMMAND_SPINCTRL_UPDATED,      boost::bind (&DCPPanel::j2k_bandwidth_changed, this));
579         _resolution->Bind       (wxEVT_COMMAND_CHOICE_SELECTED,       boost::bind (&DCPPanel::resolution_changed, this));
580         _three_d->Bind          (wxEVT_COMMAND_CHECKBOX_CLICKED,      boost::bind (&DCPPanel::three_d_changed, this));
581
582         vector<Scaler const *> const sc = Scaler::all ();
583         for (vector<Scaler const *>::const_iterator i = sc.begin(); i != sc.end(); ++i) {
584                 _scaler->Append (std_to_wx ((*i)->name()));
585         }
586
587         vector<Ratio const *> const ratio = Ratio::all ();
588         for (vector<Ratio const *>::const_iterator i = ratio.begin(); i != ratio.end(); ++i) {
589                 _container->Append (std_to_wx ((*i)->nickname ()));
590         }
591
592         list<int> const dfr = Config::instance()->allowed_dcp_frame_rates ();
593         for (list<int>::const_iterator i = dfr.begin(); i != dfr.end(); ++i) {
594                 _frame_rate_choice->Append (std_to_wx (boost::lexical_cast<string> (*i)));
595         }
596
597         _j2k_bandwidth->SetRange (1, Config::instance()->maximum_j2k_bandwidth() / 1000000);
598         _frame_rate_spin->SetRange (1, 480);
599
600         _resolution->Append (_("2K"));
601         _resolution->Append (_("4K"));
602
603         return panel;
604 }
605
606 wxPanel *
607 DCPPanel::make_audio_panel ()
608 {
609         wxPanel* panel = new wxPanel (_notebook);
610         wxSizer* sizer = new wxBoxSizer (wxVERTICAL);
611         wxGridBagSizer* grid = new wxGridBagSizer (DCPOMATIC_SIZER_X_GAP, DCPOMATIC_SIZER_Y_GAP);
612         sizer->Add (grid, 0, wxALL, 8);
613         panel->SetSizer (sizer);
614
615         int r = 0;
616         add_label_to_grid_bag_sizer (grid, panel, _("Channels"), true, wxGBPosition (r, 0));
617         _audio_channels = new wxSpinCtrl (panel, wxID_ANY);
618         grid->Add (_audio_channels, wxGBPosition (r, 1));
619         ++r;
620
621         _audio_channels->Bind (wxEVT_COMMAND_SPINCTRL_UPDATED, boost::bind (&DCPPanel::audio_channels_changed, this));
622
623         _audio_channels->SetRange (0, MAX_DCP_AUDIO_CHANNELS);
624
625         return panel;
626 }