mirror of
git://source.winehq.org/git/wine.git
synced 2024-11-05 18:01:34 +00:00
2ace16ac08
Sun Apr 28 14:32:43 1996 Alexandre Julliard <julliard@lrc.epfl.ch> * [Makefile.in] Subdir memory is now also compiled for Winelib, in order to get the Win32 heap functions. * [if1632/Makefile.in] Renamed winprocs and winprocs32 to wprocs and wprocs32 to avoid DLL names > 8 characters. * [loader/builtin.c] (New file) Grouped all built-in DLLs code in a single file. * [memory/global.c] Use the Win32 heap code instead of malloc() to allocate linear memory. This will help test the heap code. * [memory/local.c] Fixed FreeSelector() to clear DS and ES correctly for huge blocks. * [tools/build.c] [if1632/relay.c] Removed 'id' directive in spec files. For relay debugging, the DLL entry point is now computed from the CS:IP entry point address. Added 'heap' directive to specifiy a local heap for the DLL. USER and GDI heap are now created this way. * [windows/class.c] [include/class.h] Changed the class structure to use pointers instead of handles. Changed Get/SetClassWord/Long to use a switch statement; this allows changing the layout of the CLASS structure. * [windows/win.c] [include/win.h] Use a CLASS * instead of a handle for the window class. Sat Apr 27 18:10:11 Martin von Loewis <loewis@informatik.hu-berlin.de> * [if1632/kernel32.spec] [memory/global.c] [win32/memory.c] [win32/process.c] GetProcessAffinityMask,GlobalLock,IsBadReadPtr,IsBadWritePtr, LocalLock,SetThreadAffinityMask: new relays. * [win32/cursoricon32.c] Return same handle if a cursor is loaded multiple times. Sat Apr 27 15:13:37 1996 Bang Jun Young <bangjy@nownuri.nowcom.co.kr> * [resources/sysres_Ko.rc] Added support for Korean [Ko] language. Fri Apr 26 00:49:05 1996 Huw D. M. Davies <h.davies1@physics.oxford.ac.uk> * [objects/dc.c] [objects/font.c] Fixed problem with SaveDC()/RestoreDC() and font cache 'used' count. * [objects/metafile.c] [objects/dcvalues.c] Fixed broken SetTextAlign() on metafiles. * [objects/metafile.c] Delete objects in handle table at end of PlayMetaFile(). Wed Apr 24 19:21:01 Marcus Meissner <msmeissn@cip.informatik.uni-erlangen.de> * [if1632/ver.spec] [misc/ver.c] [include/ver.h] (New files) VER.DLL (partially) implemented (VerFindFile,VerInstallFile) [If it doesn't work for you, use -dll -ver and report it to me] * [if1632/user32.spec] [if1632/kernel32.spec] [if1632/shell.spec] [if1632/shell32.spec] [misc/ole2nls.c] [windows/message.c] [windows/graphics.c] Simple win32 functions, where we can just use the win16 counterpart. Misc. stubs. * [misc/lstr.c] Someone reported a _lstrlen(NULL). NULL is a valid argument. Fixed. * [misc/registry.c] Some alloclens were off by 1, one double fclose() fixed. Requesting value 0 of a key with no values returns an error (should we always return a made up value NULL? what does win3.1?) Tue Apr 23 17:00:00 1996 Alex Korobka <alex@phm30.pharm.sunysb.edu> * [misc/shell.c] Implemented FindEnvironmentString(), DoEnvironmentSubst(), ExtractIcon(), InternalExtractIcon() and ExtractAssociatedIcon(). * [misc/user.c] Do extensive cleanup on application exit. * [windows/hook.c] [windows/win.c] [windows/class.c] Added miscellaneous cleanup routines. * [controls/menu.c] More efficient popup menu window handling. Mon Apr 22 21:35:22 1996 Albrecht Kleine <kleine@ak.sax.de> * [include/windows.h][objects/oembitmap.c][include/bitmaps/obm_trtype] Added "TT-bitmap" for later usage in a ChooseFont() ownerdraw combobox.
1020 lines
31 KiB
C
1020 lines
31 KiB
C
/*
|
|
* GDI font objects
|
|
*
|
|
* Copyright 1993 Alexandre Julliard
|
|
*
|
|
* Enhacements by Juergen Marquardt 1996
|
|
*
|
|
* Implementation of a second font cache which
|
|
* will be updated dynamically
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <X11/Xatom.h>
|
|
#include "font.h"
|
|
#include "metafile.h"
|
|
#include "callback.h"
|
|
#include "options.h"
|
|
#include "xmalloc.h"
|
|
#include "stddebug.h"
|
|
#include "debug.h"
|
|
|
|
#define FONTCACHE 32 /* dynamic font cache size */
|
|
#define MAX_FONTS 256
|
|
static LPLOGFONT lpLogFontList[MAX_FONTS] = { NULL };
|
|
|
|
|
|
#define CI_NONEXISTCHAR(cs) (((cs)->width == 0) && \
|
|
(((cs)->rbearing|(cs)->lbearing| \
|
|
(cs)->ascent|(cs)->descent) == 0))
|
|
|
|
/*
|
|
* CI_GET_CHAR_INFO - return the charinfo struct for the indicated 8bit
|
|
* character. If the character is in the column and exists, then return the
|
|
* appropriate metrics (note that fonts with common per-character metrics will
|
|
* return min_bounds). If none of these hold true, try again with the default
|
|
* char.
|
|
*/
|
|
#define CI_GET_CHAR_INFO(fs,col,def,cs) \
|
|
{ \
|
|
cs = def; \
|
|
if (col >= fs->min_char_or_byte2 && col <= fs->max_char_or_byte2) { \
|
|
if (fs->per_char == NULL) { \
|
|
cs = &fs->min_bounds; \
|
|
} else { \
|
|
cs = &fs->per_char[(col - fs->min_char_or_byte2)]; \
|
|
if (CI_NONEXISTCHAR(cs)) cs = def; \
|
|
} \
|
|
} \
|
|
}
|
|
|
|
#define CI_GET_DEFAULT_INFO(fs,cs) \
|
|
CI_GET_CHAR_INFO(fs, fs->default_char, NULL, cs)
|
|
|
|
struct FontStructure {
|
|
char *window;
|
|
char *x11;
|
|
} FontNames[32];
|
|
int FontSize;
|
|
|
|
|
|
/***********************************************************************
|
|
* FONT_Init
|
|
*/
|
|
BOOL FONT_Init( void )
|
|
{
|
|
char temp[1024];
|
|
LPSTR ptr;
|
|
int i;
|
|
|
|
if (PROFILE_GetWineIniString( "fonts", NULL, "*", temp, sizeof(temp) ) > 2 )
|
|
{
|
|
for( ptr = temp, i = 1; strlen(ptr) != 0; ptr += strlen(ptr) + 1 )
|
|
if( strcmp( ptr, "default" ) )
|
|
FontNames[i++].window = xstrdup( ptr );
|
|
FontSize = i;
|
|
|
|
for( i = 1; i < FontSize; i++ )
|
|
{
|
|
PROFILE_GetWineIniString( "fonts", FontNames[i].window, "*",
|
|
temp, sizeof(temp) );
|
|
FontNames[i].x11 = xstrdup( temp );
|
|
}
|
|
PROFILE_GetWineIniString( "fonts", "default", "*", temp, sizeof(temp) );
|
|
FontNames[0].x11 = xstrdup( temp );
|
|
|
|
} else {
|
|
FontNames[0].window = NULL; FontNames[0].x11 = "*-helvetica";
|
|
FontNames[1].window = "ms sans serif"; FontNames[1].x11 = "*-helvetica";
|
|
FontNames[2].window = "ms serif"; FontNames[2].x11 = "*-times";
|
|
FontNames[3].window = "fixedsys"; FontNames[3].x11 = "*-fixed";
|
|
FontNames[4].window = "arial"; FontNames[4].x11 = "*-helvetica";
|
|
FontNames[5].window = "helv"; FontNames[5].x11 = "*-helvetica";
|
|
FontNames[6].window = "roman"; FontNames[6].x11 = "*-times";
|
|
FontSize = 7;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* FONT_TranslateName
|
|
*
|
|
* Translate a Windows face name to its X11 equivalent.
|
|
* This will probably have to be customizable.
|
|
*/
|
|
static const char *FONT_TranslateName( char *winFaceName )
|
|
{
|
|
int i;
|
|
|
|
for (i = 1; i < FontSize; i ++)
|
|
if( !strcmp( winFaceName, FontNames[i].window ) ) {
|
|
dprintf_font(stddeb, "---- Mapped %s to %s\n", winFaceName, FontNames[i].x11 );
|
|
return FontNames[i].x11;
|
|
}
|
|
return FontNames[0].x11;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* FONT_MatchFont
|
|
*
|
|
* Find a X font matching the logical font.
|
|
*/
|
|
static XFontStruct * FONT_MatchFont( LOGFONT * font, DC * dc )
|
|
{
|
|
char pattern[100];
|
|
const char *family, *weight, *charset;
|
|
char **names;
|
|
char slant, oldspacing, spacing;
|
|
int width, height, oldheight, count;
|
|
XFontStruct * fontStruct;
|
|
|
|
dprintf_font(stddeb,
|
|
"FONT_MatchFont(H,W = %d,%d; Weight = %d; Italic = %d; FaceName = '%s'\n",
|
|
font->lfHeight, font->lfWidth, font->lfWeight, font->lfItalic, font->lfFaceName);
|
|
weight = (font->lfWeight > 550) ? "bold" : "medium";
|
|
slant = font->lfItalic ? 'i' : 'r';
|
|
if (font->lfHeight == -1)
|
|
height = 0;
|
|
else
|
|
height = font->lfHeight * dc->w.VportExtX / dc->w.WndExtX;
|
|
if (height == 0) height = 120; /* Default height = 12 */
|
|
else if (height < 0)
|
|
{
|
|
/* If height is negative, it means the height of the characters */
|
|
/* *without* the internal leading. So we adjust it a bit to */
|
|
/* compensate. 5/4 seems to give good results for small fonts. */
|
|
/*
|
|
* J.M.: This causes wrong font size for bigger fonts e.g. in Winword & Write
|
|
height = 10 * (-height * 9 / 8);
|
|
* may be we have to use an non linear function
|
|
*/
|
|
/* assume internal leading is 2 pixels. Else small fonts will become
|
|
* very small. */
|
|
height = (height-2) * -10;
|
|
}
|
|
else height *= 10;
|
|
width = 10 * (font->lfWidth * dc->w.VportExtY / dc->w.WndExtY);
|
|
if (width < 0) {
|
|
dprintf_font( stddeb, "FONT_MatchFont: negative width %d(%d)\n",
|
|
width, font->lfWidth );
|
|
width = -width;
|
|
}
|
|
|
|
spacing = (font->lfPitchAndFamily & FIXED_PITCH) ? 'm' :
|
|
(font->lfPitchAndFamily & VARIABLE_PITCH) ? 'p' : '*';
|
|
|
|
|
|
charset = (font->lfCharSet == ANSI_CHARSET) ? "iso8859-1" : "*-*";
|
|
if (*font->lfFaceName) {
|
|
family = FONT_TranslateName( font->lfFaceName );
|
|
/* FIX ME: I don't if that's correct but it works J.M. */
|
|
spacing = '*';
|
|
}
|
|
else switch(font->lfPitchAndFamily & 0xf0)
|
|
{
|
|
case FF_ROMAN:
|
|
family = FONT_TranslateName( "roman" );
|
|
break;
|
|
case FF_SWISS:
|
|
family = FONT_TranslateName( "swiss" );
|
|
break;
|
|
case FF_MODERN:
|
|
family = FONT_TranslateName( "modern" );
|
|
break;
|
|
case FF_SCRIPT:
|
|
family = FONT_TranslateName( "script" );
|
|
break;
|
|
case FF_DECORATIVE:
|
|
family = FONT_TranslateName( "decorative" );
|
|
break;
|
|
default:
|
|
family = "*-*";
|
|
break;
|
|
}
|
|
|
|
oldheight = height;
|
|
oldspacing = spacing;
|
|
while (TRUE) {
|
|
/* Width==0 seems not to be a valid wildcard on SGI's, using * instead */
|
|
if ( width == 0 )
|
|
sprintf( pattern, "-%s-%s-%c-normal-*-*-%d-*-*-%c-*-%s",
|
|
family, weight, slant, height, spacing, charset);
|
|
else
|
|
sprintf( pattern, "-%s-%s-%c-normal-*-*-%d-*-*-%c-%d-%s",
|
|
family, weight, slant, height, spacing, width, charset);
|
|
dprintf_font(stddeb, "FONT_MatchFont: '%s'\n", pattern );
|
|
names = XListFonts( display, pattern, 1, &count );
|
|
if (count > 0) break;
|
|
if (spacing == 'm') /* try 'c' if no 'm' found */ {
|
|
|
|
spacing = 'c';
|
|
continue;
|
|
} else if (spacing == 'p') /* try '*' if no 'p' found */ {
|
|
spacing = '*';
|
|
continue;
|
|
}
|
|
spacing = oldspacing;
|
|
height -= 10;
|
|
if (height < 10) {
|
|
if (slant == 'i') {
|
|
/* try oblique if no italic font */
|
|
slant = 'o';
|
|
height = oldheight;
|
|
continue;
|
|
}
|
|
if (spacing == 'm' && strcmp(family, "*-*") != 0) {
|
|
/* If a fixed spacing font could not be found, ignore
|
|
* the family */
|
|
family = "*-*";
|
|
height = oldheight;
|
|
continue;
|
|
}
|
|
fprintf(stderr, "FONT_MatchFont(%s) : returning NULL\n", pattern);
|
|
return NULL;
|
|
}
|
|
}
|
|
dprintf_font(stddeb," Found '%s'\n", *names );
|
|
fontStruct = XLoadQueryFont( display, *names );
|
|
XFreeFontNames( names );
|
|
return fontStruct;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* FONT_GetMetrics
|
|
*/
|
|
void FONT_GetMetrics( LOGFONT * logfont, XFontStruct * xfont,
|
|
TEXTMETRIC * metrics )
|
|
{
|
|
int average, i, count;
|
|
unsigned long prop;
|
|
|
|
metrics->tmAscent = xfont->ascent;
|
|
metrics->tmDescent = xfont->descent;
|
|
metrics->tmHeight = xfont->ascent + xfont->descent;
|
|
|
|
metrics->tmInternalLeading = 0;
|
|
if (XGetFontProperty( xfont, XA_X_HEIGHT, &prop ))
|
|
metrics->tmInternalLeading = xfont->ascent - (short)prop;
|
|
metrics->tmExternalLeading = 0;
|
|
metrics->tmMaxCharWidth = xfont->max_bounds.width;
|
|
metrics->tmWeight = logfont->lfWeight;
|
|
metrics->tmItalic = logfont->lfItalic;
|
|
metrics->tmUnderlined = logfont->lfUnderline;
|
|
metrics->tmStruckOut = logfont->lfStrikeOut;
|
|
metrics->tmFirstChar = xfont->min_char_or_byte2;
|
|
metrics->tmLastChar = xfont->max_char_or_byte2;
|
|
metrics->tmDefaultChar = xfont->default_char;
|
|
metrics->tmBreakChar = ' ';
|
|
metrics->tmCharSet = logfont->lfCharSet;
|
|
metrics->tmOverhang = 0;
|
|
metrics->tmDigitizedAspectX = 1;
|
|
metrics->tmDigitizedAspectY = 1;
|
|
metrics->tmPitchAndFamily = (logfont->lfPitchAndFamily&0xf0)|TMPF_DEVICE;
|
|
if (logfont->lfPitchAndFamily & FIXED_PITCH)
|
|
metrics->tmPitchAndFamily |= TMPF_FIXED_PITCH;
|
|
|
|
if (!xfont->per_char) average = metrics->tmMaxCharWidth;
|
|
else
|
|
{
|
|
XCharStruct * charPtr = xfont->per_char;
|
|
average = count = 0;
|
|
for (i = metrics->tmFirstChar; i <= metrics->tmLastChar; i++)
|
|
{
|
|
if (!CI_NONEXISTCHAR( charPtr ))
|
|
{
|
|
average += charPtr->width;
|
|
count++;
|
|
}
|
|
charPtr++;
|
|
}
|
|
if (count) average = (average + count/2) / count;
|
|
}
|
|
metrics->tmAveCharWidth = average;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* GetGlyphOutLine (GDI.309)
|
|
*/
|
|
DWORD GetGlyphOutLine(HDC hdc, UINT uChar, UINT fuFormat, LPGLYPHMETRICS lpgm,
|
|
DWORD cbBuffer, LPSTR lpBuffer, LPMAT2 lpmat2)
|
|
{
|
|
fprintf( stdnimp,"GetGlyphOutLine(%04x, '%c', %04x, %p, %ld, %p, %p) // - empty stub!\n",
|
|
hdc, uChar, fuFormat, lpgm, cbBuffer, lpBuffer, lpmat2 );
|
|
return (DWORD)-1; /* failure */
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* CreateScalableFontResource (GDI.310)
|
|
*/
|
|
BOOL CreateScalableFontResource( UINT fHidden,LPSTR lpszResourceFile,
|
|
LPSTR lpszFontFile, LPSTR lpszCurrentPath )
|
|
{
|
|
/* fHidden=1 - only visible for the calling app, read-only, not
|
|
* enumbered with EnumFonts/EnumFontFamilies
|
|
* lpszCurrentPath can be NULL
|
|
*/
|
|
fprintf(stdnimp,"CreateScalableFontResource(%d,%s,%s,%s) // empty stub!\n",
|
|
fHidden, lpszResourceFile, lpszFontFile, lpszCurrentPath );
|
|
return FALSE; /* create failed */
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* CreateFontIndirect (GDI.57)
|
|
*/
|
|
HFONT CreateFontIndirect( const LOGFONT * font )
|
|
{
|
|
FONTOBJ * fontPtr;
|
|
HFONT hfont;
|
|
|
|
if (!font)
|
|
{
|
|
fprintf(stderr, "CreateFontIndirect : font is NULL : returning NULL\n");
|
|
return 0;
|
|
}
|
|
hfont = GDI_AllocObject( sizeof(FONTOBJ), FONT_MAGIC );
|
|
if (!hfont) return 0;
|
|
fontPtr = (FONTOBJ *) GDI_HEAP_LIN_ADDR( hfont );
|
|
memcpy( &fontPtr->logfont, font, sizeof(LOGFONT) );
|
|
AnsiLower( fontPtr->logfont.lfFaceName );
|
|
dprintf_font(stddeb,"CreateFontIndirect(%p (%d,%d)); return %04x\n",
|
|
font, font->lfHeight, font->lfWidth, hfont);
|
|
return hfont;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* CreateFont (GDI.56)
|
|
*/
|
|
HFONT CreateFont( INT height, INT width, INT esc, INT orient, INT weight,
|
|
BYTE italic, BYTE underline, BYTE strikeout, BYTE charset,
|
|
BYTE outpres, BYTE clippres, BYTE quality, BYTE pitch,
|
|
LPCSTR name )
|
|
{
|
|
LOGFONT logfont = { height, width, esc, orient, weight, italic, underline,
|
|
strikeout, charset, outpres, clippres, quality, pitch, };
|
|
dprintf_font(stddeb,"CreateFont(%d,%d)\n", height, width);
|
|
if (name)
|
|
{
|
|
strncpy( logfont.lfFaceName, name, LF_FACESIZE - 1 );
|
|
logfont.lfFaceName[LF_FACESIZE - 1] = '\0';
|
|
}
|
|
else logfont.lfFaceName[0] = '\0';
|
|
return CreateFontIndirect( &logfont );
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* FONT_GetObject
|
|
*/
|
|
int FONT_GetObject( FONTOBJ * font, int count, LPSTR buffer )
|
|
{
|
|
if (count > sizeof(LOGFONT)) count = sizeof(LOGFONT);
|
|
memcpy( buffer, &font->logfont, count );
|
|
return count;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* FONT_SelectObject
|
|
*/
|
|
HFONT FONT_SelectObject( DC * dc, HFONT hfont, FONTOBJ * font )
|
|
{
|
|
static X_PHYSFONT stockFonts[LAST_STOCK_FONT-FIRST_STOCK_FONT+1];
|
|
|
|
static struct {
|
|
HFONT id;
|
|
LOGFONT logfont;
|
|
int access;
|
|
int used;
|
|
X_PHYSFONT cacheFont; } cacheFonts[FONTCACHE], *cacheFontsMin;
|
|
int i;
|
|
|
|
X_PHYSFONT * stockPtr;
|
|
HFONT prevHandle = dc->w.hFont;
|
|
XFontStruct * fontStruct;
|
|
dprintf_font(stddeb,"FONT_SelectObject(%p, %04x, %p)\n", dc, hfont, font);
|
|
|
|
#if 0 /* From the code in SelectObject, this can not happen */
|
|
/* Load font if necessary */
|
|
if (!font)
|
|
{
|
|
HFONT hnewfont;
|
|
|
|
hnewfont = CreateFont(10, 7, 0, 0, FW_DONTCARE,
|
|
FALSE, FALSE, FALSE, DEFAULT_CHARSET, 0, 0,
|
|
DEFAULT_QUALITY, FF_DONTCARE, "*" );
|
|
font = (FONTOBJ *) GDI_HEAP_LIN_ADDR( hnewfont );
|
|
}
|
|
#endif
|
|
|
|
if (dc->header.wMagic == METAFILE_DC_MAGIC)
|
|
if (MF_CreateFontIndirect(dc, hfont, &(font->logfont)))
|
|
return prevHandle;
|
|
else
|
|
return 0;
|
|
|
|
if ((hfont >= FIRST_STOCK_FONT) && (hfont <= LAST_STOCK_FONT))
|
|
stockPtr = &stockFonts[hfont - FIRST_STOCK_FONT];
|
|
else {
|
|
stockPtr = NULL;
|
|
/*
|
|
* Ok, It's not a stock font but
|
|
* may be it's cached in dynamic cache
|
|
*/
|
|
for(i=0; i<FONTCACHE; i++) /* search for same handle */
|
|
if (cacheFonts[i].id==hfont) { /* Got the handle */
|
|
/*
|
|
* Check if Handle matches the font
|
|
*/
|
|
if(memcmp(&cacheFonts[i].logfont,&(font->logfont), sizeof(LOGFONT))) {
|
|
/* No: remove handle id from dynamic font cache */
|
|
cacheFonts[i].access=0;
|
|
cacheFonts[i].used=0;
|
|
cacheFonts[i].id=0;
|
|
/* may be there is an unused handle which contains the font */
|
|
for(i=0; i<FONTCACHE; i++) {
|
|
if((cacheFonts[i].used == 0) &&
|
|
(memcmp(&cacheFonts[i].logfont,&(font->logfont), sizeof(LOGFONT)))== 0) {
|
|
/* got it load from cache and set new handle id */
|
|
stockPtr = &cacheFonts[i].cacheFont;
|
|
cacheFonts[i].access=1;
|
|
cacheFonts[i].used=1;
|
|
cacheFonts[i].id=hfont;
|
|
dprintf_font(stddeb,"FONT_SelectObject: got font from unused handle\n");
|
|
break;
|
|
}
|
|
}
|
|
|
|
}
|
|
else {
|
|
/* Yes: load from dynamic font cache */
|
|
stockPtr = &cacheFonts[i].cacheFont;
|
|
cacheFonts[i].access++;
|
|
cacheFonts[i].used++;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
if (!stockPtr || !stockPtr->fstruct)
|
|
{
|
|
if (!(fontStruct = FONT_MatchFont( &font->logfont, dc )))
|
|
{
|
|
/* If it is not a stock font, we can simply return 0 */
|
|
if (!stockPtr) return 0;
|
|
/* Otherwise we must try to find a substitute */
|
|
dprintf_font(stddeb,"Loading font 'fixed' for %04x\n", hfont );
|
|
font->logfont.lfPitchAndFamily &= ~VARIABLE_PITCH;
|
|
font->logfont.lfPitchAndFamily |= FIXED_PITCH;
|
|
fontStruct = XLoadQueryFont( display, "fixed" );
|
|
if (!fontStruct)
|
|
{
|
|
fprintf( stderr, "No system font could be found. Please check your font path.\n" );
|
|
exit( 1 );
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
fontStruct = stockPtr->fstruct;
|
|
dprintf_font(stddeb,
|
|
"FONT_SelectObject: Loaded font from cache %04x %p\n",
|
|
hfont, fontStruct );
|
|
}
|
|
|
|
/* Unuse previous font */
|
|
for (i=0; i < FONTCACHE; i++) {
|
|
if (cacheFonts[i].id == prevHandle) {
|
|
if(cacheFonts[i].used == 0)
|
|
fprintf(stderr, "Trying to decrement a use count of 0.\n");
|
|
else
|
|
cacheFonts[i].used--;
|
|
}
|
|
}
|
|
|
|
/* Store font */
|
|
dc->w.hFont = hfont;
|
|
if (stockPtr)
|
|
{
|
|
if (!stockPtr->fstruct)
|
|
{
|
|
stockPtr->fstruct = fontStruct;
|
|
FONT_GetMetrics( &font->logfont, fontStruct, &stockPtr->metrics );
|
|
}
|
|
memcpy( &dc->u.x.font, stockPtr, sizeof(*stockPtr) );
|
|
}
|
|
else
|
|
{
|
|
/*
|
|
* Check in cacheFont
|
|
*/
|
|
cacheFontsMin=NULL;
|
|
for (i=0; i < FONTCACHE; i++) {
|
|
if (cacheFonts[i].used==0)
|
|
if ((!cacheFontsMin) || ((cacheFontsMin) && (cacheFontsMin->access > cacheFonts[i].access)))
|
|
cacheFontsMin=&cacheFonts[i];
|
|
}
|
|
if (!cacheFontsMin) {
|
|
fprintf(stderr,"No unused font cache entry !!!!\n" );
|
|
return prevHandle;
|
|
}
|
|
if (cacheFontsMin->id!=0) {
|
|
dprintf_font(stddeb,
|
|
"FONT_SelectObject: Freeing %04x \n",cacheFontsMin->id );
|
|
XFreeFont( display, cacheFontsMin->cacheFont.fstruct );
|
|
}
|
|
cacheFontsMin->cacheFont.fstruct = fontStruct;
|
|
FONT_GetMetrics( &font->logfont, fontStruct, &cacheFontsMin->cacheFont.metrics );
|
|
cacheFontsMin->access=1;
|
|
cacheFontsMin->used=1;
|
|
cacheFontsMin->id=hfont;
|
|
memcpy( &dc->u.x.font, &(cacheFontsMin->cacheFont), sizeof(cacheFontsMin->cacheFont) );
|
|
memcpy(&cacheFontsMin->logfont,&(font->logfont), sizeof(LOGFONT));
|
|
|
|
}
|
|
return prevHandle;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* GetTextCharacterExtra (GDI.89)
|
|
*/
|
|
short GetTextCharacterExtra( HDC hdc )
|
|
{
|
|
DC * dc = (DC *) GDI_GetObjPtr( hdc, DC_MAGIC );
|
|
if (!dc) return 0;
|
|
return abs( (dc->w.charExtra * dc->w.WndExtX + dc->w.VportExtX / 2)
|
|
/ dc->w.VportExtX );
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* SetTextCharacterExtra (GDI.8)
|
|
*/
|
|
short SetTextCharacterExtra( HDC hdc, short extra )
|
|
{
|
|
short prev;
|
|
DC * dc = (DC *) GDI_GetObjPtr( hdc, DC_MAGIC );
|
|
if (!dc) return 0;
|
|
extra = (extra * dc->w.VportExtX + dc->w.WndExtX / 2) / dc->w.WndExtX;
|
|
prev = dc->w.charExtra;
|
|
dc->w.charExtra = abs(extra);
|
|
return (prev * dc->w.WndExtX + dc->w.VportExtX / 2) / dc->w.VportExtX;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* SetTextJustification (GDI.10)
|
|
*/
|
|
short SetTextJustification( HDC hdc, short extra, short breaks )
|
|
{
|
|
DC * dc = (DC *) GDI_GetObjPtr( hdc, DC_MAGIC );
|
|
if (!dc) return 0;
|
|
|
|
extra = abs((extra * dc->w.VportExtX + dc->w.WndExtX / 2) / dc->w.WndExtX);
|
|
if (!extra) breaks = 0;
|
|
dc->w.breakTotalExtra = extra;
|
|
dc->w.breakCount = breaks;
|
|
if (breaks)
|
|
{
|
|
dc->w.breakExtra = extra / breaks;
|
|
dc->w.breakRem = extra - (dc->w.breakCount * dc->w.breakExtra);
|
|
}
|
|
else
|
|
{
|
|
dc->w.breakExtra = 0;
|
|
dc->w.breakRem = 0;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* GetTextFace (GDI.92)
|
|
*/
|
|
INT GetTextFace( HDC hdc, INT count, LPSTR name )
|
|
{
|
|
FONTOBJ *font;
|
|
|
|
DC * dc = (DC *) GDI_GetObjPtr( hdc, DC_MAGIC );
|
|
if (!dc) return 0;
|
|
if (!(font = (FONTOBJ *) GDI_GetObjPtr( dc->w.hFont, FONT_MAGIC )))
|
|
return 0;
|
|
lstrcpyn( name, font->logfont.lfFaceName, count );
|
|
return strlen(name);
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* GetTextExtent (GDI.91)
|
|
*/
|
|
DWORD GetTextExtent( HDC hdc, LPCSTR str, short count )
|
|
{
|
|
SIZE size;
|
|
if (!GetTextExtentPoint( hdc, str, count, &size )) return 0;
|
|
return MAKELONG( size.cx, size.cy );
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* GetTextExtentPoint (GDI.471)
|
|
*/
|
|
BOOL GetTextExtentPoint( HDC hdc, LPCSTR str, short count, LPSIZE size )
|
|
{
|
|
int dir, ascent, descent;
|
|
XCharStruct info;
|
|
|
|
DC * dc = (DC *) GDI_GetObjPtr( hdc, DC_MAGIC );
|
|
if (!dc) return FALSE;
|
|
XTextExtents( dc->u.x.font.fstruct, str, count, &dir,
|
|
&ascent, &descent, &info );
|
|
size->cx = abs((info.width + dc->w.breakRem + count * dc->w.charExtra)
|
|
* dc->w.WndExtX / dc->w.VportExtX);
|
|
size->cy = abs((dc->u.x.font.fstruct->ascent+dc->u.x.font.fstruct->descent)
|
|
* dc->w.WndExtY / dc->w.VportExtY);
|
|
|
|
dprintf_font(stddeb,"GetTextExtentPoint(%04x '%*.*s' %d %p): returning %d,%d\n",
|
|
hdc, count, count, str, count, size, size->cx, size->cy );
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* GetTextMetrics (GDI.93)
|
|
*/
|
|
BOOL GetTextMetrics( HDC hdc, LPTEXTMETRIC metrics )
|
|
{
|
|
DC * dc = (DC *) GDI_GetObjPtr( hdc, DC_MAGIC );
|
|
if (!dc) return FALSE;
|
|
memcpy( metrics, &dc->u.x.font.metrics, sizeof(*metrics) );
|
|
|
|
metrics->tmAscent = abs( metrics->tmAscent
|
|
* dc->w.WndExtY / dc->w.VportExtY );
|
|
metrics->tmDescent = abs( metrics->tmDescent
|
|
* dc->w.WndExtY / dc->w.VportExtY );
|
|
metrics->tmHeight = metrics->tmAscent + metrics->tmDescent;
|
|
metrics->tmInternalLeading = abs( metrics->tmInternalLeading
|
|
* dc->w.WndExtY / dc->w.VportExtY );
|
|
metrics->tmExternalLeading = abs( metrics->tmExternalLeading
|
|
* dc->w.WndExtY / dc->w.VportExtY );
|
|
metrics->tmMaxCharWidth = abs( metrics->tmMaxCharWidth
|
|
* dc->w.WndExtX / dc->w.VportExtX );
|
|
metrics->tmAveCharWidth = abs( metrics->tmAveCharWidth
|
|
* dc->w.WndExtX / dc->w.VportExtX );
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* SetMapperFlags (GDI.349)
|
|
*/
|
|
DWORD SetMapperFlags(HDC hDC, DWORD dwFlag)
|
|
{
|
|
dprintf_font(stdnimp,"SetmapperFlags(%04x, %08lX) // Empty Stub !\n",
|
|
hDC, dwFlag);
|
|
return 0L;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* GetCharABCWidths (GDI.307)
|
|
*/
|
|
BOOL GetCharABCWidths(HDC hdc, UINT wFirstChar, UINT wLastChar, LPABC lpABC)
|
|
{
|
|
/* No TrueType fonts in Wine */
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* GetCharWidth (GDI.350)
|
|
*/
|
|
BOOL GetCharWidth(HDC hdc, WORD wFirstChar, WORD wLastChar, LPINT lpBuffer)
|
|
{
|
|
int i, j;
|
|
XFontStruct *xfont;
|
|
XCharStruct *cs, *def;
|
|
|
|
DC *dc = (DC *)GDI_GetObjPtr(hdc, DC_MAGIC);
|
|
if (!dc) return FALSE;
|
|
xfont = dc->u.x.font.fstruct;
|
|
|
|
/* fixed font? */
|
|
if (xfont->per_char == NULL)
|
|
{
|
|
for (i = wFirstChar, j = 0; i <= wLastChar; i++, j++)
|
|
*(lpBuffer + j) = xfont->max_bounds.width;
|
|
return TRUE;
|
|
}
|
|
|
|
CI_GET_DEFAULT_INFO(xfont, def);
|
|
|
|
for (i = wFirstChar, j = 0; i <= wLastChar; i++, j++)
|
|
{
|
|
CI_GET_CHAR_INFO(xfont, i, def, cs);
|
|
*(lpBuffer + j) = cs ? cs->width : xfont->max_bounds.width;
|
|
if (*(lpBuffer + j) < 0)
|
|
*(lpBuffer + j) = 0;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* AddFontResource (GDI.119)
|
|
*/
|
|
INT AddFontResource( LPCSTR str )
|
|
{
|
|
fprintf( stdnimp, "STUB: AddFontResource('%s')\n", str );
|
|
return 1;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* RemoveFontResource (GDI.136)
|
|
*/
|
|
BOOL RemoveFontResource( LPSTR str )
|
|
{
|
|
fprintf( stdnimp, "STUB: RemoveFontResource('%s')\n", str );
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
/*************************************************************************
|
|
* ParseFontParms [internal]
|
|
*/
|
|
int ParseFontParms(LPSTR lpFont, WORD wParmsNo, LPSTR lpRetStr, WORD wMaxSiz)
|
|
{
|
|
int i;
|
|
if (lpFont == NULL) return 0;
|
|
if (lpRetStr == NULL) return 0;
|
|
for (i = 0; (*lpFont != '\0' && i != wParmsNo); ) {
|
|
if (*lpFont == '-') i++;
|
|
lpFont++;
|
|
}
|
|
if (i == wParmsNo) {
|
|
if (*lpFont == '-') lpFont++;
|
|
wMaxSiz--;
|
|
for (i = 0; (*lpFont != '\0' && *lpFont != '-' && i < wMaxSiz); i++)
|
|
*(lpRetStr + i) = *lpFont++;
|
|
*(lpRetStr + i) = '\0';
|
|
return i;
|
|
}
|
|
else
|
|
lpRetStr[0] = '\0';
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*************************************************************************
|
|
* InitFontsList [internal]
|
|
*/
|
|
|
|
static int logfcmp(const void *a,const void *b)
|
|
{
|
|
return strcmp( (*(LPLOGFONT *)a)->lfFaceName, (*(LPLOGFONT *)b)->lfFaceName );
|
|
}
|
|
|
|
void InitFontsList(void)
|
|
{
|
|
char str[32];
|
|
char pattern[100];
|
|
char *family, *weight, *charset;
|
|
char **names;
|
|
char slant, spacing;
|
|
int i, count;
|
|
LPLOGFONT lpNewFont;
|
|
weight = "medium";
|
|
slant = 'r';
|
|
spacing = '*';
|
|
charset = "*";
|
|
family = "*-*";
|
|
dprintf_font(stddeb,"InitFontsList !\n");
|
|
sprintf( pattern, "-%s-%s-%c-normal-*-*-*-*-*-%c-*-%s",
|
|
family, weight, slant, spacing, charset);
|
|
names = XListFonts( display, pattern, MAX_FONTS, &count );
|
|
dprintf_font(stddeb,"InitFontsList // count=%d \n", count);
|
|
for (i = 0; i < count; i++) {
|
|
lpNewFont = malloc(sizeof(LOGFONT) + LF_FACESIZE);
|
|
if (lpNewFont == NULL) {
|
|
dprintf_font(stddeb, "InitFontsList // Error alloc new font structure !\n");
|
|
break;
|
|
}
|
|
dprintf_font(stddeb,"InitFontsList // names[%d]='%s' \n", i, names[i]);
|
|
ParseFontParms(names[i], 2, str, sizeof(str));
|
|
if (strcmp(str, "fixed") == 0) strcat(str, "sys");
|
|
AnsiUpper(str);
|
|
strcpy(lpNewFont->lfFaceName, str);
|
|
ParseFontParms(names[i], 8, str, sizeof(str));
|
|
lpNewFont->lfHeight = atoi(str) / 10;
|
|
ParseFontParms(names[i], 12, str, sizeof(str));
|
|
lpNewFont->lfWidth = atoi(str) / 10;
|
|
lpNewFont->lfEscapement = 0;
|
|
lpNewFont->lfOrientation = 0;
|
|
lpNewFont->lfWeight = FW_REGULAR;
|
|
lpNewFont->lfItalic = 0;
|
|
lpNewFont->lfUnderline = 0;
|
|
lpNewFont->lfStrikeOut = 0;
|
|
ParseFontParms(names[i], 13, str, sizeof(str));
|
|
if (strcmp(str, "iso8859") == 0) {
|
|
lpNewFont->lfCharSet = ANSI_CHARSET;
|
|
} else {
|
|
lpNewFont->lfCharSet = OEM_CHARSET;
|
|
}
|
|
lpNewFont->lfOutPrecision = OUT_DEFAULT_PRECIS;
|
|
lpNewFont->lfClipPrecision = CLIP_DEFAULT_PRECIS;
|
|
lpNewFont->lfQuality = DEFAULT_QUALITY;
|
|
ParseFontParms(names[i], 11, str, sizeof(str));
|
|
switch(str[0]) {
|
|
case 'p':
|
|
lpNewFont->lfPitchAndFamily = VARIABLE_PITCH | FF_SWISS;
|
|
break;
|
|
case 'm':
|
|
case 'c':
|
|
lpNewFont->lfPitchAndFamily = FIXED_PITCH | FF_MODERN;
|
|
break;
|
|
default:
|
|
lpNewFont->lfPitchAndFamily = DEFAULT_PITCH | FF_DONTCARE;
|
|
break;
|
|
}
|
|
dprintf_font(stddeb,"InitFontsList // lpNewFont->lfHeight=%d \n", lpNewFont->lfHeight);
|
|
dprintf_font(stddeb,"InitFontsList // lpNewFont->lfWidth=%d \n", lpNewFont->lfWidth);
|
|
dprintf_font(stddeb,"InitFontsList // lfFaceName='%s' \n", lpNewFont->lfFaceName);
|
|
lpLogFontList[i] = lpNewFont;
|
|
lpLogFontList[i+1] = NULL;
|
|
}
|
|
qsort(lpLogFontList,count,sizeof(*lpLogFontList),logfcmp);
|
|
XFreeFontNames(names);
|
|
}
|
|
|
|
|
|
/*************************************************************************
|
|
* EnumFonts [GDI.70]
|
|
*/
|
|
INT EnumFonts(HDC hDC, LPCTSTR lpFaceName, FONTENUMPROC lpEnumFunc, LPARAM lpData)
|
|
{
|
|
HANDLE hLog;
|
|
HANDLE hMet;
|
|
HFONT hFont;
|
|
HFONT hOldFont;
|
|
LPLOGFONT lpLogFont;
|
|
LPTEXTMETRIC lptm;
|
|
LPSTR lpOldName;
|
|
char FaceName[LF_FACESIZE];
|
|
int nRet = 0;
|
|
int i;
|
|
|
|
dprintf_font(stddeb,"EnumFonts(%04x, %p='%s', %08lx, %08lx)\n",
|
|
hDC, lpFaceName, lpFaceName, (LONG)lpEnumFunc, lpData);
|
|
if (lpEnumFunc == 0) return 0;
|
|
hLog = GDI_HEAP_ALLOC( sizeof(LOGFONT) + LF_FACESIZE );
|
|
lpLogFont = (LPLOGFONT) GDI_HEAP_LIN_ADDR(hLog);
|
|
if (lpLogFont == NULL) {
|
|
fprintf(stderr,"EnumFonts // can't alloc LOGFONT struct !\n");
|
|
return 0;
|
|
}
|
|
hMet = GDI_HEAP_ALLOC( sizeof(TEXTMETRIC) );
|
|
lptm = (LPTEXTMETRIC) GDI_HEAP_LIN_ADDR(hMet);
|
|
if (lptm == NULL) {
|
|
GDI_HEAP_FREE(hLog);
|
|
fprintf(stderr, "EnumFonts // can't alloc TEXTMETRIC struct !\n");
|
|
return 0;
|
|
}
|
|
if (lpFaceName != NULL) {
|
|
strcpy(FaceName, lpFaceName);
|
|
AnsiUpper(FaceName);
|
|
}
|
|
lpOldName = NULL;
|
|
|
|
if (lpLogFontList[0] == NULL) InitFontsList();
|
|
for(i = 0; lpLogFontList[i] != NULL; i++) {
|
|
if (lpFaceName == NULL) {
|
|
if (lpOldName != NULL) {
|
|
if (strcmp(lpOldName,lpLogFontList[i]->lfFaceName) == 0) continue;
|
|
}
|
|
lpOldName = lpLogFontList[i]->lfFaceName;
|
|
} else {
|
|
if (strcmp(FaceName, lpLogFontList[i]->lfFaceName) != 0) continue;
|
|
}
|
|
dprintf_font(stddeb,"EnumFonts // enum '%s' !\n", lpLogFontList[i]->lfFaceName);
|
|
dprintf_font(stddeb,"EnumFonts // %p !\n", lpLogFontList[i]);
|
|
memcpy(lpLogFont, lpLogFontList[i], sizeof(LOGFONT) + LF_FACESIZE);
|
|
hFont = CreateFontIndirect(lpLogFont);
|
|
hOldFont = SelectObject(hDC, hFont);
|
|
GetTextMetrics(hDC, lptm);
|
|
SelectObject(hDC, hOldFont);
|
|
DeleteObject(hFont);
|
|
dprintf_font(stddeb,"EnumFonts // i=%d lpLogFont=%p lptm=%p\n", i, lpLogFont, lptm);
|
|
nRet = CallEnumFontsProc(lpEnumFunc, GDI_HEAP_SEG_ADDR(hLog),
|
|
GDI_HEAP_SEG_ADDR(hMet), 0, (LONG)lpData );
|
|
if (nRet == 0) {
|
|
dprintf_font(stddeb,"EnumFonts // EnumEnd requested by application !\n");
|
|
break;
|
|
}
|
|
}
|
|
GDI_HEAP_FREE(hMet);
|
|
GDI_HEAP_FREE(hLog);
|
|
return nRet;
|
|
}
|
|
|
|
|
|
/*************************************************************************
|
|
* EnumFontFamilies [GDI.330]
|
|
*/
|
|
INT EnumFontFamilies(HDC hDC, LPCTSTR lpszFamily, FONTENUMPROC lpEnumFunc, LPARAM lpData)
|
|
{
|
|
HANDLE hLog;
|
|
HANDLE hMet;
|
|
HFONT hFont;
|
|
HFONT hOldFont;
|
|
LPENUMLOGFONT lpEnumLogFont;
|
|
LPTEXTMETRIC lptm;
|
|
LPSTR lpOldName;
|
|
char FaceName[LF_FACESIZE];
|
|
int nRet = 0;
|
|
int i;
|
|
|
|
dprintf_font(stddeb,"EnumFontFamilies(%04x, %p, %08lx, %08lx)\n",
|
|
hDC, lpszFamily, (DWORD)lpEnumFunc, lpData);
|
|
if (lpEnumFunc == 0) return 0;
|
|
hLog = GDI_HEAP_ALLOC( sizeof(ENUMLOGFONT) );
|
|
lpEnumLogFont = (LPENUMLOGFONT) GDI_HEAP_LIN_ADDR(hLog);
|
|
if (lpEnumLogFont == NULL) {
|
|
fprintf(stderr,"EnumFontFamilies // can't alloc LOGFONT struct !\n");
|
|
return 0;
|
|
}
|
|
hMet = GDI_HEAP_ALLOC( sizeof(TEXTMETRIC) );
|
|
lptm = (LPTEXTMETRIC) GDI_HEAP_LIN_ADDR(hMet);
|
|
if (lptm == NULL) {
|
|
GDI_HEAP_FREE(hLog);
|
|
fprintf(stderr,"EnumFontFamilies // can't alloc TEXTMETRIC struct !\n");
|
|
return 0;
|
|
}
|
|
lpOldName = NULL;
|
|
if (lpszFamily != NULL) {
|
|
strcpy(FaceName, lpszFamily);
|
|
AnsiUpper(FaceName);
|
|
}
|
|
if (lpLogFontList[0] == NULL) InitFontsList();
|
|
for(i = 0; lpLogFontList[i] != NULL; i++) {
|
|
if (lpszFamily == NULL) {
|
|
if (lpOldName != NULL) {
|
|
if (strcmp(lpOldName,lpLogFontList[i]->lfFaceName) == 0) continue;
|
|
}
|
|
lpOldName = lpLogFontList[i]->lfFaceName;
|
|
} else {
|
|
if (strcmp(FaceName, lpLogFontList[i]->lfFaceName) != 0) continue;
|
|
}
|
|
memcpy(lpEnumLogFont, lpLogFontList[i], sizeof(LOGFONT));
|
|
strcpy(lpEnumLogFont->elfFullName,"");
|
|
strcpy(lpEnumLogFont->elfStyle,"");
|
|
hFont = CreateFontIndirect((LPLOGFONT)lpEnumLogFont);
|
|
hOldFont = SelectObject(hDC, hFont);
|
|
GetTextMetrics(hDC, lptm);
|
|
SelectObject(hDC, hOldFont);
|
|
DeleteObject(hFont);
|
|
dprintf_font(stddeb, "EnumFontFamilies // i=%d lpLogFont=%p lptm=%p\n", i, lpEnumLogFont, lptm);
|
|
|
|
nRet = CallEnumFontFamProc( lpEnumFunc,
|
|
GDI_HEAP_SEG_ADDR(hLog),
|
|
GDI_HEAP_SEG_ADDR(hMet),
|
|
0, lpData );
|
|
if (nRet == 0) {
|
|
dprintf_font(stddeb,"EnumFontFamilies // EnumEnd requested by application !\n");
|
|
break;
|
|
}
|
|
}
|
|
GDI_HEAP_FREE(hMet);
|
|
GDI_HEAP_FREE(hLog);
|
|
return nRet;
|
|
}
|
|
|
|
/*************************************************************************
|
|
* GetRasterizerCaps [GDI.313]
|
|
*/
|
|
|
|
BOOL GetRasterizerCaps(LPRASTERIZER_STATUS lprs, UINT cbNumBytes)
|
|
{
|
|
/* This is not much more than a dummy */
|
|
RASTERIZER_STATUS rs;
|
|
|
|
rs.nSize = sizeof(rs);
|
|
rs.wFlags = 0;
|
|
rs.nLanguageID = 0;
|
|
return True;
|
|
}
|
|
|
|
/*************************************************************************
|
|
* GetKerningPairs [GDI.332]
|
|
*/
|
|
int GetKerningPairs(HDC hDC,int cBufLen,LPKERNINGPAIR lpKerningPairs)
|
|
{
|
|
/* Wine fonts are ugly and don't support kerning :) */
|
|
return 0;
|
|
}
|