Merge master.
[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         int flags = wxALIGN_CENTER_VERTICAL;
62 #ifdef __WXOSX__
63         flags |= wxALIGN_RIGHT;
64 #endif  
65
66         _use_isdcf_name = new wxCheckBox (_panel, wxID_ANY, _("Use ISDCF name"));
67         grid->Add (_use_isdcf_name, wxGBPosition (r, 0), wxDefaultSpan, flags);
68         _edit_isdcf_button = new wxButton (_panel, wxID_ANY, _("Details..."));
69         grid->Add (_edit_isdcf_button, wxGBPosition (r, 1));
70         ++r;
71
72         add_label_to_grid_bag_sizer (grid, _panel, _("DCP Name"), true, wxGBPosition (r, 0));
73         _dcp_name = new wxStaticText (_panel, wxID_ANY, wxT (""), wxDefaultPosition, wxDefaultSize, wxST_ELLIPSIZE_END);
74         grid->Add (_dcp_name, wxGBPosition(r, 1), wxDefaultSpan, wxALIGN_CENTER_VERTICAL | wxEXPAND);
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         {
270                 checked_set (_use_isdcf_name, _film->use_isdcf_name ());
271                 setup_dcp_name ();
272                 bool const i = _film->use_isdcf_name ();
273                 if (!i) {
274                         _film->set_name (_film->isdcf_name (true));
275                 }
276                 _edit_isdcf_button->Enable (i);
277                 break;
278         }
279         case Film::ISDCF_METADATA:
280                 setup_dcp_name ();
281                 break;
282         case Film::VIDEO_FRAME_RATE:
283         {
284                 bool done = false;
285                 for (unsigned int i = 0; i < _frame_rate_choice->GetCount(); ++i) {
286                         if (wx_to_std (_frame_rate_choice->GetString(i)) == boost::lexical_cast<string> (_film->video_frame_rate())) {
287                                 checked_set (_frame_rate_choice, i);
288                                 done = true;
289                                 break;
290                         }
291                 }
292
293                 if (!done) {
294                         checked_set (_frame_rate_choice, -1);
295                 }
296
297                 _frame_rate_spin->SetValue (_film->video_frame_rate ());
298
299                 _best_frame_rate->Enable (_film->best_video_frame_rate () != _film->video_frame_rate ());
300                 break;
301         }
302         case Film::AUDIO_CHANNELS:
303                 checked_set (_audio_channels, _film->audio_channels ());
304                 setup_dcp_name ();
305                 break;
306         case Film::THREE_D:
307                 checked_set (_three_d, _film->three_d ());
308                 setup_dcp_name ();
309                 break;
310         case Film::INTEROP:
311                 checked_set (_standard, _film->interop() ? 1 : 0);
312                 break;
313         default:
314                 break;
315         }
316 }
317
318 void
319 DCPPanel::film_content_changed (int property)
320 {
321         if (property == FFmpegContentProperty::AUDIO_STREAM ||
322             property == SubtitleContentProperty::USE_SUBTITLES ||
323             property == VideoContentProperty::VIDEO_SCALE) {
324                 setup_dcp_name ();
325         }
326 }
327
328
329 void
330 DCPPanel::setup_container ()
331 {
332         int n = 0;
333         vector<Ratio const *> ratios = Ratio::all ();
334         vector<Ratio const *>::iterator i = ratios.begin ();
335         while (i != ratios.end() && *i != _film->container ()) {
336                 ++i;
337                 ++n;
338         }
339         
340         if (i == ratios.end()) {
341                 checked_set (_container, -1);
342         } else {
343                 checked_set (_container, n);
344         }
345         
346         setup_dcp_name ();
347 }       
348
349 /** Called when the container widget has been changed */
350 void
351 DCPPanel::container_changed ()
352 {
353         if (!_film) {
354                 return;
355         }
356
357         int const n = _container->GetSelection ();
358         if (n >= 0) {
359                 vector<Ratio const *> ratios = Ratio::all ();
360                 assert (n < int (ratios.size()));
361                 _film->set_container (ratios[n]);
362         }
363 }
364
365 /** Called when the DCP content type widget has been changed */
366 void
367 DCPPanel::dcp_content_type_changed ()
368 {
369         if (!_film) {
370                 return;
371         }
372
373         int const n = _dcp_content_type->GetSelection ();
374         if (n != wxNOT_FOUND) {
375                 _film->set_dcp_content_type (DCPContentType::from_index (n));
376         }
377 }
378
379 void
380 DCPPanel::set_film (shared_ptr<Film> film)
381 {
382         _film = film;
383         
384         film_changed (Film::NAME);
385         film_changed (Film::USE_ISDCF_NAME);
386         film_changed (Film::CONTENT);
387         film_changed (Film::DCP_CONTENT_TYPE);
388         film_changed (Film::CONTAINER);
389         film_changed (Film::RESOLUTION);
390         film_changed (Film::SCALER);
391         film_changed (Film::SIGNED);
392         film_changed (Film::BURN_SUBTITLES);
393         film_changed (Film::ENCRYPTED);
394         film_changed (Film::J2K_BANDWIDTH);
395         film_changed (Film::ISDCF_METADATA);
396         film_changed (Film::VIDEO_FRAME_RATE);
397         film_changed (Film::AUDIO_CHANNELS);
398         film_changed (Film::SEQUENCE_VIDEO);
399         film_changed (Film::THREE_D);
400         film_changed (Film::INTEROP);
401 }
402
403 void
404 DCPPanel::set_general_sensitivity (bool s)
405 {
406         _name->Enable (s);
407         _use_isdcf_name->Enable (s);
408         _edit_isdcf_button->Enable (s);
409         _dcp_content_type->Enable (s);
410
411         bool si = s;
412         if (_film && _film->encrypted ()) {
413                 si = false;
414         }
415         _burn_subtitles->Enable (s);
416         _signed->Enable (si);
417         
418         _encrypted->Enable (s);
419         _frame_rate_choice->Enable (s);
420         _frame_rate_spin->Enable (s);
421         _audio_channels->Enable (s);
422         _j2k_bandwidth->Enable (s);
423         _container->Enable (s);
424         _best_frame_rate->Enable (s && _film && _film->best_video_frame_rate () != _film->video_frame_rate ());
425         _resolution->Enable (s);
426         _scaler->Enable (s);
427         _three_d->Enable (s);
428         _standard->Enable (s);
429 }
430
431 /** Called when the scaler widget has been changed */
432 void
433 DCPPanel::scaler_changed ()
434 {
435         if (!_film) {
436                 return;
437         }
438
439         int const n = _scaler->GetSelection ();
440         if (n >= 0) {
441                 _film->set_scaler (Scaler::from_index (n));
442         }
443 }
444
445 void
446 DCPPanel::use_isdcf_name_toggled ()
447 {
448         if (!_film) {
449                 return;
450         }
451
452         _film->set_use_isdcf_name (_use_isdcf_name->GetValue ());
453 }
454
455 void
456 DCPPanel::edit_isdcf_button_clicked ()
457 {
458         if (!_film) {
459                 return;
460         }
461
462         ISDCFMetadataDialog* d = new ISDCFMetadataDialog (_panel, _film->isdcf_metadata ());
463         d->ShowModal ();
464         _film->set_isdcf_metadata (d->isdcf_metadata ());
465         d->Destroy ();
466 }
467
468 void
469 DCPPanel::setup_dcp_name ()
470 {
471         string s = _film->dcp_name (true);
472         if (s.length() > 28) {
473                 _dcp_name->SetLabel (std_to_wx (s.substr (0, 28)) + N_("..."));
474                 _dcp_name->SetToolTip (std_to_wx (s));
475         } else {
476                 _dcp_name->SetLabel (std_to_wx (s));
477         }
478 }
479
480 void
481 DCPPanel::best_frame_rate_clicked ()
482 {
483         if (!_film) {
484                 return;
485         }
486         
487         _film->set_video_frame_rate (_film->best_video_frame_rate ());
488 }
489
490 void
491 DCPPanel::three_d_changed ()
492 {
493         if (!_film) {
494                 return;
495         }
496
497         _film->set_three_d (_three_d->GetValue ());
498 }
499
500 void
501 DCPPanel::config_changed ()
502 {
503         _j2k_bandwidth->SetRange (1, Config::instance()->maximum_j2k_bandwidth() / 1000000);
504         setup_frame_rate_widget ();
505 }
506
507 void
508 DCPPanel::setup_frame_rate_widget ()
509 {
510         if (Config::instance()->allow_any_dcp_frame_rate ()) {
511                 _frame_rate_choice->Hide ();
512                 _frame_rate_spin->Show ();
513         } else {
514                 _frame_rate_choice->Show ();
515                 _frame_rate_spin->Hide ();
516         }
517
518         _frame_rate_sizer->Layout ();
519 }
520
521 wxPanel *
522 DCPPanel::make_video_panel ()
523 {
524         wxPanel* panel = new wxPanel (_notebook);
525         wxSizer* sizer = new wxBoxSizer (wxVERTICAL);
526         wxGridBagSizer* grid = new wxGridBagSizer (DCPOMATIC_SIZER_X_GAP, DCPOMATIC_SIZER_Y_GAP);
527         sizer->Add (grid, 0, wxALL, 8);
528         panel->SetSizer (sizer);
529
530         int r = 0;
531         
532         add_label_to_grid_bag_sizer (grid, panel, _("Container"), true, wxGBPosition (r, 0));
533         _container = new wxChoice (panel, wxID_ANY);
534         grid->Add (_container, wxGBPosition (r, 1));
535         ++r;
536
537         {
538                 add_label_to_grid_bag_sizer (grid, panel, _("Frame Rate"), true, wxGBPosition (r, 0));
539                 _frame_rate_sizer = new wxBoxSizer (wxHORIZONTAL);
540                 _frame_rate_choice = new wxChoice (panel, wxID_ANY);
541                 _frame_rate_sizer->Add (_frame_rate_choice, 1, wxALIGN_CENTER_VERTICAL);
542                 _frame_rate_spin = new wxSpinCtrl (panel, wxID_ANY);
543                 _frame_rate_sizer->Add (_frame_rate_spin, 1, wxALIGN_CENTER_VERTICAL);
544                 setup_frame_rate_widget ();
545                 _best_frame_rate = new wxButton (panel, wxID_ANY, _("Use best"));
546                 _frame_rate_sizer->Add (_best_frame_rate, 1, wxALIGN_CENTER_VERTICAL | wxEXPAND);
547                 grid->Add (_frame_rate_sizer, wxGBPosition (r, 1));
548         }
549         ++r;
550
551         _burn_subtitles = new wxCheckBox (panel, wxID_ANY, _("Burn subtitles into image"));
552         grid->Add (_burn_subtitles, wxGBPosition (r, 0), wxGBSpan (1, 2));
553         ++r;
554
555         _three_d = new wxCheckBox (panel, wxID_ANY, _("3D"));
556         grid->Add (_three_d, wxGBPosition (r, 0), wxGBSpan (1, 2));
557         ++r;
558
559         add_label_to_grid_bag_sizer (grid, panel, _("Resolution"), true, wxGBPosition (r, 0));
560         _resolution = new wxChoice (panel, wxID_ANY);
561         grid->Add (_resolution, wxGBPosition (r, 1));
562         ++r;
563
564         {
565                 add_label_to_grid_bag_sizer (grid, panel, _("JPEG2000 bandwidth"), true, wxGBPosition (r, 0));
566                 wxSizer* s = new wxBoxSizer (wxHORIZONTAL);
567                 _j2k_bandwidth = new wxSpinCtrl (panel, wxID_ANY);
568                 s->Add (_j2k_bandwidth, 1);
569                 add_label_to_sizer (s, panel, _("Mbit/s"), false);
570                 grid->Add (s, wxGBPosition (r, 1));
571         }
572         ++r;
573
574         add_label_to_grid_bag_sizer (grid, panel, _("Scaler"), true, wxGBPosition (r, 0));
575         _scaler = new wxChoice (panel, wxID_ANY);
576         grid->Add (_scaler, wxGBPosition (r, 1), wxDefaultSpan, wxALIGN_CENTER_VERTICAL);
577         ++r;
578
579         _container->Bind        (wxEVT_COMMAND_CHOICE_SELECTED,       boost::bind (&DCPPanel::container_changed, this));
580         _scaler->Bind           (wxEVT_COMMAND_CHOICE_SELECTED,       boost::bind (&DCPPanel::scaler_changed, this));
581         _frame_rate_choice->Bind(wxEVT_COMMAND_CHOICE_SELECTED,       boost::bind (&DCPPanel::frame_rate_choice_changed, this));
582         _frame_rate_spin->Bind  (wxEVT_COMMAND_SPINCTRL_UPDATED,      boost::bind (&DCPPanel::frame_rate_spin_changed, this));
583         _best_frame_rate->Bind  (wxEVT_COMMAND_BUTTON_CLICKED,        boost::bind (&DCPPanel::best_frame_rate_clicked, this));
584         _burn_subtitles->Bind   (wxEVT_COMMAND_CHECKBOX_CLICKED,      boost::bind (&DCPPanel::burn_subtitles_toggled, this));
585         _j2k_bandwidth->Bind    (wxEVT_COMMAND_SPINCTRL_UPDATED,      boost::bind (&DCPPanel::j2k_bandwidth_changed, this));
586         _resolution->Bind       (wxEVT_COMMAND_CHOICE_SELECTED,       boost::bind (&DCPPanel::resolution_changed, this));
587         _three_d->Bind          (wxEVT_COMMAND_CHECKBOX_CLICKED,      boost::bind (&DCPPanel::three_d_changed, this));
588
589         vector<Scaler const *> const sc = Scaler::all ();
590         for (vector<Scaler const *>::const_iterator i = sc.begin(); i != sc.end(); ++i) {
591                 _scaler->Append (std_to_wx ((*i)->name()));
592         }
593
594         vector<Ratio const *> const ratio = Ratio::all ();
595         for (vector<Ratio const *>::const_iterator i = ratio.begin(); i != ratio.end(); ++i) {
596                 _container->Append (std_to_wx ((*i)->nickname ()));
597         }
598
599         list<int> const dfr = Config::instance()->allowed_dcp_frame_rates ();
600         for (list<int>::const_iterator i = dfr.begin(); i != dfr.end(); ++i) {
601                 _frame_rate_choice->Append (std_to_wx (boost::lexical_cast<string> (*i)));
602         }
603
604         _j2k_bandwidth->SetRange (1, Config::instance()->maximum_j2k_bandwidth() / 1000000);
605         _frame_rate_spin->SetRange (1, 480);
606
607         _resolution->Append (_("2K"));
608         _resolution->Append (_("4K"));
609
610         return panel;
611 }
612
613 wxPanel *
614 DCPPanel::make_audio_panel ()
615 {
616         wxPanel* panel = new wxPanel (_notebook);
617         wxSizer* sizer = new wxBoxSizer (wxVERTICAL);
618         wxGridBagSizer* grid = new wxGridBagSizer (DCPOMATIC_SIZER_X_GAP, DCPOMATIC_SIZER_Y_GAP);
619         sizer->Add (grid, 0, wxALL, 8);
620         panel->SetSizer (sizer);
621
622         int r = 0;
623         add_label_to_grid_bag_sizer (grid, panel, _("Channels"), true, wxGBPosition (r, 0));
624         _audio_channels = new wxSpinCtrl (panel, wxID_ANY);
625         grid->Add (_audio_channels, wxGBPosition (r, 1));
626         ++r;
627
628         _audio_channels->Bind (wxEVT_COMMAND_SPINCTRL_UPDATED, boost::bind (&DCPPanel::audio_channels_changed, this));
629
630         _audio_channels->SetRange (0, MAX_DCP_AUDIO_CHANNELS);
631
632         return panel;
633 }