non/mixer/src/Module_Parameter_Editor.C

672 lines
18 KiB
C
Raw Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/*******************************************************************************/
/* 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 <FL/Fl.H>
#include <string.h>
#include <stdlib.h>
#include <stdio.h>
#include <math.h>
#include <FL/fl_draw.H>
#include <FL/Fl_Pack.H>
#include <FL/Fl_Box.H>
#include <FL/Fl_Menu_Button.H>
#include <FL/Fl_Counter.H>
#include "FL/Fl_Flowpack.H"
#include "FL/Fl_Labelpad_Group.H"
#include "FL/Fl_Value_SliderX.H"
#include "FL/Fl_DialX.H"
#include <FL/Fl_Scroll.H>
#include "Module.H"
#include "Module_Parameter_Editor.H"
#include "Controller_Module.H"
#include "Chain.H"
#include "Panner.H"
#include <FL/fl_ask.H>
#include "debug.h"
#include <FL/Fl_Menu_Button.H>
#include "FL/test_press.H"
#include "FL/menu_popup.H"
#include "SpectrumView.H"
#include "string.h"
bool
Module_Parameter_Editor::is_probably_eq ( void )
{
const char *name = _module->label();
return strcasestr( name, "eq" ) ||
strcasestr( name, "filter" ) ||
strcasestr( name, "parametric" ) ||
strcasestr( name, "band" );
}
Module_Parameter_Editor::Module_Parameter_Editor ( Module *module ) : Fl_Double_Window( 900,240)
{
_module = module;
_resized = false;
_min_width = 100;
char lab[256];
if ( strcmp( module->name(), module->label() ) )
{
snprintf( lab, sizeof( lab ), "%s : %s", module->name(), module->label() );
}
else
strcpy( lab, module->label() );
char title[512];
snprintf( title, sizeof( title ), "%s - %s - %s", "Mixer", module->chain()->name(), lab );
copy_label( title );
// fl_font( FL_HELVETICA, 14 );
_min_width = 30 + fl_width( module->label() );
{ Fl_Group *o = new Fl_Group( 0, 0, w(), 25 );
o->label( module->label() );
o->labelfont( 2 );
o->labeltype( FL_SHADOW_LABEL );
o->labelsize( 14 );
o->align( FL_ALIGN_TOP | FL_ALIGN_RIGHT | FL_ALIGN_INSIDE );
{ Fl_Menu_Button *o = mode_choice = new Fl_Menu_Button( 0, 0, 25, 25 );
o->add( "Knobs" );
o->add( "Horizontal Sliders" );
o->add( "Vertical Sliders" );
o->label( NULL );
o->value( 1 );
o->when( FL_WHEN_CHANGED );
o->callback( cb_mode_handle, this );
}
o->resizable(0);
o->end();
}
{ Fl_Scroll *o = control_scroll = new Fl_Scroll( 0, 40, w(), h() - 40 );
{ Fl_Group *o = new Fl_Group( 0, 40, w(), h() - 40 );
{ Fl_Flowpack *o = control_pack = new Fl_Flowpack( 50, 40, w() - 100, h() - 40 );
o->type( FL_HORIZONTAL );
o->flow( true );
o->vspacing( 5 );
o->hspacing( 5 );
o->end();
}
o->resizable( 0 );
o->end();
}
o->end();
}
resizable(control_scroll);
end();
make_controls();
}
Module_Parameter_Editor::~Module_Parameter_Editor ( )
{
}
void
Module_Parameter_Editor::update_spectrum ( void )
{
nframes_t sample_rate = _module->sample_rate();
SpectrumView *o = spectrum_view;
o->sample_rate( sample_rate );
nframes_t nframes = sample_rate / 10;
float *buf = new float[nframes];
memset( buf, 0, sizeof(float) * nframes );
buf[0] = 1;
bool show = false;
if ( ! _module->get_impulse_response( buf, nframes ) )
show = is_probably_eq();
else
show = true;
o->data( buf, nframes );
if ( show && ! o->parent()->visible() )
{
o->parent()->show();
update_control_visibility();
}
o->redraw();
}
void
Module_Parameter_Editor::make_controls ( void )
{
Module *module = _module;
control_pack->clear();
{ SpectrumView *o = spectrum_view = new SpectrumView( 25, 40, 300, 240, "Spectrum" );
o->labelsize(9);
o->align(FL_ALIGN_TOP);
Fl_Labelpad_Group *flg = new Fl_Labelpad_Group( (Fl_Widget*)o );
flg->hide();
control_pack->add( flg );
}
controls_by_port.clear();
/* these are for detecting related parameter groups which can be
better represented by a single control */
azimuth_port_number = -1;
float azimuth_value = 0.0f;
elevation_port_number = -1;
float elevation_value = 0.0f;
radius_port_number = -1;
float radius_value = 0.0f;
Fl_Color fc = fl_color_add_alpha( FL_CYAN, 200 );
Fl_Color bc = FL_BACKGROUND2_COLOR;
controls_by_port.resize( module->control_input.size() );
if ( mode_choice->value() == 1 )
{
control_pack->vspacing( 1 );
control_pack->hspacing( 10 );
control_pack->flow(true);
control_pack->flowdown(true);
control_pack->type( FL_HORIZONTAL );
control_pack->size( 900, 240 );
}
else if ( mode_choice->value() == 2 )
{
control_pack->vspacing( 10 );
control_pack->hspacing( 10 );
control_pack->flow(true);
control_pack->flowdown(false);
control_pack->type( FL_HORIZONTAL );
control_pack->size( 900, 250 );
}
else if ( mode_choice->value() == 0 )
{
control_pack->vspacing( 10 );
control_pack->hspacing( 10 );
control_pack->flow(true);
control_pack->flowdown(true);
control_pack->type( FL_HORIZONTAL );
control_pack->size( 700, 50 );
}
for ( unsigned int i = 0; i < module->control_input.size(); ++i )
{
Fl_Widget *w;
Module::Port *p = &module->control_input[i];
/* if ( !p->hints.visible ) */
/* continue; */
if ( !strcasecmp( "Azimuth", p->name() ) &&
180.0f == p->hints.maximum &&
-180.0f == p->hints.minimum )
{
azimuth_port_number = i;
azimuth_value = p->control_value();
continue;
}
else if ( !strcasecmp( "Elevation", p->name() ) &&
90.0f == p->hints.maximum &&
-90.0f == p->hints.minimum )
{
elevation_port_number = i;
elevation_value = p->control_value();
continue;
}
else if ( !strcasecmp( "Radius", p->name() ) )
{
radius_port_number = i;
radius_value = p->control_value();
continue;
}
if ( p->hints.type == Module::Port::Hints::BOOLEAN )
{
Fl_Button *o = new Fl_Button( 0, 0, 24, 24, p->name() );
w = o;
o->selection_color( fc );
o->type( FL_TOGGLE_BUTTON );
o->value( p->control_value() );
o->align(FL_ALIGN_TOP);
}
else if ( p->hints.type == Module::Port::Hints::INTEGER )
{
Fl_Counter *o = new Fl_Counter(0, 0, 58, 24, p->name() );
w = o;
o->type(1);
o->step(1);
o->align(FL_ALIGN_TOP);
if ( p->hints.ranged )
{
o->minimum( p->hints.minimum );
o->maximum( p->hints.maximum );
}
o->value( p->control_value() );
}
else
{
if ( mode_choice->value() == 0 )
{
Fl_DialX *o = new Fl_DialX( 0, 0, 60, 60, p->name() );
w = o;
if ( p->hints.ranged )
{
DMESSAGE( "Min: %f, max: %f", p->hints.minimum, p->hints.maximum );
o->minimum( p->hints.minimum );
o->maximum( p->hints.maximum );
}
o->color( bc );
o->selection_color( fc );
o->value( p->control_value() );
o->align(FL_ALIGN_TOP);
o->box( FL_DOWN_BOX );
/* a couple of plugins have ridiculously small units */
float r = fabs( p->hints.maximum - p->hints.minimum );
if ( r <= 0.01f )
o->precision( 4 );
else if ( r <= 0.1f )
o->precision( 3 );
else if ( r <= 100.0f )
o->precision( 2 );
else if ( r <= 5000.0f )
o->precision( 1 );
/* else if ( r <= 10000.0f ) */
/* o->precision( 1 ); */
else
o->precision( 0 );
}
else
{
Fl_Value_SliderX *o = new Fl_Value_SliderX( 0, 0, 120, 24, p->name() );
w = o;
if ( mode_choice->value() == 1 )
{
o->type( FL_HORIZONTAL );
o->align( FL_ALIGN_RIGHT );
o->size( 200, 24 );
if ( p->hints.ranged )
{
o->minimum( p->hints.minimum );
o->maximum( p->hints.maximum );
}
}
else
{
o->type( FL_VERTICAL );
o->align(FL_ALIGN_TOP);
o->size( 24, 200 );
/* have to reverse the meaning of these to get the
* orientation of the slider right */
o->maximum( p->hints.minimum );
o->minimum( p->hints.maximum );
}
if ( p->hints.type & Module::Port::Hints::LOGARITHMIC )
o->log(true);
o->precision( 2 );
/* a couple of plugins have ridiculously small units */
float r = fabs( p->hints.maximum - p->hints.minimum );
if ( r <= 0.01f )
o->precision( 4 );
else if ( r <= 0.1f )
o->precision( 3 );
else if ( r <= 100.0f )
o->precision( 2 );
else if ( r <= 5000.0f )
o->precision( 1 );
/* else if ( r <= 10000.0f ) */
/* o->precision( 1 ); */
else
o->precision( 0 );
o->textsize( 8 );
// o->box( FL_NO_BOX );
o->slider( FL_UP_BOX );
o->color( bc );
o->selection_color( fc );
o->value( p->control_value() );
}
}
// w->align(FL_ALIGN_TOP);
w->labelsize( 10 );
controls_by_port[i] = w;
w->copy_tooltip( p->osc_path() );
_callback_data.push_back( callback_data( this, i ) );
if ( p->hints.type == Module::Port::Hints::BOOLEAN )
w->callback( cb_button_handle, &_callback_data.back() );
else
w->callback( cb_value_handle, &_callback_data.back() );
{
Fl_Labelpad_Group *flg = new Fl_Labelpad_Group( w );
flg->set_visible_focus();
control_pack->add( flg );
}
}
if ( azimuth_port_number >= 0 && elevation_port_number >= 0 )
{
Panner *o = new Panner( 0,0, 502,502 );
o->box(FL_FLAT_BOX);
o->color(FL_GRAY0);
o->selection_color(FL_BACKGROUND_COLOR);
o->labeltype(FL_NORMAL_LABEL);
o->labelfont(0);
o->labelcolor(FL_FOREGROUND_COLOR);
o->align(FL_ALIGN_TOP);
o->when(FL_WHEN_CHANGED);
o->label( "Spatialization" );
o->labelsize( 10 );
_callback_data.push_back( callback_data( this, azimuth_port_number, elevation_port_number, radius_port_number ) );
o->callback( cb_panner_value_handle, &_callback_data.back() );
o->point( 0 )->azimuth( azimuth_value );
o->point( 0 )->elevation( elevation_value );
if ( radius_port_number >= 0 )
{
o->point( 0 )->radius_enabled = true;
o->point( 0 )->radius( radius_value );
}
Fl_Labelpad_Group *flg = new Fl_Labelpad_Group( o );
flg->resizable(o);
control_pack->add( flg );
controls_by_port[azimuth_port_number] = o;
controls_by_port[elevation_port_number] = o;
if ( radius_port_number >= 0 )
controls_by_port[radius_port_number] = o;
}
update_spectrum();
update_control_visibility();
}
void
Module_Parameter_Editor::update_control_visibility ( void )
{
for ( unsigned int i = 0; i < _module->control_input.size(); ++i )
{
const Module::Port *p = &_module->control_input[i];
if ( p->hints.visible )
controls_by_port[i]->parent()->show();
else
controls_by_port[i]->parent()->hide();
}
control_pack->dolayout();
int width = control_pack->w() + 100;
int height = control_pack->h() + 60;
if ( width < _min_width )
width = _min_width;
control_pack->parent()->size( control_pack->w() + 100, control_pack->h() );
control_scroll->scroll_to(0, 0 );
size( width, height );
size_range( width, height, width, height );
}
void
Module_Parameter_Editor::cb_value_handle ( Fl_Widget *w, void *v )
{
callback_data *cd = (callback_data*)v;
cd->base_widget->set_value( cd->port_number[0], ((Fl_Valuator*)w)->value() );
}
void
Module_Parameter_Editor::cb_button_handle ( Fl_Widget *w, void *v )
{
callback_data *cd = (callback_data*)v;
cd->base_widget->set_value( cd->port_number[0], ((Fl_Button*)w)->value() );
}
void
Module_Parameter_Editor::cb_panner_value_handle ( Fl_Widget *w, void *v )
{
callback_data *cd = (callback_data*)v;
cd->base_widget->set_value( cd->port_number[0], ((Panner*)w)->point( 0 )->azimuth() );
cd->base_widget->set_value( cd->port_number[1], ((Panner*)w)->point( 0 )->elevation() );
cd->base_widget->set_value( cd->port_number[2], ((Panner*)w)->point( 0 )->radius() );
}
void
Module_Parameter_Editor::cb_mode_handle ( Fl_Widget *, void *v )
{
((Module_Parameter_Editor*)v)->make_controls();
}
void
Module_Parameter_Editor::bind_control ( int i )
{
Module::Port *p = &_module->control_input[i];
if ( p->connected() )
/* can only bind once */
return;
Controller_Module *o = new Controller_Module();
o->label( p->name() );
o->chain( _module->chain() );
o->horizontal( true );
o->connect_to( p );
_module->chain()->add_control( o );
_module->redraw();
}
/* Display changes initiated via automation or from other parts of the GUI */
void
Module_Parameter_Editor::handle_control_changed ( Module::Port *p )
{
int i = _module->control_input_port_index( p );
Fl_Widget *w = controls_by_port[i];
if ( i == azimuth_port_number ||
i == elevation_port_number ||
i == radius_port_number )
{
Panner *_panner = (Panner*)w;
if ( i == azimuth_port_number )
_panner->point(0)->azimuth( p->control_value() );
else if ( i == elevation_port_number )
_panner->point(0)->elevation( p->control_value() );
else if ( i == radius_port_number )
_panner->point(0)->radius( p->control_value() );
_panner->redraw();
return;
}
if ( p->hints.type == Module::Port::Hints::BOOLEAN )
{
Fl_Button *v = (Fl_Button*)w;
v->value( p->control_value() );
}
else
{
Fl_Valuator *v = (Fl_Valuator*)w;
v->value( p->control_value() );
}
update_spectrum();
}
void
Module_Parameter_Editor::reload ( void )
{
// make_controls();
update_control_visibility();
redraw();
}
void
Module_Parameter_Editor::set_value (int i, float value )
{
if ( i >= 0 )
{
_module->control_input[i].control_value( value );
if ( _module->control_input[i].connected() )
_module->control_input[i].connected_port()->module()->handle_control_changed( _module->control_input[i].connected_port() );
}
update_spectrum();
// _module->handle_control_changed( &_module->control_input[i] );
}
void
Module_Parameter_Editor::menu_cb ( Fl_Widget *w, void *v )
{
((Module_Parameter_Editor*)v)->menu_cb((Fl_Menu_*)w);
}
void
Module_Parameter_Editor::menu_cb ( 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 );
if ( ! strcmp( picked, "Bind" ) )
{
bind_control( _selected_control );
}
}
Fl_Menu_Button &
Module_Parameter_Editor::menu ( void ) const
{
static Fl_Menu_Button m( 0, 0, 0, 0, "Control" );
m.clear();
m.add( "Bind", 0, 0, 0, FL_MENU_RADIO | (_module->control_input[_selected_control].connected() ? FL_MENU_VALUE : 0 ));
// m.add( "Unbind", 0, &Module::menu_cb, this, 0, FL_MENU_RADIO );
m.callback( menu_cb, (void*)this );
return m;
}
int
Module_Parameter_Editor::handle ( int m )
{
switch ( m )
{
case FL_PUSH:
if ( test_press( FL_BUTTON3 ) )
{
for ( unsigned int i = 0; i < controls_by_port.size(); i++ )
{
if ( Fl::event_inside( controls_by_port[i] ) )
{
_selected_control = i;
Fl_Menu_Button &m = menu();
menu_popup(&m,Fl::event_x(), Fl::event_y());
return 1;
}
}
return 0;
}
}
return Fl_Group::handle(m);
}