fontisan 0.2.17 → 0.2.22

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (316) hide show
  1. checksums.yaml +4 -4
  2. data/.rubocop_todo.yml +14 -90
  3. data/README.adoc +257 -1
  4. data/docs/.vitepress/config.ts +68 -8
  5. data/docs/.vitepress/theme/style.css +570 -272
  6. data/docs/CONVERSION_GUIDE.adoc +31 -8
  7. data/docs/EXTRACT_TTC_MIGRATION.md +1 -1
  8. data/docs/WOFF_WOFF2_FORMATS.adoc +53 -0
  9. data/docs/api/conversion-options.md +37 -14
  10. data/docs/cli/audit.md +337 -0
  11. data/docs/cli/convert.md +20 -1
  12. data/docs/cli/index.md +31 -0
  13. data/docs/guide/color.md +1 -1
  14. data/docs/guide/conversion/options.md +32 -3
  15. data/docs/guide/conversion/ttf-otf.md +1 -1
  16. data/docs/guide/conversion/type1.md +1 -1
  17. data/docs/guide/conversion/web.md +91 -32
  18. data/docs/guide/conversion.md +6 -5
  19. data/docs/guide/formats/woff.md +35 -11
  20. data/docs/guide/index.md +2 -2
  21. data/docs/guide/migrations/extract-ttc.md +1 -1
  22. data/docs/guide/quick-start.md +4 -4
  23. data/docs/guide/type1.md +4 -4
  24. data/docs/guide/woff.md +19 -17
  25. data/docs/index.md +2 -0
  26. data/docs/lychee.toml +5 -1
  27. data/docs/package.json +1 -1
  28. data/docs/public/robots.txt +4 -0
  29. data/docs/scripts/post-build.mjs +81 -0
  30. data/lib/fontisan/audit/codepoint_range_coalescer.rb +41 -0
  31. data/lib/fontisan/audit/context.rb +122 -0
  32. data/lib/fontisan/audit/differ.rb +124 -0
  33. data/lib/fontisan/audit/extractors/aggregations.rb +54 -0
  34. data/lib/fontisan/audit/extractors/base.rb +26 -0
  35. data/lib/fontisan/audit/extractors/color_capabilities.rb +141 -0
  36. data/lib/fontisan/audit/extractors/coverage.rb +48 -0
  37. data/lib/fontisan/audit/extractors/hinting.rb +197 -0
  38. data/lib/fontisan/audit/extractors/identity.rb +52 -0
  39. data/lib/fontisan/audit/extractors/language_coverage.rb +37 -0
  40. data/lib/fontisan/audit/extractors/licensing.rb +79 -0
  41. data/lib/fontisan/audit/extractors/metrics.rb +103 -0
  42. data/lib/fontisan/audit/extractors/opentype_layout.rb +69 -0
  43. data/lib/fontisan/audit/extractors/provenance.rb +29 -0
  44. data/lib/fontisan/audit/extractors/style.rb +32 -0
  45. data/lib/fontisan/audit/extractors/variation_detail.rb +99 -0
  46. data/lib/fontisan/audit/extractors.rb +27 -0
  47. data/lib/fontisan/audit/library_aggregator.rb +83 -0
  48. data/lib/fontisan/audit/library_auditor.rb +90 -0
  49. data/lib/fontisan/audit/registry.rb +60 -0
  50. data/lib/fontisan/audit/style_extractor.rb +80 -0
  51. data/lib/fontisan/audit.rb +20 -0
  52. data/lib/fontisan/base_collection.rb +23 -9
  53. data/lib/fontisan/binary/structures.rb +0 -2
  54. data/lib/fontisan/binary.rb +11 -0
  55. data/lib/fontisan/cldr/aggregator.rb +33 -0
  56. data/lib/fontisan/cldr/cache_manager.rb +110 -0
  57. data/lib/fontisan/cldr/config.rb +59 -0
  58. data/lib/fontisan/cldr/download_error.rb +9 -0
  59. data/lib/fontisan/cldr/downloader.rb +79 -0
  60. data/lib/fontisan/cldr/error.rb +8 -0
  61. data/lib/fontisan/cldr/index.rb +64 -0
  62. data/lib/fontisan/cldr/index_builder.rb +72 -0
  63. data/lib/fontisan/cldr/unicode_set_parser.rb +172 -0
  64. data/lib/fontisan/cldr/unknown_version_error.rb +9 -0
  65. data/lib/fontisan/cldr/version_resolver.rb +91 -0
  66. data/lib/fontisan/cldr.rb +23 -0
  67. data/lib/fontisan/cli/cldr_cli.rb +85 -0
  68. data/lib/fontisan/cli/ucd_cli.rb +97 -0
  69. data/lib/fontisan/cli.rb +201 -2
  70. data/lib/fontisan/collection/builder.rb +0 -4
  71. data/lib/fontisan/collection/dfont_builder.rb +0 -4
  72. data/lib/fontisan/collection/shared_logic.rb +0 -2
  73. data/lib/fontisan/collection/writer.rb +0 -3
  74. data/lib/fontisan/collection.rb +15 -0
  75. data/lib/fontisan/commands/audit_command.rb +123 -0
  76. data/lib/fontisan/commands/audit_compare_command.rb +66 -0
  77. data/lib/fontisan/commands/audit_library_command.rb +46 -0
  78. data/lib/fontisan/commands/base_command.rb +0 -3
  79. data/lib/fontisan/commands/convert_command.rb +25 -20
  80. data/lib/fontisan/commands/dump_table_command.rb +0 -3
  81. data/lib/fontisan/commands/export_command.rb +0 -4
  82. data/lib/fontisan/commands/features_command.rb +0 -3
  83. data/lib/fontisan/commands/instance_command.rb +0 -5
  84. data/lib/fontisan/commands/ls_command.rb +0 -6
  85. data/lib/fontisan/commands/optical_size_command.rb +0 -3
  86. data/lib/fontisan/commands/pack_command.rb +0 -5
  87. data/lib/fontisan/commands/scripts_command.rb +0 -2
  88. data/lib/fontisan/commands/subset_command.rb +0 -3
  89. data/lib/fontisan/commands/unicode_command.rb +0 -3
  90. data/lib/fontisan/commands/unpack_command.rb +0 -7
  91. data/lib/fontisan/commands/validate_command.rb +0 -8
  92. data/lib/fontisan/commands/variable_command.rb +0 -3
  93. data/lib/fontisan/commands.rb +29 -0
  94. data/lib/fontisan/config/cldr.yml +22 -0
  95. data/lib/fontisan/config/conversion_matrix.yml +38 -0
  96. data/lib/fontisan/config/ucd.yml +23 -0
  97. data/lib/fontisan/constants.rb +19 -0
  98. data/lib/fontisan/conversion_options.rb +30 -19
  99. data/lib/fontisan/converters/cff_table_builder.rb +0 -3
  100. data/lib/fontisan/converters/collection_converter.rb +0 -8
  101. data/lib/fontisan/converters/conversion_strategy.rb +161 -46
  102. data/lib/fontisan/converters/format_converter.rb +143 -32
  103. data/lib/fontisan/converters/glyf_table_builder.rb +0 -2
  104. data/lib/fontisan/converters/outline_converter.rb +0 -19
  105. data/lib/fontisan/converters/outline_extraction.rb +0 -5
  106. data/lib/fontisan/converters/outline_optimizer.rb +0 -5
  107. data/lib/fontisan/converters/svg_generator.rb +0 -4
  108. data/lib/fontisan/converters/table_copier.rb +0 -2
  109. data/lib/fontisan/converters/type1_converter.rb +0 -11
  110. data/lib/fontisan/converters/woff2_encoder.rb +49 -20
  111. data/lib/fontisan/converters/woff_writer.rb +211 -282
  112. data/lib/fontisan/converters.rb +21 -0
  113. data/lib/fontisan/dfont_collection.rb +29 -10
  114. data/lib/fontisan/export/exporter.rb +0 -6
  115. data/lib/fontisan/export/transformers/font_to_ttx.rb +0 -9
  116. data/lib/fontisan/export/transformers/head_transformer.rb +0 -2
  117. data/lib/fontisan/export/transformers/hhea_transformer.rb +0 -2
  118. data/lib/fontisan/export/transformers/maxp_transformer.rb +0 -2
  119. data/lib/fontisan/export/transformers/name_transformer.rb +0 -2
  120. data/lib/fontisan/export/transformers/os2_transformer.rb +0 -2
  121. data/lib/fontisan/export/transformers/post_transformer.rb +0 -2
  122. data/lib/fontisan/export/transformers.rb +17 -0
  123. data/lib/fontisan/export.rb +13 -0
  124. data/lib/fontisan/font_loader.rb +14 -19
  125. data/lib/fontisan/font_writer.rb +0 -1
  126. data/lib/fontisan/formatters/audit_diff_text_renderer.rb +122 -0
  127. data/lib/fontisan/formatters/audit_text_renderer.rb +324 -0
  128. data/lib/fontisan/formatters/library_summary_text_renderer.rb +99 -0
  129. data/lib/fontisan/formatters/text_formatter.rb +6 -0
  130. data/lib/fontisan/formatters.rb +12 -0
  131. data/lib/fontisan/hints/hint_converter.rb +0 -1
  132. data/lib/fontisan/hints/postscript_hint_applier.rb +0 -9
  133. data/lib/fontisan/hints/postscript_hint_extractor.rb +0 -2
  134. data/lib/fontisan/hints/truetype_hint_extractor.rb +0 -2
  135. data/lib/fontisan/hints.rb +16 -0
  136. data/lib/fontisan/metrics_calculator.rb +0 -2
  137. data/lib/fontisan/models/all_scripts_features_info.rb +0 -1
  138. data/lib/fontisan/models/audit/audit_axis.rb +30 -0
  139. data/lib/fontisan/models/audit/audit_block.rb +32 -0
  140. data/lib/fontisan/models/audit/audit_diff.rb +77 -0
  141. data/lib/fontisan/models/audit/audit_report.rb +153 -0
  142. data/lib/fontisan/models/audit/codepoint_range.rb +40 -0
  143. data/lib/fontisan/models/audit/codepoint_set_diff.rb +34 -0
  144. data/lib/fontisan/models/audit/color_capabilities.rb +93 -0
  145. data/lib/fontisan/models/audit/duplicate_group.rb +23 -0
  146. data/lib/fontisan/models/audit/embedding_type.rb +76 -0
  147. data/lib/fontisan/models/audit/field_change.rb +28 -0
  148. data/lib/fontisan/models/audit/fs_selection_flags.rb +61 -0
  149. data/lib/fontisan/models/audit/gasp_range.rb +63 -0
  150. data/lib/fontisan/models/audit/hinting.rb +93 -0
  151. data/lib/fontisan/models/audit/library_summary.rb +40 -0
  152. data/lib/fontisan/models/audit/licensing.rb +48 -0
  153. data/lib/fontisan/models/audit/metrics.rb +111 -0
  154. data/lib/fontisan/models/audit/named_instance.rb +41 -0
  155. data/lib/fontisan/models/audit/opentype_layout.rb +40 -0
  156. data/lib/fontisan/models/audit/script_coverage_row.rb +26 -0
  157. data/lib/fontisan/models/audit/script_features.rb +28 -0
  158. data/lib/fontisan/models/audit/variation_detail.rb +44 -0
  159. data/lib/fontisan/models/audit.rb +33 -0
  160. data/lib/fontisan/models/cldr/language_coverage.rb +31 -0
  161. data/lib/fontisan/models/cldr.rb +12 -0
  162. data/lib/fontisan/models/collection_brief_info.rb +0 -1
  163. data/lib/fontisan/models/collection_info.rb +0 -2
  164. data/lib/fontisan/models/collection_list_info.rb +0 -1
  165. data/lib/fontisan/models/collection_validation_report.rb +0 -2
  166. data/lib/fontisan/models/color_glyph.rb +0 -1
  167. data/lib/fontisan/models/font_report.rb +0 -1
  168. data/lib/fontisan/models/ttx/tables.rb +21 -0
  169. data/lib/fontisan/models/ttx/ttfont.rb +0 -8
  170. data/lib/fontisan/models/ttx.rb +14 -0
  171. data/lib/fontisan/models/ucd/ucd.rb +38 -0
  172. data/lib/fontisan/models/ucd/ucd_char.rb +67 -0
  173. data/lib/fontisan/models/ucd.rb +19 -0
  174. data/lib/fontisan/models.rb +47 -0
  175. data/lib/fontisan/open_type_collection.rb +6 -5
  176. data/lib/fontisan/open_type_font.rb +8 -2
  177. data/lib/fontisan/open_type_font_extensions.rb +9 -9
  178. data/lib/fontisan/optimizers/pattern_analyzer.rb +0 -1
  179. data/lib/fontisan/optimizers.rb +14 -0
  180. data/lib/fontisan/outline_extractor.rb +0 -2
  181. data/lib/fontisan/parsers/dfont_parser.rb +0 -1
  182. data/lib/fontisan/parsers.rb +10 -0
  183. data/lib/fontisan/pipeline/format_detector.rb +29 -102
  184. data/lib/fontisan/pipeline/output_writer.rb +11 -9
  185. data/lib/fontisan/pipeline/strategies/instance_strategy.rb +0 -4
  186. data/lib/fontisan/pipeline/strategies/named_strategy.rb +0 -4
  187. data/lib/fontisan/pipeline/strategies/preserve_strategy.rb +0 -2
  188. data/lib/fontisan/pipeline/strategies.rb +14 -0
  189. data/lib/fontisan/pipeline/transformation_pipeline.rb +0 -7
  190. data/lib/fontisan/pipeline/variation_resolver.rb +0 -7
  191. data/lib/fontisan/pipeline.rb +13 -0
  192. data/lib/fontisan/sfnt_font.rb +29 -14
  193. data/lib/fontisan/sfnt_table.rb +0 -4
  194. data/lib/fontisan/subset/builder.rb +0 -6
  195. data/lib/fontisan/subset.rb +13 -0
  196. data/lib/fontisan/svg/font_generator.rb +0 -4
  197. data/lib/fontisan/svg/glyph_generator.rb +0 -2
  198. data/lib/fontisan/svg.rb +12 -0
  199. data/lib/fontisan/tables/cbdt.rb +0 -1
  200. data/lib/fontisan/tables/cblc.rb +0 -1
  201. data/lib/fontisan/tables/cff/charset.rb +0 -1
  202. data/lib/fontisan/tables/cff/charstring.rb +0 -1
  203. data/lib/fontisan/tables/cff/charstring_rebuilder.rb +0 -4
  204. data/lib/fontisan/tables/cff/charstrings_index.rb +0 -3
  205. data/lib/fontisan/tables/cff/dict.rb +0 -1
  206. data/lib/fontisan/tables/cff/encoding.rb +0 -1
  207. data/lib/fontisan/tables/cff/header.rb +0 -2
  208. data/lib/fontisan/tables/cff/hint_operation_injector.rb +0 -2
  209. data/lib/fontisan/tables/cff/index.rb +0 -1
  210. data/lib/fontisan/tables/cff/private_dict.rb +0 -2
  211. data/lib/fontisan/tables/cff/private_dict_writer.rb +0 -2
  212. data/lib/fontisan/tables/cff/table_builder.rb +0 -6
  213. data/lib/fontisan/tables/cff/top_dict.rb +0 -2
  214. data/lib/fontisan/tables/cff.rb +22 -15
  215. data/lib/fontisan/tables/cff2/charstring_parser.rb +0 -2
  216. data/lib/fontisan/tables/cff2/table_builder.rb +0 -11
  217. data/lib/fontisan/tables/cff2/table_reader.rb +0 -2
  218. data/lib/fontisan/tables/cff2.rb +13 -14
  219. data/lib/fontisan/tables/cmap.rb +24 -2
  220. data/lib/fontisan/tables/cmap_table.rb +0 -3
  221. data/lib/fontisan/tables/colr.rb +0 -1
  222. data/lib/fontisan/tables/cpal.rb +0 -1
  223. data/lib/fontisan/tables/cvar.rb +0 -2
  224. data/lib/fontisan/tables/fvar.rb +0 -1
  225. data/lib/fontisan/tables/glyf/compound_glyph_resolver.rb +0 -2
  226. data/lib/fontisan/tables/glyf/glyph_builder.rb +0 -3
  227. data/lib/fontisan/tables/glyf.rb +0 -6
  228. data/lib/fontisan/tables/glyf_table.rb +0 -3
  229. data/lib/fontisan/tables/gpos.rb +0 -2
  230. data/lib/fontisan/tables/gsub.rb +0 -2
  231. data/lib/fontisan/tables/gvar.rb +0 -2
  232. data/lib/fontisan/tables/head.rb +0 -2
  233. data/lib/fontisan/tables/head_table.rb +0 -3
  234. data/lib/fontisan/tables/hhea.rb +0 -2
  235. data/lib/fontisan/tables/hhea_table.rb +0 -3
  236. data/lib/fontisan/tables/hmtx.rb +0 -2
  237. data/lib/fontisan/tables/hmtx_table.rb +0 -3
  238. data/lib/fontisan/tables/hvar.rb +0 -3
  239. data/lib/fontisan/tables/loca.rb +0 -2
  240. data/lib/fontisan/tables/loca_table.rb +0 -3
  241. data/lib/fontisan/tables/maxp.rb +0 -2
  242. data/lib/fontisan/tables/maxp_table.rb +0 -3
  243. data/lib/fontisan/tables/mvar.rb +0 -3
  244. data/lib/fontisan/tables/name.rb +0 -2
  245. data/lib/fontisan/tables/name_table.rb +0 -3
  246. data/lib/fontisan/tables/os2_table.rb +0 -3
  247. data/lib/fontisan/tables/post_table.rb +0 -3
  248. data/lib/fontisan/tables/sbix.rb +0 -1
  249. data/lib/fontisan/tables/svg.rb +0 -1
  250. data/lib/fontisan/tables/variation_common.rb +0 -1
  251. data/lib/fontisan/tables/vvar.rb +0 -3
  252. data/lib/fontisan/tables.rb +54 -0
  253. data/lib/fontisan/true_type_collection.rb +6 -14
  254. data/lib/fontisan/true_type_font.rb +8 -2
  255. data/lib/fontisan/true_type_font_extensions.rb +9 -9
  256. data/lib/fontisan/type1/afm_generator.rb +0 -4
  257. data/lib/fontisan/type1/conversion_options.rb +0 -2
  258. data/lib/fontisan/type1/encodings.rb +0 -2
  259. data/lib/fontisan/type1/generator.rb +0 -8
  260. data/lib/fontisan/type1/pfa_generator.rb +0 -3
  261. data/lib/fontisan/type1/pfb_generator.rb +0 -5
  262. data/lib/fontisan/type1/pfm_generator.rb +0 -4
  263. data/lib/fontisan/type1.rb +42 -69
  264. data/lib/fontisan/type1_font.rb +40 -11
  265. data/lib/fontisan/ucd/aggregator.rb +73 -0
  266. data/lib/fontisan/ucd/cache_manager.rb +111 -0
  267. data/lib/fontisan/ucd/config.rb +59 -0
  268. data/lib/fontisan/ucd/download_error.rb +9 -0
  269. data/lib/fontisan/ucd/downloader.rb +88 -0
  270. data/lib/fontisan/ucd/error.rb +8 -0
  271. data/lib/fontisan/ucd/index.rb +103 -0
  272. data/lib/fontisan/ucd/index_builder.rb +107 -0
  273. data/lib/fontisan/ucd/range_entry.rb +56 -0
  274. data/lib/fontisan/ucd/unknown_version_error.rb +9 -0
  275. data/lib/fontisan/ucd/version_resolver.rb +79 -0
  276. data/lib/fontisan/ucd.rb +23 -0
  277. data/lib/fontisan/utilities/checksum_calculator.rb +0 -1
  278. data/lib/fontisan/utilities.rb +10 -0
  279. data/lib/fontisan/utils.rb +10 -0
  280. data/lib/fontisan/validation/collection_validator.rb +0 -2
  281. data/lib/fontisan/validation.rb +9 -0
  282. data/lib/fontisan/validators/basic_validator.rb +0 -2
  283. data/lib/fontisan/validators/font_book_validator.rb +0 -2
  284. data/lib/fontisan/validators/opentype_validator.rb +0 -2
  285. data/lib/fontisan/validators/profile_loader.rb +0 -5
  286. data/lib/fontisan/validators/validator.rb +0 -2
  287. data/lib/fontisan/validators/web_font_validator.rb +0 -2
  288. data/lib/fontisan/validators.rb +14 -0
  289. data/lib/fontisan/variable/delta_applicator.rb +0 -4
  290. data/lib/fontisan/variable/instancer.rb +0 -3
  291. data/lib/fontisan/variable/static_font_builder.rb +0 -3
  292. data/lib/fontisan/variable.rb +16 -0
  293. data/lib/fontisan/variation/blend_applier.rb +0 -2
  294. data/lib/fontisan/variation/cache.rb +0 -2
  295. data/lib/fontisan/variation/converter.rb +0 -3
  296. data/lib/fontisan/variation/data_extractor.rb +0 -2
  297. data/lib/fontisan/variation/delta_applier.rb +0 -5
  298. data/lib/fontisan/variation/inspector.rb +0 -1
  299. data/lib/fontisan/variation/instance_generator.rb +0 -6
  300. data/lib/fontisan/variation/instance_writer.rb +0 -5
  301. data/lib/fontisan/variation/metrics_adjuster.rb +0 -4
  302. data/lib/fontisan/variation/optimizer.rb +0 -3
  303. data/lib/fontisan/variation/parallel_generator.rb +0 -3
  304. data/lib/fontisan/variation/subsetter.rb +0 -4
  305. data/lib/fontisan/variation/tuple_variation_header.rb +0 -2
  306. data/lib/fontisan/variation/variable_svg_generator.rb +0 -3
  307. data/lib/fontisan/variation/variation_context.rb +0 -3
  308. data/lib/fontisan/variation/variation_preserver.rb +0 -3
  309. data/lib/fontisan/variation.rb +31 -0
  310. data/lib/fontisan/version.rb +1 -1
  311. data/lib/fontisan/woff2.rb +13 -0
  312. data/lib/fontisan/woff2_font.rb +31 -9
  313. data/lib/fontisan/woff_font.rb +31 -2
  314. data/lib/fontisan.rb +124 -196
  315. metadata +114 -7
  316. data/fontisan.gemspec +0 -48
@@ -0,0 +1,41 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Fontisan
4
+ module Audit
5
+ # Coalesces a flat codepoint list into contiguous {Models::Audit::CodepointRange}s.
6
+ #
7
+ # Single static call site, deterministic output. Used by the Coverage
8
+ # extractor to produce the compact range view that is the default
9
+ # AuditReport shape.
10
+ module CodepointRangeCoalescer
11
+ module_function
12
+
13
+ # @param codepoints [Enumerable<Integer>] any enumeration of integers
14
+ # @return [Array<Models::Audit::CodepointRange>] contiguous, sorted
15
+ def call(codepoints)
16
+ return [] if codepoints.nil? || codepoints.empty?
17
+
18
+ sorted = codepoints.sort.uniq
19
+ ranges = []
20
+ range_start = sorted[0]
21
+ prev = sorted[0]
22
+
23
+ sorted[1..].each do |cp|
24
+ next if cp == prev # defensive: .uniq already handles this
25
+
26
+ if cp == prev + 1
27
+ prev = cp
28
+ else
29
+ ranges << Models::Audit::CodepointRange.new(first_cp: range_start,
30
+ last_cp: prev)
31
+ range_start = cp
32
+ prev = cp
33
+ end
34
+ end
35
+ ranges << Models::Audit::CodepointRange.new(first_cp: range_start,
36
+ last_cp: prev)
37
+ ranges
38
+ end
39
+ end
40
+ end
41
+ end
@@ -0,0 +1,122 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Fontisan
4
+ module Audit
5
+ # Value object carrying everything an extractor needs to do its job.
6
+ #
7
+ # Extractors never reach back into AuditCommand state — they read
8
+ # exclusively from the Context. Shared derived data (codepoints,
9
+ # UCD indices, source format) is memoized here so multiple
10
+ # extractors don't recompute it.
11
+ class Context
12
+ attr_reader :font, :font_path, :font_index, :num_fonts_in_source,
13
+ :options
14
+
15
+ def initialize(font:, font_path:, font_index:, num_fonts_in_source:,
16
+ options:)
17
+ @font = font
18
+ @font_path = font_path
19
+ @font_index = font_index
20
+ @num_fonts_in_source = num_fonts_in_source
21
+ @options = options
22
+ end
23
+
24
+ def codepoints
25
+ @codepoints ||= extract_codepoints
26
+ end
27
+
28
+ def ucd
29
+ @ucd ||= resolve_ucd
30
+ end
31
+
32
+ def cldr
33
+ return nil unless @options[:with_language_coverage]
34
+
35
+ @cldr ||= resolve_cldr
36
+ end
37
+
38
+ def source_format
39
+ @source_format ||= FontLoader.detect_format(@font_path)&.to_s
40
+ end
41
+
42
+ def all_codepoints?
43
+ @options[:all_codepoints] == true
44
+ end
45
+
46
+ private
47
+
48
+ def extract_codepoints
49
+ return [] unless @font.has_table?(Constants::CMAP_TAG)
50
+
51
+ @font.table(Constants::CMAP_TAG).unicode_mappings.keys
52
+ end
53
+
54
+ def resolve_ucd
55
+ version = Ucd::VersionResolver.resolve(@options[:ucd_version])
56
+
57
+ with_local_indices(version) do |blocks_path, scripts_path|
58
+ {
59
+ version: version,
60
+ blocks_index: Ucd::Index.load(blocks_path),
61
+ scripts_index: Ucd::Index.load(scripts_path),
62
+ warning: nil,
63
+ }
64
+ end
65
+ rescue Ucd::UnknownVersionError => e
66
+ { version: nil, blocks_index: nil, scripts_index: nil,
67
+ warning: "UCD version rejected: #{e.message}" }
68
+ rescue StandardError => e
69
+ version_ref = @ucd&.fetch(:version, nil)
70
+ {
71
+ version: version_ref,
72
+ blocks_index: nil,
73
+ scripts_index: nil,
74
+ warning: "UCD unavailable for version #{version_ref}: #{e.message}",
75
+ }
76
+ end
77
+
78
+ def with_local_indices(version)
79
+ unless Ucd::CacheManager.cached?(version)
80
+ Ucd::Downloader.download(version)
81
+ end
82
+ unless Ucd::CacheManager.blocks_index_path(version).exist?
83
+ Ucd::IndexBuilder.build(version)
84
+ end
85
+ yield Ucd::CacheManager.blocks_index_path(version),
86
+ Ucd::CacheManager.scripts_index_path(version)
87
+ end
88
+
89
+ def resolve_cldr
90
+ version = Cldr::VersionResolver.resolve(@options[:cldr_version])
91
+
92
+ with_local_languages_index(version) do |index_path|
93
+ {
94
+ version: version,
95
+ index: Cldr::Index.load(index_path),
96
+ warning: nil,
97
+ }
98
+ end
99
+ rescue Cldr::UnknownVersionError => e
100
+ { version: nil, index: nil,
101
+ warning: "CLDR version rejected: #{e.message}" }
102
+ rescue StandardError => e
103
+ version_ref = @cldr&.fetch(:version, nil)
104
+ {
105
+ version: version_ref,
106
+ index: nil,
107
+ warning: "CLDR unavailable for version #{version_ref}: #{e.message}",
108
+ }
109
+ end
110
+
111
+ def with_local_languages_index(version)
112
+ unless Cldr::CacheManager.cached?(version)
113
+ Cldr::Downloader.download(version)
114
+ end
115
+ unless Cldr::CacheManager.languages_index_path(version).exist?
116
+ Cldr::IndexBuilder.build(version)
117
+ end
118
+ yield Cldr::CacheManager.languages_index_path(version)
119
+ end
120
+ end
121
+ end
122
+ end
@@ -0,0 +1,124 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Fontisan
4
+ module Audit
5
+ # Computes an {Models::Audit::AuditDiff} between two AuditReports.
6
+ #
7
+ # Pure: no I/O, no font parsing. Both reports must already be built
8
+ # (Commands::AuditCompareCommand handles loading reports from disk
9
+ # or auditing fresh fonts before invoking the differ).
10
+ #
11
+ # Comparison shape:
12
+ # - Scalar fields: one FieldChange per differing field.
13
+ # - Codepoint coverage: CodepointSetDiff built from the cmap range
14
+ # lists (expanded to integer sets for set arithmetic, then
15
+ # re-coalesced to ranges for output).
16
+ # - Structural inventories (features, scripts, blocks, languages):
17
+ # simple array set-diffs.
18
+ class Differ
19
+ # Scalar AuditReport fields compared field-by-field. Excludes
20
+ # generated_at / source_sha256 / source_file (per-report identity),
21
+ # codepoints / codepoint_ranges (handled via CodepointSetDiff),
22
+ # and nested models (surfaced via structural add/remove lists).
23
+ COMPARED_FIELDS = %i[
24
+ family_name subfamily_name full_name postscript_name version
25
+ font_revision weight_class width_class italic bold panose
26
+ total_codepoints total_glyphs ucd_version cldr_version
27
+ ].freeze
28
+
29
+ def initialize(left_report, right_report)
30
+ @left = left_report
31
+ @right = right_report
32
+ end
33
+
34
+ # @return [Models::Audit::AuditDiff]
35
+ def diff
36
+ Models::Audit::AuditDiff.new(
37
+ left_source: @left.source_file,
38
+ right_source: @right.source_file,
39
+ field_changes: field_changes,
40
+ codepoints: codepoint_diff,
41
+ added_features: set_diff(features(@right), features(@left)),
42
+ removed_features: set_diff(features(@left), features(@right)),
43
+ added_scripts: set_diff(scripts(@right), scripts(@left)),
44
+ removed_scripts: set_diff(scripts(@left), scripts(@right)),
45
+ added_blocks: set_diff(blocks(@right), blocks(@left)),
46
+ removed_blocks: set_diff(blocks(@left), blocks(@right)),
47
+ added_languages: set_diff(languages(@right), languages(@left)),
48
+ removed_languages: set_diff(languages(@left), languages(@right)),
49
+ )
50
+ end
51
+
52
+ private
53
+
54
+ def field_changes
55
+ COMPARED_FIELDS.filter_map do |field|
56
+ left_val = @left.public_send(field)
57
+ right_val = @right.public_send(field)
58
+ next if left_val == right_val
59
+
60
+ Models::Audit::FieldChange.new(
61
+ field: field.to_s,
62
+ left: serialize_value(left_val),
63
+ right: serialize_value(right_val),
64
+ )
65
+ end
66
+ end
67
+
68
+ def codepoint_diff
69
+ left_set = codepoints_from_ranges(@left)
70
+ right_set = codepoints_from_ranges(@right)
71
+ added = right_set - left_set
72
+ removed = left_set - right_set
73
+ unchanged = left_set & right_set
74
+
75
+ Models::Audit::CodepointSetDiff.new(
76
+ added: CodepointRangeCoalescer.call(added.to_a),
77
+ removed: CodepointRangeCoalescer.call(removed.to_a),
78
+ added_count: added.size,
79
+ removed_count: removed.size,
80
+ unchanged_count: unchanged.size,
81
+ )
82
+ end
83
+
84
+ # Expand a report's compact codepoint range list into a Set<Integer>.
85
+ # Works for both default reports (range list populated) and
86
+ # --all-codepoints reports (range list is also populated).
87
+ def codepoints_from_ranges(report)
88
+ ranges = report.codepoint_ranges || []
89
+ ranges.each_with_object(Set.new) do |range, set|
90
+ (range.first_cp..range.last_cp).each { |cp| set << cp }
91
+ end
92
+ end
93
+
94
+ def features(report)
95
+ report.opentype_layout&.features || []
96
+ end
97
+
98
+ def scripts(report)
99
+ report.opentype_layout&.scripts || []
100
+ end
101
+
102
+ def blocks(report)
103
+ (report.blocks || []).map(&:name)
104
+ end
105
+
106
+ def languages(report)
107
+ (report.language_coverage || []).map(&:language)
108
+ end
109
+
110
+ def set_diff(minuend, subtrahend)
111
+ (Array(minuend) - Array(subtrahend)).sort
112
+ end
113
+
114
+ def serialize_value(value)
115
+ case value
116
+ when nil then ""
117
+ when String, Integer, Float then value.to_s
118
+ when true, false then value.to_s
119
+ else value.to_yaml
120
+ end
121
+ end
122
+ end
123
+ end
124
+ end
@@ -0,0 +1,54 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Fontisan
4
+ module Audit
5
+ module Extractors
6
+ # Aggregation fields: UCD block/script coverage.
7
+ #
8
+ # Returned fields:
9
+ # ucd_version, blocks, unicode_scripts
10
+ #
11
+ # OpenType script/feature inventory lives in {Extractors::OpenTypeLayout}
12
+ # (MECE: this extractor is UCD-driven, that one is SFNT-table-driven).
13
+ class Aggregations < Base
14
+ def extract(context)
15
+ ucd = context.ucd
16
+ ucd_aggregations(context.codepoints, ucd)
17
+ end
18
+
19
+ private
20
+
21
+ def ucd_aggregations(codepoints, ucd)
22
+ return empty_aggregation(ucd) if ucd[:blocks_index].nil?
23
+
24
+ blocks_hashes = Ucd::Aggregator.aggregate_blocks(codepoints,
25
+ ucd[:blocks_index])
26
+ {
27
+ ucd_version: ucd[:version],
28
+ blocks: blocks_hashes.map { |h| build_audit_block(h) },
29
+ unicode_scripts: Ucd::Aggregator.aggregate_scripts(codepoints,
30
+ ucd[:scripts_index]),
31
+ }
32
+ end
33
+
34
+ def empty_aggregation(ucd)
35
+ { ucd_version: ucd[:version], blocks: [], unicode_scripts: [] }
36
+ end
37
+
38
+ def build_audit_block(block_hash)
39
+ Models::Audit::AuditBlock.new(
40
+ name: block_hash[:name],
41
+ first_cp: block_hash[:first_cp],
42
+ last_cp: block_hash[:last_cp],
43
+ range: format("U+%<first>04X-U+%<last>04X",
44
+ first: block_hash[:first_cp], last: block_hash[:last_cp]),
45
+ total: block_hash[:total],
46
+ covered: block_hash[:covered],
47
+ fill_ratio: block_hash[:fill_ratio],
48
+ complete: block_hash[:complete],
49
+ )
50
+ end
51
+ end
52
+ end
53
+ end
54
+ end
@@ -0,0 +1,26 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Fontisan
4
+ module Audit
5
+ module Extractors
6
+ # Abstract extractor interface. Subclasses implement `#extract`.
7
+ #
8
+ # An extractor reads from a Context and returns a hash of fields
9
+ # suitable for `Models::Audit::AuditReport.new(**fields)`.
10
+ # Returning an empty hash is valid (no-op).
11
+ class Base
12
+ def extract(context)
13
+ raise NotImplementedError,
14
+ "#{self.class} must implement #extract"
15
+ end
16
+
17
+ protected
18
+
19
+ # Convenience accessor used by most extractors.
20
+ def font(context)
21
+ context.font
22
+ end
23
+ end
24
+ end
25
+ end
26
+ end
@@ -0,0 +1,141 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Fontisan
4
+ module Audit
5
+ module Extractors
6
+ # Color-font capability summary: which color formats a face carries
7
+ # (COLR v0/v1, CPAL, SVG, CBDT/CBLC, sbix) plus lightweight counts
8
+ # from each table's header.
9
+ #
10
+ # Returned fields:
11
+ # color_capabilities: Models::Audit::ColorCapabilities, or nil
12
+ # for Type 1
13
+ #
14
+ # Counts are best-effort — any table that fails to parse yields nil
15
+ # for its corresponding count fields rather than crashing the audit.
16
+ class ColorCapabilities < Base
17
+ def extract(context)
18
+ font = context.font
19
+ return { color_capabilities: nil } unless sfnt?(font)
20
+
21
+ { color_capabilities: Models::Audit::ColorCapabilities.new(**gather(font)) }
22
+ end
23
+
24
+ protected
25
+
26
+ def sfnt?(font)
27
+ font.is_a?(SfntFont)
28
+ end
29
+
30
+ private
31
+
32
+ def gather(font)
33
+ colr = colr_fields(font)
34
+ cpal = cpal_fields(font)
35
+ svg = svg_fields(font)
36
+ cbdt = cbdt_fields(font)
37
+ sbix = sbix_fields(font)
38
+
39
+ formats = Models::Audit::ColorCapabilities.derive_formats(
40
+ has_colr: colr[:has_colr], colr_version: colr[:colr_version],
41
+ has_cpal: cpal[:has_cpal], has_svg: svg[:has_svg],
42
+ has_cbdt: cbdt[:has_cbdt], has_sbix: sbix[:has_sbix]
43
+ )
44
+
45
+ colr.merge(cpal).merge(svg).merge(cbdt).merge(sbix)
46
+ .merge(color_formats: formats)
47
+ end
48
+
49
+ def colr_fields(font)
50
+ return empty_colr unless font.has_table?(Constants::COLR_TAG)
51
+
52
+ colr = font.table(Constants::COLR_TAG)
53
+ return empty_colr unless colr
54
+
55
+ {
56
+ has_colr: true,
57
+ colr_version: colr.version&.to_i,
58
+ colr_base_glyph_count: colr.num_base_glyph_records&.to_i,
59
+ colr_layer_count: colr.num_layer_records&.to_i,
60
+ }
61
+ end
62
+
63
+ def empty_colr
64
+ { has_colr: false, colr_version: nil,
65
+ colr_base_glyph_count: nil, colr_layer_count: nil }
66
+ end
67
+
68
+ def cpal_fields(font)
69
+ return empty_cpal unless font.has_table?(Constants::CPAL_TAG)
70
+
71
+ cpal = font.table(Constants::CPAL_TAG)
72
+ return empty_cpal unless cpal
73
+
74
+ {
75
+ has_cpal: true,
76
+ cpal_palette_count: cpal.num_palettes&.to_i,
77
+ cpal_color_count: cpal.num_color_records&.to_i,
78
+ }
79
+ end
80
+
81
+ def empty_cpal
82
+ { has_cpal: false, cpal_palette_count: nil, cpal_color_count: nil }
83
+ end
84
+
85
+ def svg_fields(font)
86
+ return empty_svg unless font.has_table?(Constants::SVG_TAG)
87
+
88
+ svg = font.table(Constants::SVG_TAG)
89
+ return empty_svg unless svg
90
+
91
+ {
92
+ has_svg: true,
93
+ svg_document_count: svg.num_svg_documents&.to_i,
94
+ }
95
+ end
96
+
97
+ def empty_svg
98
+ { has_svg: false, svg_document_count: nil }
99
+ end
100
+
101
+ # CBDT/CBLC are paired tables: CBLC holds the strike index,
102
+ # CBDT holds the bitmap data. has_cbdt vs has_cblc disagreement
103
+ # is reported as-is — audit consumers can spot the inconsistency.
104
+ def cbdt_fields(font)
105
+ has_cbdt = font.has_table?(Constants::CBDT_TAG)
106
+ has_cblc = font.has_table?(Constants::CBLC_TAG)
107
+ strike_count = cblc_strike_count(font) if has_cblc
108
+
109
+ {
110
+ has_cbdt: has_cbdt,
111
+ has_cblc: has_cblc,
112
+ cbdt_strike_count: strike_count,
113
+ }
114
+ end
115
+
116
+ def cblc_strike_count(font)
117
+ cblc = font.table(Constants::CBLC_TAG)
118
+ return nil unless cblc
119
+
120
+ cblc.num_sizes&.to_i
121
+ end
122
+
123
+ def sbix_fields(font)
124
+ return empty_sbix unless font.has_table?(Constants::SBIX_TAG)
125
+
126
+ sbix = font.table(Constants::SBIX_TAG)
127
+ return empty_sbix unless sbix
128
+
129
+ {
130
+ has_sbix: true,
131
+ sbix_strike_count: sbix.num_strikes&.to_i,
132
+ }
133
+ end
134
+
135
+ def empty_sbix
136
+ { has_sbix: false, sbix_strike_count: nil }
137
+ end
138
+ end
139
+ end
140
+ end
141
+ end
@@ -0,0 +1,48 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Fontisan
4
+ module Audit
5
+ module Extractors
6
+ # Coverage fields: how many codepoints and glyphs the font ships,
7
+ # the compact codepoint-range view (default), and the optional flat
8
+ # per-codepoint list (only when `--all-codepoints` is on).
9
+ #
10
+ # Returned fields:
11
+ # total_codepoints, total_glyphs, cmap_subtables,
12
+ # codepoint_ranges, codepoints
13
+ class Coverage < Base
14
+ def extract(context)
15
+ font = context.font
16
+ codepoints = context.codepoints
17
+ {
18
+ total_codepoints: codepoints.length,
19
+ total_glyphs: total_glyphs(font),
20
+ cmap_subtables: cmap_subtable_formats(font),
21
+ codepoint_ranges: CodepointRangeCoalescer.call(codepoints),
22
+ codepoints: codepoints_for_report(context, codepoints),
23
+ }
24
+ end
25
+
26
+ private
27
+
28
+ def total_glyphs(font)
29
+ return nil unless font.has_table?(Constants::MAXP_TAG)
30
+
31
+ font.table(Constants::MAXP_TAG).num_glyphs
32
+ end
33
+
34
+ def cmap_subtable_formats(font)
35
+ return [] unless font.has_table?(Constants::CMAP_TAG)
36
+
37
+ font.table(Constants::CMAP_TAG).subtable_formats
38
+ end
39
+
40
+ def codepoints_for_report(context, codepoints)
41
+ return [] unless context.all_codepoints?
42
+
43
+ codepoints.map { |cp| format("U+%<cp>04X", cp: cp) }
44
+ end
45
+ end
46
+ end
47
+ end
48
+ end