-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathMidiEventManager.cpp
288 lines (256 loc) · 8.52 KB
/
MidiEventManager.cpp
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
#include "MidiEventManager.h"
#include "math.h"
static const char keyName[12][4] = { "C", "C#", "D", "D#", "E", "F", "F#", "G", "G#", "A", "A#", "B"};
static const int ScaleC[13] = { 1,1,0,1,0,1,0,1,1,0,1,0,1 };
static const int avoidcol = GetColor(80, 80, 80);
static const int gatecol = GetColor(200, 200, 255);
static const int basecol = GetColor(80, 180, 202);
MidiEventManager::MidiEventManager() {
div = 480;
activeCh = 0;
BlockSize = 16;
BlockWidth = 16;
BlockHeight = 24;
Base = 60;
// グリッドの位置
x = 50;
y = 50;
}
void MidiEventManager::addNote(int ch, int delta, int notenum, int gate, int vel) {
// TODO ここで本来は楽器のセットアップを行いたい
// 1拍ぶんの休符を追加したい
if (seq == 0) {
//noteMap[ch].emplace(seq, NoteOnEvent(ch, div * 4, 60, 0, 0));
seq += div * 4;
return;
}
// key = resolution(480) * note(128) * beat(4)* mea(128) = 31457280 < INT_MAX(2147483647)
int mapKey1 = (seq + delta) * 128 + notenum;
int mapKey2 = (seq + delta + gate - 1) * 128 + notenum; // NoteOffのキーは次のNoteOnと重ならないように-1する
noteMap[ch].emplace(mapKey1, NoteOnEvent(ch, delta, notenum, gate, vel));
noteMap[ch].emplace(mapKey2, NoteOnEvent(ch, gate, notenum, 0, 0));
// 現在のシーケンス位置を移動
seq += delta + gate;
}
void MidiEventManager::addStartNote(int delta, int ch, int notenum, int vel) {
int key = seq + delta;
noteMap[ch].emplace(key, NoteOnEvent(ch, delta, notenum, 400, vel));
seq += delta;
}
void MidiEventManager::addEndNote(int delta, int ch, int notenum) {
int key = seq + delta - 1;
noteMap[ch].emplace(key, NoteOnEvent(ch, delta, notenum, 0, 0));
seq += delta;
}
void MidiEventManager::autoCreate(int length) {
seq = 0;
// ペンタトニック・スケールでランダムに配置(あまりに飛び飛びの演奏はやめる)
static const int majorScale[8] = { 60, 62, 64, 65, 67, 69, 71, 72 };
static const int chordC[4] = { 60, 64, 67, 72 };
static const int kokken[] = { 54, 56, 58, 61, 63, 66, 68, 70, 73, 75, 78 };
static const int gate[8] = { div / 2 , div / 2, div / 2, div / 2, div, div, div, div * 3 / 2};
static const int rest[] = { 0, 0, 0, 0, 0, div / 2};
int key, preKey;
key = preKey = GetRand(10);
while (seq < length) {
preKey = key; // 前に押した黒鍵を覚えておく
do {
key = preKey - 2 + GetRand(4);
} while (key < 0 || key > 10);
addNote(activeCh, rest[GetRand(5)], kokken[key], gate[GetRand(7)], 90 + GetRand(20));
}
}
void MidiEventManager::deleteAllEvent() {
noteMap[activeCh].clear();
seq = 0;
}
int MidiEventManager::GetNoteData(int ch, int key) const {
auto itr = noteMap[ch].find(key);
if (itr != noteMap[ch].end()) {
return itr->second.GetNote();
}
else return -1;
}
int MidiEventManager::GetGateData(int ch, int key) const {
auto itr = noteMap[ch].find(key);
if (itr != noteMap[ch].end()) {
return itr->second.GetGate();
}
else return -1;
}
int MidiEventManager::GetVelData(int ch, int key) const {
auto itr = noteMap[ch].find(key);
if (itr != noteMap[ch].end()) {
return itr->second.GetVel();
}
else return -1;
}
void MidiEventManager::Update(int focusch, int dmea) {
activeCh = focusch;
mea = dmea;
}
void MidiEventManager::keyDown() {
for (auto itr = noteMap[activeCh].begin(); itr != noteMap[activeCh].end(); itr++) {
itr->second.transpose(-12);
}
}
void MidiEventManager::keyUp() {
for (auto itr = noteMap[activeCh].begin(); itr != noteMap[activeCh].end(); itr++) {
itr->second.transpose(12);
}
}
void MidiEventManager::draw() {
// Base音を表示
DrawFormatString(x - 30, y + BlockHeight * BlockSize, WHITE, "C%d", Base / 12);
DrawFormatString(x - 30, y + BlockHeight * BlockSize / 2, WHITE, "C%d", Base / 12 + 1);
DrawFormatString(x - 30, y, WHITE, "C%d", Base / 12 + 2);
// グリッドを描画
for (int i = 0; i < BlockWidth; i++) {
for (int j = 0; j <= BlockHeight; j++) {
if (j % 12 == 0) DrawEdgeBox(i * BlockSize + x, j * BlockSize + y, (i + 1)*BlockSize + x, (j + 1)*BlockSize + y, basecol, avoidcol);
else if (ScaleC[j % 12] == 0) DrawEdgeBox(i * BlockSize + x, j * BlockSize + y, (i + 1)*BlockSize + x, (j + 1)*BlockSize + y, BLACK, avoidcol);
else DrawEdgeBox(i * BlockSize + x, j * BlockSize + y, (i + 1)*BlockSize + x, (j + 1)*BlockSize + y, WHITE, avoidcol
);
}
}
// MIDIイベントを表示(現在操作中のチャンネルのNoteONイベントのみ)
int j = 0;
// Condoctorの動きに合わせて表示するようにしたい。
int firsttick = mea * div * 4;
int endtick = (mea + 1) * div * 4 - 1;
DrawFormatString(540, 50, WHITE, " Tick Event Gate Vel");
for (auto itr = noteMap[activeCh].lower_bound(firsttick); itr != noteMap[activeCh].upper_bound(endtick); itr++) {
if (itr->second.GetVel() == 0) continue;
int tick = itr->first % (div * 4);
int delta = itr->second.GetDelta();
int note = itr->second.GetNote();
int gate = itr->second.GetGate();
int vel = itr->second.GetVel();
DrawFormatString(540, 70 + 20 * j, WHITE, "%5d %2s%d(%d) %d %d", tick, keyName[note % 12], note / 12, note, gate, vel);
if (++j == 20) break;
// Noteを描画
int n = note - Base; // Baseから何個上の音か?
if (0 <= n && n < BlockHeight) {
int x1 = BlockSize * tick / 120.0 + x;
int y1 = (BlockHeight - n) * BlockSize + y;
int x2 = x1 + BlockSize * gate / 120.0;
int y2 = y1 + BlockSize;
DrawEdgeBox(x1, y1, x2, y2, GREEN);
}
}
DrawBox(540, 50, 740, 50 + 20 * 20, GREEN, FALSE);
// DrawBox(540, 50, 740, 50 + 20, YELLOW, FALSE);
}
void MidiEventManager::LowerOctave() {
Base -= 12;
if (Base < 0) Base += 12;
}
void MidiEventManager::HigherOctave() {
Base += 12;
if (Base > 127) Base -= 12;
}
double MidiEventManager::loadMidiMsgFromSMF(int track, unsigned char* data, int size) {
int i = 0;
double bpm = 0.0f;
int a, b, c, d;
printfDx("シーケンス開始\n");
while (i < size) {
//printfDx("%02X ", data[i++]);
//if (i%16 == 0) printfDx("\n");
// デルタタイム
int delta = 0;
while ((d = data[i++]) & 0x80) {
delta = delta | (d & 0x7F);
delta <<= 7;
}
delta = delta | d;
printfDx("(%d) ", delta);
// チャンネルメッセージ
if ((data[i] & 0xF0) == 0x90) {
// ノートオン
int ch = data[i++] & 0x0F;
int note = data[i++];
int vel = data[i++];
printfDx("track:%d, Ch:%d, Note:%d, Vel:%d", track, ch, note, vel);
addStartNote(delta, track, note, vel);
continue;
}
if ((data[i] & 0xF0) == 0x80) {
// ノートオフ
int ch = data[i++] & 0x0F;
int note = data[i++];
int vel = data[i++];
printfDx("track:%d, Ch:%d, Note:%d, Vel:%d \n", track, ch, note, vel);
addEndNote(delta, track, note);
continue;
}
// メタイベント
switch (data[i++]) {
case 0xFF: // メタイベント
switch (data[i++]) {
case 0x02: // 著作権
i += data[i++]; // 飛ばす
printfDx("著作権\n");
break;
case 0x03: // シーケンス名
i += data[i++]; // 飛ばす
printfDx("シーケンス\n");
break;
case 0x51: // テンポ
i++; // データ長なので飛ばす
a = (int)data[i++] << 16;
b = (int)data[i++] << 8;
c = (int)data[i++];
bpm = 60.0 * 1000000 / (a | b | c);
// TODO なぜこれが期待した結果にならないのか良くわからない
//bpm = 60.0 * 1000000 / (((int)data[i++] << 16) | ((int)data[i++] << 8) | ((int)data[i++]));
printfDx("bpm = %.2f\n", bpm);
break;
case 0x58: // 拍子設定[4byte]
i++; // データ長なので飛ばす
printfDx("拍子:%d/%d\n", (int)pow((double)data[i++], 2.0), data[i++]);
i += 2; // 重要でない情報
break;
case 0x2F: // トラック終端
switch (data[i++]) {
case 0x00: // トラック終端
printfDx("End Of Track (i = %d, size = %d)\n", i, size);
break;
}
break;
}
break;
}
}
// 後からGateを設定する
for (auto itr = noteMap[track].begin(); itr != noteMap[track].end(); ++itr) {
if (itr->second.GetVel() > 0) {
for (auto itr2 = noteMap[track].upper_bound(itr->first); itr2 != noteMap[track].end(); ++itr2) {
if (itr2->second.GetVel() == 0) {
itr->second.SetGate(itr2->first - itr->first);
break;
}
}
}
}
return bpm;
}
int MidiEventManager::getMidiMsgForSMF(unsigned char* data) {
int i = 0;
//
// ノートオン・ノートオフ
for (auto itr = noteMap[activeCh].cbegin(); itr != noteMap[activeCh].cend(); itr++) {
// デルタタイムの計算
int delta = itr->second.GetDelta();
if (delta >> 21 != 0) data[++i] = (delta >> 21) | 128;
if (delta >> 14 != 0) data[i++] = ((delta >> 14) & 127) | 128;
if (delta >> 7 != 0) data[i++] = ((delta >> 7) & 127) | 128;
data[i++] = delta & 127;
data[i++] = (9 << 4) | itr->second.GetCh(); // 第一バイト
data[i++] = itr->second.GetNote(); // 第二バイト
data[i++] = itr->second.GetVel(); // 第三バイト
}
// メタイベント・EOF(00 FF 2F 00)
data[i++] = 0x00; data[i++] = 0xFF; data[i++] = 0x2F; data[i++] = 0x00;
return i;
}