glyph_imager 0.0.14 → 0.1.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (82) hide show
  1. data/VERSION +1 -1
  2. data/glyph_imager.gemspec +45 -36
  3. data/lib/glyph_imager.rb +19 -27
  4. data/test/fonts/Musica.ttf +0 -0
  5. data/test/test_glyph_imager.rb +9 -3
  6. data/vendor/ttfunk/.gitignore +1 -0
  7. data/vendor/ttfunk/data/fonts/DejaVuSans.ttf +0 -0
  8. data/vendor/ttfunk/data/fonts/comicsans.ttf +0 -0
  9. data/vendor/ttfunk/example.rb +45 -0
  10. data/vendor/ttfunk/lib/ttfunk.rb +102 -0
  11. data/vendor/ttfunk/lib/ttfunk/directory.rb +17 -0
  12. data/vendor/ttfunk/lib/ttfunk/encoding/mac_roman.rb +88 -0
  13. data/vendor/ttfunk/lib/ttfunk/encoding/windows_1252.rb +69 -0
  14. data/vendor/ttfunk/lib/ttfunk/reader.rb +44 -0
  15. data/vendor/ttfunk/lib/ttfunk/resource_file.rb +78 -0
  16. data/vendor/ttfunk/lib/ttfunk/subset.rb +18 -0
  17. data/vendor/ttfunk/lib/ttfunk/subset/base.rb +141 -0
  18. data/vendor/ttfunk/lib/ttfunk/subset/mac_roman.rb +50 -0
  19. data/vendor/ttfunk/lib/ttfunk/subset/unicode.rb +48 -0
  20. data/vendor/ttfunk/lib/ttfunk/subset/unicode_8bit.rb +63 -0
  21. data/vendor/ttfunk/lib/ttfunk/subset/windows_1252.rb +55 -0
  22. data/vendor/ttfunk/lib/ttfunk/subset_collection.rb +72 -0
  23. data/vendor/ttfunk/lib/ttfunk/table.rb +46 -0
  24. data/vendor/ttfunk/lib/ttfunk/table/cmap.rb +34 -0
  25. data/vendor/ttfunk/lib/ttfunk/table/cmap/format00.rb +54 -0
  26. data/vendor/ttfunk/lib/ttfunk/table/cmap/format04.rb +126 -0
  27. data/vendor/ttfunk/lib/ttfunk/table/cmap/format12.rb +37 -0
  28. data/vendor/ttfunk/lib/ttfunk/table/cmap/subtable.rb +82 -0
  29. data/vendor/ttfunk/lib/ttfunk/table/glyf.rb +64 -0
  30. data/vendor/ttfunk/lib/ttfunk/table/glyf/compound.rb +81 -0
  31. data/vendor/ttfunk/lib/ttfunk/table/glyf/simple.rb +37 -0
  32. data/vendor/ttfunk/lib/ttfunk/table/head.rb +44 -0
  33. data/vendor/ttfunk/lib/ttfunk/table/hhea.rb +41 -0
  34. data/vendor/ttfunk/lib/ttfunk/table/hmtx.rb +47 -0
  35. data/vendor/ttfunk/lib/ttfunk/table/kern.rb +79 -0
  36. data/vendor/ttfunk/lib/ttfunk/table/kern/format0.rb +62 -0
  37. data/vendor/ttfunk/lib/ttfunk/table/loca.rb +43 -0
  38. data/vendor/ttfunk/lib/ttfunk/table/maxp.rb +40 -0
  39. data/vendor/ttfunk/lib/ttfunk/table/name.rb +125 -0
  40. data/vendor/ttfunk/lib/ttfunk/table/os2.rb +78 -0
  41. data/vendor/ttfunk/lib/ttfunk/table/post.rb +91 -0
  42. data/vendor/ttfunk/lib/ttfunk/table/post/format10.rb +43 -0
  43. data/vendor/ttfunk/lib/ttfunk/table/post/format20.rb +35 -0
  44. data/vendor/ttfunk/lib/ttfunk/table/post/format25.rb +23 -0
  45. data/vendor/ttfunk/lib/ttfunk/table/post/format30.rb +17 -0
  46. data/vendor/ttfunk/lib/ttfunk/table/post/format40.rb +17 -0
  47. data/vendor/ttfunk/lib/ttfunk/table/simple.rb +14 -0
  48. metadata +46 -37
  49. data/vendor/ttf-ruby-0.1/AUTHORS +0 -1
  50. data/vendor/ttf-ruby-0.1/COPYING +0 -340
  51. data/vendor/ttf-ruby-0.1/README +0 -49
  52. data/vendor/ttf-ruby-0.1/TODO +0 -23
  53. data/vendor/ttf-ruby-0.1/VERSION +0 -1
  54. data/vendor/ttf-ruby-0.1/lib/ttf.rb +0 -20
  55. data/vendor/ttf-ruby-0.1/lib/ttf/datatypes.rb +0 -181
  56. data/vendor/ttf-ruby-0.1/lib/ttf/encodings.rb +0 -140
  57. data/vendor/ttf-ruby-0.1/lib/ttf/exceptions.rb +0 -28
  58. data/vendor/ttf-ruby-0.1/lib/ttf/font_loader.rb +0 -290
  59. data/vendor/ttf-ruby-0.1/lib/ttf/fontchunk.rb +0 -77
  60. data/vendor/ttf-ruby-0.1/lib/ttf/table/cmap.rb +0 -408
  61. data/vendor/ttf-ruby-0.1/lib/ttf/table/cvt.rb +0 -49
  62. data/vendor/ttf-ruby-0.1/lib/ttf/table/fpgm.rb +0 -48
  63. data/vendor/ttf-ruby-0.1/lib/ttf/table/gasp.rb +0 -88
  64. data/vendor/ttf-ruby-0.1/lib/ttf/table/glyf.rb +0 -452
  65. data/vendor/ttf-ruby-0.1/lib/ttf/table/head.rb +0 -86
  66. data/vendor/ttf-ruby-0.1/lib/ttf/table/hhea.rb +0 -96
  67. data/vendor/ttf-ruby-0.1/lib/ttf/table/hmtx.rb +0 -98
  68. data/vendor/ttf-ruby-0.1/lib/ttf/table/kern.rb +0 -186
  69. data/vendor/ttf-ruby-0.1/lib/ttf/table/loca.rb +0 -75
  70. data/vendor/ttf-ruby-0.1/lib/ttf/table/maxp.rb +0 -81
  71. data/vendor/ttf-ruby-0.1/lib/ttf/table/name.rb +0 -226
  72. data/vendor/ttf-ruby-0.1/lib/ttf/table/os2.rb +0 -172
  73. data/vendor/ttf-ruby-0.1/lib/ttf/table/post.rb +0 -120
  74. data/vendor/ttf-ruby-0.1/lib/ttf/table/prep.rb +0 -27
  75. data/vendor/ttf-ruby-0.1/lib/ttf/table/vhea.rb +0 -45
  76. data/vendor/ttf-ruby-0.1/lib/ttf/table/vmtx.rb +0 -36
  77. data/vendor/ttf-ruby-0.1/setup.rb +0 -1558
  78. data/vendor/ttf-ruby-0.1/tools/README +0 -44
  79. data/vendor/ttf-ruby-0.1/tools/ttfcairoglyphviewer +0 -229
  80. data/vendor/ttf-ruby-0.1/tools/ttfdump +0 -396
  81. data/vendor/ttf-ruby-0.1/tools/ttfglyph2svg +0 -144
  82. data/vendor/ttf-ruby-0.1/tools/ttfsubset +0 -273
@@ -0,0 +1,37 @@
1
+ module TTFunk
2
+ class Table
3
+ class Cmap
4
+ module Format12
5
+ attr_reader :language
6
+ attr_reader :code_map
7
+ def [](code)
8
+ @code_map[code] || 0
9
+ end
10
+ def supported?
11
+ true
12
+ end
13
+ private
14
+ def parse_cmap!
15
+ # skip reserved USHORT
16
+ io.read(2)
17
+ # read length, language and nGroups
18
+ length, @language, n_groups = read(12, "NNN")
19
+ # read the groups into code_map
20
+ cmap_data = io.read(12*n_groups)
21
+ # ruby is too slow to loop trough the code_map in real time, so we need to
22
+ # prepare data and to store this in a hash
23
+ @code_map = {}
24
+ n_groups.times{ |i|
25
+ start_index = i*12
26
+ start_char_code,end_char_code,start_glyph_id =
27
+ cmap_data[start_index..(start_index+12)].unpack("NNN")
28
+ (start_char_code..end_char_code).each { |ch|
29
+ @code_map[ch] = (ch-start_char_code) + start_glyph_id
30
+ }
31
+ }
32
+ end
33
+ end
34
+ end
35
+ end
36
+ end
37
+
@@ -0,0 +1,82 @@
1
+ require 'ttfunk/reader'
2
+
3
+ module TTFunk
4
+ class Table
5
+ class Cmap
6
+ class Subtable
7
+ include Reader
8
+
9
+ attr_reader :platform_id
10
+ attr_reader :encoding_id
11
+ attr_reader :format
12
+
13
+ ENCODING_MAPPINGS = {
14
+ :mac_roman => { :platform_id => 1, :encoding_id => 0 },
15
+ # use microsoft unicode, instead of generic unicode, for optimal windows support
16
+ :unicode => { :platform_id => 3, :encoding_id => 1 }
17
+ }
18
+
19
+ def self.encode(charmap, encoding)
20
+ case encoding
21
+ when :mac_roman
22
+ result = Format00.encode(charmap)
23
+ when :unicode
24
+ result = Format04.encode(charmap)
25
+ else
26
+ raise NotImplementedError, "encoding #{encoding.inspect} is not supported"
27
+ end
28
+
29
+ mapping = ENCODING_MAPPINGS[encoding]
30
+
31
+ # platform-id, encoding-id, offset
32
+ result[:subtable] = [mapping[:platform_id], mapping[:encoding_id],
33
+ 12, result[:subtable]].pack("nnNA*")
34
+
35
+ return result
36
+ end
37
+
38
+ def initialize(file, table_start)
39
+ @file = file
40
+ @platform_id, @encoding_id, @offset = read(8, "nnN")
41
+ @offset += table_start
42
+
43
+ parse_from(@offset) do
44
+ @format = read(2, "n").first
45
+
46
+ case @format
47
+ when 0 then extend(TTFunk::Table::Cmap::Format00)
48
+ when 4 then extend(TTFunk::Table::Cmap::Format04)
49
+ when 12 then extend(TTFunk::Table::Cmap::Format12)
50
+ end
51
+
52
+ parse_cmap!
53
+ end
54
+ end
55
+
56
+ def unicode?
57
+ platform_id == 3 && encoding_id == 1 && format == 4 ||
58
+ platform_id == 0 && format == 4 ||
59
+ platform_id == 3 && format == 12
60
+ end
61
+
62
+ def supported?
63
+ false
64
+ end
65
+
66
+ def [](code)
67
+ raise NotImplementedError, "cmap format #{@format} is not supported"
68
+ end
69
+
70
+ private
71
+
72
+ def parse_cmap!
73
+ # do nothing...
74
+ end
75
+ end
76
+ end
77
+ end
78
+ end
79
+
80
+ require 'ttfunk/table/cmap/format00'
81
+ require 'ttfunk/table/cmap/format04'
82
+ require 'ttfunk/table/cmap/format12'
@@ -0,0 +1,64 @@
1
+ require 'ttfunk/table'
2
+
3
+ module TTFunk
4
+ class Table
5
+ class Glyf < Table
6
+ # Accepts a hash mapping (old) glyph-ids to glyph objects, and a hash
7
+ # mapping old glyph-ids to new glyph-ids.
8
+ #
9
+ # Returns a hash containing:
10
+ #
11
+ # * :table - a string representing the encoded 'glyf' table containing
12
+ # the given glyphs.
13
+ # * :offsets - an array of offsets for each glyph
14
+ def self.encode(glyphs, new2old, old2new)
15
+ result = { :table => "", :offsets => [] }
16
+
17
+ new2old.keys.sort.each do |new_id|
18
+ glyph = glyphs[new2old[new_id]]
19
+ result[:offsets] << result[:table].length
20
+ result[:table] << glyph.recode(old2new) if glyph
21
+ end
22
+
23
+ # include an offset at the end of the table, for use in computing the
24
+ # size of the last glyph
25
+ result[:offsets] << result[:table].length
26
+ return result
27
+ end
28
+
29
+ def for(glyph_id)
30
+ return @cache[glyph_id] if @cache.key?(glyph_id)
31
+
32
+ index = file.glyph_locations.index_of(glyph_id)
33
+ size = file.glyph_locations.size_of(glyph_id)
34
+
35
+ if size.zero? # blank glyph, e.g. space character
36
+ @cache[glyph_id] = nil
37
+ return nil
38
+ end
39
+
40
+ parse_from(offset + index) do
41
+ raw = io.read(size)
42
+ number_of_contours, x_min, y_min, x_max, y_max = raw.unpack("n5").map { |i| to_signed(i) }
43
+
44
+ @cache[glyph_id] = if number_of_contours == -1
45
+ Compound.new(raw, x_min, y_min, x_max, y_max)
46
+ else
47
+ Simple.new(raw, number_of_contours, x_min, y_min, x_max, y_max)
48
+ end
49
+ end
50
+ end
51
+
52
+ private
53
+
54
+ def parse!
55
+ # because the glyf table is rather complex to parse, we defer
56
+ # the parse until we need a specific glyf, and then cache it.
57
+ @cache = {}
58
+ end
59
+ end
60
+ end
61
+ end
62
+
63
+ require 'ttfunk/table/glyf/compound'
64
+ require 'ttfunk/table/glyf/simple'
@@ -0,0 +1,81 @@
1
+ require 'ttfunk/reader'
2
+
3
+ module TTFunk
4
+ class Table
5
+ class Glyf
6
+ class Compound
7
+ include Reader
8
+
9
+ ARG_1_AND_2_ARE_WORDS = 0x0001
10
+ WE_HAVE_A_SCALE = 0x0008
11
+ MORE_COMPONENTS = 0x0020
12
+ WE_HAVE_AN_X_AND_Y_SCALE = 0x0040
13
+ WE_HAVE_A_TWO_BY_TWO = 0x0080
14
+ WE_HAVE_INSTRUCTIONS = 0x0100
15
+
16
+ attr_reader :raw
17
+ attr_reader :x_min, :y_min, :x_max, :y_max
18
+ attr_reader :glyph_ids
19
+
20
+ Component = Struct.new(:flags, :glyph_index, :arg1, :arg2, :transform)
21
+
22
+ def initialize(raw, x_min, y_min, x_max, y_max)
23
+ @raw = raw
24
+ @x_min, @y_min, @x_max, @y_max = x_min, y_min, x_max, y_max
25
+
26
+ # Because TTFunk only cares about glyphs insofar as they (1) provide
27
+ # a bounding box for each glyph, and (2) can be rewritten into a
28
+ # font subset, we don't really care about the rest of the glyph data
29
+ # except as a whole. Thus, we don't actually decompose the glyph
30
+ # into it's parts--all we really care about are the locations within
31
+ # the raw string where the component glyph ids are stored, so that
32
+ # when we rewrite this glyph into a subset we can rewrite the
33
+ # component glyph-ids so they are correct for the subset.
34
+
35
+ @glyph_ids = []
36
+ @glyph_id_offsets = []
37
+ offset = 10 # 2 bytes for each of num-contours, min x/y, max x/y
38
+
39
+ loop do
40
+ flags, glyph_id = @raw[offset, 4].unpack("n*")
41
+ @glyph_ids << glyph_id
42
+ @glyph_id_offsets << offset + 2
43
+
44
+ break unless flags & MORE_COMPONENTS != 0
45
+ offset += 4
46
+
47
+ if flags & ARG_1_AND_2_ARE_WORDS != 0
48
+ offset += 4
49
+ else
50
+ offset += 2
51
+ end
52
+
53
+ if flags & WE_HAVE_A_TWO_BY_TWO != 0
54
+ offset += 8
55
+ elsif flags & WE_HAVE_AN_X_AND_Y_SCALE != 0
56
+ offset += 4
57
+ elsif flags & WE_HAVE_A_SCALE != 0
58
+ offset += 2
59
+ end
60
+ end
61
+ end
62
+
63
+ def compound?
64
+ true
65
+ end
66
+
67
+ def recode(mapping)
68
+ result = @raw.dup
69
+ new_ids = glyph_ids.map { |id| mapping[id] }
70
+
71
+ new_ids.zip(@glyph_id_offsets).each do |new_id, offset|
72
+ result[offset, 2] = [new_id].pack("n")
73
+ end
74
+
75
+ return result
76
+ end
77
+ end
78
+ end
79
+ end
80
+ end
81
+
@@ -0,0 +1,37 @@
1
+ require 'ttfunk/reader'
2
+
3
+ module TTFunk
4
+ class Table
5
+ class Glyf
6
+ class Simple
7
+ attr_reader :raw
8
+ attr_reader :number_of_contours
9
+ attr_reader :x_min, :y_min, :x_max, :y_max
10
+
11
+ def initialize(raw, number_of_contours, x_min, y_min, x_max, y_max)
12
+ @raw = raw
13
+ @number_of_contours = number_of_contours
14
+ @x_min, @y_min = x_min, y_min
15
+ @x_max, @y_max = x_max, y_max
16
+
17
+ # Because TTFunk is, at this time, a library for simply pulling
18
+ # metrics out of font files, or for writing font subsets, we don't
19
+ # really care what the contours are for simple glyphs. We just
20
+ # care that we've got an entire glyph's definition. Also, a
21
+ # bounding box could be nice to know. Since we've got all that
22
+ # at this point, we don't need to worry about parsing the full
23
+ # contents of the glyph.
24
+ end
25
+
26
+ def compound?
27
+ false
28
+ end
29
+
30
+ def recode(mapping)
31
+ raw
32
+ end
33
+ end
34
+ end
35
+ end
36
+ end
37
+
@@ -0,0 +1,44 @@
1
+ require 'ttfunk/table'
2
+
3
+ module TTFunk
4
+ class Table
5
+ class Head < TTFunk::Table
6
+ attr_reader :version
7
+ attr_reader :font_revision
8
+ attr_reader :checksum_adjustment
9
+ attr_reader :magic_number
10
+ attr_reader :flags
11
+ attr_reader :units_per_em
12
+ attr_reader :created
13
+ attr_reader :modified
14
+ attr_reader :x_min
15
+ attr_reader :y_min
16
+ attr_reader :x_max
17
+ attr_reader :y_max
18
+ attr_reader :mac_style
19
+ attr_reader :lowest_rec_ppem
20
+ attr_reader :font_direction_hint
21
+ attr_reader :index_to_loc_format
22
+ attr_reader :glyph_data_format
23
+
24
+ def self.encode(head, loca)
25
+ table = head.raw
26
+ table[8,4] = "\0\0\0\0" # set checksum adjustment to 0 initially
27
+ table[-4,2] = [loca[:type]].pack("n") # set index_to_loc_format
28
+ return table
29
+ end
30
+
31
+ private
32
+
33
+ def parse!
34
+ @version, @font_revision, @check_sum_adjustment, @magic_number,
35
+ @flags, @units_per_em, @created, @modified = read(36, "N4n2q2")
36
+
37
+ @x_min, @y_min, @x_max, @y_max = read_signed(4)
38
+
39
+ @mac_style, @lowest_rec_ppem, @font_direction_hint,
40
+ @index_to_loc_format, @glyph_data_format = read(10, "n*")
41
+ end
42
+ end
43
+ end
44
+ end
@@ -0,0 +1,41 @@
1
+ require 'ttfunk/table'
2
+
3
+ module TTFunk
4
+ class Table
5
+ class Hhea < Table
6
+ attr_reader :version
7
+ attr_reader :ascent
8
+ attr_reader :descent
9
+ attr_reader :line_gap
10
+ attr_reader :advance_width_max
11
+ attr_reader :min_left_side_bearing
12
+ attr_reader :min_right_side_bearing
13
+ attr_reader :x_max_extent
14
+ attr_reader :carot_slope_rise
15
+ attr_reader :carot_slope_run
16
+ attr_reader :metric_data_format
17
+ attr_reader :number_of_metrics
18
+
19
+ def self.encode(hhea, hmtx)
20
+ raw = hhea.raw
21
+ raw[-2,2] = [hmtx[:number_of_metrics]].pack("n")
22
+ return raw
23
+ end
24
+
25
+ private
26
+
27
+ def parse!
28
+ @version = read(4, "N").first
29
+ @ascent, @descent, @line_gap = read_signed(3)
30
+ @advance_width_max = read(2, "n").first
31
+
32
+ @min_left_side_bearing, @min_right_side_bearing, @x_max_extent,
33
+ @carot_slope_rise, @carot_slope_run, @caret_offset,
34
+ reserved, reserved, reserved, reserved,
35
+ @metric_data_format = read_signed(11)
36
+
37
+ @number_of_metrics = read(2, "n").first
38
+ end
39
+ end
40
+ end
41
+ end
@@ -0,0 +1,47 @@
1
+ require 'ttfunk/table'
2
+
3
+ module TTFunk
4
+ class Table
5
+ class Hmtx < Table
6
+ attr_reader :metrics
7
+ attr_reader :left_side_bearings
8
+ attr_reader :widths
9
+
10
+ def self.encode(hmtx, mapping)
11
+ metrics = mapping.keys.sort.map do |new_id|
12
+ metric = hmtx.for(mapping[new_id])
13
+ [metric.advance_width, metric.left_side_bearing]
14
+ end
15
+
16
+ { :number_of_metrics => metrics.length,
17
+ :table => metrics.flatten.pack("n*") }
18
+ end
19
+
20
+ HorizontalMetric = Struct.new(:advance_width, :left_side_bearing)
21
+
22
+ def for(glyph_id)
23
+ @metrics[glyph_id] ||
24
+ HorizontalMetric.new(@metrics.last.advance_width,
25
+ @left_side_bearings[glyph_id - @metrics.length])
26
+ end
27
+
28
+ private
29
+
30
+ def parse!
31
+ @metrics = []
32
+
33
+ file.horizontal_header.number_of_metrics.times do
34
+ advance = read(2, "n").first
35
+ lsb = read_signed(1).first
36
+ @metrics.push HorizontalMetric.new(advance, lsb)
37
+ end
38
+
39
+ lsb_count = file.maximum_profile.num_glyphs - file.horizontal_header.number_of_metrics
40
+ @left_side_bearings = read_signed(lsb_count)
41
+
42
+ @widths = @metrics.map { |metric| metric.advance_width }
43
+ @widths += [@widths.last] * @left_side_bearings.length
44
+ end
45
+ end
46
+ end
47
+ end
@@ -0,0 +1,79 @@
1
+ require 'ttfunk/table'
2
+
3
+ module TTFunk
4
+ class Table
5
+ class Kern < Table
6
+ attr_reader :version
7
+ attr_reader :tables
8
+
9
+ def self.encode(kerning, mapping)
10
+ return nil unless kerning.exists? && kerning.tables.any?
11
+ tables = kerning.tables.map { |table| table.recode(mapping) }.compact
12
+ return nil if tables.empty?
13
+
14
+ [0, tables.length, tables.join].pack("nnA*")
15
+ end
16
+
17
+ private
18
+
19
+ def parse!
20
+ @version, num_tables = read(4, "n*")
21
+ @tables = []
22
+
23
+ if @version == 1 # Mac OS X fonts
24
+ @version = (@version << 16) + num_tables
25
+ num_tables = read(4, "N").first
26
+ parse_version_1_tables(num_tables)
27
+ else
28
+ parse_version_0_tables(num_tables)
29
+ end
30
+ end
31
+
32
+ def parse_version_0_tables(num_tables)
33
+ # It looks like some MS fonts report their kerning subtable lengths
34
+ # wrong. In one case, the length was reported to be some 19366, and yet
35
+ # the table also claimed to hold 14148 pairs (each pair consisting of 6 bytes).
36
+ # You do the math!
37
+ #
38
+ # We're going to assume that the microsoft fonts hold only a single kerning
39
+ # subtable, which occupies the entire length of the kerning table. Worst
40
+ # case, we lose any other subtables that the font contains, but it's better
41
+ # than reading a truncated kerning table.
42
+ #
43
+ # And what's more, it appears to work. So.
44
+ version, length, coverage = read(6, "n*")
45
+ format = coverage >> 8
46
+
47
+ add_table format, :version => version, :length => length,
48
+ :coverage => coverage, :data => raw[10..-1],
49
+ :vertical => (coverage & 0x1 == 0),
50
+ :minimum => (coverage & 0x2 != 0),
51
+ :cross => (coverage & 0x4 != 0),
52
+ :override => (coverage & 0x8 != 0)
53
+ end
54
+
55
+ def parse_version_1_tables(num_tables)
56
+ num_tables.times do
57
+ length, coverage, tuple_index = read(8, "Nnn")
58
+ format = coverage & 0x0FF
59
+
60
+ add_table format, :length => length, :coverage => coverage,
61
+ :tuple_index => tuple_index, :data => io.read(length-8),
62
+ :vertical => (coverage & 0x8000 != 0),
63
+ :cross => (coverage & 0x4000 != 0),
64
+ :variation => (coverage & 0x2000 != 0)
65
+ end
66
+ end
67
+
68
+ def add_table(format, attributes={})
69
+ if format == 0
70
+ @tables << Kern::Format0.new(attributes)
71
+ else
72
+ # silently ignore unsupported kerning tables
73
+ end
74
+ end
75
+ end
76
+ end
77
+ end
78
+
79
+ require 'ttfunk/table/kern/format0'