#!/usr/bin/env ../utility/fontforge-interp.sh
__license__ = """
This file is part of Gnu FreeFont.
Gnu FreeFont is free software: you can redistribute it and/or modify it under
the terms of the GNU General Public License as published by the Free Software
Foundation, either version 3 of the License, or (at your option) any later
version.
Gnu FreeFont is distributed in the hope that it will be useful, but WITHOUT
ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
You should have received a copy of the GNU General Public License along with
Gnu FreeFont. If not, see .
"""
__author__ = "Stevan White"
__email__ = "stevan.white@googlemail.com"
__copyright__ = "Copyright 2009, 2010, 2011, Stevan White"
__date__ = "$Date: 2010-09-14 13:02:02 $"
__version__ = "$Revision: 1.9 $"
__doc__ = """
Makes an HTML table showing how many glyphs are in each range in each font,
and tries to collate that with the OS/2 character range support bit flags.
Runs under FontForge.
fontforge -script ranges.py
This is a hack--in no way authoritative.
Lots of guesswork; much is wrong; the coding is gross.
See
http://www.w3.org/TR/REC-CSS2/notes.html
http://shlimazl.nm.ru/eng/fonts_ttf.htm
http://www.evertype.com/standards/iso10646/ucs-collections.html
The intervals are partly just the assigned interval, but often I have
listed the ranges that have characters assigned to them.
"""
import fontforge
from sys import stderr
from time import tzset, strftime
from OpenType.UnicodeRanges import *
def total_intervals( intervals ):
num = 0
for i in intervals:
num += i.len()
return num
def count_glyphs_in_intervals( font, intervals ):
num = 0
for r in intervals:
if r.begin < len( font ) and r.end < len( font ):
# select() will throw up if try to select value
# beyond the range of the encoding
try:
font.selection.select( ( 'ranges', None ),
r.begin, r.end )
g = font.selection.byGlyphs
for e in g:
num += 1
except ValueError:
print >> stderr, "interval " + str( r ) \
+ " not representable in " + font.fontname
exit( 1 )
return num
special_TT_points = ( 0x00, 0x01, 0x0D )
def codepointIsSpecialTT( encoding ):
""" See Recommendations for OpenType Fonts
http://www.microsoft.com/typography/otspec/recom.htm """
return encoding in special_TT_points
class SupportInfo:
def __init__( self, os2bit, supports, total ):
self.os2bit = os2bit
self.supports = supports
self.total = total
class FontSupport:
""" A record of support for all OS/2 ranges within a single font.
Uses a dictionary internally, to avoid loss of the index info.
"""
def __init__( self, fontPath, short ):
font = fontforge.open( fontPath )
self.name = font.fontname
self.short = short
self.myInfos = {}
self.totalGlyphs = 0
self.fontTotalGlyphs = 0
self.privateUseGlyphs = 0
r = font.os2_unicoderanges
# print >> stderr, font.fontname, hex( r[0] ), hex( r[1] ),hex( r[2] ),hex( r[3] );
nRanges = len( ulUnicodeRange )
for index in range( 0, nRanges ):
byte = index / 32
bit = index % 32
self.collectRangeInfo( font, r[byte], bit, index )
for g in font.glyphs():
self.fontTotalGlyphs += 1
cp = g.encoding
if ( not codepointIsInSomeRange( cp )
and not codepointIsSpecialTT( cp ) ):
print >> stderr, font.fontname, \
"no range for", hex( cp )
""" '''Would like to check that special TT slots are
present, but don't know how...'''
for cp in special_TT_points:
font.selection.all()
if not cp in font.selection.byGlyphs:
print >> stderr, font.fontname, \
"special TT glyph missing", hex( cp )
"""
def collectRangeInfo( self, font, os2supportbyte, bit, index ):
supports = ( os2supportbyte & (1 << bit) ) != 0
rangeName = ulUnicodeRange[index][1]
intervals = ulUnicodeRange[index][2]
nglyphs = count_glyphs_in_intervals( font, intervals )
self.setRangeSupport( index, supports, nglyphs )
self.totalGlyphs += nglyphs
if index == 60 or index == 90:
self.privateUseGlyphs += nglyphs
def setRangeSupport( self, idx, supports, total ):
if self.myInfos.has_key( idx ):
print >> stderr, "OS/2 index ", idx, " duplicated"
exit( 1 )
self.myInfos[idx] = SupportInfo( idx, supports, total )
def getInfo( self, idx ):
if not self.myInfos.has_key( idx ):
print >> stderr, "OS/2 index ", idx, " not found"
exit( 1 )
return self.myInfos[ idx ]
table_head = '''
OS/2 character ranges vs. font faces
OS/2 character range |
range total |
|
%s
'''
def print_font_range_table( fontSupportList ):
headings = ''
for fsl in fontSupportList:
headings += '' + fsl.short + ' | '
print table_head % ( headings )
for r in ulUnicodeRange:
idx = r[0]
range_name = r[1]
intervals = r[2]
rowclass = ' class="low"'
if len( ulUnicodeRange[idx] ) > 3 and ulUnicodeRange[ idx ][3]:
rowclass = ' class="high"'
if idx == 60 or idx == 90:
rowclass = ' class="private"'
print '%s | ' % ( rowclass, range_name )
print '%i | ' % (
total_intervals( intervals ) )
print ' | '
for fsl in fontSupportList:
supportInfo = fsl.getInfo( idx )
supportString = ''
if supportInfo.supports:
supportString = '•'
totalStr = ' '
if supportInfo.total:
totalStr = str( supportInfo.total )
print '%s | %s | ' % (
totalStr, supportString )
print '
'
print 'total in Unicode ranges | '
for fsl in fontSupportList:
print '%i | ' % (
fsl.totalGlyphs )
print '
'
print 'total in font | '
for fsl in fontSupportList:
print '%i | ' % (
fsl.fontTotalGlyphs )
print '
'
print 'total in Private Use | '
for fsl in fontSupportList:
print '%i | ' % (
fsl.privateUseGlyphs )
print '
'
# Would also like to total glyphs in ranges for each font,
# and also print total glyphs in each font.
print '
'
table_introduction = """
For historical reasons, TrueType classifies Unicode ranges according to
an extension of the old OS/2 character ranges. This table shows how many
characters FontForge finds in each of the ranges for each face in the family.
"""
table_explanation = """
Ranges for which (FontForge reports that) the font's OS/2 support
bit is set are marked with a bullet.
For many ranges, I took the liberty of reducing the set of characters
considered to those listed for the range in the current Unicode charts.
The number of characters supported can thus be less than the width of the range.
The totals include glyphs in the Private Use areas (where there
are many ligatures, alternative forms, and glyph components). The glyphs
in these areas do not correspond to Unicode characters.
Three control characters are inserted at 0x00, 0x01 and 0x0d to satisfy the
TrueType recommendations. These are counted under Basic Latin, but other
control characters there are not counted.
Note that there is a discrepancy in the Greek Symbols, Hebrew Extended and
Arabic Extended ranges, between what FontForge reports here and in its Font
Info window under OS/2 Character Ranges. I don't know why, but these ranges
are also not well defined in the TrueType standard.
Note the two characters from Devanagri. These are the danda and double-danda
used by other Indic scripts.
The ranges beyond Unicode point 0xFFFF, are
shaded.
"""
html_heading = '''
Gnu FreeFont character range support
'''
def print_font_range_report( fontSupportList ):
print html_heading
print ''
print ''
print 'Gnu FreeFont support for OpenType OS/2 character ranges'
print '
'
print ''
print table_introduction
print '
'
print_font_range_table( fontSupportList )
print ''
print table_explanation
tzset()
print 'Generated by range_report.py
on %s.' % (
strftime('%X %x %Z') )
print '
'
print ''
print ''
supportList = []
supportList.append( FontSupport( '../../sfd/FreeSerif.sfd', 'Srf' ) )
supportList.append( FontSupport( '../../sfd/FreeSerifItalic.sfd', 'Srf I' ) )
supportList.append( FontSupport( '../../sfd/FreeSerifBold.sfd', 'Srf B' ) )
supportList.append( FontSupport( '../../sfd/FreeSerifBoldItalic.sfd', 'Srf BI' ) )
supportList.append( FontSupport( '../../sfd/FreeSans.sfd', 'Sans' ) )
supportList.append( FontSupport( '../../sfd/FreeSansOblique.sfd', 'Sans O' ) )
supportList.append( FontSupport( '../../sfd/FreeSansBold.sfd', 'Sans B' ) )
supportList.append( FontSupport( '../../sfd/FreeSansBoldOblique.sfd', 'Sans BO' ) )
supportList.append( FontSupport( '../../sfd/FreeMono.sfd', 'Mono' ) )
supportList.append( FontSupport( '../../sfd/FreeMonoOblique.sfd', 'Mono O' ) )
supportList.append( FontSupport( '../../sfd/FreeMonoBold.sfd', 'Mono B' ) )
supportList.append( FontSupport( '../../sfd/FreeMonoBoldOblique.sfd', 'Mono BO' ) )
print_font_range_report( supportList )