C++程序  |  150行  |  3.86 KB

#include <stdint.h>

#include <ft2build.h>
#include FT_FREETYPE_H
#include FT_TRUETYPE_TABLES_H

#if 0
#include <freetype/freetype.h>
#include <freetype/tttables.h>
#endif

#include <harfbuzz-shaper.h>
#include "harfbuzz-unicode.h"

static HB_Bool
hb_freetype_string_to_glyphs(HB_Font font,
                             const HB_UChar16 *chars, hb_uint32 len,
                             HB_Glyph *glyphs, hb_uint32 *numGlyphs,
                             HB_Bool is_rtl) {
  FT_Face face = (FT_Face) font->userData;
  if (len > *numGlyphs)
    return 0;

  size_t i = 0, j = 0;
  while (i < len) {
    const uint32_t cp = utf16_to_code_point(chars, len, &i);
    glyphs[j++] = FT_Get_Char_Index(face, cp);
  }

  *numGlyphs = j;

  return 1;
}

static void
hb_freetype_advances_get(HB_Font font, const HB_Glyph *glyphs, hb_uint32 len,
                         HB_Fixed *advances, int flags) {
  FT_Face face = (FT_Face) font->userData;

  hb_uint32 i;
  for (i = 0; i < len; ++i) {
    const FT_Error error = FT_Load_Glyph(face, glyphs[i], FT_LOAD_DEFAULT);
    if (error) {
      advances[i] = 0;
      continue;
    }

    advances[i] = face->glyph->advance.x;
  }
}

static HB_Bool
hb_freetype_can_render(HB_Font font, const HB_UChar16 *chars, hb_uint32 len) {
  FT_Face face = (FT_Face)font->userData;

  size_t i = 0;
  while (i < len) {
    const uint32_t cp = utf16_to_code_point(chars, len, &i);
    if (FT_Get_Char_Index(face, cp) == 0)
      return 0;
  }

  return 1;
}

static HB_Error
hb_freetype_outline_point_get(HB_Font font, HB_Glyph glyph, int flags,
                              hb_uint32 point, HB_Fixed *xpos, HB_Fixed *ypos,
                              hb_uint32 *n_points) {
  HB_Error error = HB_Err_Ok;
  FT_Face face = (FT_Face) font->userData;

  int load_flags = (flags & HB_ShaperFlag_UseDesignMetrics) ? FT_LOAD_NO_HINTING : FT_LOAD_DEFAULT;

  if ((error = (HB_Error) FT_Load_Glyph(face, glyph, load_flags)))
    return error;

  if (face->glyph->format != ft_glyph_format_outline)
    return (HB_Error)HB_Err_Invalid_SubTable;

  *n_points = face->glyph->outline.n_points;
  if (!(*n_points))
    return HB_Err_Ok;

  if (point > *n_points)
    return (HB_Error)HB_Err_Invalid_SubTable;

  *xpos = face->glyph->outline.points[point].x;
  *ypos = face->glyph->outline.points[point].y;

  return HB_Err_Ok;
}

static void
hb_freetype_glyph_metrics_get(HB_Font font, HB_Glyph glyph,
                              HB_GlyphMetrics *metrics) {
  FT_Face face = (FT_Face) font->userData;

  const FT_Error error = FT_Load_Glyph(face, glyph, FT_LOAD_DEFAULT);
  if (error) {
    metrics->x = metrics->y = metrics->width = metrics->height = 0;
    metrics->xOffset = metrics->yOffset = 0;
    return;
  }

  const FT_Glyph_Metrics *ftmetrics = &face->glyph->metrics;
  metrics->width = ftmetrics->width;
  metrics->height = ftmetrics->height;
  metrics->x = ftmetrics->horiAdvance;
  metrics->y = 0;  // unclear what this is
  metrics->xOffset = ftmetrics->horiBearingX;
  metrics->yOffset = ftmetrics->horiBearingY;
}

static HB_Fixed
hb_freetype_font_metric_get(HB_Font font, HB_FontMetric metric) {
  FT_Face face = (FT_Face) font->userData;

  switch (metric) {
  case HB_FontAscent:
    // Note that we aren't scanning the VDMX table which we probably would in
    // an ideal world.
    return face->ascender;
  default:
    return 0;
  }
}

const HB_FontClass hb_freetype_class = {
  hb_freetype_string_to_glyphs,
  hb_freetype_advances_get,
  hb_freetype_can_render,
  hb_freetype_outline_point_get,
  hb_freetype_glyph_metrics_get,
  hb_freetype_font_metric_get,
};

HB_Error
hb_freetype_table_sfnt_get(void *voidface, const HB_Tag tag, HB_Byte *buffer, HB_UInt *len) {
  FT_Face face = (FT_Face) voidface;
  FT_ULong ftlen = *len;

  if (!FT_IS_SFNT(face))
    return HB_Err_Invalid_Argument;

  const FT_Error error = FT_Load_Sfnt_Table(face, tag, 0, buffer, &ftlen);
  *len = ftlen;
  return (HB_Error) error;
}