Verovio
Source code documentation
slur.h
1 // Name: slur.h
3 // Author: Rodolfo Zitellini
4 // Created: 26/06/2012
5 // Copyright (c) Authors and others. All rights reserved.
7 
8 #ifndef __VRV_SLUR_H__
9 #define __VRV_SLUR_H__
10 
11 #include "controlelement.h"
12 #include "timeinterface.h"
13 
14 namespace vrv {
15 
16 class Chord;
17 class Doc;
18 class Layer;
19 class Note;
20 class Staff;
21 
22 //----------------------------------------------------------------------------
23 // SpannedElements
24 //----------------------------------------------------------------------------
29  std::vector<const LayerElement *> elements;
30  std::set<int> layersN;
31 };
32 
33 //----------------------------------------------------------------------------
34 // NearEndCollision
35 //----------------------------------------------------------------------------
40  double metricAtStart;
41  double metricAtEnd;
42  bool endPointsAdjusted;
43 };
44 
45 // Helper enum classes
46 enum class SlurCurveDirection { None, Above, Below, AboveBelow, BelowAbove };
47 enum class PortatoSlurType { None, StemSide, Centered };
48 
49 //----------------------------------------------------------------------------
50 // Slur
51 //----------------------------------------------------------------------------
52 
53 class Slur : public ControlElement,
54  public TimeSpanningInterface,
55  public AttCurvature,
56  public AttLayerIdent,
57  public AttLineRendBase {
58 public:
63  Slur();
65  Slur(ClassId classId);
66  virtual ~Slur();
67  Object *Clone() const override { return new Slur(*this); }
68  void Reset() override;
69  std::string GetClassName() const override { return "slur"; }
71 
75  TimePointInterface *GetTimePointInterface() override { return vrv_cast<TimePointInterface *>(this); }
77  const TimePointInterface *GetTimePointInterface() const override
78  {
79  return vrv_cast<const TimePointInterface *>(this);
80  }
81  TimeSpanningInterface *GetTimeSpanningInterface() override { return vrv_cast<TimeSpanningInterface *>(this); }
82  const TimeSpanningInterface *GetTimeSpanningInterface() const override
83  {
84  return vrv_cast<const TimeSpanningInterface *>(this);
85  }
87 
91  SlurCurveDirection GetDrawingCurveDir() const { return m_drawingCurveDir; }
93  void SetDrawingCurveDir(SlurCurveDirection curveDir) { m_drawingCurveDir = curveDir; }
94  bool HasDrawingCurveDir() const { return (m_drawingCurveDir != SlurCurveDirection::None); }
95  curvature_CURVEDIR CalcDrawingCurveDir(char spanningType) const;
97 
101  bool HasMixedCurveDir() const
103  {
104  return (m_drawingCurveDir == SlurCurveDirection::AboveBelow)
105  || (m_drawingCurveDir == SlurCurveDirection::BelowAbove);
106  }
107  bool HasEndpointAboveStart() const
108  {
109  return (m_drawingCurveDir == SlurCurveDirection::Above)
110  || (m_drawingCurveDir == SlurCurveDirection::AboveBelow);
111  }
112  bool HasEndpointBelowStart() const
113  {
114  return (m_drawingCurveDir == SlurCurveDirection::Below)
115  || (m_drawingCurveDir == SlurCurveDirection::BelowAbove);
116  }
117  bool HasEndpointAboveEnd() const
118  {
119  return (m_drawingCurveDir == SlurCurveDirection::Above)
120  || (m_drawingCurveDir == SlurCurveDirection::BelowAbove);
121  }
122  bool HasEndpointBelowEnd() const
123  {
124  return (m_drawingCurveDir == SlurCurveDirection::Below)
125  || (m_drawingCurveDir == SlurCurveDirection::AboveBelow);
126  }
128 
132  bool HasInnerSlur(const Slur *innerSlur) const;
133 
137  void CalcInitialCurve(const Doc *doc, FloatingCurvePositioner *curve, NearEndCollision *nearEndCollision = NULL);
138 
142  void CalcSpannedElements(FloatingCurvePositioner *curve);
143 
147  void AddPositionerToArticulations(FloatingCurvePositioner *curve);
148 
152  Staff *CalculatePrincipalStaff(const Staff *staff, int xMin, int xMax);
154  const Staff *CalculatePrincipalStaff(const Staff *staff, int xMin, int xMax) const;
156 
160  std::pair<const Layer *, const LayerElement *> GetBoundaryLayer() const;
162  const Staff *GetBoundaryCrossStaff() const;
164 
168  void InitBezierControlSides(BezierCurve &bezier, curvature_CURVEDIR curveDir) const;
169 
170  //----------//
171  // Functors //
172  //----------//
173 
177  FunctorCode Accept(Functor &functor) override;
179  FunctorCode Accept(ConstFunctor &functor) const override;
180  FunctorCode AcceptEnd(Functor &functor) override;
181  FunctorCode AcceptEnd(ConstFunctor &functor) const override;
183 
184 private:
188  // Determine layer elements spanned by the slur
190  SpannedElements CollectSpannedElements(const Staff *staff, int xMin, int xMax) const;
191  // Filter and add layer elements spanned by the slur to the positioner
192  void AddSpannedElements(
193  FloatingCurvePositioner *curve, const SpannedElements &elements, Staff *staff, int xMin, int xMax);
194  // Determine whether a layer element should lie above or below the slur
195  bool IsElementBelow(const LayerElement *element, const Staff *startStaff, const Staff *endStaff) const;
196  bool IsElementBelow(const FloatingPositioner *positioner, const Staff *startStaff, const Staff *endStaff) const;
197  // Discard tuplets that don't have to be considered for slur adjustment
198  void DiscardTupletElements(FloatingCurvePositioner *curve, int xMin, int xMax);
200 
204  // Calculate the endpoint coordinates depending on the curve direction and spanning type of the slur
206  std::pair<Point, Point> CalcEndPoints(const Doc *doc, const Staff *staff, NearEndCollision *nearEndCollision,
207  int x1, int x2, curvature_CURVEDIR drawingCurveDir, char spanningType) const;
208  // Consider the melodic direction for the break location?
209  bool ConsiderMelodicDirection() const;
210  // Retrieve the start and end note locations of the slur
211  std::pair<int, int> GetStartEndLocs(
212  const Note *startNote, const Chord *startChord, const Note *endNote, const Chord *endChord) const;
213  // Calculate the pitch difference used to adjust for melodic direction
214  int CalcPitchDifference(const Staff *staff, int startLoc, int endLoc) const;
215  // Check if the slur resembles portato
216  PortatoSlurType IsPortatoSlur(const Doc *doc, const Note *startNote, const Chord *startChord) const;
217  // Check if the slur starts or ends on a beam
218  bool StartsOnBeam() const { return this->HasBoundaryOnBeam(true); }
219  bool EndsOnBeam() const { return this->HasBoundaryOnBeam(false); }
220  bool HasBoundaryOnBeam(bool isStart) const;
221  // Angle adjustment
222  float GetAdjustedSlurAngle(const Doc *doc, Point &p1, Point &p2, curvature_CURVEDIR curveDir) const;
224 
225 public:
226  //
227 private:
233  SlurCurveDirection m_drawingCurveDir;
234 };
235 
236 } // namespace vrv
237 
238 #endif
vrv::Slur::CalculatePrincipalStaff
Staff * CalculatePrincipalStaff(const Staff *staff, int xMin, int xMax)
Calculate the staff where the slur's floating curve positioner lives.
vrv::SpannedElements
Contains the layer elements used for collision detection.
Definition: slur.h:28
vrv::Slur::GetBoundaryLayer
std::pair< const Layer *, const LayerElement * > GetBoundaryLayer() const
Determine the slur's layer/cross staff by only considering the boundary.
vrv::Slur::AddPositionerToArticulations
void AddPositionerToArticulations(FloatingCurvePositioner *curve)
Add curve positioner to articulations.
vrv::Object
This class represents a basic object.
Definition: object.h:59
vrv::Slur::InitBezierControlSides
void InitBezierControlSides(BezierCurve &bezier, curvature_CURVEDIR curveDir) const
Set the bezier control sides depending on the curve direction.
vrv::Slur::Accept
FunctorCode Accept(Functor &functor) override
Interface for class functor visitation.
vrv::Slur
Definition: slur.h:53
vrv::Slur::CalcSpannedElements
void CalcSpannedElements(FloatingCurvePositioner *curve)
Recalculate the spanned elements of the curve positioner.
vrv::Slur::CalcInitialCurve
void CalcInitialCurve(const Doc *doc, FloatingCurvePositioner *curve, NearEndCollision *nearEndCollision=NULL)
Calculate the initial slur bezier curve and store it in the curve positioner.
vrv::Slur::Clone
Object * Clone() const override
Method call for copying child classes.
Definition: slur.h:67
vrv::Slur::Reset
void Reset() override
Virtual reset method.
vrv::ControlElement
This class represents elements appearing within a measure.
Definition: controlelement.h:28
vrv::TimeSpanningInterface
This class is an interface for spanning elements, such as slur, hairpin, etc.
Definition: timeinterface.h:141
vrv::NearEndCollision
Measure collisions near the end points.
Definition: slur.h:39