mirror of
				https://github.com/ElvishArtisan/rivendell.git
				synced 2025-10-31 06:03:51 +01:00 
			
		
		
		
	* Fixed regressions in rdlibrary(1) and rdlogedit(1) that broke display of the audio playhead cursor.
		
			
				
	
	
		
			244 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			244 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // rdwavepainter.cpp
 | |
| //
 | |
| // A Painter Class for Drawing Audio Waveforms
 | |
| //
 | |
| //   (C) Copyright 2002-2005,2016 Fred Gleason <fredg@paravelsystems.com>
 | |
| //
 | |
| //   This program is free software; you can redistribute it and/or modify
 | |
| //   it under the terms of the GNU General Public License version 2 as
 | |
| //   published by the Free Software Foundation.
 | |
| //
 | |
| //   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., 675 Mass Ave, Cambridge, MA 02139, USA.
 | |
| //
 | |
| 
 | |
| #include <math.h>
 | |
| 
 | |
| #include <q3pointarray.h>
 | |
| #include <qpixmap.h>
 | |
| #include <qfile.h>
 | |
| #include <qmessagebox.h>
 | |
| #include <qobject.h>
 | |
| 
 | |
| #include <rdwavepainter.h>
 | |
| 
 | |
| 
 | |
| RDWavePainter::RDWavePainter(QPaintDevice *pd,RDCut *cut,
 | |
| 			     RDStation *station,RDUser *user,RDConfig *config)
 | |
|   : QPainter(pd)
 | |
| {
 | |
|   wave_cut=cut;
 | |
|   wave_station=station;
 | |
|   wave_user=user;
 | |
|   wave_config=config;
 | |
|   wave_peaks=NULL;
 | |
|   LoadWave();
 | |
| }
 | |
| 
 | |
| 
 | |
| RDWavePainter::RDWavePainter(RDStation *station,RDUser *user,RDConfig *config)
 | |
| {
 | |
|   wave_cut=NULL;
 | |
|   wave_station=station;
 | |
|   wave_user=user;
 | |
|   wave_config=config;
 | |
|   wave_peaks=NULL;
 | |
| }
 | |
| 
 | |
| 
 | |
| RDWavePainter::~RDWavePainter()
 | |
| {
 | |
|   if(wave_peaks!=NULL) {
 | |
|     delete wave_peaks;
 | |
|   }
 | |
| }
 | |
| 
 | |
| 
 | |
| void RDWavePainter::nameWave(RDCut *cut)
 | |
| {
 | |
|   wave_cut=cut;
 | |
|   LoadWave();
 | |
| }
 | |
| 
 | |
| 
 | |
| bool RDWavePainter::begin(QPaintDevice *pd,RDCut *cut)
 | |
| {
 | |
|   wave_cut=cut;
 | |
|   LoadWave();
 | |
|   return QPainter::begin(pd);
 | |
| }
 | |
| 
 | |
| 
 | |
| bool RDWavePainter::begin(QPaintDevice *pd)
 | |
| {
 | |
|   return QPainter::begin(pd);
 | |
| }
 | |
| 
 | |
| 
 | |
| void RDWavePainter::drawWaveBySamples(int x,int w,int startsamp,int endsamp,
 | |
| 				      int gain,Channel channel,
 | |
| 				      const QColor &color,
 | |
| 				      int startclip,int endclip)
 | |
| {
 | |
|   int startblock=startsamp/1152;
 | |
|   int endblock=endsamp/1152;
 | |
|   int startclipblock=-1;
 | |
|   int endclipblock=-1;
 | |
| 
 | |
| 
 | |
|   if((startblock>(int)wave_peaks->energySize())||(wave_peaks->energySize()==0)) {
 | |
|     return;
 | |
|   }
 | |
|   if(startclip>=0) {
 | |
|     startclipblock=startclip/1152;
 | |
|   }
 | |
|   if(endclip>=0) {
 | |
|     endclipblock=endclip/1152;
 | |
|   }
 | |
| 
 | |
|   double time_scale=1.0;
 | |
|   double gain_scale=1.0;
 | |
|   int dx=0;
 | |
|   QPixmap *pix=(QPixmap *)device();
 | |
|   int center=pix->height()/2;
 | |
|   RDWavePainter::Channel effective_channel=channel;
 | |
|   switch(channel) {
 | |
|   case RDWavePainter::Left:
 | |
|   case RDWavePainter::Right:
 | |
|     if(wave_channels==1) {
 | |
|       effective_channel=RDWavePainter::Mono;
 | |
|     }
 | |
|     break;
 | |
| 
 | |
|   default:
 | |
|     effective_channel=channel;
 | |
|     break;
 | |
|   }
 | |
|   save();
 | |
|   resetXForm();
 | |
|   setPen(color);
 | |
|   setBrush(color);
 | |
|   Q3PointArray array(w+2);
 | |
|   array.setPoint(0,0,center);
 | |
|   array.setPoint(w+1,w+1,center);
 | |
|   switch(effective_channel) {
 | |
|   case RDWavePainter::Left:
 | |
|     time_scale=(double)(endblock-startblock)/(double)w;
 | |
|     gain_scale=(double)(pix->height()/65536.0)*
 | |
|       pow(10.0,(double)gain/2000.0);
 | |
|     for(int i=0;i<w;i++) {
 | |
|       if(((dx=(2*((int)(time_scale*(double)i)+startblock)))<
 | |
| 	  (int)wave_peaks->energySize())&&
 | |
| 	 ((startclipblock<0)||(dx>(startclipblock*2)))&&
 | |
| 	 ((endclipblock<0)||(dx<(endclipblock*2)))) {
 | |
| 	array.setPoint(i+1,i+1,
 | |
| 		       center+(int)(gain_scale*(double)wave_peaks->energy(dx)));
 | |
|       }
 | |
|       else {
 | |
| 	array.setPoint(i+1,i+1,center);
 | |
|       }
 | |
|     }
 | |
|     break;
 | |
| 
 | |
|   case RDWavePainter::Right:
 | |
|     time_scale=(double)(endblock-startblock)/(double)w;
 | |
|     gain_scale=(double)(pix->height()/65536.0)*
 | |
|       pow(10.0,(double)gain/2000.0);
 | |
|     for(int i=0;i<w;i++) {
 | |
|       if(((dx=(1+2*((int)(time_scale*(double)i)+startblock)))<
 | |
| 	  (int)wave_peaks->energySize())&&
 | |
| 	 ((startclipblock<0)||(dx>(startclipblock*2)))&&
 | |
| 	 ((endclipblock<0)||(dx<(endclipblock*2)))) {
 | |
| 	array.setPoint(i+1,i+1,
 | |
| 		       center+(int)(gain_scale*(double)wave_peaks->energy(dx)));
 | |
|       }
 | |
|       else {
 | |
| 	array.setPoint(i+1,i+1,center);
 | |
|       }
 | |
|     }
 | |
|     break;
 | |
| 
 | |
|   case RDWavePainter::Mono:
 | |
|     switch(wave_channels) {
 | |
|     case 1:
 | |
|       time_scale=(double)(endblock-startblock)/(double)w;
 | |
|       gain_scale=(double)(pix->height()/65536.0)*
 | |
| 	pow(10.0,(double)gain/2000.0);
 | |
|       for(int i=0;i<w;i++) {
 | |
| 	if(((dx=((int)(time_scale*(double)i)+startblock))<
 | |
| 	    (int)wave_peaks->energySize())&&
 | |
| 	   ((startclipblock<0)||(dx>startclipblock))&&
 | |
| 	   ((endclipblock<0)||(dx<endclipblock))) {
 | |
| 	  array.setPoint(i+1,i+1,
 | |
| 			 center+(int)(gain_scale*(double)wave_peaks->energy(dx)));
 | |
| 	}
 | |
| 	else {
 | |
| 	  array.setPoint(i+1,i+1,center);
 | |
| 	}
 | |
|       }
 | |
|       break;
 | |
| 
 | |
|     case 2:
 | |
|       time_scale=(double)(endblock-startblock)/(double)w;
 | |
|       gain_scale=(double)(pix->height()/65536.0)*
 | |
| 	pow(10.0,(double)gain/2000.0);
 | |
|       for(int i=0;i<w;i++) {
 | |
| 	if(((dx=(2*((int)(time_scale*(double)i)+startblock)))<
 | |
| 	    (int)wave_peaks->energySize())&&
 | |
| 	   ((startclipblock<0)||(dx>(startclipblock*2)))&&
 | |
| 	   ((endclipblock<0)||(dx<(endclipblock*2)))) {
 | |
| 	  array.setPoint(i+1,i+1,
 | |
| 			 center+(int)(gain_scale*
 | |
| 				      ((double)wave_peaks->energy(dx)+
 | |
| 				       (double)wave_peaks->energy(dx+1))/2.0));
 | |
| 	}
 | |
| 	else {
 | |
| 	  array.setPoint(i+1,i+1,center);
 | |
| 	}
 | |
|       }
 | |
|       break;
 | |
|     }
 | |
|     break;
 | |
|   }
 | |
|   drawPolygon(array);
 | |
|   for(int i=0;i<(w+2);i++) {
 | |
|     array.setPoint(i,array.point(i).x(),2*center-array.point(i).y());
 | |
|   }
 | |
|   drawPolygon(array);
 | |
|   restore();
 | |
| }
 | |
| 
 | |
| 
 | |
| void RDWavePainter::drawWaveByMsecs(int x,int w,int startmsecs,int endmsecs,
 | |
| 				    int gain,Channel channel,
 | |
| 				    const QColor &color,
 | |
| 				    int startclip,int endclip)
 | |
| {
 | |
|   drawWaveBySamples(x,w,
 | |
| 	     (unsigned)((double)startmsecs*(double)wave_sample_rate/1000.0),
 | |
| 	     (unsigned)((double)endmsecs*(double)wave_sample_rate/1000.0),
 | |
| 	     gain,channel,color,
 | |
| 	     (int)((double)startclip*(double)wave_sample_rate/1000.0),
 | |
| 	     (int)((double)endclip*(double)wave_sample_rate/1000.0));
 | |
| }
 | |
| 
 | |
| 
 | |
| void RDWavePainter::LoadWave()
 | |
| {
 | |
|   wave_sample_rate=wave_cut->sampleRate();
 | |
|   wave_channels=wave_cut->channels();
 | |
|   if(wave_peaks!=NULL) {
 | |
|     delete wave_peaks;
 | |
|   }
 | |
|   wave_peaks=new RDPeaksExport();
 | |
|   wave_peaks->setCartNumber(wave_cut->cartNumber());
 | |
|   wave_peaks->setCutNumber(wave_cut->cutNumber());
 | |
|   wave_peaks->runExport(wave_user->name(),wave_user->password());
 | |
| }
 |