|  |  | 
|  | /* | 
|  | * %W% %E% | 
|  | * | 
|  | * (C) Copyright IBM Corp. 1998, 1999, 2000, 2001 - All Rights Reserved | 
|  | * | 
|  | */ | 
|  |  | 
|  | #include "LETypes.h" | 
|  | #include "LayoutEngine.h" | 
|  | #include "ArabicLayoutEngine.h" | 
|  | //#include "HebrewLayoutEngine.h" | 
|  | #include "IndicLayoutEngine.h" | 
|  | #include "ThaiLayoutEngine.h" | 
|  | #include "GXLayoutEngine.h" | 
|  | #include "ScriptAndLanguageTags.h" | 
|  |  | 
|  | #include "OpenTypeUtilities.h" | 
|  | #include "GlyphSubstitutionTables.h" | 
|  | #include "MorphTables.h" | 
|  |  | 
|  | #include "DefaultCharMapper.h" | 
|  |  | 
|  | U_NAMESPACE_BEGIN | 
|  |  | 
|  | #define ARRAY_SIZE(array) (sizeof array  / sizeof array[0]) | 
|  |  | 
|  | const LEUnicode32 DefaultCharMapper::controlChars[] = { | 
|  | 0x0009, 0x000A, 0x000D, | 
|  | /*0x200C, 0x200D,*/ 0x200E, 0x200F, | 
|  | 0x2028, 0x2029, 0x202A, 0x202B, 0x202C, 0x202D, 0x202E, | 
|  | 0x206A, 0x206B, 0x206C, 0x206D, 0x206E, 0x206F | 
|  | }; | 
|  |  | 
|  | const le_int32 DefaultCharMapper::controlCharsCount = ARRAY_SIZE(controlChars); | 
|  |  | 
|  | const LEUnicode32 DefaultCharMapper::mirroredChars[] = { | 
|  | 0x0028, 0x0029, // ascii paired punctuation | 
|  | 0x003c, 0x003e, | 
|  | 0x005b, 0x005d, | 
|  | 0x007b, 0x007d, | 
|  | 0x2045, 0x2046, // math symbols (not complete) | 
|  | 0x207d, 0x207e, | 
|  | 0x208d, 0x208e, | 
|  | 0x2264, 0x2265, | 
|  | 0x3008, 0x3009, // chinese paired punctuation | 
|  | 0x300a, 0x300b, | 
|  | 0x300c, 0x300d, | 
|  | 0x300e, 0x300f, | 
|  | 0x3010, 0x3011, | 
|  | 0x3014, 0x3015, | 
|  | 0x3016, 0x3017, | 
|  | 0x3018, 0x3019, | 
|  | 0x301a, 0x301b | 
|  | }; | 
|  |  | 
|  | const le_int32 DefaultCharMapper::mirroredCharsCount = ARRAY_SIZE(mirroredChars); | 
|  |  | 
|  | LEUnicode32 DefaultCharMapper::mapChar(LEUnicode32 ch) const | 
|  | { | 
|  | if (fFilterControls) { | 
|  | le_int32 index = OpenTypeUtilities::search((le_uint32)ch, (le_uint32 *)controlChars, controlCharsCount); | 
|  |  | 
|  | if (controlChars[index] == ch) { | 
|  | return 0xFFFF; | 
|  | } | 
|  | } | 
|  |  | 
|  | if (fMirror) { | 
|  | le_int32 index = OpenTypeUtilities::search((le_uint32) ch, (le_uint32 *)mirroredChars, mirroredCharsCount); | 
|  |  | 
|  | if (mirroredChars[index] == ch) { | 
|  | le_int32 mirrorOffset = ((index & 1) == 0) ? 1 : -1; | 
|  |  | 
|  | return mirroredChars[index + mirrorOffset]; | 
|  | } | 
|  | } | 
|  |  | 
|  | return ch; | 
|  | } | 
|  |  | 
|  | LayoutEngine::LayoutEngine(const LEFontInstance *fontInstance, le_int32 scriptCode, le_int32 languageCode) | 
|  | : fGlyphCount(0), fGlyphs(NULL), fCharIndices(NULL), fPositions(NULL), | 
|  | fFontInstance(fontInstance), fScriptCode(scriptCode), fLanguageCode(languageCode) | 
|  | { | 
|  | // nothing else to do? | 
|  | } | 
|  |  | 
|  | void LayoutEngine::getCharIndices(le_int32 charIndices[], le_int32 indexBase, LEErrorCode &success) const | 
|  | { | 
|  | le_int32 i; | 
|  |  | 
|  | if LE_FAILURE(success) { | 
|  | return; | 
|  | } | 
|  |  | 
|  | if (charIndices == NULL) { | 
|  | success = LE_ILLEGAL_ARGUMENT_ERROR; | 
|  | return; | 
|  | } | 
|  |  | 
|  | if (fCharIndices == NULL) { | 
|  | success = LE_NO_LAYOUT_ERROR; | 
|  | return; | 
|  | } | 
|  |  | 
|  | for (i = 0; i < fGlyphCount; i += 1) { | 
|  | charIndices[i] = fCharIndices[i] + indexBase; | 
|  | } | 
|  | } | 
|  |  | 
|  | // Copy the glyphs into caller's (32-bit) glyph array, OR in extraBits | 
|  | void LayoutEngine::getGlyphs(le_uint32 glyphs[], le_uint32 extraBits, LEErrorCode &success) const | 
|  | { | 
|  | le_int32 i; | 
|  |  | 
|  | if (LE_FAILURE(success)) { | 
|  | return; | 
|  | } | 
|  |  | 
|  | if (glyphs == NULL) { | 
|  | success = LE_ILLEGAL_ARGUMENT_ERROR; | 
|  | return; | 
|  | } | 
|  |  | 
|  | if (fGlyphs == NULL) { | 
|  | success = LE_NO_LAYOUT_ERROR; | 
|  | return; | 
|  | } | 
|  |  | 
|  | for (i = 0; i < fGlyphCount; i += 1) { | 
|  | glyphs[i] = fGlyphs[i] | extraBits; | 
|  | } | 
|  | }; | 
|  |  | 
|  | le_int32 LayoutEngine::computeGlyphs(const LEUnicode chars[], le_int32 offset, le_int32 count, le_int32 max, le_bool rightToLeft, | 
|  | LEGlyphID *&glyphs, le_int32 *&charIndices, LEErrorCode &success) | 
|  | { | 
|  | if (LE_FAILURE(success)) { | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | if (chars == NULL || offset < 0 || count < 0 || max < 0 || offset >= max || offset + count > max) { | 
|  | success = LE_ILLEGAL_ARGUMENT_ERROR; | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | mapCharsToGlyphs(chars, offset, count, rightToLeft, rightToLeft, glyphs, charIndices, success); | 
|  |  | 
|  | return count; | 
|  | } | 
|  |  | 
|  | // Input: glyphs | 
|  | // Output: positions | 
|  | void LayoutEngine::positionGlyphs(const LEGlyphID glyphs[], le_int32 glyphCount, float x, float y, float *&positions, LEErrorCode &success) | 
|  | { | 
|  | if (LE_FAILURE(success)) { | 
|  | return; | 
|  | } | 
|  |  | 
|  | if (glyphCount < 0) { | 
|  | success = LE_ILLEGAL_ARGUMENT_ERROR; | 
|  | return; | 
|  | } | 
|  |  | 
|  | if (positions == NULL) { | 
|  | positions = new float[2 * (glyphCount + 1)]; | 
|  |  | 
|  | if (positions == NULL) { | 
|  | success = LE_MEMORY_ALLOCATION_ERROR; | 
|  | return; | 
|  | } | 
|  | } | 
|  |  | 
|  | le_int32 i; | 
|  |  | 
|  | for (i = 0; i < glyphCount; i += 1) { | 
|  | LEPoint advance; | 
|  |  | 
|  | positions[i * 2] = x; | 
|  | positions[i * 2 + 1] = y; | 
|  |  | 
|  | fFontInstance->getGlyphAdvance(glyphs[i], advance); | 
|  | x += advance.fX; | 
|  | y += advance.fY; | 
|  | } | 
|  |  | 
|  | positions[glyphCount * 2] = x; | 
|  | positions[glyphCount * 2 + 1] = y; | 
|  | } | 
|  |  | 
|  | void LayoutEngine::adjustMarkGlyphs(const LEGlyphID glyphs[], le_int32 glyphCount, le_bool reverse, LEGlyphFilter *markFilter, | 
|  | float positions[], LEErrorCode &success) | 
|  | { | 
|  | float xAdjust = 0; | 
|  | le_int32 g = 0, direction = 1; | 
|  | le_int32 p; | 
|  |  | 
|  | if (LE_FAILURE(success)) { | 
|  | return; | 
|  | } | 
|  |  | 
|  | if (positions == NULL || markFilter == NULL) { | 
|  | success = LE_ILLEGAL_ARGUMENT_ERROR; | 
|  | return; | 
|  | } | 
|  |  | 
|  | if (reverse) { | 
|  | g = glyphCount - 1; | 
|  | direction = -1; | 
|  | } | 
|  |  | 
|  | for (p = 0; p < glyphCount; p += 1, g += direction) { | 
|  | float xAdvance = positions[(p + 1) * 2] - positions[p * 2]; | 
|  |  | 
|  | positions[p * 2] += xAdjust; | 
|  |  | 
|  | if (markFilter->accept(glyphs[g])) { | 
|  | xAdjust -= xAdvance; | 
|  | } | 
|  | } | 
|  |  | 
|  | positions[glyphCount * 2] += xAdjust; | 
|  | } | 
|  |  | 
|  | const void *LayoutEngine::getFontTable(LETag tableTag) const | 
|  | { | 
|  | return fFontInstance->getFontTable(tableTag); | 
|  | } | 
|  |  | 
|  | void LayoutEngine::mapCharsToGlyphs(const LEUnicode chars[], le_int32 offset, le_int32 count, le_bool reverse, le_bool mirror, | 
|  | LEGlyphID *&glyphs, le_int32 *&charIndices, LEErrorCode &success) | 
|  | { | 
|  | if (LE_FAILURE(success)) { | 
|  | return; | 
|  | } | 
|  |  | 
|  | if (chars == NULL || offset < 0 || count < 0) { | 
|  | success = LE_ILLEGAL_ARGUMENT_ERROR; | 
|  | return; | 
|  | } | 
|  |  | 
|  | if (glyphs == NULL) { | 
|  | glyphs = new LEGlyphID[count]; | 
|  |  | 
|  | if (glyphs == NULL) { | 
|  | success = LE_MEMORY_ALLOCATION_ERROR; | 
|  | return; | 
|  | } | 
|  | } | 
|  |  | 
|  | if (charIndices == NULL) { | 
|  | le_int32 i, dir = 1, out = 0; | 
|  |  | 
|  | if (reverse) { | 
|  | out = count - 1; | 
|  | dir = -1; | 
|  | } | 
|  |  | 
|  | charIndices = new le_int32[count]; | 
|  |  | 
|  | if (charIndices == NULL) { | 
|  | success = LE_MEMORY_ALLOCATION_ERROR; | 
|  | return; | 
|  | } | 
|  |  | 
|  | for (i = 0; i < count; i += 1, out += dir) { | 
|  | charIndices[out] = i; | 
|  | } | 
|  | } | 
|  |  | 
|  | DefaultCharMapper charMapper(true, mirror); | 
|  |  | 
|  | fFontInstance->mapCharsToGlyphs(chars, offset, count, reverse, &charMapper, glyphs); | 
|  | } | 
|  |  | 
|  | // Input: characters, font? | 
|  | // Output: glyphs, positions, char indices | 
|  | // Returns: number of glyphs | 
|  | le_int32 LayoutEngine::layoutChars(const LEUnicode chars[], le_int32 offset, le_int32 count, le_int32 max, le_bool rightToLeft, | 
|  | float x, float y, LEErrorCode &success) | 
|  | { | 
|  | if (LE_FAILURE(success)) { | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | if (chars == NULL || offset < 0 || count < 0 || max < 0 || offset >= max || offset + count > max) { | 
|  | success = LE_ILLEGAL_ARGUMENT_ERROR; | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | fGlyphCount = computeGlyphs(chars, offset, count, max, rightToLeft, fGlyphs, fCharIndices, success); | 
|  | positionGlyphs(fGlyphs, fGlyphCount, x, y, fPositions, success); | 
|  | adjustGlyphPositions(chars, offset, count, rightToLeft, fGlyphs, fGlyphCount, fPositions, success); | 
|  |  | 
|  | return fGlyphCount; | 
|  | } | 
|  |  | 
|  | void LayoutEngine::reset() | 
|  | { | 
|  | fGlyphCount = 0; | 
|  |  | 
|  | if (fGlyphs != NULL) { | 
|  | delete[] fGlyphs; | 
|  | fGlyphs = NULL; | 
|  | } | 
|  |  | 
|  | if (fCharIndices != NULL) { | 
|  | delete[] fCharIndices; | 
|  | fCharIndices = NULL; | 
|  | } | 
|  |  | 
|  | if (fPositions != NULL) { | 
|  | delete[] fPositions; | 
|  | fPositions = NULL; | 
|  | } | 
|  | } | 
|  |  | 
|  | LayoutEngine *LayoutEngine::layoutEngineFactory(const LEFontInstance *fontInstance, le_int32 scriptCode, le_int32 languageCode, LEErrorCode &success) | 
|  | { | 
|  | static le_uint32 gsubTableTag = 0x47535542; // "GSUB" | 
|  | static le_uint32 mortTableTag = 0x6D6F7274; // 'mort' | 
|  |  | 
|  | if (LE_FAILURE(success)) { | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | const GlyphSubstitutionTableHeader *gsubTable = (const GlyphSubstitutionTableHeader *) fontInstance->getFontTable(gsubTableTag); | 
|  | LayoutEngine *result = NULL; | 
|  |  | 
|  | if (gsubTable != NULL && gsubTable->coversScript(OpenTypeLayoutEngine::getScriptTag(scriptCode))) { | 
|  | switch (scriptCode) { | 
|  | case bengScriptCode: | 
|  | case devaScriptCode: | 
|  | case gujrScriptCode: | 
|  | case kndaScriptCode: | 
|  | case mlymScriptCode: | 
|  | case oryaScriptCode: | 
|  | case guruScriptCode: | 
|  | case tamlScriptCode: | 
|  | case teluScriptCode: | 
|  | result = new IndicOpenTypeLayoutEngine(fontInstance, scriptCode, languageCode, gsubTable); | 
|  | break; | 
|  |  | 
|  | case arabScriptCode: | 
|  | result = new ArabicOpenTypeLayoutEngine(fontInstance, scriptCode, languageCode, gsubTable); | 
|  | break; | 
|  |  | 
|  | default: | 
|  | result = new OpenTypeLayoutEngine(fontInstance, scriptCode, languageCode, gsubTable); | 
|  | break; | 
|  | } | 
|  | } else { | 
|  | const MorphTableHeader *morphTable = (MorphTableHeader *) fontInstance->getFontTable(mortTableTag); | 
|  |  | 
|  | if (morphTable != NULL) { | 
|  | result = new GXLayoutEngine(fontInstance, scriptCode, languageCode, morphTable); | 
|  | } else { | 
|  | switch (scriptCode) { | 
|  | case bengScriptCode: | 
|  | case devaScriptCode: | 
|  | case gujrScriptCode: | 
|  | case kndaScriptCode: | 
|  | case mlymScriptCode: | 
|  | case oryaScriptCode: | 
|  | case guruScriptCode: | 
|  | case tamlScriptCode: | 
|  | case teluScriptCode: | 
|  | { | 
|  | result = new IndicOpenTypeLayoutEngine(fontInstance, scriptCode, languageCode); | 
|  | break; | 
|  | } | 
|  |  | 
|  | case arabScriptCode: | 
|  | case hebrScriptCode: | 
|  | result = new UnicodeArabicOpenTypeLayoutEngine(fontInstance, scriptCode, languageCode); | 
|  | break; | 
|  |  | 
|  | //case hebrScriptCode: | 
|  | //    return new HebrewOpenTypeLayoutEngine(fontInstance, scriptCode, languageCode); | 
|  |  | 
|  | case thaiScriptCode: | 
|  | result = new ThaiLayoutEngine(fontInstance, scriptCode, languageCode); | 
|  | break; | 
|  |  | 
|  | default: | 
|  | result = new LayoutEngine(fontInstance, scriptCode, languageCode); | 
|  | break; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | if (result == NULL) { | 
|  | success = LE_MEMORY_ALLOCATION_ERROR; | 
|  | } | 
|  |  | 
|  | return result; | 
|  | } | 
|  |  | 
|  | LayoutEngine::~LayoutEngine() { | 
|  | reset(); | 
|  | } | 
|  |  | 
|  | U_NAMESPACE_END |