Verovio
Source code documentation
iopae.h
1 // Name: musiopae.h
3 // Author: Rodolfo Zitellini
4 // Created: 2012
5 // Copyright (c) Authors and others. All rights reserved.
7 
8 #ifndef __VRV_IOPAE_H__
9 #define __VRV_IOPAE_H__
10 
11 /*
12  * There are two implementation of the Plaine & Easie parser.
13  * The new one was introduced in Verovio 3.7.
14  * In order to build with the old parser, the following define needs to be uncommented
15  */
16 //----------------------------------------------------------------------------
17 //----------------------------------------------------------------------------
18 // #define USE_PAE_OLD_PARSER
19 //----------------------------------------------------------------------------
20 //----------------------------------------------------------------------------
21 
22 #include <sstream>
23 #include <string>
24 #include <vector>
25 
26 //----------------------------------------------------------------------------
27 
28 #include "atts_cmn.h"
29 #include "clef.h"
30 #include "iobase.h"
31 #include "keysig.h"
32 #include "mensur.h"
33 #include "metersig.h"
34 #include "vrvdef.h"
35 
36 //----------------------------------------------------------------------------
37 
38 #include "jsonxx.h"
39 
40 namespace vrv {
41 
42 class Beam;
43 class Chord;
44 class DurationInterface;
45 class GraceGrp;
46 class KeyAccid;
47 class Layer;
48 class LayerElement;
49 class Mdiv;
50 class Measure;
51 class MRest;
52 class MultiRest;
53 class Note;
54 class Rest;
55 class ScoreDef;
56 class Space;
57 class Staff;
58 class StaffDef;
59 class Tie;
60 class Tuplet;
61 class KeySig;
62 class BarLine;
63 
64 //----------------------------------------------------------------------------
65 // PAEOutput
66 //----------------------------------------------------------------------------
67 
71 class PAEOutput : public Output {
72 public:
74  PAEOutput(Doc *doc);
76  virtual ~PAEOutput();
78 
82  bool Export(std::string &output);
83 
87  bool WriteObject(Object *object) override;
88 
92  bool WriteObjectEnd(Object *object) override;
93 
97  static std::string GetPaeDur(data_DURATION dur, int ndots);
98 
99 private:
103  void WriteMdiv(Mdiv *mDiv);
105  void WriteMdivEnd(Mdiv *mDiv);
106  void WriteScoreDef(ScoreDef *scoreDef);
107  void WriteStaffDef(StaffDef *staffDef);
108  void WriteMeasure(Measure *measure);
109  void WriteMeasureEnd(Measure *measure);
110  void WriteStaff(Staff *staff);
111  void WriteLayer(Layer *layer);
113 
118  void WriteBarLine(BarLine *barLine);
120  void WriteBeam(Beam *beam);
121  void WriteBeamEnd(Beam *beam);
122  void WriteChord(Chord *chord);
123  void WriteClef(Clef *clef);
124  void WriteGraceGrp(GraceGrp *graceGrp);
125  void WriteKeyAccid(KeyAccid *keyAccid);
126  void WriteKeySig(KeySig *keySig);
127  void WriteMensur(Mensur *mensur);
128  void WriteMeterSig(MeterSig *meterSig);
129  void WriteMRest(MRest *mRest);
130  void WriteMultiRest(MultiRest *multiRest);
131  void WriteNote(Note *note);
132  void WriteRest(Rest *rest);
133  void WriteSpace(Space *space);
134  void WriteTuplet(Tuplet *tuplet);
135  void WriteTupletEnd(Tuplet *tuplet);
137 
141 
147 
153 
159 
166 
172  void WriteDur(DurationInterface *interface);
174  void WriteGrace(AttGraced *attGraced);
175  bool HasFermata(Object *object);
177 
178 public:
179  //
180 private:
181  std::ostringstream m_streamStringOutput;
182  bool m_docScoreDef; // Indicates that we are writing the document scoreDef
183  bool m_mensural; // Indicates that the incipit is mensural (initial staffDef)
184  bool m_skip; // Processing a staff or a layer to skip
185  int m_layerN; // The \@n of the first layer within the first staff
186  int m_staffN; // The \@n of the first staff (initial staffDef)
187  int m_currentOct; // The current octave
188  int m_currentDur; // The current duration
189  int m_currentDots;
190  bool m_grace;
191  Measure *m_currentMeasure;
192 };
193 
194 //----------------------------------------------------------------------------
195 //----------------------------------------------------------------------------
196 #ifdef USE_PAE_OLD_PARSER
197 //----------------------------------------------------------------------------
198 //----------------------------------------------------------------------------
199 
200 //----------------------------------------------------------------------------
201 // namespace for local Plaine & Easie classes
202 //----------------------------------------------------------------------------
203 
204 namespace pae {
205 
206  // static char keySigFlats[7];
207 
208  class Note {
209 
210  public:
211  Note(const pae::Note &old)
212  { // for STL vector
213  // mnote = old.mnote;
214  // mrest = old.mrest;
215  tie = old.tie;
216  acciaccatura = old.acciaccatura;
217  appoggiatura = old.appoggiatura;
218  fermata = old.fermata;
219  trill = old.trill;
220  chord = old.chord;
221 
222  octave = old.octave;
223  beam = old.beam;
224  pitch = old.pitch;
225  duration = old.duration;
226  accidental = old.accidental;
227  accidGes = old.accidGes;
228  dots = old.dots;
229  rest = old.rest;
230 
231  clef = old.clef;
232  mensur = old.mensur;
233  meter = old.meter;
234  key = old.key;
235 
236  tuplet_notes = old.tuplet_notes;
237  tuplet_note = old.tuplet_note;
238  tuplet_val = old.tuplet_val;
239  }
240  Note() { clear(); }
241  void clear()
242  {
243  appoggiatura = 0;
244  acciaccatura = fermata = tie = trill = chord = false;
245 
246  octave = 4;
247  beam = 0;
248  pitch = PITCHNAME_NONE;
249  duration = DURATION_NONE;
250  accidental = ACCIDENTAL_WRITTEN_NONE;
251  accidGes = false;
252  dots = 0;
253  rest = false;
254 
255  tuplet_notes = 0;
256  tuplet_note = 0;
257  tuplet_val = 0;
258 
259  clef = NULL;
260  mensur = NULL;
261  meter = NULL;
262  key = NULL;
263  }
264 
265  Note &operator=(const Note &d)
266  { // for STL vector
267  // mnote = d.mnote;
268  // mrest = d.mrest;
269  tie = d.tie;
270  acciaccatura = d.acciaccatura;
271  appoggiatura = d.appoggiatura;
272  fermata = d.fermata;
273  trill = d.trill;
274  chord = d.chord;
275 
276  octave = d.octave;
277  beam = d.beam;
278  pitch = d.pitch;
279  duration = d.duration;
280  accidental = d.accidental;
281  accidGes = d.accidGes;
282  dots = d.dots;
283  rest = d.rest;
284 
285  clef = d.clef;
286  mensur = d.mensur;
287  meter = d.meter;
288  key = d.key;
289 
290  tuplet_notes = d.tuplet_notes;
291  tuplet_note = d.tuplet_note;
292  tuplet_val = d.tuplet_val;
293 
294  return *this;
295  }
296 
297  // Note *mnote;
298  // Rest *mrest; // this is not too nice
299 
300  // tuplet stuff
301  int tuplet_notes; // quantity of notes in the tuplet
302  int tuplet_note; // indicates this note is the nth in the tuplet
303  int tuplet_val; // indicates the num in the tuplet (value after ;)
304 
305  bool acciaccatura;
306  int appoggiatura;
307  bool chord;
308  bool fermata;
309  bool tie;
310  bool trill;
311 
312  char octave;
313  unsigned char beam;
314  data_PITCHNAME pitch;
315  data_DURATION duration;
316  data_ACCIDENTAL_WRITTEN accidental;
317  bool accidGes;
318  unsigned int dots;
319  bool rest;
320 
321  Clef *clef;
322  Mensur *mensur;
323  MeterSig *meter;
324  KeySig *key;
325  };
326 
327  class Measure {
328 
329  public:
330  Measure(const Measure &d)
331  { // for STL vector
332  clef = d.clef;
333  meter = d.meter;
334  mensur = d.mensur;
335  notes = d.notes;
336 
337  key = d.key;
338 
339  durations = d.durations;
340  dots = d.dots;
341  durations_offset = d.durations_offset;
342  barLine = d.barLine;
343  abbreviation_offset = d.abbreviation_offset;
344  wholerest = d.wholerest;
345  }
346  Measure() { clear(); }
347 
348  Measure &operator=(const Measure &d)
349  { // for STL vector
350  clef = d.clef;
351  meter = d.meter;
352  mensur = d.mensur;
353  notes = d.notes;
354 
355  key = d.key;
356 
357  durations = d.durations;
358  dots = d.dots;
359  durations_offset = d.durations_offset;
360  barLine = d.barLine;
361  abbreviation_offset = d.abbreviation_offset;
362  wholerest = d.wholerest;
363  return *this;
364  }
365 
366  void clear()
367  {
368  durations.clear();
369  dots.clear();
370  notes.clear();
371  durations_offset = DURATION_long;
372  reset();
373  }
374  void reset()
375  {
376  clef = NULL;
377  meter = NULL;
378  mensur = NULL;
379  key = NULL;
380  notes.clear();
381  barLine = BARRENDITION_invis;
382  wholerest = 0;
383  abbreviation_offset = -1;
384  }
385  Clef *clef;
386  Mensur *mensur;
387  MeterSig *meter;
388  KeySig *key;
389 
390  std::vector<pae::Note> notes;
391 
392  std::vector<data_DURATION> durations;
393  std::vector<int> dots; // use the same offset as durations, they are used in parallel
394  char durations_offset;
395  data_BARRENDITION barLine;
396  int abbreviation_offset;
397  int wholerest; // number of whole rests to process
398  };
399 
400 } // namespace pae
401 
402 //----------------------------------------------------------------------------
403 // PAEInput
404 //----------------------------------------------------------------------------
405 
406 class PAEInput : public Input {
407 public:
408  // constructors and destructors
409  PAEInput(Doc *doc);
410  virtual ~PAEInput();
411 
412  // dummy validation log
413  jsonxx::Object GetValidationLog();
414 
415  // dummy setter
416  void SetScoreBased(bool scoreBased) {}
417 
418 #ifndef NO_PAE_SUPPORT
419  bool Import(const std::string &pae) override;
420 
421 private:
422  // function declarations:
423 
424  void parsePlainAndEasy(std::istream &infile);
425 
426  // parsing functions
427  int getKeyInfo(const char *incipit, KeySig *key, int index = 0);
428  int getTimeInfo(const char *incipit, MeterSig *meter, Mensur *mensur, int index = 0);
429  int getClefInfo(const char *incipit, Clef *mus_clef, int index = 0);
430  int getBarLine(const char *incipit, data_BARRENDITION *output, int index);
431  int getAccidental(const char *incipit, data_ACCIDENTAL_WRITTEN *accident, int index = 0);
432  int getOctave(const char *incipit, char *octave, int index = 0);
433  int getDurations(const char *incipit, pae::Measure *measure, int index = 0);
434  int getDuration(const char *incipit, data_DURATION *duration, int *dot, int index);
435  int getTupletFermata(const char *incipit, pae::Note *note, int index = 0);
436  int getTupletFermataEnd(const char *incipit, pae::Note *note, int index = 0);
437  int getGraceNote(const char *incipit, pae::Note *note, int index = 0);
438  int getWholeRest(const char *incipit, int *wholerest, int index);
439  int getAbbreviation(const char *incipit, pae::Measure *measure, int index = 0);
440  int getNote(const char *incipit, pae::Note *note, pae::Measure *measure, int index = 0);
441 
442  data_PITCHNAME getPitch(char c_note);
443 
444  // output functions
445  void addLayerElement(LayerElement *element);
446  void parseNote(pae::Note *note);
447  void popContainer();
448  void convertMeasure(pae::Measure *measure);
449  void pushContainer(LayerElement *container);
450 
451  // input functions
452  void getAtRecordKeyValue(char *key, char *value, const char *input);
453 
454 #endif // NO_PAE_SUPPORT
455 
456 public:
457  //
458 private:
459  Staff *m_staff;
460  Measure *m_measure;
461  Layer *m_layer;
462  Tie *m_tie;
463  bool m_is_in_chord;
464  bool m_is_mensural;
465 
466  MapOfOctavedPitchAccid m_currentAccids;
467  KeySig *m_currentKeySig;
468  std::pair<int, data_ACCIDENTAL_WRITTEN> m_tieAccid;
469 
470  std::vector<LayerElement *> m_nested_objects;
471 };
472 
473 //----------------------------------------------------------------------------
474 //----------------------------------------------------------------------------
475 #else // USE_PAE_OLD_PARSER
476 //----------------------------------------------------------------------------
477 //----------------------------------------------------------------------------
478 
479 //----------------------------------------------------------------------------
480 // PAEInput
481 //----------------------------------------------------------------------------
482 
483 namespace pae {
484 
485  class Token {
486  public:
487  Token(char c, int position, Object *object = NULL);
488  virtual ~Token();
489 
491  bool Is(ClassId);
493  bool IsContainerEnd();
495  bool IsEnd();
497  bool IsSpace();
499  bool IsVoid();
501  void SetInTree();
502 
503  /* Helper to the a lowercase version of the Object classname (if any) */
504  std::string GetName();
505 
507  char m_char;
509  Object *m_object;
511  Object *m_treeObject;
513  char m_inputChar;
515  int m_position;
517  bool m_isError;
518  };
519 
520 } // namespace pae
521 
522 class PAEInput : public Input {
523 public:
524  // constructors and destructors
525  PAEInput(Doc *doc);
526  virtual ~PAEInput();
527 
533  jsonxx::Object GetValidationLog();
534 
538  void SetScoreBased(bool scoreBased) { m_scoreBased = scoreBased; }
539 
540 #ifndef NO_PAE_SUPPORT
541  bool Import(const std::string &input) override;
542 
543 private:
547  jsonxx::Object InputKeysToJson(const std::string &inputKeys);
548 
552  jsonxx::Object SingleLineToJson(const std::string &singleLine);
553 
560  void AddToken(char c, int &position);
561 
569  bool Parse();
570 
574  void ParseHeader(jsonxx::Object &header);
575 
583  bool ConvertKeySig();
585  bool ConvertClef();
586  bool ConvertMeterSigOrMensur();
587  bool ConvertMeasure();
588  bool ConvertRepeatedFigure();
589  bool ConvertRepeatedMeasure();
590  bool ConvertMRestOrMultiRest();
591  bool ConvertPitch();
592  bool ConvertOctave();
593  bool ConvertTrill();
594  bool ConvertFermata();
595  bool ConvertAccidental();
596  bool ConvertRest();
597  bool ConvertChord();
598  bool ConvertBeam();
599  bool ConvertGrace();
600  bool ConvertGraceGrp();
601  bool ConvertTuplet();
602  bool ConvertDuration();
603  bool ConvertTie();
604  bool ConvertLigature();
605  bool ConvertAccidGes();
607 
612  bool Is(pae::Token &token, const std::string &map);
614  bool Was(pae::Token &token, const std::string &map);
615  bool HasInput(char inputChar);
617 
621  bool ParseKeySig(KeySig *keySig, const std::string &paeStr, pae::Token &token);
623  bool ParseClef(Clef *clef, const std::string &paeStr, pae::Token &token, bool *mensuralScoreDef = NULL);
624  bool ParseMeterSig(MeterSig *meterSig, const std::string &paeStr, pae::Token &token);
625  bool ParseMensur(Mensur *mensur, const std::string &paeStr, pae::Token &token);
626  bool ParseMeasure(Measure *measure, const std::string &paeStr, pae::Token &token);
627  bool ParseDuration(
628  std::list<std::pair<data_DURATION, int>> &durations, const std::string &paeStr, pae::Token &token);
630 
636  bool CheckPAEChars(const std::string &input, std::string &invalidChars, const std::string &validChars = "");
639 
645  void PrepareInsertion(int position, std::list<pae::Token> &insertion);
646 
652  bool CheckHierarchy();
653 
658  bool CheckContentPreBuild();
659 
663  bool CheckContentPostBuild();
664 
668  void RemoveContainerToken(Object *object);
669 
674  pae::Token *GetTokenForTreeObject(Object *object);
675 
679  void LogPAE(int errCode, pae::Token &token, std::string value = "");
681  void LogDebugTokens(bool vertical = false);
683 
684 #endif // NO_PAE_SUPPORT
685 
690  void ClearTokenObjects();
691 
692 public:
693  static const std::map<int, std::string> s_errCodes;
694 
695 private:
708  std::list<pae::Token> m_pae;
709 
714  bool m_isMensural;
715 
721  bool m_pedanticMode;
722 
726  bool m_hasErrors;
727 
731  bool m_scoreBased;
732 
737  Clef m_clef;
739  KeySig m_keySig;
740  Mensur m_mensur;
741  MeterSig m_meterSig;
742  bool m_hasClef;
743  bool m_hasKeySig;
744  bool m_hasMeterSig;
745  bool m_hasMensur;
747 
753  jsonxx::Object m_clefLog;
755  jsonxx::Object m_keysigLog;
756  jsonxx::Object m_timesigLog;
757  jsonxx::Object m_inputLog;
758  jsonxx::Array m_dataLog;
760 };
761 
762 //----------------------------------------------------------------------------
763 //----------------------------------------------------------------------------
764 #endif // USE_PAE_OLD_PARSER
765 //----------------------------------------------------------------------------
766 //----------------------------------------------------------------------------
767 
768 } // namespace vrv
769 
770 #endif
vrv::StaffDef
This class represents a MEI staffDef.
Definition: staffdef.h:26
vrv::PAEInput::GetValidationLog
jsonxx::Object GetValidationLog()
Return a JSON object with the validation log It is a single object when an input error is encountered...
Definition: iopae.cpp:2508
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::PAEOutput
This class is a file output stream for writing PAE files.
Definition: iopae.h:71
vrv::Doc
This class is a hold the data and corresponds to the model of a MVC design pattern.
Definition: doc.h:41
vrv::PAEInput
Definition: iopae.h:522
vrv::MeterSig
This class models the MEI <meterSig> element.
Definition: metersig.h:27
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::Output
This class is a base class for output classes.
Definition: iobase.h:31
vrv::Mensur
This class models the MEI <mensur> element.
Definition: mensur.h:27
vrv::BarLine
This class models the MEI <barLine> element.
Definition: barline.h:29
vrv::Space
This class models the MEI <space> element.
Definition: space.h:23
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::PAEOutput::Export
bool Export(std::string &output)
The main method for exporting to PAE.
Definition: iopae.cpp:64
vrv::Beam
Definition: beam.h:279
vrv::Note
This class models the MEI <note> element.
Definition: note.h:47
vrv::Mdiv
This class represent a <mdiv> in page-based MEI.
Definition: mdiv.h:24
vrv::PAEOutput::WriteObject
bool WriteObject(Object *object) override
The main method for write objects.
Definition: iopae.cpp:87
vrv::PAEInput::SetScoreBased
void SetScoreBased(bool scoreBased)
Setter for scoreBased flag (false by default).
Definition: iopae.h:538
vrv::KeyAccid
This class models the MEI <keyAccid> element.
Definition: keyaccid.h:26
vrv::MultiRest
This class models the MEI <multiRest> element.
Definition: multirest.h:26
vrv::GraceGrp
Definition: gracegrp.h:20
vrv::ScoreDef
This class represents a MEI scoreDef.
Definition: scoredef.h:129
vrv::PAEOutput::WriteObjectEnd
bool WriteObjectEnd(Object *object) override
Writing object method that must be overridden in the child class.
Definition: iopae.cpp:163
vrv::Tuplet
Definition: tuplet.h:28
vrv::MRest
This class models the MEI <mRest> element.
Definition: mrest.h:27
vrv::Rest
This class models the MEI <rest> element.
Definition: rest.h:37
vrv::DurationInterface
This class is an interface for elements with duration, such as notes and rests.
Definition: durationinterface.h:31
vrv::PAEOutput::GetPaeDur
static std::string GetPaeDur(data_DURATION dur, int ndots)
Helper method to return a string representation of the PAE duration.
Definition: iopae.cpp:557
vrv::KeySig
This class models the MEI <keySig> element.
Definition: keysig.h:44
vrv::Layer
This class represents a layer in a laid-out score (Doc).
Definition: layer.h:33