The JUCE cross-platform C++ framework, with DISTRHO/KXStudio specific changes
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

329 lines
14KB

  1. /*
  2. ==============================================================================
  3. This file is part of the JUCE library.
  4. Copyright (c) 2017 - ROLI Ltd.
  5. JUCE is an open source library subject to commercial or open-source
  6. licensing.
  7. By using JUCE, you agree to the terms of both the JUCE 5 End-User License
  8. Agreement and JUCE 5 Privacy Policy (both updated and effective as of the
  9. 27th April 2017).
  10. End User License Agreement: www.juce.com/juce-5-licence
  11. Privacy Policy: www.juce.com/juce-5-privacy-policy
  12. Or: You may also use this code under the terms of the GPL v3 (see
  13. www.gnu.org/licenses).
  14. JUCE IS PROVIDED "AS IS" WITHOUT ANY WARRANTY, AND ALL WARRANTIES, WHETHER
  15. EXPRESSED OR IMPLIED, INCLUDING MERCHANTABILITY AND FITNESS FOR PURPOSE, ARE
  16. DISCLAIMED.
  17. ==============================================================================
  18. */
  19. namespace juce
  20. {
  21. //==============================================================================
  22. /**
  23. A glyph from a particular font, with a particular size, style,
  24. typeface and position.
  25. You should rarely need to use this class directly - for most purposes, the
  26. GlyphArrangement class will do what you need for text layout.
  27. @see GlyphArrangement, Font
  28. */
  29. class JUCE_API PositionedGlyph final
  30. {
  31. public:
  32. //==============================================================================
  33. PositionedGlyph() noexcept;
  34. PositionedGlyph (const Font& font, juce_wchar character, int glyphNumber,
  35. float anchorX, float baselineY, float width, bool isWhitespace);
  36. PositionedGlyph (const PositionedGlyph&) = default;
  37. PositionedGlyph& operator= (const PositionedGlyph&) = default;
  38. // VS2013 can't default move constructors and assignmants
  39. PositionedGlyph (PositionedGlyph&&) noexcept;
  40. PositionedGlyph& operator= (PositionedGlyph&&) noexcept;
  41. ~PositionedGlyph();
  42. /** Returns the character the glyph represents. */
  43. juce_wchar getCharacter() const noexcept { return character; }
  44. /** Checks whether the glyph is actually empty. */
  45. bool isWhitespace() const noexcept { return whitespace; }
  46. /** Returns the position of the glyph's left-hand edge. */
  47. float getLeft() const noexcept { return x; }
  48. /** Returns the position of the glyph's right-hand edge. */
  49. float getRight() const noexcept { return x + w; }
  50. /** Returns the y position of the glyph's baseline. */
  51. float getBaselineY() const noexcept { return y; }
  52. /** Returns the y position of the top of the glyph. */
  53. float getTop() const { return y - font.getAscent(); }
  54. /** Returns the y position of the bottom of the glyph. */
  55. float getBottom() const { return y + font.getDescent(); }
  56. /** Returns the bounds of the glyph. */
  57. Rectangle<float> getBounds() const { return { x, getTop(), w, font.getHeight() }; }
  58. //==============================================================================
  59. /** Shifts the glyph's position by a relative amount. */
  60. void moveBy (float deltaX, float deltaY);
  61. //==============================================================================
  62. /** Draws the glyph into a graphics context.
  63. (Note that this may change the context's currently selected font).
  64. */
  65. void draw (Graphics& g) const;
  66. /** Draws the glyph into a graphics context, with an extra transform applied to it.
  67. (Note that this may change the context's currently selected font).
  68. */
  69. void draw (Graphics& g, AffineTransform transform) const;
  70. /** Returns the path for this glyph.
  71. @param path the glyph's outline will be appended to this path
  72. */
  73. void createPath (Path& path) const;
  74. /** Checks to see if a point lies within this glyph. */
  75. bool hitTest (float x, float y) const;
  76. private:
  77. //==============================================================================
  78. friend class GlyphArrangement;
  79. Font font;
  80. juce_wchar character;
  81. int glyph;
  82. float x, y, w;
  83. bool whitespace;
  84. JUCE_LEAK_DETECTOR (PositionedGlyph)
  85. };
  86. //==============================================================================
  87. /**
  88. A set of glyphs, each with a position.
  89. You can create a GlyphArrangement, text to it and then draw it onto a
  90. graphics context. It's used internally by the text methods in the
  91. Graphics class, but can be used directly if more control is needed.
  92. @see Font, PositionedGlyph
  93. */
  94. class JUCE_API GlyphArrangement final
  95. {
  96. public:
  97. //==============================================================================
  98. /** Creates an empty arrangement. */
  99. GlyphArrangement();
  100. GlyphArrangement (const GlyphArrangement&) = default;
  101. GlyphArrangement& operator= (const GlyphArrangement&) = default;
  102. // VS2013 can't default move constructors and assignmants
  103. GlyphArrangement (GlyphArrangement&&);
  104. GlyphArrangement& operator= (GlyphArrangement&&);
  105. /** Destructor. */
  106. ~GlyphArrangement();
  107. //==============================================================================
  108. /** Returns the total number of glyphs in the arrangement. */
  109. int getNumGlyphs() const noexcept { return glyphs.size(); }
  110. /** Returns one of the glyphs from the arrangement.
  111. @param index the glyph's index, from 0 to (getNumGlyphs() - 1). Be
  112. careful not to pass an out-of-range index here, as it
  113. doesn't do any bounds-checking.
  114. */
  115. PositionedGlyph& getGlyph (int index) noexcept;
  116. const PositionedGlyph* begin() const { return glyphs.begin(); }
  117. const PositionedGlyph* end() const { return glyphs.end(); }
  118. //==============================================================================
  119. /** Clears all text from the arrangement and resets it. */
  120. void clear();
  121. /** Appends a line of text to the arrangement.
  122. This will add the text as a single line, where x is the left-hand edge of the
  123. first character, and y is the position for the text's baseline.
  124. If the text contains new-lines or carriage-returns, this will ignore them - use
  125. addJustifiedText() to add multi-line arrangements.
  126. */
  127. void addLineOfText (const Font& font,
  128. const String& text,
  129. float x, float y);
  130. /** Adds a line of text, truncating it if it's wider than a specified size.
  131. This is the same as addLineOfText(), but if the line's width exceeds the value
  132. specified in maxWidthPixels, it will be truncated using either ellipsis (i.e. dots: "..."),
  133. if useEllipsis is true, or if this is false, it will just drop any subsequent characters.
  134. */
  135. void addCurtailedLineOfText (const Font& font,
  136. const String& text,
  137. float x, float y,
  138. float maxWidthPixels,
  139. bool useEllipsis);
  140. /** Adds some multi-line text, breaking lines at word-boundaries if they are too wide.
  141. This will add text to the arrangement, breaking it into new lines either where there
  142. is a new-line or carriage-return character in the text, or where a line's width
  143. exceeds the value set in maxLineWidth.
  144. Each line that is added will be laid out using the flags set in horizontalLayout, so
  145. the lines can be left- or right-justified, or centred horizontally in the space
  146. between x and (x + maxLineWidth).
  147. The y coordinate is the position of the baseline of the first line of text - subsequent
  148. lines will be placed below it, separated by a distance of font.getHeight().
  149. */
  150. void addJustifiedText (const Font& font,
  151. const String& text,
  152. float x, float y,
  153. float maxLineWidth,
  154. Justification horizontalLayout);
  155. /** Tries to fit some text within a given space.
  156. This does its best to make the given text readable within the specified rectangle,
  157. so it useful for labelling things.
  158. If the text is too big, it'll be squashed horizontally or broken over multiple lines
  159. if the maximumLinesToUse value allows this. If the text just won't fit into the space,
  160. it'll cram as much as possible in there, and put some ellipsis at the end to show that
  161. it's been truncated.
  162. A Justification parameter lets you specify how the text is laid out within the rectangle,
  163. both horizontally and vertically.
  164. The minimumHorizontalScale parameter specifies how much the text can be squashed horizontally
  165. to try to squeeze it into the space. If you don't want any horizontal scaling to occur, you
  166. can set this value to 1.0f. Pass 0 if you want it to use the default value.
  167. @see Graphics::drawFittedText
  168. */
  169. void addFittedText (const Font& font,
  170. const String& text,
  171. float x, float y, float width, float height,
  172. Justification layout,
  173. int maximumLinesToUse,
  174. float minimumHorizontalScale = 0.0f);
  175. /** Appends another glyph arrangement to this one. */
  176. void addGlyphArrangement (const GlyphArrangement&);
  177. /** Appends a custom glyph to the arrangement. */
  178. void addGlyph (const PositionedGlyph&);
  179. //==============================================================================
  180. /** Draws this glyph arrangement to a graphics context.
  181. This uses cached bitmaps so is much faster than the draw (Graphics&, AffineTransform)
  182. method, which renders the glyphs as filled vectors.
  183. */
  184. void draw (const Graphics&) const;
  185. /** Draws this glyph arrangement to a graphics context.
  186. This renders the paths as filled vectors, so is far slower than the draw (Graphics&)
  187. method for non-transformed arrangements.
  188. */
  189. void draw (const Graphics&, AffineTransform) const;
  190. /** Converts the set of glyphs into a path.
  191. @param path the glyphs' outlines will be appended to this path
  192. */
  193. void createPath (Path& path) const;
  194. /** Looks for a glyph that contains the given coordinate.
  195. @returns the index of the glyph, or -1 if none were found.
  196. */
  197. int findGlyphIndexAt (float x, float y) const;
  198. //==============================================================================
  199. /** Finds the smallest rectangle that will enclose a subset of the glyphs.
  200. @param startIndex the first glyph to test
  201. @param numGlyphs the number of glyphs to include; if this is < 0, all glyphs after
  202. startIndex will be included
  203. @param includeWhitespace if true, the extent of any whitespace characters will also
  204. be taken into account
  205. */
  206. Rectangle<float> getBoundingBox (int startIndex, int numGlyphs, bool includeWhitespace) const;
  207. /** Shifts a set of glyphs by a given amount.
  208. @param startIndex the first glyph to transform
  209. @param numGlyphs the number of glyphs to move; if this is < 0, all glyphs after
  210. startIndex will be used
  211. @param deltaX the amount to add to their x-positions
  212. @param deltaY the amount to add to their y-positions
  213. */
  214. void moveRangeOfGlyphs (int startIndex, int numGlyphs,
  215. float deltaX, float deltaY);
  216. /** Removes a set of glyphs from the arrangement.
  217. @param startIndex the first glyph to remove
  218. @param numGlyphs the number of glyphs to remove; if this is < 0, all glyphs after
  219. startIndex will be deleted
  220. */
  221. void removeRangeOfGlyphs (int startIndex, int numGlyphs);
  222. /** Expands or compresses a set of glyphs horizontally.
  223. @param startIndex the first glyph to transform
  224. @param numGlyphs the number of glyphs to stretch; if this is < 0, all glyphs after
  225. startIndex will be used
  226. @param horizontalScaleFactor how much to scale their horizontal width by
  227. */
  228. void stretchRangeOfGlyphs (int startIndex, int numGlyphs,
  229. float horizontalScaleFactor);
  230. /** Justifies a set of glyphs within a given space.
  231. This moves the glyphs as a block so that the whole thing is located within the
  232. given rectangle with the specified layout.
  233. If the Justification::horizontallyJustified flag is specified, each line will
  234. be stretched out to fill the specified width.
  235. */
  236. void justifyGlyphs (int startIndex, int numGlyphs,
  237. float x, float y, float width, float height,
  238. Justification justification);
  239. private:
  240. //==============================================================================
  241. Array<PositionedGlyph> glyphs;
  242. int insertEllipsis (const Font&, float maxXPos, int startIndex, int endIndex);
  243. int fitLineIntoSpace (int start, int numGlyphs, float x, float y, float w, float h, const Font&,
  244. Justification, float minimumHorizontalScale);
  245. void spreadOutLine (int start, int numGlyphs, float targetWidth);
  246. void splitLines (const String&, Font, int start, float x, float y, float w, float h, int maxLines,
  247. float lineWidth, Justification, float minimumHorizontalScale);
  248. void addLinesWithLineBreaks (const String&, const Font&, float x, float y, float width, float height, Justification);
  249. void drawGlyphUnderline (const Graphics&, const PositionedGlyph&, int, AffineTransform) const;
  250. JUCE_LEAK_DETECTOR (GlyphArrangement)
  251. };
  252. } // namespace juce