paulstretch_cpp

PaulStretch
Log | Files | Refs | LICENSE

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