Merge master.
[dcpomatic.git] / src / wx / audio_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 <boost/lexical_cast.hpp>
21 #include <wx/spinctrl.h>
22 #include "lib/config.h"
23 #include "lib/ffmpeg_content.h"
24 #include "lib/ffmpeg_audio_stream.h"
25 #include "lib/audio_processor.h"
26 #include "lib/cinema_sound_processor.h"
27 #include "audio_dialog.h"
28 #include "audio_panel.h"
29 #include "audio_mapping_view.h"
30 #include "wx_util.h"
31 #include "gain_calculator_dialog.h"
32 #include "content_panel.h"
33
34 using std::vector;
35 using std::cout;
36 using std::string;
37 using std::list;
38 using boost::dynamic_pointer_cast;
39 using boost::lexical_cast;
40 using boost::shared_ptr;
41
42 AudioPanel::AudioPanel (ContentPanel* p)
43         : ContentSubPanel (p, _("Audio"))
44         , _audio_dialog (0)
45 {
46         wxGridBagSizer* grid = new wxGridBagSizer (DCPOMATIC_SIZER_X_GAP, DCPOMATIC_SIZER_Y_GAP);
47         _sizer->Add (grid, 0, wxALL, 8);
48
49         int r = 0;
50
51         _show = new wxButton (this, wxID_ANY, _("Show Audio..."));
52         grid->Add (_show, wxGBPosition (r, 0));
53         ++r;
54
55         add_label_to_grid_bag_sizer (grid, this, _("Gain"), true, wxGBPosition (r, 0));
56         _gain = new ContentSpinCtrlDouble<AudioContent> (
57                 this,
58                 new wxSpinCtrlDouble (this),
59                 AudioContentProperty::AUDIO_GAIN,
60                 boost::mem_fn (&AudioContent::audio_gain),
61                 boost::mem_fn (&AudioContent::set_audio_gain)
62                 );
63         
64         _gain->add (grid, wxGBPosition (r, 1));
65         add_label_to_grid_bag_sizer (grid, this, _("dB"), false, wxGBPosition (r, 2));
66         _gain_calculate_button = new wxButton (this, wxID_ANY, _("Calculate..."));
67         grid->Add (_gain_calculate_button, wxGBPosition (r, 3));
68         ++r;
69
70         add_label_to_grid_bag_sizer (grid, this, _("Delay"), true, wxGBPosition (r, 0));
71         _delay = new ContentSpinCtrl<AudioContent> (
72                 this,
73                 new wxSpinCtrl (this),
74                 AudioContentProperty::AUDIO_DELAY,
75                 boost::mem_fn (&AudioContent::audio_delay),
76                 boost::mem_fn (&AudioContent::set_audio_delay)
77                 );
78         
79         _delay->add (grid, wxGBPosition (r, 1));
80         /// TRANSLATORS: this is an abbreviation for milliseconds, the unit of time
81         add_label_to_grid_bag_sizer (grid, this, _("ms"), false, wxGBPosition (r, 2));
82         ++r;
83
84         add_label_to_grid_bag_sizer (grid, this, _("Stream"), true, wxGBPosition (r, 0));
85         _stream = new wxChoice (this, wxID_ANY);
86         grid->Add (_stream, wxGBPosition (r, 1), wxGBSpan (1, 3), wxEXPAND);
87         ++r;
88
89         add_label_to_grid_bag_sizer (grid, this, _("Process with"), true, wxGBPosition (r, 0));
90         _processor = new wxChoice (this, wxID_ANY);
91         setup_processors ();
92         grid->Add (_processor, wxGBPosition (r, 1), wxGBSpan (1, 3), wxEXPAND);
93         ++r;
94         
95         _mapping = new AudioMappingView (this);
96         _sizer->Add (_mapping, 1, wxEXPAND | wxALL, 6);
97
98         _gain->wrapped()->SetRange (-60, 60);
99         _gain->wrapped()->SetDigits (1);
100         _gain->wrapped()->SetIncrement (0.5);
101         _delay->wrapped()->SetRange (-1000, 1000);
102
103         _stream->Bind                (wxEVT_COMMAND_CHOICE_SELECTED, boost::bind (&AudioPanel::stream_changed, this));
104         _show->Bind                  (wxEVT_COMMAND_BUTTON_CLICKED,  boost::bind (&AudioPanel::show_clicked, this));
105         _gain_calculate_button->Bind (wxEVT_COMMAND_BUTTON_CLICKED,  boost::bind (&AudioPanel::gain_calculate_button_clicked, this));
106         _processor->Bind             (wxEVT_COMMAND_CHOICE_SELECTED, boost::bind (&AudioPanel::processor_changed, this));
107
108         _mapping->Changed.connect (boost::bind (&AudioPanel::mapping_changed, this, _1));
109 }
110
111
112 void
113 AudioPanel::film_changed (Film::Property property)
114 {
115         switch (property) {
116         case Film::AUDIO_CHANNELS:
117                 _mapping->set_channels (_parent->film()->audio_channels ());
118                 _sizer->Layout ();
119                 break;
120         default:
121                 break;
122         }
123 }
124
125 void
126 AudioPanel::film_content_changed (int property)
127 {
128         AudioContentList ac = _parent->selected_audio ();
129         shared_ptr<AudioContent> acs;
130         shared_ptr<FFmpegContent> fcs;
131         if (ac.size() == 1) {
132                 acs = ac.front ();
133                 fcs = dynamic_pointer_cast<FFmpegContent> (acs);
134         }
135         
136         if (property == AudioContentProperty::AUDIO_MAPPING) {
137                 _mapping->set (acs ? acs->audio_mapping () : AudioMapping ());
138                 _sizer->Layout ();
139         } else if (property == FFmpegContentProperty::AUDIO_STREAM) {
140                 _mapping->set (acs ? acs->audio_mapping () : AudioMapping ());
141                 _sizer->Layout ();
142         } else if (property == FFmpegContentProperty::AUDIO_STREAMS) {
143                 _stream->Clear ();
144                 if (fcs) {
145                         vector<shared_ptr<FFmpegAudioStream> > a = fcs->audio_streams ();
146                         for (vector<shared_ptr<FFmpegAudioStream> >::iterator i = a.begin(); i != a.end(); ++i) {
147                                 _stream->Append (std_to_wx ((*i)->name), new wxStringClientData (std_to_wx ((*i)->identifier ())));
148                         }
149                         
150                         if (fcs->audio_stream()) {
151                                 checked_set (_stream, fcs->audio_stream()->identifier ());
152                         }
153                 }
154         } else if (property == AudioContentProperty::AUDIO_PROCESSOR) {
155                 if (acs) {
156                         checked_set (_processor, acs->audio_processor() ? acs->audio_processor()->id() : N_("none"));
157                 } else {
158                         checked_set (_processor, N_("none"));
159                 }
160         }
161 }
162
163 void
164 AudioPanel::gain_calculate_button_clicked ()
165 {
166         GainCalculatorDialog* d = new GainCalculatorDialog (this);
167         d->ShowModal ();
168
169         if (d->wanted_fader() == 0 || d->actual_fader() == 0) {
170                 d->Destroy ();
171                 return;
172         }
173         
174         _gain->wrapped()->SetValue (
175                 Config::instance()->cinema_sound_processor()->db_for_fader_change (
176                         d->wanted_fader (),
177                         d->actual_fader ()
178                         )
179                 );
180
181         /* This appears to be necessary, as the change is not signalled,
182            I think.
183         */
184         _gain->view_changed ();
185         
186         d->Destroy ();
187 }
188
189 void
190 AudioPanel::show_clicked ()
191 {
192         if (_audio_dialog) {
193                 _audio_dialog->Destroy ();
194                 _audio_dialog = 0;
195         }
196
197         AudioContentList ac = _parent->selected_audio ();
198         if (ac.size() != 1) {
199                 return;
200         }
201         
202         _audio_dialog = new AudioDialog (this);
203         _audio_dialog->Show ();
204         _audio_dialog->set_content (ac.front ());
205 }
206
207 void
208 AudioPanel::stream_changed ()
209 {
210         FFmpegContentList fc = _parent->selected_ffmpeg ();
211         if (fc.size() != 1) {
212                 return;
213         }
214
215         shared_ptr<FFmpegContent> fcs = fc.front ();
216         
217         if (_stream->GetSelection() == -1) {
218                 return;
219         }
220         
221         vector<shared_ptr<FFmpegAudioStream> > a = fcs->audio_streams ();
222         vector<shared_ptr<FFmpegAudioStream> >::iterator i = a.begin ();
223         string const s = string_client_data (_stream->GetClientObject (_stream->GetSelection ()));
224         while (i != a.end() && (*i)->identifier () != s) {
225                 ++i;
226         }
227
228         if (i != a.end ()) {
229                 fcs->set_audio_stream (*i);
230         }
231 }
232
233 void
234 AudioPanel::processor_changed ()
235 {
236         string const s = string_client_data (_processor->GetClientObject (_processor->GetSelection ()));
237         AudioProcessor const * p = 0;
238         if (s != wx_to_std (N_("none"))) {
239                 p = AudioProcessor::from_id (s);
240         }
241                 
242         AudioContentList c = _parent->selected_audio ();
243         for (AudioContentList::const_iterator i = c.begin(); i != c.end(); ++i) {
244                 (*i)->set_audio_processor (p);
245         }
246 }
247
248 void
249 AudioPanel::mapping_changed (AudioMapping m)
250 {
251         AudioContentList c = _parent->selected_audio ();
252         if (c.size() == 1) {
253                 c.front()->set_audio_mapping (m);
254         }
255 }
256
257 void
258 AudioPanel::content_selection_changed ()
259 {
260         AudioContentList sel = _parent->selected_audio ();
261
262         if (_audio_dialog && sel.size() == 1) {
263                 _audio_dialog->set_content (sel.front ());
264         }
265         
266         _gain->set_content (sel);
267         _delay->set_content (sel);
268
269         _gain_calculate_button->Enable (sel.size() == 1);
270         _show->Enable (sel.size() == 1);
271         _stream->Enable (sel.size() == 1);
272         _processor->Enable (!sel.empty());
273         _mapping->Enable (sel.size() == 1);
274
275         setup_processors ();
276
277         film_content_changed (AudioContentProperty::AUDIO_MAPPING);
278         film_content_changed (AudioContentProperty::AUDIO_PROCESSOR);
279         film_content_changed (FFmpegContentProperty::AUDIO_STREAM);
280         film_content_changed (FFmpegContentProperty::AUDIO_STREAMS);
281 }
282
283 void
284 AudioPanel::setup_processors ()
285 {
286         AudioContentList sel = _parent->selected_audio ();
287
288         _processor->Clear ();
289         list<AudioProcessor const *> ap = AudioProcessor::all ();
290         _processor->Append (_("None"), new wxStringClientData (N_("none")));
291         for (list<AudioProcessor const *>::const_iterator i = ap.begin(); i != ap.end(); ++i) {
292
293                 AudioContentList::const_iterator j = sel.begin();
294                 while (j != sel.end() && (*i)->in_channels().includes ((*j)->audio_channels ())) {
295                         ++j;
296                 }
297
298                 if (j == sel.end ()) {
299                         _processor->Append (std_to_wx ((*i)->name ()), new wxStringClientData (std_to_wx ((*i)->id ())));
300                 }
301         }
302 }