284 lines
9.8 KiB
C
284 lines
9.8 KiB
C
/*
|
|
* freetype.c
|
|
* Copyright (C) 2016 Kovid Goyal <kovid at kovidgoyal.net>
|
|
*
|
|
* Distributed under terms of the GPL3 license.
|
|
*/
|
|
|
|
#include "data-types.h"
|
|
#include <structmember.h>
|
|
#include <ft2build.h>
|
|
#include FT_FREETYPE_H
|
|
typedef struct {
|
|
PyObject_HEAD
|
|
|
|
FT_Face face;
|
|
unsigned int units_per_EM;
|
|
int ascender, descender, height, max_advance_width, max_advance_height, underline_position, underline_thickness;
|
|
bool is_scalable;
|
|
} Face;
|
|
|
|
|
|
void
|
|
set_freetype_error(const char* prefix, int err_code) {
|
|
int i = 0;
|
|
#undef FTERRORS_H_
|
|
#undef __FTERRORS_H__
|
|
#define FT_ERRORDEF( e, v, s ) { e, s },
|
|
#define FT_ERROR_START_LIST {
|
|
#define FT_ERROR_END_LIST { 0, NULL } };
|
|
|
|
static const struct {
|
|
int err_code;
|
|
const char* err_msg;
|
|
} ft_errors[] =
|
|
|
|
#ifdef FT_ERRORS_H
|
|
#include FT_ERRORS_H
|
|
#else
|
|
FT_ERROR_START_LIST FT_ERROR_END_LIST
|
|
#endif
|
|
|
|
while(ft_errors[i].err_msg != NULL) {
|
|
if (ft_errors[i].err_code == err_code) {
|
|
PyErr_Format(PyExc_ValueError, "%s %s", prefix, ft_errors[i].err_msg);
|
|
return;
|
|
}
|
|
i++;
|
|
}
|
|
PyErr_Format(PyExc_ValueError, "%s (error code: %d)", prefix, err_code);
|
|
}
|
|
|
|
static FT_Library library;
|
|
|
|
|
|
static PyObject*
|
|
new(PyTypeObject *type, PyObject *args, PyObject UNUSED *kwds) {
|
|
Face *self;
|
|
char *path;
|
|
int error;
|
|
long index;
|
|
/* unsigned int columns=80, lines=24, scrollback=0; */
|
|
if (!PyArg_ParseTuple(args, "sl", &path, &index)) return NULL;
|
|
|
|
self = (Face *)type->tp_alloc(type, 0);
|
|
if (self != NULL) {
|
|
error = FT_New_Face(library, path, index, &(self->face));
|
|
if(error) { set_freetype_error("Failed to load face, with error:", error); Py_CLEAR(self); return NULL; }
|
|
#define CPY(n) self->n = self->face->n;
|
|
CPY(units_per_EM); CPY(ascender); CPY(descender); CPY(height); CPY(max_advance_width); CPY(max_advance_height); CPY(underline_position); CPY(underline_thickness);
|
|
#undef CPY
|
|
self->is_scalable = FT_IS_SCALABLE(self->face);
|
|
}
|
|
return (PyObject*)self;
|
|
}
|
|
|
|
static void
|
|
dealloc(Face* self) {
|
|
FT_Done_Face(self->face);
|
|
Py_TYPE(self)->tp_free((PyObject*)self);
|
|
}
|
|
|
|
static PyObject*
|
|
set_char_size(Face *self, PyObject *args) {
|
|
#define set_char_size_doc "set_char_size(width, height, xdpi, ydpi) -> set the character size. width, height is in 1/64th of a pt. dpi is in pixels per inch"
|
|
long char_width, char_height;
|
|
unsigned int xdpi, ydpi;
|
|
int error;
|
|
if (!PyArg_ParseTuple(args, "llII", &char_width, &char_height, &xdpi, &ydpi)) return NULL;
|
|
error = FT_Set_Char_Size(self->face, char_width, char_height, xdpi, ydpi);
|
|
if (error) { set_freetype_error("Failed to set char size, with error:", error); Py_CLEAR(self); return NULL; }
|
|
Py_RETURN_NONE;
|
|
}
|
|
|
|
static PyObject*
|
|
load_char(Face *self, PyObject *args) {
|
|
#define load_char_doc "load_char(char, hinting, hintstyle)"
|
|
int char_code, hinting, hintstyle, error;
|
|
if (!PyArg_ParseTuple(args, "Cpp", &char_code, &hinting, &hintstyle)) return NULL;
|
|
|
|
int glyph_index = FT_Get_Char_Index(self->face, char_code);
|
|
int flags = FT_LOAD_RENDER;
|
|
if (hinting) {
|
|
if (hintstyle >= 3) flags |= FT_LOAD_TARGET_NORMAL;
|
|
else if (0 < hintstyle && hintstyle < 3) flags |= FT_LOAD_TARGET_LIGHT;
|
|
} else flags |= FT_LOAD_NO_HINTING;
|
|
error = FT_Load_Glyph(self->face, glyph_index, flags);
|
|
if (error) { set_freetype_error("Failed to load glyph, with error:", error); Py_CLEAR(self); return NULL; }
|
|
Py_RETURN_NONE;
|
|
}
|
|
|
|
static PyObject*
|
|
get_char_index(Face *self, PyObject *args) {
|
|
#define get_char_index_doc ""
|
|
int code;
|
|
unsigned int ans;
|
|
if (!PyArg_ParseTuple(args, "C", &code)) return NULL;
|
|
ans = FT_Get_Char_Index(self->face, code);
|
|
|
|
return Py_BuildValue("I", ans);
|
|
}
|
|
|
|
static PyStructSequence_Field gm_fields[] = {
|
|
{"width", NULL},
|
|
{"height", NULL},
|
|
{"horiBearingX", NULL},
|
|
{"horiBearingY", NULL},
|
|
{"horiAdvance", NULL},
|
|
{"vertBearingX", NULL},
|
|
{"vertBearingY", NULL},
|
|
{"vertAdvance", NULL},
|
|
{NULL}
|
|
};
|
|
|
|
static PyStructSequence_Desc gm_desc = {"GlpyhMetrics", NULL, gm_fields, 8};
|
|
static PyTypeObject GlpyhMetricsType = {{{0}}};
|
|
|
|
static PyObject*
|
|
glyph_metrics(Face *self) {
|
|
#define glyph_metrics_doc ""
|
|
PyObject *ans = PyStructSequence_New(&GlpyhMetricsType);
|
|
if (ans != NULL) {
|
|
#define SI(num, attr) PyStructSequence_SET_ITEM(ans, num, PyLong_FromLong(self->face->glyph->metrics.attr)); if (PyStructSequence_GET_ITEM(ans, num) == NULL) { Py_CLEAR(ans); return PyErr_NoMemory(); }
|
|
SI(0, width); SI(1, height);
|
|
SI(2, horiBearingX); SI(3, horiBearingY); SI(4, horiAdvance);
|
|
SI(5, vertBearingX); SI(6, vertBearingY); SI(7, vertAdvance);
|
|
#undef SI
|
|
} else return PyErr_NoMemory();
|
|
return ans;
|
|
}
|
|
|
|
static PyStructSequence_Field bm_fields[] = {
|
|
{"rows", NULL},
|
|
{"width", NULL},
|
|
{"pitch", NULL},
|
|
{"buffer", NULL},
|
|
{"num_grays", NULL},
|
|
{"pixel_mode", NULL},
|
|
{"palette_mode", NULL},
|
|
{NULL}
|
|
};
|
|
static PyStructSequence_Desc bm_desc = {"Bitmap", NULL, bm_fields, 7};
|
|
static PyTypeObject BitmapType = {{{0}}};
|
|
|
|
static PyObject*
|
|
bitmap(Face *self) {
|
|
#define bitmap_doc ""
|
|
PyObject *ans = PyStructSequence_New(&BitmapType);
|
|
if (ans != NULL) {
|
|
#define SI(num, attr, func, conv) PyStructSequence_SET_ITEM(ans, num, func((conv)self->face->glyph->bitmap.attr)); if (PyStructSequence_GET_ITEM(ans, num) == NULL) { Py_CLEAR(ans); return PyErr_NoMemory(); }
|
|
SI(0, rows, PyLong_FromUnsignedLong, unsigned long); SI(1, width, PyLong_FromUnsignedLong, unsigned long);
|
|
SI(2, pitch, PyLong_FromLong, long);
|
|
PyObject *t = PyByteArray_FromStringAndSize((const char*)self->face->glyph->bitmap.buffer, self->face->glyph->bitmap.rows * self->face->glyph->bitmap.pitch);
|
|
if (t == NULL) { Py_CLEAR(ans); return PyErr_NoMemory(); }
|
|
PyStructSequence_SET_ITEM(ans, 3, t);
|
|
SI(4, num_grays, PyLong_FromUnsignedLong, unsigned long);
|
|
SI(5, pixel_mode, PyLong_FromUnsignedLong, unsigned long); SI(6, palette_mode, PyLong_FromUnsignedLong, unsigned long);
|
|
#undef SI
|
|
} else return PyErr_NoMemory();
|
|
return ans;
|
|
}
|
|
|
|
static PyObject*
|
|
trim_to_width(Face UNUSED *self, PyObject *args) {
|
|
#define trim_to_width_doc "Trim edges from the supplied bitmap to make it fit in the specified cell-width"
|
|
PyObject *bitmap, *t;
|
|
unsigned long cell_width, rows, width, rtrim = 0, extra, ltrim;
|
|
unsigned char *src, *dest;
|
|
bool column_has_text = false;
|
|
if (!PyArg_ParseTuple(args, "O!k", &BitmapType, &bitmap, &cell_width)) return NULL;
|
|
rows = PyLong_AsUnsignedLong(PyStructSequence_GET_ITEM(bitmap, 0));
|
|
width = PyLong_AsUnsignedLong(PyStructSequence_GET_ITEM(bitmap, 1));
|
|
extra = width - cell_width;
|
|
if (extra >= cell_width) { PyErr_SetString(PyExc_ValueError, "Too large for trimming"); return NULL; }
|
|
PyObject *ans = PyStructSequence_New(&BitmapType);
|
|
if (ans == NULL) return PyErr_NoMemory();
|
|
src = (unsigned char*)PyByteArray_AS_STRING(PyStructSequence_GET_ITEM(bitmap, 3));
|
|
PyObject *abuf = PyByteArray_FromStringAndSize(NULL, cell_width * rows);
|
|
if (abuf == NULL) { Py_CLEAR(ans); return PyErr_NoMemory(); }
|
|
dest = (unsigned char*)PyByteArray_AS_STRING(abuf);
|
|
PyStructSequence_SET_ITEM(ans, 1, PyLong_FromUnsignedLong(cell_width));
|
|
PyStructSequence_SET_ITEM(ans, 2, PyLong_FromUnsignedLong(cell_width));
|
|
PyStructSequence_SET_ITEM(ans, 3, abuf);
|
|
#define COPY(which) t = PyStructSequence_GET_ITEM(bitmap, which); Py_INCREF(t); PyStructSequence_SET_ITEM(ans, which, t);
|
|
COPY(0); COPY(4); COPY(5); COPY(6);
|
|
#undef COPY
|
|
|
|
for (long x = width - 1; !column_has_text && x > -1 && rtrim < extra; x--) {
|
|
for (unsigned long y = 0; y < rows * width; y += width) {
|
|
if (src[x + y] > 200) { column_has_text = true; break; }
|
|
}
|
|
if (!column_has_text) rtrim++;
|
|
}
|
|
rtrim = MIN(extra, rtrim);
|
|
ltrim = extra - rtrim;
|
|
for (unsigned long y = 0; y < rows; y++) {
|
|
memcpy(dest + y*cell_width, src + ltrim + y*width, cell_width);
|
|
}
|
|
|
|
return ans;
|
|
}
|
|
|
|
// Boilerplate {{{
|
|
|
|
static PyMemberDef members[] = {
|
|
#define MEM(name, type) {#name, type, offsetof(Face, name), READONLY, #name}
|
|
MEM(units_per_EM, T_UINT),
|
|
MEM(ascender, T_INT),
|
|
MEM(descender, T_INT),
|
|
MEM(height, T_INT),
|
|
MEM(max_advance_width, T_INT),
|
|
MEM(max_advance_height, T_INT),
|
|
MEM(underline_position, T_INT),
|
|
MEM(underline_thickness, T_INT),
|
|
MEM(is_scalable, T_BOOL),
|
|
{NULL} /* Sentinel */
|
|
};
|
|
|
|
static PyMethodDef methods[] = {
|
|
METHOD(set_char_size, METH_VARARGS)
|
|
METHOD(load_char, METH_VARARGS)
|
|
METHOD(get_char_index, METH_VARARGS)
|
|
METHOD(glyph_metrics, METH_NOARGS)
|
|
METHOD(bitmap, METH_NOARGS)
|
|
METHOD(trim_to_width, METH_VARARGS)
|
|
{NULL} /* Sentinel */
|
|
};
|
|
|
|
|
|
PyTypeObject Face_Type = {
|
|
PyVarObject_HEAD_INIT(NULL, 0)
|
|
.tp_name = "fast_data_types.Face",
|
|
.tp_basicsize = sizeof(Face),
|
|
.tp_dealloc = (destructor)dealloc,
|
|
.tp_flags = Py_TPFLAGS_DEFAULT,
|
|
.tp_doc = "FreeType Font face",
|
|
.tp_methods = methods,
|
|
.tp_members = members,
|
|
.tp_new = new,
|
|
};
|
|
|
|
INIT_TYPE(Face)
|
|
|
|
|
|
bool
|
|
init_freetype_library(PyObject *m) {
|
|
int error = FT_Init_FreeType(&library);
|
|
if (error) {
|
|
set_freetype_error("Failed to initialize FreeType library, with error:", error);
|
|
return false;
|
|
}
|
|
if (PyStructSequence_InitType2(&GlpyhMetricsType, &gm_desc) != 0) return false;
|
|
if (PyStructSequence_InitType2(&BitmapType, &bm_desc) != 0) return false;
|
|
PyModule_AddObject(m, "GlyphMetrics", (PyObject*)&GlpyhMetricsType);
|
|
PyModule_AddObject(m, "Bitmap", (PyObject*)&BitmapType);
|
|
PyModule_AddIntMacro(m, FT_LOAD_RENDER);
|
|
PyModule_AddIntMacro(m, FT_LOAD_TARGET_NORMAL);
|
|
PyModule_AddIntMacro(m, FT_LOAD_TARGET_LIGHT);
|
|
PyModule_AddIntMacro(m, FT_LOAD_NO_HINTING);
|
|
PyModule_AddIntMacro(m, FT_PIXEL_MODE_GRAY);
|
|
return true;
|
|
}
|
|
|
|
// }}}
|