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