do not reset session for MonitorSection just because monitor out was removed
[ardour.git] / gtk2_ardour / editor_pt_import.cc
index 79dfa22f3af6e8933d9fa43896608ef39b01badf..12b6484b783f82907651e60ac9cfaa81f9876a11 100644 (file)
@@ -58,6 +58,7 @@
 #include "interthread_progress_window.h"
 #include "mouse_cursors.h"
 #include "editor_cursors.h"
+#include "pt_import_selector.h"
 
 #include "pbd/i18n.h"
 
@@ -69,7 +70,7 @@ using namespace Gtkmm2ext;
 using namespace Editing;
 using std::string;
 
-/* Functions supporting the incorporation of PT sessions into ardour */
+/* Editor dialogs supporting the incorporation of PT sessions into ardour */
 
 void
 Editor::external_pt_dialog ()
@@ -82,254 +83,57 @@ Editor::external_pt_dialog ()
                return;
        }
 
-       Gtk::FileChooserDialog dialog(_("Import PT Session"), FILE_CHOOSER_ACTION_OPEN);
-       dialog.add_button(Gtk::Stock::CANCEL, Gtk::RESPONSE_CANCEL);
-       dialog.add_button(Gtk::Stock::OK, Gtk::RESPONSE_OK);
+       PTImportSelector dialog (import_ptf);
+       dialog.set_session (_session);
 
        while (true) {
-               int result = dialog.run();
+               int result = dialog.run ();
 
-               if (result == Gtk::RESPONSE_OK) {
-                       ptpath = dialog.get_filename ();
+               if (result == Gtk::RESPONSE_ACCEPT) {
 
-                       if (!Glib::file_test (ptpath, Glib::FILE_TEST_IS_DIR|Glib::FILE_TEST_EXISTS)) {
-                               Gtk::MessageDialog msg (string_compose (_("%1: this is only the directory/folder name, not the filename.\n"), ptpath));
-                               msg.run ();
-                               continue;
-                       }
-               }
-
-               if (ptpath.length()) {
-                       do_ptimport(ptpath, SrcBest);
-                       break;
-               }
-
-               if (result == Gtk::RESPONSE_CANCEL) {
-                       break;
-               }
-       }
-}
+                       import_pt_status.all_done = false;
 
-void
-Editor::do_ptimport (std::string ptpath,
-                      SrcQuality  quality)
-{
-       vector<boost::shared_ptr<Region> > regions;
-       boost::shared_ptr<ARDOUR::Track> track;
-       ARDOUR::PluginInfoPtr instrument;
-       vector<string> to_import;
-       string fullpath;
-       bool ok = false;
-       bool onefailed = false;
-       PTFFormat ptf;
-       samplepos_t pos = -1;
+                       ImportProgressWindow ipw (&import_pt_status, _("PT Import"), _("Cancel Import"));
+                       pthread_create_and_store ("import_pt", &import_pt_status.thread, _import_pt_thread, this);
+                       pthread_detach (import_pt_status.thread);
 
-       vector<ptflookup_t> ptfwavpair;
-       vector<ptflookup_t> ptfregpair;
+                       ipw.show();
 
-       if (ptf.load(ptpath, _session->sample_rate()) == -1) {
-               MessageDialog msg (_("Doesn't seem to be a valid PT session file"));
-               msg.run ();
-               return;
-       } else {
-               MessageDialog msg (string_compose (_("PT v%1 Session @ %2Hz\n\n%3 audio files\n%4 regions\n%5 active regions\n\nContinue..."), (int)ptf.version, ptf.sessionrate, ptf.audiofiles.size(), ptf.regions.size(), ptf.tracks.size()));
-               msg.add_button(Gtk::Stock::CANCEL, Gtk::RESPONSE_CANCEL);
-
-               int result = msg.run ();
-               if (result != Gtk::RESPONSE_OK) {
-                       return;
-               }
-       }
-       current_interthread_info = &import_status;
-       import_status.current = 1;
-       import_status.total = ptf.audiofiles.size ();
-       import_status.all_done = false;
-
-       ImportProgressWindow ipw (&import_status, _("Import"), _("Cancel Import"));
-
-       SourceList just_one;
-       SourceList imported;
-
-       for (vector<PTFFormat::wav_t>::iterator a = ptf.audiofiles.begin(); a != ptf.audiofiles.end(); ++a) {
-               ptflookup_t p;
-
-               fullpath = Glib::build_filename (Glib::path_get_dirname(ptpath), "Audio Files");
-               fullpath = Glib::build_filename (fullpath, a->filename);
-               to_import.clear ();
-               to_import.push_back (fullpath);
-               ipw.show ();
-               ok = import_sndfiles (to_import, Editing::ImportDistinctFiles, Editing::ImportAsRegion, quality, pos, 1, -1, track, false, instrument);
-               if (!import_status.sources.empty()) {
-                       p.index1 = a->index;
-                       p.id = import_status.sources.back()->id();
-
-                       ptfwavpair.push_back(p);
-                       imported.push_back(import_status.sources.back());
-               } else {
-                       onefailed = true;
-               }
-       }
-
-       if (onefailed) {
-               MessageDialog msg (_("Failed to load one or more of the audio files, but continuing to attempt import."));
-               msg.run ();
-       } else {
-               MessageDialog msg (_("Success! Import should complete soon."));
-               msg.run ();
-       }
-
-       // Create a dummy midi track first to get a midi Source
-       list<boost::shared_ptr<MidiTrack> > mt (
-               _session->new_midi_track (ChanCount (DataType::MIDI, 1),
-                                         ChanCount (DataType::MIDI, 1),
-                                         true,
-                                         instrument, (Plugin::PresetRecord*) 0,
-                                         (RouteGroup*) 0,
-                                         1,
-                                         string(),
-                                         PresentationInfo::max_order));
-       if (mt.empty()) {
-               return;
-       }
-
-       for (vector<PTFFormat::region_t>::iterator a = ptf.regions.begin();
-                       a != ptf.regions.end(); ++a) {
-               for (vector<ptflookup_t>::iterator p = ptfwavpair.begin();
-                               p != ptfwavpair.end(); ++p) {
-                       if ((p->index1 == a->wave.index) && (strcmp(a->wave.filename.c_str(), "") != 0)) {
-                               for (SourceList::iterator x = imported.begin();
-                                               x != imported.end(); ++x) {
-                                       if ((*x)->id() == p->id) {
-                                               // Matched an uncreated ptf region to ardour region
-                                               ptflookup_t rp;
-                                               PropertyList plist;
-
-                                               plist.add (ARDOUR::Properties::start, a->sampleoffset);
-                                               plist.add (ARDOUR::Properties::position, 0);
-                                               plist.add (ARDOUR::Properties::length, a->length);
-                                               plist.add (ARDOUR::Properties::name, a->name);
-                                               plist.add (ARDOUR::Properties::layer, 0);
-                                               plist.add (ARDOUR::Properties::whole_file, false);
-                                               plist.add (ARDOUR::Properties::external, true);
-
-                                               just_one.clear();
-                                               just_one.push_back(*x);
-
-                                               boost::shared_ptr<Region> r = RegionFactory::create (just_one, plist);
-                                               regions.push_back(r);
-
-                                               rp.id = regions.back()->id();
-                                               rp.index1 = a->index;
-                                               ptfregpair.push_back(rp);
-                                       }
-                               }
+                       while (!import_pt_status.all_done) {
+                               gtk_main_iteration ();
                        }
-               }
-               if (strcmp(a->wave.filename.c_str(), "") == 0) {
-                       /* Empty wave - assume MIDI region */
-                       boost::shared_ptr<MidiTrack> midi_track = mt.back();
-                       boost::shared_ptr<Playlist> playlist = midi_track->playlist();
-                       samplepos_t f = (samplepos_t)a->startpos;
-                       samplecnt_t length = (samplecnt_t)a->length;
-                       MusicSample pos (f, 0);
-                       boost::shared_ptr<Source> src = _session->create_midi_source_by_stealing_name (midi_track);
-                       PropertyList plist;
-                       plist.add (ARDOUR::Properties::start, 0);
-                       plist.add (ARDOUR::Properties::length, length);
-                       plist.add (ARDOUR::Properties::name, PBD::basename_nosuffix(src->name()));
-                       boost::shared_ptr<Region> region = (RegionFactory::create (src, plist));
-                       /* sets beat position */
-                       region->set_position (pos.sample, pos.division);
-                       midi_track->playlist()->add_region (region, pos.sample, 1.0, false, pos.division);
-
-                       boost::shared_ptr<MidiRegion> mr = boost::dynamic_pointer_cast<MidiRegion>(region);
-                       boost::shared_ptr<MidiModel> mm = mr->midi_source(0)->model();
-                       MidiModel::NoteDiffCommand *midicmd;
-                       midicmd = mm->new_note_diff_command ("Import ProTools MIDI");
 
-                       for (vector<PTFFormat::midi_ev_t>::iterator
-                                       j = a->midi.begin();
-                                       j != a->midi.end(); ++j) {
-                               Temporal::Beats start = (Temporal::Beats)(j->pos/960000.);
-                               Temporal::Beats len = (Temporal::Beats)(j->length/960000.);
-                               // PT C-2 = 0, Ardour C-1 = 0, subtract twelve to convert...
-                               midicmd->add(boost::shared_ptr<Evoral::Note<Temporal::Beats> >
-                                       (new Evoral::Note<Temporal::Beats>( (uint8_t)1, start, len, j->note - 12, j->velocity )));
+                       // wait for thread to terminate
+                       while (!import_pt_status.done) {
+                               gtk_main_iteration ();
                        }
-                       mm->apply_command (_session, midicmd);
-                       boost::shared_ptr<Region> copy (RegionFactory::create (mr, true));
-                       playlist->clear_changes ();
-                       playlist->add_region (copy, a->startpos);
-               }
-       }
-
-       boost::shared_ptr<AudioTrack> existing_track;
-       uint16_t nth = 0;
-       vector<ptflookup_t> usedtracks;
-       ptflookup_t utr;
 
-       for (vector<PTFFormat::track_t>::iterator a = ptf.tracks.begin();
-                       a != ptf.tracks.end(); ++a) {
-               for (vector<ptflookup_t>::iterator p = ptfregpair.begin();
-                               p != ptfregpair.end(); ++p) {
-
-                       if (p->index1 == a->reg.index)  {
-                               // Matched a ptf active region to an ardour region
-                               utr.index1 = a->index;
-                               utr.index2 = nth;
-                               utr.id = p->id;
-                               boost::shared_ptr<Region> r = RegionFactory::region_by_id (p->id);
-                               vector<ptflookup_t>::iterator lookuptr = usedtracks.begin();
-                               vector<ptflookup_t>::iterator found;
-                               if ((found = std::find(lookuptr, usedtracks.end(), utr)) != usedtracks.end()) {
-                                       DEBUG_TRACE (DEBUG::FileUtils, string_compose ("\twav(%1) reg(%2) ptf_tr(%3) ard_tr(%4)\n", a->reg.wave.filename.c_str(), a->reg.index, found->index1, found->index2));
-                                       existing_track =  get_nth_selected_audio_track(found->index2);
-                                       // Put on existing track
-                                       boost::shared_ptr<Playlist> playlist = existing_track->playlist();
-                                       boost::shared_ptr<Region> copy (RegionFactory::create (r, true));
-                                       playlist->clear_changes ();
-                                       playlist->add_region (copy, a->reg.startpos);
-                                       //_session->add_command (new StatefulDiffCommand (playlist));
-                               } else {
-                                       // Put on a new track
-                                       DEBUG_TRACE (DEBUG::FileUtils, string_compose ("\twav(%1) reg(%2) new_tr(%3)\n", a->reg.wave.filename.c_str(), a->reg.index, nth));
-                                       list<boost::shared_ptr<AudioTrack> > at (_session->new_audio_track (1, 2, 0, 1, string(), PresentationInfo::max_order, Normal));
-                                       if (at.empty()) {
-                                               return;
-                                       }
-                                       existing_track = at.back();
-                                       std::string trackname;
-                                       try {
-                                               trackname = Glib::convert_with_fallback (a->name, "UTF-8", "UTF-8", "_");
-                                       } catch (Glib::ConvertError& err) {
-                                               trackname = string_compose ("Invalid %1", a->index);
-                                       }
-                                       // TODO legalize track name (no slashes, no colons)
-#if 0 // TODO --  "find_route_name" is currently private
-                                       /* generate a unique name by adding a number if needed */
-                                       uint32_t id = 0;
-                                       if (!_session->find_route_name (trackname.c_str (), id, trackname, false)) {
-                                               fatal << _("PTImport: UINT_MAX routes? impossible!") << endmsg;
-                                               abort(); /*NOTREACHED*/
-                                       }
-#endif
-                                       existing_track->set_name (trackname);
-                                       boost::shared_ptr<Playlist> playlist = existing_track->playlist();
-                                       boost::shared_ptr<Region> copy (RegionFactory::create (r, true));
-                                       playlist->clear_changes ();
-                                       playlist->add_region (copy, a->reg.startpos);
-                                       //_session->add_command (new StatefulDiffCommand (playlist));
-                                       nth++;
-                               }
-                               usedtracks.push_back(utr);
+                       if (import_pt_status.cancel) {
+                               MessageDialog msg (_("PT import may have missing files, check session log for details"));
+                               msg.run ();
+                       } else {
+                               MessageDialog msg (_("PT import complete!"));
+                               msg.run ();
                        }
+                       break;
+               } else if (result == Gtk::RESPONSE_CANCEL) {
+                       break;
                }
        }
+}
 
-       import_status.sources.clear();
+void *
+Editor::_import_pt_thread (void *arg)
+{
+       SessionEvent::create_per_thread_pool ("import pt events", 64);
 
-       if (ok) {
-               _session->save_state ("");
-       }
-       import_status.all_done = true;
+       Editor *ed = (Editor *) arg;
+       return ed->import_pt_thread ();
+}
+
+void *
+Editor::import_pt_thread ()
+{
+       _session->import_pt (import_ptf, import_pt_status);
+       return 0;
 }