10#include "private/qfontdatabase_p.h"
11#include "private/qimage_p.h"
12#include <private/qstringiterator_p.h>
13#include <qguiapplication.h>
15#include <qpa/qplatformscreen.h>
16#include <QtCore/QUuid>
17#include <QtCore/QLoggingCategory>
18#include <QtGui/QPainterPath>
24#include <qscopedvaluerollback.h>
28#include <private/qcolrpaintgraphrenderer_p.h>
35#include FT_SYNTHESIS_H
36#include FT_TRUETYPE_TABLES_H
37#include FT_TYPE1_TABLES_H
40#include FT_LCD_FILTER_H
41#include FT_MULTIPLE_MASTERS_H
43#if defined(FT_CONFIG_OPTIONS_H)
44#include FT_CONFIG_OPTIONS_H
47#if defined(FT_FONT_FORMATS_H)
48#include FT_FONT_FORMATS_H
57using namespace Qt::StringLiterals;
59#define FLOOR(x) ((x) & -64
)
60#define CEIL(x) (((x)+63
) & -64
)
61#define TRUNC(x) ((x) >> 6
)
62#define ROUND(x) (((x)+32
) & -64
)
64static bool ft_getSfntTable(
void *user_data, uint tag, uchar *buffer, uint *length)
66 FT_Face face = (FT_Face)user_data;
69 if (FT_IS_SFNT(face)) {
70 FT_ULong len = *length;
71 result = FT_Load_Sfnt_Table(face, tag, 0, buffer, &len) == FT_Err_Ok;
73 Q_ASSERT(!result ||
int(*length) > 0);
83 QFontEngineFT::HintFull;
85 QFontEngineFT::HintNone;
116 for (
auto iter = faces.cbegin(); iter != faces.cend(); ++iter) {
117 iter.value()->cleanup();
118 if (!iter.value()->ref.deref())
123 for (
auto iter = staleFaces.cbegin(); iter != staleFaces.cend(); ++iter) {
125 if (!(*iter)->ref.deref())
130 FT_Done_FreeType(library);
136 return style1.faceFileName == style2.faceFileName && style1.styleName == style2.styleName;
141 return qHashMulti(seed, style.faceFileName, style.styleName);
151 if (!freetypeData->library) {
152 FT_Init_FreeType(&freetypeData->library);
153#if defined(FT_FONT_FORMATS_H)
155 FT_Bool no_darkening =
false;
156 FT_Property_Set(freetypeData->library,
"cff",
"no-stem-darkening", &no_darkening);
165 Q_ASSERT(freetypeData->library);
166 return freetypeData->library;
169int QFreetypeFace::fsType()
const
172 TT_OS2 *os2 = (TT_OS2 *)FT_Get_Sfnt_Table(face, ft_sfnt_os2);
174 fsType = os2->fsType;
178int QFreetypeFace::getPointInOutline(glyph_t glyph,
int flags, quint32 point, QFixed *xpos, QFixed *ypos, quint32 *nPoints)
180 if (
int error = FT_Load_Glyph(face, glyph, flags))
183 if (face->glyph->format != FT_GLYPH_FORMAT_OUTLINE)
184 return Err_Invalid_SubTable;
186 *nPoints = face->glyph->outline.n_points;
190 if (point > *nPoints)
191 return Err_Invalid_SubTable;
193 *xpos = QFixed::fromFixed(face->glyph->outline.points[point].x);
194 *ypos = QFixed::fromFixed(face->glyph->outline.points[point].y);
199bool QFreetypeFace::isScalableBitmap()
const
202 return !FT_IS_SCALABLE(face) && FT_HAS_COLOR(face);
211
212
213
214
215
216
217QFreetypeFace *QFreetypeFace::getFace(
const QFontEngine::FaceId &face_id,
218 const QByteArray &fontData)
220 if (face_id.filename.isEmpty() && fontData.isEmpty())
223 QtFreetypeData *freetypeData = qt_getFreetypeData();
226 for (
auto it = freetypeData->staleFaces.constBegin(); it != freetypeData->staleFaces.constEnd(); ) {
227 if ((*it)->ref.loadRelaxed() == 1) {
230 it = freetypeData->staleFaces.erase(it);
236 QFreetypeFace *freetype =
nullptr;
237 auto it = freetypeData->faces.find(face_id);
238 if (it != freetypeData->faces.end()) {
241 Q_ASSERT(freetype->ref.loadRelaxed() > 0);
242 if (freetype->ref.loadRelaxed() == 1) {
249 freetypeData->faces.erase(it);
258 const auto deleter = [](QFreetypeFace *f) {
delete f; };
259 std::unique_ptr<QFreetypeFace,
decltype(deleter)> newFreetype(
new QFreetypeFace, deleter);
262 if (!face_id.filename.isEmpty()) {
263 QString fileName = QFile::decodeName(face_id.filename);
264 if (
const char *prefix =
":qmemoryfonts/"; face_id.filename.startsWith(prefix)) {
266 QByteArray idx = face_id.filename;
267 idx.remove(0, strlen(prefix));
269 newFreetype->fontData = qt_fontdata_from_index(idx.toInt(&ok));
271 newFreetype->fontData = QByteArray();
272 }
else if (!QFileInfo(fileName).isNativePath()) {
273 QFile file(fileName);
274 if (!file.open(QIODevice::ReadOnly)) {
277 newFreetype->fontData = file.readAll();
280 newFreetype->fontData = fontData;
283 FT_Int major, minor, patch;
284 FT_Library_Version(qt_getFreetype(), &major, &minor, &patch);
285 const bool goodVersion = major > 2 || (major == 2 && minor > 13) || (major == 2 && minor == 13 && patch > 2);
287 if (!newFreetype->fontData.isEmpty()) {
288 if (FT_New_Memory_Face(freetypeData->library,
289 (
const FT_Byte *)newFreetype->fontData.constData(),
290 newFreetype->fontData.size(),
300 if (FT_Reference_Face(face))
302 }
else if (!FT_HAS_MULTIPLE_MASTERS(face)
303 || FT_New_Memory_Face(freetypeData->library,
304 (
const FT_Byte *)newFreetype->fontData.constData(),
305 newFreetype->fontData.size(),
307 &tmpFace) != FT_Err_Ok) {
311 if (FT_New_Face(freetypeData->library, face_id.filename, face_id.index, &face))
318 if (FT_Reference_Face(face))
320 }
else if (!FT_HAS_MULTIPLE_MASTERS(face)
321 || FT_New_Face(freetypeData->library, face_id.filename, face_id.index, &tmpFace) != FT_Err_Ok) {
326#if (FREETYPE_MAJOR*10000
+ FREETYPE_MINOR*100
+ FREETYPE_PATCH) >= 20900
327 if (face_id.instanceIndex >= 0) {
329 <<
"Selecting named instance" << (face_id.instanceIndex)
330 <<
"in" << face_id.filename;
331 FT_Set_Named_Instance(face, face_id.instanceIndex + 1);
339 if (tmpFace !=
nullptr) {
341 if (FT_Get_MM_Var(tmpFace, &var) == FT_Err_Ok) {
342 for (FT_UInt i = 0; i < var->num_axis; ++i) {
343 FT_Var_Axis *axis = var->axis + i;
345 QFontVariableAxis fontVariableAxis;
346 if (
const auto tag = QFont::Tag::fromValue(axis->tag)) {
347 fontVariableAxis.setTag(*tag);
349 qWarning() <<
"QFreetypeFace::getFace: Invalid variable axis tag encountered"
353 fontVariableAxis.setMinimumValue(axis->minimum / 65536.0);
354 fontVariableAxis.setMaximumValue(axis->maximum / 65536.0);
355 fontVariableAxis.setDefaultValue(axis->def / 65536.0);
356 fontVariableAxis.setName(QString::fromUtf8(axis->name));
358 newFreetype->variableAxisList.append(fontVariableAxis);
361 if (!face_id.variableAxes.isEmpty()) {
362 QVarLengthArray<FT_Fixed, 16> coords(var->num_axis);
363 FT_Get_Var_Design_Coordinates(face, var->num_axis, coords.data());
364 for (qsizetype i = 0; i < newFreetype->variableAxisList.size(); ++i) {
365 const QFontVariableAxis &axis = newFreetype->variableAxisList.at(i);
366 if (axis.tag().isValid()) {
367 const auto it = face_id.variableAxes.constFind(axis.tag());
368 if (it != face_id.variableAxes.constEnd())
369 coords[i] = FT_Fixed(*it * 65536);
372 FT_Set_Var_Design_Coordinates(face, var->num_axis, coords.data());
375 FT_Done_MM_Var(qt_getFreetype(), var);
378 FT_Done_Face(tmpFace);
381 newFreetype->face = face;
382 newFreetype->mm_var =
nullptr;
383 if (FT_IS_NAMED_INSTANCE(newFreetype->face)) {
385 ftresult = FT_Get_MM_Var(face, &newFreetype->mm_var);
386 if (ftresult != FT_Err_Ok)
387 newFreetype->mm_var =
nullptr;
390 newFreetype->ref.storeRelaxed(1);
391 newFreetype->xsize = 0;
392 newFreetype->ysize = 0;
393 newFreetype->matrix.xx = 0x10000;
394 newFreetype->matrix.yy = 0x10000;
395 newFreetype->matrix.xy = 0;
396 newFreetype->matrix.yx = 0;
397 newFreetype->unicode_map =
nullptr;
398 newFreetype->symbol_map =
nullptr;
400 memset(newFreetype->cmapCache, 0,
sizeof(newFreetype->cmapCache));
402 for (
int i = 0; i < newFreetype->face->num_charmaps; ++i) {
403 FT_CharMap cm = newFreetype->face->charmaps[i];
404 switch(cm->encoding) {
405 case FT_ENCODING_UNICODE:
406 newFreetype->unicode_map = cm;
408 case FT_ENCODING_APPLE_ROMAN:
409 case FT_ENCODING_ADOBE_LATIN_1:
410 if (!newFreetype->unicode_map || newFreetype->unicode_map->encoding != FT_ENCODING_UNICODE)
411 newFreetype->unicode_map = cm;
413 case FT_ENCODING_ADOBE_CUSTOM:
414 case FT_ENCODING_MS_SYMBOL:
415 if (!newFreetype->symbol_map)
416 newFreetype->symbol_map = cm;
423 if (!FT_IS_SCALABLE(newFreetype->face) && newFreetype->face->num_fixed_sizes == 1)
424 FT_Set_Char_Size(face, newFreetype->face->available_sizes[0].x_ppem, newFreetype->face->available_sizes[0].y_ppem, 0, 0);
426 FT_Set_Charmap(newFreetype->face, newFreetype->unicode_map);
429 freetypeData->faces.insert(face_id, newFreetype.get());
431 newFreetype.release()->release(face_id);
435 freetype = newFreetype.release();
441void QFreetypeFace::cleanup()
445 FT_Done_MM_Var(qt_getFreetype(), mm_var);
451void QFreetypeFace::release(
const QFontEngine::FaceId &face_id)
454 bool deleteThis = !ref.deref();
460 if (face && ref.loadRelaxed() == 1) {
461 QtFreetypeData *freetypeData = qt_getFreetypeData();
463 for (
auto it = freetypeData->staleFaces.constBegin(); it != freetypeData->staleFaces.constEnd(); ) {
464 if ((*it)->ref.loadRelaxed() == 1) {
470 it = freetypeData->staleFaces.erase(it);
476 for (
auto it = freetypeData->faces.constBegin();
477 it != freetypeData->faces.constEnd();
478 it = freetypeData->faces.erase(it)) {
479 if (it.value()->ref.loadRelaxed() == 1) {
480 it.value()->cleanup();
481 if (it.value() ==
this)
486 freetypeData->staleFaces.append(it.value());
490 if (freetypeData->faces.isEmpty() && freetypeData->staleFaces.isEmpty()) {
491 FT_Done_FreeType(freetypeData->library);
492 freetypeData->library =
nullptr;
502 FT_Library library = qt_getFreetype();
510 FT_Error error = FT_New_Face(library, faceFileName.toUtf8().constData(), faceIndex, &face);
511 if (error != FT_Err_Ok) {
512 qDebug() <<
"FT_New_Face failed for face index" << faceIndex <<
':' << Qt::hex << error;
516 const bool found = QLatin1StringView(face->style_name) == styleName;
517 numFaces = face->num_faces;
523 }
while (++faceIndex < numFaces);
529int QFreetypeFace::getFaceIndexByStyleName(
const QString &faceFileName,
const QString &styleName)
531 QtFreetypeData *freetypeData = qt_getFreetypeData();
534 QtFreetypeData::FaceStyle faceStyle(faceFileName, styleName);
535 int faceIndex = freetypeData->faceIndices.value(faceStyle, -1);
540 faceIndex = computeFaceIndex(faceFileName, styleName);
542 freetypeData->faceIndices.insert(faceStyle, faceIndex);
547void QFreetypeFace::computeSize(
const QFontDef &fontDef,
int *xsize,
int *ysize,
bool *outline_drawing, QFixed *scalableBitmapScaleFactor)
549 *ysize = qRound(fontDef.pixelSize * 64);
550 *xsize = *ysize * fontDef.stretch / 100;
551 *scalableBitmapScaleFactor = 1;
552 *outline_drawing =
false;
554 if (!(face->face_flags & FT_FACE_FLAG_SCALABLE)) {
556 if (!isScalableBitmap()) {
558
559
560
561 for (
int i = 1; i < face->num_fixed_sizes; i++) {
562 if (qAbs(*ysize - face->available_sizes[i].y_ppem) <
563 qAbs(*ysize - face->available_sizes[best].y_ppem) ||
564 (qAbs(*ysize - face->available_sizes[i].y_ppem) ==
565 qAbs(*ysize - face->available_sizes[best].y_ppem) &&
566 qAbs(*xsize - face->available_sizes[i].x_ppem) <
567 qAbs(*xsize - face->available_sizes[best].x_ppem))) {
573 for (
int i = 1; i < face->num_fixed_sizes; i++) {
574 if (face->available_sizes[i].y_ppem < *ysize) {
575 if (face->available_sizes[i].y_ppem > face->available_sizes[best].y_ppem)
577 }
else if (face->available_sizes[best].y_ppem < *ysize) {
579 }
else if (face->available_sizes[i].y_ppem < face->available_sizes[best].y_ppem) {
587 if (FT_Select_Size(face, best) == 0) {
588 if (isScalableBitmap())
589 *scalableBitmapScaleFactor = QFixed::fromReal((qreal)fontDef.pixelSize / face->available_sizes[best].height);
590 *xsize = face->available_sizes[best].x_ppem;
591 *ysize = face->available_sizes[best].y_ppem;
596#if defined FT_HAS_COLOR
597 if (FT_HAS_COLOR(face)) {
598 *outline_drawing =
false;
602 int maxCachedGlyphSize = QFontEngine::maxCachedGlyphSize();
603 *outline_drawing = (*xsize > (maxCachedGlyphSize << 6) || *ysize > (maxCachedGlyphSize << 6));
608QFontEngine::Properties QFreetypeFace::properties()
const
610 QFontEngine::Properties p;
611 p.postscriptName = FT_Get_Postscript_Name(face);
612 PS_FontInfoRec font_info;
613 if (FT_Get_PS_Font_Info(face, &font_info) == 0)
614 p.copyright = font_info.notice;
615 if (FT_IS_SCALABLE(face)
616#if defined(FT_HAS_COLOR)
617 && !FT_HAS_COLOR(face)
620 p.ascent = face->ascender;
621 p.descent = -face->descender;
622 p.leading = face->height - face->ascender + face->descender;
623 p.emSquare = face->units_per_EM;
624 p.boundingBox = QRectF(face->bbox.xMin, -face->bbox.yMax,
625 face->bbox.xMax - face->bbox.xMin,
626 face->bbox.yMax - face->bbox.yMin);
628 p.ascent = QFixed::fromFixed(face->size->metrics.ascender);
629 p.descent = QFixed::fromFixed(-face->size->metrics.descender);
630 p.leading = QFixed::fromFixed(face->size->metrics.height - face->size->metrics.ascender + face->size->metrics.descender);
631 p.emSquare = face->size->metrics.y_ppem;
633 p.boundingBox = QRectF(0, -p.ascent.toReal(),
634 face->size->metrics.max_advance/64, (p.ascent + p.descent).toReal() );
637 p.capHeight = p.ascent;
638 p.lineWidth = face->underline_thickness;
643bool QFreetypeFace::getSfntTable(uint tag, uchar *buffer, uint *length)
const
645 return ft_getSfntTable(face, tag, buffer, length);
649
650
651
652
653
654
655
656
657static void scaleOutline(FT_Face face, FT_GlyphSlot g, FT_Fixed x_scale, FT_Fixed y_scale)
659 x_scale = FT_MulDiv(x_scale, 1 << 10, face->units_per_EM);
660 y_scale = FT_MulDiv(y_scale, 1 << 10, face->units_per_EM);
661 FT_Vector *p = g->outline.points;
662 const FT_Vector *e = p + g->outline.n_points;
664 p->x = FT_MulFix(p->x, x_scale);
665 p->y = FT_MulFix(p->y, y_scale);
670#define GLYPH2PATH_DEBUG QT_NO_QDEBUG_MACRO
671void QFreetypeFace::addGlyphToPath(FT_Face face, FT_GlyphSlot g,
const QFixedPoint &point, QPainterPath *path, FT_Fixed x_scale, FT_Fixed y_scale)
673 const qreal factor = 1/64.;
674 scaleOutline(face, g, x_scale, y_scale);
676 QPointF cp = point.toPointF();
680 for (
int j = 0; j < g->outline.n_contours; ++j) {
681 int last_point = g->outline.contours[j];
683 QPointF start = QPointF(g->outline.points[i].x*factor, -g->outline.points[i].y*factor);
684 if (!(g->outline.tags[i] & 1)) {
685 if (!(g->outline.tags[last_point] & 1)) {
687 start = (QPointF(g->outline.points[last_point].x*factor,
688 -g->outline.points[last_point].y*factor) + start) / 2.0;
691 start = QPointF(g->outline.points[last_point].x*factor,
692 -g->outline.points[last_point].y*factor);
703 while (i < last_point) {
705 c[n] = cp + QPointF(g->outline.points[i].x*factor, -g->outline.points[i].y*factor);
706 GLYPH2PATH_DEBUG() <<
" " << i << c[n] <<
"tag =" << (
int)g->outline.tags[i]
707 <<
": on curve =" << (
bool)(g->outline.tags[i] & 1);
709 switch (g->outline.tags[i] & 3) {
714 c[3] = (c[3] + c[2])/2;
721 c[3] = (c[1] + c[2])/2;
722 c[2] = (2*c[1] + c[3])/3;
723 c[1] = (2*c[1] + c[0])/3;
736 c[2] = (2*c[1] + c[3])/3;
737 c[1] = (2*c[1] + c[0])/3;
742 path->cubicTo(c[1], c[2], c[3]);
749 path->closeSubpath();
753 c[2] = (2*c[1] + c[3])/3;
754 c[1] = (2*c[1] + c[0])/3;
757 path->cubicTo(c[1], c[2], c[3]);
763extern void qt_addBitmapToPath(qreal x0, qreal y0,
const uchar *image_data,
int bpl,
int w,
int h, QPainterPath *path);
765void QFreetypeFace::addBitmapToPath(FT_GlyphSlot slot,
const QFixedPoint &point, QPainterPath *path)
767 if (slot->format != FT_GLYPH_FORMAT_BITMAP
768 || slot->bitmap.pixel_mode != FT_PIXEL_MODE_MONO)
771 QPointF cp = point.toPointF();
772 qt_addBitmapToPath(cp.x() +
TRUNC(slot->metrics.horiBearingX), cp.y() -
TRUNC(slot->metrics.horiBearingY),
773 slot->bitmap.buffer, slot->bitmap.pitch, slot->bitmap.width, slot->bitmap.rows, path);
776static inline void convertRGBToARGB(
const uchar *src, uint *dst,
int width,
int height,
int src_pitch,
bool bgr)
778 const int offs = bgr ? -1 : 1;
779 const int w = width * 3;
782 for (
int x = 0; x < w; x += 3) {
783 uchar red = src[x + 1 - offs];
784 uchar green = src[x + 1];
785 uchar blue = src[x + 1 + offs];
786 *dd++ = (0xFFU << 24) | (red << 16) | (green << 8) | blue;
793static inline void convertRGBToARGB_V(
const uchar *src, uint *dst,
int width,
int height,
int src_pitch,
bool bgr)
795 const int offs = bgr ? -src_pitch : src_pitch;
797 for (
int x = 0; x < width; x++) {
798 uchar red = src[x + src_pitch - offs];
799 uchar green = src[x + src_pitch];
800 uchar blue = src[x + src_pitch + offs];
801 *dst++ = (0XFFU << 24) | (red << 16) | (green << 8) | blue;
809 static int type = -1;
811 if (QScreen *screen = QGuiApplication::primaryScreen())
812 type = screen->handle()->subpixelAntialiasingTypeHint();
814 return static_cast<QFontEngine::SubpixelAntialiasingType>(type);
817QFontEngineFT *QFontEngineFT::create(
const QFontDef &fontDef, FaceId faceId,
const QByteArray &fontData)
819 auto engine = std::make_unique<QFontEngineFT>(fontDef);
821 QFontEngineFT::GlyphFormat format = QFontEngineFT::Format_Mono;
822 const bool antialias = !(fontDef.styleStrategy & QFont::NoAntialias);
825 QFontEngine::SubpixelAntialiasingType subpixelType = subpixelAntialiasingTypeHint();
826 if (subpixelType == QFontEngine::Subpixel_None || (fontDef.styleStrategy & QFont::NoSubpixelAntialias)) {
827 format = QFontEngineFT::Format_A8;
828 engine->subpixelType = QFontEngine::Subpixel_None;
830 format = QFontEngineFT::Format_A32;
831 engine->subpixelType = subpixelType;
835 if (!engine->init(faceId, antialias, format, fontData) || engine->invalid()) {
836 qWarning(
"QFontEngineFT: Failed to create FreeType font engine");
840 engine->setQtDefaultHintStyle(
static_cast<QFont::HintingPreference>(fontDef.hintingPreference));
841 return engine.release();
846 FT_MM_Var *var = freetypeFace->mm_var;
847 if (var !=
nullptr && faceId.instanceIndex >= 0 && FT_UInt(faceId.instanceIndex) < var->num_namedstyles) {
848 for (FT_UInt axis = 0; axis < var->num_axis; ++axis) {
849 if (var->axis[axis].tag == QFont::Tag(
"wght").value()) {
850 return var->namedstyle[faceId.instanceIndex].coords[axis] >> 16;
854 if (
const TT_OS2 *os2 =
reinterpret_cast<
const TT_OS2 *>(FT_Get_Sfnt_Table(face, ft_sfnt_os2))) {
855 return os2->usWeightClass;
862 class QFontEngineFTRawData:
public QFontEngineFT
865 QFontEngineFTRawData(
const QFontDef &fontDef) : QFontEngineFT(fontDef)
869 void updateFamilyNameAndStyle()
871 fontDef.families = QStringList(QString::fromLatin1(freetype->face->family_name));
873 if (freetype->face->style_flags & FT_STYLE_FLAG_ITALIC)
874 fontDef.style = QFont::StyleItalic;
876 if (freetype->face->style_flags & FT_STYLE_FLAG_BOLD)
877 fontDef.weight = QFont::Bold;
879 fontDef.weight = calculateActualWeight(freetype, freetype->face, faceId());
883 const QMap<QFont::Tag,
float> &variableAxisValues,
887 faceId.filename =
"";
889 faceId.uuid = QUuid::createUuid().toByteArray();
890 faceId.variableAxes = variableAxisValues;
891 faceId.instanceIndex = instanceIndex;
893 return init(faceId,
true, Format_None, fontData);
898QFontEngineFT *QFontEngineFT::create(
const QByteArray &fontData,
900 QFont::HintingPreference hintingPreference,
901 const QMap<QFont::Tag,
float> &variableAxisValues,
905 fontDef.pixelSize = pixelSize;
906 fontDef.stretch = QFont::Unstretched;
907 fontDef.hintingPreference = hintingPreference;
908 fontDef.variableAxisValues = variableAxisValues;
910 QFontEngineFTRawData *fe =
new QFontEngineFTRawData(fontDef);
911 if (!fe->initFromData(fontData, variableAxisValues, instanceIndex)) {
916 fe->updateFamilyNameAndStyle();
917 fe->setQtDefaultHintStyle(
static_cast<QFont::HintingPreference>(fontDef.hintingPreference));
922QFontEngineFT::QFontEngineFT(
const QFontDef &fd)
923 : QFontEngine(Freetype)
930 cache_cost = 100 * 1024;
931 kerning_pairs_loaded =
false;
937 default_load_flags = FT_LOAD_IGNORE_GLOBAL_ADVANCE_WIDTH;
938 default_hint_style = ftInitialDefaultHintStyle;
939 subpixelType = Subpixel_None;
940 lcdFilterType = (
int)((quintptr) FT_LCD_FILTER_DEFAULT);
941 defaultFormat = Format_None;
942 embeddedbitmap =
false;
943 const QByteArray env = qgetenv(
"QT_NO_FT_CACHE");
944 cacheEnabled = env.isEmpty() || env.toInt() == 0;
945 m_subPixelPositionCount = 4;
946 forceAutoHint =
false;
947 stemDarkeningDriver =
false;
950QFontEngineFT::~QFontEngineFT()
953 freetype->release(face_id);
956bool QFontEngineFT::init(FaceId faceId,
bool antialias, GlyphFormat format,
957 const QByteArray &fontData)
959 return init(faceId, antialias, format, QFreetypeFace::getFace(faceId, fontData));
966 FT_MM_Var *var = freetypeFace->mm_var;
967 if (var !=
nullptr && faceId.instanceIndex >= 0 && FT_UInt(faceId.instanceIndex) < var->num_namedstyles) {
968 for (FT_UInt axis = 0; axis < var->num_axis; ++axis) {
969 if (var->axis[axis].tag == QFont::Tag(
"ital").value()) {
970 return (var->namedstyle[faceId.instanceIndex].coords[axis] >> 16) == 1;
975 return (face->style_flags & FT_STYLE_FLAG_ITALIC);
978bool QFontEngineFT::init(FaceId faceId,
bool antialias, GlyphFormat format,
979 QFreetypeFace *freetypeFace)
981 freetype = freetypeFace;
987 defaultFormat = format;
988 this->antialias = antialias;
991 glyphFormat = QFontEngine::Format_Mono;
993 glyphFormat = defaultFormat;
997 symbol = freetype->symbol_map !=
nullptr;
998 PS_FontInfoRec psrec;
1000 if (FT_Get_PS_Font_Info(freetype->face, &psrec) == FT_Err_Ok) {
1001 symbol = !fontDef.families.isEmpty() &&
bool(fontDef.families.constFirst().contains(
"symbol"_L1, Qt::CaseInsensitive));
1004 freetype->computeSize(fontDef, &xsize, &ysize, &defaultGlyphSet.outline_drawing, &scalableBitmapScaleFactor);
1006 FT_Face face = lockFace();
1008 if (FT_IS_SCALABLE(face)
1009#if defined(FT_HAS_COLOR)
1010 && !FT_HAS_COLOR(face)
1013 bool isItalic = calculateActualItalic(freetype, face, faceId);
1014 bool fake_oblique = (fontDef.style != QFont::StyleNormal) && !isItalic && !qEnvironmentVariableIsSet(
"QT_NO_SYNTHESIZED_ITALIC");
1017 FT_Set_Transform(face, &matrix,
nullptr);
1018 freetype->matrix = matrix;
1020 if ((fontDef.weight >= QFont::Bold) && !(face->style_flags & FT_STYLE_FLAG_BOLD) && !FT_IS_FIXED_WIDTH(face) && !qEnvironmentVariableIsSet(
"QT_NO_SYNTHESIZED_BOLD")) {
1021 FT_UShort actualWeight = calculateActualWeight(freetype, face, faceId);
1022 if (actualWeight < 700 &&
1023 (fontDef.pixelSize < 64 || qEnvironmentVariableIsSet(
"QT_NO_SYNTHESIZED_BOLD_LIMIT"))) {
1028 line_thickness = QFixed::fromFixed(FT_MulFix(face->underline_thickness, face->size->metrics.y_scale));
1029 QFixed center_position = QFixed::fromFixed(-FT_MulFix(face->underline_position, face->size->metrics.y_scale));
1030 underline_position = center_position - line_thickness / 2;
1033 int score = fontDef.weight * fontDef.pixelSize;
1034 line_thickness = score / 7000;
1036 if (line_thickness < 2 && score >= 1050)
1038 underline_position = ((line_thickness * 2) + 3) / 6;
1040 cacheEnabled =
false;
1041#if defined(FT_HAS_COLOR)
1042 if (FT_HAS_COLOR(face))
1043 glyphFormat = defaultFormat = GlyphFormat::Format_ARGB;
1046 if (line_thickness < 1)
1049 metrics = face->size->metrics;
1052
1053
1054
1055
1056
1057
1058 if (FT_IS_SCALABLE(face)) {
1059 for (
int i = 0; i < face->num_fixed_sizes; ++i) {
1060 if (xsize == face->available_sizes[i].x_ppem && ysize == face->available_sizes[i].y_ppem) {
1061 face->face_flags &= ~FT_FACE_FLAG_SCALABLE;
1063 FT_Select_Size(face, i);
1064 if (face->size->metrics.ascender + face->size->metrics.descender > 0) {
1065 FT_Pos leading = metrics.height - metrics.ascender + metrics.descender;
1066 metrics.ascender = face->size->metrics.ascender;
1067 metrics.descender = face->size->metrics.descender;
1068 if (metrics.descender > 0
1069 && QString::fromUtf8(face->family_name) ==
"Courier New"_L1) {
1070 metrics.descender *= -1;
1072 metrics.height = metrics.ascender - metrics.descender + leading;
1074 FT_Set_Char_Size(face, xsize, ysize, 0, 0);
1076 face->face_flags |= FT_FACE_FLAG_SCALABLE;
1081#if defined(FT_FONT_FORMATS_H)
1082 const char *fmt = FT_Get_Font_Format(face);
1083 if (fmt && qstrncmp(fmt,
"CFF", 4) == 0) {
1084 FT_Bool no_stem_darkening =
true;
1085 FT_Error err = FT_Property_Get(qt_getFreetype(),
"cff",
"no-stem-darkening", &no_stem_darkening);
1086 if (err == FT_Err_Ok)
1087 stemDarkeningDriver = !no_stem_darkening;
1089 stemDarkeningDriver =
false;
1093 fontDef.styleName = QString::fromUtf8(face->style_name);
1095 if (!freetype->hbFace) {
1096 faceData.user_data = face;
1097 faceData.get_font_table = ft_getSfntTable;
1098 (
void)harfbuzzFace();
1099 freetype->hbFace = std::move(face_);
1104 face_ = Holder(freetype->hbFace.get(), dont_delete);
1108 fsType = freetype->fsType();
1112void QFontEngineFT::setQtDefaultHintStyle(QFont::HintingPreference hintingPreference)
1114 switch (hintingPreference) {
1115 case QFont::PreferNoHinting:
1116 setDefaultHintStyle(HintNone);
1118 case QFont::PreferFullHinting:
1119 setDefaultHintStyle(HintFull);
1121 case QFont::PreferVerticalHinting:
1122 setDefaultHintStyle(HintLight);
1124 case QFont::PreferDefaultHinting:
1125 setDefaultHintStyle(ftInitialDefaultHintStyle);
1130void QFontEngineFT::setDefaultHintStyle(HintStyle style)
1132 default_hint_style = style;
1135bool QFontEngineFT::expectsGammaCorrectedBlending(QFontEngine::GlyphFormat format)
const
1138 return stemDarkeningDriver;
1141int QFontEngineFT::loadFlags(QGlyphSet *set, GlyphFormat format,
int flags,
1142 bool &hsubpixel,
int &vfactor)
const
1144 int load_flags = FT_LOAD_DEFAULT | default_load_flags;
1145 int load_target = default_hint_style == HintLight
1146 ? FT_LOAD_TARGET_LIGHT
1147 : FT_LOAD_TARGET_NORMAL;
1149 if (format == Format_Mono) {
1150 load_target = FT_LOAD_TARGET_MONO;
1151 }
else if (format == Format_A32) {
1152 if (subpixelType == Subpixel_RGB || subpixelType == Subpixel_BGR)
1154 else if (subpixelType == Subpixel_VRGB || subpixelType == Subpixel_VBGR)
1156 }
else if (format == Format_ARGB) {
1158 load_flags |= FT_LOAD_COLOR;
1162 if (set && set->outline_drawing)
1163 load_flags |= FT_LOAD_NO_BITMAP;
1165 if (default_hint_style == HintNone || (flags & DesignMetrics) || (set && set->outline_drawing))
1166 load_flags |= FT_LOAD_NO_HINTING;
1168 load_flags |= load_target;
1171 load_flags |= FT_LOAD_FORCE_AUTOHINT;
1179 return info.width > 0xFF || info.height > 0xFF || info.linearAdvance > 0x7FFF;
1188 FT_Vector_Transform(&vector, matrix);
1193 FT_Vector_Transform(&vector, matrix);
1194 if (l > vector.x) l = vector.x;
1195 if (r < vector.x) r = vector.x;
1196 if (t < vector.y) t = vector.y;
1197 if (b > vector.y) b = vector.y;
1200 FT_Vector_Transform(&vector, matrix);
1201 if (l > vector.x) l = vector.x;
1202 if (r < vector.x) r = vector.x;
1203 if (t < vector.y) t = vector.y;
1204 if (b > vector.y) b = vector.y;
1207 FT_Vector_Transform(&vector, matrix);
1208 if (l > vector.x) l = vector.x;
1209 if (r < vector.x) r = vector.x;
1210 if (t < vector.y) t = vector.y;
1211 if (b > vector.y) b = vector.y;
1218#if defined(QFONTENGINE_FT_SUPPORT_COLRV1)
1219#define FROM_FIXED_16_16(value) (value / 65536.0
)
1221static inline QTransform FTAffineToQTransform(
const FT_Affine23 &matrix)
1223 qreal m11 = FROM_FIXED_16_16(matrix.xx);
1224 qreal m21 = -FROM_FIXED_16_16(matrix.xy);
1225 qreal m12 = -FROM_FIXED_16_16(matrix.yx);
1226 qreal m22 = FROM_FIXED_16_16(matrix.yy);
1227 qreal dx = FROM_FIXED_16_16(matrix.dx);
1228 qreal dy = -FROM_FIXED_16_16(matrix.dy);
1230 return QTransform(m11, m12, m21, m22, dx, dy);
1233bool QFontEngineFT::traverseColr1(FT_OpaquePaint opaquePaint,
1234 QSet<std::pair<FT_Byte *, FT_Bool> > *loops,
1235 QColor foregroundColor,
1237 ushort paletteCount,
1238 QColrPaintGraphRenderer *paintGraphRenderer)
const
1240 FT_Face face = freetype->face;
1242 auto key = std::pair{opaquePaint.p, opaquePaint.insert_root_transform};
1243 if (loops->contains(key)) {
1244 qCWarning(lcColrv1) <<
"Cycle detected in COLRv1 graph";
1248 paintGraphRenderer->save();
1250 auto cleanup = qScopeGuard([&paintGraphRenderer, &key, &loops]() {
1252 paintGraphRenderer->restore();
1255 FT_COLR_Paint paint;
1256 if (!FT_Get_Paint(face, opaquePaint, &paint))
1259 if (paint.format == FT_COLR_PAINTFORMAT_COLR_LAYERS) {
1260 FT_OpaquePaint layerPaint;
1261 layerPaint.p =
nullptr;
1262 while (FT_Get_Paint_Layers(face, &paint.u.colr_layers.layer_iterator, &layerPaint)) {
1263 if (!traverseColr1(layerPaint, loops, foregroundColor, palette, paletteCount, paintGraphRenderer))
1266 }
else if (paint.format == FT_COLR_PAINTFORMAT_TRANSFORM
1267 || paint.format == FT_COLR_PAINTFORMAT_SCALE
1268 || paint.format == FT_COLR_PAINTFORMAT_TRANSLATE
1269 || paint.format == FT_COLR_PAINTFORMAT_ROTATE
1270 || paint.format == FT_COLR_PAINTFORMAT_SKEW) {
1273 FT_OpaquePaint nextPaint;
1274 switch (paint.format) {
1275 case FT_COLR_PAINTFORMAT_TRANSFORM:
1276 xform = FTAffineToQTransform(paint.u.transform.affine);
1277 nextPaint = paint.u.transform.paint;
1279 case FT_COLR_PAINTFORMAT_SCALE:
1281 qreal centerX = FROM_FIXED_16_16(paint.u.scale.center_x);
1282 qreal centerY = -FROM_FIXED_16_16(paint.u.scale.center_y);
1283 qreal scaleX = FROM_FIXED_16_16(paint.u.scale.scale_x);
1284 qreal scaleY = FROM_FIXED_16_16(paint.u.scale.scale_y);
1286 xform.translate(centerX, centerY);
1287 xform.scale(scaleX, scaleY);
1288 xform.translate(-centerX, -centerY);
1290 nextPaint = paint.u.scale.paint;
1293 case FT_COLR_PAINTFORMAT_ROTATE:
1295 qreal centerX = FROM_FIXED_16_16(paint.u.rotate.center_x);
1296 qreal centerY = -FROM_FIXED_16_16(paint.u.rotate.center_y);
1297 qreal angle = -FROM_FIXED_16_16(paint.u.rotate.angle) * 180.0;
1299 xform.translate(centerX, centerY);
1300 xform.rotate(angle);
1301 xform.translate(-centerX, -centerY);
1303 nextPaint = paint.u.rotate.paint;
1307 case FT_COLR_PAINTFORMAT_SKEW:
1309 qreal centerX = FROM_FIXED_16_16(paint.u.skew.center_x);
1310 qreal centerY = -FROM_FIXED_16_16(paint.u.skew.center_y);
1311 qreal angleX = FROM_FIXED_16_16(paint.u.skew.x_skew_angle) * M_PI;
1312 qreal angleY = -FROM_FIXED_16_16(paint.u.skew.y_skew_angle) * M_PI;
1314 xform.translate(centerX, centerY);
1315 xform.shear(qTan(angleX), qTan(angleY));
1316 xform.translate(-centerX, -centerY);
1318 nextPaint = paint.u.rotate.paint;
1321 case FT_COLR_PAINTFORMAT_TRANSLATE:
1323 qreal dx = FROM_FIXED_16_16(paint.u.translate.dx);
1324 qreal dy = -FROM_FIXED_16_16(paint.u.translate.dy);
1326 xform.translate(dx, dy);
1327 nextPaint = paint.u.rotate.paint;
1334 paintGraphRenderer->prependTransform(xform);
1335 if (!traverseColr1(nextPaint, loops, foregroundColor, palette, paletteCount, paintGraphRenderer))
1337 }
else if (paint.format == FT_COLR_PAINTFORMAT_LINEAR_GRADIENT
1338 || paint.format == FT_COLR_PAINTFORMAT_RADIAL_GRADIENT
1339 || paint.format == FT_COLR_PAINTFORMAT_SWEEP_GRADIENT
1340 || paint.format == FT_COLR_PAINTFORMAT_SOLID) {
1341 auto getPaletteColor = [&palette, &paletteCount, &foregroundColor](FT_UInt16 index,
1344 if (index < paletteCount) {
1345 const FT_Color &paletteColor = palette[index];
1346 color = qRgba(paletteColor.red,
1349 paletteColor.alpha);
1350 }
else if (index == 0xffff) {
1351 color = foregroundColor;
1354 if (color.isValid())
1355 color.setAlphaF(color.alphaF() * (alpha / 16384.0));
1360 auto gatherGradientStops = [&](FT_ColorStopIterator it) {
1362 ret.resize(it.num_color_stops);
1364 FT_ColorStop colorStop;
1365 while (FT_Get_Colorline_Stops(face, &colorStop, &it)) {
1366 uint index = it.current_color_stop - 1;
1367 if (qsizetype(index) < ret.size()) {
1368 QGradientStop &gradientStop = ret[index];
1369 gradientStop.first = FROM_FIXED_16_16(colorStop.stop_offset);
1370 gradientStop.second = getPaletteColor(colorStop.color.palette_index,
1371 colorStop.color.alpha);
1378 auto extendToSpread = [](FT_PaintExtend extend) {
1380 case FT_COLR_PAINT_EXTEND_REPEAT:
1381 return QGradient::RepeatSpread;
1382 case FT_COLR_PAINT_EXTEND_REFLECT:
1383 return QGradient::ReflectSpread;
1385 return QGradient::PadSpread;
1389 if (paintGraphRenderer->isRendering()) {
1390 if (paint.format == FT_COLR_PAINTFORMAT_LINEAR_GRADIENT) {
1391 const qreal p0x = FROM_FIXED_16_16(paint.u.linear_gradient.p0.x);
1392 const qreal p0y = -FROM_FIXED_16_16(paint.u.linear_gradient.p0.y);
1394 const qreal p1x = FROM_FIXED_16_16(paint.u.linear_gradient.p1.x);
1395 const qreal p1y = -FROM_FIXED_16_16(paint.u.linear_gradient.p1.y);
1397 const qreal p2x = FROM_FIXED_16_16(paint.u.linear_gradient.p2.x);
1398 const qreal p2y = -FROM_FIXED_16_16(paint.u.linear_gradient.p2.y);
1400 QPointF p0(p0x, p0y);
1401 QPointF p1(p1x, p1y);
1402 QPointF p2(p2x, p2y);
1404 const QGradient::Spread spread =
1405 extendToSpread(paint.u.linear_gradient.colorline.extend);
1406 const QGradientStops stops =
1407 gatherGradientStops(paint.u.linear_gradient.colorline.color_stop_iterator);
1408 paintGraphRenderer->setLinearGradient(p0, p1, p2, spread, stops);
1410 }
else if (paint.format == FT_COLR_PAINTFORMAT_RADIAL_GRADIENT) {
1411 const qreal c0x = FROM_FIXED_16_16(paint.u.radial_gradient.c0.x);
1412 const qreal c0y = -FROM_FIXED_16_16(paint.u.radial_gradient.c0.y);
1413 const qreal r0 = FROM_FIXED_16_16(paint.u.radial_gradient.r0);
1414 const qreal c1x = FROM_FIXED_16_16(paint.u.radial_gradient.c1.x);
1415 const qreal c1y = -FROM_FIXED_16_16(paint.u.radial_gradient.c1.y);
1416 const qreal r1 = FROM_FIXED_16_16(paint.u.radial_gradient.r1);
1418 const QPointF c0(c0x, c0y);
1419 const QPointF c1(c1x, c1y);
1420 const QGradient::Spread spread =
1421 extendToSpread(paint.u.radial_gradient.colorline.extend);
1422 const QGradientStops stops =
1423 gatherGradientStops(paint.u.radial_gradient.colorline.color_stop_iterator);
1425 paintGraphRenderer->setRadialGradient(c0, r0, c1, r1, spread, stops);
1426 }
else if (paint.format == FT_COLR_PAINTFORMAT_SWEEP_GRADIENT) {
1427 const qreal centerX = FROM_FIXED_16_16(paint.u.sweep_gradient.center.x);
1428 const qreal centerY = -FROM_FIXED_16_16(paint.u.sweep_gradient.center.y);
1429 const qreal startAngle = 180.0 * FROM_FIXED_16_16(paint.u.sweep_gradient.start_angle);
1430 const qreal endAngle = 180.0 * FROM_FIXED_16_16(paint.u.sweep_gradient.end_angle);
1432 const QPointF center(centerX, centerY);
1434 const QGradient::Spread spread = extendToSpread(paint.u.radial_gradient.colorline.extend);
1435 const QGradientStops stops = gatherGradientStops(paint.u.sweep_gradient.colorline.color_stop_iterator);
1437 paintGraphRenderer->setConicalGradient(center, startAngle, endAngle, spread, stops);
1439 }
else if (paint.format == FT_COLR_PAINTFORMAT_SOLID) {
1440 QColor color = getPaletteColor(paint.u.solid.color.palette_index,
1441 paint.u.solid.color.alpha);
1442 if (!color.isValid()) {
1443 qCWarning(lcColrv1) <<
"Invalid palette index in COLRv1 graph:"
1444 << paint.u.solid.color.palette_index;
1448 paintGraphRenderer->setSolidColor(color);
1452 paintGraphRenderer->drawCurrentPath();
1453 }
else if (paint.format == FT_COLR_PAINTFORMAT_COMPOSITE) {
1454 if (!paintGraphRenderer->isRendering()) {
1455 if (!traverseColr1(paint.u.composite.backdrop_paint,
1460 paintGraphRenderer)) {
1463 if (!traverseColr1(paint.u.composite.source_paint,
1468 paintGraphRenderer)) {
1472 QPainter::CompositionMode compositionMode = QPainter::CompositionMode_SourceOver;
1473 switch (paint.u.composite.composite_mode) {
1474 case FT_COLR_COMPOSITE_CLEAR:
1475 compositionMode = QPainter::CompositionMode_Clear;
1477 case FT_COLR_COMPOSITE_SRC:
1478 compositionMode = QPainter::CompositionMode_Source;
1480 case FT_COLR_COMPOSITE_DEST:
1481 compositionMode = QPainter::CompositionMode_Destination;
1483 case FT_COLR_COMPOSITE_SRC_OVER:
1484 compositionMode = QPainter::CompositionMode_SourceOver;
1486 case FT_COLR_COMPOSITE_DEST_OVER:
1487 compositionMode = QPainter::CompositionMode_DestinationOver;
1489 case FT_COLR_COMPOSITE_SRC_IN:
1490 compositionMode = QPainter::CompositionMode_SourceIn;
1492 case FT_COLR_COMPOSITE_DEST_IN:
1493 compositionMode = QPainter::CompositionMode_DestinationIn;
1495 case FT_COLR_COMPOSITE_SRC_OUT:
1496 compositionMode = QPainter::CompositionMode_SourceOut;
1498 case FT_COLR_COMPOSITE_DEST_OUT:
1499 compositionMode = QPainter::CompositionMode_DestinationOut;
1501 case FT_COLR_COMPOSITE_SRC_ATOP:
1502 compositionMode = QPainter::CompositionMode_SourceAtop;
1504 case FT_COLR_COMPOSITE_DEST_ATOP:
1505 compositionMode = QPainter::CompositionMode_DestinationAtop;
1507 case FT_COLR_COMPOSITE_XOR:
1508 compositionMode = QPainter::CompositionMode_Xor;
1510 case FT_COLR_COMPOSITE_PLUS:
1511 compositionMode = QPainter::CompositionMode_Plus;
1513 case FT_COLR_COMPOSITE_SCREEN:
1514 compositionMode = QPainter::CompositionMode_Screen;
1516 case FT_COLR_COMPOSITE_OVERLAY:
1517 compositionMode = QPainter::CompositionMode_Overlay;
1519 case FT_COLR_COMPOSITE_DARKEN:
1520 compositionMode = QPainter::CompositionMode_Darken;
1522 case FT_COLR_COMPOSITE_LIGHTEN:
1523 compositionMode = QPainter::CompositionMode_Lighten;
1525 case FT_COLR_COMPOSITE_COLOR_DODGE:
1526 compositionMode = QPainter::CompositionMode_ColorDodge;
1528 case FT_COLR_COMPOSITE_COLOR_BURN:
1529 compositionMode = QPainter::CompositionMode_ColorBurn;
1531 case FT_COLR_COMPOSITE_HARD_LIGHT:
1532 compositionMode = QPainter::CompositionMode_HardLight;
1534 case FT_COLR_COMPOSITE_SOFT_LIGHT:
1535 compositionMode = QPainter::CompositionMode_SoftLight;
1537 case FT_COLR_COMPOSITE_DIFFERENCE:
1538 compositionMode = QPainter::CompositionMode_Difference;
1540 case FT_COLR_COMPOSITE_EXCLUSION:
1541 compositionMode = QPainter::CompositionMode_Exclusion;
1543 case FT_COLR_COMPOSITE_MULTIPLY:
1544 compositionMode = QPainter::CompositionMode_Multiply;
1547 qCWarning(lcColrv1) <<
"Unsupported COLRv1 composition mode" << paint.u.composite.composite_mode;
1551 QColrPaintGraphRenderer compositeRenderer;
1552 compositeRenderer.setBoundingRect(paintGraphRenderer->boundingRect());
1553 compositeRenderer.beginRender(fontDef.pixelSize / face->units_per_EM,
1554 paintGraphRenderer->currentTransform());
1555 if (!traverseColr1(paint.u.composite.backdrop_paint,
1560 &compositeRenderer)) {
1564 compositeRenderer.setCompositionMode(compositionMode);
1565 if (!traverseColr1(paint.u.composite.source_paint,
1570 &compositeRenderer)) {
1573 paintGraphRenderer->drawImage(compositeRenderer.endRender());
1575 }
else if (paint.format == FT_COLR_PAINTFORMAT_GLYPH) {
1576 FT_Error error = FT_Load_Glyph(face,
1577 paint.u.glyph.glyphID,
1578 FT_LOAD_DEFAULT | FT_LOAD_NO_BITMAP | FT_LOAD_NO_SVG | FT_LOAD_IGNORE_TRANSFORM | FT_LOAD_NO_HINTING | FT_LOAD_NO_AUTOHINT | FT_LOAD_BITMAP_METRICS_ONLY);
1580 qCWarning(lcColrv1) <<
"Failed to load glyph"
1581 << paint.u.glyph.glyphID
1582 <<
"in COLRv1 graph. Error: " << error;
1587 QFreetypeFace::addGlyphToPath(face,
1591 face->units_per_EM << 6,
1592 face->units_per_EM << 6);
1594 paintGraphRenderer->appendPath(path);
1596 if (!traverseColr1(paint.u.glyph.paint, loops, foregroundColor, palette, paletteCount, paintGraphRenderer))
1598 }
else if (paint.format == FT_COLR_PAINTFORMAT_COLR_GLYPH) {
1599 FT_OpaquePaint otherOpaquePaint;
1600 otherOpaquePaint.p =
nullptr;
1601 if (!FT_Get_Color_Glyph_Paint(face,
1602 paint.u.colr_glyph.glyphID,
1603 FT_COLOR_NO_ROOT_TRANSFORM,
1604 &otherOpaquePaint)) {
1605 qCWarning(lcColrv1) <<
"Failed to load color glyph"
1606 << paint.u.colr_glyph.glyphID
1607 <<
"in COLRv1 graph.";
1611 if (!traverseColr1(otherOpaquePaint, loops, foregroundColor, palette, paletteCount, paintGraphRenderer))
1618QFontEngineFT::Glyph *QFontEngineFT::loadColrv1Glyph(QGlyphSet *set,
1621 const QColor &foregroundColor,
1622 bool fetchMetricsOnly)
const
1624 FT_Face face = freetype->face;
1627 memset(&info, 0,
sizeof(info));
1631 FT_Load_Glyph(face, glyph, FT_LOAD_DEFAULT
1634 | FT_LOAD_BITMAP_METRICS_ONLY);
1635 info.linearAdvance =
int(face->glyph->linearHoriAdvance >> 10);
1636 info.xOff =
short(TRUNC(ROUND(face->glyph->advance.x)));
1638 FT_OpaquePaint opaquePaint;
1639 opaquePaint.p =
nullptr;
1640 if (!FT_Get_Color_Glyph_Paint(face, glyph, FT_COLOR_INCLUDE_ROOT_TRANSFORM, &opaquePaint))
1645 FT_Set_Char_Size(face, face->units_per_EM << 6, face->units_per_EM << 6, 0, 0);
1649 FT_Get_Transform(face, &matrix, &delta);
1650 QTransform originalXform(FROM_FIXED_16_16(matrix.xx), -FROM_FIXED_16_16(matrix.yx),
1651 -FROM_FIXED_16_16(matrix.xy), FROM_FIXED_16_16(matrix.yy),
1652 FROM_FIXED_16_16(delta.x), FROM_FIXED_16_16(delta.y));
1656 FT_Set_Transform(face,
nullptr,
nullptr);
1658 auto cleanup = qScopeGuard([&]() {
1660 FT_Set_Char_Size(face, xsize, ysize, 0, 0);
1661 FT_Set_Transform(face, &matrix, &delta);
1664 qCDebug(lcColrv1).noquote() <<
"================== Start collecting COLRv1 metrics for" << glyph;
1665 QRect designCoordinateBounds;
1669 if (colrv1_bounds_cache_id == glyph) {
1670 designCoordinateBounds = colrv1_bounds_cache;
1675 if (FT_Get_Color_Glyph_ClipBox(face, glyph, &clipBox)) {
1676 FT_Pos left = qMin(clipBox.bottom_left.x, qMin(clipBox.bottom_right.x, qMin(clipBox.top_left.x, clipBox.top_right.x)));
1677 FT_Pos right = qMax(clipBox.bottom_left.x, qMax(clipBox.bottom_right.x, qMax(clipBox.top_left.x, clipBox.top_right.x)));
1679 FT_Pos top = qMin(-clipBox.bottom_left.y, qMin(-clipBox.bottom_right.y, qMin(-clipBox.top_left.y, -clipBox.top_right.y)));
1680 FT_Pos bottom = qMax(-clipBox.bottom_left.y, qMax(-clipBox.bottom_right.y, qMax(-clipBox.top_left.y, -clipBox.top_right.y)));
1682 qreal scale = 1.0 / 64.0;
1683 designCoordinateBounds = QRect(QPoint(qFloor(left * scale), qFloor(top * scale)),
1684 QPoint(qCeil(right * scale), qCeil(bottom * scale)));
1687 QColrPaintGraphRenderer boundingRectCalculator;
1688 boundingRectCalculator.beginCalculateBoundingBox();
1689 QSet<std::pair<FT_Byte *, FT_Bool> > loops;
1690 if (traverseColr1(opaquePaint,
1695 &boundingRectCalculator)) {
1696 designCoordinateBounds = boundingRectCalculator.boundingRect().toAlignedRect();
1700 colrv1_bounds_cache_id = glyph;
1701 colrv1_bounds_cache = designCoordinateBounds;
1704 QTransform initialTransform;
1705 initialTransform.scale(fontDef.pixelSize / face->units_per_EM,
1706 fontDef.pixelSize / face->units_per_EM);
1707 QRect bounds = initialTransform.mapRect(designCoordinateBounds);
1708 bounds = originalXform.mapRect(bounds);
1710 info.x = bounds.left();
1711 info.y = -bounds.top();
1712 info.width = bounds.width();
1713 info.height = bounds.height();
1715 qCDebug(lcColrv1) <<
"Bounds of" << glyph <<
"==" << bounds;
1718 QImage destinationImage;
1719 if (!fetchMetricsOnly && !bounds.size().isEmpty()) {
1720 FT_Palette_Data paletteData;
1721 if (FT_Palette_Data_Get(face, &paletteData))
1724 FT_Color *palette =
nullptr;
1725 FT_Error error = FT_Palette_Select(face, 0, &palette);
1727 qWarning(
"selecting palette for COLRv1 failed, err=%x face=%p, glyph=%d",
1733 if (palette ==
nullptr)
1736 ushort paletteCount = paletteData.num_palette_entries;
1738 QColrPaintGraphRenderer paintGraphRenderer;
1739 paintGraphRenderer.setBoundingRect(bounds);
1740 paintGraphRenderer.beginRender(fontDef.pixelSize / face->units_per_EM,
1744 QSet<std::pair<FT_Byte *, FT_Bool> > loops;
1745 if (!traverseColr1(opaquePaint,
1750 &paintGraphRenderer)) {
1754 destinationImage = paintGraphRenderer.endRender();
1757 if (fetchMetricsOnly || !destinationImage.isNull()) {
1762 set->setGlyph(glyph, QFixedPoint{}, g);
1765 g->linearAdvance = info.linearAdvance;
1766 g->width = info.width;
1767 g->height = info.height;
1770 g->advance = info.xOff;
1771 g->format = Format_ARGB;
1773 if (!fetchMetricsOnly && !destinationImage.isNull()) {
1774 g->data =
new uchar[info.height * info.width * 4];
1775 memcpy(g->data, destinationImage.constBits(), info.height * info.width * 4);
1785QFontEngineFT::Glyph *QFontEngineFT::loadGlyph(QGlyphSet *set, uint glyph,
1786 const QFixedPoint &subPixelPosition,
1789 bool fetchMetricsOnly,
1790 bool disableOutlineDrawing)
const
1794 if (format == Format_None)
1795 format = defaultFormat != Format_None ? defaultFormat : Format_Mono;
1796 Q_ASSERT(format != Format_None);
1798 Glyph *g = set ? set->getGlyph(glyph, subPixelPosition) :
nullptr;
1799 if (g && g->format == format && (fetchMetricsOnly || g->data))
1802 if (!g && set && set->isGlyphMissing(glyph))
1806 FT_Face face = freetype->face;
1808 FT_Matrix matrix = freetype->matrix;
1809 bool transform = matrix.xx != 0x10000
1810 || matrix.yy != 0x10000
1813 if (obliquen && transform) {
1818 slant.xx = 0x10000L;
1821 slant.yy = 0x10000L;
1823 FT_Matrix_Multiply(&matrix, &slant);
1828 v.x = format == Format_Mono ? 0 : FT_Pos(subPixelPosition.x.value());
1829 v.y = format == Format_Mono ? 0 : FT_Pos(-subPixelPosition.y.value());
1830 FT_Set_Transform(face, &matrix, &v);
1832 bool hsubpixel =
false;
1834 int load_flags = loadFlags(set, format, 0, hsubpixel, vfactor);
1836 if (transform || obliquen || (format != Format_Mono && !isScalableBitmap()))
1837 load_flags |= FT_LOAD_NO_BITMAP;
1839#if defined(QFONTENGINE_FT_SUPPORT_COLRV1)
1840 if (FT_IS_SCALABLE(freetype->face)
1841 && FT_HAS_COLOR(freetype->face)
1842 && (load_flags & FT_LOAD_COLOR)) {
1844 Glyph *ret = loadColrv1Glyph(set, g, glyph, color, fetchMetricsOnly);
1852 FT_Error err = FT_Load_Glyph(face, glyph, load_flags);
1853 if (err && (load_flags & FT_LOAD_NO_BITMAP)) {
1854 load_flags &= ~FT_LOAD_NO_BITMAP;
1855 err = FT_Load_Glyph(face, glyph, load_flags);
1857 if (err == FT_Err_Too_Few_Arguments) {
1859 load_flags |= FT_LOAD_FORCE_AUTOHINT;
1860 err = FT_Load_Glyph(face, glyph, load_flags);
1861 }
else if (err == FT_Err_Execution_Too_Long) {
1864 qWarning(
"load glyph failed due to broken hinting bytecode in font, switching to auto hinting");
1865 default_load_flags |= FT_LOAD_FORCE_AUTOHINT;
1866 load_flags |= FT_LOAD_FORCE_AUTOHINT;
1867 err = FT_Load_Glyph(face, glyph, load_flags);
1869 if (err != FT_Err_Ok) {
1870 qWarning(
"load glyph failed err=%x face=%p, glyph=%d", err, face, glyph);
1872 set->setGlyphMissing(glyph);
1876 FT_GlyphSlot slot = face->glyph;
1879 FT_GlyphSlot_Embolden(slot);
1880 if (obliquen && !transform) {
1881 FT_GlyphSlot_Oblique(slot);
1892 FT_Matrix_Multiply(&m, &matrix);
1896 info.linearAdvance = slot->linearHoriAdvance >> 10;
1900 if ((set && set->outline_drawing && !disableOutlineDrawing) || fetchMetricsOnly) {
1901 int left = slot->metrics.horiBearingX;
1902 int right = slot->metrics.horiBearingX + slot->metrics.width;
1903 int top = slot->metrics.horiBearingY;
1904 int bottom = slot->metrics.horiBearingY - slot->metrics.height;
1906 if (transform && slot->format != FT_GLYPH_FORMAT_BITMAP)
1907 transformBoundingBox(&left, &top, &right, &bottom, &matrix);
1910 right =
CEIL(right);
1911 bottom =
FLOOR(bottom);
1914 info.x =
TRUNC(left);
1915 info.y =
TRUNC(top);
1916 info.width =
TRUNC(right - left);
1917 info.height =
TRUNC(top - bottom);
1921 if (areMetricsTooLarge(info))
1926 g->linearAdvance = info.linearAdvance;
1927 g->width = info.width;
1928 g->height = info.height;
1931 g->advance = info.xOff;
1935 set->setGlyph(glyph, subPixelPosition, g);
1940 int glyph_buffer_size = 0;
1941 std::unique_ptr<uchar[]> glyph_buffer;
1942 FT_Render_Mode renderMode = (default_hint_style == HintLight) ? FT_RENDER_MODE_LIGHT : FT_RENDER_MODE_NORMAL;
1945 renderMode = FT_RENDER_MODE_MONO;
1948 if (!hsubpixel && vfactor == 1) {
1949 qWarning(
"Format_A32 requested, but subpixel layout is unknown.");
1953 renderMode = hsubpixel ? FT_RENDER_MODE_LCD : FT_RENDER_MODE_LCD_V;
1961 FT_Library_SetLcdFilter(slot->library, (FT_LcdFilter)lcdFilterType);
1963 err = FT_Render_Glyph(slot, renderMode);
1964 FT_Library_SetLcdFilter(slot->library, FT_LCD_FILTER_NONE);
1966 if (err != FT_Err_Ok) {
1967 qWarning(
"render glyph failed err=%x face=%p, glyph=%d", err, face, glyph);
1971 info.height = slot->bitmap.rows;
1972 info.width = slot->bitmap.width;
1973 info.x = slot->bitmap_left;
1974 info.y = slot->bitmap_top;
1975 if (slot->bitmap.pixel_mode == FT_PIXEL_MODE_LCD)
1976 info.width = info.width / 3;
1977 if (slot->bitmap.pixel_mode == FT_PIXEL_MODE_LCD_V)
1978 info.height = info.height / vfactor;
1980 int pitch = (format == Format_Mono ? ((info.width + 31) & ~31) >> 3 :
1981 (format == Format_A8 ? (info.width + 3) & ~3 : info.width * 4));
1983 glyph_buffer_size = info.height * pitch;
1984 glyph_buffer.reset(
new uchar[glyph_buffer_size]);
1986 if (slot->bitmap.pixel_mode == FT_PIXEL_MODE_MONO) {
1987 uchar *src = slot->bitmap.buffer;
1988 uchar *dst = glyph_buffer.get();
1989 int h = slot->bitmap.rows;
1991 if (format == Format_Mono) {
1992 int bytes = ((info.width + 7) & ~7) >> 3;
1994 memcpy (dst, src, bytes);
1996 src += slot->bitmap.pitch;
1998 }
else if (format == Format_A8) {
2000 for (
int x = 0; x <
int{info.width}; x++)
2001 dst[x] = ((src[x >> 3] & (0x80 >> (x & 7))) ? 0xff : 0x00);
2003 src += slot->bitmap.pitch;
2007 uint *dd =
reinterpret_cast<uint *>(dst);
2008 for (
int x = 0; x <
int{info.width}; x++)
2009 dd[x] = ((src[x >> 3] & (0x80 >> (x & 7))) ? 0xffffffff : 0x00000000);
2011 src += slot->bitmap.pitch;
2014 }
else if (slot->bitmap.pixel_mode == 7 ) {
2015 Q_ASSERT(format == Format_ARGB);
2016 uchar *src = slot->bitmap.buffer;
2017 uchar *dst = glyph_buffer.get();
2018 int h = slot->bitmap.rows;
2020#if Q_BYTE_ORDER == Q_BIG_ENDIAN
2021 const quint32 *srcPixel = (
const quint32 *)src;
2022 quint32 *dstPixel = (quint32 *)dst;
2023 for (
int x = 0; x <
static_cast<
int>(slot->bitmap.width); x++, srcPixel++, dstPixel++) {
2024 const quint32 pixel = *srcPixel;
2025 *dstPixel = qbswap(pixel);
2028 memcpy(dst, src, slot->bitmap.width * 4);
2030 dst += slot->bitmap.pitch;
2031 src += slot->bitmap.pitch;
2033 info.linearAdvance = info.xOff = slot->bitmap.width;
2034 }
else if (slot->bitmap.pixel_mode == FT_PIXEL_MODE_GRAY) {
2035 if (format == Format_A8) {
2036 uchar *src = slot->bitmap.buffer;
2037 uchar *dst = glyph_buffer.get();
2038 int h = slot->bitmap.rows;
2039 int bytes = info.width;
2041 memcpy (dst, src, bytes);
2043 src += slot->bitmap.pitch;
2045 }
else if (format == Format_ARGB) {
2046 uchar *src = slot->bitmap.buffer;
2047 quint32 *dstPixel =
reinterpret_cast<quint32 *>(glyph_buffer.get());
2048 int h = slot->bitmap.rows;
2050 for (
int x = 0; x <
static_cast<
int>(slot->bitmap.width); ++x) {
2051 uchar alpha = src[x];
2052 float alphaF = alpha / 255.0;
2053 dstPixel[x] = qRgba(qRound(alphaF * color.red()),
2054 qRound(alphaF * color.green()),
2055 qRound(alphaF * color.blue()),
2058 src += slot->bitmap.pitch;
2059 dstPixel += info.width;
2062 }
else if (slot->bitmap.pixel_mode == FT_PIXEL_MODE_LCD) {
2063 Q_ASSERT(format == Format_A32);
2064 convertRGBToARGB(slot->bitmap.buffer, (uint *)glyph_buffer.get(), info.width, info.height, slot->bitmap.pitch, subpixelType != Subpixel_RGB);
2065 }
else if (slot->bitmap.pixel_mode == FT_PIXEL_MODE_LCD_V) {
2066 Q_ASSERT(format == Format_A32);
2067 convertRGBToARGB_V(slot->bitmap.buffer, (uint *)glyph_buffer.get(), info.width, info.height, slot->bitmap.pitch, subpixelType != Subpixel_VRGB);
2069 qWarning(
"QFontEngine: Glyph rendered in unknown pixel_mode=%d", slot->bitmap.pixel_mode);
2078 g->linearAdvance = info.linearAdvance;
2079 g->width = info.width;
2080 g->height = info.height;
2083 g->advance = info.xOff;
2086 g->data = glyph_buffer.release();
2089 set->setGlyph(glyph, subPixelPosition, g);
2094QFontEngine::FaceId QFontEngineFT::faceId()
const
2099QFontEngine::Properties QFontEngineFT::properties()
const
2101 Properties p = freetype->properties();
2102 if (p.postscriptName.isEmpty()) {
2103 p.postscriptName = QFontEngine::convertToPostscriptFontFamilyName(fontDef.families.constFirst().toUtf8());
2106 return freetype->properties();
2109QFixed QFontEngineFT::emSquareSize()
const
2111 if (FT_IS_SCALABLE(freetype->face))
2112 return freetype->face->units_per_EM;
2114 return freetype->face->size->metrics.y_ppem;
2117bool QFontEngineFT::getSfntTableData(uint tag, uchar *buffer, uint *length)
const
2119 return ft_getSfntTable(freetype->face, tag, buffer, length);
2122int QFontEngineFT::synthesized()
const
2125 if ((fontDef.style != QFont::StyleNormal) && !(freetype->face->style_flags & FT_STYLE_FLAG_ITALIC))
2126 s = SynthesizedItalic;
2127 if ((fontDef.weight >= QFont::Bold) && !(freetype->face->style_flags & FT_STYLE_FLAG_BOLD))
2128 s |= SynthesizedBold;
2129 if (fontDef.stretch != 100 && FT_IS_SCALABLE(freetype->face))
2130 s |= SynthesizedStretch;
2134void QFontEngineFT::initializeHeightMetrics()
const
2136 m_ascent = QFixed::fromFixed(metrics.ascender);
2137 m_descent = QFixed::fromFixed(-metrics.descender);
2138 m_leading = QFixed::fromFixed(metrics.height - metrics.ascender + metrics.descender);
2140 QFontEngine::initializeHeightMetrics();
2142 if (scalableBitmapScaleFactor != 1) {
2143 m_ascent *= scalableBitmapScaleFactor;
2144 m_descent *= scalableBitmapScaleFactor;
2145 m_leading *= scalableBitmapScaleFactor;
2149QFixed QFontEngineFT::capHeight()
const
2151 TT_OS2 *os2 = (TT_OS2 *)FT_Get_Sfnt_Table(freetype->face, ft_sfnt_os2);
2152 if (os2 && os2->version >= 2) {
2154 QFixed answer = QFixed::fromFixed(FT_MulFix(os2->sCapHeight, freetype->face->size->metrics.y_scale));
2158 return calculatedCapHeight();
2161QFixed QFontEngineFT::xHeight()
const
2163 TT_OS2 *os2 = (TT_OS2 *)FT_Get_Sfnt_Table(freetype->face, ft_sfnt_os2);
2164 if (os2 && os2->sxHeight) {
2166 QFixed answer = QFixed(os2->sxHeight * freetype->face->size->metrics.y_ppem) / emSquareSize();
2171 return QFontEngine::xHeight();
2174QFixed QFontEngineFT::averageCharWidth()
const
2176 TT_OS2 *os2 = (TT_OS2 *)FT_Get_Sfnt_Table(freetype->face, ft_sfnt_os2);
2177 if (os2 && os2->xAvgCharWidth) {
2179 QFixed answer = QFixed(os2->xAvgCharWidth * freetype->face->size->metrics.x_ppem) / emSquareSize();
2184 return QFontEngine::averageCharWidth();
2187qreal QFontEngineFT::maxCharWidth()
const
2189 QFixed max_advance = QFixed::fromFixed(metrics.max_advance);
2190 if (scalableBitmapScaleFactor != 1)
2191 max_advance *= scalableBitmapScaleFactor;
2192 return max_advance.toReal();
2195QFixed QFontEngineFT::lineThickness()
const
2197 return line_thickness;
2200QFixed QFontEngineFT::underlinePosition()
const
2202 return underline_position;
2205void QFontEngineFT::doKerning(QGlyphLayout *g, QFontEngine::ShaperFlags flags)
const
2207 if (!kerning_pairs_loaded) {
2208 kerning_pairs_loaded =
true;
2210 if (freetype->face->size->metrics.x_ppem != 0) {
2211 QFixed scalingFactor = emSquareSize() / QFixed(freetype->face->size->metrics.x_ppem);
2213 const_cast<QFontEngineFT *>(
this)->loadKerningPairs(scalingFactor);
2219 if (shouldUseDesignMetrics(flags))
2220 flags |= DesignMetrics;
2222 flags &= ~DesignMetrics;
2224 QFontEngine::doKerning(g, flags);
2231 m.xx = FT_Fixed(matrix.m11() * 65536);
2232 m.xy = FT_Fixed(-matrix.m21() * 65536);
2233 m.yx = FT_Fixed(-matrix.m12() * 65536);
2234 m.yy = FT_Fixed(matrix.m22() * 65536);
2239QFontEngineFT::QGlyphSet *QFontEngineFT::TransformedGlyphSets::findSet(
const QTransform &matrix,
const QFontDef &fontDef)
2241 FT_Matrix m = QTransformToFTMatrix(matrix);
2244 for (; i < nSets; ++i) {
2245 QGlyphSet *g = sets[i];
2248 if (g->transformationMatrix.xx == m.xx
2249 && g->transformationMatrix.xy == m.xy
2250 && g->transformationMatrix.yx == m.yx
2251 && g->transformationMatrix.yy == m.yy) {
2263 moveToFront(nSets - 1);
2265 sets[0] =
new QGlyphSet;
2266 QGlyphSet *gs = sets[0];
2267 Q_ASSERT(gs !=
nullptr);
2270 gs->transformationMatrix = m;
2271 const int maxCachedSize = maxCachedGlyphSize();
2272 gs->outline_drawing = fontDef.pixelSize * fontDef.pixelSize * qAbs(matrix.determinant()) > maxCachedSize * maxCachedSize;
2277void QFontEngineFT::TransformedGlyphSets::moveToFront(
int i)
2279 QGlyphSet *g = sets[i];
2281 sets[i] = sets[i - 1];
2288QFontEngineFT::QGlyphSet *QFontEngineFT::loadGlyphSet(
const QTransform &matrix)
2290 if (matrix.type() > QTransform::TxShear || !cacheEnabled)
2294 if (!FT_IS_SCALABLE(freetype->face))
2295 return matrix.type() <= QTransform::TxTranslate ? &defaultGlyphSet :
nullptr;
2297 return transformedGlyphSets.findSet(matrix, fontDef);
2300void QFontEngineFT::getUnscaledGlyph(glyph_t glyph, QPainterPath *path, glyph_metrics_t *metrics)
2302 FT_Face face = lockFace(Unscaled);
2303 FT_Set_Transform(face,
nullptr,
nullptr);
2304 FT_Load_Glyph(face, glyph, FT_LOAD_NO_BITMAP);
2306 int left = face->glyph->metrics.horiBearingX;
2307 int right = face->glyph->metrics.horiBearingX + face->glyph->metrics.width;
2308 int top = face->glyph->metrics.horiBearingY;
2309 int bottom = face->glyph->metrics.horiBearingY - face->glyph->metrics.height;
2315 metrics->width = QFixed::fromFixed(right-left);
2316 metrics->height = QFixed::fromFixed(top-bottom);
2317 metrics->x = QFixed::fromFixed(left);
2318 metrics->y = QFixed::fromFixed(-top);
2319 metrics->xoff = QFixed::fromFixed(face->glyph->advance.x);
2321 if (!FT_IS_SCALABLE(freetype->face))
2322 QFreetypeFace::addBitmapToPath(face->glyph, p, path);
2324 QFreetypeFace::addGlyphToPath(face, face->glyph, p, path, face->units_per_EM << 6, face->units_per_EM << 6);
2326 FT_Set_Transform(face, &freetype->matrix,
nullptr);
2330bool QFontEngineFT::supportsTransformation(
const QTransform &transform)
const
2332 return transform.type() <= QTransform::TxRotate;
2335void QFontEngineFT::addOutlineToPath(qreal x, qreal y,
const QGlyphLayout &glyphs, QPainterPath *path, QTextItem::RenderFlags flags)
2337 if (!glyphs.numGlyphs)
2340 if (FT_IS_SCALABLE(freetype->face)) {
2341 QFontEngine::addOutlineToPath(x, y, glyphs, path, flags);
2343 QVarLengthArray<QFixedPoint> positions;
2344 QVarLengthArray<glyph_t> positioned_glyphs;
2346 matrix.translate(x, y);
2347 getGlyphPositions(glyphs, matrix, flags, positioned_glyphs, positions);
2349 FT_Face face = lockFace(Unscaled);
2350 for (
int gl = 0; gl < glyphs.numGlyphs; gl++) {
2351 FT_UInt glyph = positioned_glyphs[gl];
2352 FT_Load_Glyph(face, glyph, FT_LOAD_TARGET_MONO);
2353 QFreetypeFace::addBitmapToPath(face->glyph, positions[gl], path);
2359void QFontEngineFT::addGlyphsToPath(glyph_t *glyphs, QFixedPoint *positions,
int numGlyphs,
2360 QPainterPath *path, QTextItem::RenderFlags)
2362 FT_Face face = lockFace(Unscaled);
2364 for (
int gl = 0; gl < numGlyphs; gl++) {
2365 FT_UInt glyph = glyphs[gl];
2367 FT_Load_Glyph(face, glyph, FT_LOAD_NO_BITMAP);
2369 FT_GlyphSlot g = face->glyph;
2370 if (g->format != FT_GLYPH_FORMAT_OUTLINE)
2373 FT_GlyphSlot_Embolden(g);
2375 FT_GlyphSlot_Oblique(g);
2376 QFreetypeFace::addGlyphToPath(face, g, positions[gl], path, xsize, ysize);
2381glyph_t QFontEngineFT::glyphIndex(uint ucs4)
const
2383 glyph_t glyph = ucs4 < QFreetypeFace::cmapCacheSize ? freetype->cmapCache[ucs4] : 0;
2385 FT_Face face = freetype->face;
2386 glyph = FT_Get_Char_Index(face, ucs4);
2390 if (ucs4 == QChar::Nbsp || ucs4 == QChar::Tabulation) {
2391 glyph = FT_Get_Char_Index(face, QChar::Space);
2392 }
else if (freetype->symbol_map) {
2399 FT_Set_Charmap(face, freetype->symbol_map);
2400 glyph = FT_Get_Char_Index(face, ucs4);
2401 FT_Set_Charmap(face, freetype->unicode_map);
2402 if (!glyph && symbol && ucs4 < 0x100)
2403 glyph = FT_Get_Char_Index(face, ucs4 + 0xf000);
2406 if (ucs4 < QFreetypeFace::cmapCacheSize)
2407 freetype->cmapCache[ucs4] = glyph;
2413QString QFontEngineFT::glyphName(glyph_t index)
const
2416 if (index >= glyph_t(glyphCount()))
2419 FT_Face face = freetype->face;
2420 if (face->face_flags & FT_FACE_FLAG_GLYPH_NAMES) {
2421 char glyphName[128] = {};
2422 if (FT_Get_Glyph_Name(face, index, glyphName,
sizeof(glyphName)) == 0)
2423 result = QString::fromUtf8(glyphName);
2426 return result.isEmpty() ? QFontEngine::glyphName(index) : result;
2429int QFontEngineFT::stringToCMap(
const QChar *str,
int len, QGlyphLayout *glyphs,
int *nglyphs,
2430 QFontEngine::ShaperFlags flags)
const
2432 Q_ASSERT(glyphs->numGlyphs >= *nglyphs);
2433 if (*nglyphs < len) {
2438 int mappedGlyphs = 0;
2440 if (freetype->symbol_map) {
2441 FT_Face face = freetype->face;
2442 QStringIterator it(str, str + len);
2443 while (it.hasNext()) {
2444 uint uc = it.next();
2445 glyphs->glyphs[glyph_pos] = uc < QFreetypeFace::cmapCacheSize ? freetype->cmapCache[uc] : 0;
2446 if ( !glyphs->glyphs[glyph_pos] ) {
2453 glyph_t glyph = FT_Get_Char_Index(face, uc);
2456 if (!glyph && (uc == 0xa0 || uc == 0x9)) {
2458 glyph = FT_Get_Char_Index(face, uc);
2461 FT_Set_Charmap(face, freetype->symbol_map);
2462 glyph = FT_Get_Char_Index(face, uc);
2463 FT_Set_Charmap(face, freetype->unicode_map);
2464 if (!glyph && symbol && uc < 0x100)
2465 glyph = FT_Get_Char_Index(face, uc + 0xf000);
2467 glyphs->glyphs[glyph_pos] = glyph;
2468 if (uc < QFreetypeFace::cmapCacheSize)
2469 freetype->cmapCache[uc] = glyph;
2471 if (glyphs->glyphs[glyph_pos] || isIgnorableChar(uc))
2476 FT_Face face = freetype->face;
2477 QStringIterator it(str, str + len);
2478 while (it.hasNext()) {
2479 uint uc = it.next();
2480 glyphs->glyphs[glyph_pos] = uc < QFreetypeFace::cmapCacheSize ? freetype->cmapCache[uc] : 0;
2481 if (!glyphs->glyphs[glyph_pos]) {
2484 glyph_t glyph = FT_Get_Char_Index(face, uc);
2485 if (!glyph && (uc == 0xa0 || uc == 0x9)) {
2489 glyphs->glyphs[glyph_pos] = glyph;
2490 if (uc < QFreetypeFace::cmapCacheSize)
2491 freetype->cmapCache[uc] = glyph;
2494 if (glyphs->glyphs[glyph_pos] || isIgnorableChar(uc))
2500 *nglyphs = glyph_pos;
2501 glyphs->numGlyphs = glyph_pos;
2503 if (!(flags & GlyphIndicesOnly))
2504 recalcAdvances(glyphs, flags);
2506 return mappedGlyphs;
2509bool QFontEngineFT::shouldUseDesignMetrics(QFontEngine::ShaperFlags flags)
const
2511 if (!FT_IS_SCALABLE(freetype->face))
2514 return default_hint_style == HintNone || default_hint_style == HintLight || (flags & DesignMetrics);
2517QFixed QFontEngineFT::scaledBitmapMetrics(QFixed m)
const
2519 return m * scalableBitmapScaleFactor;
2522glyph_metrics_t QFontEngineFT::scaledBitmapMetrics(
const glyph_metrics_t &m,
const QTransform &t)
const
2525 trans.setMatrix(t.m11(), t.m12(), t.m13(),
2526 t.m21(), t.m22(), t.m23(),
2528 const qreal scaleFactor = scalableBitmapScaleFactor.toReal();
2529 trans.scale(scaleFactor, scaleFactor);
2531 QRectF rect(m.x.toReal(), m.y.toReal(), m.width.toReal(), m.height.toReal());
2532 QPointF offset(m.xoff.toReal(), m.yoff.toReal());
2534 rect = trans.mapRect(rect);
2535 offset = trans.map(offset);
2537 glyph_metrics_t metrics;
2538 metrics.x = QFixed::fromReal(rect.x());
2539 metrics.y = QFixed::fromReal(rect.y());
2540 metrics.width = QFixed::fromReal(rect.width());
2541 metrics.height = QFixed::fromReal(rect.height());
2542 metrics.xoff = QFixed::fromReal(offset.x());
2543 metrics.yoff = QFixed::fromReal(offset.y());
2547void QFontEngineFT::recalcAdvances(QGlyphLayout *glyphs, QFontEngine::ShaperFlags flags)
const
2549 FT_Face face =
nullptr;
2550 bool design = shouldUseDesignMetrics(flags);
2551 for (
int i = 0; i < glyphs->numGlyphs; i++) {
2552 Glyph *g = cacheEnabled ? defaultGlyphSet.getGlyph(glyphs->glyphs[i]) :
nullptr;
2554 GlyphFormat acceptableFormat = (defaultFormat != Format_None) ? defaultFormat : Format_Mono;
2555 if (g && g->format == acceptableFormat) {
2556 glyphs->advances[i] = design ? QFixed::fromFixed(g->linearAdvance) : QFixed(g->advance);
2560 g = loadGlyph(cacheEnabled ? &defaultGlyphSet :
nullptr,
2567 glyphs->advances[i] = design ? QFixed::fromFixed(g->linearAdvance) : QFixed(g->advance);
2569 glyphs->advances[i] = design ? QFixed::fromFixed(face->glyph->linearHoriAdvance >> 10)
2570 : QFixed::fromFixed(face->glyph->metrics.horiAdvance).round();
2571 if (!cacheEnabled && g != &emptyGlyph)
2575 if (scalableBitmapScaleFactor != 1)
2576 glyphs->advances[i] *= scalableBitmapScaleFactor;
2582glyph_metrics_t QFontEngineFT::boundingBox(
const QGlyphLayout &glyphs)
2584 FT_Face face =
nullptr;
2586 glyph_metrics_t overall;
2588 if (!isScalableBitmap()) {
2589 overall.y = -ascent();
2590 overall.height = ascent() + descent();
2592 overall.y = QFixed::fromFixed(-metrics.ascender);
2593 overall.height = QFixed::fromFixed(metrics.ascender - metrics.descender);
2598 for (
int i = 0; i < glyphs.numGlyphs; i++) {
2600 if (!glyphs.advances[i] || glyphs.attributes[i].dontPrint)
2602 Glyph *g = cacheEnabled ? defaultGlyphSet.getGlyph(glyphs.glyphs[i]) :
nullptr;
2606 g = loadGlyph(cacheEnabled ? &defaultGlyphSet :
nullptr,
2614 QFixed x = overall.xoff + glyphs.offsets[i].x + g->x;
2615 QFixed y = overall.yoff + glyphs.offsets[i].y - g->y;
2616 overall.x = qMin(overall.x, x);
2617 overall.y = qMin(overall.y, y);
2618 xmax = qMax(xmax, x.ceil() + g->width);
2619 ymax = qMax(ymax, y.ceil() + g->height);
2620 if (!cacheEnabled && g != &emptyGlyph)
2623 int left =
FLOOR(face->glyph->metrics.horiBearingX);
2624 int right =
CEIL(face->glyph->metrics.horiBearingX + face->glyph->metrics.width);
2625 int top =
CEIL(face->glyph->metrics.horiBearingY);
2626 int bottom =
FLOOR(face->glyph->metrics.horiBearingY - face->glyph->metrics.height);
2628 QFixed x = overall.xoff + glyphs.offsets[i].x - (-
TRUNC(left));
2629 QFixed y = overall.yoff + glyphs.offsets[i].y -
TRUNC(top);
2630 overall.x = qMin(overall.x, x);
2631 overall.y = qMin(overall.y, y);
2632 xmax = qMax(xmax, x +
TRUNC(right - left));
2633 ymax = qMax(ymax, y +
TRUNC(top - bottom));
2635 overall.xoff += glyphs.effectiveAdvance(i);
2637 overall.height = qMax(overall.height, ymax - overall.y);
2638 overall.width = xmax - overall.x;
2643 if (isScalableBitmap())
2644 overall = scaledBitmapMetrics(overall, QTransform());
2648glyph_metrics_t QFontEngineFT::boundingBox(glyph_t glyph)
2650 FT_Face face =
nullptr;
2651 glyph_metrics_t overall;
2652 Glyph *g = cacheEnabled ? defaultGlyphSet.getGlyph(glyph) :
nullptr;
2655 g = loadGlyph(cacheEnabled ? &defaultGlyphSet :
nullptr,
2665 overall.width = g->width;
2666 overall.height = g->height;
2667 overall.xoff = g->advance;
2668 if (!cacheEnabled && g != &emptyGlyph)
2671 int left =
FLOOR(face->glyph->metrics.horiBearingX);
2672 int right =
CEIL(face->glyph->metrics.horiBearingX + face->glyph->metrics.width);
2673 int top =
CEIL(face->glyph->metrics.horiBearingY);
2674 int bottom =
FLOOR(face->glyph->metrics.horiBearingY - face->glyph->metrics.height);
2676 overall.width =
TRUNC(right-left);
2677 overall.height =
TRUNC(top-bottom);
2678 overall.x =
TRUNC(left);
2679 overall.y = -
TRUNC(top);
2680 overall.xoff =
TRUNC(
ROUND(face->glyph->advance.x));
2685 if (isScalableBitmap())
2686 overall = scaledBitmapMetrics(overall, QTransform());
2690glyph_metrics_t QFontEngineFT::boundingBox(glyph_t glyph,
const QTransform &matrix)
2692 return alphaMapBoundingBox(glyph, QFixedPoint(), matrix, QFontEngine::Format_None);
2695glyph_metrics_t QFontEngineFT::alphaMapBoundingBox(glyph_t glyph,
2696 const QFixedPoint &subPixelPosition,
2697 const QTransform &matrix,
2698 QFontEngine::GlyphFormat format)
2704 const bool needsImageTransform = !FT_IS_SCALABLE(freetype->face)
2705 && matrix.type() > QTransform::TxTranslate;
2706 if (needsImageTransform && format == QFontEngine::Format_Mono)
2707 format = QFontEngine::Format_A8;
2708 Glyph *g = loadGlyphFor(glyph, subPixelPosition, format, matrix, QColor(),
true,
true);
2710 glyph_metrics_t overall;
2714 overall.width = g->width;
2715 overall.height = g->height;
2716 overall.xoff = g->advance;
2717 if (!cacheEnabled && g != &emptyGlyph)
2720 FT_Face face = lockFace();
2721 int left =
FLOOR(face->glyph->metrics.horiBearingX);
2722 int right =
CEIL(face->glyph->metrics.horiBearingX + face->glyph->metrics.width);
2723 int top =
CEIL(face->glyph->metrics.horiBearingY);
2724 int bottom =
FLOOR(face->glyph->metrics.horiBearingY - face->glyph->metrics.height);
2726 overall.width =
TRUNC(right-left);
2727 overall.height =
TRUNC(top-bottom);
2728 overall.x =
TRUNC(left);
2729 overall.y = -
TRUNC(top);
2730 overall.xoff =
TRUNC(
ROUND(face->glyph->advance.x));
2734 if (isScalableBitmap() || needsImageTransform)
2735 overall = scaledBitmapMetrics(overall, matrix);
2741 if (glyph ==
nullptr || glyph->height == 0 || glyph->width == 0)
2744 QImage::Format format = QImage::Format_Invalid;
2745 int bytesPerLine = -1;
2746 switch (glyphFormat) {
2747 case QFontEngine::Format_Mono:
2748 format = QImage::Format_Mono;
2749 bytesPerLine = ((glyph->width + 31) & ~31) >> 3;
2751 case QFontEngine::Format_A8:
2752 format = QImage::Format_Alpha8;
2753 bytesPerLine = (glyph->width + 3) & ~3;
2755 case QFontEngine::Format_A32:
2756 format = QImage::Format_RGB32;
2757 bytesPerLine = glyph->width * 4;
2763 QImage img(
static_cast<
const uchar *>(glyph->data), glyph->width, glyph->height, bytesPerLine, format);
2764 if (format == QImage::Format_Mono)
2765 img.setColor(1, QColor(Qt::white).rgba());
2769QFontEngine::Glyph *QFontEngineFT::glyphData(glyph_t glyphIndex,
2770 const QFixedPoint &subPixelPosition,
2771 QFontEngine::GlyphFormat neededFormat,
2772 const QTransform &t)
2774 Q_ASSERT(cacheEnabled);
2777 neededFormat = Format_Mono;
2778 else if (neededFormat == Format_None && defaultFormat != Format_None)
2779 neededFormat = defaultFormat;
2780 else if (neededFormat == Format_None)
2781 neededFormat = Format_A8;
2783 Glyph *glyph = loadGlyphFor(glyphIndex, subPixelPosition, neededFormat, t, QColor());
2784 if (!glyph || !glyph->width || !glyph->height)
2792 return qFuzzyCompare(t.m11(), t.m22()) && qFuzzyCompare(t.m12(), -t.m21())
2793 && qFuzzyCompare(t.m11()*t.m22() - t.m12()*t.m21(), qreal(1.0));
2796QFontEngineFT::Glyph *QFontEngineFT::loadGlyphFor(glyph_t g,
2797 const QFixedPoint &subPixelPosition,
2799 const QTransform &t,
2801 bool fetchBoundingBox,
2802 bool disableOutlineDrawing)
2804 QGlyphSet *glyphSet = loadGlyphSet(t);
2805 if (glyphSet !=
nullptr && glyphSet->outline_drawing && !disableOutlineDrawing && !fetchBoundingBox)
2808 Glyph *glyph = glyphSet !=
nullptr ? glyphSet->getGlyph(g, subPixelPosition) :
nullptr;
2809 if (!glyph || glyph->format != format || (!fetchBoundingBox && !glyph->data)) {
2810 QScopedValueRollback<HintStyle> saved_default_hint_style(default_hint_style);
2811 if (t.type() >= QTransform::TxScale && !is2dRotation(t))
2812 default_hint_style = HintNone;
2815 FT_Matrix m =
this->matrix;
2816 FT_Matrix ftMatrix = glyphSet !=
nullptr ? glyphSet->transformationMatrix : QTransformToFTMatrix(t);
2817 FT_Matrix_Multiply(&ftMatrix, &m);
2818 freetype->matrix = m;
2819 glyph = loadGlyph(glyphSet, g, subPixelPosition, color, format,
false, disableOutlineDrawing);
2826QImage QFontEngineFT::alphaMapForGlyph(glyph_t g,
const QFixedPoint &subPixelPosition)
2828 return alphaMapForGlyph(g, subPixelPosition, QTransform());
2831QImage QFontEngineFT::alphaMapForGlyph(glyph_t g,
2832 const QFixedPoint &subPixelPosition,
2833 const QTransform &t)
2835 const bool needsImageTransform = !FT_IS_SCALABLE(freetype->face)
2836 && t.type() > QTransform::TxTranslate;
2837 const GlyphFormat neededFormat = antialias || needsImageTransform ? Format_A8 : Format_Mono;
2839 Glyph *glyph = loadGlyphFor(g, subPixelPosition, neededFormat, t, QColor(),
false,
true);
2841 QImage img = alphaMapFromGlyphData(glyph, neededFormat);
2842 if (needsImageTransform)
2843 img = img.transformed(t, Qt::FastTransformation);
2847 if (!cacheEnabled && glyph != &emptyGlyph)
2853QImage QFontEngineFT::alphaRGBMapForGlyph(glyph_t g,
2854 const QFixedPoint &subPixelPosition,
2855 const QTransform &t)
2857 if (t.type() > QTransform::TxRotate)
2858 return QFontEngine::alphaRGBMapForGlyph(g, subPixelPosition, t);
2860 const bool needsImageTransform = !FT_IS_SCALABLE(freetype->face)
2861 && t.type() > QTransform::TxTranslate;
2864 const GlyphFormat neededFormat = Format_A32;
2866 Glyph *glyph = loadGlyphFor(g, subPixelPosition, neededFormat, t, QColor(),
false,
true);
2868 QImage img = alphaMapFromGlyphData(glyph, neededFormat);
2869 if (needsImageTransform)
2870 img = img.transformed(t, Qt::FastTransformation);
2874 if (!cacheEnabled && glyph != &emptyGlyph)
2880 return QFontEngine::alphaRGBMapForGlyph(g, subPixelPosition, t);
2883QImage QFontEngineFT::bitmapForGlyph(glyph_t g,
2884 const QFixedPoint &subPixelPosition,
2885 const QTransform &t,
2886 const QColor &color)
2888 Glyph *glyph = loadGlyphFor(g, subPixelPosition, defaultFormat, t, color);
2889 if (glyph ==
nullptr)
2893 if (defaultFormat == GlyphFormat::Format_ARGB)
2894 img = QImage(glyph->data, glyph->width, glyph->height, QImage::Format_ARGB32_Premultiplied).copy();
2895 else if (defaultFormat == GlyphFormat::Format_Mono)
2896 img = QImage(glyph->data, glyph->width, glyph->height, QImage::Format_Mono).copy();
2898 if (!img.isNull() && (scalableBitmapScaleFactor != 1 || (!t.isIdentity() && !isSmoothlyScalable))) {
2899 QTransform trans(t);
2900 const qreal scaleFactor = scalableBitmapScaleFactor.toReal();
2901 trans.scale(scaleFactor, scaleFactor);
2902 img = img.transformed(trans, Qt::SmoothTransformation);
2905 if (!cacheEnabled && glyph != &emptyGlyph)
2911void QFontEngineFT::removeGlyphFromCache(glyph_t glyph)
2913 defaultGlyphSet.removeGlyphFromCache(glyph, QFixedPoint());
2916int QFontEngineFT::glyphCount()
const
2919 FT_Face face = lockFace();
2921 count = face->num_glyphs;
2927FT_Face QFontEngineFT::lockFace(Scaling scale)
const
2930 FT_Face face = freetype->face;
2931 if (scale == Unscaled) {
2932 if (FT_Set_Char_Size(face, face->units_per_EM << 6, face->units_per_EM << 6, 0, 0) == 0) {
2933 freetype->xsize = face->units_per_EM << 6;
2934 freetype->ysize = face->units_per_EM << 6;
2936 }
else if (freetype->xsize != xsize || freetype->ysize != ysize) {
2937 FT_Set_Char_Size(face, xsize, ysize, 0, 0);
2938 freetype->xsize = xsize;
2939 freetype->ysize = ysize;
2941 if (freetype->matrix.xx != matrix.xx ||
2942 freetype->matrix.yy != matrix.yy ||
2943 freetype->matrix.xy != matrix.xy ||
2944 freetype->matrix.yx != matrix.yx) {
2945 freetype->matrix = matrix;
2946 FT_Set_Transform(face, &freetype->matrix,
nullptr);
2952void QFontEngineFT::unlockFace()
const
2957FT_Face QFontEngineFT::non_locked_face()
const
2959 return freetype->face;
2963QFontEngineFT::QGlyphSet::QGlyphSet()
2964 : outline_drawing(
false)
2966 transformationMatrix.xx = 0x10000;
2967 transformationMatrix.yy = 0x10000;
2968 transformationMatrix.xy = 0;
2969 transformationMatrix.yx = 0;
2970 memset(fast_glyph_data, 0,
sizeof(fast_glyph_data));
2971 fast_glyph_count = 0;
2974QFontEngineFT::QGlyphSet::~QGlyphSet()
2979void QFontEngineFT::QGlyphSet::clear()
2981 if (fast_glyph_count > 0) {
2982 for (
int i = 0; i < 256; ++i) {
2983 if (fast_glyph_data[i]) {
2984 delete fast_glyph_data[i];
2985 fast_glyph_data[i] =
nullptr;
2988 fast_glyph_count = 0;
2990 qDeleteAll(glyph_data);
2994void QFontEngineFT::QGlyphSet::removeGlyphFromCache(glyph_t index,
2995 const QFixedPoint &subPixelPosition)
2997 if (useFastGlyphData(index, subPixelPosition)) {
2998 if (fast_glyph_data[index]) {
2999 delete fast_glyph_data[index];
3000 fast_glyph_data[index] =
nullptr;
3001 if (fast_glyph_count > 0)
3005 delete glyph_data.take(GlyphAndSubPixelPosition(index, subPixelPosition));
3009void QFontEngineFT::QGlyphSet::setGlyph(glyph_t index,
3010 const QFixedPoint &subPixelPosition,
3013 if (useFastGlyphData(index, subPixelPosition)) {
3014 if (!fast_glyph_data[index])
3016 fast_glyph_data[index] = glyph;
3018 glyph_data.insert(GlyphAndSubPixelPosition(index, subPixelPosition), glyph);
3022int QFontEngineFT::getPointInOutline(glyph_t glyph,
int flags, quint32 point, QFixed *xpos, QFixed *ypos, quint32 *nPoints)
3025 bool hsubpixel =
true;
3027 int load_flags = loadFlags(
nullptr, Format_A8, flags, hsubpixel, vfactor);
3028 int result = freetype->getPointInOutline(glyph, load_flags, point, xpos, ypos, nPoints);
3033bool QFontEngineFT::initFromFontEngine(
const QFontEngineFT *fe)
3035 if (!init(fe->faceId(), fe->antialias, fe->defaultFormat, fe->freetype))
3040 freetype->ref.ref();
3042 default_load_flags = fe->default_load_flags;
3043 default_hint_style = fe->default_hint_style;
3044 antialias = fe->antialias;
3045 transform = fe->transform;
3046 embolden = fe->embolden;
3047 obliquen = fe->obliquen;
3048 subpixelType = fe->subpixelType;
3049 lcdFilterType = fe->lcdFilterType;
3050 embeddedbitmap = fe->embeddedbitmap;
3055QFontEngine *QFontEngineFT::cloneWithSize(qreal pixelSize)
const
3057 QFontDef fontDef(
this->fontDef);
3058 fontDef.pixelSize = pixelSize;
3059 QFontEngineFT *fe =
new QFontEngineFT(fontDef);
3060 if (!fe->initFromFontEngine(
this)) {
3068Qt::HANDLE QFontEngineFT::handle()
const
3070 return non_locked_face();
3073QList<QFontVariableAxis> QFontEngineFT::variableAxes()
const
3075 return freetype->variableAxes();
QHash< QFontEngine::FaceId, QFreetypeFace * > faces
QList< QFreetypeFace * > staleFaces
QHash< FaceStyle, int > faceIndices
Combined button and popup list for selecting options.
static const QFontEngine::HintStyle ftInitialDefaultHintStyle
static QFontEngine::SubpixelAntialiasingType subpixelAntialiasingTypeHint()
static FT_Matrix QTransformToFTMatrix(const QTransform &matrix)
static void transformBoundingBox(int *left, int *top, int *right, int *bottom, FT_Matrix *matrix)
static void convertRGBToARGB(const uchar *src, uint *dst, int width, int height, int src_pitch, bool bgr)
QtFreetypeData * qt_getFreetypeData()
static bool calculateActualItalic(QFreetypeFace *freetypeFace, FT_Face face, QFontEngine::FaceId faceId)
static bool is2dRotation(const QTransform &t)
static QImage alphaMapFromGlyphData(QFontEngineFT::Glyph *glyph, QFontEngine::GlyphFormat glyphFormat)
static QFontEngineFT::Glyph emptyGlyph
QByteArray qt_fontdata_from_index(int)
size_t qHash(const QtFreetypeData::FaceStyle &style, size_t seed)
static bool ft_getSfntTable(void *user_data, uint tag, uchar *buffer, uint *length)
void qt_addBitmapToPath(qreal x0, qreal y0, const uchar *image_data, int bpl, int w, int h, QPainterPath *path)
static int computeFaceIndex(const QString &faceFileName, const QString &styleName)
bool operator==(const QtFreetypeData::FaceStyle &style1, const QtFreetypeData::FaceStyle &style2)
FT_Library qt_getFreetype()
static FT_UShort calculateActualWeight(QFreetypeFace *freetypeFace, FT_Face face, QFontEngine::FaceId faceId)
static void convertRGBToARGB_V(const uchar *src, uint *dst, int width, int height, int src_pitch, bool bgr)
static void scaleOutline(FT_Face face, FT_GlyphSlot g, FT_Fixed x_scale, FT_Fixed y_scale)
static void dont_delete(void *)
static bool areMetricsTooLarge(const QFontEngineFT::GlyphInfo &info)
FaceStyle(QString faceFileName, QString styleName)