Rivendellaudio/lib/rdwavepainter.cpp
Robert Chipperfield b796bdc66e
Fix drawWaveByMsecs when startmsecs or endmsecs are negative.
On ARM, the unsigned cast seems to be respected; on x64, apparently not. In RDLogEdit's Voice Tracker, this causes incorrect waveform display on ARM systems (https://github.com/ElvishArtisan/rivendell/issues/389)
2021-10-26 19:23:09 +01:00

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,
(int)((double)startmsecs*(double)wave_sample_rate/1000.0),
(int)((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());
}