123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218 |
- #include <stdio.h>
- #include <stdlib.h>
- #include <string.h>
- #include <math.h>
- #include "gd.h"
- #include "gdhelpers.h"
- #ifndef MSWIN32
- #include <unistd.h>
- #else
- #include <io.h>
- #ifndef R_OK
- # define R_OK 04
- #endif
- #endif
- #ifdef WIN32
- #define access _access
- #ifndef R_OK
- #define R_OK 2
- #endif
- #endif
- #ifdef NUMCOLORS
- #undef NUMCOLORS
- #endif
- #define NUMCOLORS 8
- char *
- gdImageStringTTF (gdImage * im, int *brect, int fg, char *fontlist,
- double ptsize, double angle, int x, int y, char *string)
- {
-
- return gdImageStringFT (im, brect, fg, fontlist, ptsize, angle, x, y, string);
- }
- #ifndef HAVE_LIBFREETYPE
- char *
- gdImageStringFTEx (gdImage * im, int *brect, int fg, char *fontlist,
- double ptsize, double angle, int x, int y, char *string,
- gdFTStringExtraPtr strex)
- {
- return "libgd was not built with FreeType font support\n";
- }
- char *
- gdImageStringFT (gdImage * im, int *brect, int fg, char *fontlist,
- double ptsize, double angle, int x, int y, char *string)
- {
- return "libgd was not built with FreeType font support\n";
- }
- #else
- #include "gdcache.h"
- #include <ft2build.h>
- #include FT_FREETYPE_H
- #include FT_GLYPH_H
- #define FONTCACHESIZE 6
- #define TWEENCOLORCACHESIZE 32
- #define LINESPACE 1.05
- #define LISTSEPARATOR ";"
- #ifndef DEFAULT_FONTPATH
- #if defined(__APPLE__) || (defined(__MWERKS__) && defined(macintosh))
- #define DEFAULT_FONTPATH "/usr/share/fonts/truetype:/System/Library/Fonts:/Library/Fonts"
- #else
- #define DEFAULT_FONTPATH "/usr/share/fonts/truetype"
- #endif
- #endif
- #ifndef PATHSEPARATOR
- #define PATHSEPARATOR ":"
- #endif
- #ifndef TRUE
- #define FALSE 0
- #define TRUE !FALSE
- #endif
- #ifndef MAX
- #define MAX(a,b) ((a)>(b)?(a):(b))
- #endif
- #ifndef MIN
- #define MIN(a,b) ((a)<(b)?(a):(b))
- #endif
- typedef struct
- {
- char *fontlist;
- FT_Library *library;
- FT_Face face;
- FT_Bool have_char_map_unicode, have_char_map_big5, have_char_map_sjis, have_char_map_apple_roman;
- gdCache_head_t *glyphCache;
- } font_t;
- typedef struct
- {
- char *fontlist;
- int preferred_map;
- FT_Library *library;
- } fontkey_t;
- typedef struct
- {
- int pixel;
- int bgcolor;
- int fgcolor;
- gdImagePtr im;
- int tweencolor;
- } tweencolor_t;
- typedef struct
- {
- int pixel;
- int bgcolor;
- int fgcolor;
- gdImagePtr im;
- } tweencolorkey_t;
- #ifdef JISX0208
- #include "jisx0208.h"
- #endif
- extern int any2eucjp (char *, char *, unsigned int);
- gdMutexDeclare(gdFontCacheMutex);
- static gdCache_head_t *fontCache = NULL;
- static FT_Library library;
- #define Tcl_UniChar int
- #define TCL_UTF_MAX 3
- static int gdTcl_UtfToUniChar (char *str, Tcl_UniChar * chPtr)
- {
- int byte;
-
- byte = *((unsigned char *) str);
- if (byte == '&') {
- int i, n = 0;
- byte = *((unsigned char *) (str + 1));
- if (byte == '#') {
- byte = *((unsigned char *) (str + 2));
- if (byte == 'x' || byte == 'X') {
- for (i = 3; i < 8; i++) {
- byte = *((unsigned char *) (str + i));
- if (byte >= 'A' && byte <= 'F')
- byte = byte - 'A' + 10;
- else if (byte >= 'a' && byte <= 'f')
- byte = byte - 'a' + 10;
- else if (byte >= '0' && byte <= '9')
- byte = byte - '0';
- else
- break;
- n = (n * 16) + byte;
- }
- } else {
- for (i = 2; i < 8; i++) {
- byte = *((unsigned char *) (str + i));
- if (byte >= '0' && byte <= '9') {
- n = (n * 10) + (byte - '0');
- } else {
- break;
- }
- }
- }
- if (byte == ';') {
- *chPtr = (Tcl_UniChar) n;
- return ++i;
- }
- }
- }
-
- byte = *((unsigned char *) str);
- #ifdef JISX0208
- if (0xA1 <= byte && byte <= 0xFE) {
- int ku, ten;
- ku = (byte & 0x7F) - 0x20;
- ten = (str[1] & 0x7F) - 0x20;
- if ((ku < 1 || ku > 92) || (ten < 1 || ten > 94)) {
- *chPtr = (Tcl_UniChar) byte;
- return 1;
- }
- *chPtr = (Tcl_UniChar) UnicodeTbl[ku - 1][ten - 1];
- return 2;
- } else
- #endif
- if (byte < 0xC0) {
-
- *chPtr = (Tcl_UniChar) byte;
- return 1;
- } else if (byte < 0xE0) {
- if ((str[1] & 0xC0) == 0x80) {
-
- *chPtr = (Tcl_UniChar) (((byte & 0x1F) << 6) | (str[1] & 0x3F));
- return 2;
- }
-
- *chPtr = (Tcl_UniChar) byte;
- return 1;
- } else if (byte < 0xF0) {
- if (((str[1] & 0xC0) == 0x80) && ((str[2] & 0xC0) == 0x80)) {
-
- *chPtr = (Tcl_UniChar) (((byte & 0x0F) << 12) | ((str[1] & 0x3F) << 6) | (str[2] & 0x3F));
- return 3;
- }
-
- *chPtr = (Tcl_UniChar) byte;
- return 1;
- }
- #if TCL_UTF_MAX > 3
- else {
- int ch, total, trail;
- total = totalBytes[byte];
- trail = total - 1;
- if (trail > 0) {
- ch = byte & (0x3F >> trail);
- do {
- str++;
- if ((*str & 0xC0) != 0x80) {
- *chPtr = byte;
- return 1;
- }
- ch <<= 6;
- ch |= (*str & 0x3F);
- trail--;
- } while (trail > 0);
- *chPtr = ch;
- return total;
- }
- }
- #endif
- *chPtr = (Tcl_UniChar) byte;
- return 1;
- }
- static int fontTest (void *element, void *key)
- {
- font_t *a = (font_t *) element;
- fontkey_t *b = (fontkey_t *) key;
-
- if (strcmp (a->fontlist, b->fontlist) == 0) {
- switch (b->preferred_map) {
- case gdFTEX_Unicode:
- if (a->have_char_map_unicode) {
- return 1;
- }
- break;
- case gdFTEX_Shift_JIS:
- if (a->have_char_map_sjis) {
- return 1;
- }
- break;
- case gdFTEX_Big5:
- if (a->have_char_map_sjis) {
- return 1;
- }
- break;
- }
- }
- return 0;
- }
- static void *fontFetch (char **error, void *key)
- {
- font_t *a;
- fontkey_t *b = (fontkey_t *) key;
- int n;
- int font_found = 0;
- unsigned short platform, encoding;
- char *fontsearchpath, *fontlist;
- char fullname[MAXPATHLEN], cur_dir[MAXPATHLEN];
- char *name, *path=NULL, *dir;
- char *strtok_ptr;
- FT_Error err;
- FT_CharMap found = 0;
- FT_CharMap charmap;
- a = (font_t *) gdPMalloc(sizeof(font_t));
- a->fontlist = gdPEstrdup(b->fontlist);
- a->library = b->library;
-
- fontsearchpath = getenv ("GDFONTPATH");
- if (!fontsearchpath) {
- fontsearchpath = DEFAULT_FONTPATH;
- }
- fontlist = gdEstrdup(a->fontlist);
-
- for (name = gd_strtok_r (fontlist, LISTSEPARATOR, &strtok_ptr); name; name = gd_strtok_r (0, LISTSEPARATOR, &strtok_ptr)) {
- char *strtok_ptr_path;
-
- path = gdEstrdup (fontsearchpath);
-
- #ifdef NETWARE
- if (*name == '/' || (name[0] != 0 && strstr(name, ":/"))) {
- #else
- if (*name == '/' || (name[0] != 0 && name[1] == ':' && (name[2] == '/' || name[2] == '\\'))) {
- #endif
- snprintf(fullname, sizeof(fullname) - 1, "%s", name);
- if (access(fullname, R_OK) == 0) {
- font_found++;
- break;
- }
- }
- for (dir = gd_strtok_r (path, PATHSEPARATOR, &strtok_ptr_path); dir;
- dir = gd_strtok_r (0, PATHSEPARATOR, &strtok_ptr_path)) {
- if (!strcmp(dir, ".")) {
- TSRMLS_FETCH();
- #if HAVE_GETCWD
- dir = VCWD_GETCWD(cur_dir, MAXPATHLEN);
- #elif HAVE_GETWD
- dir = VCWD_GETWD(cur_dir);
- #endif
- if (!dir) {
- continue;
- }
- }
- #define GD_CHECK_FONT_PATH(ext) \
- snprintf(fullname, sizeof(fullname) - 1, "%s/%s%s", dir, name, ext); \
- if (access(fullname, R_OK) == 0) { \
- font_found++; \
- break; \
- } \
- GD_CHECK_FONT_PATH("");
- GD_CHECK_FONT_PATH(".ttf");
- GD_CHECK_FONT_PATH(".pfa");
- GD_CHECK_FONT_PATH(".pfb");
- GD_CHECK_FONT_PATH(".dfont");
- }
- gdFree(path);
- path = NULL;
- if (font_found) {
- break;
- }
- }
- if (path) {
- gdFree(path);
- }
- gdFree(fontlist);
- if (!font_found) {
- gdPFree(a->fontlist);
- gdPFree(a);
- *error = "Could not find/open font";
- return NULL;
- }
- err = FT_New_Face (*b->library, fullname, 0, &a->face);
- if (err) {
- gdPFree(a->fontlist);
- gdPFree(a);
- *error = "Could not read font";
- return NULL;
- }
-
-
- for (n = 0; n < a->face->num_charmaps; n++) {
- charmap = a->face->charmaps[n];
- platform = charmap->platform_id;
- encoding = charmap->encoding_id;
-
- a->have_char_map_unicode = 0;
- a->have_char_map_big5 = 0;
- a->have_char_map_sjis = 0;
- a->have_char_map_apple_roman = 0;
- #if (defined(FREETYPE_MAJOR) && ((FREETYPE_MAJOR == 2 && ((FREETYPE_MINOR == 1 && FREETYPE_PATCH >= 3) || FREETYPE_MINOR > 1) || FREETYPE_MAJOR > 2)))
- if (charmap->encoding == FT_ENCODING_MS_SYMBOL
- || charmap->encoding == FT_ENCODING_ADOBE_CUSTOM
- || charmap->encoding == FT_ENCODING_ADOBE_STANDARD) {
- a->have_char_map_unicode = 1;
- found = charmap;
- a->face->charmap = charmap;
- return (void *)a;
- }
- #endif
- if ((platform == 3 && encoding == 1)
- || (platform == 3 && encoding == 0)
- || (platform == 2 && encoding == 1)
- || (platform == 0))
- {
- a->have_char_map_unicode = 1;
- found = charmap;
- if (b->preferred_map == gdFTEX_Unicode) {
- break;
- }
- } else if (platform == 3 && encoding == 4) {
- a->have_char_map_big5 = 1;
- found = charmap;
- if (b->preferred_map == gdFTEX_Big5) {
- break;
- }
- } else if (platform == 3 && encoding == 2) {
- a->have_char_map_sjis = 1;
- found = charmap;
- if (b->preferred_map == gdFTEX_Shift_JIS) {
- break;
- }
- } else if ((platform == 1 && encoding == 0)
- || (platform == 2 && encoding == 0))
- {
- a->have_char_map_apple_roman = 1;
- found = charmap;
- if (b->preferred_map == gdFTEX_MacRoman) {
- break;
- }
- }
- }
- if (!found) {
- gdPFree(a->fontlist);
- gdPFree(a);
- *error = "Unable to find a CharMap that I can handle";
- return NULL;
- }
-
- a->face->charmap = found;
- return (void *) a;
- }
- static void fontRelease (void *element)
- {
- font_t *a = (font_t *) element;
- FT_Done_Face (a->face);
- gdPFree(a->fontlist);
- gdPFree((char *) element);
- }
- static int tweenColorTest (void *element, void *key)
- {
- tweencolor_t *a = (tweencolor_t *) element;
- tweencolorkey_t *b = (tweencolorkey_t *) key;
- return (a->pixel == b->pixel && a->bgcolor == b->bgcolor && a->fgcolor == b->fgcolor && a->im == b->im);
- }
- static void * tweenColorFetch (char **error, void *key)
- {
- tweencolor_t *a;
- tweencolorkey_t *b = (tweencolorkey_t *) key;
- int pixel, npixel, bg, fg;
- gdImagePtr im;
- a = (tweencolor_t *) gdMalloc (sizeof (tweencolor_t));
- pixel = a->pixel = b->pixel;
- bg = a->bgcolor = b->bgcolor;
- fg = a->fgcolor = b->fgcolor;
- im = a->im = b->im;
-
- if (fg < 0) {
- if ((pixel + pixel) >= NUMCOLORS) {
- a->tweencolor = -fg;
- } else {
- a->tweencolor = bg;
- }
- } else {
- npixel = NUMCOLORS - pixel;
- if (im->trueColor) {
-
- a->tweencolor = gdTrueColorAlpha(
- gdTrueColorGetRed(fg),
- gdTrueColorGetGreen(fg),
- gdTrueColorGetBlue(fg),
- gdAlphaMax - (gdTrueColorGetAlpha (fg) * pixel / NUMCOLORS));
- } else {
- a->tweencolor = gdImageColorResolve(im,
- (pixel * im->red[fg] + npixel * im->red[bg]) / NUMCOLORS,
- (pixel * im->green[fg] + npixel * im->green[bg]) / NUMCOLORS,
- (pixel * im->blue[fg] + npixel * im->blue[bg]) / NUMCOLORS);
- }
- }
- return (void *) a;
- }
- static void tweenColorRelease (void *element)
- {
- gdFree((char *) element);
- }
- static char * gdft_draw_bitmap (gdCache_head_t *tc_cache, gdImage * im, int fg, FT_Bitmap bitmap, int pen_x, int pen_y)
- {
- unsigned char *pixel = NULL;
- int *tpixel = NULL;
- int x, y, row, col, pc, pcr;
- tweencolor_t *tc_elem;
- tweencolorkey_t tc_key;
-
- tc_key.fgcolor = fg;
- tc_key.im = im;
-
- if (im->trueColor) {
- for (row = 0; row < bitmap.rows; row++) {
- pc = row * bitmap.pitch;
- pcr = pc;
- y = pen_y + row;
-
-
- if ((y > im->cy2) || (y < im->cy1)) {
- continue;
- }
- for (col = 0; col < bitmap.width; col++, pc++) {
- int level;
- if (bitmap.pixel_mode == ft_pixel_mode_grays) {
-
- level = (bitmap.buffer[pc] * gdAlphaMax / (bitmap.num_grays - 1));
- } else if (bitmap.pixel_mode == ft_pixel_mode_mono) {
-
- level = ((bitmap.buffer[(col>>3)+pcr]) & (1<<(~col&0x07))) ? gdAlphaTransparent : gdAlphaOpaque;
- } else {
- return "Unsupported ft_pixel_mode";
- }
- if (level == 0)
- continue;
- if ((fg >= 0) && (im->trueColor)) {
-
- level = level * (gdAlphaMax - gdTrueColorGetAlpha(fg)) / gdAlphaMax;
- }
- level = gdAlphaMax - level;
- x = pen_x + col;
-
-
- if ((x > im->cx2) || (x < im->cx1)) {
- continue;
- }
-
- tpixel = &im->tpixels[y][x];
- if (fg < 0) {
- if (level < (gdAlphaMax / 2)) {
- *tpixel = -fg;
- }
- } else {
- if (im->alphaBlendingFlag) {
- *tpixel = gdAlphaBlend(*tpixel, (level << 24) + (fg & 0xFFFFFF));
- } else {
- *tpixel = (level << 24) + (fg & 0xFFFFFF);
- }
- }
- }
- }
- return (char *) NULL;
- }
-
- for (row = 0; row < bitmap.rows; row++) {
- int pcr;
- pc = row * bitmap.pitch;
- pcr = pc;
- if (bitmap.pixel_mode==ft_pixel_mode_mono) {
- pc *= 8;
- }
- y = pen_y + row;
-
- if (y >= im->sy || y < 0) {
- continue;
- }
- for (col = 0; col < bitmap.width; col++, pc++) {
- if (bitmap.pixel_mode == ft_pixel_mode_grays) {
-
- tc_key.pixel = ((bitmap.buffer[pc] * NUMCOLORS) + bitmap.num_grays / 2) / (bitmap.num_grays - 1);
- } else if (bitmap.pixel_mode == ft_pixel_mode_mono) {
- tc_key.pixel = ((bitmap.buffer[pc / 8] << (pc % 8)) & 128) ? NUMCOLORS : 0;
-
- tc_key.pixel = ((bitmap.buffer[(col>>3)+pcr]) & (1<<(~col&0x07))) ? NUMCOLORS : 0;
- } else {
- return "Unsupported ft_pixel_mode";
- }
- if (tc_key.pixel > 0) {
- x = pen_x + col;
-
- if (x >= im->sx || x < 0) {
- continue;
- }
-
- pixel = &im->pixels[y][x];
- if (tc_key.pixel == NUMCOLORS) {
-
- *pixel = (fg < 0) ? -fg : fg;
- } else {
-
- tc_key.bgcolor = *pixel;
- tc_elem = (tweencolor_t *) gdCacheGet(tc_cache, &tc_key);
- *pixel = tc_elem->tweencolor;
- }
- }
- }
- }
- return (char *) NULL;
- }
- static int
- gdroundupdown (FT_F26Dot6 v1, int roundup)
- {
- return (!roundup) ? v1 >> 6 : (v1 + 63) >> 6;
- }
- void gdFontCacheShutdown()
- {
- gdMutexLock(gdFontCacheMutex);
- if (fontCache) {
- gdCacheDelete(fontCache);
- fontCache = NULL;
- FT_Done_FreeType(library);
- }
- gdMutexUnlock(gdFontCacheMutex);
- }
- void gdFreeFontCache()
- {
- gdFontCacheShutdown();
- }
- void gdFontCacheMutexSetup()
- {
- gdMutexSetup(gdFontCacheMutex);
- }
- void gdFontCacheMutexShutdown()
- {
- gdMutexShutdown(gdFontCacheMutex);
- }
- int gdFontCacheSetup(void)
- {
- if (fontCache) {
-
- return 0;
- }
- if (FT_Init_FreeType(&library)) {
- return -1;
- }
- fontCache = gdCacheCreate (FONTCACHESIZE, fontTest, fontFetch, fontRelease);
- return 0;
- }
- char *
- gdImageStringFT (gdImage * im, int *brect, int fg, char *fontlist,
- double ptsize, double angle, int x, int y, char *string)
- {
- return gdImageStringFTEx(im, brect, fg, fontlist, ptsize, angle, x, y, string, 0);
- }
- char *
- gdImageStringFTEx (gdImage * im, int *brect, int fg, char *fontlist, double ptsize, double angle, int x, int y, char *string, gdFTStringExtraPtr strex)
- {
- FT_BBox bbox, glyph_bbox;
- FT_Matrix matrix;
- FT_Vector pen, delta, penf;
- FT_Face face;
- FT_Glyph image;
- FT_GlyphSlot slot;
- FT_Bool use_kerning;
- FT_UInt glyph_index, previous;
- double sin_a = sin (angle);
- double cos_a = cos (angle);
- int len, i = 0, ch;
- int x1 = 0, y1 = 0;
- font_t *font;
- fontkey_t fontkey;
- char *next;
- char *tmpstr = NULL;
- int render = (im && (im->trueColor || (fg <= 255 && fg >= -255)));
- FT_BitmapGlyph bm;
-
- int render_mode = FT_LOAD_DEFAULT;
- int m, mfound;
-
- double linespace = LINESPACE;
-
-
- gdCache_head_t *tc_cache;
-
- int hdpi, vdpi;
- if (strex && ((strex->flags & gdFTEX_LINESPACE) == gdFTEX_LINESPACE)) {
- linespace = strex->linespacing;
- }
- tc_cache = gdCacheCreate(TWEENCOLORCACHESIZE, tweenColorTest, tweenColorFetch, tweenColorRelease);
-
- gdMutexLock(gdFontCacheMutex);
- if (!fontCache) {
- if (gdFontCacheSetup() != 0) {
- gdCacheDelete(tc_cache);
- gdMutexUnlock(gdFontCacheMutex);
- return "Failure to initialize font library";
- }
- }
-
-
- m = gdFTEX_Unicode;
- if (strex && (strex->flags & gdFTEX_CHARMAP)) {
- m = strex->charmap;
- }
-
- fontkey.fontlist = fontlist;
- fontkey.library = &library;
- fontkey.preferred_map = m;
- font = (font_t *) gdCacheGet (fontCache, &fontkey);
- if (!font) {
- gdCacheDelete(tc_cache);
- gdMutexUnlock(gdFontCacheMutex);
- return fontCache->error;
- }
- face = font->face;
- slot = face->glyph;
-
- hdpi = GD_RESOLUTION;
- vdpi = GD_RESOLUTION;
- if (strex && (strex->flags & gdFTEX_RESOLUTION)) {
- hdpi = strex->hdpi;
- vdpi = strex->vdpi;
- }
- if (FT_Set_Char_Size(face, 0, (FT_F26Dot6) (ptsize * 64), hdpi, vdpi)) {
- gdCacheDelete(tc_cache);
- gdMutexUnlock(gdFontCacheMutex);
- return "Could not set character size";
- }
- matrix.xx = (FT_Fixed) (cos_a * (1 << 16));
- matrix.yx = (FT_Fixed) (sin_a * (1 << 16));
- matrix.xy = -matrix.yx;
- matrix.yy = matrix.xx;
- penf.x = penf.y = 0;
- pen.x = pen.y = 0;
- bbox.xMin = bbox.xMax = bbox.yMin = bbox.yMax = 0;
- use_kerning = FT_HAS_KERNING (face);
- previous = 0;
- if (fg < 0) {
- render_mode |= FT_LOAD_MONOCHROME;
- }
-
- mfound = 0;
- for (i = 0; i < 3; i++) {
- switch (m) {
- case gdFTEX_Unicode:
- if (font->have_char_map_unicode) {
- mfound = 1;
- }
- break;
- case gdFTEX_Shift_JIS:
- if (font->have_char_map_sjis) {
- mfound = 1;
- }
- break;
- case gdFTEX_Big5:
-
- mfound = 1;
- break;
- }
- if (mfound) {
- break;
- }
- m++;
- m %= 3;
- }
- if (!mfound) {
-
- gdMutexUnlock(gdFontCacheMutex);
- return "No character set found";
- }
- #ifndef JISX0208
- if (font->have_char_map_sjis) {
- #endif
- tmpstr = (char *) gdMalloc(BUFSIZ);
- any2eucjp(tmpstr, string, BUFSIZ);
- next = tmpstr;
- #ifndef JISX0208
- } else {
- next = string;
- }
- #endif
- i = 0;
- while (*next) {
- ch = *next;
-
- if (ch == '\r') {
- penf.x = 0;
- x1 = (int)(- penf.y * sin_a + 32) / 64;
- y1 = (int)(- penf.y * cos_a + 32) / 64;
- pen.x = pen.y = 0;
- previous = 0;
- next++;
- continue;
- }
-
- if (ch == '\n') {
- if (!*(++next)) break;
-
- penf.x = 0;
- penf.y -= (long)(face->size->metrics.height * linespace);
- penf.y = (penf.y - 32) & -64;
- x1 = (int)(- penf.y * sin_a + 32) / 64;
- y1 = (int)(- penf.y * cos_a + 32) / 64;
- pen.x = pen.y = 0;
- previous = 0;
- continue;
- }
- #if (defined(FREETYPE_MAJOR) && ((FREETYPE_MAJOR == 2 && ((FREETYPE_MINOR == 1 && FREETYPE_PATCH >= 3) || FREETYPE_MINOR > 1) || FREETYPE_MAJOR > 2)))
- if (font->face->family_name && font->face->charmap->encoding &&
- font->face->charmap->encoding == FT_ENCODING_MS_SYMBOL && strcmp(font->face->family_name, "Symbol") == 0) {
-
- len = gdTcl_UtfToUniChar (next, &ch);
- ch |= 0xf000;
- next += len;
- } else
- #endif
- switch (m) {
- case gdFTEX_Unicode:
- if (font->have_char_map_unicode) {
-
- len = gdTcl_UtfToUniChar(next, &ch);
- next += len;
- }
- break;
- case gdFTEX_Shift_JIS:
- if (font->have_char_map_sjis) {
- unsigned char c;
- int jiscode;
- c = *next;
- if (0xA1 <= c && c <= 0xFE) {
- next++;
- jiscode = 0x100 * (c & 0x7F) + ((*next) & 0x7F);
- ch = (jiscode >> 8) & 0xFF;
- jiscode &= 0xFF;
- if (ch & 1) {
- jiscode += 0x40 - 0x21;
- } else {
- jiscode += 0x9E - 0x21;
- }
- if (jiscode >= 0x7F) {
- jiscode++;
- }
- ch = (ch - 0x21) / 2 + 0x81;
- if (ch >= 0xA0) {
- ch += 0x40;
- }
- ch = (ch << 8) + jiscode;
- } else {
- ch = c & 0xFF;
- }
- if (*next) next++;
- }
- break;
- case gdFTEX_Big5: {
-
- ch = (*next) & 0xFF;
- next++;
- if (ch >= 161
- && *next) {
-
- ch = (ch * 256) + ((*next) & 255);
- next++;
- }
- }
- break;
- }
-
- FT_Set_Transform(face, &matrix, NULL);
-
- glyph_index = FT_Get_Char_Index(face, ch);
-
- if (use_kerning && previous && glyph_index) {
- FT_Get_Kerning(face, previous, glyph_index, ft_kerning_default, &delta);
- pen.x += (int)(delta.x * cos_a);
- pen.y -= (int)(delta.x * sin_a);
- penf.x += delta.x;
- }
- if (brect) {
-
- if (FT_Load_Glyph(face, glyph_index, render_mode | FT_LOAD_IGNORE_TRANSFORM)) {
- if (tmpstr) {
- gdFree(tmpstr);
- }
- gdCacheDelete(tc_cache);
- gdMutexUnlock(gdFontCacheMutex);
- return "Problem loading glyph";
- }
-
- if (FT_Get_Glyph(slot, &image)) {
- if (tmpstr) {
- gdFree(tmpstr);
- }
- gdCacheDelete(tc_cache);
- gdMutexUnlock(gdFontCacheMutex);
- return "Problem loading glyph";
- }
- FT_Glyph_Get_CBox(image, ft_glyph_bbox_gridfit, &glyph_bbox);
- glyph_bbox.xMin += penf.x;
- glyph_bbox.yMin += penf.y;
- glyph_bbox.xMax += penf.x;
- glyph_bbox.yMax += penf.y;
- if (ch == ' ') {
- glyph_bbox.xMax += slot->metrics.horiAdvance;
- }
- if (!i) {
- bbox.xMin = glyph_bbox.xMin;
- bbox.yMin = glyph_bbox.yMin;
- bbox.xMax = glyph_bbox.xMax;
- bbox.yMax = glyph_bbox.yMax;
- } else {
- if (bbox.xMin > glyph_bbox.xMin) {
- bbox.xMin = glyph_bbox.xMin;
- }
- if (bbox.yMin > glyph_bbox.yMin) {
- bbox.yMin = glyph_bbox.yMin;
- }
- if (bbox.xMax < glyph_bbox.xMax) {
- bbox.xMax = glyph_bbox.xMax;
- }
- if (bbox.yMax < glyph_bbox.yMax) {
- bbox.yMax = glyph_bbox.yMax;
- }
- }
- i++;
- }
-
- penf.x += slot->metrics.horiAdvance;
- if (render) {
- if (!brect || angle != 0) {
-
- FT_Done_Glyph(image);
-
- if (FT_Load_Glyph(face, glyph_index, render_mode)) {
- if (tmpstr) {
- gdFree(tmpstr);
- }
- gdCacheDelete(tc_cache);
- gdMutexUnlock(gdFontCacheMutex);
- return "Problem loading glyph";
- }
-
- if (FT_Get_Glyph(slot, &image)) {
- if (tmpstr) {
- gdFree(tmpstr);
- }
- gdCacheDelete(tc_cache);
- gdMutexUnlock(gdFontCacheMutex);
- return "Problem loading glyph";
- }
- }
- if (image->format != ft_glyph_format_bitmap && FT_Glyph_To_Bitmap(&image, ft_render_mode_normal, 0, 1)) {
- FT_Done_Glyph(image);
- if (tmpstr) {
- gdFree(tmpstr);
- }
- gdCacheDelete(tc_cache);
- gdMutexUnlock(gdFontCacheMutex);
- return "Problem rendering glyph";
- }
-
- bm = (FT_BitmapGlyph) image;
- gdft_draw_bitmap(tc_cache, im, fg, bm->bitmap, x + x1 + ((pen.x + 31) >> 6) + bm->left, y + y1 + ((pen.y + 31) >> 6) - bm->top);
- }
-
- previous = glyph_index;
-
- pen.x += image->advance.x >> 10;
- pen.y -= image->advance.y >> 10;
- FT_Done_Glyph(image);
- }
- if (brect) {
-
- double d1 = sin (angle + 0.78539816339744830962);
- double d2 = sin (angle - 0.78539816339744830962);
-
- brect[0] = (int) (bbox.xMin * cos_a - bbox.yMin * sin_a);
- brect[1] = (int) (bbox.xMin * sin_a + bbox.yMin * cos_a);
- brect[2] = (int) (bbox.xMax * cos_a - bbox.yMin * sin_a);
- brect[3] = (int) (bbox.xMax * sin_a + bbox.yMin * cos_a);
- brect[4] = (int) (bbox.xMax * cos_a - bbox.yMax * sin_a);
- brect[5] = (int) (bbox.xMax * sin_a + bbox.yMax * cos_a);
- brect[6] = (int) (bbox.xMin * cos_a - bbox.yMax * sin_a);
- brect[7] = (int) (bbox.xMin * sin_a + bbox.yMax * cos_a);
-
- brect[0] = x + gdroundupdown(brect[0], d2 > 0);
- brect[1] = y - gdroundupdown(brect[1], d1 < 0);
- brect[2] = x + gdroundupdown(brect[2], d1 > 0);
- brect[3] = y - gdroundupdown(brect[3], d2 > 0);
- brect[4] = x + gdroundupdown(brect[4], d2 < 0);
- brect[5] = y - gdroundupdown(brect[5], d1 > 0);
- brect[6] = x + gdroundupdown(brect[6], d1 < 0);
- brect[7] = y - gdroundupdown(brect[7], d2 < 0);
- }
- if (tmpstr) {
- gdFree(tmpstr);
- }
- gdCacheDelete(tc_cache);
- gdMutexUnlock(gdFontCacheMutex);
- return (char *) NULL;
- }
- #endif
|