2000-11-28 20:56:52 +00:00
|
|
|
/*
|
|
|
|
*
|
2013-04-18 21:24:51 +00:00
|
|
|
* (C) Copyright IBM Corp. 1998-2013 - All Rights Reserved
|
2000-11-28 20:56:52 +00:00
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "LETypes.h"
|
|
|
|
#include "OpenTypeTables.h"
|
|
|
|
#include "GlyphDefinitionTables.h"
|
|
|
|
#include "GlyphPositionAdjustments.h"
|
|
|
|
#include "GlyphIterator.h"
|
2004-04-12 18:51:31 +00:00
|
|
|
#include "LEGlyphStorage.h"
|
2000-11-28 20:56:52 +00:00
|
|
|
#include "Lookups.h"
|
|
|
|
#include "LESwaps.h"
|
|
|
|
|
2001-10-16 00:39:01 +00:00
|
|
|
U_NAMESPACE_BEGIN
|
|
|
|
|
2005-08-06 00:48:27 +00:00
|
|
|
GlyphIterator::GlyphIterator(LEGlyphStorage &theGlyphStorage, GlyphPositionAdjustments *theGlyphPositionAdjustments, le_bool rightToLeft, le_uint16 theLookupFlags,
|
2013-04-18 21:24:51 +00:00
|
|
|
FeatureMask theFeatureMask, const LEReferenceTo<GlyphDefinitionTableHeader> &theGlyphDefinitionTableHeader)
|
2004-04-12 18:51:31 +00:00
|
|
|
: direction(1), position(-1), nextLimit(-1), prevLimit(-1),
|
|
|
|
glyphStorage(theGlyphStorage), glyphPositionAdjustments(theGlyphPositionAdjustments),
|
2007-10-09 19:40:14 +00:00
|
|
|
srcIndex(-1), destIndex(-1), lookupFlags(theLookupFlags), featureMask(theFeatureMask), glyphGroup(0),
|
2013-04-18 21:24:51 +00:00
|
|
|
glyphClassDefinitionTable(), markAttachClassDefinitionTable()
|
2000-11-28 20:56:52 +00:00
|
|
|
|
|
|
|
{
|
2013-04-18 21:24:51 +00:00
|
|
|
LEErrorCode success = LE_NO_ERROR; // TODO
|
2004-05-07 23:29:16 +00:00
|
|
|
le_int32 glyphCount = glyphStorage.getGlyphCount();
|
2004-04-12 18:51:31 +00:00
|
|
|
|
2013-04-18 21:24:51 +00:00
|
|
|
if (theGlyphDefinitionTableHeader.isValid()) {
|
|
|
|
glyphClassDefinitionTable = theGlyphDefinitionTableHeader
|
|
|
|
-> getGlyphClassDefinitionTable(theGlyphDefinitionTableHeader, success);
|
|
|
|
markAttachClassDefinitionTable = theGlyphDefinitionTableHeader
|
|
|
|
->getMarkAttachClassDefinitionTable(theGlyphDefinitionTableHeader, success);
|
2000-11-28 20:56:52 +00:00
|
|
|
}
|
|
|
|
|
2004-05-07 23:29:16 +00:00
|
|
|
nextLimit = glyphCount;
|
2004-04-12 18:51:31 +00:00
|
|
|
|
2001-01-19 00:30:17 +00:00
|
|
|
if (rightToLeft) {
|
2000-11-28 20:56:52 +00:00
|
|
|
direction = -1;
|
2004-04-12 18:51:31 +00:00
|
|
|
position = glyphCount;
|
2000-11-28 20:56:52 +00:00
|
|
|
nextLimit = -1;
|
2004-04-12 18:51:31 +00:00
|
|
|
prevLimit = glyphCount;
|
2000-11-28 20:56:52 +00:00
|
|
|
}
|
2013-06-25 00:46:51 +00:00
|
|
|
filterResetCache();
|
2000-11-28 20:56:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
GlyphIterator::GlyphIterator(GlyphIterator &that)
|
2004-04-12 18:51:31 +00:00
|
|
|
: glyphStorage(that.glyphStorage)
|
2000-11-28 20:56:52 +00:00
|
|
|
{
|
2002-06-10 23:40:33 +00:00
|
|
|
direction = that.direction;
|
|
|
|
position = that.position;
|
|
|
|
nextLimit = that.nextLimit;
|
|
|
|
prevLimit = that.prevLimit;
|
|
|
|
|
2000-11-28 20:56:52 +00:00
|
|
|
glyphPositionAdjustments = that.glyphPositionAdjustments;
|
2003-11-25 23:41:24 +00:00
|
|
|
srcIndex = that.srcIndex;
|
|
|
|
destIndex = that.destIndex;
|
2000-11-28 20:56:52 +00:00
|
|
|
lookupFlags = that.lookupFlags;
|
2005-08-06 00:48:27 +00:00
|
|
|
featureMask = that.featureMask;
|
2007-10-09 19:40:14 +00:00
|
|
|
glyphGroup = that.glyphGroup;
|
2000-11-28 20:56:52 +00:00
|
|
|
glyphClassDefinitionTable = that.glyphClassDefinitionTable;
|
|
|
|
markAttachClassDefinitionTable = that.markAttachClassDefinitionTable;
|
2013-06-25 00:46:51 +00:00
|
|
|
filterResetCache();
|
2000-11-28 20:56:52 +00:00
|
|
|
}
|
|
|
|
|
2005-08-06 00:48:27 +00:00
|
|
|
GlyphIterator::GlyphIterator(GlyphIterator &that, FeatureMask newFeatureMask)
|
2004-04-12 18:51:31 +00:00
|
|
|
: glyphStorage(that.glyphStorage)
|
2003-03-07 23:33:41 +00:00
|
|
|
{
|
|
|
|
direction = that.direction;
|
|
|
|
position = that.position;
|
|
|
|
nextLimit = that.nextLimit;
|
|
|
|
prevLimit = that.prevLimit;
|
|
|
|
|
|
|
|
glyphPositionAdjustments = that.glyphPositionAdjustments;
|
2004-05-07 23:29:16 +00:00
|
|
|
srcIndex = that.srcIndex;
|
|
|
|
destIndex = that.destIndex;
|
2003-03-07 23:33:41 +00:00
|
|
|
lookupFlags = that.lookupFlags;
|
2005-08-06 00:48:27 +00:00
|
|
|
featureMask = newFeatureMask;
|
2007-10-09 19:40:14 +00:00
|
|
|
glyphGroup = 0;
|
2003-03-07 23:33:41 +00:00
|
|
|
glyphClassDefinitionTable = that.glyphClassDefinitionTable;
|
|
|
|
markAttachClassDefinitionTable = that.markAttachClassDefinitionTable;
|
2013-06-25 00:46:51 +00:00
|
|
|
filterResetCache();
|
2003-03-07 23:33:41 +00:00
|
|
|
}
|
|
|
|
|
2000-11-28 20:56:52 +00:00
|
|
|
GlyphIterator::GlyphIterator(GlyphIterator &that, le_uint16 newLookupFlags)
|
2004-04-12 18:51:31 +00:00
|
|
|
: glyphStorage(that.glyphStorage)
|
2000-11-28 20:56:52 +00:00
|
|
|
{
|
2002-06-10 23:40:33 +00:00
|
|
|
direction = that.direction;
|
|
|
|
position = that.position;
|
|
|
|
nextLimit = that.nextLimit;
|
|
|
|
prevLimit = that.prevLimit;
|
|
|
|
|
2000-11-28 20:56:52 +00:00
|
|
|
glyphPositionAdjustments = that.glyphPositionAdjustments;
|
2004-05-07 23:29:16 +00:00
|
|
|
srcIndex = that.srcIndex;
|
|
|
|
destIndex = that.destIndex;
|
2000-11-28 20:56:52 +00:00
|
|
|
lookupFlags = newLookupFlags;
|
2005-08-06 00:48:27 +00:00
|
|
|
featureMask = that.featureMask;
|
2007-10-09 19:40:14 +00:00
|
|
|
glyphGroup = that.glyphGroup;
|
2000-11-28 20:56:52 +00:00
|
|
|
glyphClassDefinitionTable = that.glyphClassDefinitionTable;
|
|
|
|
markAttachClassDefinitionTable = that.markAttachClassDefinitionTable;
|
2013-06-25 00:46:51 +00:00
|
|
|
filterResetCache();
|
2000-11-28 20:56:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
GlyphIterator::~GlyphIterator()
|
|
|
|
{
|
2004-05-07 23:29:16 +00:00
|
|
|
// nothing to do, right?
|
2003-09-24 21:05:40 +00:00
|
|
|
}
|
|
|
|
|
2005-08-06 00:48:27 +00:00
|
|
|
void GlyphIterator::reset(le_uint16 newLookupFlags, FeatureMask newFeatureMask)
|
2003-09-24 21:05:40 +00:00
|
|
|
{
|
2005-08-06 00:48:27 +00:00
|
|
|
position = prevLimit;
|
|
|
|
featureMask = newFeatureMask;
|
2007-10-09 19:40:14 +00:00
|
|
|
glyphGroup = 0;
|
2005-08-06 00:48:27 +00:00
|
|
|
lookupFlags = newLookupFlags;
|
2013-06-25 00:46:51 +00:00
|
|
|
filterResetCache();
|
2003-09-24 21:05:40 +00:00
|
|
|
}
|
|
|
|
|
2008-10-28 14:50:15 +00:00
|
|
|
LEGlyphID *GlyphIterator::insertGlyphs(le_int32 count, LEErrorCode& success)
|
2003-09-24 21:05:40 +00:00
|
|
|
{
|
2008-10-28 14:50:15 +00:00
|
|
|
return glyphStorage.insertGlyphs(position, count, success);
|
2003-09-24 21:05:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
le_int32 GlyphIterator::applyInsertions()
|
|
|
|
{
|
2004-05-07 23:29:16 +00:00
|
|
|
le_int32 newGlyphCount = glyphStorage.applyInsertions();
|
2003-09-24 21:05:40 +00:00
|
|
|
|
2004-05-07 23:29:16 +00:00
|
|
|
if (direction < 0) {
|
|
|
|
prevLimit = newGlyphCount;
|
|
|
|
} else {
|
|
|
|
nextLimit = newGlyphCount;
|
|
|
|
}
|
2003-09-24 21:05:40 +00:00
|
|
|
|
2004-05-07 23:29:16 +00:00
|
|
|
return newGlyphCount;
|
2000-11-28 20:56:52 +00:00
|
|
|
}
|
|
|
|
|
2001-01-19 00:30:17 +00:00
|
|
|
le_int32 GlyphIterator::getCurrStreamPosition() const
|
2000-11-28 20:56:52 +00:00
|
|
|
{
|
|
|
|
return position;
|
|
|
|
}
|
|
|
|
|
2001-01-19 00:30:17 +00:00
|
|
|
le_bool GlyphIterator::isRightToLeft() const
|
2000-11-28 20:56:52 +00:00
|
|
|
{
|
|
|
|
return direction < 0;
|
|
|
|
}
|
|
|
|
|
2001-09-20 00:37:55 +00:00
|
|
|
le_bool GlyphIterator::ignoresMarks() const
|
|
|
|
{
|
|
|
|
return (lookupFlags & lfIgnoreMarks) != 0;
|
|
|
|
}
|
|
|
|
|
2002-06-10 23:40:33 +00:00
|
|
|
le_bool GlyphIterator::baselineIsLogicalEnd() const
|
|
|
|
{
|
|
|
|
return (lookupFlags & lfBaselineIsLogicalEnd) != 0;
|
|
|
|
}
|
|
|
|
|
2001-01-19 00:30:17 +00:00
|
|
|
LEGlyphID GlyphIterator::getCurrGlyphID() const
|
2000-11-28 20:56:52 +00:00
|
|
|
{
|
2001-01-19 00:30:17 +00:00
|
|
|
if (direction < 0) {
|
|
|
|
if (position <= nextLimit || position >= prevLimit) {
|
2000-11-28 20:56:52 +00:00
|
|
|
return 0xFFFF;
|
|
|
|
}
|
2001-01-19 00:30:17 +00:00
|
|
|
} else {
|
|
|
|
if (position <= prevLimit || position >= nextLimit) {
|
2000-11-28 20:56:52 +00:00
|
|
|
return 0xFFFF;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-05-07 23:29:16 +00:00
|
|
|
return glyphStorage[position];
|
2000-11-28 20:56:52 +00:00
|
|
|
}
|
|
|
|
|
2005-01-14 17:25:11 +00:00
|
|
|
void GlyphIterator::getCursiveEntryPoint(LEPoint &entryPoint) const
|
2002-06-10 23:40:33 +00:00
|
|
|
{
|
|
|
|
if (direction < 0) {
|
2005-01-14 17:25:11 +00:00
|
|
|
if (position <= nextLimit || position >= prevLimit) {
|
2000-11-28 20:56:52 +00:00
|
|
|
return;
|
|
|
|
}
|
2001-01-19 00:30:17 +00:00
|
|
|
} else {
|
|
|
|
if (position <= prevLimit || position >= nextLimit) {
|
2000-11-28 20:56:52 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-01-14 17:25:11 +00:00
|
|
|
glyphPositionAdjustments->getEntryPoint(position, entryPoint);
|
2000-11-28 20:56:52 +00:00
|
|
|
}
|
|
|
|
|
2005-01-14 17:25:11 +00:00
|
|
|
void GlyphIterator::getCursiveExitPoint(LEPoint &exitPoint) const
|
2002-06-10 23:40:33 +00:00
|
|
|
{
|
2005-01-14 17:25:11 +00:00
|
|
|
if (direction < 0) {
|
|
|
|
if (position <= nextLimit || position >= prevLimit) {
|
2002-06-10 23:40:33 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
} else {
|
2005-01-14 17:25:11 +00:00
|
|
|
if (position <= prevLimit || position >= nextLimit) {
|
2002-06-10 23:40:33 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-01-14 17:25:11 +00:00
|
|
|
glyphPositionAdjustments->getExitPoint(position, exitPoint);
|
2002-06-10 23:40:33 +00:00
|
|
|
}
|
|
|
|
|
2003-02-05 00:05:40 +00:00
|
|
|
void GlyphIterator::setCurrGlyphID(TTGlyphID glyphID)
|
2000-11-28 20:56:52 +00:00
|
|
|
{
|
2004-05-07 23:29:16 +00:00
|
|
|
LEGlyphID glyph = glyphStorage[position];
|
2004-04-12 18:51:31 +00:00
|
|
|
|
2004-05-07 23:29:16 +00:00
|
|
|
glyphStorage[position] = LE_SET_GLYPH(glyph, glyphID);
|
2000-11-28 20:56:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void GlyphIterator::setCurrStreamPosition(le_int32 newPosition)
|
|
|
|
{
|
2001-01-19 00:30:17 +00:00
|
|
|
if (direction < 0) {
|
|
|
|
if (newPosition >= prevLimit) {
|
2000-11-28 20:56:52 +00:00
|
|
|
position = prevLimit;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2001-01-19 00:30:17 +00:00
|
|
|
if (newPosition <= nextLimit) {
|
2000-11-28 20:56:52 +00:00
|
|
|
position = nextLimit;
|
|
|
|
return;
|
|
|
|
}
|
2001-01-19 00:30:17 +00:00
|
|
|
} else {
|
|
|
|
if (newPosition <= prevLimit) {
|
2000-11-28 20:56:52 +00:00
|
|
|
position = prevLimit;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2001-01-19 00:30:17 +00:00
|
|
|
if (newPosition >= nextLimit) {
|
2000-11-28 20:56:52 +00:00
|
|
|
position = nextLimit;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
position = newPosition - direction;
|
|
|
|
next();
|
|
|
|
}
|
|
|
|
|
2002-06-10 23:40:33 +00:00
|
|
|
void GlyphIterator::setCurrGlyphBaseOffset(le_int32 baseOffset)
|
|
|
|
{
|
|
|
|
if (direction < 0) {
|
|
|
|
if (position <= nextLimit || position >= prevLimit) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (position <= prevLimit || position >= nextLimit) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-01-14 17:25:11 +00:00
|
|
|
glyphPositionAdjustments->setBaseOffset(position, baseOffset);
|
2002-06-10 23:40:33 +00:00
|
|
|
}
|
|
|
|
|
2000-11-28 20:56:52 +00:00
|
|
|
void GlyphIterator::adjustCurrGlyphPositionAdjustment(float xPlacementAdjust, float yPlacementAdjust,
|
|
|
|
float xAdvanceAdjust, float yAdvanceAdjust)
|
|
|
|
{
|
2001-01-19 00:30:17 +00:00
|
|
|
if (direction < 0) {
|
|
|
|
if (position <= nextLimit || position >= prevLimit) {
|
2000-11-28 20:56:52 +00:00
|
|
|
return;
|
|
|
|
}
|
2001-01-19 00:30:17 +00:00
|
|
|
} else {
|
|
|
|
if (position <= prevLimit || position >= nextLimit) {
|
2000-11-28 20:56:52 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-01-14 17:25:11 +00:00
|
|
|
glyphPositionAdjustments->adjustXPlacement(position, xPlacementAdjust);
|
|
|
|
glyphPositionAdjustments->adjustYPlacement(position, yPlacementAdjust);
|
|
|
|
glyphPositionAdjustments->adjustXAdvance(position, xAdvanceAdjust);
|
|
|
|
glyphPositionAdjustments->adjustYAdvance(position, yAdvanceAdjust);
|
2000-11-28 20:56:52 +00:00
|
|
|
}
|
|
|
|
|
2004-08-20 18:12:26 +00:00
|
|
|
void GlyphIterator::setCurrGlyphPositionAdjustment(float xPlacementAdjust, float yPlacementAdjust,
|
|
|
|
float xAdvanceAdjust, float yAdvanceAdjust)
|
|
|
|
{
|
|
|
|
if (direction < 0) {
|
|
|
|
if (position <= nextLimit || position >= prevLimit) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (position <= prevLimit || position >= nextLimit) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-01-14 17:25:11 +00:00
|
|
|
glyphPositionAdjustments->setXPlacement(position, xPlacementAdjust);
|
|
|
|
glyphPositionAdjustments->setYPlacement(position, yPlacementAdjust);
|
|
|
|
glyphPositionAdjustments->setXAdvance(position, xAdvanceAdjust);
|
|
|
|
glyphPositionAdjustments->setYAdvance(position, yAdvanceAdjust);
|
2004-08-20 18:12:26 +00:00
|
|
|
}
|
|
|
|
|
2007-06-22 03:57:45 +00:00
|
|
|
void GlyphIterator::clearCursiveEntryPoint()
|
|
|
|
{
|
|
|
|
if (direction < 0) {
|
|
|
|
if (position <= nextLimit || position >= prevLimit) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (position <= prevLimit || position >= nextLimit) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
glyphPositionAdjustments->clearEntryPoint(position);
|
|
|
|
}
|
|
|
|
|
|
|
|
void GlyphIterator::clearCursiveExitPoint()
|
|
|
|
{
|
|
|
|
if (direction < 0) {
|
|
|
|
if (position <= nextLimit || position >= prevLimit) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (position <= prevLimit || position >= nextLimit) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
glyphPositionAdjustments->clearExitPoint(position);
|
|
|
|
}
|
|
|
|
|
2005-01-14 17:25:11 +00:00
|
|
|
void GlyphIterator::setCursiveEntryPoint(LEPoint &entryPoint)
|
2002-06-10 23:40:33 +00:00
|
|
|
{
|
|
|
|
if (direction < 0) {
|
|
|
|
if (position <= nextLimit || position >= prevLimit) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (position <= prevLimit || position >= nextLimit) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-01-21 18:36:02 +00:00
|
|
|
glyphPositionAdjustments->setEntryPoint(position, entryPoint, baselineIsLogicalEnd());
|
2002-06-10 23:40:33 +00:00
|
|
|
}
|
|
|
|
|
2005-01-14 17:25:11 +00:00
|
|
|
void GlyphIterator::setCursiveExitPoint(LEPoint &exitPoint)
|
2002-06-10 23:40:33 +00:00
|
|
|
{
|
|
|
|
if (direction < 0) {
|
|
|
|
if (position <= nextLimit || position >= prevLimit) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (position <= prevLimit || position >= nextLimit) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-01-21 18:36:02 +00:00
|
|
|
glyphPositionAdjustments->setExitPoint(position, exitPoint, baselineIsLogicalEnd());
|
2002-06-10 23:40:33 +00:00
|
|
|
}
|
|
|
|
|
2005-01-14 17:25:11 +00:00
|
|
|
void GlyphIterator::setCursiveGlyph()
|
2002-06-10 23:40:33 +00:00
|
|
|
{
|
|
|
|
if (direction < 0) {
|
2005-01-14 17:25:11 +00:00
|
|
|
if (position <= nextLimit || position >= prevLimit) {
|
2002-06-10 23:40:33 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
} else {
|
2005-01-14 17:25:11 +00:00
|
|
|
if (position <= prevLimit || position >= nextLimit) {
|
2002-06-10 23:40:33 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-01-21 18:36:02 +00:00
|
|
|
glyphPositionAdjustments->setCursiveGlyph(position, baselineIsLogicalEnd());
|
2002-06-10 23:40:33 +00:00
|
|
|
}
|
|
|
|
|
2013-06-25 00:46:51 +00:00
|
|
|
void GlyphIterator::filterResetCache(void) {
|
|
|
|
filterCacheValid = FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
le_bool GlyphIterator::filterGlyph(le_uint32 index)
|
2000-11-28 20:56:52 +00:00
|
|
|
{
|
2004-04-12 18:51:31 +00:00
|
|
|
LEGlyphID glyphID = glyphStorage[index];
|
2000-11-28 20:56:52 +00:00
|
|
|
|
2013-06-25 00:46:51 +00:00
|
|
|
if (!filterCacheValid || filterCache.id != glyphID) {
|
|
|
|
filterCache.id = glyphID;
|
|
|
|
|
|
|
|
le_bool &filterResult = filterCache.result; // NB: Making this a reference to accept the updated value, in case
|
|
|
|
// we want more fancy cacheing in the future.
|
|
|
|
if (LE_GET_GLYPH(glyphID) >= 0xFFFE) {
|
|
|
|
filterResult = TRUE;
|
|
|
|
} else {
|
|
|
|
LEErrorCode success = LE_NO_ERROR;
|
|
|
|
le_int32 glyphClass = gcdNoGlyphClass;
|
|
|
|
if (glyphClassDefinitionTable.isValid()) {
|
|
|
|
glyphClass = glyphClassDefinitionTable->getGlyphClass(glyphClassDefinitionTable, glyphID, success);
|
2000-11-28 20:56:52 +00:00
|
|
|
}
|
2013-06-25 00:46:51 +00:00
|
|
|
switch (glyphClass) {
|
|
|
|
case gcdNoGlyphClass:
|
|
|
|
filterResult = FALSE;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case gcdSimpleGlyph:
|
|
|
|
filterResult = (lookupFlags & lfIgnoreBaseGlyphs) != 0;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case gcdLigatureGlyph:
|
|
|
|
filterResult = (lookupFlags & lfIgnoreLigatures) != 0;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case gcdMarkGlyph:
|
|
|
|
if ((lookupFlags & lfIgnoreMarks) != 0) {
|
|
|
|
filterResult = TRUE;
|
|
|
|
} else {
|
|
|
|
le_uint16 markAttachType = (lookupFlags & lfMarkAttachTypeMask) >> lfMarkAttachTypeShift;
|
|
|
|
|
|
|
|
if ((markAttachType != 0) && (markAttachClassDefinitionTable.isValid())) {
|
|
|
|
filterResult = (markAttachClassDefinitionTable
|
|
|
|
-> getGlyphClass(markAttachClassDefinitionTable, glyphID, success) != markAttachType);
|
|
|
|
} else {
|
|
|
|
filterResult = FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case gcdComponentGlyph:
|
|
|
|
filterResult = ((lookupFlags & lfIgnoreBaseGlyphs) != 0);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
filterResult = FALSE;
|
|
|
|
break;
|
2000-11-28 20:56:52 +00:00
|
|
|
}
|
2013-06-25 00:46:51 +00:00
|
|
|
}
|
|
|
|
filterCacheValid = TRUE;
|
2000-11-28 20:56:52 +00:00
|
|
|
}
|
2013-06-25 00:46:51 +00:00
|
|
|
|
|
|
|
return filterCache.result;
|
2000-11-28 20:56:52 +00:00
|
|
|
}
|
|
|
|
|
2007-10-09 19:40:14 +00:00
|
|
|
le_bool GlyphIterator::hasFeatureTag(le_bool matchGroup) const
|
2000-11-28 20:56:52 +00:00
|
|
|
{
|
2005-08-06 00:48:27 +00:00
|
|
|
if (featureMask == 0) {
|
2003-12-08 22:43:41 +00:00
|
|
|
return TRUE;
|
2000-11-28 20:56:52 +00:00
|
|
|
}
|
|
|
|
|
2004-05-07 23:29:16 +00:00
|
|
|
LEErrorCode success = LE_NO_ERROR;
|
2005-08-06 00:48:27 +00:00
|
|
|
FeatureMask fm = glyphStorage.getAuxData(position, success);
|
2000-11-28 20:56:52 +00:00
|
|
|
|
2008-03-07 19:40:46 +00:00
|
|
|
return ((fm & featureMask) == featureMask) && (!matchGroup || (le_int32)(fm & LE_GLYPH_GROUP_MASK) == glyphGroup);
|
2000-11-28 20:56:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
le_bool GlyphIterator::findFeatureTag()
|
|
|
|
{
|
2007-10-09 19:40:14 +00:00
|
|
|
//glyphGroup = 0;
|
|
|
|
|
2001-01-19 00:30:17 +00:00
|
|
|
while (nextInternal()) {
|
2007-10-09 19:40:14 +00:00
|
|
|
if (hasFeatureTag(FALSE)) {
|
|
|
|
LEErrorCode success = LE_NO_ERROR;
|
|
|
|
|
|
|
|
glyphGroup = (glyphStorage.getAuxData(position, success) & LE_GLYPH_GROUP_MASK);
|
2003-12-08 22:43:41 +00:00
|
|
|
return TRUE;
|
2000-11-28 20:56:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-12-08 22:43:41 +00:00
|
|
|
return FALSE;
|
2000-11-28 20:56:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
le_bool GlyphIterator::nextInternal(le_uint32 delta)
|
|
|
|
{
|
|
|
|
le_int32 newPosition = position;
|
|
|
|
|
2001-01-19 00:30:17 +00:00
|
|
|
while (newPosition != nextLimit && delta > 0) {
|
|
|
|
do {
|
2000-11-28 20:56:52 +00:00
|
|
|
newPosition += direction;
|
2013-04-18 21:24:51 +00:00
|
|
|
//fprintf(stderr,"%s:%d:%s: newPosition = %d, delta = %d\n", __FILE__, __LINE__, __FUNCTION__, newPosition, delta);
|
2001-01-19 00:30:17 +00:00
|
|
|
} while (newPosition != nextLimit && filterGlyph(newPosition));
|
2000-11-28 20:56:52 +00:00
|
|
|
|
|
|
|
delta -= 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
position = newPosition;
|
|
|
|
|
2013-04-18 21:24:51 +00:00
|
|
|
//fprintf(stderr,"%s:%d:%s: exit position = %d, delta = %d\n", __FILE__, __LINE__, __FUNCTION__, position, delta);
|
2000-11-28 20:56:52 +00:00
|
|
|
return position != nextLimit;
|
|
|
|
}
|
|
|
|
|
|
|
|
le_bool GlyphIterator::next(le_uint32 delta)
|
|
|
|
{
|
2007-10-09 19:40:14 +00:00
|
|
|
return nextInternal(delta) && hasFeatureTag(TRUE);
|
2000-11-28 20:56:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
le_bool GlyphIterator::prevInternal(le_uint32 delta)
|
|
|
|
{
|
|
|
|
le_int32 newPosition = position;
|
|
|
|
|
2001-01-19 00:30:17 +00:00
|
|
|
while (newPosition != prevLimit && delta > 0) {
|
|
|
|
do {
|
2000-11-28 20:56:52 +00:00
|
|
|
newPosition -= direction;
|
2013-04-18 21:24:51 +00:00
|
|
|
//fprintf(stderr,"%s:%d:%s: newPosition = %d, delta = %d\n", __FILE__, __LINE__, __FUNCTION__, newPosition, delta);
|
2001-01-19 00:30:17 +00:00
|
|
|
} while (newPosition != prevLimit && filterGlyph(newPosition));
|
2000-11-28 20:56:52 +00:00
|
|
|
|
|
|
|
delta -= 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
position = newPosition;
|
|
|
|
|
2013-04-18 21:24:51 +00:00
|
|
|
//fprintf(stderr,"%s:%d:%s: exit position = %d, delta = %d\n", __FILE__, __LINE__, __FUNCTION__, position, delta);
|
2000-11-28 20:56:52 +00:00
|
|
|
return position != prevLimit;
|
|
|
|
}
|
|
|
|
|
|
|
|
le_bool GlyphIterator::prev(le_uint32 delta)
|
|
|
|
{
|
2007-10-09 19:40:14 +00:00
|
|
|
return prevInternal(delta) && hasFeatureTag(TRUE);
|
2000-11-28 20:56:52 +00:00
|
|
|
}
|
|
|
|
|
2001-01-19 00:30:17 +00:00
|
|
|
le_int32 GlyphIterator::getMarkComponent(le_int32 markPosition) const
|
2000-11-28 20:56:52 +00:00
|
|
|
{
|
|
|
|
le_int32 component = 0;
|
2002-06-22 00:34:39 +00:00
|
|
|
le_int32 posn;
|
|
|
|
|
|
|
|
for (posn = position; posn != markPosition; posn += direction) {
|
2004-04-12 18:51:31 +00:00
|
|
|
if (glyphStorage[posn] == 0xFFFE) {
|
2002-06-22 00:34:39 +00:00
|
|
|
component += 1;
|
|
|
|
}
|
|
|
|
}
|
2000-11-28 20:56:52 +00:00
|
|
|
|
|
|
|
return component;
|
|
|
|
}
|
|
|
|
|
2002-06-22 00:34:39 +00:00
|
|
|
// This is basically prevInternal except that it
|
|
|
|
// doesn't take a delta argument, and it doesn't
|
|
|
|
// filter out 0xFFFE glyphs.
|
|
|
|
le_bool GlyphIterator::findMark2Glyph()
|
|
|
|
{
|
|
|
|
le_int32 newPosition = position;
|
|
|
|
|
|
|
|
do {
|
|
|
|
newPosition -= direction;
|
2004-04-12 18:51:31 +00:00
|
|
|
} while (newPosition != prevLimit && glyphStorage[newPosition] != 0xFFFE && filterGlyph(newPosition));
|
2002-06-22 00:34:39 +00:00
|
|
|
|
|
|
|
position = newPosition;
|
|
|
|
|
|
|
|
return position != prevLimit;
|
|
|
|
}
|
|
|
|
|
2001-10-16 00:39:01 +00:00
|
|
|
U_NAMESPACE_END
|