299 lines
8.8 KiB
C
299 lines
8.8 KiB
C
/*
|
|
* text functions
|
|
*
|
|
* Copyright 1993, 1994 Alexandre Julliard
|
|
*
|
|
* 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 <string.h>
|
|
|
|
#include "windef.h"
|
|
#include "wingdi.h"
|
|
#include "wine/winuser16.h"
|
|
#include "winbase.h"
|
|
#include "winerror.h"
|
|
#include "gdi.h"
|
|
#include "wine/debug.h"
|
|
#include "winnls.h"
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(text);
|
|
|
|
/***********************************************************************
|
|
* FONT_mbtowc
|
|
*
|
|
* Returns a '\0' terminated Unicode translation of str using the
|
|
* charset of the currently selected font in hdc. If count is -1 then
|
|
* str is assumed to be '\0' terminated, otherwise it contains the
|
|
* number of bytes to convert. If plenW is non-NULL, on return it
|
|
* will point to the number of WCHARs (excluding the '\0') that have
|
|
* been written. If pCP is non-NULL, on return it will point to the
|
|
* codepage used in the conversion (NB, this may be CP_SYMBOL so watch
|
|
* out). The caller should free the returned LPWSTR from the process
|
|
* heap itself.
|
|
*/
|
|
LPWSTR FONT_mbtowc(HDC hdc, LPCSTR str, INT count, INT *plenW, UINT *pCP)
|
|
{
|
|
UINT cp = CP_ACP;
|
|
INT lenW, i;
|
|
LPWSTR strW;
|
|
CHARSETINFO csi;
|
|
int charset = GetTextCharset(hdc);
|
|
|
|
/* Hmm, nicely designed api this one! */
|
|
if(TranslateCharsetInfo((DWORD*)charset, &csi, TCI_SRCCHARSET))
|
|
cp = csi.ciACP;
|
|
else {
|
|
switch(charset) {
|
|
case OEM_CHARSET:
|
|
cp = GetOEMCP();
|
|
break;
|
|
case DEFAULT_CHARSET:
|
|
cp = GetACP();
|
|
break;
|
|
|
|
case VISCII_CHARSET:
|
|
case TCVN_CHARSET:
|
|
case KOI8_CHARSET:
|
|
case ISO3_CHARSET:
|
|
case ISO4_CHARSET:
|
|
case ISO10_CHARSET:
|
|
case CELTIC_CHARSET:
|
|
/* FIXME: These have no place here, but because x11drv
|
|
enumerates fonts with these (made up) charsets some apps
|
|
might use them and then the FIXME below would become
|
|
annoying. Now we could pick the intended codepage for
|
|
each of these, but since it's broken anyway we'll just
|
|
use CP_ACP and hope it'll go away...
|
|
*/
|
|
cp = CP_ACP;
|
|
break;
|
|
|
|
|
|
default:
|
|
FIXME("Can't find codepage for charset %d\n", charset);
|
|
break;
|
|
}
|
|
}
|
|
|
|
TRACE("cp == %d\n", cp);
|
|
|
|
if(count == -1) count = strlen(str);
|
|
if(cp != CP_SYMBOL) {
|
|
lenW = MultiByteToWideChar(cp, 0, str, count, NULL, 0);
|
|
strW = HeapAlloc(GetProcessHeap(), 0, (lenW + 1) * sizeof(WCHAR));
|
|
MultiByteToWideChar(cp, 0, str, count, strW, lenW);
|
|
} else {
|
|
lenW = count;
|
|
strW = HeapAlloc(GetProcessHeap(), 0, (lenW + 1) * sizeof(WCHAR));
|
|
for(i = 0; i < count; i++) strW[i] = (BYTE)str[i];
|
|
}
|
|
strW[lenW] = '\0';
|
|
TRACE("mapped %s -> %s\n", debugstr_an(str, count), debugstr_wn(strW, lenW));
|
|
if(plenW) *plenW = lenW;
|
|
if(pCP) *pCP = cp;
|
|
return strW;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* ExtTextOutA (GDI32.@)
|
|
*/
|
|
BOOL WINAPI ExtTextOutA( HDC hdc, INT x, INT y, UINT flags,
|
|
const RECT *lprect, LPCSTR str, UINT count, const INT *lpDx )
|
|
{
|
|
INT wlen;
|
|
UINT codepage;
|
|
LPWSTR p = FONT_mbtowc(hdc, str, count, &wlen, &codepage);
|
|
BOOL ret;
|
|
LPINT lpDxW = NULL;
|
|
|
|
if (lpDx) {
|
|
unsigned int i = 0, j = 0;
|
|
|
|
lpDxW = (LPINT)HeapAlloc( GetProcessHeap(), 0, wlen*sizeof(INT));
|
|
while(i < count) {
|
|
if(IsDBCSLeadByteEx(codepage, str[i])) {
|
|
lpDxW[j++] = lpDx[i] + lpDx[i+1];
|
|
i = i + 2;
|
|
} else {
|
|
lpDxW[j++] = lpDx[i];
|
|
i = i + 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
ret = ExtTextOutW( hdc, x, y, flags, lprect, p, wlen, lpDxW );
|
|
|
|
HeapFree( GetProcessHeap(), 0, p );
|
|
if (lpDxW) HeapFree( GetProcessHeap(), 0, lpDxW );
|
|
return ret;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* ExtTextOutW (GDI32.@)
|
|
*/
|
|
BOOL WINAPI ExtTextOutW( HDC hdc, INT x, INT y, UINT flags,
|
|
const RECT *lprect, LPCWSTR str, UINT count, const INT *lpDx )
|
|
{
|
|
BOOL ret = FALSE;
|
|
DC * dc = DC_GetDCUpdate( hdc );
|
|
if (dc)
|
|
{
|
|
if(PATH_IsPathOpen(dc->path))
|
|
FIXME("called on an open path\n");
|
|
else if(dc->funcs->pExtTextOut)
|
|
{
|
|
if( !(flags&(ETO_GLYPH_INDEX|ETO_IGNORELANGUAGE)) )
|
|
{
|
|
/* The caller did not specify that language processing was already done,
|
|
* and the font idetifies iteself as requiring language processing.
|
|
*/
|
|
GCP_RESULTSW gcp;
|
|
|
|
gcp.lStructSize=sizeof(gcp);
|
|
gcp.lpOutString=HeapAlloc(GetProcessHeap(), 0, count*sizeof(WCHAR));
|
|
gcp.lpOrder=NULL;
|
|
gcp.lpDx=NULL;
|
|
gcp.lpCaretPos=NULL;
|
|
gcp.lpClass=NULL;
|
|
gcp.lpGlyphs=NULL;
|
|
gcp.nGlyphs=0;
|
|
gcp.nMaxFit=0;
|
|
|
|
GetCharacterPlacementW(hdc, str, count, 0, &gcp, GCP_REORDER );
|
|
|
|
ret = dc->funcs->pExtTextOut(dc->physDev,x,y,flags|ETO_IGNORELANGUAGE,
|
|
lprect,gcp.lpOutString,count,lpDx);
|
|
HeapFree(GetProcessHeap(), 0, gcp.lpOutString);
|
|
} else
|
|
ret = dc->funcs->pExtTextOut(dc->physDev,x,y,flags,lprect,str,count,lpDx);
|
|
}
|
|
GDI_ReleaseObj( hdc );
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* TextOutA (GDI32.@)
|
|
*/
|
|
BOOL WINAPI TextOutA( HDC hdc, INT x, INT y, LPCSTR str, INT count )
|
|
{
|
|
return ExtTextOutA( hdc, x, y, 0, NULL, str, count, NULL );
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* TextOutW (GDI32.@)
|
|
*/
|
|
BOOL WINAPI TextOutW(HDC hdc, INT x, INT y, LPCWSTR str, INT count)
|
|
{
|
|
return ExtTextOutW( hdc, x, y, 0, NULL, str, count, NULL );
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* GetTextCharset [GDI32.@] Gets character set for font in DC
|
|
*
|
|
* NOTES
|
|
* Should it return a UINT32 instead of an INT32?
|
|
* => YES, as GetTextCharsetInfo returns UINT32
|
|
*
|
|
* RETURNS
|
|
* Success: Character set identifier
|
|
* Failure: DEFAULT_CHARSET
|
|
*/
|
|
UINT WINAPI GetTextCharset(
|
|
HDC hdc) /* [in] Handle to device context */
|
|
{
|
|
/* MSDN docs say this is equivalent */
|
|
return GetTextCharsetInfo(hdc, NULL, 0);
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* GetTextCharsetInfo [GDI32.@] Gets character set for font
|
|
*
|
|
* NOTES
|
|
* Should csi be an LPFONTSIGNATURE instead of an LPCHARSETINFO?
|
|
* Should it return a UINT32 instead of an INT32?
|
|
* => YES and YES, from win32.hlp from Borland
|
|
*
|
|
* This returns the actual charset selected by the driver rather than the
|
|
* value in lf.lfCharSet during CreateFont, to get that use
|
|
* GetObject(GetCurrentObject(...),...)
|
|
*
|
|
* RETURNS
|
|
* Success: Character set identifier
|
|
* Failure: DEFAULT_CHARSET
|
|
*/
|
|
UINT WINAPI GetTextCharsetInfo(
|
|
HDC hdc, /* [in] Handle to device context */
|
|
LPFONTSIGNATURE fs, /* [out] Pointer to struct to receive data */
|
|
DWORD flags) /* [in] Reserved - must be 0 */
|
|
{
|
|
UINT charSet = DEFAULT_CHARSET;
|
|
CHARSETINFO csinfo;
|
|
TEXTMETRICW tm;
|
|
|
|
if(!GetTextMetricsW(hdc, &tm)) return DEFAULT_CHARSET;
|
|
charSet = tm.tmCharSet;
|
|
|
|
if (fs != NULL) {
|
|
if (!TranslateCharsetInfo((LPDWORD)charSet, &csinfo, TCI_SRCCHARSET))
|
|
return DEFAULT_CHARSET;
|
|
memcpy(fs, &csinfo.fs, sizeof(FONTSIGNATURE));
|
|
}
|
|
return charSet;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* PolyTextOutA (GDI32.@)
|
|
*
|
|
* Draw several Strings
|
|
*/
|
|
BOOL WINAPI PolyTextOutA (
|
|
HDC hdc, /* [in] Handle to device context */
|
|
PPOLYTEXTA pptxt, /* [in] Array of strings */
|
|
INT cStrings /* [in] Number of strings in array */
|
|
)
|
|
{
|
|
for (; cStrings>0; cStrings--, pptxt++)
|
|
if (!ExtTextOutA( hdc, pptxt->x, pptxt->y, pptxt->uiFlags, &pptxt->rcl, pptxt->lpstr, pptxt->n, pptxt->pdx ))
|
|
return FALSE;
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
* PolyTextOutW (GDI32.@)
|
|
*
|
|
* Draw several Strings
|
|
*/
|
|
BOOL WINAPI PolyTextOutW (
|
|
HDC hdc, /* [in] Handle to device context */
|
|
PPOLYTEXTW pptxt, /* [in] Array of strings */
|
|
INT cStrings /* [in] Number of strings in array */
|
|
)
|
|
{
|
|
for (; cStrings>0; cStrings--, pptxt++)
|
|
if (!ExtTextOutW( hdc, pptxt->x, pptxt->y, pptxt->uiFlags, &pptxt->rcl, pptxt->lpstr, pptxt->n, pptxt->pdx ))
|
|
return FALSE;
|
|
return TRUE;
|
|
}
|