non/mixer/src/Mixer.C

1345 lines
32 KiB
C++
Raw Normal View History

2009-12-25 01:59:39 +01:00
/*******************************************************************************/
/* 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. */
/*******************************************************************************/
/* This is the main mixer group. It contains and manages Mixer_Strips. */
2010-01-24 22:54:32 +01:00
#include "const.h"
2009-12-25 01:59:39 +01:00
#include "Mixer.H"
#include "Mixer_Strip.H"
#include <FL/Fl_Pack.H>
#include <FL/Fl_Scroll.H>
#include <FL/Fl_Menu_Bar.H>
2010-01-12 03:31:15 +01:00
#include <FL/fl_ask.H>
#include <FL/Fl.H>
2013-03-15 01:32:15 +01:00
#include <FL/New_Project_Dialog.H>
#include <FL/Fl_Flowpack.H>
#include "Project.H"
2013-03-15 01:32:15 +01:00
#include <FL/Fl_Menu_Settings.H>
#include <FL/About_Dialog.H>
#include <FL/Fl_File_Chooser.H>
2012-11-14 10:54:03 +01:00
#include <FL/Fl_Theme_Chooser.H>
2013-03-15 01:32:15 +01:00
#include <FL/Fl_Value_SliderX.H>
#include <Spatialization_Console.H>
2010-02-01 07:25:16 +01:00
#include "file.h"
2013-08-06 09:03:19 +02:00
#include "Group.H"
2009-12-25 01:59:39 +01:00
#include <string.h>
#include "debug.h"
#include <unistd.h>
#include <sys/types.h>
2009-12-25 01:59:39 +01:00
2010-01-21 01:33:02 +01:00
#include "OSC/Endpoint.H"
#include <lo/lo.h>
#include "Controller_Module.H"
/* const double FEEDBACK_UPDATE_FREQ = 1.0f; */
const double FEEDBACK_UPDATE_FREQ = 1.0f / 30.0f;
extern char *user_config_dir;
extern char *instance_name;
2009-12-25 01:59:39 +01:00
2010-02-01 07:25:16 +01:00
#include "debug.h"
#include "string_util.h"
2009-12-25 01:59:39 +01:00
#include "NSM.H"
#include <FL/Fl_Tooltip.H>
2013-08-06 09:03:19 +02:00
#include "Chain.H"
extern NSM_Client *nsm;
Spatialization_Console *Mixer::spatialization_console = 0;
2010-01-21 01:33:02 +01:00
void
Mixer::show_tooltip ( const char *s )
{
mixer->_status->label( s );
}
void
Mixer::hide_tooltip ( void )
{
mixer->_status->label( 0 );
}
2010-01-21 01:33:02 +01:00
/************************/
/* OSC Message Handlers */
/************************/
#undef OSC_REPLY_OK
#undef OSC_REPLY_ERR
#undef OSC_REPLY
2010-01-21 01:33:02 +01:00
#define OSC_REPLY_OK() ((OSC::Endpoint*)user_data)->send( lo_message_get_source( msg ), path, 0, "OK" )
#define OSC_REPLY( value ) ((OSC::Endpoint*)user_data)->send( lo_message_get_source( msg ), path, value )
#define OSC_REPLY_ERR(errcode, value) ((OSC::Endpoint*)user_data)->send( lo_message_get_source( msg ), path,errcode, value )
2012-02-27 04:35:31 +01:00
#define OSC_ENDPOINT() ((OSC::Endpoint*)user_data)
2010-01-21 01:33:02 +01:00
2012-02-27 04:35:31 +01:00
static int osc_add_strip ( const char *path, const char *, lo_arg **, int , lo_message msg, void *user_data )
2010-01-21 01:33:02 +01:00
{
OSC_DMSG();
Fl::lock();
((Mixer*)(OSC_ENDPOINT())->owner)->command_add_strip();
2010-01-21 01:33:02 +01:00
Fl::unlock();
2010-01-21 01:33:02 +01:00
OSC_REPLY_OK();
return 0;
}
2013-04-09 05:56:33 +02:00
int
Mixer::osc_non_hello ( const char *, const char *, lo_arg **, int , lo_message msg, void * )
{
mixer->handle_hello( msg );
return 0;
}
void
2013-04-09 05:56:33 +02:00
Mixer::handle_hello ( lo_message msg )
{
int argc = lo_message_get_argc( msg );
lo_arg **argv = lo_message_get_argv( msg );
2013-04-09 05:56:33 +02:00
if ( argc >= 4 )
{
const char *url = &argv[0]->s;
const char *name = &argv[1]->s;
const char *version = &argv[2]->s;
const char *id = &argv[3]->s;
2013-04-09 05:56:33 +02:00
MESSAGE( "Got hello from NON peer %s (%s) @ %s with ID \"%s\"", name, version, url, id );
mixer->osc_endpoint->handle_hello( id, url );
}
}
void
Mixer::say_hello ( void )
{
lo_message m = lo_message_new();
lo_message_add( m, "sssss",
"/non/hello",
osc_endpoint->url(),
APP_NAME,
VERSION,
instance_name );
nsm->broadcast( m );
lo_message_free( m );
}
2010-01-21 01:33:02 +01:00
static
Fl_Menu_Item *
find_item( Fl_Menu_ *menu, const char *path )
{
return const_cast<Fl_Menu_Item*>(menu->find_item( path ));
}
void
Mixer::sm_active ( bool b )
{
sm_blinker->value( b );
sm_blinker->tooltip( nsm->session_manager_name() );
if ( b )
{
find_item( menubar, "&Project/&Open" )->deactivate();
find_item( menubar, "&Project/&New" )->deactivate();
}
}
2010-01-12 03:31:15 +01:00
2012-03-03 03:02:23 +01:00
void
Mixer::redraw_windows ( void )
{
window()->redraw();
if ( Fl::first_window() )
for ( Fl_Window *w = Fl::first_window(); ( w = Fl::next_window( w ) ); )
w->redraw();
}
void Mixer::command_new ( void )
{
DMESSAGE( "New project" );
char *default_path = read_line( user_config_dir, "default_path" );
char *result_path = default_path;
char *path = new_project_chooser( &result_path );
if ( path )
{
if ( ! Project::create( path, NULL ) )
fl_alert( "Error creating project!" );
free( path );
}
load_project_settings();
update_menu();
if ( result_path != default_path )
free(default_path);
if ( result_path )
{
write_line( user_config_dir, "default_path", result_path );
free( result_path );
}
}
2012-03-03 03:02:23 +01:00
2010-01-12 03:31:15 +01:00
void Mixer::cb_menu(Fl_Widget* o) {
Fl_Menu_Bar *menu = (Fl_Menu_Bar*)o;
/* const Fl_Menu_Item *mi = &menu->menu()[menu->value()]; */
char picked[256];
// const char *picked = menu->text();
menu->item_pathname( picked, sizeof( picked ) );
DMESSAGE( "Picked %s", picked );
2010-01-12 03:31:15 +01:00
if (! strcmp( picked, "&Project/&New") )
{
command_new();
2010-01-12 03:31:15 +01:00
}
else if (! strcmp( picked, "&Project/&Open" ) )
{
char *path = NULL;
// read_line( user_config_dir, "default_path", &path );
2012-02-27 04:35:31 +01:00
const char *name = fl_dir_chooser( "Open Project", path );
free( path );
mixer->hide();
if ( int err = Project::open( name ) )
{
fl_alert( "Error opening project: %s", Project::errstr( err ) );
}
update_menu();
mixer->show();
}
2010-01-12 03:31:15 +01:00
else if (! strcmp( picked, "&Project/&Save" ) )
{
command_save();
2010-01-12 03:31:15 +01:00
}
else if (! strcmp( picked, "&Project/&Quit") )
2010-01-12 06:31:02 +01:00
{
command_quit();
2010-01-12 06:31:02 +01:00
}
2010-01-12 03:31:15 +01:00
else if ( !strcmp( picked, "&Mixer/&Add Strip" ) )
{
command_add_strip();
2010-01-12 03:31:15 +01:00
}
2010-01-23 06:11:00 +01:00
else if ( !strcmp( picked, "&Mixer/Add &N Strips" ) )
{
const char *s = fl_input( "Enter number of strips to add" );
if ( s )
{
for ( int i = atoi( s ); i > 0; i-- )
command_add_strip();
2010-01-23 06:11:00 +01:00
}
}
else if ( !strcmp( picked, "&Mixer/&Import Strip" ) )
{
const char *s = fl_file_chooser( "Export strip to filename:", "*.strip", NULL, 0 );
if ( s )
{
if (! Mixer_Strip::import_strip( s ) )
fl_alert( "%s", "Failed to import strip!" );
}
}
else if ( ! strcmp( picked, "&Project/Se&ttings/Learn/By Strip Name" ) )
{
Controller_Module::learn_by_number = false;
}
else if ( ! strcmp( picked, "&Project/Se&ttings/Learn/By Strip Number" ) )
{
Controller_Module::learn_by_number = true;
}
2013-07-09 06:53:20 +02:00
else if ( ! strcmp( picked, "&Remote Control/Start Learning" ) )
{
Controller_Module::learn_mode( true );
tooltip( "Now in learn mode. Click on a highlighted control to teach it something." );
redraw();
}
2013-07-09 06:53:20 +02:00
else if ( ! strcmp( picked, "&Remote Control/Stop Learning" ) )
{
Controller_Module::learn_mode( false );
tooltip( "Learning complete" );
redraw();
}
2013-07-09 06:53:20 +02:00
else if ( !strcmp( picked, "&Remote Control/Send State" ) )
{
send_feedback();
}
else if ( ! strcmp( picked, "&Remote Control/Clear All Mappings" ) )
{
2013-07-14 23:12:44 +02:00
if ( 1 == fl_choice( "This will remove all mappings, are you sure?", "No", "Yes", NULL ) )
{
command_clear_mappings();
}
}
else if ( !strcmp( picked, "&Mixer/Paste" ) )
{
Fl::paste(*this);
}
else if (! strcmp( picked, "&Project/Se&ttings/&Rows/One") )
{
rows( 1 );
}
else if (! strcmp( picked, "&Project/Se&ttings/&Rows/Two") )
{
rows( 2 );
}
else if (! strcmp( picked, "&Project/Se&ttings/&Rows/Three") )
{
rows( 3 );
}
else if (! strcmp( picked, "&Mixer/&Spatialization Console") )
{
if ( ! spatialization_console )
{
Spatialization_Console *o = new Spatialization_Console();
spatialization_console = o;
}
if ( ! menu->mvalue()->value() )
spatialization_console->hide();
else
spatialization_console->show();
}
else if (! strcmp( picked, "&Project/Se&ttings/Make Default") )
{
save_default_project_settings();
}
else if (! strcmp( picked, "&View/&Theme") )
{
fl_theme_chooser();
}
2020-12-02 02:46:55 +01:00
else if ( ! strcmp( picked, "&Mixer/Toggle &Fader View" ) )
{
command_toggle_fader_view();
}
2010-01-24 22:54:32 +01:00
else if ( ! strcmp( picked, "&Help/&About" ) )
{
2013-04-14 06:41:39 +02:00
About_Dialog ab( PIXMAP_PATH "/non-mixer/icon-256x256.png" );
2010-02-02 04:54:09 +01:00
ab.logo_box->label( VERSION );
ab.title->label( "Non Mixer" );
2010-02-02 04:54:09 +01:00
ab.copyright->label( "Copyright (C) 2008-2013 Jonathan Moore Liles" );
2010-02-02 04:54:09 +01:00
ab.credits->label(
"Non Mixer was written from scratch by\n"
2010-02-02 04:54:09 +01:00
"Jonathan Moore Liles for his own use\n"
"(see the manual).\n"
"\n"
"Nobody planned. Nobody helped.\n"
"You can help now by donating time, money,\n"
"and/or replacing the rest of Linux Audio\n"
"with fast, light, reliable alternatives.\n" );
ab.website_url->label( "http://non.tuxfamily.org" );
2010-02-02 04:54:09 +01:00
2010-01-24 22:54:32 +01:00
ab.run();
}
else if ( !strcmp( picked, "&Help/&Manual" ))
{
char *pat;
2010-02-01 04:26:12 +01:00
asprintf( &pat, "file://%s.html", DOCUMENT_PATH "/non-mixer/MANUAL" );
open_url( pat );
free( pat );
}
2010-01-12 03:31:15 +01:00
}
void Mixer::cb_menu(Fl_Widget* o, void* v) {
((Mixer*)(v))->cb_menu(o);
}
void Mixer::update_frequency ( float v )
{
_update_interval = 1.0f / v;
2013-06-18 04:54:39 +02:00
Fl::remove_timeout( &Mixer::update_cb, this );
Fl::add_timeout( _update_interval, &Mixer::update_cb, this );
}
void
Mixer::update_cb ( void *v )
{
((Mixer*)v)->update_cb();
}
void
Mixer::update_cb ( void )
{
Fl::repeat_timeout( _update_interval, &Mixer::update_cb, this );
/* if ( active_r() && visible_r() ) */
{
2013-06-18 04:54:39 +02:00
for ( int i = 0; i < mixer_strips->children(); i++ )
{
((Mixer_Strip*)mixer_strips->child(i))->update();
}
}
}
static void
progress_cb ( int p, void *v )
{
static int oldp = 0;
if ( p != oldp )
{
oldp = p;
if ( nsm )
{
nsm->progress( p / 100.0f );
}
Fl::check();
}
}
void
Mixer::save_default_project_settings ( void )
{
char path[256];
snprintf( path, sizeof( path ), "%s/%s", user_config_dir, ".default_project_settings" );
((Fl_Menu_Settings*)menubar)->dump( menubar->find_item( "&Project/Se&ttings" ), path );
}
void
Mixer::load_default_project_settings ( void )
{
char path[256];
snprintf( path, sizeof( path ), "%s/%s", user_config_dir, ".default_project_settings" );
((Fl_Menu_Settings*)menubar)->load( menubar->find_item( "&Project/Se&ttings" ), path );
}
void
Mixer::reset_project_settings ( void )
{
rows(1);
load_default_project_settings();
}
void
Mixer::save_project_settings ( void )
{
if ( ! Project::open() )
return;
((Fl_Menu_Settings*)menubar)->dump( menubar->find_item( "&Project/Se&ttings" ), "options" );
}
void
Mixer::load_project_settings ( void )
{
reset_project_settings();
// if ( Project::open() )
((Fl_Menu_Settings*)menubar)->load( menubar->find_item( "&Project/Se&ttings" ), "options" );
update_menu();
}
2009-12-25 01:59:39 +01:00
Mixer::Mixer ( int X, int Y, int W, int H, const char *L ) :
Fl_Group( X, Y, W, H, L )
{
Loggable::dirty_callback( &Mixer::handle_dirty, this );
Loggable::progress_callback( progress_cb, NULL );
Fl_Tooltip::hoverdelay( 0 );
Fl_Tooltip::delay( 0 );
fl_show_tooltip = &Mixer::show_tooltip;
fl_hide_tooltip = &Mixer::hide_tooltip;
/* Fl_Tooltip::size( 11 ); */
/* Fl_Tooltip::textcolor( FL_FOREGROUND_COLOR ); */
/* Fl_Tooltip::color( fl_color_add_alpha( FL_DARK1, 0 ) ); */
// fl_tooltip_docked = 1;
2013-08-06 09:03:19 +02:00
// _groups.resize(16);
_rows = 1;
_strip_height = 0;
2012-06-20 09:55:58 +02:00
box( FL_FLAT_BOX );
2009-12-26 01:21:40 +01:00
labelsize( 96 );
{ Fl_Group *o = new Fl_Group( X, Y, W, 24 );
{ Fl_Menu_Bar *o = menubar = new Fl_Menu_Bar( X, Y, W, 24 );
o->add( "&Project/&New" );
o->add( "&Project/&Open" );
o->add( "&Project/Se&ttings/&Rows/One", '1', 0, 0, FL_MENU_RADIO | FL_MENU_VALUE );
o->add( "&Project/Se&ttings/&Rows/Two", '2', 0, 0, FL_MENU_RADIO );
o->add( "&Project/Se&ttings/&Rows/Three", '3', 0, 0, FL_MENU_RADIO );
o->add( "&Project/Se&ttings/Learn/By Strip Number", 0, 0, 0, FL_MENU_RADIO );
o->add( "&Project/Se&ttings/Learn/By Strip Name", 0, 0, 0, FL_MENU_RADIO | FL_MENU_VALUE );
2013-07-09 06:53:20 +02:00
o->add( "&Project/Se&ttings/Make Default", 0,0,0);
o->add( "&Project/&Save", FL_CTRL + 's', 0, 0 );
o->add( "&Project/&Quit", FL_CTRL + 'q', 0, 0 );
o->add( "&Mixer/&Add Strip", 'a', 0, 0 );
o->add( "&Mixer/Add &N Strips" );
o->add( "&Mixer/&Import Strip" );
o->add( "&Mixer/Paste", FL_CTRL + 'v', 0, 0 );
o->add( "&Mixer/&Spatialization Console", FL_F + 8, 0, 0, FL_MENU_TOGGLE );
2020-12-02 02:46:55 +01:00
o->add( "&Mixer/Toggle &Fader View", FL_ALT + 'f', 0, 0, FL_MENU_TOGGLE );
// o->add( "&Mixer/&Signal View", FL_ALT + 's', 0, 0, FL_MENU_TOGGLE );
2013-07-09 06:53:20 +02:00
o->add( "&Remote Control/Start Learning", FL_F + 9, 0, 0 );
o->add( "&Remote Control/Stop Learning", FL_F + 10, 0, 0 );
o->add( "&Remote Control/Send State" );
o->add( "&Remote Control/Clear All Mappings", 0, 0, 0 );
o->add( "&View/&Theme", 0, 0, 0 );
o->add( "&Help/&Manual" );
o->add( "&Help/&About" );
o->callback( cb_menu, this );
}
{ Fl_Box *o = project_name = new Fl_Box( X + 150, Y, W, 24 );
o->labelfont( FL_HELVETICA_ITALIC );
o->label( 0 );
o->align( FL_ALIGN_INSIDE | FL_ALIGN_CENTER );
o->labeltype( FL_SHADOW_LABEL );
Fl_Group::current()->resizable( o );
}
{ sm_blinker = new Fl_Button( ( X + W) - 37, Y + 4, 35, 15, "SM");
sm_blinker->box(FL_ROUNDED_BOX);
sm_blinker->down_box(FL_ROUNDED_BOX);
sm_blinker->color(FL_DARK2);
sm_blinker->selection_color((Fl_Color)93);
sm_blinker->labeltype(FL_NORMAL_LABEL);
sm_blinker->labelfont(3);
sm_blinker->labelsize(14);
sm_blinker->labelcolor(FL_DARK3);
sm_blinker->align(Fl_Align(FL_ALIGN_CENTER));
sm_blinker->when(FL_WHEN_RELEASE);
sm_blinker->deactivate();
} // Fl_Blink_Button* sm_blinker
o->end();
2010-02-01 03:54:15 +01:00
}
{ Fl_Scroll *o = scroll = new Fl_Scroll( X, Y + 24, W, H - ( 100 ) );
2012-06-20 09:55:58 +02:00
o->box( FL_FLAT_BOX );
// o->type( Fl_Scroll::HORIZONTAL_ALWAYS );
// o->box( Fl_Scroll::BOTH );
2009-12-25 01:59:39 +01:00
{
Fl_Flowpack *o = mixer_strips = new Fl_Flowpack( X, Y + 24, W, H - ( 18*2 + 24 ));
2012-06-20 09:55:58 +02:00
// label( "Non-Mixer" );
2009-12-26 01:21:40 +01:00
align( (Fl_Align)(FL_ALIGN_CENTER | FL_ALIGN_INSIDE) );
o->flow( false );
2012-06-20 09:55:58 +02:00
o->box( FL_FLAT_BOX );
2020-11-03 03:03:54 +01:00
o->color( fl_darker(FL_BACKGROUND_COLOR ));
2009-12-25 01:59:39 +01:00
o->type( Fl_Pack::HORIZONTAL );
o->hspacing( 2 );
o->vspacing( 2 );
2009-12-25 01:59:39 +01:00
o->end();
Fl_Group::current()->resizable( o );
}
o->end();
Fl_Group::current()->resizable( o );
}
{ Fl_Box *o = _status = new Fl_Box( X, Y + H - 18, W, 18 );
o->align( FL_ALIGN_LEFT | FL_ALIGN_INSIDE );
o->labelsize( 10 );
o->box( FL_FLAT_BOX );
o->color( FL_DARK1 );
}
2009-12-25 01:59:39 +01:00
end();
resize( X,Y,W,H );
2013-07-14 23:36:44 +02:00
update_frequency( 24 );
2009-12-25 01:59:39 +01:00
update_menu();
2009-12-25 01:59:39 +01:00
load_options();
}
2010-01-21 01:33:02 +01:00
/* translate message addressed to strip number to appropriate strip */
int
Mixer::osc_strip_by_number ( const char *path, const char *types, lo_arg **argv, int argc, lo_message msg, void *user_data )
{
int n;
char *rem;
char *client_name;
OSC::Endpoint *ep = (OSC::Endpoint*)user_data;
if ( 3 != sscanf( path, "%m[^/]/strip#/%d/%m[^\n]", &client_name, &n, &rem ) )
return -1;
Mixer_Strip *o = mixer->track_by_number( n );
if ( ! o )
{
DMESSAGE( "No strip by number %i", n );
return 0;
}
char *new_path;
char *stripname = escape_url( o->name() );
asprintf( &new_path, "%s/strip/%s/%s", client_name, stripname, rem );
free( stripname );
/* DMESSAGE( "Forwarding by-number OSC path: %s === %s", path, new_path ); */
free( rem );
lo_send_message( ep->address(), new_path, msg );
free( new_path );
return 0;
}
void
Mixer::load_translations ( void )
{
FILE *fp = fopen( "mappings", "r" );
if ( ! fp )
{
WARNING( "Error opening mappings file for reading" );
return;
}
char *to;
char *from;
while ( 2 == fscanf( fp, "%m[^|> ] |> %m[^ \n]\n", &from, &to ) )
{
osc_endpoint->add_translation( from, to );
free(from);
free(to);
}
fclose( fp );
}
void
Mixer::save_translations ( void )
{
FILE *fp = fopen( "mappings", "w" );
if ( ! fp )
{
WARNING( "Error opening mappings file for writing" );
return;
}
for ( int i = 0; i < osc_endpoint->ntranslations(); i++ )
{
const char *to;
const char *from;
if ( osc_endpoint->get_translation( i, &to, &from ) )
{
fprintf( fp, "%s |> %s\n", to, from );
}
}
fclose( fp );
}
int
Mixer::init_osc ( const char *osc_port )
{
osc_endpoint = new OSC::Endpoint();
2010-01-21 01:33:02 +01:00
if ( int r = osc_endpoint->init( LO_UDP, osc_port ) )
return r;
2010-01-21 01:33:02 +01:00
osc_endpoint->owner = this;
printf( "OSC=%s\n", osc_endpoint->url() );
2010-01-21 01:33:02 +01:00
2013-04-09 05:56:33 +02:00
osc_endpoint->add_method( "/non/hello", "ssss", &Mixer::osc_non_hello, osc_endpoint, "" );
//
2012-02-27 04:35:31 +01:00
osc_endpoint->add_method( "/non/mixer/add_strip", "", osc_add_strip, osc_endpoint, "" );
osc_endpoint->start();
osc_endpoint->add_method( NULL, NULL, osc_strip_by_number, osc_endpoint, "");
return 0;
2010-01-21 01:33:02 +01:00
}
2009-12-25 01:59:39 +01:00
Mixer::~Mixer ( )
{
DMESSAGE( "Destroying mixer" );
save_options();
2013-06-18 04:54:39 +02:00
Fl::remove_timeout( &Mixer::update_cb, this );
Fl::remove_timeout( &Mixer::send_feedback_cb, this );
/* FIXME: teardown */
mixer_strips->clear();
2009-12-25 01:59:39 +01:00
}
2013-08-06 09:03:19 +02:00
void
Mixer::add_group ( Group *g )
{
groups.push_back( g );
for ( int i = mixer_strips->children(); i--; )
((Mixer_Strip*)mixer_strips->child(i))->update_group_choice();
}
void
Mixer::remove_group ( Group *g )
{
groups.remove(g);
for ( int i = mixer_strips->children(); i--; )
((Mixer_Strip*)mixer_strips->child(i))->update_group_choice();
}
2009-12-25 01:59:39 +01:00
void Mixer::resize ( int X, int Y, int W, int H )
{
2009-12-26 01:21:40 +01:00
Fl_Group::resize( X, Y, W, H );
2009-12-25 01:59:39 +01:00
scroll->resize( X, Y + 24, W, H - 24 - 18 );
mixer_strips->resize( X, Y + 24, W, H - (18*2) - 24 );
rows( _rows );
2009-12-25 01:59:39 +01:00
}
void Mixer::add ( Mixer_Strip *ms )
{
MESSAGE( "Add mixer strip \"%s\"", ms->name() );
mixer_strips->add( ms );
ms->size( ms->w(), _strip_height );
ms->redraw();
ms->take_focus();
renumber_strips();
2009-12-25 01:59:39 +01:00
}
int
Mixer::find_strip ( const Mixer_Strip *m ) const
{
return mixer_strips->find( m );
}
2010-01-12 06:31:02 +01:00
void
Mixer::quit ( void )
{
/* TODO: save project? */
while ( Fl::first_window() ) Fl::first_window()->hide();
}
void
Mixer::renumber_strips ( void )
{
for ( int i = mixer_strips->children(); i--; )
{
Mixer_Strip *o = (Mixer_Strip*)mixer_strips->child(i);
o->number( find_strip( o ) );
}
}
2010-01-12 06:31:02 +01:00
void
Mixer::insert ( Mixer_Strip *ms, Mixer_Strip *before )
{
// mixer_strips->remove( ms );
mixer_strips->insert( *ms, before );
renumber_strips();
// scroll->redraw();
}
void
Mixer::insert ( Mixer_Strip *ms, int i )
{
Mixer_Strip *before = (Mixer_Strip*)mixer_strips->child( i );
insert( ms, before);
renumber_strips();
}
void
Mixer::move_left ( Mixer_Strip *ms )
{
int i = mixer_strips->find( ms );
if ( i > 0 )
insert( ms, i - 1 );
renumber_strips();
/* FIXME: do better */
mixer_strips->redraw();
}
void
Mixer::move_right ( Mixer_Strip *ms )
{
int i = mixer_strips->find( ms );
if ( i < mixer_strips->children() - 1 )
insert( ms, i + 2 );
renumber_strips();
/* FIXME: do better */
mixer_strips->redraw();
}
2009-12-25 01:59:39 +01:00
void Mixer::remove ( Mixer_Strip *ms )
{
MESSAGE( "Remove mixer strip \"%s\"", ms->name() );
mixer_strips->remove( ms );
2013-08-06 09:03:19 +02:00
if ( parent() )
parent()->redraw();
renumber_strips();
2009-12-25 01:59:39 +01:00
}
Mixer_Strip *
Mixer::event_inside ( void )
{
for ( int i = mixer_strips->children(); i--; )
if ( Fl::event_inside( mixer_strips->child(i) ) )
return (Mixer_Strip*)mixer_strips->child(i);
return NULL;
}
2009-12-25 01:59:39 +01:00
bool
Mixer::contains ( Mixer_Strip *ms )
{
return ms->parent() == mixer_strips;
}
2010-02-04 14:45:56 +01:00
/* set the ideal number of rows... All may not actually fit. */
void
2010-02-04 14:45:56 +01:00
Mixer::rows ( int ideal_rows )
{
2013-08-06 09:03:19 +02:00
int sh = 0;
int actual_rows = 1;
2010-02-04 14:45:56 +01:00
/* calculate how many rows will actually fit */
int can_fit = scroll->h() / ( Mixer_Strip::min_h() );
actual_rows = can_fit > 0 ? can_fit : 1;
if ( actual_rows > ideal_rows )
actual_rows = ideal_rows;
/* calculate strip height */
if ( actual_rows > 1 )
{
sh = ( scroll->h() / (float)actual_rows ) - ( mixer_strips->vspacing() * ( actual_rows - 2 ));
mixer_strips->flow(true);
}
else
actual_rows = 1;
if ( 1 == actual_rows )
2010-01-25 03:12:10 +01:00
{
sh = (scroll->h() - 18);
mixer_strips->flow( false );
2010-01-25 03:12:10 +01:00
}
int tw = 0;
for ( int i = 0; i < mixer_strips->children(); ++i )
{
Mixer_Strip *t = (Mixer_Strip*)mixer_strips->child( i );
t->size( t->w(), sh );
tw += t->w() + mixer_strips->hspacing();
}
2010-02-04 14:45:56 +01:00
if ( actual_rows > 1 )
mixer_strips->size( scroll->w() - 18, mixer_strips->h() );
else
mixer_strips->size( tw, sh );
2010-02-04 14:45:56 +01:00
_rows = ideal_rows;
2020-09-25 02:58:55 +02:00
if ( _strip_height != sh )
{
mixer_strips->redraw();
scroll->redraw();
_strip_height = sh;
}
}
int
Mixer::nstrips ( void ) const
{
return mixer_strips->children();
}
2009-12-25 01:59:39 +01:00
/** retrun a pointer to the track named /name/, or NULL if no track is named /name/ */
Mixer_Strip *
Mixer::track_by_name ( const char *name )
{
for ( int i = mixer_strips->children(); i-- ; )
{
Mixer_Strip *t = (Mixer_Strip*)mixer_strips->child( i );
if ( ! strcmp( name, t->name() ) )
return t;
}
return NULL;
}
/** retrun a pointer to the track named /name/, or NULL if no track is named /name/ */
Mixer_Strip *
Mixer::track_by_number ( int n )
{
if ( n < 0 || n >= mixer_strips->children() )
return NULL;
return (Mixer_Strip*)mixer_strips->child(n);
}
2009-12-25 01:59:39 +01:00
/** return a malloc'd string representing a unique name for a new track */
char *
Mixer::get_unique_track_name ( const char *name )
{
char pat[256];
strcpy( pat, name );
for ( int i = 1; track_by_name( pat ); ++i )
snprintf( pat, sizeof( pat ), "%s.%d", name, i );
return strdup( pat );
}
2013-08-06 09:03:19 +02:00
Group *
Mixer::group_by_name ( const char *name )
{
for ( std::list<Group*>::iterator i = groups.begin();
i != groups.end();
i++ )
if ( !strcmp( (*i)->name(), name ))
return *i;
return NULL;
}
char *
Mixer::get_unique_group_name ( const char *name )
{
char pat[256];
strcpy( pat, name );
for ( int i = 1; group_by_name( pat ); ++i )
snprintf( pat, sizeof( pat ), "%s.%d", name, i );
return strdup( pat );
}
void
Mixer::handle_dirty ( int d, void *v )
{
//Mixer *m = (Mixer*)v;
if ( !nsm )
return;
if ( d == 1 )
nsm->is_dirty();
else if ( d == 0 )
nsm->is_clean();
}
2009-12-25 01:59:39 +01:00
void
Mixer::snapshot ( void )
{
if ( spatialization_console )
spatialization_console->log_create();
2013-08-06 09:03:19 +02:00
for ( std::list<Group*>::iterator i = groups.begin(); i != groups.end(); ++i )
(*i)->log_create();
2009-12-25 01:59:39 +01:00
for ( int i = 0; i < mixer_strips->children(); ++i )
((Mixer_Strip*)mixer_strips->child( i ))->log_children();
2009-12-25 01:59:39 +01:00
}
void
Mixer::new_strip ( void )
{
2010-01-25 03:12:10 +01:00
add( new Mixer_Strip( get_unique_track_name( "Unnamed" ) ) );
2009-12-25 01:59:39 +01:00
}
bool
Mixer::save ( void )
{
MESSAGE( "Saving state" );
Loggable::snapshot_callback( &Mixer::snapshot, this );
2010-01-12 03:31:15 +01:00
Loggable::snapshot( "snapshot" );
save_translations();
return true;
}
static const char options_filename[] = "options";
void
Mixer::load_options ( void )
{
// save options
/* char *path; */
/* asprintf( &path, "%s/options", user_config_dir ); */
/* ((Fl_Menu_Settings*)menubar)->load( menubar->find_item( "&Options" ), path ); */
/* free( path ); */
}
void
Mixer::save_options ( void )
{
/* char *path; */
/* asprintf( &path, "%s/%s", user_config_dir, options_filename ); */
/* ((Fl_Menu_Settings*)menubar)->dump( menubar->find_item( "&Options" ), path ); */
/* free( path ); */
}
void
Mixer::update_menu ( void )
{
2010-02-01 03:54:15 +01:00
project_name->label( Project::name() );
const_cast<Fl_Menu_Item*>(menubar->find_item( "&Mixer/&Spatialization Console" ))
->flags = FL_MENU_TOGGLE | ( ( spatialization_console && spatialization_console->shown() ) ? FL_MENU_VALUE : 0 );
}
void
Mixer::send_feedback_cb ( void *v )
{
Mixer *m = (Mixer*)v;
m->send_feedback();
/* just to it once at the start... */
Fl::repeat_timeout( FEEDBACK_UPDATE_FREQ, send_feedback_cb, v );
}
/** unconditionally send feedback to all mapped controls. This is
* useful for updating the state of an external controller. */
void
Mixer::send_feedback ( void )
{
for ( int i = 0; i < mixer_strips->children(); i++ )
{
((Mixer_Strip*)mixer_strips->child(i))->send_feedback();
}
}
int
Mixer::handle ( int m )
{
/* if user presses certain keys when project is loading it can cause a crash. Don't respond to input. */
if ( Project::is_opening() )
return 0;
if ( Fl_Group::handle( m ) )
return 1;
switch ( m )
{
case FL_PASTE:
{
if ( ! Fl::event_inside( this ) )
return 0;
2020-11-03 03:03:54 +01:00
DMESSAGE( "Got paste into mixer, expecting strip file..." );
const char *text = Fl::event_text();
char *file;
if ( ! sscanf( text, "file://%m[^\r\n]\n", &file ) )
{
WARNING( "invalid drop \"%s\"\n", text );
return 0;
}
unescape_url( file );
printf( "pasted file \"%s\"\n", file );
if (! Mixer_Strip::import_strip( file ) )
fl_alert( "%s", "Failed to import strip!" );
return 1;
}
}
return 0;
}
#include <algorithm>
std::list <std::string>
Mixer::get_auto_connect_targets ( void )
{
std::list <std::string> sl;
std::list <std::string> rl;
for ( int i = mixer_strips->children(); i--; )
{
((Mixer_Strip*)mixer_strips->child(i))->get_output_ports(sl);
}
for ( std::list<std::string>::iterator i = sl.begin(); i != sl.end(); i++ )
{
char *s = strdup( i->c_str() );
*rindex( s, '/' ) = 0;
if ( !index( s, '/' ) )
{
char *o;
asprintf( &o, "%s/mains", s );
free(s);
s = o;
}
if ( std::find( rl.begin(), rl.end(), s ) == rl.end() )
{
rl.push_back( s );
}
free(s);
}
return rl;
}
void
Mixer::auto_connect ( void )
{
if ( Project::is_opening() )
/* it's more efficient to do this once at the end rather than as we go. */
return;
DMESSAGE("Full auto-connect cycle" );
/* give strips with group affinity the first shot */
for ( int i = 0; i < mixer_strips->children(); i++ )
{
Mixer_Strip *s = ((Mixer_Strip*)mixer_strips->child(i));
if ( s->has_group_affinity() )
s->auto_connect_outputs();
}
/* now do that catch-alls, first one wins! */
for ( int i = 0; i < mixer_strips->children(); i++ )
{
Mixer_Strip *s = ((Mixer_Strip*)mixer_strips->child(i));
if ( ! s->has_group_affinity() )
s->auto_connect_outputs();
}
}
void
Mixer::maybe_auto_connect_output ( Module::Port *p )
{
if ( Project::is_opening() )
/* it's more efficient to do this once at the end rather than as we go. */
return;
// DMESSAGE( "Single output auto connect cycle" );
/* give strips with group affinity the first shot */
for ( int i = 0; i < mixer_strips->children(); i++ )
{
Mixer_Strip *s = ((Mixer_Strip*)mixer_strips->child(i));
if ( s->has_group_affinity() )
if ( s->maybe_auto_connect_output( p ) )
return;
}
/* now do that catch-alls, first one wins! */
for ( int i = 0; i < mixer_strips->children(); i++ )
{
Mixer_Strip *s = ((Mixer_Strip*)mixer_strips->child(i));
if ( ! s->has_group_affinity() )
if ( s->maybe_auto_connect_output( p ) )
return;
}
}
/************/
/* Commands */
/************/
void
Mixer::command_toggle_fader_view ( void )
{
for ( int i = 0; i < mixer_strips->children(); i++ )
{
Mixer_Strip *s = ((Mixer_Strip*)mixer_strips->child(i));
s->command_toggle_fader_view();
}
}
void
Mixer::command_clear_mappings ( void )
{
osc_endpoint->clear_translations();
}
bool
Mixer::command_save ( void )
{
if ( ! Project::open() )
{
command_new();
update_menu();
}
save_project_settings();
return Project::save();
}
bool
Mixer::command_load ( const char *path, const char *display_name )
{
mixer->deactivate();
Project::close();
char *pwd = (char*)malloc( PATH_MAX + 1 );
getcwd( pwd, PATH_MAX );
chdir( path );
load_project_settings();
chdir( pwd );
free( pwd );
2013-08-06 09:03:19 +02:00
2012-02-27 04:35:31 +01:00
if ( Project::open( path ) )
{
// fl_alert( "Error opening project specified on commandline: %s", Project::errstr( err ) );
return false;
}
if ( display_name )
Project::name( display_name );
load_translations();
update_menu();
auto_connect();
mixer->activate();
/* Fl::remove_timeout( send_feedback_cb, this ); */
Fl::add_timeout( FEEDBACK_UPDATE_FREQ, send_feedback_cb, this );
return true;
}
bool
Mixer::command_new ( const char *path, const char *display_name )
{
if ( ! Project::create( path, "" ) )
return false;
if ( display_name )
Project::name( display_name );
load_project_settings();
update_menu();
return true;
// fl_alert( "Error creating project!" );
}
void
Mixer::command_quit ( void )
{
if ( Loggable::dirty() )
{
int i = fl_choice( "There have been changes since the last save. Quitting now will discard them", "Discard", "Cancel", NULL );
if ( i != 0 )
return;
}
quit();
}
/* */
void
Mixer::command_add_strip ( void )
{
new_strip();
}