2009-12-25 01:59:39 +01:00
|
|
|
|
|
|
|
/*******************************************************************************/
|
|
|
|
/* Copyright (C) 2008 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 "Panner.H"
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <math.h>
|
|
|
|
|
|
|
|
/* 2D Panner widget. Supports various multichannel configurations. */
|
|
|
|
|
2010-02-13 18:24:37 +01:00
|
|
|
Panner::Point *Panner::drag;
|
2009-12-25 01:59:39 +01:00
|
|
|
|
|
|
|
/* multichannel layouts, in degrees */
|
|
|
|
int Panner::_configs[][12] =
|
|
|
|
{
|
|
|
|
/* none, error condition? */
|
|
|
|
{ NONE },
|
|
|
|
/* mono, panner disabled */
|
|
|
|
{ NONE },
|
|
|
|
/* stereo */
|
|
|
|
{ L, R },
|
|
|
|
/* stereo + mono */
|
|
|
|
{ L, R, C },
|
|
|
|
/* quad */
|
|
|
|
{ FL, FR, RL, RR },
|
|
|
|
/* 5.1 */
|
|
|
|
{ FL, FR, RL, RR, C },
|
|
|
|
/* no such config */
|
|
|
|
{ NONE },
|
|
|
|
/* 7.1 */
|
|
|
|
{ FL, FR, RL, RR, C, L, R },
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/* speaker symbol */
|
|
|
|
#define BP fl_begin_polygon()
|
|
|
|
#define EP fl_end_polygon()
|
|
|
|
#define BCP fl_begin_complex_polygon()
|
|
|
|
#define ECP fl_end_complex_polygon()
|
|
|
|
#define BL fl_begin_line()
|
|
|
|
#define EL fl_end_line()
|
|
|
|
#define BC fl_begin_loop()
|
|
|
|
#define EC fl_end_loop()
|
|
|
|
#define vv(x,y) fl_vertex(x,y)
|
|
|
|
|
|
|
|
static void draw_speaker ( Fl_Color col )
|
|
|
|
{
|
|
|
|
fl_color(col);
|
|
|
|
|
|
|
|
BP; vv(0.2,0.4); vv(0.6,0.4); vv(0.6,-0.4); vv(0.2,-0.4); EP;
|
|
|
|
BP; vv(-0.6,0.8); vv(0.2,0.0); vv(-0.6,-0.8); EP;
|
|
|
|
|
|
|
|
fl_color( fl_darker( col ) );
|
|
|
|
|
|
|
|
BC; vv(0.2,0.4); vv(0.6,0.4); vv(0.6,-0.4); vv(0.2,-0.4); EC;
|
|
|
|
BC; vv(-0.6,0.8); vv(0.2,0.0); vv(-0.6,-0.8); EC;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/** set X, Y, W, and H to the bounding box of point /p/ in screen coords */
|
|
|
|
void
|
|
|
|
Panner::point_bbox ( const Point *p, int *X, int *Y, int *W, int *H ) const
|
|
|
|
{
|
|
|
|
int tx, ty, tw, th;
|
|
|
|
|
|
|
|
bbox( tx, ty, tw, th );
|
|
|
|
|
|
|
|
tw -= pw();
|
|
|
|
th -= ph();
|
|
|
|
|
|
|
|
float px, py;
|
|
|
|
|
|
|
|
p->axes( &px, &py );
|
|
|
|
|
|
|
|
*X = tx + ((tw / 2) * px + (tw / 2));
|
|
|
|
*Y = ty + ((th / 2) * py + (th / 2));
|
|
|
|
|
|
|
|
*W = pw();
|
|
|
|
*H = ph();
|
|
|
|
}
|
|
|
|
|
|
|
|
Panner::Point *
|
|
|
|
Panner::event_point ( void )
|
|
|
|
{
|
|
|
|
for ( int i = _ins; i--; )
|
|
|
|
{
|
|
|
|
int px, py, pw, ph;
|
|
|
|
|
|
|
|
Point *p = &_points[ i ];
|
|
|
|
|
|
|
|
point_bbox( p, &px, &py, &pw, &ph );
|
|
|
|
|
|
|
|
// printf( "%d, %d -- %d %d %d %d\n", Fl::event_x(), Fl::event_y(), px, py, pw, ph );
|
|
|
|
|
|
|
|
if ( Fl::event_inside( px, py, pw, ph ) )
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Panner::draw ( void )
|
|
|
|
{
|
|
|
|
draw_box();
|
|
|
|
// draw_box( FL_FLAT_BOX, x(), y(), w(), h(), FL_BLACK );
|
|
|
|
draw_label();
|
|
|
|
|
|
|
|
|
|
|
|
if ( _bypassed )
|
|
|
|
{
|
|
|
|
fl_color( 0 );
|
|
|
|
fl_font( FL_HELVETICA, 12 );
|
|
|
|
fl_draw( "(bypass)", x(), y(), w(), h(), FL_ALIGN_CENTER );
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
int tw, th, tx, ty;
|
|
|
|
|
|
|
|
bbox( tx, ty, tw, th );
|
|
|
|
|
|
|
|
fl_push_clip( tx, ty, tw, th );
|
|
|
|
|
2010-02-13 18:24:37 +01:00
|
|
|
fl_color( FL_RED );
|
2009-12-25 01:59:39 +01:00
|
|
|
|
|
|
|
const int b = 10;
|
|
|
|
|
|
|
|
tx += b;
|
|
|
|
ty += b;
|
|
|
|
tw -= b * 2;
|
|
|
|
th -= b * 2;
|
|
|
|
|
2010-02-12 11:20:43 +01:00
|
|
|
/* draw perimeter */
|
|
|
|
{
|
2010-02-13 18:24:37 +01:00
|
|
|
Fl_Color c1, c2;
|
|
|
|
int iter;
|
|
|
|
|
|
|
|
if ( Fl::belowmouse() == this )
|
|
|
|
{
|
|
|
|
iter = 12;
|
|
|
|
c1 = fl_darker( FL_RED );
|
|
|
|
c2 = FL_GRAY;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
iter = 6;
|
|
|
|
c1 = FL_GRAY;
|
|
|
|
c2 = FL_BLACK;
|
|
|
|
}
|
|
|
|
|
|
|
|
Fl_Color c = c1;
|
2010-02-12 11:20:43 +01:00
|
|
|
|
|
|
|
for ( int i = iter; i--; )
|
|
|
|
{
|
|
|
|
fl_color( c );
|
|
|
|
|
|
|
|
fl_arc( tx + (i * (tw / iter)) / 2, ty + (i * (th / iter)) / 2, tw - (i * (tw / iter)), th - (i * ( th / iter )), 0, 360 );
|
|
|
|
|
2010-02-13 18:24:37 +01:00
|
|
|
c = fl_color_average( c1, c2, (float)i / iter);
|
2010-02-12 11:20:43 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* fl_color( FL_WHITE ); */
|
|
|
|
|
|
|
|
/* fl_arc( tx, ty, tw, th, 0, 360 ); */
|
2009-12-25 01:59:39 +01:00
|
|
|
|
|
|
|
if ( _configs[ _outs ][0] >= 0 )
|
|
|
|
{
|
|
|
|
for ( int i = _outs; i--; )
|
|
|
|
{
|
|
|
|
int a = _configs[ _outs ][ i ];
|
|
|
|
|
|
|
|
Point p( 1.2f, (float)a );
|
|
|
|
|
|
|
|
float px, py;
|
|
|
|
|
|
|
|
p.axes( &px, &py );
|
|
|
|
|
|
|
|
fl_push_matrix();
|
|
|
|
|
|
|
|
const int bx = tx + ((tw / 2) * px + (tw / 2));
|
|
|
|
const int by = ty + ((th / 2) * py + (th / 2));
|
|
|
|
|
|
|
|
fl_translate( bx, by );
|
|
|
|
|
|
|
|
fl_scale( 5, 5 );
|
|
|
|
|
|
|
|
a = 90 - a;
|
|
|
|
|
|
|
|
fl_rotate( a );
|
|
|
|
|
|
|
|
draw_speaker( FL_WHITE );
|
|
|
|
|
|
|
|
fl_rotate( -a );
|
|
|
|
|
|
|
|
fl_pop_matrix();
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ensure that points are drawn *inside* the circle */
|
|
|
|
|
|
|
|
for ( int i = _ins; i--; )
|
|
|
|
{
|
|
|
|
Point *p = &_points[ i ];
|
|
|
|
|
|
|
|
Fl_Color c = (Fl_Color)(10 + i);
|
|
|
|
|
|
|
|
int px, py, pw, ph;
|
|
|
|
point_bbox( p, &px, &py, &pw, &ph );
|
|
|
|
|
|
|
|
/* draw point */
|
2010-02-13 18:24:37 +01:00
|
|
|
if ( p != drag )
|
|
|
|
fl_color( c );
|
|
|
|
else
|
|
|
|
fl_color( FL_WHITE );
|
|
|
|
|
2009-12-25 01:59:39 +01:00
|
|
|
fl_pie( px, py, pw, ph, 0, 360 );
|
|
|
|
|
|
|
|
/* draw echo */
|
|
|
|
fl_color( c = fl_darker( c ) );
|
|
|
|
fl_arc( px - 5, py - 5, pw + 10, ph + 10, 0, 360 );
|
2010-02-13 18:24:37 +01:00
|
|
|
if ( Fl::belowmouse() == this )
|
|
|
|
{
|
|
|
|
fl_color( c = fl_darker( c ) );
|
|
|
|
fl_arc( px - 10, py - 10, pw + 20, ph + 20, 0, 360 );
|
|
|
|
fl_color( c = fl_darker( c ) );
|
|
|
|
fl_arc( px - 30, py - 30, pw + 60, ph + 60, 0, 360 );
|
|
|
|
}
|
2009-12-25 01:59:39 +01:00
|
|
|
|
|
|
|
/* draw number */
|
|
|
|
char pat[4];
|
|
|
|
snprintf( pat, 4, "%d", i + 1 );
|
|
|
|
|
|
|
|
fl_color( FL_BLACK );
|
|
|
|
fl_font( FL_HELVETICA, ph + 2 );
|
|
|
|
fl_draw( pat, px + 1, py + 1, pw - 1, ph - 1, FL_ALIGN_CENTER );
|
|
|
|
|
|
|
|
/* draw line */
|
|
|
|
|
|
|
|
/* fl_color( FL_WHITE ); */
|
|
|
|
/* fl_line( bx + pw() / 2, by + ph() / 2, tx + (tw / 2), ty + (th / 2) ); */
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
fl_pop_clip();
|
|
|
|
}
|
|
|
|
|
|
|
|
/* return the current gain setting for the path in/out */
|
2010-02-14 22:20:50 +01:00
|
|
|
Panner::Point *
|
2009-12-25 01:59:39 +01:00
|
|
|
Panner::point( int i )
|
|
|
|
{
|
2010-02-14 22:20:50 +01:00
|
|
|
return &_points[ i ];
|
2009-12-25 01:59:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
Panner::handle ( int m )
|
|
|
|
{
|
|
|
|
int r = Fl_Widget::handle( m );
|
|
|
|
|
|
|
|
switch ( m )
|
|
|
|
{
|
2010-02-13 18:24:37 +01:00
|
|
|
case FL_ENTER:
|
|
|
|
case FL_LEAVE:
|
|
|
|
redraw();
|
|
|
|
return 1;
|
2009-12-25 01:59:39 +01:00
|
|
|
case FL_PUSH:
|
|
|
|
|
|
|
|
if ( Fl::event_button2() )
|
|
|
|
{
|
|
|
|
_bypassed = ! _bypassed;
|
|
|
|
redraw();
|
2010-02-24 16:05:46 +01:00
|
|
|
return 1;
|
2009-12-25 01:59:39 +01:00
|
|
|
}
|
|
|
|
else if ( Fl::event_button1() && ( drag = event_point() ) )
|
|
|
|
return 1;
|
|
|
|
else
|
|
|
|
return 0;
|
|
|
|
case FL_RELEASE:
|
2010-02-24 16:05:46 +01:00
|
|
|
if ( Fl::event_button1() && drag )
|
|
|
|
{
|
|
|
|
drag = NULL;
|
|
|
|
do_callback();
|
|
|
|
redraw();
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return 0;
|
2010-02-12 11:20:43 +01:00
|
|
|
case FL_MOUSEWHEEL:
|
|
|
|
{
|
|
|
|
/* TODO: place point on opposite face of sphere */
|
|
|
|
}
|
2009-12-25 01:59:39 +01:00
|
|
|
case FL_DRAG:
|
|
|
|
{
|
2010-02-24 16:05:46 +01:00
|
|
|
if ( ! drag )
|
|
|
|
return 0;
|
|
|
|
|
2009-12-25 01:59:39 +01:00
|
|
|
float X = Fl::event_x() - x();
|
|
|
|
float Y = Fl::event_y() - y();
|
|
|
|
|
|
|
|
int tx, ty, tw, th;
|
|
|
|
bbox( tx, ty, tw, th );
|
|
|
|
|
|
|
|
/* if ( _outs < 3 ) */
|
|
|
|
/* drag->angle( (float)(X / (tw / 2)) - 1.0f, 0.0f ); */
|
|
|
|
/* else */
|
2010-02-24 16:05:46 +01:00
|
|
|
drag->angle( (float)(X / (tw / 2)) - 1.0f, (float)(Y / (th / 2)) - 1.0f );
|
|
|
|
|
|
|
|
if ( when() & FL_WHEN_CHANGED )
|
|
|
|
do_callback();
|
2009-12-25 01:59:39 +01:00
|
|
|
|
|
|
|
redraw();
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return r;
|
|
|
|
|
|
|
|
// return 0;
|
|
|
|
}
|