[ardour-dev] new feature: export all ranges

Andre Raue dawng at raue.info
Tue Jan 3 02:46:20 PST 2006


Hello,

during the last few weaks I have been implementing a new feature in 
ardour. This makes it possible to export all ranges marked with range 
markers to separate files in a row. Below you will find the following 
information:

1) How is it expected to work / what should it do?
2) limitations / requirements
3) Status of developement
4) Next steps
5) cvs diff against 0.99 for requirements mentioned in 2)
6) cvs diff against 0.99 for the current alpha-version (new files 
excluded, 5) included)
7) diff of new file session_export_dialog.cc against export_dialog.cc
8) diff of new file session_export_dialog.h against export_dialog.h

I would be happy about some constructive criticisim. So - if you have 
some comments or ideas concerning this feature/code, please don't 
hesitate to utter them. In addition it would be great if you could 
implement the changes explained in 2) and 5) in ardour 2.0, as I need 
those for this feature.

If somebody would like to have a complete version of the changed/new 
files, please let me know.  In this case I will provide them for download.

Last but not least I would like to thank Tommi Sakari Uimonen and Taybin 
Rutkin for helping me getting started.


1) How is it expected to work / what should it do?
-------------------------------------------------------
a) mark ranges within ardour as usual
b) select menu item  "Session - Export - Export all ranges..."
c) a dialog similar to the usual export-dialog appears
d) define a destination directory and some export-attributes like 
sampling-rate etc.
e) push start-button
f) all ranges marked by range markers get exported to the destination 
directory. Each range results in a separate file. File names are the 
same as range names.
g) export-dialog gets closed


2) limitations / requirements
-------------------------------
Using the current code, I found no safe way, to decide wether a Location 
object is a range or not. What I found is "enum Flags" in location.h. 
Using these flags you can for example decide if a Location object 
IsMark, IsAutoPunch, ... But there is no flag IsRange. Thus I decided to 
add this flag. This led to changes in the files editor_mouse.cc, 
location_ui.cc and location.h. You can find these changes in 5). These 
changes are the requirements I need for the implemented feature.
The disadvantage of this solution is: Ranges defined with prior versions 
of ardour won't be exported using this feature, as they are not flagged 
as ranges. An idea for solving this problem is to add this flag for all 
Location objects with different start- and end-frame when opening a 
session that has been saved with a prior ardour-version. Is this 
possible? Does the ardour-version get stored in the session file? Are 
ranges the only locations with different start- and end-frame in 0.99?


3) Status of developement
-----------------------------
As I based my development on ardour 0.99, I decided to halt development 
when reaching a proof of concept status until ardour 2.0 will be 
released. This is the point where I am. Thus I concider the current 
version as an alpha-version.
It works but you have to know what you do and it still has some bugs. E.g.:
- the name, that you fill in for the destination-directory must not have 
a '/' at the end
- the destination-directory get's not checked for existence
- all files get the ending '.wav' even if you chose a different file-format
- the progress bar starts at 0% and ends at 100% for each single range 
instead of starting by 0% for the first range and ending at 100% for the 
last one

So, if you find some more bugs, please let me know.


4) Next steps
---------------
After getting access to the 2.0-version of the ExportDialog class I will 
go on with development. My idea is to do some refactoring so that there 
is one base class and three derived classes for export-dialogs. One for 
exporting a region, one for exporting a single range and one for 
exporting all ranges. The bugs mentioned in 3) will be removed. I hope 
to be able to provide a complete version with ardour 2.0 or the version 
that follows 2.0.

Please let me know, when the 2.0-version of the ExportDialog class is ready.


5) cvs diff against 0.99 for requirements mentioned in 2)
--------------------------------------------------------------

Index: gtk_ardour/editor_mouse.cc
===================================================================
RCS file: /home/andre/cvs_repository/Ardour/gtk_ardour/editor_mouse.cc,v
retrieving revision 1.1
retrieving revision 1.3
diff -r1.1 -r1.3
4205c4205
<             newloc = new Location(temp_location->start(), 
temp_location->end(), "unnamed");
---
 >             newloc = new Location(temp_location->start(), 
temp_location->end(), "unnamed", Location::IsRange);
Index: gtk_ardour/location_ui.cc
===================================================================
RCS file: /home/andre/cvs_repository/Ardour/gtk_ardour/location_ui.cc,v
retrieving revision 1.1
retrieving revision 1.4
diff -r1.1 -r1.4
813c813
<         Location *location = new Location (where, where, "unnamed");
---
 >         Location *location = new Location (where, where, "unnamed", 
Location::IsRange);
886d885
<
Index: libs/ardour/ardour/location.h
===================================================================
RCS file: /home/andre/cvs_repository/Ardour/libs/ardour/ardour/location.h,v
retrieving revision 1.1
retrieving revision 1.3
diff -r1.1 -r1.3
53c53,54
<         IsEnd = 0x20
---
 >         IsEnd = 0x20,
 >         IsRange = 0x40
97a99
 >     bool is_range() { return _flags & IsRange; }



6) cvs diff against 0.99 for the current alpha-version (new files excluded)
--------------------------------------------------------------------------------

Index: gtk_ardour/SConscript
===================================================================
RCS file: /home/andre/cvs_repository/Ardour/gtk_ardour/SConscript,v
retrieving revision 1.1
retrieving revision 1.2
diff -r1.1 -r1.2
94a95
 > session_export_dialog.cc
Index: gtk_ardour/ardour_ui_ed.cc
===================================================================
RCS file: /home/andre/cvs_repository/Ardour/gtk_ardour/ardour_ui_ed.cc,v
retrieving revision 1.1
retrieving revision 1.3
diff -r1.1 -r1.3
134a135
 >     export_items.push_back (MenuElem (_("Export all ranges..."), slot 
(*editor, &PublicEditor::export_ranges)));
315d315
<
Index: gtk_ardour/editor.h
===================================================================
RCS file: /home/andre/cvs_repository/Ardour/gtk_ardour/editor.h,v
retrieving revision 1.1
retrieving revision 1.3
diff -r1.1 -r1.3
93a94
 > class SessionExportDialog;
230a232
 >     void export_ranges();
1596a1599
 >     SessionExportDialog *session_export_dialog;
Index: gtk_ardour/editor_export_audio.cc
===================================================================
RCS file: 
/home/andre/cvs_repository/Ardour/gtk_ardour/editor_export_audio.cc,v
retrieving revision 1.1
retrieving revision 1.4
diff -r1.1 -r1.4
23a24
 > #include "session_export_dialog.h"
69a71,84
 > Editor::export_ranges()
 > {
 >     if (session) {
 >         if (session_export_dialog == 0) {
 >             session_export_dialog = new SessionExportDialog(*this);
 >         }
 >        
 >         session_export_dialog->connect_to_session (session);
 >         session_export_dialog->set_range (0, 
session->current_end_frame());
 >         session_export_dialog->start_export();
 >     }
 > }
 >
 > void
Index: gtk_ardour/editor_mouse.cc
===================================================================
RCS file: /home/andre/cvs_repository/Ardour/gtk_ardour/editor_mouse.cc,v
retrieving revision 1.1
retrieving revision 1.3
diff -r1.1 -r1.3
4205c4205
<             newloc = new Location(temp_location->start(), 
temp_location->end(), "unnamed");
---
 >             newloc = new Location(temp_location->start(), 
temp_location->end(), "unnamed", Location::IsRange);
Index: gtk_ardour/location_ui.cc
===================================================================
RCS file: /home/andre/cvs_repository/Ardour/gtk_ardour/location_ui.cc,v
retrieving revision 1.1
retrieving revision 1.4
diff -r1.1 -r1.4
813c813
<         Location *location = new Location (where, where, "unnamed");
---
 >         Location *location = new Location (where, where, "unnamed", 
Location::IsRange);
886d885
<
Index: gtk_ardour/public_editor.h
===================================================================
RCS file: /home/andre/cvs_repository/Ardour/gtk_ardour/public_editor.h,v
retrieving revision 1.1
retrieving revision 1.2
diff -r1.1 -r1.2
82a83
 >     virtual void export_ranges() = 0;
Index: libs/ardour/ardour/location.h
===================================================================
RCS file: /home/andre/cvs_repository/Ardour/libs/ardour/ardour/location.h,v
retrieving revision 1.1
retrieving revision 1.3
diff -r1.1 -r1.3
53c53,54
<         IsEnd = 0x20
---
 >         IsEnd = 0x20,
 >         IsRange = 0x40
97a99
 >     bool is_range() { return _flags & IsRange; }



7) diff of new file session_export_dialog.cc against export_dialog.cc
---------------------------------------------------------------------------

40c40
< #include "export_dialog.h"
---
 > #include "session_export_dialog.h"
51,54c51,56
< GdkPixmap* ExportDialog::check_pixmap = 0;
< GdkPixmap* ExportDialog::check_mask = 0;
< GdkPixmap* ExportDialog::empty_pixmap = 0;
< GdkPixmap* ExportDialog::empty_mask = 0;
---
 > void frames_to_cd_frames_string (char*, jack_nframes_t, jack_nframes_t);
 >
 > GdkPixmap* SessionExportDialog::check_pixmap = 0;
 > GdkPixmap* SessionExportDialog::check_mask = 0;
 > GdkPixmap* SessionExportDialog::empty_pixmap = 0;
 > GdkPixmap* SessionExportDialog::empty_mask = 0;
102c104
< ExportDialog::ExportDialog(PublicEditor& e, AudioRegion* r)
---
 > SessionExportDialog::SessionExportDialog(PublicEditor& e, AudioRegion* r)
159c161
<     master_selector.button_press_event.connect (slot (*this, 
&ExportDialog::master_selector_button_press_event));
---
 >     master_selector.button_press_event.connect (slot (*this, 
&SessionExportDialog::master_selector_button_press_event));
171c173
<     track_selector.button_press_event.connect (slot (*this, 
&ExportDialog::track_selector_button_press_event));
---
 >     track_selector.button_press_event.connect (slot (*this, 
&SessionExportDialog::track_selector_button_press_event));
217c219
<     track_selector_button.clicked.connect (slot (*this, 
&ExportDialog::track_selector_button_click));
---
 >     track_selector_button.clicked.connect (slot (*this, 
&SessionExportDialog::track_selector_button_click));
381,383c383,385
<     delete_event.connect (slot (*this, &ExportDialog::window_closed));
<     ok_button.clicked.connect (slot (*this, &ExportDialog::do_export));
<     cancel_button.clicked.connect (slot (*this, 
&ExportDialog::end_dialog));
---
 >     delete_event.connect (slot (*this, 
&SessionExportDialog::window_closed));
 >     ok_button.clicked.connect (slot (*this, 
&SessionExportDialog::do_export));
 >     cancel_button.clicked.connect (slot (*this, 
&SessionExportDialog::end_dialog));
386c388
<     file_browse_button.clicked.connect (slot (*this, 
&ExportDialog::initiate_browse));
---
 >     file_browse_button.clicked.connect (slot (*this, 
&SessionExportDialog::initiate_browse));
388,392c390,394
<     channel_count_combo.get_popwin()->unmap_event.connect (slot 
(*this, &ExportDialog::channels_chosen));
<     bitdepth_format_combo.get_popwin()->unmap_event.connect (slot 
(*this, &ExportDialog::bitdepth_chosen));
<     header_format_combo.get_popwin()->unmap_event.connect (slot 
(*this, &ExportDialog::header_chosen));
<     sample_rate_combo.get_popwin()->unmap_event.connect (slot (*this, 
&ExportDialog::sample_rate_chosen));
<     cue_file_combo.get_popwin()->unmap_event.connect (slot (*this, 
&ExportDialog::cue_file_type_chosen));
---
 >     channel_count_combo.get_popwin()->unmap_event.connect (slot 
(*this, &SessionExportDialog::channels_chosen));
 >     bitdepth_format_combo.get_popwin()->unmap_event.connect (slot 
(*this, &SessionExportDialog::bitdepth_chosen));
 >     header_format_combo.get_popwin()->unmap_event.connect (slot 
(*this, &SessionExportDialog::header_chosen));
 >     sample_rate_combo.get_popwin()->unmap_event.connect (slot (*this, 
&SessionExportDialog::sample_rate_chosen));
 >     cue_file_combo.get_popwin()->unmap_event.connect (slot (*this, 
&SessionExportDialog::cue_file_type_chosen));
395c397
< ExportDialog::~ExportDialog()
---
 > SessionExportDialog::~SessionExportDialog()
403c405
< ExportDialog::connect_to_session (Session *s)
---
 > SessionExportDialog::connect_to_session (Session *s)
438c440
< ExportDialog::set_state()
---
 > SessionExportDialog::set_state()
554c556
< ExportDialog::save_state()
---
 > SessionExportDialog::save_state()
592c594
< ExportDialog::set_range (jack_nframes_t start, jack_nframes_t end)
---
 > SessionExportDialog::set_range (jack_nframes_t start, jack_nframes_t end)
604c606
< ExportDialog::progress_timeout ()
---
 > SessionExportDialog::progress_timeout ()
611c613
< ExportDialog::_export_region_thread (void *arg)
---
 > SessionExportDialog::_export_region_thread (void *arg)
615c617
<     static_cast<ExportDialog*>(arg)->export_region ();
---
 >     static_cast<SessionExportDialog*>(arg)->export_region ();
620c622
< ExportDialog::export_region ()
---
 > SessionExportDialog::export_region ()
625,639d626
< void
< frames_to_cd_frames_string (char* buf, jack_nframes_t when, 
jack_nframes_t fr)
< {
<
<   long unsigned int remainder;
<   int mins, secs, frames;
<
<     mins = when / (60 * fr);
<     remainder = when - (mins * 60 * fr);
<     secs = remainder / fr;
<     remainder -= secs * fr;
<     frames = remainder / (fr / 75);
<     sprintf (buf, " %02d:%02d:%02d", mins, secs, frames);
<
< }
648c635
< ExportDialog::export_toc_file (Locations::LocationList& locations, 
const string& path)
---
 > SessionExportDialog::export_toc_file (Locations::LocationList& 
locations, const string& path)
771c758
< ExportDialog::export_cue_file (Locations::LocationList& locations, 
const string& path)
---
 > SessionExportDialog::export_cue_file (Locations::LocationList& 
locations, const string& path)
894c881
< ExportDialog::do_export_cd_markers (const string& path,const string& 
cuefile_type)
---
 > SessionExportDialog::do_export_cd_markers (const string& path,const 
string& cuefile_type)
897c884
<         session->locations()->apply (*this, 
&ExportDialog::export_toc_file, path);   
---
 >         session->locations()->apply (*this, 
&SessionExportDialog::export_toc_file, path);   
899c886
<         session->locations()->apply (*this, 
&ExportDialog::export_cue_file, path);
---
 >         session->locations()->apply (*this, 
&SessionExportDialog::export_cue_file, path);
904,906c891,894
< void
< ExportDialog::do_export ()
< {
---
 > /*
 >     initializes spec, so that the export of a range can start
 > */
 > void SessionExportDialog::initSpec(AudioExportSpecification &spec){
908,921d895
<
<     ok_button.set_sensitive(false);
<     save_state();
<
<     if (cue_file_combo.get_entry()->get_text () != _("None")) {
<         do_export_cd_markers (file_entry.get_text(), 
cue_file_combo.get_entry()->get_text ());
<     }
<
<     if (cuefile_only_checkbox.get_active()) {
<         end_dialog ();
<         return;
<     }
<
<     set_modal (true);
923d896
<     spec.path = file_entry.get_text();
1048a1022
 > }
1050,1051c1024,1028
<     progress_connection = Main::timeout.connect (slot (*this, 
&ExportDialog::progress_timeout), 100);
<     cancel_label.set_text (_("Stop Export"));
---
 > void
 > SessionExportDialog::do_export ()
 > {
 >     ok_button.set_sensitive(false);
 >     save_state();
1053,1059c1030,1031
<     if (!audio_region) {
<         if (session->start_audio_export (spec)) {
<             goto out;
<         }
<     } else {
<         pthread_t thr;
<         pthread_create_and_store ("region export", &thr, 0, 
ExportDialog::_export_region_thread, this);
---
 >     if (cue_file_combo.get_entry()->get_text () != _("None")) {
 >         do_export_cd_markers (file_entry.get_text(), 
cue_file_combo.get_entry()->get_text ());
1062,1067c1034,1054
<     gtk_main_iteration ();
<     while (spec.running) {
<         if (gtk_events_pending()) {
<             gtk_main_iteration ();
<         } else {
<             usleep (10000);
---
 >     if (cuefile_only_checkbox.get_active()) {
 >         end_dialog ();
 >         return;
 >     }
 >
 >     set_modal (true);
 >    
 >     progress_connection = Main::timeout.connect (slot (*this, 
&SessionExportDialog::progress_timeout), 100);
 >     cancel_label.set_text (_("Stop Export"));
 >
 >     session->locations()->apply(*this, 
&SessionExportDialog::process_audioranges_export, spec);
 > #if 0
 >     goto out;
 > #endif   
 >    
 >     gtk_main_iteration();
 >     while(spec.running){
 >         if(gtk_events_pending()){
 >             gtk_main_iteration();
 >         }else {
 >             usleep(10000);
1077,1078c1064,1065
< void
< ExportDialog::end_dialog ()
---
 > void
 > 
SessionExportDialog::process_audioranges_export(Locations::LocationList& 
locations, ARDOUR::AudioExportSpecification& spec)
1079a1067,1101
 >     Locations::LocationList::iterator locationIter;
 >
 >     //XXX: ToDo: check somewhere if the given destination path is 
existing
 >     // and if it is a path
 >    
 >     for (locationIter = locations.begin(); locationIter != 
locations.end(); ++locationIter) {
 >         Location *currentLocation = (*locationIter);
 >        
 >         if(currentLocation->is_range()){
 >             //XXX: ToDo: check if file_entry has "/" at the end
 >             initSpec(spec);
 >             spec.path = file_entry.get_text() + '/' + 
currentLocation->name() + ".wav";
 >             spec.start_frame = currentLocation->start();
 >             spec.end_frame = currentLocation->end();
 >
 >             session->request_locate(spec.start_frame, false);
 >
 >             //XXX: ToDo: what to do when start_audio_export fails? 
before it was
 >             // goto out; in method do_export
 >             session->start_audio_export(spec);
 >
 >             // wait until export of this range finished
 >             gtk_main_iteration();
 >             while(spec.running){
 >                 if(gtk_events_pending()){
 >                     gtk_main_iteration();
 >                 }else {
 >                     usleep(10000);
 >                 }
 >             }
 >            
 >             session->engine().freewheel (false);
 >         }
 >     }
 > }
1080a1103,1106
 >
 > void
 > SessionExportDialog::end_dialog ()
 > {
1106c1132
< ExportDialog::start_export ()
---
 > SessionExportDialog::start_export ()
1142c1168
< ExportDialog::header_chosen (GdkEventAny* ignored)
---
 > SessionExportDialog::header_chosen (GdkEventAny* ignored)
1153c1179
< ExportDialog::bitdepth_chosen (GdkEventAny* ignored)
---
 > SessionExportDialog::bitdepth_chosen (GdkEventAny* ignored)
1172c1198
< ExportDialog::cue_file_type_chosen (GdkEventAny* ignored)
---
 > SessionExportDialog::cue_file_type_chosen (GdkEventAny* ignored)
1184c1210
< ExportDialog::sample_rate_chosen (GdkEventAny* ignored)
---
 > SessionExportDialog::sample_rate_chosen (GdkEventAny* ignored)
1215c1241
< ExportDialog::channels_chosen (GdkEventAny* ignored)
---
 > SessionExportDialog::channels_chosen (GdkEventAny* ignored)
1293c1319
< ExportDialog::track_selector_button_press_event (GdkEventButton* ev)
---
 > SessionExportDialog::track_selector_button_press_event 
(GdkEventButton* ev)
1317c1343
< ExportDialog::master_selector_button_press_event (GdkEventButton* ev)
---
 > SessionExportDialog::master_selector_button_press_event 
(GdkEventButton* ev)
1341c1367
< ExportDialog::window_closed (GdkEventAny *ignored)
---
 > SessionExportDialog::window_closed (GdkEventAny *ignored)
1347c1373
< ExportDialog::initiate_browse ()
---
 > SessionExportDialog::initiate_browse ()
1353,1356c1379,1382
<         file_selector->get_cancel_button()->clicked.connect (bind 
(slot (*this, &ExportDialog::finish_browse), -1));
<         file_selector->get_ok_button()->clicked.connect (bind (slot 
(*this, &ExportDialog::finish_browse), 1));
<         file_selector->map_event.connect (bind (slot (*this, 
&ExportDialog::change_focus_policy), true));
<         file_selector->unmap_event.connect (bind (slot (*this, 
&ExportDialog::change_focus_policy), false));
---
 >         file_selector->get_cancel_button()->clicked.connect (bind 
(slot (*this, &SessionExportDialog::finish_browse), -1));
 >         file_selector->get_ok_button()->clicked.connect (bind (slot 
(*this, &SessionExportDialog::finish_browse), 1));
 >         file_selector->map_event.connect (bind (slot (*this, 
&SessionExportDialog::change_focus_policy), true));
 >         file_selector->unmap_event.connect (bind (slot (*this, 
&SessionExportDialog::change_focus_policy), false));
1362c1388
< ExportDialog::change_focus_policy (GdkEventAny *ev, bool yn)
---
 > SessionExportDialog::change_focus_policy (GdkEventAny *ev, bool yn)
1369c1395
< ExportDialog::finish_browse (int status)
---
 > SessionExportDialog::finish_browse (int status)
1384c1410
< ExportDialog::track_selector_button_click ()
---
 > SessionExportDialog::track_selector_button_click ()


8) diff of new file session_export_dialog.h against export_dialog.h
--------------------------------------------------------------------------

20,21c20,21
< #ifndef __ardour_export_dialog_h__
< #define __ardour_export_dialog_h__
---
 > #ifndef __ardour_session_export_dialog_h__
 > #define __ardour_session_export_dialog_h__
37c37
< class ExportDialog : public ArdourDialog
---
 > class SessionExportDialog : public ArdourDialog
40,41c40,41
<     ExportDialog (PublicEditor&, ARDOUR::AudioRegion* r = 0);
<     ~ExportDialog ();
---
 >     SessionExportDialog (PublicEditor&, ARDOUR::AudioRegion* r = 0);
 >     ~SessionExportDialog ();
112c112,113
<           void do_export_cd_markers (const string& path, const string& 
cuefile_type);
---
 >     void do_export_cd_markers (const string& path, const string& 
cuefile_type);
 >     void initSpec(ARDOUR::AudioExportSpecification &spec);
115a117,118
 >     void process_audioranges_export(ARDOUR::Locations::LocationList& 
locations, ARDOUR::AudioExportSpecification& spec);
 >
131c134
< #endif // __ardour_export_dialog_h__
---
 > #endif // __ardour_session_export_dialog_h__


Greetings

Andre


More information about the ardour-dev mailing list