fonttools 4.60.2__cp311-cp311-win32.whl

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 (353) hide show
  1. fontTools/__init__.py +8 -0
  2. fontTools/__main__.py +35 -0
  3. fontTools/afmLib.py +439 -0
  4. fontTools/agl.py +5233 -0
  5. fontTools/annotations.py +30 -0
  6. fontTools/cffLib/CFF2ToCFF.py +258 -0
  7. fontTools/cffLib/CFFToCFF2.py +305 -0
  8. fontTools/cffLib/__init__.py +3694 -0
  9. fontTools/cffLib/specializer.py +927 -0
  10. fontTools/cffLib/transforms.py +495 -0
  11. fontTools/cffLib/width.py +210 -0
  12. fontTools/colorLib/__init__.py +0 -0
  13. fontTools/colorLib/builder.py +664 -0
  14. fontTools/colorLib/errors.py +2 -0
  15. fontTools/colorLib/geometry.py +143 -0
  16. fontTools/colorLib/table_builder.py +223 -0
  17. fontTools/colorLib/unbuilder.py +81 -0
  18. fontTools/config/__init__.py +90 -0
  19. fontTools/cu2qu/__init__.py +15 -0
  20. fontTools/cu2qu/__main__.py +6 -0
  21. fontTools/cu2qu/benchmark.py +54 -0
  22. fontTools/cu2qu/cli.py +198 -0
  23. fontTools/cu2qu/cu2qu.c +15817 -0
  24. fontTools/cu2qu/cu2qu.cp311-win32.pyd +0 -0
  25. fontTools/cu2qu/cu2qu.py +563 -0
  26. fontTools/cu2qu/errors.py +77 -0
  27. fontTools/cu2qu/ufo.py +363 -0
  28. fontTools/designspaceLib/__init__.py +3343 -0
  29. fontTools/designspaceLib/__main__.py +6 -0
  30. fontTools/designspaceLib/split.py +475 -0
  31. fontTools/designspaceLib/statNames.py +260 -0
  32. fontTools/designspaceLib/types.py +147 -0
  33. fontTools/encodings/MacRoman.py +258 -0
  34. fontTools/encodings/StandardEncoding.py +258 -0
  35. fontTools/encodings/__init__.py +1 -0
  36. fontTools/encodings/codecs.py +135 -0
  37. fontTools/feaLib/__init__.py +4 -0
  38. fontTools/feaLib/__main__.py +78 -0
  39. fontTools/feaLib/ast.py +2143 -0
  40. fontTools/feaLib/builder.py +1814 -0
  41. fontTools/feaLib/error.py +22 -0
  42. fontTools/feaLib/lexer.c +17029 -0
  43. fontTools/feaLib/lexer.cp311-win32.pyd +0 -0
  44. fontTools/feaLib/lexer.py +287 -0
  45. fontTools/feaLib/location.py +12 -0
  46. fontTools/feaLib/lookupDebugInfo.py +12 -0
  47. fontTools/feaLib/parser.py +2394 -0
  48. fontTools/feaLib/variableScalar.py +118 -0
  49. fontTools/fontBuilder.py +1014 -0
  50. fontTools/help.py +36 -0
  51. fontTools/merge/__init__.py +248 -0
  52. fontTools/merge/__main__.py +6 -0
  53. fontTools/merge/base.py +81 -0
  54. fontTools/merge/cmap.py +173 -0
  55. fontTools/merge/layout.py +526 -0
  56. fontTools/merge/options.py +85 -0
  57. fontTools/merge/tables.py +352 -0
  58. fontTools/merge/unicode.py +78 -0
  59. fontTools/merge/util.py +143 -0
  60. fontTools/misc/__init__.py +1 -0
  61. fontTools/misc/arrayTools.py +424 -0
  62. fontTools/misc/bezierTools.c +39731 -0
  63. fontTools/misc/bezierTools.cp311-win32.pyd +0 -0
  64. fontTools/misc/bezierTools.py +1500 -0
  65. fontTools/misc/classifyTools.py +170 -0
  66. fontTools/misc/cliTools.py +53 -0
  67. fontTools/misc/configTools.py +349 -0
  68. fontTools/misc/cython.py +27 -0
  69. fontTools/misc/dictTools.py +83 -0
  70. fontTools/misc/eexec.py +119 -0
  71. fontTools/misc/encodingTools.py +72 -0
  72. fontTools/misc/enumTools.py +23 -0
  73. fontTools/misc/etree.py +456 -0
  74. fontTools/misc/filenames.py +245 -0
  75. fontTools/misc/filesystem/__init__.py +68 -0
  76. fontTools/misc/filesystem/_base.py +134 -0
  77. fontTools/misc/filesystem/_copy.py +45 -0
  78. fontTools/misc/filesystem/_errors.py +54 -0
  79. fontTools/misc/filesystem/_info.py +75 -0
  80. fontTools/misc/filesystem/_osfs.py +164 -0
  81. fontTools/misc/filesystem/_path.py +67 -0
  82. fontTools/misc/filesystem/_subfs.py +92 -0
  83. fontTools/misc/filesystem/_tempfs.py +34 -0
  84. fontTools/misc/filesystem/_tools.py +34 -0
  85. fontTools/misc/filesystem/_walk.py +55 -0
  86. fontTools/misc/filesystem/_zipfs.py +204 -0
  87. fontTools/misc/fixedTools.py +253 -0
  88. fontTools/misc/intTools.py +25 -0
  89. fontTools/misc/iterTools.py +12 -0
  90. fontTools/misc/lazyTools.py +42 -0
  91. fontTools/misc/loggingTools.py +543 -0
  92. fontTools/misc/macCreatorType.py +56 -0
  93. fontTools/misc/macRes.py +261 -0
  94. fontTools/misc/plistlib/__init__.py +681 -0
  95. fontTools/misc/plistlib/py.typed +0 -0
  96. fontTools/misc/psCharStrings.py +1511 -0
  97. fontTools/misc/psLib.py +398 -0
  98. fontTools/misc/psOperators.py +572 -0
  99. fontTools/misc/py23.py +96 -0
  100. fontTools/misc/roundTools.py +110 -0
  101. fontTools/misc/sstruct.py +227 -0
  102. fontTools/misc/symfont.py +242 -0
  103. fontTools/misc/testTools.py +233 -0
  104. fontTools/misc/textTools.py +156 -0
  105. fontTools/misc/timeTools.py +88 -0
  106. fontTools/misc/transform.py +516 -0
  107. fontTools/misc/treeTools.py +45 -0
  108. fontTools/misc/vector.py +147 -0
  109. fontTools/misc/visitor.py +158 -0
  110. fontTools/misc/xmlReader.py +188 -0
  111. fontTools/misc/xmlWriter.py +231 -0
  112. fontTools/mtiLib/__init__.py +1400 -0
  113. fontTools/mtiLib/__main__.py +5 -0
  114. fontTools/otlLib/__init__.py +1 -0
  115. fontTools/otlLib/builder.py +3465 -0
  116. fontTools/otlLib/error.py +11 -0
  117. fontTools/otlLib/maxContextCalc.py +96 -0
  118. fontTools/otlLib/optimize/__init__.py +53 -0
  119. fontTools/otlLib/optimize/__main__.py +6 -0
  120. fontTools/otlLib/optimize/gpos.py +439 -0
  121. fontTools/pens/__init__.py +1 -0
  122. fontTools/pens/areaPen.py +52 -0
  123. fontTools/pens/basePen.py +475 -0
  124. fontTools/pens/boundsPen.py +98 -0
  125. fontTools/pens/cairoPen.py +26 -0
  126. fontTools/pens/cocoaPen.py +26 -0
  127. fontTools/pens/cu2quPen.py +325 -0
  128. fontTools/pens/explicitClosingLinePen.py +101 -0
  129. fontTools/pens/filterPen.py +433 -0
  130. fontTools/pens/freetypePen.py +462 -0
  131. fontTools/pens/hashPointPen.py +89 -0
  132. fontTools/pens/momentsPen.c +13378 -0
  133. fontTools/pens/momentsPen.cp311-win32.pyd +0 -0
  134. fontTools/pens/momentsPen.py +879 -0
  135. fontTools/pens/perimeterPen.py +69 -0
  136. fontTools/pens/pointInsidePen.py +192 -0
  137. fontTools/pens/pointPen.py +643 -0
  138. fontTools/pens/qtPen.py +29 -0
  139. fontTools/pens/qu2cuPen.py +105 -0
  140. fontTools/pens/quartzPen.py +43 -0
  141. fontTools/pens/recordingPen.py +335 -0
  142. fontTools/pens/reportLabPen.py +79 -0
  143. fontTools/pens/reverseContourPen.py +96 -0
  144. fontTools/pens/roundingPen.py +130 -0
  145. fontTools/pens/statisticsPen.py +312 -0
  146. fontTools/pens/svgPathPen.py +310 -0
  147. fontTools/pens/t2CharStringPen.py +88 -0
  148. fontTools/pens/teePen.py +55 -0
  149. fontTools/pens/transformPen.py +115 -0
  150. fontTools/pens/ttGlyphPen.py +335 -0
  151. fontTools/pens/wxPen.py +29 -0
  152. fontTools/qu2cu/__init__.py +15 -0
  153. fontTools/qu2cu/__main__.py +7 -0
  154. fontTools/qu2cu/benchmark.py +56 -0
  155. fontTools/qu2cu/cli.py +125 -0
  156. fontTools/qu2cu/qu2cu.c +16682 -0
  157. fontTools/qu2cu/qu2cu.cp311-win32.pyd +0 -0
  158. fontTools/qu2cu/qu2cu.py +405 -0
  159. fontTools/subset/__init__.py +4096 -0
  160. fontTools/subset/__main__.py +6 -0
  161. fontTools/subset/cff.py +184 -0
  162. fontTools/subset/svg.py +253 -0
  163. fontTools/subset/util.py +25 -0
  164. fontTools/svgLib/__init__.py +3 -0
  165. fontTools/svgLib/path/__init__.py +65 -0
  166. fontTools/svgLib/path/arc.py +154 -0
  167. fontTools/svgLib/path/parser.py +322 -0
  168. fontTools/svgLib/path/shapes.py +183 -0
  169. fontTools/t1Lib/__init__.py +648 -0
  170. fontTools/tfmLib.py +460 -0
  171. fontTools/ttLib/__init__.py +30 -0
  172. fontTools/ttLib/__main__.py +148 -0
  173. fontTools/ttLib/macUtils.py +54 -0
  174. fontTools/ttLib/removeOverlaps.py +395 -0
  175. fontTools/ttLib/reorderGlyphs.py +285 -0
  176. fontTools/ttLib/scaleUpem.py +436 -0
  177. fontTools/ttLib/sfnt.py +661 -0
  178. fontTools/ttLib/standardGlyphOrder.py +271 -0
  179. fontTools/ttLib/tables/B_A_S_E_.py +14 -0
  180. fontTools/ttLib/tables/BitmapGlyphMetrics.py +64 -0
  181. fontTools/ttLib/tables/C_B_D_T_.py +113 -0
  182. fontTools/ttLib/tables/C_B_L_C_.py +19 -0
  183. fontTools/ttLib/tables/C_F_F_.py +61 -0
  184. fontTools/ttLib/tables/C_F_F__2.py +26 -0
  185. fontTools/ttLib/tables/C_O_L_R_.py +165 -0
  186. fontTools/ttLib/tables/C_P_A_L_.py +305 -0
  187. fontTools/ttLib/tables/D_S_I_G_.py +158 -0
  188. fontTools/ttLib/tables/D__e_b_g.py +35 -0
  189. fontTools/ttLib/tables/DefaultTable.py +49 -0
  190. fontTools/ttLib/tables/E_B_D_T_.py +835 -0
  191. fontTools/ttLib/tables/E_B_L_C_.py +718 -0
  192. fontTools/ttLib/tables/F_F_T_M_.py +52 -0
  193. fontTools/ttLib/tables/F__e_a_t.py +149 -0
  194. fontTools/ttLib/tables/G_D_E_F_.py +13 -0
  195. fontTools/ttLib/tables/G_M_A_P_.py +148 -0
  196. fontTools/ttLib/tables/G_P_K_G_.py +133 -0
  197. fontTools/ttLib/tables/G_P_O_S_.py +14 -0
  198. fontTools/ttLib/tables/G_S_U_B_.py +13 -0
  199. fontTools/ttLib/tables/G_V_A_R_.py +5 -0
  200. fontTools/ttLib/tables/G__l_a_t.py +235 -0
  201. fontTools/ttLib/tables/G__l_o_c.py +85 -0
  202. fontTools/ttLib/tables/H_V_A_R_.py +13 -0
  203. fontTools/ttLib/tables/J_S_T_F_.py +13 -0
  204. fontTools/ttLib/tables/L_T_S_H_.py +58 -0
  205. fontTools/ttLib/tables/M_A_T_H_.py +13 -0
  206. fontTools/ttLib/tables/M_E_T_A_.py +352 -0
  207. fontTools/ttLib/tables/M_V_A_R_.py +13 -0
  208. fontTools/ttLib/tables/O_S_2f_2.py +752 -0
  209. fontTools/ttLib/tables/S_I_N_G_.py +99 -0
  210. fontTools/ttLib/tables/S_T_A_T_.py +15 -0
  211. fontTools/ttLib/tables/S_V_G_.py +223 -0
  212. fontTools/ttLib/tables/S__i_l_f.py +1040 -0
  213. fontTools/ttLib/tables/S__i_l_l.py +92 -0
  214. fontTools/ttLib/tables/T_S_I_B_.py +13 -0
  215. fontTools/ttLib/tables/T_S_I_C_.py +14 -0
  216. fontTools/ttLib/tables/T_S_I_D_.py +13 -0
  217. fontTools/ttLib/tables/T_S_I_J_.py +13 -0
  218. fontTools/ttLib/tables/T_S_I_P_.py +13 -0
  219. fontTools/ttLib/tables/T_S_I_S_.py +13 -0
  220. fontTools/ttLib/tables/T_S_I_V_.py +26 -0
  221. fontTools/ttLib/tables/T_S_I__0.py +70 -0
  222. fontTools/ttLib/tables/T_S_I__1.py +163 -0
  223. fontTools/ttLib/tables/T_S_I__2.py +17 -0
  224. fontTools/ttLib/tables/T_S_I__3.py +22 -0
  225. fontTools/ttLib/tables/T_S_I__5.py +60 -0
  226. fontTools/ttLib/tables/T_T_F_A_.py +14 -0
  227. fontTools/ttLib/tables/TupleVariation.py +884 -0
  228. fontTools/ttLib/tables/V_A_R_C_.py +12 -0
  229. fontTools/ttLib/tables/V_D_M_X_.py +249 -0
  230. fontTools/ttLib/tables/V_O_R_G_.py +165 -0
  231. fontTools/ttLib/tables/V_V_A_R_.py +13 -0
  232. fontTools/ttLib/tables/__init__.py +98 -0
  233. fontTools/ttLib/tables/_a_n_k_r.py +15 -0
  234. fontTools/ttLib/tables/_a_v_a_r.py +193 -0
  235. fontTools/ttLib/tables/_b_s_l_n.py +15 -0
  236. fontTools/ttLib/tables/_c_i_d_g.py +24 -0
  237. fontTools/ttLib/tables/_c_m_a_p.py +1591 -0
  238. fontTools/ttLib/tables/_c_v_a_r.py +94 -0
  239. fontTools/ttLib/tables/_c_v_t.py +56 -0
  240. fontTools/ttLib/tables/_f_e_a_t.py +15 -0
  241. fontTools/ttLib/tables/_f_p_g_m.py +62 -0
  242. fontTools/ttLib/tables/_f_v_a_r.py +261 -0
  243. fontTools/ttLib/tables/_g_a_s_p.py +63 -0
  244. fontTools/ttLib/tables/_g_c_i_d.py +13 -0
  245. fontTools/ttLib/tables/_g_l_y_f.py +2311 -0
  246. fontTools/ttLib/tables/_g_v_a_r.py +340 -0
  247. fontTools/ttLib/tables/_h_d_m_x.py +127 -0
  248. fontTools/ttLib/tables/_h_e_a_d.py +130 -0
  249. fontTools/ttLib/tables/_h_h_e_a.py +147 -0
  250. fontTools/ttLib/tables/_h_m_t_x.py +164 -0
  251. fontTools/ttLib/tables/_k_e_r_n.py +289 -0
  252. fontTools/ttLib/tables/_l_c_a_r.py +13 -0
  253. fontTools/ttLib/tables/_l_o_c_a.py +70 -0
  254. fontTools/ttLib/tables/_l_t_a_g.py +72 -0
  255. fontTools/ttLib/tables/_m_a_x_p.py +147 -0
  256. fontTools/ttLib/tables/_m_e_t_a.py +112 -0
  257. fontTools/ttLib/tables/_m_o_r_t.py +14 -0
  258. fontTools/ttLib/tables/_m_o_r_x.py +15 -0
  259. fontTools/ttLib/tables/_n_a_m_e.py +1242 -0
  260. fontTools/ttLib/tables/_o_p_b_d.py +14 -0
  261. fontTools/ttLib/tables/_p_o_s_t.py +319 -0
  262. fontTools/ttLib/tables/_p_r_e_p.py +16 -0
  263. fontTools/ttLib/tables/_p_r_o_p.py +12 -0
  264. fontTools/ttLib/tables/_s_b_i_x.py +129 -0
  265. fontTools/ttLib/tables/_t_r_a_k.py +332 -0
  266. fontTools/ttLib/tables/_v_h_e_a.py +139 -0
  267. fontTools/ttLib/tables/_v_m_t_x.py +19 -0
  268. fontTools/ttLib/tables/asciiTable.py +20 -0
  269. fontTools/ttLib/tables/grUtils.py +92 -0
  270. fontTools/ttLib/tables/otBase.py +1458 -0
  271. fontTools/ttLib/tables/otConverters.py +2068 -0
  272. fontTools/ttLib/tables/otData.py +6400 -0
  273. fontTools/ttLib/tables/otTables.py +2703 -0
  274. fontTools/ttLib/tables/otTraverse.py +163 -0
  275. fontTools/ttLib/tables/sbixGlyph.py +149 -0
  276. fontTools/ttLib/tables/sbixStrike.py +177 -0
  277. fontTools/ttLib/tables/table_API_readme.txt +91 -0
  278. fontTools/ttLib/tables/ttProgram.py +594 -0
  279. fontTools/ttLib/ttCollection.py +125 -0
  280. fontTools/ttLib/ttFont.py +1148 -0
  281. fontTools/ttLib/ttGlyphSet.py +490 -0
  282. fontTools/ttLib/ttVisitor.py +32 -0
  283. fontTools/ttLib/woff2.py +1680 -0
  284. fontTools/ttx.py +479 -0
  285. fontTools/ufoLib/__init__.py +2575 -0
  286. fontTools/ufoLib/converters.py +407 -0
  287. fontTools/ufoLib/errors.py +30 -0
  288. fontTools/ufoLib/etree.py +6 -0
  289. fontTools/ufoLib/filenames.py +356 -0
  290. fontTools/ufoLib/glifLib.py +2120 -0
  291. fontTools/ufoLib/kerning.py +141 -0
  292. fontTools/ufoLib/plistlib.py +47 -0
  293. fontTools/ufoLib/pointPen.py +6 -0
  294. fontTools/ufoLib/utils.py +107 -0
  295. fontTools/ufoLib/validators.py +1208 -0
  296. fontTools/unicode.py +50 -0
  297. fontTools/unicodedata/Blocks.py +817 -0
  298. fontTools/unicodedata/Mirrored.py +446 -0
  299. fontTools/unicodedata/OTTags.py +50 -0
  300. fontTools/unicodedata/ScriptExtensions.py +832 -0
  301. fontTools/unicodedata/Scripts.py +3639 -0
  302. fontTools/unicodedata/__init__.py +306 -0
  303. fontTools/varLib/__init__.py +1600 -0
  304. fontTools/varLib/__main__.py +6 -0
  305. fontTools/varLib/avar/__init__.py +0 -0
  306. fontTools/varLib/avar/__main__.py +72 -0
  307. fontTools/varLib/avar/build.py +79 -0
  308. fontTools/varLib/avar/map.py +108 -0
  309. fontTools/varLib/avar/plan.py +1004 -0
  310. fontTools/varLib/avar/unbuild.py +271 -0
  311. fontTools/varLib/avarPlanner.py +8 -0
  312. fontTools/varLib/builder.py +215 -0
  313. fontTools/varLib/cff.py +631 -0
  314. fontTools/varLib/errors.py +219 -0
  315. fontTools/varLib/featureVars.py +703 -0
  316. fontTools/varLib/hvar.py +113 -0
  317. fontTools/varLib/instancer/__init__.py +2052 -0
  318. fontTools/varLib/instancer/__main__.py +5 -0
  319. fontTools/varLib/instancer/featureVars.py +190 -0
  320. fontTools/varLib/instancer/names.py +388 -0
  321. fontTools/varLib/instancer/solver.py +309 -0
  322. fontTools/varLib/interpolatable.py +1209 -0
  323. fontTools/varLib/interpolatableHelpers.py +399 -0
  324. fontTools/varLib/interpolatablePlot.py +1269 -0
  325. fontTools/varLib/interpolatableTestContourOrder.py +82 -0
  326. fontTools/varLib/interpolatableTestStartingPoint.py +107 -0
  327. fontTools/varLib/interpolate_layout.py +124 -0
  328. fontTools/varLib/iup.c +19815 -0
  329. fontTools/varLib/iup.cp311-win32.pyd +0 -0
  330. fontTools/varLib/iup.py +490 -0
  331. fontTools/varLib/merger.py +1717 -0
  332. fontTools/varLib/models.py +642 -0
  333. fontTools/varLib/multiVarStore.py +253 -0
  334. fontTools/varLib/mutator.py +529 -0
  335. fontTools/varLib/mvar.py +40 -0
  336. fontTools/varLib/plot.py +238 -0
  337. fontTools/varLib/stat.py +149 -0
  338. fontTools/varLib/varStore.py +739 -0
  339. fontTools/voltLib/__init__.py +5 -0
  340. fontTools/voltLib/__main__.py +206 -0
  341. fontTools/voltLib/ast.py +452 -0
  342. fontTools/voltLib/error.py +12 -0
  343. fontTools/voltLib/lexer.py +99 -0
  344. fontTools/voltLib/parser.py +664 -0
  345. fontTools/voltLib/voltToFea.py +911 -0
  346. fonttools-4.60.2.data/data/share/man/man1/ttx.1 +225 -0
  347. fonttools-4.60.2.dist-info/METADATA +2250 -0
  348. fonttools-4.60.2.dist-info/RECORD +353 -0
  349. fonttools-4.60.2.dist-info/WHEEL +5 -0
  350. fonttools-4.60.2.dist-info/entry_points.txt +5 -0
  351. fonttools-4.60.2.dist-info/licenses/LICENSE +21 -0
  352. fonttools-4.60.2.dist-info/licenses/LICENSE.external +388 -0
  353. fonttools-4.60.2.dist-info/top_level.txt +1 -0
@@ -0,0 +1,1511 @@
1
+ """psCharStrings.py -- module implementing various kinds of CharStrings:
2
+ CFF dictionary data and Type1/Type2 CharStrings.
3
+ """
4
+
5
+ from fontTools.misc.fixedTools import (
6
+ fixedToFloat,
7
+ floatToFixed,
8
+ floatToFixedToStr,
9
+ strToFixedToFloat,
10
+ )
11
+ from fontTools.misc.textTools import bytechr, byteord, bytesjoin, strjoin
12
+ from fontTools.pens.boundsPen import BoundsPen
13
+ import struct
14
+ import logging
15
+
16
+
17
+ log = logging.getLogger(__name__)
18
+
19
+
20
+ def read_operator(self, b0, data, index):
21
+ if b0 == 12:
22
+ op = (b0, byteord(data[index]))
23
+ index = index + 1
24
+ else:
25
+ op = b0
26
+ try:
27
+ operator = self.operators[op]
28
+ except KeyError:
29
+ return None, index
30
+ value = self.handle_operator(operator)
31
+ return value, index
32
+
33
+
34
+ def read_byte(self, b0, data, index):
35
+ return b0 - 139, index
36
+
37
+
38
+ def read_smallInt1(self, b0, data, index):
39
+ b1 = byteord(data[index])
40
+ return (b0 - 247) * 256 + b1 + 108, index + 1
41
+
42
+
43
+ def read_smallInt2(self, b0, data, index):
44
+ b1 = byteord(data[index])
45
+ return -(b0 - 251) * 256 - b1 - 108, index + 1
46
+
47
+
48
+ def read_shortInt(self, b0, data, index):
49
+ (value,) = struct.unpack(">h", data[index : index + 2])
50
+ return value, index + 2
51
+
52
+
53
+ def read_longInt(self, b0, data, index):
54
+ (value,) = struct.unpack(">l", data[index : index + 4])
55
+ return value, index + 4
56
+
57
+
58
+ def read_fixed1616(self, b0, data, index):
59
+ (value,) = struct.unpack(">l", data[index : index + 4])
60
+ return fixedToFloat(value, precisionBits=16), index + 4
61
+
62
+
63
+ def read_reserved(self, b0, data, index):
64
+ assert NotImplementedError
65
+ return NotImplemented, index
66
+
67
+
68
+ def read_realNumber(self, b0, data, index):
69
+ number = ""
70
+ while True:
71
+ b = byteord(data[index])
72
+ index = index + 1
73
+ nibble0 = (b & 0xF0) >> 4
74
+ nibble1 = b & 0x0F
75
+ if nibble0 == 0xF:
76
+ break
77
+ number = number + realNibbles[nibble0]
78
+ if nibble1 == 0xF:
79
+ break
80
+ number = number + realNibbles[nibble1]
81
+ return float(number), index
82
+
83
+
84
+ t1OperandEncoding = [None] * 256
85
+ t1OperandEncoding[0:32] = (32) * [read_operator]
86
+ t1OperandEncoding[32:247] = (247 - 32) * [read_byte]
87
+ t1OperandEncoding[247:251] = (251 - 247) * [read_smallInt1]
88
+ t1OperandEncoding[251:255] = (255 - 251) * [read_smallInt2]
89
+ t1OperandEncoding[255] = read_longInt
90
+ assert len(t1OperandEncoding) == 256
91
+
92
+ t2OperandEncoding = t1OperandEncoding[:]
93
+ t2OperandEncoding[28] = read_shortInt
94
+ t2OperandEncoding[255] = read_fixed1616
95
+
96
+ cffDictOperandEncoding = t2OperandEncoding[:]
97
+ cffDictOperandEncoding[29] = read_longInt
98
+ cffDictOperandEncoding[30] = read_realNumber
99
+ cffDictOperandEncoding[255] = read_reserved
100
+
101
+
102
+ realNibbles = [
103
+ "0",
104
+ "1",
105
+ "2",
106
+ "3",
107
+ "4",
108
+ "5",
109
+ "6",
110
+ "7",
111
+ "8",
112
+ "9",
113
+ ".",
114
+ "E",
115
+ "E-",
116
+ None,
117
+ "-",
118
+ ]
119
+ realNibblesDict = {v: i for i, v in enumerate(realNibbles)}
120
+
121
+ maxOpStack = 193
122
+
123
+
124
+ def buildOperatorDict(operatorList):
125
+ oper = {}
126
+ opc = {}
127
+ for item in operatorList:
128
+ if len(item) == 2:
129
+ oper[item[0]] = item[1]
130
+ else:
131
+ oper[item[0]] = item[1:]
132
+ if isinstance(item[0], tuple):
133
+ opc[item[1]] = item[0]
134
+ else:
135
+ opc[item[1]] = (item[0],)
136
+ return oper, opc
137
+
138
+
139
+ t2Operators = [
140
+ # opcode name
141
+ (1, "hstem"),
142
+ (3, "vstem"),
143
+ (4, "vmoveto"),
144
+ (5, "rlineto"),
145
+ (6, "hlineto"),
146
+ (7, "vlineto"),
147
+ (8, "rrcurveto"),
148
+ (10, "callsubr"),
149
+ (11, "return"),
150
+ (14, "endchar"),
151
+ (15, "vsindex"),
152
+ (16, "blend"),
153
+ (18, "hstemhm"),
154
+ (19, "hintmask"),
155
+ (20, "cntrmask"),
156
+ (21, "rmoveto"),
157
+ (22, "hmoveto"),
158
+ (23, "vstemhm"),
159
+ (24, "rcurveline"),
160
+ (25, "rlinecurve"),
161
+ (26, "vvcurveto"),
162
+ (27, "hhcurveto"),
163
+ # (28, 'shortint'), # not really an operator
164
+ (29, "callgsubr"),
165
+ (30, "vhcurveto"),
166
+ (31, "hvcurveto"),
167
+ ((12, 0), "ignore"), # dotsection. Yes, there a few very early OTF/CFF
168
+ # fonts with this deprecated operator. Just ignore it.
169
+ ((12, 3), "and"),
170
+ ((12, 4), "or"),
171
+ ((12, 5), "not"),
172
+ ((12, 8), "store"),
173
+ ((12, 9), "abs"),
174
+ ((12, 10), "add"),
175
+ ((12, 11), "sub"),
176
+ ((12, 12), "div"),
177
+ ((12, 13), "load"),
178
+ ((12, 14), "neg"),
179
+ ((12, 15), "eq"),
180
+ ((12, 18), "drop"),
181
+ ((12, 20), "put"),
182
+ ((12, 21), "get"),
183
+ ((12, 22), "ifelse"),
184
+ ((12, 23), "random"),
185
+ ((12, 24), "mul"),
186
+ ((12, 26), "sqrt"),
187
+ ((12, 27), "dup"),
188
+ ((12, 28), "exch"),
189
+ ((12, 29), "index"),
190
+ ((12, 30), "roll"),
191
+ ((12, 34), "hflex"),
192
+ ((12, 35), "flex"),
193
+ ((12, 36), "hflex1"),
194
+ ((12, 37), "flex1"),
195
+ ]
196
+
197
+
198
+ def getIntEncoder(format):
199
+ if format == "cff":
200
+ twoByteOp = bytechr(28)
201
+ fourByteOp = bytechr(29)
202
+ elif format == "t1":
203
+ twoByteOp = None
204
+ fourByteOp = bytechr(255)
205
+ else:
206
+ assert format == "t2"
207
+ twoByteOp = bytechr(28)
208
+ fourByteOp = None
209
+
210
+ def encodeInt(
211
+ value,
212
+ fourByteOp=fourByteOp,
213
+ bytechr=bytechr,
214
+ pack=struct.pack,
215
+ unpack=struct.unpack,
216
+ twoByteOp=twoByteOp,
217
+ ):
218
+ if -107 <= value <= 107:
219
+ code = bytechr(value + 139)
220
+ elif 108 <= value <= 1131:
221
+ value = value - 108
222
+ code = bytechr((value >> 8) + 247) + bytechr(value & 0xFF)
223
+ elif -1131 <= value <= -108:
224
+ value = -value - 108
225
+ code = bytechr((value >> 8) + 251) + bytechr(value & 0xFF)
226
+ elif twoByteOp is not None and -32768 <= value <= 32767:
227
+ code = twoByteOp + pack(">h", value)
228
+ elif fourByteOp is None:
229
+ # Backwards compatible hack: due to a previous bug in FontTools,
230
+ # 16.16 fixed numbers were written out as 4-byte ints. When
231
+ # these numbers were small, they were wrongly written back as
232
+ # small ints instead of 4-byte ints, breaking round-tripping.
233
+ # This here workaround doesn't do it any better, since we can't
234
+ # distinguish anymore between small ints that were supposed to
235
+ # be small fixed numbers and small ints that were just small
236
+ # ints. Hence the warning.
237
+ log.warning(
238
+ "4-byte T2 number got passed to the "
239
+ "IntType handler. This should happen only when reading in "
240
+ "old XML files.\n"
241
+ )
242
+ code = bytechr(255) + pack(">l", value)
243
+ else:
244
+ code = fourByteOp + pack(">l", value)
245
+ return code
246
+
247
+ return encodeInt
248
+
249
+
250
+ encodeIntCFF = getIntEncoder("cff")
251
+ encodeIntT1 = getIntEncoder("t1")
252
+ encodeIntT2 = getIntEncoder("t2")
253
+
254
+
255
+ def encodeFixed(f, pack=struct.pack):
256
+ """For T2 only"""
257
+ value = floatToFixed(f, precisionBits=16)
258
+ if value & 0xFFFF == 0: # check if the fractional part is zero
259
+ return encodeIntT2(value >> 16) # encode only the integer part
260
+ else:
261
+ return b"\xff" + pack(">l", value) # encode the entire fixed point value
262
+
263
+
264
+ realZeroBytes = bytechr(30) + bytechr(0xF)
265
+
266
+
267
+ def encodeFloat(f):
268
+ # For CFF only, used in cffLib
269
+ if f == 0.0: # 0.0 == +0.0 == -0.0
270
+ return realZeroBytes
271
+ # Note: 14 decimal digits seems to be the limitation for CFF real numbers
272
+ # in macOS. However, we use 8 here to match the implementation of AFDKO.
273
+ s = "%.8G" % f
274
+ if s[:2] == "0.":
275
+ s = s[1:]
276
+ elif s[:3] == "-0.":
277
+ s = "-" + s[2:]
278
+ elif s.endswith("000"):
279
+ significantDigits = s.rstrip("0")
280
+ s = "%sE%d" % (significantDigits, len(s) - len(significantDigits))
281
+ else:
282
+ dotIndex = s.find(".")
283
+ eIndex = s.find("E")
284
+ if dotIndex != -1 and eIndex != -1:
285
+ integerPart = s[:dotIndex]
286
+ fractionalPart = s[dotIndex + 1 : eIndex]
287
+ exponent = int(s[eIndex + 1 :])
288
+ newExponent = exponent - len(fractionalPart)
289
+ if newExponent == 1:
290
+ s = "%s%s0" % (integerPart, fractionalPart)
291
+ else:
292
+ s = "%s%sE%d" % (integerPart, fractionalPart, newExponent)
293
+ if s.startswith((".0", "-.0")):
294
+ sign, s = s.split(".", 1)
295
+ s = "%s%sE-%d" % (sign, s.lstrip("0"), len(s))
296
+ nibbles = []
297
+ while s:
298
+ c = s[0]
299
+ s = s[1:]
300
+ if c == "E":
301
+ c2 = s[:1]
302
+ if c2 == "-":
303
+ s = s[1:]
304
+ c = "E-"
305
+ elif c2 == "+":
306
+ s = s[1:]
307
+ if s.startswith("0"):
308
+ s = s[1:]
309
+ nibbles.append(realNibblesDict[c])
310
+ nibbles.append(0xF)
311
+ if len(nibbles) % 2:
312
+ nibbles.append(0xF)
313
+ d = bytechr(30)
314
+ for i in range(0, len(nibbles), 2):
315
+ d = d + bytechr(nibbles[i] << 4 | nibbles[i + 1])
316
+ return d
317
+
318
+
319
+ class CharStringCompileError(Exception):
320
+ pass
321
+
322
+
323
+ class SimpleT2Decompiler(object):
324
+ def __init__(self, localSubrs, globalSubrs, private=None, blender=None):
325
+ self.localSubrs = localSubrs
326
+ self.localBias = calcSubrBias(localSubrs)
327
+ self.globalSubrs = globalSubrs
328
+ self.globalBias = calcSubrBias(globalSubrs)
329
+ self.private = private
330
+ self.blender = blender
331
+ self.reset()
332
+
333
+ def reset(self):
334
+ self.callingStack = []
335
+ self.operandStack = []
336
+ self.hintCount = 0
337
+ self.hintMaskBytes = 0
338
+ self.numRegions = 0
339
+ self.vsIndex = 0
340
+
341
+ def execute(self, charString, *, pushToStack=None):
342
+ self.callingStack.append(charString)
343
+ needsDecompilation = charString.needsDecompilation()
344
+ if needsDecompilation:
345
+ program = []
346
+ pushToProgram = program.append
347
+ else:
348
+ pushToProgram = lambda x: None
349
+ if pushToStack is None:
350
+ pushToStack = self.operandStack.append
351
+ index = 0
352
+ while True:
353
+ token, isOperator, index = charString.getToken(index)
354
+ if token is None:
355
+ break # we're done!
356
+ pushToProgram(token)
357
+ if isOperator:
358
+ handlerName = "op_" + token
359
+ handler = getattr(self, handlerName, None)
360
+ if handler is not None:
361
+ rv = handler(index)
362
+ if rv:
363
+ hintMaskBytes, index = rv
364
+ pushToProgram(hintMaskBytes)
365
+ else:
366
+ self.popall()
367
+ else:
368
+ pushToStack(token)
369
+ if needsDecompilation:
370
+ charString.setProgram(program)
371
+ del self.callingStack[-1]
372
+
373
+ def pop(self):
374
+ value = self.operandStack[-1]
375
+ del self.operandStack[-1]
376
+ return value
377
+
378
+ def popall(self):
379
+ stack = self.operandStack[:]
380
+ self.operandStack[:] = []
381
+ return stack
382
+
383
+ def push(self, value):
384
+ self.operandStack.append(value)
385
+
386
+ def op_return(self, index):
387
+ if self.operandStack:
388
+ pass
389
+
390
+ def op_endchar(self, index):
391
+ pass
392
+
393
+ def op_ignore(self, index):
394
+ pass
395
+
396
+ def op_callsubr(self, index):
397
+ subrIndex = self.pop()
398
+ subr = self.localSubrs[subrIndex + self.localBias]
399
+ self.execute(subr)
400
+
401
+ def op_callgsubr(self, index):
402
+ subrIndex = self.pop()
403
+ subr = self.globalSubrs[subrIndex + self.globalBias]
404
+ self.execute(subr)
405
+
406
+ def op_hstem(self, index):
407
+ self.countHints()
408
+
409
+ def op_vstem(self, index):
410
+ self.countHints()
411
+
412
+ def op_hstemhm(self, index):
413
+ self.countHints()
414
+
415
+ def op_vstemhm(self, index):
416
+ self.countHints()
417
+
418
+ def op_hintmask(self, index):
419
+ if not self.hintMaskBytes:
420
+ self.countHints()
421
+ self.hintMaskBytes = (self.hintCount + 7) // 8
422
+ hintMaskBytes, index = self.callingStack[-1].getBytes(index, self.hintMaskBytes)
423
+ return hintMaskBytes, index
424
+
425
+ op_cntrmask = op_hintmask
426
+
427
+ def countHints(self):
428
+ args = self.popall()
429
+ self.hintCount = self.hintCount + len(args) // 2
430
+
431
+ # misc
432
+ def op_and(self, index):
433
+ raise NotImplementedError
434
+
435
+ def op_or(self, index):
436
+ raise NotImplementedError
437
+
438
+ def op_not(self, index):
439
+ raise NotImplementedError
440
+
441
+ def op_store(self, index):
442
+ raise NotImplementedError
443
+
444
+ def op_abs(self, index):
445
+ raise NotImplementedError
446
+
447
+ def op_add(self, index):
448
+ raise NotImplementedError
449
+
450
+ def op_sub(self, index):
451
+ raise NotImplementedError
452
+
453
+ def op_div(self, index):
454
+ raise NotImplementedError
455
+
456
+ def op_load(self, index):
457
+ raise NotImplementedError
458
+
459
+ def op_neg(self, index):
460
+ raise NotImplementedError
461
+
462
+ def op_eq(self, index):
463
+ raise NotImplementedError
464
+
465
+ def op_drop(self, index):
466
+ raise NotImplementedError
467
+
468
+ def op_put(self, index):
469
+ raise NotImplementedError
470
+
471
+ def op_get(self, index):
472
+ raise NotImplementedError
473
+
474
+ def op_ifelse(self, index):
475
+ raise NotImplementedError
476
+
477
+ def op_random(self, index):
478
+ raise NotImplementedError
479
+
480
+ def op_mul(self, index):
481
+ raise NotImplementedError
482
+
483
+ def op_sqrt(self, index):
484
+ raise NotImplementedError
485
+
486
+ def op_dup(self, index):
487
+ raise NotImplementedError
488
+
489
+ def op_exch(self, index):
490
+ raise NotImplementedError
491
+
492
+ def op_index(self, index):
493
+ raise NotImplementedError
494
+
495
+ def op_roll(self, index):
496
+ raise NotImplementedError
497
+
498
+ def op_blend(self, index):
499
+ if self.numRegions == 0:
500
+ self.numRegions = self.private.getNumRegions()
501
+ numBlends = self.pop()
502
+ numOps = numBlends * (self.numRegions + 1)
503
+ if self.blender is None:
504
+ del self.operandStack[
505
+ -(numOps - numBlends) :
506
+ ] # Leave the default operands on the stack.
507
+ else:
508
+ argi = len(self.operandStack) - numOps
509
+ end_args = tuplei = argi + numBlends
510
+ while argi < end_args:
511
+ next_ti = tuplei + self.numRegions
512
+ deltas = self.operandStack[tuplei:next_ti]
513
+ delta = self.blender(self.vsIndex, deltas)
514
+ self.operandStack[argi] += delta
515
+ tuplei = next_ti
516
+ argi += 1
517
+ self.operandStack[end_args:] = []
518
+
519
+ def op_vsindex(self, index):
520
+ vi = self.pop()
521
+ self.vsIndex = vi
522
+ self.numRegions = self.private.getNumRegions(vi)
523
+
524
+
525
+ t1Operators = [
526
+ # opcode name
527
+ (1, "hstem"),
528
+ (3, "vstem"),
529
+ (4, "vmoveto"),
530
+ (5, "rlineto"),
531
+ (6, "hlineto"),
532
+ (7, "vlineto"),
533
+ (8, "rrcurveto"),
534
+ (9, "closepath"),
535
+ (10, "callsubr"),
536
+ (11, "return"),
537
+ (13, "hsbw"),
538
+ (14, "endchar"),
539
+ (21, "rmoveto"),
540
+ (22, "hmoveto"),
541
+ (30, "vhcurveto"),
542
+ (31, "hvcurveto"),
543
+ ((12, 0), "dotsection"),
544
+ ((12, 1), "vstem3"),
545
+ ((12, 2), "hstem3"),
546
+ ((12, 6), "seac"),
547
+ ((12, 7), "sbw"),
548
+ ((12, 12), "div"),
549
+ ((12, 16), "callothersubr"),
550
+ ((12, 17), "pop"),
551
+ ((12, 33), "setcurrentpoint"),
552
+ ]
553
+
554
+
555
+ class T2StackUseExtractor(SimpleT2Decompiler):
556
+
557
+ def execute(self, charString):
558
+ maxStackUse = 0
559
+
560
+ def pushToStack(value):
561
+ nonlocal maxStackUse
562
+ self.operandStack.append(value)
563
+ maxStackUse = max(maxStackUse, len(self.operandStack))
564
+
565
+ super().execute(charString, pushToStack=pushToStack)
566
+ return maxStackUse
567
+
568
+
569
+ class T2WidthExtractor(SimpleT2Decompiler):
570
+ def __init__(
571
+ self,
572
+ localSubrs,
573
+ globalSubrs,
574
+ nominalWidthX,
575
+ defaultWidthX,
576
+ private=None,
577
+ blender=None,
578
+ ):
579
+ SimpleT2Decompiler.__init__(self, localSubrs, globalSubrs, private, blender)
580
+ self.nominalWidthX = nominalWidthX
581
+ self.defaultWidthX = defaultWidthX
582
+
583
+ def reset(self):
584
+ SimpleT2Decompiler.reset(self)
585
+ self.gotWidth = 0
586
+ self.width = 0
587
+
588
+ def popallWidth(self, evenOdd=0):
589
+ args = self.popall()
590
+ if not self.gotWidth:
591
+ if evenOdd ^ (len(args) % 2):
592
+ # For CFF2 charstrings, this should never happen
593
+ assert (
594
+ self.defaultWidthX is not None
595
+ ), "CFF2 CharStrings must not have an initial width value"
596
+ self.width = self.nominalWidthX + args[0]
597
+ args = args[1:]
598
+ else:
599
+ self.width = self.defaultWidthX
600
+ self.gotWidth = 1
601
+ return args
602
+
603
+ def countHints(self):
604
+ args = self.popallWidth()
605
+ self.hintCount = self.hintCount + len(args) // 2
606
+
607
+ def op_rmoveto(self, index):
608
+ self.popallWidth()
609
+
610
+ def op_hmoveto(self, index):
611
+ self.popallWidth(1)
612
+
613
+ def op_vmoveto(self, index):
614
+ self.popallWidth(1)
615
+
616
+ def op_endchar(self, index):
617
+ self.popallWidth()
618
+
619
+
620
+ class T2OutlineExtractor(T2WidthExtractor):
621
+ def __init__(
622
+ self,
623
+ pen,
624
+ localSubrs,
625
+ globalSubrs,
626
+ nominalWidthX,
627
+ defaultWidthX,
628
+ private=None,
629
+ blender=None,
630
+ ):
631
+ T2WidthExtractor.__init__(
632
+ self,
633
+ localSubrs,
634
+ globalSubrs,
635
+ nominalWidthX,
636
+ defaultWidthX,
637
+ private,
638
+ blender,
639
+ )
640
+ self.pen = pen
641
+ self.subrLevel = 0
642
+
643
+ def reset(self):
644
+ T2WidthExtractor.reset(self)
645
+ self.currentPoint = (0, 0)
646
+ self.sawMoveTo = 0
647
+ self.subrLevel = 0
648
+
649
+ def execute(self, charString):
650
+ self.subrLevel += 1
651
+ super().execute(charString)
652
+ self.subrLevel -= 1
653
+ if self.subrLevel == 0:
654
+ self.endPath()
655
+
656
+ def _nextPoint(self, point):
657
+ x, y = self.currentPoint
658
+ point = x + point[0], y + point[1]
659
+ self.currentPoint = point
660
+ return point
661
+
662
+ def rMoveTo(self, point):
663
+ self.pen.moveTo(self._nextPoint(point))
664
+ self.sawMoveTo = 1
665
+
666
+ def rLineTo(self, point):
667
+ if not self.sawMoveTo:
668
+ self.rMoveTo((0, 0))
669
+ self.pen.lineTo(self._nextPoint(point))
670
+
671
+ def rCurveTo(self, pt1, pt2, pt3):
672
+ if not self.sawMoveTo:
673
+ self.rMoveTo((0, 0))
674
+ nextPoint = self._nextPoint
675
+ self.pen.curveTo(nextPoint(pt1), nextPoint(pt2), nextPoint(pt3))
676
+
677
+ def closePath(self):
678
+ if self.sawMoveTo:
679
+ self.pen.closePath()
680
+ self.sawMoveTo = 0
681
+
682
+ def endPath(self):
683
+ # In T2 there are no open paths, so always do a closePath when
684
+ # finishing a sub path. We avoid spurious calls to closePath()
685
+ # because its a real T1 op we're emulating in T2 whereas
686
+ # endPath() is just a means to that emulation
687
+ if self.sawMoveTo:
688
+ self.closePath()
689
+
690
+ #
691
+ # hint operators
692
+ #
693
+ # def op_hstem(self, index):
694
+ # self.countHints()
695
+ # def op_vstem(self, index):
696
+ # self.countHints()
697
+ # def op_hstemhm(self, index):
698
+ # self.countHints()
699
+ # def op_vstemhm(self, index):
700
+ # self.countHints()
701
+ # def op_hintmask(self, index):
702
+ # self.countHints()
703
+ # def op_cntrmask(self, index):
704
+ # self.countHints()
705
+
706
+ #
707
+ # path constructors, moveto
708
+ #
709
+ def op_rmoveto(self, index):
710
+ self.endPath()
711
+ self.rMoveTo(self.popallWidth())
712
+
713
+ def op_hmoveto(self, index):
714
+ self.endPath()
715
+ self.rMoveTo((self.popallWidth(1)[0], 0))
716
+
717
+ def op_vmoveto(self, index):
718
+ self.endPath()
719
+ self.rMoveTo((0, self.popallWidth(1)[0]))
720
+
721
+ def op_endchar(self, index):
722
+ self.endPath()
723
+ args = self.popallWidth()
724
+ if args:
725
+ from fontTools.encodings.StandardEncoding import StandardEncoding
726
+
727
+ # endchar can do seac accent bulding; The T2 spec says it's deprecated,
728
+ # but recent software that shall remain nameless does output it.
729
+ adx, ady, bchar, achar = args
730
+ baseGlyph = StandardEncoding[bchar]
731
+ self.pen.addComponent(baseGlyph, (1, 0, 0, 1, 0, 0))
732
+ accentGlyph = StandardEncoding[achar]
733
+ self.pen.addComponent(accentGlyph, (1, 0, 0, 1, adx, ady))
734
+
735
+ #
736
+ # path constructors, lines
737
+ #
738
+ def op_rlineto(self, index):
739
+ args = self.popall()
740
+ for i in range(0, len(args), 2):
741
+ point = args[i : i + 2]
742
+ self.rLineTo(point)
743
+
744
+ def op_hlineto(self, index):
745
+ self.alternatingLineto(1)
746
+
747
+ def op_vlineto(self, index):
748
+ self.alternatingLineto(0)
749
+
750
+ #
751
+ # path constructors, curves
752
+ #
753
+ def op_rrcurveto(self, index):
754
+ """{dxa dya dxb dyb dxc dyc}+ rrcurveto"""
755
+ args = self.popall()
756
+ for i in range(0, len(args), 6):
757
+ (
758
+ dxa,
759
+ dya,
760
+ dxb,
761
+ dyb,
762
+ dxc,
763
+ dyc,
764
+ ) = args[i : i + 6]
765
+ self.rCurveTo((dxa, dya), (dxb, dyb), (dxc, dyc))
766
+
767
+ def op_rcurveline(self, index):
768
+ """{dxa dya dxb dyb dxc dyc}+ dxd dyd rcurveline"""
769
+ args = self.popall()
770
+ for i in range(0, len(args) - 2, 6):
771
+ dxb, dyb, dxc, dyc, dxd, dyd = args[i : i + 6]
772
+ self.rCurveTo((dxb, dyb), (dxc, dyc), (dxd, dyd))
773
+ self.rLineTo(args[-2:])
774
+
775
+ def op_rlinecurve(self, index):
776
+ """{dxa dya}+ dxb dyb dxc dyc dxd dyd rlinecurve"""
777
+ args = self.popall()
778
+ lineArgs = args[:-6]
779
+ for i in range(0, len(lineArgs), 2):
780
+ self.rLineTo(lineArgs[i : i + 2])
781
+ dxb, dyb, dxc, dyc, dxd, dyd = args[-6:]
782
+ self.rCurveTo((dxb, dyb), (dxc, dyc), (dxd, dyd))
783
+
784
+ def op_vvcurveto(self, index):
785
+ "dx1? {dya dxb dyb dyc}+ vvcurveto"
786
+ args = self.popall()
787
+ if len(args) % 2:
788
+ dx1 = args[0]
789
+ args = args[1:]
790
+ else:
791
+ dx1 = 0
792
+ for i in range(0, len(args), 4):
793
+ dya, dxb, dyb, dyc = args[i : i + 4]
794
+ self.rCurveTo((dx1, dya), (dxb, dyb), (0, dyc))
795
+ dx1 = 0
796
+
797
+ def op_hhcurveto(self, index):
798
+ """dy1? {dxa dxb dyb dxc}+ hhcurveto"""
799
+ args = self.popall()
800
+ if len(args) % 2:
801
+ dy1 = args[0]
802
+ args = args[1:]
803
+ else:
804
+ dy1 = 0
805
+ for i in range(0, len(args), 4):
806
+ dxa, dxb, dyb, dxc = args[i : i + 4]
807
+ self.rCurveTo((dxa, dy1), (dxb, dyb), (dxc, 0))
808
+ dy1 = 0
809
+
810
+ def op_vhcurveto(self, index):
811
+ """dy1 dx2 dy2 dx3 {dxa dxb dyb dyc dyd dxe dye dxf}* dyf? vhcurveto (30)
812
+ {dya dxb dyb dxc dxd dxe dye dyf}+ dxf? vhcurveto
813
+ """
814
+ args = self.popall()
815
+ while args:
816
+ args = self.vcurveto(args)
817
+ if args:
818
+ args = self.hcurveto(args)
819
+
820
+ def op_hvcurveto(self, index):
821
+ """dx1 dx2 dy2 dy3 {dya dxb dyb dxc dxd dxe dye dyf}* dxf?
822
+ {dxa dxb dyb dyc dyd dxe dye dxf}+ dyf?
823
+ """
824
+ args = self.popall()
825
+ while args:
826
+ args = self.hcurveto(args)
827
+ if args:
828
+ args = self.vcurveto(args)
829
+
830
+ #
831
+ # path constructors, flex
832
+ #
833
+ def op_hflex(self, index):
834
+ dx1, dx2, dy2, dx3, dx4, dx5, dx6 = self.popall()
835
+ dy1 = dy3 = dy4 = dy6 = 0
836
+ dy5 = -dy2
837
+ self.rCurveTo((dx1, dy1), (dx2, dy2), (dx3, dy3))
838
+ self.rCurveTo((dx4, dy4), (dx5, dy5), (dx6, dy6))
839
+
840
+ def op_flex(self, index):
841
+ dx1, dy1, dx2, dy2, dx3, dy3, dx4, dy4, dx5, dy5, dx6, dy6, fd = self.popall()
842
+ self.rCurveTo((dx1, dy1), (dx2, dy2), (dx3, dy3))
843
+ self.rCurveTo((dx4, dy4), (dx5, dy5), (dx6, dy6))
844
+
845
+ def op_hflex1(self, index):
846
+ dx1, dy1, dx2, dy2, dx3, dx4, dx5, dy5, dx6 = self.popall()
847
+ dy3 = dy4 = 0
848
+ dy6 = -(dy1 + dy2 + dy3 + dy4 + dy5)
849
+
850
+ self.rCurveTo((dx1, dy1), (dx2, dy2), (dx3, dy3))
851
+ self.rCurveTo((dx4, dy4), (dx5, dy5), (dx6, dy6))
852
+
853
+ def op_flex1(self, index):
854
+ dx1, dy1, dx2, dy2, dx3, dy3, dx4, dy4, dx5, dy5, d6 = self.popall()
855
+ dx = dx1 + dx2 + dx3 + dx4 + dx5
856
+ dy = dy1 + dy2 + dy3 + dy4 + dy5
857
+ if abs(dx) > abs(dy):
858
+ dx6 = d6
859
+ dy6 = -dy
860
+ else:
861
+ dx6 = -dx
862
+ dy6 = d6
863
+ self.rCurveTo((dx1, dy1), (dx2, dy2), (dx3, dy3))
864
+ self.rCurveTo((dx4, dy4), (dx5, dy5), (dx6, dy6))
865
+
866
+ # misc
867
+ def op_and(self, index):
868
+ raise NotImplementedError
869
+
870
+ def op_or(self, index):
871
+ raise NotImplementedError
872
+
873
+ def op_not(self, index):
874
+ raise NotImplementedError
875
+
876
+ def op_store(self, index):
877
+ raise NotImplementedError
878
+
879
+ def op_abs(self, index):
880
+ raise NotImplementedError
881
+
882
+ def op_add(self, index):
883
+ raise NotImplementedError
884
+
885
+ def op_sub(self, index):
886
+ raise NotImplementedError
887
+
888
+ def op_div(self, index):
889
+ num2 = self.pop()
890
+ num1 = self.pop()
891
+ d1 = num1 // num2
892
+ d2 = num1 / num2
893
+ if d1 == d2:
894
+ self.push(d1)
895
+ else:
896
+ self.push(d2)
897
+
898
+ def op_load(self, index):
899
+ raise NotImplementedError
900
+
901
+ def op_neg(self, index):
902
+ raise NotImplementedError
903
+
904
+ def op_eq(self, index):
905
+ raise NotImplementedError
906
+
907
+ def op_drop(self, index):
908
+ raise NotImplementedError
909
+
910
+ def op_put(self, index):
911
+ raise NotImplementedError
912
+
913
+ def op_get(self, index):
914
+ raise NotImplementedError
915
+
916
+ def op_ifelse(self, index):
917
+ raise NotImplementedError
918
+
919
+ def op_random(self, index):
920
+ raise NotImplementedError
921
+
922
+ def op_mul(self, index):
923
+ raise NotImplementedError
924
+
925
+ def op_sqrt(self, index):
926
+ raise NotImplementedError
927
+
928
+ def op_dup(self, index):
929
+ raise NotImplementedError
930
+
931
+ def op_exch(self, index):
932
+ raise NotImplementedError
933
+
934
+ def op_index(self, index):
935
+ raise NotImplementedError
936
+
937
+ def op_roll(self, index):
938
+ raise NotImplementedError
939
+
940
+ #
941
+ # miscellaneous helpers
942
+ #
943
+ def alternatingLineto(self, isHorizontal):
944
+ args = self.popall()
945
+ for arg in args:
946
+ if isHorizontal:
947
+ point = (arg, 0)
948
+ else:
949
+ point = (0, arg)
950
+ self.rLineTo(point)
951
+ isHorizontal = not isHorizontal
952
+
953
+ def vcurveto(self, args):
954
+ dya, dxb, dyb, dxc = args[:4]
955
+ args = args[4:]
956
+ if len(args) == 1:
957
+ dyc = args[0]
958
+ args = []
959
+ else:
960
+ dyc = 0
961
+ self.rCurveTo((0, dya), (dxb, dyb), (dxc, dyc))
962
+ return args
963
+
964
+ def hcurveto(self, args):
965
+ dxa, dxb, dyb, dyc = args[:4]
966
+ args = args[4:]
967
+ if len(args) == 1:
968
+ dxc = args[0]
969
+ args = []
970
+ else:
971
+ dxc = 0
972
+ self.rCurveTo((dxa, 0), (dxb, dyb), (dxc, dyc))
973
+ return args
974
+
975
+
976
+ class T1OutlineExtractor(T2OutlineExtractor):
977
+ def __init__(self, pen, subrs):
978
+ self.pen = pen
979
+ self.subrs = subrs
980
+ self.reset()
981
+
982
+ def reset(self):
983
+ self.flexing = 0
984
+ self.width = 0
985
+ self.sbx = 0
986
+ T2OutlineExtractor.reset(self)
987
+
988
+ def endPath(self):
989
+ if self.sawMoveTo:
990
+ self.pen.endPath()
991
+ self.sawMoveTo = 0
992
+
993
+ def popallWidth(self, evenOdd=0):
994
+ return self.popall()
995
+
996
+ def exch(self):
997
+ stack = self.operandStack
998
+ stack[-1], stack[-2] = stack[-2], stack[-1]
999
+
1000
+ #
1001
+ # path constructors
1002
+ #
1003
+ def op_rmoveto(self, index):
1004
+ if self.flexing:
1005
+ return
1006
+ self.endPath()
1007
+ self.rMoveTo(self.popall())
1008
+
1009
+ def op_hmoveto(self, index):
1010
+ if self.flexing:
1011
+ # We must add a parameter to the stack if we are flexing
1012
+ self.push(0)
1013
+ return
1014
+ self.endPath()
1015
+ self.rMoveTo((self.popall()[0], 0))
1016
+
1017
+ def op_vmoveto(self, index):
1018
+ if self.flexing:
1019
+ # We must add a parameter to the stack if we are flexing
1020
+ self.push(0)
1021
+ self.exch()
1022
+ return
1023
+ self.endPath()
1024
+ self.rMoveTo((0, self.popall()[0]))
1025
+
1026
+ def op_closepath(self, index):
1027
+ self.closePath()
1028
+
1029
+ def op_setcurrentpoint(self, index):
1030
+ args = self.popall()
1031
+ x, y = args
1032
+ self.currentPoint = x, y
1033
+
1034
+ def op_endchar(self, index):
1035
+ self.endPath()
1036
+
1037
+ def op_hsbw(self, index):
1038
+ sbx, wx = self.popall()
1039
+ self.width = wx
1040
+ self.sbx = sbx
1041
+ self.currentPoint = sbx, self.currentPoint[1]
1042
+
1043
+ def op_sbw(self, index):
1044
+ self.popall() # XXX
1045
+
1046
+ #
1047
+ def op_callsubr(self, index):
1048
+ subrIndex = self.pop()
1049
+ subr = self.subrs[subrIndex]
1050
+ self.execute(subr)
1051
+
1052
+ def op_callothersubr(self, index):
1053
+ subrIndex = self.pop()
1054
+ nArgs = self.pop()
1055
+ # print nArgs, subrIndex, "callothersubr"
1056
+ if subrIndex == 0 and nArgs == 3:
1057
+ self.doFlex()
1058
+ self.flexing = 0
1059
+ elif subrIndex == 1 and nArgs == 0:
1060
+ self.flexing = 1
1061
+ # ignore...
1062
+
1063
+ def op_pop(self, index):
1064
+ pass # ignore...
1065
+
1066
+ def doFlex(self):
1067
+ finaly = self.pop()
1068
+ finalx = self.pop()
1069
+ self.pop() # flex height is unused
1070
+
1071
+ p3y = self.pop()
1072
+ p3x = self.pop()
1073
+ bcp4y = self.pop()
1074
+ bcp4x = self.pop()
1075
+ bcp3y = self.pop()
1076
+ bcp3x = self.pop()
1077
+ p2y = self.pop()
1078
+ p2x = self.pop()
1079
+ bcp2y = self.pop()
1080
+ bcp2x = self.pop()
1081
+ bcp1y = self.pop()
1082
+ bcp1x = self.pop()
1083
+ rpy = self.pop()
1084
+ rpx = self.pop()
1085
+
1086
+ # call rrcurveto
1087
+ self.push(bcp1x + rpx)
1088
+ self.push(bcp1y + rpy)
1089
+ self.push(bcp2x)
1090
+ self.push(bcp2y)
1091
+ self.push(p2x)
1092
+ self.push(p2y)
1093
+ self.op_rrcurveto(None)
1094
+
1095
+ # call rrcurveto
1096
+ self.push(bcp3x)
1097
+ self.push(bcp3y)
1098
+ self.push(bcp4x)
1099
+ self.push(bcp4y)
1100
+ self.push(p3x)
1101
+ self.push(p3y)
1102
+ self.op_rrcurveto(None)
1103
+
1104
+ # Push back final coords so subr 0 can find them
1105
+ self.push(finalx)
1106
+ self.push(finaly)
1107
+
1108
+ def op_dotsection(self, index):
1109
+ self.popall() # XXX
1110
+
1111
+ def op_hstem3(self, index):
1112
+ self.popall() # XXX
1113
+
1114
+ def op_seac(self, index):
1115
+ "asb adx ady bchar achar seac"
1116
+ from fontTools.encodings.StandardEncoding import StandardEncoding
1117
+
1118
+ asb, adx, ady, bchar, achar = self.popall()
1119
+ baseGlyph = StandardEncoding[bchar]
1120
+ self.pen.addComponent(baseGlyph, (1, 0, 0, 1, 0, 0))
1121
+ accentGlyph = StandardEncoding[achar]
1122
+ adx = adx + self.sbx - asb # seac weirdness
1123
+ self.pen.addComponent(accentGlyph, (1, 0, 0, 1, adx, ady))
1124
+
1125
+ def op_vstem3(self, index):
1126
+ self.popall() # XXX
1127
+
1128
+
1129
+ class T2CharString(object):
1130
+ operandEncoding = t2OperandEncoding
1131
+ operators, opcodes = buildOperatorDict(t2Operators)
1132
+ decompilerClass = SimpleT2Decompiler
1133
+ outlineExtractor = T2OutlineExtractor
1134
+
1135
+ def __init__(self, bytecode=None, program=None, private=None, globalSubrs=None):
1136
+ if program is None:
1137
+ program = []
1138
+ self.bytecode = bytecode
1139
+ self.program = program
1140
+ self.private = private
1141
+ self.globalSubrs = globalSubrs if globalSubrs is not None else []
1142
+ self._cur_vsindex = None
1143
+
1144
+ def getNumRegions(self, vsindex=None):
1145
+ pd = self.private
1146
+ assert pd is not None
1147
+ if vsindex is not None:
1148
+ self._cur_vsindex = vsindex
1149
+ elif self._cur_vsindex is None:
1150
+ self._cur_vsindex = pd.vsindex if hasattr(pd, "vsindex") else 0
1151
+ return pd.getNumRegions(self._cur_vsindex)
1152
+
1153
+ def __repr__(self):
1154
+ if self.bytecode is None:
1155
+ return "<%s (source) at %x>" % (self.__class__.__name__, id(self))
1156
+ else:
1157
+ return "<%s (bytecode) at %x>" % (self.__class__.__name__, id(self))
1158
+
1159
+ def getIntEncoder(self):
1160
+ return encodeIntT2
1161
+
1162
+ def getFixedEncoder(self):
1163
+ return encodeFixed
1164
+
1165
+ def decompile(self):
1166
+ if not self.needsDecompilation():
1167
+ return
1168
+ subrs = getattr(self.private, "Subrs", [])
1169
+ decompiler = self.decompilerClass(subrs, self.globalSubrs, self.private)
1170
+ decompiler.execute(self)
1171
+
1172
+ def draw(self, pen, blender=None):
1173
+ subrs = getattr(self.private, "Subrs", [])
1174
+ extractor = self.outlineExtractor(
1175
+ pen,
1176
+ subrs,
1177
+ self.globalSubrs,
1178
+ self.private.nominalWidthX,
1179
+ self.private.defaultWidthX,
1180
+ self.private,
1181
+ blender,
1182
+ )
1183
+ extractor.execute(self)
1184
+ self.width = extractor.width
1185
+
1186
+ def calcBounds(self, glyphSet):
1187
+ boundsPen = BoundsPen(glyphSet)
1188
+ self.draw(boundsPen)
1189
+ return boundsPen.bounds
1190
+
1191
+ def compile(self, isCFF2=False):
1192
+ if self.bytecode is not None:
1193
+ return
1194
+ opcodes = self.opcodes
1195
+ program = self.program
1196
+
1197
+ if isCFF2:
1198
+ # If present, remove return and endchar operators.
1199
+ if program and program[-1] in ("return", "endchar"):
1200
+ program = program[:-1]
1201
+ elif program and not isinstance(program[-1], str):
1202
+ raise CharStringCompileError(
1203
+ "T2CharString or Subr has items on the stack after last operator."
1204
+ )
1205
+
1206
+ bytecode = []
1207
+ encodeInt = self.getIntEncoder()
1208
+ encodeFixed = self.getFixedEncoder()
1209
+ i = 0
1210
+ end = len(program)
1211
+ while i < end:
1212
+ token = program[i]
1213
+ i = i + 1
1214
+ if isinstance(token, str):
1215
+ try:
1216
+ bytecode.extend(bytechr(b) for b in opcodes[token])
1217
+ except KeyError:
1218
+ raise CharStringCompileError("illegal operator: %s" % token)
1219
+ if token in ("hintmask", "cntrmask"):
1220
+ bytecode.append(program[i]) # hint mask
1221
+ i = i + 1
1222
+ elif isinstance(token, int):
1223
+ bytecode.append(encodeInt(token))
1224
+ elif isinstance(token, float):
1225
+ bytecode.append(encodeFixed(token))
1226
+ else:
1227
+ assert 0, "unsupported type: %s" % type(token)
1228
+ try:
1229
+ bytecode = bytesjoin(bytecode)
1230
+ except TypeError:
1231
+ log.error(bytecode)
1232
+ raise
1233
+ self.setBytecode(bytecode)
1234
+
1235
+ def needsDecompilation(self):
1236
+ return self.bytecode is not None
1237
+
1238
+ def setProgram(self, program):
1239
+ self.program = program
1240
+ self.bytecode = None
1241
+
1242
+ def setBytecode(self, bytecode):
1243
+ self.bytecode = bytecode
1244
+ self.program = None
1245
+
1246
+ def getToken(self, index, len=len, byteord=byteord, isinstance=isinstance):
1247
+ if self.bytecode is not None:
1248
+ if index >= len(self.bytecode):
1249
+ return None, 0, 0
1250
+ b0 = byteord(self.bytecode[index])
1251
+ index = index + 1
1252
+ handler = self.operandEncoding[b0]
1253
+ token, index = handler(self, b0, self.bytecode, index)
1254
+ else:
1255
+ if index >= len(self.program):
1256
+ return None, 0, 0
1257
+ token = self.program[index]
1258
+ index = index + 1
1259
+ isOperator = isinstance(token, str)
1260
+ return token, isOperator, index
1261
+
1262
+ def getBytes(self, index, nBytes):
1263
+ if self.bytecode is not None:
1264
+ newIndex = index + nBytes
1265
+ bytes = self.bytecode[index:newIndex]
1266
+ index = newIndex
1267
+ else:
1268
+ bytes = self.program[index]
1269
+ index = index + 1
1270
+ assert len(bytes) == nBytes
1271
+ return bytes, index
1272
+
1273
+ def handle_operator(self, operator):
1274
+ return operator
1275
+
1276
+ def toXML(self, xmlWriter, ttFont=None):
1277
+ from fontTools.misc.textTools import num2binary
1278
+
1279
+ if self.bytecode is not None:
1280
+ xmlWriter.dumphex(self.bytecode)
1281
+ else:
1282
+ index = 0
1283
+ args = []
1284
+ while True:
1285
+ token, isOperator, index = self.getToken(index)
1286
+ if token is None:
1287
+ break
1288
+ if isOperator:
1289
+ if token in ("hintmask", "cntrmask"):
1290
+ hintMask, isOperator, index = self.getToken(index)
1291
+ bits = []
1292
+ for byte in hintMask:
1293
+ bits.append(num2binary(byteord(byte), 8))
1294
+ hintMask = strjoin(bits)
1295
+ line = " ".join(args + [token, hintMask])
1296
+ else:
1297
+ line = " ".join(args + [token])
1298
+ xmlWriter.write(line)
1299
+ xmlWriter.newline()
1300
+ args = []
1301
+ else:
1302
+ if isinstance(token, float):
1303
+ token = floatToFixedToStr(token, precisionBits=16)
1304
+ else:
1305
+ token = str(token)
1306
+ args.append(token)
1307
+ if args:
1308
+ # NOTE: only CFF2 charstrings/subrs can have numeric arguments on
1309
+ # the stack after the last operator. Compiling this would fail if
1310
+ # this is part of CFF 1.0 table.
1311
+ line = " ".join(args)
1312
+ xmlWriter.write(line)
1313
+
1314
+ def fromXML(self, name, attrs, content):
1315
+ from fontTools.misc.textTools import binary2num, readHex
1316
+
1317
+ if attrs.get("raw"):
1318
+ self.setBytecode(readHex(content))
1319
+ return
1320
+ content = strjoin(content)
1321
+ content = content.split()
1322
+ program = []
1323
+ end = len(content)
1324
+ i = 0
1325
+ while i < end:
1326
+ token = content[i]
1327
+ i = i + 1
1328
+ try:
1329
+ token = int(token)
1330
+ except ValueError:
1331
+ try:
1332
+ token = strToFixedToFloat(token, precisionBits=16)
1333
+ except ValueError:
1334
+ program.append(token)
1335
+ if token in ("hintmask", "cntrmask"):
1336
+ mask = content[i]
1337
+ maskBytes = b""
1338
+ for j in range(0, len(mask), 8):
1339
+ maskBytes = maskBytes + bytechr(binary2num(mask[j : j + 8]))
1340
+ program.append(maskBytes)
1341
+ i = i + 1
1342
+ else:
1343
+ program.append(token)
1344
+ else:
1345
+ program.append(token)
1346
+ self.setProgram(program)
1347
+
1348
+
1349
+ class T1CharString(T2CharString):
1350
+ operandEncoding = t1OperandEncoding
1351
+ operators, opcodes = buildOperatorDict(t1Operators)
1352
+
1353
+ def __init__(self, bytecode=None, program=None, subrs=None):
1354
+ super().__init__(bytecode, program)
1355
+ self.subrs = subrs
1356
+
1357
+ def getIntEncoder(self):
1358
+ return encodeIntT1
1359
+
1360
+ def getFixedEncoder(self):
1361
+ def encodeFixed(value):
1362
+ raise TypeError("Type 1 charstrings don't support floating point operands")
1363
+
1364
+ def decompile(self):
1365
+ if self.bytecode is None:
1366
+ return
1367
+ program = []
1368
+ index = 0
1369
+ while True:
1370
+ token, isOperator, index = self.getToken(index)
1371
+ if token is None:
1372
+ break
1373
+ program.append(token)
1374
+ self.setProgram(program)
1375
+
1376
+ def draw(self, pen):
1377
+ extractor = T1OutlineExtractor(pen, self.subrs)
1378
+ extractor.execute(self)
1379
+ self.width = extractor.width
1380
+
1381
+
1382
+ class DictDecompiler(object):
1383
+ operandEncoding = cffDictOperandEncoding
1384
+
1385
+ def __init__(self, strings, parent=None):
1386
+ self.stack = []
1387
+ self.strings = strings
1388
+ self.dict = {}
1389
+ self.parent = parent
1390
+
1391
+ def getDict(self):
1392
+ assert len(self.stack) == 0, "non-empty stack"
1393
+ return self.dict
1394
+
1395
+ def decompile(self, data):
1396
+ index = 0
1397
+ lenData = len(data)
1398
+ push = self.stack.append
1399
+ while index < lenData:
1400
+ b0 = byteord(data[index])
1401
+ index = index + 1
1402
+ handler = self.operandEncoding[b0]
1403
+ value, index = handler(self, b0, data, index)
1404
+ if value is not None:
1405
+ push(value)
1406
+
1407
+ def pop(self):
1408
+ value = self.stack[-1]
1409
+ del self.stack[-1]
1410
+ return value
1411
+
1412
+ def popall(self):
1413
+ args = self.stack[:]
1414
+ del self.stack[:]
1415
+ return args
1416
+
1417
+ def handle_operator(self, operator):
1418
+ operator, argType = operator
1419
+ if isinstance(argType, tuple):
1420
+ value = ()
1421
+ for i in range(len(argType) - 1, -1, -1):
1422
+ arg = argType[i]
1423
+ arghandler = getattr(self, "arg_" + arg)
1424
+ value = (arghandler(operator),) + value
1425
+ else:
1426
+ arghandler = getattr(self, "arg_" + argType)
1427
+ value = arghandler(operator)
1428
+ if operator == "blend":
1429
+ self.stack.extend(value)
1430
+ else:
1431
+ self.dict[operator] = value
1432
+
1433
+ def arg_number(self, name):
1434
+ if isinstance(self.stack[0], list):
1435
+ out = self.arg_blend_number(self.stack)
1436
+ else:
1437
+ out = self.pop()
1438
+ return out
1439
+
1440
+ def arg_blend_number(self, name):
1441
+ out = []
1442
+ blendArgs = self.pop()
1443
+ numMasters = len(blendArgs)
1444
+ out.append(blendArgs)
1445
+ out.append("blend")
1446
+ dummy = self.popall()
1447
+ return blendArgs
1448
+
1449
+ def arg_SID(self, name):
1450
+ return self.strings[self.pop()]
1451
+
1452
+ def arg_array(self, name):
1453
+ return self.popall()
1454
+
1455
+ def arg_blendList(self, name):
1456
+ """
1457
+ There may be non-blend args at the top of the stack. We first calculate
1458
+ where the blend args start in the stack. These are the last
1459
+ numMasters*numBlends) +1 args.
1460
+ The blend args starts with numMasters relative coordinate values, the BlueValues in the list from the default master font. This is followed by
1461
+ numBlends list of values. Each of value in one of these lists is the
1462
+ Variable Font delta for the matching region.
1463
+
1464
+ We re-arrange this to be a list of numMaster entries. Each entry starts with the corresponding default font relative value, and is followed by
1465
+ the delta values. We then convert the default values, the first item in each entry, to an absolute value.
1466
+ """
1467
+ vsindex = self.dict.get("vsindex", 0)
1468
+ numMasters = (
1469
+ self.parent.getNumRegions(vsindex) + 1
1470
+ ) # only a PrivateDict has blended ops.
1471
+ numBlends = self.pop()
1472
+ args = self.popall()
1473
+ numArgs = len(args)
1474
+ # The spec says that there should be no non-blended Blue Values,.
1475
+ assert numArgs == numMasters * numBlends
1476
+ value = [None] * numBlends
1477
+ numDeltas = numMasters - 1
1478
+ i = 0
1479
+ prevVal = 0
1480
+ while i < numBlends:
1481
+ newVal = args[i] + prevVal
1482
+ prevVal = newVal
1483
+ masterOffset = numBlends + (i * numDeltas)
1484
+ blendList = [newVal] + args[masterOffset : masterOffset + numDeltas]
1485
+ value[i] = blendList
1486
+ i += 1
1487
+ return value
1488
+
1489
+ def arg_delta(self, name):
1490
+ valueList = self.popall()
1491
+ out = []
1492
+ if valueList and isinstance(valueList[0], list):
1493
+ # arg_blendList() has already converted these to absolute values.
1494
+ out = valueList
1495
+ else:
1496
+ current = 0
1497
+ for v in valueList:
1498
+ current = current + v
1499
+ out.append(current)
1500
+ return out
1501
+
1502
+
1503
+ def calcSubrBias(subrs):
1504
+ nSubrs = len(subrs)
1505
+ if nSubrs < 1240:
1506
+ bias = 107
1507
+ elif nSubrs < 33900:
1508
+ bias = 1131
1509
+ else:
1510
+ bias = 32768
1511
+ return bias