spessasynth_core 3.27.5 → 3.27.6

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "spessasynth_core",
3
- "version": "3.27.5",
3
+ "version": "3.27.6",
4
4
  "description": "MIDI and SoundFont2/DLS library with no compromises",
5
5
  "main": "index.js",
6
6
  "type": "module",
@@ -41,7 +41,7 @@ class BasicMIDI extends MIDISequenceData
41
41
  isDLSRMIDI = false;
42
42
 
43
43
  /**
44
- * Copies a MIDI
44
+ * Copies a MIDI (tracks are shallowly copied!)
45
45
  * @param mid {BasicMIDI}
46
46
  * @returns {BasicMIDI}
47
47
  */
@@ -51,9 +51,27 @@ class BasicMIDI extends MIDISequenceData
51
51
  m._copyFromSequence(mid);
52
52
 
53
53
  m.isDLSRMIDI = mid.isDLSRMIDI;
54
- m.embeddedSoundFont = mid.embeddedSoundFont ? mid.embeddedSoundFont.slice(0) : undefined; // Deep copy
54
+ m.embeddedSoundFont = mid?.embeddedSoundFont ? mid.embeddedSoundFont : undefined; // Shallow copy
55
55
  m.tracks = mid.tracks.map(track => [...track]); // Shallow copy of each track array
56
-
56
+ return m;
57
+ }
58
+
59
+ /**
60
+ * Copies a MIDI with deep copy
61
+ * @param mid {BasicMIDI}
62
+ * @returns {BasicMIDI}
63
+ */
64
+ static copyFromDeep(mid)
65
+ {
66
+ const m = new BasicMIDI();
67
+ m._copyFromSequence(mid);
68
+ m.isDLSRMIDI = mid.isDLSRMIDI;
69
+ m.embeddedSoundFont = mid.embeddedSoundFont ? mid.embeddedSoundFont.slice(0) : undefined; // Deep copy
70
+ m.tracks = mid.tracks.map(track => track.map(event => new MIDIMessage(
71
+ event.ticks,
72
+ event.messageStatusByte,
73
+ event.messageData
74
+ ))); // Deep copy
57
75
  return m;
58
76
  }
59
77
 
@@ -217,7 +235,6 @@ class BasicMIDI extends MIDISequenceData
217
235
  copyrightComponents.push(readBytesAsString(
218
236
  e.messageData,
219
237
  e.messageData.length,
220
- undefined,
221
238
  false
222
239
  ));
223
240
  e.messageData.currentIndex = 0;
@@ -466,7 +483,7 @@ class BasicMIDI extends MIDISequenceData
466
483
  {
467
484
  this.rawMidiName = name.messageData;
468
485
  name.messageData.currentIndex = 0;
469
- this.midiName = readBytesAsString(name.messageData, name.messageData.length, undefined, false);
486
+ this.midiName = readBytesAsString(name.messageData, name.messageData.length, false);
470
487
  }
471
488
  }
472
489
  }
@@ -478,7 +495,7 @@ class BasicMIDI extends MIDISequenceData
478
495
  {
479
496
  this.rawMidiName = name.messageData;
480
497
  name.messageData.currentIndex = 0;
481
- this.midiName = readBytesAsString(name.messageData, name.messageData.length, undefined, false);
498
+ this.midiName = readBytesAsString(name.messageData, name.messageData.length, false);
482
499
  }
483
500
  }
484
501
  }
@@ -45,7 +45,7 @@ class MIDI extends BasicMIDI
45
45
  // possibly an RMID file (https://github.com/spessasus/sf2-rmidi-specification#readme)
46
46
  // skip size
47
47
  binaryData.currentIndex += 8;
48
- const rmid = readBytesAsString(binaryData, 4, undefined, false);
48
+ const rmid = readBytesAsString(binaryData, 4, false);
49
49
  if (rmid !== "RMID")
50
50
  {
51
51
  SpessaSynthGroupEnd();
@@ -101,18 +101,15 @@ class MIDI extends BasicMIDI
101
101
  this.copyright = readBytesAsString(
102
102
  this.RMIDInfo["ICOP"],
103
103
  this.RMIDInfo["ICOP"].length,
104
- undefined,
105
104
  false
106
105
  ).replaceAll("\n", " ");
107
106
  }
108
107
  if (this.RMIDInfo["INAM"])
109
108
  {
110
109
  this.rawMidiName = this.RMIDInfo[RMIDINFOChunks.name];
111
- // noinspection JSCheckFunctionSignatures
112
110
  this.midiName = readBytesAsString(
113
- this.rawMidiName,
111
+ /** @type {IndexedByteArray}*/this.rawMidiName,
114
112
  this.rawMidiName.length,
115
- undefined,
116
113
  false
117
114
  ).replaceAll("\n", " ");
118
115
  }
@@ -1,3 +1,5 @@
1
+ import { IndexedByteArray } from "../utils/indexed_array.js";
2
+
1
3
  /**
2
4
  * This is the base type for MIDI files. It contains all the "metadata" and information.
3
5
  * It extends to:
@@ -133,7 +135,7 @@ class MIDISequenceData
133
135
  * The RMID (Resource-Interchangeable MIDI) info data, if the file is RMID formatted.
134
136
  * Otherwise, this field is undefined.
135
137
  * Chunk type (e.g. "INAM"): Chunk data as a binary array.
136
- * @type {Object<string, IndexedByteArray>}
138
+ * @type {Record<string, IndexedByteArray>}
137
139
  */
138
140
  RMIDInfo = {};
139
141
 
@@ -217,7 +219,11 @@ class MIDISequenceData
217
219
  // copying objects
218
220
  this.loop = { ...sequence.loop };
219
221
  this.keyRange = { ...sequence.keyRange };
220
- this.RMIDInfo = { ...sequence.RMIDInfo };
222
+ this.RMIDInfo = {};
223
+ for (const [key, value] of Object.entries(sequence.RMIDInfo))
224
+ {
225
+ this.RMIDInfo[key] = new IndexedByteArray(value);
226
+ }
221
227
  }
222
228
  }
223
229
 
@@ -180,7 +180,7 @@ export function modifyMIDI(
180
180
  const midiPorts = midi.midiPorts.slice();
181
181
  /**
182
182
  * midi port: channel offset
183
- * @type {Object<number, number>}
183
+ * @type {Record<number, number>}
184
184
  */
185
185
  const midiPortChannelOffsets = {};
186
186
  let midiPortChannelOffset = 0;
@@ -48,7 +48,7 @@ const DEFAULT_COPYRIGHT = "Created using SpessaSynth";
48
48
  */
49
49
 
50
50
  /**
51
- * Writes an RMIDI file
51
+ * Writes an RMIDI file. Note that this method modifies the MIDI file in-place.
52
52
  * @this {BasicMIDI}
53
53
  * @param soundfontBinary {Uint8Array}
54
54
  * @param soundfont {BasicSoundBank}
@@ -10,7 +10,7 @@ import { SoundFontManager } from "../../synthetizer/audio_engine/engine_componen
10
10
  * Gets the used programs and keys for this MIDI file with a given sound bank
11
11
  * @this {BasicMIDI}
12
12
  * @param soundfont {SoundFontManager|BasicSoundBank} - the sound bank
13
- * @returns {Object<string, Set<string>>} Object<bank:program, Set<key-velocity>>
13
+ * @returns {Record<string, Set<string>>} Record<bank:program, Set<key-velocity>>
14
14
  */
15
15
  export function getUsedProgramsAndKeys(soundfont)
16
16
  {
@@ -81,7 +81,7 @@ export function getUsedProgramsAndKeys(soundfont)
81
81
  /**
82
82
  * find all programs used and key-velocity combos in them
83
83
  * bank:program each has a set of midiNote-velocity
84
- * @type {Object<string, Set<string>>}
84
+ * @type {Record<string, Set<string>>}
85
85
  */
86
86
  const usedProgramsAndKeys = {};
87
87
 
@@ -90,7 +90,7 @@ class XMFNode
90
90
  metadataLength;
91
91
 
92
92
  /**
93
- * @type {Object<string, any>}
93
+ * @type {Record<string, any>}
94
94
  */
95
95
  metadata = {};
96
96
 
@@ -111,8 +111,8 @@ class SpessaSynthSequencer
111
111
  midiPortChannelOffset = 0;
112
112
  /**
113
113
  * stored as:
114
- * Object<midi port, channel offset>
115
- * @type {Object<number, number>}
114
+ * Record<midi port, channel offset>
115
+ * @type {Record<number, number>}
116
116
  */
117
117
  midiPortChannelOffsets = {};
118
118
 
@@ -147,7 +147,7 @@ export function loadNewSongList(midiBuffers, autoPlay = true)
147
147
  * parse the MIDIs (only the array buffers, MIDI is unchanged)
148
148
  * @type {BasicMIDI[]}
149
149
  */
150
- this.songs = midiBuffers.map(m => BasicMIDI.copyFrom(m));
150
+ this.songs = midiBuffers;
151
151
  if (this.songs.length < 1)
152
152
  {
153
153
  return;
@@ -32,7 +32,7 @@ class BasicSoundBank
32
32
 
33
33
  /**
34
34
  * Soundfont's info stored as name: value. ifil and iver are stored as string representation of float (e.g., 2.1)
35
- * @type {Object<string, string|IndexedByteArray>}
35
+ * @type {Record<string, string|IndexedByteArray>}
36
36
  */
37
37
  soundFontInfo = {};
38
38
 
@@ -77,7 +77,7 @@ class BasicSoundBank
77
77
 
78
78
  /**
79
79
  * Creates a new basic soundfont template (or copies)
80
- * @param data {undefined|{presets: BasicPreset[], info: Object<string, string>}}
80
+ * @param data {undefined|{presets: BasicPreset[], info: Record<string, string>}}
81
81
  */
82
82
  constructor(data = undefined)
83
83
  {
@@ -234,7 +234,7 @@ export function combineZones(preset, globalize = true)
234
234
  continue;
235
235
  }
236
236
  /**
237
- * @type {Object<string, number>}
237
+ * @type {Record<string, number>}
238
238
  */
239
239
  let occurencesForValues = {};
240
240
  const defaultForChecked = generatorLimits[checkedType]?.def || 0;
@@ -12,7 +12,7 @@ export function loadSoundFont(buffer)
12
12
  {
13
13
  const check = buffer.slice(8, 12);
14
14
  const a = new IndexedByteArray(check);
15
- const id = readBytesAsString(a, 4, undefined, false).toLowerCase();
15
+ const id = readBytesAsString(a, 4, false).toLowerCase();
16
16
  if (id === "dls ")
17
17
  {
18
18
  return new DLSSoundFont(buffer);
@@ -101,7 +101,7 @@ export class SoundFont2 extends BasicSoundBank
101
101
  break;
102
102
 
103
103
  case "icmt":
104
- text = readBytesAsString(chunk.chunkData, chunk.chunkData.length, undefined, false);
104
+ text = readBytesAsString(chunk.chunkData, chunk.chunkData.length, false);
105
105
  this.soundFontInfo[chunk.header] = text;
106
106
  break;
107
107
 
@@ -33,7 +33,7 @@ export class SoundFontManager
33
33
  {
34
34
  /**
35
35
  * <"bank-program", "presetName">
36
- * @type {Object<string, string>}
36
+ * @type {Record<string, string>}
37
37
  */
38
38
  const presetList = {};
39
39
  // gather the presets in reverse and replace if necessary
@@ -3,50 +3,39 @@ import { IndexedByteArray } from "../indexed_array.js";
3
3
  /**
4
4
  * @param dataArray {IndexedByteArray}
5
5
  * @param bytes {number}
6
- * @param encoding {string} the textElement encoding
7
6
  * @param trimEnd {boolean} if we should trim once we reach an invalid byte
8
7
  * @returns {string}
9
8
  */
10
- export function readBytesAsString(dataArray, bytes, encoding = undefined, trimEnd = true)
9
+ export function readBytesAsString(dataArray, bytes, trimEnd = true)
11
10
  {
12
- if (!encoding)
11
+ let finished = false;
12
+ let string = "";
13
+ for (let i = 0; i < bytes; i++)
13
14
  {
14
- let finished = false;
15
- let string = "";
16
- for (let i = 0; i < bytes; i++)
15
+ let byte = dataArray[dataArray.currentIndex++];
16
+ if (finished)
17
17
  {
18
- let byte = dataArray[dataArray.currentIndex++];
19
- if (finished)
18
+ continue;
19
+ }
20
+ if ((byte < 32 || byte > 127) && byte !== 10) // 10 is "\n"
21
+ {
22
+ if (trimEnd)
20
23
  {
24
+ finished = true;
21
25
  continue;
22
26
  }
23
- if ((byte < 32 || byte > 127) && byte !== 10) // 10 is "\n"
27
+ else
24
28
  {
25
- if (trimEnd)
29
+ if (byte === 0)
26
30
  {
27
31
  finished = true;
28
32
  continue;
29
33
  }
30
- else
31
- {
32
- if (byte === 0)
33
- {
34
- finished = true;
35
- continue;
36
- }
37
- }
38
34
  }
39
- string += String.fromCharCode(byte);
40
35
  }
41
- return string;
42
- }
43
- else
44
- {
45
- let byteBuffer = dataArray.slice(dataArray.currentIndex, dataArray.currentIndex + bytes);
46
- dataArray.currentIndex += bytes;
47
- let decoder = new TextDecoder(encoding.replace(/[^\x20-\x7E]/g, ""));
48
- return decoder.decode(byteBuffer.buffer);
36
+ string += String.fromCharCode(byte);
49
37
  }
38
+ return string;
50
39
  }
51
40
 
52
41
  /**