68 TestFontEngine = 0x1000
73 Format_Render = Format_None,
81 DesignMetrics = 0x0002,
82 GlyphIndicesOnly = 0x0004,
83 FullStringFallback = 0x008
85 Q_DECLARE_FLAGS(ShaperFlags, ShaperFlag)
90 ~Glyph() {
delete [] data; }
91 short linearAdvance = 0;
92 unsigned short width = 0;
93 unsigned short height = 0;
97 signed char format = 0;
98 uchar *data =
nullptr;
100 Q_DISABLE_COPY(Glyph)
103 virtual ~QFontEngine();
105 inline Type type()
const {
return m_type; }
110 QByteArray postscriptName;
111 QByteArray copyright;
121 virtual Properties properties()
const;
122 virtual void getUnscaledGlyph(glyph_t glyph, QPainterPath *path, glyph_metrics_t *metrics);
123 QByteArray getSfntTable(uint tag)
const;
124 virtual bool getSfntTableData(uint tag, uchar *buffer, uint *length)
const;
127 FaceId() : index(0), instanceIndex(-1), encoding(0) {}
133 QMap<QFont::Tag,
float> variableAxes;
135 virtual FaceId faceId()
const {
return FaceId(); }
136 enum SynthesizedFlags {
137 SynthesizedItalic = 0x1,
138 SynthesizedBold = 0x2,
139 SynthesizedStretch = 0x4
141 virtual int synthesized()
const {
return 0; }
142 inline bool supportsSubPixelPositions()
const
144 return supportsHorizontalSubPixelPositions() || supportsVerticalSubPixelPositions();
146 virtual bool supportsHorizontalSubPixelPositions()
const {
return false; }
147 virtual bool supportsVerticalSubPixelPositions()
const {
return false; }
148 virtual QFixedPoint subPixelPositionFor(
const QFixedPoint &position)
const;
149 QFixed subPixelPositionForX(QFixed x)
const
151 return subPixelPositionFor(QFixedPoint(x, 0)).x;
154 bool preferTypoLineMetrics()
const;
155 bool isColorFont()
const {
return glyphFormat == Format_ARGB; }
156 static bool isIgnorableChar(
char32_t ucs4)
158 return ucs4 == QChar::LineSeparator
159 || ucs4 == QChar::LineFeed
160 || ucs4 == QChar::CarriageReturn
161 || ucs4 == QChar::ParagraphSeparator
162 || (!disableEmojiSegmenter() && (ucs4 & 0xFFF0) == 0xFE00)
163 || QChar::category(ucs4) == QChar::Other_Control;
166 static int maxCachedGlyphSize();
168 virtual QFixed emSquareSize()
const;
171 virtual glyph_t glyphIndex(uint ucs4)
const = 0;
172 virtual int stringToCMap(
const QChar *str,
int len, QGlyphLayout *glyphs,
int *nglyphs, ShaperFlags flags)
const = 0;
173 virtual void recalcAdvances(QGlyphLayout *, ShaperFlags)
const {}
174 virtual void doKerning(QGlyphLayout *, ShaperFlags)
const;
176 virtual void addGlyphsToPath(glyph_t *glyphs, QFixedPoint *positions,
int nglyphs,
177 QPainterPath *path, QTextItem::RenderFlags flags);
179 void getGlyphPositions(
const QGlyphLayout &glyphs,
const QTransform &matrix, QTextItem::RenderFlags flags,
180 QVarLengthArray<glyph_t> &glyphs_out, QVarLengthArray<QFixedPoint> &positions);
182 virtual void addOutlineToPath(qreal, qreal,
const QGlyphLayout &, QPainterPath *, QTextItem::RenderFlags flags);
183 void addBitmapFontToPath(qreal x, qreal y,
const QGlyphLayout &, QPainterPath *, QTextItem::RenderFlags);
185
186
187
189 virtual QImage alphaMapForGlyph(glyph_t);
190 virtual QImage alphaMapForGlyph(glyph_t glyph,
const QFixedPoint &subPixelPosition);
191 virtual QImage alphaMapForGlyph(glyph_t,
const QTransform &t);
192 virtual QImage alphaMapForGlyph(glyph_t,
const QFixedPoint &subPixelPosition,
const QTransform &t);
193 virtual QImage alphaRGBMapForGlyph(glyph_t,
const QFixedPoint &subPixelPosition,
const QTransform &t);
194 virtual QImage bitmapForGlyph(glyph_t,
const QFixedPoint &subPixelPosition,
const QTransform &t,
const QColor &color = QColor());
195 QImage renderedPathForGlyph(glyph_t glyph,
const QColor &color);
196 virtual Glyph *glyphData(glyph_t glyph,
const QFixedPoint &subPixelPosition, GlyphFormat neededFormat,
const QTransform &t);
197 virtual bool hasInternalCaching()
const {
return false; }
199 virtual glyph_metrics_t alphaMapBoundingBox(glyph_t glyph,
const QFixedPoint &,
const QTransform &matrix, GlyphFormat )
201 return boundingBox(glyph, matrix);
204 virtual void removeGlyphFromCache(glyph_t);
206 virtual glyph_metrics_t boundingBox(
const QGlyphLayout &glyphs);
207 virtual glyph_metrics_t boundingBox(glyph_t glyph) = 0;
208 virtual glyph_metrics_t boundingBox(glyph_t glyph,
const QTransform &matrix);
209 glyph_metrics_t tightBoundingBox(
const QGlyphLayout &glyphs, QTextItem::RenderFlags flags);
211 virtual QFixed ascent()
const;
212 virtual QFixed capHeight()
const = 0;
213 virtual QFixed descent()
const;
214 virtual QFixed leading()
const;
215 virtual QFixed xHeight()
const;
216 virtual QFixed averageCharWidth()
const;
218 virtual QFixed lineThickness()
const;
219 virtual QFixed underlinePosition()
const;
221 virtual qreal maxCharWidth()
const = 0;
222 virtual qreal minLeftBearing()
const;
223 virtual qreal minRightBearing()
const;
225 virtual void getGlyphBearings(glyph_t glyph, qreal *leftBearing =
nullptr, qreal *rightBearing =
nullptr);
227 inline bool canRender(uint ucs4)
const {
return glyphIndex(ucs4) != 0; }
228 virtual bool canRender(
const QChar *str,
int len)
const;
230 virtual bool supportsTransformation(
const QTransform &transform)
const;
232 virtual int glyphCount()
const;
233 virtual int glyphMargin(GlyphFormat format) {
return format == Format_A32 ? 2 : 0; }
234 bool hasHinting()
const;
236 virtual QFontEngine *cloneWithSize(qreal )
const {
return nullptr; }
238 virtual Qt::HANDLE handle()
const;
240 virtual QList<QFontVariableAxis> variableAxes()
const;
242 virtual QString glyphName(glyph_t index)
const;
243 virtual glyph_t findGlyph(QLatin1StringView name)
const;
245 void *harfbuzzFont()
const;
246 void *harfbuzzFace()
const;
247 bool supportsScript(QChar::Script script)
const;
249 inline static bool scriptRequiresOpenType(QChar::Script script)
251 return ((script >= QChar::Script_Syriac && script <= QChar::Script_Sinhala)
252 || script == QChar::Script_Khmer || script == QChar::Script_Nko);
255 virtual int getPointInOutline(glyph_t glyph,
int flags, quint32 point, QFixed *xpos, QFixed *ypos, quint32 *nPoints);
257 void clearGlyphCache(
const void *key);
258 void setGlyphCache(
const void *key, QFontEngineGlyphCache *data);
259 QFontEngineGlyphCache *glyphCache(
const void *key, GlyphFormat format,
const QTransform &transform,
const QColor &color = QColor())
const;
261 static const uchar *getCMap(
const uchar *table, uint tableSize,
bool *isSymbolFont,
int *cmapSize);
262 static quint32 getTrueTypeGlyphIndex(
const uchar *cmap,
int cmapSize, uint unicode);
264 static QByteArray convertToPostscriptFontFamilyName(
const QByteArray &fontFamily);
266 virtual bool hasUnreliableGlyphOutline()
const;
267 virtual bool expectsGammaCorrectedBlending(QFontEngine::GlyphFormat format)
const;
269 static bool disableEmojiSegmenter();
277 virtual void setDefaultHintStyle(HintStyle) { }
279 enum SubpixelAntialiasingType {
296 qt_destroy_func_t destroy_func;
298 Holder() : ptr(
nullptr), destroy_func(
nullptr) {}
299 explicit Holder(
void *p, qt_destroy_func_t d) : ptr(p), destroy_func(d) {}
300 ~Holder() {
if (ptr && destroy_func) destroy_func(ptr); }
301 Holder(Holder &&other)
noexcept
302 : ptr(std::exchange(other.ptr,
nullptr)),
303 destroy_func(std::exchange(other.destroy_func,
nullptr))
306 QT_MOVE_ASSIGNMENT_OPERATOR_IMPL_VIA_PURE_SWAP(Holder)
308 void swap(Holder &other)
noexcept
310 qSwap(ptr, other.ptr);
311 qSwap(destroy_func, other.destroy_func);
314 void *get()
const noexcept {
return ptr; }
315 void *release()
noexcept {
318 destroy_func =
nullptr;
321 void reset()
noexcept { Holder().swap(*
this); }
322 qt_destroy_func_t get_deleter()
const noexcept {
return destroy_func; }
324 bool operator!()
const noexcept {
return !ptr; }
327 mutable Holder font_;
328 mutable Holder face_;
332 qt_get_font_table_func_t get_font_table;
338 bool isSmoothlyScalable;
343 inline bool operator<(
const KernPair &other)
const
345 return left_right < other.left_right;
348 QList<KernPair> kerning_pairs;
349 void loadKerningPairs(QFixed scalingFactor);
351 GlyphFormat glyphFormat;
352 int m_subPixelPositionCount;
355 explicit QFontEngine(Type type);
357 QFixed firstLeftBearing(
const QGlyphLayout &glyphs);
358 QFixed lastRightBearing(
const QGlyphLayout &glyphs);
360 QFixed calculatedCapHeight()
const;
362 mutable QFixed m_ascent;
363 mutable QFixed m_descent;
364 mutable QFixed m_leading;
365 mutable bool m_heightMetricsQueried;
367 virtual void initializeHeightMetrics()
const;
368 bool processHheaTable()
const;
369 bool processOS2Table()
const;
372 struct GlyphCacheEntry {
374 GlyphCacheEntry(
const GlyphCacheEntry &);
377 GlyphCacheEntry &operator=(
const GlyphCacheEntry &);
379 QExplicitlySharedDataPointer<QFontEngineGlyphCache> cache;
380 bool operator==(
const GlyphCacheEntry &other)
const {
return cache == other.cache; }
382 typedef std::list<GlyphCacheEntry> GlyphCaches;
383 mutable QHash<
const void *, GlyphCaches> m_glyphCaches;
386 mutable qreal m_minLeftBearing;
387 mutable qreal m_minRightBearing;
388 mutable int m_hasHinting = -1;
457 explicit QFontEngineMulti(QFontEngine *engine,
int script,
const QStringList &fallbackFamilies = QStringList());
460 virtual int glyphCount()
const override;
461 virtual glyph_t glyphIndex(uint ucs4)
const override;
462 virtual int stringToCMap(
const QChar *str,
int len, QGlyphLayout *glyphs,
int *nglyphs, ShaperFlags flags)
const override;
464 virtual glyph_metrics_t boundingBox(
const QGlyphLayout &glyphs) override;
465 virtual glyph_metrics_t boundingBox(glyph_t glyph) override;
467 virtual void recalcAdvances(QGlyphLayout *, ShaperFlags)
const override;
468 virtual void doKerning(QGlyphLayout *, ShaperFlags)
const override;
469 virtual void addOutlineToPath(qreal, qreal,
const QGlyphLayout &, QPainterPath *, QTextItem::RenderFlags flags) override;
470 virtual void getGlyphBearings(glyph_t glyph, qreal *leftBearing =
nullptr, qreal *rightBearing =
nullptr) override;
472 virtual QFixed emSquareSize()
const override;
473 virtual QFixed ascent()
const override;
474 virtual QFixed capHeight()
const override;
475 virtual QFixed descent()
const override;
476 virtual QFixed leading()
const override;
477 virtual QFixed xHeight()
const override;
478 virtual QFixed averageCharWidth()
const override;
479 virtual QImage alphaMapForGlyph(glyph_t) override;
480 virtual QImage alphaMapForGlyph(glyph_t glyph,
const QFixedPoint &subPixelPosition) override;
481 virtual QImage alphaMapForGlyph(glyph_t,
const QTransform &t) override;
482 virtual QImage alphaMapForGlyph(glyph_t,
const QFixedPoint &subPixelPosition,
const QTransform &t) override;
483 virtual QImage alphaRGBMapForGlyph(glyph_t,
const QFixedPoint &subPixelPosition,
const QTransform &t) override;
485 virtual QFixed lineThickness()
const override;
486 virtual QFixed underlinePosition()
const override;
487 virtual qreal maxCharWidth()
const override;
488 virtual qreal minLeftBearing()
const override;
489 virtual qreal minRightBearing()
const override;
491 virtual QList<QFontVariableAxis> variableAxes()
const override;
493 virtual bool canRender(
const QChar *string,
int len)
const override;
494 QString glyphName(glyph_t glyph)
const override;
495 glyph_t findGlyph(QLatin1StringView name)
const override;
497 inline int fallbackFamilyCount()
const {
return m_fallbackFamilies.size(); }
498 inline QString fallbackFamilyAt(
int at)
const {
return m_fallbackFamilies.at(at); }
500 void setFallbackFamiliesList(
const QStringList &fallbackFamilies);
502 static uchar highByte(glyph_t glyph);
504 inline QFontEngine *engine(
int at)
const
505 { Q_ASSERT(at < m_engines.size());
return m_engines.at(at); }
507 void ensureEngineAt(
int at);
509 static QFontEngine *createMultiFontEngine(QFontEngine *fe,
int script);
512 virtual void ensureFallbackFamiliesQueried();
513 virtual bool shouldLoadFontEngineForCharacter(
int at, uint ucs4)
const;
514 virtual QFontEngine *loadEngine(
int at);
517 QList<QFontEngine *> m_engines;
518 QStringList m_fallbackFamilies;
520 bool m_fallbackFamiliesQueried;