#include "fitz.h" #include #include FT_FREETYPE_H #if ((FREETYPE_MAJOR == 2) && (FREETYPE_MINOR == 1)) || \ ((FREETYPE_MAJOR == 2) && (FREETYPE_MINOR == 2)) || \ ((FREETYPE_MAJOR == 2) && (FREETYPE_MINOR == 3) && (FREETYPE_PATCH < 8)) int FT_Get_Advance(FT_Face face, int gid, int masks, FT_Fixed *out) { int fterr; fterr = FT_Load_Glyph(face, gid, masks | FT_LOAD_IGNORE_TRANSFORM); if (fterr) return fterr; *out = face->glyph->advance.x * 1024; return 0; } #else #include FT_ADVANCES_H #endif typedef struct fz_textdevice_s fz_textdevice; struct fz_textdevice_s { fz_point point; fz_textspan *head; fz_textspan *span; }; fz_textspan * fz_newtextspan(void) { fz_textspan *span; span = fz_malloc(sizeof(fz_textspan)); span->font = nil; span->size = 0.0; span->len = 0; span->cap = 0; span->text = nil; span->next = nil; span->eol = 0; return span; } void fz_freetextspan(fz_textspan *span) { if (span->font) fz_dropfont(span->font); if (span->next) fz_freetextspan(span->next); fz_free(span->text); fz_free(span); } static void fz_addtextcharimp(fz_textspan *span, int c, fz_bbox bbox) { if (span->len + 1 >= span->cap) { span->cap = span->cap ? (span->cap * 3) / 2 : 80; span->text = fz_realloc(span->text, sizeof(fz_textchar) * span->cap); } span->text[span->len].c = c; span->text[span->len].bbox = bbox; span->len ++; } static fz_bbox fz_splitbbox(fz_bbox bbox, int i, int n) { int w = bbox.x1 - bbox.x0; bbox.x0 = bbox.x0 + w * i / n; bbox.x1 = bbox.x0 + w * (i + 1) / n; return bbox; } static void fz_addtextchar(fz_textspan **last, fz_font *font, float size, int c, fz_bbox bbox) { fz_textspan *span = *last; if (!span->font) { span->font = fz_keepfont(font); span->size = size; } if (span->font != font || span->size != size) { span = fz_newtextspan(); span->font = fz_keepfont(font); span->size = size; (*last)->next = span; *last = span; } switch (c) { case -1: /* ignore when one unicode character maps to multiple glyphs */ break; case 0xFB00: /* ff */ fz_addtextcharimp(span, 'f', fz_splitbbox(bbox, 0, 2)); fz_addtextcharimp(span, 'f', fz_splitbbox(bbox, 1, 2)); break; case 0xFB01: /* fi */ fz_addtextcharimp(span, 'f', fz_splitbbox(bbox, 0, 2)); fz_addtextcharimp(span, 'i', fz_splitbbox(bbox, 1, 2)); break; case 0xFB02: /* fl */ fz_addtextcharimp(span, 'f', fz_splitbbox(bbox, 0, 2)); fz_addtextcharimp(span, 'l', fz_splitbbox(bbox, 1, 2)); break; case 0xFB03: /* ffi */ fz_addtextcharimp(span, 'f', fz_splitbbox(bbox, 0, 3)); fz_addtextcharimp(span, 'f', fz_splitbbox(bbox, 1, 3)); fz_addtextcharimp(span, 'i', fz_splitbbox(bbox, 2, 3)); break; case 0xFB04: /* ffl */ fz_addtextcharimp(span, 'f', fz_splitbbox(bbox, 0, 3)); fz_addtextcharimp(span, 'f', fz_splitbbox(bbox, 1, 3)); fz_addtextcharimp(span, 'l', fz_splitbbox(bbox, 2, 3)); break; case 0xFB05: /* long st */ case 0xFB06: /* st */ fz_addtextcharimp(span, 's', fz_splitbbox(bbox, 0, 2)); fz_addtextcharimp(span, 't', fz_splitbbox(bbox, 1, 2)); break; default: fz_addtextcharimp(span, c, bbox); break; } } static void fz_addtextnewline(fz_textspan **last, fz_font *font, float size) { fz_textspan *span; span = fz_newtextspan(); span->font = fz_keepfont(font); span->size = size; (*last)->eol = 1; (*last)->next = span; *last = span; } void fz_debugtextspanxml(fz_textspan *span) { char buf[10]; int c, n, k, i; printf("\n", span->font ? span->font->name : "NULL", span->size, span->eol); for (i = 0; i < span->len; i++) { printf("\ttext[i].c; if (c < 128) putchar(c); else { n = runetochar(buf, &c); for (k = 0; k < n; k++) putchar(buf[k]); } printf("\" bbox=\"[%d %d %d %d]\">\n", span->text[i].bbox.x0, span->text[i].bbox.y0, span->text[i].bbox.x1, span->text[i].bbox.y1); } printf("\n"); if (span->next) fz_debugtextspanxml(span->next); } void fz_debugtextspan(fz_textspan *span) { char buf[10]; int c, n, k, i; for (i = 0; i < span->len; i++) { c = span->text[i].c; if (c < 128) putchar(c); else { n = runetochar(buf, &c); for (k = 0; k < n; k++) putchar(buf[k]); } } if (span->eol) putchar('\n'); if (span->next) fz_debugtextspan(span->next); } static void fz_textextractspan(fz_textspan **last, fz_text *text, fz_matrix ctm, fz_point *pen) { fz_font *font = text->font; fz_matrix tm = text->trm; fz_matrix inv = fz_invertmatrix(text->trm); float size = fz_matrixexpansion(text->trm); fz_matrix trm; float dx, dy; fz_rect rect; fz_point p; float adv; int i, fterr; if (text->len == 0) return; if (font->ftface) { FT_Set_Transform(font->ftface, NULL, NULL); fterr = FT_Set_Char_Size(font->ftface, 64, 64, 72, 72); if (fterr) fz_warn("freetype set character size: %s", ft_errorstring(fterr)); } rect = fz_emptyrect; for (i = 0; i < text->len; i++) { if (text->els[i].gid < 0) { /* TODO: split rect for one-to-many mapped chars */ fz_addtextchar(last, font, size, text->els[i].ucs, fz_roundrect(rect)); continue; } /* Get point in user space to perform heuristic space and newspan tests */ p.x = text->els[i].x; p.y = text->els[i].y; p = fz_transformpoint(inv, p); dx = pen->x - p.x; dy = pen->y - p.y; if (pen->x == -1 && pen->y == -1) dx = dy = 0; *pen = p; /* Get advance width and update pen position */ if (font->ftface) { FT_Fixed ftadv = 0; fterr = FT_Get_Advance(font->ftface, text->els[i].gid, FT_LOAD_NO_BITMAP | FT_LOAD_NO_HINTING, &ftadv); if (fterr) fz_warn("freetype get advance (gid %d): %s", text->els[i].gid, ft_errorstring(fterr)); adv = ftadv / 65536.0; pen->x += adv; } else { adv = font->t3widths[text->els[i].gid]; pen->x += adv; } /* Get bbox in device space */ tm.e = text->els[i].x; tm.f = text->els[i].y; trm = fz_concat(tm, ctm); rect.x0 = 0.0; rect.y0 = 0.0; rect.x1 = adv; rect.y1 = 1.0; rect = fz_transformrect(trm, rect); /* Add to the text span */ if (fabs(dy) > 0.001) { fz_addtextnewline(last, font, size); } else if (fabs(dx) > 0.2) { fz_rect spacerect; spacerect.x0 = -fabs(dx); spacerect.y0 = 0.0; spacerect.x1 = 0.0; spacerect.y1 = 1.0; spacerect = fz_transformrect(trm, spacerect); fz_addtextchar(last, font, size, ' ', fz_roundrect(spacerect)); } fz_addtextchar(last, font, size, text->els[i].ucs, fz_roundrect(rect)); } } static void fz_textfilltext(void *user, fz_text *text, fz_matrix ctm, fz_colorspace *colorspace, float *color, float alpha) { fz_textdevice *tdev = user; fz_textextractspan(&tdev->span, text, ctm, &tdev->point); } static void fz_textstroketext(void *user, fz_text *text, fz_strokestate *stroke, fz_matrix ctm, fz_colorspace *colorspace, float *color, float alpha) { fz_textdevice *tdev = user; fz_textextractspan(&tdev->span, text, ctm, &tdev->point); } static void fz_textcliptext(void *user, fz_text *text, fz_matrix ctm, int accumulate) { fz_textdevice *tdev = user; fz_textextractspan(&tdev->span, text, ctm, &tdev->point); } static void fz_textclipstroketext(void *user, fz_text *text, fz_strokestate *stroke, fz_matrix ctm) { fz_textdevice *tdev = user; fz_textextractspan(&tdev->span, text, ctm, &tdev->point); } static void fz_textignoretext(void *user, fz_text *text, fz_matrix ctm) { fz_textdevice *tdev = user; fz_textextractspan(&tdev->span, text, ctm, &tdev->point); } static void fz_textfreeuser(void *user) { fz_textdevice *tdev = user; tdev->span->eol = 1; /* TODO: unicode NFC normalization */ /* TODO: bidi logical reordering */ fz_free(tdev); } fz_device * fz_newtextdevice(fz_textspan *root) { fz_device *dev; fz_textdevice *tdev = fz_malloc(sizeof(fz_textdevice)); tdev->head = root; tdev->span = root; tdev->point.x = -1; tdev->point.y = -1; dev = fz_newdevice(tdev); dev->freeuser = fz_textfreeuser; dev->filltext = fz_textfilltext; dev->stroketext = fz_textstroketext; dev->cliptext = fz_textcliptext; dev->clipstroketext = fz_textclipstroketext; dev->ignoretext = fz_textignoretext; return dev; }