zynaddsubfx

ZynAddSubFX open source synthesizer
Log | Files | Refs | Submodules | LICENSE

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 }