3301 lines
106 KiB
C
3301 lines
106 KiB
C
/*
|
|
* FreeType font engine interface
|
|
*
|
|
* Copyright 2001 Huw D M Davies 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*/
|
|
|
|
#include "config.h"
|
|
#include "wine/port.h"
|
|
|
|
#include <stdarg.h>
|
|
#include <stdlib.h>
|
|
#include <sys/stat.h>
|
|
#include <string.h>
|
|
#include <dirent.h>
|
|
#include <stdio.h>
|
|
#include <assert.h>
|
|
|
|
#include "windef.h"
|
|
#include "winbase.h"
|
|
#include "winerror.h"
|
|
#include "winreg.h"
|
|
#include "wingdi.h"
|
|
#include "gdi.h"
|
|
#include "gdi_private.h"
|
|
#include "wine/unicode.h"
|
|
#include "wine/debug.h"
|
|
#include "wine/list.h"
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(font);
|
|
|
|
#ifdef HAVE_FREETYPE
|
|
|
|
#ifdef HAVE_FT2BUILD_H
|
|
#include <ft2build.h>
|
|
#endif
|
|
#ifdef HAVE_FREETYPE_FREETYPE_H
|
|
#include <freetype/freetype.h>
|
|
#endif
|
|
#ifdef HAVE_FREETYPE_FTGLYPH_H
|
|
#include <freetype/ftglyph.h>
|
|
#endif
|
|
#ifdef HAVE_FREETYPE_TTTABLES_H
|
|
#include <freetype/tttables.h>
|
|
#endif
|
|
#ifdef HAVE_FREETYPE_FTSNAMES_H
|
|
#include <freetype/ftsnames.h>
|
|
#else
|
|
# ifdef HAVE_FREETYPE_FTNAMES_H
|
|
# include <freetype/ftnames.h>
|
|
# endif
|
|
#endif
|
|
#ifdef HAVE_FREETYPE_TTNAMEID_H
|
|
#include <freetype/ttnameid.h>
|
|
#endif
|
|
#ifdef HAVE_FREETYPE_FTOUTLN_H
|
|
#include <freetype/ftoutln.h>
|
|
#endif
|
|
#ifdef HAVE_FREETYPE_INTERNAL_SFNT_H
|
|
#include <freetype/internal/sfnt.h>
|
|
#endif
|
|
#ifdef HAVE_FREETYPE_FTTRIGON_H
|
|
#include <freetype/fttrigon.h>
|
|
#endif
|
|
#ifdef HAVE_FREETYPE_FTWINFNT_H
|
|
#include <freetype/ftwinfnt.h>
|
|
#endif
|
|
|
|
#ifndef SONAME_LIBFREETYPE
|
|
#define SONAME_LIBFREETYPE "libfreetype.so"
|
|
#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;
|
|
|
|
static void *ft_handle = NULL;
|
|
|
|
#define MAKE_FUNCPTR(f) static typeof(f) * p##f = NULL
|
|
MAKE_FUNCPTR(FT_Vector_Unit);
|
|
MAKE_FUNCPTR(FT_Done_Face);
|
|
MAKE_FUNCPTR(FT_Get_Char_Index);
|
|
MAKE_FUNCPTR(FT_Get_Sfnt_Table);
|
|
MAKE_FUNCPTR(FT_Init_FreeType);
|
|
MAKE_FUNCPTR(FT_Load_Glyph);
|
|
MAKE_FUNCPTR(FT_Matrix_Multiply);
|
|
MAKE_FUNCPTR(FT_MulFix);
|
|
MAKE_FUNCPTR(FT_New_Face);
|
|
MAKE_FUNCPTR(FT_Outline_Get_Bitmap);
|
|
MAKE_FUNCPTR(FT_Outline_Transform);
|
|
MAKE_FUNCPTR(FT_Outline_Translate);
|
|
MAKE_FUNCPTR(FT_Select_Charmap);
|
|
MAKE_FUNCPTR(FT_Set_Pixel_Sizes);
|
|
MAKE_FUNCPTR(FT_Vector_Transform);
|
|
static void (*pFT_Library_Version)(FT_Library,FT_Int*,FT_Int*,FT_Int*);
|
|
static FT_Error (*pFT_Load_Sfnt_Table)(FT_Face,FT_ULong,FT_Long,FT_Byte*,FT_ULong*);
|
|
static FT_ULong (*pFT_Get_First_Char)(FT_Face,FT_UInt*);
|
|
#ifdef HAVE_FREETYPE_FTWINFNT_H
|
|
MAKE_FUNCPTR(FT_Get_WinFNT_Header);
|
|
#endif
|
|
|
|
#ifdef HAVE_FONTCONFIG_FONTCONFIG_H
|
|
#include <fontconfig/fontconfig.h>
|
|
MAKE_FUNCPTR(FcConfigGetCurrent);
|
|
MAKE_FUNCPTR(FcFontList);
|
|
MAKE_FUNCPTR(FcFontSetDestroy);
|
|
MAKE_FUNCPTR(FcInit);
|
|
MAKE_FUNCPTR(FcObjectSetAdd);
|
|
MAKE_FUNCPTR(FcObjectSetCreate);
|
|
MAKE_FUNCPTR(FcObjectSetDestroy);
|
|
MAKE_FUNCPTR(FcPatternCreate);
|
|
MAKE_FUNCPTR(FcPatternDestroy);
|
|
MAKE_FUNCPTR(FcPatternGet);
|
|
#ifndef SONAME_LIBFONTCONFIG
|
|
#define SONAME_LIBFONTCONFIG "libfontconfig.so"
|
|
#endif
|
|
#endif
|
|
|
|
#undef MAKE_FUNCPTR
|
|
|
|
#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
|
|
|
|
#define GET_BE_WORD(ptr) MAKEWORD( ((BYTE *)(ptr))[1], ((BYTE *)(ptr))[0] )
|
|
|
|
/* This is bascially a copy of FT_Bitmap_Size with an extra element added */
|
|
typedef struct {
|
|
FT_Short height;
|
|
FT_Short width;
|
|
FT_Pos size;
|
|
FT_Pos x_ppem;
|
|
FT_Pos y_ppem;
|
|
FT_Short internal_leading;
|
|
} Bitmap_Size;
|
|
|
|
/* 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 tagFace {
|
|
struct list entry;
|
|
WCHAR *StyleName;
|
|
char *file;
|
|
FT_Long face_index;
|
|
BOOL Italic;
|
|
BOOL Bold;
|
|
FONTSIGNATURE fs;
|
|
FT_Fixed font_version;
|
|
BOOL scalable;
|
|
Bitmap_Size size; /* set if face is a bitmap */
|
|
BOOL external; /* TRUE if we should manually add this font to the registry */
|
|
struct tagFamily *family;
|
|
} Face;
|
|
|
|
typedef struct tagFamily {
|
|
struct list entry;
|
|
WCHAR *FamilyName;
|
|
struct list faces;
|
|
} Family;
|
|
|
|
typedef struct {
|
|
GLYPHMETRICS gm;
|
|
INT adv; /* These three hold to widths of the unrotated chars */
|
|
INT lsb;
|
|
INT bbx;
|
|
BOOL init;
|
|
} GM;
|
|
|
|
typedef struct {
|
|
FLOAT eM11, eM12;
|
|
FLOAT eM21, eM22;
|
|
} FMAT2;
|
|
|
|
typedef struct {
|
|
DWORD hash;
|
|
LOGFONTW lf;
|
|
FMAT2 matrix;
|
|
} FONT_DESC;
|
|
|
|
typedef struct tagHFONTLIST {
|
|
struct list entry;
|
|
HFONT hfont;
|
|
} HFONTLIST;
|
|
|
|
struct tagGdiFont {
|
|
struct list entry;
|
|
FT_Face ft_face;
|
|
LPWSTR name;
|
|
int charset;
|
|
int codepage;
|
|
BOOL fake_italic;
|
|
BOOL fake_bold;
|
|
BYTE underline;
|
|
BYTE strikeout;
|
|
INT orientation;
|
|
GM *gm;
|
|
DWORD gmsize;
|
|
struct list hfontlist;
|
|
FONT_DESC font_desc;
|
|
LONG aveWidth;
|
|
SHORT yMax;
|
|
SHORT yMin;
|
|
OUTLINETEXTMETRICW *potm;
|
|
FONTSIGNATURE fs;
|
|
};
|
|
|
|
#define INIT_GM_SIZE 128
|
|
|
|
static struct list gdi_font_list = LIST_INIT(gdi_font_list);
|
|
static struct list unused_gdi_font_list = LIST_INIT(unused_gdi_font_list);
|
|
#define UNUSED_CACHE_SIZE 10
|
|
|
|
static struct list font_list = LIST_INIT(font_list);
|
|
|
|
static const WCHAR defSerif[] = {'T','i','m','e','s',' ','N','e','w',' ',
|
|
'R','o','m','a','n','\0'};
|
|
static const WCHAR defSans[] = {'A','r','i','a','l','\0'};
|
|
static const WCHAR defFixed[] = {'C','o','u','r','i','e','r',' ','N','e','w','\0'};
|
|
|
|
static const WCHAR defSystem[] = {'A','r','i','a','l','\0'};
|
|
static const WCHAR SystemW[] = {'S','y','s','t','e','m','\0'};
|
|
static const WCHAR MSSansSerifW[] = {'M','S',' ','S','a','n','s',' ',
|
|
'S','e','r','i','f','\0'};
|
|
static const WCHAR HelvW[] = {'H','e','l','v','\0'};
|
|
static const WCHAR RegularW[] = {'R','e','g','u','l','a','r','\0'};
|
|
|
|
static const WCHAR fontsW[] = {'\\','F','o','n','t','s','\0'};
|
|
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 system_fonts_reg_key[] = {'S','o','f','t','w','a','r','e','\\','F','o','n','t','s','\0'};
|
|
static const WCHAR FixedSys_Value[] = {'F','I','X','E','D','F','O','N','.','F','O','N','\0'};
|
|
static const WCHAR System_Value[] = {'F','O','N','T','S','.','F','O','N','\0'};
|
|
static const WCHAR OEMFont_Value[] = {'O','E','M','F','O','N','T','.','F','O','N','\0'};
|
|
|
|
static const WCHAR *SystemFontValues[4] = {
|
|
System_Value,
|
|
OEMFont_Value,
|
|
FixedSys_Value,
|
|
NULL
|
|
};
|
|
|
|
static const WCHAR external_fonts_reg_key[] = {'S','o','f','t','w','a','r','e','\\','W','i','n','e','\\','W','i','n','e','\\',
|
|
'F','o','n','t','s','\\','E','x','t','e','r','n','a','l',' ','F','o','n','t','s','\0'};
|
|
|
|
static const WCHAR ArabicW[] = {'A','r','a','b','i','c','\0'};
|
|
static const WCHAR BalticW[] = {'B','a','l','t','i','c','\0'};
|
|
static const WCHAR CHINESE_BIG5W[] = {'C','H','I','N','E','S','E','_','B','I','G','5','\0'};
|
|
static const WCHAR CHINESE_GB2312W[] = {'C','H','I','N','E','S','E','_','G','B','2','3','1','2','\0'};
|
|
static const WCHAR Central_EuropeanW[] = {'C','e','n','t','r','a','l',' ',
|
|
'E','u','r','o','p','e','a','n','\0'};
|
|
static const WCHAR CyrillicW[] = {'C','y','r','i','l','l','i','c','\0'};
|
|
static const WCHAR GreekW[] = {'G','r','e','e','k','\0'};
|
|
static const WCHAR HangulW[] = {'H','a','n','g','u','l','\0'};
|
|
static const WCHAR Hangul_Johab_W[] = {'H','a','n','g','u','l','(','J','o','h','a','b',')','\0'};
|
|
static const WCHAR HebrewW[] = {'H','e','b','r','e','w','\0'};
|
|
static const WCHAR JapaneseW[] = {'J','a','p','a','n','e','s','e','\0'};
|
|
static const WCHAR SymbolW[] = {'S','y','m','b','o','l','\0'};
|
|
static const WCHAR ThaiW[] = {'T','h','a','i','\0'};
|
|
static const WCHAR TurkishW[] = {'T','u','r','k','i','s','h','\0'};
|
|
static const WCHAR VietnameseW[] = {'V','i','e','t','n','a','m','e','s','e','\0'};
|
|
static const WCHAR WesternW[] = {'W','e','s','t','e','r','n','\0'};
|
|
static const WCHAR OEM_DOSW[] = {'O','E','M','/','D','O','S','\0'};
|
|
|
|
static const WCHAR *ElfScriptsW[32] = { /* these are in the order of the fsCsb[0] bits */
|
|
WesternW, /*00*/
|
|
Central_EuropeanW,
|
|
CyrillicW,
|
|
GreekW,
|
|
TurkishW,
|
|
HebrewW,
|
|
ArabicW,
|
|
BalticW,
|
|
VietnameseW, /*08*/
|
|
NULL, NULL, NULL, NULL, NULL, NULL, NULL, /*15*/
|
|
ThaiW,
|
|
JapaneseW,
|
|
CHINESE_GB2312W,
|
|
HangulW,
|
|
CHINESE_BIG5W,
|
|
Hangul_Johab_W,
|
|
NULL, NULL, /*23*/
|
|
NULL, NULL, NULL, NULL, NULL, NULL, NULL,
|
|
SymbolW /*31*/
|
|
};
|
|
|
|
typedef struct {
|
|
WCHAR *name;
|
|
INT charset;
|
|
} NameCs;
|
|
|
|
typedef struct tagFontSubst {
|
|
NameCs from;
|
|
NameCs to;
|
|
struct tagFontSubst *next;
|
|
} FontSubst;
|
|
|
|
static FontSubst *substlist = NULL;
|
|
static BOOL have_installed_roman_font = FALSE; /* CreateFontInstance will fail if this is still FALSE */
|
|
|
|
static const WCHAR font_mutex_nameW[] = {'_','_','W','I','N','E','_','F','O','N','T','_','M','U','T','E','X','_','_','\0'};
|
|
|
|
|
|
/****************************************
|
|
* 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\\Microsft\\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
|
|
*/
|
|
|
|
|
|
static inline BOOL is_win9x(void)
|
|
{
|
|
return GetVersion() & 0x80000000;
|
|
}
|
|
/*
|
|
This function builds an FT_Fixed from a float. It puts the integer part
|
|
in the highest 16 bits and the decimal part in the lowest 16 bits of the FT_Fixed.
|
|
It fails if the integer part of the float number is greater than SHORT_MAX.
|
|
*/
|
|
static inline FT_Fixed FT_FixedFromFloat(float f)
|
|
{
|
|
short value = f;
|
|
unsigned short fract = (f - value) * 0xFFFF;
|
|
return (FT_Fixed)((long)value << 16 | (unsigned long)fract);
|
|
}
|
|
|
|
/*
|
|
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)((long)f.value << 16 | (unsigned long)f.fract);
|
|
}
|
|
|
|
#define ADDFONT_EXTERNAL_FONT 0x01
|
|
#define ADDFONT_FORCE_BITMAP 0x02
|
|
static BOOL AddFontFileToList(const char *file, char *fake_family, DWORD flags)
|
|
{
|
|
FT_Face ft_face;
|
|
TT_OS2 *pOS2;
|
|
TT_Header *pHeader = NULL;
|
|
WCHAR *FamilyW, *StyleW;
|
|
DWORD len;
|
|
Family *family;
|
|
Face *face;
|
|
struct list *family_elem_ptr, *face_elem_ptr;
|
|
FT_Error err;
|
|
FT_Long face_index = 0, num_faces;
|
|
#ifdef HAVE_FREETYPE_FTWINFNT_H
|
|
FT_WinFNT_HeaderRec winfnt_header;
|
|
#endif
|
|
int i, bitmap_num;
|
|
|
|
do {
|
|
char *family_name = fake_family;
|
|
|
|
TRACE("Loading font file %s index %ld\n", debugstr_a(file), face_index);
|
|
if((err = pFT_New_Face(library, file, face_index, &ft_face)) != 0) {
|
|
WARN("Unable to load font file %s err = %x\n", debugstr_a(file), err);
|
|
return FALSE;
|
|
}
|
|
|
|
if(!FT_IS_SFNT(ft_face) && (FT_IS_SCALABLE(ft_face) || !(flags & ADDFONT_FORCE_BITMAP))) { /* for now we'll accept TT/OT or bitmap fonts*/
|
|
pFT_Done_Face(ft_face);
|
|
return FALSE;
|
|
}
|
|
|
|
/* There are too many bugs in FreeType < 2.1.9 for bitmap font support */
|
|
if(!FT_IS_SCALABLE(ft_face) && FT_SimpleVersion < ((2 << 16) | (1 << 8) | (9 << 0))) {
|
|
pFT_Done_Face(ft_face);
|
|
return FALSE;
|
|
}
|
|
|
|
if(FT_IS_SFNT(ft_face) && (!pFT_Get_Sfnt_Table(ft_face, ft_sfnt_os2) ||
|
|
!pFT_Get_Sfnt_Table(ft_face, ft_sfnt_hhea) ||
|
|
!(pHeader = pFT_Get_Sfnt_Table(ft_face, ft_sfnt_head)))) {
|
|
TRACE("Font file %s lacks either an OS2, HHEA or HEAD table.\n"
|
|
"Skipping this font.\n", debugstr_a(file));
|
|
pFT_Done_Face(ft_face);
|
|
return FALSE;
|
|
}
|
|
|
|
if(!ft_face->family_name || !ft_face->style_name) {
|
|
TRACE("Font file %s lacks either a family or style name\n", debugstr_a(file));
|
|
pFT_Done_Face(ft_face);
|
|
return FALSE;
|
|
}
|
|
|
|
if(!family_name)
|
|
family_name = ft_face->family_name;
|
|
|
|
bitmap_num = 0;
|
|
do {
|
|
My_FT_Bitmap_Size *size = NULL;
|
|
|
|
if(!FT_IS_SCALABLE(ft_face))
|
|
size = (My_FT_Bitmap_Size *)ft_face->available_sizes + bitmap_num;
|
|
|
|
len = MultiByteToWideChar(CP_ACP, 0, family_name, -1, NULL, 0);
|
|
FamilyW = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR));
|
|
MultiByteToWideChar(CP_ACP, 0, family_name, -1, FamilyW, len);
|
|
|
|
family = NULL;
|
|
LIST_FOR_EACH(family_elem_ptr, &font_list) {
|
|
family = LIST_ENTRY(family_elem_ptr, Family, entry);
|
|
if(!strcmpW(family->FamilyName, FamilyW))
|
|
break;
|
|
family = NULL;
|
|
}
|
|
if(!family) {
|
|
family = HeapAlloc(GetProcessHeap(), 0, sizeof(*family));
|
|
family->FamilyName = FamilyW;
|
|
list_init(&family->faces);
|
|
list_add_tail(&font_list, &family->entry);
|
|
} else {
|
|
HeapFree(GetProcessHeap(), 0, FamilyW);
|
|
}
|
|
|
|
len = MultiByteToWideChar(CP_ACP, 0, ft_face->style_name, -1, NULL, 0);
|
|
StyleW = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR));
|
|
MultiByteToWideChar(CP_ACP, 0, ft_face->style_name, -1, StyleW, len);
|
|
|
|
face_elem_ptr = list_head(&family->faces);
|
|
while(face_elem_ptr) {
|
|
face = LIST_ENTRY(face_elem_ptr, Face, entry);
|
|
face_elem_ptr = list_next(&family->faces, face_elem_ptr);
|
|
if(!strcmpW(face->StyleName, StyleW) &&
|
|
(FT_IS_SCALABLE(ft_face) || (size->y_ppem == face->size.y_ppem))) {
|
|
TRACE("Already loaded font %s %s original version is %lx, this version is %lx\n",
|
|
debugstr_w(family->FamilyName), debugstr_w(StyleW),
|
|
face->font_version, pHeader ? pHeader->Font_Revision : 0);
|
|
|
|
if(fake_family) {
|
|
TRACE("This font is a replacement but the original really exists, so we'll skip the replacement\n");
|
|
HeapFree(GetProcessHeap(), 0, StyleW);
|
|
pFT_Done_Face(ft_face);
|
|
return FALSE;
|
|
}
|
|
if(!pHeader || pHeader->Font_Revision <= face->font_version) {
|
|
TRACE("Original font is newer so skipping this one\n");
|
|
HeapFree(GetProcessHeap(), 0, StyleW);
|
|
pFT_Done_Face(ft_face);
|
|
return FALSE;
|
|
} else {
|
|
TRACE("Replacing original with this one\n");
|
|
list_remove(&face->entry);
|
|
HeapFree(GetProcessHeap(), 0, face->file);
|
|
HeapFree(GetProcessHeap(), 0, face->StyleName);
|
|
HeapFree(GetProcessHeap(), 0, face);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
face = HeapAlloc(GetProcessHeap(), 0, sizeof(*face));
|
|
list_add_tail(&family->faces, &face->entry);
|
|
face->StyleName = StyleW;
|
|
face->file = HeapAlloc(GetProcessHeap(),0,strlen(file)+1);
|
|
strcpy(face->file, file);
|
|
face->face_index = face_index;
|
|
face->Italic = (ft_face->style_flags & FT_STYLE_FLAG_ITALIC) ? 1 : 0;
|
|
face->Bold = (ft_face->style_flags & FT_STYLE_FLAG_BOLD) ? 1 : 0;
|
|
face->font_version = pHeader ? pHeader->Font_Revision : 0;
|
|
face->family = family;
|
|
face->external = (flags & ADDFONT_EXTERNAL_FONT) ? TRUE : FALSE;
|
|
|
|
if(FT_IS_SCALABLE(ft_face)) {
|
|
memset(&face->size, 0, sizeof(face->size));
|
|
face->scalable = TRUE;
|
|
} else {
|
|
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;
|
|
face->size.internal_leading = 0;
|
|
face->scalable = FALSE;
|
|
}
|
|
|
|
memset(&face->fs, 0, sizeof(face->fs));
|
|
|
|
pOS2 = pFT_Get_Sfnt_Table(ft_face, ft_sfnt_os2);
|
|
if(pOS2) {
|
|
face->fs.fsCsb[0] = pOS2->ulCodePageRange1;
|
|
face->fs.fsCsb[1] = pOS2->ulCodePageRange2;
|
|
face->fs.fsUsb[0] = pOS2->ulUnicodeRange1;
|
|
face->fs.fsUsb[1] = pOS2->ulUnicodeRange2;
|
|
face->fs.fsUsb[2] = pOS2->ulUnicodeRange3;
|
|
face->fs.fsUsb[3] = pOS2->ulUnicodeRange4;
|
|
if(pOS2->version == 0) {
|
|
FT_UInt dummy;
|
|
|
|
if(!pFT_Get_First_Char || (pFT_Get_First_Char( ft_face, &dummy ) < 0x100))
|
|
face->fs.fsCsb[0] |= 1;
|
|
else
|
|
face->fs.fsCsb[0] |= 1L << 31;
|
|
}
|
|
}
|
|
#ifdef HAVE_FREETYPE_FTWINFNT_H
|
|
else if(pFT_Get_WinFNT_Header && !pFT_Get_WinFNT_Header(ft_face, &winfnt_header)) {
|
|
CHARSETINFO csi;
|
|
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)winfnt_header.charset, &csi, TCI_SRCCHARSET))
|
|
memcpy(&face->fs, &csi.fs, sizeof(csi.fs));
|
|
face->size.internal_leading = winfnt_header.internal_leading;
|
|
}
|
|
#endif
|
|
TRACE("fsCsb = %08lx %08lx/%08lx %08lx %08lx %08lx\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(face->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:
|
|
face->fs.fsCsb[0] |= 1;
|
|
break;
|
|
case FT_ENCODING_MS_SYMBOL:
|
|
face->fs.fsCsb[0] |= 1L << 31;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if(face->fs.fsCsb[0] & ~(1L << 31))
|
|
have_installed_roman_font = TRUE;
|
|
} while(!FT_IS_SCALABLE(ft_face) && ++bitmap_num < ft_face->num_fixed_sizes);
|
|
|
|
num_faces = ft_face->num_faces;
|
|
pFT_Done_Face(ft_face);
|
|
TRACE("Added font %s %s\n", debugstr_w(family->FamilyName),
|
|
debugstr_w(StyleW));
|
|
} while(num_faces > ++face_index);
|
|
return TRUE;
|
|
}
|
|
|
|
static void DumpFontList(void)
|
|
{
|
|
Family *family;
|
|
Face *face;
|
|
struct list *family_elem_ptr, *face_elem_ptr;
|
|
|
|
LIST_FOR_EACH(family_elem_ptr, &font_list) {
|
|
family = LIST_ENTRY(family_elem_ptr, Family, entry);
|
|
TRACE("Family: %s\n", debugstr_w(family->FamilyName));
|
|
LIST_FOR_EACH(face_elem_ptr, &family->faces) {
|
|
face = LIST_ENTRY(face_elem_ptr, Face, entry);
|
|
TRACE("\t%s", debugstr_w(face->StyleName));
|
|
if(!face->scalable)
|
|
TRACE(" %ld", face->size.y_ppem >> 6);
|
|
TRACE("\n");
|
|
}
|
|
}
|
|
return;
|
|
}
|
|
|
|
static void DumpSubstList(void)
|
|
{
|
|
FontSubst *psub;
|
|
|
|
for(psub = substlist; psub; psub = psub->next)
|
|
if(psub->from.charset != -1 || psub->to.charset != -1)
|
|
TRACE("%s:%d -> %s:%d\n", debugstr_w(psub->from.name),
|
|
psub->from.charset, debugstr_w(psub->to.name), psub->to.charset);
|
|
else
|
|
TRACE("%s -> %s\n", debugstr_w(psub->from.name),
|
|
debugstr_w(psub->to.name));
|
|
return;
|
|
}
|
|
|
|
static LPWSTR strdupW(LPWSTR p)
|
|
{
|
|
LPWSTR ret;
|
|
DWORD len = (strlenW(p) + 1) * sizeof(WCHAR);
|
|
ret = HeapAlloc(GetProcessHeap(), 0, len);
|
|
memcpy(ret, p, len);
|
|
return ret;
|
|
}
|
|
|
|
static void split_subst_info(NameCs *nc, LPSTR str)
|
|
{
|
|
CHAR *p = strrchr(str, ',');
|
|
DWORD len;
|
|
|
|
nc->charset = -1;
|
|
if(p && *(p+1)) {
|
|
nc->charset = strtol(p+1, NULL, 10);
|
|
*p = '\0';
|
|
}
|
|
len = MultiByteToWideChar(CP_ACP, 0, str, -1, NULL, 0);
|
|
nc->name = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR));
|
|
MultiByteToWideChar(CP_ACP, 0, str, -1, nc->name, len);
|
|
}
|
|
|
|
static void LoadSubstList(void)
|
|
{
|
|
FontSubst *psub, **ppsub;
|
|
HKEY hkey;
|
|
DWORD valuelen, datalen, i = 0, type, dlen, vlen;
|
|
LPSTR value;
|
|
LPVOID data;
|
|
|
|
if(substlist) {
|
|
for(psub = substlist; psub;) {
|
|
FontSubst *ptmp;
|
|
HeapFree(GetProcessHeap(), 0, psub->to.name);
|
|
HeapFree(GetProcessHeap(), 0, psub->from.name);
|
|
ptmp = psub;
|
|
psub = psub->next;
|
|
HeapFree(GetProcessHeap(), 0, ptmp);
|
|
}
|
|
substlist = NULL;
|
|
}
|
|
|
|
if(RegOpenKeyA(HKEY_LOCAL_MACHINE,
|
|
"Software\\Microsoft\\Windows NT\\CurrentVersion\\FontSubstitutes",
|
|
&hkey) == ERROR_SUCCESS) {
|
|
|
|
RegQueryInfoKeyA(hkey, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
|
|
&valuelen, &datalen, NULL, NULL);
|
|
|
|
valuelen++; /* returned value doesn't include room for '\0' */
|
|
value = HeapAlloc(GetProcessHeap(), 0, valuelen * sizeof(CHAR));
|
|
data = HeapAlloc(GetProcessHeap(), 0, datalen);
|
|
|
|
dlen = datalen;
|
|
vlen = valuelen;
|
|
ppsub = &substlist;
|
|
while(RegEnumValueA(hkey, i++, value, &vlen, NULL, &type, data,
|
|
&dlen) == ERROR_SUCCESS) {
|
|
TRACE("Got %s=%s\n", debugstr_a(value), debugstr_a(data));
|
|
|
|
*ppsub = HeapAlloc(GetProcessHeap(), 0, sizeof(**ppsub));
|
|
(*ppsub)->next = NULL;
|
|
split_subst_info(&((*ppsub)->from), value);
|
|
split_subst_info(&((*ppsub)->to), data);
|
|
|
|
/* Win 2000 doesn't allow mapping between different charsets
|
|
or mapping of DEFAULT_CHARSET */
|
|
if(((*ppsub)->to.charset != (*ppsub)->from.charset) ||
|
|
(*ppsub)->to.charset == DEFAULT_CHARSET) {
|
|
HeapFree(GetProcessHeap(), 0, (*ppsub)->to.name);
|
|
HeapFree(GetProcessHeap(), 0, (*ppsub)->from.name);
|
|
HeapFree(GetProcessHeap(), 0, *ppsub);
|
|
*ppsub = NULL;
|
|
} else {
|
|
ppsub = &((*ppsub)->next);
|
|
}
|
|
/* reset dlen and vlen */
|
|
dlen = datalen;
|
|
vlen = valuelen;
|
|
}
|
|
HeapFree(GetProcessHeap(), 0, data);
|
|
HeapFree(GetProcessHeap(), 0, value);
|
|
RegCloseKey(hkey);
|
|
}
|
|
}
|
|
|
|
/***********************************************************
|
|
* The replacement list is a way to map an entire font
|
|
* family onto another family. For example adding
|
|
*
|
|
* [HKLM\Software\Wine\Wine\FontReplacements]
|
|
* "Wingdings"="Winedings"
|
|
*
|
|
* would enumerate the Winedings font both as Winedings and
|
|
* Wingdings. However if a real Wingdings font is present the
|
|
* replacement does not take place.
|
|
*
|
|
*/
|
|
static void LoadReplaceList(void)
|
|
{
|
|
HKEY hkey;
|
|
DWORD valuelen, datalen, i = 0, type, dlen, vlen;
|
|
LPSTR value;
|
|
LPVOID data;
|
|
Family *family;
|
|
Face *face;
|
|
struct list *family_elem_ptr, *face_elem_ptr;
|
|
WCHAR old_nameW[200];
|
|
|
|
if(RegOpenKeyA(HKEY_LOCAL_MACHINE,
|
|
"Software\\Wine\\Wine\\FontReplacements",
|
|
&hkey) == ERROR_SUCCESS) {
|
|
|
|
RegQueryInfoKeyA(hkey, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
|
|
&valuelen, &datalen, NULL, NULL);
|
|
|
|
valuelen++; /* returned value doesn't include room for '\0' */
|
|
value = HeapAlloc(GetProcessHeap(), 0, valuelen * sizeof(CHAR));
|
|
data = HeapAlloc(GetProcessHeap(), 0, datalen);
|
|
|
|
dlen = datalen;
|
|
vlen = valuelen;
|
|
while(RegEnumValueA(hkey, i++, value, &vlen, NULL, &type, data,
|
|
&dlen) == ERROR_SUCCESS) {
|
|
TRACE("Got %s=%s\n", debugstr_a(value), debugstr_a(data));
|
|
/* "NewName"="Oldname" */
|
|
if(!MultiByteToWideChar(CP_ACP, 0, data, -1, old_nameW, sizeof(old_nameW)))
|
|
break;
|
|
|
|
/* Find the old family and hence all of the font files
|
|
in that family */
|
|
LIST_FOR_EACH(family_elem_ptr, &font_list) {
|
|
family = LIST_ENTRY(family_elem_ptr, Family, entry);
|
|
if(!strcmpiW(family->FamilyName, old_nameW)) {
|
|
LIST_FOR_EACH(face_elem_ptr, &family->faces) {
|
|
face = LIST_ENTRY(face_elem_ptr, Face, entry);
|
|
TRACE("mapping %s %s to %s\n", debugstr_w(family->FamilyName),
|
|
debugstr_w(face->StyleName), value);
|
|
/* Now add a new entry with the new family name */
|
|
AddFontFileToList(face->file, value, ADDFONT_FORCE_BITMAP | (face->external ? ADDFONT_EXTERNAL_FONT : 0));
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
/* reset dlen and vlen */
|
|
dlen = datalen;
|
|
vlen = valuelen;
|
|
}
|
|
HeapFree(GetProcessHeap(), 0, data);
|
|
HeapFree(GetProcessHeap(), 0, value);
|
|
RegCloseKey(hkey);
|
|
}
|
|
}
|
|
|
|
|
|
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) {
|
|
ERR("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
|
|
AddFontFileToList(path, NULL, external_fonts ? ADDFONT_EXTERNAL_FONT : 0);
|
|
}
|
|
closedir(dir);
|
|
return TRUE;
|
|
}
|
|
|
|
static void load_fontconfig_fonts(void)
|
|
{
|
|
#ifdef HAVE_FONTCONFIG_FONTCONFIG_H
|
|
void *fc_handle = NULL;
|
|
FcConfig *config;
|
|
FcPattern *pat;
|
|
FcObjectSet *os;
|
|
FcFontSet *fontset;
|
|
FcValue v;
|
|
int i, len;
|
|
const char *ext;
|
|
|
|
fc_handle = wine_dlopen(SONAME_LIBFONTCONFIG, RTLD_NOW, NULL, 0);
|
|
if(!fc_handle) {
|
|
TRACE("Wine cannot find the fontconfig library (%s).\n",
|
|
SONAME_LIBFONTCONFIG);
|
|
return;
|
|
}
|
|
#define LOAD_FUNCPTR(f) if((p##f = wine_dlsym(fc_handle, #f, NULL, 0)) == NULL){WARN("Can't find symbol %s\n", #f); goto sym_not_found;}
|
|
LOAD_FUNCPTR(FcConfigGetCurrent);
|
|
LOAD_FUNCPTR(FcFontList);
|
|
LOAD_FUNCPTR(FcFontSetDestroy);
|
|
LOAD_FUNCPTR(FcInit);
|
|
LOAD_FUNCPTR(FcObjectSetAdd);
|
|
LOAD_FUNCPTR(FcObjectSetCreate);
|
|
LOAD_FUNCPTR(FcObjectSetDestroy);
|
|
LOAD_FUNCPTR(FcPatternCreate);
|
|
LOAD_FUNCPTR(FcPatternDestroy);
|
|
LOAD_FUNCPTR(FcPatternGet);
|
|
#undef LOAD_FUNCPTR
|
|
|
|
if(!pFcInit()) return;
|
|
|
|
config = pFcConfigGetCurrent();
|
|
pat = pFcPatternCreate();
|
|
os = pFcObjectSetCreate();
|
|
pFcObjectSetAdd(os, FC_FILE);
|
|
fontset = pFcFontList(config, pat, os);
|
|
if(!fontset) return;
|
|
for(i = 0; i < fontset->nfont; i++) {
|
|
if(pFcPatternGet(fontset->fonts[i], FC_FILE, 0, &v) != FcResultMatch)
|
|
continue;
|
|
if(v.type != FcTypeString) continue;
|
|
TRACE("fontconfig: %s\n", v.u.s);
|
|
|
|
/* We're just interested in OT/TT fonts for now, so this hack just
|
|
picks up the standard extensions to save time loading every other
|
|
font */
|
|
len = strlen(v.u.s);
|
|
if(len < 4) continue;
|
|
ext = v.u.s + len - 3;
|
|
if(!strcasecmp(ext, "ttf") || !strcasecmp(ext, "ttc") || !strcasecmp(ext, "otf"))
|
|
AddFontFileToList(v.u.s, NULL, ADDFONT_EXTERNAL_FONT);
|
|
}
|
|
pFcFontSetDestroy(fontset);
|
|
pFcObjectSetDestroy(os);
|
|
pFcPatternDestroy(pat);
|
|
sym_not_found:
|
|
#endif
|
|
return;
|
|
}
|
|
|
|
|
|
void load_system_fonts(void)
|
|
{
|
|
HKEY hkey;
|
|
WCHAR data[MAX_PATH], windowsdir[MAX_PATH], pathW[MAX_PATH];
|
|
const WCHAR **value;
|
|
DWORD dlen, type;
|
|
static const WCHAR fmtW[] = {'%','s','\\','%','s','\0'};
|
|
char *unixname;
|
|
|
|
if(RegOpenKeyW(HKEY_CURRENT_CONFIG, system_fonts_reg_key, &hkey) == ERROR_SUCCESS) {
|
|
GetWindowsDirectoryW(windowsdir, sizeof(windowsdir) / sizeof(WCHAR));
|
|
strcatW(windowsdir, fontsW);
|
|
for(value = SystemFontValues; *value; value++) {
|
|
dlen = sizeof(data);
|
|
if(RegQueryValueExW(hkey, *value, 0, &type, (void*)data, &dlen) == ERROR_SUCCESS &&
|
|
type == REG_SZ) {
|
|
sprintfW(pathW, fmtW, windowsdir, data);
|
|
if((unixname = wine_get_unix_file_name(pathW))) {
|
|
AddFontFileToList(unixname, NULL, ADDFONT_FORCE_BITMAP);
|
|
HeapFree(GetProcessHeap(), 0, unixname);
|
|
}
|
|
}
|
|
}
|
|
RegCloseKey(hkey);
|
|
}
|
|
}
|
|
|
|
/*************************************************************
|
|
*
|
|
* This adds registry entries for any externally loaded fonts
|
|
* (fonts from fontconfig or FontDirs). It also deletes entries
|
|
* of no longer existing fonts.
|
|
*
|
|
*/
|
|
void update_reg_entries(void)
|
|
{
|
|
HKEY winkey = 0, externalkey = 0;
|
|
LPWSTR valueW;
|
|
LPVOID data;
|
|
DWORD dlen, vlen, datalen, valuelen, i, type, len, len_fam;
|
|
Family *family;
|
|
Face *face;
|
|
struct list *family_elem_ptr, *face_elem_ptr;
|
|
WCHAR *file;
|
|
static const WCHAR TrueType[] = {' ','(','T','r','u','e','T','y','p','e',')','\0'};
|
|
static const WCHAR spaceW[] = {' ', '\0'};
|
|
char *path;
|
|
|
|
if(RegCreateKeyExW(HKEY_LOCAL_MACHINE, is_win9x() ? win9x_font_reg_key : winnt_font_reg_key,
|
|
0, NULL, 0, KEY_ALL_ACCESS, NULL, &winkey, NULL) != ERROR_SUCCESS) {
|
|
ERR("Can't create Windows font reg key\n");
|
|
goto end;
|
|
}
|
|
if(RegCreateKeyExW(HKEY_LOCAL_MACHINE, external_fonts_reg_key,
|
|
0, NULL, 0, KEY_ALL_ACCESS, NULL, &externalkey, NULL) != ERROR_SUCCESS) {
|
|
ERR("Can't create external font reg key\n");
|
|
goto end;
|
|
}
|
|
|
|
/* Delete all external fonts added last time */
|
|
|
|
RegQueryInfoKeyW(externalkey, 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 * sizeof(WCHAR));
|
|
|
|
dlen = datalen * sizeof(WCHAR);
|
|
vlen = valuelen;
|
|
i = 0;
|
|
while(RegEnumValueW(externalkey, i++, valueW, &vlen, NULL, &type, data,
|
|
&dlen) == ERROR_SUCCESS) {
|
|
|
|
RegDeleteValueW(winkey, valueW);
|
|
/* reset dlen and vlen */
|
|
dlen = datalen;
|
|
vlen = valuelen;
|
|
}
|
|
HeapFree(GetProcessHeap(), 0, data);
|
|
HeapFree(GetProcessHeap(), 0, valueW);
|
|
|
|
/* Delete the old external fonts key */
|
|
RegCloseKey(externalkey);
|
|
externalkey = 0;
|
|
RegDeleteKeyW(HKEY_LOCAL_MACHINE, external_fonts_reg_key);
|
|
|
|
if(RegCreateKeyExW(HKEY_LOCAL_MACHINE, external_fonts_reg_key,
|
|
0, NULL, 0, KEY_ALL_ACCESS, NULL, &externalkey, 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(family_elem_ptr, &font_list) {
|
|
family = LIST_ENTRY(family_elem_ptr, Family, entry);
|
|
len_fam = strlenW(family->FamilyName) + sizeof(TrueType) / sizeof(WCHAR) + 1;
|
|
LIST_FOR_EACH(face_elem_ptr, &family->faces) {
|
|
face = LIST_ENTRY(face_elem_ptr, Face, entry);
|
|
if(!face->external) continue;
|
|
len = len_fam;
|
|
if(strcmpiW(face->StyleName, RegularW))
|
|
len = len_fam + strlenW(face->StyleName) + 1;
|
|
valueW = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR));
|
|
strcpyW(valueW, family->FamilyName);
|
|
if(len != len_fam) {
|
|
strcatW(valueW, spaceW);
|
|
strcatW(valueW, face->StyleName);
|
|
}
|
|
strcatW(valueW, TrueType);
|
|
if((path = strrchr(face->file, '/')) == NULL)
|
|
path = face->file;
|
|
else
|
|
path++;
|
|
len = MultiByteToWideChar(CP_ACP, 0, path, -1, NULL, 0);
|
|
|
|
file = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR));
|
|
MultiByteToWideChar(CP_ACP, 0, path, -1, file, len);
|
|
RegSetValueExW(winkey, valueW, 0, REG_SZ, (BYTE*)file, len * sizeof(WCHAR));
|
|
RegSetValueExW(externalkey, valueW, 0, REG_SZ, (BYTE*)file, len * sizeof(WCHAR));
|
|
|
|
HeapFree(GetProcessHeap(), 0, file);
|
|
HeapFree(GetProcessHeap(), 0, valueW);
|
|
}
|
|
}
|
|
end:
|
|
if(externalkey)
|
|
RegCloseKey(externalkey);
|
|
if(winkey)
|
|
RegCloseKey(winkey);
|
|
return;
|
|
}
|
|
|
|
|
|
/*************************************************************
|
|
* WineEngAddFontResourceEx
|
|
*
|
|
*/
|
|
INT WineEngAddFontResourceEx(LPCWSTR file, DWORD flags, PVOID pdv)
|
|
{
|
|
if (ft_handle) /* do it only if we have freetype up and running */
|
|
{
|
|
char *unixname;
|
|
|
|
if(flags)
|
|
FIXME("Ignoring flags %lx\n", flags);
|
|
|
|
if((unixname = wine_get_unix_file_name(file)))
|
|
{
|
|
AddFontFileToList(unixname, NULL, ADDFONT_FORCE_BITMAP);
|
|
HeapFree(GetProcessHeap(), 0, unixname);
|
|
}
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
/*************************************************************
|
|
* WineEngRemoveFontResourceEx
|
|
*
|
|
*/
|
|
BOOL WineEngRemoveFontResourceEx(LPCWSTR file, DWORD flags, PVOID pdv)
|
|
{
|
|
FIXME(":stub\n");
|
|
return TRUE;
|
|
}
|
|
|
|
/*************************************************************
|
|
* WineEngInit
|
|
*
|
|
* Initialize FreeType library and create a list of available faces
|
|
*/
|
|
BOOL WineEngInit(void)
|
|
{
|
|
static const WCHAR dot_fonW[] = {'.','f','o','n','\0'};
|
|
HKEY hkey;
|
|
DWORD valuelen, datalen, i = 0, type, dlen, vlen;
|
|
LPVOID data;
|
|
WCHAR windowsdir[MAX_PATH];
|
|
char *unixname;
|
|
HANDLE font_mutex;
|
|
|
|
TRACE("\n");
|
|
|
|
ft_handle = wine_dlopen(SONAME_LIBFREETYPE, RTLD_NOW, NULL, 0);
|
|
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 = wine_dlsym(ft_handle, #f, NULL, 0)) == NULL){WARN("Can't find symbol %s\n", #f); goto sym_not_found;}
|
|
|
|
LOAD_FUNCPTR(FT_Vector_Unit)
|
|
LOAD_FUNCPTR(FT_Done_Face)
|
|
LOAD_FUNCPTR(FT_Get_Char_Index)
|
|
LOAD_FUNCPTR(FT_Get_Sfnt_Table)
|
|
LOAD_FUNCPTR(FT_Init_FreeType)
|
|
LOAD_FUNCPTR(FT_Load_Glyph)
|
|
LOAD_FUNCPTR(FT_Matrix_Multiply)
|
|
LOAD_FUNCPTR(FT_MulFix)
|
|
LOAD_FUNCPTR(FT_New_Face)
|
|
LOAD_FUNCPTR(FT_Outline_Get_Bitmap)
|
|
LOAD_FUNCPTR(FT_Outline_Transform)
|
|
LOAD_FUNCPTR(FT_Outline_Translate)
|
|
LOAD_FUNCPTR(FT_Select_Charmap)
|
|
LOAD_FUNCPTR(FT_Set_Pixel_Sizes)
|
|
LOAD_FUNCPTR(FT_Vector_Transform)
|
|
|
|
#undef LOAD_FUNCPTR
|
|
/* Don't warn if this one is missing */
|
|
pFT_Library_Version = wine_dlsym(ft_handle, "FT_Library_Version", NULL, 0);
|
|
pFT_Load_Sfnt_Table = wine_dlsym(ft_handle, "FT_Load_Sfnt_Table", NULL, 0);
|
|
pFT_Get_First_Char = wine_dlsym(ft_handle, "FT_Get_First_Char", NULL, 0);
|
|
#ifdef HAVE_FREETYPE_FTWINFNT_H
|
|
pFT_Get_WinFNT_Header = wine_dlsym(ft_handle, "FT_Get_WinFNT_Header", NULL, 0);
|
|
#endif
|
|
if(!wine_dlsym(ft_handle, "FT_Get_Postscript_Name", NULL, 0) &&
|
|
!wine_dlsym(ft_handle, "FT_Sqrt64", NULL, 0)) {
|
|
/* try to avoid 2.0.4: >= 2.0.5 has FT_Get_Postscript_Name and
|
|
<= 2.0.3 has FT_Sqrt64 */
|
|
goto sym_not_found;
|
|
}
|
|
|
|
if(pFT_Init_FreeType(&library) != 0) {
|
|
ERR("Can't init FreeType library\n");
|
|
wine_dlclose(ft_handle, NULL, 0);
|
|
ft_handle = NULL;
|
|
return FALSE;
|
|
}
|
|
FT_Version.major=FT_Version.minor=FT_Version.patch=-1;
|
|
if (pFT_Library_Version)
|
|
{
|
|
pFT_Library_Version(library,&FT_Version.major,&FT_Version.minor,&FT_Version.patch);
|
|
}
|
|
if (FT_Version.major<=0)
|
|
{
|
|
FT_Version.major=2;
|
|
FT_Version.minor=0;
|
|
FT_Version.patch=5;
|
|
}
|
|
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);
|
|
|
|
if((font_mutex = CreateMutexW(NULL, FALSE, font_mutex_nameW)) == NULL) {
|
|
ERR("Failed to create font mutex\n");
|
|
return FALSE;
|
|
}
|
|
WaitForSingleObject(font_mutex, INFINITE);
|
|
|
|
/* load the system fonts */
|
|
load_system_fonts();
|
|
|
|
/* load in the fonts from %WINDOWSDIR%\\Fonts first of all */
|
|
GetWindowsDirectoryW(windowsdir, sizeof(windowsdir) / sizeof(WCHAR));
|
|
strcatW(windowsdir, fontsW);
|
|
if((unixname = wine_get_unix_file_name(windowsdir)))
|
|
{
|
|
ReadFontDir(unixname, FALSE);
|
|
HeapFree(GetProcessHeap(), 0, unixname);
|
|
}
|
|
|
|
/* now look under HKLM\Software\Microsoft\Windows[ NT]\CurrentVersion\Fonts
|
|
for any fonts not installed in %WINDOWSDIR%\Fonts. They will have their
|
|
full path as the entry. Also look for any .fon fonts, since ReadFontDir
|
|
will skip these. */
|
|
if(RegOpenKeyW(HKEY_LOCAL_MACHINE,
|
|
is_win9x() ? win9x_font_reg_key : winnt_font_reg_key,
|
|
&hkey) == ERROR_SUCCESS) {
|
|
LPWSTR valueW;
|
|
RegQueryInfoKeyW(hkey, 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 * sizeof(WCHAR));
|
|
if (valueW && data)
|
|
{
|
|
dlen = datalen * sizeof(WCHAR);
|
|
vlen = valuelen;
|
|
while(RegEnumValueW(hkey, i++, valueW, &vlen, NULL, &type, data,
|
|
&dlen) == ERROR_SUCCESS) {
|
|
if(((LPWSTR)data)[0] && ((LPWSTR)data)[1] == ':')
|
|
{
|
|
if((unixname = wine_get_unix_file_name((LPWSTR)data)))
|
|
{
|
|
AddFontFileToList(unixname, NULL, ADDFONT_FORCE_BITMAP);
|
|
HeapFree(GetProcessHeap(), 0, unixname);
|
|
}
|
|
}
|
|
else if(dlen / 2 >= 6 && !strcmpiW(((LPWSTR)data) + dlen / 2 - 5, dot_fonW))
|
|
{
|
|
WCHAR pathW[MAX_PATH];
|
|
static const WCHAR fmtW[] = {'%','s','\\','%','s','\0'};
|
|
sprintfW(pathW, fmtW, windowsdir, data);
|
|
if((unixname = wine_get_unix_file_name(pathW)))
|
|
{
|
|
AddFontFileToList(unixname, NULL, ADDFONT_FORCE_BITMAP);
|
|
HeapFree(GetProcessHeap(), 0, unixname);
|
|
}
|
|
}
|
|
/* reset dlen and vlen */
|
|
dlen = datalen;
|
|
vlen = valuelen;
|
|
}
|
|
}
|
|
HeapFree(GetProcessHeap(), 0, data);
|
|
HeapFree(GetProcessHeap(), 0, valueW);
|
|
RegCloseKey(hkey);
|
|
}
|
|
|
|
load_fontconfig_fonts();
|
|
|
|
/* then look in any directories that we've specified in the config file */
|
|
if(RegOpenKeyA(HKEY_LOCAL_MACHINE,
|
|
"Software\\Wine\\Wine\\Config\\FontDirs",
|
|
&hkey) == ERROR_SUCCESS) {
|
|
LPSTR value;
|
|
RegQueryInfoKeyA(hkey, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
|
|
&valuelen, &datalen, NULL, NULL);
|
|
|
|
valuelen++; /* returned value doesn't include room for '\0' */
|
|
value = HeapAlloc(GetProcessHeap(), 0, valuelen);
|
|
data = HeapAlloc(GetProcessHeap(), 0, datalen);
|
|
|
|
dlen = datalen;
|
|
vlen = valuelen;
|
|
i = 0;
|
|
while(RegEnumValueA(hkey, i++, value, &vlen, NULL, &type, data,
|
|
&dlen) == ERROR_SUCCESS) {
|
|
TRACE("Got %s=%s\n", value, (LPSTR)data);
|
|
ReadFontDir((LPSTR)data, TRUE);
|
|
/* reset dlen and vlen */
|
|
dlen = datalen;
|
|
vlen = valuelen;
|
|
}
|
|
HeapFree(GetProcessHeap(), 0, data);
|
|
HeapFree(GetProcessHeap(), 0, value);
|
|
RegCloseKey(hkey);
|
|
}
|
|
|
|
DumpFontList();
|
|
LoadSubstList();
|
|
DumpSubstList();
|
|
LoadReplaceList();
|
|
update_reg_entries();
|
|
|
|
ReleaseMutex(font_mutex);
|
|
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.0.5.\n"
|
|
"http://www.freetype.org\n");
|
|
wine_dlclose(ft_handle, NULL, 0);
|
|
ft_handle = NULL;
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
static LONG calc_ppem_for_height(FT_Face ft_face, LONG height)
|
|
{
|
|
TT_OS2 *pOS2;
|
|
TT_HoriHeader *pHori;
|
|
|
|
LONG ppem;
|
|
|
|
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) {
|
|
if(pOS2->usWinAscent + pOS2->usWinDescent == 0)
|
|
ppem = ft_face->units_per_EM * height /
|
|
(pHori->Ascender - pHori->Descender);
|
|
else
|
|
ppem = ft_face->units_per_EM * height /
|
|
(pOS2->usWinAscent + pOS2->usWinDescent);
|
|
}
|
|
else
|
|
ppem = -height;
|
|
|
|
return ppem;
|
|
}
|
|
|
|
static LONG load_VDMX(GdiFont, LONG);
|
|
|
|
static FT_Face OpenFontFile(GdiFont font, char *file, FT_Long face_index, LONG width, LONG height)
|
|
{
|
|
FT_Error err;
|
|
FT_Face ft_face;
|
|
LONG ppem;
|
|
|
|
err = pFT_New_Face(library, file, face_index, &ft_face);
|
|
if(err) {
|
|
ERR("FT_New_Face rets %d\n", err);
|
|
return 0;
|
|
}
|
|
|
|
/* set it here, as load_VDMX needs it */
|
|
font->ft_face = ft_face;
|
|
|
|
if(FT_IS_SCALABLE(ft_face)) {
|
|
/* load the VDMX table if we have one */
|
|
ppem = load_VDMX(font, height);
|
|
if(ppem == 0)
|
|
ppem = calc_ppem_for_height(ft_face, height);
|
|
|
|
if((err = pFT_Set_Pixel_Sizes(ft_face, 0, ppem)) != 0)
|
|
WARN("FT_Set_Pixel_Sizes %d, %ld rets %x\n", 0, ppem, err);
|
|
} else {
|
|
if((err = pFT_Set_Pixel_Sizes(ft_face, width, height)) != 0)
|
|
WARN("FT_Set_Pixel_Sizes %ld, %ld rets %x\n", width, height, err);
|
|
}
|
|
return ft_face;
|
|
}
|
|
|
|
|
|
static int get_nearest_charset(Face *face, int *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*)acp, &csi, TCI_SRCCODEPAGE))
|
|
if(csi.fs.fsCsb[0] & face->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 %lx\n", fs0);
|
|
}
|
|
}
|
|
|
|
FIXME("returning DEFAULT_CHARSET face->fs.fsCsb[0] = %08lx file = %s\n",
|
|
face->fs.fsCsb[0], face->file);
|
|
*cp = acp;
|
|
return DEFAULT_CHARSET;
|
|
}
|
|
|
|
static GdiFont alloc_font(void)
|
|
{
|
|
GdiFont ret = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(*ret));
|
|
ret->gmsize = INIT_GM_SIZE;
|
|
ret->gm = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY,
|
|
ret->gmsize * sizeof(*ret->gm));
|
|
ret->potm = NULL;
|
|
ret->font_desc.matrix.eM11 = ret->font_desc.matrix.eM22 = 1.0;
|
|
list_init(&ret->hfontlist);
|
|
return ret;
|
|
}
|
|
|
|
static void free_font(GdiFont font)
|
|
{
|
|
if (font->ft_face) pFT_Done_Face(font->ft_face);
|
|
HeapFree(GetProcessHeap(), 0, font->potm);
|
|
HeapFree(GetProcessHeap(), 0, font->name);
|
|
HeapFree(GetProcessHeap(), 0, font->gm);
|
|
HeapFree(GetProcessHeap(), 0, font);
|
|
}
|
|
|
|
|
|
/*************************************************************
|
|
* load_VDMX
|
|
*
|
|
* load the vdmx entry for the specified height
|
|
*/
|
|
|
|
#define MS_MAKE_TAG( _x1, _x2, _x3, _x4 ) \
|
|
( ( (FT_ULong)_x4 << 24 ) | \
|
|
( (FT_ULong)_x3 << 16 ) | \
|
|
( (FT_ULong)_x2 << 8 ) | \
|
|
(FT_ULong)_x1 )
|
|
|
|
#define MS_VDMX_TAG MS_MAKE_TAG('V', 'D', 'M', 'X')
|
|
|
|
typedef struct {
|
|
BYTE bCharSet;
|
|
BYTE xRatio;
|
|
BYTE yStartRatio;
|
|
BYTE yEndRatio;
|
|
} Ratios;
|
|
|
|
|
|
static LONG load_VDMX(GdiFont font, LONG height)
|
|
{
|
|
BYTE hdr[6], tmp[2], group[4];
|
|
BYTE devXRatio, devYRatio;
|
|
USHORT numRecs, numRatios;
|
|
DWORD result, offset = -1;
|
|
LONG ppem = 0;
|
|
int i;
|
|
|
|
result = WineEngGetFontData(font, MS_VDMX_TAG, 0, hdr, 6);
|
|
|
|
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[2]);
|
|
numRatios = GET_BE_WORD(&hdr[4]);
|
|
|
|
TRACE("numRecs = %d numRatios = %d\n", numRecs, numRatios);
|
|
for(i = 0; i < numRatios; i++) {
|
|
Ratios ratio;
|
|
|
|
offset = (3 * 2) + (i * sizeof(Ratios));
|
|
WineEngGetFontData(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 != 1)
|
|
continue;
|
|
|
|
if((ratio.xRatio == 0 &&
|
|
ratio.yStartRatio == 0 &&
|
|
ratio.yEndRatio == 0) ||
|
|
(devXRatio == ratio.xRatio &&
|
|
devYRatio >= ratio.yStartRatio &&
|
|
devYRatio <= ratio.yEndRatio))
|
|
{
|
|
offset = (3 * 2) + (numRatios * 4) + (i * 2);
|
|
WineEngGetFontData(font, MS_VDMX_TAG, offset, tmp, 2);
|
|
offset = GET_BE_WORD(tmp);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if(offset == -1) {
|
|
FIXME("No suitable ratio found\n");
|
|
return ppem;
|
|
}
|
|
|
|
if(WineEngGetFontData(font, MS_VDMX_TAG, offset, group, 4) != GDI_ERROR) {
|
|
USHORT recs;
|
|
BYTE startsz, endsz;
|
|
BYTE *vTable;
|
|
|
|
recs = GET_BE_WORD(group);
|
|
startsz = group[2];
|
|
endsz = group[3];
|
|
|
|
TRACE("recs=%d startsz=%d endsz=%d\n", recs, startsz, endsz);
|
|
|
|
vTable = HeapAlloc(GetProcessHeap(), 0, recs * 6);
|
|
result = WineEngGetFontData(font, MS_VDMX_TAG, offset + 4, vTable, recs * 6);
|
|
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 * 6) + 2]);
|
|
SHORT yMin = GET_BE_WORD(&vTable[(i * 6) + 4]);
|
|
ppem = GET_BE_WORD(&vTable[i * 6]);
|
|
|
|
if(yMax + -yMin == height) {
|
|
font->yMax = yMax;
|
|
font->yMin = yMin;
|
|
TRACE("ppem %ld found; height=%ld 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 * 6) + 2]);
|
|
font->yMin = GET_BE_WORD(&vTable[(i * 6) + 4]);
|
|
TRACE("ppem %ld found; height=%ld yMax=%d yMin=%d\n", ppem, height, font->yMax, font->yMin);
|
|
break;
|
|
}
|
|
}
|
|
if(!font->yMax) {
|
|
ppem = 0;
|
|
TRACE("ppem not found for height %ld\n", height);
|
|
}
|
|
} else {
|
|
ppem = -height;
|
|
if(ppem < startsz || ppem > endsz)
|
|
goto end;
|
|
|
|
for(i = 0; i < recs; i++) {
|
|
USHORT yPelHeight;
|
|
yPelHeight = GET_BE_WORD(&vTable[i * 6]);
|
|
|
|
if(yPelHeight > ppem)
|
|
break; /* failed */
|
|
|
|
if(yPelHeight == ppem) {
|
|
font->yMax = GET_BE_WORD(&vTable[(i * 6) + 2]);
|
|
font->yMin = GET_BE_WORD(&vTable[(i * 6) + 4]);
|
|
TRACE("ppem %ld found; yMax=%d yMin=%d\n", ppem, font->yMax, font->yMin);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
end:
|
|
HeapFree(GetProcessHeap(), 0, vTable);
|
|
}
|
|
|
|
return ppem;
|
|
}
|
|
|
|
static BOOL fontcmp(GdiFont font, FONT_DESC *fd)
|
|
{
|
|
if(font->font_desc.hash != fd->hash) return TRUE;
|
|
if(memcmp(&font->font_desc.matrix, &fd->matrix, sizeof(fd->matrix))) return TRUE;
|
|
if(memcmp(&font->font_desc.lf, &fd->lf, offsetof(LOGFONTW, lfFaceName))) return TRUE;
|
|
return strcmpiW(font->font_desc.lf.lfFaceName, fd->lf.lfFaceName);
|
|
}
|
|
|
|
static void calc_hash(FONT_DESC *pfd)
|
|
{
|
|
DWORD hash = 0, *ptr, two_chars;
|
|
WORD *pwc;
|
|
unsigned int i;
|
|
|
|
for(i = 0, ptr = (DWORD*)&pfd->matrix; i < sizeof(FMAT2)/sizeof(DWORD); i++, ptr++)
|
|
hash ^= *ptr;
|
|
for(i = 0, ptr = (DWORD*)&pfd->lf; i < 7; i++, ptr++)
|
|
hash ^= *ptr;
|
|
for(i = 0, ptr = (DWORD*)&pfd->lf.lfFaceName; i < LF_FACESIZE/2; i++, ptr++) {
|
|
two_chars = *ptr;
|
|
pwc = (WCHAR *)&two_chars;
|
|
if(!*pwc) break;
|
|
*pwc = toupperW(*pwc);
|
|
pwc++;
|
|
*pwc = toupperW(*pwc);
|
|
hash ^= two_chars;
|
|
if(!*pwc) break;
|
|
}
|
|
pfd->hash = hash;
|
|
return;
|
|
}
|
|
|
|
static GdiFont find_in_cache(HFONT hfont, LOGFONTW *plf, XFORM *pxf, BOOL can_use_bitmap)
|
|
{
|
|
GdiFont ret;
|
|
FONT_DESC fd;
|
|
HFONTLIST *hflist;
|
|
struct list *font_elem_ptr, *hfontlist_elem_ptr;
|
|
|
|
memcpy(&fd.lf, plf, sizeof(LOGFONTW));
|
|
memcpy(&fd.matrix, pxf, sizeof(FMAT2));
|
|
calc_hash(&fd);
|
|
|
|
/* try the in-use list */
|
|
LIST_FOR_EACH(font_elem_ptr, &gdi_font_list) {
|
|
ret = LIST_ENTRY(font_elem_ptr, struct tagGdiFont, entry);
|
|
if(!fontcmp(ret, &fd)) {
|
|
if(!can_use_bitmap && !FT_IS_SCALABLE(ret->ft_face)) continue;
|
|
LIST_FOR_EACH(hfontlist_elem_ptr, &ret->hfontlist) {
|
|
hflist = LIST_ENTRY(hfontlist_elem_ptr, struct tagHFONTLIST, entry);
|
|
if(hflist->hfont == hfont)
|
|
return ret;
|
|
}
|
|
hflist = HeapAlloc(GetProcessHeap(), 0, sizeof(*hflist));
|
|
hflist->hfont = hfont;
|
|
list_add_head(&ret->hfontlist, &hflist->entry);
|
|
return ret;
|
|
}
|
|
}
|
|
|
|
/* then the unused list */
|
|
font_elem_ptr = list_head(&unused_gdi_font_list);
|
|
while(font_elem_ptr) {
|
|
ret = LIST_ENTRY(font_elem_ptr, struct tagGdiFont, entry);
|
|
font_elem_ptr = list_next(&unused_gdi_font_list, font_elem_ptr);
|
|
if(!fontcmp(ret, &fd)) {
|
|
if(!can_use_bitmap && !FT_IS_SCALABLE(ret->ft_face)) continue;
|
|
assert(list_empty(&ret->hfontlist));
|
|
TRACE("Found %p in unused list\n", ret);
|
|
list_remove(&ret->entry);
|
|
list_add_head(&gdi_font_list, &ret->entry);
|
|
hflist = HeapAlloc(GetProcessHeap(), 0, sizeof(*hflist));
|
|
hflist->hfont = hfont;
|
|
list_add_head(&ret->hfontlist, &hflist->entry);
|
|
return ret;
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
/*************************************************************
|
|
* WineEngCreateFontInstance
|
|
*
|
|
*/
|
|
GdiFont WineEngCreateFontInstance(DC *dc, HFONT hfont)
|
|
{
|
|
GdiFont ret;
|
|
Face *face, *best;
|
|
Family *family;
|
|
struct list *family_elem_ptr, *face_elem_ptr;
|
|
INT height, width = 0;
|
|
signed int diff = 0, newdiff;
|
|
BOOL bd, it, can_use_bitmap;
|
|
LOGFONTW lf;
|
|
CHARSETINFO csi;
|
|
HFONTLIST *hflist;
|
|
|
|
if (!GetObjectW( hfont, sizeof(lf), &lf )) return NULL;
|
|
can_use_bitmap = GetDeviceCaps(dc->hSelf, TEXTCAPS) & TC_RA_ABLE;
|
|
|
|
TRACE("%s, h=%ld, it=%d, weight=%ld, PandF=%02x, charset=%d orient %ld escapement %ld\n",
|
|
debugstr_w(lf.lfFaceName), lf.lfHeight, lf.lfItalic,
|
|
lf.lfWeight, lf.lfPitchAndFamily, lf.lfCharSet, lf.lfOrientation,
|
|
lf.lfEscapement);
|
|
|
|
/* check the cache first */
|
|
if((ret = find_in_cache(hfont, &lf, &dc->xformWorld2Vport, can_use_bitmap)) != NULL) {
|
|
TRACE("returning cached gdiFont(%p) for hFont %p\n", ret, hfont);
|
|
return ret;
|
|
}
|
|
|
|
TRACE("not in cache\n");
|
|
if(list_empty(&font_list) || !have_installed_roman_font) /* No fonts installed */
|
|
{
|
|
TRACE("No fonts installed\n");
|
|
return NULL;
|
|
}
|
|
|
|
ret = alloc_font();
|
|
|
|
memcpy(&ret->font_desc.matrix, &dc->xformWorld2Vport, sizeof(FMAT2));
|
|
memcpy(&ret->font_desc.lf, &lf, sizeof(LOGFONTW));
|
|
calc_hash(&ret->font_desc);
|
|
hflist = HeapAlloc(GetProcessHeap(), 0, sizeof(*hflist));
|
|
hflist->hfont = hfont;
|
|
list_add_head(&ret->hfontlist, &hflist->entry);
|
|
|
|
|
|
/* 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;
|
|
|
|
if(!TranslateCharsetInfo((DWORD*)(INT)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') {
|
|
FontSubst *psub;
|
|
for(psub = substlist; psub; psub = psub->next)
|
|
if(!strcmpiW(lf.lfFaceName, psub->from.name) &&
|
|
(psub->from.charset == -1 ||
|
|
psub->from.charset == lf.lfCharSet))
|
|
break;
|
|
if(psub) {
|
|
TRACE("substituting %s -> %s\n", debugstr_w(lf.lfFaceName),
|
|
debugstr_w(psub->to.name));
|
|
strcpyW(lf.lfFaceName, psub->to.name);
|
|
}
|
|
|
|
/* 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(family_elem_ptr, &font_list) {
|
|
family = LIST_ENTRY(family_elem_ptr, Family, entry);
|
|
if(!strcmpiW(family->FamilyName, lf.lfFaceName)) {
|
|
face_elem_ptr = list_head(&family->faces);
|
|
face = LIST_ENTRY(face_elem_ptr, Face, entry);
|
|
if((csi.fs.fsCsb[0] & face->fs.fsCsb[0]) || !csi.fs.fsCsb[0])
|
|
if(face->scalable || can_use_bitmap)
|
|
break;
|
|
}
|
|
family = NULL;
|
|
}
|
|
}
|
|
|
|
if(!family) {
|
|
/* 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*)acp, &csi, TCI_SRCCODEPAGE)) {
|
|
FIXME("TCI failed on codepage %d\n", acp);
|
|
csi.fs.fsCsb[0] = 0;
|
|
} else
|
|
lf.lfCharSet = csi.ciCharset;
|
|
}
|
|
|
|
/* 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, defFixed);
|
|
else if((lf.lfPitchAndFamily & 0xF0) == FF_ROMAN)
|
|
strcpyW(lf.lfFaceName, defSerif);
|
|
else if((lf.lfPitchAndFamily & 0xF0) == FF_SWISS)
|
|
strcpyW(lf.lfFaceName, defSans);
|
|
else
|
|
strcpyW(lf.lfFaceName, defSans);
|
|
LIST_FOR_EACH(family_elem_ptr, &font_list) {
|
|
family = LIST_ENTRY(family_elem_ptr, Family, entry);
|
|
if(!strcmpiW(family->FamilyName, lf.lfFaceName)) {
|
|
face_elem_ptr = list_head(&family->faces);
|
|
face = LIST_ENTRY(face_elem_ptr, Face, entry);
|
|
if(csi.fs.fsCsb[0] & face->fs.fsCsb[0])
|
|
if(face->scalable || can_use_bitmap)
|
|
break;
|
|
}
|
|
family = NULL;
|
|
}
|
|
}
|
|
|
|
if(!family) {
|
|
LIST_FOR_EACH(family_elem_ptr, &font_list) {
|
|
family = LIST_ENTRY(family_elem_ptr, Family, entry);
|
|
face_elem_ptr = list_head(&family->faces);
|
|
face = LIST_ENTRY(face_elem_ptr, Face, entry);
|
|
if(csi.fs.fsCsb[0] & face->fs.fsCsb[0])
|
|
if(face->scalable || can_use_bitmap)
|
|
break;
|
|
family = NULL;
|
|
}
|
|
}
|
|
|
|
if(!family) {
|
|
LIST_FOR_EACH(family_elem_ptr, &font_list) {
|
|
family = LIST_ENTRY(family_elem_ptr, Family, entry);
|
|
face_elem_ptr = list_head(&family->faces);
|
|
face = LIST_ENTRY(face_elem_ptr, Face, entry);
|
|
if(face->scalable || can_use_bitmap) {
|
|
csi.fs.fsCsb[0] = 0;
|
|
FIXME("just using first face for now\n");
|
|
break;
|
|
}
|
|
family = NULL;
|
|
}
|
|
if(!family) {
|
|
FIXME("can't find a single appropriate font - bailing\n");
|
|
free_font(ret);
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
it = lf.lfItalic ? 1 : 0;
|
|
bd = lf.lfWeight > 550 ? 1 : 0;
|
|
|
|
height = GDI_ROUND( (FLOAT)lf.lfHeight * dc->xformWorld2Vport.eM22 );
|
|
height = lf.lfHeight < 0 ? -abs(height) : abs(height);
|
|
|
|
face = best = NULL;
|
|
LIST_FOR_EACH(face_elem_ptr, &family->faces) {
|
|
face = LIST_ENTRY(face_elem_ptr, Face, entry);
|
|
if(!(face->Italic ^ it) && !(face->Bold ^ bd)) {
|
|
if(face->scalable)
|
|
break;
|
|
if(height > 0)
|
|
newdiff = height - (signed int)(face->size.y_ppem >> 6);
|
|
else
|
|
newdiff = -height - ((signed int)(face->size.y_ppem >> 6) - face->size.internal_leading);
|
|
if(!best || (diff > 0 && newdiff < diff && newdiff >= 0) ||
|
|
(diff < 0 && newdiff > diff)) {
|
|
TRACE("%ld is better for %d diff was %d\n", face->size.y_ppem >> 6, height, diff);
|
|
diff = newdiff;
|
|
best = face;
|
|
if(diff == 0)
|
|
break;
|
|
}
|
|
}
|
|
face = NULL;
|
|
}
|
|
if(!face && best)
|
|
face = best;
|
|
else if(!face) {
|
|
best = NULL;
|
|
LIST_FOR_EACH(face_elem_ptr, &family->faces) {
|
|
face = LIST_ENTRY(face_elem_ptr, Face, entry);
|
|
if(face->scalable)
|
|
break;
|
|
if(height > 0)
|
|
newdiff = height - (signed int)(face->size.y_ppem >> 6);
|
|
else
|
|
newdiff = -height - ((signed int)(face->size.y_ppem >> 6) - face->size.internal_leading);
|
|
if(!best || (diff > 0 && newdiff < diff && newdiff >= 0) ||
|
|
(diff < 0 && newdiff > diff)) {
|
|
TRACE("%ld is better for %d diff was %d\n", face->size.y_ppem >> 6, height, diff);
|
|
diff = newdiff;
|
|
best = face;
|
|
if(diff == 0)
|
|
break;
|
|
}
|
|
face = NULL;
|
|
}
|
|
if(!face && best)
|
|
face = best;
|
|
if(it && !face->Italic) ret->fake_italic = TRUE;
|
|
if(bd && !face->Bold) ret->fake_bold = TRUE;
|
|
}
|
|
|
|
memcpy(&ret->fs, &face->fs, sizeof(FONTSIGNATURE));
|
|
|
|
if(csi.fs.fsCsb[0]) {
|
|
ret->charset = lf.lfCharSet;
|
|
ret->codepage = csi.ciACP;
|
|
}
|
|
else
|
|
ret->charset = get_nearest_charset(face, &ret->codepage);
|
|
|
|
TRACE("Chosen: %s %s\n", debugstr_w(family->FamilyName),
|
|
debugstr_w(face->StyleName));
|
|
|
|
if(!face->scalable) {
|
|
width = face->size.x_ppem >> 6;
|
|
height = face->size.y_ppem >> 6;
|
|
}
|
|
ret->ft_face = OpenFontFile(ret, face->file, face->face_index, width, height);
|
|
|
|
if (!ret->ft_face)
|
|
{
|
|
free_font( ret );
|
|
return 0;
|
|
}
|
|
|
|
if (ret->charset == SYMBOL_CHARSET &&
|
|
!pFT_Select_Charmap(ret->ft_face, FT_ENCODING_MS_SYMBOL)) {
|
|
/* No ops */
|
|
}
|
|
else if (!pFT_Select_Charmap(ret->ft_face, FT_ENCODING_UNICODE)) {
|
|
/* No ops */
|
|
}
|
|
else {
|
|
pFT_Select_Charmap(ret->ft_face, FT_ENCODING_APPLE_ROMAN);
|
|
}
|
|
|
|
ret->orientation = FT_IS_SCALABLE(ret->ft_face) ? lf.lfOrientation : 0;
|
|
ret->name = strdupW(family->FamilyName);
|
|
ret->underline = lf.lfUnderline ? 0xff : 0;
|
|
ret->strikeout = lf.lfStrikeOut ? 0xff : 0;
|
|
|
|
TRACE("caching: gdiFont=%p hfont=%p\n", ret, hfont);
|
|
|
|
ret->aveWidth = FT_IS_SCALABLE(ret->ft_face) ? lf.lfWidth : 0;
|
|
list_add_head(&gdi_font_list, &ret->entry);
|
|
return ret;
|
|
}
|
|
|
|
static void dump_gdi_font_list(void)
|
|
{
|
|
GdiFont gdiFont;
|
|
struct list *elem_ptr;
|
|
|
|
TRACE("---------- gdiFont Cache ----------\n");
|
|
LIST_FOR_EACH(elem_ptr, &gdi_font_list) {
|
|
gdiFont = LIST_ENTRY(elem_ptr, struct tagGdiFont, entry);
|
|
TRACE("gdiFont=%p %s %ld\n",
|
|
gdiFont, debugstr_w(gdiFont->font_desc.lf.lfFaceName), gdiFont->font_desc.lf.lfHeight);
|
|
}
|
|
|
|
TRACE("---------- Unused gdiFont Cache ----------\n");
|
|
LIST_FOR_EACH(elem_ptr, &unused_gdi_font_list) {
|
|
gdiFont = LIST_ENTRY(elem_ptr, struct tagGdiFont, entry);
|
|
TRACE("gdiFont=%p %s %ld\n",
|
|
gdiFont, debugstr_w(gdiFont->font_desc.lf.lfFaceName), gdiFont->font_desc.lf.lfHeight);
|
|
}
|
|
}
|
|
|
|
/*************************************************************
|
|
* WineEngDestroyFontInstance
|
|
*
|
|
* free the gdiFont associated with this handle
|
|
*
|
|
*/
|
|
BOOL WineEngDestroyFontInstance(HFONT handle)
|
|
{
|
|
GdiFont gdiFont;
|
|
HFONTLIST *hflist;
|
|
BOOL ret = FALSE;
|
|
struct list *font_elem_ptr, *hfontlist_elem_ptr;
|
|
int i = 0;
|
|
|
|
TRACE("destroying hfont=%p\n", handle);
|
|
if(TRACE_ON(font))
|
|
dump_gdi_font_list();
|
|
|
|
font_elem_ptr = list_head(&gdi_font_list);
|
|
while(font_elem_ptr) {
|
|
gdiFont = LIST_ENTRY(font_elem_ptr, struct tagGdiFont, entry);
|
|
font_elem_ptr = list_next(&gdi_font_list, font_elem_ptr);
|
|
|
|
hfontlist_elem_ptr = list_head(&gdiFont->hfontlist);
|
|
while(hfontlist_elem_ptr) {
|
|
hflist = LIST_ENTRY(hfontlist_elem_ptr, struct tagHFONTLIST, entry);
|
|
hfontlist_elem_ptr = list_next(&gdiFont->hfontlist, hfontlist_elem_ptr);
|
|
if(hflist->hfont == handle) {
|
|
list_remove(&hflist->entry);
|
|
HeapFree(GetProcessHeap(), 0, hflist);
|
|
ret = TRUE;
|
|
}
|
|
}
|
|
if(list_empty(&gdiFont->hfontlist)) {
|
|
TRACE("Moving to Unused list\n");
|
|
list_remove(&gdiFont->entry);
|
|
list_add_head(&unused_gdi_font_list, &gdiFont->entry);
|
|
}
|
|
}
|
|
|
|
|
|
font_elem_ptr = list_head(&unused_gdi_font_list);
|
|
while(font_elem_ptr && i++ < UNUSED_CACHE_SIZE)
|
|
font_elem_ptr = list_next(&unused_gdi_font_list, font_elem_ptr);
|
|
while(font_elem_ptr) {
|
|
gdiFont = LIST_ENTRY(font_elem_ptr, struct tagGdiFont, entry);
|
|
font_elem_ptr = list_next(&unused_gdi_font_list, font_elem_ptr);
|
|
TRACE("freeing %p\n", gdiFont);
|
|
list_remove(&gdiFont->entry);
|
|
free_font(gdiFont);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static void GetEnumStructs(Face *face, LPENUMLOGFONTEXW pelf,
|
|
NEWTEXTMETRICEXW *pntm, LPDWORD ptype)
|
|
{
|
|
OUTLINETEXTMETRICW *potm = NULL;
|
|
UINT size;
|
|
TEXTMETRICW tm, *ptm;
|
|
GdiFont font = alloc_font();
|
|
LONG width, height;
|
|
|
|
if(face->scalable) {
|
|
height = 100;
|
|
width = 0;
|
|
} else {
|
|
height = face->size.y_ppem >> 6;
|
|
width = face->size.x_ppem >> 6;
|
|
}
|
|
|
|
if (!(font->ft_face = OpenFontFile(font, face->file, face->face_index, width, height)))
|
|
{
|
|
free_font(font);
|
|
return;
|
|
}
|
|
|
|
font->name = strdupW(face->family->FamilyName);
|
|
|
|
memset(&pelf->elfLogFont, 0, sizeof(LOGFONTW));
|
|
|
|
size = WineEngGetOutlineTextMetrics(font, 0, NULL);
|
|
if(size) {
|
|
potm = HeapAlloc(GetProcessHeap(), 0, size);
|
|
WineEngGetOutlineTextMetrics(font, size, potm);
|
|
ptm = (TEXTMETRICW*)&potm->otmTextMetrics;
|
|
} else {
|
|
WineEngGetTextMetrics(font, &tm);
|
|
ptm = &tm;
|
|
}
|
|
|
|
pntm->ntmTm.tmHeight = pelf->elfLogFont.lfHeight = ptm->tmHeight;
|
|
pntm->ntmTm.tmAscent = ptm->tmAscent;
|
|
pntm->ntmTm.tmDescent = ptm->tmDescent;
|
|
pntm->ntmTm.tmInternalLeading = ptm->tmInternalLeading;
|
|
pntm->ntmTm.tmExternalLeading = ptm->tmExternalLeading;
|
|
pntm->ntmTm.tmAveCharWidth = pelf->elfLogFont.lfWidth = ptm->tmAveCharWidth;
|
|
pntm->ntmTm.tmMaxCharWidth = ptm->tmMaxCharWidth;
|
|
pntm->ntmTm.tmWeight = pelf->elfLogFont.lfWeight = ptm->tmWeight;
|
|
pntm->ntmTm.tmOverhang = ptm->tmOverhang;
|
|
pntm->ntmTm.tmDigitizedAspectX = ptm->tmDigitizedAspectX;
|
|
pntm->ntmTm.tmDigitizedAspectY = ptm->tmDigitizedAspectY;
|
|
pntm->ntmTm.tmFirstChar = ptm->tmFirstChar;
|
|
pntm->ntmTm.tmLastChar = ptm->tmLastChar;
|
|
pntm->ntmTm.tmDefaultChar = ptm->tmDefaultChar;
|
|
pntm->ntmTm.tmBreakChar = ptm->tmBreakChar;
|
|
pntm->ntmTm.tmItalic = pelf->elfLogFont.lfItalic = ptm->tmItalic;
|
|
pntm->ntmTm.tmUnderlined = pelf->elfLogFont.lfUnderline = ptm->tmUnderlined;
|
|
pntm->ntmTm.tmStruckOut = pelf->elfLogFont.lfStrikeOut = ptm->tmStruckOut;
|
|
pntm->ntmTm.tmPitchAndFamily = ptm->tmPitchAndFamily;
|
|
pelf->elfLogFont.lfPitchAndFamily = (ptm->tmPitchAndFamily & 0xf1) + 1;
|
|
pntm->ntmTm.tmCharSet = pelf->elfLogFont.lfCharSet = ptm->tmCharSet;
|
|
pelf->elfLogFont.lfOutPrecision = OUT_STROKE_PRECIS;
|
|
pelf->elfLogFont.lfClipPrecision = CLIP_STROKE_PRECIS;
|
|
pelf->elfLogFont.lfQuality = DRAFT_QUALITY;
|
|
|
|
*ptype = ptm->tmPitchAndFamily & TMPF_TRUETYPE ? TRUETYPE_FONTTYPE : 0;
|
|
if(!(ptm->tmPitchAndFamily & TMPF_VECTOR))
|
|
*ptype |= RASTER_FONTTYPE;
|
|
|
|
pntm->ntmTm.ntmFlags = ptm->tmItalic ? NTM_ITALIC : 0;
|
|
if(ptm->tmWeight > 550) pntm->ntmTm.ntmFlags |= NTM_BOLD;
|
|
if(pntm->ntmTm.ntmFlags == 0) pntm->ntmTm.ntmFlags = NTM_REGULAR;
|
|
|
|
pntm->ntmTm.ntmCellHeight = pntm->ntmTm.tmHeight;
|
|
pntm->ntmTm.ntmAvgWidth = pntm->ntmTm.tmAveCharWidth;
|
|
memset(&pntm->ntmFontSig, 0, sizeof(FONTSIGNATURE));
|
|
|
|
if(potm) {
|
|
pntm->ntmTm.ntmSizeEM = potm->otmEMSquare;
|
|
|
|
strncpyW(pelf->elfLogFont.lfFaceName,
|
|
(WCHAR*)((char*)potm + (ptrdiff_t)potm->otmpFamilyName),
|
|
LF_FACESIZE);
|
|
strncpyW(pelf->elfFullName,
|
|
(WCHAR*)((char*)potm + (ptrdiff_t)potm->otmpFaceName),
|
|
LF_FULLFACESIZE);
|
|
strncpyW(pelf->elfStyle,
|
|
(WCHAR*)((char*)potm + (ptrdiff_t)potm->otmpStyleName),
|
|
LF_FACESIZE);
|
|
|
|
HeapFree(GetProcessHeap(), 0, potm);
|
|
} else {
|
|
pntm->ntmTm.ntmSizeEM = pntm->ntmTm.tmHeight - pntm->ntmTm.tmInternalLeading;
|
|
|
|
strncpyW(pelf->elfLogFont.lfFaceName, face->family->FamilyName, LF_FACESIZE);
|
|
strncpyW(pelf->elfFullName, face->family->FamilyName, LF_FACESIZE);
|
|
pelf->elfStyle[0] = '\0';
|
|
}
|
|
|
|
pelf->elfScript[0] = '\0'; /* This will get set in WineEngEnumFonts */
|
|
|
|
free_font(font);
|
|
}
|
|
|
|
/*************************************************************
|
|
* WineEngEnumFonts
|
|
*
|
|
*/
|
|
DWORD WineEngEnumFonts(LPLOGFONTW plf, FONTENUMPROCW proc, LPARAM lparam)
|
|
{
|
|
Family *family;
|
|
Face *face;
|
|
struct list *family_elem_ptr, *face_elem_ptr;
|
|
ENUMLOGFONTEXW elf;
|
|
NEWTEXTMETRICEXW ntm;
|
|
DWORD type, ret = 1;
|
|
FONTSIGNATURE fs;
|
|
CHARSETINFO csi;
|
|
LOGFONTW lf;
|
|
int i;
|
|
|
|
TRACE("facename = %s charset %d\n", debugstr_w(plf->lfFaceName), plf->lfCharSet);
|
|
|
|
if(plf->lfFaceName[0]) {
|
|
FontSubst *psub;
|
|
for(psub = substlist; psub; psub = psub->next)
|
|
if(!strcmpiW(plf->lfFaceName, psub->from.name) &&
|
|
(psub->from.charset == -1 ||
|
|
psub->from.charset == plf->lfCharSet))
|
|
break;
|
|
if(psub) {
|
|
TRACE("substituting %s -> %s\n", debugstr_w(plf->lfFaceName),
|
|
debugstr_w(psub->to.name));
|
|
memcpy(&lf, plf, sizeof(lf));
|
|
strcpyW(lf.lfFaceName, psub->to.name);
|
|
plf = &lf;
|
|
}
|
|
|
|
LIST_FOR_EACH(family_elem_ptr, &font_list) {
|
|
family = LIST_ENTRY(family_elem_ptr, Family, entry);
|
|
if(!strcmpiW(plf->lfFaceName, family->FamilyName)) {
|
|
LIST_FOR_EACH(face_elem_ptr, &family->faces) {
|
|
face = LIST_ENTRY(face_elem_ptr, Face, entry);
|
|
GetEnumStructs(face, &elf, &ntm, &type);
|
|
for(i = 0; i < 32; i++) {
|
|
if(!face->scalable && face->fs.fsCsb[0] == 0) { /* OEM bitmap */
|
|
elf.elfLogFont.lfCharSet = ntm.ntmTm.tmCharSet = OEM_CHARSET;
|
|
strcpyW(elf.elfScript, OEM_DOSW);
|
|
i = 32; /* break out of loop */
|
|
} else if(!(face->fs.fsCsb[0] & (1L << i)))
|
|
continue;
|
|
else {
|
|
fs.fsCsb[0] = 1L << i;
|
|
fs.fsCsb[1] = 0;
|
|
if(!TranslateCharsetInfo(fs.fsCsb, &csi,
|
|
TCI_SRCFONTSIG))
|
|
csi.ciCharset = DEFAULT_CHARSET;
|
|
if(i == 31) csi.ciCharset = SYMBOL_CHARSET;
|
|
if(csi.ciCharset != DEFAULT_CHARSET) {
|
|
elf.elfLogFont.lfCharSet =
|
|
ntm.ntmTm.tmCharSet = csi.ciCharset;
|
|
if(ElfScriptsW[i])
|
|
strcpyW(elf.elfScript, ElfScriptsW[i]);
|
|
else
|
|
FIXME("Unknown elfscript for bit %d\n", i);
|
|
}
|
|
}
|
|
TRACE("enuming face %s full %s style %s charset %d type %ld script %s it %d weight %ld ntmflags %08lx\n",
|
|
debugstr_w(elf.elfLogFont.lfFaceName),
|
|
debugstr_w(elf.elfFullName), debugstr_w(elf.elfStyle),
|
|
csi.ciCharset, type, debugstr_w(elf.elfScript),
|
|
elf.elfLogFont.lfItalic, elf.elfLogFont.lfWeight,
|
|
ntm.ntmTm.ntmFlags);
|
|
ret = proc(&elf.elfLogFont, (TEXTMETRICW *)&ntm, type, lparam);
|
|
if(!ret) goto end;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
LIST_FOR_EACH(family_elem_ptr, &font_list) {
|
|
family = LIST_ENTRY(family_elem_ptr, Family, entry);
|
|
face_elem_ptr = list_head(&family->faces);
|
|
face = LIST_ENTRY(face_elem_ptr, Face, entry);
|
|
GetEnumStructs(face, &elf, &ntm, &type);
|
|
for(i = 0; i < 32; i++) {
|
|
if(!face->scalable && face->fs.fsCsb[0] == 0) { /* OEM bitmap */
|
|
elf.elfLogFont.lfCharSet = ntm.ntmTm.tmCharSet = OEM_CHARSET;
|
|
strcpyW(elf.elfScript, OEM_DOSW);
|
|
i = 32; /* break out of loop */
|
|
} else if(!(face->fs.fsCsb[0] & (1L << i)))
|
|
continue;
|
|
else {
|
|
fs.fsCsb[0] = 1L << i;
|
|
fs.fsCsb[1] = 0;
|
|
if(!TranslateCharsetInfo(fs.fsCsb, &csi,
|
|
TCI_SRCFONTSIG))
|
|
csi.ciCharset = DEFAULT_CHARSET;
|
|
if(i == 31) csi.ciCharset = SYMBOL_CHARSET;
|
|
if(csi.ciCharset != DEFAULT_CHARSET) {
|
|
elf.elfLogFont.lfCharSet = ntm.ntmTm.tmCharSet =
|
|
csi.ciCharset;
|
|
if(ElfScriptsW[i])
|
|
strcpyW(elf.elfScript, ElfScriptsW[i]);
|
|
else
|
|
FIXME("Unknown elfscript for bit %d\n", i);
|
|
}
|
|
}
|
|
TRACE("enuming face %s full %s style %s charset = %d type %ld script %s it %d weight %ld ntmflags %08lx\n",
|
|
debugstr_w(elf.elfLogFont.lfFaceName),
|
|
debugstr_w(elf.elfFullName), debugstr_w(elf.elfStyle),
|
|
csi.ciCharset, type, debugstr_w(elf.elfScript),
|
|
elf.elfLogFont.lfItalic, elf.elfLogFont.lfWeight,
|
|
ntm.ntmTm.ntmFlags);
|
|
ret = proc(&elf.elfLogFont, (TEXTMETRICW *)&ntm, type, lparam);
|
|
if(!ret) goto end;
|
|
}
|
|
}
|
|
}
|
|
end:
|
|
return ret;
|
|
}
|
|
|
|
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));
|
|
return;
|
|
}
|
|
|
|
static FT_UInt get_glyph_index(GdiFont font, UINT glyph)
|
|
{
|
|
if(font->ft_face->charmap->encoding == FT_ENCODING_NONE) {
|
|
WCHAR wc = (WCHAR)glyph;
|
|
unsigned char buf;
|
|
WideCharToMultiByte(font->codepage, 0, &wc, 1, &buf, sizeof(buf), 0, 0);
|
|
return pFT_Get_Char_Index(font->ft_face, buf);
|
|
}
|
|
|
|
if(font->charset == SYMBOL_CHARSET && glyph < 0x100)
|
|
glyph = glyph + 0xf000;
|
|
return pFT_Get_Char_Index(font->ft_face, glyph);
|
|
}
|
|
|
|
/*************************************************************
|
|
* WineEngGetGlyphIndices
|
|
*
|
|
* FIXME: add support for GGI_MARK_NONEXISTING_GLYPHS
|
|
*/
|
|
DWORD WineEngGetGlyphIndices(GdiFont font, LPCWSTR lpstr, INT count,
|
|
LPWORD pgi, DWORD flags)
|
|
{
|
|
INT i;
|
|
|
|
for(i = 0; i < count; i++)
|
|
pgi[i] = get_glyph_index(font, lpstr[i]);
|
|
|
|
return count;
|
|
}
|
|
|
|
/*************************************************************
|
|
* WineEngGetGlyphOutline
|
|
*
|
|
* Behaves in exactly the same way as the win32 api GetGlyphOutline
|
|
* except that the first parameter is the HWINEENGFONT of the font in
|
|
* question rather than an HDC.
|
|
*
|
|
*/
|
|
DWORD WineEngGetGlyphOutline(GdiFont font, UINT glyph, UINT format,
|
|
LPGLYPHMETRICS lpgm, DWORD buflen, LPVOID buf,
|
|
const MAT2* lpmat)
|
|
{
|
|
static const FT_Matrix identityMat = {(1 << 16), 0, 0, (1 << 16)};
|
|
FT_Face ft_face = font->ft_face;
|
|
FT_UInt glyph_index;
|
|
DWORD width, height, pitch, needed = 0;
|
|
FT_Bitmap ft_bitmap;
|
|
FT_Error err;
|
|
INT left, right, top = 0, bottom = 0;
|
|
FT_Angle angle = 0;
|
|
FT_Int load_flags = FT_LOAD_DEFAULT | FT_LOAD_IGNORE_GLOBAL_ADVANCE_WIDTH;
|
|
float widthRatio = 1.0;
|
|
FT_Matrix transMat = identityMat;
|
|
BOOL needsTransform = FALSE;
|
|
|
|
|
|
TRACE("%p, %04x, %08x, %p, %08lx, %p, %p\n", font, glyph, format, lpgm,
|
|
buflen, buf, lpmat);
|
|
|
|
if(format & GGO_GLYPH_INDEX) {
|
|
glyph_index = glyph;
|
|
format &= ~GGO_GLYPH_INDEX;
|
|
} else
|
|
glyph_index = get_glyph_index(font, glyph);
|
|
|
|
if(glyph_index >= font->gmsize) {
|
|
font->gmsize = (glyph_index / INIT_GM_SIZE + 1) * INIT_GM_SIZE;
|
|
font->gm = HeapReAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, font->gm,
|
|
font->gmsize * sizeof(*font->gm));
|
|
} else {
|
|
if(format == GGO_METRICS && font->gm[glyph_index].init) {
|
|
memcpy(lpgm, &font->gm[glyph_index].gm, sizeof(*lpgm));
|
|
return 1; /* FIXME */
|
|
}
|
|
}
|
|
|
|
if(font->orientation || (format != GGO_METRICS && format != GGO_BITMAP) || font->aveWidth || lpmat)
|
|
load_flags |= FT_LOAD_NO_BITMAP;
|
|
|
|
err = pFT_Load_Glyph(ft_face, glyph_index, load_flags);
|
|
|
|
if(err) {
|
|
FIXME("FT_Load_Glyph on index %x returns %d\n", glyph_index, err);
|
|
return GDI_ERROR;
|
|
}
|
|
|
|
/* Scaling factor */
|
|
if (font->aveWidth && font->potm) {
|
|
widthRatio = (float)font->aveWidth * font->font_desc.matrix.eM11 / (float) font->potm->otmTextMetrics.tmAveCharWidth;
|
|
}
|
|
|
|
left = (INT)(ft_face->glyph->metrics.horiBearingX * widthRatio) & -64;
|
|
right = (INT)((ft_face->glyph->metrics.horiBearingX + ft_face->glyph->metrics.width) * widthRatio + 63) & -64;
|
|
|
|
font->gm[glyph_index].adv = (INT)((ft_face->glyph->metrics.horiAdvance * widthRatio) + 63) >> 6;
|
|
font->gm[glyph_index].lsb = left >> 6;
|
|
font->gm[glyph_index].bbx = (right - left) >> 6;
|
|
|
|
/* Scaling transform */
|
|
if(font->aveWidth) {
|
|
FT_Matrix scaleMat;
|
|
scaleMat.xx = FT_FixedFromFloat(widthRatio);
|
|
scaleMat.xy = 0;
|
|
scaleMat.yx = 0;
|
|
scaleMat.yy = (1 << 16);
|
|
|
|
pFT_Matrix_Multiply(&scaleMat, &transMat);
|
|
needsTransform = TRUE;
|
|
}
|
|
|
|
/* Rotation transform */
|
|
if(font->orientation) {
|
|
FT_Matrix rotationMat;
|
|
FT_Vector vecAngle;
|
|
angle = FT_FixedFromFloat((float)font->orientation / 10.0);
|
|
pFT_Vector_Unit(&vecAngle, angle);
|
|
rotationMat.xx = vecAngle.x;
|
|
rotationMat.xy = -vecAngle.y;
|
|
rotationMat.yx = -rotationMat.xy;
|
|
rotationMat.yy = rotationMat.xx;
|
|
|
|
pFT_Matrix_Multiply(&rotationMat, &transMat);
|
|
needsTransform = TRUE;
|
|
}
|
|
|
|
/* Extra transformation specified by caller */
|
|
if (lpmat) {
|
|
FT_Matrix extraMat;
|
|
extraMat.xx = FT_FixedFromFIXED(lpmat->eM11);
|
|
extraMat.xy = FT_FixedFromFIXED(lpmat->eM21);
|
|
extraMat.yx = FT_FixedFromFIXED(lpmat->eM12);
|
|
extraMat.yy = FT_FixedFromFIXED(lpmat->eM22);
|
|
pFT_Matrix_Multiply(&extraMat, &transMat);
|
|
needsTransform = TRUE;
|
|
}
|
|
|
|
if(!needsTransform) {
|
|
top = (ft_face->glyph->metrics.horiBearingY + 63) & -64;
|
|
bottom = (ft_face->glyph->metrics.horiBearingY -
|
|
ft_face->glyph->metrics.height) & -64;
|
|
lpgm->gmCellIncX = font->gm[glyph_index].adv;
|
|
lpgm->gmCellIncY = 0;
|
|
} else {
|
|
INT xc, yc;
|
|
FT_Vector vec;
|
|
for(xc = 0; xc < 2; xc++) {
|
|
for(yc = 0; yc < 2; yc++) {
|
|
vec.x = (ft_face->glyph->metrics.horiBearingX +
|
|
xc * ft_face->glyph->metrics.width);
|
|
vec.y = ft_face->glyph->metrics.horiBearingY -
|
|
yc * ft_face->glyph->metrics.height;
|
|
TRACE("Vec %ld,%ld\n", vec.x, vec.y);
|
|
pFT_Vector_Transform(&vec, &transMat);
|
|
if(xc == 0 && yc == 0) {
|
|
left = right = vec.x;
|
|
top = bottom = vec.y;
|
|
} else {
|
|
if(vec.x < left) left = vec.x;
|
|
else if(vec.x > right) right = vec.x;
|
|
if(vec.y < bottom) bottom = vec.y;
|
|
else if(vec.y > top) top = vec.y;
|
|
}
|
|
}
|
|
}
|
|
left = left & -64;
|
|
right = (right + 63) & -64;
|
|
bottom = bottom & -64;
|
|
top = (top + 63) & -64;
|
|
|
|
TRACE("transformed box: (%d,%d - %d,%d)\n", left, top, right, bottom);
|
|
vec.x = ft_face->glyph->metrics.horiAdvance;
|
|
vec.y = 0;
|
|
pFT_Vector_Transform(&vec, &transMat);
|
|
lpgm->gmCellIncX = (vec.x+63) >> 6;
|
|
lpgm->gmCellIncY = -((vec.y+63) >> 6);
|
|
}
|
|
lpgm->gmBlackBoxX = (right - left) >> 6;
|
|
lpgm->gmBlackBoxY = (top - bottom) >> 6;
|
|
lpgm->gmptGlyphOrigin.x = left >> 6;
|
|
lpgm->gmptGlyphOrigin.y = top >> 6;
|
|
|
|
memcpy(&font->gm[glyph_index].gm, lpgm, sizeof(*lpgm));
|
|
font->gm[glyph_index].init = TRUE;
|
|
|
|
if(format == GGO_METRICS)
|
|
return 1; /* FIXME */
|
|
|
|
if (buf && !buflen){
|
|
return GDI_ERROR;
|
|
}
|
|
|
|
if(ft_face->glyph->format != ft_glyph_format_outline && format != GGO_BITMAP) {
|
|
TRACE("loaded a bitmap\n");
|
|
return GDI_ERROR;
|
|
}
|
|
|
|
switch(format) {
|
|
case GGO_BITMAP:
|
|
width = lpgm->gmBlackBoxX;
|
|
height = lpgm->gmBlackBoxY;
|
|
pitch = ((width + 31) >> 5) << 2;
|
|
needed = pitch * height;
|
|
|
|
if(!buf || !buflen) break;
|
|
|
|
switch(ft_face->glyph->format) {
|
|
case ft_glyph_format_bitmap:
|
|
{
|
|
BYTE *src = ft_face->glyph->bitmap.buffer, *dst = buf;
|
|
INT w = (ft_face->glyph->bitmap.width + 7) >> 3;
|
|
INT h = ft_face->glyph->bitmap.rows;
|
|
while(h--) {
|
|
memcpy(dst, src, w);
|
|
src += ft_face->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(needsTransform) {
|
|
pFT_Outline_Transform(&ft_face->glyph->outline, &transMat);
|
|
}
|
|
|
|
pFT_Outline_Translate(&ft_face->glyph->outline, -left, -bottom );
|
|
|
|
/* Note: FreeType will only set 'black' bits for us. */
|
|
memset(buf, 0, needed);
|
|
pFT_Outline_Get_Bitmap(library, &ft_face->glyph->outline, &ft_bitmap);
|
|
break;
|
|
|
|
default:
|
|
FIXME("loaded glyph format %x\n", ft_face->glyph->format);
|
|
return GDI_ERROR;
|
|
}
|
|
break;
|
|
|
|
case GGO_GRAY2_BITMAP:
|
|
case GGO_GRAY4_BITMAP:
|
|
case GGO_GRAY8_BITMAP:
|
|
case WINE_GGO_GRAY16_BITMAP:
|
|
{
|
|
unsigned int mult, row, col;
|
|
BYTE *start, *ptr;
|
|
|
|
width = lpgm->gmBlackBoxX;
|
|
height = lpgm->gmBlackBoxY;
|
|
pitch = (width + 3) / 4 * 4;
|
|
needed = pitch * height;
|
|
|
|
if(!buf || !buflen) break;
|
|
ft_bitmap.width = width;
|
|
ft_bitmap.rows = height;
|
|
ft_bitmap.pitch = pitch;
|
|
ft_bitmap.pixel_mode = ft_pixel_mode_grays;
|
|
ft_bitmap.buffer = buf;
|
|
|
|
if(needsTransform) {
|
|
pFT_Outline_Transform(&ft_face->glyph->outline, &transMat);
|
|
}
|
|
|
|
pFT_Outline_Translate(&ft_face->glyph->outline, -left, -bottom );
|
|
|
|
pFT_Outline_Get_Bitmap(library, &ft_face->glyph->outline, &ft_bitmap);
|
|
|
|
if(format == GGO_GRAY2_BITMAP)
|
|
mult = 5;
|
|
else if(format == GGO_GRAY4_BITMAP)
|
|
mult = 17;
|
|
else if(format == GGO_GRAY8_BITMAP)
|
|
mult = 65;
|
|
else if(format == WINE_GGO_GRAY16_BITMAP)
|
|
break;
|
|
else {
|
|
assert(0);
|
|
break;
|
|
}
|
|
|
|
start = buf;
|
|
for(row = 0; row < height; row++) {
|
|
ptr = start;
|
|
for(col = 0; col < width; col++, ptr++) {
|
|
*ptr = (*(unsigned int*)ptr * mult + 128) / 256;
|
|
}
|
|
start += pitch;
|
|
}
|
|
break;
|
|
}
|
|
|
|
case GGO_NATIVE:
|
|
{
|
|
int contour, point = 0, first_pt;
|
|
FT_Outline *outline = &ft_face->glyph->outline;
|
|
TTPOLYGONHEADER *pph;
|
|
TTPOLYCURVE *ppc;
|
|
DWORD pph_start, cpfx, type;
|
|
|
|
if(buflen == 0) buf = NULL;
|
|
|
|
if (needsTransform && buf) {
|
|
pFT_Outline_Transform(outline, &transMat);
|
|
}
|
|
|
|
for(contour = 0; contour < outline->n_contours; contour++) {
|
|
pph_start = needed;
|
|
pph = (TTPOLYGONHEADER *)((char *)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 *)((char *)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;
|
|
}
|
|
break;
|
|
}
|
|
case GGO_BEZIER:
|
|
{
|
|
/* 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 Beizer 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;
|
|
FT_Outline *outline = &ft_face->glyph->outline;
|
|
TTPOLYGONHEADER *pph;
|
|
TTPOLYCURVE *ppc;
|
|
DWORD pph_start, cpfx, type;
|
|
FT_Vector cubic_control[4];
|
|
if(buflen == 0) buf = NULL;
|
|
|
|
if (needsTransform && buf) {
|
|
pFT_Outline_Transform(outline, &transMat);
|
|
}
|
|
|
|
for(contour = 0; contour < outline->n_contours; contour++) {
|
|
pph_start = needed;
|
|
pph = (TTPOLYGONHEADER *)((char *)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 *)((char *)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;
|
|
}
|
|
break;
|
|
}
|
|
|
|
default:
|
|
FIXME("Unsupported format %d\n", format);
|
|
return GDI_ERROR;
|
|
}
|
|
return needed;
|
|
}
|
|
|
|
static BOOL get_bitmap_text_metrics(GdiFont font)
|
|
{
|
|
FT_Face ft_face = font->ft_face;
|
|
#ifdef HAVE_FREETYPE_FTWINFNT_H
|
|
FT_WinFNT_HeaderRec winfnt_header;
|
|
#endif
|
|
const DWORD size = offsetof(OUTLINETEXTMETRICW, otmFiller);
|
|
font->potm = HeapAlloc(GetProcessHeap(), 0, size);
|
|
font->potm->otmSize = size;
|
|
|
|
#define TM font->potm->otmTextMetrics
|
|
#ifdef HAVE_FREETYPE_FTWINFNT_H
|
|
if(pFT_Get_WinFNT_Header && !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.tmUnderlined = font->underline;
|
|
TM.tmStruckOut = font->strikeout;
|
|
TM.tmPitchAndFamily = winfnt_header.pitch_and_family;
|
|
TM.tmCharSet = winfnt_header.charset;
|
|
}
|
|
else
|
|
#endif
|
|
{
|
|
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;
|
|
TM.tmUnderlined = font->underline;
|
|
TM.tmStruckOut = font->strikeout;
|
|
/* NB inverted meaning of TMPF_FIXED_PITCH */
|
|
TM.tmPitchAndFamily = ft_face->face_flags & FT_FACE_FLAG_FIXED_WIDTH ? 0 : TMPF_FIXED_PITCH;
|
|
TM.tmCharSet = font->charset;
|
|
}
|
|
#undef TM
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/*************************************************************
|
|
* WineEngGetTextMetrics
|
|
*
|
|
*/
|
|
BOOL WineEngGetTextMetrics(GdiFont font, LPTEXTMETRICW ptm)
|
|
{
|
|
if(!font->potm) {
|
|
if(!WineEngGetOutlineTextMetrics(font, 0, NULL))
|
|
if(!get_bitmap_text_metrics(font))
|
|
return FALSE;
|
|
}
|
|
if(!font->potm) return FALSE;
|
|
memcpy(ptm, &font->potm->otmTextMetrics, sizeof(*ptm));
|
|
|
|
if (font->aveWidth) {
|
|
ptm->tmAveCharWidth = font->aveWidth * font->font_desc.matrix.eM11;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
/*************************************************************
|
|
* WineEngGetOutlineTextMetrics
|
|
*
|
|
*/
|
|
UINT WineEngGetOutlineTextMetrics(GdiFont font, UINT cbSize,
|
|
OUTLINETEXTMETRICW *potm)
|
|
{
|
|
FT_Face ft_face = font->ft_face;
|
|
UINT needed, lenfam, lensty, ret;
|
|
TT_OS2 *pOS2;
|
|
TT_HoriHeader *pHori;
|
|
TT_Postscript *pPost;
|
|
FT_Fixed x_scale, y_scale;
|
|
WCHAR *family_nameW, *style_nameW;
|
|
static const WCHAR spaceW[] = {' ', '\0'};
|
|
char *cp;
|
|
INT ascent, descent;
|
|
|
|
TRACE("font=%p\n", font);
|
|
|
|
if(!FT_IS_SCALABLE(ft_face))
|
|
return 0;
|
|
|
|
if(font->potm) {
|
|
if(cbSize >= font->potm->otmSize)
|
|
memcpy(potm, font->potm, font->potm->otmSize);
|
|
return font->potm->otmSize;
|
|
}
|
|
|
|
|
|
needed = sizeof(*potm);
|
|
|
|
lenfam = (strlenW(font->name) + 1) * sizeof(WCHAR);
|
|
family_nameW = strdupW(font->name);
|
|
|
|
lensty = MultiByteToWideChar(CP_ACP, 0, ft_face->style_name, -1, NULL, 0)
|
|
* sizeof(WCHAR);
|
|
style_nameW = HeapAlloc(GetProcessHeap(), 0, lensty);
|
|
MultiByteToWideChar(CP_ACP, 0, ft_face->style_name, -1,
|
|
style_nameW, lensty);
|
|
|
|
/* These names should be read from the TT name table */
|
|
|
|
/* length of otmpFamilyName */
|
|
needed += lenfam;
|
|
|
|
/* length of otmpFaceName */
|
|
if(!strcasecmp(ft_face->style_name, "regular")) {
|
|
needed += lenfam; /* just the family name */
|
|
} else {
|
|
needed += lenfam + lensty; /* family + " " + style */
|
|
}
|
|
|
|
/* length of otmpStyleName */
|
|
needed += lensty;
|
|
|
|
/* length of otmpFullName */
|
|
needed += lenfam + lensty;
|
|
|
|
|
|
x_scale = ft_face->size->metrics.x_scale;
|
|
y_scale = ft_face->size->metrics.y_scale;
|
|
|
|
pOS2 = pFT_Get_Sfnt_Table(ft_face, ft_sfnt_os2);
|
|
if(!pOS2) {
|
|
FIXME("Can't find OS/2 table - not TT font?\n");
|
|
ret = 0;
|
|
goto end;
|
|
}
|
|
|
|
pHori = pFT_Get_Sfnt_Table(ft_face, ft_sfnt_hhea);
|
|
if(!pHori) {
|
|
FIXME("Can't find HHEA table - not TT font?\n");
|
|
ret = 0;
|
|
goto end;
|
|
}
|
|
|
|
pPost = pFT_Get_Sfnt_Table(ft_face, ft_sfnt_post); /* we can live with this failing */
|
|
|
|
TRACE("OS/2 winA = %d winD = %d typoA = %d typoD = %d typoLG = %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,
|
|
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->potm = HeapAlloc(GetProcessHeap(), 0, needed);
|
|
font->potm->otmSize = needed;
|
|
|
|
#define TM font->potm->otmTextMetrics
|
|
|
|
if(pOS2->usWinAscent + pOS2->usWinDescent == 0) {
|
|
ascent = pHori->Ascender;
|
|
descent = -pHori->Descender;
|
|
} else {
|
|
ascent = pOS2->usWinAscent;
|
|
descent = pOS2->usWinDescent;
|
|
}
|
|
|
|
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 = (pFT_MulFix(ascent, y_scale) + 32) >> 6;
|
|
TM.tmDescent = (pFT_MulFix(descent, y_scale) + 32) >> 6;
|
|
TM.tmInternalLeading = (pFT_MulFix(ascent + descent
|
|
- ft_face->units_per_EM, y_scale) + 32) >> 6;
|
|
}
|
|
|
|
TM.tmHeight = TM.tmAscent + TM.tmDescent;
|
|
|
|
/* MSDN says:
|
|
el = MAX(0, LineGap - ((WinAscent + WinDescent) - (Ascender - Descender)))
|
|
*/
|
|
TM.tmExternalLeading = max(0, (pFT_MulFix(pHori->Line_Gap -
|
|
((ascent + descent) -
|
|
(pHori->Ascender - pHori->Descender)), y_scale) + 32) >> 6);
|
|
|
|
TM.tmAveCharWidth = (pFT_MulFix(pOS2->xAvgCharWidth, x_scale) + 32) >> 6;
|
|
if (TM.tmAveCharWidth == 0) {
|
|
TM.tmAveCharWidth = 1;
|
|
}
|
|
TM.tmMaxCharWidth = (pFT_MulFix(ft_face->bbox.xMax - ft_face->bbox.xMin, x_scale) + 32) >> 6;
|
|
TM.tmWeight = font->fake_bold ? FW_BOLD : pOS2->usWeightClass;
|
|
TM.tmOverhang = 0;
|
|
TM.tmDigitizedAspectX = 300;
|
|
TM.tmDigitizedAspectY = 300;
|
|
TM.tmFirstChar = pOS2->usFirstCharIndex;
|
|
TM.tmLastChar = pOS2->usLastCharIndex;
|
|
TM.tmDefaultChar = pOS2->usDefaultChar;
|
|
TM.tmBreakChar = pOS2->usBreakChar ? pOS2->usBreakChar : ' ';
|
|
TM.tmItalic = font->fake_italic ? 255 : ((ft_face->style_flags & FT_STYLE_FLAG_ITALIC) ? 255 : 0);
|
|
TM.tmUnderlined = font->underline;
|
|
TM.tmStruckOut = font->strikeout;
|
|
|
|
/* Yes TPMF_FIXED_PITCH is correct; braindead api */
|
|
if(!FT_IS_FIXED_WIDTH(ft_face))
|
|
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:
|
|
case PAN_FAMILY_PICTORIAL:
|
|
TM.tmPitchAndFamily |= FF_DECORATIVE;
|
|
break;
|
|
case PAN_FAMILY_TEXT_DISPLAY:
|
|
if(TM.tmPitchAndFamily == 0) /* fixed */
|
|
TM.tmPitchAndFamily = FF_MODERN;
|
|
else {
|
|
switch(pOS2->panose[PAN_SERIFSTYLE_INDEX]) {
|
|
case PAN_SERIF_NORMAL_SANS:
|
|
case PAN_SERIF_OBTUSE_SANS:
|
|
case PAN_SERIF_PERP_SANS:
|
|
TM.tmPitchAndFamily |= FF_SWISS;
|
|
break;
|
|
default:
|
|
TM.tmPitchAndFamily |= FF_ROMAN;
|
|
}
|
|
}
|
|
break;
|
|
default:
|
|
TM.tmPitchAndFamily |= FF_DONTCARE;
|
|
}
|
|
|
|
if(FT_IS_SCALABLE(ft_face))
|
|
TM.tmPitchAndFamily |= TMPF_VECTOR;
|
|
if(FT_IS_SFNT(ft_face))
|
|
TM.tmPitchAndFamily |= TMPF_TRUETYPE;
|
|
|
|
TM.tmCharSet = font->charset;
|
|
#undef TM
|
|
|
|
font->potm->otmFiller = 0;
|
|
memcpy(&font->potm->otmPanoseNumber, pOS2->panose, PANOSE_COUNT);
|
|
font->potm->otmfsSelection = pOS2->fsSelection;
|
|
font->potm->otmfsType = pOS2->fsType;
|
|
font->potm->otmsCharSlopeRise = pHori->caret_Slope_Rise;
|
|
font->potm->otmsCharSlopeRun = pHori->caret_Slope_Run;
|
|
font->potm->otmItalicAngle = 0; /* POST table */
|
|
font->potm->otmEMSquare = ft_face->units_per_EM;
|
|
font->potm->otmAscent = (pFT_MulFix(pOS2->sTypoAscender, y_scale) + 32) >> 6;
|
|
font->potm->otmDescent = (pFT_MulFix(pOS2->sTypoDescender, y_scale) + 32) >> 6;
|
|
font->potm->otmLineGap = (pFT_MulFix(pOS2->sTypoLineGap, y_scale) + 32) >> 6;
|
|
font->potm->otmsCapEmHeight = (pFT_MulFix(pOS2->sCapHeight, y_scale) + 32) >> 6;
|
|
font->potm->otmsXHeight = (pFT_MulFix(pOS2->sxHeight, y_scale) + 32) >> 6;
|
|
font->potm->otmrcFontBox.left = (pFT_MulFix(ft_face->bbox.xMin, x_scale) + 32) >> 6;
|
|
font->potm->otmrcFontBox.right = (pFT_MulFix(ft_face->bbox.xMax, x_scale) + 32) >> 6;
|
|
font->potm->otmrcFontBox.top = (pFT_MulFix(ft_face->bbox.yMax, y_scale) + 32) >> 6;
|
|
font->potm->otmrcFontBox.bottom = (pFT_MulFix(ft_face->bbox.yMin, y_scale) + 32) >> 6;
|
|
font->potm->otmMacAscent = 0; /* where do these come from ? */
|
|
font->potm->otmMacDescent = 0;
|
|
font->potm->otmMacLineGap = 0;
|
|
font->potm->otmusMinimumPPEM = 0; /* TT Header */
|
|
font->potm->otmptSubscriptSize.x = (pFT_MulFix(pOS2->ySubscriptXSize, x_scale) + 32) >> 6;
|
|
font->potm->otmptSubscriptSize.y = (pFT_MulFix(pOS2->ySubscriptYSize, y_scale) + 32) >> 6;
|
|
font->potm->otmptSubscriptOffset.x = (pFT_MulFix(pOS2->ySubscriptXOffset, x_scale) + 32) >> 6;
|
|
font->potm->otmptSubscriptOffset.y = (pFT_MulFix(pOS2->ySubscriptYOffset, y_scale) + 32) >> 6;
|
|
font->potm->otmptSuperscriptSize.x = (pFT_MulFix(pOS2->ySuperscriptXSize, x_scale) + 32) >> 6;
|
|
font->potm->otmptSuperscriptSize.y = (pFT_MulFix(pOS2->ySuperscriptYSize, y_scale) + 32) >> 6;
|
|
font->potm->otmptSuperscriptOffset.x = (pFT_MulFix(pOS2->ySuperscriptXOffset, x_scale) + 32) >> 6;
|
|
font->potm->otmptSuperscriptOffset.y = (pFT_MulFix(pOS2->ySuperscriptYOffset, y_scale) + 32) >> 6;
|
|
font->potm->otmsStrikeoutSize = (pFT_MulFix(pOS2->yStrikeoutSize, y_scale) + 32) >> 6;
|
|
font->potm->otmsStrikeoutPosition = (pFT_MulFix(pOS2->yStrikeoutPosition, y_scale) + 32) >> 6;
|
|
if(!pPost) {
|
|
font->potm->otmsUnderscoreSize = 0;
|
|
font->potm->otmsUnderscorePosition = 0;
|
|
} else {
|
|
font->potm->otmsUnderscoreSize = (pFT_MulFix(pPost->underlineThickness, y_scale) + 32) >> 6;
|
|
font->potm->otmsUnderscorePosition = (pFT_MulFix(pPost->underlinePosition, y_scale) + 32) >> 6;
|
|
}
|
|
|
|
/* otmp* members should clearly have type ptrdiff_t, but M$ knows best */
|
|
cp = (char*)font->potm + sizeof(*font->potm);
|
|
font->potm->otmpFamilyName = (LPSTR)(cp - (char*)font->potm);
|
|
strcpyW((WCHAR*)cp, family_nameW);
|
|
cp += lenfam;
|
|
font->potm->otmpStyleName = (LPSTR)(cp - (char*)font->potm);
|
|
strcpyW((WCHAR*)cp, style_nameW);
|
|
cp += lensty;
|
|
font->potm->otmpFaceName = (LPSTR)(cp - (char*)font->potm);
|
|
strcpyW((WCHAR*)cp, family_nameW);
|
|
if(strcasecmp(ft_face->style_name, "regular")) {
|
|
strcatW((WCHAR*)cp, spaceW);
|
|
strcatW((WCHAR*)cp, style_nameW);
|
|
cp += lenfam + lensty;
|
|
} else
|
|
cp += lenfam;
|
|
font->potm->otmpFullName = (LPSTR)(cp - (char*)font->potm);
|
|
strcpyW((WCHAR*)cp, family_nameW);
|
|
strcatW((WCHAR*)cp, spaceW);
|
|
strcatW((WCHAR*)cp, style_nameW);
|
|
ret = needed;
|
|
|
|
if(potm && needed <= cbSize)
|
|
memcpy(potm, font->potm, font->potm->otmSize);
|
|
|
|
end:
|
|
HeapFree(GetProcessHeap(), 0, style_nameW);
|
|
HeapFree(GetProcessHeap(), 0, family_nameW);
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
/*************************************************************
|
|
* WineEngGetCharWidth
|
|
*
|
|
*/
|
|
BOOL WineEngGetCharWidth(GdiFont font, UINT firstChar, UINT lastChar,
|
|
LPINT buffer)
|
|
{
|
|
UINT c;
|
|
GLYPHMETRICS gm;
|
|
FT_UInt glyph_index;
|
|
|
|
TRACE("%p, %d, %d, %p\n", font, firstChar, lastChar, buffer);
|
|
|
|
for(c = firstChar; c <= lastChar; c++) {
|
|
glyph_index = get_glyph_index(font, c);
|
|
WineEngGetGlyphOutline(font, glyph_index, GGO_METRICS | GGO_GLYPH_INDEX,
|
|
&gm, 0, NULL, NULL);
|
|
buffer[c - firstChar] = font->gm[glyph_index].adv;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
/*************************************************************
|
|
* WineEngGetCharABCWidths
|
|
*
|
|
*/
|
|
BOOL WineEngGetCharABCWidths(GdiFont font, UINT firstChar, UINT lastChar,
|
|
LPABC buffer)
|
|
{
|
|
UINT c;
|
|
GLYPHMETRICS gm;
|
|
FT_UInt glyph_index;
|
|
|
|
TRACE("%p, %d, %d, %p\n", font, firstChar, lastChar, buffer);
|
|
|
|
if(!FT_IS_SCALABLE(font->ft_face))
|
|
return FALSE;
|
|
|
|
for(c = firstChar; c <= lastChar; c++) {
|
|
glyph_index = get_glyph_index(font, c);
|
|
WineEngGetGlyphOutline(font, glyph_index, GGO_METRICS | GGO_GLYPH_INDEX,
|
|
&gm, 0, NULL, NULL);
|
|
buffer[c - firstChar].abcA = font->gm[glyph_index].lsb;
|
|
buffer[c - firstChar].abcB = font->gm[glyph_index].bbx;
|
|
buffer[c - firstChar].abcC = font->gm[glyph_index].adv - font->gm[glyph_index].lsb -
|
|
font->gm[glyph_index].bbx;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
/*************************************************************
|
|
* WineEngGetTextExtentPoint
|
|
*
|
|
*/
|
|
BOOL WineEngGetTextExtentPoint(GdiFont font, LPCWSTR wstr, INT count,
|
|
LPSIZE size)
|
|
{
|
|
INT idx;
|
|
GLYPHMETRICS gm;
|
|
TEXTMETRICW tm;
|
|
FT_UInt glyph_index;
|
|
|
|
TRACE("%p, %s, %d, %p\n", font, debugstr_wn(wstr, count), count,
|
|
size);
|
|
|
|
size->cx = 0;
|
|
WineEngGetTextMetrics(font, &tm);
|
|
size->cy = tm.tmHeight;
|
|
|
|
for(idx = 0; idx < count; idx++) {
|
|
glyph_index = get_glyph_index(font, wstr[idx]);
|
|
WineEngGetGlyphOutline(font, glyph_index, GGO_METRICS | GGO_GLYPH_INDEX,
|
|
&gm, 0, NULL, NULL);
|
|
size->cx += font->gm[glyph_index].adv;
|
|
}
|
|
TRACE("return %ld,%ld\n", size->cx, size->cy);
|
|
return TRUE;
|
|
}
|
|
|
|
/*************************************************************
|
|
* WineEngGetTextExtentPointI
|
|
*
|
|
*/
|
|
BOOL WineEngGetTextExtentPointI(GdiFont font, const WORD *indices, INT count,
|
|
LPSIZE size)
|
|
{
|
|
INT idx;
|
|
GLYPHMETRICS gm;
|
|
TEXTMETRICW tm;
|
|
|
|
TRACE("%p, %p, %d, %p\n", font, indices, count, size);
|
|
|
|
size->cx = 0;
|
|
WineEngGetTextMetrics(font, &tm);
|
|
size->cy = tm.tmHeight;
|
|
|
|
for(idx = 0; idx < count; idx++) {
|
|
WineEngGetGlyphOutline(font, indices[idx],
|
|
GGO_METRICS | GGO_GLYPH_INDEX, &gm, 0, NULL,
|
|
NULL);
|
|
size->cx += font->gm[indices[idx]].adv;
|
|
}
|
|
TRACE("return %ld,%ld\n", size->cx, size->cy);
|
|
return TRUE;
|
|
}
|
|
|
|
/*************************************************************
|
|
* WineEngGetFontData
|
|
*
|
|
*/
|
|
DWORD WineEngGetFontData(GdiFont font, DWORD table, DWORD offset, LPVOID buf,
|
|
DWORD cbData)
|
|
{
|
|
FT_Face ft_face = font->ft_face;
|
|
DWORD len;
|
|
FT_Error err;
|
|
|
|
TRACE("font=%p, table=%08lx, offset=%08lx, buf=%p, cbData=%lx\n",
|
|
font, table, offset, buf, cbData);
|
|
|
|
if(!FT_IS_SFNT(ft_face))
|
|
return GDI_ERROR;
|
|
|
|
if(!buf || !cbData)
|
|
len = 0;
|
|
else
|
|
len = cbData;
|
|
|
|
if(table) { /* MS tags differ in endidness from FT ones */
|
|
table = table >> 24 | table << 24 |
|
|
(table >> 8 & 0xff00) | (table << 8 & 0xff0000);
|
|
}
|
|
|
|
/* If the FT_Load_Sfnt_Table function is there we'll use it */
|
|
if(pFT_Load_Sfnt_Table)
|
|
err = pFT_Load_Sfnt_Table(ft_face, table, offset, buf, &len);
|
|
else { /* Do it the hard way */
|
|
TT_Face tt_face = (TT_Face) ft_face;
|
|
SFNT_Interface *sfnt;
|
|
if (FT_Version.major==2 && FT_Version.minor==0)
|
|
{
|
|
/* 2.0.x */
|
|
sfnt = *(SFNT_Interface**)((char*)tt_face + 528);
|
|
}
|
|
else
|
|
{
|
|
/* A field was added in the middle of the structure in 2.1.x */
|
|
sfnt = *(SFNT_Interface**)((char*)tt_face + 532);
|
|
}
|
|
err = sfnt->load_any(tt_face, table, offset, buf, &len);
|
|
}
|
|
if(err) {
|
|
TRACE("Can't find table %08lx.\n", table);
|
|
return GDI_ERROR;
|
|
}
|
|
return len;
|
|
}
|
|
|
|
/*************************************************************
|
|
* WineEngGetTextFace
|
|
*
|
|
*/
|
|
INT WineEngGetTextFace(GdiFont font, INT count, LPWSTR str)
|
|
{
|
|
if(str) {
|
|
lstrcpynW(str, font->name, count);
|
|
return strlenW(font->name);
|
|
} else
|
|
return strlenW(font->name) + 1;
|
|
}
|
|
|
|
UINT WineEngGetTextCharsetInfo(GdiFont font, LPFONTSIGNATURE fs, DWORD flags)
|
|
{
|
|
if (fs) memcpy(fs, &font->fs, sizeof(FONTSIGNATURE));
|
|
return font->charset;
|
|
}
|
|
|
|
#else /* HAVE_FREETYPE */
|
|
|
|
BOOL WineEngInit(void)
|
|
{
|
|
return FALSE;
|
|
}
|
|
GdiFont WineEngCreateFontInstance(DC *dc, HFONT hfont)
|
|
{
|
|
return NULL;
|
|
}
|
|
BOOL WineEngDestroyFontInstance(HFONT hfont)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
DWORD WineEngEnumFonts(LPLOGFONTW plf, FONTENUMPROCW proc, LPARAM lparam)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
DWORD WineEngGetGlyphIndices(GdiFont font, LPCWSTR lpstr, INT count,
|
|
LPWORD pgi, DWORD flags)
|
|
{
|
|
return GDI_ERROR;
|
|
}
|
|
|
|
DWORD WineEngGetGlyphOutline(GdiFont font, UINT glyph, UINT format,
|
|
LPGLYPHMETRICS lpgm, DWORD buflen, LPVOID buf,
|
|
const MAT2* lpmat)
|
|
{
|
|
ERR("called but we don't have FreeType\n");
|
|
return GDI_ERROR;
|
|
}
|
|
|
|
BOOL WineEngGetTextMetrics(GdiFont font, LPTEXTMETRICW ptm)
|
|
{
|
|
ERR("called but we don't have FreeType\n");
|
|
return FALSE;
|
|
}
|
|
|
|
UINT WineEngGetOutlineTextMetrics(GdiFont font, UINT cbSize,
|
|
OUTLINETEXTMETRICW *potm)
|
|
{
|
|
ERR("called but we don't have FreeType\n");
|
|
return 0;
|
|
}
|
|
|
|
BOOL WineEngGetCharWidth(GdiFont font, UINT firstChar, UINT lastChar,
|
|
LPINT buffer)
|
|
{
|
|
ERR("called but we don't have FreeType\n");
|
|
return FALSE;
|
|
}
|
|
|
|
BOOL WineEngGetCharABCWidths(GdiFont font, UINT firstChar, UINT lastChar,
|
|
LPABC buffer)
|
|
{
|
|
ERR("called but we don't have FreeType\n");
|
|
return FALSE;
|
|
}
|
|
|
|
BOOL WineEngGetTextExtentPoint(GdiFont font, LPCWSTR wstr, INT count,
|
|
LPSIZE size)
|
|
{
|
|
ERR("called but we don't have FreeType\n");
|
|
return FALSE;
|
|
}
|
|
|
|
BOOL WineEngGetTextExtentPointI(GdiFont font, const WORD *indices, INT count,
|
|
LPSIZE size)
|
|
{
|
|
ERR("called but we don't have FreeType\n");
|
|
return FALSE;
|
|
}
|
|
|
|
DWORD WineEngGetFontData(GdiFont font, DWORD table, DWORD offset, LPVOID buf,
|
|
DWORD cbData)
|
|
{
|
|
ERR("called but we don't have FreeType\n");
|
|
return GDI_ERROR;
|
|
}
|
|
|
|
INT WineEngGetTextFace(GdiFont font, INT count, LPWSTR str)
|
|
{
|
|
ERR("called but we don't have FreeType\n");
|
|
return 0;
|
|
}
|
|
|
|
INT WineEngAddFontResourceEx(LPCWSTR file, DWORD flags, PVOID pdv)
|
|
{
|
|
FIXME(":stub\n");
|
|
return 1;
|
|
}
|
|
|
|
INT WineEngRemoveFontResourceEx(LPCWSTR file, DWORD flags, PVOID pdv)
|
|
{
|
|
FIXME(":stub\n");
|
|
return TRUE;
|
|
}
|
|
|
|
UINT WineEngGetTextCharsetInfo(GdiFont font, LPFONTSIGNATURE fs, DWORD flags)
|
|
{
|
|
FIXME(":stub\n");
|
|
return DEFAULT_CHARSET;
|
|
}
|
|
|
|
#endif /* HAVE_FREETYPE */
|