-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathsequence.h
301 lines (265 loc) · 9.76 KB
/
sequence.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
/*
Drumstick MIDI File Player Multiplatform Program
Copyright (C) 2006-2024, Pedro Lopez-Cabanillas <[email protected]>
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 SEQUENCE_H
#define SEQUENCE_H
#include <QList>
#include <QMap>
#include <QObject>
#include <QTextCodec>
#include <chrono>
#include <drumstick/qsmf.h>
#include <drumstick/rmid.h>
#include <drumstick/qwrk.h>
#include <drumstick/rtmidioutput.h>
#include <uchardet.h>
#include "events.h"
class Sequence : public QObject
{
Q_OBJECT
public:
/**
* For Karaoke files, there may be additional metadata
* FileType: @KMIDI KARAOKE FILE
* Version: @V0100
* Information: @I<text>
* Language: @L<lang>
* Title: @T<title, author, ...>
* What?: @W<bla bla bla>
*/
enum TextType {
None = 0, Text = 1, Copyright = 2, TrackName = 3,
InstrumentName = 4, Lyric = 5, Marker = 6, Cue = 7,
KarFileType = 8, KarVersion = 9, KarInformation = 10,
KarLanguage = 11, KarTitles = 12, KarWhatever = 13,
FIRST_TYPE = Text, LAST_TYPE = KarWhatever
};
Q_ENUM(TextType)
Sequence(QObject* parent = 0);
virtual ~Sequence();
void clear();
void appendEvent(MIDIEvent* ev);
void loadPattern(QList<MIDIEvent*> pattern);
void loadFile(const QString& fileName);
qreal tempoFactor() const;
void setTempoFactor(const qreal factor);
MIDIEvent *nextEvent();
int eventTime(MIDIEvent* ev) const;
std::chrono::microseconds timeOfEvent(MIDIEvent *ev) const;
std::chrono::microseconds deltaTimeOfEvent(MIDIEvent *ev) const;
std::chrono::microseconds timeOfTicks(const uint64_t ticks) const;
bool hasMoreEvents();
int getFormat() const { return m_format; }
int getDivision() const { return m_division; }
bool isEmpty();
void resetPosition();
void setTickPosition(long tick);
qreal currentTempo() const;
QString getName() const { return m_lblName; }
int songLengthTicks() const;
void updateTempo(qreal newTempo);
qreal ticks2micros() const { return m_ticks2micros; }
bool channelUsed(int channel);
QString channelLabel(int channel);
int lowestMidiNote();
int highestMidiNote();
int numUchardetErrors();
int currentMIB() const;
QByteArray currentCharset() const;
void setCurrentCharset(const QByteArray& charset);
int getNumTracks() const;
int trackMaxPoints();
int typeMaxPoints();
QByteArray trackName(int track) const;
int trackChannel(int track) const;
QString currentFile() const;
QString currentFullFileName() const;
QList<QPair<int,QByteArray>> getRawText(const int track, const TextType type);
QStringList getText(const TextType type);
void findCodec();
int getNumAlterations() const;
BeatEvent* nearestBeatByTicks(int ticks);
BeatEvent* nextBar(BeatEvent* latest);
BeatEvent* previousBar(BeatEvent* latest);
BeatEvent* jumpToBar(int bar);
BeatEvent* firstBeat();
int lastBar();
static QByteArrayList getExtraCodecNames();
static int getMibForName(const QByteArray name);
QTextCodec *codec() const;
void setCodec(QTextCodec *newCodec);
QString getFileFormat() const;
QString getMetadataInfo() const;
QString loadingErrors() const;
int errorsCount() const;
qsizetype size() const;
qreal initialTempo() const;
bool simpleTimeProcess() const;
QString duration() const;
signals:
void loadingStart(int size);
void loadingProgress(int pos);
void loadingFinished();
public slots:
/* Rmidi slots */
void dataHandler(const QString& dataType, const QByteArray& data);
void infoHandler(const QString& infoType, const QByteArray& data);
/* SMF slots */
void appendSMFEvent(MIDIEvent *ev);
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 smfMetaEvent(int type, const QByteArray &data);
void smfTempoEvent(int tempo);
void smfTimeSigEvent(int b0, int b1, int b2, int b3);
void smfTrackStartEvent();
void smfTrackEnd();
void smfErrorHandler(const QString& errorStr);
void smfUpdateLoadProgress();
void smfKeySig(int, int);
/* WRK slots */
void appendWRKEvent(long ticks, MIDIEvent* ev);
void appendWRKmetadata(int track, long time, TextType typ, const QByteArray &data);
void wrkUpdateLoadProgress();
void wrkErrorHandler(const QString& errorStr);
void wrkFileHeader(int verh, int verl);
void wrkEndOfFile();
void wrkStreamEndEvent(long time);
void wrkTrackHeader(const QByteArray& name1, const QByteArray& name2,
int trackno, int channel, int pitch,
int velocity, int port,
bool selected, bool muted, bool loop);
void wrkTimeBase(int timebase);
void wrkGlobalVars();
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 wrkTextEvent(int track, long time, int typ, const QByteArray& data);
void wrkComments(const QByteArray& cmt);
void wrkVariableRecord(const QString& name, const QByteArray& data);
void wrkTempoEvent(long time, int tempo);
void wrkTrackPatch(int track, int patch);
void wrkNewTrackHeader(const QByteArray& name,
int trackno, int channel, int pitch,
int velocity, int port,
bool selected, bool muted, bool loop);
void wrkTrackName(int trackno, const QByteArray& name);
void wrkTrackVol(int track, int vol);
void wrkTrackBank(int track, int bank);
void wrkSegment(int track, long time, const QByteArray& name);
void wrkChord(int track, long time, const QString& name, const QByteArray& data);
void wrkExpression(int track, long time, int code, const QByteArray& text);
void wrkTimeSignatureEvent(int bar, int num, int den);
void wrkKeySig(int bar, int alt);
void wrkMarker(long time, int smpte, const QByteArray &data);
private: // methods
void sort();
void initCodecs();
void insertBeats(qint64 ticks);
void timeCalculations();
void addMetaData(int time, int type, const QByteArray &data);
void appendStringToList(QStringList &list, QString &s, TextType type);
private: // members
QList<MIDIEvent*> m_list;
drumstick::File::Rmidi* m_rmid;
drumstick::File::QSmf* m_smf;
drumstick::File::QWrk* m_wrk;
int m_uchardetErrors;
int m_format;
int m_numTracks;
int m_ticksDuration;
int m_division;
int m_pos;
int m_curTrack;
int m_beatMax;
int m_barCount;
int m_beatCount;
int m_lowestMidiNote;
int m_highestMidiNote;
qreal m_tempo;
qreal m_tempoFactor;
qreal m_ticks2micros;
qreal m_duration;
qint64 m_lastBeat;
qint64 m_beatLength;
qint64 m_tick;
QString m_lblName;
static const QMap<QByteArray,int> m_umibs;
QMap<QByteArray,int> m_supportedCodecs;
QTextCodec *m_codec;
int m_mib;
QByteArray m_charset;
uchardet_t m_handle;
QMap<int, SysExEvent> m_savedSysexEvents;
struct TrackMapRec {
int channel;
int pitch;
int velocity;
};
QMap<int,TrackMapRec> m_trackMap;
struct TimeSigRec {
int bar;
int num;
int den;
long time;
};
QList<TimeSigRec> m_bars;
struct TextRec {
TextRec(QByteArray data)
: m_tick(0)
, m_track(0)
, m_type(TextType::None)
, m_text(data)
{}
TextRec(int tick, int track, TextType e, QByteArray data)
: m_tick(tick)
, m_track(track)
, m_type(e)
, m_text(data)
{}
int m_tick;
int m_track;
TextType m_type;
QByteArray m_text;
};
QList<TextRec> m_textEvents;
QString m_currentFile;
QString m_currentFileFull;
QString m_fileFormat;
QByteArray m_trackLabel;
bool m_channelUsed[drumstick::rt::MIDI_STD_CHANNELS];
int m_channelEvents[drumstick::rt::MIDI_STD_CHANNELS];
QByteArray m_channelLabel[drumstick::rt::MIDI_STD_CHANNELS];
int m_channelPatches[drumstick::rt::MIDI_STD_CHANNELS];
QMap<int, QByteArray> m_trkName;
QMap<int, int> m_trkScore;
QMap<int, int> m_typScore;
QMap<int, int> m_trkChannel;
QMap<QString, QString> m_infoMap;
QList<QString> m_loadingErrors;
int m_numTempoChanges{0};
qreal m_initialTempo{5e5};
};
#endif // SEQUENCE_H