@scorelabs/core 1.0.7 → 1.0.10
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/dist/importers/MusicXMLParser.d.ts +7 -2
- package/dist/importers/MusicXMLParser.js +261 -103
- package/dist/index.d.ts +2 -2
- package/dist/index.js +2 -2
- package/dist/models/Instrument.d.ts +8 -19
- package/dist/models/Instrument.js +77 -23
- package/dist/models/Measure.d.ts +5 -1
- package/dist/models/Measure.js +132 -19
- package/dist/models/Note.d.ts +7 -7
- package/dist/models/Note.js +2 -2
- package/dist/models/NoteSet.d.ts +44 -42
- package/dist/models/NoteSet.js +5 -2
- package/dist/models/Pitch.js +4 -0
- package/dist/models/PreMeasure.d.ts +24 -0
- package/dist/models/PreMeasure.js +30 -0
- package/dist/models/Score.d.ts +20 -5
- package/dist/models/Score.js +190 -5
- package/dist/models/index.d.ts +1 -0
- package/dist/models/index.js +1 -0
- package/dist/models/types.d.ts +2 -213
- package/dist/models/types.js +2 -213
- package/dist/types/Accidental.d.ts +7 -0
- package/dist/types/Accidental.js +8 -0
- package/dist/types/Arpeggio.d.ts +5 -0
- package/dist/types/Arpeggio.js +6 -0
- package/dist/types/Articulation.d.ts +10 -0
- package/dist/types/Articulation.js +11 -0
- package/dist/types/BarlineStyle.d.ts +9 -0
- package/dist/types/BarlineStyle.js +10 -0
- package/dist/types/Bowing.d.ts +4 -0
- package/dist/types/Bowing.js +5 -0
- package/dist/types/Clef.d.ts +10 -0
- package/dist/types/Clef.js +11 -0
- package/dist/types/Duration.d.ts +20 -0
- package/dist/types/Duration.js +60 -0
- package/dist/types/Dynamic.d.ts +12 -0
- package/dist/types/Dynamic.js +13 -0
- package/dist/types/Fretboard.d.ts +19 -0
- package/dist/types/Fretboard.js +1 -0
- package/dist/types/Genre.d.ts +27 -0
- package/dist/types/Genre.js +28 -0
- package/dist/types/Glissando.d.ts +8 -0
- package/dist/types/Glissando.js +5 -0
- package/dist/types/Hairpin.d.ts +10 -0
- package/dist/types/Hairpin.js +11 -0
- package/dist/types/InstrumentPreset.d.ts +11 -0
- package/dist/types/InstrumentPreset.js +12 -0
- package/dist/types/InstrumentType.d.ts +8 -0
- package/dist/types/InstrumentType.js +9 -0
- package/dist/types/KeySignature.d.ts +3 -0
- package/dist/types/KeySignature.js +1 -0
- package/dist/types/Lyric.d.ts +11 -0
- package/dist/types/Lyric.js +7 -0
- package/dist/types/NoteheadShape.d.ts +8 -0
- package/dist/types/NoteheadShape.js +9 -0
- package/dist/types/Ornament.d.ts +8 -0
- package/dist/types/Ornament.js +9 -0
- package/dist/types/Ottava.d.ts +10 -0
- package/dist/types/Ottava.js +7 -0
- package/dist/types/Pedal.d.ts +4 -0
- package/dist/types/Pedal.js +1 -0
- package/dist/types/Repeat.d.ts +8 -0
- package/dist/types/Repeat.js +1 -0
- package/dist/types/Slur.d.ts +4 -0
- package/dist/types/Slur.js +1 -0
- package/dist/types/StemDirection.d.ts +4 -0
- package/dist/types/StemDirection.js +5 -0
- package/dist/types/Tempo.d.ts +8 -0
- package/dist/types/Tempo.js +1 -0
- package/dist/types/TimeSignature.d.ts +6 -0
- package/dist/types/TimeSignature.js +3 -0
- package/dist/types/Tuplet.d.ts +5 -0
- package/dist/types/Tuplet.js +1 -0
- package/dist/types/User.d.ts +21 -0
- package/dist/types/User.js +7 -0
- package/dist/types/index.d.ts +27 -0
- package/dist/types/index.js +27 -0
- package/package.json +3 -1
package/dist/models/NoteSet.js
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import { Note } from './Note';
|
|
2
|
-
import { Duration } from './types';
|
|
2
|
+
import { Duration, } from './types';
|
|
3
3
|
export class NoteSet {
|
|
4
4
|
notes;
|
|
5
5
|
constructor(notes) {
|
|
@@ -178,6 +178,9 @@ export class NoteSet {
|
|
|
178
178
|
get slur() {
|
|
179
179
|
return this.notes[0].slur;
|
|
180
180
|
}
|
|
181
|
+
get articulations() {
|
|
182
|
+
return this.notes[0].articulations;
|
|
183
|
+
}
|
|
181
184
|
get articulation() {
|
|
182
185
|
return this.notes[0].articulation;
|
|
183
186
|
}
|
|
@@ -264,7 +267,7 @@ export class NoteSet {
|
|
|
264
267
|
return new NoteSet([new Note(Duration.Quarter, undefined, true)]);
|
|
265
268
|
}
|
|
266
269
|
let notesArray;
|
|
267
|
-
if (
|
|
270
|
+
if ('notes' in data && Array.isArray(data.notes)) {
|
|
268
271
|
notesArray = data.notes;
|
|
269
272
|
}
|
|
270
273
|
else if (Array.isArray(data)) {
|
package/dist/models/Pitch.js
CHANGED
|
@@ -33,7 +33,9 @@ export class Pitch {
|
|
|
33
33
|
getStaffPosition(clef) {
|
|
34
34
|
const middleLineAbsStep = {
|
|
35
35
|
[Clef.Treble]: 4 * 7 + 6, // B4
|
|
36
|
+
[Clef.Treble8vaBassa]: 3 * 7 + 6, // B3
|
|
36
37
|
[Clef.Bass]: 3 * 7 + 1, // D3
|
|
38
|
+
[Clef.Bass8vaBassa]: 2 * 7 + 1, // D2
|
|
37
39
|
[Clef.Alto]: 4 * 7 + 0, // C4
|
|
38
40
|
[Clef.Tenor]: 3 * 7 + 5, // A3
|
|
39
41
|
[Clef.Percussion]: 4 * 7 + 6, // B4
|
|
@@ -100,7 +102,9 @@ export class Pitch {
|
|
|
100
102
|
static fromStaffPosition(clef, staffPosition) {
|
|
101
103
|
const middleLineAbsStep = {
|
|
102
104
|
[Clef.Treble]: 4 * 7 + 6, // B4
|
|
105
|
+
[Clef.Treble8vaBassa]: 3 * 7 + 6, // B3
|
|
103
106
|
[Clef.Bass]: 3 * 7 + 1, // D3
|
|
107
|
+
[Clef.Bass8vaBassa]: 2 * 7 + 1, // D2
|
|
104
108
|
[Clef.Alto]: 4 * 7 + 0, // C4
|
|
105
109
|
[Clef.Tenor]: 3 * 7 + 5, // A3
|
|
106
110
|
[Clef.Percussion]: 4 * 7 + 0,
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
import { Clef, KeySignature, TimeSignature } from './types';
|
|
2
|
+
export interface PreMeasureJSON {
|
|
3
|
+
clef?: string;
|
|
4
|
+
keySignature?: KeySignature;
|
|
5
|
+
timeSignature?: TimeSignature;
|
|
6
|
+
}
|
|
7
|
+
/**
|
|
8
|
+
* Represents the musical context before the first measure of a system.
|
|
9
|
+
* This includes the clef, key signature, and time signature.
|
|
10
|
+
*/
|
|
11
|
+
export declare class PreMeasure {
|
|
12
|
+
readonly clef?: Clef | undefined;
|
|
13
|
+
readonly keySignature?: KeySignature | undefined;
|
|
14
|
+
readonly timeSignature?: TimeSignature | undefined;
|
|
15
|
+
constructor(clef?: Clef | undefined, keySignature?: KeySignature | undefined, timeSignature?: TimeSignature | undefined);
|
|
16
|
+
/**
|
|
17
|
+
* Creates a PreMeasure from JSON data.
|
|
18
|
+
*/
|
|
19
|
+
static fromJSON(data: Partial<PreMeasureJSON>): PreMeasure;
|
|
20
|
+
/**
|
|
21
|
+
* Converts the PreMeasure to a JSON-serializable object.
|
|
22
|
+
*/
|
|
23
|
+
toJSON(): PreMeasureJSON;
|
|
24
|
+
}
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Represents the musical context before the first measure of a system.
|
|
3
|
+
* This includes the clef, key signature, and time signature.
|
|
4
|
+
*/
|
|
5
|
+
export class PreMeasure {
|
|
6
|
+
clef;
|
|
7
|
+
keySignature;
|
|
8
|
+
timeSignature;
|
|
9
|
+
constructor(clef, keySignature, timeSignature) {
|
|
10
|
+
this.clef = clef;
|
|
11
|
+
this.keySignature = keySignature;
|
|
12
|
+
this.timeSignature = timeSignature;
|
|
13
|
+
}
|
|
14
|
+
/**
|
|
15
|
+
* Creates a PreMeasure from JSON data.
|
|
16
|
+
*/
|
|
17
|
+
static fromJSON(data) {
|
|
18
|
+
return new PreMeasure(data.clef, data.keySignature, data.timeSignature);
|
|
19
|
+
}
|
|
20
|
+
/**
|
|
21
|
+
* Converts the PreMeasure to a JSON-serializable object.
|
|
22
|
+
*/
|
|
23
|
+
toJSON() {
|
|
24
|
+
return {
|
|
25
|
+
clef: this.clef,
|
|
26
|
+
keySignature: this.keySignature,
|
|
27
|
+
timeSignature: this.timeSignature,
|
|
28
|
+
};
|
|
29
|
+
}
|
|
30
|
+
}
|
package/dist/models/Score.d.ts
CHANGED
|
@@ -2,7 +2,7 @@ import { Measure } from './Measure';
|
|
|
2
2
|
import { NoteSet } from './NoteSet';
|
|
3
3
|
import { Part, PartJSON } from './Part';
|
|
4
4
|
import { Staff } from './Staff';
|
|
5
|
-
import { Duration, Genre, KeySignature, TimeSignature } from './types';
|
|
5
|
+
import { Clef, Duration, Genre, KeySignature, TimeSignature } from './types';
|
|
6
6
|
/**
|
|
7
7
|
* Represents a complete musical score.
|
|
8
8
|
*/
|
|
@@ -19,13 +19,13 @@ export declare class Score {
|
|
|
19
19
|
readonly lyricist: string;
|
|
20
20
|
readonly swing: boolean;
|
|
21
21
|
readonly subtitle: string;
|
|
22
|
-
readonly genre: Genre
|
|
22
|
+
readonly genre: Genre;
|
|
23
23
|
readonly tempoText: string;
|
|
24
|
-
constructor(title: string, composer: string, timeSignature: TimeSignature, keySignature: KeySignature, parts: Part[], bpm?: number, tempoDuration?: Duration, tempoIsDotted?: boolean, copyright?: string, lyricist?: string, swing?: boolean, subtitle?: string, genre?: Genre
|
|
24
|
+
constructor(title: string, composer: string, timeSignature: TimeSignature, keySignature: KeySignature, parts: Part[], bpm?: number, tempoDuration?: Duration, tempoIsDotted?: boolean, copyright?: string, lyricist?: string, swing?: boolean, subtitle?: string, genre?: Genre, tempoText?: string);
|
|
25
25
|
withTitle(title: string): Score;
|
|
26
26
|
withComposer(composer: string): Score;
|
|
27
27
|
withSubtitle(subtitle: string): Score;
|
|
28
|
-
withGenre(genre: Genre
|
|
28
|
+
withGenre(genre: Genre): Score;
|
|
29
29
|
getMeasureCount(): number;
|
|
30
30
|
getTimeSignatureAt(measureIndex: number): TimeSignature;
|
|
31
31
|
getKeySignatureAt(measureIndex: number): KeySignature;
|
|
@@ -38,6 +38,21 @@ export declare class Score {
|
|
|
38
38
|
static fromJSON(data: ScoreJSON): Score;
|
|
39
39
|
transpose(semitones: number): Score;
|
|
40
40
|
replaceNote(partIndex: number, staffIndex: number, measureIndex: number, noteIndex: number, newNote: NoteSet, voiceIndex?: number): Score;
|
|
41
|
+
withKeySignature(keySig: KeySignature): Score;
|
|
42
|
+
withTimeSignature(timeSig: TimeSignature): Score;
|
|
43
|
+
updateMeasureSignatures(measureIndex: number, signatures: {
|
|
44
|
+
keySignature?: KeySignature;
|
|
45
|
+
timeSignature?: TimeSignature;
|
|
46
|
+
clef?: Clef;
|
|
47
|
+
}, autoBeam?: boolean): Score;
|
|
48
|
+
/**
|
|
49
|
+
* Reflows the notes starting from a specific measure index.
|
|
50
|
+
* This is useful when time signatures change and notes need to be redistributed.
|
|
51
|
+
*/
|
|
52
|
+
reflow(fromMeasureIndex: number, autoBeam?: boolean): Score;
|
|
53
|
+
private isTied;
|
|
54
|
+
private areSamePitch;
|
|
55
|
+
private reflowStaff;
|
|
41
56
|
replaceMeasure(partIndex: number, staffIndex: number, measureIndex: number, newMeasure: Measure, autoBeam?: boolean): Score;
|
|
42
57
|
deleteNote(partIndex: number, staffIndex: number, measureIndex: number, noteIndex: number, voiceIndex?: number): Score;
|
|
43
58
|
changeNoteDuration(partIndex: number, staffIndex: number, measureIndex: number, noteIndex: number, newDuration: Duration, isDotted?: boolean, voiceIndex?: number): Score;
|
|
@@ -70,6 +85,6 @@ export interface ScoreJSON {
|
|
|
70
85
|
lyricist?: string;
|
|
71
86
|
swing?: boolean;
|
|
72
87
|
subtitle?: string;
|
|
73
|
-
genre?: Genre
|
|
88
|
+
genre?: Genre;
|
|
74
89
|
tempoText?: string;
|
|
75
90
|
}
|
package/dist/models/Score.js
CHANGED
|
@@ -1,3 +1,4 @@
|
|
|
1
|
+
import { Measure } from './Measure';
|
|
1
2
|
import { Part } from './Part';
|
|
2
3
|
import { Duration, Genre, decomposeDuration } from './types';
|
|
3
4
|
/**
|
|
@@ -120,6 +121,170 @@ export class Score {
|
|
|
120
121
|
const updatedMeasure = measure.replaceNoteSet(noteIndex, newNote, voiceIndex);
|
|
121
122
|
return this.replaceMeasure(partIndex, staffIndex, measureIndex, updatedMeasure, true);
|
|
122
123
|
}
|
|
124
|
+
withKeySignature(keySig) {
|
|
125
|
+
return new Score(this.title, this.composer, this.timeSignature, keySig, this.parts, this.bpm, this.tempoDuration, this.tempoIsDotted, this.copyright, this.lyricist, this.swing, this.subtitle, this.genre, this.tempoText);
|
|
126
|
+
}
|
|
127
|
+
withTimeSignature(timeSig) {
|
|
128
|
+
return new Score(this.title, this.composer, timeSig, this.keySignature, this.parts, this.bpm, this.tempoDuration, this.tempoIsDotted, this.copyright, this.lyricist, this.swing, this.subtitle, this.genre, this.tempoText);
|
|
129
|
+
}
|
|
130
|
+
updateMeasureSignatures(measureIndex, signatures, autoBeam = true) {
|
|
131
|
+
let score = this;
|
|
132
|
+
for (let pIdx = 0; pIdx < this.parts.length; pIdx++) {
|
|
133
|
+
for (let sIdx = 0; sIdx < this.parts[pIdx].staves.length; sIdx++) {
|
|
134
|
+
const currentMsr = score.parts[pIdx].staves[sIdx].measures[measureIndex];
|
|
135
|
+
if (currentMsr) {
|
|
136
|
+
let updatedMsr = currentMsr;
|
|
137
|
+
if (signatures.keySignature !== undefined) {
|
|
138
|
+
updatedMsr = updatedMsr.withKeySignature(signatures.keySignature);
|
|
139
|
+
}
|
|
140
|
+
if (signatures.timeSignature !== undefined) {
|
|
141
|
+
updatedMsr = updatedMsr.withTimeSignature(signatures.timeSignature);
|
|
142
|
+
}
|
|
143
|
+
if (signatures.clef !== undefined) {
|
|
144
|
+
updatedMsr = updatedMsr.withClef(signatures.clef);
|
|
145
|
+
}
|
|
146
|
+
score = score.replaceMeasure(pIdx, sIdx, measureIndex, updatedMsr, autoBeam);
|
|
147
|
+
}
|
|
148
|
+
}
|
|
149
|
+
}
|
|
150
|
+
if (signatures.timeSignature !== undefined) {
|
|
151
|
+
score = score.reflow(measureIndex, autoBeam);
|
|
152
|
+
}
|
|
153
|
+
return score;
|
|
154
|
+
}
|
|
155
|
+
/**
|
|
156
|
+
* Reflows the notes starting from a specific measure index.
|
|
157
|
+
* This is useful when time signatures change and notes need to be redistributed.
|
|
158
|
+
*/
|
|
159
|
+
reflow(fromMeasureIndex, autoBeam = true) {
|
|
160
|
+
let currentScore = this;
|
|
161
|
+
const allStaves = this.getAllStaves();
|
|
162
|
+
for (const { partIndex, staffIndex } of allStaves) {
|
|
163
|
+
currentScore = currentScore.reflowStaff(partIndex, staffIndex, fromMeasureIndex, autoBeam);
|
|
164
|
+
}
|
|
165
|
+
// Harmonize measure count across all staves
|
|
166
|
+
const maxMeasures = Math.max(...currentScore.parts.flatMap((p) => p.staves.map((s) => s.measures.length)));
|
|
167
|
+
const minMeasures = Math.min(...currentScore.parts.flatMap((p) => p.staves.map((s) => s.measures.length)));
|
|
168
|
+
if (maxMeasures !== minMeasures) {
|
|
169
|
+
for (let pIdx = 0; pIdx < currentScore.parts.length; pIdx++) {
|
|
170
|
+
for (let sIdx = 0; sIdx < currentScore.parts[pIdx].staves.length; sIdx++) {
|
|
171
|
+
const staff = currentScore.parts[pIdx].staves[sIdx];
|
|
172
|
+
if (staff.measures.length < maxMeasures) {
|
|
173
|
+
let updatedMsrs = [...staff.measures];
|
|
174
|
+
for (let i = staff.measures.length; i < maxMeasures; i++) {
|
|
175
|
+
const ts = currentScore.getTimeSignatureAt(i);
|
|
176
|
+
const targetDur = ts.beats * (4 / ts.beatType);
|
|
177
|
+
updatedMsrs.push(new Measure([[]]).fillVoiceWithRests(0, targetDur));
|
|
178
|
+
}
|
|
179
|
+
currentScore = currentScore.replaceStaff(pIdx, sIdx, staff.withMeasures(updatedMsrs));
|
|
180
|
+
}
|
|
181
|
+
}
|
|
182
|
+
}
|
|
183
|
+
}
|
|
184
|
+
return currentScore;
|
|
185
|
+
}
|
|
186
|
+
isTied(ns) {
|
|
187
|
+
return !!ns.notes[0].tie;
|
|
188
|
+
}
|
|
189
|
+
areSamePitch(ns1, ns2) {
|
|
190
|
+
if (ns1.isRest !== ns2.isRest)
|
|
191
|
+
return false;
|
|
192
|
+
if (ns1.isRest)
|
|
193
|
+
return true;
|
|
194
|
+
if (ns1.notes.length !== ns2.notes.length)
|
|
195
|
+
return false;
|
|
196
|
+
const p1 = ns1.notes.map((n) => n.pitch?.midiNumber || -1).sort((a, b) => a - b);
|
|
197
|
+
const p2 = ns2.notes.map((n) => n.pitch?.midiNumber || -1).sort((a, b) => a - b);
|
|
198
|
+
return p1.every((v, i) => v === p2[i]);
|
|
199
|
+
}
|
|
200
|
+
reflowStaff(pIdx, sIdx, fromMeasureIndex, autoBeam) {
|
|
201
|
+
const staff = this.parts[pIdx].staves[sIdx];
|
|
202
|
+
const measures = staff.measures;
|
|
203
|
+
const maxVoices = Math.max(...measures.map((m) => m.voices.length), 1);
|
|
204
|
+
let updatedMeasures = [...measures];
|
|
205
|
+
for (let vIdx = 0; vIdx < maxVoices; vIdx++) {
|
|
206
|
+
// 1. Collect notes and merge tied notes into logical streams
|
|
207
|
+
let stream = [];
|
|
208
|
+
for (let mIdx = fromMeasureIndex; mIdx < measures.length; mIdx++) {
|
|
209
|
+
const msr = measures[mIdx];
|
|
210
|
+
const voice = msr.voices[vIdx] || [];
|
|
211
|
+
for (const ns of voice) {
|
|
212
|
+
const d = ns.getDurationValue();
|
|
213
|
+
if (stream.length > 0 &&
|
|
214
|
+
this.isTied(stream[stream.length - 1].ns) &&
|
|
215
|
+
this.areSamePitch(stream[stream.length - 1].ns, ns)) {
|
|
216
|
+
stream[stream.length - 1].duration += d;
|
|
217
|
+
}
|
|
218
|
+
else {
|
|
219
|
+
stream.push({ ns, duration: d });
|
|
220
|
+
}
|
|
221
|
+
}
|
|
222
|
+
}
|
|
223
|
+
// 2. Redistribute logical notes into measures according to time signatures
|
|
224
|
+
let currentMIdx = fromMeasureIndex;
|
|
225
|
+
let streamIdx = 0;
|
|
226
|
+
while (streamIdx < stream.length) {
|
|
227
|
+
if (currentMIdx >= updatedMeasures.length) {
|
|
228
|
+
updatedMeasures.push(new Measure([[]]));
|
|
229
|
+
}
|
|
230
|
+
let msr = updatedMeasures[currentMIdx];
|
|
231
|
+
const ts = this.getTimeSignatureAt(currentMIdx);
|
|
232
|
+
const targetDur = ts.beats * (4 / ts.beatType);
|
|
233
|
+
const newVoices = [...msr.voices];
|
|
234
|
+
while (newVoices.length <= vIdx)
|
|
235
|
+
newVoices.push([]);
|
|
236
|
+
newVoices[vIdx] = [];
|
|
237
|
+
msr = msr.withVoices(newVoices);
|
|
238
|
+
let currentDur = 0;
|
|
239
|
+
while (streamIdx < stream.length &&
|
|
240
|
+
(currentDur < targetDur - 0.001 || stream[streamIdx].duration === 0)) {
|
|
241
|
+
const logicalNote = stream[streamIdx];
|
|
242
|
+
if (logicalNote.duration === 0) {
|
|
243
|
+
// Grace note or zero-duration element
|
|
244
|
+
msr = msr.withVoices(msr.voices.map((v, i) => (i === vIdx ? [...v, logicalNote.ns] : v)));
|
|
245
|
+
streamIdx++;
|
|
246
|
+
continue;
|
|
247
|
+
}
|
|
248
|
+
const remainingRoom = targetDur - currentDur;
|
|
249
|
+
const toTake = Math.min(logicalNote.duration, remainingRoom);
|
|
250
|
+
const parts = decomposeDuration(toTake);
|
|
251
|
+
const newNoteSets = parts.map((p, idx) => {
|
|
252
|
+
const isLastOfLogical = Math.abs(toTake - logicalNote.duration) < 0.001 && idx === parts.length - 1;
|
|
253
|
+
return logicalNote.ns
|
|
254
|
+
.withDuration(p.duration, p.isDotted)
|
|
255
|
+
.withTie(isLastOfLogical ? !!logicalNote.ns.notes[0].tie : true);
|
|
256
|
+
});
|
|
257
|
+
msr = msr.withVoices(msr.voices.map((v, i) => (i === vIdx ? [...v, ...newNoteSets] : v)));
|
|
258
|
+
logicalNote.duration -= toTake;
|
|
259
|
+
currentDur += toTake;
|
|
260
|
+
if (logicalNote.duration < 0.001) {
|
|
261
|
+
streamIdx++;
|
|
262
|
+
}
|
|
263
|
+
}
|
|
264
|
+
// Fill with rests if score is not full
|
|
265
|
+
if (currentDur < targetDur - 0.001) {
|
|
266
|
+
msr = msr.fillVoiceWithRests(vIdx, targetDur);
|
|
267
|
+
}
|
|
268
|
+
if (autoBeam) {
|
|
269
|
+
msr = msr.autoBeam(ts);
|
|
270
|
+
}
|
|
271
|
+
updatedMeasures[currentMIdx] = msr;
|
|
272
|
+
currentMIdx++;
|
|
273
|
+
}
|
|
274
|
+
// Ensure any remaining measures in this staff are also cleared/filled with rests for this voice
|
|
275
|
+
for (let mIdx = currentMIdx; mIdx < updatedMeasures.length; mIdx++) {
|
|
276
|
+
const ts = this.getTimeSignatureAt(mIdx);
|
|
277
|
+
const targetDur = ts.beats * (4 / ts.beatType);
|
|
278
|
+
updatedMeasures[mIdx] = updatedMeasures[mIdx]
|
|
279
|
+
.withVoices(updatedMeasures[mIdx].voices.map((v, i) => (i === vIdx ? [] : v)))
|
|
280
|
+
.fillVoiceWithRests(vIdx, targetDur);
|
|
281
|
+
if (autoBeam) {
|
|
282
|
+
updatedMeasures[mIdx] = updatedMeasures[mIdx].autoBeam(ts);
|
|
283
|
+
}
|
|
284
|
+
}
|
|
285
|
+
}
|
|
286
|
+
return this.replaceStaff(pIdx, sIdx, staff.withMeasures(updatedMeasures));
|
|
287
|
+
}
|
|
123
288
|
replaceMeasure(partIndex, staffIndex, measureIndex, newMeasure, autoBeam = true) {
|
|
124
289
|
if (partIndex < 0 || partIndex >= this.parts.length)
|
|
125
290
|
return this;
|
|
@@ -273,17 +438,27 @@ export class Score {
|
|
|
273
438
|
const measures = this.parts[0]?.staves[0]?.measures || [];
|
|
274
439
|
const sequence = [];
|
|
275
440
|
let i = 0;
|
|
276
|
-
|
|
441
|
+
const startRepeatStack = [0]; // Implicit start at 0
|
|
442
|
+
let justJumpedTo = -1;
|
|
277
443
|
const repeatCount = new Map();
|
|
278
444
|
let safetyCounter = 0;
|
|
279
445
|
const MAX_SEQUENCE = 10000;
|
|
280
446
|
while (i < measures.length && safetyCounter < MAX_SEQUENCE) {
|
|
281
447
|
safetyCounter++;
|
|
282
448
|
const m = measures[i];
|
|
283
|
-
|
|
284
|
-
|
|
449
|
+
// Handle Start Repeats
|
|
450
|
+
if (m.repeats.some((r) => r.type === 'start')) {
|
|
451
|
+
// Only push if we didn't just jump here (avoid re-pushing on loop reentry)
|
|
452
|
+
if (i !== justJumpedTo) {
|
|
453
|
+
startRepeatStack.push(i);
|
|
454
|
+
}
|
|
455
|
+
}
|
|
456
|
+
// Reset jump flag after processing start repeats check
|
|
457
|
+
if (i !== justJumpedTo)
|
|
458
|
+
justJumpedTo = -1;
|
|
285
459
|
const endRepeat = m.repeats.find((r) => r.type === 'end');
|
|
286
460
|
let iteration = 1;
|
|
461
|
+
// Calculate current iteration for Volta logic
|
|
287
462
|
if (endRepeat)
|
|
288
463
|
iteration = (repeatCount.get(i) || 0) + 1;
|
|
289
464
|
else {
|
|
@@ -307,11 +482,21 @@ export class Score {
|
|
|
307
482
|
const currentCount = repeatCount.get(i) || 0;
|
|
308
483
|
if (currentCount + 1 < maxTimes) {
|
|
309
484
|
repeatCount.set(i, currentCount + 1);
|
|
310
|
-
|
|
485
|
+
// Jump to the nearest start repeat
|
|
486
|
+
const target = startRepeatStack[startRepeatStack.length - 1];
|
|
487
|
+
i = target;
|
|
488
|
+
justJumpedTo = target;
|
|
311
489
|
continue;
|
|
312
490
|
}
|
|
313
|
-
else
|
|
491
|
+
else {
|
|
492
|
+
// Finished loop
|
|
314
493
|
repeatCount.set(i, 0);
|
|
494
|
+
// Pop the start repeat if we have one (keeping the implicit 0)
|
|
495
|
+
if (startRepeatStack.length > 1) {
|
|
496
|
+
startRepeatStack.pop();
|
|
497
|
+
}
|
|
498
|
+
// Do not pop 0, so unmatched repeats will default to 0
|
|
499
|
+
}
|
|
315
500
|
}
|
|
316
501
|
i++;
|
|
317
502
|
}
|
package/dist/models/index.d.ts
CHANGED
package/dist/models/index.js
CHANGED
package/dist/models/types.d.ts
CHANGED
|
@@ -1,216 +1,5 @@
|
|
|
1
1
|
/**
|
|
2
2
|
* Shared types and enums for ScoreLabs
|
|
3
|
+
* All types have been moved to ../types/ for better organization.
|
|
3
4
|
*/
|
|
4
|
-
export
|
|
5
|
-
Up = "up",
|
|
6
|
-
Down = "down"
|
|
7
|
-
}
|
|
8
|
-
export declare enum Clef {
|
|
9
|
-
Treble = "treble",
|
|
10
|
-
Bass = "bass",
|
|
11
|
-
Alto = "alto",
|
|
12
|
-
Tenor = "tenor",
|
|
13
|
-
Percussion = "percussion",
|
|
14
|
-
Tab = "tab"
|
|
15
|
-
}
|
|
16
|
-
export declare enum Duration {
|
|
17
|
-
Whole = "whole",
|
|
18
|
-
Half = "half",
|
|
19
|
-
Quarter = "quarter",
|
|
20
|
-
Eighth = "eighth",
|
|
21
|
-
Sixteenth = "sixteenth",
|
|
22
|
-
ThirtySecond = "thirty-second",
|
|
23
|
-
SixtyFourth = "sixty-fourth",
|
|
24
|
-
OneHundredTwentyEighth = "one-hundred-twenty-eighth",
|
|
25
|
-
TwoHundredFiftySixth = "two-hundred-fifty-sixth"
|
|
26
|
-
}
|
|
27
|
-
export declare enum Accidental {
|
|
28
|
-
Sharp = "sharp",
|
|
29
|
-
Flat = "flat",
|
|
30
|
-
Natural = "natural",
|
|
31
|
-
DoubleSharp = "double-sharp",
|
|
32
|
-
DoubleFlat = "double-flat"
|
|
33
|
-
}
|
|
34
|
-
export declare enum Articulation {
|
|
35
|
-
Staccato = "staccato",
|
|
36
|
-
Accent = "accent",
|
|
37
|
-
Tenuto = "tenuto",
|
|
38
|
-
Marcato = "marcato",
|
|
39
|
-
Fermata = "fermata",
|
|
40
|
-
Staccatissimo = "staccatissimo",
|
|
41
|
-
Caesura = "caesura",
|
|
42
|
-
BreathMark = "breath-mark"
|
|
43
|
-
}
|
|
44
|
-
export declare enum Bowing {
|
|
45
|
-
DownBow = "down-bow",
|
|
46
|
-
UpBow = "up-bow"
|
|
47
|
-
}
|
|
48
|
-
export declare enum Syllabic {
|
|
49
|
-
Single = "single",
|
|
50
|
-
Begin = "begin",
|
|
51
|
-
Middle = "middle",
|
|
52
|
-
End = "end"
|
|
53
|
-
}
|
|
54
|
-
export interface Lyric {
|
|
55
|
-
text: string;
|
|
56
|
-
syllabic?: Syllabic;
|
|
57
|
-
isExtension?: boolean;
|
|
58
|
-
}
|
|
59
|
-
export declare enum NoteheadShape {
|
|
60
|
-
Normal = "normal",
|
|
61
|
-
Cross = "cross",// X shape (percussion, spoken)
|
|
62
|
-
Diamond = "diamond",// Harmonics
|
|
63
|
-
Slash = "slash",// Rhythmic notation
|
|
64
|
-
Triangle = "triangle",// Percussion
|
|
65
|
-
Square = "square"
|
|
66
|
-
}
|
|
67
|
-
export declare enum Dynamic {
|
|
68
|
-
PPP = "ppp",
|
|
69
|
-
PP = "pp",
|
|
70
|
-
P = "p",
|
|
71
|
-
MP = "mp",
|
|
72
|
-
MF = "mf",
|
|
73
|
-
F = "f",
|
|
74
|
-
FF = "ff",
|
|
75
|
-
FFF = "fff",
|
|
76
|
-
SFZ = "sfz",
|
|
77
|
-
FP = "fp"
|
|
78
|
-
}
|
|
79
|
-
export declare enum Genre {
|
|
80
|
-
Blues = "Blues",
|
|
81
|
-
Children = "Children",
|
|
82
|
-
Christian = "Christian",
|
|
83
|
-
Christmas = "Christmas",
|
|
84
|
-
Classical = "Classical",
|
|
85
|
-
ContestFestival = "Contest/Festival",
|
|
86
|
-
Country = "Country",
|
|
87
|
-
Educational = "Educational",
|
|
88
|
-
FilmTV = "Film/TV",
|
|
89
|
-
Folk = "Folk",
|
|
90
|
-
Games = "Games",
|
|
91
|
-
Gospel = "Gospel",
|
|
92
|
-
Holiday = "Holiday",
|
|
93
|
-
Jazz = "Jazz",
|
|
94
|
-
Latin = "Latin",
|
|
95
|
-
Musicals = "Musicals",
|
|
96
|
-
Pop = "Pop",
|
|
97
|
-
RBHipHop = "R&B/Hip-Hop",
|
|
98
|
-
Rock = "Rock",
|
|
99
|
-
Standards = "Standards",
|
|
100
|
-
Traditional = "Traditional",
|
|
101
|
-
Wedding = "Wedding",
|
|
102
|
-
World = "World",
|
|
103
|
-
Worship = "Worship",
|
|
104
|
-
Unknown = "unknown"
|
|
105
|
-
}
|
|
106
|
-
export interface TimeSignature {
|
|
107
|
-
beats: number;
|
|
108
|
-
beatType: number;
|
|
109
|
-
symbol?: 'common' | 'cut' | 'normal';
|
|
110
|
-
}
|
|
111
|
-
export interface KeySignature {
|
|
112
|
-
fifths: number;
|
|
113
|
-
}
|
|
114
|
-
export interface Tempo {
|
|
115
|
-
bpm: number;
|
|
116
|
-
duration: Duration;
|
|
117
|
-
isDotted: boolean;
|
|
118
|
-
text?: string;
|
|
119
|
-
}
|
|
120
|
-
export interface Slur {
|
|
121
|
-
placement: 'start' | 'stop';
|
|
122
|
-
direction?: 'up' | 'down';
|
|
123
|
-
}
|
|
124
|
-
export interface Tuplet {
|
|
125
|
-
actual: number;
|
|
126
|
-
normal: number;
|
|
127
|
-
type: 'start' | 'stop' | 'middle';
|
|
128
|
-
}
|
|
129
|
-
export declare enum HairpinType {
|
|
130
|
-
Crescendo = "crescendo",
|
|
131
|
-
Decrescendo = "decrescendo"
|
|
132
|
-
}
|
|
133
|
-
export interface Hairpin {
|
|
134
|
-
type: HairpinType;
|
|
135
|
-
placement: 'start' | 'stop';
|
|
136
|
-
}
|
|
137
|
-
export interface Repeat {
|
|
138
|
-
type: 'start' | 'end';
|
|
139
|
-
times?: number;
|
|
140
|
-
}
|
|
141
|
-
export interface Volta {
|
|
142
|
-
type: 'start' | 'stop' | 'both';
|
|
143
|
-
numbers: number[];
|
|
144
|
-
}
|
|
145
|
-
export declare enum BarlineStyle {
|
|
146
|
-
Regular = "regular",
|
|
147
|
-
Double = "light-light",// Double barline (often used for key changes)
|
|
148
|
-
Final = "light-heavy",
|
|
149
|
-
Dotted = "dotted",
|
|
150
|
-
Dashed = "dashed",
|
|
151
|
-
Heavy = "heavy",
|
|
152
|
-
None = "none"
|
|
153
|
-
}
|
|
154
|
-
export declare enum GlissandoType {
|
|
155
|
-
Wavy = "wavy",
|
|
156
|
-
Straight = "straight"
|
|
157
|
-
}
|
|
158
|
-
export interface Glissando {
|
|
159
|
-
type: GlissandoType;
|
|
160
|
-
placement: 'start' | 'stop';
|
|
161
|
-
}
|
|
162
|
-
export declare enum Arpeggio {
|
|
163
|
-
Normal = "normal",
|
|
164
|
-
Up = "up",
|
|
165
|
-
Down = "down"
|
|
166
|
-
}
|
|
167
|
-
export declare enum OttavaType {
|
|
168
|
-
OttavaAlta = "8va",
|
|
169
|
-
OttavaBassa = "8vb",
|
|
170
|
-
QuindicesimaAlta = "15ma",
|
|
171
|
-
QuindicesimaBassa = "15mb"
|
|
172
|
-
}
|
|
173
|
-
export interface Ottava {
|
|
174
|
-
type: OttavaType;
|
|
175
|
-
placement: 'start' | 'stop';
|
|
176
|
-
}
|
|
177
|
-
export interface Pedal {
|
|
178
|
-
type: 'sustain' | 'una-corda';
|
|
179
|
-
placement: 'start' | 'stop';
|
|
180
|
-
}
|
|
181
|
-
export declare enum Ornament {
|
|
182
|
-
Trill = "trill",
|
|
183
|
-
Mordent = "mordent",
|
|
184
|
-
InvertedMordent = "inverted-mordent",
|
|
185
|
-
Turn = "turn",
|
|
186
|
-
InvertedTurn = "inverted-turn",
|
|
187
|
-
Tremolo = "tremolo"
|
|
188
|
-
}
|
|
189
|
-
export declare const DURATION_VALUES: Record<Duration, number>;
|
|
190
|
-
/**
|
|
191
|
-
* Decomposes a duration value (quarter = 1) into a list of duration/dot pairs.
|
|
192
|
-
*/
|
|
193
|
-
export declare function decomposeDuration(value: number): {
|
|
194
|
-
duration: Duration;
|
|
195
|
-
isDotted: boolean;
|
|
196
|
-
val: number;
|
|
197
|
-
}[];
|
|
198
|
-
export interface FretboardDot {
|
|
199
|
-
string: number;
|
|
200
|
-
fret: number;
|
|
201
|
-
label?: string;
|
|
202
|
-
}
|
|
203
|
-
export interface FretboardBarre {
|
|
204
|
-
fret: number;
|
|
205
|
-
startString: number;
|
|
206
|
-
endString: number;
|
|
207
|
-
}
|
|
208
|
-
export interface FretboardDiagram {
|
|
209
|
-
strings: number;
|
|
210
|
-
frets: number;
|
|
211
|
-
startingFret?: number;
|
|
212
|
-
dots: FretboardDot[];
|
|
213
|
-
barres?: FretboardBarre[];
|
|
214
|
-
openStrings?: number[];
|
|
215
|
-
mutedStrings?: number[];
|
|
216
|
-
}
|
|
5
|
+
export * from '../types/index.js';
|