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,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,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,91 @@
1
+ require 'ttfunk/table'
2
+
3
+ module TTFunk
4
+ class Table
5
+ class Post < Table
6
+ attr_reader :format
7
+ attr_reader :italic_angle
8
+ attr_reader :underline_position
9
+ attr_reader :underline_thickness
10
+ attr_reader :fixed_pitch
11
+ attr_reader :min_mem_type42
12
+ attr_reader :max_mem_type42
13
+ attr_reader :min_mem_type1
14
+ attr_reader :max_mem_type1
15
+
16
+ attr_reader :subtable
17
+
18
+ def self.encode(post, mapping)
19
+ return nil unless post.exists?
20
+ post.recode(mapping)
21
+ end
22
+
23
+ def fixed_pitch?
24
+ @fixed_pitch != 0
25
+ end
26
+
27
+ def glyph_for(code)
28
+ ".notdef"
29
+ end
30
+
31
+ def recode(mapping)
32
+ return raw if format == 0x00030000
33
+
34
+ table = raw[0,32]
35
+ table[0,4] = [0x00020000].pack("N")
36
+
37
+ index = []
38
+ strings = []
39
+
40
+ mapping.keys.sort.each do |new_id|
41
+ post_glyph = glyph_for(mapping[new_id])
42
+ position = Format10::POSTSCRIPT_GLYPHS.index(post_glyph)
43
+ if position
44
+ index << position
45
+ else
46
+ index << 257 + strings.length
47
+ strings << post_glyph
48
+ end
49
+ end
50
+
51
+ table << [mapping.length, *index].pack("n*")
52
+ strings.each do |string|
53
+ table << [string.length, string].pack("CA*")
54
+ end
55
+
56
+ return table
57
+ end
58
+
59
+ private
60
+
61
+ def parse!
62
+ @format, @italic_angle, @underline_position, @underline_thickness,
63
+ @fixed_pitch, @min_mem_type42, @max_mem_type42,
64
+ @min_mem_type1, @max_mem_type1 = read(32, "N2n2N*")
65
+
66
+ end_of_table = offset + length
67
+
68
+ @subtable = case @format
69
+ when 0x00010000 then extend(Post::Format10)
70
+ when 0x00020000 then extend(Post::Format20)
71
+ when 0x00025000 then extend(Post::Format25)
72
+ when 0x00030000 then extend(Post::Format30)
73
+ when 0x00040000 then extend(Post::Format40)
74
+ end
75
+
76
+ parse_format!
77
+ end
78
+
79
+ def parse_format!
80
+ warn "postscript table format 0x%08X is not supported" % @format
81
+ end
82
+ end
83
+
84
+ end
85
+ end
86
+
87
+ require 'ttfunk/table/post/format10'
88
+ require 'ttfunk/table/post/format20'
89
+ require 'ttfunk/table/post/format25'
90
+ require 'ttfunk/table/post/format30'
91
+ require 'ttfunk/table/post/format40'
@@ -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