Prevent selection of too few DCP channels (#611).
[dcpomatic.git] / src / wx / dcp_panel.cc
1 /*
2     Copyright (C) 2012-2015 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 "key_dialog.h"
23 #include "isdcf_metadata_dialog.h"
24 #include "audio_dialog.h"
25 #include "lib/ratio.h"
26 #include "lib/config.h"
27 #include "lib/dcp_content_type.h"
28 #include "lib/util.h"
29 #include "lib/film.h"
30 #include "lib/ffmpeg_content.h"
31 #include "lib/audio_processor.h"
32 #include <dcp/key.h>
33 #include <dcp/raw_convert.h>
34 #include <wx/wx.h>
35 #include <wx/notebook.h>
36 #include <wx/gbsizer.h>
37 #include <wx/spinctrl.h>
38 #include <boost/lexical_cast.hpp>
39 #include <boost/foreach.hpp>
40
41 using std::cout;
42 using std::list;
43 using std::string;
44 using std::vector;
45 using std::pair;
46 using std::make_pair;
47 using boost::lexical_cast;
48 using boost::shared_ptr;
49
50 DCPPanel::DCPPanel (wxNotebook* n, boost::shared_ptr<Film> f)
51         : _audio_dialog (0)
52         , _film (f)
53         , _generally_sensitive (true)
54 {
55         _panel = new wxPanel (n);
56         _sizer = new wxBoxSizer (wxVERTICAL);
57         _panel->SetSizer (_sizer);
58
59         wxGridBagSizer* grid = new wxGridBagSizer (DCPOMATIC_SIZER_X_GAP, DCPOMATIC_SIZER_Y_GAP);
60         _sizer->Add (grid, 0, wxEXPAND | wxALL, 8);
61
62         int r = 0;
63         
64         add_label_to_grid_bag_sizer (grid, _panel, _("Name"), true, wxGBPosition (r, 0));
65         _name = new wxTextCtrl (_panel, wxID_ANY);
66         grid->Add (_name, wxGBPosition(r, 1), wxDefaultSpan, wxEXPAND | wxLEFT | wxRIGHT);
67         ++r;
68         
69         int flags = wxALIGN_CENTER_VERTICAL;
70 #ifdef __WXOSX__
71         flags |= wxALIGN_RIGHT;
72 #endif  
73
74         _use_isdcf_name = new wxCheckBox (_panel, wxID_ANY, _("Use ISDCF name"));
75         grid->Add (_use_isdcf_name, wxGBPosition (r, 0), wxDefaultSpan, flags);
76
77         {
78                 wxBoxSizer* s = new wxBoxSizer (wxHORIZONTAL);
79                 _edit_isdcf_button = new wxButton (_panel, wxID_ANY, _("Details..."));
80                 s->Add (_edit_isdcf_button, 1, wxEXPAND | wxRIGHT, DCPOMATIC_SIZER_X_GAP);
81                 _copy_isdcf_name_button = new wxButton (_panel, wxID_ANY, _("Copy as name"));
82                 s->Add (_copy_isdcf_name_button, 1, wxEXPAND | wxLEFT, DCPOMATIC_SIZER_X_GAP);
83                 grid->Add (s, wxGBPosition (r, 1), wxDefaultSpan, wxEXPAND);
84                 ++r;
85         }
86
87         /* wxST_ELLIPSIZE_MIDDLE works around a bug in GTK2 and/or wxWidgets, see
88            http://trac.wxwidgets.org/ticket/12539
89         */
90         _dcp_name = new wxStaticText (
91                 _panel, wxID_ANY, wxT (""), wxDefaultPosition, wxDefaultSize,
92                 wxALIGN_CENTRE_HORIZONTAL | wxST_NO_AUTORESIZE | wxST_ELLIPSIZE_MIDDLE
93                 );
94
95         grid->Add (_dcp_name, wxGBPosition(r, 0), wxGBSpan (1, 2), wxALIGN_CENTER_VERTICAL | wxEXPAND);
96         ++r;
97
98         add_label_to_grid_bag_sizer (grid, _panel, _("Content Type"), true, wxGBPosition (r, 0));
99         _dcp_content_type = new wxChoice (_panel, wxID_ANY);
100         grid->Add (_dcp_content_type, wxGBPosition (r, 1));
101         ++r;
102
103         _notebook = new wxNotebook (_panel, wxID_ANY);
104         _sizer->Add (_notebook, 1, wxEXPAND | wxTOP, 6);
105
106         _notebook->AddPage (make_video_panel (), _("Video"), false);
107         _notebook->AddPage (make_audio_panel (), _("Audio"), false);
108         
109         _signed = new wxCheckBox (_panel, wxID_ANY, _("Signed"));
110         grid->Add (_signed, wxGBPosition (r, 0), wxGBSpan (1, 2));
111         ++r;
112         
113         _encrypted = new wxCheckBox (_panel, wxID_ANY, _("Encrypted"));
114         grid->Add (_encrypted, wxGBPosition (r, 0), wxGBSpan (1, 2));
115         ++r;
116
117         wxClientDC dc (_panel);
118         wxSize size = dc.GetTextExtent (wxT ("GGGGGGGG..."));
119         size.SetHeight (-1);
120
121         {
122                add_label_to_grid_bag_sizer (grid, _panel, _("Key"), true, wxGBPosition (r, 0));
123                wxBoxSizer* s = new wxBoxSizer (wxHORIZONTAL);
124                _key = new wxStaticText (_panel, wxID_ANY, "", wxDefaultPosition, size);
125                s->Add (_key, 1, wxALIGN_CENTER_VERTICAL);
126                _edit_key = new wxButton (_panel, wxID_ANY, _("Edit..."));
127                s->Add (_edit_key);
128                grid->Add (s, wxGBPosition (r, 1));
129                ++r;
130         }
131         
132         add_label_to_grid_bag_sizer (grid, _panel, _("Standard"), true, wxGBPosition (r, 0));
133         _standard = new wxChoice (_panel, wxID_ANY);
134         grid->Add (_standard, wxGBPosition (r, 1), wxDefaultSpan, wxALIGN_CENTER_VERTICAL);
135         ++r;
136
137         _name->Bind              (wxEVT_COMMAND_TEXT_UPDATED,         boost::bind (&DCPPanel::name_changed, this));
138         _use_isdcf_name->Bind    (wxEVT_COMMAND_CHECKBOX_CLICKED,     boost::bind (&DCPPanel::use_isdcf_name_toggled, this));
139         _edit_isdcf_button->Bind (wxEVT_COMMAND_BUTTON_CLICKED,       boost::bind (&DCPPanel::edit_isdcf_button_clicked, this));
140         _copy_isdcf_name_button->Bind(wxEVT_COMMAND_BUTTON_CLICKED,   boost::bind (&DCPPanel::copy_isdcf_name_button_clicked, this));
141         _dcp_content_type->Bind  (wxEVT_COMMAND_CHOICE_SELECTED,      boost::bind (&DCPPanel::dcp_content_type_changed, this));
142         _signed->Bind            (wxEVT_COMMAND_CHECKBOX_CLICKED,     boost::bind (&DCPPanel::signed_toggled, this));
143         _encrypted->Bind         (wxEVT_COMMAND_CHECKBOX_CLICKED,     boost::bind (&DCPPanel::encrypted_toggled, this));
144         _edit_key->Bind          (wxEVT_COMMAND_BUTTON_CLICKED,       boost::bind (&DCPPanel::edit_key_clicked, this));
145         _standard->Bind          (wxEVT_COMMAND_CHOICE_SELECTED,      boost::bind (&DCPPanel::standard_changed, this));
146
147         vector<DCPContentType const *> const ct = DCPContentType::all ();
148         for (vector<DCPContentType const *>::const_iterator i = ct.begin(); i != ct.end(); ++i) {
149                 _dcp_content_type->Append (std_to_wx ((*i)->pretty_name ()));
150         }
151
152         _standard->Append (_("SMPTE"));
153         _standard->Append (_("Interop"));
154
155         Config::instance()->Changed.connect (boost::bind (&DCPPanel::config_changed, this));
156 }
157
158 void
159 DCPPanel::edit_key_clicked ()
160 {
161         KeyDialog* d = new KeyDialog (_panel, _film->key ());
162         if (d->ShowModal () == wxID_OK) {
163                 _film->set_key (d->key ());
164         }
165         d->Destroy ();
166 }
167
168 void
169 DCPPanel::name_changed ()
170 {
171         if (!_film) {
172                 return;
173         }
174
175         _film->set_name (string (_name->GetValue().mb_str()));
176 }
177
178 void
179 DCPPanel::j2k_bandwidth_changed ()
180 {
181         if (!_film) {
182                 return;
183         }
184         
185         _film->set_j2k_bandwidth (_j2k_bandwidth->GetValue() * 1000000);
186 }
187
188 void
189 DCPPanel::signed_toggled ()
190 {
191         if (!_film) {
192                 return;
193         }
194
195         _film->set_signed (_signed->GetValue ());
196 }
197
198 void
199 DCPPanel::burn_subtitles_toggled ()
200 {
201         if (!_film) {
202                 return;
203         }
204
205         _film->set_burn_subtitles (_burn_subtitles->GetValue ());
206 }
207
208 void
209 DCPPanel::encrypted_toggled ()
210 {
211         if (!_film) {
212                 return;
213         }
214
215         _film->set_encrypted (_encrypted->GetValue ());
216 }
217                                
218 /** Called when the frame rate choice widget has been changed */
219 void
220 DCPPanel::frame_rate_choice_changed ()
221 {
222         if (!_film) {
223                 return;
224         }
225
226         _film->set_video_frame_rate (
227                 boost::lexical_cast<int> (
228                         wx_to_std (_frame_rate_choice->GetString (_frame_rate_choice->GetSelection ()))
229                         )
230                 );
231 }
232
233 /** Called when the frame rate spin widget has been changed */
234 void
235 DCPPanel::frame_rate_spin_changed ()
236 {
237         if (!_film) {
238                 return;
239         }
240
241         _film->set_video_frame_rate (_frame_rate_spin->GetValue ());
242 }
243
244 void
245 DCPPanel::audio_channels_changed ()
246 {
247         if (!_film) {
248                 return;
249         }
250
251         _film->set_audio_channels (dcp::raw_convert<int> (string_client_data (_audio_channels->GetClientObject (_audio_channels->GetSelection ()))));
252 }
253
254 void
255 DCPPanel::resolution_changed ()
256 {
257         if (!_film) {
258                 return;
259         }
260
261         _film->set_resolution (_resolution->GetSelection() == 0 ? RESOLUTION_2K : RESOLUTION_4K);
262 }
263
264 void
265 DCPPanel::standard_changed ()
266 {
267         if (!_film) {
268                 return;
269         }
270
271         _film->set_interop (_standard->GetSelection() == 1);
272 }
273
274 void
275 DCPPanel::film_changed (int p)
276 {
277         switch (p) {
278         case Film::NONE:
279                 break;
280         case Film::CONTAINER:
281                 setup_container ();
282                 break;
283         case Film::NAME:
284                 checked_set (_name, _film->name());
285                 setup_dcp_name ();
286                 break;
287         case Film::DCP_CONTENT_TYPE:
288                 checked_set (_dcp_content_type, DCPContentType::as_index (_film->dcp_content_type ()));
289                 setup_dcp_name ();
290                 break;
291         case Film::BURN_SUBTITLES:
292                 checked_set (_burn_subtitles, _film->burn_subtitles ());
293                 break;
294         case Film::SIGNED:
295                 checked_set (_signed, _film->is_signed ());
296                 break;
297         case Film::ENCRYPTED:
298                 checked_set (_encrypted, _film->encrypted ());
299                 if (_film->encrypted ()) {
300                         _film->set_signed (true);
301                         _signed->Enable (false);
302                         _key->Enable (_generally_sensitive);
303                         _edit_key->Enable (_generally_sensitive);
304                 } else {
305                         _signed->Enable (_generally_sensitive);
306                         _key->Enable (false);
307                         _edit_key->Enable (false);
308                 }
309                 break;
310         case Film::KEY:
311                 checked_set (_key, _film->key().hex().substr (0, 8) + "...");
312                 break;
313         case Film::RESOLUTION:
314                 checked_set (_resolution, _film->resolution() == RESOLUTION_2K ? 0 : 1);
315                 setup_container ();
316                 setup_dcp_name ();
317                 break;
318         case Film::J2K_BANDWIDTH:
319                 checked_set (_j2k_bandwidth, _film->j2k_bandwidth() / 1000000);
320                 break;
321         case Film::USE_ISDCF_NAME:
322         {
323                 checked_set (_use_isdcf_name, _film->use_isdcf_name ());
324                 setup_dcp_name ();
325                 _edit_isdcf_button->Enable (_film->use_isdcf_name ());
326                 break;
327         }
328         case Film::ISDCF_METADATA:
329                 setup_dcp_name ();
330                 break;
331         case Film::VIDEO_FRAME_RATE:
332         {
333                 bool done = false;
334                 for (unsigned int i = 0; i < _frame_rate_choice->GetCount(); ++i) {
335                         if (wx_to_std (_frame_rate_choice->GetString(i)) == boost::lexical_cast<string> (_film->video_frame_rate())) {
336                                 checked_set (_frame_rate_choice, i);
337                                 done = true;
338                                 break;
339                         }
340                 }
341
342                 if (!done) {
343                         checked_set (_frame_rate_choice, -1);
344                 }
345
346                 _frame_rate_spin->SetValue (_film->video_frame_rate ());
347
348                 _best_frame_rate->Enable (_film->best_video_frame_rate () != _film->video_frame_rate ());
349                 break;
350         }
351         case Film::AUDIO_CHANNELS:
352                 checked_set (_audio_channels, dcp::raw_convert<string> (_film->audio_channels ()));
353                 setup_dcp_name ();
354                 break;
355         case Film::THREE_D:
356                 checked_set (_three_d, _film->three_d ());
357                 setup_dcp_name ();
358                 break;
359         case Film::INTEROP:
360                 checked_set (_standard, _film->interop() ? 1 : 0);
361                 setup_dcp_name ();
362                 break;
363         case Film::AUDIO_PROCESSOR:
364                 if (_film->audio_processor ()) {
365                         checked_set (_audio_processor, _film->audio_processor()->id());
366                 } else {
367                         checked_set (_audio_processor, 0);
368                 }
369                 setup_audio_channels_choice ();
370                 film_changed (Film::AUDIO_CHANNELS);
371                 break;
372         default:
373                 break;
374         }
375 }
376
377 void
378 DCPPanel::film_content_changed (int property)
379 {
380         if (property == AudioContentProperty::AUDIO_STREAMS ||
381             property == SubtitleContentProperty::USE_SUBTITLES ||
382             property == VideoContentProperty::VIDEO_SCALE) {
383                 setup_dcp_name ();
384         }
385 }
386
387
388 void
389 DCPPanel::setup_container ()
390 {
391         int n = 0;
392         vector<Ratio const *> ratios = Ratio::all ();
393         vector<Ratio const *>::iterator i = ratios.begin ();
394         while (i != ratios.end() && *i != _film->container ()) {
395                 ++i;
396                 ++n;
397         }
398         
399         if (i == ratios.end()) {
400                 checked_set (_container, -1);
401                 checked_set (_container_size, wxT (""));
402         } else {
403                 checked_set (_container, n);
404                 dcp::Size const size = fit_ratio_within (_film->container()->ratio(), _film->full_frame ());
405                 checked_set (_container_size, wxString::Format ("%dx%d", size.width, size.height));
406         }
407         
408         setup_dcp_name ();
409 }       
410
411 /** Called when the container widget has been changed */
412 void
413 DCPPanel::container_changed ()
414 {
415         if (!_film) {
416                 return;
417         }
418
419         int const n = _container->GetSelection ();
420         if (n >= 0) {
421                 vector<Ratio const *> ratios = Ratio::all ();
422                 DCPOMATIC_ASSERT (n < int (ratios.size()));
423                 _film->set_container (ratios[n]);
424         }
425 }
426
427 /** Called when the DCP content type widget has been changed */
428 void
429 DCPPanel::dcp_content_type_changed ()
430 {
431         if (!_film) {
432                 return;
433         }
434
435         int const n = _dcp_content_type->GetSelection ();
436         if (n != wxNOT_FOUND) {
437                 _film->set_dcp_content_type (DCPContentType::from_index (n));
438         }
439 }
440
441 void
442 DCPPanel::set_film (shared_ptr<Film> film)
443 {
444         _film = film;
445         
446         film_changed (Film::NAME);
447         film_changed (Film::USE_ISDCF_NAME);
448         film_changed (Film::CONTENT);
449         film_changed (Film::DCP_CONTENT_TYPE);
450         film_changed (Film::CONTAINER);
451         film_changed (Film::RESOLUTION);
452         film_changed (Film::SIGNED);
453         film_changed (Film::BURN_SUBTITLES);
454         film_changed (Film::ENCRYPTED);
455         film_changed (Film::KEY);
456         film_changed (Film::J2K_BANDWIDTH);
457         film_changed (Film::ISDCF_METADATA);
458         film_changed (Film::VIDEO_FRAME_RATE);
459         film_changed (Film::AUDIO_CHANNELS);
460         film_changed (Film::SEQUENCE_VIDEO);
461         film_changed (Film::THREE_D);
462         film_changed (Film::INTEROP);
463         film_changed (Film::AUDIO_PROCESSOR);
464 }
465
466 void
467 DCPPanel::set_general_sensitivity (bool s)
468 {
469         _name->Enable (s);
470         _use_isdcf_name->Enable (s);
471         _edit_isdcf_button->Enable (s);
472         _dcp_content_type->Enable (s);
473         _copy_isdcf_name_button->Enable (s);
474
475         bool si = s;
476         if (_film && _film->encrypted ()) {
477                 si = false;
478         }
479         _burn_subtitles->Enable (s);
480         _signed->Enable (si);
481         
482         _encrypted->Enable (s);
483         _key->Enable (s && _film && _film->encrypted ());
484         _edit_key->Enable (s && _film && _film->encrypted ());
485         _frame_rate_choice->Enable (s);
486         _frame_rate_spin->Enable (s);
487         _audio_channels->Enable (s);
488         _audio_processor->Enable (s);
489         _j2k_bandwidth->Enable (s);
490         _container->Enable (s);
491         _best_frame_rate->Enable (s && _film && _film->best_video_frame_rate () != _film->video_frame_rate ());
492         _resolution->Enable (s);
493         _three_d->Enable (s);
494         _standard->Enable (s);
495 }
496
497 void
498 DCPPanel::use_isdcf_name_toggled ()
499 {
500         if (!_film) {
501                 return;
502         }
503
504         _film->set_use_isdcf_name (_use_isdcf_name->GetValue ());
505 }
506
507 void
508 DCPPanel::edit_isdcf_button_clicked ()
509 {
510         if (!_film) {
511                 return;
512         }
513
514         ISDCFMetadataDialog* d = new ISDCFMetadataDialog (_panel, _film->isdcf_metadata ());
515         d->ShowModal ();
516         _film->set_isdcf_metadata (d->isdcf_metadata ());
517         d->Destroy ();
518 }
519
520 void
521 DCPPanel::setup_dcp_name ()
522 {
523         _dcp_name->SetLabel (std_to_wx (_film->dcp_name (true)));
524 }
525
526 void
527 DCPPanel::best_frame_rate_clicked ()
528 {
529         if (!_film) {
530                 return;
531         }
532         
533         _film->set_video_frame_rate (_film->best_video_frame_rate ());
534 }
535
536 void
537 DCPPanel::three_d_changed ()
538 {
539         if (!_film) {
540                 return;
541         }
542
543         _film->set_three_d (_three_d->GetValue ());
544 }
545
546 void
547 DCPPanel::config_changed ()
548 {
549         _j2k_bandwidth->SetRange (1, Config::instance()->maximum_j2k_bandwidth() / 1000000);
550         setup_frame_rate_widget ();
551 }
552
553 void
554 DCPPanel::setup_frame_rate_widget ()
555 {
556         if (Config::instance()->allow_any_dcp_frame_rate ()) {
557                 _frame_rate_choice->Hide ();
558                 _frame_rate_spin->Show ();
559         } else {
560                 _frame_rate_choice->Show ();
561                 _frame_rate_spin->Hide ();
562         }
563
564         _frame_rate_sizer->Layout ();
565 }
566
567 wxPanel *
568 DCPPanel::make_video_panel ()
569 {
570         wxPanel* panel = new wxPanel (_notebook);
571         wxSizer* sizer = new wxBoxSizer (wxVERTICAL);
572         wxGridBagSizer* grid = new wxGridBagSizer (DCPOMATIC_SIZER_X_GAP, DCPOMATIC_SIZER_Y_GAP);
573         sizer->Add (grid, 0, wxALL, 8);
574         panel->SetSizer (sizer);
575
576         int r = 0;
577         
578         add_label_to_grid_bag_sizer (grid, panel, _("Container"), true, wxGBPosition (r, 0));
579         {
580                 wxBoxSizer* s = new wxBoxSizer (wxHORIZONTAL);
581                 _container = new wxChoice (panel, wxID_ANY);
582                 s->Add (_container, 1, wxEXPAND | wxRIGHT, DCPOMATIC_SIZER_X_GAP);
583                 _container_size = new wxStaticText (panel, wxID_ANY, wxT (""));
584                 s->Add (_container_size, 1, wxLEFT | wxALIGN_CENTER_VERTICAL);
585                 grid->Add (s, wxGBPosition (r,1 ), wxDefaultSpan, wxEXPAND);
586                 ++r;
587         }
588
589         add_label_to_grid_bag_sizer (grid, panel, _("Frame Rate"), true, wxGBPosition (r, 0));
590         {
591                 _frame_rate_sizer = new wxBoxSizer (wxHORIZONTAL);
592                 _frame_rate_choice = new wxChoice (panel, wxID_ANY);
593                 _frame_rate_sizer->Add (_frame_rate_choice, 1, wxALIGN_CENTER_VERTICAL);
594                 _frame_rate_spin = new wxSpinCtrl (panel, wxID_ANY);
595                 _frame_rate_sizer->Add (_frame_rate_spin, 1, wxALIGN_CENTER_VERTICAL);
596                 setup_frame_rate_widget ();
597                 _best_frame_rate = new wxButton (panel, wxID_ANY, _("Use best"));
598                 _frame_rate_sizer->Add (_best_frame_rate, 1, wxALIGN_CENTER_VERTICAL);
599                 grid->Add (_frame_rate_sizer, wxGBPosition (r, 1));
600                 ++r;
601         }
602
603         _burn_subtitles = new wxCheckBox (panel, wxID_ANY, _("Burn subtitles into image"));
604         grid->Add (_burn_subtitles, wxGBPosition (r, 0), wxGBSpan (1, 2));
605         ++r;
606
607         _three_d = new wxCheckBox (panel, wxID_ANY, _("3D"));
608         grid->Add (_three_d, wxGBPosition (r, 0), wxGBSpan (1, 2));
609         ++r;
610
611         add_label_to_grid_bag_sizer (grid, panel, _("Resolution"), true, wxGBPosition (r, 0));
612         _resolution = new wxChoice (panel, wxID_ANY);
613         grid->Add (_resolution, wxGBPosition (r, 1));
614         ++r;
615
616         {
617                 add_label_to_grid_bag_sizer (grid, panel, _("JPEG2000 bandwidth"), true, wxGBPosition (r, 0));
618                 wxSizer* s = new wxBoxSizer (wxHORIZONTAL);
619                 _j2k_bandwidth = new wxSpinCtrl (panel, wxID_ANY);
620                 s->Add (_j2k_bandwidth, 1);
621                 add_label_to_sizer (s, panel, _("Mbit/s"), false);
622                 grid->Add (s, wxGBPosition (r, 1));
623         }
624         ++r;
625
626         _container->Bind        (wxEVT_COMMAND_CHOICE_SELECTED,       boost::bind (&DCPPanel::container_changed, this));
627         _frame_rate_choice->Bind(wxEVT_COMMAND_CHOICE_SELECTED,       boost::bind (&DCPPanel::frame_rate_choice_changed, this));
628         _frame_rate_spin->Bind  (wxEVT_COMMAND_SPINCTRL_UPDATED,      boost::bind (&DCPPanel::frame_rate_spin_changed, this));
629         _best_frame_rate->Bind  (wxEVT_COMMAND_BUTTON_CLICKED,        boost::bind (&DCPPanel::best_frame_rate_clicked, this));
630         _burn_subtitles->Bind   (wxEVT_COMMAND_CHECKBOX_CLICKED,      boost::bind (&DCPPanel::burn_subtitles_toggled, this));
631         _j2k_bandwidth->Bind    (wxEVT_COMMAND_SPINCTRL_UPDATED,      boost::bind (&DCPPanel::j2k_bandwidth_changed, this));
632         /* Also listen to wxEVT_COMMAND_TEXT_UPDATED so that typing numbers directly in is always noticed */
633         _j2k_bandwidth->Bind    (wxEVT_COMMAND_TEXT_UPDATED,          boost::bind (&DCPPanel::j2k_bandwidth_changed, this));
634         _resolution->Bind       (wxEVT_COMMAND_CHOICE_SELECTED,       boost::bind (&DCPPanel::resolution_changed, this));
635         _three_d->Bind          (wxEVT_COMMAND_CHECKBOX_CLICKED,      boost::bind (&DCPPanel::three_d_changed, this));
636
637         vector<Ratio const *> const ratio = Ratio::all ();
638         for (vector<Ratio const *>::const_iterator i = ratio.begin(); i != ratio.end(); ++i) {
639                 _container->Append (std_to_wx ((*i)->nickname ()));
640         }
641
642         list<int> const dfr = Config::instance()->allowed_dcp_frame_rates ();
643         for (list<int>::const_iterator i = dfr.begin(); i != dfr.end(); ++i) {
644                 _frame_rate_choice->Append (std_to_wx (boost::lexical_cast<string> (*i)));
645         }
646
647         _j2k_bandwidth->SetRange (1, Config::instance()->maximum_j2k_bandwidth() / 1000000);
648         _frame_rate_spin->SetRange (1, 480);
649
650         _resolution->Append (_("2K"));
651         _resolution->Append (_("4K"));
652
653         return panel;
654 }
655
656 void
657 DCPPanel::setup_audio_channels_choice ()
658 {
659         int min = 2;
660         if (_film && _film->audio_processor ()) {
661                 min = _film->audio_processor()->out_channels ();
662         }
663
664         if (min % 2 == 1) {
665                 ++min;
666         }
667
668         vector<pair<string, string> > items;
669         for (int i = min; i <= 12; i += 2) {
670                 items.push_back (make_pair (dcp::raw_convert<string> (i), dcp::raw_convert<string> (i)));
671         }
672
673         checked_set (_audio_channels, items);
674 }
675
676 wxPanel *
677 DCPPanel::make_audio_panel ()
678 {
679         wxPanel* panel = new wxPanel (_notebook);
680         wxSizer* sizer = new wxBoxSizer (wxVERTICAL);
681         wxGridBagSizer* grid = new wxGridBagSizer (DCPOMATIC_SIZER_X_GAP, DCPOMATIC_SIZER_Y_GAP);
682         sizer->Add (grid, 0, wxALL, 8);
683         panel->SetSizer (sizer);
684
685         int r = 0;
686         
687         add_label_to_grid_bag_sizer (grid, panel, _("Channels"), true, wxGBPosition (r, 0));
688         _audio_channels = new wxChoice (panel, wxID_ANY);
689         setup_audio_channels_choice ();
690         grid->Add (_audio_channels, wxGBPosition (r, 1));
691         ++r;
692
693         add_label_to_grid_bag_sizer (grid, panel, _("Processor"), true, wxGBPosition (r, 0));
694         _audio_processor = new wxChoice (panel, wxID_ANY);
695         _audio_processor->Append (_("None"), new wxStringClientData (N_("none")));
696         BOOST_FOREACH (AudioProcessor const * ap, AudioProcessor::all ()) {
697                 _audio_processor->Append (std_to_wx (ap->name ()), new wxStringClientData (std_to_wx (ap->id ())));
698         }
699         grid->Add (_audio_processor, wxGBPosition (r, 1));
700         ++r;
701
702         _show_audio = new wxButton (panel, wxID_ANY, _("Show audio..."));
703         grid->Add (_show_audio, wxGBPosition (r, 0), wxGBSpan (1, 2));
704         ++r;
705
706         _audio_channels->Bind (wxEVT_COMMAND_CHOICE_SELECTED, boost::bind (&DCPPanel::audio_channels_changed, this));
707         _audio_processor->Bind (wxEVT_COMMAND_CHOICE_SELECTED, boost::bind (&DCPPanel::audio_processor_changed, this));
708         _show_audio->Bind (wxEVT_COMMAND_BUTTON_CLICKED, boost::bind (&DCPPanel::show_audio_clicked, this));
709
710         return panel;
711 }
712
713 void
714 DCPPanel::copy_isdcf_name_button_clicked ()
715 {
716         _film->set_name (_film->isdcf_name (false));
717         _film->set_use_isdcf_name (false);
718 }
719
720 void
721 DCPPanel::audio_processor_changed ()
722 {
723         if (!_film) {
724                 return;
725         }
726
727         string const s = string_client_data (_audio_processor->GetClientObject (_audio_processor->GetSelection ()));
728         _film->set_audio_processor (AudioProcessor::from_id (s));
729 }
730
731 void
732 DCPPanel::show_audio_clicked ()
733 {
734         if (!_film) {
735                 return;
736         }
737
738         if (_audio_dialog) {
739                 _audio_dialog->Destroy ();
740                 _audio_dialog = 0;
741         }
742         
743         AudioDialog* d = new AudioDialog (_panel, _film);
744         d->Show ();
745         d->set_playlist (_film->playlist ());
746 }