Pillow/src/_imagingft.c

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

1656 lines
49 KiB
C
Raw Normal View History

2010-07-31 06:52:47 +04:00
/*
* PIL FreeType Driver
*
* a FreeType 2.X driver for PIL
*
* history:
* 2001-02-17 fl Created (based on old experimental freetype 1.0 code)
* 2001-04-18 fl Fixed some egcs compiler nits
* 2002-11-08 fl Added unicode support; more font metrics, etc
* 2003-05-20 fl Fixed compilation under 1.5.2 and newer non-unicode builds
* 2003-09-27 fl Added charmap encoding support
* 2004-05-15 fl Fixed compilation for FreeType 2.1.8
* 2004-09-10 fl Added support for monochrome bitmaps
* 2006-06-18 fl Fixed glyph bearing calculation
* 2007-12-23 fl Fixed crash in family/style attribute fetch
* 2008-01-02 fl Handle Unicode filenames properly
*
* Copyright (c) 1998-2007 by Secret Labs AB
*/
2019-03-26 05:50:57 +03:00
#define PY_SSIZE_T_CLEAN
2010-07-31 06:52:47 +04:00
#include "Python.h"
#include "thirdparty/pythoncapi_compat.h"
#include "libImaging/Imaging.h"
2010-07-31 06:52:47 +04:00
#include <ft2build.h>
#include FT_FREETYPE_H
#include FT_GLYPH_H
#include FT_BITMAP_H
2019-07-28 23:40:03 +03:00
#include FT_STROKER_H
2019-06-12 13:27:11 +03:00
#include FT_MULTIPLE_MASTERS_H
#include FT_SFNT_NAMES_H
2020-03-28 07:58:35 +03:00
#ifdef FT_COLOR_H
#include FT_COLOR_H
#endif
2010-07-31 06:52:47 +04:00
/* -------------------------------------------------------------------- */
/* error table */
#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, 0 } \
} \
;
#ifdef HAVE_RAQM
#ifdef HAVE_RAQM_SYSTEM
#include <raqm.h>
#else
#include "thirdparty/raqm/raqm.h"
#ifdef HAVE_FRIBIDI_SYSTEM
#include <fribidi.h>
#else
#include "thirdparty/fribidi-shim/fribidi.h"
#include <hb.h>
#endif
#endif
#endif
static int have_raqm = 0;
2016-01-27 08:09:26 +03:00
2017-06-13 19:03:23 +03:00
#define LAYOUT_FALLBACK 0
#define LAYOUT_RAQM 1
2016-01-27 08:09:26 +03:00
typedef struct {
2019-05-31 13:31:41 +03:00
int index, x_offset, x_advance, y_offset, y_advance;
2016-01-27 08:09:26 +03:00
unsigned int cluster;
} GlyphInfo;
2010-07-31 06:52:47 +04:00
struct {
int code;
const char *message;
} ft_errors[] =
2013-11-28 16:58:43 +04:00
#include FT_ERRORS_H
2010-07-31 06:52:47 +04:00
/* -------------------------------------------------------------------- */
/* font objects */
static FT_Library library;
#ifdef Py_GIL_DISABLED
static PyMutex ft_library_mutex;
#endif
2010-07-31 06:52:47 +04:00
typedef struct {
PyObject_HEAD FT_Face face;
unsigned char *font_bytes;
2017-06-13 19:03:23 +03:00
int layout_engine;
2010-07-31 06:52:47 +04:00
} FontObject;
static PyTypeObject Font_Type;
2010-07-31 06:52:47 +04:00
2020-09-05 08:21:40 +03:00
/* round a 26.6 pixel coordinate to the nearest integer */
#define PIXEL(x) ((((x) + 32) & -64) >> 6)
2010-07-31 06:52:47 +04:00
static PyObject *
geterror(int code) {
int i;
2020-05-10 12:56:36 +03:00
for (i = 0; ft_errors[i].message; i++) {
2010-07-31 06:52:47 +04:00
if (ft_errors[i].code == code) {
PyErr_SetString(PyExc_OSError, ft_errors[i].message);
2010-07-31 06:52:47 +04:00
return NULL;
}
2020-05-10 12:56:36 +03:00
}
2010-07-31 06:52:47 +04:00
PyErr_SetString(PyExc_OSError, "unknown freetype error");
2010-07-31 06:52:47 +04:00
return NULL;
}
static PyObject *
getfont(PyObject *self_, PyObject *args, PyObject *kw) {
/* create a font object from a file name and a size (in pixels) */
FontObject *self;
int error = 0;
2010-07-31 06:52:47 +04:00
char *filename = NULL;
2023-04-22 06:45:18 +03:00
float size;
FT_Size_RequestRec req;
FT_Long width;
2019-03-26 05:50:57 +03:00
Py_ssize_t index = 0;
Py_ssize_t layout_engine = 0;
2013-04-25 20:48:43 +04:00
unsigned char *encoding;
2013-04-26 08:15:32 +04:00
unsigned char *font_bytes;
2019-03-26 05:50:57 +03:00
Py_ssize_t font_bytes_size = 0;
2010-07-31 06:52:47 +04:00
static char *kwlist[] = {
"filename", "size", "index", "encoding", "font_bytes", "layout_engine", NULL
};
2010-07-31 06:52:47 +04:00
if (!library) {
PyErr_SetString(PyExc_OSError, "failed to initialize FreeType library");
return NULL;
}
#if PY_MAJOR_VERSION > 3 || PY_MINOR_VERSION > 11
PyConfig config;
PyConfig_InitPythonConfig(&config);
if (!PyArg_ParseTupleAndKeywords(
args,
kw,
"etf|nsy#n",
kwlist,
config.filesystem_encoding,
&filename,
&size,
&index,
&encoding,
&font_bytes,
&font_bytes_size,
&layout_engine
)) {
PyConfig_Clear(&config);
return NULL;
}
PyConfig_Clear(&config);
#else
2019-09-26 15:12:28 +03:00
if (!PyArg_ParseTupleAndKeywords(
args,
kw,
2023-04-22 06:45:18 +03:00
"etf|nsy#n",
2019-09-26 15:12:28 +03:00
kwlist,
Py_FileSystemDefaultEncoding,
&filename,
&size,
&index,
&encoding,
&font_bytes,
2017-06-13 19:03:23 +03:00
&font_bytes_size,
&layout_engine
)) {
return NULL;
}
#endif
2010-07-31 06:52:47 +04:00
self = PyObject_New(FontObject, &Font_Type);
if (!self) {
2020-05-10 12:56:36 +03:00
if (filename) {
PyMem_Free(filename);
2020-05-10 12:56:36 +03:00
}
2010-07-31 06:52:47 +04:00
return NULL;
}
2016-09-03 05:23:42 +03:00
2016-05-30 13:56:28 +03:00
self->face = NULL;
2017-06-13 19:03:23 +03:00
self->layout_engine = layout_engine;
2016-09-03 05:23:42 +03:00
2013-04-26 08:15:32 +04:00
if (filename && font_bytes_size <= 0) {
self->font_bytes = NULL;
MUTEX_LOCK(&ft_library_mutex);
error = FT_New_Face(library, filename, index, &self->face);
MUTEX_UNLOCK(&ft_library_mutex);
} else {
/* need to have allocated storage for font_bytes for the life of the object.*/
/* Don't free this before FT_Done_Face */
self->font_bytes = PyMem_Malloc(font_bytes_size);
if (!self->font_bytes) {
error = FT_Err_Out_Of_Memory;
}
if (!error) {
memcpy(self->font_bytes, font_bytes, (size_t)font_bytes_size);
MUTEX_LOCK(&ft_library_mutex);
2015-10-11 13:24:35 +03:00
error = FT_New_Memory_Face(
library,
(FT_Byte *)self->font_bytes,
font_bytes_size,
index,
&self->face
);
MUTEX_UNLOCK(&ft_library_mutex);
}
}
2020-05-10 12:56:36 +03:00
if (!error) {
2023-04-22 06:45:18 +03:00
width = size * 64;
req.type = FT_SIZE_REQUEST_TYPE_NOMINAL;
req.width = width;
req.height = width;
req.horiResolution = 0;
req.vertResolution = 0;
error = FT_Request_Size(self->face, &req);
2020-05-10 12:56:36 +03:00
}
2010-07-31 06:52:47 +04:00
if (!error && encoding && strlen((char *)encoding) == 4) {
FT_Encoding encoding_tag =
FT_MAKE_TAG(encoding[0], encoding[1], encoding[2], encoding[3]);
error = FT_Select_Charmap(self->face, encoding_tag);
}
2020-05-10 12:56:36 +03:00
if (filename) {
PyMem_Free(filename);
2020-05-10 12:56:36 +03:00
}
2010-07-31 06:52:47 +04:00
if (error) {
if (self->font_bytes) {
PyMem_Free(self->font_bytes);
self->font_bytes = NULL;
2015-10-11 13:24:35 +03:00
}
Py_DECREF(self);
2010-07-31 06:52:47 +04:00
return geterror(error);
}
return (PyObject *)self;
}
2020-11-25 19:27:12 +03:00
#ifdef HAVE_RAQM
2016-02-16 10:52:59 +03:00
static size_t
text_layout_raqm(
PyObject *string,
FontObject *self,
const char *dir,
PyObject *features,
2020-03-28 07:58:35 +03:00
const char *lang,
GlyphInfo **glyph_info
) {
2019-04-15 14:01:44 +03:00
size_t i = 0, count = 0, start = 0;
raqm_t *rq;
raqm_glyph_t *glyphs = NULL;
2016-01-27 08:09:26 +03:00
raqm_direction_t direction;
2010-07-31 06:52:47 +04:00
2020-11-25 14:21:42 +03:00
rq = raqm_create();
if (rq == NULL) {
PyErr_SetString(PyExc_ValueError, "raqm_create() failed.");
goto failed;
}
2024-06-15 09:05:58 +03:00
Py_ssize_t size;
int set_text;
2019-04-07 22:56:57 +03:00
if (PyUnicode_Check(string)) {
Py_UCS4 *text = PyUnicode_AsUCS4Copy(string);
2024-06-15 09:05:58 +03:00
size = PyUnicode_GET_LENGTH(string);
2019-04-07 22:56:57 +03:00
if (!text || !size) {
/* return 0 and clean up, no glyphs==no size,
and raqm fails with empty strings */
goto failed;
}
2024-06-15 09:05:58 +03:00
set_text = raqm_set_text(rq, text, size);
2019-04-07 22:56:57 +03:00
PyMem_Free(text);
2024-08-16 11:31:07 +03:00
} else if (PyBytes_Check(string)) {
2024-06-15 09:05:58 +03:00
char *buffer;
PyBytes_AsStringAndSize(string, &buffer, &size);
if (!buffer || !size) {
/* return 0 and clean up, no glyphs==no size,
and raqm fails with empty strings */
2019-04-07 22:56:57 +03:00
goto failed;
}
2024-06-15 09:05:58 +03:00
set_text = raqm_set_text_utf8(rq, buffer, size);
2024-08-16 11:31:07 +03:00
} else {
PyErr_SetString(PyExc_TypeError, "expected string or bytes");
goto failed;
2024-06-15 09:05:58 +03:00
}
if (!set_text) {
PyErr_SetString(PyExc_ValueError, "raqm_set_text() failed");
goto failed;
}
if (lang && !raqm_set_language(rq, lang, start, size)) {
PyErr_SetString(PyExc_ValueError, "raqm_set_language() failed");
goto failed;
}
direction = RAQM_DIRECTION_DEFAULT;
if (dir) {
2020-05-11 00:46:12 +03:00
if (strcmp(dir, "rtl") == 0) {
direction = RAQM_DIRECTION_RTL;
2020-05-11 00:46:12 +03:00
} else if (strcmp(dir, "ltr") == 0) {
direction = RAQM_DIRECTION_LTR;
2020-05-11 00:46:12 +03:00
} else if (strcmp(dir, "ttb") == 0) {
direction = RAQM_DIRECTION_TTB;
#if !defined(RAQM_VERSION_ATLEAST)
/* RAQM_VERSION_ATLEAST was added in Raqm 0.7.0 */
2021-03-25 02:04:41 +03:00
PyErr_SetString(
PyExc_ValueError, "libraqm 0.7 or greater required for 'ttb' direction"
);
2021-03-25 02:04:41 +03:00
goto failed;
#endif
2019-05-31 13:31:41 +03:00
} else {
PyErr_SetString(
PyExc_ValueError, "direction must be either 'rtl', 'ltr' or 'ttb'"
);
goto failed;
}
}
2020-11-25 14:21:42 +03:00
if (!raqm_set_par_direction(rq, direction)) {
PyErr_SetString(PyExc_ValueError, "raqm_set_par_direction() failed");
goto failed;
}
2016-01-27 08:09:26 +03:00
if (features != Py_None) {
2019-04-15 14:01:44 +03:00
int j, len;
PyObject *seq = PySequence_Fast(features, "expected a sequence");
if (!seq) {
goto failed;
}
len = PySequence_Fast_GET_SIZE(seq);
2019-04-15 14:01:44 +03:00
for (j = 0; j < len; j++) {
PyObject *item = PySequence_Fast_GET_ITEM(seq, j);
char *feature = NULL;
Py_ssize_t size = 0;
2016-01-25 14:16:35 +03:00
PyObject *bytes;
2016-01-25 14:16:35 +03:00
if (!PyUnicode_Check(item)) {
Py_DECREF(seq);
PyErr_SetString(PyExc_TypeError, "expected a string");
goto failed;
}
bytes = PyUnicode_AsUTF8String(item);
if (bytes == NULL) {
Py_DECREF(seq);
goto failed;
2016-01-25 14:16:35 +03:00
}
feature = PyBytes_AS_STRING(bytes);
size = PyBytes_GET_SIZE(bytes);
2020-11-25 14:21:42 +03:00
if (!raqm_add_font_feature(rq, feature, size)) {
Py_DECREF(seq);
Py_DECREF(bytes);
PyErr_SetString(PyExc_ValueError, "raqm_add_font_feature() failed");
goto failed;
}
Py_DECREF(bytes);
}
Py_DECREF(seq);
}
2020-11-25 14:21:42 +03:00
if (!raqm_set_freetype_face(rq, self->face)) {
PyErr_SetString(PyExc_RuntimeError, "raqm_set_freetype_face() failed.");
goto failed;
}
2020-11-25 14:21:42 +03:00
if (!raqm_layout(rq)) {
PyErr_SetString(PyExc_RuntimeError, "raqm_layout() failed.");
goto failed;
}
2021-03-25 02:04:41 +03:00
glyphs = raqm_get_glyphs(rq, &count);
if (glyphs == NULL) {
PyErr_SetString(PyExc_ValueError, "raqm_get_glyphs() failed.");
count = 0;
goto failed;
}
2010-07-31 06:52:47 +04:00
2016-01-27 08:09:26 +03:00
(*glyph_info) = PyMem_New(GlyphInfo, count);
if ((*glyph_info) == NULL) {
PyErr_SetString(PyExc_MemoryError, "PyMem_New() failed");
count = 0;
goto failed;
}
2021-03-25 02:04:41 +03:00
for (i = 0; i < count; i++) {
(*glyph_info)[i].index = glyphs[i].index;
(*glyph_info)[i].x_offset = glyphs[i].x_offset;
(*glyph_info)[i].x_advance = glyphs[i].x_advance;
(*glyph_info)[i].y_offset = glyphs[i].y_offset;
(*glyph_info)[i].y_advance = glyphs[i].y_advance;
(*glyph_info)[i].cluster = glyphs[i].cluster;
}
2016-01-27 08:09:26 +03:00
failed:
2020-11-25 14:21:42 +03:00
raqm_destroy(rq);
2016-01-27 08:09:26 +03:00
return count;
}
2016-01-27 08:09:26 +03:00
2020-11-25 19:27:12 +03:00
#endif
static size_t
text_layout_fallback(
PyObject *string,
FontObject *self,
const char *dir,
PyObject *features,
2020-03-28 07:58:35 +03:00
const char *lang,
GlyphInfo **glyph_info,
int mask,
int color
) {
2024-06-15 09:05:58 +03:00
int error, load_flags, i;
char *buffer = NULL;
2016-01-27 08:09:26 +03:00
FT_ULong ch;
Py_ssize_t count;
FT_GlyphSlot glyph;
FT_Bool kerning = FT_HAS_KERNING(self->face);
FT_UInt last_index = 0;
2016-12-13 08:07:29 +03:00
if (features != Py_None || dir != NULL || lang != NULL) {
PyErr_SetString(
PyExc_KeyError,
"setting text direction, language or font features is not supported "
"without libraqm"
);
}
2016-01-27 08:09:26 +03:00
2024-06-15 09:05:58 +03:00
if (PyUnicode_Check(string)) {
count = PyUnicode_GET_LENGTH(string);
2024-08-16 11:31:07 +03:00
} else if (PyBytes_Check(string)) {
2024-06-15 09:05:58 +03:00
PyBytes_AsStringAndSize(string, &buffer, &count);
2024-08-16 11:31:07 +03:00
} else {
PyErr_SetString(PyExc_TypeError, "expected string or bytes");
return 0;
}
if (count == 0) {
2016-01-27 08:09:26 +03:00
return 0;
}
2016-01-27 08:09:26 +03:00
(*glyph_info) = PyMem_New(GlyphInfo, count);
if ((*glyph_info) == NULL) {
PyErr_SetString(PyExc_MemoryError, "PyMem_New() failed");
return 0;
}
load_flags = FT_LOAD_DEFAULT;
if (mask) {
2016-01-27 08:09:26 +03:00
load_flags |= FT_LOAD_TARGET_MONO;
}
2020-03-28 07:58:35 +03:00
if (color) {
load_flags |= FT_LOAD_COLOR;
}
2024-06-15 09:05:58 +03:00
for (i = 0; i < count; i++) {
if (buffer) {
ch = buffer[i];
} else {
ch = PyUnicode_READ_CHAR(string, i);
}
2016-01-27 08:09:26 +03:00
(*glyph_info)[i].index = FT_Get_Char_Index(self->face, ch);
error = FT_Load_Glyph(self->face, (*glyph_info)[i].index, load_flags);
if (error) {
geterror(error);
return 0;
}
glyph = self->face->glyph;
(*glyph_info)[i].x_offset = 0;
(*glyph_info)[i].y_offset = 0;
if (kerning && last_index && (*glyph_info)[i].index) {
FT_Vector delta;
if (FT_Get_Kerning(
self->face,
last_index,
(*glyph_info)[i].index,
2020-03-29 14:24:18 +03:00
ft_kerning_default,
&delta
) == 0) {
2020-03-29 14:24:18 +03:00
(*glyph_info)[i - 1].x_advance += PIXEL(delta.x);
(*glyph_info)[i - 1].y_advance += PIXEL(delta.y);
}
2016-01-27 08:09:26 +03:00
}
(*glyph_info)[i].x_advance = glyph->metrics.horiAdvance;
// y_advance is only used in ttb, which is not supported by basic layout
(*glyph_info)[i].y_advance = 0;
2016-01-27 08:09:26 +03:00
last_index = (*glyph_info)[i].index;
(*glyph_info)[i].cluster = ch;
}
return count;
}
static size_t
text_layout(
PyObject *string,
FontObject *self,
const char *dir,
PyObject *features,
2020-03-28 07:58:35 +03:00
const char *lang,
GlyphInfo **glyph_info,
int mask,
int color
) {
size_t count;
2020-11-25 19:27:12 +03:00
#ifdef HAVE_RAQM
if (have_raqm && self->layout_engine == LAYOUT_RAQM) {
count = text_layout_raqm(string, self, dir, features, lang, glyph_info);
2020-11-25 19:27:12 +03:00
} else
#endif
{
2020-03-28 07:58:35 +03:00
count = text_layout_fallback(
string, self, dir, features, lang, glyph_info, mask, color
);
2017-06-13 19:03:23 +03:00
}
return count;
}
static PyObject *
font_getlength(FontObject *self, PyObject *args) {
int length; /* length along primary axis, in 26.6 precision */
GlyphInfo *glyph_info = NULL; /* computed text layout */
size_t i, count; /* glyph_info index and length */
int horizontal_dir; /* is primary axis horizontal? */
int mask = 0; /* is FT_LOAD_TARGET_MONO enabled? */
int color = 0; /* is FT_LOAD_COLOR enabled? */
const char *mode = NULL;
const char *dir = NULL;
const char *lang = NULL;
PyObject *features = Py_None;
PyObject *string;
/* calculate size and bearing for a given string */
if (!PyArg_ParseTuple(
args, "O|zzOz:getlength", &string, &mode, &dir, &features, &lang
)) {
return NULL;
}
horizontal_dir = dir && strcmp(dir, "ttb") == 0 ? 0 : 1;
mask = mode && strcmp(mode, "1") == 0;
color = mode && strcmp(mode, "RGBA") == 0;
count = text_layout(string, self, dir, features, lang, &glyph_info, mask, color);
if (PyErr_Occurred()) {
return NULL;
}
length = 0;
for (i = 0; i < count; i++) {
if (horizontal_dir) {
length += glyph_info[i].x_advance;
} else {
length -= glyph_info[i].y_advance;
}
}
if (glyph_info) {
PyMem_Free(glyph_info);
glyph_info = NULL;
}
return PyLong_FromLong(length);
}
2023-06-10 10:10:42 +03:00
static int
bounding_box_and_anchors(
FT_Face face,
const char *anchor,
int horizontal_dir,
GlyphInfo *glyph_info,
size_t count,
int load_flags,
int *width,
int *height,
int *x_offset,
int *y_offset
) {
int position; /* pen position along primary axis, in 26.6 precision */
int advanced; /* pen position along primary axis, in pixels */
2020-09-05 08:21:40 +03:00
int px, py; /* position of current glyph, in pixels */
int x_min, x_max, y_min, y_max; /* text bounding box, in pixels */
int x_anchor, y_anchor; /* offset of point drawn at (0, 0), in pixels */
int error;
FT_Glyph glyph;
FT_BBox bbox; /* glyph bounding box */
size_t i; /* glyph_info index */
/*
* text bounds are given by:
* - bounding boxes of individual glyphs
* - pen line, i.e. 0 to `advanced` along primary axis
* this means point (0, 0) is part of the text bounding box
*/
position = x_min = x_max = y_min = y_max = 0;
2019-05-31 13:31:41 +03:00
for (i = 0; i < count; i++) {
if (horizontal_dir) {
2020-09-05 08:21:40 +03:00
px = PIXEL(position + glyph_info[i].x_offset);
py = PIXEL(glyph_info[i].y_offset);
2019-05-31 13:31:41 +03:00
position += glyph_info[i].x_advance;
advanced = PIXEL(position);
if (advanced > x_max) {
x_max = advanced;
2020-05-10 12:56:36 +03:00
}
2019-05-31 13:31:41 +03:00
} else {
2020-09-05 08:21:40 +03:00
px = PIXEL(glyph_info[i].x_offset);
py = PIXEL(position + glyph_info[i].y_offset);
2019-05-31 13:31:41 +03:00
position += glyph_info[i].y_advance;
advanced = PIXEL(position);
if (advanced < y_min) {
y_min = advanced;
2020-05-10 12:56:36 +03:00
}
2019-05-31 13:31:41 +03:00
}
error = FT_Load_Glyph(face, glyph_info[i].index, load_flags);
2020-05-10 12:56:36 +03:00
if (error) {
2023-06-10 10:10:42 +03:00
geterror(error);
return 1;
2020-05-10 12:56:36 +03:00
}
error = FT_Get_Glyph(face->glyph, &glyph);
if (error) {
2023-06-10 10:10:42 +03:00
geterror(error);
return 1;
}
2020-04-14 11:07:28 +03:00
FT_Glyph_Get_CBox(glyph, FT_GLYPH_BBOX_PIXELS, &bbox);
2020-09-05 08:21:40 +03:00
bbox.xMax += px;
if (bbox.xMax > x_max) {
x_max = bbox.xMax;
}
2020-09-05 08:21:40 +03:00
bbox.xMin += px;
if (bbox.xMin < x_min) {
x_min = bbox.xMin;
}
2020-09-05 08:21:40 +03:00
bbox.yMax += py;
if (bbox.yMax > y_max) {
y_max = bbox.yMax;
}
2020-09-05 08:21:40 +03:00
bbox.yMin += py;
if (bbox.yMin < y_min) {
y_min = bbox.yMin;
2019-05-31 13:31:41 +03:00
}
FT_Done_Glyph(glyph);
}
2023-06-10 10:10:42 +03:00
if (anchor == NULL) {
anchor = horizontal_dir ? "la" : "lt";
}
if (strlen(anchor) != 2) {
goto bad_anchor;
2017-07-18 16:40:47 +03:00
}
2018-01-27 09:02:56 +03:00
x_anchor = y_anchor = 0;
2023-06-10 10:10:42 +03:00
if (count) {
2019-05-31 13:31:41 +03:00
if (horizontal_dir) {
switch (anchor[0]) {
case 'l': // left
x_anchor = 0;
break;
case 'm': // middle (left + right) / 2
x_anchor = PIXEL(position / 2);
break;
case 'r': // right
x_anchor = PIXEL(position);
break;
case 's': // vertical baseline
default:
goto bad_anchor;
}
switch (anchor[1]) {
case 'a': // ascender
2023-06-10 10:10:42 +03:00
y_anchor = PIXEL(face->size->metrics.ascender);
break;
case 't': // top
y_anchor = y_max;
break;
case 'm': // middle (ascender + descender) / 2
y_anchor = PIXEL(
(face->size->metrics.ascender + face->size->metrics.descender) /
2
);
break;
case 's': // horizontal baseline
y_anchor = 0;
break;
case 'b': // bottom
y_anchor = y_min;
break;
case 'd': // descender
2023-06-10 10:10:42 +03:00
y_anchor = PIXEL(face->size->metrics.descender);
break;
default:
goto bad_anchor;
2020-05-10 12:56:36 +03:00
}
2019-05-31 13:31:41 +03:00
} else {
switch (anchor[0]) {
case 'l': // left
x_anchor = x_min;
break;
case 'm': // middle (left + right) / 2
x_anchor = (x_min + x_max) / 2;
break;
case 'r': // right
x_anchor = x_max;
break;
case 's': // vertical baseline
x_anchor = 0;
break;
default:
goto bad_anchor;
}
switch (anchor[1]) {
case 't': // top
y_anchor = 0;
break;
case 'm': // middle (top + bottom) / 2
y_anchor = PIXEL(position / 2);
break;
case 'b': // bottom
y_anchor = PIXEL(position);
break;
case 'a': // ascender
case 's': // horizontal baseline
case 'd': // descender
default:
goto bad_anchor;
2020-05-10 12:56:36 +03:00
}
2019-05-31 13:31:41 +03:00
}
}
2023-06-10 10:10:42 +03:00
*width = x_max - x_min;
*height = y_max - y_min;
*x_offset = -x_anchor + x_min;
*y_offset = -(-y_anchor + y_max);
return 0;
bad_anchor:
PyErr_Format(PyExc_ValueError, "bad anchor specified: %s", anchor);
2023-06-10 10:10:42 +03:00
return 1;
}
static PyObject *
font_getsize(FontObject *self, PyObject *args) {
int width, height, x_offset, y_offset;
int load_flags; /* FreeType load_flags parameter */
2023-06-10 10:10:42 +03:00
int error;
GlyphInfo *glyph_info = NULL; /* computed text layout */
size_t count; /* glyph_info length */
int horizontal_dir; /* is primary axis horizontal? */
int mask = 0; /* is FT_LOAD_TARGET_MONO enabled? */
int color = 0; /* is FT_LOAD_COLOR enabled? */
const char *mode = NULL;
const char *dir = NULL;
const char *lang = NULL;
const char *anchor = NULL;
PyObject *features = Py_None;
PyObject *string;
/* calculate size and bearing for a given string */
if (!PyArg_ParseTuple(
args, "O|zzOzz:getsize", &string, &mode, &dir, &features, &lang, &anchor
)) {
2023-06-10 10:10:42 +03:00
return NULL;
}
horizontal_dir = dir && strcmp(dir, "ttb") == 0 ? 0 : 1;
mask = mode && strcmp(mode, "1") == 0;
color = mode && strcmp(mode, "RGBA") == 0;
count = text_layout(string, self, dir, features, lang, &glyph_info, mask, color);
if (PyErr_Occurred()) {
return NULL;
}
load_flags = FT_LOAD_DEFAULT;
if (mask) {
load_flags |= FT_LOAD_TARGET_MONO;
}
if (color) {
load_flags |= FT_LOAD_COLOR;
}
error = bounding_box_and_anchors(
self->face,
anchor,
horizontal_dir,
glyph_info,
count,
load_flags,
&width,
&height,
&x_offset,
&y_offset
);
2023-06-10 10:10:42 +03:00
if (glyph_info) {
PyMem_Free(glyph_info);
glyph_info = NULL;
}
if (error) {
return NULL;
}
return Py_BuildValue("(ii)(ii)", width, height, x_offset, y_offset);
}
2016-01-27 08:09:26 +03:00
static PyObject *
font_render(FontObject *self, PyObject *args) {
int x, y; /* pen position, in 26.6 precision */
2020-09-05 08:21:40 +03:00
int px, py; /* position of current glyph, in pixels */
int x_min, y_max; /* text offset in 26.6 precision */
int load_flags; /* FreeType load_flags parameter */
int error;
2019-07-28 23:40:03 +03:00
FT_Glyph glyph;
FT_GlyphSlot glyph_slot;
FT_Bitmap bitmap;
FT_Bitmap bitmap_converted; /* initialized lazily, for non-8bpp fonts */
2019-07-28 23:40:03 +03:00
FT_BitmapGlyph bitmap_glyph;
FT_Stroker stroker = NULL;
int bitmap_converted_ready = 0; /* has bitmap_converted been initialized */
GlyphInfo *glyph_info = NULL; /* computed text layout */
size_t i, count; /* glyph_info index and length */
int xx, yy; /* pixel offset of current glyph bitmap */
int x0, x1; /* horizontal bounds of glyph bitmap to copy */
unsigned int bitmap_y; /* glyph bitmap y index */
unsigned char *source; /* glyph bitmap source buffer */
unsigned char convert_scale; /* scale factor for non-8bpp bitmaps */
2023-06-10 10:10:42 +03:00
PyObject *image;
Imaging im;
int mask = 0; /* is FT_LOAD_TARGET_MONO enabled? */
2020-03-28 07:58:35 +03:00
int color = 0; /* is FT_LOAD_COLOR enabled? */
2024-09-11 12:53:52 +03:00
float stroke_width = 0;
2020-10-07 11:39:18 +03:00
PY_LONG_LONG foreground_ink_long = 0;
unsigned int foreground_ink;
2020-03-28 07:58:35 +03:00
const char *mode = NULL;
2016-01-27 08:09:26 +03:00
const char *dir = NULL;
const char *lang = NULL;
2023-06-10 10:10:42 +03:00
const char *anchor = NULL;
PyObject *features = Py_None;
PyObject *string;
2023-06-10 10:10:42 +03:00
PyObject *fill;
float x_start = 0;
float y_start = 0;
2023-06-10 10:10:42 +03:00
int width, height, x_offset, y_offset;
int horizontal_dir; /* is primary axis horizontal? */
2016-01-27 08:09:26 +03:00
/* render string into given buffer (the buffer *must* have
the right size, or this will crash) */
2016-01-27 08:09:26 +03:00
2020-03-28 07:58:35 +03:00
if (!PyArg_ParseTuple(
args,
2024-09-11 12:53:52 +03:00
"OO|zzOzfzLffO:render",
2020-03-28 07:58:35 +03:00
&string,
2023-06-10 10:10:42 +03:00
&fill,
2020-03-28 07:58:35 +03:00
&mode,
&dir,
&features,
&lang,
2020-10-07 11:39:18 +03:00
&stroke_width,
2023-06-10 10:10:42 +03:00
&anchor,
&foreground_ink_long,
&x_start,
&y_start
)) {
2016-01-27 08:09:26 +03:00
return NULL;
}
2016-01-27 08:09:26 +03:00
2020-03-28 07:58:35 +03:00
mask = mode && strcmp(mode, "1") == 0;
color = mode && strcmp(mode, "RGBA") == 0;
2020-10-07 11:39:18 +03:00
foreground_ink = foreground_ink_long;
2020-03-28 07:58:35 +03:00
#ifdef FT_COLOR_H
if (color) {
FT_Color foreground_color;
2020-10-07 11:39:18 +03:00
FT_Byte *ink = (FT_Byte *)&foreground_ink;
foreground_color.red = ink[0];
foreground_color.green = ink[1];
foreground_color.blue = ink[2];
2020-03-28 07:58:35 +03:00
foreground_color.alpha =
(FT_Byte)255; /* ink alpha is handled in ImageDraw.text */
FT_Palette_Set_Foreground_Color(self->face, foreground_color);
}
#endif
count = text_layout(string, self, dir, features, lang, &glyph_info, mask, color);
2017-08-31 22:31:05 +03:00
if (PyErr_Occurred()) {
2016-01-27 08:09:26 +03:00
return NULL;
}
2023-06-10 10:10:42 +03:00
load_flags = stroke_width ? FT_LOAD_NO_BITMAP : FT_LOAD_DEFAULT;
if (mask) {
load_flags |= FT_LOAD_TARGET_MONO;
}
if (color) {
load_flags |= FT_LOAD_COLOR;
}
horizontal_dir = dir && strcmp(dir, "ttb") == 0 ? 0 : 1;
error = bounding_box_and_anchors(
self->face,
anchor,
horizontal_dir,
glyph_info,
count,
load_flags,
&width,
&height,
&x_offset,
&y_offset
);
2023-06-10 10:10:42 +03:00
if (error) {
PyMem_Del(glyph_info);
return NULL;
}
2024-09-11 12:53:52 +03:00
width += ceil(stroke_width * 2 + x_start);
height += ceil(stroke_width * 2 + y_start);
image = PyObject_CallFunction(fill, "ii", width, height);
2024-10-07 11:17:59 +03:00
if (image == NULL) {
2023-06-15 07:27:33 +03:00
PyMem_Del(glyph_info);
return NULL;
}
2024-09-02 12:04:57 +03:00
PyObject *imagePtr = PyObject_GetAttrString(image, "ptr");
im = (Imaging)PyCapsule_GetPointer(imagePtr, IMAGING_MAGIC);
Py_XDECREF(imagePtr);
2023-06-10 10:10:42 +03:00
2024-09-11 12:53:52 +03:00
x_offset = round(x_offset - stroke_width);
y_offset = round(y_offset - stroke_width);
2023-06-10 10:10:42 +03:00
if (count == 0 || width == 0 || height == 0) {
PyMem_Del(glyph_info);
return Py_BuildValue("N(ii)", image, x_offset, y_offset);
}
2016-01-27 08:09:26 +03:00
2019-07-28 23:40:03 +03:00
if (stroke_width) {
error = FT_Stroker_New(library, &stroker);
if (error) {
2023-06-15 05:52:57 +03:00
geterror(error);
goto glyph_error;
2019-07-28 23:40:03 +03:00
}
FT_Stroker_Set(
stroker,
2024-09-11 12:53:52 +03:00
(FT_Fixed)round(stroke_width * 64),
2019-07-28 23:40:03 +03:00
FT_STROKER_LINECAP_ROUND,
FT_STROKER_LINEJOIN_ROUND,
0
);
2019-07-28 23:40:03 +03:00
}
/*
* calculate x_min and y_max
* must match font_getsize or there may be clipping!
*/
x = y = x_min = y_max = 0;
for (i = 0; i < count; i++) {
2020-09-05 08:21:40 +03:00
px = PIXEL(x + glyph_info[i].x_offset);
py = PIXEL(y + glyph_info[i].y_offset);
error =
FT_Load_Glyph(self->face, glyph_info[i].index, load_flags | FT_LOAD_RENDER);
2019-07-28 23:40:03 +03:00
if (error) {
2023-06-15 05:52:57 +03:00
geterror(error);
goto glyph_error;
2019-07-28 23:40:03 +03:00
}
2016-01-27 08:09:26 +03:00
2019-07-28 23:40:03 +03:00
glyph_slot = self->face->glyph;
bitmap = glyph_slot->bitmap;
2020-09-05 08:21:40 +03:00
if (glyph_slot->bitmap_top + py > y_max) {
y_max = glyph_slot->bitmap_top + py;
}
2020-09-05 08:21:40 +03:00
if (glyph_slot->bitmap_left + px < x_min) {
x_min = glyph_slot->bitmap_left + px;
2020-05-10 12:56:36 +03:00
}
x += glyph_info[i].x_advance;
y += glyph_info[i].y_advance;
}
/* set pen position to text origin */
2024-09-11 12:53:52 +03:00
x = round((-x_min + stroke_width + x_start) * 64);
y = round((-y_max + (-stroke_width) - y_start) * 64);
2020-02-16 12:40:23 +03:00
if (stroker == NULL) {
load_flags |= FT_LOAD_RENDER;
}
2019-05-31 13:31:41 +03:00
for (i = 0; i < count; i++) {
2020-09-05 08:21:40 +03:00
px = PIXEL(x + glyph_info[i].x_offset);
py = PIXEL(y + glyph_info[i].y_offset);
error = FT_Load_Glyph(self->face, glyph_info[i].index, load_flags);
2019-07-28 23:40:03 +03:00
if (error) {
2023-06-15 05:52:57 +03:00
geterror(error);
goto glyph_error;
2019-07-28 23:40:03 +03:00
}
glyph_slot = self->face->glyph;
if (stroker != NULL) {
error = FT_Get_Glyph(glyph_slot, &glyph);
if (!error) {
error = FT_Glyph_Stroke(&glyph, stroker, 1);
}
if (!error) {
FT_Vector origin = {0, 0};
error = FT_Glyph_To_Bitmap(&glyph, FT_RENDER_MODE_NORMAL, &origin, 1);
}
if (error) {
2023-06-15 05:52:57 +03:00
geterror(error);
goto glyph_error;
2019-07-28 23:40:03 +03:00
}
bitmap_glyph = (FT_BitmapGlyph)glyph;
bitmap = bitmap_glyph->bitmap;
2020-09-05 08:21:40 +03:00
xx = px + bitmap_glyph->left;
yy = -(py + bitmap_glyph->top);
2019-07-28 23:40:03 +03:00
} else {
bitmap = glyph_slot->bitmap;
2020-09-05 08:21:40 +03:00
xx = px + glyph_slot->bitmap_left;
yy = -(py + glyph_slot->bitmap_top);
2019-07-28 23:40:03 +03:00
}
2013-04-12 09:25:16 +04:00
// Null buffer, is dereferenced in FT_Bitmap_Convert
if (!bitmap.buffer && bitmap.rows) {
2022-11-05 07:41:17 +03:00
PyErr_SetString(PyExc_OSError, "Bitmap missing for glyph");
goto glyph_error;
}
/* convert non-8bpp bitmaps */
switch (bitmap.pixel_mode) {
case FT_PIXEL_MODE_MONO:
convert_scale = 255;
break;
case FT_PIXEL_MODE_GRAY2:
convert_scale = 255 / 3;
break;
case FT_PIXEL_MODE_GRAY4:
convert_scale = 255 / 15;
break;
default:
convert_scale = 1;
2019-05-31 13:31:41 +03:00
}
switch (bitmap.pixel_mode) {
case FT_PIXEL_MODE_MONO:
case FT_PIXEL_MODE_GRAY2:
case FT_PIXEL_MODE_GRAY4:
if (!bitmap_converted_ready) {
FT_Bitmap_Init(&bitmap_converted);
bitmap_converted_ready = 1;
}
error = FT_Bitmap_Convert(library, &bitmap, &bitmap_converted, 1);
if (error) {
geterror(error);
goto glyph_error;
}
bitmap = bitmap_converted;
/* bitmap is now FT_PIXEL_MODE_GRAY, fall through */
case FT_PIXEL_MODE_GRAY:
break;
case FT_PIXEL_MODE_BGRA:
if (color) {
break;
}
/* we didn't ask for color, fall through to default */
default:
2022-11-04 03:48:18 +03:00
PyErr_SetString(PyExc_OSError, "unsupported bitmap pixel mode");
goto glyph_error;
}
/* clip glyph bitmap width to target image bounds */
2013-04-12 09:25:16 +04:00
x0 = 0;
2019-07-28 23:40:03 +03:00
x1 = bitmap.width;
2020-05-10 12:56:36 +03:00
if (xx < 0) {
2013-04-12 09:25:16 +04:00
x0 = -xx;
2020-05-10 12:56:36 +03:00
}
if (xx + x1 > im->xsize) {
2013-04-12 09:25:16 +04:00
x1 = im->xsize - xx;
2020-05-10 12:56:36 +03:00
}
2013-04-12 09:25:16 +04:00
2019-07-28 23:40:03 +03:00
source = (unsigned char *)bitmap.buffer;
for (bitmap_y = 0; bitmap_y < bitmap.rows; bitmap_y++, yy++) {
/* clip glyph bitmap height to target image bounds */
2019-05-31 13:31:41 +03:00
if (yy >= 0 && yy < im->ysize) {
/* blend this glyph into the buffer */
int k;
unsigned char *target;
unsigned int tmp;
2020-03-28 07:58:35 +03:00
if (color) {
/* target[RGB] returns the color, target[A] returns the mask */
/* target bands get split again in ImageDraw.text */
target = (unsigned char *)im->image[yy] + xx * 4;
2019-05-31 13:31:41 +03:00
} else {
target = im->image8[yy] + xx;
}
if (color && bitmap.pixel_mode == FT_PIXEL_MODE_BGRA) {
/* paste color glyph */
2019-05-31 13:31:41 +03:00
for (k = x0; k < x1; k++) {
unsigned int src_alpha = source[k * 4 + 3];
/* paste only if source has data */
if (src_alpha > 0) {
/* unpremultiply BGRa */
int src_red =
CLIP8((255 * (int)source[k * 4 + 2]) / src_alpha);
int src_green =
CLIP8((255 * (int)source[k * 4 + 1]) / src_alpha);
int src_blue =
CLIP8((255 * (int)source[k * 4 + 0]) / src_alpha);
/* blend required if target has data */
if (target[k * 4 + 3] > 0) {
/* blend RGBA colors */
target[k * 4 + 0] =
BLEND(src_alpha, target[k * 4 + 0], src_red, tmp);
target[k * 4 + 1] =
BLEND(src_alpha, target[k * 4 + 1], src_green, tmp);
target[k * 4 + 2] =
BLEND(src_alpha, target[k * 4 + 2], src_blue, tmp);
target[k * 4 + 3] = CLIP8(
src_alpha +
MULDIV255(target[k * 4 + 3], (255 - src_alpha), tmp)
);
} else {
/* paste unpremultiplied RGBA values */
target[k * 4 + 0] = src_red;
target[k * 4 + 1] = src_green;
target[k * 4 + 2] = src_blue;
target[k * 4 + 3] = src_alpha;
}
2020-05-10 12:56:36 +03:00
}
2010-07-31 06:52:47 +04:00
}
2023-03-31 03:08:58 +03:00
} else if (bitmap.pixel_mode == FT_PIXEL_MODE_GRAY) {
if (color) {
2020-10-07 11:39:18 +03:00
unsigned char *ink = (unsigned char *)&foreground_ink;
2020-03-28 07:58:35 +03:00
for (k = x0; k < x1; k++) {
unsigned int src_alpha = source[k] * convert_scale;
if (src_alpha > 0) {
if (target[k * 4 + 3] > 0) {
target[k * 4 + 0] = BLEND(
src_alpha, target[k * 4 + 0], ink[0], tmp
);
target[k * 4 + 1] = BLEND(
src_alpha, target[k * 4 + 1], ink[1], tmp
);
target[k * 4 + 2] = BLEND(
src_alpha, target[k * 4 + 2], ink[2], tmp
);
target[k * 4 + 3] = CLIP8(
src_alpha +
MULDIV255(
target[k * 4 + 3], (255 - src_alpha), tmp
)
);
} else {
target[k * 4 + 0] = ink[0];
target[k * 4 + 1] = ink[1];
target[k * 4 + 2] = ink[2];
target[k * 4 + 3] = src_alpha;
}
2020-03-28 07:58:35 +03:00
}
2010-07-31 06:52:47 +04:00
}
2020-03-28 07:58:35 +03:00
} else {
for (k = x0; k < x1; k++) {
unsigned int src_alpha = source[k] * convert_scale;
if (src_alpha > 0) {
target[k] =
target[k] > 0
? CLIP8(
src_alpha +
MULDIV255(
target[k], (255 - src_alpha), tmp
)
)
: src_alpha;
}
2020-05-10 12:56:36 +03:00
}
2010-07-31 06:52:47 +04:00
}
} else {
2022-11-04 03:48:18 +03:00
PyErr_SetString(PyExc_OSError, "unsupported bitmap pixel mode");
goto glyph_error;
2010-07-31 06:52:47 +04:00
}
}
2019-07-28 23:40:03 +03:00
source += bitmap.pitch;
2010-07-31 06:52:47 +04:00
}
2016-02-03 14:24:10 +03:00
x += glyph_info[i].x_advance;
2020-04-14 04:16:32 +03:00
y += glyph_info[i].y_advance;
if (stroker != NULL) {
FT_Done_Glyph(glyph);
}
2010-07-31 06:52:47 +04:00
}
2016-02-16 10:52:59 +03:00
if (bitmap_converted_ready) {
FT_Bitmap_Done(library, &bitmap_converted);
}
2019-07-28 23:40:03 +03:00
FT_Stroker_Done(stroker);
2016-01-27 08:09:26 +03:00
PyMem_Del(glyph_info);
return Py_BuildValue("N(ii)", image, x_offset, y_offset);
glyph_error:
Py_DECREF(image);
if (stroker != NULL) {
FT_Done_Glyph(glyph);
}
if (bitmap_converted_ready) {
FT_Bitmap_Done(library, &bitmap_converted);
}
FT_Stroker_Done(stroker);
PyMem_Del(glyph_info);
return NULL;
2010-07-31 06:52:47 +04:00
}
2019-06-12 13:27:11 +03:00
#if FREETYPE_MAJOR > 2 || (FREETYPE_MAJOR == 2 && FREETYPE_MINOR > 9) || \
(FREETYPE_MAJOR == 2 && FREETYPE_MINOR == 9 && FREETYPE_PATCH == 1)
static PyObject *
font_getvarnames(FontObject *self) {
2019-06-12 13:27:11 +03:00
int error;
FT_UInt i, j, num_namedstyles, name_count;
FT_MM_Var *master;
FT_SfntName name;
PyObject *list_names, *list_name;
2021-01-03 06:17:51 +03:00
2019-06-12 13:27:11 +03:00
error = FT_Get_MM_Var(self->face, &master);
2020-05-10 12:56:36 +03:00
if (error) {
2019-06-12 13:27:11 +03:00
return geterror(error);
2021-01-03 06:17:51 +03:00
}
2019-06-12 13:27:11 +03:00
num_namedstyles = master->num_namedstyles;
list_names = PyList_New(num_namedstyles);
if (list_names == NULL) {
FT_Done_MM_Var(library, master);
return NULL;
}
int *list_names_filled = PyMem_Malloc(num_namedstyles * sizeof(int));
if (list_names_filled == NULL) {
Py_DECREF(list_names);
FT_Done_MM_Var(library, master);
return PyErr_NoMemory();
}
2024-08-07 11:44:43 +03:00
for (unsigned int i = 0; i < num_namedstyles; i++) {
list_names_filled[i] = 0;
}
2019-06-12 13:27:11 +03:00
name_count = FT_Get_Sfnt_Name_Count(self->face);
for (i = 0; i < name_count; i++) {
error = FT_Get_Sfnt_Name(self->face, i, &name);
2020-05-10 12:56:36 +03:00
if (error) {
PyMem_Free(list_names_filled);
2023-03-11 16:11:48 +03:00
Py_DECREF(list_names);
FT_Done_MM_Var(library, master);
2019-06-12 13:27:11 +03:00
return geterror(error);
2020-05-10 12:56:36 +03:00
}
2019-06-12 13:27:11 +03:00
for (j = 0; j < num_namedstyles; j++) {
if (list_names_filled[j]) {
2019-06-12 13:27:11 +03:00
continue;
2020-05-10 12:56:36 +03:00
}
2019-06-12 13:27:11 +03:00
if (master->namedstyle[j].strid == name.name_id) {
2019-09-26 15:12:28 +03:00
list_name = Py_BuildValue("y#", name.string, name.string_len);
if (list_name == NULL) {
PyMem_Free(list_names_filled);
Py_DECREF(list_names);
FT_Done_MM_Var(library, master);
return NULL;
}
2019-06-12 13:27:11 +03:00
PyList_SetItem(list_names, j, list_name);
list_names_filled[j] = 1;
2019-06-12 13:27:11 +03:00
break;
}
}
}
PyMem_Free(list_names_filled);
2019-06-12 13:27:11 +03:00
FT_Done_MM_Var(library, master);
return list_names;
2021-01-03 06:17:51 +03:00
}
2019-06-12 13:27:11 +03:00
static PyObject *
font_getvaraxes(FontObject *self) {
int error;
2019-06-12 13:27:11 +03:00
FT_UInt i, j, num_axis, name_count;
FT_MM_Var *master;
FT_Var_Axis axis;
FT_SfntName name;
PyObject *list_axes, *list_axis, *axis_name;
error = FT_Get_MM_Var(self->face, &master);
2020-05-10 12:56:36 +03:00
if (error) {
2019-06-12 13:27:11 +03:00
return geterror(error);
2021-01-03 06:17:51 +03:00
}
2019-06-12 13:27:11 +03:00
num_axis = master->num_axis;
name_count = FT_Get_Sfnt_Name_Count(self->face);
list_axes = PyList_New(num_axis);
if (list_axes == NULL) {
FT_Done_MM_Var(library, master);
return NULL;
}
2019-06-12 13:27:11 +03:00
for (i = 0; i < num_axis; i++) {
axis = master->axis[i];
list_axis = PyDict_New();
if (list_axis == NULL) {
Py_DECREF(list_axes);
FT_Done_MM_Var(library, master);
return NULL;
}
PyObject *minimum = PyLong_FromLong(axis.minimum / 65536);
PyDict_SetItemString(list_axis, "minimum", minimum ? minimum : Py_None);
Py_XDECREF(minimum);
2023-03-22 10:56:52 +03:00
PyObject *def = PyLong_FromLong(axis.def / 65536);
PyDict_SetItemString(list_axis, "default", def ? def : Py_None);
Py_XDECREF(def);
2023-03-11 10:32:43 +03:00
PyObject *maximum = PyLong_FromLong(axis.maximum / 65536);
PyDict_SetItemString(list_axis, "maximum", maximum ? maximum : Py_None);
Py_XDECREF(maximum);
for (j = 0; j < name_count; j++) {
error = FT_Get_Sfnt_Name(self->face, j, &name);
2020-05-10 12:56:36 +03:00
if (error) {
Py_DECREF(list_axis);
Py_DECREF(list_axes);
FT_Done_MM_Var(library, master);
2019-06-12 13:27:11 +03:00
return geterror(error);
}
if (name.name_id == axis.strid) {
axis_name = Py_BuildValue("y#", name.string, name.string_len);
if (axis_name == NULL) {
Py_DECREF(list_axis);
Py_DECREF(list_axes);
FT_Done_MM_Var(library, master);
return NULL;
}
2024-07-15 13:48:39 +03:00
PyDict_SetItemString(list_axis, "name", axis_name);
Py_DECREF(axis_name);
break;
}
2019-06-12 13:27:11 +03:00
}
PyList_SetItem(list_axes, i, list_axis);
}
FT_Done_MM_Var(library, master);
return list_axes;
2020-05-10 12:56:36 +03:00
}
2019-06-12 13:27:11 +03:00
static PyObject *
font_setvarname(FontObject *self, PyObject *args) {
int error;
int instance_index;
if (!PyArg_ParseTuple(args, "i", &instance_index)) {
return NULL;
}
error = FT_Set_Named_Instance(self->face, instance_index);
if (error) {
return geterror(error);
2021-01-03 06:17:51 +03:00
}
2019-06-12 13:27:11 +03:00
Py_INCREF(Py_None);
return Py_None;
2020-05-10 12:56:36 +03:00
}
2019-06-12 13:27:11 +03:00
static PyObject *
font_setvaraxes(FontObject *self, PyObject *args) {
int error;
PyObject *axes, *item;
Py_ssize_t i, num_coords;
FT_Fixed *coords;
FT_Fixed coord;
2020-05-10 12:56:36 +03:00
if (!PyArg_ParseTuple(args, "O", &axes)) {
2019-06-12 13:27:11 +03:00
return NULL;
}
if (!PyList_Check(axes)) {
PyErr_SetString(PyExc_TypeError, "argument must be a list");
return NULL;
2021-01-03 06:17:51 +03:00
}
2019-06-12 13:27:11 +03:00
num_coords = PyObject_Length(axes);
coords = (FT_Fixed *)malloc(num_coords * sizeof(FT_Fixed));
2019-06-25 23:33:49 +03:00
if (coords == NULL) {
return PyErr_NoMemory();
2021-01-03 06:17:51 +03:00
}
2019-06-12 13:27:11 +03:00
for (i = 0; i < num_coords; i++) {
item = PyList_GetItemRef(axes, i);
if (item == NULL) {
free(coords);
return NULL;
}
2020-05-10 12:56:36 +03:00
if (PyFloat_Check(item)) {
2019-06-12 13:27:11 +03:00
coord = PyFloat_AS_DOUBLE(item);
} else if (PyLong_Check(item)) {
2019-09-26 15:12:28 +03:00
coord = (float)PyLong_AS_LONG(item);
2020-05-10 12:56:36 +03:00
} else if (PyNumber_Check(item)) {
2019-06-12 13:27:11 +03:00
coord = PyFloat_AsDouble(item);
2021-01-03 06:17:51 +03:00
} else {
Py_DECREF(item);
2019-06-25 23:33:49 +03:00
free(coords);
2019-06-12 13:27:11 +03:00
PyErr_SetString(PyExc_TypeError, "list must contain numbers");
return NULL;
2020-05-10 12:56:36 +03:00
}
Py_DECREF(item);
2019-06-12 13:27:11 +03:00
coords[i] = coord * 65536;
2021-01-03 06:17:51 +03:00
}
2019-06-12 13:27:11 +03:00
error = FT_Set_Var_Design_Coordinates(self->face, num_coords, coords);
free(coords);
2020-05-10 12:56:36 +03:00
if (error) {
2019-06-12 13:27:11 +03:00
return geterror(error);
}
2021-01-03 06:17:51 +03:00
2019-06-12 13:27:11 +03:00
Py_INCREF(Py_None);
return Py_None;
2021-01-03 06:17:51 +03:00
}
2019-06-12 13:27:11 +03:00
#endif
2010-07-31 06:52:47 +04:00
static void
font_dealloc(FontObject *self) {
2016-05-30 13:56:28 +03:00
if (self->face) {
MUTEX_LOCK(&ft_library_mutex);
2016-05-30 13:56:28 +03:00
FT_Done_Face(self->face);
MUTEX_UNLOCK(&ft_library_mutex);
2016-05-30 13:56:28 +03:00
}
if (self->font_bytes) {
PyMem_Free(self->font_bytes);
2015-10-11 13:24:35 +03:00
}
2010-07-31 06:52:47 +04:00
PyObject_Del(self);
}
static PyMethodDef font_methods[] = {
{"render", (PyCFunction)font_render, METH_VARARGS},
{"getsize", (PyCFunction)font_getsize, METH_VARARGS},
{"getlength", (PyCFunction)font_getlength, METH_VARARGS},
2019-06-12 13:27:11 +03:00
#if FREETYPE_MAJOR > 2 || (FREETYPE_MAJOR == 2 && FREETYPE_MINOR > 9) || \
(FREETYPE_MAJOR == 2 && FREETYPE_MINOR == 9 && FREETYPE_PATCH == 1)
{"getvarnames", (PyCFunction)font_getvarnames, METH_NOARGS},
{"getvaraxes", (PyCFunction)font_getvaraxes, METH_NOARGS},
2019-06-12 13:27:11 +03:00
{"setvarname", (PyCFunction)font_setvarname, METH_VARARGS},
{"setvaraxes", (PyCFunction)font_setvaraxes, METH_VARARGS},
#endif
{NULL, NULL}
};
2010-07-31 06:52:47 +04:00
static PyObject *
font_getattr_family(FontObject *self, void *closure) {
2020-05-10 12:56:36 +03:00
if (self->face->family_name) {
return PyUnicode_FromString(self->face->family_name);
2020-05-10 12:56:36 +03:00
}
Py_RETURN_NONE;
}
2010-07-31 06:52:47 +04:00
static PyObject *
font_getattr_style(FontObject *self, void *closure) {
2020-05-10 12:56:36 +03:00
if (self->face->style_name) {
return PyUnicode_FromString(self->face->style_name);
2020-05-10 12:56:36 +03:00
}
Py_RETURN_NONE;
}
2010-07-31 06:52:47 +04:00
static PyObject *
font_getattr_ascent(FontObject *self, void *closure) {
2019-09-26 15:12:28 +03:00
return PyLong_FromLong(PIXEL(self->face->size->metrics.ascender));
}
2010-07-31 06:52:47 +04:00
static PyObject *
font_getattr_descent(FontObject *self, void *closure) {
2019-09-26 15:12:28 +03:00
return PyLong_FromLong(-PIXEL(self->face->size->metrics.descender));
}
2010-07-31 06:52:47 +04:00
2015-11-18 18:02:16 +03:00
static PyObject *
font_getattr_height(FontObject *self, void *closure) {
2019-09-26 15:12:28 +03:00
return PyLong_FromLong(PIXEL(self->face->size->metrics.height));
2015-11-18 18:02:16 +03:00
}
static PyObject *
font_getattr_x_ppem(FontObject *self, void *closure) {
2019-09-26 15:12:28 +03:00
return PyLong_FromLong(self->face->size->metrics.x_ppem);
2015-11-18 18:02:16 +03:00
}
static PyObject *
font_getattr_y_ppem(FontObject *self, void *closure) {
2019-09-26 15:12:28 +03:00
return PyLong_FromLong(self->face->size->metrics.y_ppem);
2015-11-18 18:02:16 +03:00
}
static PyObject *
font_getattr_glyphs(FontObject *self, void *closure) {
2019-09-26 15:12:28 +03:00
return PyLong_FromLong(self->face->num_glyphs);
2010-07-31 06:52:47 +04:00
}
static struct PyGetSetDef font_getsetters[] = {
{"family", (getter)font_getattr_family},
{"style", (getter)font_getattr_style},
{"ascent", (getter)font_getattr_ascent},
{"descent", (getter)font_getattr_descent},
2015-11-18 18:02:16 +03:00
{"height", (getter)font_getattr_height},
{"x_ppem", (getter)font_getattr_x_ppem},
{"y_ppem", (getter)font_getattr_y_ppem},
{"glyphs", (getter)font_getattr_glyphs},
{NULL}
};
static PyTypeObject Font_Type = {
PyVarObject_HEAD_INIT(NULL, 0) "Font", /*tp_name*/
sizeof(FontObject), /*tp_basicsize*/
0, /*tp_itemsize*/
2010-07-31 06:52:47 +04:00
/* methods */
(destructor)font_dealloc, /*tp_dealloc*/
0, /*tp_vectorcall_offset*/
0, /*tp_getattr*/
0, /*tp_setattr*/
0, /*tp_as_async*/
0, /*tp_repr*/
0, /*tp_as_number*/
0, /*tp_as_sequence*/
0, /*tp_as_mapping*/
0, /*tp_hash*/
0, /*tp_call*/
0, /*tp_str*/
0, /*tp_getattro*/
0, /*tp_setattro*/
0, /*tp_as_buffer*/
Py_TPFLAGS_DEFAULT, /*tp_flags*/
0, /*tp_doc*/
0, /*tp_traverse*/
0, /*tp_clear*/
0, /*tp_richcompare*/
0, /*tp_weaklistoffset*/
0, /*tp_iter*/
0, /*tp_iternext*/
font_methods, /*tp_methods*/
0, /*tp_members*/
font_getsetters, /*tp_getset*/
2010-07-31 06:52:47 +04:00
};
static PyMethodDef _functions[] = {
{"getfont", (PyCFunction)getfont, METH_VARARGS | METH_KEYWORDS}, {NULL, NULL}
};
2010-07-31 06:52:47 +04:00
static int
setup_module(PyObject *m) {
2010-07-31 06:52:47 +04:00
PyObject *d;
PyObject *v;
int major, minor, patch;
d = PyModule_GetDict(m);
/* Ready object type */
PyType_Ready(&Font_Type);
2010-07-31 06:52:47 +04:00
2020-05-10 12:56:36 +03:00
if (FT_Init_FreeType(&library)) {
2015-06-02 15:50:22 +03:00
return 0; /* leave it uninitialized */
2020-05-10 12:56:36 +03:00
}
2010-07-31 06:52:47 +04:00
FT_Library_Version(library, &major, &minor, &patch);
v = PyUnicode_FromFormat("%d.%d.%d", major, minor, patch);
2023-03-21 03:59:00 +03:00
PyDict_SetItemString(d, "freetype2_version", v ? v : Py_None);
2023-03-20 09:34:34 +03:00
Py_XDECREF(v);
#ifdef HAVE_RAQM
2020-11-25 19:27:12 +03:00
#if defined(HAVE_RAQM_SYSTEM) || defined(HAVE_FRIBIDI_SYSTEM)
have_raqm = 1;
#else
load_fribidi();
have_raqm = !!p_fribidi;
#endif
#else
have_raqm = 0;
#endif
2020-11-25 14:21:42 +03:00
/* if we have Raqm, we have all three (but possibly no version info) */
v = PyBool_FromLong(have_raqm);
PyDict_SetItemString(d, "HAVE_RAQM", v);
2020-11-25 14:21:42 +03:00
PyDict_SetItemString(d, "HAVE_FRIBIDI", v);
PyDict_SetItemString(d, "HAVE_HARFBUZZ", v);
2023-03-11 10:32:43 +03:00
Py_DECREF(v);
if (have_raqm) {
2023-03-21 03:59:00 +03:00
v = NULL;
#ifdef RAQM_VERSION_MAJOR
v = PyUnicode_FromString(raqm_version_string());
#endif
2023-03-21 03:59:00 +03:00
PyDict_SetItemString(d, "raqm_version", v ? v : Py_None);
2023-03-20 09:34:34 +03:00
Py_XDECREF(v);
2023-03-21 03:59:00 +03:00
v = NULL;
#ifdef FRIBIDI_MAJOR_VERSION
2021-03-25 02:25:38 +03:00
{
const char *a = strchr(fribidi_version_info, ')');
const char *b = strchr(fribidi_version_info, '\n');
if (a && b && a + 2 < b) {
v = PyUnicode_FromStringAndSize(a + 2, b - (a + 2));
}
}
#endif
2023-03-21 03:59:00 +03:00
PyDict_SetItemString(d, "fribidi_version", v ? v : Py_None);
Py_XDECREF(v);
2023-03-21 03:59:00 +03:00
v = NULL;
#ifdef HB_VERSION_STRING
v = PyUnicode_FromString(hb_version_string());
#endif
2023-03-21 03:59:00 +03:00
PyDict_SetItemString(d, "harfbuzz_version", v ? v : Py_None);
Py_XDECREF(v);
}
2016-12-15 13:43:02 +03:00
return 0;
2010-07-31 06:52:47 +04:00
}
PyMODINIT_FUNC
PyInit__imagingft(void) {
PyObject *m;
static PyModuleDef module_def = {
PyModuleDef_HEAD_INIT,
"_imagingft", /* m_name */
NULL, /* m_doc */
-1, /* m_size */
_functions, /* m_methods */
};
m = PyModule_Create(&module_def);
2020-05-10 12:56:36 +03:00
if (setup_module(m) < 0) {
return NULL;
2020-05-10 12:56:36 +03:00
}
#ifdef Py_GIL_DISABLED
PyUnstable_Module_SetGIL(m, Py_MOD_GIL_NOT_USED);
#endif
return m;
}