123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659 |
- /*
- * organic.cpp - additive synthesizer for organ-like sounds
- *
- * Copyright (c) 2006-2008 Andreas Brandmaier <andy/at/brandmaier/dot/de>
- *
- * This file is part of LMMS - https://lmms.io
- *
- * This program is free software; you can redistribute it and/or
- * modify it under the terms of the GNU General Public
- * License as published by the Free Software Foundation; either
- * version 2 of the License, or (at your option) any later version.
- *
- * 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 (see COPYING); if not, write to the
- * Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
- * Boston, MA 02110-1301 USA.
- *
- */
- #include "organic.h"
- #include <QDomElement>
- #include <QPainter>
- #include "Engine.h"
- #include "AudioEngine.h"
- #include "InstrumentTrack.h"
- #include "Knob.h"
- #include "NotePlayHandle.h"
- #include "Oscillator.h"
- #include "PixmapButton.h"
- #include "ToolTip.h"
- #include "embed.h"
- #include "plugin_export.h"
- extern "C"
- {
- Plugin::Descriptor PLUGIN_EXPORT organic_plugin_descriptor =
- {
- STRINGIFY( PLUGIN_NAME ),
- "Organic",
- QT_TRANSLATE_NOOP( "PluginBrowser",
- "Additive Synthesizer for organ-like sounds" ),
- "Andreas Brandmaier <andreas/at/brandmaier.de>",
- 0x0100,
- Plugin::Instrument,
- new PluginPixmapLoader( "logo" ),
- nullptr,
- nullptr,
- } ;
- }
- QPixmap * organicInstrumentView::s_artwork = nullptr;
- float * organicInstrument::s_harmonics = nullptr;
- /***********************************************************************
- *
- * class OrganicInstrument
- *
- * lmms - plugin
- *
- ***********************************************************************/
- organicInstrument::organicInstrument( InstrumentTrack * _instrument_track ) :
- Instrument( _instrument_track, &organic_plugin_descriptor ),
- m_modulationAlgo( Oscillator::SignalMix, Oscillator::SignalMix, Oscillator::SignalMix),
- m_fx1Model( 0.0f, 0.0f, 0.99f, 0.01f , this, tr( "Distortion" ) ),
- m_volModel( 100.0f, 0.0f, 200.0f, 1.0f, this, tr( "Volume" ) )
- {
- m_numOscillators = NUM_OSCILLATORS;
- m_osc = new OscillatorObject*[ m_numOscillators ];
- for (int i=0; i < m_numOscillators; i++)
- {
- m_osc[i] = new OscillatorObject( this, i );
- m_osc[i]->m_numOscillators = m_numOscillators;
- // Connect events
- connect( &m_osc[i]->m_oscModel, SIGNAL( dataChanged() ),
- m_osc[i], SLOT ( oscButtonChanged() ) );
- connect( &m_osc[i]->m_harmModel, SIGNAL( dataChanged() ),
- m_osc[i], SLOT( updateDetuning() ) );
- connect( &m_osc[i]->m_volModel, SIGNAL( dataChanged() ),
- m_osc[i], SLOT( updateVolume() ) );
- connect( &m_osc[i]->m_panModel, SIGNAL( dataChanged() ),
- m_osc[i], SLOT( updateVolume() ) );
- connect( &m_osc[i]->m_detuneModel, SIGNAL( dataChanged() ),
- m_osc[i], SLOT( updateDetuning() ) );
- m_osc[i]->updateVolume();
- }
- /* m_osc[0]->m_harmonic = log2f( 0.5f ); // one octave below
- m_osc[1]->m_harmonic = log2f( 0.75f ); // a fifth below
- m_osc[2]->m_harmonic = log2f( 1.0f ); // base freq
- m_osc[3]->m_harmonic = log2f( 2.0f ); // first overtone
- m_osc[4]->m_harmonic = log2f( 3.0f ); // second overtone
- m_osc[5]->m_harmonic = log2f( 4.0f ); // .
- m_osc[6]->m_harmonic = log2f( 5.0f ); // .
- m_osc[7]->m_harmonic = log2f( 6.0f ); // .*/
-
- if( s_harmonics == nullptr )
- {
- s_harmonics = new float[ NUM_HARMONICS ];
- s_harmonics[0] = log2f( 0.5f );
- s_harmonics[1] = log2f( 0.75f );
- s_harmonics[2] = log2f( 1.0f );
- s_harmonics[3] = log2f( 2.0f );
- s_harmonics[4] = log2f( 3.0f );
- s_harmonics[5] = log2f( 4.0f );
- s_harmonics[6] = log2f( 5.0f );
- s_harmonics[7] = log2f( 6.0f );
- s_harmonics[8] = log2f( 7.0f );
- s_harmonics[9] = log2f( 8.0f );
- s_harmonics[10] = log2f( 9.0f );
- s_harmonics[11] = log2f( 10.0f );
- s_harmonics[12] = log2f( 11.0f );
- s_harmonics[13] = log2f( 12.0f );
- s_harmonics[14] = log2f( 13.0f );
- s_harmonics[15] = log2f( 14.0f );
- s_harmonics[16] = log2f( 15.0f );
- s_harmonics[17] = log2f( 16.0f );
- }
- for (int i=0; i < m_numOscillators; i++) {
- m_osc[i]->updateVolume();
- m_osc[i]->updateDetuning();
- }
-
- connect( Engine::audioEngine(), SIGNAL( sampleRateChanged() ),
- this, SLOT( updateAllDetuning() ) );
- }
- organicInstrument::~organicInstrument()
- {
- delete[] m_osc;
- }
- void organicInstrument::saveSettings( QDomDocument & _doc, QDomElement & _this )
- {
- _this.setAttribute( "num_osc", QString::number( m_numOscillators ) );
- m_fx1Model.saveSettings( _doc, _this, "foldback" );
- m_volModel.saveSettings( _doc, _this, "vol" );
- for( int i = 0; i < m_numOscillators; ++i )
- {
- QString is = QString::number( i );
- m_osc[i]->m_volModel.saveSettings( _doc, _this, "vol" + is );
- m_osc[i]->m_panModel.saveSettings( _doc, _this, "pan" + is );
- m_osc[i]->m_harmModel.saveSettings( _doc, _this, "newharmonic" + is );
- m_osc[i]->m_detuneModel.saveSettings( _doc, _this, "newdetune"
- + is );
- m_osc[i]->m_oscModel.saveSettings( _doc, _this, "wavetype"
- + is );
- }
- }
- void organicInstrument::loadSettings( const QDomElement & _this )
- {
- // m_numOscillators = _this.attribute( "num_osc" ).
- // toInt();
- for( int i = 0; i < m_numOscillators; ++i )
- {
- QString is = QString::number( i );
- m_osc[i]->m_volModel.loadSettings( _this, "vol" + is );
- if( _this.hasAttribute( "detune" + is ) )
- {
- m_osc[i]->m_detuneModel.setValue( _this.attribute( "detune" ).toInt() * 12 );
- }
- else
- {
- m_osc[i]->m_detuneModel.loadSettings( _this, "newdetune" + is );
- }
- m_osc[i]->m_panModel.loadSettings( _this, "pan" + is );
- m_osc[i]->m_oscModel.loadSettings( _this, "wavetype" + is );
-
- if( _this.hasAttribute( "newharmonic" + is ) )
- {
- m_osc[i]->m_harmModel.loadSettings( _this, "newharmonic" + is );
- }
- else
- {
- m_osc[i]->m_harmModel.setValue( static_cast<float>( i ) );
- }
- }
-
- m_volModel.loadSettings( _this, "vol" );
- m_fx1Model.loadSettings( _this, "foldback" );
- }
- QString organicInstrument::nodeName() const
- {
- return( organic_plugin_descriptor.name );
- }
- void organicInstrument::playNote( NotePlayHandle * _n,
- sampleFrame * _working_buffer )
- {
- const fpp_t frames = _n->framesLeftForCurrentPeriod();
- const f_cnt_t offset = _n->noteOffset();
-
- if( _n->totalFramesPlayed() == 0 || _n->m_pluginData == nullptr )
- {
- Oscillator * oscs_l[NUM_OSCILLATORS];
- Oscillator * oscs_r[NUM_OSCILLATORS];
- _n->m_pluginData = new oscPtr;
- for( int i = m_numOscillators - 1; i >= 0; --i )
- {
- static_cast<oscPtr *>( _n->m_pluginData )->phaseOffsetLeft[i]
- = rand() / ( RAND_MAX + 1.0f );
- static_cast<oscPtr *>( _n->m_pluginData )->phaseOffsetRight[i]
- = rand() / ( RAND_MAX + 1.0f );
-
- // initialise ocillators
-
- if( i == m_numOscillators - 1 )
- {
- // create left oscillator
- oscs_l[i] = new Oscillator(
- &m_osc[i]->m_waveShape,
- &m_modulationAlgo,
- _n->frequency(),
- m_osc[i]->m_detuningLeft,
- static_cast<oscPtr *>( _n->m_pluginData )->phaseOffsetLeft[i],
- m_osc[i]->m_volumeLeft );
- // create right oscillator
- oscs_r[i] = new Oscillator(
- &m_osc[i]->m_waveShape,
- &m_modulationAlgo,
- _n->frequency(),
- m_osc[i]->m_detuningRight,
- static_cast<oscPtr *>( _n->m_pluginData )->phaseOffsetRight[i],
- m_osc[i]->m_volumeRight );
- }
- else
- {
- // create left oscillator
- oscs_l[i] = new Oscillator(
- &m_osc[i]->m_waveShape,
- &m_modulationAlgo,
- _n->frequency(),
- m_osc[i]->m_detuningLeft,
- static_cast<oscPtr *>( _n->m_pluginData )->phaseOffsetLeft[i],
- m_osc[i]->m_volumeLeft,
- oscs_l[i + 1] );
- // create right oscillator
- oscs_r[i] = new Oscillator(
- &m_osc[i]->m_waveShape,
- &m_modulationAlgo,
- _n->frequency(),
- m_osc[i]->m_detuningRight,
- static_cast<oscPtr *>( _n->m_pluginData )->phaseOffsetRight[i],
- m_osc[i]->m_volumeRight,
- oscs_r[i + 1] );
- }
-
-
- }
- static_cast<oscPtr *>( _n->m_pluginData )->oscLeft = oscs_l[0];
- static_cast<oscPtr *>( _n->m_pluginData )->oscRight = oscs_r[0];
- }
- Oscillator * osc_l = static_cast<oscPtr *>( _n->m_pluginData )->oscLeft;
- Oscillator * osc_r = static_cast<oscPtr *>( _n->m_pluginData)->oscRight;
- osc_l->update( _working_buffer + offset, frames, 0 );
- osc_r->update( _working_buffer + offset, frames, 1 );
- // -- fx section --
-
- // fxKnob is [0;1]
- float t = m_fx1Model.value();
-
- for (int i=0 ; i < frames + offset ; i++)
- {
- _working_buffer[i][0] = waveshape( _working_buffer[i][0], t ) *
- m_volModel.value() / 100.0f;
- _working_buffer[i][1] = waveshape( _working_buffer[i][1], t ) *
- m_volModel.value() / 100.0f;
- }
-
- // -- --
- instrumentTrack()->processAudioBuffer( _working_buffer, frames + offset, _n );
- }
- void organicInstrument::deleteNotePluginData( NotePlayHandle * _n )
- {
- delete static_cast<Oscillator *>( static_cast<oscPtr *>(
- _n->m_pluginData )->oscLeft );
- delete static_cast<Oscillator *>( static_cast<oscPtr *>(
- _n->m_pluginData )->oscRight );
-
- delete static_cast<oscPtr *>( _n->m_pluginData );
- }
- /*float inline organicInstrument::foldback(float in, float threshold)
- {
- if (in>threshold || in<-threshold)
- {
- in= fabs(fabs(fmod(in - threshold, threshold*4)) - threshold*2) - threshold;
- }
- return in;
- }
- */
- float inline organicInstrument::waveshape(float in, float amount)
- {
- float k = 2.0f * amount / ( 1.0f - amount );
- return( ( 1.0f + k ) * in / ( 1.0f + k * fabs( in ) ) );
- }
- void organicInstrument::randomiseSettings()
- {
- for( int i = 0; i < m_numOscillators; i++ )
- {
- m_osc[i]->m_volModel.setValue( intRand( 0, 100 ) );
- m_osc[i]->m_detuneModel.setValue( intRand( -5, 5 ) );
- m_osc[i]->m_panModel.setValue( 0 );
- m_osc[i]->m_oscModel.setValue( intRand( 0, 5 ) );
- }
- }
- void organicInstrument::updateAllDetuning()
- {
- for( int i = 0; i < m_numOscillators; ++i )
- {
- m_osc[i]->updateDetuning();
- }
- }
- int organicInstrument::intRand( int min, int max )
- {
- // int randn = min+int((max-min)*rand()/(RAND_MAX + 1.0));
- // cout << randn << endl;
- int randn = ( rand() % (max - min) ) + min;
- return( randn );
- }
- PluginView * organicInstrument::instantiateView( QWidget * _parent )
- {
- return( new organicInstrumentView( this, _parent ) );
- }
- class organicKnob : public Knob
- {
- public:
- organicKnob( QWidget * _parent ) :
- Knob( knobStyled, _parent )
- {
- setFixedSize( 21, 21 );
- }
- };
- organicInstrumentView::organicInstrumentView( Instrument * _instrument,
- QWidget * _parent ) :
- InstrumentViewFixedSize( _instrument, _parent ),
- m_oscKnobs( nullptr )
- {
- organicInstrument * oi = castModel<organicInstrument>();
- setAutoFillBackground( true );
- QPalette pal;
- pal.setBrush( backgroundRole(), PLUGIN_NAME::getIconPixmap(
- "artwork" ) );
- setPalette( pal );
- // setup knob for FX1
- m_fx1Knob = new organicKnob( this );
- m_fx1Knob->move( 15, 201 );
- m_fx1Knob->setFixedSize( 37, 47 );
- m_fx1Knob->setHintText( tr( "Distortion:" ), QString() );
- m_fx1Knob->setObjectName( "fx1Knob" );
- // setup volume-knob
- m_volKnob = new organicKnob( this );
- m_volKnob->setVolumeKnob( true );
- m_volKnob->move( 60, 201 );
- m_volKnob->setFixedSize( 37, 47 );
- m_volKnob->setHintText( tr( "Volume:" ), "%" );
- m_volKnob->setObjectName( "volKnob" );
- // randomise
- m_randBtn = new PixmapButton( this, tr( "Randomise" ) );
- m_randBtn->move( 148, 224 );
- m_randBtn->setActiveGraphic( PLUGIN_NAME::getIconPixmap(
- "randomise_pressed" ) );
- m_randBtn->setInactiveGraphic( PLUGIN_NAME::getIconPixmap(
- "randomise" ) );
-
- connect( m_randBtn, SIGNAL ( clicked() ),
- oi, SLOT( randomiseSettings() ) );
- if( s_artwork == nullptr )
- {
- s_artwork = new QPixmap( PLUGIN_NAME::getIconPixmap(
- "artwork" ) );
- }
- }
- organicInstrumentView::~organicInstrumentView()
- {
- delete[] m_oscKnobs;
- }
- void organicInstrumentView::modelChanged()
- {
- organicInstrument * oi = castModel<organicInstrument>();
-
- const float y=91.0f;
- const float rowHeight = 26.0f;
- const float x=53.0f;
- const float colWidth = 24.0f;
- m_numOscillators = oi->m_numOscillators;
-
- m_fx1Knob->setModel( &oi->m_fx1Model );
- m_volKnob->setModel( &oi->m_volModel );
- if( m_oscKnobs != nullptr )
- {
- delete[] m_oscKnobs;
- }
-
- m_oscKnobs = new OscillatorKnobs[ m_numOscillators ];
- // Create knobs, now that we know how many to make
- for( int i = 0; i < m_numOscillators; ++i )
- {
- // setup harmonic knob
- Knob * harmKnob = new organicKnob( this );
- harmKnob->move( x + i * colWidth, y - rowHeight );
- harmKnob->setObjectName( "harmKnob" );
- connect( &oi->m_osc[i]->m_harmModel, SIGNAL( dataChanged() ),
- this, SLOT( updateKnobHint() ) );
-
- // setup waveform-knob
- Knob * oscKnob = new organicKnob( this );
- oscKnob->move( x + i * colWidth, y );
- connect( &oi->m_osc[i]->m_oscModel, SIGNAL( dataChanged() ),
- this, SLOT( updateKnobHint() ) );
- oscKnob->setHintText( tr( "Osc %1 waveform:" ).arg( i + 1 ), QString() );
-
- // setup volume-knob
- Knob * volKnob = new Knob( knobStyled, this );
- volKnob->setVolumeKnob( true );
- volKnob->move( x + i * colWidth, y + rowHeight*1 );
- volKnob->setFixedSize( 21, 21 );
- volKnob->setHintText( tr( "Osc %1 volume:" ).arg(
- i + 1 ), "%" );
-
- // setup panning-knob
- Knob * panKnob = new organicKnob( this );
- panKnob->move( x + i * colWidth, y + rowHeight*2 );
- panKnob->setHintText( tr("Osc %1 panning:").arg(
- i + 1 ), "" );
-
- // setup knob for fine-detuning
- Knob * detuneKnob = new organicKnob( this );
- detuneKnob->move( x + i * colWidth, y + rowHeight*3 );
- detuneKnob->setHintText( tr( "Osc %1 stereo detuning" ).arg( i + 1 )
- , " " +
- tr( "cents" ) );
- m_oscKnobs[i] = OscillatorKnobs( harmKnob, volKnob, oscKnob, panKnob, detuneKnob );
- // Attach to models
- m_oscKnobs[i].m_harmKnob->setModel( &oi->m_osc[i]->m_harmModel );
- m_oscKnobs[i].m_volKnob->setModel( &oi->m_osc[i]->m_volModel );
- m_oscKnobs[i].m_oscKnob->setModel( &oi->m_osc[i]->m_oscModel );
- m_oscKnobs[i].m_panKnob->setModel( &oi->m_osc[i]->m_panModel );
- m_oscKnobs[i].m_detuneKnob->setModel( &oi->m_osc[i]->m_detuneModel );
- }
- updateKnobHint();
- }
- void organicInstrumentView::updateKnobHint()
- {
- organicInstrument * oi = castModel<organicInstrument>();
- for( int i = 0; i < m_numOscillators; ++i )
- {
- const float harm = oi->m_osc[i]->m_harmModel.value();
- const float wave = oi->m_osc[i]->m_oscModel.value();
-
- m_oscKnobs[i].m_harmKnob->setHintText( tr( "Osc %1 harmonic:" ).arg( i + 1 ), " (" +
- HARMONIC_NAMES[ static_cast<int>( harm ) ] + ")" );
- m_oscKnobs[i].m_oscKnob->setHintText( tr( "Osc %1 waveform:" ).arg( i + 1 ), " (" +
- WAVEFORM_NAMES[ static_cast<int>( wave ) ] + ")" );
- }
- }
- OscillatorObject::OscillatorObject( Model * _parent, int _index ) :
- Model( _parent ),
- m_waveShape( Oscillator::SineWave, 0, Oscillator::NumWaveShapes-1, this ),
- m_oscModel( 0.0f, 0.0f, 5.0f, 1.0f,
- this, tr( "Osc %1 waveform" ).arg( _index + 1 ) ),
- m_harmModel( static_cast<float>( _index ), 0.0f, 17.0f, 1.0f,
- this, tr( "Osc %1 harmonic" ).arg( _index + 1 ) ),
- m_volModel( 100.0f, 0.0f, 100.0f, 1.0f,
- this, tr( "Osc %1 volume" ).arg( _index + 1 ) ),
- m_panModel( DefaultPanning, PanningLeft, PanningRight, 1.0f,
- this, tr( "Osc %1 panning" ).arg( _index + 1 ) ),
- m_detuneModel( 0.0f, -1200.0f, 1200.0f, 1.0f,
- this, tr( "Osc %1 fine detuning left" ).arg( _index + 1 ) )
- {
- }
- OscillatorObject::~OscillatorObject()
- {
- }
- void OscillatorObject::oscButtonChanged()
- {
- static Oscillator::WaveShapes shapes[] =
- {
- Oscillator::SineWave,
- Oscillator::SawWave,
- Oscillator::SquareWave,
- Oscillator::TriangleWave,
- Oscillator::MoogSawWave,
- Oscillator::ExponentialWave
- } ;
- m_waveShape.setValue( shapes[(int)roundf( m_oscModel.value() )] );
- }
- void OscillatorObject::updateVolume()
- {
- m_volumeLeft = ( 1.0f - m_panModel.value() / (float)PanningRight )
- * m_volModel.value() / m_numOscillators / 100.0f;
- m_volumeRight = ( 1.0f + m_panModel.value() / (float)PanningRight )
- * m_volModel.value() / m_numOscillators / 100.0f;
- }
- void OscillatorObject::updateDetuning()
- {
- m_detuningLeft = powf( 2.0f, organicInstrument::s_harmonics[ static_cast<int>( m_harmModel.value() ) ]
- + (float)m_detuneModel.value() * CENT ) /
- Engine::audioEngine()->processingSampleRate();
- m_detuningRight = powf( 2.0f, organicInstrument::s_harmonics[ static_cast<int>( m_harmModel.value() ) ]
- - (float)m_detuneModel.value() * CENT ) /
- Engine::audioEngine()->processingSampleRate();
- }
- extern "C"
- {
- // necessary for getting instance out of shared lib
- PLUGIN_EXPORT Plugin * lmms_plugin_main( Model *m, void * )
- {
- return( new organicInstrument( static_cast<InstrumentTrack *>( m ) ) );
- }
- }
- /*
- * some notes & ideas for the future of this plugin:
- *
- * - 32.692 Hz in the bass to 5919.85 Hz of treble in a Hammond organ
- * => implement harmonic foldback
- *
- m_osc[i].m_oscModel->setInitValue( 0.0f );
- * - randomize preset
- */
|