/* This file is part of Patchage. * Copyright (C) 2007-2009 Dave Robillard * * Patchage is free software; you can redistribute it and/or modify it under the * terms of the GNU General Public License as published by the Free Software * Foundation; either version 2 of the License, or (at your option) any later * version. * * Patchage is distributed in the hope that it will be useful, but WITHOUT ANY * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS * FOR A PARTICULAR PURPOSE. See the GNU General Public License for details. * * You should have received a copy of the GNU General Public License along * with this program; if not, write to the Free Software Foundation, Inc., * 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA */ #include "raul/SharedPtr.hpp" #include "patchage-config.h" #include "PatchageCanvas.hpp" #include "Patchage.hpp" #if defined(HAVE_JACK_DBUS) #include "JackDbusDriver.hpp" #elif defined(USE_LIBJACK) #include "JackDriver.hpp" #endif #include "PatchageModule.hpp" #include "PatchagePort.hpp" #ifdef HAVE_ALSA #include "AlsaDriver.hpp" #endif PatchageCanvas::PatchageCanvas(Patchage* app, int width, int height) : FlowCanvas::Canvas(width, height) , _app(app) { } boost::shared_ptr PatchageCanvas::get_item(const string& name) { ItemList::iterator m = _items.begin(); for ( ; m != _items.end(); ++m) if ((*m)->name() == name) break; return (m == _items.end()) ? boost::shared_ptr() : *m; } boost::shared_ptr PatchageCanvas::find_module(const string& name, ModuleType type) { for (ItemList::iterator m = _items.begin(); m != _items.end(); ++m) { boost::shared_ptr pm = boost::dynamic_pointer_cast(*m); if (pm && pm->name() == name && (pm->type() == type || pm->type() == InputOutput)) { return pm; } } return boost::shared_ptr(); } boost::shared_ptr PatchageCanvas::find_port(const PortID& id) { string module_name; string port_name; #if defined(USE_LIBJACK) || defined(HAVE_JACK_DBUS) jack_port_t* jack_port = NULL; #endif SharedPtr module; boost::shared_ptr pp; // TODO: filthy. keep a port map and make this O(log(n)) switch (id.type) { #if defined(USE_LIBJACK) && !defined(HAVE_JACK_DBUS) case PortID::JACK_ID: jack_port = jack_port_by_id(_app->jack_driver()->client(), id.id.jack_id); if (!jack_port) return boost::shared_ptr(); _app->jack_driver()->port_names(id, module_name, port_name); module = find_module(module_name, (jack_port_flags(jack_port) & JackPortIsInput) ? Input : Output); if (module) return PtrCast(module->get_port(port_name)); else return boost::shared_ptr(); break; #endif #ifdef HAVE_ALSA case PortID::ALSA_ADDR: for (ItemList::iterator m = _items.begin(); m != _items.end(); ++m) { SharedPtr module = PtrCast(*m); if (!module) continue; for (PortVector::const_iterator p = module->ports().begin(); p != module->ports().end(); ++p) { pp = boost::dynamic_pointer_cast(*p); if (!pp) continue; if (pp->type() == ALSA_MIDI) { /*cerr << "ALSA PORT: " << (int)pp->alsa_addr()->client << ":" << (int)pp->alsa_addr()->port << endl;*/ if (pp->alsa_addr() && pp->alsa_addr()->client == id.id.alsa_addr.client && pp->alsa_addr()->port == id.id.alsa_addr.port) { if (!id.is_input && module->type() == Input) { //cerr << "WRONG DIRECTION, SKIPPED PORT" << endl; } else { return pp; } } } } } #endif // HAVE_ALSA default: break; } return boost::shared_ptr(); } void PatchageCanvas::connect(boost::shared_ptr port1, boost::shared_ptr port2) { boost::shared_ptr p1 = boost::dynamic_pointer_cast(port1); boost::shared_ptr p2 = boost::dynamic_pointer_cast(port2); if (!p1 || !p2) return; if ((p1->type() == JACK_AUDIO && p2->type() == JACK_AUDIO) || ((p1->type() == JACK_MIDI && p2->type() == JACK_MIDI))) { #if defined(USE_LIBJACK) || defined(HAVE_JACK_DBUS) _app->jack_driver()->connect(p1, p2); #endif #ifdef HAVE_ALSA } else if (p1->type() == ALSA_MIDI && p2->type() == ALSA_MIDI) { _app->alsa_driver()->connect(p1, p2); #endif } else { status_message("WARNING: Cannot make connection, incompatible port types."); } } void PatchageCanvas::disconnect(boost::shared_ptr port1, boost::shared_ptr port2) { boost::shared_ptr input = boost::dynamic_pointer_cast(port1); boost::shared_ptr output = boost::dynamic_pointer_cast(port2); if (!input || !output) return; if (input->is_output() && output->is_input()) { // Damn, guessed wrong boost::shared_ptr swap = input; input = output; output = swap; } if (!input || !output || input->is_output() || output->is_input()) { status_message("ERROR: Attempt to disconnect mismatched/unknown ports"); return; } if ((input->type() == JACK_AUDIO && output->type() == JACK_AUDIO) || (input->type() == JACK_MIDI && output->type() == JACK_MIDI)) { #if defined(USE_LIBJACK) || defined(HAVE_JACK_DBUS) _app->jack_driver()->disconnect(output, input); #endif #ifdef HAVE_ALSA } else if (input->type() == ALSA_MIDI && output->type() == ALSA_MIDI) { _app->alsa_driver()->disconnect(output, input); #endif } else { status_message("ERROR: Attempt to disconnect ports with mismatched types"); } } void PatchageCanvas::status_message(const string& msg) { _app->status_msg(string("[Canvas] ").append(msg)); } boost::shared_ptr PatchageCanvas::get_port(const string& node_name, const string& port_name) { for (ItemList::iterator i = _items.begin(); i != _items.end(); ++i) { const boost::shared_ptr item = *i; const boost::shared_ptr module = boost::dynamic_pointer_cast(item); if (!module) continue; const boost::shared_ptr port = module->get_port(port_name); if (module->name() == node_name && port) return port; } return boost::shared_ptr(); }