22#include <harfbuzz/hb.h>
23#include <harfbuzz/hb-ft.h>
30#include FT_SFNT_NAMES_H
31#include FT_TRUETYPE_TABLES_H
59 TT_OS2* os2 =
reinterpret_cast<TT_OS2*
>( FT_Get_Sfnt_Table(
m_face, FT_SFNT_OS2 ) );
66 if( os2->fsType & FT_FSTYPE_BITMAP_EMBEDDING_ONLY )
73 FT_UShort embeddingBits = os2->fsType & 0x000F;
76 if( embeddingBits == FT_FSTYPE_INSTALLABLE_EMBEDDING )
80 if( embeddingBits & FT_FSTYPE_EDITABLE_EMBEDDING )
84 if( embeddingBits & FT_FSTYPE_PREVIEW_AND_PRINT_EMBEDDING )
93 const std::vector<wxString>* aEmbeddedFiles,
94 bool aForDrawingSheet )
96 std::unique_ptr<OUTLINE_FONT> font = std::make_unique<OUTLINE_FONT>();
103 fc::FF_RESULT retval =
Fontconfig()->
FindFont( aFontName, fontFile, faceIndex, aBold, aItalic,
106 if( retval == fc::FF_RESULT::FF_ERROR )
109 if( retval == fc::FF_RESULT::FF_MISSING_BOLD || retval == fc::FF_RESULT::FF_MISSING_BOLD_ITAL )
112 if( retval == fc::FF_RESULT::FF_MISSING_ITAL || retval == fc::FF_RESULT::FF_MISSING_BOLD_ITAL )
115 if( font->loadFace( fontFile, faceIndex ) != 0 )
118 font->m_fontName = aFontName;
119 font->m_fontFileName = fontFile;
120 font->m_forDrawingSheet = aForDrawingSheet;
122 return font.release();
130 FT_Error e = FT_New_Face(
m_freeType, aFontFileName.mb_str( wxConvUTF8 ), aFaceIndex, &
m_face );
152 if( FT_Select_Charmap( aFace, FT_ENCODING_UNICODE ) == 0 )
157 static const FT_ULong probes[] = {
'A',
'a',
'0',
' ' };
159 for( FT_ULong codepoint : probes )
161 if( FT_Get_Char_Index( aFace, codepoint ) != 0 )
168 if( FT_Select_Charmap( aFace, FT_ENCODING_MS_SYMBOL ) == 0 )
172 FT_Select_Charmap( aFace, FT_ENCODING_UNICODE );
178 double glyphToFontHeight = 1.0;
183 return aFontMetrics.
GetInterline( aGlyphHeight * glyphToFontHeight );
191 case FT_ORIENTATION_TRUETYPE:
return c.
m_Winding == 1;
192 case FT_ORIENTATION_POSTSCRIPT:
return c.
m_Winding == -1;
193 default:
return false;
211 for(
const std::unique_ptr<KIFONT::GLYPH>& glyph : aGlyphs )
213 BOX2D bbox = glyph->BoundingBox();
216 if( minX > bbox.
GetX() )
219 if( minY > bbox.
GetY() )
237 const wxString& aText,
const VECTOR2I& aPosition,
239 const METRICS& aFontMetrics )
const
241 wxArrayString strings;
242 std::vector<VECTOR2I> positions;
243 std::vector<VECTOR2I> extents;
249 getLinePositions( aText, aPosition, strings, positions, extents, aAttrs, aFontMetrics );
251 for(
size_t i = 0; i < strings.GetCount(); i++ )
253 (void)
drawMarkup(
nullptr, aGlyphs, strings.Item( i ), positions[i], aAttrs.
m_Size,
260 const wxString& aText,
const VECTOR2I& aSize,
262 bool aMirror,
const VECTOR2I& aOrigin,
267 constexpr double TAB_WIDTH = 4 * 0.6;
275 aBBox->
SetEnd( aPosition );
278 for( wxUniChar c : aText )
283 if( !textRun.IsEmpty() )
285 position =
getTextAsGlyphs( aBBox, aGlyphs, textRun, aSize, position, aAngle,
286 aMirror, aOrigin, aTextStyle );
290 int tabWidth =
KiROUND( aSize.
x * TAB_WIDTH );
291 int currentIntrusion = ( position.
x - aOrigin.
x ) % tabWidth;
293 position.
x += tabWidth - currentIntrusion;
301 if( !textRun.IsEmpty() )
303 position =
getTextAsGlyphs( aBBox, aGlyphs, textRun, aSize, position, aAngle, aMirror,
304 aOrigin, aTextStyle );
312 const wxString& aText,
const VECTOR2I& aSize,
314 bool aMirror,
const VECTOR2I& aOrigin,
320 aOrigin, aTextStyle );
362 static std::unordered_map<HARFBUZZ_CACHE_KEY, HARFBUZZ_CACHE_ENTRY> s_harfbuzzCache;
364 std::string textUtf8 =
UTF8( aText );
371 hb_buffer_t* buf = hb_buffer_create();
372 hb_buffer_add_utf8( buf, textUtf8.c_str(), -1, 0, -1 );
373 hb_buffer_guess_segment_properties( buf );
376 hb_font_t* referencedFont = hb_ft_font_create_referenced( aFace );
378 hb_shape( referencedFont, buf,
nullptr, 0 );
380 unsigned int glyphCount;
381 hb_glyph_info_t* glyphInfo = hb_buffer_get_glyph_infos( buf, &glyphCount );
382 hb_glyph_position_t* glyphPos = hb_buffer_get_glyph_positions( buf, &glyphCount );
384 entry.
m_GlyphInfo.assign( glyphInfo, glyphInfo + glyphCount );
388 hb_buffer_destroy( buf );
389 hb_font_destroy( referencedFont );
437 std::vector<std::unique_ptr<GLYPH>>* aGlyphs,
438 const wxString& aText,
const VECTOR2I& aSize,
440 bool aMirror,
const VECTOR2I& aOrigin,
456 unsigned int glyphCount =
static_cast<unsigned int>( hbShape.
m_GlyphInfo.size() );
457 const hb_glyph_info_t* glyphInfo = hbShape.
m_GlyphInfo.data();
466 aGlyphs->reserve( glyphCount );
470 static std::unordered_map<GLYPH_CACHE_KEY, GLYPH_DATA> s_glyphCache;
472 for(
unsigned int i = 0; i < glyphCount; i++ )
486 const float angle = (float)( -
M_PI * 12.0f ) / 180.0f;
487 matrix.xx = (FT_Fixed) ( cos( angle ) * 0x10000L );
488 matrix.xy = (FT_Fixed) ( -sin( angle ) * 0x10000L );
489 matrix.yx = (FT_Fixed) ( 0 * 0x10000L );
490 matrix.yy = (FT_Fixed) ( 1 * 0x10000L );
492 FT_Set_Transform( face, &matrix,
nullptr );
495 FT_Load_Glyph( face, glyphInfo[i].codepoint, FT_LOAD_NO_BITMAP );
498 FT_Outline_Embolden( &face->glyph->outline, 1 << 6 );
505 BOX2D tofuBox( { scaler * 0.03, 0.0 },
506 { hb_advance - scaler * 0.02, scaler * 0.72 } );
513 outline.
m_Points.push_back( tofuBox.GetPosition() );
514 outline.
m_Points.push_back( { tofuBox.GetSize().x, tofuBox.GetPosition().y } );
515 outline.
m_Points.push_back( tofuBox.GetSize() );
516 outline.
m_Points.push_back( { tofuBox.GetPosition().x, tofuBox.GetSize().y } );
517 glyphData.
m_Contours.push_back( std::move( outline ) );
520 tofuBox.Move( { scaler * 0.06, scaler * 0.06 } );
521 tofuBox.SetSize( { tofuBox.GetWidth() - scaler * 0.06,
522 tofuBox.GetHeight() - scaler * 0.06 } );
525 hole.
m_Points.push_back( tofuBox.GetPosition() );
526 hole.
m_Points.push_back( { tofuBox.GetSize().x, tofuBox.GetPosition().y } );
527 hole.
m_Points.push_back( tofuBox.GetSize() );
528 hole.
m_Points.push_back( { tofuBox.GetPosition().x, tofuBox.GetSize().y } );
529 glyphData.
m_Contours.push_back( std::move( hole ) );
533 std::unique_ptr<OUTLINE_GLYPH> glyph = std::make_unique<OUTLINE_GLYPH>();
534 std::vector<SHAPE_LINE_CHAIN> holes;
538 std::vector<VECTOR2D> points = c.
m_Points;
556 pt.
x = aOrigin.
x - ( pt.
x - aOrigin.
x );
567 holes.push_back( std::move( shape ) );
569 glyph->AddOutline( std::move( shape ) );
574 bool added_hole =
false;
576 if( hole.PointCount() )
578 for(
int ii = 0; ii < glyph->OutlineCount(); ++ii )
580 if( glyph->Outline( ii ).PointInside( hole.GetPoint( 0 ) ) )
582 glyph->AddHole( std::move( hole ), ii );
592 glyph->AddOutline( std::move( hole ) );
598 glyph->CacheTriangulation();
606 aGlyphs->push_back( std::move( glyph ) );
609 const hb_glyph_position_t& pos = glyphPos[i];
619 aBBox->
Merge( aPosition -
VECTOR2I( 0, ascender * abs( scaleFactor.
y ) ) );
627#undef OUTLINEFONT_RENDER_AS_PIXELS
628#ifdef OUTLINEFONT_RENDER_AS_PIXELS
632void OUTLINE_FONT::RenderToOpenGLCanvas(
KIGFX::OPENGL_GAL& aGal,
const wxString& aString,
634 const EDA_ANGLE& aOrientation,
bool aIsMirrored )
const
636 hb_buffer_t* buf = hb_buffer_create();
637 hb_buffer_add_utf8( buf,
UTF8( aString ).c_str(), -1, 0, -1 );
640 hb_buffer_guess_segment_properties( buf );
642 unsigned int glyphCount;
643 hb_glyph_info_t* glyphInfo = hb_buffer_get_glyph_infos( buf, &glyphCount );
644 hb_glyph_position_t* glyphPos = hb_buffer_get_glyph_positions( buf, &glyphCount );
648 hb_font_t* referencedFont = hb_ft_font_create_referenced(
m_face );
650 hb_shape( referencedFont, buf,
nullptr, 0 );
652 const double mirror_factor = ( aIsMirrored ? 1 : -1 );
653 const double x_scaleFactor = mirror_factor * aGlyphSize.
x / mScaler;
654 const double y_scaleFactor = aGlyphSize.
y / mScaler;
656 hb_position_t cursor_x = 0;
657 hb_position_t cursor_y = 0;
659 for(
unsigned int i = 0; i < glyphCount; i++ )
661 const hb_glyph_position_t& pos = glyphPos[i];
662 int codepoint = glyphInfo[i].codepoint;
664 FT_Error e = FT_Load_Glyph(
m_face, codepoint, FT_LOAD_DEFAULT );
668 e = FT_Get_Glyph(
m_face->glyph, &glyph );
671 wxPoint pt( aPosition );
672 pt.x += ( cursor_x >> 6 ) * x_scaleFactor;
673 pt.y += ( cursor_y >> 6 ) * y_scaleFactor;
675 cursor_x += pos.x_advance;
676 cursor_y += pos.y_advance;
679 hb_buffer_destroy( buf );
constexpr BOX2I KiROUND(const BOX2D &aBoxD)
constexpr void SetOrigin(const Vec &pos)
constexpr BOX2< Vec > & Normalize()
Ensure that the height and width are positive.
constexpr coord_type GetY() const
constexpr coord_type GetX() const
constexpr BOX2< Vec > & Merge(const BOX2< Vec > &aRect)
Modify the position and size of the rectangle in order to contain aRect.
constexpr coord_type GetRight() const
constexpr void SetEnd(coord_type x, coord_type y)
constexpr coord_type GetBottom() const
void getLinePositions(const wxString &aText, const VECTOR2I &aPosition, wxArrayString &aTextLines, std::vector< VECTOR2I > &aPositions, std::vector< VECTOR2I > &aExtents, const TEXT_ATTRIBUTES &aAttrs, const METRICS &aFontMetrics) const
VECTOR2I drawMarkup(BOX2I *aBoundingBox, std::vector< std::unique_ptr< GLYPH > > *aGlyphs, const wxString &aText, const VECTOR2I &aPosition, const VECTOR2I &aSize, const EDA_ANGLE &aAngle, bool aMirror, const VECTOR2I &aOrigin, TEXT_STYLE_FLAGS aTextStyle, const METRICS &aFontMetrics, std::optional< VECTOR2I > aMousePos=std::nullopt, wxString *aActiveUrl=nullptr) const
double GetInterline(double aFontHeight) const
bool OutlineToSegments(std::vector< CONTOUR > *aContours)
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
static void SelectCharmap(FT_Face aFace)
Select the charmap used to map characters to glyphs for aFace.
static std::mutex m_freeTypeMutex
Mutex for freetype access, FT_Library and FT_Face are not thread safe.
double GetInterline(double aGlyphHeight, const METRICS &aFontMetrics) const override
Compute the distance (interline) between 2 lines of text (for multiline texts).
static FT_Library m_freeType
BOX2I getBoundingBox(const std::vector< std::unique_ptr< GLYPH > > &aGlyphs) const
FT_Error loadFace(const wxString &aFontFileName, int aFaceIndex)
static constexpr double m_superscriptVerticalOffset
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
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 override
Convert text string to an array of GLYPHs.
const FT_Face & GetFace() const
int faceSize(int aSize) const
void GetLinesAsGlyphs(std::vector< std::unique_ptr< GLYPH > > *aGlyphs, const wxString &aText, const VECTOR2I &aPosition, const TEXT_ATTRIBUTES &aAttrs, const METRICS &aFontMetrics) const
static OUTLINE_FONT * LoadFont(const wxString &aFontFileName, bool aBold, bool aItalic, const std::vector< wxString > *aEmbeddedFiles, bool aForDrawingSheet)
Load an outline font.
static constexpr double m_outlineFontSizeCompensation
EMBEDDING_PERMISSION GetEmbeddingPermission() const
int subscriptSize() const
static constexpr double m_subscriptVerticalOffset
OpenGL implementation of the Graphics Abstraction Layer.
Represent a polyline containing arcs as well as line segments: A chain of connected line and/or arc s...
void SetClosed(bool aClosed)
Mark the line chain as closed (i.e.
void ReservePoints(size_t aSize)
Allocate a number of points all at once (for performance).
void Append(int aX, int aY, bool aAllowDuplication=false)
Append a new point at the end of the line chain.
An 8 bit string that is assuredly encoded in UTF8, and supplies special conversion support to and fro...
FF_RESULT FindFont(const wxString &aFontName, wxString &aFontFile, int &aFaceIndex, bool aBold, bool aItalic, const std::vector< wxString > *aEmbeddedFiles=nullptr)
Given a fully-qualified font name ("Times:Bold:Italic") find the closest matching font and return its...
unsigned int TEXT_STYLE_FLAGS
bool IsSuperscript(TEXT_STYLE_FLAGS aFlags)
bool IsSubscript(TEXT_STYLE_FLAGS aFlags)
FONTCONFIG * Fontconfig()
static constexpr std::size_t hash_val(const Types &... args)
constexpr int GLYPH_RESOLUTION
constexpr double GLYPH_SIZE_SCALER
static bool contourIsHole(const CONTOUR &c)
static bool contourIsFilled(const CONTOUR &c)
static const HARFBUZZ_CACHE_ENTRY & getHarfbuzzShape(FT_Face aFace, const wxString &aText, int aScaler)
bool operator==(const GLYPH_CACHE_KEY &rhs) const
std::vector< hb_glyph_info_t > m_GlyphInfo
std::vector< hb_glyph_position_t > m_GlyphPositions
bool operator==(const HARFBUZZ_CACHE_KEY &rhs) const
std::vector< VECTOR2D > m_Points
FT_Orientation m_Orientation
std::vector< CONTOUR > m_Contours
std::vector< std::unique_ptr< SHAPE_POLY_SET::TRIANGULATED_POLYGON > > m_TriangulationData
std::size_t operator()(const GLYPH_CACHE_KEY &k) const
std::size_t operator()(const HARFBUZZ_CACHE_KEY &k) const
void RotatePoint(int *pX, int *pY, const EDA_ANGLE &aAngle)
Calculate the new point of coord coord pX, pY, for a rotation center 0, 0.
VECTOR2< int32_t > VECTOR2I
VECTOR2< double > VECTOR2D