|
- /*
- * 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
|