All useage of ArdourPrompter checks for a valid result.
[ardour.git] / gtk2_ardour / sfdb_ui.cc
1 /*
2     Copyright (C) 2005 Paul Davis 
3     Written by Taybin Rutkin
4
5     This program is free software; you can redistribute it and/or modify
6     it under the terms of the GNU General Public License as published by
7     the Free Software Foundation; either version 2 of the License, or
8     (at your option) any later version.
9
10     This program is distributed in the hope that it will be useful,
11     but WITHOUT ANY WARRANTY; without even the implied warranty of
12     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13     GNU General Public License for more details.
14
15     You should have received a copy of the GNU General Public License
16     along with this program; if not, write to the Free Software
17     Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
18
19
20 */
21
22 #include <map>
23 #include <cerrno>
24
25 #include <sndfile.h>
26
27 #include <pbd/basename.h>
28
29 #include <gtkmm/box.h>
30 #include <gtkmm/stock.h>
31
32 #include <ardour/audio_library.h>
33 #include <ardour/audioregion.h>
34 #include <ardour/sndfile_helpers.h>
35 #include <ardour/sndfilesource.h>
36
37 #include "gui_thread.h"
38 #include "prompter.h"
39 #include "sfdb_ui.h"
40
41 #include "i18n.h"
42
43 using namespace ARDOUR;
44
45 std::string length2string (const int32_t frames, const int32_t sample_rate);
46
47 SoundFileBox::SoundFileBox ()
48         :
49         _session(0),
50         current_pid(0),
51         fields(Gtk::ListStore::create(label_columns)),
52         main_box (false, 3),
53         top_box (true, 4),
54         bottom_box (true, 4),
55         play_btn(_("Play")),
56         stop_btn(_("Stop")),
57         add_field_btn(_("Add Field...")),
58         remove_field_btn(_("Remove Field"))
59 {
60         set_name (X_("SoundFileBox"));
61         border_frame.set_label (_("Soundfile Info"));
62         border_frame.add (main_box);
63
64         pack_start (border_frame);
65         set_border_width (4);
66
67         main_box.set_border_width (4);
68
69         main_box.pack_start(length, false, false);
70         main_box.pack_start(format, false, false);
71         main_box.pack_start(channels, false, false);
72         main_box.pack_start(samplerate, false, false);
73         main_box.pack_start(field_view, true, true);
74         main_box.pack_start(top_box, false, false);
75         main_box.pack_start(bottom_box, false, false);
76
77         field_view.set_model (fields);
78         field_view.set_size_request(200, 150);
79         field_view.append_column (_("Field"), label_columns.field);
80         field_view.append_column_editable (_("Value"), label_columns.data);
81
82         top_box.set_homogeneous(true);
83         top_box.pack_start(add_field_btn);
84         top_box.pack_start(remove_field_btn);
85
86         remove_field_btn.set_sensitive(false);
87
88         bottom_box.set_homogeneous(true);
89         bottom_box.pack_start(play_btn);
90         bottom_box.pack_start(stop_btn);
91
92         play_btn.signal_clicked().connect (mem_fun (*this, &SoundFileBox::play_btn_clicked));
93         stop_btn.signal_clicked().connect (mem_fun (*this, &SoundFileBox::stop_btn_clicked));
94
95         add_field_btn.signal_clicked().connect
96                         (mem_fun (*this, &SoundFileBox::add_field_clicked));
97         remove_field_btn.signal_clicked().connect
98                         (mem_fun (*this, &SoundFileBox::remove_field_clicked));
99
100         field_view.get_selection()->signal_changed().connect (mem_fun (*this, &SoundFileBox::field_selected));
101         Library->fields_changed.connect (mem_fun (*this, &SoundFileBox::setup_fields));
102
103         show_all();
104         stop_btn.hide();
105 }
106
107 void
108 SoundFileBox::set_session(Session* s)
109 {
110         _session = s;
111
112     if (!_session) {
113                 play_btn.set_sensitive(false);
114         } else {
115                 _session->AuditionActive.connect(mem_fun (*this, &SoundFileBox::audition_status_changed));
116         }
117 }
118
119 bool
120 SoundFileBox::setup_labels (string filename) 
121 {
122         path = filename;
123
124     SNDFILE *sf;
125
126         sf_info.format = 0; // libsndfile says to clear this before sf_open().
127
128     if ((sf = sf_open ((char *) filename.c_str(), SFM_READ, &sf_info)) < 0) {
129         return false;
130     }
131
132         sf_close (sf);
133
134     if (sf_info.frames == 0 && sf_info.channels == 0 &&
135                 sf_info.samplerate == 0 && sf_info.format == 0 &&
136                 sf_info.sections == 0) {
137                 /* .. ok, it's not a sound file */
138             return false;
139         }
140
141         length.set_alignment (0.0f, 0.0f);
142         length.set_text (string_compose("Length: %1", length2string(sf_info.frames, sf_info.samplerate)));
143
144         format.set_alignment (0.0f, 0.0f);
145         format.set_text (string_compose("Format: %1, %2", 
146                                 sndfile_major_format(sf_info.format),
147                                 sndfile_minor_format(sf_info.format)));
148
149         channels.set_alignment (0.0f, 0.0f);
150         channels.set_text (string_compose("Channels: %1", sf_info.channels));
151
152         samplerate.set_alignment (0.0f, 0.0f);
153         samplerate.set_text (string_compose("Samplerate: %1", sf_info.samplerate));
154
155         setup_fields ();
156
157         return true;
158 }
159
160 void
161 SoundFileBox::setup_fields ()
162 {
163         ENSURE_GUI_THREAD(mem_fun (*this, &SoundFileBox::setup_fields));
164
165         fields->clear ();
166
167         vector<string> field_list;
168         Library->get_fields(field_list);
169
170         vector<string>::iterator i;
171         Gtk::TreeModel::iterator iter;
172         Gtk::TreeModel::Row row;
173         for (i = field_list.begin(); i != field_list.end(); ++i) {
174                 if (!(*i == _("channels") || *i == _("samplerate") ||
175                         *i == _("resolution") || *i == _("format"))) {
176                         iter = fields->append();
177                         row = *iter;
178
179                         string value = Library->get_field(path, *i);
180                         row[label_columns.field] = *i;
181                         row[label_columns.data]  = value;
182                 }
183         }
184 }
185
186 void
187 SoundFileBox::play_btn_clicked ()
188 {
189         if (!_session) {
190                 return;
191         }
192
193         _session->cancel_audition();
194
195         if (access(path.c_str(), R_OK)) {
196                 warning << string_compose(_("Could not read file: %1 (%2)."), path, strerror(errno)) << endmsg;
197                 return;
198         }
199
200         static std::map<string, AudioRegion*> region_cache;
201
202         if (region_cache.find (path) == region_cache.end()) {
203                 AudioRegion::SourceList srclist;
204                 SndFileSource* sfs;
205
206                 for (int n = 0; n < sf_info.channels; ++n) {
207                         try {
208                                 sfs = new SndFileSource(path+":"+string_compose("%1", n), false);
209                                 srclist.push_back(sfs);
210
211                         } catch (failed_constructor& err) {
212                                 error << _("Could not access soundfile: ") << path << endmsg;
213                                 return;
214                         }
215                 }
216
217                 if (srclist.empty()) {
218                         return;
219                 }
220
221                 string result;
222                 _session->region_name (result, PBD::basename(srclist[0]->name()), false);
223                 AudioRegion* a_region = new AudioRegion(srclist, 0, srclist[0]->length(), result, 0, Region::DefaultFlags, false);
224                 region_cache[path] = a_region;
225         }
226
227         play_btn.hide();
228         stop_btn.show();
229
230         _session->audition_region(*region_cache[path]);
231 }
232
233 void
234 SoundFileBox::stop_btn_clicked ()
235 {
236         if (_session) {
237                 _session->cancel_audition();
238                 play_btn.show();
239                 stop_btn.hide();
240         }
241 }
242
243 void
244 SoundFileBox::add_field_clicked ()
245 {
246     ArdourPrompter prompter (true);
247     string name;
248
249     prompter.set_prompt (_("Name for field"));
250
251     switch (prompter.run ()) {
252                 case Gtk::RESPONSE_ACCEPT:
253                 prompter.get_result (name);
254                         if (name.length()) {
255                                 Library->add_field (name);
256                                 Library->save_changes ();
257                         }
258                 break;
259
260             default:
261                 break;
262     }
263 }
264
265 void
266 SoundFileBox::remove_field_clicked ()
267 {}
268
269 void
270 SoundFileBox::audition_status_changed (bool active)
271 {
272     ENSURE_GUI_THREAD(bind (mem_fun (*this, &SoundFileBox::audition_status_changed), active));
273
274         if (!active) {
275                 stop_btn_clicked ();
276         }
277 }
278
279 void
280 SoundFileBox::field_selected ()
281 {}
282
283 SoundFileBrowser::SoundFileBrowser (std::string title)
284         :
285         ArdourDialog(title),
286         chooser(Gtk::FILE_CHOOSER_ACTION_OPEN)
287 {
288         get_vbox()->pack_start(chooser);
289         chooser.set_preview_widget(preview);
290
291         chooser.signal_update_preview().connect(mem_fun(*this, &SoundFileBrowser::update_preview));
292 }
293
294 void
295 SoundFileBrowser::set_session (Session* s)
296 {
297         preview.set_session(s);
298 }
299
300 void
301 SoundFileBrowser::update_preview ()
302 {
303         chooser.set_preview_widget_active(preview.setup_labels(chooser.get_filename()));
304 }
305
306 SoundFileChooser::SoundFileChooser (std::string title)
307         :
308         SoundFileBrowser(title)
309 {
310         add_button (Gtk::Stock::OPEN, Gtk::RESPONSE_OK);
311         add_button (Gtk::Stock::CANCEL, Gtk::RESPONSE_CANCEL);
312
313         show_all ();
314 }
315
316 SoundFileOmega::SoundFileOmega (std::string title)
317         :
318         SoundFileBrowser(title),
319         embed_btn (_("Embed")),
320         import_btn (_("Import")),
321         split_check (_("Split Channels"))
322 {
323         get_action_area()->pack_start(embed_btn);
324         get_action_area()->pack_start(import_btn);
325         add_button (Gtk::Stock::CLOSE, Gtk::RESPONSE_CLOSE);
326
327         chooser.set_extra_widget(split_check);
328
329         embed_btn.signal_clicked().connect (mem_fun (*this, &SoundFileOmega::embed_clicked));
330         import_btn.signal_clicked().connect (mem_fun (*this, &SoundFileOmega::import_clicked));
331
332         show_all ();
333 }
334
335 void
336 SoundFileOmega::embed_clicked ()
337 {
338         Embedded (chooser.get_filenames(), split_check.get_active());
339 }
340
341 void
342 SoundFileOmega::import_clicked ()
343 {
344         Imported (chooser.get_filenames(), split_check.get_active());
345 }
346
347 std::string
348 length2string (const int32_t frames, const int32_t sample_rate)
349 {
350     int secs = (int) (frames / (float) sample_rate);
351     int hrs =  secs / 3600;
352     secs -= (hrs * 3600);
353     int mins = secs / 60;
354     secs -= (mins * 60);
355
356     int total_secs = (hrs * 3600) + (mins * 60) + secs;
357     int frames_remaining = frames - (total_secs * sample_rate);
358     float fractional_secs = (float) frames_remaining / sample_rate;
359
360     char duration_str[32];
361     sprintf (duration_str, "%02d:%02d:%05.2f", hrs, mins, (float) secs + fractional_secs);
362
363     return duration_str;
364 }
365