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 virtual QFixed emSquareSize()
const;
169 virtual glyph_t glyphIndex(uint ucs4)
const = 0;
170 virtual int stringToCMap(
const QChar *str,
int len, QGlyphLayout *glyphs,
int *nglyphs, ShaperFlags flags)
const = 0;
171 virtual void recalcAdvances(QGlyphLayout *, ShaperFlags)
const {}
172 virtual void doKerning(QGlyphLayout *, ShaperFlags)
const;
174 virtual void addGlyphsToPath(glyph_t *glyphs, QFixedPoint *positions,
int nglyphs,
175 QPainterPath *path, QTextItem::RenderFlags flags);
177 void getGlyphPositions(
const QGlyphLayout &glyphs,
const QTransform &matrix, QTextItem::RenderFlags flags,
178 QVarLengthArray<glyph_t> &glyphs_out, QVarLengthArray<QFixedPoint> &positions);
180 virtual void addOutlineToPath(qreal, qreal,
const QGlyphLayout &, QPainterPath *, QTextItem::RenderFlags flags);
181 void addBitmapFontToPath(qreal x, qreal y,
const QGlyphLayout &, QPainterPath *, QTextItem::RenderFlags);
183
184
185
187 virtual QImage alphaMapForGlyph(glyph_t);
188 virtual QImage alphaMapForGlyph(glyph_t glyph,
const QFixedPoint &subPixelPosition);
189 virtual QImage alphaMapForGlyph(glyph_t,
const QTransform &t);
190 virtual QImage alphaMapForGlyph(glyph_t,
const QFixedPoint &subPixelPosition,
const QTransform &t);
191 virtual QImage alphaRGBMapForGlyph(glyph_t,
const QFixedPoint &subPixelPosition,
const QTransform &t);
192 virtual QImage bitmapForGlyph(glyph_t,
const QFixedPoint &subPixelPosition,
const QTransform &t,
const QColor &color = QColor());
193 QImage renderedPathForGlyph(glyph_t glyph,
const QColor &color);
194 virtual Glyph *glyphData(glyph_t glyph,
const QFixedPoint &subPixelPosition, GlyphFormat neededFormat,
const QTransform &t);
195 virtual bool hasInternalCaching()
const {
return false; }
197 virtual glyph_metrics_t alphaMapBoundingBox(glyph_t glyph,
const QFixedPoint &,
const QTransform &matrix, GlyphFormat )
199 return boundingBox(glyph, matrix);
202 virtual void removeGlyphFromCache(glyph_t);
204 virtual glyph_metrics_t boundingBox(
const QGlyphLayout &glyphs);
205 virtual glyph_metrics_t boundingBox(glyph_t glyph) = 0;
206 virtual glyph_metrics_t boundingBox(glyph_t glyph,
const QTransform &matrix);
207 glyph_metrics_t tightBoundingBox(
const QGlyphLayout &glyphs);
209 virtual QFixed ascent()
const;
210 virtual QFixed capHeight()
const = 0;
211 virtual QFixed descent()
const;
212 virtual QFixed leading()
const;
213 virtual QFixed xHeight()
const;
214 virtual QFixed averageCharWidth()
const;
216 virtual QFixed lineThickness()
const;
217 virtual QFixed underlinePosition()
const;
219 virtual qreal maxCharWidth()
const = 0;
220 virtual qreal minLeftBearing()
const;
221 virtual qreal minRightBearing()
const;
223 virtual void getGlyphBearings(glyph_t glyph, qreal *leftBearing =
nullptr, qreal *rightBearing =
nullptr);
225 inline bool canRender(uint ucs4)
const {
return glyphIndex(ucs4) != 0; }
226 virtual bool canRender(
const QChar *str,
int len)
const;
228 virtual bool supportsTransformation(
const QTransform &transform)
const;
230 virtual int glyphCount()
const;
231 virtual int glyphMargin(GlyphFormat format) {
return format == Format_A32 ? 2 : 0; }
233 virtual QFontEngine *cloneWithSize(qreal )
const {
return nullptr; }
235 virtual Qt::HANDLE handle()
const;
237 virtual QList<QFontVariableAxis> variableAxes()
const;
239 virtual QString glyphName(glyph_t index)
const;
240 virtual glyph_t findGlyph(QLatin1StringView name)
const;
242 void *harfbuzzFont()
const;
243 void *harfbuzzFace()
const;
244 bool supportsScript(QChar::Script script)
const;
246 inline static bool scriptRequiresOpenType(QChar::Script script)
248 return ((script >= QChar::Script_Syriac && script <= QChar::Script_Sinhala)
249 || script == QChar::Script_Khmer || script == QChar::Script_Nko);
252 virtual int getPointInOutline(glyph_t glyph,
int flags, quint32 point, QFixed *xpos, QFixed *ypos, quint32 *nPoints);
254 void clearGlyphCache(
const void *key);
255 void setGlyphCache(
const void *key, QFontEngineGlyphCache *data);
256 QFontEngineGlyphCache *glyphCache(
const void *key, GlyphFormat format,
const QTransform &transform,
const QColor &color = QColor())
const;
258 static const uchar *getCMap(
const uchar *table, uint tableSize,
bool *isSymbolFont,
int *cmapSize);
259 static quint32 getTrueTypeGlyphIndex(
const uchar *cmap,
int cmapSize, uint unicode);
261 static QByteArray convertToPostscriptFontFamilyName(
const QByteArray &fontFamily);
263 virtual bool hasUnreliableGlyphOutline()
const;
264 virtual bool expectsGammaCorrectedBlending()
const;
266 static bool disableEmojiSegmenter();
274 virtual void setDefaultHintStyle(HintStyle) { }
276 enum SubpixelAntialiasingType {
293 qt_destroy_func_t destroy_func;
295 Holder() : ptr(
nullptr), destroy_func(
nullptr) {}
296 explicit Holder(
void *p, qt_destroy_func_t d) : ptr(p), destroy_func(d) {}
297 ~Holder() {
if (ptr && destroy_func) destroy_func(ptr); }
298 Holder(Holder &&other)
noexcept
299 : ptr(std::exchange(other.ptr,
nullptr)),
300 destroy_func(std::exchange(other.destroy_func,
nullptr))
303 QT_MOVE_ASSIGNMENT_OPERATOR_IMPL_VIA_PURE_SWAP(Holder)
305 void swap(Holder &other)
noexcept
307 qSwap(ptr, other.ptr);
308 qSwap(destroy_func, other.destroy_func);
311 void *get()
const noexcept {
return ptr; }
312 void *release()
noexcept {
315 destroy_func =
nullptr;
318 void reset()
noexcept { Holder().swap(*
this); }
319 qt_destroy_func_t get_deleter()
const noexcept {
return destroy_func; }
321 bool operator!()
const noexcept {
return !ptr; }
324 mutable Holder font_;
325 mutable Holder face_;
329 qt_get_font_table_func_t get_font_table;
335 bool isSmoothlyScalable;
340 inline bool operator<(
const KernPair &other)
const
342 return left_right < other.left_right;
345 QList<KernPair> kerning_pairs;
346 void loadKerningPairs(QFixed scalingFactor);
348 GlyphFormat glyphFormat;
349 int m_subPixelPositionCount;
352 explicit QFontEngine(Type type);
354 QFixed firstLeftBearing(
const QGlyphLayout &glyphs);
355 QFixed lastRightBearing(
const QGlyphLayout &glyphs);
357 QFixed calculatedCapHeight()
const;
359 mutable QFixed m_ascent;
360 mutable QFixed m_descent;
361 mutable QFixed m_leading;
362 mutable bool m_heightMetricsQueried;
364 virtual void initializeHeightMetrics()
const;
365 bool processHheaTable()
const;
366 bool processOS2Table()
const;
369 struct GlyphCacheEntry {
371 GlyphCacheEntry(
const GlyphCacheEntry &);
374 GlyphCacheEntry &operator=(
const GlyphCacheEntry &);
376 QExplicitlySharedDataPointer<QFontEngineGlyphCache> cache;
377 bool operator==(
const GlyphCacheEntry &other)
const {
return cache == other.cache; }
379 typedef std::list<GlyphCacheEntry> GlyphCaches;
380 mutable QHash<
const void *, GlyphCaches> m_glyphCaches;
383 mutable qreal m_minLeftBearing;
384 mutable qreal m_minRightBearing;
453 explicit QFontEngineMulti(QFontEngine *engine,
int script,
const QStringList &fallbackFamilies = QStringList());
456 virtual int glyphCount()
const override;
457 virtual glyph_t glyphIndex(uint ucs4)
const override;
458 virtual int stringToCMap(
const QChar *str,
int len, QGlyphLayout *glyphs,
int *nglyphs, ShaperFlags flags)
const override;
460 virtual glyph_metrics_t boundingBox(
const QGlyphLayout &glyphs) override;
461 virtual glyph_metrics_t boundingBox(glyph_t glyph) override;
463 virtual void recalcAdvances(QGlyphLayout *, ShaperFlags)
const override;
464 virtual void doKerning(QGlyphLayout *, ShaperFlags)
const override;
465 virtual void addOutlineToPath(qreal, qreal,
const QGlyphLayout &, QPainterPath *, QTextItem::RenderFlags flags) override;
466 virtual void getGlyphBearings(glyph_t glyph, qreal *leftBearing =
nullptr, qreal *rightBearing =
nullptr) override;
468 virtual QFixed emSquareSize()
const override;
469 virtual QFixed ascent()
const override;
470 virtual QFixed capHeight()
const override;
471 virtual QFixed descent()
const override;
472 virtual QFixed leading()
const override;
473 virtual QFixed xHeight()
const override;
474 virtual QFixed averageCharWidth()
const override;
475 virtual QImage alphaMapForGlyph(glyph_t) override;
476 virtual QImage alphaMapForGlyph(glyph_t glyph,
const QFixedPoint &subPixelPosition) override;
477 virtual QImage alphaMapForGlyph(glyph_t,
const QTransform &t) override;
478 virtual QImage alphaMapForGlyph(glyph_t,
const QFixedPoint &subPixelPosition,
const QTransform &t) override;
479 virtual QImage alphaRGBMapForGlyph(glyph_t,
const QFixedPoint &subPixelPosition,
const QTransform &t) override;
481 virtual QFixed lineThickness()
const override;
482 virtual QFixed underlinePosition()
const override;
483 virtual qreal maxCharWidth()
const override;
484 virtual qreal minLeftBearing()
const override;
485 virtual qreal minRightBearing()
const override;
487 virtual QList<QFontVariableAxis> variableAxes()
const override;
489 virtual bool canRender(
const QChar *string,
int len)
const override;
490 QString glyphName(glyph_t glyph)
const override;
491 glyph_t findGlyph(QLatin1StringView name)
const override;
493 inline int fallbackFamilyCount()
const {
return m_fallbackFamilies.size(); }
494 inline QString fallbackFamilyAt(
int at)
const {
return m_fallbackFamilies.at(at); }
496 void setFallbackFamiliesList(
const QStringList &fallbackFamilies);
498 static uchar highByte(glyph_t glyph);
500 inline QFontEngine *engine(
int at)
const
501 { Q_ASSERT(at < m_engines.size());
return m_engines.at(at); }
503 void ensureEngineAt(
int at);
505 static QFontEngine *createMultiFontEngine(QFontEngine *fe,
int script);
508 virtual void ensureFallbackFamiliesQueried();
509 virtual bool shouldLoadFontEngineForCharacter(
int at, uint ucs4)
const;
510 virtual QFontEngine *loadEngine(
int at);
513 QList<QFontEngine *> m_engines;
514 QStringList m_fallbackFamilies;
516 bool m_fallbackFamiliesQueried;