FreeEdit.cpp (5733B)
1 /* 2 Copyright (C) 2009 Nasca Octavian Paul 3 Author: Nasca Octavian Paul 4 5 This program is free software; you can redistribute it and/or modify 6 it under the terms of version 2 of the GNU General Public License 7 as published by the Free Software Foundation. 8 9 This program is distributed in the hope that it will be useful, 10 but WITHOUT ANY WARRANTY; without even the implied warranty of 11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 12 GNU General Public License (version 2) for more details. 13 14 You should have received a copy of the GNU General Public License (version 2) 15 along with this program; if not, write to the Free Software Foundation, 16 Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA 17 */ 18 19 #include <stdio.h> 20 #include <math.h> 21 #include "FreeEdit.h" 22 23 FreeEdit::FreeEdit(){ 24 enabled=false; 25 smooth=0.0; 26 interp_mode=LINEAR; 27 npos=FREE_EDIT_MAX_POINTS; 28 pos=new FreeEditPos[npos]; 29 for (int i=0;i<npos;i++){ 30 pos[i].x=pos[i].y=0; 31 pos[i].enabled=false; 32 }; 33 pos[0].x=0; 34 pos[0].y=0.5; 35 pos[0].enabled=true; 36 pos[1].x=1; 37 pos[1].y=0.5; 38 pos[1].enabled=true; 39 40 curve.data=NULL; 41 curve.size=0; 42 }; 43 44 void FreeEdit::deep_copy_from(const FreeEdit &other){ 45 enabled=other.enabled; 46 smooth=other.smooth; 47 interp_mode=other.interp_mode; 48 npos=other.npos; 49 pos=new FreeEditPos[npos]; 50 for (int i=0;i<npos;i++){ 51 pos[i].x=other.pos[i].x; 52 pos[i].y=other.pos[i].y; 53 pos[i].enabled=other.pos[i].enabled; 54 }; 55 curve.size=other.curve.size; 56 if (other.curve.data&&other.curve.size){ 57 curve.data=new REALTYPE[curve.size]; 58 for (int i=0;i<curve.size;i++) curve.data[i]=other.curve.data[i]; 59 }else curve.data=NULL; 60 extreme_x=other.extreme_x; 61 extreme_y=other.extreme_y; 62 }; 63 FreeEdit::FreeEdit (const FreeEdit &other){ 64 deep_copy_from(other); 65 }; 66 const FreeEdit &FreeEdit::operator=(const FreeEdit &other){ 67 if (this == &other) return *this; 68 deep_copy_from(other); 69 return *this; 70 }; 71 void FreeEdit::get_curve(int datasize,REALTYPE *data,bool real_values){ 72 int npos_used=0; 73 for (int i=0;i<npos;i++) if (is_enabled(i)) npos_used++; 74 if (!npos_used){ 75 for (int i=0;i<datasize;i++) data[i]=(real_values?extreme_y.get_min():0.0); 76 return; 77 }; 78 79 //get the enabled points 80 REALTYPE posx[npos_used],posy[npos_used]; 81 int k=0; 82 for (int i=0;i<npos;i++){ 83 if (is_enabled(i)){ 84 posx[k]=get_posx(i); 85 posy[k]=get_posy(i); 86 k++; 87 }; 88 }; 89 90 //sort the points 91 for (int j=0;j<npos_used-1;j++){ 92 for (int i=j+1;i<npos_used;i++){ 93 if (posx[i]<posx[j]){ 94 swap(posx[i],posx[j]); 95 swap(posy[i],posy[j]); 96 }; 97 }; 98 }; 99 100 //generate the curve 101 int p1=0,p2=1; 102 for (int i=0;i<datasize;i++){ 103 REALTYPE x=(REALTYPE)i/(REALTYPE)datasize; 104 while ((x>posx[p2])&&(p2<npos_used)){ 105 p1=p2; 106 p2++; 107 }; 108 REALTYPE px1=posx[p1]; 109 REALTYPE px2=posx[p2]; 110 REALTYPE diffx=px2-px1; 111 REALTYPE x0=0; 112 if (diffx>1e-5) x0=(x-px1)/diffx; 113 if (interp_mode==COSINE) x0=(1.0-cos(x0*M_PI))*0.5; 114 REALTYPE y=y=posy[p1]*(1.0-x0)+posy[p2]*x0; 115 data[i]=y; 116 }; 117 118 119 //smooth the curve 120 if (smooth>0.01){ 121 const int max_times=4; 122 REALTYPE a=exp(log(0.25)/(smooth*smooth*datasize*0.25)); 123 if ((a<=0.0)||(a>=1.0)) return; 124 a=pow(a,max_times); 125 for (k=0;k<max_times;k++){ 126 for (int i=1;i<datasize;i++) data[i]=data[i]*(1.0-a)+data[i-1]*a; 127 for (int i=datasize-2;i>=0;i--) data[i]=data[i]*(1.0-a)+data[i+1]*a; 128 }; 129 }; 130 131 if (real_values){ 132 for (int i=0;i<datasize;i++) data[i]=extreme_y.coord_to_real_value(data[i]); 133 if (extreme_y.get_scale()==FE_DB){ 134 for (int i=0;i<datasize;i++) data[i]=dB2rap(data[i]); 135 }; 136 }; 137 138 }; 139 140 void FreeEdit::update_curve(int size){ 141 if (curve.data) delete []curve.data; 142 if (size<2) size=2; 143 curve.size=size; 144 curve.data=new REALTYPE[size]; 145 146 get_curve(curve.size,curve.data,true); 147 148 // for(int i=0;i<size;i++) printf("_%d %g\n",i,curve.data[i]); 149 }; 150 151 REALTYPE FreeEdit::get_value(REALTYPE x){ 152 if (!curve.data) { 153 return 0.0;// update_curve(); 154 }; 155 if (extreme_x.get_scale()==FE_LOG){ 156 if (x<=0.0) x=1e-9; 157 }; 158 159 // printf("%g\n",curve.data[1]); 160 161 x=extreme_x.real_value_to_coord(x); 162 if (x<0) x=0.0; 163 else if (x>1.0) x=1.0; 164 REALTYPE rx=x*curve.size; 165 REALTYPE rxh=floor(rx); 166 int k=(int)rxh; 167 REALTYPE rxl=rx-rxh; 168 if (k<0) k=0; 169 if (k>(curve.size-1)) k=curve.size-1; 170 int k1=k+1; if (k1>(curve.size-1)) k1=curve.size-1; 171 return curve.data[k]*(1.0-rxl)+curve.data[k1]*rxl; 172 }; 173 174 void FreeEdit::add2XML(XMLwrapper *xml){ 175 xml->addparbool("enabled",enabled); 176 xml->addparreal("smooth",smooth); 177 xml->addpar("interp_mode",interp_mode); 178 xml->beginbranch("POINTS"); 179 for (int i=0;i<FREE_EDIT_MAX_POINTS;i++){ 180 if (!pos[i].enabled) continue; 181 xml->beginbranch("POINT",i); 182 xml->addparbool("enabled",pos[i].enabled); 183 xml->addparreal("x",pos[i].x); 184 xml->addparreal("y",pos[i].y); 185 186 xml->endbranch(); 187 }; 188 xml->endbranch(); 189 190 xml->beginbranch("EXTREME_X"); 191 extreme_x.add2XML(xml); 192 xml->endbranch(); 193 194 xml->beginbranch("EXTREME_Y"); 195 extreme_y.add2XML(xml); 196 xml->endbranch(); 197 }; 198 199 200 void FreeEdit::getfromXML(XMLwrapper *xml){ 201 enabled=xml->getparbool("enabled",enabled); 202 smooth=xml->getparreal("smooth",smooth); 203 interp_mode=(INTERP_MODE)xml->getpar("interp_mode",interp_mode,0,1); 204 205 206 207 if (xml->enterbranch("POINTS")){ 208 for (int i=0;i<FREE_EDIT_MAX_POINTS;i++){ 209 if (xml->enterbranch("POINT",i)){ 210 pos[i].enabled=xml->getparbool("enabled",pos[i].enabled); 211 pos[i].x=xml->getparreal("x",pos[i].x); 212 pos[i].y=xml->getparreal("y",pos[i].y); 213 214 xml->exitbranch(); 215 }; 216 }; 217 xml->exitbranch(); 218 }; 219 220 if (xml->enterbranch("EXTREME_X")){ 221 extreme_x.getfromXML(xml); 222 xml->exitbranch(); 223 }; 224 225 if (xml->enterbranch("EXTREME_Y")){ 226 extreme_y.getfromXML(xml); 227 xml->exitbranch(); 228 }; 229 230 update_curve(); 231 }; 232 233