Verovio
Source code documentation
midifunctor.h
1 // Name: midifunctor.h
3 // Author: David Bauer
4 // Created: 2023
5 // Copyright (c) Authors and others. All rights reserved.
7 
8 #ifndef __VRV_MIDIFUNCTOR_H__
9 #define __VRV_MIDIFUNCTOR_H__
10 
11 #include "functor.h"
12 
13 //----------------------------------------------------------------------------
14 
15 #include "alignfunctor.h"
16 
17 namespace smf {
18 class MidiFile;
19 }
20 
21 namespace vrv {
22 
23 class FeatureExtractor;
24 class Timemap;
25 
26 //----------------------------------------------------------------------------
27 // InitOnsetOffsetFunctor
28 //----------------------------------------------------------------------------
29 
34 public:
40  virtual ~InitOnsetOffsetFunctor() = default;
42 
43  /*
44  * Abstract base implementation
45  */
46  bool ImplementsEndInterface() const override { return true; }
47 
48  /*
49  * Functor interface
50  */
52  FunctorCode VisitChordEnd(Chord *chord) override;
53  FunctorCode VisitLayer(Layer *layer) override;
54  FunctorCode VisitLayerElement(LayerElement *layerElement) override;
55  FunctorCode VisitMeasure(Measure *measure) override;
56  FunctorCode VisitStaff(Staff *staff) override;
57  FunctorCode VisitTabGrpEnd(TabGrp *tabGrp) override;
59 
60 protected:
61  //
62 private:
63  //
64 public:
65  //
66 private:
67  // The current score time in the measure (incremented by each element)
68  Fraction m_currentScoreTime;
69  // The current real time in seconds in the measure (incremented by each element)
70  double m_currentRealTimeSeconds;
71  // The current time alignment parameters
72  AlignMeterParams m_meterParams;
73  // The current notation type
74  data_NOTATIONTYPE m_notationType;
75  // The current tempo
76  double m_currentTempo;
77 };
78 
79 //----------------------------------------------------------------------------
80 // InitMaxMeasureDurationFunctor
81 //----------------------------------------------------------------------------
82 
87 public:
93  virtual ~InitMaxMeasureDurationFunctor() = default;
95 
96  /*
97  * Abstract base implementation
98  */
99  bool ImplementsEndInterface() const override { return true; }
100 
101  /*
102  * Set the tempo
103  */
105  void SetCurrentTempo(double tempo) { m_currentTempo = tempo; }
106  void SetTempoAdjustment(double adjustment) { m_tempoAdjustment = adjustment; }
108 
109  /*
110  * Functor interface
111  */
113  FunctorCode VisitLayerElement(LayerElement *layerElement) override;
114  FunctorCode VisitMeasure(Measure *measure) override;
115  FunctorCode VisitMeasureEnd(Measure *measure) override;
116  FunctorCode VisitScoreDef(ScoreDef *scoreDef) override;
117  FunctorCode VisitTempo(Tempo *tempo) override;
119 
120 protected:
121  //
122 private:
123  // Returns the current tempo with adjustment
124  double GetAdjustedTempo() const { return m_currentTempo * m_tempoAdjustment; }
125 
126 public:
127  //
128 private:
129  // The current score time
130  Fraction m_currentScoreTime;
131  // The current time in seconds
132  double m_currentRealTimeSeconds;
133  // The current tempo
134  double m_currentTempo;
135  // The tempo adjustment
136  double m_tempoAdjustment;
137  // The factor for multibar rests
138  int m_multiRestFactor;
139 };
140 
141 //----------------------------------------------------------------------------
142 // InitTimemapTiesFunctor
143 //----------------------------------------------------------------------------
144 
149 public:
155  virtual ~InitTimemapTiesFunctor() = default;
157 
158  /*
159  * Abstract base implementation
160  */
161  bool ImplementsEndInterface() const override { return false; }
162 
163  /*
164  * Functor interface
165  */
167  FunctorCode VisitTie(Tie *tie) override;
169 
170 protected:
171  //
172 private:
173  //
174 public:
175  //
176 private:
177  //
178 };
179 //----------------------------------------------------------------------------
180 // InitTimemapAdjustNotesFunctor
181 //----------------------------------------------------------------------------
182 
187 public:
193  virtual ~InitTimemapAdjustNotesFunctor() = default;
195 
196  /*
197  * Abstract base implementation
198  */
199  bool ImplementsEndInterface() const override { return true; }
200 
201  /*
202  * Setter for various properties
203  */
205  void SetNoCue(bool noCue) { m_noCue = noCue; }
207 
208  /*
209  * Functor interface
210  */
212  FunctorCode VisitArpeg(Arpeg *arpeg) override;
213  FunctorCode VisitChord(Chord *chord) override;
214  FunctorCode VisitGraceGrpEnd(GraceGrp *graceGrp) override;
215  FunctorCode VisitMeasure(Measure *measure) override;
216  FunctorCode VisitNote(Note *note) override;
218 
219 protected:
220  //
221 private:
225  struct Grace {
226  std::list<Note *> notes;
227  data_DURATION duration;
228  data_PERCENT time;
229  };
230 
234  void SetGraceNotesFor(Note *refNote);
235 
239  void SetNoteStartStop(Note *note, const Fraction &startTime, const Fraction &stopTime);
240  void SetNoteStart(Note *note, const Fraction &startTime);
241 
242 public:
243  //
244 private:
245  // Indicates whether cue notes should be included
246  bool m_noCue;
247  // Grace note/chord sequence
248  std::list<Grace> m_graces;
249  // Indicates whether the last grace note/chord was accented
250  bool m_accentedGraceNote;
251  // The current tempo
252  double m_currentTempo;
253  // The last (non grace) note
254  Note *m_lastNote;
255 };
256 
257 //----------------------------------------------------------------------------
258 // InitMIDIFunctor
259 //----------------------------------------------------------------------------
260 
264 struct OctaveInfo {
265  const Octave *octave;
266  int staffN;
267  int layerN;
268  int octaveShift;
269  bool isActive;
270 };
271 
278 public:
282  InitMIDIFunctor();
284  virtual ~InitMIDIFunctor() = default;
286 
287  /*
288  * Abstract base implementation
289  */
290  bool ImplementsEndInterface() const override { return false; }
291 
292  /*
293  * Setter and getter for properties
294  */
296  void SetCurrentTempo(double tempo) { m_currentTempo = tempo; }
297  const std::list<OctaveInfo> &GetOctaves() const { return m_octaves; }
299 
300  /*
301  * Functor interface
302  */
304  FunctorCode VisitMeasure(const Measure *measure) override;
305  FunctorCode VisitOctave(const Octave *octave) override;
307 
308 protected:
309  //
310 private:
311  //
312 public:
313  //
314 private:
315  // The current tempo
316  double m_currentTempo;
317  // Octave info which is collected
318  std::list<OctaveInfo> m_octaves;
319 };
320 
321 //----------------------------------------------------------------------------
322 // GenerateMIDIFunctor
323 //----------------------------------------------------------------------------
324 
328 struct MIDINote {
329  int pitch;
330  double duration;
331 };
332 
333 using MIDINoteSequence = std::list<MIDINote>;
334 
338 struct MIDIHeldNote {
339  int m_pitch = 0;
340  double m_stopTime = 0;
341 };
342 
347 public:
351  GenerateMIDIFunctor(smf::MidiFile *midiFile);
353  virtual ~GenerateMIDIFunctor() = default;
355 
356  /*
357  * Abstract base implementation
358  */
359  bool ImplementsEndInterface() const override { return true; }
360 
361  /*
362  * Getter for properties
363  */
365  std::set<int> GetTempoEventTicks() const { return m_tempoEventTicks; }
367 
368  /*
369  * Setter for various properties
370  */
372  void SetChannel(int channel) { m_midiChannel = channel; }
373  void SetControlEvents(bool controlEvents) { m_controlEvents = controlEvents; }
374  void SetNoCue(bool noCue) { m_noCue = noCue; }
375  void SetCurrentTempo(double tempo) { m_currentTempo = tempo; }
376  void SetDeferredNotes(const std::map<const Note *, double> &deferredNotes) { m_deferredNotes = deferredNotes; }
377  void SetLayerN(int layerN) { m_layerN = layerN; }
378  void SetOctaves(const std::list<OctaveInfo> &octaves) { m_octaves = octaves; }
379  void SetStaffN(int staffN) { m_staffN = staffN; }
380  void SetTempoEventTicks(const std::set<int> &ticks) { m_tempoEventTicks = ticks; }
381  void SetTrack(int track) { m_midiTrack = track; }
382  void SetTransSemi(int transSemi) { m_transSemi = transSemi; }
384 
385  /*
386  * Functor interface
387  */
389  FunctorCode VisitBeatRpt(const BeatRpt *beatRpt) override;
390  FunctorCode VisitBTrem(const BTrem *bTrem) override;
391  FunctorCode VisitChord(const Chord *chord) override;
392  FunctorCode VisitFTrem(const FTrem *fTrem) override;
393  FunctorCode VisitHalfmRpt(const HalfmRpt *halfmRpt) override;
394  FunctorCode VisitLayer(const Layer *layer) override;
395  FunctorCode VisitLayerEnd(const Layer *layer) override;
396  FunctorCode VisitLayerElement(const LayerElement *layerElement) override;
397  FunctorCode VisitMeasure(const Measure *measure) override;
398  FunctorCode VisitMRpt(const MRpt *mRpt) override;
399  FunctorCode VisitNote(const Note *note) override;
400  FunctorCode VisitPedal(const Pedal *pedal) override;
401  FunctorCode VisitScoreDef(const ScoreDef *scoreDef) override;
402  FunctorCode VisitStaff(const Staff *staff) override;
403  FunctorCode VisitStaffDef(const StaffDef *staffDef) override;
404  FunctorCode VisitSyl(const Syl *syl) override;
405  FunctorCode VisitVerse(const Verse *verse) override;
407 
408 protected:
409  //
410 private:
414  void DeferMIDINote(const Note *refNote, double shift, bool includeChordSiblings);
415 
419  void HandleOctave(const LayerElement *layerElement);
420 
424  int GetMIDIPitch(const Note *note) { return note->GetMIDIPitch(m_transSemi, m_octaveShift); }
425 
426 public:
427  //
428 private:
429  // The MidiFile we are writing to
430  smf::MidiFile *m_midiFile;
431  // The MIDI track number
432  int m_midiTrack;
433  // The MIDI channel number
434  int m_midiChannel;
435  // The score time from the start of the music to the start of the current measure
436  double m_totalTime;
437  // The current staff number
438  int m_staffN;
439  // The current layer number
440  int m_layerN;
441  // The semi tone transposition for the current track
442  int m_transSemi;
443  // The octave shift for the current track
444  int m_octaveShift;
445  // The current tempo
446  double m_currentTempo;
447  // Tempo events are always added on track 0
448  // This set contains the ticks of all added tempo events to avoid multiple events at the same time
449  std::set<int> m_tempoEventTicks;
450  // The last (non grace) note that was performed
451  const Note *m_lastNote;
452  // Expanded notes due to ornaments and tremolandi
453  std::map<const Note *, MIDINoteSequence> m_expandedNotes;
454  // Deferred notes which start slightly later
455  std::map<const Note *, double> m_deferredNotes;
456  // Octave info which is used to determine the octave shift
457  std::list<OctaveInfo> m_octaves;
458  // Indicates whether cue notes should be included
459  bool m_noCue;
460  // Tablature held notes indexed by (course - 1)
461  std::vector<MIDIHeldNote> m_heldNotes;
462  // A flag indicating we want to process control events
463  bool m_controlEvents;
464 };
465 
466 //----------------------------------------------------------------------------
467 // GenerateTimemapFunctor
468 //----------------------------------------------------------------------------
469 
474 public:
480  virtual ~GenerateTimemapFunctor() = default;
482 
483  /*
484  * Abstract base implementation
485  */
486  bool ImplementsEndInterface() const override { return false; }
487 
488  /*
489  * Set the cue exclusion flag
490  */
491  void SetNoCue(bool noCue) { m_noCue = noCue; }
492 
493  /*
494  * Functor interface
495  */
497  FunctorCode VisitLayerElement(const LayerElement *layerElement) override;
498  FunctorCode VisitMeasure(const Measure *measure) override;
499  FunctorCode VisitNote(const Note *note) override;
500  FunctorCode VisitRest(const Rest *rest) override;
502 
503 protected:
504  //
505 private:
509  void AddTimemapEntry(const Object *object);
510 
511 public:
512  //
513 private:
514  // The score time from the start of the piece to the previous barline in quarter notes
515  Fraction m_scoreTimeOffset;
516  // Real time from the start of the piece to the previous barline in ms
517  double m_realTimeOffsetMilliseconds;
518  // The current tempo
519  double m_currentTempo;
520  // Indicates whether cue notes should be included
521  bool m_noCue;
522  // The timemap
523  Timemap *m_timemap;
524 };
525 
526 //----------------------------------------------------------------------------
527 // GenerateFeaturesFunctor
528 //----------------------------------------------------------------------------
529 
534 public:
540  virtual ~GenerateFeaturesFunctor() = default;
542 
543  /*
544  * Abstract base implementation
545  */
546  bool ImplementsEndInterface() const override { return false; }
547 
548  /*
549  * Functor interface
550  */
552  FunctorCode VisitObject(const Object *object) override;
554 
555 protected:
556  //
557 private:
558  //
559 public:
560  //
561 private:
562  // The FeatureExtractor to which extraction is delegated
563  FeatureExtractor *m_extractor;
564 };
565 
566 } // namespace vrv
567 
568 #endif // __VRV_MIDIFUNCTOR_H__
vrv::MIDIHeldNote
Helper struct for held notes in tablature.
Definition: midifunctor.h:338
vrv::Staff
This class represents a staff in a laid-out score (Doc).
Definition: staff.h:102
vrv::Measure
This class represents a measure in a page-based score (Doc).
Definition: measure.h:37
vrv::InitMaxMeasureDurationFunctor::ImplementsEndInterface
bool ImplementsEndInterface() const override
Return true if the functor implements the end interface.
Definition: midifunctor.h:99
vrv::GenerateFeaturesFunctor::ImplementsEndInterface
bool ImplementsEndInterface() const override
Return true if the functor implements the end interface.
Definition: midifunctor.h:546
vrv::Doc
This class is a hold the data and corresponds to the model of a MVC design pattern.
Definition: doc.h:41
vrv::InitTimemapTiesFunctor::ImplementsEndInterface
bool ImplementsEndInterface() const override
Return true if the functor implements the end interface.
Definition: midifunctor.h:161
vrv::AlignMeterParams
Regroup pointers to meterSig, mensur and proport objects.
Definition: alignfunctor.h:21
vrv::GenerateTimemapFunctor::ImplementsEndInterface
bool ImplementsEndInterface() const override
Return true if the functor implements the end interface.
Definition: midifunctor.h:486
vrv::Chord
This class represents a collection of notes in the same layer with the same onset time.
Definition: chord.h:32
vrv::Object
This class represents a basic object.
Definition: object.h:59
vrv::Timemap
This class holds a timemap for exporting onset / offset values.
Definition: timemap.h:52
vrv::MIDINote
Helper struct to store note sequences which replace notes in MIDI output due to expanded ornaments an...
Definition: midifunctor.h:328
vrv::InitOnsetOffsetFunctor::ImplementsEndInterface
bool ImplementsEndInterface() const override
Return true if the functor implements the end interface.
Definition: midifunctor.h:46
vrv::Tie
This class models the MEI <tie> element.
Definition: tie.h:27
vrv::Fraction
Definition: fraction.h:19
vrv::OctaveInfo
Helper struct to store octave data.
Definition: midifunctor.h:264
vrv::TabGrp
This class models the MEI <tabGrp> element.
Definition: tabgrp.h:23
vrv::InitTimemapAdjustNotesFunctor
This class adjusts note duration for grace notes and arpeggios.
Definition: midifunctor.h:186
vrv::FeatureExtractor
Definition: featureextractor.h:21
vrv::Functor
This abstract class is the base class for all mutable functors.
Definition: functor.h:101
vrv::InitTimemapAdjustNotesFunctor::ImplementsEndInterface
bool ImplementsEndInterface() const override
Return true if the functor implements the end interface.
Definition: midifunctor.h:199
vrv::GenerateMIDIFunctor::ImplementsEndInterface
bool ImplementsEndInterface() const override
Return true if the functor implements the end interface.
Definition: midifunctor.h:359
vrv::Octave
This class models the MEI <octave> element.
Definition: octave.h:25
vrv::ConstFunctor
This abstract class is the base class for all const functors.
Definition: functor.h:126
vrv::InitMIDIFunctor
This class initializes the MIDI export.
Definition: midifunctor.h:277
vrv::InitOnsetOffsetFunctor
This class prepares Note onsets.
Definition: midifunctor.h:33
vrv::GenerateFeaturesFunctor
This class exports the object to a JSON feature file.
Definition: midifunctor.h:533
vrv::GenerateTimemapFunctor
This class exports the object to a JSON timemap file.
Definition: midifunctor.h:473
vrv::InitMIDIFunctor::ImplementsEndInterface
bool ImplementsEndInterface() const override
Return true if the functor implements the end interface.
Definition: midifunctor.h:290
vrv::InitMaxMeasureDurationFunctor
This class calculates the maximum duration of each measure.
Definition: midifunctor.h:86
vrv::InitTimemapTiesFunctor
This class adjusts note timings based on ties.
Definition: midifunctor.h:148
vrv::LayerElement
This class is a base class for the Layer (<layer>) content.
Definition: layerelement.h:46
vrv::Layer
This class represents a layer in a laid-out score (Doc).
Definition: layer.h:33
vrv::DocFunctor
This abstract class is the base class for all mutable functors that need access to the document.
Definition: functor.h:151
vrv::GenerateMIDIFunctor
This class performs the export to a MidiFile.
Definition: midifunctor.h:346