/* * FreeType font engine interface * * Copyright 2001 Huw D M Davies for CodeWeavers. * Copyright 2006 Dmitry Timoshkov for CodeWeavers. * * This file contains the WineEng* functions. * * 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 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA */ #include "config.h" #include "wine/port.h" #include #include #ifdef HAVE_SYS_STAT_H # include #endif #ifdef HAVE_SYS_MMAN_H # include #endif #include #ifdef HAVE_DIRENT_H # include #endif #include #include #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 #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 */ #ifdef HAVE_FT2BUILD_H #include #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 */ #include "windef.h" #include "winbase.h" #include "winternl.h" #include "winerror.h" #include "winreg.h" #include "wingdi.h" #include "gdi_private.h" #include "wine/unicode.h" #include "wine/debug.h" #include "wine/list.h" #include "resource.h" WINE_DEFAULT_DEBUG_CHANNEL(font); #ifdef HAVE_FREETYPE #ifndef HAVE_FT_TRUETYPEENGINETYPE typedef enum { FT_TRUETYPE_ENGINE_TYPE_NONE = 0, FT_TRUETYPE_ENGINE_TYPE_UNPATENTED, FT_TRUETYPE_ENGINE_TYPE_PATENTED } FT_TrueTypeEngineType; #endif static FT_Library library = 0; typedef struct { FT_Int major; FT_Int minor; FT_Int patch; } FT_Version_t; static FT_Version_t FT_Version; static DWORD FT_SimpleVersion; #define FT_VERSION_VALUE(major, minor, patch) (((major) << 16) | ((minor) << 8) | (patch)) static void *ft_handle = NULL; #define MAKE_FUNCPTR(f) static typeof(f) * p##f = NULL MAKE_FUNCPTR(FT_Done_Face); MAKE_FUNCPTR(FT_Get_Char_Index); MAKE_FUNCPTR(FT_Get_First_Char); MAKE_FUNCPTR(FT_Get_Next_Char); MAKE_FUNCPTR(FT_Get_Sfnt_Name); MAKE_FUNCPTR(FT_Get_Sfnt_Name_Count); MAKE_FUNCPTR(FT_Get_Sfnt_Table); MAKE_FUNCPTR(FT_Get_WinFNT_Header); MAKE_FUNCPTR(FT_Init_FreeType); MAKE_FUNCPTR(FT_Library_Version); MAKE_FUNCPTR(FT_Load_Glyph); MAKE_FUNCPTR(FT_Load_Sfnt_Table); MAKE_FUNCPTR(FT_Matrix_Multiply); #ifdef FT_MULFIX_INLINED #define pFT_MulFix FT_MULFIX_INLINED #else MAKE_FUNCPTR(FT_MulFix); #endif MAKE_FUNCPTR(FT_New_Face); MAKE_FUNCPTR(FT_New_Memory_Face); MAKE_FUNCPTR(FT_Outline_Get_Bitmap); MAKE_FUNCPTR(FT_Outline_Get_CBox); MAKE_FUNCPTR(FT_Outline_Transform); MAKE_FUNCPTR(FT_Outline_Translate); MAKE_FUNCPTR(FT_Render_Glyph); MAKE_FUNCPTR(FT_Set_Charmap); MAKE_FUNCPTR(FT_Set_Pixel_Sizes); MAKE_FUNCPTR(FT_Vector_Length); MAKE_FUNCPTR(FT_Vector_Transform); MAKE_FUNCPTR(FT_Vector_Unit); static FT_Error (*pFT_Outline_Embolden)(FT_Outline *, FT_Pos); static FT_TrueTypeEngineType (*pFT_Get_TrueType_Engine_Type)(FT_Library); #ifdef FT_LCD_FILTER_H static FT_Error (*pFT_Library_SetLcdFilter)(FT_Library, FT_LcdFilter); #endif static FT_Error (*pFT_Property_Set)(FT_Library, const FT_String *, const FT_String *, const void *); #ifdef SONAME_LIBFONTCONFIG #include MAKE_FUNCPTR(FcConfigSubstitute); MAKE_FUNCPTR(FcDefaultSubstitute); MAKE_FUNCPTR(FcFontList); MAKE_FUNCPTR(FcFontMatch); MAKE_FUNCPTR(FcFontSetDestroy); MAKE_FUNCPTR(FcInit); MAKE_FUNCPTR(FcPatternAddString); MAKE_FUNCPTR(FcPatternCreate); MAKE_FUNCPTR(FcPatternDestroy); MAKE_FUNCPTR(FcPatternGetBool); MAKE_FUNCPTR(FcPatternGetInteger); MAKE_FUNCPTR(FcPatternGetString); #ifndef FC_NAMELANG #define FC_NAMELANG "namelang" #endif #ifndef FC_PRGNAME #define FC_PRGNAME "prgname" #endif #endif /* SONAME_LIBFONTCONFIG */ #undef MAKE_FUNCPTR #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 #ifndef ft_encoding_none #define FT_ENCODING_NONE ft_encoding_none #endif #ifndef ft_encoding_ms_symbol #define FT_ENCODING_MS_SYMBOL ft_encoding_symbol #endif #ifndef ft_encoding_unicode #define FT_ENCODING_UNICODE ft_encoding_unicode #endif #ifndef ft_encoding_apple_roman #define FT_ENCODING_APPLE_ROMAN ft_encoding_apple_roman #endif #ifdef WORDS_BIGENDIAN #define GET_BE_WORD(x) (x) #define GET_BE_DWORD(x) (x) #else #define GET_BE_WORD(x) RtlUshortByteSwap(x) #define GET_BE_DWORD(x) RtlUlongByteSwap(x) #endif /* 'gasp' flags */ #define GASP_GRIDFIT 0x01 #define GASP_DOGRAY 0x02 /* 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; typedef struct gdi_font_face Face; typedef struct gdi_font_family Family; struct font_private_data { FT_Face ft_face; struct font_mapping *mapping; }; static inline FT_Face get_ft_face( struct gdi_font *font ) { return ((struct font_private_data *)font->private)->ft_face; } static const struct font_backend_funcs font_funcs; 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'}; static const WCHAR external_fonts_reg_key[] = {'S','o','f','t','w','a','r','e','\\','W','i','n','e','\\', 'F','o','n','t','s','\\','E','x','t','e','r','n','a','l',' ','F','o','n','t','s','\0'}; /* Interesting and well-known (frequently-assumed!) font names */ 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}; 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}; static const WCHAR times_new_roman[] = {'T','i','m','e','s',' ','N','e','w',' ','R','o','m','a','n',0}; static const WCHAR SymbolW[] = {'S','y','m','b','o','l','\0'}; static const WCHAR *default_serif_list[] = { times_new_roman, liberation_serif, bitstream_vera_serif, NULL }; static const WCHAR *default_fixed_list[] = { courier_new, liberation_mono, bitstream_vera_sans_mono, NULL }; static const WCHAR *default_sans_list[] = { arial, liberation_sans, bitstream_vera_sans, NULL }; static const WCHAR *default_serif = times_new_roman; static const WCHAR *default_fixed = courier_new; static const WCHAR *default_sans = arial; /* 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}; 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 ); static UINT default_aa_flags; static HKEY hkey_font_cache; static BOOL antialias_fakes = TRUE; static const WCHAR font_mutex_nameW[] = {'_','_','W','I','N','E','_','F','O','N','T','_','M','U','T','E','X','_','_','\0'}; static BOOL CDECL freetype_set_outline_text_metrics( struct gdi_font *font ); static BOOL CDECL freetype_set_bitmap_text_metrics( struct gdi_font *font ); static void remove_face_from_cache( Face *face ); /**************************************** * 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 * OEMFONT.FON Terminal * LogPixels Current dpi set by the display control panel applet * (HKLM\\Software\\Microsoft\\Windows NT\\CurrentVersion\\FontDPI * 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 */ #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; ResFileRefNum res_ref; 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; ResType fond_res = FT_MAKE_TAG('F','O','N','D'); fond = Get1IndResource(fond_res, idx); 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; ResType sfnt_res = FT_MAKE_TAG('s','f','n','t'); 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); sfnt = GetResource(sfnt_res, font_id); 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 */ /* This function builds an FT_Fixed from a double. It fails if the absolute value of the float number is greater than 32768. */ static inline FT_Fixed FT_FixedFromFloat(double f) { return f * 0x10000; } /* 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) { return (FT_Fixed)((int)f.value << 16 | (unsigned int)f.fract); } 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; TRACE("hinting is %senabled\n", enabled ? "" : "NOT "); } return enabled; } static BOOL is_subpixel_rendering_enabled( void ) { static int enabled = -1; if (enabled == -1) { /* FreeType >= 2.8.1 offers LCD-optimezed rendering without lcd filters. */ if (FT_SimpleVersion >= FT_VERSION_VALUE(2, 8, 1)) 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; TRACE("subpixel rendering is %senabled\n", enabled ? "" : "NOT "); } return enabled; } static const struct list *get_face_list_from_family(const Family *family) { if (family->replacement) return &family->replacement->faces; return &family->faces; } 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; } 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; } 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; } static int match_name_table_language( const FT_SfntName *name, LANGID lang ) { LANGID name_lang; int res = 0; switch (name->platform_id) { case TT_PLATFORM_MICROSOFT: res += 5; /* prefer the Microsoft name */ switch (name->encoding_id) { case TT_MS_ID_UNICODE_CS: case TT_MS_ID_SYMBOL_CS: name_lang = name->language_id; break; default: return 0; } break; case TT_PLATFORM_MACINTOSH: if (!IsValidCodePage( get_mac_code_page( name ))) return 0; if (name->language_id >= ARRAY_SIZE( mac_langid_table )) return 0; name_lang = mac_langid_table[name->language_id]; break; case TT_PLATFORM_APPLE_UNICODE: res += 2; /* prefer Unicode encodings */ switch (name->encoding_id) { case TT_APPLE_ID_DEFAULT: case TT_APPLE_ID_ISO_10646: case TT_APPLE_ID_UNICODE_2_0: if (name->language_id >= ARRAY_SIZE( mac_langid_table )) return 0; name_lang = mac_langid_table[name->language_id]; break; default: return 0; } break; default: return 0; } 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; else if (lang == MAKELANGID( LANG_NEUTRAL, SUBLANG_NEUTRAL )) res += 5 * (0x100000 - name_lang); return res; } static WCHAR *copy_name_table_string( const FT_SfntName *name ) { WCHAR *ret; WORD codepage; int i; switch (name->platform_id) { case TT_PLATFORM_APPLE_UNICODE: 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; } static WCHAR *get_face_name(FT_Face ft_face, FT_UShort name_id, LANGID language_id) { FT_SfntName name; FT_UInt num_names, name_index; int res, best_lang = 0, best_index = -1; if (!FT_IS_SFNT(ft_face)) return NULL; num_names = pFT_Get_Sfnt_Name_Count( ft_face ); 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) { best_lang = res; best_index = name_index; } } 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; } 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 ); } 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; } static inline BOOL faces_equal( const Face *f1, const Face *f2 ) { if (strcmpiW( f1->full_name, f2->full_name )) return FALSE; if (f1->scalable) return TRUE; if (f1->size.y_ppem != f2->size.y_ppem) return FALSE; return !memcmp( &f1->fs, &f2->fs, sizeof(f1->fs) ); } static void release_face( Face *face ) { if (--face->refcount) return; if (face->family) { if (face->flags & ADDFONT_ADD_TO_CACHE) remove_face_from_cache( face ); list_remove( &face->entry ); release_family( face->family ); } HeapFree( GetProcessHeap(), 0, face->file ); HeapFree( GetProcessHeap(), 0, face->style_name ); HeapFree( GetProcessHeap(), 0, face->full_name ); HeapFree( GetProcessHeap(), 0, face->cached_enum_data ); HeapFree( GetProcessHeap(), 0, face ); } 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: WARN( "Don't know how to order face %s with flags 0x%08x\n", debugstr_w(face->full_name), face->ntmFlags ); 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 )) { TRACE( "Already loaded face %s in family %s, original version %x, new version %x\n", debugstr_w(face->full_name), debugstr_w(family->family_name), cursor->version, face->version ); if (face->file && !strcmpiW( face->file, cursor->file )) { cursor->refcount++; TRACE("Font %s already in list, refcount now %d\n", debugstr_w(face->file), cursor->refcount); return FALSE; } if (face->version <= cursor->version) { TRACE("Original font %s is newer so skipping %s\n", debugstr_w(cursor->file), debugstr_w(face->file)); return FALSE; } else { TRACE("Replacing original %s with %s\n", debugstr_w(cursor->file), debugstr_w(face->file)); list_add_before( &cursor->entry, &face->entry ); face->family = family; family->refcount++; face->refcount++; release_face( cursor ); return TRUE; } } if (style_order( face ) < style_order( cursor )) break; } TRACE( "Adding face %s in family %s from %s\n", debugstr_w(face->full_name), debugstr_w(family->family_name), debugstr_w(face->file) ); list_add_before( &cursor->entry, &face->entry ); face->family = family; family->refcount++; face->refcount++; return TRUE; } struct cached_face { DWORD index; DWORD flags; DWORD ntmflags; DWORD version; struct bitmap_font_size size; FONTSIGNATURE fs; WCHAR full_name[1]; /* WCHAR file_name[]; */ }; static void load_face(HKEY hkey_family, Family *family, void *buffer, DWORD buffer_size, BOOL scalable) { DWORD type, size, needed, index = 0; Face *face; HKEY hkey_strike; WCHAR name[256]; struct cached_face *cached = (struct cached_face *)buffer; size = sizeof(name); needed = buffer_size - sizeof(DWORD); while (!RegEnumValueW( hkey_family, index++, name, &size, NULL, &type, buffer, &needed )) { if (type == REG_BINARY && needed > sizeof(*cached)) { ((DWORD *)buffer)[needed / sizeof(DWORD)] = 0; 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", 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) ); release_face( face ); } size = sizeof(name); needed = buffer_size - sizeof(DWORD); } /* load bitmap strikes */ index = 0; needed = buffer_size; while (!RegEnumKeyExW(hkey_family, index++, buffer, &needed, NULL, NULL, NULL, NULL)) { if (!RegOpenKeyExW(hkey_family, buffer, 0, KEY_ALL_ACCESS, &hkey_strike)) { load_face(hkey_strike, family, buffer, buffer_size, FALSE); RegCloseKey(hkey_strike); } needed = buffer_size; } } /* 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 ) { if (family->family_name[0] != '@') continue; 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 ); if (strcmpiW( family->family_name, vert_family->family_name + 1 ) > 0) { 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 ); } static void load_font_list_from_cache(HKEY hkey_font_cache) { DWORD size, family_index = 0; Family *family; HKEY hkey_family; WCHAR buffer[4096]; size = sizeof(buffer); while (!RegEnumKeyExW(hkey_font_cache, family_index++, buffer, &size, NULL, NULL, NULL, NULL)) { WCHAR *second_name = NULL; WCHAR *family_name = strdupW( buffer ); RegOpenKeyExW(hkey_font_cache, family_name, 0, KEY_ALL_ACCESS, &hkey_family); TRACE("opened family key %s\n", debugstr_w(family_name)); size = sizeof(buffer); if (!RegQueryValueExW( hkey_family, NULL, NULL, NULL, (BYTE *)buffer, &size )) second_name = strdupW( buffer ); family = create_family( family_name, second_name ); load_face(hkey_family, family, buffer, sizeof(buffer), TRUE); HeapFree( GetProcessHeap(), 0, family_name ); HeapFree( GetProcessHeap(), 0, second_name ); RegCloseKey(hkey_family); release_family( family ); size = sizeof(buffer); } reorder_vertical_fonts(); } 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) { HKEY hkey_family, hkey_face; DWORD len, buffer[1024]; struct cached_face *cached = (struct cached_face *)buffer; RegCreateKeyExW( hkey_font_cache, face->family->family_name, 0, NULL, REG_OPTION_VOLATILE, KEY_ALL_ACCESS, NULL, &hkey_family, NULL ); if (face->family->second_name[0]) RegSetValueExW( hkey_family, NULL, 0, REG_SZ, (BYTE *)face->family->second_name, (strlenW( face->family->second_name ) + 1) * sizeof(WCHAR) ); if (!face->scalable) { static const WCHAR fmtW[] = {'%','d',0}; WCHAR name[10]; 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; cached->version = face->version; cached->fs = face->fs; if (!face->scalable) cached->size = face->size; 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); RegCloseKey(hkey_family); } static void remove_face_from_cache( Face *face ) { HKEY hkey_family; RegOpenKeyExW( hkey_font_cache, face->family->family_name, 0, KEY_ALL_ACCESS, &hkey_family ); if (face->scalable) { RegDeleteValueW( hkey_family, face->style_name ); } else { static const WCHAR fmtW[] = {'%','d',0}; WCHAR name[10]; sprintfW( name, fmtW, face->size.y_ppem ); RegDeleteKeyW( hkey_family, name ); } RegCloseKey(hkey_family); } static WCHAR *get_vertical_name( WCHAR *name ) { SIZE_T length; if (!name) return NULL; if (name[0] == '@') return name; length = strlenW( name ) + 1; name = HeapReAlloc( GetProcessHeap(), 0, name, (length + 1) * sizeof(WCHAR) ); memmove( name + 1, name, length * sizeof(WCHAR) ); name[0] = '@'; return name; } static Family *get_family( FT_Face ft_face, BOOL vertical ) { Family *family; WCHAR *family_name, *second_name; family_name = ft_face_get_family_name( ft_face, GetSystemDefaultLCID() ); second_name = ft_face_get_family_name( ft_face, MAKELANGID(LANG_ENGLISH, SUBLANG_DEFAULT) ); /* try to find another secondary name, preferring the lowest langids */ if (!strcmpiW( family_name, second_name )) { 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; } if (vertical) { family_name = get_vertical_name( family_name ); second_name = get_vertical_name( second_name ); } if ((family = find_family_from_name( family_name ))) family->refcount++; else family = create_family( family_name, second_name ); HeapFree( GetProcessHeap(), 0, family_name ); HeapFree( GetProcessHeap(), 0, second_name ); return family; } 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; } static inline DWORD get_ntm_flags( FT_Face ft_face ) { DWORD flags = 0; FT_ULong table_size = 0; FT_WinFNT_HeaderRec winfnt_header; if (ft_face->style_flags & FT_STYLE_FLAG_ITALIC) flags |= NTM_ITALIC; if (ft_face->style_flags & FT_STYLE_FLAG_BOLD) flags |= NTM_BOLD; /* 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; 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; } static inline void get_bitmap_size( FT_Face ft_face, struct bitmap_font_size *face_size ) { My_FT_Bitmap_Size *size; FT_WinFNT_HeaderRec winfnt_header; 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; if (!pFT_Get_WinFNT_Header( ft_face, &winfnt_header )) { face_size->internal_leading = winfnt_header.internal_leading; if (winfnt_header.external_leading > 0 && (face_size->height == winfnt_header.pixel_height + winfnt_header.external_leading)) face_size->height = winfnt_header.pixel_height; } } 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) { if (os2->usFirstCharIndex >= 0xf000 && os2->usFirstCharIndex < 0xf100) fs->fsCsb[0] = FS_SYMBOL; else fs->fsCsb[0] = FS_LATIN1; } 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; } } } } static Face *create_face_from_ft_face( FT_Face ft_face, FT_Long face_index, const WCHAR *filename, DWORD flags ) { 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() ); 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 ); if (!HIWORD( flags )) flags |= ADDFONT_AA_FLAGS( default_aa_flags ); 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 ); TRACE("fsCsb = %08x %08x/%08x %08x %08x %08x\n", fs.fsCsb[0], fs.fsCsb[1], fs.fsUsb[0], fs.fsUsb[1], fs.fsUsb[2], fs.fsUsb[3]); HeapFree( GetProcessHeap(), 0, style_name ); HeapFree( GetProcessHeap(), 0, full_name ); return face; } static void AddFaceToList(FT_Face ft_face, const WCHAR *file, void *font_data_ptr, DWORD font_data_size, FT_Long face_index, DWORD flags ) { Face *face; Family *family; 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; } family = get_family( ft_face, flags & ADDFONT_VERTICAL_FONT ); if (insert_face_in_family_list( face, family )) { if (flags & ADDFONT_ADD_TO_CACHE) add_face_to_cache( face ); TRACE( "Added face %s to family %s\n", debugstr_w(face->full_name), debugstr_w(family->family_name) ); } release_face( face ); release_family( family ); } 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 */ if (!FT_IS_SCALABLE( ft_face ) && FT_SimpleVersion < FT_VERSION_VALUE(2, 1, 9)) { 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; } static INT AddFontToList(const WCHAR *dos_name, const char *unix_name, void *font_data_ptr, DWORD font_data_size, DWORD flags) { FT_Face ft_face; FT_Long face_index = 0, num_faces; INT ret = 0; WCHAR *filename = NULL; /* we always load external fonts from files - otherwise we would get a crash in update_reg_entries */ assert(unix_name || !(flags & ADDFONT_EXTERNAL_FONT)); #ifdef HAVE_CARBON_CARBON_H if(unix_name) { char **mac_list = expand_mac_font(unix_name); if(mac_list) { BOOL had_one = FALSE; char **cursor; for(cursor = mac_list; *cursor; cursor++) { had_one = TRUE; AddFontToList(NULL, *cursor, NULL, 0, flags); HeapFree(GetProcessHeap(), 0, *cursor); } HeapFree(GetProcessHeap(), 0, mac_list); if(had_one) return 1; } } #endif /* HAVE_CARBON_CARBON_H */ if (!dos_name && unix_name) dos_name = filename = wine_get_dos_file_name( unix_name ); do { FONTSIGNATURE fs; ft_face = new_ft_face( unix_name, font_data_ptr, font_data_size, face_index, flags & ADDFONT_ALLOW_BITMAP ); if (!ft_face) break; if(ft_face->family_name[0] == '.') /* Ignore fonts with names beginning with a dot */ { TRACE("Ignoring %s since its family name begins with a dot\n", debugstr_a(unix_name)); pFT_Done_Face(ft_face); break; } AddFaceToList(ft_face, dos_name, font_data_ptr, font_data_size, face_index, flags); ++ret; get_fontsig(ft_face, &fs); if (fs.fsCsb[0] & FS_DBCS_MASK) { AddFaceToList(ft_face, dos_name, font_data_ptr, font_data_size, face_index, flags | ADDFONT_VERTICAL_FONT); ++ret; } num_faces = ft_face->num_faces; pFT_Done_Face(ft_face); } while(num_faces > ++face_index); HeapFree( GetProcessHeap(), 0, filename ); return ret; } /************************************************************* * freetype_add_font */ static INT CDECL freetype_add_font( const WCHAR *file, DWORD flags ) { int ret = 0; char *unixname = wine_get_unix_file_name( file ); if (unixname) { ret = AddFontToList( file, unixname, NULL, 0, flags ); HeapFree( GetProcessHeap(), 0, unixname ); } return ret; } /************************************************************* * 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 ); } /************************************************************* * freetype_remove_font */ static INT CDECL freetype_remove_font( const WCHAR *file, DWORD flags ) { 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; if (!strcmpiW( face->file, file )) { TRACE( "removing matching face %s refcount %d\n", debugstr_w(face->file), face->refcount ); release_face( face ); count++; } } release_family( family ); } return count; } #ifdef __ANDROID__ static BOOL ReadFontDir(const char *dirname, BOOL external_fonts) { DIR *dir; struct dirent *dent; char path[MAX_PATH]; TRACE("Loading fonts from %s\n", debugstr_a(dirname)); dir = opendir(dirname); if(!dir) { WARN("Can't open directory %s\n", debugstr_a(dirname)); return FALSE; } while((dent = readdir(dir)) != NULL) { struct stat statbuf; if(!strcmp(dent->d_name, ".") || !strcmp(dent->d_name, "..")) continue; TRACE("Found %s in %s\n", debugstr_a(dent->d_name), debugstr_a(dirname)); sprintf(path, "%s/%s", dirname, dent->d_name); if(stat(path, &statbuf) == -1) { WARN("Can't stat %s\n", debugstr_a(path)); continue; } if(S_ISDIR(statbuf.st_mode)) ReadFontDir(path, external_fonts); else { DWORD addfont_flags = ADDFONT_ADD_TO_CACHE; if(external_fonts) addfont_flags |= ADDFONT_EXTERNAL_FONT; AddFontToList(NULL, path, NULL, 0, addfont_flags); } } closedir(dir); return TRUE; } #endif #ifdef SONAME_LIBFONTCONFIG 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; case FC_RGBA_NONE: aa_flags = aa_flags ? aa_flags : GGO_GRAY4_BITMAP; break; } } return aa_flags; } static void init_fontconfig(void) { void *fc_handle = dlopen(SONAME_LIBFONTCONFIG, RTLD_NOW); if (!fc_handle) { TRACE("Wine cannot find the fontconfig library (%s).\n", SONAME_LIBFONTCONFIG); return; } #define LOAD_FUNCPTR(f) if((p##f = dlsym(fc_handle, #f)) == NULL){WARN("Can't find symbol %s\n", #f); return;} LOAD_FUNCPTR(FcConfigSubstitute); LOAD_FUNCPTR(FcDefaultSubstitute); LOAD_FUNCPTR(FcFontList); LOAD_FUNCPTR(FcFontMatch); LOAD_FUNCPTR(FcFontSetDestroy); LOAD_FUNCPTR(FcInit); LOAD_FUNCPTR(FcPatternAddString); 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 ); if (!default_aa_flags) { FcPattern *pattern = pFcPatternCreate(); pFcConfigSubstitute( NULL, pattern, FcMatchPattern ); default_aa_flags = parse_aa_pattern( pattern ); pFcPatternDestroy( pattern ); } TRACE( "enabled, default flags = %x\n", default_aa_flags ); fontconfig_enabled = TRUE; } } static void load_fontconfig_fonts(void) { FcPattern *pat; FcFontSet *fontset; int i, len; char *file; const char *ext; if (!fontconfig_enabled) return; pat = pFcPatternCreate(); if (!pat) return; fontset = pFcFontList(NULL, pat, NULL); if (!fontset) { pFcPatternDestroy(pat); return; } for(i = 0; i < fontset->nfont; i++) { FcBool scalable; DWORD aa_flags; if(pFcPatternGetString(fontset->fonts[i], FC_FILE, 0, (FcChar8**)&file) != FcResultMatch) continue; pFcConfigSubstitute( NULL, fontset->fonts[i], FcMatchFont ); /* We're just interested in OT/TT fonts for now, so this hack just 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; } aa_flags = parse_aa_pattern( fontset->fonts[i] ); TRACE("fontconfig: %s aa %x\n", file, aa_flags); len = strlen( file ); if(len < 4) continue; ext = &file[ len - 3 ]; if(_strnicmp(ext, "pfa", -1) && _strnicmp(ext, "pfb", -1)) AddFontToList(NULL, file, NULL, 0, ADDFONT_EXTERNAL_FONT | ADDFONT_ADD_TO_CACHE | ADDFONT_AA_FLAGS(aa_flags) ); } pFcFontSetDestroy(fontset); pFcPatternDestroy(pat); } #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); AddFontToList(NULL, path, NULL, 0, ADDFONT_EXTERNAL_FONT | ADDFONT_ADD_TO_CACHE); } 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); #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 we have to go CFFontDescriptor -> CTFont -> ATSFont -> FSRef -> CFURL. */ { CTFontRef font; ATSFontRef atsFont; OSStatus status; FSRef fsref; font = CTFontCreateWithFontDescriptor(desc, 0, NULL); if (!font) continue; atsFont = CTFontGetPlatformFont(font, NULL); if (!atsFont) { CFRelease(font); continue; } status = ATSFontGetFileReference(atsFont, &fsref); CFRelease(font); if (status != noErr) continue; url = CFURLCreateFromFSRef(NULL, &fsref); } #endif 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 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; } /************************************************************* * * This adds registry entries for any externally loaded fonts * (fonts from fontconfig or FontDirs). It also deletes entries * of no longer existing fonts. * */ static void update_reg_entries(void) { HKEY winnt_key = 0, win9x_key = 0, external_key = 0; LPWSTR valueW; DWORD len; Family *family; Face *face; WCHAR *file, *path; static const WCHAR TrueType[] = {' ','(','T','r','u','e','T','y','p','e',')','\0'}; 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(RegCreateKeyExW(HKEY_CURRENT_USER, external_fonts_reg_key, 0, NULL, 0, KEY_ALL_ACCESS, NULL, &external_key, NULL) != ERROR_SUCCESS) { ERR("Can't create external font reg key\n"); goto end; } /* enumerate the fonts and add external ones to the two keys */ LIST_FOR_EACH_ENTRY( family, &font_list, Family, entry ) { LIST_FOR_EACH_ENTRY( face, &family->faces, Face, entry ) { if (!(face->flags & ADDFONT_EXTERNAL_FONT)) continue; len = strlenW( face->full_name ) + 1; if (face->scalable) len += ARRAY_SIZE(TrueType); valueW = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); strcpyW( valueW, face->full_name ); if (face->scalable) strcatW(valueW, TrueType); if ((path = get_full_path_name(face->file))) { file = path; } else if ((file = strrchrW(face->file, '\\'))) { file++; } else { file = face->file; } len = strlenW(file) + 1; 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)); HeapFree(GetProcessHeap(), 0, path); HeapFree(GetProcessHeap(), 0, valueW); } } end: if(external_key) RegCloseKey(external_key); if(win9x_key) RegCloseKey(win9x_key); if(winnt_key) RegCloseKey(winnt_key); } static void delete_external_font_keys(void) { HKEY winnt_key = 0, win9x_key = 0, external_key = 0; DWORD dlen, plen, vlen, datalen, valuelen, i, type, path_type; LPWSTR valueW; LPVOID data; BYTE *path; 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)); data = HeapAlloc(GetProcessHeap(), 0, datalen); path = HeapAlloc(GetProcessHeap(), 0, datalen); dlen = datalen; vlen = valuelen; i = 0; while(RegEnumValueW(external_key, i++, valueW, &vlen, NULL, &type, data, &dlen) == ERROR_SUCCESS) { 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); /* reset dlen and vlen */ dlen = datalen; vlen = valuelen; } HeapFree(GetProcessHeap(), 0, path); 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); } static BOOL init_freetype(void) { ft_handle = dlopen(SONAME_LIBFREETYPE, RTLD_NOW); 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; } #define LOAD_FUNCPTR(f) if((p##f = dlsym(ft_handle, #f)) == NULL){WARN("Can't find symbol %s\n", #f); goto sym_not_found;} LOAD_FUNCPTR(FT_Done_Face) LOAD_FUNCPTR(FT_Get_Char_Index) LOAD_FUNCPTR(FT_Get_First_Char) LOAD_FUNCPTR(FT_Get_Next_Char) LOAD_FUNCPTR(FT_Get_Sfnt_Name) LOAD_FUNCPTR(FT_Get_Sfnt_Name_Count) LOAD_FUNCPTR(FT_Get_Sfnt_Table) LOAD_FUNCPTR(FT_Get_WinFNT_Header) LOAD_FUNCPTR(FT_Init_FreeType) LOAD_FUNCPTR(FT_Library_Version) LOAD_FUNCPTR(FT_Load_Glyph) LOAD_FUNCPTR(FT_Load_Sfnt_Table) LOAD_FUNCPTR(FT_Matrix_Multiply) #ifndef FT_MULFIX_INLINED LOAD_FUNCPTR(FT_MulFix) #endif LOAD_FUNCPTR(FT_New_Face) LOAD_FUNCPTR(FT_New_Memory_Face) LOAD_FUNCPTR(FT_Outline_Get_Bitmap) LOAD_FUNCPTR(FT_Outline_Get_CBox) LOAD_FUNCPTR(FT_Outline_Transform) LOAD_FUNCPTR(FT_Outline_Translate) LOAD_FUNCPTR(FT_Render_Glyph) LOAD_FUNCPTR(FT_Set_Charmap) LOAD_FUNCPTR(FT_Set_Pixel_Sizes) LOAD_FUNCPTR(FT_Vector_Length) LOAD_FUNCPTR(FT_Vector_Transform) LOAD_FUNCPTR(FT_Vector_Unit) #undef LOAD_FUNCPTR /* Don't warn if these ones are missing */ pFT_Outline_Embolden = dlsym(ft_handle, "FT_Outline_Embolden"); pFT_Get_TrueType_Engine_Type = dlsym(ft_handle, "FT_Get_TrueType_Engine_Type"); #ifdef FT_LCD_FILTER_H pFT_Library_SetLcdFilter = dlsym(ft_handle, "FT_Library_SetLcdFilter"); #endif pFT_Property_Set = dlsym(ft_handle, "FT_Property_Set"); if(pFT_Init_FreeType(&library) != 0) { ERR("Can't init FreeType library\n"); dlclose(ft_handle); ft_handle = NULL; return FALSE; } pFT_Library_Version(library,&FT_Version.major,&FT_Version.minor,&FT_Version.patch); TRACE("FreeType version is %d.%d.%d\n",FT_Version.major,FT_Version.minor,FT_Version.patch); FT_SimpleVersion = ((FT_Version.major << 16) & 0xff0000) | ((FT_Version.minor << 8) & 0x00ff00) | ((FT_Version.patch ) & 0x0000ff); /* In FreeType < 2.8.1 v40's FT_LOAD_TARGET_MONO has broken advance widths. */ 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 ); } 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" "FreeType to at least version 2.1.4.\n" "http://www.freetype.org\n"); dlclose(ft_handle); ft_handle = NULL; return FALSE; } static void init_font_list(void) { delete_external_font_keys(); load_system_bitmap_fonts(); load_file_system_fonts(); load_registry_fonts(); #ifdef SONAME_LIBFONTCONFIG load_fontconfig_fonts(); #elif defined(HAVE_CARBON_CARBON_H) load_mac_fonts(); #elif defined(__ANDROID__) ReadFontDir("/system/fonts", TRUE); #endif } static BOOL move_to_front(const WCHAR *name) { Family *family, *cursor2; LIST_FOR_EACH_ENTRY_SAFE(family, cursor2, &font_list, Family, entry) { if (!strncmpiW( family->family_name, name, LF_FACESIZE - 1 )) { list_remove(&family->entry); list_add_head(&font_list, &family->entry); return TRUE; } } return FALSE; } static const WCHAR *set_default(const WCHAR **name_list) { const WCHAR **entry = name_list; while (*entry) { if (move_to_front(*entry)) return *entry; entry++; } return *name_list; } static void reorder_font_list(void) { default_serif = set_default( default_serif_list ); default_fixed = set_default( default_fixed_list ); default_sans = set_default( default_sans_list ); } /************************************************************* * WineEngInit * * Initialize FreeType library and create a list of available faces */ BOOL WineEngInit( const struct font_backend_funcs **funcs ) { HKEY hkey; DWORD disposition; HANDLE font_mutex; if(!init_freetype()) return FALSE; #ifdef SONAME_LIBFONTCONFIG init_fontconfig(); #endif *funcs = &font_funcs; 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); } if((font_mutex = CreateMutexW(NULL, FALSE, font_mutex_nameW)) == NULL) { ERR("Failed to create font mutex\n"); return FALSE; } WaitForSingleObject(font_mutex, INFINITE); create_font_cache_key(&hkey_font_cache, &disposition); if(disposition == REG_CREATED_NEW_KEY) init_font_list(); else load_font_list_from_cache(hkey_font_cache); reorder_font_list(); if(disposition == REG_CREATED_NEW_KEY) update_reg_entries(); ReleaseMutex(font_mutex); return TRUE; } /* 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); } static LONG calc_ppem_for_height(FT_Face ft_face, LONG height) { TT_OS2 *pOS2; TT_HoriHeader *pHori; LONG ppem; const LONG MAX_PPEM = (1 << 16) - 1; pOS2 = pFT_Get_Sfnt_Table(ft_face, ft_sfnt_os2); pHori = pFT_Get_Sfnt_Table(ft_face, ft_sfnt_hhea); 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] * */ if(height > 0) { USHORT windescent = get_fixed_windescent(pOS2->usWinDescent); if(pOS2->usWinAscent + windescent == 0) ppem = MulDiv(ft_face->units_per_EM, height, pHori->Ascender - pHori->Descender); else ppem = MulDiv(ft_face->units_per_EM, height, pOS2->usWinAscent + windescent); if(ppem > MAX_PPEM) { WARN("Ignoring too large height %d, ppem %d\n", height, ppem); ppem = 1; } } else if(height >= -MAX_PPEM) ppem = -height; else { WARN("Ignoring too large height %d\n", height); ppem = 1; } return ppem; } static struct font_mapping *map_font_file( const char *name ) { 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; } static void unmap_font_file( struct font_mapping *mapping ) { if (!--mapping->refcount) { list_remove( &mapping->entry ); munmap( mapping->data, mapping->size ); HeapFree( GetProcessHeap(), 0, mapping ); } } static LONG load_VDMX(struct gdi_font *font, LONG height); static UINT get_nearest_charset(const WCHAR *family_name, Face *face, UINT *cp) { /* 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 */ CHARSETINFO csi; int acp = GetACP(), i; DWORD fs0; *cp = acp; if(TranslateCharsetInfo((DWORD*)(INT_PTR)acp, &csi, TCI_SRCCODEPAGE)) { const struct gdi_font_link *font_link; if (csi.fs.fsCsb[0] & face->fs.fsCsb[0]) return csi.ciCharset; font_link = find_gdi_font_link(family_name); if (font_link != NULL && csi.fs.fsCsb[0] & font_link->fs.fsCsb[0]) return csi.ciCharset; } for(i = 0; i < 32; i++) { fs0 = 1L << i; if(face->fs.fsCsb[0] & fs0) { if(TranslateCharsetInfo(&fs0, &csi, TCI_SRCFONTSIG)) { *cp = csi.ciACP; return csi.ciCharset; } else FIXME("TCI failing on %x\n", fs0); } } FIXME("returning DEFAULT_CHARSET face->fs.fsCsb[0] = %08x file = %s\n", face->fs.fsCsb[0], debugstr_w(face->file)); *cp = acp; return DEFAULT_CHARSET; } /************************************************************* * freetype_destroy_font */ static void CDECL freetype_destroy_font( struct gdi_font *font ) { struct font_private_data *data = font->private; if (data->ft_face) pFT_Done_Face( data->ft_face ); if (data->mapping) unmap_font_file( data->mapping ); HeapFree( GetProcessHeap(), 0, data ); } /************************************************************* * freetype_get_font_data */ static DWORD CDECL freetype_get_font_data( struct gdi_font *font, DWORD table, DWORD offset, void *buf, DWORD cbData) { FT_Face ft_face = get_ft_face( font ); FT_ULong len; FT_Error err; if (!FT_IS_SFNT(ft_face)) return GDI_ERROR; if(!buf) len = 0; else len = cbData; /* 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; } /* make sure value of len is the value freetype says it needs */ if (buf && len) { FT_ULong needed = 0; err = pFT_Load_Sfnt_Table(ft_face, RtlUlongByteSwap(table), offset, NULL, &needed); if( !err && needed < len) len = needed; } err = pFT_Load_Sfnt_Table(ft_face, RtlUlongByteSwap(table), offset, buf, &len); if (err) { TRACE("Can't find table %s\n", debugstr_an((char*)&table, 4)); return GDI_ERROR; } return len; } /************************************************************* * load_VDMX * * load the vdmx entry for the specified height */ typedef struct { WORD version; WORD numRecs; WORD numRatios; } VDMX_Header; typedef struct { BYTE bCharSet; BYTE xRatio; BYTE yStartRatio; BYTE yEndRatio; } Ratios; typedef struct { WORD recs; BYTE startsz; BYTE endsz; } VDMX_group; typedef struct { WORD yPelHeight; WORD yMax; WORD yMin; } VDMX_vTable; static LONG load_VDMX(struct gdi_font *font, LONG height) { VDMX_Header hdr; VDMX_group group; BYTE devXRatio, devYRatio; USHORT numRecs, numRatios; DWORD result, offset = -1; LONG ppem = 0; int i; result = freetype_get_font_data(font, MS_VDMX_TAG, 0, &hdr, sizeof(hdr)); if(result == GDI_ERROR) /* no vdmx table present, use linear scaling */ return ppem; /* FIXME: need the real device aspect ratio */ devXRatio = 1; devYRatio = 1; numRecs = GET_BE_WORD(hdr.numRecs); numRatios = GET_BE_WORD(hdr.numRatios); TRACE("version = %d numRecs = %d numRatios = %d\n", GET_BE_WORD(hdr.version), numRecs, numRatios); for(i = 0; i < numRatios; i++) { Ratios ratio; offset = sizeof(hdr) + (i * sizeof(Ratios)); freetype_get_font_data(font, MS_VDMX_TAG, offset, &ratio, sizeof(Ratios)); offset = -1; TRACE("Ratios[%d] %d %d : %d -> %d\n", i, ratio.bCharSet, ratio.xRatio, ratio.yStartRatio, ratio.yEndRatio); if (!ratio.bCharSet) continue; if((ratio.xRatio == 0 && ratio.yStartRatio == 0 && ratio.yEndRatio == 0) || (devXRatio == ratio.xRatio && devYRatio >= ratio.yStartRatio && devYRatio <= ratio.yEndRatio)) { WORD group_offset; offset = sizeof(hdr) + numRatios * sizeof(ratio) + i * sizeof(group_offset); freetype_get_font_data(font, MS_VDMX_TAG, offset, &group_offset, sizeof(group_offset)); offset = GET_BE_WORD(group_offset); break; } } if(offset == -1) return 0; if(freetype_get_font_data(font, MS_VDMX_TAG, offset, &group, sizeof(group)) != GDI_ERROR) { USHORT recs; BYTE startsz, endsz; WORD *vTable; recs = GET_BE_WORD(group.recs); startsz = group.startsz; endsz = group.endsz; TRACE("recs=%d startsz=%d endsz=%d\n", recs, startsz, endsz); vTable = HeapAlloc(GetProcessHeap(), 0, recs * sizeof(VDMX_vTable)); result = freetype_get_font_data(font, MS_VDMX_TAG, offset + sizeof(group), vTable, recs * sizeof(VDMX_vTable)); if(result == GDI_ERROR) { FIXME("Failed to retrieve vTable\n"); goto end; } if(height > 0) { for(i = 0; i < recs; i++) { 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]); if(yMax + -yMin == height) { font->yMax = yMax; font->yMin = yMin; TRACE("ppem %d found; height=%d yMax=%d yMin=%d\n", ppem, height, font->yMax, font->yMin); break; } if(yMax + -yMin > height) { if(--i < 0) { ppem = 0; goto end; /* failed */ } font->yMax = GET_BE_WORD(vTable[(i * 3) + 1]); font->yMin = GET_BE_WORD(vTable[(i * 3) + 2]); ppem = GET_BE_WORD(vTable[i * 3]); TRACE("ppem %d found; height=%d yMax=%d yMin=%d\n", ppem, height, font->yMax, font->yMin); break; } } if(!font->yMax) { ppem = 0; TRACE("ppem not found for height %d\n", height); } } else { ppem = -height; if(ppem < startsz || ppem > endsz) { ppem = 0; goto end; } for(i = 0; i < recs; i++) { USHORT yPelHeight; yPelHeight = GET_BE_WORD(vTable[i * 3]); if(yPelHeight > ppem) { ppem = 0; break; /* failed */ } if(yPelHeight == ppem) { 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); break; } } } end: HeapFree(GetProcessHeap(), 0, vTable); } return ppem; } static BOOL select_charmap(FT_Face ft_face, FT_Encoding encoding) { FT_Error ft_err = FT_Err_Invalid_CharMap_Handle; FT_CharMap cmap0, cmap1, cmap2, cmap3, cmap_def; FT_Int i; cmap0 = cmap1 = cmap2 = cmap3 = cmap_def = NULL; for (i = 0; i < ft_face->num_charmaps; i++) { if (ft_face->charmaps[i]->encoding == encoding) { TRACE("found cmap with platform_id %u, encoding_id %u\n", ft_face->charmaps[i]->platform_id, ft_face->charmaps[i]->encoding_id); 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; } } 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); } return ft_err == FT_Err_Ok; } static FT_Encoding pick_charmap( FT_Face face, int charset ) { static const FT_Encoding regular_order[] = { FT_ENCODING_UNICODE, FT_ENCODING_APPLE_ROMAN, FT_ENCODING_MS_SYMBOL, 0 }; 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++; } if (!face->charmap && face->num_charmaps) { if (!pFT_Set_Charmap(face, face->charmaps[0])) return face->charmap->encoding; } return *encs; } static BOOL get_gasp_flags( struct gdi_font *font, WORD *flags ) { FT_Face ft_face = get_ft_face( font ); 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; size = freetype_get_font_data( font, MS_GASP_TAG, 0, NULL, 0 ); 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; } freetype_get_font_data( font, MS_GASP_TAG, 0, ptr, size ); 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) ); if (ft_face->size->metrics.y_ppem <= GET_BE_WORD( *ptr )) break; ptr += 2; } TRACE( "got flags %04x for ppem %d\n", *flags, ft_face->size->metrics.y_ppem ); ret = TRUE; done: HeapFree( GetProcessHeap(), 0, alloced ); return ret; } #ifdef SONAME_LIBFONTCONFIG static Family* get_fontconfig_family(DWORD pitch_and_family, const CHARSETINFO *csi, BOOL want_vertical) { 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; const struct gdi_font_link *font_link; const struct list *face_list; 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); } if (!ret) continue; family = find_family_from_any_name(nameW); if (!family) continue; font_link = find_gdi_font_link( family->family_name ); 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: pFcPatternDestroy(pat); pFcPatternDestroy(best); return family; } #endif 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 ); } /************************************************************* * freetype_load_font */ static BOOL CDECL freetype_load_font( struct gdi_font *font ) { struct font_private_data *data; INT width = 0, height; FT_Face ft_face; void *data_ptr; SIZE_T data_size; if (!(data = HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(*data) ))) return FALSE; font->private = data; if (font->file[0]) { char *filename = wine_get_unix_file_name( font->file ); data->mapping = map_font_file( filename ); HeapFree( GetProcessHeap(), 0, filename ); if (!data->mapping) { WARN("failed to map %s\n", debugstr_w(font->file)); return FALSE; } data_ptr = data->mapping->data; data_size = data->mapping->size; } else { data_ptr = font->data_ptr; data_size = font->data_size; } if (pFT_New_Memory_Face( library, data_ptr, data_size, font->face_index, &ft_face )) return FALSE; 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) { 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() ); set_gdi_font_names( font, family_name, style_name, full_name ); HeapFree( GetProcessHeap(), 0, family_name ); HeapFree( GetProcessHeap(), 0, style_name ); HeapFree( GetProcessHeap(), 0, full_name ); } if (font->scalable) { /* load the VDMX table if we have one */ 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 ); } else { struct bitmap_font_size size; get_bitmap_size( ft_face, &size ); width = size.x_ppem >> 6; height = size.y_ppem >> 6; font->ppem = height; } pFT_Set_Pixel_Sizes( ft_face, width, height ); pick_charmap( ft_face, font->charset ); return TRUE; } /************************************************************* * freetype_SelectFont */ static struct gdi_font * CDECL freetype_SelectFont( DC *dc, HFONT hfont, UINT *aa_flags, UINT default_aa_flags ) { struct gdi_font *font; Face *face, *best, *best_bitmap; Family *family, *last_resort_family; const struct list *face_list; INT height; unsigned int score = 0, new_score; signed int diff = 0, newdiff; BOOL bd, it, can_use_bitmap, want_vertical; LOGFONTW lf; CHARSETINFO csi; FMAT2 dcmat; const WCHAR *orig_name = NULL; const struct gdi_font_link *font_link; GetObjectW( hfont, sizeof(lf), &lf ); lf.lfWidth = abs(lf.lfWidth); can_use_bitmap = !!(GetDeviceCaps(dc->hSelf, TEXTCAPS) & TC_RA_ABLE); TRACE("%s, h=%d, it=%d, weight=%d, PandF=%02x, charset=%d orient %d escapement %d\n", debugstr_w(lf.lfFaceName), lf.lfHeight, lf.lfItalic, lf.lfWeight, lf.lfPitchAndFamily, lf.lfCharSet, lf.lfOrientation, lf.lfEscapement); if(dc->GraphicsMode == GM_ADVANCED) { memcpy(&dcmat, &dc->xformWorld2Vport, sizeof(FMAT2)); /* 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); dcmat.eM11 = dcmat.eM22 = dcmat.eM11 < 0 ? -1 : 1; } } else { /* Windows 3.1 compatibility mode GM_COMPATIBLE has only limited font scaling abilities. */ dcmat.eM11 = dcmat.eM22 = 1.0; dcmat.eM21 = dcmat.eM12 = 0; lf.lfOrientation = lf.lfEscapement; 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); } } TRACE("DC transform %f %f %f %f\n", dcmat.eM11, dcmat.eM12, dcmat.eM21, dcmat.eM22); /* check the cache first */ if ((font = find_cached_gdi_font( &lf, &dcmat, can_use_bitmap ))) { TRACE("returning cached gdiFont(%p) for hFont %p\n", font, hfont); goto done; } /* 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; it = !!lf.lfItalic; bd = lf.lfWeight > 550; if(!TranslateCharsetInfo((DWORD*)(INT_PTR)lf.lfCharSet, &csi, TCI_SRCCHARSET)) { 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; } } family = NULL; if(lf.lfFaceName[0] != '\0') { LPWSTR FaceName = lf.lfFaceName; int subst_charset; const WCHAR *subst = get_gdi_font_subst( FaceName, lf.lfCharSet, &subst_charset ); if(subst) { TRACE("substituting %s,%d -> %s,%d\n", debugstr_w(FaceName), lf.lfCharSet, debugstr_w(subst), (subst_charset != -1) ? subst_charset : lf.lfCharSet); if (subst_charset != -1) lf.lfCharSet = subst_charset; orig_name = FaceName; } /* 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. */ LIST_FOR_EACH_ENTRY( family, &font_list, Family, entry ) { if (!strncmpiW( family->family_name, FaceName, LF_FACESIZE - 1 ) || (subst && !strncmpiW( family->family_name, subst, LF_FACESIZE - 1 ))) { font_link = find_gdi_font_link( family->family_name ); face_list = get_face_list_from_family(family); LIST_FOR_EACH_ENTRY( face, face_list, Face, entry ) { 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; } } } /* Search by full face name. */ LIST_FOR_EACH_ENTRY( family, &font_list, Family, entry ) { face_list = get_face_list_from_family(family); LIST_FOR_EACH_ENTRY( face, face_list, Face, entry ) { if (!strncmpiW( face->full_name, FaceName, LF_FACESIZE - 1 ) && (face->scalable || can_use_bitmap)) { if (csi.fs.fsCsb[0] & face->fs.fsCsb[0] || !csi.fs.fsCsb[0]) goto found_face; font_link = find_gdi_font_link( family->family_name ); if (font_link != NULL && csi.fs.fsCsb[0] & font_link->fs.fsCsb[0]) goto found_face; } } } /* * Try check the SystemLink list first for a replacement font. * We may find good replacements there. */ if ((family = find_family_from_font_links( FaceName, subst, csi.fs ))) goto found; } orig_name = NULL; /* substitution is no longer relevant */ /* If requested charset was DEFAULT_CHARSET then try using charset corresponding to the current ansi codepage */ if (!csi.fs.fsCsb[0]) { INT acp = GetACP(); if(!TranslateCharsetInfo((DWORD*)(INT_PTR)acp, &csi, TCI_SRCCODEPAGE)) { FIXME("TCI failed on codepage %d\n", acp); csi.fs.fsCsb[0] = 0; } else lf.lfCharSet = csi.ciCharset; } want_vertical = (lf.lfFaceName[0] == '@'); /* Face families are in the top 4 bits of lfPitchAndFamily, so mask with 0xF0 before testing */ if((lf.lfPitchAndFamily & FIXED_PITCH) || (lf.lfPitchAndFamily & 0xF0) == FF_MODERN) strcpyW(lf.lfFaceName, default_fixed); else if((lf.lfPitchAndFamily & 0xF0) == FF_ROMAN) strcpyW(lf.lfFaceName, default_serif); else if((lf.lfPitchAndFamily & 0xF0) == FF_SWISS) strcpyW(lf.lfFaceName, default_sans); else strcpyW(lf.lfFaceName, default_sans); LIST_FOR_EACH_ENTRY( family, &font_list, Family, entry ) { if (!strncmpiW( family->family_name, lf.lfFaceName, LF_FACESIZE - 1 )) { font_link = find_gdi_font_link( family->family_name ); face_list = get_face_list_from_family(family); LIST_FOR_EACH_ENTRY( face, face_list, Face, entry ) { 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; } } } #ifdef SONAME_LIBFONTCONFIG /* Try FontConfig substitutions if the face isn't found */ family = get_fontconfig_family(lf.lfPitchAndFamily, &csi, want_vertical); if (family) goto found; #endif last_resort_family = NULL; LIST_FOR_EACH_ENTRY( family, &font_list, Family, entry ) { font_link = find_gdi_font_link( family->family_name ); face_list = get_face_list_from_family(family); LIST_FOR_EACH_ENTRY( face, face_list, Face, entry ) { if(!(face->flags & ADDFONT_VERTICAL_FONT) == !want_vertical && (csi.fs.fsCsb[0] & face->fs.fsCsb[0] || (font_link != NULL && csi.fs.fsCsb[0] & font_link->fs.fsCsb[0]))) { if(face->scalable) goto found; if(can_use_bitmap && !last_resort_family) last_resort_family = family; } } } if(last_resort_family) { family = last_resort_family; csi.fs.fsCsb[0] = 0; goto found; } LIST_FOR_EACH_ENTRY( family, &font_list, Family, entry ) { face_list = get_face_list_from_family(family); LIST_FOR_EACH_ENTRY( face, face_list, Face, entry ) { if(face->scalable && !(face->flags & ADDFONT_VERTICAL_FONT) == !want_vertical) { csi.fs.fsCsb[0] = 0; WARN("just using first face for now\n"); goto found; } if(can_use_bitmap && !last_resort_family) last_resort_family = family; } } if(!last_resort_family) { FIXME("can't find a single appropriate font - bailing\n"); return NULL; } WARN("could only find a bitmap font - this will probably look awful!\n"); family = last_resort_family; csi.fs.fsCsb[0] = 0; found: height = lf.lfHeight; face = best = best_bitmap = NULL; font_link = find_gdi_font_link( family->family_name ); face_list = get_face_list_from_family(family); LIST_FOR_EACH_ENTRY(face, face_list, Face, entry) { 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]) { BOOL italic, bold; italic = (face->ntmFlags & NTM_ITALIC) ? 1 : 0; bold = (face->ntmFlags & NTM_BOLD) ? 1 : 0; new_score = (italic ^ it) + (bold ^ bd); if(!best || new_score <= score) { TRACE("(it=%d, bd=%d) is selected for (it=%d, bd=%d)\n", italic, bold, it, bd); score = new_score; best = face; 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; } } } } } if(best) face = best->scalable ? best : best_bitmap; found_face: height = lf.lfHeight; TRACE("not in cache\n"); font = create_gdi_font( face, orig_name, &lf ); font->matrix = dcmat; font->can_use_bitmap = can_use_bitmap; if(csi.fs.fsCsb[0]) { font->charset = lf.lfCharSet; font->codepage = csi.ciACP; } else font->charset = get_nearest_charset( family->family_name, face, &font->codepage ); TRACE( "Chosen: %s (%s/%p:%u)\n", debugstr_w(face->full_name), debugstr_w(face->file), face->data_ptr, face->face_index ); font->aveWidth = height ? lf.lfWidth : 0; if(!face->scalable) { /* Windows uses integer scaling factors for bitmap fonts */ INT scale, scaled_height; struct gdi_font *cachedfont; /* FIXME: rotation of bitmap fonts is ignored */ 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; 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. */ if((cachedfont = find_cached_gdi_font( &lf, &dcmat, can_use_bitmap ))) { TRACE("Found cached font after non-scalable matrix rescale!\n"); free_gdi_font( font ); font = cachedfont; goto done; } if (height != 0) height = diff; height += face->size.height; scale = (height + face->size.height - 1) / face->size.height; scaled_height = scale * face->size.height; /* 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--; font->scale_y = scale; } TRACE("font scale y: %f\n", font->scale_y); if (!freetype_load_font( font )) { free_gdi_font( font ); return NULL; } if (face->flags & ADDFONT_VERTICAL_FONT) /* We need to try to load the GSUB table */ font->vert_feature = get_GSUB_vert_feature( font ); create_child_font_list( font ); TRACE("caching: gdiFont=%p hfont=%p\n", font, hfont); cache_gdi_font( font ); done: if (font) { switch (lf.lfQuality) { case NONANTIALIASED_QUALITY: case ANTIALIASED_QUALITY: if (!*aa_flags) *aa_flags = default_aa_flags; break; case CLEARTYPE_QUALITY: case CLEARTYPE_NATURAL_QUALITY: default: if (!*aa_flags) *aa_flags = font->aa_flags; if (!*aa_flags) *aa_flags = default_aa_flags; /* fixup the antialiasing flags for that font */ switch (*aa_flags) { 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: if ((!antialias_fakes || (!font->fake_bold && !font->fake_italic)) && is_hinting_enabled()) { WORD gasp_flags; if (get_gasp_flags( font, &gasp_flags ) && !(gasp_flags & GASP_DOGRAY)) { TRACE( "font %s %d aa disabled by GASP\n", debugstr_w(lf.lfFaceName), lf.lfHeight ); *aa_flags = GGO_BITMAP; } } } } TRACE( "%p %s %d aa %x\n", hfont, debugstr_w(lf.lfFaceName), lf.lfHeight, *aa_flags ); } return font; } 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)); } static FT_UInt get_glyph_index_symbol( struct gdi_font *font, UINT glyph ) { FT_Face ft_face = get_ft_face( font ); 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 */ if (!(ret = pFT_Get_Char_Index(ft_face, glyph))) ret = pFT_Get_Char_Index(ft_face, glyph - 0xf000); return ret; } /************************************************************* * freetype_get_glyph_index */ static BOOL CDECL freetype_get_glyph_index( struct gdi_font *font, UINT *glyph, BOOL use_encoding ) { FT_Face ft_face = get_ft_face( font ); if (!use_encoding ^ (ft_face->charmap->encoding == FT_ENCODING_NONE)) return FALSE; if (ft_face->charmap->encoding == FT_ENCODING_MS_SYMBOL) { if (!(*glyph = get_glyph_index_symbol( font, *glyph ))) { WCHAR wc = *glyph; char ch; if (WideCharToMultiByte( CP_ACP, 0, &wc, 1, &ch, 1, NULL, NULL )) *glyph = get_glyph_index_symbol( font, (unsigned char)ch ); } return TRUE; } *glyph = pFT_Get_Char_Index( ft_face, *glyph ); return TRUE; } /************************************************************* * freetype_get_default_glyph */ static UINT CDECL freetype_get_default_glyph( struct gdi_font *font ) { FT_Face ft_face = get_ft_face( font ); FT_WinFNT_HeaderRec winfnt; TT_OS2 *pOS2; if ((pOS2 = pFT_Get_Sfnt_Table( ft_face, ft_sfnt_os2 ))) { UINT glyph = pOS2->usDefaultChar; freetype_get_glyph_index( font, &glyph, TRUE ); return glyph; } if (!pFT_Get_WinFNT_Header( ft_face, &winfnt )) return winfnt.default_char + winfnt.first_char; return 32; } 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)); } 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; } /* get_glyph_outline() glyph transform matrices index */ enum matrices_index { matrix_hori, matrix_vert, matrix_unrotated }; static BOOL get_transform_matrices( struct gdi_font *font, BOOL vertical, const MAT2 *user_transform, 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 */ if (font->aveWidth) { 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; } else width_ratio = font->scale_y; /* Scaling transform */ if (width_ratio != 1.0 || font->scale_y != 1.0) { FT_Matrix scale_mat; scale_mat.xx = FT_FixedFromFloat( width_ratio ); scale_mat.xy = 0; scale_mat.yx = 0; scale_mat.yy = FT_FixedFromFloat( font->scale_y ); pFT_Matrix_Multiply( &scale_mat, &matrices[matrix_unrotated] ); needs_transform = TRUE; } /* Slant transform */ if (font->fake_italic) { 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]; if (font->scalable && font->lf.lfOrientation % 3600) { FT_Matrix rotation_mat; FT_Vector angle; pFT_Vector_Unit( &angle, MulDiv( 1 << 16, font->lf.lfOrientation, 10 ) ); 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 */ if (!is_identity_FMAT2( &font->matrix )) { FT_Matrix world_mat; 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 ); for (i = 0; i < 3; i++) pFT_Matrix_Multiply( &world_mat, &matrices[i] ); needs_transform = TRUE; } /* Extra transformation specified by caller */ if (user_transform) { 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; } static BOOL get_bold_glyph_outline(FT_GlyphSlot glyph, LONG ppem, FT_Glyph_Metrics *metrics) { FT_Error err; FT_Pos strength; FT_BBox bbox; if(glyph->format != FT_GLYPH_FORMAT_OUTLINE) return FALSE; if(!pFT_Outline_Embolden) return FALSE; 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; } 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; } 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; } static FT_Vector get_advance_metric(struct gdi_font *incoming_font, struct gdi_font *font, 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. */ if (freetype_set_outline_text_metrics(incoming_font) && !(incoming_font->otm.otmTextMetrics.tmPitchAndFamily & TMPF_FIXED_PITCH)) { UINT avg_advance; em_scale = MulDiv(incoming_font->ppem, 1 << 16, get_ft_face(incoming_font)->units_per_EM); avg_advance = pFT_MulFix(incoming_font->ntmAvgWidth, em_scale); fixed_pitch_full = (avg_advance > 0 && (base_advance + 63) >> 6 == pFT_MulFix(incoming_font->ntmAvgWidth*2, em_scale)); 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; vec.x = incoming_font->ntmAvgWidth; vec.y = 0; pFT_Vector_Transform(&vec, transMat); adv.x = (pFT_MulFix(vec.x, em_scale) * 2) << 6; } } if (font->fake_bold) { 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; } } adv.x = (adv.x + 63) & -64; adv.y = -((adv.y + 63) & -64); return adv; } 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; } static void compute_metrics( struct gdi_font *incoming_font, struct gdi_font *font, 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; if (vertical && freetype_set_outline_text_metrics( font )) { if (vertical_metrics) lsb = metrics->horiBearingY + metrics->vertBearingY; else lsb = metrics->vertAdvance + (font->otm.otmDescent << 6); vec.x = lsb; vec.y = font->otm.otmDescent << 6; 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 ); } 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; } 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; } 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; } /************************************************************* * freetype_get_glyph_outline */ static DWORD CDECL freetype_get_glyph_outline( struct gdi_font *font, UINT glyph, UINT format, GLYPHMETRICS *lpgm, ABC *abc, DWORD buflen, void *buf, const MAT2 *lpmat, BOOL tategaki ) { struct gdi_font *base_font = font->base_font ? font->base_font : font; FT_Face ft_face = get_ft_face( base_font ); FT_Glyph_Metrics metrics; FT_Error err; FT_BBox bbox; FT_Int load_flags = get_load_flags(format); FT_Matrix matrices[3]; BOOL needsTransform = FALSE; BOOL vertical_metrics; TRACE("%p, %04x, %08x, %p, %08x, %p, %p\n", font, glyph, format, lpgm, buflen, buf, lpmat); TRACE("font transform %f %f %f %f\n", font->matrix.eM11, font->matrix.eM12, font->matrix.eM21, font->matrix.eM22); needsTransform = get_transform_matrices( font, tategaki, lpmat, matrices ); 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 */ if (vertical_metrics && FT_SimpleVersion < FT_VERSION_VALUE(2, 4, 0)) vertical_metrics = FALSE; if (needsTransform || format != GGO_BITMAP) load_flags |= FT_LOAD_NO_BITMAP; if (vertical_metrics) load_flags |= FT_LOAD_VERTICAL_LAYOUT; err = pFT_Load_Glyph(ft_face, glyph, load_flags); if (err && !(load_flags & FT_LOAD_NO_HINTING)) { WARN("Failed to load glyph %#x, retrying without hinting. Error %#x.\n", glyph, err); load_flags |= FT_LOAD_NO_HINTING; err = pFT_Load_Glyph(ft_face, glyph, load_flags); } if(err) { WARN("Failed to load glyph %#x, error %#x.\n", glyph, err); return GDI_ERROR; } metrics = ft_face->glyph->metrics; if(font->fake_bold) { if (!get_bold_glyph_outline(ft_face->glyph, font->ppem, &metrics) && metrics.width) metrics.width += 1 << 6; } /* 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. */ if (freetype_set_outline_text_metrics(base_font) || freetype_set_bitmap_text_metrics(base_font)) { TEXTMETRICW *ptm = &base_font->otm.otmTextMetrics; INT top = min( metrics.horiBearingY, ptm->tmAscent << 6 ); INT bottom = max( metrics.horiBearingY - metrics.height, -(ptm->tmDescent << 6) ); 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 ); */ } bbox = get_transformed_bbox( &metrics, needsTransform, matrices ); compute_metrics( base_font, font, bbox, &metrics, tategaki, vertical_metrics, needsTransform, matrices, lpgm, abc ); switch (format) { case GGO_METRICS: return 1; /* FIXME */ case GGO_BITMAP: return get_mono_glyph_bitmap( ft_face->glyph, bbox, font->fake_bold, needsTransform, matrices, buflen, buf ); case GGO_GRAY2_BITMAP: case GGO_GRAY4_BITMAP: case GGO_GRAY8_BITMAP: case WINE_GGO_GRAY16_BITMAP: return get_antialias_glyph_bitmap( ft_face->glyph, bbox, format, font->fake_bold, needsTransform, matrices, buflen, buf ); case WINE_GGO_HRGB_BITMAP: case WINE_GGO_HBGR_BITMAP: case WINE_GGO_VRGB_BITMAP: case WINE_GGO_VBGR_BITMAP: return get_subpixel_glyph_bitmap( ft_face->glyph, bbox, format, font->fake_bold, needsTransform, matrices, lpgm, buflen, buf ); case GGO_NATIVE: if (ft_face->glyph->format == ft_glyph_format_outline) { FT_Outline *outline = &ft_face->glyph->outline; UINT needed; if (buflen == 0) buf = NULL; if (needsTransform && buf) pFT_Outline_Transform( outline, &matrices[matrix_vert] ); needed = get_native_glyph_outline(outline, buflen, NULL); 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; case GGO_BEZIER: if (ft_face->glyph->format == ft_glyph_format_outline) { FT_Outline *outline = &ft_face->glyph->outline; UINT needed; if (buflen == 0) buf = NULL; if (needsTransform && buf) pFT_Outline_Transform( outline, &matrices[matrix_vert] ); needed = get_bezier_glyph_outline(outline, buflen, NULL); 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; default: FIXME("Unsupported format %d\n", format); return GDI_ERROR; } } /************************************************************* * freetype_set_bitmap_text_metrics */ static BOOL CDECL freetype_set_bitmap_text_metrics( struct gdi_font *font ) { FT_Face ft_face = get_ft_face( font ); FT_WinFNT_HeaderRec winfnt_header; if (font->otm.otmSize) return TRUE; /* already set */ font->otm.otmSize = offsetof( OUTLINETEXTMETRICW, otmFiller ); #define TM font->otm.otmTextMetrics if(!pFT_Get_WinFNT_Header(ft_face, &winfnt_header)) { 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; TM.tmDefaultChar = winfnt_header.default_char + winfnt_header.first_char; TM.tmBreakChar = winfnt_header.break_char + winfnt_header.first_char; 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 */ TM.tmPitchAndFamily = FT_IS_FIXED_WIDTH(ft_face) ? 0 : TMPF_FIXED_PITCH; TM.tmCharSet = font->charset; } TM.tmUnderlined = font->lf.lfUnderline ? 0xff : 0; TM.tmStruckOut = font->lf.lfStrikeOut ? 0xff : 0; if(font->fake_bold) TM.tmWeight = FW_BOLD; #undef TM return TRUE; } 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; } /************************************************************* * freetype_set_outline_text_metrics */ static BOOL CDECL freetype_set_outline_text_metrics( struct gdi_font *font ) { FT_Face ft_face = get_ft_face( font ); UINT needed; TT_OS2 *pOS2; TT_HoriHeader *pHori; TT_Postscript *pPost; FT_Fixed em_scale; INT ascent, descent; USHORT windescent; TRACE("font=%p\n", font); if (!font->scalable) return FALSE; if (font->otm.otmSize) return TRUE; /* already set */ /* 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() ))) { static const WCHAR fake_nameW[] = {'f','a','k','e',' ','n','a','m','e', 0}; 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); } 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); em_scale = (FT_Fixed)MulDiv(font->ppem, 1 << 16, ft_face->units_per_EM); pOS2 = pFT_Get_Sfnt_Table(ft_face, ft_sfnt_os2); if(!pOS2) { FIXME("Can't find OS/2 table - not TT font?\n"); return FALSE; } pHori = pFT_Get_Sfnt_Table(ft_face, ft_sfnt_hhea); if(!pHori) { FIXME("Can't find HHEA table - not TT font?\n"); return FALSE; } pPost = pFT_Get_Sfnt_Table(ft_face, ft_sfnt_post); /* we can live with this failing */ 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", pOS2->usWinAscent, pOS2->usWinDescent, pOS2->sTypoAscender, pOS2->sTypoDescender, pOS2->sTypoLineGap, pOS2->xAvgCharWidth, ft_face->ascender, ft_face->descender, ft_face->height, pHori->Ascender, pHori->Descender, pHori->Line_Gap, ft_face->bbox.yMax, ft_face->bbox.yMin); font->otm.otmSize = needed; #define TM font->otm.otmTextMetrics windescent = get_fixed_windescent(pOS2->usWinDescent); if(pOS2->usWinAscent + windescent == 0) { ascent = pHori->Ascender; descent = -pHori->Descender; } else { ascent = pOS2->usWinAscent; descent = windescent; } font->ntmCellHeight = ascent + descent; font->ntmAvgWidth = pOS2->xAvgCharWidth; #define SCALE_X(x) (pFT_MulFix(x, em_scale)) #define SCALE_Y(y) (pFT_MulFix(y, em_scale)) if(font->yMax) { TM.tmAscent = font->yMax; TM.tmDescent = -font->yMin; TM.tmInternalLeading = (TM.tmAscent + TM.tmDescent) - ft_face->size->metrics.y_ppem; } else { TM.tmAscent = SCALE_Y(ascent); TM.tmDescent = SCALE_Y(descent); TM.tmInternalLeading = SCALE_Y(ascent + descent - ft_face->units_per_EM); } TM.tmHeight = TM.tmAscent + TM.tmDescent; /* MSDN says: el = MAX(0, LineGap - ((WinAscent + WinDescent) - (Ascender - Descender))) */ TM.tmExternalLeading = max(0, SCALE_Y(pHori->Line_Gap - ((ascent + descent) - (pHori->Ascender - pHori->Descender)))); TM.tmAveCharWidth = SCALE_X(pOS2->xAvgCharWidth); if (TM.tmAveCharWidth == 0) { TM.tmAveCharWidth = 1; } TM.tmMaxCharWidth = SCALE_X(ft_face->bbox.xMax - ft_face->bbox.xMin); TM.tmWeight = FW_REGULAR; if (font->fake_bold) 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; } TM.tmOverhang = 0; TM.tmDigitizedAspectX = 96; /* FIXME */ TM.tmDigitizedAspectY = 96; /* FIXME */ /* It appears that for fonts with SYMBOL_CHARSET Windows always sets * symbol range to 0 - f0ff */ if (face_has_symbol_charmap(ft_face) || (pOS2->usFirstCharIndex >= 0xf000 && pOS2->usFirstCharIndex < 0xf100)) { TM.tmFirstChar = 0; switch(GetACP()) { case 1255: /* Hebrew */ TM.tmLastChar = 0xf896; break; case 1257: /* Baltic */ TM.tmLastChar = 0xf8fd; break; default: TM.tmLastChar = 0xf0ff; } TM.tmBreakChar = 0x20; TM.tmDefaultChar = 0x1f; } else { 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; } 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; /* Yes TPMF_FIXED_PITCH is correct; braindead api */ if(!FT_IS_FIXED_WIDTH(ft_face) && (pOS2->version == 0xFFFFU || pOS2->panose[PAN_PROPORTION_INDEX] != PAN_PROP_MONOSPACED)) TM.tmPitchAndFamily = TMPF_FIXED_PITCH; else TM.tmPitchAndFamily = 0; switch(pOS2->panose[PAN_FAMILYTYPE_INDEX]) { case PAN_FAMILY_SCRIPT: TM.tmPitchAndFamily |= FF_SCRIPT; break; case PAN_FAMILY_DECORATIVE: TM.tmPitchAndFamily |= FF_DECORATIVE; break; case PAN_ANY: case PAN_NO_FIT: case PAN_FAMILY_TEXT_DISPLAY: case PAN_FAMILY_PICTORIAL: /* symbol fonts get treated as if they were text */ /* which is clearly not what the panose spec says. */ default: if(TM.tmPitchAndFamily == 0 || /* fixed */ pOS2->panose[PAN_PROPORTION_INDEX] == PAN_PROP_MONOSPACED) TM.tmPitchAndFamily = FF_MODERN; 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: case PAN_SERIF_FLARED: case PAN_SERIF_ROUNDED: TM.tmPitchAndFamily |= FF_SWISS; break; } } break; } if(FT_IS_SCALABLE(ft_face)) TM.tmPitchAndFamily |= TMPF_VECTOR; if(FT_IS_SFNT(ft_face)) { if (font->ntmFlags & NTM_PS_OPENTYPE) TM.tmPitchAndFamily |= TMPF_DEVICE; else TM.tmPitchAndFamily |= TMPF_TRUETYPE; } TM.tmCharSet = font->charset; 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; /* Only return valid bits that define embedding and subsetting restrictions */ 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); if(!pPost) { font->otm.otmsUnderscoreSize = 0; font->otm.otmsUnderscorePosition = 0; } else { font->otm.otmsUnderscoreSize = SCALE_Y(pPost->underlineThickness); font->otm.otmsUnderscorePosition = SCALE_Y(pPost->underlinePosition); } #undef SCALE_X #undef SCALE_Y #undef TM return TRUE; } /************************************************************* * freetype_get_char_width_info */ static BOOL CDECL freetype_get_char_width_info( struct gdi_font *font, struct char_width_info *info ) { FT_Face ft_face = get_ft_face( font ); TT_HoriHeader *pHori; TRACE("%p, %p\n", font, info); if ((pHori = pFT_Get_Sfnt_Table(ft_face, ft_sfnt_hhea))) { FT_Fixed em_scale = MulDiv(font->ppem, 1 << 16, ft_face->units_per_EM); info->lsb = (SHORT)pFT_MulFix(pHori->min_Left_Side_Bearing, em_scale); info->rsb = (SHORT)pFT_MulFix(pHori->min_Right_Side_Bearing, em_scale); return TRUE; } return FALSE; } /************************************************************* * freetype_get_unicode_ranges * * Retrieve a list of supported Unicode ranges for a given font. * Can be called with NULL gs to calculate the buffer size. Returns * the number of ranges found. */ static DWORD CDECL freetype_get_unicode_ranges( struct gdi_font *font, GLYPHSET *gs ) { FT_Face ft_face = get_ft_face( font ); DWORD num_ranges = 0; if (ft_face->charmap->encoding == FT_ENCODING_UNICODE) { FT_UInt glyph_code; FT_ULong char_code, char_code_prev; glyph_code = 0; char_code_prev = char_code = pFT_Get_First_Char(ft_face, &glyph_code); TRACE("face encoding FT_ENCODING_UNICODE, number of glyphs %ld, first glyph %u, first char %04lx\n", ft_face->num_glyphs, glyph_code, char_code); 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; char_code = pFT_Get_Next_Char(ft_face, char_code, &glyph_code); } } else { DWORD encoding = RtlUlongByteSwap(ft_face->charmap->encoding); FIXME("encoding %s not supported\n", debugstr_an((char *)&encoding, 4)); } return num_ranges; } /************************************************************************* * 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; }; static DWORD parse_format0_kern_subtable(struct gdi_font *font, const struct TT_format0_kern_subtable *tt_f0_ks, const USHORT *glyph_to_char, KERNINGPAIR *kern_pair, DWORD cPairs) { FT_Face ft_face = get_ft_face( font ); USHORT i, nPairs; const struct TT_kern_pair *tt_kern_pair; TRACE("font height %d, units_per_EM %d\n", font->ppem, ft_face->units_per_EM); 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)]; /* this algorithm appears to better match what Windows does */ kern_pair->iKernAmount = (short)GET_BE_WORD(tt_kern_pair[i].value) * font->ppem; if (kern_pair->iKernAmount < 0) { kern_pair->iKernAmount -= ft_face->units_per_EM / 2; kern_pair->iKernAmount -= font->ppem; } else if (kern_pair->iKernAmount > 0) { kern_pair->iKernAmount += ft_face->units_per_EM / 2; kern_pair->iKernAmount += font->ppem; } kern_pair->iKernAmount /= ft_face->units_per_EM; 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; } /************************************************************* * freetype_get_kerning_pairs */ static DWORD CDECL freetype_get_kerning_pairs( struct gdi_font *font, KERNINGPAIR **pairs ) { FT_Face ft_face = get_ft_face( font ); DWORD length, count = 0; 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; length = freetype_get_font_data(font, MS_KERN_TAG, 0, NULL, 0); if (length == GDI_ERROR) { TRACE("no kerning data in the font\n"); return 0; } buf = HeapAlloc(GetProcessHeap(), 0, length); if (!buf) return 0; freetype_get_font_data(font, MS_KERN_TAG, 0, buf, length); /* 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; } if (ft_face->charmap->encoding == FT_ENCODING_UNICODE) { FT_UInt glyph_code; FT_ULong char_code; glyph_code = 0; char_code = pFT_Get_First_Char(ft_face, &glyph_code); TRACE("face encoding FT_ENCODING_UNICODE, number of glyphs %ld, first glyph %u, first char %lu\n", ft_face->num_glyphs, glyph_code, char_code); 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; char_code = pFT_Get_Next_Char(ft_face, char_code, &glyph_code); } } else { DWORD encoding = RtlUlongByteSwap(ft_face->charmap->encoding); ULONG n; FIXME("encoding %s not supported\n", debugstr_an((char *)&encoding, 4)); 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) { DWORD new_chunk, old_total = count; new_chunk = parse_format0_kern_subtable(font, (const struct TT_format0_kern_subtable *)(tt_kern_subtable + 1), glyph_to_char, NULL, 0); count += new_chunk; if (!*pairs) *pairs = HeapAlloc(GetProcessHeap(), 0, count * sizeof(**pairs)); else *pairs = HeapReAlloc(GetProcessHeap(), 0, *pairs, count * sizeof(**pairs)); parse_format0_kern_subtable(font, (const struct TT_format0_kern_subtable *)(tt_kern_subtable + 1), glyph_to_char, *pairs + old_total, new_chunk); } 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); return count; } static const struct font_backend_funcs font_funcs = { freetype_SelectFont, freetype_add_font, freetype_add_mem_font, freetype_remove_font, freetype_load_font, freetype_get_font_data, freetype_get_glyph_index, freetype_get_default_glyph, freetype_get_glyph_outline, freetype_get_unicode_ranges, freetype_get_char_width_info, freetype_set_outline_text_metrics, freetype_set_bitmap_text_metrics, freetype_get_kerning_pairs, freetype_destroy_font }; #else /* HAVE_FREETYPE */ /*************************************************************************/ BOOL WineEngInit( const struct font_backend_funcs **funcs ) { return FALSE; } #endif /* HAVE_FREETYPE */