This commit is contained in:
ftang%netscape.com 1999-05-05 23:55:50 +00:00
Родитель 17ab1677de
Коммит d53e2d0843
2 изменённых файлов: 666 добавлений и 0 удалений

Просмотреть файл

@ -0,0 +1,596 @@
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*-
*
* The contents of this file are subject to the Netscape Public License
* Version 1.0 (the "NPL"); you may not use this file except in
* compliance with the NPL. You may obtain a copy of the NPL at
* http://www.mozilla.org/NPL/
*
* Software distributed under the NPL is distributed on an "AS IS" basis,
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the NPL
* for the specific language governing rights and limitations under the
* NPL.
*
* The Initial Developer of this code under the NPL is Netscape
* Communications Corporation. Portions created by Netscape are
* Copyright (C) 1999 Netscape Communications Corporation. All Rights
* Reserved.
*/
#include "nsUnicodeRenderingToolkit.h"
#include "nsUnicodeFontMappingMac.h"
#include "nsUnicodeFallbackCache.h"
#include "nsDrawingSurfaceMac.h"
#include "nsTransform2D.h"
#define BAD_FONT_NUM -1
#define BAD_SCRIPT 0x7F
#define STACK_TREASHOLD 1000
#define BAD_TEXT_ENCODING 0xFFFFFFFF
//------------------------------------------------------------------------
static UnicodeToTextInfo gConverters[32] = {
nsnull, nsnull, nsnull, nsnull, nsnull, nsnull, nsnull, nsnull,
nsnull, nsnull, nsnull, nsnull, nsnull, nsnull, nsnull, nsnull
};
//------------------------------------------------------------------------
UnicodeToTextInfo nsUnicodeRenderingToolkit :: GetConverterByScript(ScriptCode sc)
{
NS_PRECONDITION(sc < 32, "illegal script id");
if(sc >= 32)
return nsnull;
if(gConverters[sc] != nsnull) {
return gConverters[sc];
}
OSStatus err = noErr;
//
TextEncoding scriptEncoding;
err = ::UpgradeScriptInfoToTextEncoding(sc, kTextLanguageDontCare, kTextRegionDontCare, nsnull, &scriptEncoding);
if( noErr == err )
err = ::CreateUnicodeToTextInfoByEncoding(scriptEncoding, &gConverters[sc] );
if(noErr != err)
gConverters[sc] = nsnull;
return gConverters[sc];
}
//------------------------------------------------------------------------
#pragma mark -
//------------------------------------------------------------------------
nsUnicodeFallbackCache* nsUnicodeRenderingToolkit :: GetTECFallbackCache()
{
static nsUnicodeFallbackCache* gTECFallbackCache = nsnull;
if( ! gTECFallbackCache)
gTECFallbackCache = new nsUnicodeFallbackCache();
return gTECFallbackCache;
}
//------------------------------------------------------------------------
PRBool nsUnicodeRenderingToolkit :: TECFallbackGetWidth(
const PRUnichar *aCharPt,
short& oWidth,
short origFontNum,
const short* scriptFallbackFonts)
{
OSStatus err = noErr;
char buf[20];
ByteCount processBytes = 0;
ByteCount outLen = 0;
ScriptCode fallbackScript;
nsUnicodeFallbackCache* cache = GetTECFallbackCache();
if( cache->Get(*aCharPt, fallbackScript))
{
if(BAD_SCRIPT == fallbackScript)
return PR_FALSE;
UnicodeToTextInfo fallbackConverter = GetConverterByScript(fallbackScript);
if( fallbackConverter ) {
err = ::ConvertFromUnicodeToText(fallbackConverter, (ByteCount)(2), (ConstUniCharArrayPtr) aCharPt,
kUnicodeLooseMappingsMask , 0, NULL, 0, NULL,
STACK_TREASHOLD, &processBytes, &outLen,
(LogicalAddress)buf);
if( outLen > 0) {
::TextFont(scriptFallbackFonts[fallbackScript]);
GetScriptTextWidth(buf, outLen, oWidth);
::TextFont(origFontNum);
}
}
return PR_TRUE;
}
for( fallbackScript = 0 ; fallbackScript < 32; fallbackScript++)
{
if(BAD_FONT_NUM != scriptFallbackFonts[fallbackScript])
{
UnicodeToTextInfo fallbackConverter = GetConverterByScript(fallbackScript);
if( fallbackConverter ) {
err = ::ConvertFromUnicodeToText(fallbackConverter, (ByteCount)(2), (ConstUniCharArrayPtr) aCharPt,
kUnicodeLooseMappingsMask , 0, NULL, 0, NULL,
STACK_TREASHOLD, &processBytes, &outLen,
(LogicalAddress)buf);
if(outLen > 0)
{
NS_PRECONDITION(0 == (processBytes % 2), "strange conversion result");
::TextFont(scriptFallbackFonts[fallbackScript]);
GetScriptTextWidth(buf, outLen, oWidth);
::TextFont(origFontNum);
break;
}
}
}
}
if( 0 == outLen )
fallbackScript = BAD_SCRIPT;
// put into cache
cache->Set(*aCharPt, fallbackScript);
return (BAD_SCRIPT != fallbackScript);
}
//------------------------------------------------------------------------
PRBool nsUnicodeRenderingToolkit :: TECFallbackDrawChar(
const PRUnichar *aCharPt,
PRInt32 x,
PRInt32 y,
short& oWidth,
short origFontNum,
const short* scriptFallbackFonts)
{
OSStatus err = noErr;
char buf[20];
ByteCount processBytes = 0;
ByteCount outLen = 0;
ScriptCode fallbackScript;
nsUnicodeFallbackCache* cache = GetTECFallbackCache();
// since we always call TECFallbackGetWidth before TECFallbackDrawChar
// we could assume that we can always get the script code from cache.
if( cache->Get(*aCharPt, fallbackScript))
{
if(BAD_SCRIPT == fallbackScript)
return PR_FALSE;
UnicodeToTextInfo fallbackConverter = GetConverterByScript(fallbackScript);
if( fallbackConverter ) {
err = ::ConvertFromUnicodeToText(fallbackConverter, (ByteCount)(2), (ConstUniCharArrayPtr) aCharPt,
kUnicodeLooseMappingsMask , 0, NULL, 0, NULL,
STACK_TREASHOLD, &processBytes, &outLen,
(LogicalAddress)buf);
if( outLen > 0) {
::TextFont(scriptFallbackFonts[fallbackScript]);
DrawScriptText(buf, outLen, x, y, oWidth);
::TextFont(origFontNum);
}
}
return PR_TRUE;
}
return PR_FALSE;
}
//------------------------------------------------------------------------
PRBool nsUnicodeRenderingToolkit :: ATSUIFallbackGetWidth(
const PRUnichar *aCharPt,
short& oWidth,
short origFontNum,
short aSize, PRBool aBold, PRBool aItalic, nscolor aColor)
{
if( 0xFFFD == *aCharPt)
return PR_FALSE;
if (nsATSUIUtils::IsAvailable())
{
mATSUIToolkit.PrepareToDraw(mPort, mContext );
nsresult res = mATSUIToolkit.GetWidth(aCharPt, oWidth, aSize,
origFontNum,
aBold, aItalic, aColor);
if(NS_SUCCEEDED(res))
return PR_TRUE;
}
return PR_FALSE;
}
//------------------------------------------------------------------------
PRBool nsUnicodeRenderingToolkit :: ATSUIFallbackDrawChar(
const PRUnichar *aCharPt,
PRInt32 x, PRInt32 y,
short& oWidth,
short origFontNum,
short aSize, PRBool aBold, PRBool aItalic, nscolor aColor)
{
if( 0xFFFD == *aCharPt)
return PR_FALSE;
if (nsATSUIUtils::IsAvailable())
{
mATSUIToolkit.PrepareToDraw(mPort, mContext );
nsresult res = mATSUIToolkit.DrawString(aCharPt, x, y, oWidth, aSize,
origFontNum,
aBold, aItalic, aColor);
if(NS_SUCCEEDED(res))
return PR_TRUE;
}
return PR_FALSE;
}
static char *question = "<?>";
//------------------------------------------------------------------------
PRBool nsUnicodeRenderingToolkit :: QuestionMarkFallbackGetWidth(
const PRUnichar *aCharPt,
short& oWidth)
{
GetScriptTextWidth(question, 3,oWidth);
return PR_TRUE;
}
//------------------------------------------------------------------------
PRBool nsUnicodeRenderingToolkit :: QuestionMarkFallbackDrawChar(
const PRUnichar *aCharPt,
PRInt32 x,
PRInt32 y,
short& oWidth)
{
DrawScriptText(question, 3, x, y, oWidth);
return PR_TRUE;
}
//------------------------------------------------------------------------
void nsUnicodeRenderingToolkit :: GetScriptTextWidth(
const char* buf,
ByteCount aLen,
short& oWidth)
{
oWidth = ::TextWidth(buf, 0, aLen);
}
//------------------------------------------------------------------------
void nsUnicodeRenderingToolkit :: DrawScriptText(
const char* buf,
ByteCount aLen,
PRInt32 x,
PRInt32 y,
short& oWidth)
{
oWidth = ::TextWidth(buf, 0, aLen);
::MoveTo(x, y);
::DrawText(buf,0,aLen);
}
//------------------------------------------------------------------------
nsresult nsUnicodeRenderingToolkit :: GetTextSegmentWidth(
const PRUnichar *aString, PRUint32 aLength,
short fontNum, const short *scriptFallbackFonts,
PRUint32& oWidth)
{
if(aLength == 0) {
oWidth = 0;
return NS_OK;
}
NS_PRECONDITION(BAD_FONT_NUM != fontNum, "illegal font num");
short textWidth = 0;
PRUint32 processLen = 0;
char *heapBuf = nsnull;
PRUint32 heapBufSize = 0;
short thisWidth = 0;
char stackBuf[STACK_TREASHOLD];
char *buf ;
ByteCount processBytes;
ByteCount outLen;
OSStatus err = noErr;
::TextFont(fontNum);
ScriptCode script = ::FontToScript(fontNum);
UnicodeToTextInfo converter = GetConverterByScript(script);
// find buf from stack or heap. We only need to do this once in this function.
// put this out of the loop for performance...
ByteCount bufLen = aLength * 2 + 10;
if( bufLen > STACK_TREASHOLD)
{
if(bufLen > heapBufSize )
{
if(heapBuf)
delete[] heapBuf;
heapBuf = new char [bufLen];
heapBufSize = bufLen;
if(nsnull == heapBuf) {
return NS_ERROR_OUT_OF_MEMORY;
}
}
buf = heapBuf;
} else {
bufLen = STACK_TREASHOLD;
buf = stackBuf;
}
do {
if(converter)
{
outLen = 0;
err = noErr;
processBytes = 0;
err = ::ConvertFromUnicodeToText(converter, (ByteCount)(2*(aLength - processLen)), (ConstUniCharArrayPtr) aString,
0 , 0, NULL, 0, NULL,
bufLen, &processBytes, &outLen,
(LogicalAddress)buf);
// no mater if failed or not, as long as it convert some text, we process it.
if(outLen > 0)
{
GetScriptTextWidth(buf, outLen, thisWidth);
textWidth += thisWidth;
NS_PRECONDITION(0 == (processBytes % 2), "strange conversion result");
PRInt32 processUnicode = processBytes / 2;
processLen += processUnicode;
aString += processUnicode;
}
}
// Cannot precess by TEC, process one char a time by fallback mechanism
if( processLen < aLength)
{
PRBool fallbackDone = PR_FALSE;
// Fallback by try different Script code
fallbackDone = TECFallbackGetWidth(aString, thisWidth, fontNum, scriptFallbackFonts);
// Fallback by using ATSUI
if(! fallbackDone) {
nsFont *font;
mGS->mFontMetrics->GetFont(font);
fallbackDone = ATSUIFallbackGetWidth(aString, thisWidth, fontNum,
font->size,
(font->weight > NS_FONT_WEIGHT_NORMAL),
((NS_FONT_STYLE_ITALIC == font->style) || (NS_FONT_STYLE_OBLIQUE == font->style)),
mGS->mColor );
}
// Fallback to question mark
if(! fallbackDone)
QuestionMarkFallbackGetWidth(aString, thisWidth);
textWidth += thisWidth;
// for fallback measure/drawing, we always do one char a time.
aString++;
processLen++;
}
} while (processLen < aLength);
// release buffer if it is from heap
if(heapBuf)
delete[] heapBuf;
oWidth = textWidth;
return NS_OK;
}
//------------------------------------------------------------------------
nsresult nsUnicodeRenderingToolkit :: DrawTextSegment(
const PRUnichar *aString, PRUint32 aLength,
short fontNum, const short *scriptFallbackFonts,
PRInt32 x, PRInt32 y, PRUint32& oWidth)
{
if(aLength == 0) {
oWidth = 0;
return NS_OK;
}
NS_PRECONDITION(BAD_FONT_NUM != fontNum, "illegal font num");
short textWidth = 0;
PRUint32 processLen = 0;
char *heapBuf = nsnull;
PRUint32 heapBufSize = 0;
short thisWidth = 0;
char stackBuf[STACK_TREASHOLD];
char *buf ;
ByteCount processBytes;
ByteCount outLen;
OSStatus err = noErr;
::TextFont(fontNum);
ScriptCode script = ::FontToScript(fontNum);
UnicodeToTextInfo converter = GetConverterByScript(script);
// find buf from stack or heap. We only need to do this once in this function.
// put this out of the loop for performance...
ByteCount bufLen = aLength * 2 + 10;
if( bufLen > STACK_TREASHOLD)
{
if(bufLen > heapBufSize )
{
if(heapBuf)
delete[] heapBuf;
heapBuf = new char [bufLen];
heapBufSize = bufLen;
if(nsnull == heapBuf) {
return NS_ERROR_OUT_OF_MEMORY;
}
}
buf = heapBuf;
} else {
bufLen = STACK_TREASHOLD;
buf = stackBuf;
}
do {
if(converter)
{
outLen = 0;
err = noErr;
processBytes = 0;
err = ::ConvertFromUnicodeToText(converter, (ByteCount)(2*(aLength - processLen)), (ConstUniCharArrayPtr) aString,
0 , 0, NULL, 0, NULL,
bufLen, &processBytes, &outLen,
(LogicalAddress)buf);
// no mater if failed or not, as long as it convert some text, we process it.
if(outLen > 0)
{
DrawScriptText(buf, outLen, x, y, thisWidth);
textWidth += thisWidth;
x += thisWidth;
NS_PRECONDITION(0 == (processBytes % 2), "strange conversion result");
PRInt32 processUnicode = processBytes / 2;
processLen += processUnicode;
aString += processUnicode;
}
}
// Cannot precess by TEC, process one char a time by fallback mechanism
if( processLen < aLength)
{
PRBool fallbackDone = PR_FALSE;
// Fallback by try different Script code
fallbackDone = TECFallbackDrawChar(aString, x, y, thisWidth, fontNum, scriptFallbackFonts);
// Fallback by using ATSUI
if(! fallbackDone) {
nsFont *font;
mGS->mFontMetrics->GetFont(font);
fallbackDone = ATSUIFallbackDrawChar(aString, x, y, thisWidth, fontNum,
font->size,
(font->weight > NS_FONT_WEIGHT_NORMAL),
((NS_FONT_STYLE_ITALIC == font->style) || (NS_FONT_STYLE_OBLIQUE == font->style)),
mGS->mColor );
}
// Fallback to question mark
if(! fallbackDone)
QuestionMarkFallbackDrawChar(aString, x, y, thisWidth);
textWidth += thisWidth;
x += thisWidth;
// for fallback measure/drawing, we always do one char a time.
aString++;
processLen++;
}
} while (processLen < aLength);
// release buffer if it is from heap
if(heapBuf)
delete[] heapBuf;
oWidth = textWidth;
return NS_OK;
}
//------------------------------------------------------------------------
NS_IMETHODIMP nsUnicodeRenderingToolkit :: GetWidth(const PRUnichar *aString, PRUint32 aLength, nscoord &aWidth, PRInt32 *aFontID)
{
nsresult res = NS_OK;
nsFontMetricsMac *metrics = (nsFontMetricsMac*) mGS->mFontMetrics;
nsUnicodeFontMappingMac* fontmap = metrics->GetUnicodeFontMapping();
PRUint32 i;
short fontNum[2];
fontNum[0] = fontNum[1] = BAD_FONT_NUM;
PRUint32 start;
PRUint32 textWidth = 0;
PRUint32 thisWidth = 0;
const short *scriptFallbackFonts = fontmap->GetScriptFallbackFonts();
for(i =0, start = 0; i < aLength; i++)
{
fontNum[ i % 2] = fontmap->GetFontID(aString[i]);
if((fontNum[0] != fontNum[1]) && (0 != i))
{ // start new font run...
PRUint32 thisWidth = 0;
res = GetTextSegmentWidth(aString+start, i-start, fontNum[ (i+1) % 2], scriptFallbackFonts, thisWidth);
if(NS_FAILED (res))
return res;
textWidth += thisWidth;
start = i;
}
}
res = GetTextSegmentWidth(aString+start, aLength-start, fontNum[ (i+1) % 2], scriptFallbackFonts, thisWidth);
if(NS_FAILED (res))
return res;
textWidth += thisWidth;
aWidth = NSToCoordRound(float(textWidth) * mP2T);
return res;
}
//------------------------------------------------------------------------
NS_IMETHODIMP nsUnicodeRenderingToolkit :: DrawString(const PRUnichar *aString, PRUint32 aLength,
nscoord aX, nscoord aY, PRInt32 aFontID,
const nscoord* aSpacing)
{
nsresult res = NS_OK;
nsFontMetricsMac *metrics = (nsFontMetricsMac*) mGS->mFontMetrics;
nsUnicodeFontMappingMac* fontmap = metrics->GetUnicodeFontMapping();
PRInt32 x = aX;
PRInt32 y = aY;
nscoord ascent = 0;
mGS->mFontMetrics->GetMaxAscent(ascent);
y += ascent;
mGS->mTMatrix->TransformCoord(&x,&y);
PRUint32 i;
PRInt32 currentX = x;
PRUint32 thisWidth = 0;
const short *scriptFallbackFonts = fontmap->GetScriptFallbackFonts();
if(aSpacing) {
int buffer[STACK_TREASHOLD];
int* spacing = (aLength <= STACK_TREASHOLD ? buffer : new int[aLength]);
if (spacing)
{
mGS->mTMatrix->ScaleXCoords(aSpacing, aLength, spacing);
for(i =0; i < aLength; i++)
{
short curFontNum = fontmap->GetFontID(aString[i]);
res = DrawTextSegment(aString+i, 1, curFontNum, scriptFallbackFonts, currentX, y, thisWidth);
if(NS_FAILED(res))
goto end_of_func;
currentX += spacing[i];
}
if (spacing != buffer)
delete[] spacing;
}
else {
res = NS_ERROR_OUT_OF_MEMORY;
goto end_of_func;
}
} else {
short fontNum[2];
fontNum[0] = fontNum[1] = BAD_FONT_NUM;
PRUint32 start;
for(i =0, start = 0; i < aLength; i++)
{
fontNum[ i % 2] = fontmap->GetFontID(aString[i]);
if((fontNum[0] != fontNum[1]) && (0 != i))
{ // start new font run...
res = DrawTextSegment(aString+start, i-start, fontNum[ (i+1) % 2], scriptFallbackFonts, currentX, y, thisWidth);
if(NS_FAILED(res))
goto end_of_func;
currentX += thisWidth;
start = i;
}
}
res = DrawTextSegment(aString+start, aLength-start, fontNum[ (i+1) % 2], scriptFallbackFonts, currentX, y, thisWidth);
if(NS_FAILED(res))
goto end_of_func;
}
end_of_func:
return res;
}
//------------------------------------------------------------------------
NS_IMETHODIMP nsUnicodeRenderingToolkit :: PrepareToDraw(float aP2T, nsIDeviceContext* aContext, GraphicState* aGS, GrafPtr aPort)
{
mP2T = aP2T;
mContext = aContext;
mGS = aGS;
mPort = aPort;
return NS_OK;
}

Просмотреть файл

@ -0,0 +1,70 @@
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*-
*
* The contents of this file are subject to the Netscape Public License
* Version 1.0 (the "NPL"); you may not use this file except in
* compliance with the NPL. You may obtain a copy of the NPL at
* http://www.mozilla.org/NPL/
*
* Software distributed under the NPL is distributed on an "AS IS" basis,
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the NPL
* for the specific language governing rights and limitations under the
* NPL.
*
* The Initial Developer of this code under the NPL is Netscape
* Communications Corporation. Portions created by Netscape are
* Copyright (C) 1999 Netscape Communications Corporation. All Rights
* Reserved.
*/
#ifndef nsUnicodeRenderingToolkit_h__
#define nsUnicodeRenderingToolkit_h__
#include "nsATSUIUtils.h"
#include <UnicodeConverter.h>
class nsUnicodeFallbackCache;
class nsIDeviceContext;
class GraphicState;
class nsUnicodeRenderingToolkit
{
nsUnicodeRenderingToolkit() {};
~nsUnicodeRenderingToolkit() {};
public:
NS_IMETHOD PrepareToDraw(float aP2T, nsIDeviceContext* aContext, GraphicState* aGS, GrafPtr aPort);
NS_IMETHOD GetWidth(const PRUnichar *aString, PRUint32 aLength, nscoord &aWidth,
PRInt32 *aFontID);
NS_IMETHOD DrawString(const PRUnichar *aString, PRUint32 aLength, nscoord aX, nscoord aY,
PRInt32 aFontID,
const nscoord* aSpacing);
private:
// Unicode text measure/drawing functions
UnicodeToTextInfo GetConverterByScript(ScriptCode sc);
PRBool TECFallbackGetWidth(const PRUnichar *pChar, short& oWidth,
short fontNum, const short* scriptFallbackFonts);
PRBool TECFallbackDrawChar(const PRUnichar *pChar, PRInt32 x, PRInt32 y, short& oWidth,
short fontNum, const short* scriptFallbackFonts);
PRBool ATSUIFallbackGetWidth(const PRUnichar *pChar, short& oWidth, short fontNum, short aSize, PRBool aBold, PRBool aItalic, nscolor aColor);
PRBool ATSUIFallbackDrawChar(const PRUnichar *pChar, PRInt32 x, PRInt32 y, short& oWidth, short fontNum, short aSize, PRBool aBold, PRBool aItalic, nscolor aColor);
PRBool QuestionMarkFallbackGetWidth(const PRUnichar *pChar, short& oWidth);
PRBool QuestionMarkFallbackDrawChar(const PRUnichar *pChar, PRInt32 x, PRInt32 y, short& oWidth);
void GetScriptTextWidth(const char* aText, ByteCount aLen, short& aWidth);
void DrawScriptText(const char* aText, ByteCount aLen, PRInt32 x, PRInt32 y, short& aWidth);
nsresult GetTextSegmentWidth( const PRUnichar *aString, PRUint32 aLength, short fontNum, const short *scriptFallbackFonts,
PRUint32& oWidth);
nsresult DrawTextSegment( const PRUnichar *aString, PRUint32 aLength, short fontNum, const short *scriptFallbackFonts,
PRInt32 x, PRInt32 y, PRUint32& oWidth);
nsUnicodeFallbackCache* GetTECFallbackCache();
private:
float mP2T; // Pixel to Twip conversion factor
nsIDeviceContext * mContext;
GraphicState * mGS; // current graphic state - shortcut for mCurrentSurface->GetGS()
GrafPtr mPort; // current grafPort - shortcut for mCurrentSurface->GetPort()
nsATSUIToolkit mATSUIToolkit;
};
#endif /* nsUnicodeRenderingToolkit_h__ */