drumstick 2.10.0
C++ MIDI libraries using Qt objects, idioms, and style.
guiplayer.cpp

SMF playback, graphic user interface program.

SMF playback, graphic user interface program

/*
SMF GUI Player test using the MIDI Sequencer C++ library
Copyright (C) 2006-2024, Pedro Lopez-Cabanillas <plcl@users.sf.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 3 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. If not, see <http://www.gnu.org/licenses/>.
*/
#ifndef INCLUDED_GUIPLAYER_H
#define INCLUDED_GUIPLAYER_H
#include <QMainWindow>
#include <QProgressDialog>
#include <QObject>
#include <QString>
#include <QList>
#include <QHash>
#include <QPointer>
namespace drumstick {
namespace ALSA {
class MidiClient;
class MidiPort;
class MidiQueue;
class SequencerEvent;
class SysExEvent;
}
namespace File {
class QSmf;
class QWrk;
class Rmidi;
}
}
namespace Ui {
class GUIPlayerClass;
}
class Player;
class About;
class Song;
class GUIPlayer : public QMainWindow
{
Q_OBJECT
public:
enum PlayerState {
InvalidState,
EmptyState,
PlayingState,
PausedState,
StoppedState
};
Q_ENUM(PlayerState)
GUIPlayer(QWidget *parent = nullptr, Qt::WindowFlags flags = Qt::Window);
~GUIPlayer();
void appendSMFEvent(drumstick::ALSA::SequencerEvent* ev);
void appendWRKEvent(unsigned long ticks, drumstick::ALSA::SequencerEvent* ev);
void subscribe(const QString& portName);
void updateTimeLabel(int mins, int secs, int cnts);
void updateTempoLabel(float ftempo);
void dragEnterEvent(QDragEnterEvent* event) override;
void dropEvent(QDropEvent* event) override;
void closeEvent(QCloseEvent* event) override;
void openFile(const QString& fileName);
void readSettings();
void writeSettings();
void updateState(PlayerState newState);
void progressDialogInit(const QString& type, int max);
void progressDialogUpdate(int pos);
void progressDialogClose();
static const QString QSTR_DOMAIN;
static const QString QSTR_APPNAME;
public Q_SLOTS:
void about();
void play();
void pause();
void stop();
void open();
void setup();
void tempoReset();
void volumeReset();
void tempoSlider(int value);
void volumeSlider(int value);
void pitchShift(int value);
void songFinished();
void playerStopped();
void sequencerEvent(drumstick::ALSA::SequencerEvent* ev);
/* RMI slots */
void dataHandler(const QString &dataType, const QByteArray &data);
/* SMF slots */
void smfHeaderEvent(int format, int ntrks, int division);
void smfNoteOnEvent(int chan, int pitch, int vol);
void smfNoteOffEvent(int chan, int pitch, int vol);
void smfKeyPressEvent(int chan, int pitch, int press);
void smfCtlChangeEvent(int chan, int ctl, int value);
void smfPitchBendEvent(int chan, int value);
void smfProgramEvent(int chan, int patch);
void smfChanPressEvent(int chan, int press);
void smfSysexEvent(const QByteArray& data);
void smfTempoEvent(int tempo);
void smfErrorHandler(const QString& errorStr);
void smfTrackStarted();
void smfTrackEnded();
void smfUpdateLoadProgress();
/* WRK slots */
void wrkUpdateLoadProgress();
void wrkErrorHandler(const QString& errorStr);
void wrkFileHeader(int verh, int verl);
void wrkEndOfFile();
void wrkStreamEndEvent(long time);
void wrkTrackHeader(const QString& name1, const QString& name2,
int trackno, int channel, int pitch,
int velocity, int port,
bool selected, bool muted, bool loop);
void wrkTimeBase(int timebase);
void wrkNoteEvent(int track, long time, int chan, int pitch, int vol, int dur);
void wrkKeyPressEvent(int track, long time, int chan, int pitch, int press);
void wrkCtlChangeEvent(int track, long time, int chan, int ctl, int value);
void wrkPitchBendEvent(int track, long time, int chan, int value);
void wrkProgramEvent(int track, long time, int chan, int patch);
void wrkChanPressEvent(int track, long time, int chan, int press);
void wrkSysexEvent(int track, long time, int bank);
void wrkSysexEventBank(int bank, const QString& name, bool autosend, int port, const QByteArray& data);
void wrkTempoEvent(long time, int tempo);
void wrkTrackPatch(int track, int patch);
void wrkNewTrackHeader(const QString& name,
int trackno, int channel, int pitch,
int velocity, int port,
bool selected, bool muted, bool loop);
void wrkTrackVol(int track, int vol);
void wrkTrackBank(int track, int bank);
private:
int m_portId;
int m_queueId;
int m_initialTempo;
int m_currentTrack;
float m_tempoFactor;
unsigned long m_tick;
PlayerState m_state;
Player* m_player;
Ui::GUIPlayerClass* m_ui;
QPointer<QProgressDialog> m_pd;
Song* m_song;
QString m_subscription;
QString m_lastDirectory;
QString m_loadingMessages;
QHash<int, drumstick::ALSA::SysExEvent> m_savedSysexEvents;
struct TrackMapRec {
int channel;
int pitch;
int velocity;
};
QHash<int,TrackMapRec> m_trackMap;
};
#endif // INCLUDED_GUIPLAYER_H
Client management.
Definition: alsaclient.h:219
Port management.
Definition: alsaport.h:125
Queue management.
Definition: alsaqueue.h:201
Base class for the event's hierarchy.
Definition: alsaevent.h:68
Standard MIDI Files input/output.
Definition: qsmf.h:103
Cakewalk WRK file format (input only)
Definition: qwrk.h:96
RIFF MIDI file format (input only)
Definition: rmid.h:57
Drumstick common.
Definition: alsaclient.cpp:71
/*
SMF GUI Player test using the MIDI Sequencer C++ library
Copyright (C) 2006-2024, Pedro Lopez-Cabanillas <plcl@users.sf.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 3 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. If not, see <http://www.gnu.org/licenses/>.
*/
#include <QApplication>
#include <QCloseEvent>
#include <QDebug>
#include <QDragEnterEvent>
#include <QDropEvent>
#include <QFileDialog>
#include <QFileInfo>
#include <QInputDialog>
#include <QMessageBox>
#include <QMimeData>
#include <QSettings>
#include <QStatusBar>
#include <QTextCodec>
#include <QToolTip>
#include <QUrl>
#include <qmath.h>
#include "guiplayer.h"
#include "iconutils.h"
#include "player.h"
#include "playerabout.h"
#include "song.h"
#include "ui_guiplayer.h"
#include <drumstick/qsmf.h>
#include <drumstick/qwrk.h>
#include <drumstick/rmid.h>
DISABLE_WARNING_PUSH
DISABLE_WARNING_DEPRECATED_DECLARATIONS
using namespace drumstick;
using namespace ALSA;
using namespace File;
const QString GUIPlayer::QSTR_DOMAIN = QStringLiteral("drumstick.sourceforge.net");
const QString GUIPlayer::QSTR_APPNAME = QStringLiteral("drumstick-guiplayer");
GUIPlayer::GUIPlayer(QWidget *parent, Qt::WindowFlags flags)
: QMainWindow(parent, flags),
m_portId(-1),
m_queueId(-1),
m_initialTempo(0),
m_currentTrack(0),
m_tempoFactor(1.0),
m_tick(0),
m_state(InvalidState),
m_smf(nullptr),
m_wrk(nullptr),
m_Client(nullptr),
m_Port(nullptr),
m_Queue(nullptr),
m_player(nullptr),
m_ui(new Ui::GUIPlayerClass),
m_pd(nullptr),
m_song(new Song)
{
m_ui->setupUi(this);
setAcceptDrops(true);
connect(m_ui->actionAbout, &QAction::triggered, this, &GUIPlayer::about);
connect(m_ui->actionAboutQt, &QAction::triggered, qApp, QApplication::aboutQt);
connect(m_ui->actionPlay, &QAction::triggered, this, &GUIPlayer::play);
connect(m_ui->actionPause, &QAction::triggered, this, &GUIPlayer::pause);
connect(m_ui->actionStop, &QAction::triggered, this, &GUIPlayer::stop);
connect(m_ui->actionOpen, &QAction::triggered, this, &GUIPlayer::open);
connect(m_ui->actionMIDISetup, &QAction::triggered, this, &GUIPlayer::setup);
connect(m_ui->actionQuit, &QAction::triggered, this, &GUIPlayer::close);
connect(m_ui->btnTempo, &QPushButton::clicked, this, &GUIPlayer::tempoReset);
connect(m_ui->btnVolume, &QPushButton::clicked, this, &GUIPlayer::volumeReset);
connect(m_ui->sliderTempo, &QSlider::valueChanged, this, &GUIPlayer::tempoSlider);
connect(m_ui->volumeSlider, &QSlider::valueChanged, this, &GUIPlayer::volumeSlider);
connect(m_ui->spinPitch, QOverload<int>::of(&QSpinBox::valueChanged),
this, &GUIPlayer::pitchShift);
connect(m_ui->toolBar->toggleViewAction(), &QAction::toggled,
m_ui->actionShowToolbar, &QAction::setChecked);
m_ui->actionPlay->setIcon(QIcon(IconUtils::GetPixmap(this, ":/resources/play.png")));
m_ui->actionPlay->setShortcut( Qt::Key_MediaPlay );
m_ui->actionStop->setIcon(QIcon(IconUtils::GetPixmap(this, ":/resources/stop.png")));
m_ui->actionStop->setShortcut( Qt::Key_MediaStop );
m_ui->actionPause->setIcon(QIcon(IconUtils::GetPixmap(this, ":/resources/pause.png")));
m_ui->actionMIDISetup->setIcon(QIcon(IconUtils::GetPixmap(this, ":/resources/setup.png")));
m_Client = new MidiClient(this);
m_Client->open();
m_Client->setPoolOutput(50); // small size, for near real-time pitchShift
m_Client->setClientName("MIDI Player");
connect(m_Client,
&MidiClient::eventReceived,
this,
&GUIPlayer::sequencerEvent,
static_cast<Qt::ConnectionType>(Qt::QueuedConnection | Qt::UniqueConnection));
m_Port = new MidiPort(this);
m_Port->attach( m_Client );
m_Port->setPortName("MIDI Player Output Port");
m_Port->setCapability( SND_SEQ_PORT_CAP_READ |
SND_SEQ_PORT_CAP_SUBS_READ |
SND_SEQ_PORT_CAP_WRITE );
m_Port->setPortType( SND_SEQ_PORT_TYPE_APPLICATION |
SND_SEQ_PORT_TYPE_MIDI_GENERIC );
m_Queue = m_Client->createQueue(QSTR_APPNAME);
m_queueId = m_Queue->getId();
m_portId = m_Port->getPortId();
m_rmi = new Rmidi(this);
connect(m_rmi, &Rmidi::signalRiffData, this, &GUIPlayer::dataHandler);
m_smf = new QSmf(this);
connect(m_smf, &QSmf::signalSMFHeader, this, &GUIPlayer::smfHeaderEvent);
connect(m_smf, &QSmf::signalSMFNoteOn, this, &GUIPlayer::smfNoteOnEvent);
connect(m_smf, &QSmf::signalSMFNoteOff, this, &GUIPlayer::smfNoteOffEvent);
connect(m_smf, &QSmf::signalSMFKeyPress, this, &GUIPlayer::smfKeyPressEvent);
connect(m_smf, &QSmf::signalSMFCtlChange, this, &GUIPlayer::smfCtlChangeEvent);
connect(m_smf, &QSmf::signalSMFPitchBend, this, &GUIPlayer::smfPitchBendEvent);
connect(m_smf, &QSmf::signalSMFProgram, this, &GUIPlayer::smfProgramEvent);
connect(m_smf, &QSmf::signalSMFChanPress, this, &GUIPlayer::smfChanPressEvent);
connect(m_smf, &QSmf::signalSMFSysex, this, &GUIPlayer::smfSysexEvent);
connect(m_smf, &QSmf::signalSMFText, this, &GUIPlayer::smfUpdateLoadProgress);
connect(m_smf, &QSmf::signalSMFTempo, this, &GUIPlayer::smfTempoEvent);
connect(m_smf, &QSmf::signalSMFTrackStart, this, &GUIPlayer::smfUpdateLoadProgress);
connect(m_smf, &QSmf::signalSMFTrackStart, this, &GUIPlayer::smfTrackStarted);
connect(m_smf, &QSmf::signalSMFTrackEnd, this, &GUIPlayer::smfTrackEnded);
connect(m_smf, &QSmf::signalSMFendOfTrack, this, &GUIPlayer::smfUpdateLoadProgress);
connect(m_smf, &QSmf::signalSMFError, this, &GUIPlayer::smfErrorHandler);
m_wrk = new QWrk(this);
m_wrk->setTextCodec(QTextCodec::codecForLocale());
connect(m_wrk, &QWrk::signalWRKError, this, &GUIPlayer::wrkErrorHandler);
connect(m_wrk, &QWrk::signalWRKUnknownChunk, this, &GUIPlayer::wrkUpdateLoadProgress);
connect(m_wrk, &QWrk::signalWRKHeader, this, &GUIPlayer::wrkFileHeader);
connect(m_wrk, &QWrk::signalWRKEnd, this, &GUIPlayer::wrkEndOfFile);
connect(m_wrk, &QWrk::signalWRKStreamEnd, this, &GUIPlayer::wrkStreamEndEvent);
connect(m_wrk, &QWrk::signalWRKGlobalVars, this, &GUIPlayer::wrkUpdateLoadProgress);
connect(m_wrk, &QWrk::signalWRKTrack, this, &GUIPlayer::wrkTrackHeader);
connect(m_wrk, &QWrk::signalWRKTimeBase, this, &GUIPlayer::wrkTimeBase);
connect(m_wrk, &QWrk::signalWRKNote, this, &GUIPlayer::wrkNoteEvent);
connect(m_wrk, &QWrk::signalWRKKeyPress, this, &GUIPlayer::wrkKeyPressEvent);
connect(m_wrk, &QWrk::signalWRKCtlChange, this, &GUIPlayer::wrkCtlChangeEvent);
connect(m_wrk, &QWrk::signalWRKPitchBend, this, &GUIPlayer::wrkPitchBendEvent);
connect(m_wrk, &QWrk::signalWRKProgram, this, &GUIPlayer::wrkProgramEvent);
connect(m_wrk, &QWrk::signalWRKChanPress, this, &GUIPlayer::wrkChanPressEvent);
connect(m_wrk, &QWrk::signalWRKSysexEvent, this, &GUIPlayer::wrkSysexEvent);
connect(m_wrk, &QWrk::signalWRKSysex, this, &GUIPlayer::wrkSysexEventBank);
connect(m_wrk, &QWrk::signalWRKText, this, &GUIPlayer::wrkUpdateLoadProgress);
connect(m_wrk, &QWrk::signalWRKTimeSig, this, &GUIPlayer::wrkUpdateLoadProgress);
connect(m_wrk, &QWrk::signalWRKKeySig, this, &GUIPlayer::wrkUpdateLoadProgress);
connect(m_wrk, &QWrk::signalWRKTempo, this, &GUIPlayer::wrkTempoEvent);
connect(m_wrk, &QWrk::signalWRKTrackPatch, this, &GUIPlayer::wrkTrackPatch);
connect(m_wrk, &QWrk::signalWRKComments, this, &GUIPlayer::wrkUpdateLoadProgress);
connect(m_wrk, &QWrk::signalWRKVariableRecord, this, &GUIPlayer::wrkUpdateLoadProgress);
connect(m_wrk, &QWrk::signalWRKNewTrack, this, &GUIPlayer::wrkNewTrackHeader);
connect(m_wrk, &QWrk::signalWRKTrackName, this, &GUIPlayer::wrkUpdateLoadProgress);
connect(m_wrk, &QWrk::signalWRKTrackVol, this, &GUIPlayer::wrkTrackVol);
connect(m_wrk, &QWrk::signalWRKTrackBank, this, &GUIPlayer::wrkTrackBank);
connect(m_wrk, &QWrk::signalWRKSegment, this, &GUIPlayer::wrkUpdateLoadProgress);
connect(m_wrk, &QWrk::signalWRKChord, this, &GUIPlayer::wrkUpdateLoadProgress);
connect(m_wrk, &QWrk::signalWRKExpression, this, &GUIPlayer::wrkUpdateLoadProgress);
m_player = new Player(m_Client, m_portId);
connect(m_player, &Player::playbackStopped, this, &GUIPlayer::playerStopped, Qt::QueuedConnection);
m_Client->setRealTimeInput(false);
m_Client->startSequencerInput();
tempoReset();
volumeReset();
updateState(EmptyState);
readSettings();
}
GUIPlayer::~GUIPlayer()
{
m_Client->stopSequencerInput();
m_Port->unsubscribeAll();
m_Port->detach();
m_Client->close();
delete m_player;
delete m_ui;
delete m_song;
}
void GUIPlayer::subscribe(const QString& portName)
{
try {
if (!m_subscription.isEmpty()) {
m_Port->unsubscribeTo(m_subscription);
}
m_subscription = portName;
m_Port->subscribeTo(m_subscription);
} catch (const SequencerError& err) {
qWarning() << "SequencerError exception. Error code: " << err.code()
<< " (" << err.qstrError() << ")";
qWarning() << "Location: " << err.location();
}
}
void GUIPlayer::updateTimeLabel(int mins, int secs, int cnts)
{
static QChar fill('0');
QString stime = QString("%1:%2.%3").arg(mins,2,10,fill)
.arg(secs,2,10,fill)
.arg(cnts,2,10,fill);
m_ui->lblTime->setText(stime);
}
void GUIPlayer::updateState(PlayerState newState)
{
if (m_state == newState)
return;
switch (newState) {
case EmptyState:
m_ui->actionPlay->setEnabled(false);
m_ui->actionPause->setEnabled(false);
m_ui->actionStop->setEnabled(false);
statusBar()->showMessage("Please, load a song");
break;
case PlayingState:
m_ui->actionPlay->setEnabled(false);
m_ui->actionPause->setEnabled(true);
m_ui->actionStop->setEnabled(true);
statusBar()->showMessage("Playing");
break;
case PausedState:
m_ui->actionPlay->setEnabled(false);
m_ui->actionStop->setEnabled(true);
statusBar()->showMessage("Paused");
break;
case StoppedState:
m_ui->actionPause->setChecked(false);
m_ui->actionPause->setEnabled(false);
m_ui->actionStop->setEnabled(false);
m_ui->actionPlay->setEnabled(true);
statusBar()->showMessage("Stopped");
break;
default:
statusBar()->showMessage("Not initialized");
break;
}
m_state = newState;
}
void GUIPlayer::play()
{
if (!m_song->isEmpty()) {
if (m_player->getInitialPosition() == 0) {
if (m_initialTempo == 0)
return;
QueueTempo firstTempo = m_Queue->getTempo();
firstTempo.setPPQ(m_song->getDivision());
firstTempo.setTempo(m_initialTempo);
firstTempo.setTempoFactor(m_tempoFactor);
m_Queue->setTempo(firstTempo);
m_Client->drainOutput();
m_player->sendVolumeEvents();
}
m_player->start();
updateState(PlayingState);
}
}
void GUIPlayer::pause()
{
if (m_state == PlayingState || m_player->isRunning()) {
m_player->stop();
m_player->setPosition(m_Queue->getStatus().getTickTime());
updateState(PausedState);
} else if (!m_song->isEmpty()) {
m_player->start();
updateState(PlayingState);
}
}
void GUIPlayer::stop()
{
if (m_state == PlayingState || m_state == PausedState ||
m_player->isRunning()) {
m_Queue->stop();
m_Queue->clear();
m_player->stop();
}
if (m_initialTempo != 0)
songFinished();
else
updateState(StoppedState);
}
void GUIPlayer::progressDialogInit(const QString& type, int max)
{
m_pd = new QProgressDialog("", "", 0, max, this);
m_pd->setWindowTitle(QString("Loading %1 file...").arg(type));
m_pd->setMinimumDuration(1000);
m_pd->setValue(0);
}
void GUIPlayer::progressDialogUpdate(int pos)
{
if (m_pd != nullptr) {
m_pd->setValue(pos);
qApp->processEvents();
}
}
void GUIPlayer::progressDialogClose()
{
delete m_pd; // set to 0 by QPointer<>
}
void GUIPlayer::openFile(const QString& fileName)
{
QFileInfo finfo(fileName);
if (finfo.exists()) {
m_song->clear();
m_loadingMessages.clear();
m_tick = 0;
m_initialTempo = 0;
m_currentTrack = 0;
try {
QString ext = finfo.suffix().toLower();
if (ext == "wrk") {
progressDialogInit("Cakewalk", finfo.size());
m_wrk->readFromFile(fileName);
}
else if (ext == "mid" || ext == "midi" || ext == "kar") {
progressDialogInit("MIDI", finfo.size());
m_smf->readFromFile(fileName);
}
else if (ext == "rmi") {
progressDialogInit("RIFF MIDI", finfo.size());
m_rmi->readFromFile(fileName);
}
progressDialogUpdate(finfo.size());
if (m_song->isEmpty()) {
m_ui->lblName->clear();
} else {
m_song->sort();
m_player->setSong(m_song);
m_ui->lblName->setText(finfo.fileName());
m_lastDirectory = finfo.absolutePath();
}
} catch (...) {
m_song->clear();
m_ui->lblName->clear();
}
progressDialogClose();
if (m_initialTempo == 0) {
m_initialTempo = 500000;
}
updateTimeLabel(0,0,0);
updateTempoLabel(6.0e7f / m_initialTempo);
m_ui->progressBar->setValue(0);
if (!m_loadingMessages.isEmpty()) {
m_loadingMessages.insert(0,
"Warning, this file may be non-standard or damaged.<br>");
QMessageBox::warning(this, QSTR_APPNAME, m_loadingMessages);
}
if (m_song->isEmpty())
updateState(EmptyState);
else
updateState(StoppedState);
}
}
void GUIPlayer::open()
{
QString fileName = QFileDialog::getOpenFileName(this,
"Open MIDI File", m_lastDirectory,
"All files (*.kar *.mid *.midi *rmi *.wrk);;"
"Karaoke files (*.kar);;"
"MIDI Files (*.mid *.midi);;"
"RIFF MIDI Files (*.rmi);;"
"Cakewalk files (*.wrk)" );
if (! fileName.isEmpty() ) {
stop();
openFile(fileName);
}
}
void GUIPlayer::setup()
{
bool ok;
int current;
QStringList items;
QListIterator<PortInfo> it(m_Client->getAvailableOutputs());
while(it.hasNext()) {
PortInfo p = it.next();
items << QString("%1:%2").arg(p.getClientName()).arg(p.getPort());
}
current = items.indexOf(m_subscription);
QString item = QInputDialog::getItem(this, "Player subscription",
"Output port:", items,
current, false, &ok);
if (ok && !item.isEmpty())
subscribe(item);
}
void GUIPlayer::songFinished()
{
m_player->resetPosition();
updateState(StoppedState);
}
void GUIPlayer::playerStopped()
{
int portId = m_Port->getPortId();
for (int channel = 0; channel < 16; ++channel) {
ControllerEvent ev1(channel, MIDI_CTL_ALL_NOTES_OFF, 0);
ev1.setSource(portId);
ev1.setSubscribers();
ev1.setDirect();
m_Client->outputDirect(&ev1);
ControllerEvent ev2(channel, MIDI_CTL_ALL_SOUNDS_OFF, 0);
ev2.setSource(portId);
ev2.setSubscribers();
ev2.setDirect();
m_Client->outputDirect(&ev2);
}
m_Client->drainOutput();
}
void GUIPlayer::updateTempoLabel(float ftempo)
{
QString stempo = QString("%1 bpm").arg(ftempo, 0, 'f', 2);
m_ui->lblOther->setText(stempo);
}
void GUIPlayer::sequencerEvent(SequencerEvent *ev)
{
if ((ev->getSequencerType() == SND_SEQ_EVENT_ECHO) && (m_tick != 0)){
auto t = ev->getTick();
int pos = 100 * t / m_tick;
const snd_seq_real_time_t* rt = m_Queue->getStatus().getRealtime();
int mins = rt->tv_sec / 60;
int secs = rt->tv_sec % 60;
int cnts = qFloor( rt->tv_nsec / 1.0e7 );
updateTempoLabel(m_Queue->getTempo().getRealBPM());
updateTimeLabel(mins, secs, cnts);
m_ui->progressBar->setValue(pos);
if (t >= m_tick) {
songFinished();
}
}
delete ev;
}
void GUIPlayer::dataHandler(const QString &dataType, const QByteArray &data)
{
if (dataType == "RMID") {
QDataStream ds(data);
m_smf->readFromStream(&ds);
}
}
void GUIPlayer::pitchShift(int value)
{
m_player->setPitchShift(value);
}
void GUIPlayer::tempoReset()
{
m_ui->sliderTempo->setValue(100);
tempoSlider(100);
}
void GUIPlayer::volumeReset()
{
m_ui->volumeSlider->setValue(100);
volumeSlider(100);
}
void GUIPlayer::tempoSlider(int value)
{
m_tempoFactor = (value*value + 100.0*value + 20000.0) / 40000.0;
QueueTempo qtempo = m_Queue->getTempo();
qtempo.setTempoFactor(m_tempoFactor);
m_Queue->setTempo(qtempo);
m_Client->drainOutput();
if (!m_player->isRunning())
updateTempoLabel(qtempo.getRealBPM());
// Slider tooltip
QString tip = QString("%1 %").arg(m_tempoFactor*100.0, 0, 'f', 0);
m_ui->sliderTempo->setToolTip(tip);
QToolTip::showText(QCursor::pos(), tip, this);
}
void GUIPlayer::volumeSlider(int value)
{
QString tip = QString::number(value)+'%';
m_ui->lblVolume->setText(tip);
m_ui->volumeSlider->setToolTip(tip);
m_player->setVolumeFactor(value);
QToolTip::showText(QCursor::pos(), tip, this);
}
void GUIPlayer::dragEnterEvent( QDragEnterEvent * event )
{
if (event->mimeData()->hasUrls()) {
event->acceptProposedAction();
}
}
void GUIPlayer::dropEvent( QDropEvent * event )
{
if ( event->mimeData()->hasUrls() ) {
QList<QUrl> urls = event->mimeData()->urls();
if (!urls.empty()) {
QString fileName = urls.first().toLocalFile();
if ( fileName.endsWith(".mid", Qt::CaseInsensitive) ||
fileName.endsWith(".midi", Qt::CaseInsensitive) ||
fileName.endsWith(".kar", Qt::CaseInsensitive) ||
fileName.endsWith(".rmi", Qt::CaseInsensitive) ||
fileName.endsWith(".wrk", Qt::CaseInsensitive) ) {
stop();
event->accept();
openFile(fileName);
} else {
QMessageBox::warning(this, QSTR_APPNAME,
QString("Dropped file %1 is not supported").arg(fileName));
}
}
}
}
void GUIPlayer::readSettings()
{
QSettings settings;
settings.beginGroup("Window");
restoreGeometry(settings.value("Geometry").toByteArray());
restoreState(settings.value("State").toByteArray());
settings.endGroup();
settings.beginGroup("Preferences");
m_lastDirectory = settings.value("LastDirectory").toString();
QString midiConn = settings.value("MIDIConnection").toString();
settings.endGroup();
if (midiConn.length() > 0)
subscribe(midiConn);
}
void GUIPlayer::writeSettings()
{
QSettings settings;
settings.beginGroup("Window");
settings.setValue("Geometry", saveGeometry());
settings.setValue("State", saveState());
settings.endGroup();
settings.beginGroup("Preferences");
settings.setValue("LastDirectory", m_lastDirectory);
settings.setValue("MIDIConnection", m_subscription);
settings.endGroup();
}
void GUIPlayer::closeEvent( QCloseEvent *event )
{
stop();
m_player->wait();
writeSettings();
event->accept();
}
void GUIPlayer::about()
{
About aboutDlg(this);
aboutDlg.exec();
}
/* **************************************** *
* SMF (Standard MIDI file) format handling
* **************************************** */
void GUIPlayer::smfUpdateLoadProgress()
{
progressDialogUpdate(m_smf->getFilePos());
}
void GUIPlayer::appendSMFEvent(SequencerEvent* ev)
{
unsigned long tick = m_smf->getCurrentTime();
ev->setSource(m_portId);
if (ev->getSequencerType() != SND_SEQ_EVENT_TEMPO) {
ev->setSubscribers();
}
ev->scheduleTick(m_queueId, tick, false);
m_song->append(ev);
if (tick > m_tick)
m_tick = tick;
smfUpdateLoadProgress();
}
void GUIPlayer::smfHeaderEvent(int format, int ntrks, int division)
{
m_song->setHeader(format, ntrks, division);
smfUpdateLoadProgress();
}
void GUIPlayer::smfNoteOnEvent(int chan, int pitch, int vol)
{
SequencerEvent* ev = new NoteOnEvent (chan, pitch, vol);
appendSMFEvent(ev);
}
void GUIPlayer::smfNoteOffEvent(int chan, int pitch, int vol)
{
SequencerEvent* ev = new NoteOffEvent (chan, pitch, vol);
appendSMFEvent(ev);
}
void GUIPlayer::smfKeyPressEvent(int chan, int pitch, int press)
{
SequencerEvent* ev = new KeyPressEvent (chan, pitch, press);
appendSMFEvent(ev);
}
void GUIPlayer::smfCtlChangeEvent(int chan, int ctl, int value)
{
SequencerEvent* ev = new ControllerEvent (chan, ctl, value);
appendSMFEvent(ev);
}
void GUIPlayer::smfPitchBendEvent(int chan, int value)
{
SequencerEvent* ev = new PitchBendEvent (chan, value);
appendSMFEvent(ev);
}
void GUIPlayer::smfProgramEvent(int chan, int patch)
{
SequencerEvent* ev = new ProgramChangeEvent (chan, patch);
appendSMFEvent(ev);
}
void GUIPlayer::smfChanPressEvent(int chan, int press)
{
SequencerEvent* ev = new ChanPressEvent (chan, press);
appendSMFEvent(ev);
}
void GUIPlayer::smfSysexEvent(const QByteArray& data)
{
SequencerEvent* ev = new SysExEvent (data);
appendSMFEvent(ev);
}
void GUIPlayer::smfTempoEvent(int tempo)
{
if ( m_initialTempo == 0 ) {
m_initialTempo = tempo;
}
SequencerEvent* ev = new TempoEvent (m_queueId, tempo);
appendSMFEvent(ev);
}
void GUIPlayer::smfErrorHandler(const QString& errorStr)
{
if (m_loadingMessages.length() < 1024)
m_loadingMessages.append(QString("%1 at file offset %2<br>")
.arg(errorStr).arg(m_smf->getFilePos()));
}
void GUIPlayer::smfTrackStarted()
{
m_currentTrack++;
}
void GUIPlayer::smfTrackEnded()
{
if (m_currentTrack == m_smf->getTracks()) {
SequencerEvent* ev = new SystemEvent(SND_SEQ_EVENT_ECHO);
appendSMFEvent(ev);
}
}
/* ********************************* *
* Cakewalk WRK file format handling
* ********************************* */
void GUIPlayer::wrkUpdateLoadProgress()
{
if (m_pd != nullptr) {
progressDialogUpdate(m_wrk->getFilePos());
}
}
void
GUIPlayer::appendWRKEvent(unsigned long ticks, SequencerEvent* ev)
{
ev->setSource(m_portId);
if (ev->getSequencerType() != SND_SEQ_EVENT_TEMPO) {
ev->setSubscribers();
}
ev->scheduleTick(m_queueId, ticks, false);
m_song->append(ev);
if (ticks > m_tick)
m_tick = ticks;
wrkUpdateLoadProgress();
}
void GUIPlayer::wrkErrorHandler(const QString& errorStr)
{
if (m_loadingMessages.length() < 1024) {
m_loadingMessages.append(QString("%1 at file offset %2<br>")
.arg(errorStr).arg(m_wrk->getFilePos()));
}
}
void GUIPlayer::wrkFileHeader(int /*verh*/, int /*verl*/)
{
m_song->setHeader(1, 0, 120);
wrkUpdateLoadProgress();
// qDebug() << Q_FUNC_INFO;
}
void GUIPlayer::wrkTimeBase(int timebase)
{
m_song->setDivision(timebase);
wrkUpdateLoadProgress();
// qDebug() << Q_FUNC_INFO << timebase;
}
void GUIPlayer::wrkStreamEndEvent(long time)
{
unsigned long ticks = time;
if (ticks > m_tick)
m_tick = ticks;
wrkUpdateLoadProgress();
// qDebug() << Q_FUNC_INFO << time;
}
void GUIPlayer::wrkTrackHeader( const QString& /*name1*/,
const QString& /*name2*/,
int trackno, int channel,
int pitch, int velocity, int /*port*/,
bool /*selected*/, bool /*muted*/, bool /*loop*/ )
{
TrackMapRec rec;
rec.channel = channel;
rec.pitch = pitch;
rec.velocity = velocity;
m_trackMap[trackno] = rec;
wrkUpdateLoadProgress();
// qDebug() << Q_FUNC_INFO << trackno << channel << pitch << velocity;
}
void GUIPlayer::wrkNoteEvent(int track, long time, int chan, int pitch, int vol, int dur)
{
TrackMapRec rec = m_trackMap[track];
int channel = (rec.channel > -1) ? rec.channel : chan;
int key = qBound(0, pitch + rec.pitch, 127);
int velocity = qBound(0, vol + rec.velocity, 127);
SequencerEvent* ev = new NoteEvent(channel, key, velocity, dur);
appendWRKEvent(time, ev);
// qDebug() << Q_FUNC_INFO << channel << key << velocity << dur;
}
void GUIPlayer::wrkKeyPressEvent(int track, long time, int chan, int pitch, int press)
{
TrackMapRec rec = m_trackMap[track];
int key = pitch + rec.pitch;
int channel = (rec.channel > -1) ? rec.channel : chan;
SequencerEvent* ev = new KeyPressEvent(channel, key, press);
appendWRKEvent(time, ev);
// qDebug() << Q_FUNC_INFO;
}
void GUIPlayer::wrkCtlChangeEvent(int track, long time, int chan, int ctl, int value)
{
TrackMapRec rec = m_trackMap[track];
int channel = (rec.channel > -1) ? rec.channel : chan;
SequencerEvent* ev = new ControllerEvent(channel, ctl, value);
appendWRKEvent(time, ev);
// qDebug() << Q_FUNC_INFO;
}
void GUIPlayer::wrkPitchBendEvent(int track, long time, int chan, int value)
{
TrackMapRec rec = m_trackMap[track];
int channel = (rec.channel > -1) ? rec.channel : chan;
SequencerEvent* ev = new PitchBendEvent(channel, value);
appendWRKEvent(time, ev);
// qDebug() << Q_FUNC_INFO;
}
void GUIPlayer::wrkProgramEvent(int track, long time, int chan, int patch)
{
TrackMapRec rec = m_trackMap[track];
int channel = (rec.channel > -1) ? rec.channel : chan;
SequencerEvent* ev = new ProgramChangeEvent(channel, patch);
appendWRKEvent(time, ev);
// qDebug() << Q_FUNC_INFO;
}
void GUIPlayer::wrkChanPressEvent(int track, long time, int chan, int press)
{
TrackMapRec rec = m_trackMap[track];
int channel = (rec.channel > -1) ? rec.channel : chan;
SequencerEvent* ev = new ChanPressEvent(channel, press);
appendWRKEvent(time, ev);
// qDebug() << Q_FUNC_INFO;
}
void GUIPlayer::wrkSysexEvent(int track, long time, int bank)
{
Q_UNUSED(track)
qDebug() << Q_FUNC_INFO;
if (m_savedSysexEvents.contains(bank)) {
SysExEvent* ev = m_savedSysexEvents[bank].clone();
appendWRKEvent(time, ev);
wrkUpdateLoadProgress();
}
}
void GUIPlayer::wrkSysexEventBank(int bank, const QString& /*name*/,
bool autosend, int /*port*/, const QByteArray& data)
{
//qDebug() << Q_FUNC_INFO;
SysExEvent* ev = new SysExEvent(data);
if (autosend) {
appendWRKEvent(0, ev);
} else {
m_savedSysexEvents[bank] = *ev;
delete ev;
}
wrkUpdateLoadProgress();
}
void GUIPlayer::wrkTempoEvent(long time, int tempo)
{
double bpm = tempo / 100.0;
if ( m_initialTempo < 0 )
m_initialTempo = qRound( bpm );
SequencerEvent* ev = new TempoEvent(m_queueId, qRound ( 6e7 / bpm ) );
appendWRKEvent(time, ev);
// qDebug() << Q_FUNC_INFO;
}
void GUIPlayer::wrkTrackPatch(int track, int patch)
{
TrackMapRec rec = m_trackMap[track];
int channel = (rec.channel > -1) ? rec.channel : 0;
wrkProgramEvent(track, 0, channel, patch);
// qDebug() << Q_FUNC_INFO;
}
void GUIPlayer::wrkNewTrackHeader( const QString& /*name*/,
int trackno, int channel,
int pitch, int velocity, int /*port*/,
bool /*selected*/, bool /*muted*/, bool /*loop*/ )
{
TrackMapRec rec;
rec.channel = channel;
rec.pitch = pitch;
rec.velocity = velocity;
m_trackMap[trackno] = rec;
wrkUpdateLoadProgress();
// qDebug() << Q_FUNC_INFO << trackno << channel << pitch << velocity;
}
void GUIPlayer::wrkTrackVol(int track, int vol)
{
int lsb, msb;
TrackMapRec rec = m_trackMap[track];
int channel = (rec.channel > -1) ? rec.channel : 0;
if (vol < 128)
wrkCtlChangeEvent(track, 0, channel, MIDI_CTL_MSB_MAIN_VOLUME, vol);
else {
lsb = vol % 0x80;
msb = vol / 0x80;
wrkCtlChangeEvent(track, 0, channel, MIDI_CTL_LSB_MAIN_VOLUME, lsb);
wrkCtlChangeEvent(track, 0, channel, MIDI_CTL_MSB_MAIN_VOLUME, msb);
}
// qDebug() << Q_FUNC_INFO;
}
void GUIPlayer::wrkTrackBank(int track, int bank)
{
// assume GM/GS bank method
int lsb, msb;
TrackMapRec rec = m_trackMap[track];
int channel = (rec.channel > -1) ? rec.channel : 0;
lsb = bank % 0x80;
msb = bank / 0x80;
wrkCtlChangeEvent(track, 0, channel, MIDI_CTL_MSB_BANK, msb);
wrkCtlChangeEvent(track, 0, channel, MIDI_CTL_LSB_BANK, lsb);
// qDebug() << Q_FUNC_INFO;
}
void GUIPlayer::wrkEndOfFile()
{
if (m_initialTempo < 0)
m_initialTempo = 120;
SequencerEvent* ev = new SystemEvent(SND_SEQ_EVENT_ECHO);
appendWRKEvent(m_tick, ev);
// qDebug() << Q_FUNC_INFO;
}
DISABLE_WARNING_POP
Classes managing ALSA Sequencer clients.
Classes managing ALSA Sequencer events.
Classes managing ALSA Sequencer ports.
Classes managing ALSA Sequencer queues.
The QSettings class provides persistent platform-independent application settings.
Standard MIDI Files Input/Output.
Cakewalk WRK Files Input.
RIFF MIDI Files Input.
SequencerError Exception class.