2022-01-01 01:21:03 +00:00
|
|
|
/*
|
|
|
|
* This program source code file is part of KICAD, a free EDA CAD application.
|
|
|
|
*
|
|
|
|
* Copyright (C) 2021 Ola Rinta-Koski
|
2023-08-06 19:20:53 +00:00
|
|
|
* Copyright (C) 2021-2023 Kicad Developers, see AUTHORS.txt for contributors.
|
2022-01-01 01:21:03 +00:00
|
|
|
*
|
|
|
|
* Outline font class
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License
|
|
|
|
* as published by the Free Software Foundation; either version 2
|
|
|
|
* of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, you may find one here:
|
|
|
|
* http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
|
|
|
|
* or you may search the http://www.gnu.org website for the version 2 license,
|
|
|
|
* or you may write to the Free Software Foundation, Inc.,
|
|
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef OUTLINE_FONT_H_
|
|
|
|
#define OUTLINE_FONT_H_
|
|
|
|
|
|
|
|
#include <gal/graphics_abstraction_layer.h>
|
|
|
|
#include <geometry/shape_poly_set.h>
|
2022-01-08 17:35:40 +00:00
|
|
|
#ifdef _MSC_VER
|
|
|
|
#include <ft2build.h>
|
|
|
|
#else
|
2022-01-01 01:21:03 +00:00
|
|
|
#include <freetype2/ft2build.h>
|
2022-01-08 17:35:40 +00:00
|
|
|
#endif
|
2022-01-01 01:21:03 +00:00
|
|
|
#include FT_FREETYPE_H
|
|
|
|
#include FT_OUTLINE_H
|
|
|
|
//#include <gal/opengl/opengl_freetype.h>
|
|
|
|
#include <font/font.h>
|
|
|
|
#include <font/glyph.h>
|
|
|
|
#include <font/outline_decomposer.h>
|
|
|
|
|
2023-02-17 12:24:22 +00:00
|
|
|
#include <mutex>
|
|
|
|
|
2022-01-01 01:21:03 +00:00
|
|
|
namespace KIFONT
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* Class OUTLINE_FONT implements outline font drawing.
|
|
|
|
*/
|
|
|
|
class OUTLINE_FONT : public FONT
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
OUTLINE_FONT();
|
|
|
|
|
2022-03-07 14:01:37 +00:00
|
|
|
static wxString FontConfigVersion();
|
|
|
|
|
2022-01-21 13:14:23 +00:00
|
|
|
static wxString FreeTypeVersion();
|
|
|
|
|
2022-03-07 14:01:37 +00:00
|
|
|
static wxString HarfBuzzVersion();
|
|
|
|
|
|
|
|
static wxString FontLibraryVersion();
|
|
|
|
|
2022-01-01 01:21:03 +00:00
|
|
|
bool IsOutline() const override { return true; }
|
|
|
|
|
|
|
|
bool IsBold() const override
|
|
|
|
{
|
2023-01-25 21:10:31 +00:00
|
|
|
return m_face && ( m_fakeBold || ( m_face->style_flags & FT_STYLE_FLAG_BOLD ) );
|
2022-01-01 01:21:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool IsItalic() const override
|
|
|
|
{
|
2023-01-25 21:10:31 +00:00
|
|
|
return m_face && ( m_fakeItal || ( m_face->style_flags & FT_STYLE_FLAG_ITALIC ) );
|
|
|
|
}
|
|
|
|
|
|
|
|
void SetFakeBold()
|
|
|
|
{
|
|
|
|
m_fakeBold = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SetFakeItal()
|
|
|
|
{
|
|
|
|
m_fakeItal = true;
|
2022-01-01 01:21:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Load an outline font. TrueType (.ttf) and OpenType (.otf) are supported.
|
|
|
|
* @param aFontFileName is the (platform-specific) fully qualified name of the font file
|
|
|
|
*/
|
|
|
|
static OUTLINE_FONT* LoadFont( const wxString& aFontFileName, bool aBold, bool aItalic );
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Compute the distance (interline) between 2 lines of text (for multiline texts). This is
|
|
|
|
* the distance between baselines, not the space between line bounding boxes.
|
|
|
|
*/
|
2023-08-06 19:20:53 +00:00
|
|
|
double GetInterline( double aGlyphHeight, const METRICS& aFontMetrics ) const override;
|
2022-01-01 01:21:03 +00:00
|
|
|
|
2022-01-07 17:42:43 +00:00
|
|
|
VECTOR2I GetTextAsGlyphs( BOX2I* aBoundingBox, std::vector<std::unique_ptr<GLYPH>>* aGlyphs,
|
2022-01-25 22:33:37 +00:00
|
|
|
const wxString& aText, const VECTOR2I& aSize,
|
|
|
|
const VECTOR2I& aPosition, const EDA_ANGLE& aAngle, bool aMirror,
|
|
|
|
const VECTOR2I& aOrigin, TEXT_STYLE_FLAGS aTextStyle ) const override;
|
2022-01-01 01:21:03 +00:00
|
|
|
|
2022-01-25 22:33:37 +00:00
|
|
|
void GetLinesAsGlyphs( std::vector<std::unique_ptr<GLYPH>>* aGlyphs, const wxString& aText,
|
2023-08-06 19:20:53 +00:00
|
|
|
const VECTOR2I& aPosition, const TEXT_ATTRIBUTES& aAttrs,
|
|
|
|
const METRICS& aFontMetrics ) const;
|
2022-01-01 01:21:03 +00:00
|
|
|
|
|
|
|
const FT_Face& GetFace() const { return m_face; }
|
|
|
|
|
|
|
|
#if 0
|
2022-01-25 22:33:37 +00:00
|
|
|
void RenderToOpenGLCanvas( KIGFX::OPENGL_FREETYPE& aTarget, const wxString& aString,
|
2022-01-04 23:00:00 +00:00
|
|
|
const VECTOR2D& aSize, const wxPoint& aPosition,
|
|
|
|
const EDA_ANGLE& aAngle, bool aMirror ) const;
|
2022-01-01 01:21:03 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
protected:
|
2023-03-13 16:35:28 +00:00
|
|
|
FT_Error loadFace( const wxString& aFontFileName, int aFaceIndex );
|
2022-01-01 01:21:03 +00:00
|
|
|
|
|
|
|
BOX2I getBoundingBox( const std::vector<std::unique_ptr<GLYPH>>& aGlyphs ) const;
|
|
|
|
|
2022-04-20 11:00:19 +00:00
|
|
|
VECTOR2I getTextAsGlyphs( BOX2I* aBoundingBox, std::vector<std::unique_ptr<GLYPH>>* aGlyphs,
|
|
|
|
const wxString& aText, const VECTOR2I& aSize,
|
|
|
|
const VECTOR2I& aPosition, const EDA_ANGLE& aAngle, bool aMirror,
|
|
|
|
const VECTOR2I& aOrigin, TEXT_STYLE_FLAGS aTextStyle ) const;
|
|
|
|
|
2023-02-23 22:58:37 +00:00
|
|
|
private:
|
2023-04-17 14:15:29 +00:00
|
|
|
VECTOR2I getTextAsGlyphsUnlocked( BOX2I* aBoundingBox,
|
|
|
|
std::vector<std::unique_ptr<GLYPH>>* aGlyphs,
|
|
|
|
const wxString& aText, const VECTOR2I& aSize,
|
|
|
|
const VECTOR2I& aPosition, const EDA_ANGLE& aAngle,
|
|
|
|
bool aMirror, const VECTOR2I& aOrigin,
|
|
|
|
TEXT_STYLE_FLAGS aTextStyle ) const;
|
2023-02-23 22:58:37 +00:00
|
|
|
|
2022-01-01 01:21:03 +00:00
|
|
|
private:
|
|
|
|
// FreeType variables
|
2023-02-17 14:39:52 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Mutex for freetype access, FT_Library and FT_Face are not thread safe
|
|
|
|
*/
|
2023-02-17 12:24:22 +00:00
|
|
|
static std::mutex m_freeTypeMutex;
|
2022-01-01 01:21:03 +00:00
|
|
|
static FT_Library m_freeType;
|
|
|
|
FT_Face m_face;
|
2023-02-17 14:39:52 +00:00
|
|
|
|
2022-01-01 01:21:03 +00:00
|
|
|
const int m_faceSize;
|
2023-01-25 21:10:31 +00:00
|
|
|
bool m_fakeBold;
|
|
|
|
bool m_fakeItal;
|
2022-01-01 01:21:03 +00:00
|
|
|
|
|
|
|
// cache for glyphs converted to straight segments
|
|
|
|
// key is glyph index (FT_GlyphSlot field glyph_index)
|
|
|
|
std::map<unsigned int, GLYPH_POINTS_LIST> m_contourCache;
|
2022-01-15 12:51:09 +00:00
|
|
|
|
|
|
|
// The height of the KiCad stroke font is the distance between stroke endpoints for a vertical
|
|
|
|
// line of cap-height. So the cap-height of the font is actually stroke-width taller than its
|
|
|
|
// height.
|
|
|
|
// Outline fonts are normally scaled on full-height (including ascenders and descenders), so we
|
|
|
|
// need to compensate to keep them from being much smaller than their stroked counterparts.
|
|
|
|
static constexpr double m_outlineFontSizeCompensation = 1.4;
|
|
|
|
|
2022-02-07 00:24:08 +00:00
|
|
|
// FT_Set_Char_Size() gets character width and height specified in 1/64ths of a point
|
2022-01-15 12:51:09 +00:00
|
|
|
static constexpr int m_charSizeScaler = 64;
|
|
|
|
|
|
|
|
// The KiCad stroke font uses a subscript/superscript size ratio of 0.7. This ratio is also
|
|
|
|
// commonly used in LaTeX, but fonts with designed-in subscript and superscript glyphs are more
|
|
|
|
// likely to use 0.58.
|
|
|
|
// For auto-generated subscript and superscript glyphs in outline fonts we split the difference
|
|
|
|
// with 0.64.
|
|
|
|
static constexpr double m_subscriptSuperscriptSize = 0.64;
|
|
|
|
|
2022-08-27 19:57:34 +00:00
|
|
|
static constexpr double m_underlineOffsetScaler = -0.16;
|
|
|
|
|
2022-01-15 12:51:09 +00:00
|
|
|
int faceSize( int aSize ) const
|
|
|
|
{
|
|
|
|
return aSize * m_charSizeScaler * m_outlineFontSizeCompensation;
|
|
|
|
};
|
|
|
|
int faceSize() const { return faceSize( m_faceSize ); }
|
|
|
|
|
|
|
|
// also for superscripts
|
|
|
|
int subscriptSize( int aSize ) const
|
|
|
|
{
|
|
|
|
return KiROUND( faceSize( aSize ) * m_subscriptSuperscriptSize );
|
|
|
|
}
|
|
|
|
int subscriptSize() const { return subscriptSize( m_faceSize ); }
|
|
|
|
|
|
|
|
static constexpr double m_subscriptVerticalOffset = -0.25;
|
|
|
|
static constexpr double m_superscriptVerticalOffset = 0.45;
|
2022-01-01 01:21:03 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
} //namespace KIFONT
|
|
|
|
|
|
|
|
#endif // OUTLINE_FONT_H_
|