ttfunk 1.0.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (43) hide show
  1. data/CHANGELOG +2 -0
  2. data/README.rdoc +39 -0
  3. data/data/fonts/DejaVuSans.ttf +0 -0
  4. data/data/fonts/comicsans.ttf +0 -0
  5. data/examples/metrics.rb +46 -0
  6. data/lib/ttfunk/directory.rb +17 -0
  7. data/lib/ttfunk/encoding/mac_roman.rb +88 -0
  8. data/lib/ttfunk/encoding/windows_1252.rb +69 -0
  9. data/lib/ttfunk/reader.rb +44 -0
  10. data/lib/ttfunk/resource_file.rb +78 -0
  11. data/lib/ttfunk/subset/base.rb +141 -0
  12. data/lib/ttfunk/subset/mac_roman.rb +50 -0
  13. data/lib/ttfunk/subset/unicode.rb +48 -0
  14. data/lib/ttfunk/subset/unicode_8bit.rb +63 -0
  15. data/lib/ttfunk/subset/windows_1252.rb +55 -0
  16. data/lib/ttfunk/subset.rb +18 -0
  17. data/lib/ttfunk/subset_collection.rb +72 -0
  18. data/lib/ttfunk/table/cmap/format00.rb +54 -0
  19. data/lib/ttfunk/table/cmap/format04.rb +126 -0
  20. data/lib/ttfunk/table/cmap/subtable.rb +79 -0
  21. data/lib/ttfunk/table/cmap.rb +34 -0
  22. data/lib/ttfunk/table/glyf/compound.rb +81 -0
  23. data/lib/ttfunk/table/glyf/simple.rb +37 -0
  24. data/lib/ttfunk/table/glyf.rb +64 -0
  25. data/lib/ttfunk/table/head.rb +44 -0
  26. data/lib/ttfunk/table/hhea.rb +41 -0
  27. data/lib/ttfunk/table/hmtx.rb +47 -0
  28. data/lib/ttfunk/table/kern/format0.rb +62 -0
  29. data/lib/ttfunk/table/kern.rb +79 -0
  30. data/lib/ttfunk/table/loca.rb +43 -0
  31. data/lib/ttfunk/table/maxp.rb +40 -0
  32. data/lib/ttfunk/table/name.rb +125 -0
  33. data/lib/ttfunk/table/os2.rb +78 -0
  34. data/lib/ttfunk/table/post/format10.rb +43 -0
  35. data/lib/ttfunk/table/post/format20.rb +35 -0
  36. data/lib/ttfunk/table/post/format25.rb +23 -0
  37. data/lib/ttfunk/table/post/format30.rb +17 -0
  38. data/lib/ttfunk/table/post/format40.rb +17 -0
  39. data/lib/ttfunk/table/post.rb +91 -0
  40. data/lib/ttfunk/table/simple.rb +14 -0
  41. data/lib/ttfunk/table.rb +46 -0
  42. data/lib/ttfunk.rb +102 -0
  43. metadata +121 -0
@@ -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,62 @@
1
+ require 'ttfunk/reader'
2
+
3
+ module TTFunk
4
+ class Table
5
+ class Kern
6
+ class Format0
7
+ include Reader
8
+
9
+ attr_reader :attributes
10
+ attr_reader :pairs
11
+
12
+ def initialize(attributes={})
13
+ @attributes = attributes
14
+
15
+ num_pairs, search_range, entry_selector, range_shift, *pairs =
16
+ attributes.delete(:data).unpack("n*")
17
+
18
+ @pairs = {}
19
+ num_pairs.times do |i|
20
+ break if i*3+2 > pairs.length # sanity check, in case there's a bad length somewhere
21
+ left = pairs[i*3]
22
+ right = pairs[i*3+1]
23
+ value = to_signed(pairs[i*3+2])
24
+ @pairs[[left, right]] = value
25
+ end
26
+ end
27
+
28
+ def vertical?
29
+ @attributes[:vertical]
30
+ end
31
+
32
+ def horizontal?
33
+ !vertical?
34
+ end
35
+
36
+ def cross_stream?
37
+ @attributes[:cross]
38
+ end
39
+
40
+ def recode(mapping)
41
+ subset = []
42
+ pairs.each do |(left, right), value|
43
+ if mapping[left] && mapping[right]
44
+ subset << [mapping[left], mapping[right], value]
45
+ end
46
+ end
47
+
48
+ return nil if subset.empty?
49
+
50
+ num_pairs = subset.length
51
+ search_range = 2 * 2 ** (Math.log(num_pairs) / Math.log(2)).to_i
52
+ entry_selector = (Math.log(search_range / 2) / Math.log(2)).to_i
53
+ range_shift = (2 * num_pairs) - search_range
54
+
55
+ [attributes[:version], num_pairs * 6 + 14, attributes[:coverage],
56
+ num_pairs, search_range, entry_selector, range_shift, subset].
57
+ flatten.pack("n*")
58
+ end
59
+ end
60
+ end
61
+ end
62
+ 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'
@@ -0,0 +1,43 @@
1
+ require 'ttfunk/table'
2
+
3
+ module TTFunk
4
+ class Table
5
+ class Loca < Table
6
+ attr_reader :offsets
7
+
8
+ # Accepts an array of offsets, with each index corresponding to the
9
+ # glyph id with that index.
10
+ #
11
+ # Returns a hash containing:
12
+ #
13
+ # * :table - the string representing the table's contents
14
+ # * :type - the type of offset (to be encoded in the 'head' table)
15
+ def self.encode(offsets)
16
+ if offsets.any? { |ofs| ofs > 0xFFFF }
17
+ { :type => 1, :table => offsets.pack("N*") }
18
+ else
19
+ { :type => 0, :table => offsets.map { |o| o/2 }.pack("n*") }
20
+ end
21
+ end
22
+
23
+ def index_of(glyph_id)
24
+ @offsets[glyph_id]
25
+ end
26
+
27
+ def size_of(glyph_id)
28
+ @offsets[glyph_id+1] - @offsets[glyph_id]
29
+ end
30
+
31
+ private
32
+
33
+ def parse!
34
+ type = file.header.index_to_loc_format == 0 ? "n" : "N"
35
+ @offsets = read(length, "#{type}*")
36
+
37
+ if file.header.index_to_loc_format == 0
38
+ @offsets.map! { |v| v * 2 }
39
+ end
40
+ end
41
+ end
42
+ end
43
+ end
@@ -0,0 +1,40 @@
1
+ require 'ttfunk/table'
2
+
3
+ module TTFunk
4
+ class Table
5
+ class Maxp < Table
6
+ attr_reader :version
7
+ attr_reader :num_glyphs
8
+ attr_reader :max_points
9
+ attr_reader :max_contours
10
+ attr_reader :max_component_points
11
+ attr_reader :max_component_contours
12
+ attr_reader :max_zones
13
+ attr_reader :max_twilight_points
14
+ attr_reader :max_storage
15
+ attr_reader :max_function_defs
16
+ attr_reader :max_instruction_defs
17
+ attr_reader :max_stack_elements
18
+ attr_reader :max_size_of_instructions
19
+ attr_reader :max_component_elements
20
+ attr_reader :max_component_depth
21
+
22
+ def self.encode(maxp, mapping)
23
+ num_glyphs = mapping.length
24
+ raw = maxp.raw
25
+ raw[4,2] = [num_glyphs].pack("n")
26
+ return raw
27
+ end
28
+
29
+ private
30
+
31
+ def parse!
32
+ @version, @num_glyphs, @max_points, @max_contours, @max_component_points,
33
+ @max_component_contours, @max_zones, @max_twilight_points, @max_storage,
34
+ @max_function_defs, @max_instruction_defs, @max_stack_elements,
35
+ @max_size_of_instructions, @max_component_elements, @max_component_depth =
36
+ read(length, "Nn*")
37
+ end
38
+ end
39
+ end
40
+ end
@@ -0,0 +1,125 @@
1
+ require 'ttfunk/table'
2
+
3
+ module TTFunk
4
+ class Table
5
+ class Name < Table
6
+ class String < ::String
7
+ attr_reader :platform_id
8
+ attr_reader :encoding_id
9
+ attr_reader :language_id
10
+
11
+ def initialize(text, platform_id, encoding_id, language_id)
12
+ super(text)
13
+ @platform_id = platform_id
14
+ @encoding_id = encoding_id
15
+ @language_id = language_id
16
+ end
17
+
18
+ def strip_extended
19
+ stripped = gsub(/[\x00-\x19\x80-\xff]/n, "")
20
+ stripped = "[not-postscript]" if stripped.empty?
21
+ return stripped
22
+ end
23
+ end
24
+
25
+ attr_reader :strings
26
+
27
+ attr_reader :copyright
28
+ attr_reader :font_family
29
+ attr_reader :font_subfamily
30
+ attr_reader :unique_subfamily
31
+ attr_reader :font_name
32
+ attr_reader :version
33
+ attr_reader :trademark
34
+ attr_reader :manufacturer
35
+ attr_reader :designer
36
+ attr_reader :description
37
+ attr_reader :vendor_url
38
+ attr_reader :designer_url
39
+ attr_reader :license
40
+ attr_reader :license_url
41
+ attr_reader :preferred_family
42
+ attr_reader :preferred_subfamily
43
+ attr_reader :compatible_full
44
+ attr_reader :sample_text
45
+
46
+ @@subset_tag = "AAAAAA"
47
+
48
+ def self.encode(names)
49
+ tag = @@subset_tag.dup
50
+ @@subset_tag.succ!
51
+
52
+ postscript_name = Name::String.new("#{tag}+#{names.postscript_name}", 1, 0, 0)
53
+
54
+ strings = names.strings.dup
55
+ strings[6] = [postscript_name]
56
+ str_count = strings.inject(0) { |sum, (id, list)| sum + list.length }
57
+
58
+ table = [0, str_count, 6 + 12 * str_count].pack("n*")
59
+ strtable = ""
60
+
61
+ strings.each do |id, list|
62
+ list.each do |string|
63
+ table << [string.platform_id, string.encoding_id, string.language_id, id, string.length, strtable.length].pack("n*")
64
+ strtable << string
65
+ end
66
+ end
67
+
68
+ table << strtable
69
+ end
70
+
71
+ def postscript_name
72
+ return @postscript_name if @postscript_name
73
+ font_family.first || "unnamed"
74
+ end
75
+
76
+ private
77
+
78
+ def parse!
79
+ format, count, string_offset = read(6, "n*")
80
+
81
+ entries = []
82
+ count.times do
83
+ platform, encoding, language, id, length, start_offset = read(12, "n*")
84
+ entries << {
85
+ :platform_id => platform,
86
+ :encoding_id => encoding,
87
+ :language_id => language,
88
+ :name_id => id,
89
+ :length => length,
90
+ :offset => offset + string_offset + start_offset
91
+ }
92
+ end
93
+
94
+ @strings = Hash.new { |h,k| h[k] = [] }
95
+
96
+ count.times do |i|
97
+ io.pos = entries[i][:offset]
98
+ text = io.read(entries[i][:length])
99
+ @strings[entries[i][:name_id]] << Name::String.new(text,
100
+ entries[i][:platform_id], entries[i][:encoding_id], entries[i][:language_id])
101
+ end
102
+
103
+ @copyright = @strings[0]
104
+ @font_family = @strings[1]
105
+ @font_subfamily = @strings[2]
106
+ @unique_subfamily = @strings[3]
107
+ @font_name = @strings[4]
108
+ @version = @strings[5]
109
+ @postscript_name = @strings[6].first.strip_extended # should only be ONE postscript name
110
+ @trademark = @strings[7]
111
+ @manufacturer = @strings[8]
112
+ @designer = @strings[9]
113
+ @description = @strings[10]
114
+ @vendor_url = @strings[11]
115
+ @designer_url = @strings[12]
116
+ @license = @strings[13]
117
+ @license_url = @strings[14]
118
+ @preferred_family = @strings[15]
119
+ @preferred_subfamily = @strings[17]
120
+ @compatible_full = @strings[18]
121
+ @sample_text = @strings[19]
122
+ end
123
+ end
124
+ end
125
+ end
@@ -0,0 +1,78 @@
1
+ require 'ttfunk/table'
2
+
3
+ module TTFunk
4
+ class Table
5
+ class OS2 < Table
6
+ attr_reader :version
7
+
8
+ attr_reader :ave_char_width
9
+ attr_reader :weight_class
10
+ attr_reader :width_class
11
+ attr_reader :type
12
+ attr_reader :y_subscript_x_size
13
+ attr_reader :y_subscript_y_size
14
+ attr_reader :y_subscript_x_offset
15
+ attr_reader :y_subscript_y_offset
16
+ attr_reader :y_superscript_x_size
17
+ attr_reader :y_superscript_y_size
18
+ attr_reader :y_superscript_x_offset
19
+ attr_reader :y_superscript_y_offset
20
+ attr_reader :y_strikeout_size
21
+ attr_reader :y_strikeout_position
22
+ attr_reader :family_class
23
+ attr_reader :panose
24
+ attr_reader :char_range
25
+ attr_reader :vendor_id
26
+ attr_reader :selection
27
+ attr_reader :first_char_index
28
+ attr_reader :last_char_index
29
+
30
+ attr_reader :ascent
31
+ attr_reader :descent
32
+ attr_reader :line_gap
33
+ attr_reader :win_ascent
34
+ attr_reader :win_descent
35
+ attr_reader :code_page_range
36
+
37
+ attr_reader :x_height
38
+ attr_reader :cap_height
39
+ attr_reader :default_char
40
+ attr_reader :break_char
41
+ attr_reader :max_context
42
+
43
+ def tag
44
+ "OS/2"
45
+ end
46
+
47
+ private
48
+
49
+ def parse!
50
+ @version = read(2, "n").first
51
+
52
+ @ave_char_width = read_signed(1)
53
+ @weight_class, @width_class = read(4, "nn")
54
+ @type, @y_subscript_x_size, @y_subscript_y_size, @y_subscript_x_offset,
55
+ @y_subscript_y_offset, @y_superscript_x_size, @y_superscript_y_size,
56
+ @y_superscript_x_offset, @y_superscript_y_offset, @y_strikeout_size,
57
+ @y_strikeout_position, @family_class = read_signed(12)
58
+ @panose = io.read(10)
59
+
60
+ @char_range = io.read(16)
61
+ @vendor_id = io.read(4)
62
+
63
+ @selection, @first_char_index, @last_char_index = read(6, "n*")
64
+
65
+ if @version > 0
66
+ @ascent, @descent, @line_gap = read_signed(3)
67
+ @win_ascent, @win_descent = read(4, "nn")
68
+ @code_page_range = io.read(8)
69
+
70
+ if @version > 1
71
+ @x_height, @cap_height = read_signed(2)
72
+ @default_char, @break_char, @max_context = read(6, "nnn")
73
+ end
74
+ end
75
+ end
76
+ end
77
+ end
78
+ end
@@ -0,0 +1,43 @@
1
+ module TTFunk
2
+ class Table
3
+ class Post
4
+ module Format10
5
+ POSTSCRIPT_GLYPHS = %w(
6
+ .notdef .null nonmarkingreturn space exclam quotedbl numbersign dollar percent
7
+ ampersand quotesingle parenleft parenright asterisk plus comma hyphen period slash
8
+ zero one two three four five six seven eight nine colon semicolon less equal greater
9
+ question at A B C D E F G H I J K L M N O P Q R S T U V W X Y Z
10
+ bracketleft backslash bracketright asciicircum underscore grave
11
+ a b c d e f g h i j k l m n o p q r s t u v w x y z
12
+ braceleft bar braceright asciitilde Adieresis Aring Ccedilla Eacute Ntilde Odieresis
13
+ Udieresis aacute agrave acircumflex adieresis atilde aring ccedilla eacute egrave
14
+ ecircumflex edieresis iacute igrave icircumflex idieresis ntilde oacute ograve
15
+ ocircumflex odieresis otilde uacute ugrave ucircumflex udieresis dagger degree cent
16
+ sterling section bullet paragraph germandbls registered copyright trademark acute
17
+ dieresis notequal AE Oslash infinity plusminus lessequal greaterequal yen mu
18
+ partialdiff summation product pi integral ordfeminine ordmasculine Omega ae oslash
19
+ questiondown exclamdown logicalnot radical florin approxequal Delta guillemotleft
20
+ guillemotright ellipsis nonbreakingspace Agrave Atilde Otilde OE oe endash emdash
21
+ quotedblleft quotedblright quoteleft quoteright divide lozenge ydieresis Ydieresis
22
+ fraction currency guilsinglleft guilsinglright fi fl daggerdbl periodcentered
23
+ quotesinglbase quotedblbase perthousand Acircumflex Ecircumflex Aacute Edieresis
24
+ Egrave Iacute Icircumflex Idieresis Igrave Oacute Ocircumflex apple Ograve Uacute
25
+ Ucircumflex Ugrave dotlessi circumflex tilde macron breve dotaccent ring cedilla
26
+ hungarumlaut ogonek caron Lslash lslash Scaron scaron Zcaron zcaron brokenbar Eth
27
+ eth Yacute yacute Thorn thorn minus multiply onesuperior twosuperior threesuperior
28
+ onehalf onequarter threequarters franc Gbreve gbreve Idotaccent Scedilla scedilla
29
+ Cacute cacute Ccaron ccaron dcroat)
30
+
31
+ def glyph_for(code)
32
+ POSTSCRIPT_GLYPHS[code] || ".notdef"
33
+ end
34
+
35
+ private
36
+
37
+ def parse_format!
38
+ # do nothing. Format 1 is easy-sauce.
39
+ end
40
+ end
41
+ end
42
+ end
43
+ end
@@ -0,0 +1,35 @@
1
+ require 'ttfunk/table/post/format10'
2
+ require 'stringio'
3
+
4
+ module TTFunk
5
+ class Table
6
+ class Post
7
+ module Format20
8
+ include Format10
9
+
10
+ def glyph_for(code)
11
+ index = @glyph_name_index[code]
12
+ if index <= 257
13
+ POSTSCRIPT_GLYPHS[index]
14
+ else
15
+ @names[index - 258] || ".notdef"
16
+ end
17
+ end
18
+
19
+ private
20
+
21
+ def parse_format!
22
+ number_of_glyphs = read(2, 'n').first
23
+ @glyph_name_index = read(number_of_glyphs*2, 'n*')
24
+ @names = []
25
+
26
+ strings = StringIO.new(io.read(offset + length - io.pos))
27
+ while !strings.eof?
28
+ length = strings.read(1).unpack("C").first
29
+ @names << strings.read(length)
30
+ end
31
+ end
32
+ end
33
+ end
34
+ end
35
+ end
@@ -0,0 +1,23 @@
1
+ require 'ttfunk/table/post/format10'
2
+ require 'stringio'
3
+
4
+ module TTFunk
5
+ class Table
6
+ class Post
7
+ module Format25
8
+ include Format10
9
+
10
+ def glyph_for(code)
11
+ POSTSCRIPT_GLYPHS[code + @offsets[code]] || ".notdef"
12
+ end
13
+
14
+ private
15
+
16
+ def parse_format!
17
+ number_of_glyphs = read(2, 'n').first
18
+ @offsets = read(@number_of_glyphs, "c*")
19
+ end
20
+ end
21
+ end
22
+ end
23
+ end
@@ -0,0 +1,17 @@
1
+ module TTFunk
2
+ class Table
3
+ class Post
4
+ module Format30
5
+ def glyph_for(code)
6
+ ".notdef"
7
+ end
8
+
9
+ private
10
+
11
+ def parse_format!
12
+ # do nothing. Format 3 is easy-sauce.
13
+ end
14
+ end
15
+ end
16
+ end
17
+ end