Verovio
Source code documentation
iomusxml.h
1 // Name: iomusxml.h
3 // Author: Laurent Pugin and Klaus Rettinghaus
4 // Created: 22/09/2015
5 // Copyright (c) Authors and others. All rights reserved.
7 
8 #ifndef __VRV_IOMUSXML_H__
9 #define __VRV_IOMUSXML_H__
10 
11 #include <map>
12 #include <optional>
13 #include <queue>
14 #include <string>
15 #include <vector>
16 
17 //----------------------------------------------------------------------------
18 
19 #include "attdef.h"
20 #include "iobase.h"
21 #include "metersig.h"
22 #include "vrvdef.h"
23 
24 //----------------------------------------------------------------------------
25 
26 #include "pugixml.hpp"
27 
28 namespace vrv {
29 
30 class Arpeg;
31 class BeamSpan;
32 class BracketSpan;
33 class Clef;
34 class ControlElement;
35 class Dir;
36 class Dynam;
37 class F;
38 class Fermata;
39 class Fb;
40 class Gliss;
41 class Hairpin;
42 class Harm;
43 class InstrDef;
44 class KeySig;
45 class Label;
46 class LabelAbbr;
47 class Layer;
48 class LayerElement;
49 class Measure;
50 class MeterSigGrp;
51 class Octave;
52 class Pedal;
53 class Section;
54 class Slur;
55 class StaffGrp;
56 class Tempo;
57 class Tie;
58 class Trill;
59 
60 //----------------------------------------------------------------------------
61 // namespace for local MusicXML classes
62 //----------------------------------------------------------------------------
63 
64 #ifndef NO_MUSICXML_SUPPORT
65 
66 namespace musicxml {
67 
68  struct OpenSlur {
69  OpenSlur(const std::string &measureNum, short int number, curvature_CURVEDIR curvedir)
70  {
71  m_measureNum = measureNum;
72  m_number = number;
73  m_curvedir = curvedir;
74  }
75 
76  std::string m_measureNum;
77  short int m_number;
78  curvature_CURVEDIR m_curvedir;
79  };
80 
81  struct CloseSlur {
82  CloseSlur(const std::string &measureNum, short int number, curvature_CURVEDIR curvedir)
83  {
84  m_measureNum = measureNum;
85  m_number = number;
86  m_curvedir = curvedir;
87  }
88 
89  std::string m_measureNum;
90  short int m_number;
91  curvature_CURVEDIR m_curvedir;
92  };
93 
94  struct OpenSpanner {
95  OpenSpanner(const int &dirN, const int &lastMeasureCount)
96  {
97  m_dirN = dirN;
98  m_lastMeasureCount = lastMeasureCount;
99  }
100 
101  int m_dirN;
102  int m_lastMeasureCount;
103  };
104 
105  struct OpenArpeggio {
106  OpenArpeggio(const int &arpegN, const Fraction &timeStamp)
107  {
108  m_arpegN = arpegN;
109  m_timeStamp = timeStamp;
110  }
111 
112  int m_arpegN;
113  Fraction m_timeStamp;
114  };
115 
116  struct EndingInfo {
117  EndingInfo(const std::string &endingNumber, const std::string &endingType, const std::string &endingText)
118  {
119  m_endingNumber = endingNumber;
120  m_endingType = endingType;
121  m_endingText = endingText;
122  }
123 
124  std::string m_endingNumber;
125  std::string m_endingType;
126  std::string m_endingText;
127  };
128 
129  struct ClefChange {
130  ClefChange(const std::string &measureNum, Staff *staff, Layer *layer, Clef *clef, const int &scoreOnset,
131  bool afterBarline)
132  {
133  m_measureNum = measureNum;
134  m_staff = staff;
135  m_layer = layer;
136  m_clef = clef;
137  m_scoreOnset = scoreOnset;
138  m_afterBarline = afterBarline;
139  }
140 
141  std::string m_measureNum;
142  Staff *m_staff;
143  Layer *m_layer;
144  Clef *m_clef;
145  int m_scoreOnset; // the score position of clef change
146  bool m_afterBarline = false; // musicXML attribute
147  };
148 
149  struct OpenDashes {
150  OpenDashes(const int dirN, int staffNum, const int measureCount)
151  {
152  m_dirN = dirN;
153  m_staffNum = staffNum;
154  m_measureCount = measureCount;
155  }
156 
157  int m_dirN; // direction number
158  int m_staffNum;
159  int m_measureCount; // measure number of dashes start
160  };
161 
162  struct OpenTie {
163  OpenTie(Tie *tie, Note *note, int layerNum)
164  {
165  m_tie = tie;
166  m_note = note;
167  m_layerNum = layerNum;
168  }
169 
170  Tie *m_tie = NULL;
171  Note *m_note = NULL;
172  int m_layerNum = 0;
173  };
174 
175  struct CloseTie {
176  CloseTie(Note *note, int layerNum)
177  {
178  m_note = note;
179  m_layerNum = layerNum;
180  }
181 
182  Note *m_note = NULL;
183  int m_layerNum = 0;
184  };
185 
186 } // namespace musicxml
187 
188 #endif // NO_MUSICXML_SUPPORT
189 
190 //----------------------------------------------------------------------------
191 // MusicXmlInput
192 //----------------------------------------------------------------------------
193 
194 class MusicXmlInput : public Input {
195 public:
196  // constructors and destructors
197  MusicXmlInput(Doc *doc);
198  virtual ~MusicXmlInput();
199 
200 private:
201  /*
202  * Objects that were not successfully added and should be destroyed at the end of the import
203  */
204  ListOfObjects m_garbage;
205 
206 #ifndef NO_MUSICXML_SUPPORT
207 public:
208  bool Import(const std::string &musicxml) override;
209 
210 private:
211  /*
212  * Top level method called from ImportFile or Import
213  */
214  bool ReadMusicXml(pugi::xml_node root);
215 
216  /*
217  * Method to fill MEI header with title
218  */
219  void ReadMusicXmlTitle(pugi::xml_node title);
220 
221  /*
222  * @name Top level methods for reading MusicXML part and measure elements.
223  */
225  bool ReadMusicXmlPart(pugi::xml_node node, Section *section, short int nbStaves, const short int staffOffset);
226  bool ReadMusicXmlMeasure(pugi::xml_node node, Section *section, Measure *measure, short int nbStaves,
227  const short int staffOffset, int index);
229 
230  /*
231  * Methods for reading the first MusicXML attributes element as MEI staffDef.
232  * Returns the number of staves in the part.
233  */
234  short int ReadMusicXmlPartAttributesAsStaffDef(
235  pugi::xml_node node, StaffGrp *staffGrp, const short int staffOffset);
236 
237  /*
238  * @name Methods for reading the content of a MusicXML measure.
239  */
241  void ReadMusicXmlAttributes(pugi::xml_node, Section *section, Measure *measure, const std::string &measureNum);
242  void ReadMusicXmlBackup(pugi::xml_node, Measure *measure, const std::string &measureNum);
243  void ReadMusicXmlBarLine(pugi::xml_node, Measure *measure, const std::string &measureNum);
244  void ReadMusicXmlDirection(
245  pugi::xml_node, Measure *measure, const std::string &measureNum, const short int staffOffset);
246  void ReadMusicXmlFigures(pugi::xml_node, Measure *measure, const std::string &measureNum);
247  void ReadMusicXmlForward(pugi::xml_node, Measure *measure, const std::string &measureNum);
248  void ReadMusicXmlHarmony(pugi::xml_node, Measure *measure, const std::string &measureNum);
249  void ReadMusicXmlNote(
250  pugi::xml_node, Measure *measure, const std::string &measureNum, const short int staffOffset, Section *section);
251  void ReadMusicXmlPrint(pugi::xml_node, Section *section);
252  bool ReadMusicXmlBeamsAndTuplets(const pugi::xml_node &node, Layer *layer, bool isChord);
253  void ReadMusicXmlTupletStart(const pugi::xml_node &node, const pugi::xml_node &tupletStart, Layer *layer);
254  void ReadMusicXmlBeamStart(const pugi::xml_node &node, const pugi::xml_node &beamStart, Layer *layer);
255  void ReadMusicXMLMeterSig(const pugi::xml_node &node, Object *parent);
256  void ReadMusicXmlTies(const pugi::xml_node &node, Layer *layer, Note *note, const std::string &measureNum);
258 
262  void ProcessClefChangeQueue(Section *section);
263 
267  void AddClefs(Measure *measure, const musicxml::ClefChange &clefChange);
268 
272  void InsertClefToLayer(Staff *staff, Layer *layer, Clef *clef, int scoreOnSet);
273 
274  /*
275  * @name Helper function to insert clef into correct position in layer/other parent based on the insertAfter
276  * variable.
277  */
279  void InsertClefIntoObject(Object *layerElement, Clef *clef, Layer *layer, int scoreOnset, bool insertAfter);
280  void InsertClefIntoObject(Object *parent, Clef *clef, Object *relevantChild, bool insertAfter);
282 
283  /*
284  * Add a Measure to the section.
285  * If the measure already exists it will move all its content.
286  * The measure can contain only staves. Other elements must be stacked on m_floatingElements.
287  */
288  void AddMeasure(Section *section, Measure *measure, int i);
289 
290  /*
291  * Add a Layer element to the layer or to the LayerElement at the top of m_elementStack.
292  */
293  void AddLayerElement(Layer *layer, LayerElement *element, int duration = 0);
294 
295  /*
296  * Returns the appropriate layer for a node looking at its MusicXML staff and voice elements.
297  */
298  Layer *SelectLayer(pugi::xml_node node, Measure *measure);
299 
300  /*
301  * Returns the appropriate first layer of a staff.
302  */
303  Layer *SelectLayer(short int staffNb, Measure *measure);
304 
305  /*
306  * Returns the layer with \@n=layerNb on the staff.
307  * Creates the layer if not found.
308  */
309  Layer *SelectLayer(short int layerNb, Staff *staff);
310 
311  /*
312  * @name Methods for converting the content of MusicXML attributes.
313  */
315  Clef *ConvertClef(const pugi::xml_node &clef);
316  KeySig *ConvertKey(const pugi::xml_node &key);
318 
319  /*
320  * Remove the last ClassId element on top of m_elementStack.
321  * For example, when closing a beam, we need to remove it from the stack, but it is not
322  * necessary the top one (for example we can have an opened chord there).
323  */
324  void RemoveLastFromStack(ClassId classId, Layer *layer);
325 
329  bool IsInStack(ClassId classId, Layer *layer);
330 
331  /*
332  * @name Helper methods for checking presence of values of attributes or elements
333  */
335  bool HasAttributeWithValue(const pugi::xml_node node, const std::string &attribute, const std::string &value) const;
336  bool IsElement(const pugi::xml_node node, const std::string &name) const;
337  bool HasContentWithValue(const pugi::xml_node node, const std::string &value) const;
339 
340  /*
341  * @name Helper method to check whether an ending measure is already present in m_endingStack.
342  */
344  bool NotInEndingStack(const Measure *measure) const;
346 
347  /*
348  * @name Helper methods for retrieving attribute values or element content
349  */
351  std::string GetContent(const pugi::xml_node node) const;
352  std::string GetContentOfChild(const pugi::xml_node node, const std::string &child) const;
354 
355  /*
356  * @name Methods for opening and closing ties and slurs.
357  * Opened ties and slurs are stacked together with musicxml::OpenTie
358  * and musicxml::OpenSlur objects.
359  * Slur starts and ends are matched based on its number.
360  */
362  void OpenTie(Note *note, Tie *tie, int layerNum);
363  void CloseTie(Note *note, int layerNum);
364  void OpenSlur(Measure *measure, short int number, Slur *slur, curvature_CURVEDIR dir);
365  void CloseSlur(Measure *measure, short int number, LayerElement *element, curvature_CURVEDIR dir);
366  void CloseBeamSpan(Staff *staff, Layer *layer, LayerElement *element);
367  void MatchTies(bool matchLayers);
369 
370  /*
371  * @name Helper methods for rendering text elements
372  */
375  std::string GetWordsOrDynamicsText(const pugi::xml_node node) const;
376  void TextRendition(const pugi::xpath_node_set words, ControlElement *element) const;
377  std::string StyleLabel(pugi::xml_node display);
378  void PrintMetronome(pugi::xml_node metronome, Tempo *tempo);
379 
380  /*
381  * @name Helper methods for filling in space elements
382  */
385  void FillSpace(Layer *layer, int dur);
386 
387  /*
388  * @name Helper method for generating additional IDs
389  */
392  void GenerateID(pugi::xml_node node);
393 
394  /*
395  * @name Helper method for meterSigGrp. Separates beat/beat-type into MeterSig and adds them to the MeterSigGrp.
396  * Returns total meterCount and meterUnit for the group
397  */
399  std::pair<std::vector<int>, int> GetMeterSigGrpValues(const pugi::xml_node &node, MeterSigGrp *parent);
401 
402  /*
403  * @name Helper method for multirests. Returns number of measure hidden by MRest before
404  * measure with certain index
405  */
407  int GetMrestMeasuresCountBeforeIndex(int index) const;
409 
410  /*
411  * @name Helper method for multirests. Checks whether measure should be hidden as part of MRest
412  */
414  bool IsMultirestMeasure(int index) const;
416 
417  /*
418  * @name Helper method for styling fermatas
419  */
421  void SetFermataExternalSymbols(Fermata *fermata, const std::string &shape);
422  void ShapeFermata(Fermata *fermata, pugi::xml_node node);
424 
425  /*
426  * @name Helper method for getting glyph numbers for ornaments based on approach/depart attributes
427  */
429  std::string GetOrnamentGlyphNumber(int attributes) const;
431 
432  /*
433  * @name Helper method for setting \@staff attribute for chords
434  */
436  void SetChordStaff(Layer *layer);
438 
439  /*
440  * @name Helper method for comparing written/gestural accidental attributes
441  */
443  static bool IsSameAccidWrittenGestural(data_ACCIDENTAL_WRITTEN written, data_ACCIDENTAL_GESTURAL gestural);
445 
446  /*
447  * @name Helper for detecting the slur curve direction
448  */
449  static curvature_CURVEDIR CombineCurvedir(curvature_CURVEDIR startDir, curvature_CURVEDIR stopDir);
450 
451  /*
452  * @name Methods for converting MusicXML values to MEI attributes.
453  */
455  static data_ACCIDENTAL_WRITTEN ConvertAccidentalToAccid(const std::string &value);
456  static data_ACCIDENTAL_GESTURAL ConvertAlterToAccid(const float value);
457  static data_ARTICULATION ConvertArticulations(const std::string &value);
458  static data_BARRENDITION ConvertStyleToRend(const std::string &value, const bool repeat);
459  static data_BOOLEAN ConvertWordToBool(const std::string &value);
460  static data_DURATION ConvertTypeToDur(const std::string &value);
461  static data_HEADSHAPE ConvertNotehead(const std::string &value);
462  static data_LINESTARTENDSYMBOL ConvertLineEndSymbol(const std::string &value);
463  static data_MIDIVALUE ConvertDynamicsToMidiVal(const float dynamics);
464  static data_PITCHNAME ConvertStepToPitchName(const std::string &value);
465  static data_TEXTRENDITION ConvertEnclosure(const std::string &value);
466  static beamRend_FORM ConvertBeamFanToForm(const std::string &value);
467  static curvature_CURVEDIR InferCurvedir(const pugi::xml_node slurOrTie);
468  static fermataVis_SHAPE ConvertFermataShape(const std::string &value);
469  static pedalLog_DIR ConvertPedalTypeToDir(const std::string &value);
470  static tupletVis_NUMFORMAT ConvertTupletNumberValue(const std::string &value);
471  static std::u32string ConvertTypeToVerovioText(const std::string &value);
472  static std::string ConvertAlterToSymbol(const std::string &value, bool plusMinus = false);
473  static std::string ConvertKindToSymbol(const std::string &value);
474  static std::string ConvertKindToText(const std::string &value);
475  static std::string ConvertDegreeToText(const pugi::xml_node harmony);
476  static std::string ConvertFigureGlyph(const std::string &value);
478 
479  /*
480  * @name Methods for converting between MusicXML <pitch> and MIDI note numbers.
481  */
483  static int PitchToMidi(const std::string &step, int alter, int octave);
484  static void MidiToPitch(int midi, std::string &step, int &alter, int &octave);
486 
487 public:
488  //
489 private:
490  /* octave offset */
491  std::vector<int> m_octDis;
492  /* measure repeats */
493  bool m_mRpt = false;
494  /* measure repeats */
495  bool m_slash = false;
496  /* MIDI ticks */
497  int m_ppq = -1;
498  /* measure time */
499  int m_durTotal = 0;
500  /* measure time */
501  int m_durFb = 0;
502  /* meter signature */
503  std::vector<int> m_meterCount = { 4 };
504  int m_meterUnit = 4;
505  MeterCountSign m_meterSign = MeterCountSign::None;
506  /* part information */
507  Label *m_label = NULL;
508  LabelAbbr *m_labelAbbr = NULL;
509  InstrDef *m_instrdef = NULL;
510  /* LastElementID */
511  std::string m_ID;
512  /* A map of stacks for piling open LayerElements (beams, tuplets, chords, btrem, ftrem) separately per layer */
513  std::map<Layer *, std::vector<LayerElement *>> m_elementStackMap;
514  /* A maps of time stamps (score time) to indicate write pointer of a given layer */
515  std::map<Layer *, int> m_layerEndTimes;
516  std::map<Layer *, std::multimap<int, LayerElement *>> m_layerTimes;
517  /* To remember layer of last element (note) to handle chords */
518  Layer *m_prevLayer = NULL;
519  /* To remember current layer to properly handle layers/staves/cross-staff elements */
520  Layer *m_currentLayer = NULL;
521  bool m_isLayerInitialized = false;
522  /* The stack for open slurs */
523  std::vector<std::pair<Slur *, musicxml::OpenSlur>> m_slurStack;
524  /* The stack for slur stops that might come before the slur has been opened */
525  std::vector<std::pair<LayerElement *, musicxml::CloseSlur>> m_slurStopStack;
526  /* The stack for open ties */
527  std::vector<musicxml::OpenTie> m_tieStack;
528  /* The stack for tie stops that might come before that tie was opened */
529  std::vector<musicxml::CloseTie> m_tieStopStack;
530  /* The stack for hairpins */
531  std::vector<std::pair<Hairpin *, musicxml::OpenSpanner>> m_hairpinStack;
532  /* The stack for hairpin stops that might occur before a hairpin was started staffNumber, tStamp2, (hairpinNumber,
533  * measureCount) */
534  std::vector<std::tuple<int, double, musicxml::OpenSpanner>> m_hairpinStopStack;
535  /* The stack for the beamspans with numbers of staff/layer where starting element is located*/
536  std::vector<std::pair<BeamSpan *, std::pair<int, int>>> m_beamspanStack;
537  std::vector<std::pair<BracketSpan *, musicxml::OpenSpanner>> m_bracketStack;
538  std::vector<std::pair<Trill *, musicxml::OpenSpanner>> m_trillStack;
539  /* Current ending info for start/stop */
540  std::optional<musicxml::EndingInfo> m_currentEndingStart;
541  std::optional<musicxml::EndingInfo> m_currentEndingStop;
542  /* The stack of endings to be inserted at the end of XML import */
543  std::vector<std::pair<std::vector<Measure *>, musicxml::EndingInfo>> m_endingStack;
544  /* The stack of open dashes (direction-type) containing *ControlElement, OpenDashes */
545  std::vector<std::pair<ControlElement *, musicxml::OpenDashes>> m_openDashesStack;
546  /* The stacks for ControlElements */
547  std::vector<Dir *> m_dirStack;
548  std::vector<Dynam *> m_dynamStack;
549  std::vector<Gliss *> m_glissStack;
550  std::vector<Harm *> m_harmStack;
551  std::vector<Octave *> m_octaveStack;
552  std::vector<Pedal *> m_pedalStack;
553  std::vector<Tempo *> m_tempoStack;
554  /*
555  * The stack of floating elements (tie, slur, etc.) to be added at the
556  * end of each measure
557  */
558  std::vector<std::pair<std::string, ControlElement *>> m_controlElements;
559  /* stack of clef changes to be inserted to all layers of a given staff */
560  std::queue<musicxml::ClefChange> m_clefChangeQueue;
561  /* stack of new arpeggios that get more notes added. */
562  std::vector<std::pair<Arpeg *, musicxml::OpenArpeggio>> m_ArpeggioStack;
563  /* a map for the measure counts storing the index of each measure created */
564  std::map<Measure *, int> m_measureCounts;
565  /* measure rests */
566  std::map<int, int> m_multiRests;
567 
568 #endif // NO_MUSICXML_SUPPORT
569 };
570 
571 } // namespace vrv
572 
573 #endif // __VRV_IOMUSXML_H__
vrv::MusicXmlInput
Definition: iomusxml.h:194
vrv::LabelAbbr
This class models the MEI <labelAbbr> element.
Definition: labelabbr.h:22
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::Doc
This class is a hold the data and corresponds to the model of a MVC design pattern.
Definition: doc.h:41
vrv::Tempo
This class is an interface for <tempo> elements at the measure level.
Definition: tempo.h:27
vrv::Section
This class represents a MEI section.
Definition: section.h:28
vrv::Object
This class represents a basic object.
Definition: object.h:59
vrv::MeterSigGrp
This class represents a MEI meterSigGrp.
Definition: metersiggrp.h:28
vrv::Slur
Definition: slur.h:53
vrv::Tie
This class models the MEI <tie> element.
Definition: tie.h:27
vrv::InstrDef
This class models the MEI <instrDef> element.
Definition: instrdef.h:24
vrv::Clef
This class models the MEI <clef> element.
Definition: clef.h:27
vrv::Input
This class is a base class for input classes.
Definition: iobase.h:63
vrv::Note
This class models the MEI <note> element.
Definition: note.h:47
vrv::Label
This class models the MEI <label> element.
Definition: label.h:22
vrv::ControlElement
This class represents elements appearing within a measure.
Definition: controlelement.h:28
vrv::KeySig
This class models the MEI <keySig> element.
Definition: keysig.h:44
vrv::StaffGrp
This class represents a MEI staffGrp.
Definition: staffgrp.h:31
vrv::Fermata
This class models the MEI <fermata> element.
Definition: fermata.h:26
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