/*******************************************************************************/ /* Copyright (C) 2009 Jonathan Moore Liles */ /* */ /* This program 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. */ /* */ /* This program 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 */ /* more details. */ /* */ /* You should have received a copy of the GNU General Public License along */ /* with This program; see the file COPYING. If not,write to the Free Software */ /* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ /*******************************************************************************/ #include "Module.H" #include #include #include #include #include #include "Module_Parameter_Editor.H" #include "Chain.H" #include "JACK_Module.H" #include "Gain_Module.H" #include "Mono_Pan_Module.H" #include "Meter_Module.H" #include "Plugin_Module.H" #include "AUX_Module.H" #include #include "FL/test_press.H" #include "FL/menu_popup.H" #include "Mixer.H" #include "Plugin_Chooser.H" #include "OSC/Endpoint.H" #include "string_util.h" nframes_t Module::_sample_rate = 0; Module *Module::_copied_module_empty = 0; char *Module::_copied_module_settings = 0; Module::Module ( int W, int H, const char *L ) : Fl_Group( 0, 0, W, H, L ) { init(); } Module::Module ( bool is_default, int W, int H, const char *L ) : Fl_Group( 0, 0, W, H, L ), Loggable( !is_default ) { init(); this->is_default( is_default ); } Module::Module ( ) : Fl_Group( 0, 0, 50, 50, "Unnamed" ) { init(); } Module::~Module ( ) { /* we assume that the engine for this chain is already locked */ if ( _editor ) { delete _editor; _editor = NULL; } for ( unsigned int i = 0; i < audio_input.size(); ++i ) audio_input[i].disconnect(); for ( unsigned int i = 0; i < audio_output.size(); ++i ) audio_output[i].disconnect(); for ( unsigned int i = 0; i < control_input.size(); ++i ) { /* destroy connected Controller_Module */ if ( control_input[i].connected() ) { Module *o = (Module*)control_input[i].connected_port()->module(); if ( ! o->is_default() ) { control_input[i].disconnect(); DMESSAGE( "Deleting connected module %s", o->label() ); delete o; } else { control_input[i].disconnect(); } } control_input[i].destroy_osc_port(); } for ( unsigned int i = 0; i < control_output.size(); ++i ) control_output[i].disconnect(); audio_input.clear(); audio_output.clear(); control_input.clear(); control_output.clear(); if ( parent() ) parent()->remove( this ); } void Module::init ( void ) { _is_default = false; _editor = 0; _chain = 0; _instances = 1; _bypass = 0; box( FL_UP_BOX ); labeltype( FL_NO_LABEL ); align( FL_ALIGN_CENTER | FL_ALIGN_INSIDE ); set_visible_focus(); selection_color( FL_RED ); color( fl_color_average( FL_WHITE, FL_CYAN, 0.40 ) ); } void Module::get ( Log_Entry &e ) const { // e.add( ":name", label() ); // e.add( ":color", (unsigned long)color()); { char *s = get_parameters(); if ( strlen( s ) ) e.add( ":parameter_values", s ); delete[] s; } e.add( ":is_default", is_default() ); e.add( ":chain", chain() ); e.add( ":active", ! bypass() ); } void Module::copy ( void ) const { Module *m = clone_empty(); if ( ! m ) { DMESSAGE( "Module \"%s\" doesn't support cloning", name() ); } Log_Entry *ne = new Log_Entry(); _copied_module_empty = m; { Log_Entry e; get( e ); for ( int i = 0; i < e.size(); ++i ) { const char *s, *v; e.get( i, &s, &v ); /* we don't want this module to get added to the current chain... */ if ( !( !strcmp( s, ":chain" ) || !strcmp( s, ":is_default" ) ) ) { DMESSAGE( "%s = %s", s, v ); ne->add_raw( s, v ); } } } _copied_module_settings = ne->print(); } void Module::paste_before ( void ) { Module *m = _copied_module_empty; Log_Entry le( _copied_module_settings ); le.remove( ":chain" ); char *print = le.print(); DMESSAGE( "Pasting settings: %s", print ); free( print ); m->set( le ); if ( ! chain()->insert( this, m ) ) { fl_alert( "Copied module cannot be inserted at this point in the chain" ); } free( _copied_module_settings ); _copied_module_settings = NULL; _copied_module_empty = NULL; /* set up for another paste */ m->copy(); } char * Module::Port::osc_number_path ( void ) { int n = _module->chain()->strip()->number(); char *rem; char *client_name; char *strip_name; if ( 3 != sscanf( _scaled_signal->path(), "%a[^/]/strip/%a[^/]/%a[^\n]", &client_name, &strip_name, &rem ) ) return NULL; free( strip_name ); char *path; asprintf( &path, "%s/strip#/%i/%s", client_name, n, rem ); free( client_name ); free( rem ); return path; } void Module::Port::send_feedback ( void ) { float f = control_value(); if ( hints.ranged ) { // scale value to range. float scale = hints.maximum - hints.minimum; float offset = hints.minimum; f = ( f - offset ) / scale; } if ( f > 1.0 ) f = 1.0; else if ( f < 0.0 ) f = 0.0; if ( _scaled_signal ) { /* send feedback for by_name signal */ mixer->osc_endpoint->send_feedback( _scaled_signal->path(), f ); /* send feedback for by number signal */ { char *path = osc_number_path(); mixer->osc_endpoint->send_feedback( path, f ); free(path); } } } void Module::send_feedback ( void ) { for ( int i = 0; i < ncontrol_inputs(); i++ ) control_input[i].send_feedback(); } void Module::handle_control_changed ( Port *p ) { if ( _editor ) _editor->handle_control_changed ( p ); p->send_feedback(); } bool Module::Port::connected_osc ( void ) const { if ( _scaled_signal ) return _scaled_signal->connected(); else return false; } void Module::Port::learn_osc ( void ) { _scaled_signal->learn_connection(); } char * Module::Port::generate_osc_path () { const Port *p = this; char *path = NULL; // /strip/STRIPNAME/MODULENAME/CONTROLNAME if ( ! p->hints.visible ) { return NULL; } int n = module()->chain()->get_module_instance_number( module() ); if ( n > 0 ) asprintf( &path, "/strip/%s/%s.%i/%s", module()->chain()->name(), p->module()->label(), n, p->name() ); else asprintf( &path, "/strip/%s/%s/%s", module()->chain()->name(), p->module()->label(), p->name() ); char *s = escape_url( path ); free( path ); path = s; return path; } void Module::Port::handle_signal_connection_state_changed ( OSC::Signal *, void *o ) { ((Module::Port*)o)->module()->redraw(); } void Module::Port::change_osc_path ( char *path ) { if ( path ) { char *scaled_path = path; char *unscaled_path = NULL; asprintf( &unscaled_path, "%s/unscaled", path ); if ( NULL == _scaled_signal ) { float scaled_default = 0.5f; if ( hints.ranged ) { float scale = hints.maximum - hints.minimum; float offset = hints.minimum; scaled_default = ( hints.default_value - offset ) / scale; } _scaled_signal = mixer->osc_endpoint->add_signal( scaled_path, OSC::Signal::Input, 0.0, 1.0, scaled_default, &Module::Port::osc_control_change_cv, this ); _scaled_signal->connection_state_callback( handle_signal_connection_state_changed, this ); _unscaled_signal = mixer->osc_endpoint->add_signal( unscaled_path, OSC::Signal::Input, hints.minimum, hints.maximum, hints.default_value, &Module::Port::osc_control_change_exact, this ); } else { DMESSAGE( "Renaming OSC signals" ); _scaled_signal->rename( scaled_path ); _unscaled_signal->rename( unscaled_path ); } free( unscaled_path ); /* this was path, it's ok to free because it was malloc()'d in generate_osc_path */ free( scaled_path ); } } int Module::Port::osc_control_change_exact ( float v, void *user_data ) { Module::Port *p = (Module::Port*)user_data; Fl::lock(); float f = v; if ( p->hints.ranged ) { if ( f > p->hints.maximum ) f = p->hints.maximum; else if ( f < p->hints.minimum ) f = p->hints.minimum; } p->control_value( f ); Fl::unlock(); // mixer->osc_endpoint->send( lo_message_get_source( msg ), "/reply", path, f ); return 0; } int Module::Port::osc_control_change_cv ( float v, void *user_data ) { Module::Port *p = (Module::Port*)user_data; float f = v; Fl::lock(); // clamp value to control voltage range. if ( f > 1.0 ) f = 1.0; else if ( f < 0.0 ) f = 0.0; if ( p->hints.ranged ) { // scale value to range. float scale = p->hints.maximum - p->hints.minimum; float offset = p->hints.minimum; f = ( f * scale ) + offset; } p->control_value( f ); Fl::unlock(); // mixer->osc_endpoint->send( lo_message_get_source( msg ), "/reply", path, f ); return 0; } void Module::set ( Log_Entry &e ) { for ( int i = 0; i < e.size(); ++i ) { const char *s, *v; e.get( i, &s, &v ); if ( ! ( strcmp( s, ":is_default" ) ) ) { is_default( atoi( v ) ); } else if ( ! strcmp( s, ":chain" ) ) { /* This trickiness is because we may need to know the name of our chain before we actually get added to it. */ int i; sscanf( v, "%X", &i ); Chain *t = (Chain*)Loggable::find( i ); assert( t ); chain( t ); } } for ( int i = 0; i < e.size(); ++i ) { const char *s, *v; e.get( i, &s, &v ); /* if ( ! strcmp( s, ":name" ) ) */ /* label( v ); */ if ( ! strcmp( s, ":parameter_values" ) ) { set_parameters( v ); } else if ( ! ( strcmp( s, ":active" ) ) ) { bypass( ! atoi( v ) ); } else if ( ! strcmp( s, ":chain" ) ) { int i; sscanf( v, "%X", &i ); Chain *t = (Chain*)Loggable::find( i ); assert( t ); t->add( this ); } } } void Module::chain ( Chain *v ) { if ( _chain != v ) { DMESSAGE( "Adding module %s in to chain %s", label(), v ? v->name() : "NULL" ); _chain = v; for ( int i = 0; i < ncontrol_inputs(); ++i ) { control_input[i].update_osc_port(); } } else { DMESSAGE( "Module %s already belongs to chain %s", label(), v ? v->name() : "NULL" ); } } /* return a string serializing this module's parameter settings. The format is 1.0:2.0:... Where 1.0 is the value of the first control input, 2.0 is the value of the second control input etc. */ char * Module::get_parameters ( void ) const { char *s = new char[1024]; s[0] = 0; char *sp = s; if ( control_input.size() ) { for ( unsigned int i = 0; i < control_input.size(); ++i ) sp += snprintf( sp, 1024 - (sp - s),"%f:", control_input[i].control_value() ); *(sp - 1) = '\0'; } return s; } void Module::set_parameters ( const char *parameters ) { char *s = strdup( parameters ); char *start = s; unsigned int i = 0; for ( char *sp = s; ; ++sp ) { if ( ':' == *sp || '\0' == *sp ) { char was = *sp; *sp = '\0'; DMESSAGE( start ); if ( i < control_input.size() ) control_input[i].control_value( atof( start ) ); else { WARNING( "Module has no parameter at index %i", i ); break; } i++; if ( '\0' == was ) break; start = sp + 1; } } free( s ); } void Module::draw_box ( int tx, int ty, int tw, int th ) { fl_color( fl_contrast( FL_FOREGROUND_COLOR, color() ) ); fl_push_clip( tx, ty, tw, th ); Fl_Color c = color(); c = active() && ! bypass() ? c : FL_GRAY; int spacing = w() / instances(); for ( int i = instances(); i--; ) { fl_draw_box( box(), tx + (spacing * i), ty, tw / instances(), th, c ); } if ( this == Fl::focus() ) { fl_draw_box( FL_UP_FRAME, x(), y(), w(), h(), selection_color() ); } if ( audio_input.size() && audio_output.size() ) { /* maybe draw control indicators */ if ( control_input.size() ) { fl_draw_box( FL_ROUNDED_BOX, tx + 4, ty + 4, 5, 5, is_being_controlled() ? FL_YELLOW : fl_inactive( FL_YELLOW ) ); fl_draw_box( FL_ROUNDED_BOX, tx + 4, ty + th - 8, 5, 5, is_being_controlled_osc() ? FL_YELLOW : fl_inactive( FL_YELLOW ) ); } if ( control_output.size() ) fl_draw_box( FL_ROUNDED_BOX, tx + tw - 8, ty + 4, 5, 5, is_controlling() ? FL_YELLOW : fl_inactive( FL_YELLOW ) ); } Fl_Group::draw_children(); fl_pop_clip(); } void Module::draw_label ( int tx, int ty, int tw, int th ) { bbox( tx, ty, tw, th ); const char *lp = label(); Fl_Color c = fl_contrast( FL_FOREGROUND_COLOR, bypass() ? FL_BLACK : color() ); fl_color( active_r() ? c : fl_inactive(c) ); fl_font( FL_HELVETICA, 12 ); int LW = fl_width( lp ); char *s = NULL; if ( LW > tw ) { s = new char[strlen(lp)]; char *sp = s; for ( ; *lp; ++lp ) switch ( *lp ) { case 'i': case 'e': case 'o': case 'u': case 'a': break; default: *(sp++) = *lp; } *sp = '\0'; } fl_draw( s ? s : lp, tx, ty, tw, th, align() | FL_ALIGN_CLIP ); if ( s ) delete[] s; } void Module::insert_menu_cb ( const Fl_Menu_ *m ) { const char * picked = m->mvalue()->label(); DMESSAGE("picked = %s", picked ); Module *mod = NULL; if ( !strcmp( picked, "Aux" ) ) { int n = 0; for ( int i = 0; i < chain()->modules(); i++ ) { if ( !strcmp( chain()->module(i)->name(), "AUX" ) ) n++; } AUX_Module *jm = new AUX_Module(); jm->chain( chain() ); jm->number( n ); jm->configure_inputs( ninputs() ); jm->configure_outputs( ninputs() ); jm->initialize(); mod = jm; } else if ( !strcmp( picked, "Gain" ) ) mod = new Gain_Module(); else if ( !strcmp( picked, "Meter" ) ) mod = new Meter_Module(); else if ( !strcmp( picked, "Mono Pan" )) mod = new Mono_Pan_Module(); else if ( !strcmp(picked, "Plugin" )) { unsigned long id = Plugin_Chooser::plugin_chooser( this->ninputs() ); if ( id == 0 ) return; Plugin_Module *m = new Plugin_Module(); m->load( id ); mod = m; } if ( mod ) { if ( ! chain()->insert( this, mod ) ) { fl_alert( "Cannot insert this module at this point in the chain" ); delete mod; return; } redraw(); } } void Module::insert_menu_cb ( Fl_Widget *w, void *v ) { ((Module*)v)->insert_menu_cb( (Fl_Menu_*) w ); } void Module::menu_cb ( const Fl_Menu_ *m ) { char picked[256]; if ( ! m->mvalue() || m->mvalue()->flags & FL_SUBMENU_POINTER || m->mvalue()->flags & FL_SUBMENU ) return; strncpy( picked, m->mvalue()->label(), sizeof( picked ) ); // m->item_pathname( picked, sizeof( picked ) ); DMESSAGE( "%s", picked ); Logger log( this ); if ( ! strcmp( picked, "Edit Parameters" ) ) command_open_parameter_editor(); else if ( ! strcmp( picked, "Bypass" ) ) if ( ! bypassable() ) { fl_alert( "Due to its channel configuration, this module cannot be bypassed." ); } else { bypass( ! ( m->mvalue()->flags & FL_MENU_VALUE ) ); } else if ( ! strcmp( picked, "Cut" ) ) { copy(); chain()->remove( this ); Fl::delete_widget( this ); } else if ( ! strcmp( picked, "Copy" ) ) { copy(); } else if ( ! strcmp( picked, "Paste" ) ) { paste_before(); } else if ( ! strcmp( picked, "Remove" ) ) command_remove(); } void Module::menu_cb ( Fl_Widget *w, void *v ) { ((Module*)v)->menu_cb( (Fl_Menu_*) w ); } /** build the context menu */ Fl_Menu_Button & Module::menu ( void ) const { static Fl_Menu_Button m( 0, 0, 0, 0, "Module" ); static Fl_Menu_Button *insert_menu = NULL; if ( ! insert_menu ) { insert_menu = new Fl_Menu_Button( 0, 0, 0, 0 ); insert_menu->add( "Gain", 0, 0 ); insert_menu->add( "Meter", 0, 0 ); insert_menu->add( "Mono Pan", 0, 0 ); insert_menu->add( "Aux", 0, 0 ); insert_menu->add( "Plugin", 0, 0 ); insert_menu->callback( &Module::insert_menu_cb, (void*)this ); } m.clear(); m.add( "Insert", 0, &Module::menu_cb, (void*)this, 0); m.add( "Insert", 0, &Module::menu_cb, const_cast< Fl_Menu_Item *>( insert_menu->menu() ), FL_SUBMENU_POINTER ); m.add( "Edit Parameters", ' ', &Module::menu_cb, (void*)this, 0 ); m.add( "Bypass", 'b', &Module::menu_cb, (void*)this, FL_MENU_TOGGLE | ( bypass() ? FL_MENU_VALUE : 0 ) ); m.add( "Cut", FL_CTRL + 'x', &Module::menu_cb, (void*)this, is_default() ? FL_MENU_INACTIVE : 0 ); m.add( "Copy", FL_CTRL + 'c', &Module::menu_cb, (void*)this, is_default() ? FL_MENU_INACTIVE : 0 ); m.add( "Paste", FL_CTRL + 'v', &Module::menu_cb, (void*)this, _copied_module_empty ? 0 : FL_MENU_INACTIVE ); m.add( "Remove", FL_Delete, &Module::menu_cb, (void*)this ); // menu_set_callback( menu, &Module::menu_cb, (void*)this ); m.callback( &Module::insert_menu_cb, (void*)this ); return m; } void Module::handle_chain_name_changed ( ) { // pass it along to our connected Controller_Modules, if any. for ( int i = 0; i < ncontrol_inputs(); ++i ) { if ( control_input[i].connected() ) control_input[i].connected_port()->module()->handle_chain_name_changed(); control_input[i].update_osc_port(); } } int Module::handle ( int m ) { static unsigned long _event_state = 0; unsigned long evstate = Fl::event_state(); if ( Fl_Group::handle( m ) ) return 1; switch ( m ) { case FL_KEYBOARD: { if ( Fl::event_key() == FL_Menu ) { menu_popup( &menu(), x(), y() ); return 1; } else return menu().test_shortcut() != 0; } case FL_PUSH: take_focus(); _event_state = evstate; return 1; // if ( Fl::visible_focus() && handle( FL_FOCUS )) Fl::focus(this); case FL_DRAG: _event_state = evstate; return 1; case FL_RELEASE: { unsigned long e = _event_state; _event_state = 0; if ( ! Fl::event_inside( this ) ) return 1; if ( e & FL_BUTTON1 ) { command_open_parameter_editor(); return 1; } else if ( e & FL_BUTTON3 && e & FL_CTRL ) { command_remove(); return 1; } else if ( e & FL_BUTTON3 ) { menu_popup( &menu() ); return 1; } else if ( e & FL_BUTTON2 ) { if ( !bypassable() ) { fl_alert( "Due to its channel configuration, this module cannot be bypassed." ); } else { bypass( !bypass() ); redraw(); } return 1; } /* else */ /* { */ /* take_focus(); */ /* } */ return 0; } case FL_FOCUS: case FL_UNFOCUS: redraw(); return 1; } return 0; } /************/ /* Commands */ /************/ void Module::command_open_parameter_editor ( void ) { if ( _editor ) { _editor->show(); } else if ( ncontrol_inputs() && nvisible_control_inputs() ) { DMESSAGE( "Opening module parameters for \"%s\"", label() ); _editor = new Module_Parameter_Editor( this ); _editor->show(); } } void Module::command_activate ( void ) { bypass( false ); } void Module::command_deactivate ( void ) { bypass( true ); } void Module::command_remove ( void ) { if ( is_default() ) fl_alert( "Default modules may not be deleted." ); else { chain()->remove( this ); Fl::delete_widget( this ); } }