mirror of
https://github.com/LMMS/lmms.git
synced 2026-04-03 05:44:38 -04:00
482 lines
12 KiB
C++
482 lines
12 KiB
C++
/*
|
|
* FxMixerView.cpp - effect-mixer-view for LMMS
|
|
*
|
|
* Copyright (c) 2008-2014 Tobias Doerffel <tobydox/at/users.sourceforge.net>
|
|
*
|
|
* This file is part of Linux MultiMedia Studio - http://lmms.sourceforge.net
|
|
*
|
|
* 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 <QtGlobal>
|
|
#include <QDebug>
|
|
|
|
#include <QtGui/QButtonGroup>
|
|
#include <QtGui/QInputDialog>
|
|
#include <QtGui/QLayout>
|
|
#include <QtGui/QMdiArea>
|
|
#include <QtGui/QMdiSubWindow>
|
|
#include <QtGui/QPainter>
|
|
#include <QtGui/QPushButton>
|
|
#include <QtGui/QToolButton>
|
|
#include <QtGui/QStackedLayout>
|
|
#include <QtGui/QScrollArea>
|
|
#include <QtGui/QStyle>
|
|
#include <QtGui/QKeyEvent>
|
|
|
|
#include "FxMixerView.h"
|
|
#include "knob.h"
|
|
#include "engine.h"
|
|
#include "embed.h"
|
|
#include "MainWindow.h"
|
|
#include "gui_templates.h"
|
|
#include "InstrumentTrack.h"
|
|
#include "song.h"
|
|
#include "bb_track_container.h"
|
|
|
|
FxMixerView::FxMixerView() :
|
|
QWidget(),
|
|
ModelView( NULL, this ),
|
|
SerializingObjectHook()
|
|
{
|
|
FxMixer * m = engine::fxMixer();
|
|
m->setHook( this );
|
|
|
|
//QPalette pal = palette();
|
|
//pal.setColor( QPalette::Background, QColor( 72, 76, 88 ) );
|
|
//setPalette( pal );
|
|
setAutoFillBackground( true );
|
|
|
|
setWindowTitle( tr( "FX-Mixer" ) );
|
|
setWindowIcon( embed::getIconPixmap( "fx_mixer" ) );
|
|
|
|
// main-layout
|
|
QHBoxLayout * ml = new QHBoxLayout;
|
|
|
|
// Channel area
|
|
m_channelAreaWidget = new QWidget;
|
|
chLayout = new QHBoxLayout(m_channelAreaWidget);
|
|
chLayout->setSizeConstraint(QLayout::SetMinimumSize);
|
|
chLayout->setSpacing( 0 );
|
|
chLayout->setMargin( 0 );
|
|
m_channelAreaWidget->setLayout(chLayout);
|
|
|
|
// add master channel
|
|
m_fxChannelViews.resize(m->numChannels());
|
|
m_fxChannelViews[0] = new FxChannelView(this, this, 0);
|
|
|
|
FxChannelView * masterView = m_fxChannelViews[0];
|
|
ml->addWidget( masterView->m_fxLine, 0, Qt::AlignTop );
|
|
|
|
QSize fxLineSize = masterView->m_fxLine->size();
|
|
|
|
// add mixer channels
|
|
for( int i = 1; i < m_fxChannelViews.size(); ++i )
|
|
{
|
|
m_fxChannelViews[i] = new FxChannelView(m_channelAreaWidget, this, i);
|
|
chLayout->addWidget(m_fxChannelViews[i]->m_fxLine);
|
|
}
|
|
|
|
// add the scrolling section to the main layout
|
|
// class solely for scroll area to pass key presses down
|
|
class ChannelArea : public QScrollArea
|
|
{
|
|
public:
|
|
ChannelArea(QWidget * parent, FxMixerView * mv) :
|
|
QScrollArea(parent), m_mv(mv) {}
|
|
~ChannelArea() {}
|
|
virtual void keyPressEvent(QKeyEvent * e)
|
|
{
|
|
m_mv->keyPressEvent(e);
|
|
}
|
|
private:
|
|
FxMixerView * m_mv;
|
|
};
|
|
channelArea = new ChannelArea(this, this);
|
|
channelArea->setWidget(m_channelAreaWidget);
|
|
channelArea->setVerticalScrollBarPolicy( Qt::ScrollBarAlwaysOff );
|
|
channelArea->setFrameStyle( QFrame::NoFrame );
|
|
channelArea->setMinimumWidth( fxLineSize.width() * 6 );
|
|
channelArea->setFixedHeight( fxLineSize.height() +
|
|
style()->pixelMetric( QStyle::PM_ScrollBarExtent ) );
|
|
ml->addWidget(channelArea);
|
|
|
|
// show the add new effect channel button
|
|
QPushButton * newChannelBtn = new QPushButton("new", this );
|
|
newChannelBtn->setFont(QFont("sans-serif", 10, 1, false));
|
|
newChannelBtn->setFixedSize(fxLineSize);
|
|
connect( newChannelBtn, SIGNAL(clicked()), this, SLOT(addNewChannel()));
|
|
ml->addWidget( newChannelBtn, 0, Qt::AlignTop );
|
|
|
|
|
|
// Create EffectRack and set initial index to master channel
|
|
m_rackView = new EffectRackView( &m->m_fxChannels[0]->m_fxChain, this );
|
|
ml->addWidget( m_rackView, 0, Qt::AlignTop );
|
|
setCurrentFxLine( m_fxChannelViews[0]->m_fxLine );
|
|
|
|
setLayout( ml );
|
|
updateGeometry();
|
|
|
|
// timer for updating faders
|
|
connect( engine::mainWindow(), SIGNAL( periodicUpdate() ),
|
|
this, SLOT( updateFaders() ) );
|
|
|
|
|
|
// add ourself to workspace
|
|
QMdiSubWindow * subWin =
|
|
engine::mainWindow()->workspace()->addSubWindow( this );
|
|
Qt::WindowFlags flags = subWin->windowFlags();
|
|
flags &= ~Qt::WindowMaximizeButtonHint;
|
|
subWin->setWindowFlags( flags );
|
|
layout()->setSizeConstraint( QLayout::SetMinAndMaxSize );
|
|
subWin->layout()->setSizeConstraint( QLayout::SetMinAndMaxSize );
|
|
|
|
parentWidget()->setAttribute( Qt::WA_DeleteOnClose, false );
|
|
parentWidget()->move( 5, 310 );
|
|
|
|
// we want to receive dataChanged-signals in order to update
|
|
setModel( m );
|
|
}
|
|
|
|
FxMixerView::~FxMixerView()
|
|
{
|
|
}
|
|
|
|
|
|
|
|
void FxMixerView::addNewChannel()
|
|
{
|
|
// add new fx mixer channel and redraw the form.
|
|
FxMixer * mix = engine::fxMixer();
|
|
|
|
int newChannelIndex = mix->createChannel();
|
|
m_fxChannelViews.push_back(new FxChannelView(m_channelAreaWidget, this,
|
|
newChannelIndex));
|
|
chLayout->addWidget(m_fxChannelViews[newChannelIndex]->m_fxLine);
|
|
|
|
updateFxLine(newChannelIndex);
|
|
|
|
updateMaxChannelSelector();
|
|
}
|
|
|
|
|
|
void FxMixerView::refreshDisplay()
|
|
{
|
|
// delete all views and re-add them
|
|
for( int i = 1; i<m_fxChannelViews.size(); ++i )
|
|
{
|
|
chLayout->removeWidget(m_fxChannelViews[i]->m_fxLine);
|
|
delete m_fxChannelViews[i]->m_fader;
|
|
delete m_fxChannelViews[i]->m_muteBtn;
|
|
delete m_fxChannelViews[i]->m_fxLine;
|
|
delete m_fxChannelViews[i];
|
|
}
|
|
m_channelAreaWidget->adjustSize();
|
|
|
|
// re-add the views
|
|
m_fxChannelViews.resize(engine::fxMixer()->numChannels());
|
|
for( int i = 1; i < m_fxChannelViews.size(); ++i )
|
|
{
|
|
m_fxChannelViews[i] = new FxChannelView(m_channelAreaWidget, this, i);
|
|
chLayout->addWidget(m_fxChannelViews[i]->m_fxLine);
|
|
}
|
|
|
|
updateMaxChannelSelector();
|
|
}
|
|
|
|
|
|
// update the and max. channel number for every instrument
|
|
void FxMixerView::updateMaxChannelSelector()
|
|
{
|
|
QVector<track *> songTrackList = engine::getSong()->tracks();
|
|
QVector<track *> bbTrackList = engine::getBBTrackContainer()->tracks();
|
|
|
|
QVector<track *> trackLists[] = {songTrackList, bbTrackList};
|
|
for(int tl=0; tl<2; ++tl)
|
|
{
|
|
QVector<track *> trackList = trackLists[tl];
|
|
for(int i=0; i<trackList.size(); ++i)
|
|
{
|
|
if( trackList[i]->type() == track::InstrumentTrack )
|
|
{
|
|
InstrumentTrack * inst = (InstrumentTrack *) trackList[i];
|
|
inst->effectChannelModel()->setRange(0,
|
|
m_fxChannelViews.size()-1,1);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void FxMixerView::saveSettings( QDomDocument & _doc, QDomElement & _this )
|
|
{
|
|
MainWindow::saveWidgetState( this, _this );
|
|
}
|
|
|
|
|
|
|
|
|
|
void FxMixerView::loadSettings( const QDomElement & _this )
|
|
{
|
|
MainWindow::restoreWidgetState( this, _this );
|
|
}
|
|
|
|
|
|
FxMixerView::FxChannelView::FxChannelView(QWidget * _parent, FxMixerView * _mv,
|
|
int _chIndex )
|
|
{
|
|
m_fxLine = new FxLine(_parent, _mv, _chIndex);
|
|
|
|
FxMixer * m = engine::fxMixer();
|
|
m_fader = new fader( &m->effectChannel(_chIndex)->m_volumeModel,
|
|
tr( "FX Fader %1" ).arg( _chIndex ), m_fxLine );
|
|
m_fader->move( 15-m_fader->width()/2,
|
|
m_fxLine->height()-
|
|
m_fader->height()-5 );
|
|
|
|
m_muteBtn = new pixmapButton( m_fxLine, tr( "Mute" ) );
|
|
m_muteBtn->setModel( &m->effectChannel(_chIndex)->m_muteModel );
|
|
m_muteBtn->setActiveGraphic(
|
|
embed::getIconPixmap( "led_off" ) );
|
|
m_muteBtn->setInactiveGraphic(
|
|
embed::getIconPixmap( "led_green" ) );
|
|
m_muteBtn->setCheckable( true );
|
|
m_muteBtn->move( 9, m_fader->y()-16);
|
|
toolTip::add( m_muteBtn, tr( "Mute this FX channel" ) );
|
|
}
|
|
|
|
|
|
void FxMixerView::setCurrentFxLine( FxLine * _line )
|
|
{
|
|
// select
|
|
m_currentFxLine = _line;
|
|
m_rackView->setModel( &engine::fxMixer()->m_fxChannels[_line->channelIndex()]->m_fxChain );
|
|
|
|
// set up send knob
|
|
for(int i = 0; i < m_fxChannelViews.size(); ++i)
|
|
{
|
|
updateFxLine(i);
|
|
}
|
|
}
|
|
|
|
|
|
void FxMixerView::updateFxLine(int index)
|
|
{
|
|
FxMixer * mix = engine::fxMixer();
|
|
|
|
// does current channel send to this channel?
|
|
int selIndex = m_currentFxLine->channelIndex();
|
|
FxLine * thisLine = m_fxChannelViews[index]->m_fxLine;
|
|
FloatModel * sendModel = mix->channelSendModel(selIndex, index);
|
|
if( sendModel == NULL )
|
|
{
|
|
// does not send, hide send knob
|
|
thisLine->m_sendKnob->setVisible(false);
|
|
}
|
|
else
|
|
{
|
|
// it does send, show knob and connect
|
|
thisLine->m_sendKnob->setVisible(true);
|
|
thisLine->m_sendKnob->setModel(sendModel);
|
|
}
|
|
|
|
// disable the send button if it would cause an infinite loop
|
|
thisLine->m_sendBtn->setVisible(! mix->isInfiniteLoop(selIndex, index));
|
|
thisLine->m_sendBtn->updateLightStatus();
|
|
thisLine->update();
|
|
}
|
|
|
|
|
|
void FxMixerView::deleteChannel(int index)
|
|
{
|
|
// can't delete master
|
|
if( index == 0 ) return;
|
|
|
|
// remember selected line
|
|
int selLine = m_currentFxLine->channelIndex();
|
|
|
|
// delete the real channel
|
|
engine::fxMixer()->deleteChannel(index);
|
|
|
|
// delete the view
|
|
chLayout->removeWidget(m_fxChannelViews[index]->m_fxLine);
|
|
delete m_fxChannelViews[index]->m_fader;
|
|
delete m_fxChannelViews[index]->m_muteBtn;
|
|
delete m_fxChannelViews[index]->m_fxLine;
|
|
delete m_fxChannelViews[index];
|
|
m_channelAreaWidget->adjustSize();
|
|
|
|
// make sure every channel knows what index it is
|
|
for(int i=0; i<m_fxChannelViews.size(); ++i)
|
|
{
|
|
if( i > index )
|
|
{
|
|
m_fxChannelViews[i]->m_fxLine->setChannelIndex(i-1);
|
|
}
|
|
}
|
|
m_fxChannelViews.remove(index);
|
|
|
|
// select the next channel
|
|
if( selLine >= m_fxChannelViews.size() )
|
|
{
|
|
selLine = m_fxChannelViews.size()-1;
|
|
}
|
|
setCurrentFxLine(selLine);
|
|
|
|
updateMaxChannelSelector();
|
|
}
|
|
|
|
|
|
|
|
void FxMixerView::moveChannelLeft(int index)
|
|
{
|
|
// can't move master or first channel left or last channel right
|
|
if( index <= 1 || index >= m_fxChannelViews.size() ) return;
|
|
|
|
int selIndex = m_currentFxLine->channelIndex();
|
|
|
|
FxMixer * mix = engine::fxMixer();
|
|
mix->moveChannelLeft(index);
|
|
|
|
// refresh the two mixer views
|
|
for( int i = index-1; i <= index; ++i )
|
|
{
|
|
// delete the mixer view
|
|
int replaceIndex = chLayout->indexOf(m_fxChannelViews[i]->m_fxLine);
|
|
|
|
chLayout->removeWidget(m_fxChannelViews[i]->m_fxLine);
|
|
delete m_fxChannelViews[i]->m_fader;
|
|
delete m_fxChannelViews[i]->m_muteBtn;
|
|
delete m_fxChannelViews[i]->m_fxLine;
|
|
delete m_fxChannelViews[i];
|
|
|
|
// add it again
|
|
m_fxChannelViews[i] = new FxChannelView(m_channelAreaWidget, this, i);
|
|
chLayout->insertWidget(replaceIndex, m_fxChannelViews[i]->m_fxLine);
|
|
}
|
|
|
|
// keep selected channel
|
|
if( selIndex == index )
|
|
{
|
|
selIndex = index-1;
|
|
}
|
|
else if( selIndex == index - 1 )
|
|
{
|
|
selIndex = index;
|
|
}
|
|
setCurrentFxLine(selIndex);
|
|
}
|
|
|
|
|
|
|
|
void FxMixerView::moveChannelRight(int index)
|
|
{
|
|
moveChannelLeft(index+1);
|
|
}
|
|
|
|
|
|
|
|
void FxMixerView::keyPressEvent(QKeyEvent * e)
|
|
{
|
|
switch(e->key())
|
|
{
|
|
case Qt::Key_Delete:
|
|
deleteChannel(m_currentFxLine->channelIndex());
|
|
break;
|
|
case Qt::Key_Left:
|
|
if( e->modifiers() & Qt::AltModifier )
|
|
{
|
|
moveChannelLeft( m_currentFxLine->channelIndex() );
|
|
}
|
|
else
|
|
{
|
|
// select channel to the left
|
|
setCurrentFxLine( m_currentFxLine->channelIndex()-1 );
|
|
}
|
|
break;
|
|
case Qt::Key_Right:
|
|
if( e->modifiers() & Qt::AltModifier )
|
|
{
|
|
moveChannelRight( m_currentFxLine->channelIndex() );
|
|
}
|
|
else
|
|
{
|
|
// select channel to the right
|
|
setCurrentFxLine( m_currentFxLine->channelIndex()+1 );
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
void FxMixerView::setCurrentFxLine( int _line )
|
|
{
|
|
if( _line >= 0 && _line < m_fxChannelViews.size() )
|
|
{
|
|
setCurrentFxLine( m_fxChannelViews[_line]->m_fxLine );
|
|
}
|
|
}
|
|
|
|
|
|
|
|
void FxMixerView::clear()
|
|
{
|
|
m_rackView->clearViews();
|
|
|
|
engine::fxMixer()->clear();
|
|
|
|
refreshDisplay();
|
|
}
|
|
|
|
|
|
|
|
|
|
void FxMixerView::updateFaders()
|
|
{
|
|
FxMixer * m = engine::fxMixer();
|
|
for( int i = 0; i < m_fxChannelViews.size(); ++i )
|
|
{
|
|
const float opl = m_fxChannelViews[i]->m_fader->getPeak_L();
|
|
const float opr = m_fxChannelViews[i]->m_fader->getPeak_R();
|
|
const float fall_off = 1.2;
|
|
if( m->m_fxChannels[i]->m_peakLeft > opl )
|
|
{
|
|
m_fxChannelViews[i]->m_fader->setPeak_L(
|
|
m->m_fxChannels[i]->m_peakLeft );
|
|
}
|
|
else
|
|
{
|
|
m_fxChannelViews[i]->m_fader->setPeak_L( opl/fall_off );
|
|
}
|
|
if( m->m_fxChannels[i]->m_peakRight > opr )
|
|
{
|
|
m_fxChannelViews[i]->m_fader->setPeak_R(
|
|
m->m_fxChannels[i]->m_peakRight );
|
|
}
|
|
else
|
|
{
|
|
m_fxChannelViews[i]->m_fader->setPeak_R( opr/fall_off );
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
#include "moc_FxMixerView.cxx"
|
|
|