Qt
Internal/Contributor docs for the Qt SDK. Note: These are NOT official API docs; those are found at https://doc.qt.io/
Loading...
Searching...
No Matches
qharfbuzzng.cpp
Go to the documentation of this file.
1// Copyright (C) 2020 The Qt Company Ltd.
2// Copyright (C) 2013 Konstantin Ritt
3// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
4
6
7#include <qstring.h>
8
9#include <private/qstringiterator_p.h>
10
11#include "qfontengine_p.h"
12
14
15// Unicode routines
16
18 HB_SCRIPT_UNKNOWN,
19 HB_SCRIPT_INHERITED,
20 HB_SCRIPT_COMMON,
21
22 HB_SCRIPT_LATIN,
23 HB_SCRIPT_GREEK,
24 HB_SCRIPT_CYRILLIC,
25 HB_SCRIPT_ARMENIAN,
26 HB_SCRIPT_HEBREW,
27 HB_SCRIPT_ARABIC,
28 HB_SCRIPT_SYRIAC,
29 HB_SCRIPT_THAANA,
30 HB_SCRIPT_DEVANAGARI,
31 HB_SCRIPT_BENGALI,
32 HB_SCRIPT_GURMUKHI,
33 HB_SCRIPT_GUJARATI,
34 HB_SCRIPT_ORIYA,
35 HB_SCRIPT_TAMIL,
36 HB_SCRIPT_TELUGU,
37 HB_SCRIPT_KANNADA,
38 HB_SCRIPT_MALAYALAM,
39 HB_SCRIPT_SINHALA,
40 HB_SCRIPT_THAI,
41 HB_SCRIPT_LAO,
42 HB_SCRIPT_TIBETAN,
43 HB_SCRIPT_MYANMAR,
44 HB_SCRIPT_GEORGIAN,
45 HB_SCRIPT_HANGUL,
46 HB_SCRIPT_ETHIOPIC,
47 HB_SCRIPT_CHEROKEE,
48 HB_SCRIPT_CANADIAN_SYLLABICS,
49 HB_SCRIPT_OGHAM,
50 HB_SCRIPT_RUNIC,
51 HB_SCRIPT_KHMER,
52 HB_SCRIPT_MONGOLIAN,
53 HB_SCRIPT_HIRAGANA,
54 HB_SCRIPT_KATAKANA,
55 HB_SCRIPT_BOPOMOFO,
56 HB_SCRIPT_HAN,
57 HB_SCRIPT_YI,
58 HB_SCRIPT_OLD_ITALIC,
59 HB_SCRIPT_GOTHIC,
60 HB_SCRIPT_DESERET,
61 HB_SCRIPT_TAGALOG,
62 HB_SCRIPT_HANUNOO,
63 HB_SCRIPT_BUHID,
64 HB_SCRIPT_TAGBANWA,
65 HB_SCRIPT_COPTIC,
66
67 // Unicode 4.0 additions
68 HB_SCRIPT_LIMBU,
69 HB_SCRIPT_TAI_LE,
70 HB_SCRIPT_LINEAR_B,
71 HB_SCRIPT_UGARITIC,
72 HB_SCRIPT_SHAVIAN,
73 HB_SCRIPT_OSMANYA,
74 HB_SCRIPT_CYPRIOT,
75 HB_SCRIPT_BRAILLE,
76
77 // Unicode 4.1 additions
78 HB_SCRIPT_BUGINESE,
79 HB_SCRIPT_NEW_TAI_LUE,
80 HB_SCRIPT_GLAGOLITIC,
81 HB_SCRIPT_TIFINAGH,
82 HB_SCRIPT_SYLOTI_NAGRI,
83 HB_SCRIPT_OLD_PERSIAN,
84 HB_SCRIPT_KHAROSHTHI,
85
86 // Unicode 5.0 additions
87 HB_SCRIPT_BALINESE,
88 HB_SCRIPT_CUNEIFORM,
89 HB_SCRIPT_PHOENICIAN,
90 HB_SCRIPT_PHAGS_PA,
91 HB_SCRIPT_NKO,
92
93 // Unicode 5.1 additions
94 HB_SCRIPT_SUNDANESE,
95 HB_SCRIPT_LEPCHA,
96 HB_SCRIPT_OL_CHIKI,
97 HB_SCRIPT_VAI,
98 HB_SCRIPT_SAURASHTRA,
99 HB_SCRIPT_KAYAH_LI,
100 HB_SCRIPT_REJANG,
101 HB_SCRIPT_LYCIAN,
102 HB_SCRIPT_CARIAN,
103 HB_SCRIPT_LYDIAN,
104 HB_SCRIPT_CHAM,
105
106 // Unicode 5.2 additions
107 HB_SCRIPT_TAI_THAM,
108 HB_SCRIPT_TAI_VIET,
109 HB_SCRIPT_AVESTAN,
110 HB_SCRIPT_EGYPTIAN_HIEROGLYPHS,
111 HB_SCRIPT_SAMARITAN,
112 HB_SCRIPT_LISU,
113 HB_SCRIPT_BAMUM,
114 HB_SCRIPT_JAVANESE,
115 HB_SCRIPT_MEETEI_MAYEK,
116 HB_SCRIPT_IMPERIAL_ARAMAIC,
117 HB_SCRIPT_OLD_SOUTH_ARABIAN,
118 HB_SCRIPT_INSCRIPTIONAL_PARTHIAN,
119 HB_SCRIPT_INSCRIPTIONAL_PAHLAVI,
120 HB_SCRIPT_OLD_TURKIC,
121 HB_SCRIPT_KAITHI,
122
123 // Unicode 6.0 additions
124 HB_SCRIPT_BATAK,
125 HB_SCRIPT_BRAHMI,
126 HB_SCRIPT_MANDAIC,
127
128 // Unicode 6.1 additions
129 HB_SCRIPT_CHAKMA,
130 HB_SCRIPT_MEROITIC_CURSIVE,
131 HB_SCRIPT_MEROITIC_HIEROGLYPHS,
132 HB_SCRIPT_MIAO,
133 HB_SCRIPT_SHARADA,
134 HB_SCRIPT_SORA_SOMPENG,
135 HB_SCRIPT_TAKRI,
136
137 // Unicode 7.0 additions
138 HB_SCRIPT_CAUCASIAN_ALBANIAN,
139 HB_SCRIPT_BASSA_VAH,
140 HB_SCRIPT_DUPLOYAN,
141 HB_SCRIPT_ELBASAN,
142 HB_SCRIPT_GRANTHA,
143 HB_SCRIPT_PAHAWH_HMONG,
144 HB_SCRIPT_KHOJKI,
145 HB_SCRIPT_LINEAR_A,
146 HB_SCRIPT_MAHAJANI,
147 HB_SCRIPT_MANICHAEAN,
148 HB_SCRIPT_MENDE_KIKAKUI,
149 HB_SCRIPT_MODI,
150 HB_SCRIPT_MRO,
151 HB_SCRIPT_OLD_NORTH_ARABIAN,
152 HB_SCRIPT_NABATAEAN,
153 HB_SCRIPT_PALMYRENE,
154 HB_SCRIPT_PAU_CIN_HAU,
155 HB_SCRIPT_OLD_PERMIC,
156 HB_SCRIPT_PSALTER_PAHLAVI,
157 HB_SCRIPT_SIDDHAM,
158 HB_SCRIPT_KHUDAWADI,
159 HB_SCRIPT_TIRHUTA,
160 HB_SCRIPT_WARANG_CITI,
161
162 // Unicode 8.0 additions
163 HB_SCRIPT_AHOM,
164 HB_SCRIPT_ANATOLIAN_HIEROGLYPHS,
165 HB_SCRIPT_HATRAN,
166 HB_SCRIPT_MULTANI,
167 HB_SCRIPT_OLD_HUNGARIAN,
168 HB_SCRIPT_SIGNWRITING,
169
170 // Unicode 9.0 additions
171 HB_SCRIPT_ADLAM,
172 HB_SCRIPT_BHAIKSUKI,
173 HB_SCRIPT_MARCHEN,
174 HB_SCRIPT_NEWA,
175 HB_SCRIPT_OSAGE,
176 HB_SCRIPT_TANGUT,
177
178 // Unicode 10.0 additions
179 HB_SCRIPT_MASARAM_GONDI,
180 HB_SCRIPT_NUSHU,
181 HB_SCRIPT_SOYOMBO,
182 HB_SCRIPT_ZANABAZAR_SQUARE,
183
184 // Unicode 11.0 additions
185 HB_SCRIPT_DOGRA,
186 HB_SCRIPT_GUNJALA_GONDI,
187 HB_SCRIPT_HANIFI_ROHINGYA,
188 HB_SCRIPT_MAKASAR,
189 HB_SCRIPT_MEDEFAIDRIN,
190 HB_SCRIPT_OLD_SOGDIAN,
191 HB_SCRIPT_SOGDIAN,
192
193 // Unicode 12.0 additions
194 HB_SCRIPT_ELYMAIC,
195 HB_SCRIPT_NANDINAGARI,
196 HB_SCRIPT_NYIAKENG_PUACHUE_HMONG,
197 HB_SCRIPT_WANCHO,
198
199 // Unicode 13.0 additions (not present in harfbuzz-ng 2.6.6 and earlier)
200#if !HB_VERSION_ATLEAST(2, 6, 7)
201 hb_script_t(HB_TAG('C', 'h', 'r', 's')), // Script_Chorasmian
202 hb_script_t(HB_TAG('D', 'i', 'a', 'k')), // Script_DivesAkuru
203 hb_script_t(HB_TAG('K', 'i', 't', 's')), // Script_KhitanSmallScript
204 hb_script_t(HB_TAG('Y', 'e', 'z', 'i')), // Script_Yezidi
205#else
206 HB_SCRIPT_CHORASMIAN,
207 HB_SCRIPT_DIVES_AKURU,
208 HB_SCRIPT_KHITAN_SMALL_SCRIPT,
209 HB_SCRIPT_YEZIDI,
210#endif
211 // Unicode 14.0 additions (not present in harfbuzz-ng 2.9.1 and earlier)
212#if !HB_VERSION_ATLEAST(3, 0, 0)
213 hb_script_t(HB_TAG('C','p','m','n')), // Script_CyproMinoan
214 hb_script_t(HB_TAG('O','u','g','r')), // Script_OldUyghur
215 hb_script_t(HB_TAG('T','n','s','a')), // Script_Tangsa
216 hb_script_t(HB_TAG('T','o','t','o')), // Script_Toto
217 hb_script_t(HB_TAG('V','i','t','h')), // Script_Vithkuqi
218#else
219 HB_SCRIPT_CYPRO_MINOAN,
220 HB_SCRIPT_OLD_UYGHUR,
221 HB_SCRIPT_TANGSA,
222 HB_SCRIPT_TOTO,
223 HB_SCRIPT_VITHKUQI,
224#endif
225 // Unicode 15.0 additions (not present in harfbuzz-ng 5.1.0 and earlier)
226#if !HB_VERSION_ATLEAST(5, 2, 0)
227 hb_script_t(HB_TAG('K','a','w','i')), // Script_Kawi
228 hb_script_t(HB_TAG('N','a','g','m')), // Script_NagMundari
229#else
230 HB_SCRIPT_KAWI,
231 HB_SCRIPT_NAG_MUNDARI,
232#endif
233
234 // Unicode 16.0 additions (not present in harfbuzz-ng 9.0.0 and earlier)
235#if !HB_VERSION_ATLEAST(9, 1, 0)
236 hb_script_t(HB_TAG('G','a','r','a')), // Script_Garay
237 hb_script_t(HB_TAG('G','u','k','h')), // Script_GurungKhema
238 hb_script_t(HB_TAG('K','r','a','i')), // Script_KiratRai
239 hb_script_t(HB_TAG('O','n','a','o')), // Script_OlOnal
240 hb_script_t(HB_TAG('S','u','n','u')), // Script_Sunuwar
241 hb_script_t(HB_TAG('T','o','d','r')), // Script_Todhri
242 hb_script_t(HB_TAG('T','u','t','g')), // Script_TuluTigalari
243#else
244 HB_SCRIPT_GARAY,
245 HB_SCRIPT_GURUNG_KHEMA,
246 HB_SCRIPT_KIRAT_RAI,
247 HB_SCRIPT_OL_ONAL,
248 HB_SCRIPT_SUNUWAR,
249 HB_SCRIPT_TODHRI,
250 HB_SCRIPT_TULU_TIGALARI,
251#endif
252#if !HB_VERSION_ATLEAST(11, 5, 0)
253 hb_script_t(HB_TAG('B','e','r','f')), // Script_BeriaErfe
254 hb_script_t(HB_TAG('S','i','d','t')), // Script_Sidetic
255 hb_script_t(HB_TAG('T','a','y','o')), // Script_TaiYo
256 hb_script_t(HB_TAG('T','o','l','s')), // Script_TolongSiki
257#else
258 HB_SCRIPT_BERIA_ERFE,
259 HB_SCRIPT_SIDETIC,
260 HB_SCRIPT_TAI_YO,
261 HB_SCRIPT_TOLONG_SIKI,
262#endif
263};
264static_assert(QChar::ScriptCount == sizeof(_qtscript_to_hbscript) / sizeof(_qtscript_to_hbscript[0]));
265
267{
268 return _qtscript_to_hbscript[script];
269}
270
272{
273 uint i = QChar::ScriptCount - 1;
274 while (i > QChar::Script_Unknown && _qtscript_to_hbscript[i] != script)
275 --i;
276 return QChar::Script(i);
277}
278
279
281_hb_qt_unicode_combining_class(hb_unicode_funcs_t * /*ufuncs*/,
282 hb_codepoint_t unicode,
283 void * /*user_data*/)
284{
285 return hb_unicode_combining_class_t(QChar::combiningClass(unicode));
286}
287
289 HB_UNICODE_GENERAL_CATEGORY_NON_SPACING_MARK, // Mn
290 HB_UNICODE_GENERAL_CATEGORY_SPACING_MARK, // Mc
291 HB_UNICODE_GENERAL_CATEGORY_ENCLOSING_MARK, // Me
292
293 HB_UNICODE_GENERAL_CATEGORY_DECIMAL_NUMBER, // Nd
294 HB_UNICODE_GENERAL_CATEGORY_LETTER_NUMBER, // Nl
295 HB_UNICODE_GENERAL_CATEGORY_OTHER_NUMBER, // No
296
297 HB_UNICODE_GENERAL_CATEGORY_SPACE_SEPARATOR, // Zs
298 HB_UNICODE_GENERAL_CATEGORY_LINE_SEPARATOR, // Zl
299 HB_UNICODE_GENERAL_CATEGORY_PARAGRAPH_SEPARATOR, // Zp
300
301 HB_UNICODE_GENERAL_CATEGORY_CONTROL, // Cc
302 HB_UNICODE_GENERAL_CATEGORY_FORMAT, // Cf
303 HB_UNICODE_GENERAL_CATEGORY_SURROGATE, // Cs
304 HB_UNICODE_GENERAL_CATEGORY_PRIVATE_USE, // Co
305 HB_UNICODE_GENERAL_CATEGORY_UNASSIGNED, // Cn
306
307 HB_UNICODE_GENERAL_CATEGORY_UPPERCASE_LETTER, // Lu
308 HB_UNICODE_GENERAL_CATEGORY_LOWERCASE_LETTER, // Ll
309 HB_UNICODE_GENERAL_CATEGORY_TITLECASE_LETTER, // Lt
310 HB_UNICODE_GENERAL_CATEGORY_MODIFIER_LETTER, // Lm
311 HB_UNICODE_GENERAL_CATEGORY_OTHER_LETTER, // Lo
312
313 HB_UNICODE_GENERAL_CATEGORY_CONNECT_PUNCTUATION, // Pc
314 HB_UNICODE_GENERAL_CATEGORY_DASH_PUNCTUATION, // Pd
315 HB_UNICODE_GENERAL_CATEGORY_OPEN_PUNCTUATION, // Ps
316 HB_UNICODE_GENERAL_CATEGORY_CLOSE_PUNCTUATION, // Pe
317 HB_UNICODE_GENERAL_CATEGORY_INITIAL_PUNCTUATION, // Pi
318 HB_UNICODE_GENERAL_CATEGORY_FINAL_PUNCTUATION, // Pf
319 HB_UNICODE_GENERAL_CATEGORY_OTHER_PUNCTUATION, // Po
320
321 HB_UNICODE_GENERAL_CATEGORY_MATH_SYMBOL, // Sm
322 HB_UNICODE_GENERAL_CATEGORY_CURRENCY_SYMBOL, // Sc
323 HB_UNICODE_GENERAL_CATEGORY_MODIFIER_SYMBOL, // Sk
324 HB_UNICODE_GENERAL_CATEGORY_OTHER_SYMBOL // So
325};
326
328_hb_qt_unicode_general_category(hb_unicode_funcs_t * /*ufuncs*/,
329 hb_codepoint_t unicode,
330 void * /*user_data*/)
331{
332 return _qtcategory_to_hbcategory[QChar::category(unicode)];
333}
334
335static hb_codepoint_t
336_hb_qt_unicode_mirroring(hb_unicode_funcs_t * /*ufuncs*/,
337 hb_codepoint_t unicode,
338 void * /*user_data*/)
339{
340 return QChar::mirroredChar(unicode);
341}
342
343static hb_script_t
344_hb_qt_unicode_script(hb_unicode_funcs_t * /*ufuncs*/,
345 hb_codepoint_t unicode,
346 void * /*user_data*/)
347{
348 return _qtscript_to_hbscript[QChar::script(unicode)];
349}
350
351static hb_bool_t
352_hb_qt_unicode_compose(hb_unicode_funcs_t * /*ufuncs*/,
353 hb_codepoint_t a, hb_codepoint_t b,
354 hb_codepoint_t *ab,
355 void * /*user_data*/)
356{
357 // ### optimize
358 QString s;
359 s.reserve(4);
360 s += QChar::fromUcs4(a);
361 s += QChar::fromUcs4(b);
362 QString normalized = s.normalized(QString::NormalizationForm_C);
363
364 QStringIterator it(normalized);
365 Q_ASSERT(it.hasNext()); // size>0
366 *ab = it.next();
367
368 return !it.hasNext(); // size==1
369}
370
371static hb_bool_t
372_hb_qt_unicode_decompose(hb_unicode_funcs_t * /*ufuncs*/,
373 hb_codepoint_t ab,
374 hb_codepoint_t *a, hb_codepoint_t *b,
375 void * /*user_data*/)
376{
377 // ### optimize
378 if (QChar::decompositionTag(ab) != QChar::Canonical) // !NFD
379 return false;
380
381 QString normalized = QChar::decomposition(ab);
382 if (normalized.isEmpty())
383 return false;
384
385 QStringIterator it(normalized);
386 Q_ASSERT(it.hasNext()); // size>0
387 *a = it.next();
388
389 if (!it.hasNext()) { // size==1
390 *b = 0;
391 return *a != ab;
392 }
393
394 // size>1
395 *b = it.next();
396 if (!it.hasNext()) { // size==2
397 // Here's the ugly part: if ab decomposes to a single character and
398 // that character decomposes again, we have to detect that and undo
399 // the second part :-(
400 const QString recomposed = normalized.normalized(QString::NormalizationForm_C);
401 QStringIterator jt(recomposed);
402 Q_ASSERT(jt.hasNext()); // size>0
403 const hb_codepoint_t c = jt.next();
404 if (c != *a && c != ab) {
405 *a = c;
406 *b = 0;
407 }
408 return true;
409 }
410
411 // size>2
412 // If decomposed to more than two characters, take the last one,
413 // and recompose the rest to get the first component
414 do {
415 *b = it.next();
416 } while (it.hasNext());
417 normalized.chop(QChar::requiresSurrogates(*b) ? 2 : 1);
418 const QString recomposed = normalized.normalized(QString::NormalizationForm_C);
419 QStringIterator jt(recomposed);
420 Q_ASSERT(jt.hasNext()); // size>0
421 // We expect that recomposed has exactly one character now
422 *a = jt.next();
423 return true;
424}
425
426
429 {
430 funcs = hb_unicode_funcs_create(NULL);
431 hb_unicode_funcs_set_combining_class_func(funcs, _hb_qt_unicode_combining_class, NULL, NULL);
432 hb_unicode_funcs_set_general_category_func(funcs, _hb_qt_unicode_general_category, NULL, NULL);
433 hb_unicode_funcs_set_mirroring_func(funcs, _hb_qt_unicode_mirroring, NULL, NULL);
434 hb_unicode_funcs_set_script_func(funcs, _hb_qt_unicode_script, NULL, NULL);
435 hb_unicode_funcs_set_compose_func(funcs, _hb_qt_unicode_compose, NULL, NULL);
436 hb_unicode_funcs_set_decompose_func(funcs, _hb_qt_unicode_decompose, NULL, NULL);
437 }
439 {
440 hb_unicode_funcs_destroy(funcs);
441 }
442
444};
445
446Q_GLOBAL_STATIC(_hb_unicode_funcs_t, qt_ufuncs)
447
448hb_unicode_funcs_t *hb_qt_get_unicode_funcs()
449{
450 return qt_ufuncs()->funcs;
451}
452
453
454// Font routines
455
456static hb_bool_t
457_hb_qt_get_font_h_extents(hb_font_t * /*font*/, void *font_data,
458 hb_font_extents_t *metrics,
459 void * /*user_data*/)
460{
461 QFontEngine *fe = static_cast<QFontEngine *>(font_data);
462 Q_ASSERT(fe);
463
464 metrics->ascender = fe->ascent().value();
465 metrics->descender = fe->descent().value();
466 metrics->line_gap = fe->leading().value();
467
468 return true;
469}
470
471static hb_bool_t
472_hb_qt_font_get_nominal_glyph(hb_font_t * /*font*/, void *font_data,
473 hb_codepoint_t unicode,
474 hb_codepoint_t *glyph,
475 void * /*user_data*/)
476{
477 QFontEngine *fe = static_cast<QFontEngine *>(font_data);
478 Q_ASSERT(fe);
479
480 *glyph = fe->glyphIndex(unicode);
481
482 return *glyph != 0;
483}
484
485static hb_bool_t
486_hb_qt_font_get_variation_glyph(hb_font_t * /*font*/, void *font_data,
487 hb_codepoint_t unicode, hb_codepoint_t /*variation_selector*/,
488 hb_codepoint_t *glyph,
489 void * /*user_data*/)
490{
491 QFontEngine *fe = static_cast<QFontEngine *>(font_data);
492 Q_ASSERT(fe);
493
494 // ### TODO add support for variation selectors
495 *glyph = fe->glyphIndex(unicode);
496
497 return *glyph != 0;
498}
499
500static hb_position_t
501_hb_qt_font_get_glyph_h_advance(hb_font_t *font, void *font_data,
502 hb_codepoint_t glyph,
503 void * /*user_data*/)
504{
505 QFontEngine *fe = static_cast<QFontEngine *>(font_data);
506 Q_ASSERT(fe);
507
508 QFixed advance;
509
510 QGlyphLayout g;
511 g.numGlyphs = 1;
512 g.glyphs = &glyph;
513 g.advances = &advance;
514
515 fe->recalcAdvances(&g, QFontEngine::ShaperFlags(hb_qt_font_get_use_design_metrics(font)));
516
517 return advance.value();
518}
519
520static hb_position_t
521_hb_qt_font_get_glyph_h_kerning(hb_font_t *font, void *font_data,
522 hb_codepoint_t first_glyph, hb_codepoint_t second_glyph,
523 void * /*user_data*/)
524{
525 QFontEngine *fe = static_cast<QFontEngine *>(font_data);
526 Q_ASSERT(fe);
527
528 glyph_t glyphs[2] = { first_glyph, second_glyph };
529 QFixed advance;
530
531 QGlyphLayout g;
532 g.numGlyphs = 2;
533 g.glyphs = glyphs;
534 g.advances = &advance;
535
536 fe->doKerning(&g, QFontEngine::ShaperFlags(hb_qt_font_get_use_design_metrics(font)));
537
538 return advance.value();
539}
540
541static hb_bool_t
542_hb_qt_font_get_glyph_extents(hb_font_t * /*font*/, void *font_data,
543 hb_codepoint_t glyph,
544 hb_glyph_extents_t *extents,
545 void * /*user_data*/)
546{
547 QFontEngine *fe = static_cast<QFontEngine *>(font_data);
548 Q_ASSERT(fe);
549
550 glyph_metrics_t gm = fe->boundingBox(glyph);
551
552 extents->x_bearing = gm.x.value();
553 extents->y_bearing = gm.y.value();
554 extents->width = gm.width.value();
555 extents->height = gm.height.value();
556
557 return true;
558}
559
560static hb_bool_t
561_hb_qt_font_get_glyph_contour_point(hb_font_t * /*font*/, void *font_data,
562 hb_codepoint_t glyph,
563 unsigned int point_index, hb_position_t *x, hb_position_t *y,
564 void * /*user_data*/)
565{
566 QFontEngine *fe = static_cast<QFontEngine *>(font_data);
567 Q_ASSERT(fe);
568
569 QFixed xpos, ypos;
570 quint32 numPoints = 1;
571 if (Q_LIKELY(fe->getPointInOutline(glyph, 0, point_index, &xpos, &ypos, &numPoints) == 0)) {
572 *x = xpos.value();
573 *y = ypos.value();
574 return true;
575 }
576
577 *x = *y = 0;
578 return false;
579}
580
581
582static hb_user_data_key_t _useDesignMetricsKey;
583
584void hb_qt_font_set_use_design_metrics(hb_font_t *font, uint value)
585{
586 hb_font_set_user_data(font, &_useDesignMetricsKey, (void *)quintptr(value), NULL, true);
587}
588
590{
591 return quintptr(hb_font_get_user_data(font, &_useDesignMetricsKey));
592}
593
594
597 {
598 funcs = hb_font_funcs_create();
599
600 hb_font_funcs_set_font_h_extents_func(funcs, _hb_qt_get_font_h_extents, NULL, NULL);
601 hb_font_funcs_set_nominal_glyph_func(funcs, _hb_qt_font_get_nominal_glyph, NULL, NULL);
602 hb_font_funcs_set_variation_glyph_func(funcs, _hb_qt_font_get_variation_glyph, NULL, NULL);
603 hb_font_funcs_set_glyph_h_advance_func(funcs, _hb_qt_font_get_glyph_h_advance, NULL, NULL);
604 hb_font_funcs_set_glyph_h_kerning_func(funcs, _hb_qt_font_get_glyph_h_kerning, NULL, NULL);
605 hb_font_funcs_set_glyph_extents_func(funcs, _hb_qt_font_get_glyph_extents, NULL, NULL);
606 hb_font_funcs_set_glyph_contour_point_func(funcs, _hb_qt_font_get_glyph_contour_point, NULL, NULL);
607
608 hb_font_funcs_make_immutable(funcs);
609 }
611 {
612 hb_font_funcs_destroy(funcs);
613 }
614
616};
617
618Q_GLOBAL_STATIC(_hb_qt_font_funcs_t, qt_ffuncs)
619
620
621static hb_blob_t *
622_hb_qt_reference_table(hb_face_t * /*face*/, hb_tag_t tag, void *user_data)
623{
624 QFontEngine::FaceData *data = static_cast<QFontEngine::FaceData *>(user_data);
625 Q_ASSERT(data);
626
627 qt_get_font_table_func_t get_font_table = data->get_font_table;
628 Q_ASSERT(get_font_table);
629
630 uint length = 0;
631 if (Q_UNLIKELY(!get_font_table(data->user_data, tag, 0, &length)))
632 return hb_blob_get_empty();
633
634 char *buffer = static_cast<char *>(malloc(length));
635 if (q_check_ptr(buffer) == nullptr)
636 return nullptr;
637
638 if (Q_UNLIKELY(!get_font_table(data->user_data, tag, reinterpret_cast<uchar *>(buffer), &length)))
639 return nullptr;
640
641 return hb_blob_create(const_cast<const char *>(buffer), length,
642 HB_MEMORY_MODE_WRITABLE,
643 buffer, free);
644}
645
646static inline hb_face_t *
648{
649 QFontEngine::FaceData *data = static_cast<QFontEngine::FaceData *>(malloc(sizeof(QFontEngine::FaceData)));
650 Q_CHECK_PTR(data);
651 data->user_data = fe->faceData.user_data;
652 data->get_font_table = fe->faceData.get_font_table;
653
654 hb_face_t *face = hb_face_create_for_tables(_hb_qt_reference_table, (void *)data, free);
655
656 hb_face_set_index(face, fe->faceId().index);
657 hb_face_set_upem(face, fe->emSquareSize().truncate());
658
659 return face;
660}
661
662static void
663_hb_qt_face_release(void *user_data)
664{
665 hb_face_destroy(static_cast<hb_face_t *>(user_data));
666}
667
669{
670 Q_ASSERT(fe && fe->type() != QFontEngine::Multi);
671
672 if (Q_UNLIKELY(!fe->face_))
673 fe->face_ = QFontEngine::Holder(_hb_qt_face_create(fe), _hb_qt_face_release);
674
675 return static_cast<hb_face_t *>(fe->face_.get());
676}
677
678
679static inline hb_font_t *
681{
682 hb_face_t *face = hb_qt_face_get_for_engine(fe);
683
684 hb_font_t *font = hb_font_create(face);
685
686 const qreal y_ppem = fe->fontDef.pixelSize;
687 const qreal x_ppem = (fe->fontDef.pixelSize * fe->fontDef.stretch) / 100.0;
688
689 hb_font_set_funcs(font, qt_ffuncs()->funcs, fe, nullptr);
690 hb_font_set_scale(font, QFixed::fromReal(x_ppem).value(), -QFixed::fromReal(y_ppem).value());
691 hb_font_set_ppem(font, int(x_ppem), int(y_ppem));
692
693 hb_font_set_ptem(font, fe->fontDef.pointSize);
694
695 return font;
696}
697
698static void
699_hb_qt_font_release(void *user_data)
700{
701 hb_font_destroy(static_cast<hb_font_t *>(user_data));
702}
703
705{
706 Q_ASSERT(fe && fe->type() != QFontEngine::Multi);
707
708 if (Q_UNLIKELY(!fe->font_))
709 fe->font_ = QFontEngine::Holder(_hb_qt_font_create(fe), _hb_qt_font_release);
710
711 return static_cast<hb_font_t *>(fe->font_.get());
712}
713
714QT_END_NAMESPACE
friend class QFontEngine
Definition qpainter.h:432
Combined button and popup list for selecting options.
bool(* qt_get_font_table_func_t)(void *user_data, uint tag, uchar *buffer, uint *length)
static hb_codepoint_t _hb_qt_unicode_mirroring(hb_unicode_funcs_t *, hb_codepoint_t unicode, void *)
static hb_unicode_combining_class_t _hb_qt_unicode_combining_class(hb_unicode_funcs_t *, hb_codepoint_t unicode, void *)
static QT_BEGIN_NAMESPACE const hb_script_t _qtscript_to_hbscript[]
uint hb_qt_font_get_use_design_metrics(hb_font_t *font)
static void _hb_qt_font_release(void *user_data)
static void _hb_qt_face_release(void *user_data)
static hb_unicode_general_category_t _hb_qt_unicode_general_category(hb_unicode_funcs_t *, hb_codepoint_t unicode, void *)
static hb_bool_t _hb_qt_unicode_compose(hb_unicode_funcs_t *, hb_codepoint_t a, hb_codepoint_t b, hb_codepoint_t *ab, void *)
static hb_blob_t * _hb_qt_reference_table(hb_face_t *, hb_tag_t tag, void *user_data)
QChar::Script hb_qt_script_from_script(hb_script_t script)
static hb_script_t _hb_qt_unicode_script(hb_unicode_funcs_t *, hb_codepoint_t unicode, void *)
hb_script_t hb_qt_script_to_script(QChar::Script script)
static hb_face_t * _hb_qt_face_create(QFontEngine *fe)
hb_font_t * hb_qt_font_get_for_engine(QFontEngine *fe)
void hb_qt_font_set_use_design_metrics(hb_font_t *font, uint value)
static hb_bool_t _hb_qt_unicode_decompose(hb_unicode_funcs_t *, hb_codepoint_t ab, hb_codepoint_t *a, hb_codepoint_t *b, void *)
hb_face_t * hb_qt_face_get_for_engine(QFontEngine *fe)
static hb_font_t * _hb_qt_font_create(QFontEngine *fe)
static const hb_unicode_general_category_t _qtcategory_to_hbcategory[]
hb_font_funcs_t * funcs
hb_unicode_funcs_t * funcs