| 
							- /*
 -  * Author: Harry van Haaren 2013
 -  *         harryhaaren@gmail.com
 -  * 
 -  * 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; if not, write to the Free Software
 -  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
 -  * MA 02110-1301, USA.
 -  * 
 -  */
 - 
 - 
 - #ifndef AVTK_FILTERGRAPH_H
 - #define AVTK_FILTERGRAPH_H
 - 
 - #include <FL/Fl_Widget.H>
 - #include <FL/Fl_Slider.H>
 - #include <valarray>
 - #include <string>
 - 
 - namespace Avtk
 - {
 -   
 - class Filtergraph : public Fl_Slider
 - {
 -   public:
 -     enum Type {
 -       FILTER_LOWPASS = 0,
 -       FILTER_HIGHPASS,
 -       FILTER_FLAT,
 -     };
 -     
 -     Filtergraph(int _x, int _y, int _w, int _h, const char *_label = 0, Type _type = FILTER_LOWPASS):
 -         Fl_Slider(_x, _y, _w, _h, _label)
 -     {
 -       graphType = _type;
 -       
 -       x = _x;
 -       y = _y;
 -       w = _w;
 -       h = _h;
 -       
 -       label = _label;
 -       
 -       mouseClickedX = 0;
 -       mouseClickedY = 0;
 -       mouseClicked = false;
 -       
 -       active = true;
 -       highlight = false;
 -       
 -       value( 0.5 );
 -       freq = value();
 -       
 -       gain = 0;
 -       bandwidth = 0;
 -     }
 -     
 -     void setGain(float g) {gain = g; redraw();}
 -     void setBandwidth(float b) {bandwidth = b; redraw();}
 -     float getGain() {return gain;}
 -     float getBandwidth() {return bandwidth;}
 -     
 -     void setType(Type t)
 -     {
 -       graphType = t;
 -       redraw();
 -     }
 -     
 -     void setActive(bool a)
 -     {
 -       active = a;
 -       redraw();
 -     }
 -     
 -     bool getActive()
 -     {
 -       return active;
 -     }
 -     
 -     Type graphType;
 -     bool active;
 -     bool highlight;
 -     int x, y, w, h;
 -     const char* label;
 -     
 -     int mouseClickedX;
 -     int mouseClickedY;
 -     bool mouseClicked;
 -     
 -     float freq;
 -     float gain;
 -     float bandwidth;
 -     
 -     void draw()
 -     {
 -       if (damage() & FL_DAMAGE_ALL)
 -       {
 -         cairo_t *cr = Fl::cairo_cc();
 -         
 -         cairo_save( cr );
 -         
 -         cairo_set_line_width(cr, 1.5);
 -         
 -         
 -         // fill background
 -         cairo_rectangle( cr, x, y, w, h);
 -         cairo_set_source_rgb( cr, 28 / 255.f,  28 / 255.f ,  28 / 255.f  );
 -         cairo_fill( cr );
 -         
 -         
 -         // set up dashed lines, 1 px off, 1 px on
 -         double dashes[1];
 -         dashes[0] = 2.0;
 -         
 -         cairo_set_dash ( cr, dashes, 1, 0.0);
 -         cairo_set_line_width( cr, 1.0);
 -         
 -         // loop over each 2nd line, drawing dots
 -         cairo_set_line_width(cr, 1.0);
 -         cairo_set_source_rgb(cr, 0.4,0.4,0.4);
 -         for ( int i = 0; i < 4; i++ )
 -         {
 -           cairo_move_to( cr, x + ((w / 4.f)*i), y );
 -           cairo_line_to( cr, x + ((w / 4.f)*i), y + h );
 -         }
 -         for ( int i = 0; i < 4; i++ )
 -         {
 -           cairo_move_to( cr, x    , y + ((h / 4.f)*i) );
 -           cairo_line_to( cr, x + w, y + ((h / 4.f)*i) );
 -         }
 -         
 -         
 -         cairo_set_source_rgba( cr,  66 / 255.f,  66 / 255.f ,  66 / 255.f , 0.5 );
 -         cairo_stroke(cr);
 -         cairo_set_dash ( cr, dashes, 0, 0.0);
 -         
 -         // change filter type (low|high) based on value:
 -         // 0 to < 0.5 = lowpass
 -         // 0.5 == no change
 -         // 0.5 < 1.0 == highpass
 -         if ( value() < 0.45 )
 -         {
 -           graphType = FILTER_LOWPASS;
 -           freq = value()*2.f;
 -         }
 -         else if ( value() > 0.55 )
 -         {
 -           graphType = FILTER_HIGHPASS;
 -           freq = (value()-0.5)*2.f;
 -         }
 -         else
 -         {
 -           graphType = FILTER_FLAT;
 -         }
 -         
 -         switch( graphType )
 -         {
 -           case FILTER_LOWPASS :   drawLowpass(cr);      break;
 -           case FILTER_HIGHPASS:   drawHighpass(cr);     break;
 -           case FILTER_FLAT    :   drawFlat(cr);         break;
 -           default:
 -             cout << "Filtergraph: unknown filter type selected!" << endl;
 -         }
 -         
 -         // stroke outline
 -         cairo_rectangle(cr, x, y, w, h);
 -         cairo_set_source_rgba( cr,  126 / 255.f,  126 / 255.f ,  126 / 255.f , 0.8 );
 -         cairo_set_line_width(cr, 1.9);
 -         cairo_stroke( cr );
 -         
 -         if ( !active )
 -         {
 -           // big grey X
 -           cairo_set_line_width(cr, 20.0);
 -           cairo_set_source_rgba(cr, 0.4,0.4,0.4, 0.7);
 -           
 -           cairo_move_to( cr, x + (3 * w / 4.f), y + ( h / 4.f ) );
 -           cairo_line_to( cr, x + (w / 4.f), y + ( 3 *h / 4.f ) );
 -           
 -           cairo_move_to( cr, x + (w / 4.f), y + ( h / 4.f ) );
 -           cairo_line_to( cr, x + (3 * w / 4.f), y + ( 3 *h / 4.f ) );
 -           cairo_set_line_cap ( cr, CAIRO_LINE_CAP_BUTT);
 -           cairo_stroke( cr );
 -         }
 -         
 -         cairo_restore( cr );
 -       }
 -     }
 -     
 -     void resize(int X, int Y, int W, int H)
 -     {
 -       Fl_Widget::resize(X,Y,W,H);
 -       x = X;
 -       y = Y;
 -       w = W;
 -       h = H;
 -       redraw();
 -     }
 -     
 -     int handle(int event)
 -     {
 -       switch(event)
 -       {
 -         case FL_PUSH:
 -           highlight = 0;
 -           if ( Fl::event_button() == FL_RIGHT_MOUSE )
 -           {
 -             active = !active;
 -             redraw();
 -             do_callback();
 -           }
 -           redraw();
 -           return 1;
 -         case FL_DRAG:
 -           {
 -             if ( Fl::event_state(FL_BUTTON1) )
 -             {
 -               if ( mouseClicked == false ) // catch the "click" event
 -               {
 -                 mouseClickedX = Fl::event_x();
 -                 mouseClickedY = Fl::event_y();
 -                 mouseClicked = true;
 -               }
 -               
 -               float deltaX = mouseClickedX - Fl::event_x();
 -               float deltaY = mouseClickedY - Fl::event_y();
 -               
 -               float valX = value();
 -               valX -= deltaX / 100.f;
 -               float valY = gain;
 -               valY += deltaY / 100.f;
 -               
 -               if ( valX > 1.0 ) valX = 1.0;
 -               if ( valX < 0.0 ) valX = 0.0;
 -               
 -               if ( valY > 1.0 ) valY = 1.0;
 -               if ( valY < 0.0 ) valY = 0.0;
 -               
 -               //handle_drag( value + deltaY );
 -               set_value( valX );
 -               gain = valY;
 -               
 -               mouseClickedX = Fl::event_x();
 -               mouseClickedY = Fl::event_y();
 -               redraw();
 -               do_callback();
 -             }
 -           }
 -           return 1;
 -         case FL_RELEASE:
 -           if (highlight) {
 -             highlight = 0;
 -             redraw();
 -             do_callback();
 -           }
 -           mouseClicked = false;
 -           return 1;
 -         case FL_SHORTCUT:
 -           if ( test_shortcut() )
 -           {
 -             do_callback();
 -             return 1;
 -           }
 -           return 0;
 -         default:
 -           return Fl_Widget::handle(event);
 -       }
 -     }
 -     
 -   private:
 -     void drawLowpass(cairo_t* cr)
 -     {
 -       // draw the cutoff line:
 -       // move to bottom left, draw line to middle left
 -       cairo_move_to( cr, x , y + h );
 -       cairo_line_to( cr, x , y + (h*0.47));
 -       
 -       float cutoff = 0.1 + freq * 0.85;
 -       
 -       // Curve
 -       cairo_curve_to( cr, x + w * cutoff    , y+(h*0.5)  ,   // control point 1
 -                           x + w * cutoff    , y+(h * 0.3),   // control point 2
 -                           x + w * cutoff + 5, y+ h       );  // end of curve 1
 -       
 -       cairo_close_path(cr);
 -       
 -       cairo_set_source_rgba( cr, 0 / 255.f, 153 / 255.f , 255 / 255.f , 0.21 );
 -       cairo_fill_preserve(cr);
 -       cairo_set_source_rgba( cr, 0 / 255.f, 153 / 255.f , 255 / 255.f , 1 );
 -       cairo_set_line_width(cr, 1.5);
 -       cairo_set_line_join( cr, CAIRO_LINE_JOIN_ROUND);
 -       cairo_set_line_cap ( cr, CAIRO_LINE_CAP_ROUND);
 -       cairo_stroke( cr );
 -     }
 -         
 -     void drawHighpass(cairo_t* cr)
 -     {
 -       // draw the cutoff line:
 -       float cutoff = 0.95 - (freq*0.8);
 -       
 -       // move to bottom right
 -       cairo_move_to( cr, x + w, y + h );
 -       cairo_line_to( cr, x + w, y + (h*0.47));
 -       
 -       // Curve
 -       cairo_curve_to( cr, x + w - (w*cutoff)    , y+(h*0.5)  ,   // control point 1
 -                           x + w - (w*cutoff)    , y+(h * 0.3),   // control point 2
 -                           x + w - (w*cutoff) - 5, y+ h      );   // end of curve 1
 -       
 -       cairo_close_path(cr);
 -       
 -       // stroke
 -       cairo_set_source_rgba( cr, 0 / 255.f, 153 / 255.f , 255 / 255.f , 0.21 );
 -       cairo_fill_preserve(cr);
 -       cairo_set_source_rgba( cr, 0 / 255.f, 153 / 255.f , 255 / 255.f , 1 );
 -       cairo_set_line_width(cr, 1.5);
 -       cairo_set_line_join( cr, CAIRO_LINE_JOIN_ROUND);
 -       cairo_set_line_cap ( cr, CAIRO_LINE_CAP_ROUND);
 -       cairo_stroke( cr );
 -     }
 -     
 -     void drawFlat(cairo_t* cr)
 -     {
 -       // move to bottom right
 -       cairo_move_to( cr, x + w, y + h );
 -       cairo_line_to( cr, x + w, y + (h*0.47));
 -       cairo_line_to( cr, x    , y + (h*0.47));
 -       cairo_line_to( cr, x    , y + h       );
 -       cairo_close_path(cr);
 -       
 -       // stroke
 -       cairo_set_source_rgba( cr, 0 / 255.f, 153 / 255.f , 255 / 255.f , 0.21 );
 -       cairo_fill_preserve(cr);
 -       cairo_set_source_rgba( cr, 0 / 255.f, 153 / 255.f , 255 / 255.f , 1 );
 -       cairo_set_line_width(cr, 1.5);
 -       cairo_set_line_join( cr, CAIRO_LINE_JOIN_ROUND);
 -       cairo_set_line_cap ( cr, CAIRO_LINE_CAP_ROUND);
 -       cairo_stroke( cr );
 -     }
 -     
 - };
 - 
 - } // Avtk
 - 
 - #endif // AVTK_FILTERGRAPH_H
 
 
  |