提交 6917a045 编写于 作者: B Behdad Esfahbod

[coretext] Unbreak glyph positioning in presence of notdef runs

As discovered on Chrome Mac:
https://code.google.com/p/chromium/issues/detail?id=452326

This was originally broken in:

	commit 5a0eed3b
	Author: Behdad Esfahbod <behdad@behdad.org>
	Date:   Mon Aug 11 23:47:16 2014 -0400

	    [coretext] Implement vertical shaping
上级 1eff4350
...@@ -787,6 +787,7 @@ retry: ...@@ -787,6 +787,7 @@ retry:
buffer->len = 0; buffer->len = 0;
uint32_t status_and = ~0, status_or = 0; uint32_t status_and = ~0, status_or = 0;
double advances_so_far = 0;
const CFRange range_all = CFRangeMake (0, 0); const CFRange range_all = CFRangeMake (0, 0);
...@@ -797,6 +798,10 @@ retry: ...@@ -797,6 +798,10 @@ retry:
status_or |= run_status; status_or |= run_status;
status_and &= run_status; status_and &= run_status;
DEBUG_MSG (CORETEXT, run, "CTRunStatus: %x", run_status); DEBUG_MSG (CORETEXT, run, "CTRunStatus: %x", run_status);
double run_advance = CTRunGetTypographicBounds (run, range_all, NULL, NULL, NULL);
if (HB_DIRECTION_IS_VERTICAL (buffer->props.direction))
run_advance = -run_advance;
DEBUG_MSG (CORETEXT, run, "Run advance: %g", run_advance);
/* CoreText does automatic font fallback (AKA "cascading") for characters /* CoreText does automatic font fallback (AKA "cascading") for characters
* not supported by the requested font, and provides no way to turn it off, * not supported by the requested font, and provides no way to turn it off,
...@@ -905,6 +910,7 @@ retry: ...@@ -905,6 +910,7 @@ retry:
} }
if (HB_DIRECTION_IS_BACKWARD (buffer->props.direction)) if (HB_DIRECTION_IS_BACKWARD (buffer->props.direction))
buffer->reverse_range (old_len, buffer->len); buffer->reverse_range (old_len, buffer->len);
advances_so_far += run_advance;
continue; continue;
} }
} }
...@@ -934,7 +940,7 @@ retry: ...@@ -934,7 +940,7 @@ retry:
scratch_size = scratch_size_saved; \ scratch_size = scratch_size_saved; \
scratch = scratch_saved; scratch = scratch_saved;
{ { /* Setup glyphs */
SCRATCH_SAVE(); SCRATCH_SAVE();
const CGGlyph* glyphs = USE_PTR ? CTRunGetGlyphsPtr (run) : NULL; const CGGlyph* glyphs = USE_PTR ? CTRunGetGlyphsPtr (run) : NULL;
if (!glyphs) { if (!glyphs) {
...@@ -958,6 +964,11 @@ retry: ...@@ -958,6 +964,11 @@ retry:
SCRATCH_RESTORE(); SCRATCH_RESTORE();
} }
{ {
/* Setup positions.
* Note that CoreText does not return advances for glyphs. As such,
* for all but last glyph, we use the delta position to next glyph as
* advance (in the advance direction only), and for last glyph we set
* whatever is needed to make the whole run's advance add up. */
SCRATCH_SAVE(); SCRATCH_SAVE();
const CGPoint* positions = USE_PTR ? CTRunGetPositionsPtr (run) : NULL; const CGPoint* positions = USE_PTR ? CTRunGetPositionsPtr (run) : NULL;
if (!positions) { if (!positions) {
...@@ -965,34 +976,42 @@ retry: ...@@ -965,34 +976,42 @@ retry:
CTRunGetPositions (run, range_all, position_buf); CTRunGetPositions (run, range_all, position_buf);
positions = position_buf; positions = position_buf;
} }
double run_advance = CTRunGetTypographicBounds (run, range_all, NULL, NULL, NULL);
DEBUG_MSG (CORETEXT, run, "Run advance: %g", run_advance);
hb_glyph_info_t *info = run_info; hb_glyph_info_t *info = run_info;
CGFloat x_mult = font_data->x_mult, y_mult = font_data->y_mult; CGFloat x_mult = font_data->x_mult, y_mult = font_data->y_mult;
if (HB_DIRECTION_IS_HORIZONTAL (buffer->props.direction)) if (HB_DIRECTION_IS_HORIZONTAL (buffer->props.direction))
{ {
hb_position_t x_offset = (positions[0].x - advances_so_far) * x_mult;
for (unsigned int j = 0; j < num_glyphs; j++) for (unsigned int j = 0; j < num_glyphs; j++)
{ {
double advance = (j + 1 < num_glyphs ? positions[j + 1].x : positions[0].x + run_advance) - positions[j].x; double advance;
if (likely (j + 1 < num_glyphs))
advance = positions[j + 1].x - positions[j].x;
else /* last glyph */
advance = run_advance - (positions[j].x - positions[0].x);
info->mask = advance * x_mult; info->mask = advance * x_mult;
info->var1.u32 = positions[0].x * x_mult; /* Yes, zero. */ info->var1.u32 = x_offset;
info->var2.u32 = positions[j].y * y_mult; info->var2.u32 = positions[j].y * y_mult;
info++; info++;
} }
} }
else else
{ {
run_advance = -run_advance; hb_position_t y_offset = (positions[0].y - advances_so_far) * y_mult;
for (unsigned int j = 0; j < num_glyphs; j++) for (unsigned int j = 0; j < num_glyphs; j++)
{ {
double advance = (j + 1 < num_glyphs ? positions[j + 1].y : positions[0].y + run_advance) - positions[j].y; double advance;
if (likely (j + 1 < num_glyphs))
advance = positions[j + 1].y - positions[j].y;
else /* last glyph */
advance = run_advance - (positions[j].y - positions[0].y);
info->mask = advance * y_mult; info->mask = advance * y_mult;
info->var1.u32 = positions[j].x * x_mult; info->var1.u32 = positions[j].x * x_mult;
info->var2.u32 = positions[0].y * y_mult; /* Yes, zero. */ info->var2.u32 = y_offset;
info++; info++;
} }
} }
SCRATCH_RESTORE(); SCRATCH_RESTORE();
advances_so_far += run_advance;
} }
#undef SCRATCH_RESTORE #undef SCRATCH_RESTORE
#undef SCRATCH_SAVE #undef SCRATCH_SAVE
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册