You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
208 lines
5.4 KiB
208 lines
5.4 KiB
/*
|
|
*******************************************************************************
|
|
*
|
|
* © 2016 and later: Unicode, Inc. and others.
|
|
* License & terms of use: http://www.unicode.org/copyright.html
|
|
*
|
|
*******************************************************************************
|
|
*******************************************************************************
|
|
*
|
|
* Copyright (C) 1999-2007, International Business Machines
|
|
* Corporation and others. All Rights Reserved.
|
|
*
|
|
*******************************************************************************
|
|
* file name: GnomeFontInstance.cpp
|
|
*
|
|
* created on: 08/30/2001
|
|
* created by: Eric R. Mader
|
|
*/
|
|
|
|
#include <gnome.h>
|
|
#include <ft2build.h>
|
|
#include FT_FREETYPE_H
|
|
#include FT_GLYPH_H
|
|
#include FT_RENDER_H
|
|
#include FT_TRUETYPE_TABLES_H
|
|
#include <cairo.h>
|
|
#include <cairo-ft.h>
|
|
|
|
#include "layout/LETypes.h"
|
|
#include "layout/LESwaps.h"
|
|
|
|
#include "GnomeFontInstance.h"
|
|
#include "sfnt.h"
|
|
#include "cmaps.h"
|
|
|
|
GnomeSurface::GnomeSurface(GtkWidget *theWidget)
|
|
: fWidget(theWidget)
|
|
{
|
|
fCairo = gdk_cairo_create(fWidget->window);
|
|
}
|
|
|
|
GnomeSurface::~GnomeSurface()
|
|
{
|
|
cairo_destroy(fCairo);
|
|
}
|
|
|
|
void GnomeSurface::drawGlyphs(const LEFontInstance *font, const LEGlyphID *glyphs, le_int32 count,
|
|
const float *positions, le_int32 x, le_int32 y, le_int32 /*width*/, le_int32 /*height*/)
|
|
{
|
|
GnomeFontInstance *gFont = (GnomeFontInstance *) font;
|
|
|
|
gFont->rasterizeGlyphs(fCairo, glyphs, count, positions, x, y);
|
|
}
|
|
|
|
GnomeFontInstance::GnomeFontInstance(FT_Library engine, const char *fontPathName, le_int16 pointSize, LEErrorCode &status)
|
|
: FontTableCache(), fPointSize(pointSize), fUnitsPerEM(0), fAscent(0), fDescent(0), fLeading(0),
|
|
fDeviceScaleX(1), fDeviceScaleY(1), fMapper(NULL)
|
|
{
|
|
FT_Error error;
|
|
|
|
fFace = NULL;
|
|
fCairoFace = NULL;
|
|
|
|
error = FT_New_Face(engine, fontPathName, 0, &fFace);
|
|
|
|
if (error != 0) {
|
|
printf("OOPS! Got error code %d\n", error);
|
|
status = LE_FONT_FILE_NOT_FOUND_ERROR;
|
|
return;
|
|
}
|
|
|
|
// FIXME: what about the display resolution?
|
|
fDeviceScaleX = ((float) 96) / 72;
|
|
fDeviceScaleY = ((float) 96) / 72;
|
|
|
|
error = FT_Set_Char_Size(fFace, 0, pointSize << 6, 92, 92);
|
|
|
|
fCairoFace = cairo_ft_font_face_create_for_ft_face(fFace, 0);
|
|
|
|
fUnitsPerEM = fFace->units_per_EM;
|
|
|
|
fAscent = (le_int32) (yUnitsToPoints(fFace->ascender) * fDeviceScaleY);
|
|
fDescent = (le_int32) -(yUnitsToPoints(fFace->descender) * fDeviceScaleY);
|
|
fLeading = (le_int32) (yUnitsToPoints(fFace->height) * fDeviceScaleY) - fAscent - fDescent;
|
|
|
|
// printf("Face = %s, unitsPerEM = %d, ascent = %d, descent = %d\n", fontPathName, fUnitsPerEM, fAscent, fDescent);
|
|
|
|
if (error != 0) {
|
|
status = LE_MEMORY_ALLOCATION_ERROR;
|
|
return;
|
|
}
|
|
|
|
status = initMapper();
|
|
}
|
|
|
|
GnomeFontInstance::~GnomeFontInstance()
|
|
{
|
|
cairo_font_face_destroy(fCairoFace);
|
|
|
|
if (fFace != NULL) {
|
|
FT_Done_Face(fFace);
|
|
}
|
|
}
|
|
|
|
LEErrorCode GnomeFontInstance::initMapper()
|
|
{
|
|
LETag cmapTag = LE_CMAP_TABLE_TAG;
|
|
const CMAPTable *cmap = (const CMAPTable *) readFontTable(cmapTag);
|
|
|
|
if (cmap == NULL) {
|
|
return LE_MISSING_FONT_TABLE_ERROR;
|
|
}
|
|
|
|
fMapper = CMAPMapper::createUnicodeMapper(cmap);
|
|
|
|
if (fMapper == NULL) {
|
|
return LE_MISSING_FONT_TABLE_ERROR;
|
|
}
|
|
|
|
return LE_NO_ERROR;
|
|
}
|
|
|
|
const void *GnomeFontInstance::getFontTable(LETag tableTag) const
|
|
{
|
|
return FontTableCache::find(tableTag);
|
|
}
|
|
|
|
const void *GnomeFontInstance::readFontTable(LETag tableTag) const
|
|
{
|
|
FT_ULong len = 0;
|
|
FT_Byte *result = NULL;
|
|
|
|
FT_Load_Sfnt_Table(fFace, tableTag, 0, NULL, &len);
|
|
|
|
if (len > 0) {
|
|
result = LE_NEW_ARRAY(FT_Byte, len);
|
|
FT_Load_Sfnt_Table(fFace, tableTag, 0, result, &len);
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
void GnomeFontInstance::getGlyphAdvance(LEGlyphID glyph, LEPoint &advance) const
|
|
{
|
|
advance.fX = 0;
|
|
advance.fY = 0;
|
|
|
|
if (glyph >= 0xFFFE) {
|
|
return;
|
|
}
|
|
|
|
FT_Error error;
|
|
|
|
error = FT_Load_Glyph(fFace, glyph, FT_LOAD_DEFAULT);
|
|
|
|
if (error != 0) {
|
|
return;
|
|
}
|
|
|
|
advance.fX = fFace->glyph->metrics.horiAdvance >> 6;
|
|
return;
|
|
}
|
|
|
|
le_bool GnomeFontInstance::getGlyphPoint(LEGlyphID glyph, le_int32 pointNumber, LEPoint &point) const
|
|
{
|
|
FT_Error error;
|
|
|
|
error = FT_Load_Glyph(fFace, glyph, FT_LOAD_DEFAULT);
|
|
|
|
if (error != 0) {
|
|
return FALSE;
|
|
}
|
|
|
|
if (pointNumber >= fFace->glyph->outline.n_points) {
|
|
return FALSE;
|
|
}
|
|
|
|
point.fX = fFace->glyph->outline.points[pointNumber].x >> 6;
|
|
point.fY = fFace->glyph->outline.points[pointNumber].y >> 6;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
void GnomeFontInstance::rasterizeGlyphs(cairo_t *cairo, const LEGlyphID *glyphs, le_int32 glyphCount, const float *positions,
|
|
le_int32 x, le_int32 y) const
|
|
{
|
|
cairo_glyph_t *glyph_t = LE_NEW_ARRAY(cairo_glyph_t, glyphCount);
|
|
le_int32 in, out;
|
|
|
|
for (in = 0, out = 0; in < glyphCount; in += 1) {
|
|
TTGlyphID glyph = LE_GET_GLYPH(glyphs[in]);
|
|
|
|
if (glyph < 0xFFFE) {
|
|
glyph_t[out].index = glyph;
|
|
glyph_t[out].x = x + positions[in*2];
|
|
glyph_t[out].y = y + positions[in*2 + 1];
|
|
|
|
out += 1;
|
|
}
|
|
}
|
|
|
|
cairo_set_font_face(cairo, fCairoFace);
|
|
cairo_set_font_size(cairo, getXPixelsPerEm() * getScaleFactorX());
|
|
cairo_show_glyphs(cairo, glyph_t, out);
|
|
|
|
LE_DELETE_ARRAY(glyph_t);
|
|
}
|