Verovio
Source code documentation
All Classes Namespaces Functions Variables Typedefs Enumerations Modules
comparison.h
1 // Name: comparison.h
3 // Author: Laurent Pugin
4 // Created: 2015
5 // Copyright (c) Authors and others. All rights reserved.
7 
8 #ifndef __VRV_COMPARISON_H__
9 #define __VRV_COMPARISON_H__
10 
11 #include "artic.h"
12 #include "atts_shared.h"
13 #include "durationinterface.h"
14 #include "horizontalaligner.h"
15 #include "measure.h"
16 #include "note.h"
17 #include "object.h"
18 #include "staffdef.h"
19 #include "staffgrp.h"
20 #include "symbol.h"
21 #include "timeinterface.h"
22 
23 namespace vrv {
24 
25 enum DurExtreme { LONGEST = 0, SHORTEST };
26 
27 //----------------------------------------------------------------------------
28 // Comparison
29 //----------------------------------------------------------------------------
30 
31 class Comparison {
32 
33 public:
34  virtual bool operator()(const Object *object) = 0;
35  // For classes that do a reverse comparison, return reversed result
36  bool Result(bool comparison) { return (m_reverse) ? !comparison : comparison; }
37  // Set reverse comparison.
38  // This is possible only for Comparison classes that allow it explicitly
39  void ReverseComparison()
40  {
41  assert(m_supportReverse);
42  m_reverse = true;
43  }
44 
45 protected:
46  // This is set to true in contructor of classes that allow it
47  bool m_supportReverse = false;
48 
49 private:
50  // The flag indicating if a reverse comparison needs to be done
51  bool m_reverse = false;
52 };
53 
54 //----------------------------------------------------------------------------
55 // ClassIdComparison
56 //----------------------------------------------------------------------------
57 
58 class ClassIdComparison : public Comparison {
59 
60 public:
61  ClassIdComparison(ClassId classId)
62  {
63  m_classId = classId;
64  m_supportReverse = true;
65  }
66 
67  bool operator()(const Object *object) override { return Result(this->MatchesType(object)); }
68 
69  ClassId GetType() { return m_classId; }
70 
71  bool MatchesType(const Object *object) { return (object->Is(m_classId)); }
72 
73 protected:
74  ClassId m_classId;
75 };
76 
77 //----------------------------------------------------------------------------
78 // ClassIdsComparison
79 //----------------------------------------------------------------------------
80 
82 
83 public:
84  ClassIdsComparison(const std::vector<ClassId> &classIds)
85  {
86  m_classIds = classIds;
87  m_supportReverse = true;
88  }
89 
90  bool operator()(const Object *object) override { return Result(this->MatchesType(object)); }
91 
92  bool MatchesType(const Object *object) { return (object->Is(m_classIds)); }
93 
94 protected:
95  std::vector<ClassId> m_classIds;
96 };
97 
98 //----------------------------------------------------------------------------
99 // InterfaceComparison
100 //----------------------------------------------------------------------------
101 
103 
104 public:
105  InterfaceComparison(InterfaceId interfaceId) { m_interfaceId = interfaceId; }
106 
107  bool operator()(const Object *object) override
108  {
109  if (object->HasInterface(m_interfaceId)) {
110  return true;
111  }
112  return false;
113  }
114 
115 protected:
116  InterfaceId m_interfaceId;
117 };
118 
119 //----------------------------------------------------------------------------
120 // ChildOfClassIdComparison
121 //----------------------------------------------------------------------------
122 
124 
125 public:
126  ChildOfClassIdComparison(ClassId classId) { m_classId = classId; }
127 
128  bool operator()(const Object *object) override
129  {
130  return (object->GetParent() && object->GetParent()->GetClassId() == m_classId);
131  }
132 
133 protected:
134  ClassId m_classId;
135 };
136 
137 //----------------------------------------------------------------------------
138 // PointingToComparison
139 //----------------------------------------------------------------------------
140 
142 
143 public:
144  PointingToComparison(ClassId classId, const Object *pointingTo) : ClassIdComparison(classId)
145  {
146  m_pointingTo = pointingTo;
147  }
148 
149  bool operator()(const Object *object) override
150  {
151  if (!MatchesType(object)) return false;
152  const TimePointInterface *interface = object->GetTimePointInterface();
153  if (!interface) return false;
154  return (interface->GetStart() == m_pointingTo);
155  }
156 
157 protected:
158  const Object *m_pointingTo;
159 };
160 
161 //----------------------------------------------------------------------------
162 // SpanningToComparison
163 //----------------------------------------------------------------------------
164 
166 
167 public:
168  SpanningToComparison(ClassId classId, const Object *pointingTo) : ClassIdComparison(classId)
169  {
170  m_pointingTo = pointingTo;
171  }
172 
173  bool operator()(const Object *object) override
174  {
175  if (!MatchesType(object)) return false;
176  const TimeSpanningInterface *interface = object->GetTimeSpanningInterface();
177  if (!interface) return false;
178  return (interface->GetEnd() == m_pointingTo);
179  }
180 
181 protected:
182  const Object *m_pointingTo;
183 };
184 
185 //----------------------------------------------------------------------------
186 // IsEditorialElementComparison
187 //----------------------------------------------------------------------------
188 
193 
194 public:
195  IsEditorialElementComparison() : Comparison() { m_supportReverse = true; }
196 
197  bool operator()(const Object *object) override
198  {
199  if (object->IsEditorialElement()) return Result(true);
200  return Result(false);
201  }
202 };
203 
204 //----------------------------------------------------------------------------
205 // IsEmptyComparison
206 //----------------------------------------------------------------------------
207 
212 
213 public:
214  IsEmptyComparison(ClassId classId) : ClassIdComparison(classId) { m_supportReverse = true; }
215 
216  bool operator()(const Object *object) override
217  {
218  if (!MatchesType(object)) return false;
219  if (object->GetChildCount() == 0) {
220  return Result(true);
221  }
222  else {
223  return Result(false);
224  }
225  }
226 };
227 
228 //----------------------------------------------------------------------------
229 // IsAttributeComparison
230 //----------------------------------------------------------------------------
231 
236 
237 public:
238  IsAttributeComparison(ClassId classId) : ClassIdComparison(classId) {}
239 
240  bool operator()(const Object *object) override
241  {
242  if (!MatchesType(object)) return false;
243  if (object->IsAttribute()) return true;
244  return false;
245  }
246 };
247 
248 //----------------------------------------------------------------------------
249 // AttNIntegerComparison
250 //----------------------------------------------------------------------------
251 
256 
257 public:
258  AttNIntegerComparison(ClassId classId, const int n) : ClassIdComparison(classId) { m_n = n; }
259 
260  void SetN(int n) { m_n = n; }
261 
262  bool operator()(const Object *object) override
263  {
264  if (!MatchesType(object)) return false;
265  // This should not happen, but just in case
266  if (!object->HasAttClass(ATT_NINTEGER)) return false;
267  const AttNInteger *element = dynamic_cast<const AttNInteger *>(object);
268  assert(element);
269  return (element->GetN() == m_n);
270  }
271 
272 private:
273  int m_n;
274 };
275 
276 //----------------------------------------------------------------------------
277 // AttNIntegerAnyComparison
278 //----------------------------------------------------------------------------
279 
284 
285 public:
286  AttNIntegerAnyComparison(ClassId classId, std::vector<int> ns) : ClassIdComparison(classId) { m_ns = ns; }
287 
288  void SetNs(std::vector<int> ns) { m_ns = ns; }
289  void AppendN(int n) { m_ns.push_back(n); }
290 
291  bool operator()(const Object *object) override
292  {
293  if (!MatchesType(object)) return false;
294  // This should not happen, but just in case
295  if (!object->HasAttClass(ATT_NINTEGER)) return false;
296  const AttNInteger *element = dynamic_cast<const AttNInteger *>(object);
297  assert(element);
298  return (std::find(m_ns.begin(), m_ns.end(), element->GetN()) != m_ns.end());
299  }
300 
301 private:
302  std::vector<int> m_ns;
303 };
304 
305 //----------------------------------------------------------------------------
306 // AttNNumberLikeComparison
307 //----------------------------------------------------------------------------
308 
313 
314 public:
315  AttNNumberLikeComparison(ClassId classId, const std::string n) : ClassIdComparison(classId) { m_n = n; }
316 
317  void SetN(std::string n) { m_n = n; }
318 
319  bool operator()(const Object *object) override
320  {
321  if (!MatchesType(object)) return false;
322  // This should not happen, but just in case
323  if (!object->HasAttClass(ATT_NNUMBERLIKE)) return false;
324  const AttNNumberLike *element = dynamic_cast<const AttNNumberLike *>(object);
325  assert(element);
326  return (element->GetN() == m_n);
327  }
328 
329 private:
330  std::string m_n;
331 };
332 
333 //----------------------------------------------------------------------------
334 // AttDurExtremeComparison
335 //----------------------------------------------------------------------------
336 
343 
344 public:
345  AttDurExtremeComparison(DurExtreme extremeType) : ClassIdComparison(OBJECT)
346  {
347  m_extremeType = extremeType;
348  m_extremeDur = (m_extremeType == LONGEST) ? -VRV_UNSET : VRV_UNSET;
349  }
350 
351  bool operator()(const Object *object) override
352  {
353  if (!object->HasInterface(INTERFACE_DURATION)) return false;
354  const DurationInterface *interface = dynamic_cast<const DurationInterface *>(object);
355  assert(interface);
356  if (interface->HasDur()) {
357  if ((m_extremeType == LONGEST) && (interface->GetActualDur() < m_extremeDur)) {
358  m_extremeDur = interface->GetActualDur();
359  return true;
360  }
361  else if ((m_extremeType == SHORTEST) && (interface->GetActualDur() > m_extremeDur)) {
362  m_extremeDur = interface->GetActualDur();
363  return true;
364  }
365  }
366  return false;
367  }
368 
369 private:
370  int m_extremeDur;
371  DurExtreme m_extremeType;
372 };
373 
374 //----------------------------------------------------------------------------
375 // AttVisibilityComparison
376 //----------------------------------------------------------------------------
381 
382 public:
383  AttVisibilityComparison(ClassId classId, data_BOOLEAN isVisible) : ClassIdComparison(classId)
384  {
385  m_isVisible = isVisible;
386  };
387 
388  bool operator()(const Object *object) override
389  {
390  if (!MatchesType(object)) return false;
391  if (!object->HasAttClass(ATT_VISIBILITY)) return false;
392  const AttVisibility *visibility = dynamic_cast<const AttVisibility *>(object);
393  assert(visibility);
394  return (visibility->GetVisible() == m_isVisible);
395  }
396 
397 private:
398  data_BOOLEAN m_isVisible;
399 };
400 
401 //----------------------------------------------------------------------------
402 // AttFormeworkComparison
403 //----------------------------------------------------------------------------
404 
409 
410 public:
411  AttFormeworkComparison(ClassId classId, data_PGFUNC func) : ClassIdComparison(classId) { m_func = func; }
412 
413  bool operator()(const Object *object) override
414  {
415  if (!MatchesType(object)) return false;
416  // This should not happen, but just in case
417  if (!object->HasAttClass(ATT_FORMEWORK)) return false;
418  const AttFormework *element = dynamic_cast<const AttFormework *>(object);
419  assert(element);
420  return (element->GetFunc() == m_func);
421  }
422 
423 private:
424  data_PGFUNC m_func;
425 };
426 
427 //----------------------------------------------------------------------------
428 // CrossAlignmentReferenceComparison
429 //----------------------------------------------------------------------------
430 
435 public:
436  CrossAlignmentReferenceComparison() : ClassIdComparison(ALIGNMENT_REFERENCE) {}
437 
438  bool operator()(const Object *object) override
439  {
440  if (!this->MatchesType(object)) return false;
441  const AlignmentReference *ref = vrv_cast<const AlignmentReference *>(object);
442  assert(ref);
443  return ref->HasCrossStaffElements();
444  }
445 };
446 
447 //----------------------------------------------------------------------------
448 // MeasureAlignerTypeComparison
449 //----------------------------------------------------------------------------
450 
455 
456 public:
457  MeasureAlignerTypeComparison(const AlignmentType type) : ClassIdComparison(ALIGNMENT) { m_type = type; }
458 
459  void SetType(AlignmentType type) { m_type = type; }
460 
461  bool operator()(const Object *object) override
462  {
463  if (!MatchesType(object)) return false;
464  const Alignment *alignment = vrv_cast<const Alignment *>(object);
465  assert(alignment);
466  return (alignment->GetType() == m_type);
467  }
468 
469 private:
470  AlignmentType m_type;
471 };
472 
473 //----------------------------------------------------------------------------
474 // MeasureOnsetOffsetComparison
475 //----------------------------------------------------------------------------
476 
481 
482 public:
483  MeasureOnsetOffsetComparison(const int time) : ClassIdComparison(MEASURE) { m_time = time; }
484 
485  void SetTime(int time) { m_time = time; }
486 
487  bool operator()(const Object *object) override
488  {
489  if (!MatchesType(object)) return false;
490  const Measure *measure = vrv_cast<const Measure *>(object);
491  assert(measure);
492  return (measure->EnclosesTime(m_time) > 0);
493  }
494 
495 private:
496  int m_time;
497 };
498 
499 //----------------------------------------------------------------------------
500 // NoteOrRestOnsetOffsetComparison
501 //----------------------------------------------------------------------------
502 
507 
508 public:
509  NoteOrRestOnsetOffsetComparison(const int time) : ClassIdsComparison({ NOTE, REST }) { m_time = time; }
510 
511  void SetTime(int time) { m_time = time; }
512 
513  bool operator()(const Object *object) override
514  {
515  if (!MatchesType(object)) return false;
516  const DurationInterface *interface = object->GetDurationInterface();
517  assert(interface);
518  return ((m_time >= interface->GetRealTimeOnsetMilliseconds())
519  && (m_time <= interface->GetRealTimeOffsetMilliseconds()));
520  }
521 
522 private:
523  int m_time;
524 };
525 
526 //----------------------------------------------------------------------------
527 // IDComparison
528 //----------------------------------------------------------------------------
529 
534 
535 public:
536  IDComparison(ClassId classId, const std::string &id) : ClassIdComparison(classId) { m_id = id; }
537 
538  void SetID(const std::string &id) { m_id = id; }
539 
540  bool operator()(const Object *object) override
541  {
542  if (!MatchesType(object)) return false;
543  return (object->GetID() == m_id);
544  }
545 
546 private:
547  std::string m_id;
548 };
549 
550 //----------------------------------------------------------------------------
551 // VisibleStaffDefOrGrpObject
552 //----------------------------------------------------------------------------
558 
559 public:
560  VisibleStaffDefOrGrpObject() : ClassIdsComparison({ STAFFDEF, STAFFGRP }) {}
561 
562  void Skip(const Object *objectToExclude) { m_objectToExclude = objectToExclude; }
563 
564  bool operator()(const Object *object) override
565  {
566  if (object == m_objectToExclude || !ClassIdsComparison::operator()(object)) return false;
567 
568  if (object->Is(STAFFDEF)) {
569  const StaffDef *staffDef = vrv_cast<const StaffDef *>(object);
570  return staffDef && staffDef->GetDrawingVisibility() != OPTIMIZATION_HIDDEN;
571  }
572 
573  const StaffGrp *staffGrp = vrv_cast<const StaffGrp *>(object);
574  return staffGrp && staffGrp->GetDrawingVisibility() != OPTIMIZATION_HIDDEN;
575  }
576 
577 protected:
578  const Object *m_objectToExclude;
579 };
580 
581 //----------------------------------------------------------------------------
582 // Filters class
583 //----------------------------------------------------------------------------
584 
588 class Filters {
589 public:
590  enum class Type { AllOf, AnyOf };
591 
592 public:
593  Filters() = default;
594  explicit Filters(const std::initializer_list<Comparison *> &comp)
595  {
596  std::copy(comp.begin(), comp.end(), std::back_inserter(m_filters));
597  }
598 
599  void Add(Comparison *comp) { m_filters.push_back(comp); }
600  void Clear() { m_filters.clear(); }
601  void SetType(Type type) { m_type = type; }
602 
606  bool Apply(const Object *object) const
607  {
608  auto condition = [object](Comparison *iter) {
609  // ignore any class comparison which does not match the object class
610  ClassIdComparison *cmp = dynamic_cast<ClassIdComparison *>(iter);
611  if (!cmp || (cmp->GetType() != object->GetClassId())) {
612  return true;
613  }
614  return (*iter)(object);
615  };
616  switch (m_type) {
617  case Type::AnyOf: {
618  return std::any_of(m_filters.cbegin(), m_filters.cend(), condition);
619  }
620  case Type::AllOf:
621  default: {
622  return std::all_of(m_filters.cbegin(), m_filters.cend(), condition);
623  }
624  }
625  }
626 
627  Filters &operator=(const std::initializer_list<Comparison *> &other)
628  {
629  m_filters.clear();
630  std::copy(other.begin(), other.end(), std::back_inserter(m_filters));
631  return *this;
632  }
633 
634 private:
635  std::vector<Comparison *> m_filters;
636  Type m_type = Type::AllOf;
637 };
638 
639 } // namespace vrv
640 
641 #endif
vrv::StaffDef
This class represents a MEI staffDef.
Definition: staffdef.h:26
vrv::Object::GetParent
Object * GetParent()
Get the parent of the Object.
Definition: object.h:379
vrv::Measure
This class represents a measure in a page-based score (Doc).
Definition: measure.h:37
vrv::InterfaceComparison
Definition: comparison.h:102
vrv::AttNIntegerComparison
This class evaluates if the object is of a certain ClassId and has a of value n.
Definition: comparison.h:255
vrv::ClassIdsComparison
Definition: comparison.h:81
vrv::Add
Definition: add.h:20
vrv::Object
This class represents a basic object.
Definition: object.h:59
vrv::IsEmptyComparison
This class evaluates if the object is of a certain ClassId and is empty.
Definition: comparison.h:211
vrv::ClassIdComparison
Definition: comparison.h:58
vrv::Filters::Apply
bool Apply(const Object *object) const
Apply comparison filter based on the specified type.
Definition: comparison.h:606
vrv::Measure::EnclosesTime
int EnclosesTime(int time) const
Check if the measure encloses the given time (in millisecond) Return the playing repeat time (1-based...
vrv::AttNNumberLikeComparison
This class evaluates if the object is of a certain ClassId and has a of value n.
Definition: comparison.h:312
vrv::AlignmentReference
This class stores a references of LayerElements for a staff.
Definition: horizontalaligner.h:290
vrv::Filters
This class is used to store comparison filters and apply them when necessary.
Definition: comparison.h:588
vrv::SpanningToComparison
Definition: comparison.h:165
vrv::MeasureOnsetOffsetComparison
This class evaluates if the object is a measure enclosing the given time.
Definition: comparison.h:480
vrv::VisibleStaffDefOrGrpObject
This class evaluates if the object is a visible StaffDef or StaffGrp.
Definition: comparison.h:557
vrv::AttNIntegerAnyComparison
This class evaluates if the object is of a certain ClassId and has a of value n.
Definition: comparison.h:283
vrv::IsEditorialElementComparison
This class evaluates if the object is an editorial element.
Definition: comparison.h:192
vrv::Alignment
This class stores an alignment position elements will point to.
Definition: horizontalaligner.h:73
vrv::PointingToComparison
Definition: comparison.h:141
vrv::AlignmentReference::HasCrossStaffElements
bool HasCrossStaffElements() const
Return true if the reference has elements from cross-staff.
vrv::AttVisibilityComparison
This class evaluates if the object is visible.
Definition: comparison.h:380
vrv::ChildOfClassIdComparison
Definition: comparison.h:123
vrv::IDComparison
This class evaluates if the object is of a certain ClassId has a certain ID.
Definition: comparison.h:533
vrv::AttDurExtremeComparison
This class evaluates if the object the extreme duration so far The object has to have a DurationInter...
Definition: comparison.h:342
vrv::CrossAlignmentReferenceComparison
This class evaluates if alignment reference contains cross-staff elements.
Definition: comparison.h:434
vrv::NoteOrRestOnsetOffsetComparison
This class evaluates if the object is a note being played at the given time.
Definition: comparison.h:506
vrv::DurationInterface
This class is an interface for elements with duration, such as notes and rests.
Definition: durationinterface.h:31
vrv::IsAttributeComparison
This class evaluates if the object is of a certain ClassId and is an attribute in the original MEI.
Definition: comparison.h:235
vrv::TimeSpanningInterface
This class is an interface for spanning elements, such as slur, hairpin, etc.
Definition: timeinterface.h:141
vrv::StaffGrp
This class represents a MEI staffGrp.
Definition: staffgrp.h:31
vrv::AttFormeworkComparison
This class evaluates if the object is of a certain ClassId and has a @func of value func.
Definition: comparison.h:408
vrv::Comparison
Definition: comparison.h:31
vrv::MeasureAlignerTypeComparison
This class evaluates if the object is an Alignment of a certain type.
Definition: comparison.h:454
vrv::TimePointInterface
This class is an interface for elements having a single time point, such as tempo,...
Definition: timeinterface.h:35