123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677 |
- /*
- * ZynAddSubFx.cpp - ZynAddSubxFX-embedding plugin
- *
- * Copyright (c) 2008-2014 Tobias Doerffel <tobydox/at/users.sourceforge.net>
- *
- * 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 "lmmsconfig.h"
- #include <QDir>
- #include <QDomDocument>
- #include <QTemporaryFile>
- #include <QtGlobal>
- #include <QDropEvent>
- #include <QGridLayout>
- #include <QPushButton>
- #include "ZynAddSubFx.h"
- #include "ConfigManager.h"
- #include "Engine.h"
- #include "Knob.h"
- #include "LedCheckbox.h"
- #include "DataFile.h"
- #include "InstrumentPlayHandle.h"
- #include "InstrumentTrack.h"
- #include "gui_templates.h"
- #include "Song.h"
- #include "StringPairDrag.h"
- #include "RemoteZynAddSubFx.h"
- #include "LocalZynAddSubFx.h"
- #include "AudioEngine.h"
- #include "ControllerConnection.h"
- #include "Clipboard.h"
- #include "embed.h"
- #include "plugin_export.h"
- extern "C"
- {
- Plugin::Descriptor PLUGIN_EXPORT zynaddsubfx_plugin_descriptor =
- {
- STRINGIFY( PLUGIN_NAME ),
- "ZynAddSubFX",
- QT_TRANSLATE_NOOP( "PluginBrowser",
- "Embedded ZynAddSubFX" ),
- "Tobias Doerffel <tobydox/at/users.sf.net>",
- 0x0100,
- Plugin::Instrument,
- new PluginPixmapLoader( "logo" ),
- "xiz",
- nullptr,
- } ;
- }
- ZynAddSubFxRemotePlugin::ZynAddSubFxRemotePlugin() :
- RemotePlugin()
- {
- init( "RemoteZynAddSubFx", false );
- }
- ZynAddSubFxRemotePlugin::~ZynAddSubFxRemotePlugin()
- {
- }
- bool ZynAddSubFxRemotePlugin::processMessage( const message & _m )
- {
- switch( _m.id )
- {
- case IdHideUI:
- emit clickedCloseButton();
- return true;
- default:
- break;
- }
- return RemotePlugin::processMessage( _m );
- }
- ZynAddSubFxInstrument::ZynAddSubFxInstrument(
- InstrumentTrack * _instrumentTrack ) :
- Instrument( _instrumentTrack, &zynaddsubfx_plugin_descriptor ),
- m_hasGUI( false ),
- m_plugin( nullptr ),
- m_remotePlugin( nullptr ),
- m_portamentoModel( 0, 0, 127, 1, this, tr( "Portamento" ) ),
- m_filterFreqModel( 64, 0, 127, 1, this, tr( "Filter frequency" ) ),
- m_filterQModel( 64, 0, 127, 1, this, tr( "Filter resonance" ) ),
- m_bandwidthModel( 64, 0, 127, 1, this, tr( "Bandwidth" ) ),
- m_fmGainModel( 127, 0, 127, 1, this, tr( "FM gain" ) ),
- m_resCenterFreqModel( 64, 0, 127, 1, this, tr( "Resonance center frequency" ) ),
- m_resBandwidthModel( 64, 0, 127, 1, this, tr( "Resonance bandwidth" ) ),
- m_forwardMidiCcModel( true, this, tr( "Forward MIDI control change events" ) )
- {
- initPlugin();
- connect( &m_portamentoModel, SIGNAL( dataChanged() ),
- this, SLOT( updatePortamento() ), Qt::DirectConnection );
- connect( &m_filterFreqModel, SIGNAL( dataChanged() ),
- this, SLOT( updateFilterFreq() ), Qt::DirectConnection );
- connect( &m_filterQModel, SIGNAL( dataChanged() ),
- this, SLOT( updateFilterQ() ), Qt::DirectConnection );
- connect( &m_bandwidthModel, SIGNAL( dataChanged() ),
- this, SLOT( updateBandwidth() ), Qt::DirectConnection );
- connect( &m_fmGainModel, SIGNAL( dataChanged() ),
- this, SLOT( updateFmGain() ), Qt::DirectConnection );
- connect( &m_resCenterFreqModel, SIGNAL( dataChanged() ),
- this, SLOT( updateResCenterFreq() ), Qt::DirectConnection );
- connect( &m_resBandwidthModel, SIGNAL( dataChanged() ),
- this, SLOT( updateResBandwidth() ), Qt::DirectConnection );
- // now we need a play-handle which cares for calling play()
- InstrumentPlayHandle * iph = new InstrumentPlayHandle( this, _instrumentTrack );
- Engine::audioEngine()->addPlayHandle( iph );
- connect( Engine::audioEngine(), SIGNAL( sampleRateChanged() ),
- this, SLOT( reloadPlugin() ) );
- connect( instrumentTrack()->pitchRangeModel(), SIGNAL( dataChanged() ),
- this, SLOT( updatePitchRange() ), Qt::DirectConnection );
- }
- ZynAddSubFxInstrument::~ZynAddSubFxInstrument()
- {
- Engine::audioEngine()->removePlayHandlesOfTypes( instrumentTrack(),
- PlayHandle::TypeNotePlayHandle
- | PlayHandle::TypeInstrumentPlayHandle );
- m_pluginMutex.lock();
- delete m_plugin;
- delete m_remotePlugin;
- m_plugin = nullptr;
- m_remotePlugin = nullptr;
- m_pluginMutex.unlock();
- }
- void ZynAddSubFxInstrument::saveSettings( QDomDocument & _doc,
- QDomElement & _this )
- {
- m_portamentoModel.saveSettings( _doc, _this, "portamento" );
- m_filterFreqModel.saveSettings( _doc, _this, "filterfreq" );
- m_filterQModel.saveSettings( _doc, _this, "filterq" );
- m_bandwidthModel.saveSettings( _doc, _this, "bandwidth" );
- m_fmGainModel.saveSettings( _doc, _this, "fmgain" );
- m_resCenterFreqModel.saveSettings( _doc, _this, "rescenterfreq" );
- m_resBandwidthModel.saveSettings( _doc, _this, "resbandwidth" );
- QString modifiedControllers;
- for( QMap<int, bool>::ConstIterator it = m_modifiedControllers.begin();
- it != m_modifiedControllers.end(); ++it )
- {
- if( it.value() )
- {
- modifiedControllers += QString( "%1," ).arg( it.key() );
- }
- }
- _this.setAttribute( "modifiedcontrollers", modifiedControllers );
- m_forwardMidiCcModel.saveSettings( _doc, _this, "forwardmidicc" );
- QTemporaryFile tf;
- if( tf.open() )
- {
- const std::string fn = QSTR_TO_STDSTR(
- QDir::toNativeSeparators( tf.fileName() ) );
- m_pluginMutex.lock();
- if( m_remotePlugin )
- {
- m_remotePlugin->lock();
- m_remotePlugin->sendMessage( RemotePlugin::message( IdSaveSettingsToFile ).addString( fn ) );
- m_remotePlugin->waitForMessage( IdSaveSettingsToFile );
- m_remotePlugin->unlock();
- }
- else
- {
- m_plugin->saveXML( fn );
- }
- m_pluginMutex.unlock();
- QByteArray a = tf.readAll();
- QDomDocument doc( "mydoc" );
- if( doc.setContent( a ) )
- {
- QDomNode n = _doc.importNode( doc.documentElement(), true );
- _this.appendChild( n );
- }
- }
- }
- void ZynAddSubFxInstrument::loadSettings( const QDomElement & _this )
- {
- if( !_this.hasChildNodes() )
- {
- return;
- }
- m_portamentoModel.loadSettings( _this, "portamento" );
- m_filterFreqModel.loadSettings( _this, "filterfreq" );
- m_filterQModel.loadSettings( _this, "filterq" );
- m_bandwidthModel.loadSettings( _this, "bandwidth" );
- m_fmGainModel.loadSettings( _this, "fmgain" );
- m_resCenterFreqModel.loadSettings( _this, "rescenterfreq" );
- m_resBandwidthModel.loadSettings( _this, "resbandwidth" );
- m_forwardMidiCcModel.loadSettings( _this, "forwardmidicc" );
- QDomDocument doc;
- QDomElement data = _this.firstChildElement( "ZynAddSubFX-data" );
- if( data.isNull() )
- {
- data = _this.firstChildElement();
- }
- doc.appendChild( doc.importNode( data, true ) );
- QTemporaryFile tf;
- if( tf.open() )
- {
- QByteArray a = doc.toString( 0 ).toUtf8();
- tf.write( a );
- tf.flush();
- const std::string fn = QSTR_TO_STDSTR( QDir::toNativeSeparators( tf.fileName() ) );
- m_pluginMutex.lock();
- if( m_remotePlugin )
- {
- m_remotePlugin->lock();
- m_remotePlugin->sendMessage( RemotePlugin::message( IdLoadSettingsFromFile ).addString( fn ) );
- m_remotePlugin->waitForMessage( IdLoadSettingsFromFile );
- m_remotePlugin->unlock();
- }
- else
- {
- m_plugin->loadXML( fn );
- }
- m_pluginMutex.unlock();
- m_modifiedControllers.clear();
- for( const QString & c : _this.attribute( "modifiedcontrollers" ).split( ',' ) )
- {
- if( !c.isEmpty() )
- {
- switch( c.toInt() )
- {
- case C_portamento: updatePortamento(); break;
- case C_filtercutoff: updateFilterFreq(); break;
- case C_filterq: updateFilterQ(); break;
- case C_bandwidth: updateBandwidth(); break;
- case C_fmamp: updateFmGain(); break;
- case C_resonance_center: updateResCenterFreq(); break;
- case C_resonance_bandwidth: updateResBandwidth(); break;
- default:
- break;
- }
- }
- }
- emit settingsChanged();
- }
- emit instrumentTrack()->pitchModel()->dataChanged();
- }
- void ZynAddSubFxInstrument::loadFile( const QString & _file )
- {
- const std::string fn = QSTR_TO_STDSTR( _file );
- if( m_remotePlugin )
- {
- m_remotePlugin->lock();
- m_remotePlugin->sendMessage( RemotePlugin::message( IdLoadPresetFile ).addString( fn ) );
- m_remotePlugin->waitForMessage( IdLoadPresetFile );
- m_remotePlugin->unlock();
- }
- else
- {
- m_pluginMutex.lock();
- m_plugin->loadPreset( fn );
- m_pluginMutex.unlock();
- }
- instrumentTrack()->setName( QFileInfo( _file ).baseName().replace( QRegExp( "^[0-9]{4}-" ), QString() ) );
- m_modifiedControllers.clear();
- emit settingsChanged();
- }
- QString ZynAddSubFxInstrument::nodeName() const
- {
- return zynaddsubfx_plugin_descriptor.name;
- }
- void ZynAddSubFxInstrument::play( sampleFrame * _buf )
- {
- if (!m_pluginMutex.tryLock(Engine::getSong()->isExporting() ? -1 : 0)) {return;}
- if( m_remotePlugin )
- {
- m_remotePlugin->process( nullptr, _buf );
- }
- else
- {
- m_plugin->processAudio( _buf );
- }
- m_pluginMutex.unlock();
- instrumentTrack()->processAudioBuffer( _buf, Engine::audioEngine()->framesPerPeriod(), nullptr );
- }
- bool ZynAddSubFxInstrument::handleMidiEvent( const MidiEvent& event, const TimePos& time, f_cnt_t offset )
- {
- // do not forward external MIDI Control Change events if the according
- // LED is not checked
- if( event.type() == MidiControlChange &&
- event.sourcePort() != this &&
- m_forwardMidiCcModel.value() == false )
- {
- return true;
- }
- MidiEvent localEvent = event;
- localEvent.setChannel( 0 );
- m_pluginMutex.lock();
- if( m_remotePlugin )
- {
- m_remotePlugin->processMidiEvent( localEvent, 0 );
- }
- else
- {
- m_plugin->processMidiEvent( localEvent );
- }
- m_pluginMutex.unlock();
- return true;
- }
- void ZynAddSubFxInstrument::reloadPlugin()
- {
- // save state of current plugin instance
- DataFile m( DataFile::InstrumentTrackSettings );
- saveSettings( m, m.content() );
- // init plugin (will delete current one and create a new instance)
- initPlugin();
- // and load the settings again
- loadSettings( m.content() );
- }
- void ZynAddSubFxInstrument::updatePitchRange()
- {
- m_pluginMutex.lock();
- if( m_remotePlugin )
- {
- m_remotePlugin->sendMessage( RemotePlugin::message( IdZasfSetPitchWheelBendRange ).
- addInt( instrumentTrack()->midiPitchRange() ) );
- }
- else
- {
- m_plugin->setPitchWheelBendRange( instrumentTrack()->midiPitchRange() );
- }
- m_pluginMutex.unlock();
- }
- #define GEN_CC_SLOT(slotname,midictl,modelname) \
- void ZynAddSubFxInstrument::slotname() \
- { \
- sendControlChange( midictl, modelname.value() ); \
- m_modifiedControllers[midictl] = true; \
- }
- GEN_CC_SLOT(updatePortamento,C_portamento,m_portamentoModel);
- GEN_CC_SLOT(updateFilterFreq,C_filtercutoff,m_filterFreqModel);
- GEN_CC_SLOT(updateFilterQ,C_filterq,m_filterQModel);
- GEN_CC_SLOT(updateBandwidth,C_bandwidth,m_bandwidthModel);
- GEN_CC_SLOT(updateFmGain,C_fmamp,m_fmGainModel);
- GEN_CC_SLOT(updateResCenterFreq,C_resonance_center,m_resCenterFreqModel);
- GEN_CC_SLOT(updateResBandwidth,C_resonance_bandwidth,m_resBandwidthModel);
- void ZynAddSubFxInstrument::initPlugin()
- {
- m_pluginMutex.lock();
- delete m_plugin;
- delete m_remotePlugin;
- m_plugin = nullptr;
- m_remotePlugin = nullptr;
- if( m_hasGUI )
- {
- m_remotePlugin = new ZynAddSubFxRemotePlugin();
- m_remotePlugin->lock();
- m_remotePlugin->waitForInitDone( false );
- m_remotePlugin->sendMessage(
- RemotePlugin::message( IdZasfLmmsWorkingDirectory ).
- addString(
- QSTR_TO_STDSTR(
- QString( ConfigManager::inst()->workingDir() ) ) ) );
- m_remotePlugin->sendMessage(
- RemotePlugin::message( IdZasfPresetDirectory ).
- addString(
- QSTR_TO_STDSTR(
- QDir( ConfigManager::inst()->factoryPresetsDir() +
- "/ZynAddSubFX" ).absolutePath() ) ) );
- m_remotePlugin->updateSampleRate( Engine::audioEngine()->processingSampleRate() );
- // temporary workaround until the VST synchronization feature gets stripped out of the RemotePluginClient class
- // causing not to send buffer size information requests
- m_remotePlugin->sendMessage( RemotePlugin::message( IdBufferSizeInformation ).addInt( Engine::audioEngine()->framesPerPeriod() ) );
- m_remotePlugin->showUI();
- m_remotePlugin->unlock();
- }
- else
- {
- m_plugin = new LocalZynAddSubFx;
- m_plugin->setSampleRate( Engine::audioEngine()->processingSampleRate() );
- m_plugin->setBufferSize( Engine::audioEngine()->framesPerPeriod() );
- }
- m_pluginMutex.unlock();
- }
- void ZynAddSubFxInstrument::sendControlChange( MidiControllers midiCtl, float value )
- {
- handleMidiEvent( MidiEvent( MidiControlChange, instrumentTrack()->midiPort()->realOutputChannel(), midiCtl, (int) value, this ) );
- }
- PluginView * ZynAddSubFxInstrument::instantiateView( QWidget * _parent )
- {
- return new ZynAddSubFxView( this, _parent );
- }
- ZynAddSubFxView::ZynAddSubFxView( Instrument * _instrument, QWidget * _parent ) :
- InstrumentViewFixedSize( _instrument, _parent )
- {
- setAutoFillBackground( true );
- QPalette pal;
- pal.setBrush( backgroundRole(), PLUGIN_NAME::getIconPixmap(
- "artwork" ) );
- setPalette( pal );
- QGridLayout * l = new QGridLayout( this );
- l->setContentsMargins( 20, 80, 10, 10 );
- l->setVerticalSpacing( 16 );
- l->setHorizontalSpacing( 10 );
- m_portamento = new Knob( knobBright_26, this );
- m_portamento->setHintText( tr( "Portamento:" ), "" );
- m_portamento->setLabel( tr( "PORT" ) );
- m_filterFreq = new Knob( knobBright_26, this );
- m_filterFreq->setHintText( tr( "Filter frequency:" ), "" );
- m_filterFreq->setLabel( tr( "FREQ" ) );
- m_filterQ = new Knob( knobBright_26, this );
- m_filterQ->setHintText( tr( "Filter resonance:" ), "" );
- m_filterQ->setLabel( tr( "RES" ) );
- m_bandwidth = new Knob( knobBright_26, this );
- m_bandwidth->setHintText( tr( "Bandwidth:" ), "" );
- m_bandwidth->setLabel( tr( "BW" ) );
- m_fmGain = new Knob( knobBright_26, this );
- m_fmGain->setHintText( tr( "FM gain:" ), "" );
- m_fmGain->setLabel( tr( "FM GAIN" ) );
- m_resCenterFreq = new Knob( knobBright_26, this );
- m_resCenterFreq->setHintText( tr( "Resonance center frequency:" ), "" );
- m_resCenterFreq->setLabel( tr( "RES CF" ) );
- m_resBandwidth = new Knob( knobBright_26, this );
- m_resBandwidth->setHintText( tr( "Resonance bandwidth:" ), "" );
- m_resBandwidth->setLabel( tr( "RES BW" ) );
- m_forwardMidiCC = new LedCheckBox( tr( "Forward MIDI control changes" ), this );
- m_toggleUIButton = new QPushButton( tr( "Show GUI" ), this );
- m_toggleUIButton->setCheckable( true );
- m_toggleUIButton->setChecked( false );
- m_toggleUIButton->setIcon( embed::getIconPixmap( "zoom" ) );
- m_toggleUIButton->setFont( pointSize<8>( m_toggleUIButton->font() ) );
- connect( m_toggleUIButton, SIGNAL( toggled( bool ) ), this,
- SLOT( toggleUI() ) );
- l->addWidget( m_toggleUIButton, 0, 0, 1, 4 );
- l->setRowStretch( 1, 5 );
- l->addWidget( m_portamento, 2, 0 );
- l->addWidget( m_filterFreq, 2, 1 );
- l->addWidget( m_filterQ, 2, 2 );
- l->addWidget( m_bandwidth, 2, 3 );
- l->addWidget( m_fmGain, 3, 0 );
- l->addWidget( m_resCenterFreq, 3, 1 );
- l->addWidget( m_resBandwidth, 3, 2 );
- l->addWidget( m_forwardMidiCC, 4, 0, 1, 4 );
- l->setRowStretch( 5, 10 );
- l->setColumnStretch( 4, 10 );
- setAcceptDrops( true );
- }
- ZynAddSubFxView::~ZynAddSubFxView()
- {
- }
- void ZynAddSubFxView::dragEnterEvent( QDragEnterEvent * _dee )
- {
- // For mimeType() and MimeType enum class
- using namespace Clipboard;
- if( _dee->mimeData()->hasFormat( mimeType( MimeType::StringPair ) ) )
- {
- QString txt = _dee->mimeData()->data(
- mimeType( MimeType::StringPair ) );
- if( txt.section( ':', 0, 0 ) == "pluginpresetfile" )
- {
- _dee->acceptProposedAction();
- }
- else
- {
- _dee->ignore();
- }
- }
- else
- {
- _dee->ignore();
- }
- }
- void ZynAddSubFxView::dropEvent( QDropEvent * _de )
- {
- const QString type = StringPairDrag::decodeKey( _de );
- const QString value = StringPairDrag::decodeValue( _de );
- if( type == "pluginpresetfile" )
- {
- castModel<ZynAddSubFxInstrument>()->loadFile( value );
- _de->accept();
- return;
- }
- _de->ignore();
- }
- void ZynAddSubFxView::modelChanged()
- {
- ZynAddSubFxInstrument * m = castModel<ZynAddSubFxInstrument>();
- // set models for controller knobs
- m_portamento->setModel( &m->m_portamentoModel );
- m_filterFreq->setModel( &m->m_filterFreqModel );
- m_filterQ->setModel( &m->m_filterQModel );
- m_bandwidth->setModel( &m->m_bandwidthModel );
- m_fmGain->setModel( &m->m_fmGainModel );
- m_resCenterFreq->setModel( &m->m_resCenterFreqModel );
- m_resBandwidth->setModel( &m->m_resBandwidthModel );
- m_forwardMidiCC->setModel( &m->m_forwardMidiCcModel );
- m_toggleUIButton->setChecked( m->m_hasGUI );
- }
- void ZynAddSubFxView::toggleUI()
- {
- ZynAddSubFxInstrument * model = castModel<ZynAddSubFxInstrument>();
- if( model->m_hasGUI != m_toggleUIButton->isChecked() )
- {
- model->m_hasGUI = m_toggleUIButton->isChecked();
- model->reloadPlugin();
- if( model->m_remotePlugin )
- {
- connect( model->m_remotePlugin, SIGNAL( clickedCloseButton() ),
- m_toggleUIButton, SLOT( toggle() ) );
- }
- }
- }
- extern "C"
- {
- // necessary for getting instance out of shared lib
- PLUGIN_EXPORT Plugin * lmms_plugin_main(Model * m, void *)
- {
- return new ZynAddSubFxInstrument(static_cast<InstrumentTrack *>(m));
- }
- }
|