6#include "QskGlyphTable.h"
8#include "QskInternalMacros.h"
12#include <qpainterpath.h>
16#include <private/qrawfont_p.h>
19typedef QHash< QString, uint > GlyphNameTable;
29namespace PostTableParser
33 static inline QString toString(
const uint8_t* s )
36 return QString::fromUtf8(
37 reinterpret_cast< const char*
> ( s + 1 ), *s );
40 static GlyphNameTable glyphNames(
const QByteArray& blob )
44 const auto* glyphData =
reinterpret_cast< const uint16_t*
>( blob.constData() + 32 );
45 if (
const auto nglyphs = qFromBigEndian( *glyphData++ ) )
47 QVarLengthArray< const uint8_t* > strings;
48 strings.reserve( nglyphs );
50 const auto from =
reinterpret_cast< const uint8_t*
>( glyphData + nglyphs );
51 const auto to =
reinterpret_cast< const uint8_t*
>( blob.data() + blob.size() );
55 for (
auto s = from; s < to; s += *s + 1 )
58 for (
int i = 0; i < nglyphs; i++ )
60 const int idx = qFromBigEndian( glyphData[i] ) - 258;
62 if ( idx >= 0 && idx < strings.size() )
63 names.insert( toString( strings[idx] ), i );
72namespace CFFTableParser
76 static uint32_t offsetAt(
const uint8_t* d,
int size )
85 return ( d[0] << 8 ) | d[1];
87 return ( d[0] << 16 ) | ( d[1] << 8 ) | d[2];
89 return ( d[0] << 24 ) | ( d[1] << 16 ) | ( d[2] << 8 ) | d[3];
93 static int indexDataSize(
const uint8_t* d )
95 const int nitems = ( d[0] << 8 ) | d[1];
99 const int size = d[2];
101 const uint8_t* lx = d + 3 + nitems * size;
102 return lx + size - 1 + offsetAt( lx, size ) - d;
105 static const uint8_t* indexData(
const uint8_t* d )
107 const int nitems = ( d[0] << 8 ) | d[1];
113 const int size = d[0];
115 const uint8_t* _contents = d + ( nitems + 1 ) * size;
116 return _contents + offsetAt(d + 1, size);
119 static const uint8_t* skipHeader(
const uint8_t* d )
124 static const uint8_t* skipIndex(
const uint8_t* d )
126 return d + indexDataSize( d );
129 static QVector< int > stringIds(
const uint8_t* data,
int nglyphs )
131 const int format = data[0];
133 QVector< int > _sids;
136 const uint8_t* p = data + 1;
141 for (; _sids.size() < nglyphs; p += 2)
143 int sid = ( p[0] << 8 ) | p[1];
150 for (; _sids.size() < nglyphs; p += 3)
152 const int sid = ( p[0] << 8 ) | p[1];
155 for (
int i = 0; i <= n; i++ )
162 for (; _sids.size() < nglyphs; p += 4)
164 const int sid = ( p[0] << 8 ) | p[1];
165 const int n = ( p[2] << 8 ) | p[3];
167 for (
int i = 0; i <= n; i++ )
177 static int dictValue(
const uint8_t* d,
int op )
184 const uint8_t* data = indexData( d );
191 const auto valueType = data[0];
193 if ( valueType == 28 )
195 value = ( data[1] << 8 ) | data[2];
198 else if ( valueType == 29 )
200 value = ( data[1] << 24 ) | ( data[2] << 16 )
201 | ( data[3] << 8 ) | data[4];
205 else if ( valueType == 30 )
214 if ( ( b & 0x0f ) == 0x0f || ( b & 0xf0 ) == 0xf0 )
220 else if ( valueType >= 31 && valueType <= 246 )
222 value = data[0] - 139;
225 else if ( valueType >= 247 && valueType <= 250 )
227 value = ( ( data[0] - 247 ) << 8 ) + data[1] + 108;
230 else if ( valueType >= 251 && valueType <= 254 )
232 value = -( ( data[0] - 251 ) << 8 ) - data[1] - 108;
241 data += ( data[0] == 12 ) ? 2 : 1;
250 StringTable(
const uint8_t* data )
252 const int nitems = ( data[0] << 8 ) | data[1];
255 _contents = data + 2;
264 _contents = _offset + nitems * _offsize + _offsize - 1;
268 inline QString operator[](
int which )
const
270 const auto x = _offset + which * _offsize;
272 const auto d1 = _contents + offsetAt(x, _offsize);
273 const auto d2 = _contents + offsetAt(x + _offsize, _offsize);
275 return QString::fromUtf8(
276 reinterpret_cast< const char*
>( d1 ), d2 - d1 );
280 const uint8_t* _contents =
nullptr;
281 const uint8_t* _offset =
nullptr;
285 static GlyphNameTable glyphNames(
const QByteArray& blob,
int glyphCount )
287 auto data =
reinterpret_cast< const uint8_t*
>( blob.constData() );
289 auto nameIndex = skipHeader( data );
290 auto dictIndex = skipIndex( nameIndex );
291 auto stringIndex = skipIndex( dictIndex );
293 auto charset = data + dictValue( dictIndex, 15 );
295 const QVector< int > sids = stringIds( charset, glyphCount );
297 const StringTable table( stringIndex );
299 GlyphNameTable names;
301 for (
int i = 0; i < glyphCount; i++ )
308 const auto idx = sids[i] - 391;
311 names.insert( table[idx], i );
318static uint qskGlyphCount(
const QRawFont& font )
325 const auto fontEngine = QRawFontPrivate::get( font )->fontEngine;
326 return fontEngine ? fontEngine->glyphCount() : 0;
329static qreal qskPixelSize(
const QRawFont& font )
331 const auto fontEngine = QRawFontPrivate::get( font )->fontEngine;
332 return fontEngine ? fontEngine->fontDef.pixelSize : 0.0;
335static GlyphNameTable qskGlyphNameTable(
const QRawFont& font )
337 const auto count = qskGlyphCount( font );
345 auto blob = font.fontTable(
"CFF ");
346 if ( !blob.isEmpty() )
347 return CFFTableParser::glyphNames( blob, count );
349 blob = font.fontTable(
"post" );
350 if ( !blob.isEmpty() )
351 return PostTableParser::glyphNames( blob );
354 return GlyphNameTable();
357static inline quint32 qskGlyphIndex(
const QRawFont& font,
char32_t ucs4 )
359 if ( qskGlyphCount( font ) == 0 )
362 const auto idxs = font.glyphIndexesForString(
363 QString::fromUcs4( &ucs4, 1 ) );
366 Q_ASSERT( idxs.size() == 1 );
368 return idxs.size() == 1 ? idxs[0] : 0;
371class QskGlyphTable::PrivateData
374 inline const GlyphNameTable& glyphNameTable()
const
378 auto that =
const_cast< PrivateData*
>( this );
379 that->nameTable = qskGlyphNameTable( font );
380 that->validNames =
true;
387 GlyphNameTable nameTable;
388 bool validNames =
false;
391QskGlyphTable::QskGlyphTable()
392 : m_data( new PrivateData )
396QskGlyphTable::QskGlyphTable(
const QRawFont& font )
405 *m_data = *other.m_data;
408QskGlyphTable::~QskGlyphTable()
414 if ( m_data != other.m_data )
415 *m_data = *other.m_data;
420void QskGlyphTable::setIconFont(
const QRawFont& font )
422 if ( font != m_data->font )
425 m_data->nameTable.clear();
426 m_data->validNames =
false;
430QRawFont QskGlyphTable::iconFont()
const
435QPainterPath QskGlyphTable::glyphPath( uint glyphIndex )
const
444 if (
auto fontEngine = QRawFontPrivate::get( m_data->font )->fontEngine )
446 QFixedPoint position;
447 quint32 idx = glyphIndex;
449 fontEngine->addGlyphsToPath( &idx, &position, 1, &path, {} );
455QskGraphic QskGlyphTable::glyphGraphic( uint glyphIndex )
const
459 if ( glyphIndex > 0 && qskGlyphCount( m_data->font ) > 0 )
461 const auto path = glyphPath( glyphIndex );
463 if ( !path.isEmpty() )
466 const auto sz = qskPixelSize( m_data->font );
467 graphic.setViewBox( QRectF( 0.0, -sz, sz, sz ) );
469 QPainter painter( &graphic );
470 painter.setRenderHint( QPainter::Antialiasing,
true );
471 painter.fillPath( path, Qt::black );
478uint QskGlyphTable::glyphCount()
const
480 return qskGlyphCount( m_data->font );
483uint QskGlyphTable::codeToIndex(
char32_t ucs4 )
const
485 return qskGlyphIndex( m_data->font, ucs4 );
488uint QskGlyphTable::nameToIndex(
const QString& name )
const
496 return m_data->glyphNameTable().value( name.toLatin1(), 0 );
499QHash< QString, uint > QskGlyphTable::nameTable()
const
501 return m_data->glyphNameTable();
A paint device for scalable graphics.