qt_demoe/qwtdemo/examples/radio/ampfrm.cpp

202 lines
5.3 KiB
C++

#include "ampfrm.h"
#include <qwt_knob.h>
#include <qwt_thermo.h>
#include <qwt_round_scale_draw.h>
#include <qwt_math.h>
#include <qlayout.h>
#include <qlabel.h>
#include <qfont.h>
#include <qpen.h>
#include <qevent.h>
#if QT_VERSION < 0x040600
#define qFastSin(x) ::sin(x)
#define qFastCos(x) ::cos(x)
#endif
class Knob: public QWidget
{
public:
Knob( const QString &title, double min, double max, QWidget *parent ):
QWidget( parent )
{
d_knob = new QwtKnob( this );
d_knob->setScale( min, max );
d_knob->setTotalSteps( 0 ); // disable
d_knob->setScaleMaxMajor( 10 );
d_knob->setKnobStyle( QwtKnob::Raised );
d_knob->setKnobWidth( 50 );
d_knob->setBorderWidth( 2 );
d_knob->setMarkerStyle( QwtKnob::Notch );
d_knob->setMarkerSize( 8 );
d_knob->scaleDraw()->setTickLength( QwtScaleDiv::MinorTick, 4 );
d_knob->scaleDraw()->setTickLength( QwtScaleDiv::MediumTick, 4 );
d_knob->scaleDraw()->setTickLength( QwtScaleDiv::MajorTick, 6 );
d_label = new QLabel( title, this );
d_label->setAlignment( Qt::AlignTop | Qt::AlignHCenter );
setSizePolicy( QSizePolicy::MinimumExpanding,
QSizePolicy::MinimumExpanding );
}
virtual QSize sizeHint() const
{
QSize sz1 = d_knob->sizeHint();
QSize sz2 = d_label->sizeHint();
const int w = qMax( sz1.width(), sz2.width() );
const int h = sz1.height() + sz2.height();
int off = qCeil( d_knob->scaleDraw()->extent( d_knob->font() ) );
off -= 10; // spacing
return QSize( w, h - off );
}
void setValue( double value )
{
d_knob->setValue( value );
}
double value() const
{
return d_knob->value();
}
protected:
virtual void resizeEvent( QResizeEvent *e )
{
const QSize sz = e->size();
int h = d_label->sizeHint().height();
d_label->setGeometry( 0, sz.height() - h, sz.width(), h );
h = d_knob->sizeHint().height();
int off = qCeil( d_knob->scaleDraw()->extent( d_knob->font() ) );
off -= 10; // spacing
d_knob->setGeometry( 0, d_label->pos().y() - h + off,
sz.width(), h );
}
private:
QwtKnob *d_knob;
QLabel *d_label;
};
class Thermo: public QWidget
{
public:
Thermo( const QString &title, QWidget *parent ):
QWidget( parent )
{
d_thermo = new QwtThermo( this );
d_thermo->setPipeWidth( 6 );
d_thermo->setScale( -40, 10 );
d_thermo->setFillBrush( Qt::green );
d_thermo->setAlarmBrush( Qt::red );
d_thermo->setAlarmLevel( 0.0 );
d_thermo->setAlarmEnabled( true );
QLabel *label = new QLabel( title, this );
label->setAlignment( Qt::AlignTop | Qt::AlignLeft );
QVBoxLayout *layout = new QVBoxLayout( this );
layout->setMargin( 0 );
layout->setSpacing( 0 );
layout->addWidget( d_thermo, 10 );
layout->addWidget( label );
}
void setValue( double value )
{
d_thermo->setValue( value );
}
private:
QwtThermo *d_thermo;
};
AmpFrame::AmpFrame( QWidget *p ):
QFrame( p )
{
d_knbVolume = new Knob( "Volume", 0.0, 10.0, this );
d_knbBalance = new Knob( "Balance", -10.0, 10.0, this );
d_knbTreble = new Knob( "Treble", -10.0, 10.0, this );
d_knbBass = new Knob( "Bass", -10.0, 10.0, this );
d_thmLeft = new Thermo( "Left [dB]", this );
d_thmRight = new Thermo( "Right [dB]", this );
QHBoxLayout *layout = new QHBoxLayout( this );
layout->setSpacing( 0 );
layout->setMargin( 10 );
layout->addWidget( d_knbVolume );
layout->addWidget( d_knbBalance);
layout->addWidget( d_knbTreble);
layout->addWidget( d_knbBass );
layout->addSpacing( 20 );
layout->addStretch( 10 );
layout->addWidget( d_thmLeft );
layout->addSpacing( 10 );
layout->addWidget( d_thmRight );
d_knbVolume->setValue( 7.0 );
( void )startTimer( 50 );
}
void AmpFrame::timerEvent( QTimerEvent * )
{
static double phs = 0;
//
// This amplifier generates its own input signal...
//
const double sig_bass = ( 1.0 + 0.1 * d_knbBass->value() )
* qFastSin( 13.0 * phs );
const double sig_mid_l = qFastSin( 17.0 * phs );
const double sig_mid_r = qFastCos( 17.5 * phs );
const double sig_trbl_l = 0.5 * ( 1.0 + 0.1 * d_knbTreble->value() )
* qFastSin( 35.0 * phs );
const double sig_trbl_r = 0.5 * ( 1.0 + 0.1 * d_knbTreble->value() )
* qFastSin( 34.0 * phs );
double sig_l = 0.05 * d_master * d_knbVolume->value()
* qwtSqr( sig_bass + sig_mid_l + sig_trbl_l );
double sig_r = 0.05 * d_master * d_knbVolume->value()
* qwtSqr( sig_bass + sig_mid_r + sig_trbl_r );
double balance = 0.1 * d_knbBalance->value();
if ( balance > 0 )
sig_l *= ( 1.0 - balance );
else
sig_r *= ( 1.0 + balance );
if ( sig_l > 0.01 )
sig_l = 20.0 * log10( sig_l );
else
sig_l = -40.0;
if ( sig_r > 0.01 )
sig_r = 20.0 * log10( sig_r );
else
sig_r = - 40.0;
d_thmLeft->setValue( sig_l );
d_thmRight->setValue( sig_r );
phs += M_PI / 100;
if ( phs > M_PI )
phs = 0;
}
void AmpFrame::setMaster( double v )
{
d_master = v;
}