QSkinny 0.8.0
C++/Qt UI toolkit
Loading...
Searching...
No Matches
QskGlyphTable.cpp
1/******************************************************************************
2 * QSkinny - Copyright (C) The authors
3 * SPDX-License-Identifier: BSD-3-Clause
4 *****************************************************************************/
5
6#include "QskGlyphTable.h"
7#include "QskGraphic.h"
8#include "QskInternalMacros.h"
9
10#include <qrawfont.h>
11#include <qpainter.h>
12#include <qpainterpath.h>
13#include <qendian.h>
14
15QSK_QT_PRIVATE_BEGIN
16#include <private/qrawfont_p.h>
17QSK_QT_PRIVATE_END
18
19typedef QHash< QString, uint > GlyphNameTable;
20
21/*
22 The "parsers" below do no validation checks as the font has
23 already been validated by QRawFont
24
25 Hope QRawFont will extend its API some day ( the underlying
26 font libraries do support glyph names ) and we can remove the nasty
27 code below. see https://bugreports.qt.io/browse/QTBUG-132629
28 */
29namespace PostTableParser
30{
31 // https://learn.microsoft.com/en-us/typography/opentype/spec/post
32
33 static inline QString toString( const uint8_t* s )
34 {
35 // Pascal string. Valid characters are: [A–Z] [a–z] [0–9] '.' '_'
36 return QString::fromUtf8(
37 reinterpret_cast< const char* > ( s + 1 ), *s );
38 }
39
40 static GlyphNameTable glyphNames( const QByteArray& blob )
41 {
42 GlyphNameTable names;
43
44 const auto* glyphData = reinterpret_cast< const uint16_t* >( blob.constData() + 32 );
45 if ( const auto nglyphs = qFromBigEndian( *glyphData++ ) )
46 {
47 QVarLengthArray< const uint8_t* > strings;
48 strings.reserve( nglyphs );
49
50 const auto from = reinterpret_cast< const uint8_t* >( glyphData + nglyphs );
51 const auto to = reinterpret_cast< const uint8_t* >( blob.data() + blob.size() );
52
53 if ( to > from )
54 {
55 for ( auto s = from; s < to; s += *s + 1 )
56 strings += s;
57
58 for ( int i = 0; i < nglyphs; i++ )
59 {
60 const int idx = qFromBigEndian( glyphData[i] ) - 258;
61
62 if ( idx >= 0 && idx < strings.size() )
63 names.insert( toString( strings[idx] ), i );
64 }
65 }
66 }
67
68 return names;
69 }
70}
71
72namespace CFFTableParser
73{
74 // https://adobe-type-tools.github.io/font-tech-notes/pdfs/5176.CFF.pdf
75
76 static uint32_t offsetAt( const uint8_t* d, int size )
77 {
78 switch (size)
79 {
80 case 0:
81 return 0;
82 case 1:
83 return d[0];
84 case 2:
85 return ( d[0] << 8 ) | d[1];
86 case 3:
87 return ( d[0] << 16 ) | ( d[1] << 8 ) | d[2];
88 default:
89 return ( d[0] << 24 ) | ( d[1] << 16 ) | ( d[2] << 8 ) | d[3];
90 }
91 }
92
93 static int indexDataSize( const uint8_t* d )
94 {
95 const int nitems = ( d[0] << 8 ) | d[1];
96 if ( nitems == 0 )
97 return 2;
98
99 const int size = d[2];
100
101 const uint8_t* lx = d + 3 + nitems * size;
102 return lx + size - 1 + offsetAt( lx, size ) - d;
103 }
104
105 static const uint8_t* indexData( const uint8_t* d )
106 {
107 const int nitems = ( d[0] << 8 ) | d[1];
108 d += 2;
109
110 if ( nitems == 0 )
111 return d;
112
113 const int size = d[0];
114
115 const uint8_t* _contents = d + ( nitems + 1 ) * size;
116 return _contents + offsetAt(d + 1, size);
117 }
118
119 static const uint8_t* skipHeader( const uint8_t* d )
120 {
121 return d + d[2];
122 }
123
124 static const uint8_t* skipIndex( const uint8_t* d )
125 {
126 return d + indexDataSize( d );
127 }
128
129 static QVector< int > stringIds( const uint8_t* data, int nglyphs )
130 {
131 const int format = data[0];
132
133 QVector< int > _sids;
134 _sids += 0;
135
136 const uint8_t* p = data + 1;
137 switch( format )
138 {
139 case 0:
140 {
141 for (; _sids.size() < nglyphs; p += 2)
142 {
143 int sid = ( p[0] << 8 ) | p[1];
144 _sids += sid;
145 }
146 break;
147 }
148 case 1:
149 {
150 for (; _sids.size() < nglyphs; p += 3)
151 {
152 const int sid = ( p[0] << 8 ) | p[1];
153 const int n = p[2];
154
155 for ( int i = 0; i <= n; i++ )
156 _sids += sid + i;
157 }
158 break;
159 }
160 case 2:
161 {
162 for (; _sids.size() < nglyphs; p += 4)
163 {
164 const int sid = ( p[0] << 8 ) | p[1];
165 const int n = ( p[2] << 8 ) | p[3];
166
167 for ( int i = 0; i <= n; i++ )
168 _sids += sid + i;
169 }
170 break;
171 }
172 }
173
174 return _sids;
175 }
176
177 static int dictValue( const uint8_t* d, int op )
178 {
179 /*
180 see https://adobe-type-tools.github.io/font-tech-notes/pdfs/5176.CFF.pdf
181 We are intersted in the offset ( operator 15 ).
182 */
183
184 const uint8_t* data = indexData( d );
185
186 int value = 0;
187
188 Q_FOREVER
189 {
190 // operand
191 const auto valueType = data[0];
192
193 if ( valueType == 28 )
194 {
195 value = ( data[1] << 8 ) | data[2];
196 data += 3;
197 }
198 else if ( valueType == 29 )
199 {
200 value = ( data[1] << 24 ) | ( data[2] << 16 )
201 | ( data[3] << 8 ) | data[4];
202
203 data += 5;
204 }
205 else if ( valueType == 30 )
206 {
207 /*
208 Assuming, that an offset is never a double
209 we skip the operand without reading it
210 */
211 while( ++data )
212 {
213 const int b = *data;
214 if ( ( b & 0x0f ) == 0x0f || ( b & 0xf0 ) == 0xf0 )
215 break; // 0xf nibble indicating the end
216 }
217
218 data++;
219 }
220 else if ( valueType >= 31 && valueType <= 246 )
221 {
222 value = data[0] - 139;
223 data++;
224 }
225 else if ( valueType >= 247 && valueType <= 250 )
226 {
227 value = ( ( data[0] - 247 ) << 8 ) + data[1] + 108;
228 data += 2;
229 }
230 else if ( valueType >= 251 && valueType <= 254 )
231 {
232 value = -( ( data[0] - 251 ) << 8 ) - data[1] - 108;
233 data += 2;
234 }
235
236 // operator
237
238 if ( op == data[0] )
239 return value;
240
241 data += ( data[0] == 12 ) ? 2 : 1;
242 }
243
244 return 0; // default value
245 }
246
247 class StringTable
248 {
249 public:
250 StringTable( const uint8_t* data )
251 {
252 const int nitems = ( data[0] << 8 ) | data[1];
253 if ( nitems == 0 )
254 {
255 _contents = data + 2;
256 _offset = nullptr;
257 _offsize = 0;
258 }
259 else
260 {
261 _offsize = data[2];
262 _offset = data + 3;
263
264 _contents = _offset + nitems * _offsize + _offsize - 1;
265 }
266 }
267
268 inline QString operator[]( int which ) const
269 {
270 const auto x = _offset + which * _offsize;
271
272 const auto d1 = _contents + offsetAt(x, _offsize);
273 const auto d2 = _contents + offsetAt(x + _offsize, _offsize);
274
275 return QString::fromUtf8(
276 reinterpret_cast< const char* >( d1 ), d2 - d1 );
277 }
278
279 private:
280 const uint8_t* _contents = nullptr;
281 const uint8_t* _offset = nullptr;
282 int _offsize = -1;
283 };
284
285 static GlyphNameTable glyphNames( const QByteArray& blob, int glyphCount )
286 {
287 auto data = reinterpret_cast< const uint8_t* >( blob.constData() );
288
289 auto nameIndex = skipHeader( data );
290 auto dictIndex = skipIndex( nameIndex );
291 auto stringIndex = skipIndex( dictIndex );
292
293 auto charset = data + dictValue( dictIndex, 15 ); // 15: charset offset
294
295 const QVector< int > sids = stringIds( charset, glyphCount );
296
297 const StringTable table( stringIndex );
298
299 GlyphNameTable names;
300
301 for ( int i = 0; i < glyphCount; i++ )
302 {
303 /*
304 The first 391 SIDs are reserved for standard strings
305 ( Appendix A ) that are not from the charset table.
306 */
307
308 const auto idx = sids[i] - 391;
309
310 if ( idx >= 0 )
311 names.insert( table[idx], i );
312 }
313
314 return names;
315 }
316}
317
318static uint qskGlyphCount( const QRawFont& font )
319{
320 /*
321 we could also read the count from the "maxp" table:
322 https://learn.microsoft.com/en-us/typography/opentype/spec/maxp
323 */
324
325 const auto fontEngine = QRawFontPrivate::get( font )->fontEngine;
326 return fontEngine ? fontEngine->glyphCount() : 0;
327}
328
329static qreal qskPixelSize( const QRawFont& font )
330{
331 const auto fontEngine = QRawFontPrivate::get( font )->fontEngine;
332 return fontEngine ? fontEngine->fontDef.pixelSize : 0.0;
333}
334
335static GlyphNameTable qskGlyphNameTable( const QRawFont& font )
336{
337 const auto count = qskGlyphCount( font );
338 if ( count > 0 )
339 {
340 /*
341 The Compact Font Format ( CFF ) table has been introduced with
342 the OpenType specification. For not complying fonts the names
343 might be found in the Post table
344 */
345 auto blob = font.fontTable( "CFF ");
346 if ( !blob.isEmpty() )
347 return CFFTableParser::glyphNames( blob, count );
348
349 blob = font.fontTable( "post" );
350 if ( !blob.isEmpty() )
351 return PostTableParser::glyphNames( blob );
352 }
353
354 return GlyphNameTable();
355}
356
357static inline quint32 qskGlyphIndex( const QRawFont& font, char32_t ucs4 )
358{
359 if ( qskGlyphCount( font ) == 0 )
360 return 0;
361
362 const auto idxs = font.glyphIndexesForString(
363 QString::fromUcs4( &ucs4, 1 ) );
364
365 // fingers crossed: icon fonts will map code points to single glyphs only
366 Q_ASSERT( idxs.size() == 1 );
367
368 return idxs.size() == 1 ? idxs[0] : 0;
369}
370
371class QskGlyphTable::PrivateData
372{
373 public:
374 inline const GlyphNameTable& glyphNameTable() const
375 {
376 if ( !validNames )
377 {
378 auto that = const_cast< PrivateData* >( this );
379 that->nameTable = qskGlyphNameTable( font );
380 that->validNames = true;
381 }
382
383 return nameTable;
384 }
385
386 QRawFont font;
387 GlyphNameTable nameTable;
388 bool validNames = false;
389};
390
391QskGlyphTable::QskGlyphTable()
392 : m_data( new PrivateData )
393{
394}
395
396QskGlyphTable::QskGlyphTable( const QRawFont& font )
397 : QskGlyphTable()
398{
399 m_data->font = font;
400}
401
402QskGlyphTable::QskGlyphTable( const QskGlyphTable& other )
403 : QskGlyphTable()
404{
405 *m_data = *other.m_data;
406}
407
408QskGlyphTable::~QskGlyphTable()
409{
410}
411
412QskGlyphTable& QskGlyphTable::operator=( const QskGlyphTable& other )
413{
414 if ( m_data != other.m_data )
415 *m_data = *other.m_data;
416
417 return *this;
418}
419
420void QskGlyphTable::setIconFont( const QRawFont& font )
421{
422 if ( font != m_data->font )
423 {
424 m_data->font = font;
425 m_data->nameTable.clear();
426 m_data->validNames = false;
427 }
428}
429
430QRawFont QskGlyphTable::iconFont() const
431{
432 return m_data->font;
433}
434
435QPainterPath QskGlyphTable::glyphPath( uint glyphIndex ) const
436{
437 QPainterPath path;
438
439 /*
440 Unfortunately QRawFont::pathForGlyph runs into failing checks
441 when being called from a different thread - f.e the scene graph thread.
442 So we need to bypass QRawFont and retrieve from its fontEngine.
443 */
444 if ( auto fontEngine = QRawFontPrivate::get( m_data->font )->fontEngine )
445 {
446 QFixedPoint position;
447 quint32 idx = glyphIndex;
448
449 fontEngine->addGlyphsToPath( &idx, &position, 1, &path, {} );
450 }
451
452 return path;
453}
454
455QskGraphic QskGlyphTable::glyphGraphic( uint glyphIndex ) const
456{
457 QskGraphic graphic;
458
459 if ( glyphIndex > 0 && qskGlyphCount( m_data->font ) > 0 )
460 {
461 const auto path = glyphPath( glyphIndex );
462
463 if ( !path.isEmpty() )
464 {
465 // vertical glyph coordinates are in the range [-sz, 0.0]
466 const auto sz = qskPixelSize( m_data->font );
467 graphic.setViewBox( QRectF( 0.0, -sz, sz, sz ) );
468
469 QPainter painter( &graphic );
470 painter.setRenderHint( QPainter::Antialiasing, true );
471 painter.fillPath( path, Qt::black );
472 }
473 }
474
475 return graphic;
476}
477
478uint QskGlyphTable::glyphCount() const
479{
480 return qskGlyphCount( m_data->font );
481}
482
483uint QskGlyphTable::codeToIndex( char32_t ucs4 ) const
484{
485 return qskGlyphIndex( m_data->font, ucs4 );
486}
487
488uint QskGlyphTable::nameToIndex( const QString& name ) const
489{
490 /*
491 Names/Codes that do not correspond to any glyph in should
492 be mapped to glyph index 0.
493
494 see https://learn.microsoft.com/en-us/typography/opentype/spec/cmap
495 */
496 return m_data->glyphNameTable().value( name.toLatin1(), 0 );
497}
498
499QHash< QString, uint > QskGlyphTable::nameTable() const
500{
501 return m_data->glyphNameTable();
502}
A paint device for scalable graphics.
Definition QskGraphic.h:28