|
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328 |
- /*
- ==============================================================================
-
- This file is part of the JUCE library.
- Copyright (c) 2020 - Raw Material Software Limited
-
- JUCE is an open source library subject to commercial or open-source
- licensing.
-
- By using JUCE, you agree to the terms of both the JUCE 6 End-User License
- Agreement and JUCE Privacy Policy (both effective as of the 16th June 2020).
-
- End User License Agreement: www.juce.com/juce-6-licence
- Privacy Policy: www.juce.com/juce-privacy-policy
-
- Or: You may also use this code under the terms of the GPL v3 (see
- www.gnu.org/licenses).
-
- JUCE IS PROVIDED "AS IS" WITHOUT ANY WARRANTY, AND ALL WARRANTIES, WHETHER
- EXPRESSED OR IMPLIED, INCLUDING MERCHANTABILITY AND FITNESS FOR PURPOSE, ARE
- DISCLAIMED.
-
- ==============================================================================
- */
-
- namespace juce
- {
-
- //==============================================================================
- /**
- A glyph from a particular font, with a particular size, style,
- typeface and position.
-
- You should rarely need to use this class directly - for most purposes, the
- GlyphArrangement class will do what you need for text layout.
-
- @see GlyphArrangement, Font
-
- @tags{Graphics}
- */
- class JUCE_API PositionedGlyph final
- {
- public:
- //==============================================================================
- PositionedGlyph() noexcept;
-
- PositionedGlyph (const Font& font, juce_wchar character, int glyphNumber,
- float anchorX, float baselineY, float width, bool isWhitespace);
-
- PositionedGlyph (const PositionedGlyph&) = default;
- PositionedGlyph& operator= (const PositionedGlyph&) = default;
- PositionedGlyph (PositionedGlyph&&) noexcept = default;
- PositionedGlyph& operator= (PositionedGlyph&&) noexcept = default;
-
- ~PositionedGlyph();
-
- /** Returns the character the glyph represents. */
- juce_wchar getCharacter() const noexcept { return character; }
- /** Checks whether the glyph is actually empty. */
- bool isWhitespace() const noexcept { return whitespace; }
-
- /** Returns the position of the glyph's left-hand edge. */
- float getLeft() const noexcept { return x; }
- /** Returns the position of the glyph's right-hand edge. */
- float getRight() const noexcept { return x + w; }
- /** Returns the y position of the glyph's baseline. */
- float getBaselineY() const noexcept { return y; }
- /** Returns the y position of the top of the glyph. */
- float getTop() const { return y - font.getAscent(); }
- /** Returns the y position of the bottom of the glyph. */
- float getBottom() const { return y + font.getDescent(); }
- /** Returns the bounds of the glyph. */
- Rectangle<float> getBounds() const { return { x, getTop(), w, font.getHeight() }; }
-
- //==============================================================================
- /** Shifts the glyph's position by a relative amount. */
- void moveBy (float deltaX, float deltaY);
-
- //==============================================================================
- /** Draws the glyph into a graphics context.
- (Note that this may change the context's currently selected font).
- */
- void draw (Graphics& g) const;
-
- /** Draws the glyph into a graphics context, with an extra transform applied to it.
- (Note that this may change the context's currently selected font).
- */
- void draw (Graphics& g, AffineTransform transform) const;
-
- /** Returns the path for this glyph.
- @param path the glyph's outline will be appended to this path
- */
- void createPath (Path& path) const;
-
- /** Checks to see if a point lies within this glyph. */
- bool hitTest (float x, float y) const;
-
- private:
- //==============================================================================
- friend class GlyphArrangement;
- Font font;
- juce_wchar character;
- int glyph;
- float x, y, w;
- bool whitespace;
-
- JUCE_LEAK_DETECTOR (PositionedGlyph)
- };
-
-
- //==============================================================================
- /**
- A set of glyphs, each with a position.
-
- You can create a GlyphArrangement, text to it and then draw it onto a
- graphics context. It's used internally by the text methods in the
- Graphics class, but can be used directly if more control is needed.
-
- @see Font, PositionedGlyph
-
- @tags{Graphics}
- */
- class JUCE_API GlyphArrangement final
- {
- public:
- //==============================================================================
- /** Creates an empty arrangement. */
- GlyphArrangement();
-
- GlyphArrangement (const GlyphArrangement&) = default;
- GlyphArrangement& operator= (const GlyphArrangement&) = default;
- GlyphArrangement (GlyphArrangement&&) = default;
- GlyphArrangement& operator= (GlyphArrangement&&) = default;
-
- /** Destructor. */
- ~GlyphArrangement() = default;
-
- //==============================================================================
- /** Returns the total number of glyphs in the arrangement. */
- int getNumGlyphs() const noexcept { return glyphs.size(); }
-
- /** Returns one of the glyphs from the arrangement.
-
- @param index the glyph's index, from 0 to (getNumGlyphs() - 1). Be
- careful not to pass an out-of-range index here, as it
- doesn't do any bounds-checking.
- */
- PositionedGlyph& getGlyph (int index) noexcept;
-
- const PositionedGlyph* begin() const { return glyphs.begin(); }
- const PositionedGlyph* end() const { return glyphs.end(); }
-
- //==============================================================================
- /** Clears all text from the arrangement and resets it. */
- void clear();
-
- /** Appends a line of text to the arrangement.
-
- This will add the text as a single line, where x is the left-hand edge of the
- first character, and y is the position for the text's baseline.
-
- If the text contains new-lines or carriage-returns, this will ignore them - use
- addJustifiedText() to add multi-line arrangements.
- */
- void addLineOfText (const Font& font,
- const String& text,
- float x, float y);
-
- /** Adds a line of text, truncating it if it's wider than a specified size.
-
- This is the same as addLineOfText(), but if the line's width exceeds the value
- specified in maxWidthPixels, it will be truncated using either ellipsis (i.e. dots: "..."),
- if useEllipsis is true, or if this is false, it will just drop any subsequent characters.
- */
- void addCurtailedLineOfText (const Font& font,
- const String& text,
- float x, float y,
- float maxWidthPixels,
- bool useEllipsis);
-
- /** Adds some multi-line text, breaking lines at word-boundaries if they are too wide.
-
- This will add text to the arrangement, breaking it into new lines either where there
- is a new-line or carriage-return character in the text, or where a line's width
- exceeds the value set in maxLineWidth.
-
- Each line that is added will be laid out using the flags set in horizontalLayout, so
- the lines can be left- or right-justified, or centred horizontally in the space
- between x and (x + maxLineWidth).
-
- The y coordinate is the position of the baseline of the first line of text - subsequent
- lines will be placed below it, separated by a distance of font.getHeight() + leading.
- */
- void addJustifiedText (const Font& font,
- const String& text,
- float x, float y,
- float maxLineWidth,
- Justification horizontalLayout,
- float leading = 0.0f);
-
- /** Tries to fit some text within a given space.
-
- This does its best to make the given text readable within the specified rectangle,
- so it's useful for labelling things.
-
- If the text is too big, it'll be squashed horizontally or broken over multiple lines
- if the maximumLinesToUse value allows this. If the text just won't fit into the space,
- it'll cram as much as possible in there, and put some ellipsis at the end to show that
- it's been truncated.
-
- A Justification parameter lets you specify how the text is laid out within the rectangle,
- both horizontally and vertically.
-
- The minimumHorizontalScale parameter specifies how much the text can be squashed horizontally
- to try to squeeze it into the space. If you don't want any horizontal scaling to occur, you
- can set this value to 1.0f. Pass 0 if you want it to use the default value.
-
- @see Graphics::drawFittedText
- */
- void addFittedText (const Font& font,
- const String& text,
- float x, float y, float width, float height,
- Justification layout,
- int maximumLinesToUse,
- float minimumHorizontalScale = 0.0f);
-
- /** Appends another glyph arrangement to this one. */
- void addGlyphArrangement (const GlyphArrangement&);
-
- /** Appends a custom glyph to the arrangement. */
- void addGlyph (const PositionedGlyph&);
-
- //==============================================================================
- /** Draws this glyph arrangement to a graphics context.
-
- This uses cached bitmaps so is much faster than the draw (Graphics&, AffineTransform)
- method, which renders the glyphs as filled vectors.
- */
- void draw (const Graphics&) const;
-
- /** Draws this glyph arrangement to a graphics context.
-
- This renders the paths as filled vectors, so is far slower than the draw (Graphics&)
- method for non-transformed arrangements.
- */
- void draw (const Graphics&, AffineTransform) const;
-
- /** Converts the set of glyphs into a path.
- @param path the glyphs' outlines will be appended to this path
- */
- void createPath (Path& path) const;
-
- /** Looks for a glyph that contains the given coordinate.
- @returns the index of the glyph, or -1 if none were found.
- */
- int findGlyphIndexAt (float x, float y) const;
-
- //==============================================================================
- /** Finds the smallest rectangle that will enclose a subset of the glyphs.
-
-
- @param startIndex the first glyph to test
- @param numGlyphs the number of glyphs to include; if this is < 0, all glyphs after
- startIndex will be included
- @param includeWhitespace if true, the extent of any whitespace characters will also
- be taken into account
- */
- Rectangle<float> getBoundingBox (int startIndex, int numGlyphs, bool includeWhitespace) const;
-
- /** Shifts a set of glyphs by a given amount.
-
- @param startIndex the first glyph to transform
- @param numGlyphs the number of glyphs to move; if this is < 0, all glyphs after
- startIndex will be used
- @param deltaX the amount to add to their x-positions
- @param deltaY the amount to add to their y-positions
- */
- void moveRangeOfGlyphs (int startIndex, int numGlyphs,
- float deltaX, float deltaY);
-
- /** Removes a set of glyphs from the arrangement.
-
- @param startIndex the first glyph to remove
- @param numGlyphs the number of glyphs to remove; if this is < 0, all glyphs after
- startIndex will be deleted
- */
- void removeRangeOfGlyphs (int startIndex, int numGlyphs);
-
- /** Expands or compresses a set of glyphs horizontally.
-
- @param startIndex the first glyph to transform
- @param numGlyphs the number of glyphs to stretch; if this is < 0, all glyphs after
- startIndex will be used
- @param horizontalScaleFactor how much to scale their horizontal width by
- */
- void stretchRangeOfGlyphs (int startIndex, int numGlyphs,
- float horizontalScaleFactor);
-
- /** Justifies a set of glyphs within a given space.
-
- This moves the glyphs as a block so that the whole thing is located within the
- given rectangle with the specified layout.
-
- If the Justification::horizontallyJustified flag is specified, each line will
- be stretched out to fill the specified width.
- */
- void justifyGlyphs (int startIndex, int numGlyphs,
- float x, float y, float width, float height,
- Justification justification);
-
-
- private:
- //==============================================================================
- Array<PositionedGlyph> glyphs;
-
- int insertEllipsis (const Font&, float maxXPos, int startIndex, int endIndex);
- int fitLineIntoSpace (int start, int numGlyphs, float x, float y, float w, float h, const Font&,
- Justification, float minimumHorizontalScale);
- void spreadOutLine (int start, int numGlyphs, float targetWidth);
- void splitLines (const String&, Font, int start, float x, float y, float w, float h, int maxLines,
- float lineWidth, Justification, float minimumHorizontalScale);
- void addLinesWithLineBreaks (const String&, const Font&, float x, float y, float width, float height, Justification);
- void drawGlyphUnderline (const Graphics&, const PositionedGlyph&, int, AffineTransform) const;
-
- JUCE_LEAK_DETECTOR (GlyphArrangement)
- };
-
- } // namespace juce
|