2001-09-12 22:21:06 +02:00
|
|
|
/*
|
|
|
|
* FreeType font engine interface
|
|
|
|
*
|
|
|
|
* Copyright 2001 Huw D M Davies for CodeWeavers.
|
2006-09-20 17:18:51 +02:00
|
|
|
* Copyright 2006 Dmitry Timoshkov for CodeWeavers.
|
2001-09-12 22:21:06 +02:00
|
|
|
*
|
|
|
|
* This file contains the WineEng* functions.
|
2002-03-10 00:29:33 +01:00
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library 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
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with this library; if not, write to the Free Software
|
2006-05-18 14:49:52 +02:00
|
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
|
2001-09-12 22:21:06 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
2004-01-15 01:35:38 +01:00
|
|
|
#include "wine/port.h"
|
2001-09-12 22:21:06 +02:00
|
|
|
|
2003-09-06 01:08:26 +02:00
|
|
|
#include <stdarg.h>
|
|
|
|
#include <stdlib.h>
|
2005-05-06 18:22:54 +02:00
|
|
|
#ifdef HAVE_SYS_STAT_H
|
|
|
|
# include <sys/stat.h>
|
|
|
|
#endif
|
2006-12-06 21:31:53 +01:00
|
|
|
#ifdef HAVE_SYS_MMAN_H
|
|
|
|
# include <sys/mman.h>
|
|
|
|
#endif
|
2003-09-06 01:08:26 +02:00
|
|
|
#include <string.h>
|
2008-10-19 18:55:13 +02:00
|
|
|
#ifdef HAVE_DIRENT_H
|
|
|
|
# include <dirent.h>
|
|
|
|
#endif
|
2003-09-06 01:08:26 +02:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <assert.h>
|
|
|
|
|
2006-12-21 16:21:13 +01:00
|
|
|
#ifdef HAVE_CARBON_CARBON_H
|
|
|
|
#define LoadResource __carbon_LoadResource
|
|
|
|
#define CompareString __carbon_CompareString
|
|
|
|
#define GetCurrentThread __carbon_GetCurrentThread
|
|
|
|
#define GetCurrentProcess __carbon_GetCurrentProcess
|
|
|
|
#define AnimatePalette __carbon_AnimatePalette
|
|
|
|
#define EqualRgn __carbon_EqualRgn
|
|
|
|
#define FillRgn __carbon_FillRgn
|
|
|
|
#define FrameRgn __carbon_FrameRgn
|
|
|
|
#define GetPixel __carbon_GetPixel
|
|
|
|
#define InvertRgn __carbon_InvertRgn
|
|
|
|
#define LineTo __carbon_LineTo
|
|
|
|
#define OffsetRgn __carbon_OffsetRgn
|
|
|
|
#define PaintRgn __carbon_PaintRgn
|
|
|
|
#define Polygon __carbon_Polygon
|
|
|
|
#define ResizePalette __carbon_ResizePalette
|
|
|
|
#define SetRectRgn __carbon_SetRectRgn
|
|
|
|
#include <Carbon/Carbon.h>
|
|
|
|
#undef LoadResource
|
|
|
|
#undef CompareString
|
|
|
|
#undef GetCurrentThread
|
|
|
|
#undef _CDECL
|
|
|
|
#undef GetCurrentProcess
|
|
|
|
#undef AnimatePalette
|
|
|
|
#undef EqualRgn
|
|
|
|
#undef FillRgn
|
|
|
|
#undef FrameRgn
|
|
|
|
#undef GetPixel
|
|
|
|
#undef InvertRgn
|
|
|
|
#undef LineTo
|
|
|
|
#undef OffsetRgn
|
|
|
|
#undef PaintRgn
|
|
|
|
#undef Polygon
|
|
|
|
#undef ResizePalette
|
|
|
|
#undef SetRectRgn
|
|
|
|
#endif /* HAVE_CARBON_CARBON_H */
|
|
|
|
|
2013-11-30 11:22:25 +01:00
|
|
|
#ifdef HAVE_FT2BUILD_H
|
|
|
|
#include <ft2build.h>
|
|
|
|
#include FT_FREETYPE_H
|
|
|
|
#include FT_GLYPH_H
|
|
|
|
#include FT_TYPES_H
|
|
|
|
#include FT_TRUETYPE_TABLES_H
|
|
|
|
#include FT_SFNT_NAMES_H
|
|
|
|
#include FT_TRUETYPE_IDS_H
|
|
|
|
#include FT_OUTLINE_H
|
|
|
|
#include FT_TRIGONOMETRY_H
|
|
|
|
#include FT_MODULE_H
|
|
|
|
#include FT_WINFONTS_H
|
|
|
|
#ifdef FT_LCD_FILTER_H
|
|
|
|
#include FT_LCD_FILTER_H
|
|
|
|
#endif
|
|
|
|
#endif /* HAVE_FT2BUILD_H */
|
|
|
|
|
2001-09-12 22:21:06 +02:00
|
|
|
#include "windef.h"
|
2003-09-06 01:08:26 +02:00
|
|
|
#include "winbase.h"
|
2006-04-04 13:12:41 +02:00
|
|
|
#include "winternl.h"
|
2001-09-12 22:21:06 +02:00
|
|
|
#include "winerror.h"
|
|
|
|
#include "winreg.h"
|
|
|
|
#include "wingdi.h"
|
2004-01-15 01:35:38 +01:00
|
|
|
#include "gdi_private.h"
|
2001-09-12 22:21:06 +02:00
|
|
|
#include "wine/unicode.h"
|
2002-04-03 22:02:39 +02:00
|
|
|
#include "wine/debug.h"
|
2004-08-04 20:16:13 +02:00
|
|
|
#include "wine/list.h"
|
2001-09-12 22:21:06 +02:00
|
|
|
|
2011-11-03 14:11:35 +01:00
|
|
|
#include "resource.h"
|
|
|
|
|
2002-03-10 00:29:33 +01:00
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(font);
|
2001-09-12 22:21:06 +02:00
|
|
|
|
|
|
|
#ifdef HAVE_FREETYPE
|
|
|
|
|
2006-02-22 13:24:25 +01:00
|
|
|
#ifndef HAVE_FT_TRUETYPEENGINETYPE
|
|
|
|
typedef enum
|
|
|
|
{
|
|
|
|
FT_TRUETYPE_ENGINE_TYPE_NONE = 0,
|
|
|
|
FT_TRUETYPE_ENGINE_TYPE_UNPATENTED,
|
|
|
|
FT_TRUETYPE_ENGINE_TYPE_PATENTED
|
|
|
|
} FT_TrueTypeEngineType;
|
2006-01-24 11:04:20 +01:00
|
|
|
#endif
|
|
|
|
|
2001-09-12 22:21:06 +02:00
|
|
|
static FT_Library library = 0;
|
2002-08-17 20:34:34 +02:00
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
FT_Int major;
|
|
|
|
FT_Int minor;
|
|
|
|
FT_Int patch;
|
|
|
|
} FT_Version_t;
|
|
|
|
static FT_Version_t FT_Version;
|
2004-06-16 22:06:26 +02:00
|
|
|
static DWORD FT_SimpleVersion;
|
2019-02-08 02:51:32 +01:00
|
|
|
#define FT_VERSION_VALUE(major, minor, patch) (((major) << 16) | ((minor) << 8) | (patch))
|
2001-09-12 22:21:06 +02:00
|
|
|
|
2002-04-03 22:02:39 +02:00
|
|
|
static void *ft_handle = NULL;
|
|
|
|
|
2002-09-23 22:45:57 +02:00
|
|
|
#define MAKE_FUNCPTR(f) static typeof(f) * p##f = NULL
|
|
|
|
MAKE_FUNCPTR(FT_Done_Face);
|
|
|
|
MAKE_FUNCPTR(FT_Get_Char_Index);
|
2011-10-11 11:14:56 +02:00
|
|
|
MAKE_FUNCPTR(FT_Get_First_Char);
|
|
|
|
MAKE_FUNCPTR(FT_Get_Next_Char);
|
2006-04-05 13:45:25 +02:00
|
|
|
MAKE_FUNCPTR(FT_Get_Sfnt_Name);
|
|
|
|
MAKE_FUNCPTR(FT_Get_Sfnt_Name_Count);
|
2002-09-23 22:45:57 +02:00
|
|
|
MAKE_FUNCPTR(FT_Get_Sfnt_Table);
|
2011-10-11 11:14:56 +02:00
|
|
|
MAKE_FUNCPTR(FT_Get_WinFNT_Header);
|
2002-09-23 22:45:57 +02:00
|
|
|
MAKE_FUNCPTR(FT_Init_FreeType);
|
2011-10-11 11:14:56 +02:00
|
|
|
MAKE_FUNCPTR(FT_Library_Version);
|
2002-09-23 22:45:57 +02:00
|
|
|
MAKE_FUNCPTR(FT_Load_Glyph);
|
2011-10-10 15:15:56 +02:00
|
|
|
MAKE_FUNCPTR(FT_Load_Sfnt_Table);
|
2003-10-10 02:06:35 +02:00
|
|
|
MAKE_FUNCPTR(FT_Matrix_Multiply);
|
2008-09-03 13:14:17 +02:00
|
|
|
#ifdef FT_MULFIX_INLINED
|
|
|
|
#define pFT_MulFix FT_MULFIX_INLINED
|
|
|
|
#else
|
2002-09-23 22:45:57 +02:00
|
|
|
MAKE_FUNCPTR(FT_MulFix);
|
2008-09-03 13:14:17 +02:00
|
|
|
#endif
|
2002-09-23 22:45:57 +02:00
|
|
|
MAKE_FUNCPTR(FT_New_Face);
|
2006-12-06 21:31:53 +01:00
|
|
|
MAKE_FUNCPTR(FT_New_Memory_Face);
|
2002-09-23 22:45:57 +02:00
|
|
|
MAKE_FUNCPTR(FT_Outline_Get_Bitmap);
|
2013-10-17 14:25:10 +02:00
|
|
|
MAKE_FUNCPTR(FT_Outline_Get_CBox);
|
2002-09-23 22:45:57 +02:00
|
|
|
MAKE_FUNCPTR(FT_Outline_Transform);
|
|
|
|
MAKE_FUNCPTR(FT_Outline_Translate);
|
2011-10-10 15:15:56 +02:00
|
|
|
MAKE_FUNCPTR(FT_Render_Glyph);
|
2007-10-02 05:04:25 +02:00
|
|
|
MAKE_FUNCPTR(FT_Set_Charmap);
|
2002-09-23 22:45:57 +02:00
|
|
|
MAKE_FUNCPTR(FT_Set_Pixel_Sizes);
|
2016-01-13 15:09:08 +01:00
|
|
|
MAKE_FUNCPTR(FT_Vector_Length);
|
2003-10-10 02:06:35 +02:00
|
|
|
MAKE_FUNCPTR(FT_Vector_Transform);
|
2011-10-10 15:15:56 +02:00
|
|
|
MAKE_FUNCPTR(FT_Vector_Unit);
|
2013-10-17 14:25:10 +02:00
|
|
|
static FT_Error (*pFT_Outline_Embolden)(FT_Outline *, FT_Pos);
|
2006-02-22 13:24:25 +01:00
|
|
|
static FT_TrueTypeEngineType (*pFT_Get_TrueType_Engine_Type)(FT_Library);
|
2013-11-30 11:22:25 +01:00
|
|
|
#ifdef FT_LCD_FILTER_H
|
2008-12-23 12:34:01 +01:00
|
|
|
static FT_Error (*pFT_Library_SetLcdFilter)(FT_Library, FT_LcdFilter);
|
|
|
|
#endif
|
2019-02-19 11:25:41 +01:00
|
|
|
static FT_Error (*pFT_Property_Set)(FT_Library, const FT_String *, const FT_String *, const void *);
|
2002-04-03 22:02:39 +02:00
|
|
|
|
2007-07-02 17:31:48 +02:00
|
|
|
#ifdef SONAME_LIBFONTCONFIG
|
2003-11-09 01:30:13 +01:00
|
|
|
#include <fontconfig/fontconfig.h>
|
2012-11-01 12:35:52 +01:00
|
|
|
MAKE_FUNCPTR(FcConfigSubstitute);
|
2018-11-27 16:50:20 +01:00
|
|
|
MAKE_FUNCPTR(FcDefaultSubstitute);
|
2003-11-09 01:30:13 +01:00
|
|
|
MAKE_FUNCPTR(FcFontList);
|
2018-11-27 16:50:20 +01:00
|
|
|
MAKE_FUNCPTR(FcFontMatch);
|
2003-11-09 01:30:13 +01:00
|
|
|
MAKE_FUNCPTR(FcFontSetDestroy);
|
|
|
|
MAKE_FUNCPTR(FcInit);
|
2018-11-27 16:50:20 +01:00
|
|
|
MAKE_FUNCPTR(FcPatternAddString);
|
2003-11-09 01:30:13 +01:00
|
|
|
MAKE_FUNCPTR(FcPatternCreate);
|
|
|
|
MAKE_FUNCPTR(FcPatternDestroy);
|
2006-12-19 22:21:27 +01:00
|
|
|
MAKE_FUNCPTR(FcPatternGetBool);
|
2012-11-01 12:35:52 +01:00
|
|
|
MAKE_FUNCPTR(FcPatternGetInteger);
|
2006-04-28 13:51:36 +02:00
|
|
|
MAKE_FUNCPTR(FcPatternGetString);
|
2018-12-19 12:01:35 +01:00
|
|
|
#ifndef FC_NAMELANG
|
|
|
|
#define FC_NAMELANG "namelang"
|
2003-11-09 01:30:13 +01:00
|
|
|
#endif
|
2018-12-19 12:01:35 +01:00
|
|
|
#ifndef FC_PRGNAME
|
|
|
|
#define FC_PRGNAME "prgname"
|
|
|
|
#endif
|
|
|
|
#endif /* SONAME_LIBFONTCONFIG */
|
2003-11-09 01:30:13 +01:00
|
|
|
|
|
|
|
#undef MAKE_FUNCPTR
|
|
|
|
|
2007-09-20 12:10:31 +02:00
|
|
|
#ifndef FT_MAKE_TAG
|
|
|
|
#define FT_MAKE_TAG( ch0, ch1, ch2, ch3 ) \
|
|
|
|
( ((DWORD)(BYTE)(ch0) << 24) | ((DWORD)(BYTE)(ch1) << 16) | \
|
|
|
|
((DWORD)(BYTE)(ch2) << 8) | (DWORD)(BYTE)(ch3) )
|
|
|
|
#endif
|
|
|
|
|
2004-10-18 21:38:30 +02:00
|
|
|
#ifndef ft_encoding_none
|
2004-09-15 20:03:32 +02:00
|
|
|
#define FT_ENCODING_NONE ft_encoding_none
|
|
|
|
#endif
|
2004-10-18 21:38:30 +02:00
|
|
|
#ifndef ft_encoding_ms_symbol
|
2004-09-15 20:03:32 +02:00
|
|
|
#define FT_ENCODING_MS_SYMBOL ft_encoding_symbol
|
|
|
|
#endif
|
2004-10-18 21:38:30 +02:00
|
|
|
#ifndef ft_encoding_unicode
|
2004-09-15 20:03:32 +02:00
|
|
|
#define FT_ENCODING_UNICODE ft_encoding_unicode
|
|
|
|
#endif
|
2004-10-18 21:38:30 +02:00
|
|
|
#ifndef ft_encoding_apple_roman
|
2004-09-15 20:03:32 +02:00
|
|
|
#define FT_ENCODING_APPLE_ROMAN ft_encoding_apple_roman
|
|
|
|
#endif
|
2003-11-09 01:30:13 +01:00
|
|
|
|
2006-04-04 13:12:41 +02:00
|
|
|
#ifdef WORDS_BIGENDIAN
|
|
|
|
#define GET_BE_WORD(x) (x)
|
2016-08-12 12:14:50 +02:00
|
|
|
#define GET_BE_DWORD(x) (x)
|
2006-04-04 13:12:41 +02:00
|
|
|
#else
|
|
|
|
#define GET_BE_WORD(x) RtlUshortByteSwap(x)
|
2016-08-12 12:14:50 +02:00
|
|
|
#define GET_BE_DWORD(x) RtlUlongByteSwap(x)
|
2006-04-04 13:12:41 +02:00
|
|
|
#endif
|
2002-05-23 04:53:10 +02:00
|
|
|
|
2016-08-12 12:14:49 +02:00
|
|
|
/* 'gasp' flags */
|
|
|
|
#define GASP_GRIDFIT 0x01
|
|
|
|
#define GASP_DOGRAY 0x02
|
|
|
|
|
2004-07-14 23:42:35 +02:00
|
|
|
/* FT_Bitmap_Size gained 3 new elements between FreeType 2.1.4 and 2.1.5
|
|
|
|
So to let this compile on older versions of FreeType we'll define the
|
|
|
|
new structure here. */
|
|
|
|
typedef struct {
|
|
|
|
FT_Short height, width;
|
|
|
|
FT_Pos size, x_ppem, y_ppem;
|
|
|
|
} My_FT_Bitmap_Size;
|
|
|
|
|
2020-10-30 14:31:13 +01:00
|
|
|
typedef struct gdi_font_face Face;
|
2001-09-12 22:21:06 +02:00
|
|
|
|
2020-10-30 14:22:51 +01:00
|
|
|
typedef struct gdi_font_family Family;
|
2001-09-12 22:21:06 +02:00
|
|
|
|
2020-10-29 14:53:18 +01:00
|
|
|
struct font_private_data
|
|
|
|
{
|
2001-09-12 22:21:06 +02:00
|
|
|
FT_Face ft_face;
|
2006-12-06 21:31:53 +01:00
|
|
|
struct font_mapping *mapping;
|
2001-09-12 22:21:06 +02:00
|
|
|
};
|
|
|
|
|
2020-10-29 14:53:18 +01:00
|
|
|
static inline FT_Face get_ft_face( struct gdi_font *font )
|
|
|
|
{
|
|
|
|
return ((struct font_private_data *)font->private)->ft_face;
|
|
|
|
}
|
2020-10-20 22:05:37 +02:00
|
|
|
|
|
|
|
static const struct font_backend_funcs font_funcs;
|
2011-10-18 11:44:41 +02:00
|
|
|
|
2004-03-09 04:43:54 +01:00
|
|
|
static const WCHAR win9x_font_reg_key[] = {'S','o','f','t','w','a','r','e','\\','M','i','c','r','o','s','o','f','t','\\',
|
|
|
|
'W','i','n','d','o','w','s','\\',
|
|
|
|
'C','u','r','r','e','n','t','V','e','r','s','i','o','n','\\',
|
|
|
|
'F','o','n','t','s','\0'};
|
|
|
|
|
|
|
|
static const WCHAR winnt_font_reg_key[] = {'S','o','f','t','w','a','r','e','\\','M','i','c','r','o','s','o','f','t','\\',
|
|
|
|
'W','i','n','d','o','w','s',' ','N','T','\\',
|
|
|
|
'C','u','r','r','e','n','t','V','e','r','s','i','o','n','\\',
|
|
|
|
'F','o','n','t','s','\0'};
|
|
|
|
|
2005-06-14 14:33:19 +02:00
|
|
|
static const WCHAR external_fonts_reg_key[] = {'S','o','f','t','w','a','r','e','\\','W','i','n','e','\\',
|
2004-03-09 04:43:54 +01:00
|
|
|
'F','o','n','t','s','\\','E','x','t','e','r','n','a','l',' ','F','o','n','t','s','\0'};
|
2002-01-29 04:02:50 +01:00
|
|
|
|
2009-05-09 19:12:40 +02:00
|
|
|
/* Interesting and well-known (frequently-assumed!) font names */
|
2011-10-12 12:55:12 +02:00
|
|
|
static const WCHAR arial[] = {'A','r','i','a','l',0};
|
|
|
|
static const WCHAR bitstream_vera_sans[] = {'B','i','t','s','t','r','e','a','m',' ','V','e','r','a',' ','S','a','n','s',0};
|
|
|
|
static const WCHAR bitstream_vera_sans_mono[] = {'B','i','t','s','t','r','e','a','m',' ','V','e','r','a',' ','S','a','n','s',' ','M','o','n','o',0};
|
|
|
|
static const WCHAR bitstream_vera_serif[] = {'B','i','t','s','t','r','e','a','m',' ','V','e','r','a',' ','S','e','r','i','f',0};
|
|
|
|
static const WCHAR courier_new[] = {'C','o','u','r','i','e','r',' ','N','e','w',0};
|
2011-10-12 12:55:13 +02:00
|
|
|
static const WCHAR liberation_mono[] = {'L','i','b','e','r','a','t','i','o','n',' ','M','o','n','o',0};
|
|
|
|
static const WCHAR liberation_sans[] = {'L','i','b','e','r','a','t','i','o','n',' ','S','a','n','s',0};
|
|
|
|
static const WCHAR liberation_serif[] = {'L','i','b','e','r','a','t','i','o','n',' ','S','e','r','i','f',0};
|
2011-10-12 12:55:12 +02:00
|
|
|
static const WCHAR times_new_roman[] = {'T','i','m','e','s',' ','N','e','w',' ','R','o','m','a','n',0};
|
2011-11-03 14:11:35 +01:00
|
|
|
static const WCHAR SymbolW[] = {'S','y','m','b','o','l','\0'};
|
2011-10-12 12:55:12 +02:00
|
|
|
|
|
|
|
static const WCHAR *default_serif_list[] =
|
|
|
|
{
|
|
|
|
times_new_roman,
|
2011-10-12 12:55:13 +02:00
|
|
|
liberation_serif,
|
2011-10-12 12:55:12 +02:00
|
|
|
bitstream_vera_serif,
|
|
|
|
NULL
|
|
|
|
};
|
|
|
|
|
|
|
|
static const WCHAR *default_fixed_list[] =
|
|
|
|
{
|
|
|
|
courier_new,
|
2011-10-12 12:55:13 +02:00
|
|
|
liberation_mono,
|
2011-10-12 12:55:12 +02:00
|
|
|
bitstream_vera_sans_mono,
|
|
|
|
NULL
|
|
|
|
};
|
|
|
|
|
|
|
|
static const WCHAR *default_sans_list[] =
|
|
|
|
{
|
|
|
|
arial,
|
2011-10-12 12:55:13 +02:00
|
|
|
liberation_sans,
|
2011-10-12 12:55:12 +02:00
|
|
|
bitstream_vera_sans,
|
|
|
|
NULL
|
|
|
|
};
|
|
|
|
|
2018-12-20 11:32:14 +01:00
|
|
|
static const WCHAR *default_serif = times_new_roman;
|
|
|
|
static const WCHAR *default_fixed = courier_new;
|
|
|
|
static const WCHAR *default_sans = arial;
|
|
|
|
|
2011-10-06 23:26:06 +02:00
|
|
|
/* Registry font cache key and value names */
|
|
|
|
static const WCHAR wine_fonts_key[] = {'S','o','f','t','w','a','r','e','\\','W','i','n','e','\\',
|
|
|
|
'F','o','n','t','s',0};
|
|
|
|
static const WCHAR wine_fonts_cache_key[] = {'C','a','c','h','e',0};
|
|
|
|
|
|
|
|
|
2006-12-06 21:31:53 +01:00
|
|
|
struct font_mapping
|
|
|
|
{
|
|
|
|
struct list entry;
|
|
|
|
int refcount;
|
|
|
|
dev_t dev;
|
|
|
|
ino_t ino;
|
|
|
|
void *data;
|
|
|
|
size_t size;
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct list mappings_list = LIST_INIT( mappings_list );
|
|
|
|
|
2012-11-05 22:32:51 +01:00
|
|
|
static UINT default_aa_flags;
|
2013-01-15 13:56:52 +01:00
|
|
|
static HKEY hkey_font_cache;
|
2015-09-29 09:27:15 +02:00
|
|
|
static BOOL antialias_fakes = TRUE;
|
2012-11-05 22:32:51 +01:00
|
|
|
|
2004-03-09 04:43:54 +01:00
|
|
|
static const WCHAR font_mutex_nameW[] = {'_','_','W','I','N','E','_','F','O','N','T','_','M','U','T','E','X','_','_','\0'};
|
|
|
|
|
2020-10-26 11:43:24 +01:00
|
|
|
static BOOL CDECL freetype_set_outline_text_metrics( struct gdi_font *font );
|
|
|
|
static BOOL CDECL freetype_set_bitmap_text_metrics( struct gdi_font *font );
|
2013-01-15 12:34:40 +01:00
|
|
|
static void remove_face_from_cache( Face *face );
|
2004-06-16 22:06:26 +02:00
|
|
|
|
|
|
|
/****************************************
|
|
|
|
* Notes on .fon files
|
|
|
|
*
|
|
|
|
* The fonts System, FixedSys and Terminal are special. There are typically multiple
|
|
|
|
* versions installed for different resolutions and codepages. Windows stores which one to use
|
|
|
|
* in HKEY_CURRENT_CONFIG\\Software\\Fonts.
|
|
|
|
* Key Meaning
|
|
|
|
* FIXEDFON.FON FixedSys
|
|
|
|
* FONTS.FON System
|
2005-01-05 18:12:07 +01:00
|
|
|
* OEMFONT.FON Terminal
|
2004-06-16 22:06:26 +02:00
|
|
|
* LogPixels Current dpi set by the display control panel applet
|
2008-03-25 18:35:45 +01:00
|
|
|
* (HKLM\\Software\\Microsoft\\Windows NT\\CurrentVersion\\FontDPI
|
2004-06-16 22:06:26 +02:00
|
|
|
* also has a LogPixels value that appears to mirror this)
|
|
|
|
*
|
|
|
|
* On my system these values have data: vgafix.fon, vgasys.fon, vga850.fon and 96 respectively
|
|
|
|
* (vgaoem.fon would be your oemfont.fon if you have a US setup).
|
|
|
|
* If the resolution is changed to be >= 109dpi then the fonts goto 8514fix, 8514sys and 8514oem
|
|
|
|
* (not sure what's happening to the oem codepage here). 109 is nicely halfway between 96 and 120dpi,
|
|
|
|
* so that makes sense.
|
|
|
|
*
|
|
|
|
* Additionally Windows also loads the fonts listed in the [386enh] section of system.ini (this doesn't appear
|
|
|
|
* to be mapped into the registry on Windows 2000 at least).
|
|
|
|
* I have
|
|
|
|
* woafont=app850.fon
|
|
|
|
* ega80woa.fon=ega80850.fon
|
|
|
|
* ega40woa.fon=ega40850.fon
|
|
|
|
* cga80woa.fon=cga80850.fon
|
|
|
|
* cga40woa.fon=cga40850.fon
|
|
|
|
*/
|
|
|
|
|
2006-12-21 16:21:13 +01:00
|
|
|
#ifdef HAVE_CARBON_CARBON_H
|
|
|
|
static char *find_cache_dir(void)
|
|
|
|
{
|
|
|
|
FSRef ref;
|
|
|
|
OSErr err;
|
|
|
|
static char cached_path[MAX_PATH];
|
|
|
|
static const char *wine = "/Wine", *fonts = "/Fonts";
|
|
|
|
|
|
|
|
if(*cached_path) return cached_path;
|
|
|
|
|
|
|
|
err = FSFindFolder(kUserDomain, kCachedDataFolderType, kCreateFolder, &ref);
|
|
|
|
if(err != noErr)
|
|
|
|
{
|
|
|
|
WARN("can't create cached data folder\n");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
err = FSRefMakePath(&ref, (unsigned char*)cached_path, sizeof(cached_path));
|
|
|
|
if(err != noErr)
|
|
|
|
{
|
|
|
|
WARN("can't create cached data path\n");
|
|
|
|
*cached_path = '\0';
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
if(strlen(cached_path) + strlen(wine) + strlen(fonts) + 1 > sizeof(cached_path))
|
|
|
|
{
|
|
|
|
ERR("Could not create full path\n");
|
|
|
|
*cached_path = '\0';
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
strcat(cached_path, wine);
|
|
|
|
|
|
|
|
if(mkdir(cached_path, 0700) == -1 && errno != EEXIST)
|
|
|
|
{
|
|
|
|
WARN("Couldn't mkdir %s\n", cached_path);
|
|
|
|
*cached_path = '\0';
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
strcat(cached_path, fonts);
|
|
|
|
if(mkdir(cached_path, 0700) == -1 && errno != EEXIST)
|
|
|
|
{
|
|
|
|
WARN("Couldn't mkdir %s\n", cached_path);
|
|
|
|
*cached_path = '\0';
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
return cached_path;
|
|
|
|
}
|
|
|
|
|
|
|
|
/******************************************************************
|
|
|
|
* expand_mac_font
|
|
|
|
*
|
|
|
|
* Extracts individual TrueType font files from a Mac suitcase font
|
|
|
|
* and saves them into the user's caches directory (see
|
|
|
|
* find_cache_dir()).
|
|
|
|
* Returns a NULL terminated array of filenames.
|
|
|
|
*
|
|
|
|
* We do this because they are apps that try to read ttf files
|
|
|
|
* themselves and they don't like Mac suitcase files.
|
|
|
|
*/
|
|
|
|
static char **expand_mac_font(const char *path)
|
|
|
|
{
|
|
|
|
FSRef ref;
|
2019-07-10 10:50:17 +02:00
|
|
|
ResFileRefNum res_ref;
|
2006-12-21 16:21:13 +01:00
|
|
|
OSStatus s;
|
|
|
|
unsigned int idx;
|
|
|
|
const char *out_dir;
|
|
|
|
const char *filename;
|
|
|
|
int output_len;
|
|
|
|
struct {
|
|
|
|
char **array;
|
|
|
|
unsigned int size, max_size;
|
|
|
|
} ret;
|
|
|
|
|
|
|
|
TRACE("path %s\n", path);
|
|
|
|
|
|
|
|
s = FSPathMakeRef((unsigned char*)path, &ref, FALSE);
|
|
|
|
if(s != noErr)
|
|
|
|
{
|
|
|
|
WARN("failed to get ref\n");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
s = FSOpenResourceFile(&ref, 0, NULL, fsRdPerm, &res_ref);
|
|
|
|
if(s != noErr)
|
|
|
|
{
|
|
|
|
TRACE("no data fork, so trying resource fork\n");
|
|
|
|
res_ref = FSOpenResFile(&ref, fsRdPerm);
|
|
|
|
if(res_ref == -1)
|
|
|
|
{
|
|
|
|
TRACE("unable to open resource fork\n");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ret.size = 0;
|
|
|
|
ret.max_size = 10;
|
|
|
|
ret.array = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, ret.max_size * sizeof(*ret.array));
|
|
|
|
if(!ret.array)
|
|
|
|
{
|
|
|
|
CloseResFile(res_ref);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
out_dir = find_cache_dir();
|
|
|
|
|
|
|
|
filename = strrchr(path, '/');
|
|
|
|
if(!filename) filename = path;
|
|
|
|
else filename++;
|
|
|
|
|
|
|
|
/* output filename has the form out_dir/filename_%04x.ttf */
|
|
|
|
output_len = strlen(out_dir) + 1 + strlen(filename) + 5 + 5;
|
|
|
|
|
|
|
|
UseResFile(res_ref);
|
|
|
|
idx = 1;
|
|
|
|
while(1)
|
|
|
|
{
|
|
|
|
FamRec *fam_rec;
|
|
|
|
unsigned short *num_faces_ptr, num_faces, face;
|
|
|
|
AsscEntry *assoc;
|
|
|
|
Handle fond;
|
2007-09-20 12:10:31 +02:00
|
|
|
ResType fond_res = FT_MAKE_TAG('F','O','N','D');
|
2006-12-21 16:21:13 +01:00
|
|
|
|
2007-06-25 23:34:34 +02:00
|
|
|
fond = Get1IndResource(fond_res, idx);
|
2006-12-21 16:21:13 +01:00
|
|
|
if(!fond) break;
|
|
|
|
TRACE("got fond resource %d\n", idx);
|
|
|
|
HLock(fond);
|
|
|
|
|
|
|
|
fam_rec = *(FamRec**)fond;
|
|
|
|
num_faces_ptr = (unsigned short *)(fam_rec + 1);
|
|
|
|
num_faces = GET_BE_WORD(*num_faces_ptr);
|
|
|
|
num_faces++;
|
|
|
|
assoc = (AsscEntry*)(num_faces_ptr + 1);
|
|
|
|
TRACE("num faces %04x\n", num_faces);
|
|
|
|
for(face = 0; face < num_faces; face++, assoc++)
|
|
|
|
{
|
|
|
|
Handle sfnt;
|
2007-09-20 12:10:31 +02:00
|
|
|
ResType sfnt_res = FT_MAKE_TAG('s','f','n','t');
|
2006-12-21 16:21:13 +01:00
|
|
|
unsigned short size, font_id;
|
|
|
|
char *output;
|
|
|
|
|
|
|
|
size = GET_BE_WORD(assoc->fontSize);
|
|
|
|
font_id = GET_BE_WORD(assoc->fontID);
|
|
|
|
if(size != 0)
|
|
|
|
{
|
|
|
|
TRACE("skipping id %04x because it's not scalable (fixed size %d)\n", font_id, size);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
TRACE("trying to load sfnt id %04x\n", font_id);
|
2007-06-25 23:34:34 +02:00
|
|
|
sfnt = GetResource(sfnt_res, font_id);
|
2006-12-21 16:21:13 +01:00
|
|
|
if(!sfnt)
|
|
|
|
{
|
|
|
|
TRACE("can't get sfnt resource %04x\n", font_id);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
output = HeapAlloc(GetProcessHeap(), 0, output_len);
|
|
|
|
if(output)
|
|
|
|
{
|
|
|
|
int fd;
|
|
|
|
|
|
|
|
sprintf(output, "%s/%s_%04x.ttf", out_dir, filename, font_id);
|
|
|
|
|
|
|
|
fd = open(output, O_CREAT | O_EXCL | O_WRONLY, 0600);
|
|
|
|
if(fd != -1 || errno == EEXIST)
|
|
|
|
{
|
|
|
|
if(fd != -1)
|
|
|
|
{
|
|
|
|
unsigned char *sfnt_data;
|
|
|
|
|
|
|
|
HLock(sfnt);
|
|
|
|
sfnt_data = *(unsigned char**)sfnt;
|
|
|
|
write(fd, sfnt_data, GetHandleSize(sfnt));
|
|
|
|
HUnlock(sfnt);
|
|
|
|
close(fd);
|
|
|
|
}
|
|
|
|
if(ret.size >= ret.max_size - 1) /* Always want the last element to be NULL */
|
|
|
|
{
|
|
|
|
ret.max_size *= 2;
|
|
|
|
ret.array = HeapReAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, ret.array, ret.max_size * sizeof(*ret.array));
|
|
|
|
}
|
|
|
|
ret.array[ret.size++] = output;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
WARN("unable to create %s\n", output);
|
|
|
|
HeapFree(GetProcessHeap(), 0, output);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ReleaseResource(sfnt);
|
|
|
|
}
|
|
|
|
HUnlock(fond);
|
|
|
|
ReleaseResource(fond);
|
|
|
|
idx++;
|
|
|
|
}
|
|
|
|
CloseResFile(res_ref);
|
|
|
|
|
|
|
|
return ret.array;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* HAVE_CARBON_CARBON_H */
|
2004-06-16 22:06:26 +02:00
|
|
|
|
2003-10-10 02:06:35 +02:00
|
|
|
/*
|
2008-06-30 00:23:07 +02:00
|
|
|
This function builds an FT_Fixed from a double. It fails if the absolute
|
|
|
|
value of the float number is greater than 32768.
|
2003-10-10 02:06:35 +02:00
|
|
|
*/
|
2008-06-24 09:10:27 +02:00
|
|
|
static inline FT_Fixed FT_FixedFromFloat(double f)
|
2003-10-10 02:06:35 +02:00
|
|
|
{
|
2008-06-30 00:23:07 +02:00
|
|
|
return f * 0x10000;
|
2003-10-10 02:06:35 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
This function builds an FT_Fixed from a FIXED. It simply put f.value
|
|
|
|
in the highest 16 bits and f.fract in the lowest 16 bits of the FT_Fixed.
|
|
|
|
*/
|
|
|
|
static inline FT_Fixed FT_FixedFromFIXED(FIXED f)
|
|
|
|
{
|
2009-06-02 10:16:38 +02:00
|
|
|
return (FT_Fixed)((int)f.value << 16 | (unsigned int)f.fract);
|
2003-10-10 02:06:35 +02:00
|
|
|
}
|
|
|
|
|
2012-11-02 17:21:28 +01:00
|
|
|
static BOOL is_hinting_enabled(void)
|
|
|
|
{
|
|
|
|
static int enabled = -1;
|
|
|
|
|
|
|
|
if (enabled == -1)
|
|
|
|
{
|
|
|
|
/* Use the >= 2.2.0 function if available */
|
|
|
|
if (pFT_Get_TrueType_Engine_Type)
|
|
|
|
{
|
|
|
|
FT_TrueTypeEngineType type = pFT_Get_TrueType_Engine_Type(library);
|
|
|
|
enabled = (type == FT_TRUETYPE_ENGINE_TYPE_PATENTED);
|
|
|
|
}
|
|
|
|
else enabled = FALSE;
|
2012-11-05 22:31:31 +01:00
|
|
|
TRACE("hinting is %senabled\n", enabled ? "" : "NOT ");
|
2012-11-02 17:21:28 +01:00
|
|
|
}
|
|
|
|
return enabled;
|
|
|
|
}
|
|
|
|
|
|
|
|
static BOOL is_subpixel_rendering_enabled( void )
|
|
|
|
{
|
|
|
|
static int enabled = -1;
|
|
|
|
if (enabled == -1)
|
2012-11-05 22:31:31 +01:00
|
|
|
{
|
2018-10-09 07:11:07 +02:00
|
|
|
/* FreeType >= 2.8.1 offers LCD-optimezed rendering without lcd filters. */
|
2019-02-08 02:51:32 +01:00
|
|
|
if (FT_SimpleVersion >= FT_VERSION_VALUE(2, 8, 1))
|
2018-10-09 07:11:07 +02:00
|
|
|
enabled = TRUE;
|
|
|
|
#ifdef FT_LCD_FILTER_H
|
|
|
|
else if (pFT_Library_SetLcdFilter &&
|
|
|
|
pFT_Library_SetLcdFilter( NULL, 0 ) != FT_Err_Unimplemented_Feature)
|
|
|
|
enabled = TRUE;
|
|
|
|
#endif
|
|
|
|
else enabled = FALSE;
|
|
|
|
|
2012-11-05 22:31:31 +01:00
|
|
|
TRACE("subpixel rendering is %senabled\n", enabled ? "" : "NOT ");
|
|
|
|
}
|
2012-11-02 17:21:28 +01:00
|
|
|
return enabled;
|
|
|
|
}
|
|
|
|
|
2006-04-05 13:41:49 +02:00
|
|
|
|
2012-03-14 13:01:49 +01:00
|
|
|
static const struct list *get_face_list_from_family(const Family *family)
|
|
|
|
{
|
2020-10-30 14:39:07 +01:00
|
|
|
if (family->replacement) return &family->replacement->faces;
|
|
|
|
return &family->faces;
|
2012-03-14 13:01:49 +01:00
|
|
|
}
|
|
|
|
|
2006-04-05 13:41:49 +02:00
|
|
|
static LPWSTR strdupW(LPCWSTR p)
|
|
|
|
{
|
|
|
|
LPWSTR ret;
|
|
|
|
DWORD len = (strlenW(p) + 1) * sizeof(WCHAR);
|
|
|
|
ret = HeapAlloc(GetProcessHeap(), 0, len);
|
|
|
|
memcpy(ret, p, len);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2011-12-20 13:04:56 +01:00
|
|
|
static WCHAR *towstr(UINT cp, const char *str)
|
|
|
|
{
|
|
|
|
int len;
|
|
|
|
WCHAR *wstr;
|
|
|
|
|
|
|
|
len = MultiByteToWideChar(cp, 0, str, -1, NULL, 0);
|
|
|
|
wstr = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR));
|
|
|
|
MultiByteToWideChar(cp, 0, str, -1, wstr, len);
|
|
|
|
return wstr;
|
|
|
|
}
|
|
|
|
|
2008-09-18 13:23:08 +02:00
|
|
|
|
2013-06-20 15:04:11 +02:00
|
|
|
static const LANGID mac_langid_table[] =
|
|
|
|
{
|
|
|
|
MAKELANGID(LANG_ENGLISH,SUBLANG_DEFAULT), /* TT_MAC_LANGID_ENGLISH */
|
|
|
|
MAKELANGID(LANG_FRENCH,SUBLANG_DEFAULT), /* TT_MAC_LANGID_FRENCH */
|
|
|
|
MAKELANGID(LANG_GERMAN,SUBLANG_DEFAULT), /* TT_MAC_LANGID_GERMAN */
|
|
|
|
MAKELANGID(LANG_ITALIAN,SUBLANG_DEFAULT), /* TT_MAC_LANGID_ITALIAN */
|
|
|
|
MAKELANGID(LANG_DUTCH,SUBLANG_DEFAULT), /* TT_MAC_LANGID_DUTCH */
|
|
|
|
MAKELANGID(LANG_SWEDISH,SUBLANG_DEFAULT), /* TT_MAC_LANGID_SWEDISH */
|
|
|
|
MAKELANGID(LANG_SPANISH,SUBLANG_DEFAULT), /* TT_MAC_LANGID_SPANISH */
|
|
|
|
MAKELANGID(LANG_DANISH,SUBLANG_DEFAULT), /* TT_MAC_LANGID_DANISH */
|
|
|
|
MAKELANGID(LANG_PORTUGUESE,SUBLANG_DEFAULT), /* TT_MAC_LANGID_PORTUGUESE */
|
|
|
|
MAKELANGID(LANG_NORWEGIAN,SUBLANG_DEFAULT), /* TT_MAC_LANGID_NORWEGIAN */
|
|
|
|
MAKELANGID(LANG_HEBREW,SUBLANG_DEFAULT), /* TT_MAC_LANGID_HEBREW */
|
|
|
|
MAKELANGID(LANG_JAPANESE,SUBLANG_DEFAULT), /* TT_MAC_LANGID_JAPANESE */
|
|
|
|
MAKELANGID(LANG_ARABIC,SUBLANG_DEFAULT), /* TT_MAC_LANGID_ARABIC */
|
|
|
|
MAKELANGID(LANG_FINNISH,SUBLANG_DEFAULT), /* TT_MAC_LANGID_FINNISH */
|
|
|
|
MAKELANGID(LANG_GREEK,SUBLANG_DEFAULT), /* TT_MAC_LANGID_GREEK */
|
|
|
|
MAKELANGID(LANG_ICELANDIC,SUBLANG_DEFAULT), /* TT_MAC_LANGID_ICELANDIC */
|
|
|
|
MAKELANGID(LANG_MALTESE,SUBLANG_DEFAULT), /* TT_MAC_LANGID_MALTESE */
|
|
|
|
MAKELANGID(LANG_TURKISH,SUBLANG_DEFAULT), /* TT_MAC_LANGID_TURKISH */
|
|
|
|
MAKELANGID(LANG_CROATIAN,SUBLANG_DEFAULT), /* TT_MAC_LANGID_CROATIAN */
|
|
|
|
MAKELANGID(LANG_CHINESE_TRADITIONAL,SUBLANG_DEFAULT), /* TT_MAC_LANGID_CHINESE_TRADITIONAL */
|
|
|
|
MAKELANGID(LANG_URDU,SUBLANG_DEFAULT), /* TT_MAC_LANGID_URDU */
|
|
|
|
MAKELANGID(LANG_HINDI,SUBLANG_DEFAULT), /* TT_MAC_LANGID_HINDI */
|
|
|
|
MAKELANGID(LANG_THAI,SUBLANG_DEFAULT), /* TT_MAC_LANGID_THAI */
|
|
|
|
MAKELANGID(LANG_KOREAN,SUBLANG_DEFAULT), /* TT_MAC_LANGID_KOREAN */
|
|
|
|
MAKELANGID(LANG_LITHUANIAN,SUBLANG_DEFAULT), /* TT_MAC_LANGID_LITHUANIAN */
|
|
|
|
MAKELANGID(LANG_POLISH,SUBLANG_DEFAULT), /* TT_MAC_LANGID_POLISH */
|
|
|
|
MAKELANGID(LANG_HUNGARIAN,SUBLANG_DEFAULT), /* TT_MAC_LANGID_HUNGARIAN */
|
|
|
|
MAKELANGID(LANG_ESTONIAN,SUBLANG_DEFAULT), /* TT_MAC_LANGID_ESTONIAN */
|
|
|
|
MAKELANGID(LANG_LATVIAN,SUBLANG_DEFAULT), /* TT_MAC_LANGID_LETTISH */
|
|
|
|
MAKELANGID(LANG_SAMI,SUBLANG_DEFAULT), /* TT_MAC_LANGID_SAAMISK */
|
|
|
|
MAKELANGID(LANG_FAEROESE,SUBLANG_DEFAULT), /* TT_MAC_LANGID_FAEROESE */
|
|
|
|
MAKELANGID(LANG_FARSI,SUBLANG_DEFAULT), /* TT_MAC_LANGID_FARSI */
|
|
|
|
MAKELANGID(LANG_RUSSIAN,SUBLANG_DEFAULT), /* TT_MAC_LANGID_RUSSIAN */
|
|
|
|
MAKELANGID(LANG_CHINESE_SIMPLIFIED,SUBLANG_DEFAULT), /* TT_MAC_LANGID_CHINESE_SIMPLIFIED */
|
|
|
|
MAKELANGID(LANG_DUTCH,SUBLANG_DUTCH_BELGIAN), /* TT_MAC_LANGID_FLEMISH */
|
|
|
|
MAKELANGID(LANG_IRISH,SUBLANG_DEFAULT), /* TT_MAC_LANGID_IRISH */
|
|
|
|
MAKELANGID(LANG_ALBANIAN,SUBLANG_DEFAULT), /* TT_MAC_LANGID_ALBANIAN */
|
|
|
|
MAKELANGID(LANG_ROMANIAN,SUBLANG_DEFAULT), /* TT_MAC_LANGID_ROMANIAN */
|
|
|
|
MAKELANGID(LANG_CZECH,SUBLANG_DEFAULT), /* TT_MAC_LANGID_CZECH */
|
|
|
|
MAKELANGID(LANG_SLOVAK,SUBLANG_DEFAULT), /* TT_MAC_LANGID_SLOVAK */
|
|
|
|
MAKELANGID(LANG_SLOVENIAN,SUBLANG_DEFAULT), /* TT_MAC_LANGID_SLOVENIAN */
|
|
|
|
0, /* TT_MAC_LANGID_YIDDISH */
|
|
|
|
MAKELANGID(LANG_SERBIAN,SUBLANG_DEFAULT), /* TT_MAC_LANGID_SERBIAN */
|
|
|
|
MAKELANGID(LANG_MACEDONIAN,SUBLANG_DEFAULT), /* TT_MAC_LANGID_MACEDONIAN */
|
|
|
|
MAKELANGID(LANG_BULGARIAN,SUBLANG_DEFAULT), /* TT_MAC_LANGID_BULGARIAN */
|
|
|
|
MAKELANGID(LANG_UKRAINIAN,SUBLANG_DEFAULT), /* TT_MAC_LANGID_UKRAINIAN */
|
|
|
|
MAKELANGID(LANG_BELARUSIAN,SUBLANG_DEFAULT), /* TT_MAC_LANGID_BYELORUSSIAN */
|
|
|
|
MAKELANGID(LANG_UZBEK,SUBLANG_DEFAULT), /* TT_MAC_LANGID_UZBEK */
|
|
|
|
MAKELANGID(LANG_KAZAK,SUBLANG_DEFAULT), /* TT_MAC_LANGID_KAZAKH */
|
|
|
|
MAKELANGID(LANG_AZERI,SUBLANG_AZERI_CYRILLIC), /* TT_MAC_LANGID_AZERBAIJANI */
|
|
|
|
0, /* TT_MAC_LANGID_AZERBAIJANI_ARABIC_SCRIPT */
|
|
|
|
MAKELANGID(LANG_ARMENIAN,SUBLANG_DEFAULT), /* TT_MAC_LANGID_ARMENIAN */
|
|
|
|
MAKELANGID(LANG_GEORGIAN,SUBLANG_DEFAULT), /* TT_MAC_LANGID_GEORGIAN */
|
|
|
|
0, /* TT_MAC_LANGID_MOLDAVIAN */
|
|
|
|
MAKELANGID(LANG_KYRGYZ,SUBLANG_DEFAULT), /* TT_MAC_LANGID_KIRGHIZ */
|
|
|
|
MAKELANGID(LANG_TAJIK,SUBLANG_DEFAULT), /* TT_MAC_LANGID_TAJIKI */
|
|
|
|
MAKELANGID(LANG_TURKMEN,SUBLANG_DEFAULT), /* TT_MAC_LANGID_TURKMEN */
|
|
|
|
MAKELANGID(LANG_MONGOLIAN,SUBLANG_DEFAULT), /* TT_MAC_LANGID_MONGOLIAN */
|
|
|
|
MAKELANGID(LANG_MONGOLIAN,SUBLANG_MONGOLIAN_CYRILLIC_MONGOLIA), /* TT_MAC_LANGID_MONGOLIAN_CYRILLIC_SCRIPT */
|
|
|
|
MAKELANGID(LANG_PASHTO,SUBLANG_DEFAULT), /* TT_MAC_LANGID_PASHTO */
|
|
|
|
0, /* TT_MAC_LANGID_KURDISH */
|
|
|
|
MAKELANGID(LANG_KASHMIRI,SUBLANG_DEFAULT), /* TT_MAC_LANGID_KASHMIRI */
|
|
|
|
MAKELANGID(LANG_SINDHI,SUBLANG_DEFAULT), /* TT_MAC_LANGID_SINDHI */
|
|
|
|
MAKELANGID(LANG_TIBETAN,SUBLANG_DEFAULT), /* TT_MAC_LANGID_TIBETAN */
|
|
|
|
MAKELANGID(LANG_NEPALI,SUBLANG_DEFAULT), /* TT_MAC_LANGID_NEPALI */
|
|
|
|
MAKELANGID(LANG_SANSKRIT,SUBLANG_DEFAULT), /* TT_MAC_LANGID_SANSKRIT */
|
|
|
|
MAKELANGID(LANG_MARATHI,SUBLANG_DEFAULT), /* TT_MAC_LANGID_MARATHI */
|
|
|
|
MAKELANGID(LANG_BENGALI,SUBLANG_DEFAULT), /* TT_MAC_LANGID_BENGALI */
|
|
|
|
MAKELANGID(LANG_ASSAMESE,SUBLANG_DEFAULT), /* TT_MAC_LANGID_ASSAMESE */
|
|
|
|
MAKELANGID(LANG_GUJARATI,SUBLANG_DEFAULT), /* TT_MAC_LANGID_GUJARATI */
|
|
|
|
MAKELANGID(LANG_PUNJABI,SUBLANG_DEFAULT), /* TT_MAC_LANGID_PUNJABI */
|
|
|
|
MAKELANGID(LANG_ORIYA,SUBLANG_DEFAULT), /* TT_MAC_LANGID_ORIYA */
|
|
|
|
MAKELANGID(LANG_MALAYALAM,SUBLANG_DEFAULT), /* TT_MAC_LANGID_MALAYALAM */
|
|
|
|
MAKELANGID(LANG_KANNADA,SUBLANG_DEFAULT), /* TT_MAC_LANGID_KANNADA */
|
|
|
|
MAKELANGID(LANG_TAMIL,SUBLANG_DEFAULT), /* TT_MAC_LANGID_TAMIL */
|
|
|
|
MAKELANGID(LANG_TELUGU,SUBLANG_DEFAULT), /* TT_MAC_LANGID_TELUGU */
|
|
|
|
MAKELANGID(LANG_SINHALESE,SUBLANG_DEFAULT), /* TT_MAC_LANGID_SINHALESE */
|
|
|
|
0, /* TT_MAC_LANGID_BURMESE */
|
|
|
|
MAKELANGID(LANG_KHMER,SUBLANG_DEFAULT), /* TT_MAC_LANGID_KHMER */
|
|
|
|
MAKELANGID(LANG_LAO,SUBLANG_DEFAULT), /* TT_MAC_LANGID_LAO */
|
|
|
|
MAKELANGID(LANG_VIETNAMESE,SUBLANG_DEFAULT), /* TT_MAC_LANGID_VIETNAMESE */
|
|
|
|
MAKELANGID(LANG_INDONESIAN,SUBLANG_DEFAULT), /* TT_MAC_LANGID_INDONESIAN */
|
|
|
|
0, /* TT_MAC_LANGID_TAGALOG */
|
|
|
|
MAKELANGID(LANG_MALAY,SUBLANG_DEFAULT), /* TT_MAC_LANGID_MALAY_ROMAN_SCRIPT */
|
|
|
|
0, /* TT_MAC_LANGID_MALAY_ARABIC_SCRIPT */
|
|
|
|
MAKELANGID(LANG_AMHARIC,SUBLANG_DEFAULT), /* TT_MAC_LANGID_AMHARIC */
|
|
|
|
MAKELANGID(LANG_TIGRIGNA,SUBLANG_DEFAULT), /* TT_MAC_LANGID_TIGRINYA */
|
|
|
|
0, /* TT_MAC_LANGID_GALLA */
|
|
|
|
0, /* TT_MAC_LANGID_SOMALI */
|
|
|
|
MAKELANGID(LANG_SWAHILI,SUBLANG_DEFAULT), /* TT_MAC_LANGID_SWAHILI */
|
|
|
|
0, /* TT_MAC_LANGID_RUANDA */
|
|
|
|
0, /* TT_MAC_LANGID_RUNDI */
|
|
|
|
0, /* TT_MAC_LANGID_CHEWA */
|
|
|
|
MAKELANGID(LANG_MALAGASY,SUBLANG_DEFAULT), /* TT_MAC_LANGID_MALAGASY */
|
|
|
|
MAKELANGID(LANG_ESPERANTO,SUBLANG_DEFAULT), /* TT_MAC_LANGID_ESPERANTO */
|
|
|
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, /* 95-111 */
|
|
|
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, /* 112-127 */
|
|
|
|
MAKELANGID(LANG_WELSH,SUBLANG_DEFAULT), /* TT_MAC_LANGID_WELSH */
|
|
|
|
MAKELANGID(LANG_BASQUE,SUBLANG_DEFAULT), /* TT_MAC_LANGID_BASQUE */
|
|
|
|
MAKELANGID(LANG_CATALAN,SUBLANG_DEFAULT), /* TT_MAC_LANGID_CATALAN */
|
|
|
|
0, /* TT_MAC_LANGID_LATIN */
|
|
|
|
MAKELANGID(LANG_QUECHUA,SUBLANG_DEFAULT), /* TT_MAC_LANGID_QUECHUA */
|
|
|
|
0, /* TT_MAC_LANGID_GUARANI */
|
|
|
|
0, /* TT_MAC_LANGID_AYMARA */
|
|
|
|
MAKELANGID(LANG_TATAR,SUBLANG_DEFAULT), /* TT_MAC_LANGID_TATAR */
|
|
|
|
MAKELANGID(LANG_UIGHUR,SUBLANG_DEFAULT), /* TT_MAC_LANGID_UIGHUR */
|
|
|
|
0, /* TT_MAC_LANGID_DZONGKHA */
|
|
|
|
0, /* TT_MAC_LANGID_JAVANESE */
|
|
|
|
0, /* TT_MAC_LANGID_SUNDANESE */
|
|
|
|
MAKELANGID(LANG_GALICIAN,SUBLANG_DEFAULT), /* TT_MAC_LANGID_GALICIAN */
|
|
|
|
MAKELANGID(LANG_AFRIKAANS,SUBLANG_DEFAULT), /* TT_MAC_LANGID_AFRIKAANS */
|
|
|
|
MAKELANGID(LANG_BRETON,SUBLANG_DEFAULT), /* TT_MAC_LANGID_BRETON */
|
|
|
|
MAKELANGID(LANG_INUKTITUT,SUBLANG_DEFAULT), /* TT_MAC_LANGID_INUKTITUT */
|
|
|
|
MAKELANGID(LANG_SCOTTISH_GAELIC,SUBLANG_DEFAULT), /* TT_MAC_LANGID_SCOTTISH_GAELIC */
|
|
|
|
MAKELANGID(LANG_MANX_GAELIC,SUBLANG_DEFAULT), /* TT_MAC_LANGID_MANX_GAELIC */
|
|
|
|
MAKELANGID(LANG_IRISH,SUBLANG_IRISH_IRELAND), /* TT_MAC_LANGID_IRISH_GAELIC */
|
|
|
|
0, /* TT_MAC_LANGID_TONGAN */
|
|
|
|
0, /* TT_MAC_LANGID_GREEK_POLYTONIC */
|
|
|
|
MAKELANGID(LANG_GREENLANDIC,SUBLANG_DEFAULT), /* TT_MAC_LANGID_GREELANDIC */
|
|
|
|
MAKELANGID(LANG_AZERI,SUBLANG_AZERI_LATIN), /* TT_MAC_LANGID_AZERBAIJANI_ROMAN_SCRIPT */
|
|
|
|
};
|
|
|
|
|
|
|
|
static inline WORD get_mac_code_page( const FT_SfntName *name )
|
|
|
|
{
|
|
|
|
if (name->encoding_id == TT_MAC_ID_SIMPLIFIED_CHINESE) return 10008; /* special case */
|
|
|
|
return 10000 + name->encoding_id;
|
|
|
|
}
|
|
|
|
|
2013-06-20 13:36:34 +02:00
|
|
|
static int match_name_table_language( const FT_SfntName *name, LANGID lang )
|
2006-04-05 13:45:25 +02:00
|
|
|
{
|
2013-06-20 13:36:34 +02:00
|
|
|
LANGID name_lang;
|
2013-08-16 19:11:33 +02:00
|
|
|
int res = 0;
|
2006-04-05 13:45:25 +02:00
|
|
|
|
2013-06-20 13:36:34 +02:00
|
|
|
switch (name->platform_id)
|
2006-04-05 13:45:25 +02:00
|
|
|
{
|
2013-06-20 13:36:34 +02:00
|
|
|
case TT_PLATFORM_MICROSOFT:
|
2013-08-16 19:11:33 +02:00
|
|
|
res += 5; /* prefer the Microsoft name */
|
2013-06-20 13:36:34 +02:00
|
|
|
switch (name->encoding_id)
|
2006-04-05 13:45:25 +02:00
|
|
|
{
|
2013-06-20 13:36:34 +02:00
|
|
|
case TT_MS_ID_UNICODE_CS:
|
|
|
|
case TT_MS_ID_SYMBOL_CS:
|
|
|
|
name_lang = name->language_id;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return 0;
|
2006-04-05 13:45:25 +02:00
|
|
|
}
|
2013-06-20 13:36:34 +02:00
|
|
|
break;
|
2013-06-20 15:04:11 +02:00
|
|
|
case TT_PLATFORM_MACINTOSH:
|
|
|
|
if (!IsValidCodePage( get_mac_code_page( name ))) return 0;
|
2018-10-05 21:35:35 +02:00
|
|
|
if (name->language_id >= ARRAY_SIZE( mac_langid_table )) return 0;
|
2013-06-20 15:04:11 +02:00
|
|
|
name_lang = mac_langid_table[name->language_id];
|
|
|
|
break;
|
2013-06-20 21:07:33 +02:00
|
|
|
case TT_PLATFORM_APPLE_UNICODE:
|
2013-08-16 19:11:33 +02:00
|
|
|
res += 2; /* prefer Unicode encodings */
|
2013-06-20 21:07:33 +02:00
|
|
|
switch (name->encoding_id)
|
|
|
|
{
|
|
|
|
case TT_APPLE_ID_DEFAULT:
|
|
|
|
case TT_APPLE_ID_ISO_10646:
|
|
|
|
case TT_APPLE_ID_UNICODE_2_0:
|
2018-10-05 21:35:35 +02:00
|
|
|
if (name->language_id >= ARRAY_SIZE( mac_langid_table )) return 0;
|
2013-06-20 21:07:33 +02:00
|
|
|
name_lang = mac_langid_table[name->language_id];
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
break;
|
2013-06-20 13:36:34 +02:00
|
|
|
default:
|
|
|
|
return 0;
|
2006-04-05 13:45:25 +02:00
|
|
|
}
|
2013-08-16 19:11:33 +02:00
|
|
|
if (name_lang == lang) res += 30;
|
|
|
|
else if (PRIMARYLANGID( name_lang ) == PRIMARYLANGID( lang )) res += 20;
|
|
|
|
else if (name_lang == MAKELANGID( LANG_ENGLISH, SUBLANG_DEFAULT )) res += 10;
|
2020-09-17 19:30:43 +02:00
|
|
|
else if (lang == MAKELANGID( LANG_NEUTRAL, SUBLANG_NEUTRAL )) res += 5 * (0x100000 - name_lang);
|
2013-08-16 19:11:33 +02:00
|
|
|
return res;
|
2013-06-20 13:36:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static WCHAR *copy_name_table_string( const FT_SfntName *name )
|
|
|
|
{
|
|
|
|
WCHAR *ret;
|
2013-06-20 15:04:11 +02:00
|
|
|
WORD codepage;
|
2013-06-20 13:36:34 +02:00
|
|
|
int i;
|
|
|
|
|
2013-06-20 15:04:11 +02:00
|
|
|
switch (name->platform_id)
|
|
|
|
{
|
2013-06-20 21:07:33 +02:00
|
|
|
case TT_PLATFORM_APPLE_UNICODE:
|
2013-06-20 15:04:11 +02:00
|
|
|
case TT_PLATFORM_MICROSOFT:
|
|
|
|
ret = HeapAlloc( GetProcessHeap(), 0, name->string_len + sizeof(WCHAR) );
|
|
|
|
for (i = 0; i < name->string_len / 2; i++)
|
|
|
|
ret[i] = (name->string[i * 2] << 8) | name->string[i * 2 + 1];
|
|
|
|
ret[i] = 0;
|
|
|
|
return ret;
|
|
|
|
case TT_PLATFORM_MACINTOSH:
|
|
|
|
codepage = get_mac_code_page( name );
|
|
|
|
i = MultiByteToWideChar( codepage, 0, (char *)name->string, name->string_len, NULL, 0 );
|
|
|
|
ret = HeapAlloc( GetProcessHeap(), 0, (i + 1) * sizeof(WCHAR) );
|
|
|
|
MultiByteToWideChar( codepage, 0, (char *)name->string, name->string_len, ret, i );
|
|
|
|
ret[i] = 0;
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
return NULL;
|
2008-09-18 13:23:08 +02:00
|
|
|
}
|
2006-04-05 13:45:25 +02:00
|
|
|
|
2013-06-20 13:36:34 +02:00
|
|
|
static WCHAR *get_face_name(FT_Face ft_face, FT_UShort name_id, LANGID language_id)
|
2008-09-18 13:23:08 +02:00
|
|
|
{
|
|
|
|
FT_SfntName name;
|
2013-06-20 13:36:34 +02:00
|
|
|
FT_UInt num_names, name_index;
|
|
|
|
int res, best_lang = 0, best_index = -1;
|
2008-09-18 13:23:08 +02:00
|
|
|
|
2013-06-20 13:36:34 +02:00
|
|
|
if (!FT_IS_SFNT(ft_face)) return NULL;
|
2008-09-18 13:23:08 +02:00
|
|
|
|
2013-06-20 13:36:34 +02:00
|
|
|
num_names = pFT_Get_Sfnt_Name_Count( ft_face );
|
2008-09-18 13:23:08 +02:00
|
|
|
|
2013-06-20 13:36:34 +02:00
|
|
|
for (name_index = 0; name_index < num_names; name_index++)
|
|
|
|
{
|
|
|
|
if (pFT_Get_Sfnt_Name( ft_face, name_index, &name )) continue;
|
|
|
|
if (name.name_id != name_id) continue;
|
|
|
|
res = match_name_table_language( &name, language_id );
|
|
|
|
if (res > best_lang)
|
2008-09-18 13:23:08 +02:00
|
|
|
{
|
2013-06-20 13:36:34 +02:00
|
|
|
best_lang = res;
|
|
|
|
best_index = name_index;
|
2008-09-18 13:23:08 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-06-20 13:36:34 +02:00
|
|
|
if (best_index != -1 && !pFT_Get_Sfnt_Name( ft_face, best_index, &name ))
|
|
|
|
{
|
|
|
|
WCHAR *ret = copy_name_table_string( &name );
|
|
|
|
TRACE( "name %u found platform %u lang %04x %s\n",
|
|
|
|
name_id, name.platform_id, name.language_id, debugstr_w( ret ));
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
return NULL;
|
2006-04-05 13:45:25 +02:00
|
|
|
}
|
|
|
|
|
2020-09-07 15:18:54 +02:00
|
|
|
static WCHAR *ft_face_get_family_name( FT_Face ft_face, LANGID langid )
|
|
|
|
{
|
|
|
|
WCHAR *family_name;
|
|
|
|
|
|
|
|
if ((family_name = get_face_name( ft_face, TT_NAME_ID_FONT_FAMILY, langid )))
|
|
|
|
return family_name;
|
|
|
|
|
|
|
|
return towstr( CP_ACP, ft_face->family_name );
|
|
|
|
}
|
|
|
|
|
|
|
|
static WCHAR *ft_face_get_style_name( FT_Face ft_face, LANGID langid )
|
|
|
|
{
|
|
|
|
WCHAR *style_name;
|
|
|
|
|
|
|
|
if ((style_name = get_face_name( ft_face, TT_NAME_ID_FONT_SUBFAMILY, langid )))
|
|
|
|
return style_name;
|
|
|
|
|
|
|
|
return towstr( CP_ACP, ft_face->style_name );
|
|
|
|
}
|
|
|
|
|
2020-09-07 15:18:56 +02:00
|
|
|
static WCHAR *ft_face_get_full_name( FT_Face ft_face, LANGID langid )
|
|
|
|
{
|
|
|
|
static const WCHAR space_w[] = {' ',0};
|
|
|
|
WCHAR *full_name, *style_name;
|
|
|
|
SIZE_T length;
|
|
|
|
|
|
|
|
if ((full_name = get_face_name( ft_face, TT_NAME_ID_FULL_NAME, langid )))
|
|
|
|
return full_name;
|
|
|
|
|
|
|
|
full_name = ft_face_get_family_name( ft_face, langid );
|
|
|
|
style_name = ft_face_get_style_name( ft_face, langid );
|
|
|
|
|
|
|
|
length = strlenW( full_name ) + strlenW( space_w ) + strlenW( style_name ) + 1;
|
|
|
|
full_name = HeapReAlloc( GetProcessHeap(), 0, full_name, length * sizeof(WCHAR) );
|
|
|
|
|
|
|
|
strcatW( full_name, space_w );
|
|
|
|
strcatW( full_name, style_name );
|
|
|
|
HeapFree( GetProcessHeap(), 0, style_name );
|
|
|
|
|
|
|
|
WARN( "full name not found, using %s instead\n", debugstr_w(full_name) );
|
|
|
|
return full_name;
|
|
|
|
}
|
|
|
|
|
2012-03-30 10:52:02 +02:00
|
|
|
static inline BOOL faces_equal( const Face *f1, const Face *f2 )
|
|
|
|
{
|
2020-09-07 15:18:59 +02:00
|
|
|
if (strcmpiW( f1->full_name, f2->full_name )) return FALSE;
|
2012-03-30 10:52:02 +02:00
|
|
|
if (f1->scalable) return TRUE;
|
2012-04-09 11:24:53 +02:00
|
|
|
if (f1->size.y_ppem != f2->size.y_ppem) return FALSE;
|
2012-03-30 10:52:02 +02:00
|
|
|
return !memcmp( &f1->fs, &f2->fs, sizeof(f1->fs) );
|
|
|
|
}
|
|
|
|
|
2013-01-14 15:19:14 +01:00
|
|
|
static void release_face( Face *face )
|
2012-03-30 10:52:02 +02:00
|
|
|
{
|
2013-01-14 15:19:14 +01:00
|
|
|
if (--face->refcount) return;
|
|
|
|
if (face->family)
|
2012-03-30 10:52:02 +02:00
|
|
|
{
|
2013-01-15 12:34:40 +01:00
|
|
|
if (face->flags & ADDFONT_ADD_TO_CACHE) remove_face_from_cache( face );
|
2012-03-30 10:52:02 +02:00
|
|
|
list_remove( &face->entry );
|
2013-01-14 15:19:14 +01:00
|
|
|
release_family( face->family );
|
2012-03-30 10:52:02 +02:00
|
|
|
}
|
2013-01-14 15:19:14 +01:00
|
|
|
HeapFree( GetProcessHeap(), 0, face->file );
|
2020-09-07 15:18:57 +02:00
|
|
|
HeapFree( GetProcessHeap(), 0, face->style_name );
|
2020-09-07 15:18:56 +02:00
|
|
|
HeapFree( GetProcessHeap(), 0, face->full_name );
|
2013-01-14 15:19:14 +01:00
|
|
|
HeapFree( GetProcessHeap(), 0, face->cached_enum_data );
|
|
|
|
HeapFree( GetProcessHeap(), 0, face );
|
2012-03-30 10:52:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline int style_order(const Face *face)
|
|
|
|
{
|
|
|
|
switch (face->ntmFlags & (NTM_REGULAR | NTM_BOLD | NTM_ITALIC))
|
|
|
|
{
|
|
|
|
case NTM_REGULAR:
|
|
|
|
return 0;
|
|
|
|
case NTM_BOLD:
|
|
|
|
return 1;
|
|
|
|
case NTM_ITALIC:
|
|
|
|
return 2;
|
|
|
|
case NTM_BOLD | NTM_ITALIC:
|
|
|
|
return 3;
|
|
|
|
default:
|
2020-09-07 15:18:56 +02:00
|
|
|
WARN( "Don't know how to order face %s with flags 0x%08x\n", debugstr_w(face->full_name), face->ntmFlags );
|
2012-03-30 10:52:02 +02:00
|
|
|
return 9999;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static BOOL insert_face_in_family_list( Face *face, Family *family )
|
|
|
|
{
|
|
|
|
Face *cursor;
|
|
|
|
|
|
|
|
LIST_FOR_EACH_ENTRY( cursor, &family->faces, Face, entry )
|
|
|
|
{
|
|
|
|
if (faces_equal( face, cursor ))
|
|
|
|
{
|
2020-10-30 14:31:13 +01:00
|
|
|
TRACE( "Already loaded face %s in family %s, original version %x, new version %x\n",
|
2020-09-07 15:18:58 +02:00
|
|
|
debugstr_w(face->full_name), debugstr_w(family->family_name),
|
2020-10-30 14:31:13 +01:00
|
|
|
cursor->version, face->version );
|
2012-03-30 10:52:02 +02:00
|
|
|
|
2020-10-06 15:40:25 +02:00
|
|
|
if (face->file && !strcmpiW( face->file, cursor->file ))
|
2013-03-04 11:35:50 +01:00
|
|
|
{
|
|
|
|
cursor->refcount++;
|
|
|
|
TRACE("Font %s already in list, refcount now %d\n",
|
|
|
|
debugstr_w(face->file), cursor->refcount);
|
|
|
|
return FALSE;
|
|
|
|
}
|
2020-10-30 14:31:13 +01:00
|
|
|
if (face->version <= cursor->version)
|
2012-03-30 10:52:02 +02:00
|
|
|
{
|
2012-04-09 11:25:11 +02:00
|
|
|
TRACE("Original font %s is newer so skipping %s\n",
|
2012-11-01 13:16:26 +01:00
|
|
|
debugstr_w(cursor->file), debugstr_w(face->file));
|
2012-03-30 10:52:02 +02:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2012-04-09 11:25:11 +02:00
|
|
|
TRACE("Replacing original %s with %s\n",
|
2012-11-01 13:16:26 +01:00
|
|
|
debugstr_w(cursor->file), debugstr_w(face->file));
|
2012-03-30 10:52:02 +02:00
|
|
|
list_add_before( &cursor->entry, &face->entry );
|
|
|
|
face->family = family;
|
2013-01-14 15:19:14 +01:00
|
|
|
family->refcount++;
|
|
|
|
face->refcount++;
|
|
|
|
release_face( cursor );
|
2012-03-30 10:52:02 +02:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (style_order( face ) < style_order( cursor )) break;
|
|
|
|
}
|
|
|
|
|
2020-09-07 15:18:56 +02:00
|
|
|
TRACE( "Adding face %s in family %s from %s\n", debugstr_w(face->full_name),
|
2020-09-07 15:18:58 +02:00
|
|
|
debugstr_w(family->family_name), debugstr_w(face->file) );
|
2012-03-30 10:52:02 +02:00
|
|
|
list_add_before( &cursor->entry, &face->entry );
|
|
|
|
face->family = family;
|
2013-01-14 15:19:14 +01:00
|
|
|
family->refcount++;
|
|
|
|
face->refcount++;
|
2012-03-30 10:52:02 +02:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2020-10-06 19:58:11 +02:00
|
|
|
struct cached_face
|
2011-10-06 23:26:07 +02:00
|
|
|
{
|
2020-10-30 14:31:13 +01:00
|
|
|
DWORD index;
|
|
|
|
DWORD flags;
|
|
|
|
DWORD ntmflags;
|
|
|
|
DWORD version;
|
|
|
|
struct bitmap_font_size size;
|
|
|
|
FONTSIGNATURE fs;
|
|
|
|
WCHAR full_name[1];
|
|
|
|
/* WCHAR file_name[]; */
|
2020-10-06 19:58:11 +02:00
|
|
|
};
|
2011-10-06 23:26:06 +02:00
|
|
|
|
2020-10-06 19:58:11 +02:00
|
|
|
static void load_face(HKEY hkey_family, Family *family, void *buffer, DWORD buffer_size, BOOL scalable)
|
2011-10-06 23:26:07 +02:00
|
|
|
{
|
2020-10-06 19:58:11 +02:00
|
|
|
DWORD type, size, needed, index = 0;
|
|
|
|
Face *face;
|
2012-11-01 12:35:32 +01:00
|
|
|
HKEY hkey_strike;
|
2020-10-06 19:58:11 +02:00
|
|
|
WCHAR name[256];
|
|
|
|
struct cached_face *cached = (struct cached_face *)buffer;
|
2011-10-06 23:26:07 +02:00
|
|
|
|
2020-10-06 19:58:11 +02:00
|
|
|
size = sizeof(name);
|
|
|
|
needed = buffer_size - sizeof(DWORD);
|
|
|
|
while (!RegEnumValueW( hkey_family, index++, name, &size, NULL, &type, buffer, &needed ))
|
2011-10-06 23:26:07 +02:00
|
|
|
{
|
2020-10-06 19:58:11 +02:00
|
|
|
if (type == REG_BINARY && needed > sizeof(*cached))
|
2020-09-07 15:18:56 +02:00
|
|
|
{
|
2020-10-06 19:58:11 +02:00
|
|
|
((DWORD *)buffer)[needed / sizeof(DWORD)] = 0;
|
|
|
|
|
2020-10-30 14:31:13 +01:00
|
|
|
face = create_face( name, cached->full_name, cached->full_name + strlenW(cached->full_name) + 1,
|
|
|
|
cached->index, cached->fs, cached->ntmflags, cached->version,
|
|
|
|
cached->flags, scalable ? NULL : &cached->size );
|
|
|
|
if (!scalable)
|
|
|
|
TRACE("Adding bitmap size h %d w %d size %d x_ppem %d y_ppem %d\n",
|
2020-10-06 19:58:11 +02:00
|
|
|
face->size.height, face->size.width, face->size.size >> 6,
|
|
|
|
face->size.x_ppem >> 6, face->size.y_ppem >> 6);
|
|
|
|
|
|
|
|
TRACE("fsCsb = %08x %08x/%08x %08x %08x %08x\n",
|
|
|
|
face->fs.fsCsb[0], face->fs.fsCsb[1],
|
|
|
|
face->fs.fsUsb[0], face->fs.fsUsb[1],
|
|
|
|
face->fs.fsUsb[2], face->fs.fsUsb[3]);
|
|
|
|
|
|
|
|
if (insert_face_in_family_list(face, family))
|
|
|
|
TRACE( "Added face %s to family %s\n", debugstr_w(face->full_name), debugstr_w(family->family_name) );
|
2011-10-06 23:26:07 +02:00
|
|
|
|
2020-10-06 19:58:11 +02:00
|
|
|
release_face( face );
|
2011-10-06 23:26:07 +02:00
|
|
|
}
|
2020-10-06 19:58:11 +02:00
|
|
|
size = sizeof(name);
|
|
|
|
needed = buffer_size - sizeof(DWORD);
|
2011-10-06 23:26:07 +02:00
|
|
|
}
|
|
|
|
|
2012-11-01 12:35:32 +01:00
|
|
|
/* load bitmap strikes */
|
2011-10-06 23:26:07 +02:00
|
|
|
|
2020-10-06 19:58:11 +02:00
|
|
|
index = 0;
|
2012-11-01 12:35:32 +01:00
|
|
|
needed = buffer_size;
|
2020-10-06 19:58:11 +02:00
|
|
|
while (!RegEnumKeyExW(hkey_family, index++, buffer, &needed, NULL, NULL, NULL, NULL))
|
2012-11-01 12:35:32 +01:00
|
|
|
{
|
2020-10-06 19:58:11 +02:00
|
|
|
if (!RegOpenKeyExW(hkey_family, buffer, 0, KEY_ALL_ACCESS, &hkey_strike))
|
2011-10-06 23:26:07 +02:00
|
|
|
{
|
2020-10-06 19:58:11 +02:00
|
|
|
load_face(hkey_strike, family, buffer, buffer_size, FALSE);
|
2011-10-06 23:26:07 +02:00
|
|
|
RegCloseKey(hkey_strike);
|
|
|
|
}
|
2012-11-01 12:35:32 +01:00
|
|
|
needed = buffer_size;
|
2011-10-06 23:26:07 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-06-20 12:12:16 +02:00
|
|
|
/* move vertical fonts after their horizontal counterpart */
|
|
|
|
/* assumes that font_list is already sorted by family name */
|
|
|
|
static void reorder_vertical_fonts(void)
|
|
|
|
{
|
|
|
|
Family *family, *next, *vert_family;
|
|
|
|
struct list *ptr, *vptr;
|
|
|
|
struct list vertical_families = LIST_INIT( vertical_families );
|
|
|
|
|
|
|
|
LIST_FOR_EACH_ENTRY_SAFE( family, next, &font_list, Family, entry )
|
|
|
|
{
|
2020-09-07 15:18:58 +02:00
|
|
|
if (family->family_name[0] != '@') continue;
|
2013-06-20 12:12:16 +02:00
|
|
|
list_remove( &family->entry );
|
|
|
|
list_add_tail( &vertical_families, &family->entry );
|
|
|
|
}
|
|
|
|
|
|
|
|
ptr = list_head( &font_list );
|
|
|
|
vptr = list_head( &vertical_families );
|
|
|
|
while (ptr && vptr)
|
|
|
|
{
|
|
|
|
family = LIST_ENTRY( ptr, Family, entry );
|
|
|
|
vert_family = LIST_ENTRY( vptr, Family, entry );
|
2020-09-07 15:18:58 +02:00
|
|
|
if (strcmpiW( family->family_name, vert_family->family_name + 1 ) > 0)
|
2013-06-20 12:12:16 +02:00
|
|
|
{
|
|
|
|
list_remove( vptr );
|
|
|
|
list_add_before( ptr, vptr );
|
|
|
|
vptr = list_head( &vertical_families );
|
|
|
|
}
|
|
|
|
else ptr = list_next( &font_list, ptr );
|
|
|
|
}
|
|
|
|
list_move_tail( &font_list, &vertical_families );
|
|
|
|
}
|
|
|
|
|
2011-10-06 23:26:07 +02:00
|
|
|
static void load_font_list_from_cache(HKEY hkey_font_cache)
|
|
|
|
{
|
2012-11-01 12:35:32 +01:00
|
|
|
DWORD size, family_index = 0;
|
2011-10-06 23:26:07 +02:00
|
|
|
Family *family;
|
|
|
|
HKEY hkey_family;
|
2012-11-01 12:35:32 +01:00
|
|
|
WCHAR buffer[4096];
|
2011-10-06 23:26:07 +02:00
|
|
|
|
2012-11-01 12:35:32 +01:00
|
|
|
size = sizeof(buffer);
|
|
|
|
while (!RegEnumKeyExW(hkey_font_cache, family_index++, buffer, &size, NULL, NULL, NULL, NULL))
|
2011-10-06 23:26:07 +02:00
|
|
|
{
|
2020-09-17 19:30:43 +02:00
|
|
|
WCHAR *second_name = NULL;
|
2012-11-01 12:35:32 +01:00
|
|
|
WCHAR *family_name = strdupW( buffer );
|
2011-10-06 23:26:07 +02:00
|
|
|
|
|
|
|
RegOpenKeyExW(hkey_font_cache, family_name, 0, KEY_ALL_ACCESS, &hkey_family);
|
|
|
|
TRACE("opened family key %s\n", debugstr_w(family_name));
|
2012-11-01 12:35:32 +01:00
|
|
|
size = sizeof(buffer);
|
2020-10-06 19:58:11 +02:00
|
|
|
if (!RegQueryValueExW( hkey_family, NULL, NULL, NULL, (BYTE *)buffer, &size ))
|
2020-09-17 19:30:43 +02:00
|
|
|
second_name = strdupW( buffer );
|
2011-10-06 23:26:07 +02:00
|
|
|
|
2020-09-17 19:30:43 +02:00
|
|
|
family = create_family( family_name, second_name );
|
2011-10-06 23:26:07 +02:00
|
|
|
|
2020-10-06 19:58:11 +02:00
|
|
|
load_face(hkey_family, family, buffer, sizeof(buffer), TRUE);
|
2020-09-07 15:18:58 +02:00
|
|
|
|
|
|
|
HeapFree( GetProcessHeap(), 0, family_name );
|
2020-09-17 19:30:43 +02:00
|
|
|
HeapFree( GetProcessHeap(), 0, second_name );
|
2020-09-07 15:18:58 +02:00
|
|
|
|
2011-10-06 23:26:07 +02:00
|
|
|
RegCloseKey(hkey_family);
|
2013-01-14 15:19:14 +01:00
|
|
|
release_family( family );
|
2012-11-01 12:35:32 +01:00
|
|
|
size = sizeof(buffer);
|
2011-10-06 23:26:07 +02:00
|
|
|
}
|
2013-06-20 12:12:16 +02:00
|
|
|
|
|
|
|
reorder_vertical_fonts();
|
2011-10-06 23:26:07 +02:00
|
|
|
}
|
|
|
|
|
2011-10-06 23:26:06 +02:00
|
|
|
static LONG create_font_cache_key(HKEY *hkey, DWORD *disposition)
|
|
|
|
{
|
|
|
|
LONG ret;
|
|
|
|
HKEY hkey_wine_fonts;
|
|
|
|
|
|
|
|
/* We don't want to create the fonts key as volatile, so open this first */
|
|
|
|
ret = RegCreateKeyExW(HKEY_CURRENT_USER, wine_fonts_key, 0, NULL, 0,
|
|
|
|
KEY_ALL_ACCESS, NULL, &hkey_wine_fonts, NULL);
|
|
|
|
if(ret != ERROR_SUCCESS)
|
|
|
|
{
|
|
|
|
WARN("Can't create %s\n", debugstr_w(wine_fonts_key));
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = RegCreateKeyExW(hkey_wine_fonts, wine_fonts_cache_key, 0, NULL, REG_OPTION_VOLATILE,
|
|
|
|
KEY_ALL_ACCESS, NULL, hkey, disposition);
|
|
|
|
RegCloseKey(hkey_wine_fonts);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void add_face_to_cache(Face *face)
|
|
|
|
{
|
2013-01-15 13:56:52 +01:00
|
|
|
HKEY hkey_family, hkey_face;
|
2020-10-06 19:58:11 +02:00
|
|
|
DWORD len, buffer[1024];
|
|
|
|
struct cached_face *cached = (struct cached_face *)buffer;
|
2011-10-06 23:26:06 +02:00
|
|
|
|
2020-09-07 15:18:58 +02:00
|
|
|
RegCreateKeyExW( hkey_font_cache, face->family->family_name, 0, NULL, REG_OPTION_VOLATILE,
|
|
|
|
KEY_ALL_ACCESS, NULL, &hkey_family, NULL );
|
2020-09-17 19:30:43 +02:00
|
|
|
if (face->family->second_name[0])
|
2020-10-06 19:58:11 +02:00
|
|
|
RegSetValueExW( hkey_family, NULL, 0, REG_SZ, (BYTE *)face->family->second_name,
|
2020-09-17 19:30:43 +02:00
|
|
|
(strlenW( face->family->second_name ) + 1) * sizeof(WCHAR) );
|
2011-10-06 23:26:06 +02:00
|
|
|
|
2020-10-06 19:58:11 +02:00
|
|
|
if (!face->scalable)
|
2011-10-06 23:26:06 +02:00
|
|
|
{
|
2020-10-06 19:58:11 +02:00
|
|
|
static const WCHAR fmtW[] = {'%','d',0};
|
|
|
|
WCHAR name[10];
|
2011-10-06 23:26:06 +02:00
|
|
|
|
2020-10-06 19:58:11 +02:00
|
|
|
sprintfW( name, fmtW, face->size.y_ppem );
|
|
|
|
RegCreateKeyExW( hkey_family, name, 0, NULL, REG_OPTION_VOLATILE, KEY_ALL_ACCESS,
|
|
|
|
NULL, &hkey_face, NULL);
|
|
|
|
}
|
|
|
|
else hkey_face = hkey_family;
|
|
|
|
|
|
|
|
memset( cached, 0, sizeof(*cached) );
|
|
|
|
cached->index = face->face_index;
|
|
|
|
cached->flags = face->flags;
|
|
|
|
cached->ntmflags = face->ntmFlags;
|
2020-10-30 14:31:13 +01:00
|
|
|
cached->version = face->version;
|
2020-10-06 19:58:11 +02:00
|
|
|
cached->fs = face->fs;
|
2020-10-30 14:31:13 +01:00
|
|
|
if (!face->scalable) cached->size = face->size;
|
2020-10-06 19:58:11 +02:00
|
|
|
strcpyW( cached->full_name, face->full_name );
|
|
|
|
len = strlenW( face->full_name ) + 1;
|
|
|
|
strcpyW( cached->full_name + len, face->file );
|
|
|
|
len += strlenW( face->file ) + 1;
|
|
|
|
|
|
|
|
RegSetValueExW( hkey_face, face->style_name, 0, REG_BINARY, (BYTE *)cached,
|
|
|
|
offsetof( struct cached_face, full_name[len] ));
|
|
|
|
|
|
|
|
if (hkey_face != hkey_family) RegCloseKey(hkey_face);
|
2011-10-06 23:26:06 +02:00
|
|
|
RegCloseKey(hkey_family);
|
|
|
|
}
|
2006-04-05 13:45:25 +02:00
|
|
|
|
2013-01-15 12:34:40 +01:00
|
|
|
static void remove_face_from_cache( Face *face )
|
|
|
|
{
|
|
|
|
HKEY hkey_family;
|
|
|
|
|
2020-09-07 15:18:58 +02:00
|
|
|
RegOpenKeyExW( hkey_font_cache, face->family->family_name, 0, KEY_ALL_ACCESS, &hkey_family );
|
2013-01-15 12:34:40 +01:00
|
|
|
|
|
|
|
if (face->scalable)
|
|
|
|
{
|
2020-10-06 19:58:11 +02:00
|
|
|
RegDeleteValueW( hkey_family, face->style_name );
|
2013-01-15 12:34:40 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-10-06 19:58:11 +02:00
|
|
|
static const WCHAR fmtW[] = {'%','d',0};
|
|
|
|
WCHAR name[10];
|
|
|
|
sprintfW( name, fmtW, face->size.y_ppem );
|
|
|
|
RegDeleteKeyW( hkey_family, name );
|
2013-01-15 12:34:40 +01:00
|
|
|
}
|
|
|
|
RegCloseKey(hkey_family);
|
|
|
|
}
|
|
|
|
|
2020-09-07 15:18:55 +02:00
|
|
|
static WCHAR *get_vertical_name( WCHAR *name )
|
2011-12-20 13:05:53 +01:00
|
|
|
{
|
2020-09-07 15:18:55 +02:00
|
|
|
SIZE_T length;
|
|
|
|
if (!name) return NULL;
|
|
|
|
if (name[0] == '@') return name;
|
2011-12-20 13:05:53 +01:00
|
|
|
|
2020-09-07 15:18:55 +02:00
|
|
|
length = strlenW( name ) + 1;
|
|
|
|
name = HeapReAlloc( GetProcessHeap(), 0, name, (length + 1) * sizeof(WCHAR) );
|
|
|
|
memmove( name + 1, name, length * sizeof(WCHAR) );
|
|
|
|
name[0] = '@';
|
|
|
|
return name;
|
2011-12-20 13:05:53 +01:00
|
|
|
}
|
|
|
|
|
2020-09-07 15:18:54 +02:00
|
|
|
static Family *get_family( FT_Face ft_face, BOOL vertical )
|
2012-03-27 12:48:38 +02:00
|
|
|
{
|
2020-09-07 15:18:54 +02:00
|
|
|
Family *family;
|
2020-09-17 19:30:43 +02:00
|
|
|
WCHAR *family_name, *second_name;
|
2012-03-27 12:48:38 +02:00
|
|
|
|
2020-09-07 15:18:54 +02:00
|
|
|
family_name = ft_face_get_family_name( ft_face, GetSystemDefaultLCID() );
|
2020-09-17 19:30:43 +02:00
|
|
|
second_name = ft_face_get_family_name( ft_face, MAKELANGID(LANG_ENGLISH, SUBLANG_DEFAULT) );
|
2020-09-07 15:18:54 +02:00
|
|
|
|
2020-09-17 19:30:43 +02:00
|
|
|
/* try to find another secondary name, preferring the lowest langids */
|
|
|
|
if (!strcmpiW( family_name, second_name ))
|
2012-03-27 12:48:38 +02:00
|
|
|
{
|
2020-09-17 19:30:43 +02:00
|
|
|
HeapFree( GetProcessHeap(), 0, second_name );
|
|
|
|
second_name = ft_face_get_family_name( ft_face, MAKELANGID(LANG_NEUTRAL, SUBLANG_NEUTRAL) );
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!strcmpiW( family_name, second_name ))
|
|
|
|
{
|
|
|
|
HeapFree( GetProcessHeap(), 0, second_name );
|
|
|
|
second_name = NULL;
|
2012-03-27 12:48:38 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (vertical)
|
|
|
|
{
|
2020-09-07 15:18:55 +02:00
|
|
|
family_name = get_vertical_name( family_name );
|
2020-09-17 19:30:43 +02:00
|
|
|
second_name = get_vertical_name( second_name );
|
2012-03-27 12:48:38 +02:00
|
|
|
}
|
2012-03-27 12:48:39 +02:00
|
|
|
|
2020-09-07 15:18:58 +02:00
|
|
|
if ((family = find_family_from_name( family_name ))) family->refcount++;
|
2020-10-30 14:15:34 +01:00
|
|
|
else family = create_family( family_name, second_name );
|
2012-03-27 12:48:39 +02:00
|
|
|
|
2020-09-07 15:18:58 +02:00
|
|
|
HeapFree( GetProcessHeap(), 0, family_name );
|
2020-09-17 19:30:43 +02:00
|
|
|
HeapFree( GetProcessHeap(), 0, second_name );
|
2020-09-07 15:18:58 +02:00
|
|
|
|
2012-03-27 12:48:39 +02:00
|
|
|
return family;
|
|
|
|
}
|
|
|
|
|
2012-03-27 12:48:41 +02:00
|
|
|
static inline FT_Fixed get_font_version( FT_Face ft_face )
|
|
|
|
{
|
|
|
|
FT_Fixed version = 0;
|
|
|
|
TT_Header *header;
|
|
|
|
|
|
|
|
header = pFT_Get_Sfnt_Table( ft_face, ft_sfnt_head );
|
|
|
|
if (header) version = header->Font_Revision;
|
|
|
|
|
|
|
|
return version;
|
|
|
|
}
|
2012-03-27 12:48:39 +02:00
|
|
|
|
2012-03-27 12:48:42 +02:00
|
|
|
static inline DWORD get_ntm_flags( FT_Face ft_face )
|
|
|
|
{
|
|
|
|
DWORD flags = 0;
|
|
|
|
FT_ULong table_size = 0;
|
2015-12-31 05:33:15 +01:00
|
|
|
FT_WinFNT_HeaderRec winfnt_header;
|
2012-03-27 12:48:42 +02:00
|
|
|
|
|
|
|
if (ft_face->style_flags & FT_STYLE_FLAG_ITALIC) flags |= NTM_ITALIC;
|
|
|
|
if (ft_face->style_flags & FT_STYLE_FLAG_BOLD) flags |= NTM_BOLD;
|
2015-12-31 05:33:15 +01:00
|
|
|
|
|
|
|
/* fixup the flag for our fake-bold implementation. */
|
|
|
|
if (!FT_IS_SCALABLE( ft_face ) &&
|
|
|
|
!pFT_Get_WinFNT_Header( ft_face, &winfnt_header ) &&
|
|
|
|
winfnt_header.weight > FW_NORMAL )
|
|
|
|
flags |= NTM_BOLD;
|
|
|
|
|
2012-03-27 12:48:42 +02:00
|
|
|
if (flags == 0) flags = NTM_REGULAR;
|
|
|
|
|
|
|
|
if (!pFT_Load_Sfnt_Table( ft_face, FT_MAKE_TAG( 'C','F','F',' ' ), 0, NULL, &table_size ))
|
|
|
|
flags |= NTM_PS_OPENTYPE;
|
|
|
|
|
|
|
|
return flags;
|
|
|
|
}
|
|
|
|
|
2020-10-30 14:31:13 +01:00
|
|
|
static inline void get_bitmap_size( FT_Face ft_face, struct bitmap_font_size *face_size )
|
2012-03-27 12:48:43 +02:00
|
|
|
{
|
2013-07-31 16:16:26 +02:00
|
|
|
My_FT_Bitmap_Size *size;
|
2012-03-27 12:48:43 +02:00
|
|
|
FT_WinFNT_HeaderRec winfnt_header;
|
|
|
|
|
2013-07-31 16:16:26 +02:00
|
|
|
size = (My_FT_Bitmap_Size *)ft_face->available_sizes;
|
|
|
|
TRACE("Adding bitmap size h %d w %d size %ld x_ppem %ld y_ppem %ld\n",
|
|
|
|
size->height, size->width, size->size >> 6,
|
|
|
|
size->x_ppem >> 6, size->y_ppem >> 6);
|
|
|
|
face_size->height = size->height;
|
|
|
|
face_size->width = size->width;
|
|
|
|
face_size->size = size->size;
|
|
|
|
face_size->x_ppem = size->x_ppem;
|
|
|
|
face_size->y_ppem = size->y_ppem;
|
2012-03-27 12:48:43 +02:00
|
|
|
|
2013-07-31 16:16:28 +02:00
|
|
|
if (!pFT_Get_WinFNT_Header( ft_face, &winfnt_header )) {
|
2013-07-31 16:16:26 +02:00
|
|
|
face_size->internal_leading = winfnt_header.internal_leading;
|
2013-07-31 16:16:28 +02:00
|
|
|
if (winfnt_header.external_leading > 0 &&
|
|
|
|
(face_size->height ==
|
|
|
|
winfnt_header.pixel_height + winfnt_header.external_leading))
|
|
|
|
face_size->height = winfnt_header.pixel_height;
|
|
|
|
}
|
2012-03-27 12:48:43 +02:00
|
|
|
}
|
|
|
|
|
2012-03-27 12:48:44 +02:00
|
|
|
static inline void get_fontsig( FT_Face ft_face, FONTSIGNATURE *fs )
|
|
|
|
{
|
|
|
|
TT_OS2 *os2;
|
|
|
|
CHARSETINFO csi;
|
|
|
|
FT_WinFNT_HeaderRec winfnt_header;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
memset( fs, 0, sizeof(*fs) );
|
|
|
|
|
|
|
|
os2 = pFT_Get_Sfnt_Table( ft_face, ft_sfnt_os2 );
|
|
|
|
if (os2)
|
|
|
|
{
|
|
|
|
fs->fsUsb[0] = os2->ulUnicodeRange1;
|
|
|
|
fs->fsUsb[1] = os2->ulUnicodeRange2;
|
|
|
|
fs->fsUsb[2] = os2->ulUnicodeRange3;
|
|
|
|
fs->fsUsb[3] = os2->ulUnicodeRange4;
|
|
|
|
|
|
|
|
if (os2->version == 0)
|
|
|
|
{
|
2018-11-29 10:28:15 +01:00
|
|
|
if (os2->usFirstCharIndex >= 0xf000 && os2->usFirstCharIndex < 0xf100)
|
2012-03-27 12:48:44 +02:00
|
|
|
fs->fsCsb[0] = FS_SYMBOL;
|
2018-11-29 10:28:15 +01:00
|
|
|
else
|
|
|
|
fs->fsCsb[0] = FS_LATIN1;
|
2012-03-27 12:48:44 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fs->fsCsb[0] = os2->ulCodePageRange1;
|
|
|
|
fs->fsCsb[1] = os2->ulCodePageRange2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (!pFT_Get_WinFNT_Header( ft_face, &winfnt_header ))
|
|
|
|
{
|
|
|
|
TRACE("pix_h %d charset %d dpi %dx%d pt %d\n", winfnt_header.pixel_height, winfnt_header.charset,
|
|
|
|
winfnt_header.vertical_resolution,winfnt_header.horizontal_resolution, winfnt_header.nominal_point_size);
|
|
|
|
if (TranslateCharsetInfo( (DWORD*)(UINT_PTR)winfnt_header.charset, &csi, TCI_SRCCHARSET ))
|
|
|
|
*fs = csi.fs;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fs->fsCsb[0] == 0)
|
|
|
|
{
|
|
|
|
/* let's see if we can find any interesting cmaps */
|
|
|
|
for (i = 0; i < ft_face->num_charmaps; i++)
|
|
|
|
{
|
|
|
|
switch (ft_face->charmaps[i]->encoding)
|
|
|
|
{
|
|
|
|
case FT_ENCODING_UNICODE:
|
|
|
|
case FT_ENCODING_APPLE_ROMAN:
|
|
|
|
fs->fsCsb[0] |= FS_LATIN1;
|
|
|
|
break;
|
|
|
|
case FT_ENCODING_MS_SYMBOL:
|
|
|
|
fs->fsCsb[0] |= FS_SYMBOL;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-30 14:31:13 +01:00
|
|
|
static Face *create_face_from_ft_face( FT_Face ft_face, FT_Long face_index,
|
|
|
|
const WCHAR *filename, DWORD flags )
|
2011-12-20 13:05:43 +01:00
|
|
|
{
|
2020-10-30 14:31:13 +01:00
|
|
|
struct bitmap_font_size size;
|
|
|
|
struct gdi_font_face *face;
|
|
|
|
FONTSIGNATURE fs;
|
|
|
|
WCHAR *style_name = ft_face_get_style_name( ft_face, GetSystemDefaultLangID() );
|
|
|
|
WCHAR *full_name = ft_face_get_full_name( ft_face, GetSystemDefaultLangID() );
|
2011-12-20 13:05:43 +01:00
|
|
|
|
2020-10-30 14:31:13 +01:00
|
|
|
if (flags & ADDFONT_VERTICAL_FONT) full_name = get_vertical_name( full_name );
|
|
|
|
get_fontsig( ft_face, &fs );
|
|
|
|
if (!FT_IS_SCALABLE( ft_face )) get_bitmap_size( ft_face, &size );
|
2013-01-14 15:28:42 +01:00
|
|
|
if (!HIWORD( flags )) flags |= ADDFONT_AA_FLAGS( default_aa_flags );
|
2020-10-30 14:31:13 +01:00
|
|
|
|
|
|
|
face = create_face( style_name, full_name, filename, face_index, fs,
|
|
|
|
get_ntm_flags( ft_face ), get_font_version( ft_face ),
|
|
|
|
flags, FT_IS_SCALABLE(ft_face) ? NULL : &size );
|
2012-03-27 12:48:47 +02:00
|
|
|
|
2012-03-27 12:48:46 +02:00
|
|
|
TRACE("fsCsb = %08x %08x/%08x %08x %08x %08x\n",
|
2020-10-30 14:31:13 +01:00
|
|
|
fs.fsCsb[0], fs.fsCsb[1], fs.fsUsb[0], fs.fsUsb[1], fs.fsUsb[2], fs.fsUsb[3]);
|
2011-12-20 13:05:43 +01:00
|
|
|
|
2020-10-30 14:31:13 +01:00
|
|
|
HeapFree( GetProcessHeap(), 0, style_name );
|
|
|
|
HeapFree( GetProcessHeap(), 0, full_name );
|
2012-03-27 12:48:47 +02:00
|
|
|
return face;
|
|
|
|
}
|
|
|
|
|
2020-10-06 15:40:25 +02:00
|
|
|
static void AddFaceToList(FT_Face ft_face, const WCHAR *file, void *font_data_ptr, DWORD font_data_size,
|
|
|
|
FT_Long face_index, DWORD flags )
|
2012-03-27 12:48:47 +02:00
|
|
|
{
|
|
|
|
Face *face;
|
|
|
|
Family *family;
|
|
|
|
|
2020-10-30 14:31:13 +01:00
|
|
|
face = create_face_from_ft_face( ft_face, face_index, file, flags );
|
|
|
|
if (face && !file)
|
|
|
|
{
|
|
|
|
face->data_ptr = font_data_ptr;
|
|
|
|
face->data_size = font_data_size;
|
|
|
|
}
|
2013-01-15 14:13:09 +01:00
|
|
|
family = get_family( ft_face, flags & ADDFONT_VERTICAL_FONT );
|
2015-01-19 15:11:51 +01:00
|
|
|
|
2013-01-14 15:19:14 +01:00
|
|
|
if (insert_face_in_family_list( face, family ))
|
2012-03-27 12:48:47 +02:00
|
|
|
{
|
2013-01-14 15:19:14 +01:00
|
|
|
if (flags & ADDFONT_ADD_TO_CACHE)
|
|
|
|
add_face_to_cache( face );
|
2020-09-07 15:18:58 +02:00
|
|
|
TRACE( "Added face %s to family %s\n", debugstr_w(face->full_name), debugstr_w(family->family_name) );
|
2013-01-14 15:19:14 +01:00
|
|
|
}
|
|
|
|
release_face( face );
|
|
|
|
release_family( family );
|
2011-12-20 13:05:43 +01:00
|
|
|
}
|
|
|
|
|
2012-03-28 15:49:23 +02:00
|
|
|
static FT_Face new_ft_face( const char *file, void *font_data_ptr, DWORD font_data_size,
|
|
|
|
FT_Long face_index, BOOL allow_bitmap )
|
|
|
|
{
|
|
|
|
FT_Error err;
|
|
|
|
TT_OS2 *pOS2;
|
|
|
|
FT_Face ft_face;
|
|
|
|
|
|
|
|
if (file)
|
|
|
|
{
|
|
|
|
TRACE("Loading font file %s index %ld\n", debugstr_a(file), face_index);
|
|
|
|
err = pFT_New_Face(library, file, face_index, &ft_face);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
TRACE("Loading font from ptr %p size %d, index %ld\n", font_data_ptr, font_data_size, face_index);
|
|
|
|
err = pFT_New_Memory_Face(library, font_data_ptr, font_data_size, face_index, &ft_face);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (err != 0)
|
|
|
|
{
|
|
|
|
WARN("Unable to load font %s/%p err = %x\n", debugstr_a(file), font_data_ptr, err);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* There are too many bugs in FreeType < 2.1.9 for bitmap font support */
|
2019-02-08 02:51:32 +01:00
|
|
|
if (!FT_IS_SCALABLE( ft_face ) && FT_SimpleVersion < FT_VERSION_VALUE(2, 1, 9))
|
2012-03-28 15:49:23 +02:00
|
|
|
{
|
|
|
|
WARN("FreeType version < 2.1.9, skipping bitmap font %s/%p\n", debugstr_a(file), font_data_ptr);
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!FT_IS_SFNT( ft_face ))
|
|
|
|
{
|
|
|
|
if (FT_IS_SCALABLE( ft_face ) || !allow_bitmap )
|
|
|
|
{
|
|
|
|
WARN("Ignoring font %s/%p\n", debugstr_a(file), font_data_ptr);
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (!(pOS2 = pFT_Get_Sfnt_Table( ft_face, ft_sfnt_os2 )) ||
|
|
|
|
!pFT_Get_Sfnt_Table( ft_face, ft_sfnt_hhea ) ||
|
|
|
|
!pFT_Get_Sfnt_Table( ft_face, ft_sfnt_head ))
|
|
|
|
{
|
|
|
|
TRACE("Font %s/%p lacks either an OS2, HHEA or HEAD table.\n"
|
|
|
|
"Skipping this font.\n", debugstr_a(file), font_data_ptr);
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Wine uses ttfs as an intermediate step in building its bitmap fonts;
|
|
|
|
we don't want to load these. */
|
|
|
|
if (!memcmp( pOS2->achVendID, "Wine", sizeof(pOS2->achVendID) ))
|
|
|
|
{
|
|
|
|
FT_ULong len = 0;
|
|
|
|
|
|
|
|
if (!pFT_Load_Sfnt_Table( ft_face, FT_MAKE_TAG('E','B','S','C'), 0, NULL, &len ))
|
|
|
|
{
|
|
|
|
TRACE("Skipping Wine bitmap-only TrueType font %s\n", debugstr_a(file));
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!ft_face->family_name || !ft_face->style_name)
|
|
|
|
{
|
|
|
|
TRACE("Font %s/%p lacks either a family or style name\n", debugstr_a(file), font_data_ptr);
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ft_face;
|
|
|
|
fail:
|
|
|
|
pFT_Done_Face( ft_face );
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2020-10-06 15:12:09 +02:00
|
|
|
static INT AddFontToList(const WCHAR *dos_name, const char *unix_name, void *font_data_ptr,
|
|
|
|
DWORD font_data_size, DWORD flags)
|
2001-09-12 22:21:06 +02:00
|
|
|
{
|
|
|
|
FT_Face ft_face;
|
2002-06-22 03:19:29 +02:00
|
|
|
FT_Long face_index = 0, num_faces;
|
2011-12-20 13:05:53 +01:00
|
|
|
INT ret = 0;
|
2020-10-06 15:40:25 +02:00
|
|
|
WCHAR *filename = NULL;
|
2012-11-05 22:32:51 +01:00
|
|
|
|
2007-09-13 19:42:55 +02:00
|
|
|
/* we always load external fonts from files - otherwise we would get a crash in update_reg_entries */
|
2020-10-06 15:12:09 +02:00
|
|
|
assert(unix_name || !(flags & ADDFONT_EXTERNAL_FONT));
|
2007-09-13 19:42:55 +02:00
|
|
|
|
2006-12-21 16:21:13 +01:00
|
|
|
#ifdef HAVE_CARBON_CARBON_H
|
2020-10-06 15:12:09 +02:00
|
|
|
if(unix_name)
|
2006-12-21 16:21:13 +01:00
|
|
|
{
|
2020-10-06 15:12:09 +02:00
|
|
|
char **mac_list = expand_mac_font(unix_name);
|
2006-12-21 16:21:13 +01:00
|
|
|
if(mac_list)
|
|
|
|
{
|
|
|
|
BOOL had_one = FALSE;
|
|
|
|
char **cursor;
|
|
|
|
for(cursor = mac_list; *cursor; cursor++)
|
|
|
|
{
|
|
|
|
had_one = TRUE;
|
2020-10-06 15:12:09 +02:00
|
|
|
AddFontToList(NULL, *cursor, NULL, 0, flags);
|
2006-12-21 16:21:13 +01:00
|
|
|
HeapFree(GetProcessHeap(), 0, *cursor);
|
|
|
|
}
|
|
|
|
HeapFree(GetProcessHeap(), 0, mac_list);
|
|
|
|
if(had_one)
|
2007-05-04 09:26:06 +02:00
|
|
|
return 1;
|
2006-12-21 16:21:13 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif /* HAVE_CARBON_CARBON_H */
|
|
|
|
|
2020-10-06 15:40:25 +02:00
|
|
|
if (!dos_name && unix_name) dos_name = filename = wine_get_dos_file_name( unix_name );
|
|
|
|
|
2002-06-22 03:19:29 +02:00
|
|
|
do {
|
2014-01-09 13:07:45 +01:00
|
|
|
FONTSIGNATURE fs;
|
|
|
|
|
2020-10-06 15:12:09 +02:00
|
|
|
ft_face = new_ft_face( unix_name, font_data_ptr, font_data_size, face_index, flags & ADDFONT_ALLOW_BITMAP );
|
2020-10-06 15:40:25 +02:00
|
|
|
if (!ft_face) break;
|
2003-06-23 22:51:06 +02:00
|
|
|
|
2008-03-21 11:32:45 +01:00
|
|
|
if(ft_face->family_name[0] == '.') /* Ignore fonts with names beginning with a dot */
|
|
|
|
{
|
2020-10-06 15:12:09 +02:00
|
|
|
TRACE("Ignoring %s since its family name begins with a dot\n", debugstr_a(unix_name));
|
2008-03-21 11:32:45 +01:00
|
|
|
pFT_Done_Face(ft_face);
|
2020-10-06 15:40:25 +02:00
|
|
|
break;
|
2008-03-21 11:32:45 +01:00
|
|
|
}
|
|
|
|
|
2020-10-06 15:40:25 +02:00
|
|
|
AddFaceToList(ft_face, dos_name, font_data_ptr, font_data_size, face_index, flags);
|
2011-12-20 13:05:53 +01:00
|
|
|
++ret;
|
|
|
|
|
2014-01-09 13:07:45 +01:00
|
|
|
get_fontsig(ft_face, &fs);
|
|
|
|
if (fs.fsCsb[0] & FS_DBCS_MASK)
|
2011-12-20 13:05:53 +01:00
|
|
|
{
|
2020-10-06 15:40:25 +02:00
|
|
|
AddFaceToList(ft_face, dos_name, font_data_ptr, font_data_size, face_index,
|
2013-01-15 14:13:09 +01:00
|
|
|
flags | ADDFONT_VERTICAL_FONT);
|
2011-12-20 13:05:53 +01:00
|
|
|
++ret;
|
|
|
|
}
|
2003-06-23 22:51:06 +02:00
|
|
|
|
2002-06-22 03:19:29 +02:00
|
|
|
num_faces = ft_face->num_faces;
|
|
|
|
pFT_Done_Face(ft_face);
|
|
|
|
} while(num_faces > ++face_index);
|
2020-10-06 15:40:25 +02:00
|
|
|
HeapFree( GetProcessHeap(), 0, filename );
|
2011-12-20 13:05:53 +01:00
|
|
|
return ret;
|
2001-09-12 22:21:06 +02:00
|
|
|
}
|
|
|
|
|
2020-10-26 12:04:36 +01:00
|
|
|
/*************************************************************
|
|
|
|
* freetype_add_font
|
|
|
|
*/
|
|
|
|
static INT CDECL freetype_add_font( const WCHAR *file, DWORD flags )
|
2020-04-17 12:39:01 +02:00
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
char *unixname = wine_get_unix_file_name( file );
|
|
|
|
|
|
|
|
if (unixname)
|
|
|
|
{
|
2020-10-06 15:12:09 +02:00
|
|
|
ret = AddFontToList( file, unixname, NULL, 0, flags );
|
2020-04-17 12:39:01 +02:00
|
|
|
HeapFree( GetProcessHeap(), 0, unixname );
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2020-10-26 12:05:19 +01:00
|
|
|
/*************************************************************
|
|
|
|
* freetype_add_mem_font
|
|
|
|
*/
|
|
|
|
static INT CDECL freetype_add_mem_font( void *ptr, SIZE_T size, DWORD flags )
|
|
|
|
{
|
|
|
|
return AddFontToList( NULL, NULL, ptr, size, flags );
|
|
|
|
}
|
|
|
|
|
2020-10-26 12:04:36 +01:00
|
|
|
/*************************************************************
|
|
|
|
* freetype_remove_font
|
|
|
|
*/
|
|
|
|
static INT CDECL freetype_remove_font( const WCHAR *file, DWORD flags )
|
2013-01-15 12:34:40 +01:00
|
|
|
{
|
|
|
|
Family *family, *family_next;
|
|
|
|
Face *face, *face_next;
|
|
|
|
int count = 0;
|
|
|
|
|
|
|
|
LIST_FOR_EACH_ENTRY_SAFE( family, family_next, &font_list, Family, entry )
|
|
|
|
{
|
|
|
|
family->refcount++;
|
|
|
|
LIST_FOR_EACH_ENTRY_SAFE( face, face_next, &family->faces, Face, entry )
|
|
|
|
{
|
|
|
|
if (!face->file) continue;
|
|
|
|
if (LOWORD(face->flags) != LOWORD(flags)) continue;
|
2020-10-06 15:40:25 +02:00
|
|
|
if (!strcmpiW( face->file, file ))
|
2013-01-15 12:34:40 +01:00
|
|
|
{
|
2013-03-04 11:35:28 +01:00
|
|
|
TRACE( "removing matching face %s refcount %d\n", debugstr_w(face->file), face->refcount );
|
2013-01-15 12:34:40 +01:00
|
|
|
release_face( face );
|
|
|
|
count++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
release_family( family );
|
|
|
|
}
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
2020-10-27 10:27:32 +01:00
|
|
|
#ifdef __ANDROID__
|
2004-03-09 04:43:54 +01:00
|
|
|
static BOOL ReadFontDir(const char *dirname, BOOL external_fonts)
|
2001-09-12 22:21:06 +02:00
|
|
|
{
|
|
|
|
DIR *dir;
|
|
|
|
struct dirent *dent;
|
|
|
|
char path[MAX_PATH];
|
|
|
|
|
2002-01-29 04:02:50 +01:00
|
|
|
TRACE("Loading fonts from %s\n", debugstr_a(dirname));
|
|
|
|
|
2001-09-12 22:21:06 +02:00
|
|
|
dir = opendir(dirname);
|
|
|
|
if(!dir) {
|
2006-04-04 11:54:14 +02:00
|
|
|
WARN("Can't open directory %s\n", debugstr_a(dirname));
|
2001-09-12 22:21:06 +02:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
while((dent = readdir(dir)) != NULL) {
|
2002-01-29 04:02:50 +01:00
|
|
|
struct stat statbuf;
|
|
|
|
|
2001-09-12 22:21:06 +02:00
|
|
|
if(!strcmp(dent->d_name, ".") || !strcmp(dent->d_name, ".."))
|
|
|
|
continue;
|
2002-01-29 04:02:50 +01:00
|
|
|
|
|
|
|
TRACE("Found %s in %s\n", debugstr_a(dent->d_name), debugstr_a(dirname));
|
|
|
|
|
2001-09-12 22:21:06 +02:00
|
|
|
sprintf(path, "%s/%s", dirname, dent->d_name);
|
2002-01-29 04:02:50 +01:00
|
|
|
|
|
|
|
if(stat(path, &statbuf) == -1)
|
|
|
|
{
|
|
|
|
WARN("Can't stat %s\n", debugstr_a(path));
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if(S_ISDIR(statbuf.st_mode))
|
2004-03-09 04:43:54 +01:00
|
|
|
ReadFontDir(path, external_fonts);
|
2002-01-29 04:02:50 +01:00
|
|
|
else
|
2011-10-06 23:26:06 +02:00
|
|
|
{
|
|
|
|
DWORD addfont_flags = ADDFONT_ADD_TO_CACHE;
|
|
|
|
if(external_fonts) addfont_flags |= ADDFONT_EXTERNAL_FONT;
|
2020-10-06 15:12:09 +02:00
|
|
|
AddFontToList(NULL, path, NULL, 0, addfont_flags);
|
2011-10-06 23:26:06 +02:00
|
|
|
}
|
2001-09-12 22:21:06 +02:00
|
|
|
}
|
2002-11-14 00:54:50 +01:00
|
|
|
closedir(dir);
|
2001-09-12 22:21:06 +02:00
|
|
|
return TRUE;
|
|
|
|
}
|
2020-10-27 10:27:32 +01:00
|
|
|
#endif
|
2020-04-17 12:39:15 +02:00
|
|
|
|
2012-04-05 00:41:10 +02:00
|
|
|
#ifdef SONAME_LIBFONTCONFIG
|
2012-11-05 22:32:51 +01:00
|
|
|
|
|
|
|
static BOOL fontconfig_enabled;
|
|
|
|
|
|
|
|
static UINT parse_aa_pattern( FcPattern *pattern )
|
|
|
|
{
|
|
|
|
FcBool antialias;
|
|
|
|
int rgba;
|
|
|
|
UINT aa_flags = 0;
|
|
|
|
|
|
|
|
if (pFcPatternGetBool( pattern, FC_ANTIALIAS, 0, &antialias ) == FcResultMatch)
|
|
|
|
aa_flags = antialias ? GGO_GRAY4_BITMAP : GGO_BITMAP;
|
|
|
|
|
|
|
|
if (pFcPatternGetInteger( pattern, FC_RGBA, 0, &rgba ) == FcResultMatch)
|
|
|
|
{
|
|
|
|
switch (rgba)
|
|
|
|
{
|
|
|
|
case FC_RGBA_RGB: aa_flags = WINE_GGO_HRGB_BITMAP; break;
|
|
|
|
case FC_RGBA_BGR: aa_flags = WINE_GGO_HBGR_BITMAP; break;
|
|
|
|
case FC_RGBA_VRGB: aa_flags = WINE_GGO_VRGB_BITMAP; break;
|
|
|
|
case FC_RGBA_VBGR: aa_flags = WINE_GGO_VBGR_BITMAP; break;
|
2018-10-29 13:50:21 +01:00
|
|
|
case FC_RGBA_NONE: aa_flags = aa_flags ? aa_flags : GGO_GRAY4_BITMAP; break;
|
2012-11-05 22:32:51 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return aa_flags;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void init_fontconfig(void)
|
|
|
|
{
|
2020-04-06 11:46:10 +02:00
|
|
|
void *fc_handle = dlopen(SONAME_LIBFONTCONFIG, RTLD_NOW);
|
2012-11-05 22:32:51 +01:00
|
|
|
|
|
|
|
if (!fc_handle)
|
|
|
|
{
|
|
|
|
TRACE("Wine cannot find the fontconfig library (%s).\n", SONAME_LIBFONTCONFIG);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-04-06 11:46:10 +02:00
|
|
|
#define LOAD_FUNCPTR(f) if((p##f = dlsym(fc_handle, #f)) == NULL){WARN("Can't find symbol %s\n", #f); return;}
|
2012-11-05 22:32:51 +01:00
|
|
|
LOAD_FUNCPTR(FcConfigSubstitute);
|
2018-11-27 16:50:20 +01:00
|
|
|
LOAD_FUNCPTR(FcDefaultSubstitute);
|
2012-11-05 22:32:51 +01:00
|
|
|
LOAD_FUNCPTR(FcFontList);
|
2018-11-27 16:50:20 +01:00
|
|
|
LOAD_FUNCPTR(FcFontMatch);
|
2012-11-05 22:32:51 +01:00
|
|
|
LOAD_FUNCPTR(FcFontSetDestroy);
|
|
|
|
LOAD_FUNCPTR(FcInit);
|
2018-11-27 16:50:20 +01:00
|
|
|
LOAD_FUNCPTR(FcPatternAddString);
|
2012-11-05 22:32:51 +01:00
|
|
|
LOAD_FUNCPTR(FcPatternCreate);
|
|
|
|
LOAD_FUNCPTR(FcPatternDestroy);
|
|
|
|
LOAD_FUNCPTR(FcPatternGetBool);
|
|
|
|
LOAD_FUNCPTR(FcPatternGetInteger);
|
|
|
|
LOAD_FUNCPTR(FcPatternGetString);
|
|
|
|
#undef LOAD_FUNCPTR
|
|
|
|
|
|
|
|
if (pFcInit())
|
|
|
|
{
|
|
|
|
FcPattern *pattern = pFcPatternCreate();
|
|
|
|
pFcConfigSubstitute( NULL, pattern, FcMatchFont );
|
|
|
|
default_aa_flags = parse_aa_pattern( pattern );
|
|
|
|
pFcPatternDestroy( pattern );
|
2018-10-23 18:07:11 +02:00
|
|
|
|
|
|
|
if (!default_aa_flags)
|
|
|
|
{
|
|
|
|
FcPattern *pattern = pFcPatternCreate();
|
|
|
|
pFcConfigSubstitute( NULL, pattern, FcMatchPattern );
|
|
|
|
default_aa_flags = parse_aa_pattern( pattern );
|
|
|
|
pFcPatternDestroy( pattern );
|
|
|
|
}
|
|
|
|
|
2012-11-05 22:32:51 +01:00
|
|
|
TRACE( "enabled, default flags = %x\n", default_aa_flags );
|
|
|
|
fontconfig_enabled = TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-11-09 01:30:13 +01:00
|
|
|
static void load_fontconfig_fonts(void)
|
|
|
|
{
|
|
|
|
FcPattern *pat;
|
|
|
|
FcFontSet *fontset;
|
|
|
|
int i, len;
|
2006-10-30 22:05:43 +01:00
|
|
|
char *file;
|
|
|
|
const char *ext;
|
2003-11-09 01:30:13 +01:00
|
|
|
|
2012-11-05 22:32:51 +01:00
|
|
|
if (!fontconfig_enabled) return;
|
2012-11-01 12:35:52 +01:00
|
|
|
|
2003-11-09 01:30:13 +01:00
|
|
|
pat = pFcPatternCreate();
|
2018-10-29 13:50:20 +01:00
|
|
|
if (!pat) return;
|
|
|
|
|
|
|
|
fontset = pFcFontList(NULL, pat, NULL);
|
|
|
|
if (!fontset)
|
|
|
|
{
|
|
|
|
pFcPatternDestroy(pat);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2003-11-09 01:30:13 +01:00
|
|
|
for(i = 0; i < fontset->nfont; i++) {
|
2006-12-19 22:21:27 +01:00
|
|
|
FcBool scalable;
|
2012-11-05 22:32:51 +01:00
|
|
|
DWORD aa_flags;
|
2006-12-19 22:21:27 +01:00
|
|
|
|
2006-04-28 13:51:36 +02:00
|
|
|
if(pFcPatternGetString(fontset->fonts[i], FC_FILE, 0, (FcChar8**)&file) != FcResultMatch)
|
2003-11-09 01:30:13 +01:00
|
|
|
continue;
|
|
|
|
|
2012-11-01 12:35:52 +01:00
|
|
|
pFcConfigSubstitute( NULL, fontset->fonts[i], FcMatchFont );
|
|
|
|
|
2003-11-09 01:30:13 +01:00
|
|
|
/* We're just interested in OT/TT fonts for now, so this hack just
|
2006-12-19 22:21:27 +01:00
|
|
|
picks up the scalable fonts without extensions .pf[ab] to save time
|
|
|
|
loading every other font */
|
|
|
|
|
|
|
|
if(pFcPatternGetBool(fontset->fonts[i], FC_SCALABLE, 0, &scalable) == FcResultMatch && !scalable)
|
|
|
|
{
|
|
|
|
TRACE("not scalable\n");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2012-11-05 22:32:51 +01:00
|
|
|
aa_flags = parse_aa_pattern( fontset->fonts[i] );
|
|
|
|
TRACE("fontconfig: %s aa %x\n", file, aa_flags);
|
2012-11-01 12:35:52 +01:00
|
|
|
|
2005-08-08 17:03:42 +02:00
|
|
|
len = strlen( file );
|
2003-11-09 01:30:13 +01:00
|
|
|
if(len < 4) continue;
|
2005-08-08 17:03:42 +02:00
|
|
|
ext = &file[ len - 3 ];
|
2019-04-08 14:30:23 +02:00
|
|
|
if(_strnicmp(ext, "pfa", -1) && _strnicmp(ext, "pfb", -1))
|
2020-10-06 15:12:09 +02:00
|
|
|
AddFontToList(NULL, file, NULL, 0,
|
2012-11-01 12:35:52 +01:00
|
|
|
ADDFONT_EXTERNAL_FONT | ADDFONT_ADD_TO_CACHE | ADDFONT_AA_FLAGS(aa_flags) );
|
2003-11-09 01:30:13 +01:00
|
|
|
}
|
|
|
|
pFcFontSetDestroy(fontset);
|
|
|
|
pFcPatternDestroy(pat);
|
|
|
|
}
|
2004-03-09 04:43:54 +01:00
|
|
|
|
2012-04-05 00:41:10 +02:00
|
|
|
#elif defined(HAVE_CARBON_CARBON_H)
|
|
|
|
|
|
|
|
static void load_mac_font_callback(const void *value, void *context)
|
|
|
|
{
|
|
|
|
CFStringRef pathStr = value;
|
|
|
|
CFIndex len;
|
|
|
|
char* path;
|
|
|
|
|
|
|
|
len = CFStringGetMaximumSizeOfFileSystemRepresentation(pathStr);
|
|
|
|
path = HeapAlloc(GetProcessHeap(), 0, len);
|
|
|
|
if (path && CFStringGetFileSystemRepresentation(pathStr, path, len))
|
|
|
|
{
|
|
|
|
TRACE("font file %s\n", path);
|
2020-10-06 15:12:09 +02:00
|
|
|
AddFontToList(NULL, path, NULL, 0, ADDFONT_EXTERNAL_FONT | ADDFONT_ADD_TO_CACHE);
|
2012-04-05 00:41:10 +02:00
|
|
|
}
|
|
|
|
HeapFree(GetProcessHeap(), 0, path);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void load_mac_fonts(void)
|
|
|
|
{
|
|
|
|
CFStringRef removeDupesKey;
|
|
|
|
CFBooleanRef removeDupesValue;
|
|
|
|
CFDictionaryRef options;
|
|
|
|
CTFontCollectionRef col;
|
|
|
|
CFArrayRef descs;
|
|
|
|
CFMutableSetRef paths;
|
|
|
|
CFIndex i;
|
|
|
|
|
|
|
|
removeDupesKey = kCTFontCollectionRemoveDuplicatesOption;
|
|
|
|
removeDupesValue = kCFBooleanTrue;
|
|
|
|
options = CFDictionaryCreate(NULL, (const void**)&removeDupesKey, (const void**)&removeDupesValue, 1,
|
|
|
|
&kCFTypeDictionaryKeyCallBacks, &kCFTypeDictionaryValueCallBacks);
|
|
|
|
col = CTFontCollectionCreateFromAvailableFonts(options);
|
|
|
|
if (options) CFRelease(options);
|
|
|
|
if (!col)
|
|
|
|
{
|
|
|
|
WARN("CTFontCollectionCreateFromAvailableFonts failed\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
descs = CTFontCollectionCreateMatchingFontDescriptors(col);
|
|
|
|
CFRelease(col);
|
|
|
|
if (!descs)
|
|
|
|
{
|
|
|
|
WARN("CTFontCollectionCreateMatchingFontDescriptors failed\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
paths = CFSetCreateMutable(NULL, 0, &kCFTypeSetCallBacks);
|
|
|
|
if (!paths)
|
|
|
|
{
|
|
|
|
WARN("CFSetCreateMutable failed\n");
|
|
|
|
CFRelease(descs);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < CFArrayGetCount(descs); i++)
|
|
|
|
{
|
|
|
|
CTFontDescriptorRef desc;
|
|
|
|
CFURLRef url;
|
|
|
|
CFStringRef ext;
|
|
|
|
CFStringRef path;
|
|
|
|
|
|
|
|
desc = CFArrayGetValueAtIndex(descs, i);
|
|
|
|
|
2016-11-29 18:22:32 +01:00
|
|
|
#if defined(MAC_OS_X_VERSION_10_6) && MAC_OS_X_VERSION_MAX_ALLOWED >= MAC_OS_X_VERSION_10_6
|
|
|
|
url = CTFontDescriptorCopyAttribute(desc, kCTFontURLAttribute);
|
|
|
|
#else
|
|
|
|
/* CTFontDescriptor doesn't support kCTFontURLAttribute prior to 10.6, so
|
2012-04-05 00:41:10 +02:00
|
|
|
we have to go CFFontDescriptor -> CTFont -> ATSFont -> FSRef -> CFURL. */
|
|
|
|
{
|
2016-11-29 18:22:32 +01:00
|
|
|
CTFontRef font;
|
|
|
|
ATSFontRef atsFont;
|
|
|
|
OSStatus status;
|
|
|
|
FSRef fsref;
|
2012-04-05 00:41:10 +02:00
|
|
|
|
2016-11-29 18:22:32 +01:00
|
|
|
font = CTFontCreateWithFontDescriptor(desc, 0, NULL);
|
|
|
|
if (!font) continue;
|
2012-04-05 00:41:10 +02:00
|
|
|
|
2016-11-29 18:22:32 +01:00
|
|
|
atsFont = CTFontGetPlatformFont(font, NULL);
|
|
|
|
if (!atsFont)
|
|
|
|
{
|
|
|
|
CFRelease(font);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
status = ATSFontGetFileReference(atsFont, &fsref);
|
|
|
|
CFRelease(font);
|
|
|
|
if (status != noErr) continue;
|
|
|
|
|
|
|
|
url = CFURLCreateFromFSRef(NULL, &fsref);
|
|
|
|
}
|
|
|
|
#endif
|
2012-04-05 00:41:10 +02:00
|
|
|
if (!url) continue;
|
|
|
|
|
|
|
|
ext = CFURLCopyPathExtension(url);
|
|
|
|
if (ext)
|
|
|
|
{
|
|
|
|
BOOL skip = (CFStringCompare(ext, CFSTR("pfa"), kCFCompareCaseInsensitive) == kCFCompareEqualTo ||
|
|
|
|
CFStringCompare(ext, CFSTR("pfb"), kCFCompareCaseInsensitive) == kCFCompareEqualTo);
|
|
|
|
CFRelease(ext);
|
|
|
|
if (skip)
|
|
|
|
{
|
|
|
|
CFRelease(url);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
path = CFURLCopyFileSystemPath(url, kCFURLPOSIXPathStyle);
|
|
|
|
CFRelease(url);
|
|
|
|
if (!path) continue;
|
|
|
|
|
|
|
|
CFSetAddValue(paths, path);
|
|
|
|
CFRelease(path);
|
|
|
|
}
|
|
|
|
|
|
|
|
CFRelease(descs);
|
|
|
|
|
|
|
|
CFSetApplyFunction(paths, load_mac_font_callback, NULL);
|
|
|
|
CFRelease(paths);
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
2019-10-16 13:49:24 +02:00
|
|
|
static WCHAR *get_full_path_name(const WCHAR *name)
|
|
|
|
{
|
|
|
|
WCHAR *full_path;
|
|
|
|
DWORD len;
|
|
|
|
|
|
|
|
if (!(len = GetFullPathNameW(name, 0, NULL, NULL)))
|
|
|
|
{
|
|
|
|
ERR("GetFullPathNameW() failed, name %s.\n", debugstr_w(name));
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(full_path = HeapAlloc(GetProcessHeap(), 0, len * sizeof(*full_path))))
|
|
|
|
{
|
|
|
|
ERR("Could not get memory.\n");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (GetFullPathNameW(name, len, full_path, NULL) != len - 1)
|
|
|
|
{
|
|
|
|
ERR("Unexpected GetFullPathNameW() result, name %s.\n", debugstr_w(name));
|
|
|
|
HeapFree(GetProcessHeap(), 0, full_path);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return full_path;
|
|
|
|
}
|
|
|
|
|
2004-03-09 04:43:54 +01:00
|
|
|
/*************************************************************
|
|
|
|
*
|
|
|
|
* This adds registry entries for any externally loaded fonts
|
|
|
|
* (fonts from fontconfig or FontDirs). It also deletes entries
|
|
|
|
* of no longer existing fonts.
|
|
|
|
*
|
|
|
|
*/
|
2005-04-11 15:00:28 +02:00
|
|
|
static void update_reg_entries(void)
|
2004-03-09 04:43:54 +01:00
|
|
|
{
|
2008-03-07 18:02:32 +01:00
|
|
|
HKEY winnt_key = 0, win9x_key = 0, external_key = 0;
|
2004-03-09 04:43:54 +01:00
|
|
|
LPWSTR valueW;
|
2012-09-07 12:40:12 +02:00
|
|
|
DWORD len;
|
2004-03-09 04:43:54 +01:00
|
|
|
Family *family;
|
|
|
|
Face *face;
|
2020-10-06 15:12:09 +02:00
|
|
|
WCHAR *file, *path;
|
2004-04-19 22:12:14 +02:00
|
|
|
static const WCHAR TrueType[] = {' ','(','T','r','u','e','T','y','p','e',')','\0'};
|
2004-03-09 04:43:54 +01:00
|
|
|
|
2008-03-07 18:02:32 +01:00
|
|
|
if(RegCreateKeyExW(HKEY_LOCAL_MACHINE, winnt_font_reg_key,
|
|
|
|
0, NULL, 0, KEY_ALL_ACCESS, NULL, &winnt_key, NULL) != ERROR_SUCCESS) {
|
2004-03-09 04:43:54 +01:00
|
|
|
ERR("Can't create Windows font reg key\n");
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
|
2008-03-07 18:02:32 +01:00
|
|
|
if(RegCreateKeyExW(HKEY_LOCAL_MACHINE, win9x_font_reg_key,
|
|
|
|
0, NULL, 0, KEY_ALL_ACCESS, NULL, &win9x_key, NULL) != ERROR_SUCCESS) {
|
|
|
|
ERR("Can't create Windows font reg key\n");
|
|
|
|
goto end;
|
2004-03-09 04:43:54 +01:00
|
|
|
}
|
|
|
|
|
2005-06-28 21:15:17 +02:00
|
|
|
if(RegCreateKeyExW(HKEY_CURRENT_USER, external_fonts_reg_key,
|
2008-03-07 18:02:32 +01:00
|
|
|
0, NULL, 0, KEY_ALL_ACCESS, NULL, &external_key, NULL) != ERROR_SUCCESS) {
|
2004-03-09 04:43:54 +01:00
|
|
|
ERR("Can't create external font reg key\n");
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* enumerate the fonts and add external ones to the two keys */
|
|
|
|
|
2012-12-07 14:29:28 +01:00
|
|
|
LIST_FOR_EACH_ENTRY( family, &font_list, Family, entry ) {
|
|
|
|
LIST_FOR_EACH_ENTRY( face, &family->faces, Face, entry ) {
|
2013-01-14 15:28:42 +01:00
|
|
|
if (!(face->flags & ADDFONT_EXTERNAL_FONT)) continue;
|
2012-09-07 12:40:12 +02:00
|
|
|
|
2020-09-07 15:18:56 +02:00
|
|
|
len = strlenW( face->full_name ) + 1;
|
2016-08-14 22:52:16 +02:00
|
|
|
if (face->scalable)
|
2018-10-05 21:35:35 +02:00
|
|
|
len += ARRAY_SIZE(TrueType);
|
2016-08-14 22:52:16 +02:00
|
|
|
|
|
|
|
valueW = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR));
|
2020-09-07 15:18:56 +02:00
|
|
|
strcpyW( valueW, face->full_name );
|
2016-08-14 22:52:16 +02:00
|
|
|
|
|
|
|
if (face->scalable)
|
|
|
|
strcatW(valueW, TrueType);
|
2008-03-10 13:31:43 +01:00
|
|
|
|
2020-10-06 15:12:09 +02:00
|
|
|
if ((path = get_full_path_name(face->file)))
|
2019-10-16 13:49:24 +02:00
|
|
|
{
|
2012-11-01 13:16:26 +01:00
|
|
|
file = path;
|
2019-10-16 13:49:24 +02:00
|
|
|
}
|
2020-10-06 15:12:09 +02:00
|
|
|
else if ((file = strrchrW(face->file, '\\')))
|
2019-10-16 13:49:24 +02:00
|
|
|
{
|
2012-11-01 13:16:26 +01:00
|
|
|
file++;
|
2019-10-16 13:49:24 +02:00
|
|
|
}
|
2004-03-09 04:43:54 +01:00
|
|
|
else
|
2019-10-16 13:49:24 +02:00
|
|
|
{
|
2012-11-01 13:16:26 +01:00
|
|
|
file = face->file;
|
2019-10-16 13:49:24 +02:00
|
|
|
}
|
2004-03-09 04:43:54 +01:00
|
|
|
|
2012-11-01 13:16:26 +01:00
|
|
|
len = strlenW(file) + 1;
|
2008-03-07 18:02:32 +01:00
|
|
|
RegSetValueExW(winnt_key, valueW, 0, REG_SZ, (BYTE*)file, len * sizeof(WCHAR));
|
|
|
|
RegSetValueExW(win9x_key, valueW, 0, REG_SZ, (BYTE*)file, len * sizeof(WCHAR));
|
|
|
|
RegSetValueExW(external_key, valueW, 0, REG_SZ, (BYTE*)file, len * sizeof(WCHAR));
|
2004-03-09 04:43:54 +01:00
|
|
|
|
2012-11-01 13:16:26 +01:00
|
|
|
HeapFree(GetProcessHeap(), 0, path);
|
2004-03-09 04:43:54 +01:00
|
|
|
HeapFree(GetProcessHeap(), 0, valueW);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
end:
|
2008-03-07 18:02:32 +01:00
|
|
|
if(external_key) RegCloseKey(external_key);
|
|
|
|
if(win9x_key) RegCloseKey(win9x_key);
|
|
|
|
if(winnt_key) RegCloseKey(winnt_key);
|
2004-03-09 04:43:54 +01:00
|
|
|
}
|
|
|
|
|
2008-03-07 18:02:32 +01:00
|
|
|
static void delete_external_font_keys(void)
|
|
|
|
{
|
|
|
|
HKEY winnt_key = 0, win9x_key = 0, external_key = 0;
|
2017-04-25 12:39:29 +02:00
|
|
|
DWORD dlen, plen, vlen, datalen, valuelen, i, type, path_type;
|
2008-03-07 18:02:32 +01:00
|
|
|
LPWSTR valueW;
|
|
|
|
LPVOID data;
|
2017-04-25 12:39:29 +02:00
|
|
|
BYTE *path;
|
2008-03-07 18:02:32 +01:00
|
|
|
|
|
|
|
if(RegCreateKeyExW(HKEY_LOCAL_MACHINE, winnt_font_reg_key,
|
|
|
|
0, NULL, 0, KEY_ALL_ACCESS, NULL, &winnt_key, NULL) != ERROR_SUCCESS) {
|
|
|
|
ERR("Can't create Windows font reg key\n");
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(RegCreateKeyExW(HKEY_LOCAL_MACHINE, win9x_font_reg_key,
|
|
|
|
0, NULL, 0, KEY_ALL_ACCESS, NULL, &win9x_key, NULL) != ERROR_SUCCESS) {
|
|
|
|
ERR("Can't create Windows font reg key\n");
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(RegCreateKeyW(HKEY_CURRENT_USER, external_fonts_reg_key, &external_key) != ERROR_SUCCESS) {
|
|
|
|
ERR("Can't create external font reg key\n");
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Delete all external fonts added last time */
|
|
|
|
|
|
|
|
RegQueryInfoKeyW(external_key, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
|
|
|
|
&valuelen, &datalen, NULL, NULL);
|
|
|
|
valuelen++; /* returned value doesn't include room for '\0' */
|
|
|
|
valueW = HeapAlloc(GetProcessHeap(), 0, valuelen * sizeof(WCHAR));
|
2017-04-25 12:39:23 +02:00
|
|
|
data = HeapAlloc(GetProcessHeap(), 0, datalen);
|
2017-04-25 12:39:29 +02:00
|
|
|
path = HeapAlloc(GetProcessHeap(), 0, datalen);
|
2008-03-07 18:02:32 +01:00
|
|
|
|
2017-04-25 12:39:23 +02:00
|
|
|
dlen = datalen;
|
2008-03-07 18:02:32 +01:00
|
|
|
vlen = valuelen;
|
|
|
|
i = 0;
|
|
|
|
while(RegEnumValueW(external_key, i++, valueW, &vlen, NULL, &type, data,
|
|
|
|
&dlen) == ERROR_SUCCESS) {
|
2017-04-25 12:39:29 +02:00
|
|
|
plen = dlen;
|
|
|
|
if (RegQueryValueExW(winnt_key, valueW, 0, &path_type, path, &plen) == ERROR_SUCCESS &&
|
|
|
|
type == path_type && dlen == plen && !memcmp(data, path, plen))
|
|
|
|
RegDeleteValueW(winnt_key, valueW);
|
|
|
|
|
|
|
|
plen = dlen;
|
|
|
|
if (RegQueryValueExW(win9x_key, valueW, 0, &path_type, path, &plen) == ERROR_SUCCESS &&
|
|
|
|
type == path_type && dlen == plen && !memcmp(data, path, plen))
|
|
|
|
RegDeleteValueW(win9x_key, valueW);
|
2008-03-07 18:02:32 +01:00
|
|
|
|
|
|
|
/* reset dlen and vlen */
|
|
|
|
dlen = datalen;
|
|
|
|
vlen = valuelen;
|
|
|
|
}
|
2017-04-25 12:39:29 +02:00
|
|
|
HeapFree(GetProcessHeap(), 0, path);
|
2008-03-07 18:02:32 +01:00
|
|
|
HeapFree(GetProcessHeap(), 0, data);
|
|
|
|
HeapFree(GetProcessHeap(), 0, valueW);
|
|
|
|
|
|
|
|
/* Delete the old external fonts key */
|
|
|
|
RegCloseKey(external_key);
|
|
|
|
RegDeleteKeyW(HKEY_CURRENT_USER, external_fonts_reg_key);
|
|
|
|
|
|
|
|
end:
|
|
|
|
if(win9x_key) RegCloseKey(win9x_key);
|
|
|
|
if(winnt_key) RegCloseKey(winnt_key);
|
|
|
|
}
|
2004-03-09 04:43:54 +01:00
|
|
|
|
2008-03-12 14:45:13 +01:00
|
|
|
static BOOL init_freetype(void)
|
|
|
|
{
|
2020-04-06 11:46:10 +02:00
|
|
|
ft_handle = dlopen(SONAME_LIBFREETYPE, RTLD_NOW);
|
2002-04-03 22:02:39 +02:00
|
|
|
if(!ft_handle) {
|
|
|
|
WINE_MESSAGE(
|
|
|
|
"Wine cannot find the FreeType font library. To enable Wine to\n"
|
|
|
|
"use TrueType fonts please install a version of FreeType greater than\n"
|
|
|
|
"or equal to 2.0.5.\n"
|
|
|
|
"http://www.freetype.org\n");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2020-04-06 11:46:10 +02:00
|
|
|
#define LOAD_FUNCPTR(f) if((p##f = dlsym(ft_handle, #f)) == NULL){WARN("Can't find symbol %s\n", #f); goto sym_not_found;}
|
2002-04-03 22:02:39 +02:00
|
|
|
|
|
|
|
LOAD_FUNCPTR(FT_Done_Face)
|
|
|
|
LOAD_FUNCPTR(FT_Get_Char_Index)
|
2011-10-11 11:14:56 +02:00
|
|
|
LOAD_FUNCPTR(FT_Get_First_Char)
|
|
|
|
LOAD_FUNCPTR(FT_Get_Next_Char)
|
2006-04-05 13:45:25 +02:00
|
|
|
LOAD_FUNCPTR(FT_Get_Sfnt_Name)
|
|
|
|
LOAD_FUNCPTR(FT_Get_Sfnt_Name_Count)
|
2002-04-03 22:02:39 +02:00
|
|
|
LOAD_FUNCPTR(FT_Get_Sfnt_Table)
|
2011-10-11 11:14:56 +02:00
|
|
|
LOAD_FUNCPTR(FT_Get_WinFNT_Header)
|
2002-04-03 22:02:39 +02:00
|
|
|
LOAD_FUNCPTR(FT_Init_FreeType)
|
2011-10-11 11:14:56 +02:00
|
|
|
LOAD_FUNCPTR(FT_Library_Version)
|
2002-04-03 22:02:39 +02:00
|
|
|
LOAD_FUNCPTR(FT_Load_Glyph)
|
2011-10-10 15:15:56 +02:00
|
|
|
LOAD_FUNCPTR(FT_Load_Sfnt_Table)
|
2003-10-10 02:06:35 +02:00
|
|
|
LOAD_FUNCPTR(FT_Matrix_Multiply)
|
2008-09-03 13:14:17 +02:00
|
|
|
#ifndef FT_MULFIX_INLINED
|
2002-04-03 22:02:39 +02:00
|
|
|
LOAD_FUNCPTR(FT_MulFix)
|
2008-09-03 13:14:17 +02:00
|
|
|
#endif
|
2002-04-03 22:02:39 +02:00
|
|
|
LOAD_FUNCPTR(FT_New_Face)
|
2006-12-06 21:31:53 +01:00
|
|
|
LOAD_FUNCPTR(FT_New_Memory_Face)
|
2002-04-03 22:02:39 +02:00
|
|
|
LOAD_FUNCPTR(FT_Outline_Get_Bitmap)
|
2013-10-17 14:25:10 +02:00
|
|
|
LOAD_FUNCPTR(FT_Outline_Get_CBox)
|
2002-04-03 22:02:39 +02:00
|
|
|
LOAD_FUNCPTR(FT_Outline_Transform)
|
|
|
|
LOAD_FUNCPTR(FT_Outline_Translate)
|
2011-10-10 15:15:56 +02:00
|
|
|
LOAD_FUNCPTR(FT_Render_Glyph)
|
2007-10-02 05:04:25 +02:00
|
|
|
LOAD_FUNCPTR(FT_Set_Charmap)
|
2002-04-03 22:02:39 +02:00
|
|
|
LOAD_FUNCPTR(FT_Set_Pixel_Sizes)
|
2016-01-13 15:09:08 +01:00
|
|
|
LOAD_FUNCPTR(FT_Vector_Length)
|
2003-10-10 02:06:35 +02:00
|
|
|
LOAD_FUNCPTR(FT_Vector_Transform)
|
2011-10-10 15:15:56 +02:00
|
|
|
LOAD_FUNCPTR(FT_Vector_Unit)
|
2002-04-03 22:02:39 +02:00
|
|
|
#undef LOAD_FUNCPTR
|
2008-03-12 14:45:13 +01:00
|
|
|
/* Don't warn if these ones are missing */
|
2020-04-06 11:46:10 +02:00
|
|
|
pFT_Outline_Embolden = dlsym(ft_handle, "FT_Outline_Embolden");
|
|
|
|
pFT_Get_TrueType_Engine_Type = dlsym(ft_handle, "FT_Get_TrueType_Engine_Type");
|
2013-11-30 11:22:25 +01:00
|
|
|
#ifdef FT_LCD_FILTER_H
|
2020-04-06 11:46:10 +02:00
|
|
|
pFT_Library_SetLcdFilter = dlsym(ft_handle, "FT_Library_SetLcdFilter");
|
2008-12-23 12:34:01 +01:00
|
|
|
#endif
|
2020-04-06 11:46:10 +02:00
|
|
|
pFT_Property_Set = dlsym(ft_handle, "FT_Property_Set");
|
2002-06-01 01:06:46 +02:00
|
|
|
|
2002-04-03 22:02:39 +02:00
|
|
|
if(pFT_Init_FreeType(&library) != 0) {
|
2001-09-12 22:21:06 +02:00
|
|
|
ERR("Can't init FreeType library\n");
|
2020-04-06 11:46:10 +02:00
|
|
|
dlclose(ft_handle);
|
2002-06-22 03:19:29 +02:00
|
|
|
ft_handle = NULL;
|
2001-09-12 22:21:06 +02:00
|
|
|
return FALSE;
|
|
|
|
}
|
2011-10-11 11:14:56 +02:00
|
|
|
pFT_Library_Version(library,&FT_Version.major,&FT_Version.minor,&FT_Version.patch);
|
2008-03-12 14:45:13 +01:00
|
|
|
|
2002-08-17 20:34:34 +02:00
|
|
|
TRACE("FreeType version is %d.%d.%d\n",FT_Version.major,FT_Version.minor,FT_Version.patch);
|
2004-06-16 22:06:26 +02:00
|
|
|
FT_SimpleVersion = ((FT_Version.major << 16) & 0xff0000) |
|
|
|
|
((FT_Version.minor << 8) & 0x00ff00) |
|
|
|
|
((FT_Version.patch ) & 0x0000ff);
|
2001-09-12 22:21:06 +02:00
|
|
|
|
2019-03-12 13:42:19 +01:00
|
|
|
/* In FreeType < 2.8.1 v40's FT_LOAD_TARGET_MONO has broken advance widths. */
|
2019-02-19 11:25:41 +01:00
|
|
|
if (pFT_Property_Set && FT_SimpleVersion < FT_VERSION_VALUE(2, 8, 1))
|
|
|
|
{
|
|
|
|
FT_UInt interpreter_version = 35;
|
|
|
|
pFT_Property_Set( library, "truetype", "interpreter-version", &interpreter_version );
|
|
|
|
}
|
|
|
|
|
2008-03-12 14:45:13 +01:00
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
sym_not_found:
|
|
|
|
WINE_MESSAGE(
|
|
|
|
"Wine cannot find certain functions that it needs inside the FreeType\n"
|
|
|
|
"font library. To enable Wine to use TrueType fonts please upgrade\n"
|
2011-10-10 15:15:56 +02:00
|
|
|
"FreeType to at least version 2.1.4.\n"
|
2008-03-12 14:45:13 +01:00
|
|
|
"http://www.freetype.org\n");
|
2020-04-06 11:46:10 +02:00
|
|
|
dlclose(ft_handle);
|
2008-03-12 14:45:13 +01:00
|
|
|
ft_handle = NULL;
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2011-10-06 23:26:05 +02:00
|
|
|
static void init_font_list(void)
|
2008-03-12 14:45:13 +01:00
|
|
|
{
|
2008-03-07 18:02:32 +01:00
|
|
|
delete_external_font_keys();
|
2020-10-27 10:27:06 +01:00
|
|
|
load_system_bitmap_fonts();
|
2020-10-27 10:27:32 +01:00
|
|
|
load_file_system_fonts();
|
2020-10-27 10:27:20 +01:00
|
|
|
load_registry_fonts();
|
2002-06-22 03:19:29 +02:00
|
|
|
|
2012-04-05 00:41:10 +02:00
|
|
|
#ifdef SONAME_LIBFONTCONFIG
|
2003-11-09 01:30:13 +01:00
|
|
|
load_fontconfig_fonts();
|
2012-04-05 00:41:10 +02:00
|
|
|
#elif defined(HAVE_CARBON_CARBON_H)
|
|
|
|
load_mac_fonts();
|
2016-04-07 09:59:48 +02:00
|
|
|
#elif defined(__ANDROID__)
|
|
|
|
ReadFontDir("/system/fonts", TRUE);
|
2012-04-05 00:41:10 +02:00
|
|
|
#endif
|
2011-10-06 23:26:05 +02:00
|
|
|
}
|
|
|
|
|
2011-10-06 23:26:08 +02:00
|
|
|
static BOOL move_to_front(const WCHAR *name)
|
|
|
|
{
|
|
|
|
Family *family, *cursor2;
|
|
|
|
LIST_FOR_EACH_ENTRY_SAFE(family, cursor2, &font_list, Family, entry)
|
|
|
|
{
|
2020-09-07 15:18:58 +02:00
|
|
|
if (!strncmpiW( family->family_name, name, LF_FACESIZE - 1 ))
|
2011-10-06 23:26:08 +02:00
|
|
|
{
|
|
|
|
list_remove(&family->entry);
|
|
|
|
list_add_head(&font_list, &family->entry);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2018-12-20 11:32:14 +01:00
|
|
|
static const WCHAR *set_default(const WCHAR **name_list)
|
2011-10-06 23:26:08 +02:00
|
|
|
{
|
2018-12-20 11:32:14 +01:00
|
|
|
const WCHAR **entry = name_list;
|
|
|
|
|
|
|
|
while (*entry)
|
2011-10-12 12:55:12 +02:00
|
|
|
{
|
2018-12-20 11:32:14 +01:00
|
|
|
if (move_to_front(*entry)) return *entry;
|
|
|
|
entry++;
|
2011-10-12 12:55:12 +02:00
|
|
|
}
|
2011-10-06 23:26:08 +02:00
|
|
|
|
2018-12-20 11:32:14 +01:00
|
|
|
return *name_list;
|
2011-10-12 12:55:12 +02:00
|
|
|
}
|
2011-10-06 23:26:08 +02:00
|
|
|
|
2011-10-12 12:55:12 +02:00
|
|
|
static void reorder_font_list(void)
|
|
|
|
{
|
2018-12-20 11:32:14 +01:00
|
|
|
default_serif = set_default( default_serif_list );
|
|
|
|
default_fixed = set_default( default_fixed_list );
|
|
|
|
default_sans = set_default( default_sans_list );
|
2011-10-06 23:26:08 +02:00
|
|
|
}
|
|
|
|
|
2011-10-06 23:26:05 +02:00
|
|
|
/*************************************************************
|
|
|
|
* WineEngInit
|
|
|
|
*
|
|
|
|
* Initialize FreeType library and create a list of available faces
|
|
|
|
*/
|
2020-10-20 22:05:37 +02:00
|
|
|
BOOL WineEngInit( const struct font_backend_funcs **funcs )
|
2011-10-06 23:26:05 +02:00
|
|
|
{
|
2015-09-29 09:27:15 +02:00
|
|
|
HKEY hkey;
|
2011-10-06 23:26:06 +02:00
|
|
|
DWORD disposition;
|
2011-10-06 23:26:05 +02:00
|
|
|
HANDLE font_mutex;
|
|
|
|
|
|
|
|
if(!init_freetype()) return FALSE;
|
|
|
|
|
2013-01-15 14:28:10 +01:00
|
|
|
#ifdef SONAME_LIBFONTCONFIG
|
|
|
|
init_fontconfig();
|
|
|
|
#endif
|
|
|
|
|
2020-10-20 22:05:37 +02:00
|
|
|
*funcs = &font_funcs;
|
|
|
|
|
2015-09-29 09:27:15 +02:00
|
|
|
if (!RegOpenKeyExW(HKEY_CURRENT_USER, wine_fonts_key, 0, KEY_READ, &hkey))
|
|
|
|
{
|
|
|
|
static const WCHAR antialias_fake_bold_or_italic[] = { 'A','n','t','i','a','l','i','a','s','F','a','k','e',
|
|
|
|
'B','o','l','d','O','r','I','t','a','l','i','c',0 };
|
|
|
|
static const WCHAR true_options[] = { 'y','Y','t','T','1',0 };
|
|
|
|
DWORD type, size;
|
|
|
|
WCHAR buffer[20];
|
|
|
|
|
|
|
|
size = sizeof(buffer);
|
|
|
|
if (!RegQueryValueExW(hkey, antialias_fake_bold_or_italic, NULL, &type, (BYTE*)buffer, &size) &&
|
|
|
|
type == REG_SZ && size >= 1)
|
|
|
|
{
|
|
|
|
antialias_fakes = (strchrW(true_options, buffer[0]) != NULL);
|
|
|
|
}
|
|
|
|
RegCloseKey(hkey);
|
|
|
|
}
|
|
|
|
|
2011-10-06 23:26:05 +02:00
|
|
|
if((font_mutex = CreateMutexW(NULL, FALSE, font_mutex_nameW)) == NULL)
|
|
|
|
{
|
|
|
|
ERR("Failed to create font mutex\n");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
WaitForSingleObject(font_mutex, INFINITE);
|
|
|
|
|
2011-10-06 23:26:06 +02:00
|
|
|
create_font_cache_key(&hkey_font_cache, &disposition);
|
|
|
|
|
2011-10-06 23:26:07 +02:00
|
|
|
if(disposition == REG_CREATED_NEW_KEY)
|
|
|
|
init_font_list();
|
|
|
|
else
|
|
|
|
load_font_list_from_cache(hkey_font_cache);
|
2002-01-29 04:02:50 +01:00
|
|
|
|
2011-10-06 23:26:08 +02:00
|
|
|
reorder_font_list();
|
|
|
|
|
2011-10-06 23:26:07 +02:00
|
|
|
if(disposition == REG_CREATED_NEW_KEY)
|
|
|
|
update_reg_entries();
|
2004-03-09 04:43:54 +01:00
|
|
|
|
|
|
|
ReleaseMutex(font_mutex);
|
2001-09-12 22:21:06 +02:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2015-09-27 21:52:56 +02:00
|
|
|
/* Some fonts have large usWinDescent values, as a result of storing signed short
|
|
|
|
in unsigned field. That's probably caused by sTypoDescent vs usWinDescent confusion in
|
|
|
|
some font generation tools. */
|
|
|
|
static inline USHORT get_fixed_windescent(USHORT windescent)
|
|
|
|
{
|
|
|
|
return abs((SHORT)windescent);
|
|
|
|
}
|
2001-09-12 22:21:06 +02:00
|
|
|
|
2002-01-29 04:02:50 +01:00
|
|
|
static LONG calc_ppem_for_height(FT_Face ft_face, LONG height)
|
2001-09-12 22:21:06 +02:00
|
|
|
{
|
|
|
|
TT_OS2 *pOS2;
|
2003-12-08 22:53:15 +01:00
|
|
|
TT_HoriHeader *pHori;
|
|
|
|
|
2001-09-12 22:21:06 +02:00
|
|
|
LONG ppem;
|
2013-05-08 14:11:20 +02:00
|
|
|
const LONG MAX_PPEM = (1 << 16) - 1;
|
2001-09-12 22:21:06 +02:00
|
|
|
|
2002-04-03 22:02:39 +02:00
|
|
|
pOS2 = pFT_Get_Sfnt_Table(ft_face, ft_sfnt_os2);
|
2003-12-08 22:53:15 +01:00
|
|
|
pHori = pFT_Get_Sfnt_Table(ft_face, ft_sfnt_hhea);
|
2001-09-12 22:21:06 +02:00
|
|
|
|
|
|
|
if(height == 0) height = 16;
|
|
|
|
|
|
|
|
/* Calc. height of EM square:
|
|
|
|
*
|
|
|
|
* For +ve lfHeight we have
|
|
|
|
* lfHeight = (winAscent + winDescent) * ppem / units_per_em
|
|
|
|
* Re-arranging gives:
|
|
|
|
* ppem = units_per_em * lfheight / (winAscent + winDescent)
|
|
|
|
*
|
|
|
|
* For -ve lfHeight we have
|
|
|
|
* |lfHeight| = ppem
|
|
|
|
* [i.e. |lfHeight| = (winAscent + winDescent - il) * ppem / units_per_em
|
|
|
|
* with il = winAscent + winDescent - units_per_em]
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2003-12-08 22:53:15 +01:00
|
|
|
if(height > 0) {
|
2015-09-27 21:52:56 +02:00
|
|
|
USHORT windescent = get_fixed_windescent(pOS2->usWinDescent);
|
|
|
|
if(pOS2->usWinAscent + windescent == 0)
|
2008-06-24 09:10:27 +02:00
|
|
|
ppem = MulDiv(ft_face->units_per_EM, height,
|
|
|
|
pHori->Ascender - pHori->Descender);
|
2003-12-08 22:53:15 +01:00
|
|
|
else
|
2008-06-24 09:10:27 +02:00
|
|
|
ppem = MulDiv(ft_face->units_per_EM, height,
|
2015-09-27 21:52:56 +02:00
|
|
|
pOS2->usWinAscent + windescent);
|
2013-05-08 14:11:20 +02:00
|
|
|
if(ppem > MAX_PPEM) {
|
|
|
|
WARN("Ignoring too large height %d, ppem %d\n", height, ppem);
|
|
|
|
ppem = 1;
|
|
|
|
}
|
2003-12-08 22:53:15 +01:00
|
|
|
}
|
2013-05-08 14:11:20 +02:00
|
|
|
else if(height >= -MAX_PPEM)
|
2001-09-12 22:21:06 +02:00
|
|
|
ppem = -height;
|
2013-05-08 14:11:20 +02:00
|
|
|
else {
|
|
|
|
WARN("Ignoring too large height %d\n", height);
|
|
|
|
ppem = 1;
|
|
|
|
}
|
2001-09-12 22:21:06 +02:00
|
|
|
|
2002-01-29 04:02:50 +01:00
|
|
|
return ppem;
|
|
|
|
}
|
|
|
|
|
2007-09-13 19:42:55 +02:00
|
|
|
static struct font_mapping *map_font_file( const char *name )
|
2006-12-06 21:31:53 +01:00
|
|
|
{
|
|
|
|
struct font_mapping *mapping;
|
|
|
|
struct stat st;
|
|
|
|
int fd;
|
|
|
|
|
|
|
|
if ((fd = open( name, O_RDONLY )) == -1) return NULL;
|
|
|
|
if (fstat( fd, &st ) == -1) goto error;
|
|
|
|
|
|
|
|
LIST_FOR_EACH_ENTRY( mapping, &mappings_list, struct font_mapping, entry )
|
|
|
|
{
|
|
|
|
if (mapping->dev == st.st_dev && mapping->ino == st.st_ino)
|
|
|
|
{
|
|
|
|
mapping->refcount++;
|
|
|
|
close( fd );
|
|
|
|
return mapping;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!(mapping = HeapAlloc( GetProcessHeap(), 0, sizeof(*mapping) )))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
mapping->data = mmap( NULL, st.st_size, PROT_READ, MAP_PRIVATE, fd, 0 );
|
|
|
|
close( fd );
|
|
|
|
|
|
|
|
if (mapping->data == MAP_FAILED)
|
|
|
|
{
|
|
|
|
HeapFree( GetProcessHeap(), 0, mapping );
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
mapping->refcount = 1;
|
|
|
|
mapping->dev = st.st_dev;
|
|
|
|
mapping->ino = st.st_ino;
|
|
|
|
mapping->size = st.st_size;
|
|
|
|
list_add_tail( &mappings_list, &mapping->entry );
|
|
|
|
return mapping;
|
|
|
|
|
|
|
|
error:
|
|
|
|
close( fd );
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2007-09-13 19:42:55 +02:00
|
|
|
static void unmap_font_file( struct font_mapping *mapping )
|
2006-12-06 21:31:53 +01:00
|
|
|
{
|
|
|
|
if (!--mapping->refcount)
|
|
|
|
{
|
|
|
|
list_remove( &mapping->entry );
|
|
|
|
munmap( mapping->data, mapping->size );
|
|
|
|
HeapFree( GetProcessHeap(), 0, mapping );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-29 14:53:18 +01:00
|
|
|
static LONG load_VDMX(struct gdi_font *font, LONG height);
|
2002-01-29 04:02:50 +01:00
|
|
|
|
2020-10-21 11:04:35 +02:00
|
|
|
static UINT get_nearest_charset(const WCHAR *family_name, Face *face, UINT *cp)
|
2002-06-22 03:19:29 +02:00
|
|
|
{
|
|
|
|
/* Only get here if lfCharSet == DEFAULT_CHARSET or we couldn't find
|
|
|
|
a single face with the requested charset. The idea is to check if
|
|
|
|
the selected font supports the current ANSI codepage, if it does
|
|
|
|
return the corresponding charset, else return the first charset */
|
2002-01-29 04:02:50 +01:00
|
|
|
|
2002-06-22 03:19:29 +02:00
|
|
|
CHARSETINFO csi;
|
|
|
|
int acp = GetACP(), i;
|
|
|
|
DWORD fs0;
|
|
|
|
|
2004-08-26 20:24:03 +02:00
|
|
|
*cp = acp;
|
2006-07-19 09:17:49 +02:00
|
|
|
if(TranslateCharsetInfo((DWORD*)(INT_PTR)acp, &csi, TCI_SRCCODEPAGE))
|
2012-03-08 14:33:02 +01:00
|
|
|
{
|
2020-10-31 10:19:08 +01:00
|
|
|
const struct gdi_font_link *font_link;
|
2012-03-08 14:33:02 +01:00
|
|
|
|
|
|
|
if (csi.fs.fsCsb[0] & face->fs.fsCsb[0])
|
2002-06-22 03:19:29 +02:00
|
|
|
return csi.ciCharset;
|
|
|
|
|
2020-10-31 10:19:08 +01:00
|
|
|
font_link = find_gdi_font_link(family_name);
|
2012-03-08 14:33:02 +01:00
|
|
|
if (font_link != NULL && csi.fs.fsCsb[0] & font_link->fs.fsCsb[0])
|
|
|
|
return csi.ciCharset;
|
|
|
|
}
|
|
|
|
|
2002-06-22 03:19:29 +02:00
|
|
|
for(i = 0; i < 32; i++) {
|
|
|
|
fs0 = 1L << i;
|
2003-06-23 22:51:06 +02:00
|
|
|
if(face->fs.fsCsb[0] & fs0) {
|
2004-08-26 20:24:03 +02:00
|
|
|
if(TranslateCharsetInfo(&fs0, &csi, TCI_SRCFONTSIG)) {
|
|
|
|
*cp = csi.ciACP;
|
2002-06-22 03:19:29 +02:00
|
|
|
return csi.ciCharset;
|
2004-08-26 20:24:03 +02:00
|
|
|
}
|
2002-06-22 03:19:29 +02:00
|
|
|
else
|
2006-10-12 22:56:56 +02:00
|
|
|
FIXME("TCI failing on %x\n", fs0);
|
2002-06-22 03:19:29 +02:00
|
|
|
}
|
|
|
|
}
|
2002-01-29 04:02:50 +01:00
|
|
|
|
2006-10-12 22:56:56 +02:00
|
|
|
FIXME("returning DEFAULT_CHARSET face->fs.fsCsb[0] = %08x file = %s\n",
|
2012-11-01 13:16:26 +01:00
|
|
|
face->fs.fsCsb[0], debugstr_w(face->file));
|
2004-08-26 20:24:03 +02:00
|
|
|
*cp = acp;
|
2002-01-29 04:02:50 +01:00
|
|
|
return DEFAULT_CHARSET;
|
|
|
|
}
|
|
|
|
|
2020-10-20 22:05:37 +02:00
|
|
|
/*************************************************************
|
|
|
|
* freetype_destroy_font
|
|
|
|
*/
|
2020-10-29 14:53:18 +01:00
|
|
|
static void CDECL freetype_destroy_font( struct gdi_font *font )
|
2002-01-29 04:02:50 +01:00
|
|
|
{
|
2020-10-29 14:53:18 +01:00
|
|
|
struct font_private_data *data = font->private;
|
2005-09-07 11:21:50 +02:00
|
|
|
|
2020-10-29 14:53:18 +01:00
|
|
|
if (data->ft_face) pFT_Done_Face( data->ft_face );
|
|
|
|
if (data->mapping) unmap_font_file( data->mapping );
|
|
|
|
HeapFree( GetProcessHeap(), 0, data );
|
2002-01-29 04:02:50 +01:00
|
|
|
}
|
|
|
|
|
2020-10-22 11:57:48 +02:00
|
|
|
/*************************************************************
|
|
|
|
* freetype_get_font_data
|
|
|
|
*/
|
|
|
|
static DWORD CDECL freetype_get_font_data( struct gdi_font *font, DWORD table, DWORD offset,
|
|
|
|
void *buf, DWORD cbData)
|
2011-10-20 17:58:09 +02:00
|
|
|
{
|
2020-10-29 14:53:18 +01:00
|
|
|
FT_Face ft_face = get_ft_face( font );
|
2011-10-20 17:58:09 +02:00
|
|
|
FT_ULong len;
|
|
|
|
FT_Error err;
|
|
|
|
|
|
|
|
if (!FT_IS_SFNT(ft_face)) return GDI_ERROR;
|
|
|
|
|
|
|
|
if(!buf)
|
|
|
|
len = 0;
|
|
|
|
else
|
|
|
|
len = cbData;
|
|
|
|
|
2016-08-12 12:14:50 +02:00
|
|
|
/* if font is a member of TTC, 'ttcf' tag allows reading from beginning of TTC file,
|
|
|
|
0 tag means to read from start of collection member data. */
|
|
|
|
if (font->ttc_item_offset)
|
|
|
|
{
|
|
|
|
if (table == MS_TTCF_TAG)
|
|
|
|
table = 0;
|
|
|
|
else if (table == 0)
|
|
|
|
offset += font->ttc_item_offset;
|
|
|
|
}
|
|
|
|
|
2011-10-20 17:58:09 +02:00
|
|
|
/* make sure value of len is the value freetype says it needs */
|
|
|
|
if (buf && len)
|
|
|
|
{
|
|
|
|
FT_ULong needed = 0;
|
2020-10-22 11:57:48 +02:00
|
|
|
err = pFT_Load_Sfnt_Table(ft_face, RtlUlongByteSwap(table), offset, NULL, &needed);
|
2011-10-20 17:58:09 +02:00
|
|
|
if( !err && needed < len) len = needed;
|
|
|
|
}
|
2020-10-22 11:57:48 +02:00
|
|
|
err = pFT_Load_Sfnt_Table(ft_face, RtlUlongByteSwap(table), offset, buf, &len);
|
2011-10-20 17:58:09 +02:00
|
|
|
if (err)
|
|
|
|
{
|
2016-08-14 22:52:15 +02:00
|
|
|
TRACE("Can't find table %s\n", debugstr_an((char*)&table, 4));
|
2011-10-20 17:58:09 +02:00
|
|
|
return GDI_ERROR;
|
|
|
|
}
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
2002-01-29 04:02:50 +01:00
|
|
|
/*************************************************************
|
|
|
|
* load_VDMX
|
|
|
|
*
|
|
|
|
* load the vdmx entry for the specified height
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
|
2015-06-29 20:32:07 +02:00
|
|
|
typedef struct {
|
|
|
|
WORD version;
|
|
|
|
WORD numRecs;
|
|
|
|
WORD numRatios;
|
|
|
|
} VDMX_Header;
|
|
|
|
|
2002-01-29 04:02:50 +01:00
|
|
|
typedef struct {
|
|
|
|
BYTE bCharSet;
|
|
|
|
BYTE xRatio;
|
|
|
|
BYTE yStartRatio;
|
|
|
|
BYTE yEndRatio;
|
|
|
|
} Ratios;
|
|
|
|
|
2006-04-04 13:12:41 +02:00
|
|
|
typedef struct {
|
|
|
|
WORD recs;
|
|
|
|
BYTE startsz;
|
|
|
|
BYTE endsz;
|
|
|
|
} VDMX_group;
|
2002-01-29 04:02:50 +01:00
|
|
|
|
2015-06-29 20:32:07 +02:00
|
|
|
typedef struct {
|
|
|
|
WORD yPelHeight;
|
|
|
|
WORD yMax;
|
|
|
|
WORD yMin;
|
|
|
|
} VDMX_vTable;
|
|
|
|
|
2020-10-29 14:53:18 +01:00
|
|
|
static LONG load_VDMX(struct gdi_font *font, LONG height)
|
2002-01-29 04:02:50 +01:00
|
|
|
{
|
2015-06-29 20:32:07 +02:00
|
|
|
VDMX_Header hdr;
|
2006-04-04 13:12:41 +02:00
|
|
|
VDMX_group group;
|
2002-01-29 04:02:50 +01:00
|
|
|
BYTE devXRatio, devYRatio;
|
|
|
|
USHORT numRecs, numRatios;
|
2004-09-08 03:23:57 +02:00
|
|
|
DWORD result, offset = -1;
|
2002-01-29 04:02:50 +01:00
|
|
|
LONG ppem = 0;
|
2004-09-08 03:23:57 +02:00
|
|
|
int i;
|
2002-01-29 04:02:50 +01:00
|
|
|
|
2020-10-29 14:53:18 +01:00
|
|
|
result = freetype_get_font_data(font, MS_VDMX_TAG, 0, &hdr, sizeof(hdr));
|
2002-01-29 04:02:50 +01:00
|
|
|
|
|
|
|
if(result == GDI_ERROR) /* no vdmx table present, use linear scaling */
|
|
|
|
return ppem;
|
|
|
|
|
|
|
|
/* FIXME: need the real device aspect ratio */
|
|
|
|
devXRatio = 1;
|
|
|
|
devYRatio = 1;
|
|
|
|
|
2015-06-29 20:32:07 +02:00
|
|
|
numRecs = GET_BE_WORD(hdr.numRecs);
|
|
|
|
numRatios = GET_BE_WORD(hdr.numRatios);
|
2002-01-29 04:02:50 +01:00
|
|
|
|
2015-06-29 20:32:07 +02:00
|
|
|
TRACE("version = %d numRecs = %d numRatios = %d\n", GET_BE_WORD(hdr.version), numRecs, numRatios);
|
2002-01-29 04:02:50 +01:00
|
|
|
for(i = 0; i < numRatios; i++) {
|
|
|
|
Ratios ratio;
|
2002-06-01 01:06:46 +02:00
|
|
|
|
2015-06-29 20:32:07 +02:00
|
|
|
offset = sizeof(hdr) + (i * sizeof(Ratios));
|
2020-10-29 14:53:18 +01:00
|
|
|
freetype_get_font_data(font, MS_VDMX_TAG, offset, &ratio, sizeof(Ratios));
|
2002-01-29 04:02:50 +01:00
|
|
|
offset = -1;
|
|
|
|
|
|
|
|
TRACE("Ratios[%d] %d %d : %d -> %d\n", i, ratio.bCharSet, ratio.xRatio, ratio.yStartRatio, ratio.yEndRatio);
|
|
|
|
|
2013-08-28 15:21:13 +02:00
|
|
|
if (!ratio.bCharSet) continue;
|
|
|
|
|
2002-06-01 01:06:46 +02:00
|
|
|
if((ratio.xRatio == 0 &&
|
2002-01-29 04:02:50 +01:00
|
|
|
ratio.yStartRatio == 0 &&
|
|
|
|
ratio.yEndRatio == 0) ||
|
2002-06-01 01:06:46 +02:00
|
|
|
(devXRatio == ratio.xRatio &&
|
2002-01-29 04:02:50 +01:00
|
|
|
devYRatio >= ratio.yStartRatio &&
|
2002-06-01 01:06:46 +02:00
|
|
|
devYRatio <= ratio.yEndRatio))
|
2002-01-29 04:02:50 +01:00
|
|
|
{
|
2015-06-29 20:32:07 +02:00
|
|
|
WORD group_offset;
|
2014-11-25 20:08:22 +01:00
|
|
|
|
2015-06-29 20:32:07 +02:00
|
|
|
offset = sizeof(hdr) + numRatios * sizeof(ratio) + i * sizeof(group_offset);
|
2020-10-29 14:53:18 +01:00
|
|
|
freetype_get_font_data(font, MS_VDMX_TAG, offset, &group_offset, sizeof(group_offset));
|
2015-06-29 20:32:07 +02:00
|
|
|
offset = GET_BE_WORD(group_offset);
|
2002-01-29 04:02:50 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-28 15:21:13 +02:00
|
|
|
if(offset == -1) return 0;
|
2002-01-29 04:02:50 +01:00
|
|
|
|
2020-10-29 14:53:18 +01:00
|
|
|
if(freetype_get_font_data(font, MS_VDMX_TAG, offset, &group, sizeof(group)) != GDI_ERROR) {
|
2002-01-29 04:02:50 +01:00
|
|
|
USHORT recs;
|
|
|
|
BYTE startsz, endsz;
|
2006-04-04 13:12:41 +02:00
|
|
|
WORD *vTable;
|
2002-01-29 04:02:50 +01:00
|
|
|
|
2006-04-04 13:12:41 +02:00
|
|
|
recs = GET_BE_WORD(group.recs);
|
|
|
|
startsz = group.startsz;
|
|
|
|
endsz = group.endsz;
|
2002-01-29 04:02:50 +01:00
|
|
|
|
|
|
|
TRACE("recs=%d startsz=%d endsz=%d\n", recs, startsz, endsz);
|
|
|
|
|
2015-06-29 20:32:07 +02:00
|
|
|
vTable = HeapAlloc(GetProcessHeap(), 0, recs * sizeof(VDMX_vTable));
|
2020-10-29 14:53:18 +01:00
|
|
|
result = freetype_get_font_data(font, MS_VDMX_TAG, offset + sizeof(group), vTable, recs * sizeof(VDMX_vTable));
|
2002-01-29 04:02:50 +01:00
|
|
|
if(result == GDI_ERROR) {
|
|
|
|
FIXME("Failed to retrieve vTable\n");
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(height > 0) {
|
|
|
|
for(i = 0; i < recs; i++) {
|
2006-04-04 13:12:41 +02:00
|
|
|
SHORT yMax = GET_BE_WORD(vTable[(i * 3) + 1]);
|
|
|
|
SHORT yMin = GET_BE_WORD(vTable[(i * 3) + 2]);
|
|
|
|
ppem = GET_BE_WORD(vTable[i * 3]);
|
2002-01-29 04:02:50 +01:00
|
|
|
|
|
|
|
if(yMax + -yMin == height) {
|
2020-10-29 14:53:18 +01:00
|
|
|
font->yMax = yMax;
|
|
|
|
font->yMin = yMin;
|
|
|
|
TRACE("ppem %d found; height=%d yMax=%d yMin=%d\n", ppem, height, font->yMax, font->yMin);
|
2002-01-29 04:02:50 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
if(yMax + -yMin > height) {
|
|
|
|
if(--i < 0) {
|
|
|
|
ppem = 0;
|
|
|
|
goto end; /* failed */
|
|
|
|
}
|
2020-10-29 14:53:18 +01:00
|
|
|
font->yMax = GET_BE_WORD(vTable[(i * 3) + 1]);
|
|
|
|
font->yMin = GET_BE_WORD(vTable[(i * 3) + 2]);
|
2006-09-20 12:53:06 +02:00
|
|
|
ppem = GET_BE_WORD(vTable[i * 3]);
|
2020-10-29 14:53:18 +01:00
|
|
|
TRACE("ppem %d found; height=%d yMax=%d yMin=%d\n", ppem, height, font->yMax, font->yMin);
|
2002-01-29 04:02:50 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2020-10-29 14:53:18 +01:00
|
|
|
if(!font->yMax) {
|
2002-01-29 04:02:50 +01:00
|
|
|
ppem = 0;
|
2006-10-12 22:56:56 +02:00
|
|
|
TRACE("ppem not found for height %d\n", height);
|
2002-01-29 04:02:50 +01:00
|
|
|
}
|
2013-08-28 15:21:11 +02:00
|
|
|
} else {
|
|
|
|
ppem = -height;
|
|
|
|
if(ppem < startsz || ppem > endsz)
|
2013-08-28 15:21:12 +02:00
|
|
|
{
|
|
|
|
ppem = 0;
|
|
|
|
goto end;
|
|
|
|
}
|
2013-08-28 15:21:11 +02:00
|
|
|
|
|
|
|
for(i = 0; i < recs; i++) {
|
|
|
|
USHORT yPelHeight;
|
|
|
|
yPelHeight = GET_BE_WORD(vTable[i * 3]);
|
|
|
|
|
|
|
|
if(yPelHeight > ppem)
|
2013-08-28 15:21:12 +02:00
|
|
|
{
|
|
|
|
ppem = 0;
|
|
|
|
break; /* failed */
|
|
|
|
}
|
2013-08-28 15:21:11 +02:00
|
|
|
|
|
|
|
if(yPelHeight == ppem) {
|
2020-10-29 14:53:18 +01:00
|
|
|
font->yMax = GET_BE_WORD(vTable[(i * 3) + 1]);
|
|
|
|
font->yMin = GET_BE_WORD(vTable[(i * 3) + 2]);
|
|
|
|
TRACE("ppem %d found; yMax=%d yMin=%d\n", ppem, font->yMax, font->yMin);
|
2013-08-28 15:21:11 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2002-01-29 04:02:50 +01:00
|
|
|
}
|
|
|
|
end:
|
|
|
|
HeapFree(GetProcessHeap(), 0, vTable);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ppem;
|
|
|
|
}
|
|
|
|
|
2007-10-02 05:04:25 +02:00
|
|
|
static BOOL select_charmap(FT_Face ft_face, FT_Encoding encoding)
|
|
|
|
{
|
|
|
|
FT_Error ft_err = FT_Err_Invalid_CharMap_Handle;
|
2017-09-26 11:13:46 +02:00
|
|
|
FT_CharMap cmap0, cmap1, cmap2, cmap3, cmap_def;
|
|
|
|
FT_Int i;
|
2007-10-02 05:04:25 +02:00
|
|
|
|
2017-09-26 11:13:46 +02:00
|
|
|
cmap0 = cmap1 = cmap2 = cmap3 = cmap_def = NULL;
|
2007-10-02 05:04:25 +02:00
|
|
|
|
2017-09-26 11:13:46 +02:00
|
|
|
for (i = 0; i < ft_face->num_charmaps; i++)
|
|
|
|
{
|
|
|
|
if (ft_face->charmaps[i]->encoding == encoding)
|
2007-10-02 05:04:25 +02:00
|
|
|
{
|
2017-09-26 11:13:46 +02:00
|
|
|
TRACE("found cmap with platform_id %u, encoding_id %u\n",
|
|
|
|
ft_face->charmaps[i]->platform_id, ft_face->charmaps[i]->encoding_id);
|
2007-10-02 05:04:25 +02:00
|
|
|
|
2017-09-26 11:13:46 +02:00
|
|
|
switch (ft_face->charmaps[i]->platform_id)
|
|
|
|
{
|
|
|
|
default:
|
|
|
|
cmap_def = ft_face->charmaps[i];
|
|
|
|
break;
|
|
|
|
case 0: /* Apple Unicode */
|
|
|
|
cmap0 = ft_face->charmaps[i];
|
|
|
|
break;
|
|
|
|
case 1: /* Macintosh */
|
|
|
|
cmap1 = ft_face->charmaps[i];
|
|
|
|
break;
|
|
|
|
case 2: /* ISO */
|
|
|
|
cmap2 = ft_face->charmaps[i];
|
|
|
|
break;
|
|
|
|
case 3: /* Microsoft */
|
|
|
|
cmap3 = ft_face->charmaps[i];
|
|
|
|
break;
|
2007-10-02 05:04:25 +02:00
|
|
|
}
|
|
|
|
}
|
2017-09-26 11:13:46 +02:00
|
|
|
|
|
|
|
if (cmap3) /* prefer Microsoft cmap table */
|
|
|
|
ft_err = pFT_Set_Charmap(ft_face, cmap3);
|
|
|
|
else if (cmap1)
|
|
|
|
ft_err = pFT_Set_Charmap(ft_face, cmap1);
|
|
|
|
else if (cmap2)
|
|
|
|
ft_err = pFT_Set_Charmap(ft_face, cmap2);
|
|
|
|
else if (cmap0)
|
|
|
|
ft_err = pFT_Set_Charmap(ft_face, cmap0);
|
|
|
|
else if (cmap_def)
|
|
|
|
ft_err = pFT_Set_Charmap(ft_face, cmap_def);
|
2007-10-02 05:04:25 +02:00
|
|
|
}
|
|
|
|
|
2017-09-26 11:13:46 +02:00
|
|
|
return ft_err == FT_Err_Ok;
|
2007-10-02 05:04:25 +02:00
|
|
|
}
|
|
|
|
|
2011-10-18 11:44:41 +02:00
|
|
|
|
2012-10-26 15:04:04 +02:00
|
|
|
static FT_Encoding pick_charmap( FT_Face face, int charset )
|
|
|
|
{
|
2012-10-26 15:04:05 +02:00
|
|
|
static const FT_Encoding regular_order[] = { FT_ENCODING_UNICODE, FT_ENCODING_APPLE_ROMAN, FT_ENCODING_MS_SYMBOL, 0 };
|
2012-10-26 15:04:04 +02:00
|
|
|
static const FT_Encoding symbol_order[] = { FT_ENCODING_MS_SYMBOL, FT_ENCODING_UNICODE, FT_ENCODING_APPLE_ROMAN, 0 };
|
|
|
|
const FT_Encoding *encs = regular_order;
|
|
|
|
|
|
|
|
if (charset == SYMBOL_CHARSET) encs = symbol_order;
|
|
|
|
|
|
|
|
while (*encs != 0)
|
|
|
|
{
|
|
|
|
if (select_charmap( face, *encs )) break;
|
|
|
|
encs++;
|
|
|
|
}
|
2017-09-26 11:13:47 +02:00
|
|
|
|
|
|
|
if (!face->charmap && face->num_charmaps)
|
|
|
|
{
|
|
|
|
if (!pFT_Set_Charmap(face, face->charmaps[0]))
|
|
|
|
return face->charmap->encoding;
|
|
|
|
}
|
|
|
|
|
2012-10-26 15:04:04 +02:00
|
|
|
return *encs;
|
|
|
|
}
|
2011-10-18 11:44:41 +02:00
|
|
|
|
2020-10-29 14:53:18 +01:00
|
|
|
static BOOL get_gasp_flags( struct gdi_font *font, WORD *flags )
|
2012-11-02 17:21:28 +01:00
|
|
|
{
|
2020-10-29 14:53:18 +01:00
|
|
|
FT_Face ft_face = get_ft_face( font );
|
2012-11-02 17:21:28 +01:00
|
|
|
DWORD size;
|
|
|
|
WORD buf[16]; /* Enough for seven ranges before we need to alloc */
|
|
|
|
WORD *alloced = NULL, *ptr = buf;
|
|
|
|
WORD num_recs, version;
|
|
|
|
BOOL ret = FALSE;
|
|
|
|
|
|
|
|
*flags = 0;
|
2020-10-29 14:53:18 +01:00
|
|
|
size = freetype_get_font_data( font, MS_GASP_TAG, 0, NULL, 0 );
|
2012-11-02 17:21:28 +01:00
|
|
|
if (size == GDI_ERROR) return FALSE;
|
|
|
|
if (size < 4 * sizeof(WORD)) return FALSE;
|
|
|
|
if (size > sizeof(buf))
|
|
|
|
{
|
|
|
|
ptr = alloced = HeapAlloc( GetProcessHeap(), 0, size );
|
|
|
|
if (!ptr) return FALSE;
|
|
|
|
}
|
|
|
|
|
2020-10-29 14:53:18 +01:00
|
|
|
freetype_get_font_data( font, MS_GASP_TAG, 0, ptr, size );
|
2012-11-02 17:21:28 +01:00
|
|
|
|
|
|
|
version = GET_BE_WORD( *ptr++ );
|
|
|
|
num_recs = GET_BE_WORD( *ptr++ );
|
|
|
|
|
|
|
|
if (version > 1 || size < (num_recs * 2 + 2) * sizeof(WORD))
|
|
|
|
{
|
|
|
|
FIXME( "Unsupported gasp table: ver %d size %d recs %d\n", version, size, num_recs );
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (num_recs--)
|
|
|
|
{
|
|
|
|
*flags = GET_BE_WORD( *(ptr + 1) );
|
2020-10-29 14:53:18 +01:00
|
|
|
if (ft_face->size->metrics.y_ppem <= GET_BE_WORD( *ptr )) break;
|
2012-11-02 17:21:28 +01:00
|
|
|
ptr += 2;
|
|
|
|
}
|
2020-10-29 14:53:18 +01:00
|
|
|
TRACE( "got flags %04x for ppem %d\n", *flags, ft_face->size->metrics.y_ppem );
|
2012-11-02 17:21:28 +01:00
|
|
|
ret = TRUE;
|
|
|
|
|
|
|
|
done:
|
|
|
|
HeapFree( GetProcessHeap(), 0, alloced );
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2018-11-27 16:50:20 +01:00
|
|
|
#ifdef SONAME_LIBFONTCONFIG
|
2020-06-04 11:43:14 +02:00
|
|
|
static Family* get_fontconfig_family(DWORD pitch_and_family, const CHARSETINFO *csi, BOOL want_vertical)
|
2018-11-27 16:50:20 +01:00
|
|
|
{
|
|
|
|
const char *name;
|
|
|
|
WCHAR nameW[LF_FACESIZE];
|
|
|
|
FcChar8 *str;
|
|
|
|
FcPattern *pat = NULL, *best = NULL;
|
|
|
|
FcResult result;
|
|
|
|
FcBool r;
|
|
|
|
int ret, i;
|
|
|
|
Family *family = NULL;
|
|
|
|
|
|
|
|
if (!csi->fs.fsCsb[0]) return NULL;
|
|
|
|
|
|
|
|
if((pitch_and_family & FIXED_PITCH) ||
|
|
|
|
(pitch_and_family & 0xF0) == FF_MODERN)
|
|
|
|
name = "monospace";
|
|
|
|
else if((pitch_and_family & 0xF0) == FF_ROMAN)
|
|
|
|
name = "serif";
|
|
|
|
else
|
|
|
|
name = "sans-serif";
|
|
|
|
|
|
|
|
pat = pFcPatternCreate();
|
|
|
|
if (!pat) return NULL;
|
|
|
|
r = pFcPatternAddString(pat, FC_FAMILY, (const FcChar8 *)name);
|
|
|
|
if (!r) goto end;
|
|
|
|
r = pFcPatternAddString(pat, FC_NAMELANG, (const FcChar8 *)"en-us");
|
|
|
|
if (!r) goto end;
|
|
|
|
r = pFcPatternAddString(pat, FC_PRGNAME, (const FcChar8 *)"wine");
|
|
|
|
if (!r) goto end;
|
|
|
|
r = pFcConfigSubstitute(NULL, pat, FcMatchPattern);
|
|
|
|
if (!r) goto end;
|
|
|
|
pFcDefaultSubstitute(pat);
|
|
|
|
|
|
|
|
best = pFcFontMatch(NULL, pat, &result);
|
|
|
|
if (!best || result != FcResultMatch) goto end;
|
|
|
|
|
|
|
|
for (i = 0;
|
|
|
|
!family && pFcPatternGetString(best, FC_FAMILY, i, &str) == FcResultMatch;
|
|
|
|
i++)
|
|
|
|
{
|
|
|
|
Face *face;
|
2020-10-31 10:19:08 +01:00
|
|
|
const struct gdi_font_link *font_link;
|
2018-11-27 16:50:20 +01:00
|
|
|
const struct list *face_list;
|
|
|
|
|
2020-06-04 11:43:14 +02:00
|
|
|
if (!want_vertical)
|
|
|
|
{
|
|
|
|
ret = MultiByteToWideChar(CP_UTF8, 0, (const char*)str, -1,
|
|
|
|
nameW, ARRAY_SIZE(nameW));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
nameW[0] = '@';
|
|
|
|
ret = MultiByteToWideChar(CP_UTF8, 0, (const char*)str, -1,
|
|
|
|
nameW + 1, ARRAY_SIZE(nameW) - 1);
|
|
|
|
}
|
2018-11-27 16:50:20 +01:00
|
|
|
if (!ret) continue;
|
|
|
|
family = find_family_from_any_name(nameW);
|
|
|
|
if (!family) continue;
|
|
|
|
|
2020-10-31 10:19:08 +01:00
|
|
|
font_link = find_gdi_font_link( family->family_name );
|
2018-11-27 16:50:20 +01:00
|
|
|
face_list = get_face_list_from_family(family);
|
|
|
|
LIST_FOR_EACH_ENTRY( face, face_list, Face, entry ) {
|
|
|
|
if (!face->scalable)
|
|
|
|
continue;
|
|
|
|
if (csi->fs.fsCsb[0] & face->fs.fsCsb[0])
|
|
|
|
goto found;
|
|
|
|
if (font_link != NULL &&
|
|
|
|
csi->fs.fsCsb[0] & font_link->fs.fsCsb[0])
|
|
|
|
goto found;
|
|
|
|
}
|
|
|
|
family = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
found:
|
|
|
|
if (family)
|
|
|
|
TRACE("got %s\n", wine_dbgstr_w(nameW));
|
|
|
|
|
|
|
|
end:
|
2018-12-22 09:35:33 +01:00
|
|
|
pFcPatternDestroy(pat);
|
|
|
|
pFcPatternDestroy(best);
|
2018-11-27 16:50:20 +01:00
|
|
|
return family;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2020-10-27 14:10:41 +01:00
|
|
|
static DWORD get_ttc_offset( FT_Face ft_face, UINT face_index )
|
|
|
|
{
|
|
|
|
FT_ULong len;
|
|
|
|
DWORD header, offset;
|
|
|
|
|
|
|
|
/* see if it's a TTC */
|
|
|
|
len = sizeof(header);
|
|
|
|
if (pFT_Load_Sfnt_Table( ft_face, 0, 0, (void *)&header, &len )) return 0;
|
|
|
|
if (header != MS_TTCF_TAG) return 0;
|
|
|
|
|
|
|
|
len = sizeof(offset);
|
|
|
|
if (pFT_Load_Sfnt_Table( ft_face, 0, (3 + face_index) * sizeof(DWORD), (void *)&offset, &len ))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
return GET_BE_DWORD( offset );
|
|
|
|
}
|
|
|
|
|
2020-10-27 14:25:47 +01:00
|
|
|
/*************************************************************
|
|
|
|
* freetype_load_font
|
|
|
|
*/
|
2020-10-29 14:53:18 +01:00
|
|
|
static BOOL CDECL freetype_load_font( struct gdi_font *font )
|
2020-10-27 14:10:41 +01:00
|
|
|
{
|
2020-10-29 14:53:18 +01:00
|
|
|
struct font_private_data *data;
|
2020-10-27 14:10:41 +01:00
|
|
|
INT width = 0, height;
|
|
|
|
FT_Face ft_face;
|
|
|
|
void *data_ptr;
|
|
|
|
SIZE_T data_size;
|
|
|
|
|
2020-10-29 14:53:18 +01:00
|
|
|
if (!(data = HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(*data) ))) return FALSE;
|
|
|
|
font->private = data;
|
2020-10-27 14:25:32 +01:00
|
|
|
|
2020-10-29 14:53:18 +01:00
|
|
|
if (font->file[0])
|
2020-10-27 14:10:41 +01:00
|
|
|
{
|
2020-10-29 14:53:18 +01:00
|
|
|
char *filename = wine_get_unix_file_name( font->file );
|
|
|
|
data->mapping = map_font_file( filename );
|
2020-10-27 14:10:41 +01:00
|
|
|
HeapFree( GetProcessHeap(), 0, filename );
|
2020-10-29 14:53:18 +01:00
|
|
|
if (!data->mapping)
|
2020-10-27 14:10:41 +01:00
|
|
|
{
|
2020-10-29 14:53:18 +01:00
|
|
|
WARN("failed to map %s\n", debugstr_w(font->file));
|
2020-10-27 14:10:41 +01:00
|
|
|
return FALSE;
|
|
|
|
}
|
2020-10-29 14:53:18 +01:00
|
|
|
data_ptr = data->mapping->data;
|
|
|
|
data_size = data->mapping->size;
|
2020-10-27 14:10:41 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-10-29 14:53:18 +01:00
|
|
|
data_ptr = font->data_ptr;
|
|
|
|
data_size = font->data_size;
|
2020-10-27 14:10:41 +01:00
|
|
|
}
|
|
|
|
|
2020-10-29 14:53:18 +01:00
|
|
|
if (pFT_New_Memory_Face( library, data_ptr, data_size, font->face_index, &ft_face )) return FALSE;
|
2020-10-27 14:10:41 +01:00
|
|
|
|
2020-10-29 14:53:18 +01:00
|
|
|
data->ft_face = ft_face;
|
|
|
|
font->scalable = FT_IS_SCALABLE( ft_face );
|
|
|
|
if (!font->fs.fsCsb[0]) get_fontsig( ft_face, &font->fs );
|
|
|
|
if (!font->ntmFlags) font->ntmFlags = get_ntm_flags( ft_face );
|
|
|
|
if (!font->aa_flags) font->aa_flags = ADDFONT_AA_FLAGS( default_aa_flags );
|
|
|
|
if (!font->otm.otmpFamilyName)
|
2020-10-27 14:10:41 +01:00
|
|
|
{
|
|
|
|
WCHAR *family_name = ft_face_get_family_name( ft_face, GetSystemDefaultLCID() );
|
|
|
|
WCHAR *style_name = ft_face_get_style_name( ft_face, GetSystemDefaultLangID() );
|
|
|
|
WCHAR *full_name = ft_face_get_full_name( ft_face, GetSystemDefaultLangID() );
|
|
|
|
|
2020-10-29 14:53:18 +01:00
|
|
|
set_gdi_font_names( font, family_name, style_name, full_name );
|
2020-10-27 14:10:41 +01:00
|
|
|
HeapFree( GetProcessHeap(), 0, family_name );
|
|
|
|
HeapFree( GetProcessHeap(), 0, style_name );
|
|
|
|
HeapFree( GetProcessHeap(), 0, full_name );
|
|
|
|
}
|
|
|
|
|
2020-10-29 14:53:18 +01:00
|
|
|
if (font->scalable)
|
2020-10-27 14:10:41 +01:00
|
|
|
{
|
|
|
|
/* load the VDMX table if we have one */
|
2020-10-29 14:53:18 +01:00
|
|
|
font->ppem = load_VDMX( font, font->lf.lfHeight );
|
|
|
|
if (font->ppem == 0) font->ppem = calc_ppem_for_height( ft_face, font->lf.lfHeight );
|
|
|
|
TRACE( "height %d => ppem %d\n", font->lf.lfHeight, font->ppem );
|
|
|
|
height = font->ppem;
|
|
|
|
font->ttc_item_offset = get_ttc_offset( ft_face, font->face_index );
|
2020-10-27 14:10:41 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-10-30 14:31:13 +01:00
|
|
|
struct bitmap_font_size size;
|
2020-10-27 14:10:41 +01:00
|
|
|
|
|
|
|
get_bitmap_size( ft_face, &size );
|
|
|
|
width = size.x_ppem >> 6;
|
|
|
|
height = size.y_ppem >> 6;
|
2020-10-29 14:53:18 +01:00
|
|
|
font->ppem = height;
|
2020-10-27 14:10:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
pFT_Set_Pixel_Sizes( ft_face, width, height );
|
2020-10-29 14:53:18 +01:00
|
|
|
pick_charmap( ft_face, font->charset );
|
2020-10-27 14:10:41 +01:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2001-09-12 22:21:06 +02:00
|
|
|
/*************************************************************
|
2011-10-18 13:10:58 +02:00
|
|
|
* freetype_SelectFont
|
2001-09-12 22:21:06 +02:00
|
|
|
*/
|
2020-10-21 11:01:50 +02:00
|
|
|
static struct gdi_font * CDECL freetype_SelectFont( DC *dc, HFONT hfont, UINT *aa_flags,
|
|
|
|
UINT default_aa_flags )
|
2001-09-12 22:21:06 +02:00
|
|
|
{
|
2020-10-29 14:53:18 +01:00
|
|
|
struct gdi_font *font;
|
2006-08-29 08:21:52 +02:00
|
|
|
Face *face, *best, *best_bitmap;
|
2005-10-31 22:03:28 +01:00
|
|
|
Family *family, *last_resort_family;
|
2012-12-07 14:29:28 +01:00
|
|
|
const struct list *face_list;
|
2020-10-27 14:10:41 +01:00
|
|
|
INT height;
|
2006-08-29 08:21:52 +02:00
|
|
|
unsigned int score = 0, new_score;
|
2004-06-16 22:06:26 +02:00
|
|
|
signed int diff = 0, newdiff;
|
2012-02-08 17:33:00 +01:00
|
|
|
BOOL bd, it, can_use_bitmap, want_vertical;
|
2002-06-04 03:02:51 +02:00
|
|
|
LOGFONTW lf;
|
2002-06-22 03:19:29 +02:00
|
|
|
CHARSETINFO csi;
|
2008-07-07 17:18:10 +02:00
|
|
|
FMAT2 dcmat;
|
2020-10-30 14:15:34 +01:00
|
|
|
const WCHAR *orig_name = NULL;
|
2020-10-31 10:19:08 +01:00
|
|
|
const struct gdi_font_link *font_link;
|
2011-10-18 13:10:58 +02:00
|
|
|
|
|
|
|
GetObjectW( hfont, sizeof(lf), &lf );
|
2008-04-07 14:23:55 +02:00
|
|
|
lf.lfWidth = abs(lf.lfWidth);
|
|
|
|
|
2020-10-21 11:05:28 +02:00
|
|
|
can_use_bitmap = !!(GetDeviceCaps(dc->hSelf, TEXTCAPS) & TC_RA_ABLE);
|
2001-09-12 22:21:06 +02:00
|
|
|
|
2006-10-12 22:56:56 +02:00
|
|
|
TRACE("%s, h=%d, it=%d, weight=%d, PandF=%02x, charset=%d orient %d escapement %d\n",
|
2002-06-04 03:02:51 +02:00
|
|
|
debugstr_w(lf.lfFaceName), lf.lfHeight, lf.lfItalic,
|
|
|
|
lf.lfWeight, lf.lfPitchAndFamily, lf.lfCharSet, lf.lfOrientation,
|
|
|
|
lf.lfEscapement);
|
2002-01-29 04:02:50 +01:00
|
|
|
|
2008-07-07 17:18:10 +02:00
|
|
|
if(dc->GraphicsMode == GM_ADVANCED)
|
2012-02-16 13:22:02 +01:00
|
|
|
{
|
2008-07-07 17:18:10 +02:00
|
|
|
memcpy(&dcmat, &dc->xformWorld2Vport, sizeof(FMAT2));
|
2012-02-16 13:22:02 +01:00
|
|
|
/* Try to avoid not necessary glyph transformations */
|
|
|
|
if (dcmat.eM21 == 0.0 && dcmat.eM12 == 0.0 && dcmat.eM11 == dcmat.eM22)
|
|
|
|
{
|
|
|
|
lf.lfHeight *= fabs(dcmat.eM11);
|
|
|
|
lf.lfWidth *= fabs(dcmat.eM11);
|
2012-12-10 14:33:53 +01:00
|
|
|
dcmat.eM11 = dcmat.eM22 = dcmat.eM11 < 0 ? -1 : 1;
|
2012-02-16 13:22:02 +01:00
|
|
|
}
|
|
|
|
}
|
2008-07-07 17:18:10 +02:00
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Windows 3.1 compatibility mode GM_COMPATIBLE has only limited
|
|
|
|
font scaling abilities. */
|
2008-11-01 10:44:05 +01:00
|
|
|
dcmat.eM11 = dcmat.eM22 = 1.0;
|
2012-02-16 13:22:02 +01:00
|
|
|
dcmat.eM21 = dcmat.eM12 = 0;
|
2012-12-10 14:33:53 +01:00
|
|
|
lf.lfOrientation = lf.lfEscapement;
|
2012-02-16 13:22:02 +01:00
|
|
|
if (dc->vport2WorldValid)
|
|
|
|
{
|
|
|
|
if (dc->xformWorld2Vport.eM11 * dc->xformWorld2Vport.eM22 < 0)
|
|
|
|
lf.lfOrientation = -lf.lfOrientation;
|
|
|
|
lf.lfHeight *= fabs(dc->xformWorld2Vport.eM22);
|
|
|
|
lf.lfWidth *= fabs(dc->xformWorld2Vport.eM22);
|
|
|
|
}
|
2008-11-01 10:44:05 +01:00
|
|
|
}
|
|
|
|
|
2008-07-07 17:18:10 +02:00
|
|
|
TRACE("DC transform %f %f %f %f\n", dcmat.eM11, dcmat.eM12,
|
|
|
|
dcmat.eM21, dcmat.eM22);
|
2008-06-24 09:10:27 +02:00
|
|
|
|
2002-01-29 04:02:50 +01:00
|
|
|
/* check the cache first */
|
2020-10-29 14:53:18 +01:00
|
|
|
if ((font = find_cached_gdi_font( &lf, &dcmat, can_use_bitmap ))) {
|
|
|
|
TRACE("returning cached gdiFont(%p) for hFont %p\n", font, hfont);
|
2011-10-18 13:10:58 +02:00
|
|
|
goto done;
|
2002-01-29 04:02:50 +01:00
|
|
|
}
|
|
|
|
|
2002-06-22 03:19:29 +02:00
|
|
|
/* If lfFaceName is "Symbol" then Windows fixes up lfCharSet to
|
|
|
|
SYMBOL_CHARSET so that Symbol gets picked irrespective of the
|
|
|
|
original value lfCharSet. Note this is a special case for
|
|
|
|
Symbol and doesn't happen at least for "Wingdings*" */
|
|
|
|
|
|
|
|
if(!strcmpiW(lf.lfFaceName, SymbolW))
|
|
|
|
lf.lfCharSet = SYMBOL_CHARSET;
|
|
|
|
|
2020-10-26 11:37:03 +01:00
|
|
|
it = !!lf.lfItalic;
|
|
|
|
bd = lf.lfWeight > 550;
|
|
|
|
|
2006-07-19 09:17:49 +02:00
|
|
|
if(!TranslateCharsetInfo((DWORD*)(INT_PTR)lf.lfCharSet, &csi, TCI_SRCCHARSET)) {
|
2002-06-22 03:19:29 +02:00
|
|
|
switch(lf.lfCharSet) {
|
|
|
|
case DEFAULT_CHARSET:
|
|
|
|
csi.fs.fsCsb[0] = 0;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
FIXME("Untranslated charset %d\n", lf.lfCharSet);
|
|
|
|
csi.fs.fsCsb[0] = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-08-18 00:08:31 +02:00
|
|
|
family = NULL;
|
2002-06-04 03:02:51 +02:00
|
|
|
if(lf.lfFaceName[0] != '\0') {
|
2008-03-11 11:34:48 +01:00
|
|
|
LPWSTR FaceName = lf.lfFaceName;
|
2020-10-30 14:15:34 +01:00
|
|
|
int subst_charset;
|
|
|
|
const WCHAR *subst = get_gdi_font_subst( FaceName, lf.lfCharSet, &subst_charset );
|
2007-03-07 06:45:11 +01:00
|
|
|
|
2020-10-30 14:15:34 +01:00
|
|
|
if(subst) {
|
2008-06-22 15:41:47 +02:00
|
|
|
TRACE("substituting %s,%d -> %s,%d\n", debugstr_w(FaceName), lf.lfCharSet,
|
2020-10-30 14:15:34 +01:00
|
|
|
debugstr_w(subst), (subst_charset != -1) ? subst_charset : lf.lfCharSet);
|
|
|
|
if (subst_charset != -1) lf.lfCharSet = subst_charset;
|
|
|
|
orig_name = FaceName;
|
2002-04-03 22:51:20 +02:00
|
|
|
}
|
|
|
|
|
2002-06-22 03:19:29 +02:00
|
|
|
/* We want a match on name and charset or just name if
|
|
|
|
charset was DEFAULT_CHARSET. If the latter then
|
|
|
|
we fixup the returned charset later in get_nearest_charset
|
|
|
|
where we'll either use the charset of the current ansi codepage
|
|
|
|
or if that's unavailable the first charset that the font supports.
|
|
|
|
*/
|
2012-12-07 14:29:28 +01:00
|
|
|
LIST_FOR_EACH_ENTRY( family, &font_list, Family, entry ) {
|
2020-09-07 15:18:58 +02:00
|
|
|
if (!strncmpiW( family->family_name, FaceName, LF_FACESIZE - 1 ) ||
|
2020-10-30 14:15:34 +01:00
|
|
|
(subst && !strncmpiW( family->family_name, subst, LF_FACESIZE - 1 )))
|
2008-06-22 15:41:47 +02:00
|
|
|
{
|
2020-10-31 10:19:08 +01:00
|
|
|
font_link = find_gdi_font_link( family->family_name );
|
2012-03-08 14:33:15 +01:00
|
|
|
face_list = get_face_list_from_family(family);
|
2012-12-07 14:29:28 +01:00
|
|
|
LIST_FOR_EACH_ENTRY( face, face_list, Face, entry ) {
|
2012-03-08 14:33:02 +01:00
|
|
|
if (!(face->scalable || can_use_bitmap))
|
|
|
|
continue;
|
|
|
|
if (csi.fs.fsCsb[0] & face->fs.fsCsb[0])
|
|
|
|
goto found;
|
|
|
|
if (font_link != NULL &&
|
|
|
|
csi.fs.fsCsb[0] & font_link->fs.fsCsb[0])
|
|
|
|
goto found;
|
|
|
|
if (!csi.fs.fsCsb[0])
|
|
|
|
goto found;
|
2005-08-19 11:58:32 +02:00
|
|
|
}
|
2004-08-18 00:08:31 +02:00
|
|
|
}
|
2002-01-29 04:02:50 +01:00
|
|
|
}
|
2007-03-07 06:45:11 +01:00
|
|
|
|
2011-05-10 22:38:07 +02:00
|
|
|
/* Search by full face name. */
|
2012-12-07 14:29:28 +01:00
|
|
|
LIST_FOR_EACH_ENTRY( family, &font_list, Family, entry ) {
|
2012-03-08 14:33:15 +01:00
|
|
|
face_list = get_face_list_from_family(family);
|
2012-12-07 14:29:28 +01:00
|
|
|
LIST_FOR_EACH_ENTRY( face, face_list, Face, entry ) {
|
2020-09-07 15:18:56 +02:00
|
|
|
if (!strncmpiW( face->full_name, FaceName, LF_FACESIZE - 1 ) && (face->scalable || can_use_bitmap))
|
2011-05-10 22:38:07 +02:00
|
|
|
{
|
2012-03-08 14:33:02 +01:00
|
|
|
if (csi.fs.fsCsb[0] & face->fs.fsCsb[0] || !csi.fs.fsCsb[0])
|
|
|
|
goto found_face;
|
2020-10-31 10:19:08 +01:00
|
|
|
font_link = find_gdi_font_link( family->family_name );
|
2012-03-08 14:33:02 +01:00
|
|
|
if (font_link != NULL &&
|
|
|
|
csi.fs.fsCsb[0] & font_link->fs.fsCsb[0])
|
2011-05-10 22:38:07 +02:00
|
|
|
goto found_face;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-03-07 06:45:11 +01:00
|
|
|
/*
|
|
|
|
* Try check the SystemLink list first for a replacement font.
|
|
|
|
* We may find good replacements there.
|
|
|
|
*/
|
2020-10-31 10:19:08 +01:00
|
|
|
if ((family = find_family_from_font_links( FaceName, subst, csi.fs ))) goto found;
|
2002-01-29 04:02:50 +01:00
|
|
|
}
|
|
|
|
|
2020-10-30 14:15:34 +01:00
|
|
|
orig_name = NULL; /* substitution is no longer relevant */
|
2008-06-22 15:41:47 +02:00
|
|
|
|
2005-08-19 11:58:32 +02:00
|
|
|
/* If requested charset was DEFAULT_CHARSET then try using charset
|
|
|
|
corresponding to the current ansi codepage */
|
2008-10-09 07:30:00 +02:00
|
|
|
if (!csi.fs.fsCsb[0])
|
2008-05-13 15:10:05 +02:00
|
|
|
{
|
2005-08-19 11:58:32 +02:00
|
|
|
INT acp = GetACP();
|
2006-07-19 09:17:49 +02:00
|
|
|
if(!TranslateCharsetInfo((DWORD*)(INT_PTR)acp, &csi, TCI_SRCCODEPAGE)) {
|
2005-08-19 11:58:32 +02:00
|
|
|
FIXME("TCI failed on codepage %d\n", acp);
|
|
|
|
csi.fs.fsCsb[0] = 0;
|
|
|
|
} else
|
|
|
|
lf.lfCharSet = csi.ciCharset;
|
|
|
|
}
|
2003-10-04 06:00:08 +02:00
|
|
|
|
2012-02-08 17:33:00 +01:00
|
|
|
want_vertical = (lf.lfFaceName[0] == '@');
|
|
|
|
|
2005-08-19 11:58:32 +02:00
|
|
|
/* Face families are in the top 4 bits of lfPitchAndFamily,
|
|
|
|
so mask with 0xF0 before testing */
|
2003-10-04 06:00:08 +02:00
|
|
|
|
2005-08-19 11:58:32 +02:00
|
|
|
if((lf.lfPitchAndFamily & FIXED_PITCH) ||
|
|
|
|
(lf.lfPitchAndFamily & 0xF0) == FF_MODERN)
|
2018-12-20 11:32:14 +01:00
|
|
|
strcpyW(lf.lfFaceName, default_fixed);
|
2005-08-19 11:58:32 +02:00
|
|
|
else if((lf.lfPitchAndFamily & 0xF0) == FF_ROMAN)
|
2018-12-20 11:32:14 +01:00
|
|
|
strcpyW(lf.lfFaceName, default_serif);
|
2005-08-19 11:58:32 +02:00
|
|
|
else if((lf.lfPitchAndFamily & 0xF0) == FF_SWISS)
|
2018-12-20 11:32:14 +01:00
|
|
|
strcpyW(lf.lfFaceName, default_sans);
|
2005-08-19 11:58:32 +02:00
|
|
|
else
|
2018-12-20 11:32:14 +01:00
|
|
|
strcpyW(lf.lfFaceName, default_sans);
|
2012-12-07 14:29:28 +01:00
|
|
|
LIST_FOR_EACH_ENTRY( family, &font_list, Family, entry ) {
|
2020-09-07 15:18:58 +02:00
|
|
|
if (!strncmpiW( family->family_name, lf.lfFaceName, LF_FACESIZE - 1 ))
|
|
|
|
{
|
2020-10-31 10:19:08 +01:00
|
|
|
font_link = find_gdi_font_link( family->family_name );
|
2012-03-08 14:33:15 +01:00
|
|
|
face_list = get_face_list_from_family(family);
|
2012-12-07 14:29:28 +01:00
|
|
|
LIST_FOR_EACH_ENTRY( face, face_list, Face, entry ) {
|
2012-03-08 14:33:02 +01:00
|
|
|
if (!(face->scalable || can_use_bitmap))
|
|
|
|
continue;
|
|
|
|
if (csi.fs.fsCsb[0] & face->fs.fsCsb[0])
|
|
|
|
goto found;
|
|
|
|
if (font_link != NULL && csi.fs.fsCsb[0] & font_link->fs.fsCsb[0])
|
|
|
|
goto found;
|
2004-08-18 00:08:31 +02:00
|
|
|
}
|
2005-08-19 11:58:32 +02:00
|
|
|
}
|
2002-06-22 03:19:29 +02:00
|
|
|
}
|
|
|
|
|
2018-11-27 16:50:20 +01:00
|
|
|
#ifdef SONAME_LIBFONTCONFIG
|
|
|
|
/* Try FontConfig substitutions if the face isn't found */
|
2020-06-04 11:43:14 +02:00
|
|
|
family = get_fontconfig_family(lf.lfPitchAndFamily, &csi, want_vertical);
|
2018-11-27 16:50:20 +01:00
|
|
|
if (family) goto found;
|
|
|
|
#endif
|
|
|
|
|
2005-10-31 22:03:28 +01:00
|
|
|
last_resort_family = NULL;
|
2012-12-07 14:29:28 +01:00
|
|
|
LIST_FOR_EACH_ENTRY( family, &font_list, Family, entry ) {
|
2020-10-31 10:19:08 +01:00
|
|
|
font_link = find_gdi_font_link( family->family_name );
|
2012-03-08 14:33:15 +01:00
|
|
|
face_list = get_face_list_from_family(family);
|
2012-12-07 14:29:28 +01:00
|
|
|
LIST_FOR_EACH_ENTRY( face, face_list, Face, entry ) {
|
2013-01-15 14:13:09 +01:00
|
|
|
if(!(face->flags & ADDFONT_VERTICAL_FONT) == !want_vertical &&
|
2012-03-08 14:33:02 +01:00
|
|
|
(csi.fs.fsCsb[0] & face->fs.fsCsb[0] ||
|
|
|
|
(font_link != NULL && csi.fs.fsCsb[0] & font_link->fs.fsCsb[0]))) {
|
2005-10-31 22:03:28 +01:00
|
|
|
if(face->scalable)
|
2005-08-19 11:58:32 +02:00
|
|
|
goto found;
|
2005-10-31 22:03:28 +01:00
|
|
|
if(can_use_bitmap && !last_resort_family)
|
|
|
|
last_resort_family = family;
|
|
|
|
}
|
2005-08-19 11:58:32 +02:00
|
|
|
}
|
2001-09-12 22:21:06 +02:00
|
|
|
}
|
|
|
|
|
2005-10-31 22:03:28 +01:00
|
|
|
if(last_resort_family) {
|
|
|
|
family = last_resort_family;
|
|
|
|
csi.fs.fsCsb[0] = 0;
|
|
|
|
goto found;
|
|
|
|
}
|
|
|
|
|
2012-12-07 14:29:28 +01:00
|
|
|
LIST_FOR_EACH_ENTRY( family, &font_list, Family, entry ) {
|
2012-03-08 14:33:15 +01:00
|
|
|
face_list = get_face_list_from_family(family);
|
2012-12-07 14:29:28 +01:00
|
|
|
LIST_FOR_EACH_ENTRY( face, face_list, Face, entry ) {
|
2013-01-15 14:13:09 +01:00
|
|
|
if(face->scalable && !(face->flags & ADDFONT_VERTICAL_FONT) == !want_vertical) {
|
2004-08-11 20:49:34 +02:00
|
|
|
csi.fs.fsCsb[0] = 0;
|
2006-11-09 05:15:16 +01:00
|
|
|
WARN("just using first face for now\n");
|
2005-08-19 11:58:32 +02:00
|
|
|
goto found;
|
2004-08-11 20:49:34 +02:00
|
|
|
}
|
2005-10-31 22:03:28 +01:00
|
|
|
if(can_use_bitmap && !last_resort_family)
|
|
|
|
last_resort_family = family;
|
2004-08-11 20:49:34 +02:00
|
|
|
}
|
2001-09-12 22:21:06 +02:00
|
|
|
}
|
2005-10-31 22:03:28 +01:00
|
|
|
if(!last_resort_family) {
|
|
|
|
FIXME("can't find a single appropriate font - bailing\n");
|
2020-10-22 11:54:35 +02:00
|
|
|
return NULL;
|
2005-10-31 22:03:28 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
WARN("could only find a bitmap font - this will probably look awful!\n");
|
|
|
|
family = last_resort_family;
|
|
|
|
csi.fs.fsCsb[0] = 0;
|
2001-09-12 22:21:06 +02:00
|
|
|
|
2005-08-19 11:58:32 +02:00
|
|
|
found:
|
2001-09-12 22:21:06 +02:00
|
|
|
|
2008-06-24 09:11:40 +02:00
|
|
|
height = lf.lfHeight;
|
2004-06-16 22:06:26 +02:00
|
|
|
|
2006-08-29 08:21:52 +02:00
|
|
|
face = best = best_bitmap = NULL;
|
2020-10-31 10:19:08 +01:00
|
|
|
font_link = find_gdi_font_link( family->family_name );
|
2012-03-08 14:33:15 +01:00
|
|
|
face_list = get_face_list_from_family(family);
|
|
|
|
LIST_FOR_EACH_ENTRY(face, face_list, Face, entry)
|
2006-08-29 08:21:52 +02:00
|
|
|
{
|
2012-03-08 14:33:02 +01:00
|
|
|
if (csi.fs.fsCsb[0] & face->fs.fsCsb[0] ||
|
|
|
|
(font_link != NULL && csi.fs.fsCsb[0] & font_link->fs.fsCsb[0]) ||
|
|
|
|
!csi.fs.fsCsb[0])
|
2006-08-29 08:21:52 +02:00
|
|
|
{
|
2008-04-07 09:11:42 +02:00
|
|
|
BOOL italic, bold;
|
|
|
|
|
|
|
|
italic = (face->ntmFlags & NTM_ITALIC) ? 1 : 0;
|
|
|
|
bold = (face->ntmFlags & NTM_BOLD) ? 1 : 0;
|
|
|
|
new_score = (italic ^ it) + (bold ^ bd);
|
2006-08-29 08:21:52 +02:00
|
|
|
if(!best || new_score <= score)
|
|
|
|
{
|
|
|
|
TRACE("(it=%d, bd=%d) is selected for (it=%d, bd=%d)\n",
|
2008-04-07 09:11:42 +02:00
|
|
|
italic, bold, it, bd);
|
2006-08-29 08:21:52 +02:00
|
|
|
score = new_score;
|
2004-06-16 22:06:26 +02:00
|
|
|
best = face;
|
2006-08-29 08:21:52 +02:00
|
|
|
if(best->scalable && score == 0) break;
|
|
|
|
if(!best->scalable)
|
|
|
|
{
|
|
|
|
if(height > 0)
|
|
|
|
newdiff = height - (signed int)(best->size.height);
|
|
|
|
else
|
|
|
|
newdiff = -height - ((signed int)(best->size.height) - best->size.internal_leading);
|
|
|
|
if(!best_bitmap || new_score < score ||
|
|
|
|
(diff > 0 && newdiff < diff && newdiff >= 0) || (diff < 0 && newdiff > diff))
|
|
|
|
{
|
|
|
|
TRACE("%d is better for %d diff was %d\n", best->size.height, height, diff);
|
|
|
|
diff = newdiff;
|
|
|
|
best_bitmap = best;
|
|
|
|
if(score == 0 && diff == 0) break;
|
|
|
|
}
|
2005-08-19 11:58:32 +02:00
|
|
|
}
|
2004-08-10 00:54:33 +02:00
|
|
|
}
|
|
|
|
}
|
2002-01-29 04:02:50 +01:00
|
|
|
}
|
2006-08-29 08:21:52 +02:00
|
|
|
if(best)
|
|
|
|
face = best->scalable ? best : best_bitmap;
|
2002-06-22 03:19:29 +02:00
|
|
|
|
2011-05-10 22:38:07 +02:00
|
|
|
found_face:
|
|
|
|
height = lf.lfHeight;
|
|
|
|
|
2020-10-26 11:37:03 +01:00
|
|
|
TRACE("not in cache\n");
|
2020-10-31 15:58:54 +01:00
|
|
|
font = create_gdi_font( face, orig_name, &lf );
|
2020-10-29 14:53:18 +01:00
|
|
|
|
|
|
|
font->matrix = dcmat;
|
|
|
|
font->can_use_bitmap = can_use_bitmap;
|
2004-08-26 20:24:03 +02:00
|
|
|
if(csi.fs.fsCsb[0]) {
|
2020-10-29 14:53:18 +01:00
|
|
|
font->charset = lf.lfCharSet;
|
|
|
|
font->codepage = csi.ciACP;
|
2004-08-26 20:24:03 +02:00
|
|
|
}
|
2002-06-22 03:19:29 +02:00
|
|
|
else
|
2020-10-29 14:53:18 +01:00
|
|
|
font->charset = get_nearest_charset( family->family_name, face, &font->codepage );
|
2001-09-12 22:21:06 +02:00
|
|
|
|
2020-10-30 14:31:13 +01:00
|
|
|
TRACE( "Chosen: %s (%s/%p:%u)\n", debugstr_w(face->full_name), debugstr_w(face->file),
|
|
|
|
face->data_ptr, face->face_index );
|
2001-09-12 22:21:06 +02:00
|
|
|
|
2020-10-29 14:53:18 +01:00
|
|
|
font->aveWidth = height ? lf.lfWidth : 0;
|
2008-01-09 14:58:00 +01:00
|
|
|
|
2004-06-16 22:06:26 +02:00
|
|
|
if(!face->scalable) {
|
2008-01-20 14:21:45 +01:00
|
|
|
/* Windows uses integer scaling factors for bitmap fonts */
|
|
|
|
INT scale, scaled_height;
|
2020-10-21 11:01:50 +02:00
|
|
|
struct gdi_font *cachedfont;
|
2008-01-20 14:21:45 +01:00
|
|
|
|
2008-08-21 08:02:15 +02:00
|
|
|
/* FIXME: rotation of bitmap fonts is ignored */
|
2020-10-29 14:53:18 +01:00
|
|
|
height = abs(GDI_ROUND( (double)height * font->matrix.eM22 ));
|
|
|
|
if (font->aveWidth)
|
|
|
|
font->aveWidth = (double)font->aveWidth * font->matrix.eM11;
|
|
|
|
font->matrix.eM11 = font->matrix.eM22 = 1.0;
|
2010-07-20 00:17:10 +02:00
|
|
|
dcmat.eM11 = dcmat.eM22 = 1.0;
|
|
|
|
/* As we changed the matrix, we need to search the cache for the font again,
|
|
|
|
* otherwise we might explode the cache. */
|
2020-10-21 11:01:50 +02:00
|
|
|
if((cachedfont = find_cached_gdi_font( &lf, &dcmat, can_use_bitmap ))) {
|
2010-07-20 00:17:10 +02:00
|
|
|
TRACE("Found cached font after non-scalable matrix rescale!\n");
|
2020-10-29 14:53:18 +01:00
|
|
|
free_gdi_font( font );
|
|
|
|
font = cachedfont;
|
2011-10-18 13:10:58 +02:00
|
|
|
goto done;
|
2010-07-20 00:17:10 +02:00
|
|
|
}
|
2008-08-21 08:02:15 +02:00
|
|
|
|
2008-01-20 14:21:45 +01:00
|
|
|
if (height != 0) height = diff;
|
|
|
|
height += face->size.height;
|
|
|
|
|
|
|
|
scale = (height + face->size.height - 1) / face->size.height;
|
|
|
|
scaled_height = scale * face->size.height;
|
2009-02-20 15:40:44 +01:00
|
|
|
/* Only jump to the next height if the difference <= 25% original height */
|
|
|
|
if (scale > 2 && scaled_height - height > face->size.height / 4) scale--;
|
|
|
|
/* The jump between unscaled and doubled is delayed by 1 */
|
|
|
|
else if (scale == 2 && scaled_height - height > (face->size.height / 4 - 1)) scale--;
|
2020-10-29 14:53:18 +01:00
|
|
|
font->scale_y = scale;
|
2004-06-16 22:06:26 +02:00
|
|
|
}
|
2020-10-29 14:53:18 +01:00
|
|
|
TRACE("font scale y: %f\n", font->scale_y);
|
2008-01-20 14:21:45 +01:00
|
|
|
|
2020-10-29 14:53:18 +01:00
|
|
|
if (!freetype_load_font( font ))
|
2002-04-03 23:06:09 +02:00
|
|
|
{
|
2020-10-29 14:53:18 +01:00
|
|
|
free_gdi_font( font );
|
2020-10-22 11:54:35 +02:00
|
|
|
return NULL;
|
2002-04-03 23:06:09 +02:00
|
|
|
}
|
2003-10-28 01:08:28 +01:00
|
|
|
|
2020-10-27 14:10:41 +01:00
|
|
|
if (face->flags & ADDFONT_VERTICAL_FONT) /* We need to try to load the GSUB table */
|
2020-10-29 14:53:18 +01:00
|
|
|
font->vert_feature = get_GSUB_vert_feature( font );
|
2001-09-12 22:21:06 +02:00
|
|
|
|
2020-10-29 14:53:18 +01:00
|
|
|
create_child_font_list( font );
|
2002-01-29 04:02:50 +01:00
|
|
|
|
2020-10-29 14:53:18 +01:00
|
|
|
TRACE("caching: gdiFont=%p hfont=%p\n", font, hfont);
|
2004-08-11 01:42:18 +02:00
|
|
|
|
2020-10-29 14:53:18 +01:00
|
|
|
cache_gdi_font( font );
|
2011-10-18 13:10:58 +02:00
|
|
|
done:
|
2020-10-29 14:53:18 +01:00
|
|
|
if (font)
|
2011-10-18 13:10:58 +02:00
|
|
|
{
|
2012-11-05 12:32:15 +01:00
|
|
|
switch (lf.lfQuality)
|
2012-11-02 17:21:28 +01:00
|
|
|
{
|
2012-11-05 12:32:15 +01:00
|
|
|
case NONANTIALIASED_QUALITY:
|
|
|
|
case ANTIALIASED_QUALITY:
|
2020-10-20 22:05:37 +02:00
|
|
|
if (!*aa_flags) *aa_flags = default_aa_flags;
|
2012-11-05 12:32:15 +01:00
|
|
|
break;
|
|
|
|
case CLEARTYPE_QUALITY:
|
|
|
|
case CLEARTYPE_NATURAL_QUALITY:
|
|
|
|
default:
|
2020-10-29 14:53:18 +01:00
|
|
|
if (!*aa_flags) *aa_flags = font->aa_flags;
|
2020-10-20 22:05:37 +02:00
|
|
|
if (!*aa_flags) *aa_flags = default_aa_flags;
|
2012-11-05 12:32:15 +01:00
|
|
|
|
|
|
|
/* fixup the antialiasing flags for that font */
|
|
|
|
switch (*aa_flags)
|
2012-11-02 17:21:28 +01:00
|
|
|
{
|
2012-11-05 12:32:15 +01:00
|
|
|
case WINE_GGO_HRGB_BITMAP:
|
|
|
|
case WINE_GGO_HBGR_BITMAP:
|
|
|
|
case WINE_GGO_VRGB_BITMAP:
|
|
|
|
case WINE_GGO_VBGR_BITMAP:
|
|
|
|
if (is_subpixel_rendering_enabled()) break;
|
|
|
|
*aa_flags = GGO_GRAY4_BITMAP;
|
|
|
|
/* fall through */
|
|
|
|
case GGO_GRAY2_BITMAP:
|
|
|
|
case GGO_GRAY4_BITMAP:
|
|
|
|
case GGO_GRAY8_BITMAP:
|
|
|
|
case WINE_GGO_GRAY16_BITMAP:
|
2020-10-29 14:53:18 +01:00
|
|
|
if ((!antialias_fakes || (!font->fake_bold && !font->fake_italic)) && is_hinting_enabled())
|
2012-11-05 12:32:15 +01:00
|
|
|
{
|
|
|
|
WORD gasp_flags;
|
2020-10-29 14:53:18 +01:00
|
|
|
if (get_gasp_flags( font, &gasp_flags ) && !(gasp_flags & GASP_DOGRAY))
|
2012-11-05 22:32:51 +01:00
|
|
|
{
|
|
|
|
TRACE( "font %s %d aa disabled by GASP\n",
|
|
|
|
debugstr_w(lf.lfFaceName), lf.lfHeight );
|
2012-11-05 12:32:15 +01:00
|
|
|
*aa_flags = GGO_BITMAP;
|
2012-11-05 22:32:51 +01:00
|
|
|
}
|
2012-11-05 12:32:15 +01:00
|
|
|
}
|
2012-11-02 17:21:28 +01:00
|
|
|
}
|
|
|
|
}
|
2012-11-05 22:32:51 +01:00
|
|
|
TRACE( "%p %s %d aa %x\n", hfont, debugstr_w(lf.lfFaceName), lf.lfHeight, *aa_flags );
|
2011-10-18 13:10:58 +02:00
|
|
|
}
|
2020-10-29 14:53:18 +01:00
|
|
|
return font;
|
2001-09-12 22:21:06 +02:00
|
|
|
}
|
|
|
|
|
2002-01-29 04:02:50 +01:00
|
|
|
static void FTVectorToPOINTFX(FT_Vector *vec, POINTFX *pt)
|
|
|
|
{
|
|
|
|
pt->x.value = vec->x >> 6;
|
|
|
|
pt->x.fract = (vec->x & 0x3f) << 10;
|
|
|
|
pt->x.fract |= ((pt->x.fract >> 6) | (pt->x.fract >> 12));
|
|
|
|
pt->y.value = vec->y >> 6;
|
|
|
|
pt->y.fract = (vec->y & 0x3f) << 10;
|
|
|
|
pt->y.fract |= ((pt->y.fract >> 6) | (pt->y.fract >> 12));
|
|
|
|
}
|
|
|
|
|
2020-10-29 14:53:18 +01:00
|
|
|
static FT_UInt get_glyph_index_symbol( struct gdi_font *font, UINT glyph )
|
2018-06-22 12:35:43 +02:00
|
|
|
{
|
2020-10-29 14:53:18 +01:00
|
|
|
FT_Face ft_face = get_ft_face( font );
|
2018-06-22 12:35:43 +02:00
|
|
|
FT_UInt ret;
|
|
|
|
|
|
|
|
if (glyph < 0x100) glyph += 0xf000;
|
|
|
|
/* there are a number of old pre-Unicode "broken" TTFs, which
|
|
|
|
do have symbols at U+00XX instead of U+f0XX */
|
2020-10-29 14:53:18 +01:00
|
|
|
if (!(ret = pFT_Get_Char_Index(ft_face, glyph)))
|
|
|
|
ret = pFT_Get_Char_Index(ft_face, glyph - 0xf000);
|
2018-06-22 12:35:43 +02:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2020-10-22 13:13:52 +02:00
|
|
|
/*************************************************************
|
|
|
|
* freetype_get_glyph_index
|
|
|
|
*/
|
2020-10-29 14:53:18 +01:00
|
|
|
static BOOL CDECL freetype_get_glyph_index( struct gdi_font *font, UINT *glyph, BOOL use_encoding )
|
2013-12-11 13:11:15 +01:00
|
|
|
{
|
2020-10-29 14:53:18 +01:00
|
|
|
FT_Face ft_face = get_ft_face( font );
|
2013-12-11 13:11:15 +01:00
|
|
|
|
2020-10-29 14:53:18 +01:00
|
|
|
if (!use_encoding ^ (ft_face->charmap->encoding == FT_ENCODING_NONE)) return FALSE;
|
2013-12-11 13:11:15 +01:00
|
|
|
|
2020-10-29 14:53:18 +01:00
|
|
|
if (ft_face->charmap->encoding == FT_ENCODING_MS_SYMBOL)
|
2013-12-11 13:11:15 +01:00
|
|
|
{
|
2020-10-22 13:13:52 +02:00
|
|
|
if (!(*glyph = get_glyph_index_symbol( font, *glyph )))
|
|
|
|
{
|
|
|
|
WCHAR wc = *glyph;
|
|
|
|
char ch;
|
2013-05-22 12:46:28 +02:00
|
|
|
|
2020-10-22 13:13:52 +02:00
|
|
|
if (WideCharToMultiByte( CP_ACP, 0, &wc, 1, &ch, 1, NULL, NULL ))
|
|
|
|
*glyph = get_glyph_index_symbol( font, (unsigned char)ch );
|
|
|
|
}
|
|
|
|
return TRUE;
|
2013-05-22 12:46:28 +02:00
|
|
|
}
|
2020-10-29 14:53:18 +01:00
|
|
|
*glyph = pFT_Get_Char_Index( ft_face, *glyph );
|
2020-10-22 13:13:52 +02:00
|
|
|
return TRUE;
|
2013-05-22 12:46:28 +02:00
|
|
|
}
|
|
|
|
|
2002-04-03 22:41:14 +02:00
|
|
|
/*************************************************************
|
2020-10-22 13:13:52 +02:00
|
|
|
* freetype_get_default_glyph
|
2002-04-03 22:41:14 +02:00
|
|
|
*/
|
2020-10-29 14:53:18 +01:00
|
|
|
static UINT CDECL freetype_get_default_glyph( struct gdi_font *font )
|
2002-04-03 22:41:14 +02:00
|
|
|
{
|
2020-10-29 14:53:18 +01:00
|
|
|
FT_Face ft_face = get_ft_face( font );
|
2020-10-22 13:13:52 +02:00
|
|
|
FT_WinFNT_HeaderRec winfnt;
|
|
|
|
TT_OS2 *pOS2;
|
2002-04-03 22:41:14 +02:00
|
|
|
|
2020-10-29 14:53:18 +01:00
|
|
|
if ((pOS2 = pFT_Get_Sfnt_Table( ft_face, ft_sfnt_os2 )))
|
2006-08-23 16:49:13 +02:00
|
|
|
{
|
2020-10-22 13:13:52 +02:00
|
|
|
UINT glyph = pOS2->usDefaultChar;
|
2020-10-29 14:53:18 +01:00
|
|
|
freetype_get_glyph_index( font, &glyph, TRUE );
|
2020-10-22 13:13:52 +02:00
|
|
|
return glyph;
|
2006-08-23 16:49:13 +02:00
|
|
|
}
|
2020-10-29 14:53:18 +01:00
|
|
|
if (!pFT_Get_WinFNT_Header( ft_face, &winfnt )) return winfnt.default_char + winfnt.first_char;
|
2020-10-22 13:13:52 +02:00
|
|
|
return 32;
|
2002-04-03 22:41:14 +02:00
|
|
|
}
|
|
|
|
|
2020-10-22 13:13:52 +02:00
|
|
|
|
2008-06-24 09:11:40 +02:00
|
|
|
static inline BOOL is_identity_FMAT2(const FMAT2 *matrix)
|
|
|
|
{
|
|
|
|
static const FMAT2 identity = { 1.0, 0.0, 0.0, 1.0 };
|
|
|
|
return !memcmp(matrix, &identity, sizeof(FMAT2));
|
|
|
|
}
|
|
|
|
|
2016-01-13 15:09:08 +01:00
|
|
|
static inline FT_Vector normalize_vector(FT_Vector *vec)
|
|
|
|
{
|
|
|
|
FT_Vector out;
|
|
|
|
FT_Fixed len;
|
|
|
|
len = pFT_Vector_Length(vec);
|
|
|
|
if (len) {
|
|
|
|
out.x = (vec->x << 6) / len;
|
|
|
|
out.y = (vec->y << 6) / len;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
out.x = out.y = 0;
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
2019-02-07 15:43:38 +01:00
|
|
|
/* get_glyph_outline() glyph transform matrices index */
|
|
|
|
enum matrices_index
|
|
|
|
{
|
|
|
|
matrix_hori,
|
|
|
|
matrix_vert,
|
|
|
|
matrix_unrotated
|
|
|
|
};
|
|
|
|
|
2020-10-26 11:43:24 +01:00
|
|
|
static BOOL get_transform_matrices( struct gdi_font *font, BOOL vertical, const MAT2 *user_transform,
|
2019-02-07 15:43:38 +01:00
|
|
|
FT_Matrix matrices[3] )
|
|
|
|
{
|
|
|
|
static const FT_Matrix identity_mat = { (1 << 16), 0, 0, (1 << 16) };
|
|
|
|
BOOL needs_transform = FALSE;
|
|
|
|
double width_ratio;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
matrices[matrix_unrotated] = identity_mat;
|
|
|
|
|
|
|
|
/* Scaling factor */
|
2020-10-26 11:43:24 +01:00
|
|
|
if (font->aveWidth)
|
2019-02-07 15:43:38 +01:00
|
|
|
{
|
2020-10-26 11:43:24 +01:00
|
|
|
if (!freetype_set_outline_text_metrics( font )) freetype_set_bitmap_text_metrics( font );
|
|
|
|
width_ratio = (double)font->aveWidth;
|
|
|
|
width_ratio /= (double)font->otm.otmTextMetrics.tmAveCharWidth;
|
2019-02-07 15:43:38 +01:00
|
|
|
}
|
|
|
|
else
|
2020-10-26 11:43:24 +01:00
|
|
|
width_ratio = font->scale_y;
|
2019-02-07 15:43:38 +01:00
|
|
|
|
|
|
|
/* Scaling transform */
|
2020-10-26 11:43:24 +01:00
|
|
|
if (width_ratio != 1.0 || font->scale_y != 1.0)
|
2019-02-07 15:43:38 +01:00
|
|
|
{
|
|
|
|
FT_Matrix scale_mat;
|
|
|
|
scale_mat.xx = FT_FixedFromFloat( width_ratio );
|
|
|
|
scale_mat.xy = 0;
|
|
|
|
scale_mat.yx = 0;
|
2020-10-26 11:43:24 +01:00
|
|
|
scale_mat.yy = FT_FixedFromFloat( font->scale_y );
|
2019-02-07 15:43:38 +01:00
|
|
|
|
|
|
|
pFT_Matrix_Multiply( &scale_mat, &matrices[matrix_unrotated] );
|
|
|
|
needs_transform = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Slant transform */
|
2020-10-26 11:43:24 +01:00
|
|
|
if (font->fake_italic)
|
2019-02-07 15:43:38 +01:00
|
|
|
{
|
|
|
|
FT_Matrix slant_mat;
|
|
|
|
slant_mat.xx = (1 << 16);
|
|
|
|
slant_mat.xy = (1 << 16) >> 2;
|
|
|
|
slant_mat.yx = 0;
|
|
|
|
slant_mat.yy = (1 << 16);
|
|
|
|
|
|
|
|
pFT_Matrix_Multiply( &slant_mat, &matrices[matrix_unrotated] );
|
|
|
|
needs_transform = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Rotation transform */
|
|
|
|
matrices[matrix_hori] = matrices[matrix_unrotated];
|
2020-10-26 11:43:24 +01:00
|
|
|
if (font->scalable && font->lf.lfOrientation % 3600)
|
2019-02-07 15:43:38 +01:00
|
|
|
{
|
|
|
|
FT_Matrix rotation_mat;
|
|
|
|
FT_Vector angle;
|
|
|
|
|
2020-10-26 11:43:24 +01:00
|
|
|
pFT_Vector_Unit( &angle, MulDiv( 1 << 16, font->lf.lfOrientation, 10 ) );
|
2019-02-07 15:43:38 +01:00
|
|
|
rotation_mat.xx = angle.x;
|
|
|
|
rotation_mat.xy = -angle.y;
|
|
|
|
rotation_mat.yx = angle.y;
|
|
|
|
rotation_mat.yy = angle.x;
|
|
|
|
pFT_Matrix_Multiply( &rotation_mat, &matrices[matrix_hori] );
|
|
|
|
needs_transform = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Vertical transform */
|
|
|
|
matrices[matrix_vert] = matrices[matrix_hori];
|
|
|
|
if (vertical)
|
|
|
|
{
|
|
|
|
FT_Matrix vertical_mat = { 0, -(1 << 16), 1 << 16, 0 }; /* 90 degrees rotation */
|
|
|
|
|
|
|
|
pFT_Matrix_Multiply( &vertical_mat, &matrices[matrix_vert] );
|
|
|
|
needs_transform = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* World transform */
|
2020-10-26 11:43:24 +01:00
|
|
|
if (!is_identity_FMAT2( &font->matrix ))
|
2019-02-07 15:43:38 +01:00
|
|
|
{
|
|
|
|
FT_Matrix world_mat;
|
2020-10-26 11:43:24 +01:00
|
|
|
world_mat.xx = FT_FixedFromFloat( font->matrix.eM11 );
|
|
|
|
world_mat.xy = -FT_FixedFromFloat( font->matrix.eM21 );
|
|
|
|
world_mat.yx = -FT_FixedFromFloat( font->matrix.eM12 );
|
|
|
|
world_mat.yy = FT_FixedFromFloat( font->matrix.eM22 );
|
2019-02-07 15:43:38 +01:00
|
|
|
|
|
|
|
for (i = 0; i < 3; i++)
|
|
|
|
pFT_Matrix_Multiply( &world_mat, &matrices[i] );
|
|
|
|
needs_transform = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Extra transformation specified by caller */
|
2020-10-29 14:47:18 +01:00
|
|
|
if (user_transform)
|
2019-02-07 15:43:38 +01:00
|
|
|
{
|
|
|
|
FT_Matrix user_mat;
|
|
|
|
user_mat.xx = FT_FixedFromFIXED( user_transform->eM11 );
|
|
|
|
user_mat.xy = FT_FixedFromFIXED( user_transform->eM21 );
|
|
|
|
user_mat.yx = FT_FixedFromFIXED( user_transform->eM12 );
|
|
|
|
user_mat.yy = FT_FixedFromFIXED( user_transform->eM22 );
|
|
|
|
|
|
|
|
for (i = 0; i < 3; i++)
|
|
|
|
pFT_Matrix_Multiply( &user_mat, &matrices[i] );
|
|
|
|
needs_transform = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return needs_transform;
|
|
|
|
}
|
|
|
|
|
2016-01-13 15:09:09 +01:00
|
|
|
static BOOL get_bold_glyph_outline(FT_GlyphSlot glyph, LONG ppem, FT_Glyph_Metrics *metrics)
|
2013-10-17 14:25:10 +02:00
|
|
|
{
|
|
|
|
FT_Error err;
|
2016-01-13 15:09:09 +01:00
|
|
|
FT_Pos strength;
|
|
|
|
FT_BBox bbox;
|
2013-10-17 14:25:10 +02:00
|
|
|
|
2016-01-13 15:09:09 +01:00
|
|
|
if(glyph->format != FT_GLYPH_FORMAT_OUTLINE)
|
|
|
|
return FALSE;
|
|
|
|
if(!pFT_Outline_Embolden)
|
|
|
|
return FALSE;
|
2013-10-17 14:25:10 +02:00
|
|
|
|
2016-01-13 15:09:09 +01:00
|
|
|
strength = MulDiv(ppem, 1 << 6, 24);
|
|
|
|
err = pFT_Outline_Embolden(&glyph->outline, strength);
|
|
|
|
if(err) {
|
|
|
|
TRACE("FT_Ouline_Embolden returns %d\n", err);
|
|
|
|
return FALSE;
|
2013-10-17 14:25:10 +02:00
|
|
|
}
|
2016-01-13 15:09:09 +01:00
|
|
|
|
|
|
|
pFT_Outline_Get_CBox(&glyph->outline, &bbox);
|
|
|
|
metrics->width = bbox.xMax - bbox.xMin;
|
|
|
|
metrics->height = bbox.yMax - bbox.yMin;
|
|
|
|
metrics->horiBearingX = bbox.xMin;
|
|
|
|
metrics->horiBearingY = bbox.yMax;
|
|
|
|
metrics->vertBearingX = metrics->horiBearingX - metrics->horiAdvance / 2;
|
|
|
|
metrics->vertBearingY = (metrics->vertAdvance - metrics->height) / 2;
|
|
|
|
return TRUE;
|
2013-10-17 14:25:10 +02:00
|
|
|
}
|
|
|
|
|
2011-11-16 12:20:50 +01:00
|
|
|
static inline BYTE get_max_level( UINT format )
|
|
|
|
{
|
|
|
|
switch( format )
|
|
|
|
{
|
|
|
|
case GGO_GRAY2_BITMAP: return 4;
|
|
|
|
case GGO_GRAY4_BITMAP: return 16;
|
|
|
|
case GGO_GRAY8_BITMAP: return 64;
|
|
|
|
}
|
|
|
|
return 255;
|
|
|
|
}
|
|
|
|
|
2020-10-26 11:43:24 +01:00
|
|
|
static FT_Vector get_advance_metric(struct gdi_font *incoming_font, struct gdi_font *font,
|
2016-01-13 15:09:06 +01:00
|
|
|
const FT_Glyph_Metrics *metrics,
|
|
|
|
const FT_Matrix *transMat, BOOL vertical_metrics)
|
|
|
|
{
|
|
|
|
FT_Vector adv;
|
|
|
|
FT_Fixed base_advance, em_scale = 0;
|
|
|
|
BOOL fixed_pitch_full = FALSE;
|
|
|
|
|
|
|
|
if (vertical_metrics)
|
|
|
|
base_advance = metrics->vertAdvance;
|
|
|
|
else
|
|
|
|
base_advance = metrics->horiAdvance;
|
|
|
|
|
|
|
|
adv.x = base_advance;
|
|
|
|
adv.y = 0;
|
|
|
|
|
|
|
|
/* In fixed-pitch font, we adjust the fullwidth character advance so that
|
|
|
|
they have double halfwidth character width. E.g. if the font is 19 ppem,
|
|
|
|
we return 20 (not 19) for fullwidth characters as we return 10 for
|
|
|
|
halfwidth characters. */
|
2020-10-26 11:43:24 +01:00
|
|
|
if (freetype_set_outline_text_metrics(incoming_font) &&
|
|
|
|
!(incoming_font->otm.otmTextMetrics.tmPitchAndFamily & TMPF_FIXED_PITCH)) {
|
2016-01-13 15:09:06 +01:00
|
|
|
UINT avg_advance;
|
2020-10-26 11:43:24 +01:00
|
|
|
em_scale = MulDiv(incoming_font->ppem, 1 << 16,
|
2020-10-29 14:53:18 +01:00
|
|
|
get_ft_face(incoming_font)->units_per_EM);
|
2020-10-26 11:43:24 +01:00
|
|
|
avg_advance = pFT_MulFix(incoming_font->ntmAvgWidth, em_scale);
|
2016-01-13 15:09:06 +01:00
|
|
|
fixed_pitch_full = (avg_advance > 0 &&
|
|
|
|
(base_advance + 63) >> 6 ==
|
2020-10-26 11:43:24 +01:00
|
|
|
pFT_MulFix(incoming_font->ntmAvgWidth*2, em_scale));
|
2016-01-13 15:09:06 +01:00
|
|
|
if (fixed_pitch_full && !transMat)
|
|
|
|
adv.x = (avg_advance * 2) << 6;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (transMat) {
|
|
|
|
pFT_Vector_Transform(&adv, transMat);
|
|
|
|
if (fixed_pitch_full && adv.y == 0) {
|
|
|
|
FT_Vector vec;
|
2020-10-26 11:43:24 +01:00
|
|
|
vec.x = incoming_font->ntmAvgWidth;
|
2016-01-13 15:09:06 +01:00
|
|
|
vec.y = 0;
|
|
|
|
pFT_Vector_Transform(&vec, transMat);
|
|
|
|
adv.x = (pFT_MulFix(vec.x, em_scale) * 2) << 6;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-26 11:43:24 +01:00
|
|
|
if (font->fake_bold) {
|
2016-01-13 15:09:08 +01:00
|
|
|
if (!transMat)
|
|
|
|
adv.x += 1 << 6;
|
|
|
|
else {
|
|
|
|
FT_Vector fake_bold_adv, vec = { 1 << 6, 0 };
|
|
|
|
pFT_Vector_Transform(&vec, transMat);
|
|
|
|
fake_bold_adv = normalize_vector(&vec);
|
|
|
|
adv.x += fake_bold_adv.x;
|
|
|
|
adv.y += fake_bold_adv.y;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-01-13 15:09:06 +01:00
|
|
|
adv.x = (adv.x + 63) & -64;
|
|
|
|
adv.y = -((adv.y + 63) & -64);
|
|
|
|
return adv;
|
|
|
|
}
|
|
|
|
|
2019-02-08 02:51:30 +01:00
|
|
|
static FT_BBox get_transformed_bbox( const FT_Glyph_Metrics *metrics,
|
|
|
|
BOOL needs_transform, const FT_Matrix metrices[3] )
|
|
|
|
{
|
|
|
|
FT_BBox bbox = { 0, 0, 0, 0 };
|
|
|
|
|
|
|
|
if (!needs_transform)
|
|
|
|
{
|
|
|
|
bbox.xMin = (metrics->horiBearingX) & -64;
|
|
|
|
bbox.xMax = (metrics->horiBearingX + metrics->width + 63) & -64;
|
|
|
|
bbox.yMax = (metrics->horiBearingY + 63) & -64;
|
|
|
|
bbox.yMin = (metrics->horiBearingY - metrics->height) & -64;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
FT_Vector vec;
|
|
|
|
INT xc, yc;
|
|
|
|
|
|
|
|
for (xc = 0; xc < 2; xc++)
|
|
|
|
{
|
|
|
|
for (yc = 0; yc < 2; yc++)
|
|
|
|
{
|
|
|
|
vec.x = metrics->horiBearingX + xc * metrics->width;
|
|
|
|
vec.y = metrics->horiBearingY - yc * metrics->height;
|
|
|
|
TRACE( "Vec %ld,i %ld\n", vec.x, vec.y );
|
|
|
|
pFT_Vector_Transform( &vec, &metrices[matrix_vert] );
|
|
|
|
if (xc == 0 && yc == 0)
|
|
|
|
{
|
|
|
|
bbox.xMin = bbox.xMax = vec.x;
|
|
|
|
bbox.yMin = bbox.yMax = vec.y;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (vec.x < bbox.xMin) bbox.xMin = vec.x;
|
|
|
|
else if (vec.x > bbox.xMax) bbox.xMax = vec.x;
|
|
|
|
if (vec.y < bbox.yMin) bbox.yMin = vec.y;
|
|
|
|
else if (vec.y > bbox.yMax) bbox.yMax = vec.y;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
bbox.xMin = bbox.xMin & -64;
|
|
|
|
bbox.xMax = (bbox.xMax + 63) & -64;
|
|
|
|
bbox.yMin = bbox.yMin & -64;
|
|
|
|
bbox.yMax = (bbox.yMax + 63) & -64;
|
|
|
|
TRACE( "transformed box: (%ld, %ld - %ld, %ld)\n", bbox.xMin, bbox.yMax, bbox.xMax, bbox.yMin );
|
|
|
|
}
|
|
|
|
|
|
|
|
return bbox;
|
|
|
|
}
|
|
|
|
|
2020-10-26 11:43:24 +01:00
|
|
|
static void compute_metrics( struct gdi_font *incoming_font, struct gdi_font *font,
|
2019-02-08 02:51:30 +01:00
|
|
|
FT_BBox bbox, const FT_Glyph_Metrics *metrics,
|
|
|
|
BOOL vertical, BOOL vertical_metrics,
|
|
|
|
BOOL needs_transform, const FT_Matrix matrices[3],
|
|
|
|
GLYPHMETRICS *gm, ABC *abc )
|
|
|
|
{
|
|
|
|
FT_Vector adv, vec, origin;
|
|
|
|
|
|
|
|
if (!needs_transform)
|
|
|
|
{
|
|
|
|
adv = get_advance_metric( incoming_font, font, metrics, NULL, vertical_metrics );
|
|
|
|
gm->gmCellIncX = adv.x >> 6;
|
|
|
|
gm->gmCellIncY = 0;
|
|
|
|
origin.x = bbox.xMin;
|
|
|
|
origin.y = bbox.yMax;
|
|
|
|
abc->abcA = origin.x >> 6;
|
|
|
|
abc->abcB = (metrics->width + 63) >> 6;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
FT_Pos lsb;
|
|
|
|
|
2020-10-26 11:43:24 +01:00
|
|
|
if (vertical && freetype_set_outline_text_metrics( font ))
|
2019-02-08 02:51:30 +01:00
|
|
|
{
|
|
|
|
if (vertical_metrics)
|
|
|
|
lsb = metrics->horiBearingY + metrics->vertBearingY;
|
|
|
|
else
|
2020-10-26 11:43:24 +01:00
|
|
|
lsb = metrics->vertAdvance + (font->otm.otmDescent << 6);
|
2019-02-08 02:51:30 +01:00
|
|
|
vec.x = lsb;
|
2020-10-26 11:43:24 +01:00
|
|
|
vec.y = font->otm.otmDescent << 6;
|
2019-02-08 02:51:30 +01:00
|
|
|
TRACE( "Vec %ld,%ld\n", vec.x>>6, vec.y>>6 );
|
|
|
|
pFT_Vector_Transform( &vec, &matrices[matrix_hori] );
|
|
|
|
origin.x = (vec.x + bbox.xMin) & -64;
|
|
|
|
origin.y = (vec.y + bbox.yMax + 63) & -64;
|
|
|
|
lsb -= metrics->horiBearingY;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
origin.x = bbox.xMin;
|
|
|
|
origin.y = bbox.yMax;
|
|
|
|
lsb = metrics->horiBearingX;
|
|
|
|
}
|
|
|
|
|
|
|
|
adv = get_advance_metric( incoming_font, font, metrics, &matrices[matrix_hori],
|
|
|
|
vertical_metrics );
|
|
|
|
gm->gmCellIncX = adv.x >> 6;
|
|
|
|
gm->gmCellIncY = adv.y >> 6;
|
|
|
|
|
|
|
|
adv = get_advance_metric( incoming_font, font, metrics, &matrices[matrix_unrotated],
|
|
|
|
vertical_metrics );
|
|
|
|
adv.x = pFT_Vector_Length( &adv );
|
|
|
|
adv.y = 0;
|
|
|
|
|
|
|
|
vec.x = lsb;
|
|
|
|
vec.y = 0;
|
|
|
|
pFT_Vector_Transform( &vec, &matrices[matrix_unrotated] );
|
|
|
|
if (lsb > 0) abc->abcA = pFT_Vector_Length( &vec ) >> 6;
|
|
|
|
else abc->abcA = -((pFT_Vector_Length( &vec ) + 63) >> 6);
|
|
|
|
|
|
|
|
/* We use lsb again to avoid rounding errors */
|
|
|
|
vec.x = lsb + (vertical ? metrics->height : metrics->width);
|
|
|
|
vec.y = 0;
|
|
|
|
pFT_Vector_Transform( &vec, &matrices[matrix_unrotated] );
|
|
|
|
abc->abcB = ((pFT_Vector_Length( &vec ) + 63) >> 6) - abc->abcA;
|
|
|
|
}
|
|
|
|
if (!abc->abcB) abc->abcB = 1;
|
|
|
|
abc->abcC = (adv.x >> 6) - abc->abcA - abc->abcB;
|
|
|
|
|
|
|
|
gm->gmptGlyphOrigin.x = origin.x >> 6;
|
|
|
|
gm->gmptGlyphOrigin.y = origin.y >> 6;
|
|
|
|
gm->gmBlackBoxX = (bbox.xMax - bbox.xMin) >> 6;
|
|
|
|
gm->gmBlackBoxY = (bbox.yMax - bbox.yMin) >> 6;
|
|
|
|
if (!gm->gmBlackBoxX) gm->gmBlackBoxX = 1;
|
|
|
|
if (!gm->gmBlackBoxY) gm->gmBlackBoxY = 1;
|
|
|
|
|
|
|
|
TRACE( "gm: %u, %u, %s, %d, %d abc %d, %u, %d\n",
|
|
|
|
gm->gmBlackBoxX, gm->gmBlackBoxY, wine_dbgstr_point(&gm->gmptGlyphOrigin),
|
|
|
|
gm->gmCellIncX, gm->gmCellIncY, abc->abcA, abc->abcB, abc->abcC );
|
|
|
|
}
|
|
|
|
|
2019-02-08 02:51:31 +01:00
|
|
|
|
|
|
|
static const BYTE masks[8] = {0x80, 0x40, 0x20, 0x10, 0x08, 0x04, 0x02, 0x01};
|
|
|
|
|
|
|
|
static DWORD get_mono_glyph_bitmap( FT_GlyphSlot glyph, FT_BBox bbox,
|
|
|
|
BOOL fake_bold, BOOL needs_transform, FT_Matrix matrices[3],
|
|
|
|
DWORD buflen, BYTE *buf )
|
|
|
|
{
|
|
|
|
DWORD width = (bbox.xMax - bbox.xMin ) >> 6;
|
|
|
|
DWORD height = (bbox.yMax - bbox.yMin ) >> 6;
|
|
|
|
DWORD pitch = ((width + 31) >> 5) << 2;
|
|
|
|
DWORD needed = pitch * height;
|
|
|
|
FT_Bitmap ft_bitmap;
|
|
|
|
BYTE *src, *dst;
|
|
|
|
INT w, h, x;
|
|
|
|
|
|
|
|
if (!buf || !buflen) return needed;
|
|
|
|
if (!needed) return GDI_ERROR; /* empty glyph */
|
|
|
|
if (needed > buflen) return GDI_ERROR;
|
|
|
|
|
|
|
|
switch (glyph->format)
|
|
|
|
{
|
|
|
|
case FT_GLYPH_FORMAT_BITMAP:
|
|
|
|
src = glyph->bitmap.buffer;
|
|
|
|
dst = buf;
|
|
|
|
w = min( pitch, (glyph->bitmap.width + 7) >> 3 );
|
|
|
|
h = min( height, glyph->bitmap.rows );
|
|
|
|
while (h--)
|
|
|
|
{
|
|
|
|
if (!fake_bold)
|
|
|
|
memcpy( dst, src, w );
|
|
|
|
else
|
|
|
|
{
|
|
|
|
dst[0] = 0;
|
|
|
|
for (x = 0; x < w; x++)
|
|
|
|
{
|
|
|
|
dst[x] = (dst[x] & 0x80) | (src[x] >> 1) | src[x];
|
|
|
|
if (x + 1 < pitch)
|
|
|
|
dst[x + 1] = (src[x] & 0x01) << 7;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
src += glyph->bitmap.pitch;
|
|
|
|
dst += pitch;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case FT_GLYPH_FORMAT_OUTLINE:
|
|
|
|
ft_bitmap.width = width;
|
|
|
|
ft_bitmap.rows = height;
|
|
|
|
ft_bitmap.pitch = pitch;
|
|
|
|
ft_bitmap.pixel_mode = FT_PIXEL_MODE_MONO;
|
|
|
|
ft_bitmap.buffer = buf;
|
|
|
|
|
|
|
|
if (needs_transform)
|
|
|
|
pFT_Outline_Transform( &glyph->outline, &matrices[matrix_vert] );
|
|
|
|
pFT_Outline_Translate( &glyph->outline, -bbox.xMin, -bbox.yMin );
|
|
|
|
|
|
|
|
/* Note: FreeType will only set 'black' bits for us. */
|
|
|
|
memset( buf, 0, buflen );
|
|
|
|
pFT_Outline_Get_Bitmap( library, &glyph->outline, &ft_bitmap );
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
FIXME( "loaded glyph format %x\n", glyph->format );
|
|
|
|
return GDI_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
return needed;
|
|
|
|
}
|
|
|
|
|
|
|
|
static DWORD get_antialias_glyph_bitmap( FT_GlyphSlot glyph, FT_BBox bbox, UINT format,
|
|
|
|
BOOL fake_bold, BOOL needs_transform, FT_Matrix matrices[3],
|
|
|
|
DWORD buflen, BYTE *buf )
|
|
|
|
{
|
|
|
|
DWORD width = (bbox.xMax - bbox.xMin ) >> 6;
|
|
|
|
DWORD height = (bbox.yMax - bbox.yMin ) >> 6;
|
|
|
|
DWORD pitch = (width + 3) / 4 * 4;
|
|
|
|
DWORD needed = pitch * height;
|
|
|
|
FT_Bitmap ft_bitmap;
|
|
|
|
INT w, h, x, max_level;
|
|
|
|
BYTE *src, *dst;
|
|
|
|
|
|
|
|
if (!buf || !buflen) return needed;
|
|
|
|
if (!needed) return GDI_ERROR; /* empty glyph */
|
|
|
|
if (needed > buflen) return GDI_ERROR;
|
|
|
|
|
|
|
|
max_level = get_max_level( format );
|
|
|
|
|
|
|
|
switch (glyph->format)
|
|
|
|
{
|
|
|
|
case FT_GLYPH_FORMAT_BITMAP:
|
|
|
|
src = glyph->bitmap.buffer;
|
|
|
|
dst = buf;
|
|
|
|
memset( buf, 0, buflen );
|
|
|
|
|
|
|
|
w = min( pitch, glyph->bitmap.width );
|
|
|
|
h = min( height, glyph->bitmap.rows );
|
|
|
|
while (h--)
|
|
|
|
{
|
|
|
|
for (x = 0; x < w; x++)
|
|
|
|
{
|
|
|
|
if (src[x / 8] & masks[x % 8])
|
|
|
|
{
|
|
|
|
dst[x] = max_level;
|
|
|
|
if (fake_bold && x + 1 < pitch) dst[x + 1] = max_level;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
src += glyph->bitmap.pitch;
|
|
|
|
dst += pitch;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case FT_GLYPH_FORMAT_OUTLINE:
|
|
|
|
ft_bitmap.width = width;
|
|
|
|
ft_bitmap.rows = height;
|
|
|
|
ft_bitmap.pitch = pitch;
|
|
|
|
ft_bitmap.pixel_mode = FT_PIXEL_MODE_GRAY;
|
|
|
|
ft_bitmap.buffer = buf;
|
|
|
|
|
|
|
|
if (needs_transform)
|
|
|
|
pFT_Outline_Transform( &glyph->outline, &matrices[matrix_vert] );
|
|
|
|
pFT_Outline_Translate( &glyph->outline, -bbox.xMin, -bbox.yMin );
|
|
|
|
|
|
|
|
memset( buf, 0, buflen );
|
|
|
|
pFT_Outline_Get_Bitmap( library, &glyph->outline, &ft_bitmap );
|
|
|
|
|
|
|
|
if (max_level != 255)
|
|
|
|
{
|
|
|
|
INT row, col;
|
|
|
|
BYTE *ptr, *start;
|
|
|
|
|
|
|
|
for (row = 0, start = buf; row < height; row++)
|
|
|
|
{
|
|
|
|
for (col = 0, ptr = start; col < width; col++, ptr++)
|
|
|
|
*ptr = (((int)*ptr) * (max_level + 1)) / 256;
|
|
|
|
start += pitch;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
FIXME("loaded glyph format %x\n", glyph->format);
|
|
|
|
return GDI_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
return needed;
|
|
|
|
}
|
|
|
|
|
|
|
|
static DWORD get_subpixel_glyph_bitmap( FT_GlyphSlot glyph, FT_BBox bbox, UINT format,
|
|
|
|
BOOL fake_bold, BOOL needs_transform, FT_Matrix matrices[3],
|
|
|
|
GLYPHMETRICS *gm, DWORD buflen, BYTE *buf )
|
|
|
|
{
|
|
|
|
DWORD width = (bbox.xMax - bbox.xMin ) >> 6;
|
|
|
|
DWORD height = (bbox.yMax - bbox.yMin ) >> 6;
|
|
|
|
DWORD pitch, needed = 0;
|
|
|
|
BYTE *src, *dst;
|
|
|
|
INT w, h, x;
|
|
|
|
|
|
|
|
switch (glyph->format)
|
|
|
|
{
|
|
|
|
case FT_GLYPH_FORMAT_BITMAP:
|
|
|
|
pitch = width * 4;
|
|
|
|
needed = pitch * height;
|
|
|
|
|
|
|
|
if (!buf || !buflen) break;
|
|
|
|
if (!needed) return GDI_ERROR; /* empty glyph */
|
|
|
|
if (needed > buflen) return GDI_ERROR;
|
|
|
|
|
|
|
|
src = glyph->bitmap.buffer;
|
|
|
|
dst = buf;
|
|
|
|
memset( buf, 0, buflen );
|
|
|
|
|
|
|
|
w = min( width, glyph->bitmap.width );
|
|
|
|
h = min( height, glyph->bitmap.rows );
|
|
|
|
while (h--)
|
|
|
|
{
|
|
|
|
for (x = 0; x < w; x++)
|
|
|
|
{
|
|
|
|
if ( src[x / 8] & masks[x % 8] )
|
|
|
|
{
|
|
|
|
((unsigned int *)dst)[x] = ~0u;
|
|
|
|
if (fake_bold && x + 1 < width) ((unsigned int *)dst)[x + 1] = ~0u;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
src += glyph->bitmap.pitch;
|
|
|
|
dst += pitch;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case FT_GLYPH_FORMAT_OUTLINE:
|
|
|
|
{
|
|
|
|
INT src_pitch, src_width, src_height, x_shift, y_shift;
|
|
|
|
INT sub_stride, hmul, vmul;
|
|
|
|
const INT *sub_order;
|
|
|
|
const INT rgb_order[3] = { 0, 1, 2 };
|
|
|
|
const INT bgr_order[3] = { 2, 1, 0 };
|
|
|
|
FT_Render_Mode render_mode =
|
|
|
|
(format == WINE_GGO_HRGB_BITMAP ||
|
|
|
|
format == WINE_GGO_HBGR_BITMAP) ? FT_RENDER_MODE_LCD : FT_RENDER_MODE_LCD_V;
|
|
|
|
|
|
|
|
if (!width || !height) /* empty glyph */
|
|
|
|
{
|
|
|
|
if (!buf || !buflen) break;
|
|
|
|
return GDI_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( render_mode == FT_RENDER_MODE_LCD)
|
|
|
|
{
|
|
|
|
gm->gmBlackBoxX += 2;
|
|
|
|
gm->gmptGlyphOrigin.x -= 1;
|
|
|
|
bbox.xMin -= (1 << 6);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
gm->gmBlackBoxY += 2;
|
|
|
|
gm->gmptGlyphOrigin.y += 1;
|
|
|
|
bbox.yMax += (1 << 6);
|
|
|
|
}
|
|
|
|
|
|
|
|
width = gm->gmBlackBoxX;
|
|
|
|
height = gm->gmBlackBoxY;
|
|
|
|
pitch = width * 4;
|
|
|
|
needed = pitch * height;
|
|
|
|
|
|
|
|
if (!buf || !buflen) return needed;
|
|
|
|
if (needed > buflen) return GDI_ERROR;
|
|
|
|
|
|
|
|
if (needs_transform)
|
|
|
|
pFT_Outline_Transform( &glyph->outline, &matrices[matrix_vert] );
|
|
|
|
|
|
|
|
#ifdef FT_LCD_FILTER_H
|
|
|
|
if (pFT_Library_SetLcdFilter)
|
|
|
|
pFT_Library_SetLcdFilter( library, FT_LCD_FILTER_DEFAULT );
|
|
|
|
#endif
|
|
|
|
pFT_Render_Glyph( glyph, render_mode );
|
|
|
|
|
|
|
|
src_pitch = glyph->bitmap.pitch;
|
|
|
|
src_width = glyph->bitmap.width;
|
|
|
|
src_height = glyph->bitmap.rows;
|
|
|
|
src = glyph->bitmap.buffer;
|
|
|
|
dst = buf;
|
|
|
|
memset( buf, 0, buflen );
|
|
|
|
|
|
|
|
sub_order = (format == WINE_GGO_HRGB_BITMAP ||
|
|
|
|
format == WINE_GGO_VRGB_BITMAP) ? rgb_order : bgr_order;
|
|
|
|
sub_stride = render_mode == FT_RENDER_MODE_LCD ? 1 : src_pitch;
|
|
|
|
hmul = render_mode == FT_RENDER_MODE_LCD ? 3 : 1;
|
|
|
|
vmul = render_mode == FT_RENDER_MODE_LCD ? 1 : 3;
|
|
|
|
|
|
|
|
x_shift = glyph->bitmap_left - (bbox.xMin >> 6);
|
|
|
|
if ( x_shift < 0 )
|
|
|
|
{
|
|
|
|
src += hmul * -x_shift;
|
|
|
|
src_width -= hmul * -x_shift;
|
|
|
|
}
|
|
|
|
else if ( x_shift > 0 )
|
|
|
|
{
|
|
|
|
dst += x_shift * sizeof(unsigned int);
|
|
|
|
width -= x_shift;
|
|
|
|
}
|
|
|
|
|
|
|
|
y_shift = (bbox.yMax >> 6) - glyph->bitmap_top;
|
|
|
|
if ( y_shift < 0 )
|
|
|
|
{
|
|
|
|
src += src_pitch * vmul * -y_shift;
|
|
|
|
src_height -= vmul * -y_shift;
|
|
|
|
}
|
|
|
|
else if ( y_shift > 0 )
|
|
|
|
{
|
|
|
|
dst += y_shift * pitch;
|
|
|
|
height -= y_shift;
|
|
|
|
}
|
|
|
|
|
|
|
|
w = min( width, src_width / hmul );
|
|
|
|
h = min( height, src_height / vmul );
|
|
|
|
while (h--)
|
|
|
|
{
|
|
|
|
for (x = 0; x < w; x++)
|
|
|
|
{
|
|
|
|
((unsigned int *)dst)[x] =
|
|
|
|
((unsigned int)src[hmul * x + sub_stride * sub_order[0]] << 16) |
|
|
|
|
((unsigned int)src[hmul * x + sub_stride * sub_order[1]] << 8) |
|
|
|
|
((unsigned int)src[hmul * x + sub_stride * sub_order[2]]);
|
|
|
|
}
|
|
|
|
src += src_pitch * vmul;
|
|
|
|
dst += pitch;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
FIXME ( "loaded glyph format %x\n", glyph->format );
|
|
|
|
return GDI_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
return needed;
|
|
|
|
}
|
|
|
|
|
2014-10-09 18:00:55 +02:00
|
|
|
static unsigned int get_native_glyph_outline(FT_Outline *outline, unsigned int buflen, char *buf)
|
|
|
|
{
|
|
|
|
TTPOLYGONHEADER *pph;
|
|
|
|
TTPOLYCURVE *ppc;
|
|
|
|
unsigned int needed = 0, point = 0, contour, first_pt;
|
|
|
|
unsigned int pph_start, cpfx;
|
|
|
|
DWORD type;
|
|
|
|
|
|
|
|
for (contour = 0; contour < outline->n_contours; contour++)
|
|
|
|
{
|
|
|
|
/* Ignore contours containing one point */
|
|
|
|
if (point == outline->contours[contour])
|
|
|
|
{
|
|
|
|
point++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
pph_start = needed;
|
|
|
|
pph = (TTPOLYGONHEADER *)(buf + needed);
|
|
|
|
first_pt = point;
|
|
|
|
if (buf)
|
|
|
|
{
|
|
|
|
pph->dwType = TT_POLYGON_TYPE;
|
|
|
|
FTVectorToPOINTFX(&outline->points[point], &pph->pfxStart);
|
|
|
|
}
|
|
|
|
needed += sizeof(*pph);
|
|
|
|
point++;
|
|
|
|
while (point <= outline->contours[contour])
|
|
|
|
{
|
|
|
|
ppc = (TTPOLYCURVE *)(buf + needed);
|
|
|
|
type = outline->tags[point] & FT_Curve_Tag_On ?
|
|
|
|
TT_PRIM_LINE : TT_PRIM_QSPLINE;
|
|
|
|
cpfx = 0;
|
|
|
|
do
|
|
|
|
{
|
|
|
|
if (buf)
|
|
|
|
FTVectorToPOINTFX(&outline->points[point], &ppc->apfx[cpfx]);
|
|
|
|
cpfx++;
|
|
|
|
point++;
|
|
|
|
} while (point <= outline->contours[contour] &&
|
|
|
|
(outline->tags[point] & FT_Curve_Tag_On) ==
|
|
|
|
(outline->tags[point-1] & FT_Curve_Tag_On));
|
|
|
|
/* At the end of a contour Windows adds the start point, but
|
|
|
|
only for Beziers */
|
|
|
|
if (point > outline->contours[contour] &&
|
|
|
|
!(outline->tags[point-1] & FT_Curve_Tag_On))
|
|
|
|
{
|
|
|
|
if (buf)
|
|
|
|
FTVectorToPOINTFX(&outline->points[first_pt], &ppc->apfx[cpfx]);
|
|
|
|
cpfx++;
|
|
|
|
}
|
|
|
|
else if (point <= outline->contours[contour] &&
|
|
|
|
outline->tags[point] & FT_Curve_Tag_On)
|
|
|
|
{
|
|
|
|
/* add closing pt for bezier */
|
|
|
|
if (buf)
|
|
|
|
FTVectorToPOINTFX(&outline->points[point], &ppc->apfx[cpfx]);
|
|
|
|
cpfx++;
|
|
|
|
point++;
|
|
|
|
}
|
|
|
|
if (buf)
|
|
|
|
{
|
|
|
|
ppc->wType = type;
|
|
|
|
ppc->cpfx = cpfx;
|
|
|
|
}
|
|
|
|
needed += sizeof(*ppc) + (cpfx - 1) * sizeof(POINTFX);
|
|
|
|
}
|
|
|
|
if (buf)
|
|
|
|
pph->cb = needed - pph_start;
|
|
|
|
}
|
|
|
|
return needed;
|
|
|
|
}
|
|
|
|
|
|
|
|
static unsigned int get_bezier_glyph_outline(FT_Outline *outline, unsigned int buflen, char *buf)
|
|
|
|
{
|
|
|
|
/* Convert the quadratic Beziers to cubic Beziers.
|
|
|
|
The parametric eqn for a cubic Bezier is, from PLRM:
|
|
|
|
r(t) = at^3 + bt^2 + ct + r0
|
|
|
|
with the control points:
|
|
|
|
r1 = r0 + c/3
|
|
|
|
r2 = r1 + (c + b)/3
|
|
|
|
r3 = r0 + c + b + a
|
|
|
|
|
|
|
|
A quadratic Bezier has the form:
|
|
|
|
p(t) = (1-t)^2 p0 + 2(1-t)t p1 + t^2 p2
|
|
|
|
|
|
|
|
So equating powers of t leads to:
|
|
|
|
r1 = 2/3 p1 + 1/3 p0
|
|
|
|
r2 = 2/3 p1 + 1/3 p2
|
|
|
|
and of course r0 = p0, r3 = p2
|
|
|
|
*/
|
|
|
|
int contour, point = 0, first_pt;
|
|
|
|
TTPOLYGONHEADER *pph;
|
|
|
|
TTPOLYCURVE *ppc;
|
|
|
|
DWORD pph_start, cpfx, type;
|
|
|
|
FT_Vector cubic_control[4];
|
|
|
|
unsigned int needed = 0;
|
|
|
|
|
|
|
|
for (contour = 0; contour < outline->n_contours; contour++)
|
|
|
|
{
|
|
|
|
pph_start = needed;
|
|
|
|
pph = (TTPOLYGONHEADER *)(buf + needed);
|
|
|
|
first_pt = point;
|
|
|
|
if (buf)
|
|
|
|
{
|
|
|
|
pph->dwType = TT_POLYGON_TYPE;
|
|
|
|
FTVectorToPOINTFX(&outline->points[point], &pph->pfxStart);
|
|
|
|
}
|
|
|
|
needed += sizeof(*pph);
|
|
|
|
point++;
|
|
|
|
while (point <= outline->contours[contour])
|
|
|
|
{
|
|
|
|
ppc = (TTPOLYCURVE *)(buf + needed);
|
|
|
|
type = outline->tags[point] & FT_Curve_Tag_On ?
|
|
|
|
TT_PRIM_LINE : TT_PRIM_CSPLINE;
|
|
|
|
cpfx = 0;
|
|
|
|
do
|
|
|
|
{
|
|
|
|
if (type == TT_PRIM_LINE)
|
|
|
|
{
|
|
|
|
if (buf)
|
|
|
|
FTVectorToPOINTFX(&outline->points[point], &ppc->apfx[cpfx]);
|
|
|
|
cpfx++;
|
|
|
|
point++;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Unlike QSPLINEs, CSPLINEs always have their endpoint
|
|
|
|
so cpfx = 3n */
|
|
|
|
|
|
|
|
/* FIXME: Possible optimization in endpoint calculation
|
|
|
|
if there are two consecutive curves */
|
|
|
|
cubic_control[0] = outline->points[point-1];
|
|
|
|
if (!(outline->tags[point-1] & FT_Curve_Tag_On))
|
|
|
|
{
|
|
|
|
cubic_control[0].x += outline->points[point].x + 1;
|
|
|
|
cubic_control[0].y += outline->points[point].y + 1;
|
|
|
|
cubic_control[0].x >>= 1;
|
|
|
|
cubic_control[0].y >>= 1;
|
|
|
|
}
|
|
|
|
if (point+1 > outline->contours[contour])
|
|
|
|
cubic_control[3] = outline->points[first_pt];
|
|
|
|
else
|
|
|
|
{
|
|
|
|
cubic_control[3] = outline->points[point+1];
|
|
|
|
if (!(outline->tags[point+1] & FT_Curve_Tag_On))
|
|
|
|
{
|
|
|
|
cubic_control[3].x += outline->points[point].x + 1;
|
|
|
|
cubic_control[3].y += outline->points[point].y + 1;
|
|
|
|
cubic_control[3].x >>= 1;
|
|
|
|
cubic_control[3].y >>= 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* r1 = 1/3 p0 + 2/3 p1
|
|
|
|
r2 = 1/3 p2 + 2/3 p1 */
|
|
|
|
cubic_control[1].x = (2 * outline->points[point].x + 1) / 3;
|
|
|
|
cubic_control[1].y = (2 * outline->points[point].y + 1) / 3;
|
|
|
|
cubic_control[2] = cubic_control[1];
|
|
|
|
cubic_control[1].x += (cubic_control[0].x + 1) / 3;
|
|
|
|
cubic_control[1].y += (cubic_control[0].y + 1) / 3;
|
|
|
|
cubic_control[2].x += (cubic_control[3].x + 1) / 3;
|
|
|
|
cubic_control[2].y += (cubic_control[3].y + 1) / 3;
|
|
|
|
if (buf)
|
|
|
|
{
|
|
|
|
FTVectorToPOINTFX(&cubic_control[1], &ppc->apfx[cpfx]);
|
|
|
|
FTVectorToPOINTFX(&cubic_control[2], &ppc->apfx[cpfx+1]);
|
|
|
|
FTVectorToPOINTFX(&cubic_control[3], &ppc->apfx[cpfx+2]);
|
|
|
|
}
|
|
|
|
cpfx += 3;
|
|
|
|
point++;
|
|
|
|
}
|
|
|
|
} while (point <= outline->contours[contour] &&
|
|
|
|
(outline->tags[point] & FT_Curve_Tag_On) ==
|
|
|
|
(outline->tags[point-1] & FT_Curve_Tag_On));
|
|
|
|
/* At the end of a contour Windows adds the start point,
|
|
|
|
but only for Beziers and we've already done that.
|
|
|
|
*/
|
|
|
|
if (point <= outline->contours[contour] &&
|
|
|
|
outline->tags[point] & FT_Curve_Tag_On)
|
|
|
|
{
|
|
|
|
/* This is the closing pt of a bezier, but we've already
|
|
|
|
added it, so just inc point and carry on */
|
|
|
|
point++;
|
|
|
|
}
|
|
|
|
if (buf)
|
|
|
|
{
|
|
|
|
ppc->wType = type;
|
|
|
|
ppc->cpfx = cpfx;
|
|
|
|
}
|
|
|
|
needed += sizeof(*ppc) + (cpfx - 1) * sizeof(POINTFX);
|
|
|
|
}
|
|
|
|
if (buf)
|
|
|
|
pph->cb = needed - pph_start;
|
|
|
|
}
|
|
|
|
return needed;
|
|
|
|
}
|
|
|
|
|
2018-11-13 10:27:09 +01:00
|
|
|
static FT_Int get_load_flags( UINT format )
|
|
|
|
{
|
|
|
|
FT_Int load_flags = FT_LOAD_IGNORE_GLOBAL_ADVANCE_WIDTH;
|
|
|
|
|
|
|
|
if (format & GGO_UNHINTED)
|
|
|
|
return load_flags | FT_LOAD_NO_HINTING;
|
|
|
|
|
|
|
|
switch (format & ~GGO_GLYPH_INDEX)
|
|
|
|
{
|
|
|
|
case GGO_BITMAP:
|
|
|
|
load_flags |= FT_LOAD_TARGET_MONO;
|
|
|
|
break;
|
|
|
|
case GGO_GRAY2_BITMAP:
|
|
|
|
case GGO_GRAY4_BITMAP:
|
|
|
|
case GGO_GRAY8_BITMAP:
|
|
|
|
case WINE_GGO_GRAY16_BITMAP:
|
|
|
|
load_flags |= FT_LOAD_TARGET_NORMAL;
|
|
|
|
break;
|
|
|
|
case WINE_GGO_HRGB_BITMAP:
|
|
|
|
case WINE_GGO_HBGR_BITMAP:
|
|
|
|
load_flags |= FT_LOAD_TARGET_LCD;
|
|
|
|
break;
|
|
|
|
case WINE_GGO_VRGB_BITMAP:
|
|
|
|
case WINE_GGO_VBGR_BITMAP:
|
|
|
|
load_flags |= FT_LOAD_TARGET_LCD_V;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return load_flags;
|
|
|
|
}
|
|
|
|
|
2020-10-22 11:56:47 +02:00
|
|
|
/*************************************************************
|
|
|
|
* freetype_get_glyph_outline
|
|
|
|
*/
|
2020-10-29 14:47:18 +01:00
|
|
|
static DWORD CDECL freetype_get_glyph_outline( struct gdi_font *font, UINT glyph, UINT format,
|
2020-10-22 11:56:47 +02:00
|
|
|
GLYPHMETRICS *lpgm, ABC *abc, DWORD buflen, void *buf,
|
2020-10-29 14:47:18 +01:00
|
|
|
const MAT2 *lpmat, BOOL tategaki )
|
2001-09-12 22:21:06 +02:00
|
|
|
{
|
2020-10-29 14:47:18 +01:00
|
|
|
struct gdi_font *base_font = font->base_font ? font->base_font : font;
|
2020-10-29 14:53:18 +01:00
|
|
|
FT_Face ft_face = get_ft_face( base_font );
|
2013-04-30 09:20:57 +02:00
|
|
|
FT_Glyph_Metrics metrics;
|
2002-01-29 04:02:50 +01:00
|
|
|
FT_Error err;
|
2019-02-08 02:51:30 +01:00
|
|
|
FT_BBox bbox;
|
2018-11-13 10:27:09 +01:00
|
|
|
FT_Int load_flags = get_load_flags(format);
|
2019-02-07 15:43:38 +01:00
|
|
|
FT_Matrix matrices[3];
|
2003-10-10 02:06:35 +02:00
|
|
|
BOOL needsTransform = FALSE;
|
2013-05-24 18:00:41 +02:00
|
|
|
BOOL vertical_metrics;
|
2003-10-10 02:06:35 +02:00
|
|
|
|
2020-10-29 14:53:18 +01:00
|
|
|
TRACE("%p, %04x, %08x, %p, %08x, %p, %p\n", font, glyph, format, lpgm, buflen, buf, lpmat);
|
2001-09-12 22:21:06 +02:00
|
|
|
|
2008-06-24 09:10:27 +02:00
|
|
|
TRACE("font transform %f %f %f %f\n",
|
2020-10-29 14:47:18 +01:00
|
|
|
font->matrix.eM11, font->matrix.eM12,
|
|
|
|
font->matrix.eM21, font->matrix.eM22);
|
2002-01-29 04:02:50 +01:00
|
|
|
|
2020-10-29 14:47:18 +01:00
|
|
|
needsTransform = get_transform_matrices( font, tategaki, lpmat, matrices );
|
2003-10-10 02:06:35 +02:00
|
|
|
|
2013-05-24 18:00:41 +02:00
|
|
|
vertical_metrics = (tategaki && FT_HAS_VERTICAL(ft_face));
|
|
|
|
/* there is a freetype bug where vertical metrics are only
|
|
|
|
properly scaled and correct in 2.4.0 or greater */
|
2019-02-08 02:51:32 +01:00
|
|
|
if (vertical_metrics && FT_SimpleVersion < FT_VERSION_VALUE(2, 4, 0))
|
2013-05-24 18:00:41 +02:00
|
|
|
vertical_metrics = FALSE;
|
|
|
|
|
2012-11-15 13:29:21 +01:00
|
|
|
if (needsTransform || format != GGO_BITMAP) load_flags |= FT_LOAD_NO_BITMAP;
|
2013-05-24 18:00:41 +02:00
|
|
|
if (vertical_metrics) load_flags |= FT_LOAD_VERTICAL_LAYOUT;
|
2008-08-21 08:01:05 +02:00
|
|
|
|
2020-10-29 14:47:18 +01:00
|
|
|
err = pFT_Load_Glyph(ft_face, glyph, load_flags);
|
2020-02-05 11:17:35 +01:00
|
|
|
if (err && !(load_flags & FT_LOAD_NO_HINTING))
|
|
|
|
{
|
2020-10-29 14:47:18 +01:00
|
|
|
WARN("Failed to load glyph %#x, retrying without hinting. Error %#x.\n", glyph, err);
|
2020-02-05 11:17:35 +01:00
|
|
|
load_flags |= FT_LOAD_NO_HINTING;
|
2020-10-29 14:47:18 +01:00
|
|
|
err = pFT_Load_Glyph(ft_face, glyph, load_flags);
|
2020-02-05 11:17:35 +01:00
|
|
|
}
|
2008-08-21 08:01:05 +02:00
|
|
|
|
|
|
|
if(err) {
|
2020-10-29 14:47:18 +01:00
|
|
|
WARN("Failed to load glyph %#x, error %#x.\n", glyph, err);
|
2008-08-21 08:01:05 +02:00
|
|
|
return GDI_ERROR;
|
|
|
|
}
|
|
|
|
|
2013-10-17 14:25:10 +02:00
|
|
|
metrics = ft_face->glyph->metrics;
|
2020-10-29 14:47:18 +01:00
|
|
|
if(font->fake_bold) {
|
|
|
|
if (!get_bold_glyph_outline(ft_face->glyph, font->ppem, &metrics) && metrics.width)
|
2016-01-13 15:09:11 +01:00
|
|
|
metrics.width += 1 << 6;
|
|
|
|
}
|
2013-10-17 14:25:10 +02:00
|
|
|
|
2013-04-30 09:20:57 +02:00
|
|
|
/* Some poorly-created fonts contain glyphs that exceed the boundaries set
|
|
|
|
* by the text metrics. The proper behavior is to clip the glyph metrics to
|
|
|
|
* fit within the maximums specified in the text metrics. */
|
2020-10-29 14:47:18 +01:00
|
|
|
if (freetype_set_outline_text_metrics(base_font) ||
|
|
|
|
freetype_set_bitmap_text_metrics(base_font)) {
|
|
|
|
TEXTMETRICW *ptm = &base_font->otm.otmTextMetrics;
|
2019-02-08 02:51:30 +01:00
|
|
|
INT top = min( metrics.horiBearingY, ptm->tmAscent << 6 );
|
|
|
|
INT bottom = max( metrics.horiBearingY - metrics.height, -(ptm->tmDescent << 6) );
|
2013-04-30 09:20:57 +02:00
|
|
|
metrics.horiBearingY = top;
|
|
|
|
metrics.height = top - bottom;
|
|
|
|
|
|
|
|
/* TODO: Are we supposed to clip the width as well...? */
|
|
|
|
/* metrics.width = min( metrics.width, ptm->tmMaxCharWidth << 6 ); */
|
|
|
|
}
|
|
|
|
|
2019-02-08 02:51:30 +01:00
|
|
|
bbox = get_transformed_bbox( &metrics, needsTransform, matrices );
|
2020-10-29 14:47:18 +01:00
|
|
|
compute_metrics( base_font, font, bbox, &metrics, tategaki,
|
|
|
|
vertical_metrics, needsTransform, matrices, lpgm, abc );
|
2001-09-12 22:21:06 +02:00
|
|
|
|
2019-02-08 02:51:31 +01:00
|
|
|
switch (format)
|
|
|
|
{
|
2020-10-29 14:47:18 +01:00
|
|
|
case GGO_METRICS:
|
|
|
|
return 1; /* FIXME */
|
|
|
|
|
2002-01-29 04:02:50 +01:00
|
|
|
case GGO_BITMAP:
|
2020-10-29 14:47:18 +01:00
|
|
|
return get_mono_glyph_bitmap( ft_face->glyph, bbox, font->fake_bold,
|
|
|
|
needsTransform, matrices, buflen, buf );
|
2002-01-29 04:02:50 +01:00
|
|
|
|
|
|
|
case GGO_GRAY2_BITMAP:
|
|
|
|
case GGO_GRAY4_BITMAP:
|
|
|
|
case GGO_GRAY8_BITMAP:
|
|
|
|
case WINE_GGO_GRAY16_BITMAP:
|
2020-10-29 14:47:18 +01:00
|
|
|
return get_antialias_glyph_bitmap( ft_face->glyph, bbox, format, font->fake_bold,
|
|
|
|
needsTransform, matrices, buflen, buf );
|
2002-01-29 04:02:50 +01:00
|
|
|
|
2008-12-10 15:50:44 +01:00
|
|
|
case WINE_GGO_HRGB_BITMAP:
|
|
|
|
case WINE_GGO_HBGR_BITMAP:
|
|
|
|
case WINE_GGO_VRGB_BITMAP:
|
|
|
|
case WINE_GGO_VBGR_BITMAP:
|
2020-10-29 14:47:18 +01:00
|
|
|
return get_subpixel_glyph_bitmap( ft_face->glyph, bbox, format, font->fake_bold,
|
|
|
|
needsTransform, matrices, lpgm, buflen, buf );
|
2008-12-10 15:50:44 +01:00
|
|
|
|
2002-01-29 04:02:50 +01:00
|
|
|
case GGO_NATIVE:
|
2020-10-29 14:47:18 +01:00
|
|
|
if (ft_face->glyph->format == ft_glyph_format_outline)
|
|
|
|
{
|
|
|
|
FT_Outline *outline = &ft_face->glyph->outline;
|
|
|
|
UINT needed;
|
2002-01-29 04:02:50 +01:00
|
|
|
|
2020-10-29 14:47:18 +01:00
|
|
|
if (buflen == 0) buf = NULL;
|
2002-01-29 04:02:50 +01:00
|
|
|
|
2020-10-29 14:47:18 +01:00
|
|
|
if (needsTransform && buf)
|
|
|
|
pFT_Outline_Transform( outline, &matrices[matrix_vert] );
|
2003-06-23 22:51:06 +02:00
|
|
|
|
2020-10-29 14:47:18 +01:00
|
|
|
needed = get_native_glyph_outline(outline, buflen, NULL);
|
2012-11-13 16:51:29 +01:00
|
|
|
|
2020-10-29 14:47:18 +01:00
|
|
|
if (!buf || !buflen) return needed;
|
|
|
|
if (needed > buflen) return GDI_ERROR;
|
|
|
|
return get_native_glyph_outline(outline, buflen, buf);
|
|
|
|
}
|
|
|
|
TRACE("loaded a bitmap\n");
|
|
|
|
return GDI_ERROR;
|
2014-10-09 18:00:55 +02:00
|
|
|
|
2002-06-22 03:19:29 +02:00
|
|
|
case GGO_BEZIER:
|
2020-10-29 14:47:18 +01:00
|
|
|
if (ft_face->glyph->format == ft_glyph_format_outline)
|
|
|
|
{
|
|
|
|
FT_Outline *outline = &ft_face->glyph->outline;
|
|
|
|
UINT needed;
|
2002-06-22 03:19:29 +02:00
|
|
|
|
2020-10-29 14:47:18 +01:00
|
|
|
if (buflen == 0) buf = NULL;
|
2003-06-23 22:51:06 +02:00
|
|
|
|
2020-10-29 14:47:18 +01:00
|
|
|
if (needsTransform && buf)
|
|
|
|
pFT_Outline_Transform( outline, &matrices[matrix_vert] );
|
2014-10-09 18:00:55 +02:00
|
|
|
|
2020-10-29 14:47:18 +01:00
|
|
|
needed = get_bezier_glyph_outline(outline, buflen, NULL);
|
2014-10-09 18:00:55 +02:00
|
|
|
|
2020-10-29 14:47:18 +01:00
|
|
|
if (!buf || !buflen) return needed;
|
|
|
|
if (needed > buflen) return GDI_ERROR;
|
|
|
|
return get_bezier_glyph_outline(outline, buflen, buf);
|
|
|
|
}
|
|
|
|
TRACE("loaded a bitmap\n");
|
|
|
|
return GDI_ERROR;
|
2002-06-22 03:19:29 +02:00
|
|
|
|
2002-01-29 04:02:50 +01:00
|
|
|
default:
|
2001-09-12 22:21:06 +02:00
|
|
|
FIXME("Unsupported format %d\n", format);
|
|
|
|
return GDI_ERROR;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-26 11:43:24 +01:00
|
|
|
/*************************************************************
|
|
|
|
* freetype_set_bitmap_text_metrics
|
|
|
|
*/
|
|
|
|
static BOOL CDECL freetype_set_bitmap_text_metrics( struct gdi_font *font )
|
2004-06-16 22:06:26 +02:00
|
|
|
{
|
2020-10-29 14:53:18 +01:00
|
|
|
FT_Face ft_face = get_ft_face( font );
|
2004-06-16 22:06:26 +02:00
|
|
|
FT_WinFNT_HeaderRec winfnt_header;
|
|
|
|
|
2020-10-26 11:43:24 +01:00
|
|
|
if (font->otm.otmSize) return TRUE; /* already set */
|
|
|
|
font->otm.otmSize = offsetof( OUTLINETEXTMETRICW, otmFiller );
|
|
|
|
|
|
|
|
#define TM font->otm.otmTextMetrics
|
2011-10-11 11:14:56 +02:00
|
|
|
if(!pFT_Get_WinFNT_Header(ft_face, &winfnt_header))
|
2004-06-16 22:06:26 +02:00
|
|
|
{
|
|
|
|
TM.tmHeight = winfnt_header.pixel_height;
|
|
|
|
TM.tmAscent = winfnt_header.ascent;
|
|
|
|
TM.tmDescent = TM.tmHeight - TM.tmAscent;
|
|
|
|
TM.tmInternalLeading = winfnt_header.internal_leading;
|
|
|
|
TM.tmExternalLeading = winfnt_header.external_leading;
|
|
|
|
TM.tmAveCharWidth = winfnt_header.avg_width;
|
|
|
|
TM.tmMaxCharWidth = winfnt_header.max_width;
|
|
|
|
TM.tmWeight = winfnt_header.weight;
|
|
|
|
TM.tmOverhang = 0;
|
|
|
|
TM.tmDigitizedAspectX = winfnt_header.horizontal_resolution;
|
|
|
|
TM.tmDigitizedAspectY = winfnt_header.vertical_resolution;
|
|
|
|
TM.tmFirstChar = winfnt_header.first_char;
|
|
|
|
TM.tmLastChar = winfnt_header.last_char;
|
2004-08-09 20:52:51 +02:00
|
|
|
TM.tmDefaultChar = winfnt_header.default_char + winfnt_header.first_char;
|
|
|
|
TM.tmBreakChar = winfnt_header.break_char + winfnt_header.first_char;
|
2004-06-16 22:06:26 +02:00
|
|
|
TM.tmItalic = winfnt_header.italic;
|
|
|
|
TM.tmPitchAndFamily = winfnt_header.pitch_and_family;
|
|
|
|
TM.tmCharSet = winfnt_header.charset;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
TM.tmAscent = ft_face->size->metrics.ascender >> 6;
|
|
|
|
TM.tmDescent = -ft_face->size->metrics.descender >> 6;
|
|
|
|
TM.tmHeight = TM.tmAscent + TM.tmDescent;
|
|
|
|
TM.tmInternalLeading = TM.tmHeight - ft_face->size->metrics.y_ppem;
|
|
|
|
TM.tmExternalLeading = (ft_face->size->metrics.height >> 6) - TM.tmHeight;
|
|
|
|
TM.tmMaxCharWidth = ft_face->size->metrics.max_advance >> 6;
|
|
|
|
TM.tmAveCharWidth = TM.tmMaxCharWidth * 2 / 3; /* FIXME */
|
|
|
|
TM.tmWeight = ft_face->style_flags & FT_STYLE_FLAG_BOLD ? FW_BOLD : FW_NORMAL;
|
|
|
|
TM.tmOverhang = 0;
|
|
|
|
TM.tmDigitizedAspectX = 96; /* FIXME */
|
|
|
|
TM.tmDigitizedAspectY = 96; /* FIXME */
|
|
|
|
TM.tmFirstChar = 1;
|
|
|
|
TM.tmLastChar = 255;
|
|
|
|
TM.tmDefaultChar = 32;
|
|
|
|
TM.tmBreakChar = 32;
|
|
|
|
TM.tmItalic = ft_face->style_flags & FT_STYLE_FLAG_ITALIC ? 1 : 0;
|
|
|
|
/* NB inverted meaning of TMPF_FIXED_PITCH */
|
2016-03-01 05:49:52 +01:00
|
|
|
TM.tmPitchAndFamily = FT_IS_FIXED_WIDTH(ft_face) ? 0 : TMPF_FIXED_PITCH;
|
2020-10-26 11:43:24 +01:00
|
|
|
TM.tmCharSet = font->charset;
|
2004-06-16 22:06:26 +02:00
|
|
|
}
|
2020-10-26 11:43:24 +01:00
|
|
|
TM.tmUnderlined = font->lf.lfUnderline ? 0xff : 0;
|
|
|
|
TM.tmStruckOut = font->lf.lfStrikeOut ? 0xff : 0;
|
2018-02-18 13:34:42 +01:00
|
|
|
|
2020-10-26 11:43:24 +01:00
|
|
|
if(font->fake_bold)
|
2018-02-18 13:34:42 +01:00
|
|
|
TM.tmWeight = FW_BOLD;
|
2004-06-16 22:06:26 +02:00
|
|
|
#undef TM
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2008-01-08 16:02:09 +01:00
|
|
|
|
2009-02-18 11:33:12 +01:00
|
|
|
static BOOL face_has_symbol_charmap(FT_Face ft_face)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for(i = 0; i < ft_face->num_charmaps; i++)
|
|
|
|
{
|
|
|
|
if(ft_face->charmaps[i]->encoding == FT_ENCODING_MS_SYMBOL)
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
2002-06-22 03:19:29 +02:00
|
|
|
|
2020-10-26 11:43:24 +01:00
|
|
|
/*************************************************************
|
|
|
|
* freetype_set_outline_text_metrics
|
|
|
|
*/
|
|
|
|
static BOOL CDECL freetype_set_outline_text_metrics( struct gdi_font *font )
|
2001-09-12 22:21:06 +02:00
|
|
|
{
|
2020-10-29 14:53:18 +01:00
|
|
|
FT_Face ft_face = get_ft_face( font );
|
2020-10-26 11:43:24 +01:00
|
|
|
UINT needed;
|
2001-09-12 22:21:06 +02:00
|
|
|
TT_OS2 *pOS2;
|
|
|
|
TT_HoriHeader *pHori;
|
2002-11-14 00:54:50 +01:00
|
|
|
TT_Postscript *pPost;
|
2013-03-14 12:08:14 +01:00
|
|
|
FT_Fixed em_scale;
|
2003-12-08 22:53:15 +01:00
|
|
|
INT ascent, descent;
|
2015-09-27 21:52:56 +02:00
|
|
|
USHORT windescent;
|
2001-09-12 22:21:06 +02:00
|
|
|
|
2002-01-29 04:02:50 +01:00
|
|
|
TRACE("font=%p\n", font);
|
|
|
|
|
2020-10-26 11:43:24 +01:00
|
|
|
if (!font->scalable) return FALSE;
|
|
|
|
if (font->otm.otmSize) return TRUE; /* already set */
|
2001-09-12 22:21:06 +02:00
|
|
|
|
2020-10-26 11:43:24 +01:00
|
|
|
/* note: we store actual pointers in the names instead of offsets,
|
|
|
|
they are fixed up when returned to the app */
|
|
|
|
if (!(font->otm.otmpFullName = (char *)get_face_name( ft_face, TT_NAME_ID_UNIQUE_ID,
|
|
|
|
GetSystemDefaultLangID() )))
|
2012-09-07 12:40:34 +02:00
|
|
|
{
|
2018-11-30 08:14:04 +01:00
|
|
|
static const WCHAR fake_nameW[] = {'f','a','k','e',' ','n','a','m','e', 0};
|
2020-10-26 11:43:24 +01:00
|
|
|
FIXME("failed to read full_nameW for font %s!\n", wine_dbgstr_w(get_gdi_font_name(font)));
|
|
|
|
font->otm.otmpFullName = (char *)strdupW(fake_nameW);
|
2012-09-07 12:40:34 +02:00
|
|
|
}
|
2020-10-26 11:43:24 +01:00
|
|
|
needed = sizeof(font->otm) + (strlenW( (WCHAR *)font->otm.otmpFamilyName ) + 1 +
|
|
|
|
strlenW( (WCHAR *)font->otm.otmpStyleName ) + 1 +
|
|
|
|
strlenW( (WCHAR *)font->otm.otmpFaceName ) + 1 +
|
|
|
|
strlenW( (WCHAR *)font->otm.otmpFullName ) + 1) * sizeof(WCHAR);
|
2001-09-12 22:21:06 +02:00
|
|
|
|
2020-10-26 11:43:24 +01:00
|
|
|
em_scale = (FT_Fixed)MulDiv(font->ppem, 1 << 16, ft_face->units_per_EM);
|
2001-09-12 22:21:06 +02:00
|
|
|
|
2002-04-03 22:02:39 +02:00
|
|
|
pOS2 = pFT_Get_Sfnt_Table(ft_face, ft_sfnt_os2);
|
2001-09-12 22:21:06 +02:00
|
|
|
if(!pOS2) {
|
|
|
|
FIXME("Can't find OS/2 table - not TT font?\n");
|
2020-10-26 11:43:24 +01:00
|
|
|
return FALSE;
|
2001-09-12 22:21:06 +02:00
|
|
|
}
|
|
|
|
|
2002-04-03 22:02:39 +02:00
|
|
|
pHori = pFT_Get_Sfnt_Table(ft_face, ft_sfnt_hhea);
|
2001-09-12 22:21:06 +02:00
|
|
|
if(!pHori) {
|
|
|
|
FIXME("Can't find HHEA table - not TT font?\n");
|
2020-10-26 11:43:24 +01:00
|
|
|
return FALSE;
|
2001-09-12 22:21:06 +02:00
|
|
|
}
|
|
|
|
|
2002-11-14 00:54:50 +01:00
|
|
|
pPost = pFT_Get_Sfnt_Table(ft_face, ft_sfnt_post); /* we can live with this failing */
|
|
|
|
|
2015-09-27 21:52:56 +02:00
|
|
|
TRACE("OS/2 winA = %u winD = %u typoA = %d typoD = %d typoLG = %d avgW %d FT_Face a = %d, d = %d, h = %d: HORZ a = %d, d = %d lg = %d maxY = %ld minY = %ld\n",
|
2002-06-22 03:19:29 +02:00
|
|
|
pOS2->usWinAscent, pOS2->usWinDescent,
|
|
|
|
pOS2->sTypoAscender, pOS2->sTypoDescender, pOS2->sTypoLineGap,
|
2012-04-24 10:03:35 +02:00
|
|
|
pOS2->xAvgCharWidth,
|
2002-06-22 03:19:29 +02:00
|
|
|
ft_face->ascender, ft_face->descender, ft_face->height,
|
|
|
|
pHori->Ascender, pHori->Descender, pHori->Line_Gap,
|
|
|
|
ft_face->bbox.yMax, ft_face->bbox.yMin);
|
|
|
|
|
2020-10-26 11:43:24 +01:00
|
|
|
font->otm.otmSize = needed;
|
2002-06-22 03:19:29 +02:00
|
|
|
|
2020-10-26 11:43:24 +01:00
|
|
|
#define TM font->otm.otmTextMetrics
|
2002-06-22 03:19:29 +02:00
|
|
|
|
2015-09-27 21:52:56 +02:00
|
|
|
windescent = get_fixed_windescent(pOS2->usWinDescent);
|
|
|
|
if(pOS2->usWinAscent + windescent == 0) {
|
2003-12-08 22:53:15 +01:00
|
|
|
ascent = pHori->Ascender;
|
|
|
|
descent = -pHori->Descender;
|
|
|
|
} else {
|
|
|
|
ascent = pOS2->usWinAscent;
|
2015-09-27 21:52:56 +02:00
|
|
|
descent = windescent;
|
2003-12-08 22:53:15 +01:00
|
|
|
}
|
|
|
|
|
2020-10-26 11:43:24 +01:00
|
|
|
font->ntmCellHeight = ascent + descent;
|
|
|
|
font->ntmAvgWidth = pOS2->xAvgCharWidth;
|
2012-04-24 10:03:35 +02:00
|
|
|
|
2013-03-14 12:08:14 +01:00
|
|
|
#define SCALE_X(x) (pFT_MulFix(x, em_scale))
|
|
|
|
#define SCALE_Y(y) (pFT_MulFix(y, em_scale))
|
2013-03-14 12:08:12 +01:00
|
|
|
|
2020-10-26 11:43:24 +01:00
|
|
|
if(font->yMax) {
|
|
|
|
TM.tmAscent = font->yMax;
|
|
|
|
TM.tmDescent = -font->yMin;
|
2002-06-22 03:19:29 +02:00
|
|
|
TM.tmInternalLeading = (TM.tmAscent + TM.tmDescent) - ft_face->size->metrics.y_ppem;
|
|
|
|
} else {
|
2013-03-14 12:08:12 +01:00
|
|
|
TM.tmAscent = SCALE_Y(ascent);
|
|
|
|
TM.tmDescent = SCALE_Y(descent);
|
|
|
|
TM.tmInternalLeading = SCALE_Y(ascent + descent - ft_face->units_per_EM);
|
2002-06-22 03:19:29 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
TM.tmHeight = TM.tmAscent + TM.tmDescent;
|
|
|
|
|
|
|
|
/* MSDN says:
|
|
|
|
el = MAX(0, LineGap - ((WinAscent + WinDescent) - (Ascender - Descender)))
|
|
|
|
*/
|
2013-03-14 12:08:12 +01:00
|
|
|
TM.tmExternalLeading = max(0, SCALE_Y(pHori->Line_Gap -
|
|
|
|
((ascent + descent) -
|
|
|
|
(pHori->Ascender - pHori->Descender))));
|
2002-06-22 03:19:29 +02:00
|
|
|
|
2013-03-14 12:08:12 +01:00
|
|
|
TM.tmAveCharWidth = SCALE_X(pOS2->xAvgCharWidth);
|
2003-10-10 02:06:35 +02:00
|
|
|
if (TM.tmAveCharWidth == 0) {
|
|
|
|
TM.tmAveCharWidth = 1;
|
|
|
|
}
|
2013-03-14 12:08:12 +01:00
|
|
|
TM.tmMaxCharWidth = SCALE_X(ft_face->bbox.xMax - ft_face->bbox.xMin);
|
2009-07-31 15:21:51 +02:00
|
|
|
TM.tmWeight = FW_REGULAR;
|
2020-10-26 11:43:24 +01:00
|
|
|
if (font->fake_bold)
|
2009-07-31 15:21:51 +02:00
|
|
|
TM.tmWeight = FW_BOLD;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (ft_face->style_flags & FT_STYLE_FLAG_BOLD)
|
|
|
|
{
|
|
|
|
if (pOS2->usWeightClass > FW_MEDIUM)
|
|
|
|
TM.tmWeight = pOS2->usWeightClass;
|
|
|
|
}
|
|
|
|
else if (pOS2->usWeightClass <= FW_MEDIUM)
|
|
|
|
TM.tmWeight = pOS2->usWeightClass;
|
|
|
|
}
|
2002-06-22 03:19:29 +02:00
|
|
|
TM.tmOverhang = 0;
|
2012-05-08 12:29:24 +02:00
|
|
|
TM.tmDigitizedAspectX = 96; /* FIXME */
|
|
|
|
TM.tmDigitizedAspectY = 96; /* FIXME */
|
2007-03-15 07:12:09 +01:00
|
|
|
/* It appears that for fonts with SYMBOL_CHARSET Windows always sets
|
|
|
|
* symbol range to 0 - f0ff
|
|
|
|
*/
|
2009-02-18 11:33:12 +01:00
|
|
|
|
|
|
|
if (face_has_symbol_charmap(ft_face) || (pOS2->usFirstCharIndex >= 0xf000 && pOS2->usFirstCharIndex < 0xf100))
|
2008-06-18 05:35:58 +02:00
|
|
|
{
|
2007-03-15 07:12:09 +01:00
|
|
|
TM.tmFirstChar = 0;
|
2009-02-17 12:49:02 +01:00
|
|
|
switch(GetACP())
|
|
|
|
{
|
2014-05-07 15:51:11 +02:00
|
|
|
case 1255: /* Hebrew */
|
|
|
|
TM.tmLastChar = 0xf896;
|
|
|
|
break;
|
2009-02-17 12:49:02 +01:00
|
|
|
case 1257: /* Baltic */
|
|
|
|
TM.tmLastChar = 0xf8fd;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
TM.tmLastChar = 0xf0ff;
|
|
|
|
}
|
2009-02-16 13:39:15 +01:00
|
|
|
TM.tmBreakChar = 0x20;
|
|
|
|
TM.tmDefaultChar = 0x1f;
|
2008-06-18 05:35:58 +02:00
|
|
|
}
|
2007-03-15 07:12:09 +01:00
|
|
|
else
|
2008-06-18 05:35:58 +02:00
|
|
|
{
|
2009-02-16 13:39:15 +01:00
|
|
|
TM.tmFirstChar = pOS2->usFirstCharIndex; /* Should be the first char in the cmap */
|
|
|
|
TM.tmLastChar = pOS2->usLastCharIndex; /* Should be min(cmap_last, os2_last) */
|
|
|
|
|
|
|
|
if(pOS2->usFirstCharIndex <= 1)
|
|
|
|
TM.tmBreakChar = pOS2->usFirstCharIndex + 2;
|
|
|
|
else if (pOS2->usFirstCharIndex > 0xff)
|
|
|
|
TM.tmBreakChar = 0x20;
|
|
|
|
else
|
|
|
|
TM.tmBreakChar = pOS2->usFirstCharIndex;
|
|
|
|
TM.tmDefaultChar = TM.tmBreakChar - 1;
|
2008-06-18 05:35:58 +02:00
|
|
|
}
|
2020-10-26 11:43:24 +01:00
|
|
|
TM.tmItalic = font->fake_italic ? 255 : ((ft_face->style_flags & FT_STYLE_FLAG_ITALIC) ? 255 : 0);
|
|
|
|
TM.tmUnderlined = font->lf.lfUnderline ? 255 : 0;
|
|
|
|
TM.tmStruckOut = font->lf.lfStrikeOut ? 255 : 0;
|
2002-06-22 03:19:29 +02:00
|
|
|
|
|
|
|
/* Yes TPMF_FIXED_PITCH is correct; braindead api */
|
2005-02-25 14:59:22 +01:00
|
|
|
if(!FT_IS_FIXED_WIDTH(ft_face) &&
|
|
|
|
(pOS2->version == 0xFFFFU ||
|
|
|
|
pOS2->panose[PAN_PROPORTION_INDEX] != PAN_PROP_MONOSPACED))
|
2002-06-22 03:19:29 +02:00
|
|
|
TM.tmPitchAndFamily = TMPF_FIXED_PITCH;
|
|
|
|
else
|
|
|
|
TM.tmPitchAndFamily = 0;
|
|
|
|
|
2009-02-11 14:49:52 +01:00
|
|
|
switch(pOS2->panose[PAN_FAMILYTYPE_INDEX])
|
|
|
|
{
|
2002-06-22 03:19:29 +02:00
|
|
|
case PAN_FAMILY_SCRIPT:
|
|
|
|
TM.tmPitchAndFamily |= FF_SCRIPT;
|
2009-02-11 14:49:52 +01:00
|
|
|
break;
|
|
|
|
|
2002-06-22 03:19:29 +02:00
|
|
|
case PAN_FAMILY_DECORATIVE:
|
|
|
|
TM.tmPitchAndFamily |= FF_DECORATIVE;
|
2009-02-11 14:49:52 +01:00
|
|
|
break;
|
|
|
|
|
|
|
|
case PAN_ANY:
|
|
|
|
case PAN_NO_FIT:
|
2002-06-22 03:19:29 +02:00
|
|
|
case PAN_FAMILY_TEXT_DISPLAY:
|
2009-02-11 14:49:52 +01:00
|
|
|
case PAN_FAMILY_PICTORIAL: /* symbol fonts get treated as if they were text */
|
|
|
|
/* which is clearly not what the panose spec says. */
|
|
|
|
default:
|
2009-02-19 11:41:34 +01:00
|
|
|
if(TM.tmPitchAndFamily == 0 || /* fixed */
|
|
|
|
pOS2->panose[PAN_PROPORTION_INDEX] == PAN_PROP_MONOSPACED)
|
2002-06-22 03:19:29 +02:00
|
|
|
TM.tmPitchAndFamily = FF_MODERN;
|
2009-02-11 14:49:52 +01:00
|
|
|
else
|
|
|
|
{
|
|
|
|
switch(pOS2->panose[PAN_SERIFSTYLE_INDEX])
|
|
|
|
{
|
|
|
|
case PAN_ANY:
|
|
|
|
case PAN_NO_FIT:
|
|
|
|
default:
|
|
|
|
TM.tmPitchAndFamily |= FF_DONTCARE;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PAN_SERIF_COVE:
|
|
|
|
case PAN_SERIF_OBTUSE_COVE:
|
|
|
|
case PAN_SERIF_SQUARE_COVE:
|
|
|
|
case PAN_SERIF_OBTUSE_SQUARE_COVE:
|
|
|
|
case PAN_SERIF_SQUARE:
|
|
|
|
case PAN_SERIF_THIN:
|
|
|
|
case PAN_SERIF_BONE:
|
|
|
|
case PAN_SERIF_EXAGGERATED:
|
|
|
|
case PAN_SERIF_TRIANGLE:
|
|
|
|
TM.tmPitchAndFamily |= FF_ROMAN;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PAN_SERIF_NORMAL_SANS:
|
|
|
|
case PAN_SERIF_OBTUSE_SANS:
|
|
|
|
case PAN_SERIF_PERP_SANS:
|
2009-02-12 13:23:01 +01:00
|
|
|
case PAN_SERIF_FLARED:
|
|
|
|
case PAN_SERIF_ROUNDED:
|
2009-02-11 14:49:52 +01:00
|
|
|
TM.tmPitchAndFamily |= FF_SWISS;
|
|
|
|
break;
|
|
|
|
}
|
2002-06-22 03:19:29 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(FT_IS_SCALABLE(ft_face))
|
|
|
|
TM.tmPitchAndFamily |= TMPF_VECTOR;
|
2007-09-07 23:30:40 +02:00
|
|
|
|
2002-06-22 03:19:29 +02:00
|
|
|
if(FT_IS_SFNT(ft_face))
|
2007-09-07 23:30:40 +02:00
|
|
|
{
|
2020-10-26 11:43:24 +01:00
|
|
|
if (font->ntmFlags & NTM_PS_OPENTYPE)
|
2007-09-07 23:30:40 +02:00
|
|
|
TM.tmPitchAndFamily |= TMPF_DEVICE;
|
|
|
|
else
|
|
|
|
TM.tmPitchAndFamily |= TMPF_TRUETYPE;
|
|
|
|
}
|
2002-06-22 03:19:29 +02:00
|
|
|
|
2020-10-26 11:43:24 +01:00
|
|
|
TM.tmCharSet = font->charset;
|
2002-06-22 03:19:29 +02:00
|
|
|
|
2020-10-26 11:43:24 +01:00
|
|
|
font->otm.otmFiller = 0;
|
|
|
|
memcpy(&font->otm.otmPanoseNumber, pOS2->panose, PANOSE_COUNT);
|
|
|
|
font->otm.otmfsSelection = pOS2->fsSelection;
|
|
|
|
if (font->fake_italic)
|
|
|
|
font->otm.otmfsSelection |= 1;
|
|
|
|
if (font->fake_bold)
|
|
|
|
font->otm.otmfsSelection |= 1 << 5;
|
2017-03-05 11:44:59 +01:00
|
|
|
/* Only return valid bits that define embedding and subsetting restrictions */
|
2020-10-26 11:43:24 +01:00
|
|
|
font->otm.otmfsType = pOS2->fsType & 0x30e;
|
|
|
|
font->otm.otmsCharSlopeRise = pHori->caret_Slope_Rise;
|
|
|
|
font->otm.otmsCharSlopeRun = pHori->caret_Slope_Run;
|
|
|
|
font->otm.otmItalicAngle = 0; /* POST table */
|
|
|
|
font->otm.otmEMSquare = ft_face->units_per_EM;
|
|
|
|
font->otm.otmAscent = SCALE_Y(pOS2->sTypoAscender);
|
|
|
|
font->otm.otmDescent = SCALE_Y(pOS2->sTypoDescender);
|
|
|
|
font->otm.otmLineGap = SCALE_Y(pOS2->sTypoLineGap);
|
|
|
|
font->otm.otmsCapEmHeight = SCALE_Y(pOS2->sCapHeight);
|
|
|
|
font->otm.otmsXHeight = SCALE_Y(pOS2->sxHeight);
|
|
|
|
font->otm.otmrcFontBox.left = SCALE_X(ft_face->bbox.xMin);
|
|
|
|
font->otm.otmrcFontBox.right = SCALE_X(ft_face->bbox.xMax);
|
|
|
|
font->otm.otmrcFontBox.top = SCALE_Y(ft_face->bbox.yMax);
|
|
|
|
font->otm.otmrcFontBox.bottom = SCALE_Y(ft_face->bbox.yMin);
|
|
|
|
font->otm.otmMacAscent = TM.tmAscent;
|
|
|
|
font->otm.otmMacDescent = -TM.tmDescent;
|
|
|
|
font->otm.otmMacLineGap = SCALE_Y(pHori->Line_Gap);
|
|
|
|
font->otm.otmusMinimumPPEM = 0; /* TT Header */
|
|
|
|
font->otm.otmptSubscriptSize.x = SCALE_X(pOS2->ySubscriptXSize);
|
|
|
|
font->otm.otmptSubscriptSize.y = SCALE_Y(pOS2->ySubscriptYSize);
|
|
|
|
font->otm.otmptSubscriptOffset.x = SCALE_X(pOS2->ySubscriptXOffset);
|
|
|
|
font->otm.otmptSubscriptOffset.y = SCALE_Y(pOS2->ySubscriptYOffset);
|
|
|
|
font->otm.otmptSuperscriptSize.x = SCALE_X(pOS2->ySuperscriptXSize);
|
|
|
|
font->otm.otmptSuperscriptSize.y = SCALE_Y(pOS2->ySuperscriptYSize);
|
|
|
|
font->otm.otmptSuperscriptOffset.x = SCALE_X(pOS2->ySuperscriptXOffset);
|
|
|
|
font->otm.otmptSuperscriptOffset.y = SCALE_Y(pOS2->ySuperscriptYOffset);
|
|
|
|
font->otm.otmsStrikeoutSize = SCALE_Y(pOS2->yStrikeoutSize);
|
|
|
|
font->otm.otmsStrikeoutPosition = SCALE_Y(pOS2->yStrikeoutPosition);
|
2002-11-14 00:54:50 +01:00
|
|
|
if(!pPost) {
|
2020-10-26 11:43:24 +01:00
|
|
|
font->otm.otmsUnderscoreSize = 0;
|
|
|
|
font->otm.otmsUnderscorePosition = 0;
|
2002-11-14 00:54:50 +01:00
|
|
|
} else {
|
2020-10-26 11:43:24 +01:00
|
|
|
font->otm.otmsUnderscoreSize = SCALE_Y(pPost->underlineThickness);
|
|
|
|
font->otm.otmsUnderscorePosition = SCALE_Y(pPost->underlinePosition);
|
2002-11-14 00:54:50 +01:00
|
|
|
}
|
2013-03-14 12:08:12 +01:00
|
|
|
#undef SCALE_X
|
|
|
|
#undef SCALE_Y
|
2008-06-24 09:10:47 +02:00
|
|
|
#undef TM
|
2020-10-26 11:43:24 +01:00
|
|
|
return TRUE;
|
2001-09-12 22:21:06 +02:00
|
|
|
}
|
|
|
|
|
2019-04-09 11:32:59 +02:00
|
|
|
/*************************************************************
|
2020-10-28 10:14:01 +01:00
|
|
|
* freetype_get_char_width_info
|
2019-04-09 11:32:59 +02:00
|
|
|
*/
|
2020-10-28 10:14:01 +01:00
|
|
|
static BOOL CDECL freetype_get_char_width_info( struct gdi_font *font, struct char_width_info *info )
|
2019-04-09 11:32:59 +02:00
|
|
|
{
|
2020-10-29 14:53:18 +01:00
|
|
|
FT_Face ft_face = get_ft_face( font );
|
2019-04-09 11:32:59 +02:00
|
|
|
TT_HoriHeader *pHori;
|
|
|
|
|
2020-10-20 22:05:37 +02:00
|
|
|
TRACE("%p, %p\n", font, info);
|
2019-04-09 11:32:59 +02:00
|
|
|
|
2020-10-29 14:53:18 +01:00
|
|
|
if ((pHori = pFT_Get_Sfnt_Table(ft_face, ft_sfnt_hhea)))
|
2019-04-09 11:32:59 +02:00
|
|
|
{
|
2020-10-29 14:53:18 +01:00
|
|
|
FT_Fixed em_scale = MulDiv(font->ppem, 1 << 16, ft_face->units_per_EM);
|
2019-04-09 11:32:59 +02:00
|
|
|
info->lsb = (SHORT)pFT_MulFix(pHori->min_Left_Side_Bearing, em_scale);
|
|
|
|
info->rsb = (SHORT)pFT_MulFix(pHori->min_Right_Side_Bearing, em_scale);
|
2020-10-28 10:14:01 +01:00
|
|
|
return TRUE;
|
2019-04-09 11:32:59 +02:00
|
|
|
}
|
2020-10-28 10:14:01 +01:00
|
|
|
return FALSE;
|
2019-04-09 11:32:59 +02:00
|
|
|
}
|
|
|
|
|
2001-10-23 22:06:32 +02:00
|
|
|
|
2020-10-28 10:13:53 +01:00
|
|
|
/*************************************************************
|
|
|
|
* freetype_get_unicode_ranges
|
|
|
|
*
|
|
|
|
* Retrieve a list of supported Unicode ranges for a given font.
|
2007-03-01 20:30:12 +01:00
|
|
|
* Can be called with NULL gs to calculate the buffer size. Returns
|
|
|
|
* the number of ranges found.
|
|
|
|
*/
|
2020-10-28 10:13:53 +01:00
|
|
|
static DWORD CDECL freetype_get_unicode_ranges( struct gdi_font *font, GLYPHSET *gs )
|
2007-03-01 20:30:12 +01:00
|
|
|
{
|
2020-10-29 14:53:18 +01:00
|
|
|
FT_Face ft_face = get_ft_face( font );
|
2007-03-01 20:30:12 +01:00
|
|
|
DWORD num_ranges = 0;
|
|
|
|
|
2020-10-29 14:53:18 +01:00
|
|
|
if (ft_face->charmap->encoding == FT_ENCODING_UNICODE)
|
2007-03-01 20:30:12 +01:00
|
|
|
{
|
|
|
|
FT_UInt glyph_code;
|
|
|
|
FT_ULong char_code, char_code_prev;
|
|
|
|
|
|
|
|
glyph_code = 0;
|
2020-10-29 14:53:18 +01:00
|
|
|
char_code_prev = char_code = pFT_Get_First_Char(ft_face, &glyph_code);
|
2007-03-01 20:30:12 +01:00
|
|
|
|
|
|
|
TRACE("face encoding FT_ENCODING_UNICODE, number of glyphs %ld, first glyph %u, first char %04lx\n",
|
2020-10-29 14:53:18 +01:00
|
|
|
ft_face->num_glyphs, glyph_code, char_code);
|
2007-03-01 20:30:12 +01:00
|
|
|
|
|
|
|
if (!glyph_code) return 0;
|
|
|
|
|
|
|
|
if (gs)
|
|
|
|
{
|
|
|
|
gs->ranges[0].wcLow = (USHORT)char_code;
|
|
|
|
gs->ranges[0].cGlyphs = 0;
|
|
|
|
gs->cGlyphsSupported = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
num_ranges = 1;
|
|
|
|
while (glyph_code)
|
|
|
|
{
|
|
|
|
if (char_code < char_code_prev)
|
|
|
|
{
|
|
|
|
ERR("expected increasing char code from FT_Get_Next_Char\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
if (char_code - char_code_prev > 1)
|
|
|
|
{
|
|
|
|
num_ranges++;
|
|
|
|
if (gs)
|
|
|
|
{
|
|
|
|
gs->ranges[num_ranges - 1].wcLow = (USHORT)char_code;
|
|
|
|
gs->ranges[num_ranges - 1].cGlyphs = 1;
|
|
|
|
gs->cGlyphsSupported++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (gs)
|
|
|
|
{
|
|
|
|
gs->ranges[num_ranges - 1].cGlyphs++;
|
|
|
|
gs->cGlyphsSupported++;
|
|
|
|
}
|
|
|
|
char_code_prev = char_code;
|
2020-10-29 14:53:18 +01:00
|
|
|
char_code = pFT_Get_Next_Char(ft_face, char_code, &glyph_code);
|
2007-03-01 20:30:12 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
2017-04-06 13:07:05 +02:00
|
|
|
{
|
2020-10-29 14:53:18 +01:00
|
|
|
DWORD encoding = RtlUlongByteSwap(ft_face->charmap->encoding);
|
2017-04-06 13:07:05 +02:00
|
|
|
FIXME("encoding %s not supported\n", debugstr_an((char *)&encoding, 4));
|
|
|
|
}
|
2007-03-01 20:30:12 +01:00
|
|
|
|
|
|
|
return num_ranges;
|
|
|
|
}
|
|
|
|
|
2006-09-20 17:18:51 +02:00
|
|
|
/*************************************************************************
|
|
|
|
* Kerning support for TrueType fonts
|
|
|
|
*/
|
|
|
|
|
|
|
|
struct TT_kern_table
|
|
|
|
{
|
|
|
|
USHORT version;
|
|
|
|
USHORT nTables;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct TT_kern_subtable
|
|
|
|
{
|
|
|
|
USHORT version;
|
|
|
|
USHORT length;
|
|
|
|
union
|
|
|
|
{
|
|
|
|
USHORT word;
|
|
|
|
struct
|
|
|
|
{
|
|
|
|
USHORT horizontal : 1;
|
|
|
|
USHORT minimum : 1;
|
|
|
|
USHORT cross_stream: 1;
|
|
|
|
USHORT override : 1;
|
|
|
|
USHORT reserved1 : 4;
|
|
|
|
USHORT format : 8;
|
|
|
|
} bits;
|
|
|
|
} coverage;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct TT_format0_kern_subtable
|
|
|
|
{
|
|
|
|
USHORT nPairs;
|
|
|
|
USHORT searchRange;
|
|
|
|
USHORT entrySelector;
|
|
|
|
USHORT rangeShift;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct TT_kern_pair
|
|
|
|
{
|
|
|
|
USHORT left;
|
|
|
|
USHORT right;
|
|
|
|
short value;
|
|
|
|
};
|
|
|
|
|
2020-10-29 14:53:18 +01:00
|
|
|
static DWORD parse_format0_kern_subtable(struct gdi_font *font,
|
2006-09-20 17:18:51 +02:00
|
|
|
const struct TT_format0_kern_subtable *tt_f0_ks,
|
|
|
|
const USHORT *glyph_to_char,
|
|
|
|
KERNINGPAIR *kern_pair, DWORD cPairs)
|
|
|
|
{
|
2020-10-29 14:53:18 +01:00
|
|
|
FT_Face ft_face = get_ft_face( font );
|
2006-09-20 17:18:51 +02:00
|
|
|
USHORT i, nPairs;
|
|
|
|
const struct TT_kern_pair *tt_kern_pair;
|
|
|
|
|
2020-10-29 14:53:18 +01:00
|
|
|
TRACE("font height %d, units_per_EM %d\n", font->ppem, ft_face->units_per_EM);
|
2006-09-20 17:18:51 +02:00
|
|
|
|
|
|
|
nPairs = GET_BE_WORD(tt_f0_ks->nPairs);
|
|
|
|
|
|
|
|
TRACE("nPairs %u, searchRange %u, entrySelector %u, rangeShift %u\n",
|
|
|
|
nPairs, GET_BE_WORD(tt_f0_ks->searchRange),
|
|
|
|
GET_BE_WORD(tt_f0_ks->entrySelector), GET_BE_WORD(tt_f0_ks->rangeShift));
|
|
|
|
|
|
|
|
if (!kern_pair || !cPairs)
|
|
|
|
return nPairs;
|
|
|
|
|
|
|
|
tt_kern_pair = (const struct TT_kern_pair *)(tt_f0_ks + 1);
|
|
|
|
|
|
|
|
nPairs = min(nPairs, cPairs);
|
|
|
|
|
|
|
|
for (i = 0; i < nPairs; i++)
|
|
|
|
{
|
|
|
|
kern_pair->wFirst = glyph_to_char[GET_BE_WORD(tt_kern_pair[i].left)];
|
|
|
|
kern_pair->wSecond = glyph_to_char[GET_BE_WORD(tt_kern_pair[i].right)];
|
2006-09-27 16:43:23 +02:00
|
|
|
/* this algorithm appears to better match what Windows does */
|
2020-10-29 14:53:18 +01:00
|
|
|
kern_pair->iKernAmount = (short)GET_BE_WORD(tt_kern_pair[i].value) * font->ppem;
|
2006-09-27 16:43:23 +02:00
|
|
|
if (kern_pair->iKernAmount < 0)
|
|
|
|
{
|
2020-10-29 14:53:18 +01:00
|
|
|
kern_pair->iKernAmount -= ft_face->units_per_EM / 2;
|
|
|
|
kern_pair->iKernAmount -= font->ppem;
|
2006-09-27 16:43:23 +02:00
|
|
|
}
|
|
|
|
else if (kern_pair->iKernAmount > 0)
|
|
|
|
{
|
2020-10-29 14:53:18 +01:00
|
|
|
kern_pair->iKernAmount += ft_face->units_per_EM / 2;
|
|
|
|
kern_pair->iKernAmount += font->ppem;
|
2006-09-27 16:43:23 +02:00
|
|
|
}
|
2020-10-29 14:53:18 +01:00
|
|
|
kern_pair->iKernAmount /= ft_face->units_per_EM;
|
2006-09-20 17:18:51 +02:00
|
|
|
|
|
|
|
TRACE("left %u right %u value %d\n",
|
|
|
|
kern_pair->wFirst, kern_pair->wSecond, kern_pair->iKernAmount);
|
|
|
|
|
|
|
|
kern_pair++;
|
|
|
|
}
|
|
|
|
TRACE("copied %u entries\n", nPairs);
|
|
|
|
return nPairs;
|
|
|
|
}
|
|
|
|
|
2011-10-20 16:39:53 +02:00
|
|
|
/*************************************************************
|
2020-10-26 11:50:07 +01:00
|
|
|
* freetype_get_kerning_pairs
|
2011-10-20 16:39:53 +02:00
|
|
|
*/
|
2020-10-29 14:53:18 +01:00
|
|
|
static DWORD CDECL freetype_get_kerning_pairs( struct gdi_font *font, KERNINGPAIR **pairs )
|
2006-09-20 17:18:51 +02:00
|
|
|
{
|
2020-10-29 14:53:18 +01:00
|
|
|
FT_Face ft_face = get_ft_face( font );
|
2020-10-26 11:50:07 +01:00
|
|
|
DWORD length, count = 0;
|
2006-09-20 17:18:51 +02:00
|
|
|
void *buf;
|
|
|
|
const struct TT_kern_table *tt_kern_table;
|
|
|
|
const struct TT_kern_subtable *tt_kern_subtable;
|
|
|
|
USHORT i, nTables;
|
|
|
|
USHORT *glyph_to_char;
|
|
|
|
|
2020-10-29 14:53:18 +01:00
|
|
|
length = freetype_get_font_data(font, MS_KERN_TAG, 0, NULL, 0);
|
2006-09-20 17:18:51 +02:00
|
|
|
|
|
|
|
if (length == GDI_ERROR)
|
|
|
|
{
|
|
|
|
TRACE("no kerning data in the font\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
buf = HeapAlloc(GetProcessHeap(), 0, length);
|
2020-10-22 11:54:35 +02:00
|
|
|
if (!buf) return 0;
|
2006-09-20 17:18:51 +02:00
|
|
|
|
2020-10-29 14:53:18 +01:00
|
|
|
freetype_get_font_data(font, MS_KERN_TAG, 0, buf, length);
|
2006-09-20 17:18:51 +02:00
|
|
|
|
|
|
|
/* build a glyph index to char code map */
|
|
|
|
glyph_to_char = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(USHORT) * 65536);
|
|
|
|
if (!glyph_to_char)
|
|
|
|
{
|
|
|
|
HeapFree(GetProcessHeap(), 0, buf);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-10-29 14:53:18 +01:00
|
|
|
if (ft_face->charmap->encoding == FT_ENCODING_UNICODE)
|
2006-09-20 17:18:51 +02:00
|
|
|
{
|
|
|
|
FT_UInt glyph_code;
|
|
|
|
FT_ULong char_code;
|
|
|
|
|
|
|
|
glyph_code = 0;
|
2020-10-29 14:53:18 +01:00
|
|
|
char_code = pFT_Get_First_Char(ft_face, &glyph_code);
|
2006-09-20 17:18:51 +02:00
|
|
|
|
|
|
|
TRACE("face encoding FT_ENCODING_UNICODE, number of glyphs %ld, first glyph %u, first char %lu\n",
|
2020-10-29 14:53:18 +01:00
|
|
|
ft_face->num_glyphs, glyph_code, char_code);
|
2006-09-20 17:18:51 +02:00
|
|
|
|
|
|
|
while (glyph_code)
|
|
|
|
{
|
|
|
|
/*TRACE("Char %04lX -> Index %u%s\n", char_code, glyph_code, glyph_to_char[glyph_code] ? " !" : "" );*/
|
|
|
|
|
|
|
|
/* FIXME: This doesn't match what Windows does: it does some fancy
|
|
|
|
* things with duplicate glyph index to char code mappings, while
|
|
|
|
* we just avoid overriding existing entries.
|
|
|
|
*/
|
|
|
|
if (glyph_code <= 65535 && !glyph_to_char[glyph_code])
|
|
|
|
glyph_to_char[glyph_code] = (USHORT)char_code;
|
|
|
|
|
2020-10-29 14:53:18 +01:00
|
|
|
char_code = pFT_Get_Next_Char(ft_face, char_code, &glyph_code);
|
2006-09-20 17:18:51 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-10-29 14:53:18 +01:00
|
|
|
DWORD encoding = RtlUlongByteSwap(ft_face->charmap->encoding);
|
2006-09-20 17:18:51 +02:00
|
|
|
ULONG n;
|
|
|
|
|
2017-04-06 13:07:05 +02:00
|
|
|
FIXME("encoding %s not supported\n", debugstr_an((char *)&encoding, 4));
|
2006-09-20 17:18:51 +02:00
|
|
|
for (n = 0; n <= 65535; n++)
|
|
|
|
glyph_to_char[n] = (USHORT)n;
|
|
|
|
}
|
|
|
|
|
|
|
|
tt_kern_table = buf;
|
|
|
|
nTables = GET_BE_WORD(tt_kern_table->nTables);
|
|
|
|
TRACE("version %u, nTables %u\n",
|
|
|
|
GET_BE_WORD(tt_kern_table->version), nTables);
|
|
|
|
|
|
|
|
tt_kern_subtable = (const struct TT_kern_subtable *)(tt_kern_table + 1);
|
|
|
|
|
|
|
|
for (i = 0; i < nTables; i++)
|
|
|
|
{
|
|
|
|
struct TT_kern_subtable tt_kern_subtable_copy;
|
|
|
|
|
|
|
|
tt_kern_subtable_copy.version = GET_BE_WORD(tt_kern_subtable->version);
|
|
|
|
tt_kern_subtable_copy.length = GET_BE_WORD(tt_kern_subtable->length);
|
|
|
|
tt_kern_subtable_copy.coverage.word = GET_BE_WORD(tt_kern_subtable->coverage.word);
|
|
|
|
|
|
|
|
TRACE("version %u, length %u, coverage %u, subtable format %u\n",
|
|
|
|
tt_kern_subtable_copy.version, tt_kern_subtable_copy.length,
|
|
|
|
tt_kern_subtable_copy.coverage.word, tt_kern_subtable_copy.coverage.bits.format);
|
|
|
|
|
|
|
|
/* According to the TrueType specification this is the only format
|
|
|
|
* that will be properly interpreted by Windows and OS/2
|
|
|
|
*/
|
|
|
|
if (tt_kern_subtable_copy.coverage.bits.format == 0)
|
|
|
|
{
|
2020-10-26 11:50:07 +01:00
|
|
|
DWORD new_chunk, old_total = count;
|
2006-09-20 17:18:51 +02:00
|
|
|
|
|
|
|
new_chunk = parse_format0_kern_subtable(font, (const struct TT_format0_kern_subtable *)(tt_kern_subtable + 1),
|
|
|
|
glyph_to_char, NULL, 0);
|
2020-10-26 11:50:07 +01:00
|
|
|
count += new_chunk;
|
2006-09-20 17:18:51 +02:00
|
|
|
|
2020-10-26 11:50:07 +01:00
|
|
|
if (!*pairs)
|
|
|
|
*pairs = HeapAlloc(GetProcessHeap(), 0, count * sizeof(**pairs));
|
2006-09-20 17:18:51 +02:00
|
|
|
else
|
2020-10-26 11:50:07 +01:00
|
|
|
*pairs = HeapReAlloc(GetProcessHeap(), 0, *pairs, count * sizeof(**pairs));
|
2006-09-20 17:18:51 +02:00
|
|
|
|
|
|
|
parse_format0_kern_subtable(font, (const struct TT_format0_kern_subtable *)(tt_kern_subtable + 1),
|
2020-10-26 11:50:07 +01:00
|
|
|
glyph_to_char, *pairs + old_total, new_chunk);
|
2006-09-20 17:18:51 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
TRACE("skipping kerning table format %u\n", tt_kern_subtable_copy.coverage.bits.format);
|
|
|
|
|
|
|
|
tt_kern_subtable = (const struct TT_kern_subtable *)((const char *)tt_kern_subtable + tt_kern_subtable_copy.length);
|
|
|
|
}
|
|
|
|
|
|
|
|
HeapFree(GetProcessHeap(), 0, glyph_to_char);
|
|
|
|
HeapFree(GetProcessHeap(), 0, buf);
|
2020-10-26 11:50:07 +01:00
|
|
|
return count;
|
2006-09-20 17:18:51 +02:00
|
|
|
}
|
2005-10-06 14:28:11 +02:00
|
|
|
|
2020-10-20 22:05:37 +02:00
|
|
|
static const struct font_backend_funcs font_funcs =
|
|
|
|
{
|
|
|
|
freetype_SelectFont,
|
2020-10-26 12:04:36 +01:00
|
|
|
freetype_add_font,
|
2020-10-26 12:05:19 +01:00
|
|
|
freetype_add_mem_font,
|
2020-10-26 12:04:36 +01:00
|
|
|
freetype_remove_font,
|
2020-10-27 14:25:47 +01:00
|
|
|
freetype_load_font,
|
2020-10-22 11:57:48 +02:00
|
|
|
freetype_get_font_data,
|
2020-10-22 13:13:52 +02:00
|
|
|
freetype_get_glyph_index,
|
|
|
|
freetype_get_default_glyph,
|
2020-10-22 11:56:47 +02:00
|
|
|
freetype_get_glyph_outline,
|
2020-10-28 10:13:53 +01:00
|
|
|
freetype_get_unicode_ranges,
|
2020-10-28 10:14:01 +01:00
|
|
|
freetype_get_char_width_info,
|
2020-10-26 11:43:24 +01:00
|
|
|
freetype_set_outline_text_metrics,
|
|
|
|
freetype_set_bitmap_text_metrics,
|
2020-10-26 11:50:07 +01:00
|
|
|
freetype_get_kerning_pairs,
|
2020-10-20 22:05:37 +02:00
|
|
|
freetype_destroy_font
|
2011-10-18 11:44:41 +02:00
|
|
|
};
|
|
|
|
|
2001-09-12 22:21:06 +02:00
|
|
|
#else /* HAVE_FREETYPE */
|
|
|
|
|
2006-05-12 00:34:55 +02:00
|
|
|
/*************************************************************************/
|
|
|
|
|
2020-10-20 22:05:37 +02:00
|
|
|
BOOL WineEngInit( const struct font_backend_funcs **funcs )
|
2001-09-12 22:21:06 +02:00
|
|
|
{
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* HAVE_FREETYPE */
|