Fl_Resonance_Graph.cpp (6967B)
1 /* 2 ZynAddSubFX - a software synthesizer 3 4 Fl_Resonance_Graph.cpp - OSC Resonance Graph View 5 Copyright (C) 2016 Mark McCurry 6 7 This program is free software; you can redistribute it and/or 8 modify it under the terms of the GNU General Public License 9 as published by the Free Software Foundation; either version 2 10 of the License, or (at your option) any later version. 11 */ 12 #include "Fl_Resonance_Graph.H" 13 #include <FL/Fl.H> 14 #include <FL/fl_draw.H> 15 #include <FL/Fl_Value_Output.H> 16 #include <rtosc/rtosc.h> 17 using namespace zyn; 18 19 Fl_Resonance_Graph::Fl_Resonance_Graph(int x,int y, int w, int h, const char *label) 20 :Fl_Box(x,y,w,h,label), Fl_Osc_Widget(this), khzvalue(NULL), dbvalue(NULL), 21 oldx(-1), oldy(-1), cbwidget(NULL), applybutton(NULL), Pcenterfreq(0), Poctavesfreq(0), 22 PmaxdB(0) 23 { 24 memset(Prespoints, 64, N_RES_POINTS); 25 //Get values 26 oscRegister("Prespoints"); 27 oscRegister("Pcenterfreq"); 28 oscRegister("Poctavesfreq"); 29 oscRegister("PmaxdB"); 30 31 cbwidget=NULL; 32 applybutton=NULL; 33 } 34 35 Fl_Resonance_Graph::~Fl_Resonance_Graph(void) 36 { 37 } 38 39 void Fl_Resonance_Graph::init(Fl_Value_Output *khzvalue_,Fl_Value_Output *dbvalue_) 40 { 41 khzvalue=khzvalue_; 42 dbvalue=dbvalue_; 43 oldx=-1; 44 khzval=-1; 45 } 46 47 void Fl_Resonance_Graph::draw_freq_line(float freq,int type) 48 { 49 const float freqx=getfreqpos(freq);//XXX 50 switch(type){ 51 case 0:fl_line_style(FL_SOLID);break; 52 case 1:fl_line_style(FL_DOT);break; 53 case 2:fl_line_style(FL_DASH);break; 54 } 55 56 57 if ((freqx>0.0)&&(freqx<1.0)) 58 fl_line(x()+(int) (freqx*w()),y(), 59 x()+(int) (freqx*w()),y()+h()); 60 } 61 62 void Fl_Resonance_Graph::draw() 63 { 64 const int ox=x(),oy=y(),lx=w(),ly=h(); 65 66 fl_color(FL_DARK1); 67 fl_rectf(ox,oy,lx,ly); 68 69 70 //draw the lines 71 fl_color(FL_GRAY); 72 73 fl_line_style(FL_SOLID); 74 fl_line(ox+2,oy+ly/2,ox+lx-2,oy+ly/2); 75 76 77 //Draw 1kHz line 78 const float freqx=getfreqpos(1000.0);//XXX 79 if ((freqx>0.0)&&(freqx<1.0)) 80 fl_line(ox+(int) (freqx*lx),oy, 81 ox+(int) (freqx*lx),oy+ly); 82 83 //Draw other frequency lines 84 for (int i=1; i<10; ++i){ 85 if(i==1) { 86 draw_freq_line(i*100.0,0); 87 draw_freq_line(i*1000.0,0); 88 } else 89 if (i==5) { 90 draw_freq_line(i*100.0,2); 91 draw_freq_line(i*1000.0,2); 92 } else { 93 draw_freq_line(i*100.0,1); 94 draw_freq_line(i*1000.0,1); 95 } 96 } 97 98 draw_freq_line(10000.0,0); 99 draw_freq_line(20000.0,1); 100 101 //Draw dotted grid 102 fl_line_style(FL_DOT); 103 int GY=10;if (ly<GY*3) GY=-1; 104 for (int i=1; i<GY; ++i){ 105 int tmp=(int)(ly/(float)GY*i); 106 fl_line(ox+2,oy+tmp,ox+lx-2,oy+tmp); 107 } 108 109 110 111 //draw the data 112 fl_color(FL_RED); 113 fl_line_style(FL_SOLID,2); 114 fl_begin_line(); 115 int oiy = ly*Prespoints[0]/128.0;//XXX easy 116 for (int i=1; i<N_RES_POINTS; ++i){ 117 const int ix=(i*1.0/N_RES_POINTS*lx); 118 const int iy= ly*Prespoints[i]/128.0;//XXX easy 119 fl_vertex(ox+ix,oy+ly-oiy); 120 oiy=iy; 121 } 122 fl_end_line(); 123 fl_line_style(FL_SOLID,0); 124 } 125 126 int Fl_Resonance_Graph::handle(int event) 127 { 128 int x_=Fl::event_x()-x(); 129 int y_=Fl::event_y()-y(); 130 if((x_>=0)&&(x_<w()) && (y_>=0)&&(y_<h())){ 131 khzvalue->value(getfreqx(x_*1.0/w())/1000.0);//XXX 132 dbvalue->value((1.0-y_*2.0/h())*PmaxdB);//XXX 133 } 134 135 if((event==FL_PUSH)||(event==FL_DRAG)){ 136 const bool leftbutton = Fl::event_button() == FL_LEFT_MOUSE; 137 138 if (x_<0) x_=0; 139 if (y_<0) y_=0; 140 if (x_>=w()) x_=w(); 141 if (y_>=h()-1) y_=h()-1; 142 143 if ((oldx<0)||(oldx==x_)){ 144 int sn=(int)(x_*1.0/w()*N_RES_POINTS); 145 int sp=127-(int)(y_*1.0/h()*127); 146 if(leftbutton) 147 setPoint(sn,sp); 148 //oscWrite("setpoint", "ii", sn, sp);//respar->setpoint(sn,sp);//XXX easy 149 else 150 setPoint(sn,sp); 151 //oscWrite("setpoint", "ii", sn, 64);//respar->setpoint(sn,64);//XXX easy 152 } else { 153 int x1=oldx; 154 int x2=x_; 155 int y1=oldy; 156 int y2=y_; 157 if (oldx>x_){ 158 x1=x_;y1=y_; 159 x2=oldx;y2=oldy; 160 } 161 for (int i=0;i<x2-x1;i++){ 162 int sn=(int)((i+x1)*1.0/w()*N_RES_POINTS); 163 float yy=(y2-y1)*1.0/(x2-x1)*i; 164 int sp=127-(int)((y1+yy)/h()*127); 165 if(leftbutton) //respar->setpoint(sn,sp);//XXX easy 166 setPoint(sn, sp); 167 //oscWrite("setpoint", "ii", sn, sp); 168 else //respar->setpoint(sn,64);//XXX easy 169 setPoint(sn, sp); 170 //oscWrite("setpoint", "ii", sn, sp); 171 } 172 } 173 174 oldx=x_;oldy=y_; 175 redraw(); 176 } 177 178 if(event==FL_RELEASE) { 179 oldx=-1; 180 if(cbwidget) { 181 cbwidget->do_callback(); 182 if(applybutton) { 183 applybutton->color(FL_RED); 184 applybutton->redraw(); 185 } 186 } 187 } 188 189 return 1; 190 } 191 192 void Fl_Resonance_Graph::setcbwidget(Fl_Widget *cbwidget,Fl_Widget *applybutton) 193 { 194 this->cbwidget=cbwidget; 195 this->applybutton=applybutton; 196 } 197 198 void Fl_Resonance_Graph::update(void) 199 { 200 oscWrite("Prespoints"); 201 oscWrite("Pcenterfreq"); 202 oscWrite("Poctavesfreq"); 203 oscWrite("PmaxdB"); 204 } 205 206 void Fl_Resonance_Graph::OSC_raw(const char *msg) 207 { 208 //TODO check the types (OSC regex) 209 if(strstr(msg, "Prespoints")) { 210 rtosc_blob_t arg = rtosc_argument(msg, 0).b; 211 assert(arg.len == N_RES_POINTS); 212 memcpy(Prespoints, arg.data, N_RES_POINTS); 213 } else if(strstr(msg, "Pcenterfreq")) 214 Pcenterfreq = rtosc_argument(msg, 0).i; 215 else if(strstr(msg, "Poctavesfreq")) 216 Poctavesfreq = rtosc_argument(msg, 0).i; 217 else if(strstr(msg, "PmaxdB")) 218 PmaxdB = rtosc_argument(msg, 0).i; 219 else 220 puts("I got an unknown message..."); 221 222 redraw(); 223 } 224 225 float Fl_Resonance_Graph::getfreqx(float x) const 226 { 227 const float octf = powf(2.0f, getoctavesfreq()); 228 return getcenterfreq() / sqrt(octf) * powf(octf, limit(x, 0.0f, 1.0f)); 229 } 230 231 /* 232 * Get the x coordinate from frequency (used by the UI) 233 */ 234 float Fl_Resonance_Graph::getfreqpos(float freq) const 235 { 236 return (logf(freq) - logf(getfreqx(0.0f))) / logf(2.0f) / getoctavesfreq(); 237 } 238 239 /* 240 * Get the center frequency of the resonance graph 241 */ 242 float Fl_Resonance_Graph::getcenterfreq() const 243 { 244 return 10000.0f * powf(10, -(1.0f - Pcenterfreq / 127.0f) * 2.0f); 245 } 246 247 /* 248 * Get the number of octave that the resonance functions applies to 249 */ 250 float Fl_Resonance_Graph::getoctavesfreq() const 251 { 252 return 0.25f + 10.0f * Poctavesfreq / 127.0f; 253 } 254 255 void Fl_Resonance_Graph::setPoint(int idx, int val) 256 { 257 Prespoints[idx] = val; 258 oscWrite(std::string("Prespoints")+to_s(idx), "i", val); 259 redraw(); 260 }