/* Latin.java -- Latin specific glyph handling
Copyright (C) 2006 Free Software Foundation, Inc.
This file is part of GNU Classpath.
GNU Classpath is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2, or (at your option)
any later version.
GNU Classpath 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
General Public License for more details.
You should have received a copy of the GNU General Public License
along with GNU Classpath; see the file COPYING. If not, write to the
Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
02110-1301 USA.
Linking this library statically or dynamically with other modules is
making a combined work based on this library. Thus, the terms and
conditions of the GNU General Public License cover the whole
combination.
As a special exception, the copyright holders of this library give you
permission to link this library with independent modules to produce an
executable, regardless of the license terms of these independent
modules, and to copy and distribute the resulting executable under
terms of your choice, provided that you also meet, for each linked
independent module, the terms and conditions of the license of that
module. An independent module is a module which is not derived from
or based on this library. If you modify this library, you may extend
this exception to your version of the library, but you are not
obligated to do so. If you do not wish to do so, delete this
exception statement from your version. */
package gnu.java.awt.font.autofit;
import java.awt.geom.AffineTransform;
import java.util.HashSet;
import gnu.java.awt.font.opentype.OpenTypeFont;
import gnu.java.awt.font.opentype.truetype.Point;
import gnu.java.awt.font.opentype.truetype.Zone;
/**
* Implements Latin specific glyph handling.
*/
class Latin
implements Script, Constants
{
static final int MAX_WIDTHS = 16;
private final static int MAX_TEST_CHARS = 12;
/**
* The types of the 6 blue zones.
*/
private static final int CAPITAL_TOP = 0;
private static final int CAPITAL_BOTTOM = 1;
private static final int SMALL_F_TOP = 2;
private static final int SMALL_TOP = 3;
private static final int SMALL_BOTTOM = 4;
private static final int SMALL_MINOR = 5;
static final int BLUE_MAX = 6;
/**
* The test chars for the blue zones.
*
* @see #initBlues(LatinMetrics, OpenTypeFont)
*/
private static final String[] TEST_CHARS =
new String[]{"THEZOCQS", "HEZLOCUS", "fijkdbh",
"xzroesc", "xzroesc", "pqgjy"};
public void applyHints(GlyphHints hints, ScriptMetrics metrics)
{
// TODO Auto-generated method stub
}
public void doneMetrics(ScriptMetrics metrics)
{
// TODO Auto-generated method stub
}
/**
* Initializes the hints
object.
*
* @param hints the hints to initialize
* @param metrics the metrics to use
*/
public void initHints(GlyphHints hints, ScriptMetrics metrics)
{
hints.rescale(metrics);
LatinMetrics lm = (LatinMetrics) metrics;
hints.xScale = lm.axis[DIMENSION_HORZ].scale;
hints.xDelta = lm.axis[DIMENSION_HORZ].delta;
hints.yScale = lm.axis[DIMENSION_VERT].scale;
hints.yDelta = lm.axis[DIMENSION_VERT].delta;
// TODO: Set the scaler and other flags.
}
/**
* Initializes the script metrics.
*
* @param metrics the script metrics to initialize
* @param face the font
*/
public void initMetrics(ScriptMetrics metrics, OpenTypeFont face)
{
assert metrics instanceof LatinMetrics;
LatinMetrics lm = (LatinMetrics) metrics;
lm.unitsPerEm = face.unitsPerEm;
// TODO: Check for latin charmap.
initWidths(lm, face, 'o');
initBlues(lm, face);
}
public void scaleMetrics(ScriptMetrics metrics)
{
// TODO Auto-generated method stub
}
/**
* Determines the standard stem widths.
*
* @param metrics the metrics to use
* @param face the font face
* @param ch the character that is used for getting the widths
*/
private void initWidths(LatinMetrics metrics, OpenTypeFont face, char ch)
{
GlyphHints hints = new GlyphHints();
metrics.axis[DIMENSION_HORZ].widthCount = 0;
metrics.axis[DIMENSION_VERT].widthCount = 0;
int glyphIndex = face.getGlyph(ch);
// TODO: Avoid that AffineTransform constructor and change
// getRawGlyphOutline() to accept null or remove that parameter altogether.
// Consider this when the thing is done and we know what we need that for.
Zone outline = face.getRawGlyphOutline(glyphIndex, IDENTITY);
LatinMetrics dummy = new LatinMetrics();
Scaler scaler = dummy.scaler;
dummy.unitsPerEm = metrics.unitsPerEm;
scaler.xScale = scaler.yScale = 10000;
scaler.xDelta = scaler.yDelta = 0;
scaler.face = face;
hints.rescale(dummy);
hints.reload(outline);
for (int dim = 0; dim < DIMENSION_MAX; dim++)
{
LatinAxis axis = metrics.axis[dim];
AxisHints axHints = hints.axis[dim];
int numWidths = 0;
computeSegments(hints, dim);
linkSegments(hints, dim);
Segment[] segs = axHints.segments;
HashSet touched = new HashSet();
for (int i = 0; i < segs.length; i++)
{
Segment seg = segs[i];
Segment link = seg.link;
if (link != null && link.link == seg && ! touched.contains(link))
{
int dist = Math.abs(seg.pos - link.pos);
if (numWidths < MAX_WIDTHS)
axis.widths[numWidths++] = new Width(dist);
}
touched.add(seg);
}
}
for (int dim = 0; dim < DIMENSION_MAX; dim++)
{
LatinAxis axis = metrics.axis[dim];
int stdw = axis.widthCount > 0 ? axis.widths[0].org
: constant(metrics, 50);
axis.edgeDistanceTreshold= stdw / 5;
}
}
void linkSegments(GlyphHints hints, int dim)
{
AxisHints axis = hints.axis[dim];
Segment[] segments = axis.segments;
int numSegs = axis.numSegments;
int majorDir = axis.majorDir;
int lenThreshold = constant((LatinMetrics) hints.metrics, 8);
lenThreshold = Math.min(1, lenThreshold);
int lenScore = constant((LatinMetrics) hints.metrics, 3000);
for (int i1 = 0; i1 < numSegs; i1++)
{
Segment seg1 = segments[i1];
// The fake segments are introduced to hint the metrics.
// Never link them to anything.
if (seg1.first == seg1.last || seg1.dir != majorDir)
continue;
for (int i2 = 0; i2 < numSegs; i2++)
{
Segment seg2 = segments[i2];
if (seg2 != seg1 && seg1.dir + seg2.dir == 0)
{
int pos1 = seg1.pos;
int pos2 = seg2.pos;
// The vertical coords are swapped compared to how FT handles
// this.
int dist = dim == DIMENSION_VERT ? pos1 - pos2 : pos2 - pos1;
if (dist >= 0)
{
int min = seg1.minPos;
int max = seg1.maxPos;
int len, score;
if (min < seg2.minPos)
min = seg2.minPos;
if (max > seg2.maxPos)
max = seg2.maxPos;
len = max - min;
if (len > lenThreshold)
{
score = dist + lenScore / len;
if (score < seg1.score)
{
seg1.score = score;
seg1.link = seg2;
}
if (score < seg2.score)
{
seg2.score = score;
seg2.link = seg1;
}
}
}
}
}
}
for (int i1 = 0; i1 < numSegs; i1++)
{
Segment seg1 = segments[i1];
Segment seg2 = seg1.link;
if (seg2 != null)
{
seg2.numLinked++;
if (seg2.link != seg1)
{
seg1.link = null;
seg1.serif = seg2.link;
}
}
// Uncomment to show all segments.
// System.err.println("segment#" + i1 + ": " + seg1);
}
}
/**
* Initializes the blue zones of the font.
*
* @param metrics the metrics to use
* @param face the font face to analyze
*/
private void initBlues(LatinMetrics metrics, OpenTypeFont face)
{
int[] flats = new int[MAX_TEST_CHARS];
int[] rounds = new int[MAX_TEST_CHARS];
int numFlats;
int numRounds;
LatinBlue blue;
LatinAxis axis = metrics.axis[DIMENSION_VERT];
// We compute the blues simply by loading each character in the test
// strings, then compute its topmost or bottommost points.
for (int bb = 0; bb < BLUE_MAX; bb++)
{
String p = TEST_CHARS[bb];
int blueRef;
int blueShoot;
numFlats = 0;
numRounds = 0;
for (int i = 0; i < p.length(); i++)
{
// Load the character.
int glyphIndex = face.getGlyph(p.charAt(i));
Zone glyph =
face.getRawGlyphOutline(glyphIndex, IDENTITY);
// Now compute the min and max points.
int numPoints = glyph.getSize() - 4; // 4 phantom points.
Point[] points = glyph.getPoints();
Point point = points[0];
int extremum = 0;
int index = 1;
if (isTopBlue(bb))
{
for (; index < numPoints; index++)
{
point = points[index];
// We have the vertical direction swapped. The higher
// points have smaller (negative) Y.
if (point.getOrigY() < points[extremum].getOrigY())
extremum = index;
}
}
else
{
for (; index < numPoints; index++)
{
point = points[index];
// We have the vertical direction swapped. The higher
// points have smaller (negative) Y.
if (point.getOrigY() > points[extremum].getOrigY())
extremum = index;
}
}
// Debug, prints out the maxima.
// System.err.println("extremum for " + bb + " / "+ p.charAt(i)
// + ": " + points[extremum]);
// Now determine if the point is part of a straight or round
// segment.
boolean round;
int idx = extremum;
int first, last, prev, next, end;
int dist;
last = -1;
first = 0;
for (int n = 0; n < glyph.getNumContours(); n++)
{
end = glyph.getContourEnd(n);
// System.err.println("contour end for " + n + ": " + end);
if (end >= idx)
{
last = end;
break;
}
first = end + 1;
}
// Should never happen.
assert last >= 0;
// Now look for the previous and next points that are not on the
// same Y coordinate. Threshold the 'closeness'.
prev = idx;
next = prev;
do
{
if (prev > first)
prev--;
else
prev = last;
dist = points[prev].getOrigY() - points[extremum].getOrigY();
if (dist < -5 || dist > 5)
break;
} while (prev != idx);
do
{
if (next < last)
next++;
else
next = first;
dist = points[next].getOrigY() - points[extremum].getOrigY();
if (dist < -5 || dist > 5)
break;
} while (next != idx);
round = points[prev].isControlPoint()
|| points[next].isControlPoint();
if (round)
{
rounds[numRounds++] = points[extremum].getOrigY();
// System.err.println("new round extremum: " + bb + ": "
// + points[extremum].getOrigY());
}
else
{
flats[numFlats++] = points[extremum].getOrigY();
// System.err.println("new flat extremum: " + bb + ": "
// + points[extremum].getOrigY());
}
}
// We have computed the contents of the rounds and flats tables.
// Now determine the reference and overshoot position of the blues --
// we simply take the median after a simple sort.
Utils.sort(numRounds, rounds);
Utils.sort(numFlats, flats);
blue = axis.blues[axis.blueCount] = new LatinBlue();
axis.blueCount++;
if (numFlats == 0)
{
blue.ref = blue.shoot = new Width(rounds[numRounds / 2]);
}
else if (numRounds == 0)
{
blue.ref = blue.shoot = new Width(flats[numFlats / 2]);
}
else
{
blue.ref = new Width(flats[numFlats / 2]);
blue.shoot = new Width(rounds[numRounds / 2]);
}
// There are sometimes problems: if the overshoot position of top
// zones is under its reference position, or the opposite for bottom
// zones. We must check everything there and correct problems.
if (blue.shoot != blue.ref)
{
int ref = blue.ref.org;
int shoot = blue.shoot.org;
// Inversed vertical coordinates!
boolean overRef = shoot < ref;
if (isTopBlue(bb) ^ overRef)
{
blue.shoot = blue.ref = new Width((shoot + ref) / 2);
}
}
blue.flags = 0;
if (isTopBlue(bb))
blue.flags |= LatinBlue.FLAG_TOP;
// The following flag is used later to adjust y and x scales in
// order to optimize the pixel grid alignment of the top small
// letters.
if (bb == SMALL_TOP)
{
blue.flags |= LatinBlue.FLAG_ADJUSTMENT;
}
// Debug: print out the blue zones.
System.err.println("blue zone #" + bb + ": " + blue);
}
}
private static final AffineTransform IDENTITY = new AffineTransform();
private int constant(LatinMetrics metrics, int c)
{
return c * (metrics.unitsPerEm / 2048);
}
private void computeSegments(GlyphHints hints, int dim)
{
Point[] points = hints.points;
if (dim == DIMENSION_HORZ)
{
for (int i = 0; i < hints.numPoints; i++)
{
points[i].setU(points[i].getOrigX());
points[i].setV(points[i].getOrigY());
}
}
else
{
for (int i = 0; i < hints.numPoints; i++)
{
points[i].setU(points[i].getOrigY());
points[i].setV(points[i].getOrigX());
}
}
// Now look at each contour.
AxisHints axis = hints.axis[dim];
int majorDir = Math.abs(axis.majorDir);
int segmentDir = majorDir;
Point[] contours = hints.contours;
int numContours = hints.numContours;
Segment segment = null;
for (int i = 0; i < numContours; i++)
{
int minPos = 32000;
int maxPos = -32000;
Point point = contours[i];
Point last = point.getPrev();
if (point == last) // Skip singletons.
continue;
if (Math.abs(last.getOutDir()) == majorDir
&& Math.abs(point.getOutDir()) == majorDir)
{
// We are already on an edge. Locate its start.
last = point;
while (true)
{
point = point.getPrev();
if (Math.abs(point.getOutDir()) != majorDir)
{
point = point.getNext();
break;
}
if (point == last)
break;
}
}
last = point;
boolean passed = false;
boolean onEdge = false;
while (true)
{
int u, v;
if (onEdge)
{
u = point.getU();
if (u < minPos)
minPos = u;
if (u > maxPos)
maxPos = u;
if (point.getOutDir() != segmentDir || point == last)
{
// Leaving an edge. Record new segment.
segment.last = point;
// (minPos + maxPos) / 2.
segment.pos = (minPos + maxPos) >> 1;
if (segment.first.isControlPoint()
|| point.isControlPoint())
segment.flags |= Segment.FLAG_EDGE_ROUND;
minPos = maxPos = point.getV();
v = segment.first.getV();
if (v < minPos)
minPos = v;
if (v > maxPos)
maxPos = v;
segment.minPos = minPos;
segment.maxPos = maxPos;
onEdge = false;
segment = null;
}
}
if (point == last)
{
if (passed)
break;
passed = true;
}
if (! onEdge && Math.abs(point.getOutDir()) == majorDir)
{
// This is the start of a new segment.
segmentDir = point.getOutDir();
segment = axis.newSegment();
segment.dir = segmentDir;
segment.flags = Segment.FLAG_EDGE_NORMAL;
minPos = maxPos = point.getU();
segment.first = point;
segment.last = point;
segment.contour = contours[i];
segment.score = 32000;
segment.len = 0;
segment.link = null;
onEdge = true;
}
point = point.getNext();
}
}
}
private boolean isTopBlue(int b)
{
return b == CAPITAL_TOP || b == SMALL_F_TOP || b == SMALL_TOP;
}
}