Verovio
Source code documentation
boundingbox.h
1 // Name: boundingbox.h
3 // Author: Laurent Pugin
4 // Created: 2016
5 // Copyright (c) Authors and others. All rights reserved.
7 
8 #ifndef __VRV_BOUNDING_BOX_H__
9 #define __VRV_BOUNDING_BOX_H__
10 
11 //----------------------------------------------------------------------------
12 
13 #include "vrvdef.h"
14 
15 namespace vrv {
16 
17 #define BEZIER_APPROXIMATION 50.0
18 
19 class BeamDrawingInterface;
20 class Doc;
21 class FloatingCurvePositioner;
22 class Glyph;
23 class Resources;
24 
25 //----------------------------------------------------------------------------
26 // BoundingBox
27 //----------------------------------------------------------------------------
28 
32 class BoundingBox {
33 public:
38  BoundingBox();
40  virtual ~BoundingBox() {}
41  virtual ClassId GetClassId() const = 0;
42  bool Is(ClassId classId) const { return (this->GetClassId() == classId); }
43  bool Is(const std::vector<ClassId> &classIds) const;
45 
49  virtual void UpdateContentBBoxX(int x1, int x2);
51  virtual void UpdateContentBBoxY(int y1, int y2);
52  virtual void UpdateSelfBBoxX(int x1, int x2);
53  virtual void UpdateSelfBBoxY(int y1, int y2);
54  void SetEmptyBB();
55  //
56  bool HasSelfBB() const;
57  bool HasSelfHorizontalBB() const;
58  bool HasSelfVerticalBB() const;
59  bool HasContentBB() const;
60  bool HasContentHorizontalBB() const;
61  bool HasContentVerticalBB() const;
62  bool HasEmptyBB() const;
64 
68  void SetBoundingBoxGlyph(char32_t smuflGlyph, int fontSize);
70  char32_t GetBoundingBoxGlyph() const { return m_smuflGlyph; }
71  int GetBoundingBoxGlyphFontSize() const { return m_smuflGlyphFontSize; }
73 
77  virtual void ResetBoundingBox();
78 
83  virtual int GetDrawingX() const = 0;
85  virtual int GetDrawingY() const = 0;
87 
92  virtual void ResetCachedDrawingX() const = 0;
94  virtual void ResetCachedDrawingY() const = 0;
96 
100  int GetSelfBottom() const { return (this->GetDrawingY() + m_selfBB_y1); }
102  int GetSelfTop() const { return (this->GetDrawingY() + m_selfBB_y2); }
103  int GetSelfLeft() const { return (this->GetDrawingX() + m_selfBB_x1); }
104  int GetSelfRight() const { return (this->GetDrawingX() + m_selfBB_x2); }
105  int GetContentBottom() const { return (this->GetDrawingY() + m_contentBB_y1); }
106  int GetContentTop() const { return (this->GetDrawingY() + m_contentBB_y2); }
107  int GetContentLeft() const { return (this->GetDrawingX() + m_contentBB_x1); }
108  int GetContentRight() const { return (this->GetDrawingX() + m_contentBB_x2); }
109  //
110  int GetSelfX1() const { return m_selfBB_x1; }
111  int GetSelfX2() const { return m_selfBB_x2; }
112  int GetSelfY1() const { return m_selfBB_y1; }
113  int GetSelfY2() const { return m_selfBB_y2; }
114  int GetContentX1() const { return m_contentBB_x1; }
115  int GetContentX2() const { return m_contentBB_x2; }
116  int GetContentY1() const { return m_contentBB_y1; }
117  int GetContentY2() const { return m_contentBB_y2; }
119 
123  int GetBottomBy(Accessor type) const { return ((type == SELF) ? GetSelfBottom() : GetContentBottom()); }
125  int GetTopBy(Accessor type) const { return ((type == SELF) ? GetSelfTop() : GetContentTop()); }
126  int GetLeftBy(Accessor type) const { return ((type == SELF) ? GetSelfLeft() : GetContentLeft()); }
127  int GetRightBy(Accessor type) const { return ((type == SELF) ? GetSelfRight() : GetContentRight()); }
128  int GetX1By(Accessor type) const { return ((type == SELF) ? GetSelfX1() : GetContentX1()); }
129  int GetX2By(Accessor type) const { return ((type == SELF) ? GetSelfX2() : GetContentX2()); }
130  int GetY1By(Accessor type) const { return ((type == SELF) ? GetSelfY1() : GetContentY1()); }
131  int GetY2By(Accessor type) const { return ((type == SELF) ? GetSelfY2() : GetContentY2()); }
133 
138  bool HorizontalContentOverlap(const BoundingBox *other, int margin = 0) const;
140  bool VerticalContentOverlap(const BoundingBox *other, int margin = 0) const;
141  bool HorizontalSelfOverlap(const BoundingBox *other, int margin = 0) const;
142  bool VerticalSelfOverlap(const BoundingBox *other, int margin = 0) const;
144 
148  int HorizontalLeftOverlap(const BoundingBox *other, const Doc *doc, int margin = 0, int vMargin = 0) const;
150  int HorizontalRightOverlap(const BoundingBox *other, const Doc *doc, int margin = 0, int vMargin = 0) const;
151  int VerticalTopOverlap(const BoundingBox *other, const Doc *doc, int margin = 0, int hMargin = 0) const;
152  int VerticalBottomOverlap(const BoundingBox *other, const Doc *doc, int margin = 0, int hMargin = 0) const;
154 
159  int GetCutOutTop(const Resources &resources) const;
161  int GetCutOutBottom(const Resources &resources) const;
162  int GetCutOutLeft(const Resources &resources) const;
163  int GetCutOutRight(const Resources &resources) const;
164  // Restricted version which only considers the cutout rectangles from the top or bottom
165  int GetCutOutLeft(const Resources &resources, bool fromTop) const;
166  int GetCutOutRight(const Resources &resources, bool fromTop) const;
168 
172  bool Encloses(const Point point) const;
173 
178  int Intersects(const FloatingCurvePositioner *curve, Accessor type, int margin = 0) const;
179 
185  int Intersects(const BeamDrawingInterface *beamInterface, Accessor type, int margin = 0,
186  bool fromBeamContentSide = false) const;
187 
188  //----------------//
189  // Static methods //
190  //----------------//
191 
192  static std::pair<double, int> ApproximateBezierExtrema(
193  const Point bezier[4], bool isMaxExtrema, int approximationSteps = BEZIER_APPROXIMATION);
194 
198  static double CalcDistance(const Point &p1, const Point &p2);
199 
203  static bool ArePointsClose(const Point &p1, const Point &p2, int margin);
204 
208  static double CalcSlope(const Point &p1, const Point &p2);
209 
213  static Point CalcPositionAfterRotation(Point point, float alpha, Point center);
214 
218  static double CalcBezierParamAtPosition(const Point bezier[4], int x);
219 
223  static int CalcBezierAtPosition(const Point bezier[4], int x);
224 
228  static void CalcLinearInterpolation(Point &dest, const Point &a, const Point &b, double t);
229 
233  static Point CalcPointAtBezier(const Point bezier[4], double t);
234 
238  static double GetBezierThicknessCoefficient(const Point bezier[4], int currentThickness, int penWidth);
239 
243  static Point CalcDeCasteljau(const Point bezier[4], double t);
244 
248  static void CalcThickBezier(const Point bezier[4], int thickness, Point topBezier[4], Point bottomBezier[4]);
249 
253  static void ApproximateBezierBoundingBox(
254  const Point bezier[4], Point &pos, int &width, int &height, int &minYPos, int &maxYPos);
255 
259  static int RectLeftOverlap(const Point rect1[2], const Point rect2[2], int margin, int vMargin);
261  static int RectRightOverlap(const Point rect1[2], const Point rect2[2], int margin, int vMargin);
262  static int RectTopOverlap(const Point rect1[2], const Point rect2[2], int margin, int hMargin);
263  static int RectBottomOverlap(const Point rect1[2], const Point rect2[2], int margin, int hMargin);
265 
270  static std::set<double> SolveCubicPolynomial(double a, double b, double c, double d);
271 
272 private:
279  int GetRectangles(const SMuFLGlyphAnchor &anchor, Point rect[2][2], const Resources &resources) const;
281  int GetRectangles(const SMuFLGlyphAnchor &anchor1, const SMuFLGlyphAnchor &anchor2, Point rect[3][2],
282  const Resources &resources) const;
284 
289  bool GetGlyph2PointRectangles(
290  const SMuFLGlyphAnchor &anchor1, const SMuFLGlyphAnchor &anchor2, const Glyph *glyph1, Point rect[3][2]) const;
291 
296  bool GetGlyph1PointRectangles(const SMuFLGlyphAnchor &anchor, const Glyph *glyph, Point rect[2][2]) const;
297 
298 public:
299  //
300 protected:
306  mutable int m_cachedDrawingX;
308  mutable int m_cachedDrawingY;
310 private:
314  int m_contentBB_x1, m_contentBB_y1, m_contentBB_x2, m_contentBB_y2;
316  int m_selfBB_x1, m_selfBB_y1, m_selfBB_x2, m_selfBB_y2;
318 
323  char32_t m_smuflGlyph;
324 
328  int m_smuflGlyphFontSize;
329 };
330 
331 //----------------------------------------------------------------------------
332 // SegmentedLine
333 //----------------------------------------------------------------------------
334 
338 public:
343  SegmentedLine(int start, int end);
345  virtual ~SegmentedLine() {}
347 
351  bool IsEmpty() const { return (m_segments.empty()); }
352 
356  bool IsUnsegmented() const { return (m_segments.size() == 1); }
357 
361  int GetSegmentCount() const { return (int)m_segments.size(); }
362 
366  std::pair<int, int> GetStartEnd(int idx) const;
367 
371  void AddGap(int start, int end);
372 
373 protected:
374  //
375 private:
376  //
377 public:
378  //
379 protected:
380  //
381 private:
386  ArrayOfIntPairs m_segments;
387 
391  bool m_increasing;
392 };
393 
394 } // namespace vrv
395 
396 #endif
vrv::BoundingBox::CalcPointAtBezier
static Point CalcPointAtBezier(const Point bezier[4], double t)
Calculate point (X,Y) coordinaties on the bezier curve.
vrv::BoundingBox::CalcLinearInterpolation
static void CalcLinearInterpolation(Point &dest, const Point &a, const Point &b, double t)
Calculate linear interpolation between two points at time t.
vrv::BoundingBox::SolveCubicPolynomial
static std::set< double > SolveCubicPolynomial(double a, double b, double c, double d)
Solve the cubic equation ax^3 + bx^2 + cx + d = 0 Returns up to three real roots.
vrv::BoundingBox::CalcBezierParamAtPosition
static double CalcBezierParamAtPosition(const Point bezier[4], int x)
Calculate the t parameter of a bezier at position x.
vrv::SegmentedLine::IsUnsegmented
bool IsUnsegmented() const
Check if the line is one single segment.
Definition: boundingbox.h:356
vrv::BoundingBox::CalcSlope
static double CalcSlope(const Point &p1, const Point &p2)
Calculate the slope represented by two points.
vrv::Doc
This class is a hold the data and corresponds to the model of a MVC design pattern.
Definition: doc.h:41
vrv::SegmentedLine::IsEmpty
bool IsEmpty() const
Check if the segmented line is empty.
Definition: boundingbox.h:351
vrv::BoundingBox::CalcDistance
static double CalcDistance(const Point &p1, const Point &p2)
Calculate the euclidean distance between two points.
vrv::BoundingBox::Encloses
bool Encloses(const Point point) const
Return true if the bounding box encloses the point.
vrv::BoundingBox::CalcBezierAtPosition
static int CalcBezierAtPosition(const Point bezier[4], int x)
Calculate the y position of a bezier at position x.
vrv::BoundingBox::ResetBoundingBox
virtual void ResetBoundingBox()
Reset the bounding box values.
vrv::FloatingCurvePositioner
This class represents a basic object for a curve (slur, tie) in the layout domain.
Definition: floatingobject.h:329
vrv::SegmentedLine::AddGap
void AddGap(int start, int end)
Add a gap in the line.
vrv::SegmentedLine
Definition: boundingbox.h:337
vrv::BoundingBox::CalcThickBezier
static void CalcThickBezier(const Point bezier[4], int thickness, Point topBezier[4], Point bottomBezier[4])
Calculate the position of the bezier above and below for a thick bezier.
vrv::BoundingBox::ArePointsClose
static bool ArePointsClose(const Point &p1, const Point &p2, int margin)
vrv::BoundingBox::ApproximateBezierBoundingBox
static void ApproximateBezierBoundingBox(const Point bezier[4], Point &pos, int &width, int &height, int &minYPos, int &maxYPos)
Approximate the bounding box of a bezier taking into accound the height and the width.
vrv::SegmentedLine::GetSegmentCount
int GetSegmentCount() const
The number of segments.
Definition: boundingbox.h:361
vrv::Resources
This class provides resource values.
Definition: resources.h:30
vrv::BoundingBox::m_cachedDrawingX
int m_cachedDrawingX
The cached version of the drawingX and drawingY values.
Definition: boundingbox.h:307
vrv::BoundingBox::RectLeftOverlap
static int RectLeftOverlap(const Point rect1[2], const Point rect2[2], int margin, int vMargin)
Calculate the left / right / top / bottom overlap of two rectangle taking into account the margin / v...
vrv::BeamDrawingInterface
This class is an interface for MEI beam elements (beam, beamSpan).
Definition: drawinginterface.h:97
vrv::BoundingBox::Intersects
int Intersects(const FloatingCurvePositioner *curve, Accessor type, int margin=0) const
Return intersection between the bounding box and the curve represented by the FloatingPositioner.
vrv::Glyph
This class is used for storing a music font glyph.
Definition: glyph.h:31
vrv::SegmentedLine::GetStartEnd
std::pair< int, int > GetStartEnd(int idx) const
Get the start and end of a segment.
vrv::BoundingBox::GetBezierThicknessCoefficient
static double GetBezierThicknessCoefficient(const Point bezier[4], int currentThickness, int penWidth)
Calculate thickness coefficient to be applient for bezier curve to fit MEI units thickness.
vrv::BoundingBox
This class represents a basic object in the layout domain.
Definition: boundingbox.h:32
vrv::Point
Simple class for representing points.
Definition: devicecontextbase.h:203
vrv::BoundingBox::CalcDeCasteljau
static Point CalcDeCasteljau(const Point bezier[4], double t)
Calculate the point bezier point position for a t between 0.0 and 1.0.
vrv::BoundingBox::CalcPositionAfterRotation
static Point CalcPositionAfterRotation(Point point, float alpha, Point center)
Calculate the position of a point after a rotation of alpha (in radian) around the center.
vrv::BoundingBox::SetBoundingBoxGlyph
void SetBoundingBoxGlyph(char32_t smuflGlyph, int fontSize)
Set and get the smuflGlyph / fontsize for a bounding box that is the one of a single SMuFL glyph.