BarcodePDF417.cs 66 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602
  1. // Generates the 2D barcode PDF417. Supports dimensioning auto-sizing, fixed
  2. // and variable sizes, automatic and manual error levels, raw codeword input,
  3. // codeword size optimization and bitmap inversion.
  4. // Author: Paulo Soares (psoares@consiste.pt)
  5. // Modifications by Alexander Tzyganenko
  6. //
  7. using System;
  8. using System.Collections;
  9. using System.Text;
  10. using System.Drawing;
  11. using System.Drawing.Drawing2D;
  12. using System.ComponentModel;
  13. using FastReport.Utils;
  14. namespace FastReport.Barcode
  15. {
  16. /// <summary>
  17. /// Specifies the error correction level used for PDF417 barcode.
  18. /// </summary>
  19. public enum PDF417ErrorCorrection
  20. {
  21. /// <summary>
  22. /// Indicates that correction level should be calculated automatically.
  23. /// </summary>
  24. Auto,
  25. /// <summary>
  26. /// Specifies level 0.
  27. /// </summary>
  28. Level0,
  29. /// <summary>
  30. /// Specifies level 1.
  31. /// </summary>
  32. Level1,
  33. /// <summary>
  34. /// Specifies level 2.
  35. /// </summary>
  36. Level2,
  37. /// <summary>
  38. /// Specifies level 3.
  39. /// </summary>
  40. Level3,
  41. /// <summary>
  42. /// Specifies level 4.
  43. /// </summary>
  44. Level4,
  45. /// <summary>
  46. /// Specifies level 5.
  47. /// </summary>
  48. Level5,
  49. /// <summary>
  50. /// Specifies level 6.
  51. /// </summary>
  52. Level6,
  53. /// <summary>
  54. /// Specifies level 7.
  55. /// </summary>
  56. Level7,
  57. /// <summary>
  58. /// Specifies level 8.
  59. /// </summary>
  60. Level8
  61. }
  62. /// <summary>
  63. /// Specifies the compaction mode used for PDF417 barcode.
  64. /// </summary>
  65. public enum PDF417CompactionMode
  66. {
  67. /// <summary>
  68. /// Indicates that compaction mode should be calculated automatically.
  69. /// </summary>
  70. Auto,
  71. /// <summary>
  72. /// Specifies the text compaction mode.
  73. /// </summary>
  74. Text,
  75. /// <summary>
  76. /// Specifies the numeric compaction mode.
  77. /// </summary>
  78. Numeric,
  79. /// <summary>
  80. /// Specifies the binary compaction mode.
  81. /// </summary>
  82. Binary
  83. }
  84. /// <summary>
  85. /// Generates the 2D PDF417 barcode.
  86. /// </summary>
  87. /// <example>This example shows how to configure the BarcodeObject to display PDF417 barcode.
  88. /// <code>
  89. /// BarcodeObject barcode;
  90. /// ...
  91. /// barcode.Barcode = new BarcodePDF417();
  92. /// (barcode.Barcode as BarcodePDF417).CompactionMode = PDF417CompactionMode.Text;
  93. /// </code>
  94. /// </example>
  95. public class BarcodePDF417 : Barcode2DBase
  96. {
  97. #region Fields
  98. private byte[] outBits;
  99. private int bitColumns;
  100. private int codeRows;
  101. private int codeColumns;
  102. private int rows;
  103. private int columns;
  104. private int[] codewords = new int[MAX_DATA_CODEWORDS + 2];
  105. private int lenCodewords;
  106. private int errorLevel;
  107. private PDF417ErrorCorrection errorCorrection;
  108. private PDF417CompactionMode compactionMode;
  109. private byte[] bytes;
  110. private float aspectRatio;
  111. private Size pixelSize;
  112. private int codePage;
  113. private int bitPtr;
  114. private int cwPtr;
  115. private SegmentList segmentList;
  116. private const int START_PATTERN = 0x1fea8;
  117. private const int STOP_PATTERN = 0x3fa29;
  118. private const int START_CODE_SIZE = 17;
  119. private const int STOP_SIZE = 18;
  120. private const int MOD = 929;
  121. private const int ALPHA = 0x10000;
  122. private const int LOWER = 0x20000;
  123. private const int MIXED = 0x40000;
  124. private const int PUNCTUATION = 0x80000;
  125. private const int ISBYTE = 0x100000;
  126. private const int BYTESHIFT = 913;
  127. private const int PL = 25;
  128. private const int LL = 27;
  129. private const int AS = 27;
  130. private const int ML = 28;
  131. private const int AL = 28;
  132. private const int PS = 29;
  133. private const int PAL = 29;
  134. private const int SPACE = 26;
  135. private const int TEXT_MODE = 900;
  136. private const int BYTE_MODE_6 = 924;
  137. private const int BYTE_MODE = 901;
  138. private const int NUMERIC_MODE = 902;
  139. private const int ABSOLUTE_MAX_TEXT_SIZE = 5420;
  140. private const int MAX_DATA_CODEWORDS = 926;
  141. private const int MACRO_SEGMENT_ID = 928;
  142. private const int MACRO_LAST_SEGMENT = 922;
  143. private const string MIXED_SET = "0123456789&\r\t,:#-.$/+%*=^";
  144. private const string PUNCTUATION_SET = ";<>@[\\]_`~!\r\t,:\n-.$/\"|*()?{}'";
  145. private static readonly int[][] CLUSTERS = new int[][]
  146. {new int[]{
  147. 0x1d5c0, 0x1eaf0, 0x1f57c, 0x1d4e0, 0x1ea78, 0x1f53e, 0x1a8c0, 0x1d470,
  148. 0x1a860, 0x15040, 0x1a830, 0x15020, 0x1adc0, 0x1d6f0, 0x1eb7c, 0x1ace0,
  149. 0x1d678, 0x1eb3e, 0x158c0, 0x1ac70, 0x15860, 0x15dc0, 0x1aef0, 0x1d77c,
  150. 0x15ce0, 0x1ae78, 0x1d73e, 0x15c70, 0x1ae3c, 0x15ef0, 0x1af7c, 0x15e78,
  151. 0x1af3e, 0x15f7c, 0x1f5fa, 0x1d2e0, 0x1e978, 0x1f4be, 0x1a4c0, 0x1d270,
  152. 0x1e93c, 0x1a460, 0x1d238, 0x14840, 0x1a430, 0x1d21c, 0x14820, 0x1a418,
  153. 0x14810, 0x1a6e0, 0x1d378, 0x1e9be, 0x14cc0, 0x1a670, 0x1d33c, 0x14c60,
  154. 0x1a638, 0x1d31e, 0x14c30, 0x1a61c, 0x14ee0, 0x1a778, 0x1d3be, 0x14e70,
  155. 0x1a73c, 0x14e38, 0x1a71e, 0x14f78, 0x1a7be, 0x14f3c, 0x14f1e, 0x1a2c0,
  156. 0x1d170, 0x1e8bc, 0x1a260, 0x1d138, 0x1e89e, 0x14440, 0x1a230, 0x1d11c,
  157. 0x14420, 0x1a218, 0x14410, 0x14408, 0x146c0, 0x1a370, 0x1d1bc, 0x14660,
  158. 0x1a338, 0x1d19e, 0x14630, 0x1a31c, 0x14618, 0x1460c, 0x14770, 0x1a3bc,
  159. 0x14738, 0x1a39e, 0x1471c, 0x147bc, 0x1a160, 0x1d0b8, 0x1e85e, 0x14240,
  160. 0x1a130, 0x1d09c, 0x14220, 0x1a118, 0x1d08e, 0x14210, 0x1a10c, 0x14208,
  161. 0x1a106, 0x14360, 0x1a1b8, 0x1d0de, 0x14330, 0x1a19c, 0x14318, 0x1a18e,
  162. 0x1430c, 0x14306, 0x1a1de, 0x1438e, 0x14140, 0x1a0b0, 0x1d05c, 0x14120,
  163. 0x1a098, 0x1d04e, 0x14110, 0x1a08c, 0x14108, 0x1a086, 0x14104, 0x141b0,
  164. 0x14198, 0x1418c, 0x140a0, 0x1d02e, 0x1a04c, 0x1a046, 0x14082, 0x1cae0,
  165. 0x1e578, 0x1f2be, 0x194c0, 0x1ca70, 0x1e53c, 0x19460, 0x1ca38, 0x1e51e,
  166. 0x12840, 0x19430, 0x12820, 0x196e0, 0x1cb78, 0x1e5be, 0x12cc0, 0x19670,
  167. 0x1cb3c, 0x12c60, 0x19638, 0x12c30, 0x12c18, 0x12ee0, 0x19778, 0x1cbbe,
  168. 0x12e70, 0x1973c, 0x12e38, 0x12e1c, 0x12f78, 0x197be, 0x12f3c, 0x12fbe,
  169. 0x1dac0, 0x1ed70, 0x1f6bc, 0x1da60, 0x1ed38, 0x1f69e, 0x1b440, 0x1da30,
  170. 0x1ed1c, 0x1b420, 0x1da18, 0x1ed0e, 0x1b410, 0x1da0c, 0x192c0, 0x1c970,
  171. 0x1e4bc, 0x1b6c0, 0x19260, 0x1c938, 0x1e49e, 0x1b660, 0x1db38, 0x1ed9e,
  172. 0x16c40, 0x12420, 0x19218, 0x1c90e, 0x16c20, 0x1b618, 0x16c10, 0x126c0,
  173. 0x19370, 0x1c9bc, 0x16ec0, 0x12660, 0x19338, 0x1c99e, 0x16e60, 0x1b738,
  174. 0x1db9e, 0x16e30, 0x12618, 0x16e18, 0x12770, 0x193bc, 0x16f70, 0x12738,
  175. 0x1939e, 0x16f38, 0x1b79e, 0x16f1c, 0x127bc, 0x16fbc, 0x1279e, 0x16f9e,
  176. 0x1d960, 0x1ecb8, 0x1f65e, 0x1b240, 0x1d930, 0x1ec9c, 0x1b220, 0x1d918,
  177. 0x1ec8e, 0x1b210, 0x1d90c, 0x1b208, 0x1b204, 0x19160, 0x1c8b8, 0x1e45e,
  178. 0x1b360, 0x19130, 0x1c89c, 0x16640, 0x12220, 0x1d99c, 0x1c88e, 0x16620,
  179. 0x12210, 0x1910c, 0x16610, 0x1b30c, 0x19106, 0x12204, 0x12360, 0x191b8,
  180. 0x1c8de, 0x16760, 0x12330, 0x1919c, 0x16730, 0x1b39c, 0x1918e, 0x16718,
  181. 0x1230c, 0x12306, 0x123b8, 0x191de, 0x167b8, 0x1239c, 0x1679c, 0x1238e,
  182. 0x1678e, 0x167de, 0x1b140, 0x1d8b0, 0x1ec5c, 0x1b120, 0x1d898, 0x1ec4e,
  183. 0x1b110, 0x1d88c, 0x1b108, 0x1d886, 0x1b104, 0x1b102, 0x12140, 0x190b0,
  184. 0x1c85c, 0x16340, 0x12120, 0x19098, 0x1c84e, 0x16320, 0x1b198, 0x1d8ce,
  185. 0x16310, 0x12108, 0x19086, 0x16308, 0x1b186, 0x16304, 0x121b0, 0x190dc,
  186. 0x163b0, 0x12198, 0x190ce, 0x16398, 0x1b1ce, 0x1638c, 0x12186, 0x16386,
  187. 0x163dc, 0x163ce, 0x1b0a0, 0x1d858, 0x1ec2e, 0x1b090, 0x1d84c, 0x1b088,
  188. 0x1d846, 0x1b084, 0x1b082, 0x120a0, 0x19058, 0x1c82e, 0x161a0, 0x12090,
  189. 0x1904c, 0x16190, 0x1b0cc, 0x19046, 0x16188, 0x12084, 0x16184, 0x12082,
  190. 0x120d8, 0x161d8, 0x161cc, 0x161c6, 0x1d82c, 0x1d826, 0x1b042, 0x1902c,
  191. 0x12048, 0x160c8, 0x160c4, 0x160c2, 0x18ac0, 0x1c570, 0x1e2bc, 0x18a60,
  192. 0x1c538, 0x11440, 0x18a30, 0x1c51c, 0x11420, 0x18a18, 0x11410, 0x11408,
  193. 0x116c0, 0x18b70, 0x1c5bc, 0x11660, 0x18b38, 0x1c59e, 0x11630, 0x18b1c,
  194. 0x11618, 0x1160c, 0x11770, 0x18bbc, 0x11738, 0x18b9e, 0x1171c, 0x117bc,
  195. 0x1179e, 0x1cd60, 0x1e6b8, 0x1f35e, 0x19a40, 0x1cd30, 0x1e69c, 0x19a20,
  196. 0x1cd18, 0x1e68e, 0x19a10, 0x1cd0c, 0x19a08, 0x1cd06, 0x18960, 0x1c4b8,
  197. 0x1e25e, 0x19b60, 0x18930, 0x1c49c, 0x13640, 0x11220, 0x1cd9c, 0x1c48e,
  198. 0x13620, 0x19b18, 0x1890c, 0x13610, 0x11208, 0x13608, 0x11360, 0x189b8,
  199. 0x1c4de, 0x13760, 0x11330, 0x1cdde, 0x13730, 0x19b9c, 0x1898e, 0x13718,
  200. 0x1130c, 0x1370c, 0x113b8, 0x189de, 0x137b8, 0x1139c, 0x1379c, 0x1138e,
  201. 0x113de, 0x137de, 0x1dd40, 0x1eeb0, 0x1f75c, 0x1dd20, 0x1ee98, 0x1f74e,
  202. 0x1dd10, 0x1ee8c, 0x1dd08, 0x1ee86, 0x1dd04, 0x19940, 0x1ccb0, 0x1e65c,
  203. 0x1bb40, 0x19920, 0x1eedc, 0x1e64e, 0x1bb20, 0x1dd98, 0x1eece, 0x1bb10,
  204. 0x19908, 0x1cc86, 0x1bb08, 0x1dd86, 0x19902, 0x11140, 0x188b0, 0x1c45c,
  205. 0x13340, 0x11120, 0x18898, 0x1c44e, 0x17740, 0x13320, 0x19998, 0x1ccce,
  206. 0x17720, 0x1bb98, 0x1ddce, 0x18886, 0x17710, 0x13308, 0x19986, 0x17708,
  207. 0x11102, 0x111b0, 0x188dc, 0x133b0, 0x11198, 0x188ce, 0x177b0, 0x13398,
  208. 0x199ce, 0x17798, 0x1bbce, 0x11186, 0x13386, 0x111dc, 0x133dc, 0x111ce,
  209. 0x177dc, 0x133ce, 0x1dca0, 0x1ee58, 0x1f72e, 0x1dc90, 0x1ee4c, 0x1dc88,
  210. 0x1ee46, 0x1dc84, 0x1dc82, 0x198a0, 0x1cc58, 0x1e62e, 0x1b9a0, 0x19890,
  211. 0x1ee6e, 0x1b990, 0x1dccc, 0x1cc46, 0x1b988, 0x19884, 0x1b984, 0x19882,
  212. 0x1b982, 0x110a0, 0x18858, 0x1c42e, 0x131a0, 0x11090, 0x1884c, 0x173a0,
  213. 0x13190, 0x198cc, 0x18846, 0x17390, 0x1b9cc, 0x11084, 0x17388, 0x13184,
  214. 0x11082, 0x13182, 0x110d8, 0x1886e, 0x131d8, 0x110cc, 0x173d8, 0x131cc,
  215. 0x110c6, 0x173cc, 0x131c6, 0x110ee, 0x173ee, 0x1dc50, 0x1ee2c, 0x1dc48,
  216. 0x1ee26, 0x1dc44, 0x1dc42, 0x19850, 0x1cc2c, 0x1b8d0, 0x19848, 0x1cc26,
  217. 0x1b8c8, 0x1dc66, 0x1b8c4, 0x19842, 0x1b8c2, 0x11050, 0x1882c, 0x130d0,
  218. 0x11048, 0x18826, 0x171d0, 0x130c8, 0x19866, 0x171c8, 0x1b8e6, 0x11042,
  219. 0x171c4, 0x130c2, 0x171c2, 0x130ec, 0x171ec, 0x171e6, 0x1ee16, 0x1dc22,
  220. 0x1cc16, 0x19824, 0x19822, 0x11028, 0x13068, 0x170e8, 0x11022, 0x13062,
  221. 0x18560, 0x10a40, 0x18530, 0x10a20, 0x18518, 0x1c28e, 0x10a10, 0x1850c,
  222. 0x10a08, 0x18506, 0x10b60, 0x185b8, 0x1c2de, 0x10b30, 0x1859c, 0x10b18,
  223. 0x1858e, 0x10b0c, 0x10b06, 0x10bb8, 0x185de, 0x10b9c, 0x10b8e, 0x10bde,
  224. 0x18d40, 0x1c6b0, 0x1e35c, 0x18d20, 0x1c698, 0x18d10, 0x1c68c, 0x18d08,
  225. 0x1c686, 0x18d04, 0x10940, 0x184b0, 0x1c25c, 0x11b40, 0x10920, 0x1c6dc,
  226. 0x1c24e, 0x11b20, 0x18d98, 0x1c6ce, 0x11b10, 0x10908, 0x18486, 0x11b08,
  227. 0x18d86, 0x10902, 0x109b0, 0x184dc, 0x11bb0, 0x10998, 0x184ce, 0x11b98,
  228. 0x18dce, 0x11b8c, 0x10986, 0x109dc, 0x11bdc, 0x109ce, 0x11bce, 0x1cea0,
  229. 0x1e758, 0x1f3ae, 0x1ce90, 0x1e74c, 0x1ce88, 0x1e746, 0x1ce84, 0x1ce82,
  230. 0x18ca0, 0x1c658, 0x19da0, 0x18c90, 0x1c64c, 0x19d90, 0x1cecc, 0x1c646,
  231. 0x19d88, 0x18c84, 0x19d84, 0x18c82, 0x19d82, 0x108a0, 0x18458, 0x119a0,
  232. 0x10890, 0x1c66e, 0x13ba0, 0x11990, 0x18ccc, 0x18446, 0x13b90, 0x19dcc,
  233. 0x10884, 0x13b88, 0x11984, 0x10882, 0x11982, 0x108d8, 0x1846e, 0x119d8,
  234. 0x108cc, 0x13bd8, 0x119cc, 0x108c6, 0x13bcc, 0x119c6, 0x108ee, 0x119ee,
  235. 0x13bee, 0x1ef50, 0x1f7ac, 0x1ef48, 0x1f7a6, 0x1ef44, 0x1ef42, 0x1ce50,
  236. 0x1e72c, 0x1ded0, 0x1ef6c, 0x1e726, 0x1dec8, 0x1ef66, 0x1dec4, 0x1ce42,
  237. 0x1dec2, 0x18c50, 0x1c62c, 0x19cd0, 0x18c48, 0x1c626, 0x1bdd0, 0x19cc8,
  238. 0x1ce66, 0x1bdc8, 0x1dee6, 0x18c42, 0x1bdc4, 0x19cc2, 0x1bdc2, 0x10850,
  239. 0x1842c, 0x118d0, 0x10848, 0x18426, 0x139d0, 0x118c8, 0x18c66, 0x17bd0,
  240. 0x139c8, 0x19ce6, 0x10842, 0x17bc8, 0x1bde6, 0x118c2, 0x17bc4, 0x1086c,
  241. 0x118ec, 0x10866, 0x139ec, 0x118e6, 0x17bec, 0x139e6, 0x17be6, 0x1ef28,
  242. 0x1f796, 0x1ef24, 0x1ef22, 0x1ce28, 0x1e716, 0x1de68, 0x1ef36, 0x1de64,
  243. 0x1ce22, 0x1de62, 0x18c28, 0x1c616, 0x19c68, 0x18c24, 0x1bce8, 0x19c64,
  244. 0x18c22, 0x1bce4, 0x19c62, 0x1bce2, 0x10828, 0x18416, 0x11868, 0x18c36,
  245. 0x138e8, 0x11864, 0x10822, 0x179e8, 0x138e4, 0x11862, 0x179e4, 0x138e2,
  246. 0x179e2, 0x11876, 0x179f6, 0x1ef12, 0x1de34, 0x1de32, 0x19c34, 0x1bc74,
  247. 0x1bc72, 0x11834, 0x13874, 0x178f4, 0x178f2, 0x10540, 0x10520, 0x18298,
  248. 0x10510, 0x10508, 0x10504, 0x105b0, 0x10598, 0x1058c, 0x10586, 0x105dc,
  249. 0x105ce, 0x186a0, 0x18690, 0x1c34c, 0x18688, 0x1c346, 0x18684, 0x18682,
  250. 0x104a0, 0x18258, 0x10da0, 0x186d8, 0x1824c, 0x10d90, 0x186cc, 0x10d88,
  251. 0x186c6, 0x10d84, 0x10482, 0x10d82, 0x104d8, 0x1826e, 0x10dd8, 0x186ee,
  252. 0x10dcc, 0x104c6, 0x10dc6, 0x104ee, 0x10dee, 0x1c750, 0x1c748, 0x1c744,
  253. 0x1c742, 0x18650, 0x18ed0, 0x1c76c, 0x1c326, 0x18ec8, 0x1c766, 0x18ec4,
  254. 0x18642, 0x18ec2, 0x10450, 0x10cd0, 0x10448, 0x18226, 0x11dd0, 0x10cc8,
  255. 0x10444, 0x11dc8, 0x10cc4, 0x10442, 0x11dc4, 0x10cc2, 0x1046c, 0x10cec,
  256. 0x10466, 0x11dec, 0x10ce6, 0x11de6, 0x1e7a8, 0x1e7a4, 0x1e7a2, 0x1c728,
  257. 0x1cf68, 0x1e7b6, 0x1cf64, 0x1c722, 0x1cf62, 0x18628, 0x1c316, 0x18e68,
  258. 0x1c736, 0x19ee8, 0x18e64, 0x18622, 0x19ee4, 0x18e62, 0x19ee2, 0x10428,
  259. 0x18216, 0x10c68, 0x18636, 0x11ce8, 0x10c64, 0x10422, 0x13de8, 0x11ce4,
  260. 0x10c62, 0x13de4, 0x11ce2, 0x10436, 0x10c76, 0x11cf6, 0x13df6, 0x1f7d4,
  261. 0x1f7d2, 0x1e794, 0x1efb4, 0x1e792, 0x1efb2, 0x1c714, 0x1cf34, 0x1c712,
  262. 0x1df74, 0x1cf32, 0x1df72, 0x18614, 0x18e34, 0x18612, 0x19e74, 0x18e32,
  263. 0x1bef4
  264. }, new int[]{
  265. 0x1f560, 0x1fab8, 0x1ea40, 0x1f530, 0x1fa9c, 0x1ea20, 0x1f518, 0x1fa8e,
  266. 0x1ea10, 0x1f50c, 0x1ea08, 0x1f506, 0x1ea04, 0x1eb60, 0x1f5b8, 0x1fade,
  267. 0x1d640, 0x1eb30, 0x1f59c, 0x1d620, 0x1eb18, 0x1f58e, 0x1d610, 0x1eb0c,
  268. 0x1d608, 0x1eb06, 0x1d604, 0x1d760, 0x1ebb8, 0x1f5de, 0x1ae40, 0x1d730,
  269. 0x1eb9c, 0x1ae20, 0x1d718, 0x1eb8e, 0x1ae10, 0x1d70c, 0x1ae08, 0x1d706,
  270. 0x1ae04, 0x1af60, 0x1d7b8, 0x1ebde, 0x15e40, 0x1af30, 0x1d79c, 0x15e20,
  271. 0x1af18, 0x1d78e, 0x15e10, 0x1af0c, 0x15e08, 0x1af06, 0x15f60, 0x1afb8,
  272. 0x1d7de, 0x15f30, 0x1af9c, 0x15f18, 0x1af8e, 0x15f0c, 0x15fb8, 0x1afde,
  273. 0x15f9c, 0x15f8e, 0x1e940, 0x1f4b0, 0x1fa5c, 0x1e920, 0x1f498, 0x1fa4e,
  274. 0x1e910, 0x1f48c, 0x1e908, 0x1f486, 0x1e904, 0x1e902, 0x1d340, 0x1e9b0,
  275. 0x1f4dc, 0x1d320, 0x1e998, 0x1f4ce, 0x1d310, 0x1e98c, 0x1d308, 0x1e986,
  276. 0x1d304, 0x1d302, 0x1a740, 0x1d3b0, 0x1e9dc, 0x1a720, 0x1d398, 0x1e9ce,
  277. 0x1a710, 0x1d38c, 0x1a708, 0x1d386, 0x1a704, 0x1a702, 0x14f40, 0x1a7b0,
  278. 0x1d3dc, 0x14f20, 0x1a798, 0x1d3ce, 0x14f10, 0x1a78c, 0x14f08, 0x1a786,
  279. 0x14f04, 0x14fb0, 0x1a7dc, 0x14f98, 0x1a7ce, 0x14f8c, 0x14f86, 0x14fdc,
  280. 0x14fce, 0x1e8a0, 0x1f458, 0x1fa2e, 0x1e890, 0x1f44c, 0x1e888, 0x1f446,
  281. 0x1e884, 0x1e882, 0x1d1a0, 0x1e8d8, 0x1f46e, 0x1d190, 0x1e8cc, 0x1d188,
  282. 0x1e8c6, 0x1d184, 0x1d182, 0x1a3a0, 0x1d1d8, 0x1e8ee, 0x1a390, 0x1d1cc,
  283. 0x1a388, 0x1d1c6, 0x1a384, 0x1a382, 0x147a0, 0x1a3d8, 0x1d1ee, 0x14790,
  284. 0x1a3cc, 0x14788, 0x1a3c6, 0x14784, 0x14782, 0x147d8, 0x1a3ee, 0x147cc,
  285. 0x147c6, 0x147ee, 0x1e850, 0x1f42c, 0x1e848, 0x1f426, 0x1e844, 0x1e842,
  286. 0x1d0d0, 0x1e86c, 0x1d0c8, 0x1e866, 0x1d0c4, 0x1d0c2, 0x1a1d0, 0x1d0ec,
  287. 0x1a1c8, 0x1d0e6, 0x1a1c4, 0x1a1c2, 0x143d0, 0x1a1ec, 0x143c8, 0x1a1e6,
  288. 0x143c4, 0x143c2, 0x143ec, 0x143e6, 0x1e828, 0x1f416, 0x1e824, 0x1e822,
  289. 0x1d068, 0x1e836, 0x1d064, 0x1d062, 0x1a0e8, 0x1d076, 0x1a0e4, 0x1a0e2,
  290. 0x141e8, 0x1a0f6, 0x141e4, 0x141e2, 0x1e814, 0x1e812, 0x1d034, 0x1d032,
  291. 0x1a074, 0x1a072, 0x1e540, 0x1f2b0, 0x1f95c, 0x1e520, 0x1f298, 0x1f94e,
  292. 0x1e510, 0x1f28c, 0x1e508, 0x1f286, 0x1e504, 0x1e502, 0x1cb40, 0x1e5b0,
  293. 0x1f2dc, 0x1cb20, 0x1e598, 0x1f2ce, 0x1cb10, 0x1e58c, 0x1cb08, 0x1e586,
  294. 0x1cb04, 0x1cb02, 0x19740, 0x1cbb0, 0x1e5dc, 0x19720, 0x1cb98, 0x1e5ce,
  295. 0x19710, 0x1cb8c, 0x19708, 0x1cb86, 0x19704, 0x19702, 0x12f40, 0x197b0,
  296. 0x1cbdc, 0x12f20, 0x19798, 0x1cbce, 0x12f10, 0x1978c, 0x12f08, 0x19786,
  297. 0x12f04, 0x12fb0, 0x197dc, 0x12f98, 0x197ce, 0x12f8c, 0x12f86, 0x12fdc,
  298. 0x12fce, 0x1f6a0, 0x1fb58, 0x16bf0, 0x1f690, 0x1fb4c, 0x169f8, 0x1f688,
  299. 0x1fb46, 0x168fc, 0x1f684, 0x1f682, 0x1e4a0, 0x1f258, 0x1f92e, 0x1eda0,
  300. 0x1e490, 0x1fb6e, 0x1ed90, 0x1f6cc, 0x1f246, 0x1ed88, 0x1e484, 0x1ed84,
  301. 0x1e482, 0x1ed82, 0x1c9a0, 0x1e4d8, 0x1f26e, 0x1dba0, 0x1c990, 0x1e4cc,
  302. 0x1db90, 0x1edcc, 0x1e4c6, 0x1db88, 0x1c984, 0x1db84, 0x1c982, 0x1db82,
  303. 0x193a0, 0x1c9d8, 0x1e4ee, 0x1b7a0, 0x19390, 0x1c9cc, 0x1b790, 0x1dbcc,
  304. 0x1c9c6, 0x1b788, 0x19384, 0x1b784, 0x19382, 0x1b782, 0x127a0, 0x193d8,
  305. 0x1c9ee, 0x16fa0, 0x12790, 0x193cc, 0x16f90, 0x1b7cc, 0x193c6, 0x16f88,
  306. 0x12784, 0x16f84, 0x12782, 0x127d8, 0x193ee, 0x16fd8, 0x127cc, 0x16fcc,
  307. 0x127c6, 0x16fc6, 0x127ee, 0x1f650, 0x1fb2c, 0x165f8, 0x1f648, 0x1fb26,
  308. 0x164fc, 0x1f644, 0x1647e, 0x1f642, 0x1e450, 0x1f22c, 0x1ecd0, 0x1e448,
  309. 0x1f226, 0x1ecc8, 0x1f666, 0x1ecc4, 0x1e442, 0x1ecc2, 0x1c8d0, 0x1e46c,
  310. 0x1d9d0, 0x1c8c8, 0x1e466, 0x1d9c8, 0x1ece6, 0x1d9c4, 0x1c8c2, 0x1d9c2,
  311. 0x191d0, 0x1c8ec, 0x1b3d0, 0x191c8, 0x1c8e6, 0x1b3c8, 0x1d9e6, 0x1b3c4,
  312. 0x191c2, 0x1b3c2, 0x123d0, 0x191ec, 0x167d0, 0x123c8, 0x191e6, 0x167c8,
  313. 0x1b3e6, 0x167c4, 0x123c2, 0x167c2, 0x123ec, 0x167ec, 0x123e6, 0x167e6,
  314. 0x1f628, 0x1fb16, 0x162fc, 0x1f624, 0x1627e, 0x1f622, 0x1e428, 0x1f216,
  315. 0x1ec68, 0x1f636, 0x1ec64, 0x1e422, 0x1ec62, 0x1c868, 0x1e436, 0x1d8e8,
  316. 0x1c864, 0x1d8e4, 0x1c862, 0x1d8e2, 0x190e8, 0x1c876, 0x1b1e8, 0x1d8f6,
  317. 0x1b1e4, 0x190e2, 0x1b1e2, 0x121e8, 0x190f6, 0x163e8, 0x121e4, 0x163e4,
  318. 0x121e2, 0x163e2, 0x121f6, 0x163f6, 0x1f614, 0x1617e, 0x1f612, 0x1e414,
  319. 0x1ec34, 0x1e412, 0x1ec32, 0x1c834, 0x1d874, 0x1c832, 0x1d872, 0x19074,
  320. 0x1b0f4, 0x19072, 0x1b0f2, 0x120f4, 0x161f4, 0x120f2, 0x161f2, 0x1f60a,
  321. 0x1e40a, 0x1ec1a, 0x1c81a, 0x1d83a, 0x1903a, 0x1b07a, 0x1e2a0, 0x1f158,
  322. 0x1f8ae, 0x1e290, 0x1f14c, 0x1e288, 0x1f146, 0x1e284, 0x1e282, 0x1c5a0,
  323. 0x1e2d8, 0x1f16e, 0x1c590, 0x1e2cc, 0x1c588, 0x1e2c6, 0x1c584, 0x1c582,
  324. 0x18ba0, 0x1c5d8, 0x1e2ee, 0x18b90, 0x1c5cc, 0x18b88, 0x1c5c6, 0x18b84,
  325. 0x18b82, 0x117a0, 0x18bd8, 0x1c5ee, 0x11790, 0x18bcc, 0x11788, 0x18bc6,
  326. 0x11784, 0x11782, 0x117d8, 0x18bee, 0x117cc, 0x117c6, 0x117ee, 0x1f350,
  327. 0x1f9ac, 0x135f8, 0x1f348, 0x1f9a6, 0x134fc, 0x1f344, 0x1347e, 0x1f342,
  328. 0x1e250, 0x1f12c, 0x1e6d0, 0x1e248, 0x1f126, 0x1e6c8, 0x1f366, 0x1e6c4,
  329. 0x1e242, 0x1e6c2, 0x1c4d0, 0x1e26c, 0x1cdd0, 0x1c4c8, 0x1e266, 0x1cdc8,
  330. 0x1e6e6, 0x1cdc4, 0x1c4c2, 0x1cdc2, 0x189d0, 0x1c4ec, 0x19bd0, 0x189c8,
  331. 0x1c4e6, 0x19bc8, 0x1cde6, 0x19bc4, 0x189c2, 0x19bc2, 0x113d0, 0x189ec,
  332. 0x137d0, 0x113c8, 0x189e6, 0x137c8, 0x19be6, 0x137c4, 0x113c2, 0x137c2,
  333. 0x113ec, 0x137ec, 0x113e6, 0x137e6, 0x1fba8, 0x175f0, 0x1bafc, 0x1fba4,
  334. 0x174f8, 0x1ba7e, 0x1fba2, 0x1747c, 0x1743e, 0x1f328, 0x1f996, 0x132fc,
  335. 0x1f768, 0x1fbb6, 0x176fc, 0x1327e, 0x1f764, 0x1f322, 0x1767e, 0x1f762,
  336. 0x1e228, 0x1f116, 0x1e668, 0x1e224, 0x1eee8, 0x1f776, 0x1e222, 0x1eee4,
  337. 0x1e662, 0x1eee2, 0x1c468, 0x1e236, 0x1cce8, 0x1c464, 0x1dde8, 0x1cce4,
  338. 0x1c462, 0x1dde4, 0x1cce2, 0x1dde2, 0x188e8, 0x1c476, 0x199e8, 0x188e4,
  339. 0x1bbe8, 0x199e4, 0x188e2, 0x1bbe4, 0x199e2, 0x1bbe2, 0x111e8, 0x188f6,
  340. 0x133e8, 0x111e4, 0x177e8, 0x133e4, 0x111e2, 0x177e4, 0x133e2, 0x177e2,
  341. 0x111f6, 0x133f6, 0x1fb94, 0x172f8, 0x1b97e, 0x1fb92, 0x1727c, 0x1723e,
  342. 0x1f314, 0x1317e, 0x1f734, 0x1f312, 0x1737e, 0x1f732, 0x1e214, 0x1e634,
  343. 0x1e212, 0x1ee74, 0x1e632, 0x1ee72, 0x1c434, 0x1cc74, 0x1c432, 0x1dcf4,
  344. 0x1cc72, 0x1dcf2, 0x18874, 0x198f4, 0x18872, 0x1b9f4, 0x198f2, 0x1b9f2,
  345. 0x110f4, 0x131f4, 0x110f2, 0x173f4, 0x131f2, 0x173f2, 0x1fb8a, 0x1717c,
  346. 0x1713e, 0x1f30a, 0x1f71a, 0x1e20a, 0x1e61a, 0x1ee3a, 0x1c41a, 0x1cc3a,
  347. 0x1dc7a, 0x1883a, 0x1987a, 0x1b8fa, 0x1107a, 0x130fa, 0x171fa, 0x170be,
  348. 0x1e150, 0x1f0ac, 0x1e148, 0x1f0a6, 0x1e144, 0x1e142, 0x1c2d0, 0x1e16c,
  349. 0x1c2c8, 0x1e166, 0x1c2c4, 0x1c2c2, 0x185d0, 0x1c2ec, 0x185c8, 0x1c2e6,
  350. 0x185c4, 0x185c2, 0x10bd0, 0x185ec, 0x10bc8, 0x185e6, 0x10bc4, 0x10bc2,
  351. 0x10bec, 0x10be6, 0x1f1a8, 0x1f8d6, 0x11afc, 0x1f1a4, 0x11a7e, 0x1f1a2,
  352. 0x1e128, 0x1f096, 0x1e368, 0x1e124, 0x1e364, 0x1e122, 0x1e362, 0x1c268,
  353. 0x1e136, 0x1c6e8, 0x1c264, 0x1c6e4, 0x1c262, 0x1c6e2, 0x184e8, 0x1c276,
  354. 0x18de8, 0x184e4, 0x18de4, 0x184e2, 0x18de2, 0x109e8, 0x184f6, 0x11be8,
  355. 0x109e4, 0x11be4, 0x109e2, 0x11be2, 0x109f6, 0x11bf6, 0x1f9d4, 0x13af8,
  356. 0x19d7e, 0x1f9d2, 0x13a7c, 0x13a3e, 0x1f194, 0x1197e, 0x1f3b4, 0x1f192,
  357. 0x13b7e, 0x1f3b2, 0x1e114, 0x1e334, 0x1e112, 0x1e774, 0x1e332, 0x1e772,
  358. 0x1c234, 0x1c674, 0x1c232, 0x1cef4, 0x1c672, 0x1cef2, 0x18474, 0x18cf4,
  359. 0x18472, 0x19df4, 0x18cf2, 0x19df2, 0x108f4, 0x119f4, 0x108f2, 0x13bf4,
  360. 0x119f2, 0x13bf2, 0x17af0, 0x1bd7c, 0x17a78, 0x1bd3e, 0x17a3c, 0x17a1e,
  361. 0x1f9ca, 0x1397c, 0x1fbda, 0x17b7c, 0x1393e, 0x17b3e, 0x1f18a, 0x1f39a,
  362. 0x1f7ba, 0x1e10a, 0x1e31a, 0x1e73a, 0x1ef7a, 0x1c21a, 0x1c63a, 0x1ce7a,
  363. 0x1defa, 0x1843a, 0x18c7a, 0x19cfa, 0x1bdfa, 0x1087a, 0x118fa, 0x139fa,
  364. 0x17978, 0x1bcbe, 0x1793c, 0x1791e, 0x138be, 0x179be, 0x178bc, 0x1789e,
  365. 0x1785e, 0x1e0a8, 0x1e0a4, 0x1e0a2, 0x1c168, 0x1e0b6, 0x1c164, 0x1c162,
  366. 0x182e8, 0x1c176, 0x182e4, 0x182e2, 0x105e8, 0x182f6, 0x105e4, 0x105e2,
  367. 0x105f6, 0x1f0d4, 0x10d7e, 0x1f0d2, 0x1e094, 0x1e1b4, 0x1e092, 0x1e1b2,
  368. 0x1c134, 0x1c374, 0x1c132, 0x1c372, 0x18274, 0x186f4, 0x18272, 0x186f2,
  369. 0x104f4, 0x10df4, 0x104f2, 0x10df2, 0x1f8ea, 0x11d7c, 0x11d3e, 0x1f0ca,
  370. 0x1f1da, 0x1e08a, 0x1e19a, 0x1e3ba, 0x1c11a, 0x1c33a, 0x1c77a, 0x1823a,
  371. 0x1867a, 0x18efa, 0x1047a, 0x10cfa, 0x11dfa, 0x13d78, 0x19ebe, 0x13d3c,
  372. 0x13d1e, 0x11cbe, 0x13dbe, 0x17d70, 0x1bebc, 0x17d38, 0x1be9e, 0x17d1c,
  373. 0x17d0e, 0x13cbc, 0x17dbc, 0x13c9e, 0x17d9e, 0x17cb8, 0x1be5e, 0x17c9c,
  374. 0x17c8e, 0x13c5e, 0x17cde, 0x17c5c, 0x17c4e, 0x17c2e, 0x1c0b4, 0x1c0b2,
  375. 0x18174, 0x18172, 0x102f4, 0x102f2, 0x1e0da, 0x1c09a, 0x1c1ba, 0x1813a,
  376. 0x1837a, 0x1027a, 0x106fa, 0x10ebe, 0x11ebc, 0x11e9e, 0x13eb8, 0x19f5e,
  377. 0x13e9c, 0x13e8e, 0x11e5e, 0x13ede, 0x17eb0, 0x1bf5c, 0x17e98, 0x1bf4e,
  378. 0x17e8c, 0x17e86, 0x13e5c, 0x17edc, 0x13e4e, 0x17ece, 0x17e58, 0x1bf2e,
  379. 0x17e4c, 0x17e46, 0x13e2e, 0x17e6e, 0x17e2c, 0x17e26, 0x10f5e, 0x11f5c,
  380. 0x11f4e, 0x13f58, 0x19fae, 0x13f4c, 0x13f46, 0x11f2e, 0x13f6e, 0x13f2c,
  381. 0x13f26
  382. }, new int[]{
  383. 0x1abe0, 0x1d5f8, 0x153c0, 0x1a9f0, 0x1d4fc, 0x151e0, 0x1a8f8, 0x1d47e,
  384. 0x150f0, 0x1a87c, 0x15078, 0x1fad0, 0x15be0, 0x1adf8, 0x1fac8, 0x159f0,
  385. 0x1acfc, 0x1fac4, 0x158f8, 0x1ac7e, 0x1fac2, 0x1587c, 0x1f5d0, 0x1faec,
  386. 0x15df8, 0x1f5c8, 0x1fae6, 0x15cfc, 0x1f5c4, 0x15c7e, 0x1f5c2, 0x1ebd0,
  387. 0x1f5ec, 0x1ebc8, 0x1f5e6, 0x1ebc4, 0x1ebc2, 0x1d7d0, 0x1ebec, 0x1d7c8,
  388. 0x1ebe6, 0x1d7c4, 0x1d7c2, 0x1afd0, 0x1d7ec, 0x1afc8, 0x1d7e6, 0x1afc4,
  389. 0x14bc0, 0x1a5f0, 0x1d2fc, 0x149e0, 0x1a4f8, 0x1d27e, 0x148f0, 0x1a47c,
  390. 0x14878, 0x1a43e, 0x1483c, 0x1fa68, 0x14df0, 0x1a6fc, 0x1fa64, 0x14cf8,
  391. 0x1a67e, 0x1fa62, 0x14c7c, 0x14c3e, 0x1f4e8, 0x1fa76, 0x14efc, 0x1f4e4,
  392. 0x14e7e, 0x1f4e2, 0x1e9e8, 0x1f4f6, 0x1e9e4, 0x1e9e2, 0x1d3e8, 0x1e9f6,
  393. 0x1d3e4, 0x1d3e2, 0x1a7e8, 0x1d3f6, 0x1a7e4, 0x1a7e2, 0x145e0, 0x1a2f8,
  394. 0x1d17e, 0x144f0, 0x1a27c, 0x14478, 0x1a23e, 0x1443c, 0x1441e, 0x1fa34,
  395. 0x146f8, 0x1a37e, 0x1fa32, 0x1467c, 0x1463e, 0x1f474, 0x1477e, 0x1f472,
  396. 0x1e8f4, 0x1e8f2, 0x1d1f4, 0x1d1f2, 0x1a3f4, 0x1a3f2, 0x142f0, 0x1a17c,
  397. 0x14278, 0x1a13e, 0x1423c, 0x1421e, 0x1fa1a, 0x1437c, 0x1433e, 0x1f43a,
  398. 0x1e87a, 0x1d0fa, 0x14178, 0x1a0be, 0x1413c, 0x1411e, 0x141be, 0x140bc,
  399. 0x1409e, 0x12bc0, 0x195f0, 0x1cafc, 0x129e0, 0x194f8, 0x1ca7e, 0x128f0,
  400. 0x1947c, 0x12878, 0x1943e, 0x1283c, 0x1f968, 0x12df0, 0x196fc, 0x1f964,
  401. 0x12cf8, 0x1967e, 0x1f962, 0x12c7c, 0x12c3e, 0x1f2e8, 0x1f976, 0x12efc,
  402. 0x1f2e4, 0x12e7e, 0x1f2e2, 0x1e5e8, 0x1f2f6, 0x1e5e4, 0x1e5e2, 0x1cbe8,
  403. 0x1e5f6, 0x1cbe4, 0x1cbe2, 0x197e8, 0x1cbf6, 0x197e4, 0x197e2, 0x1b5e0,
  404. 0x1daf8, 0x1ed7e, 0x169c0, 0x1b4f0, 0x1da7c, 0x168e0, 0x1b478, 0x1da3e,
  405. 0x16870, 0x1b43c, 0x16838, 0x1b41e, 0x1681c, 0x125e0, 0x192f8, 0x1c97e,
  406. 0x16de0, 0x124f0, 0x1927c, 0x16cf0, 0x1b67c, 0x1923e, 0x16c78, 0x1243c,
  407. 0x16c3c, 0x1241e, 0x16c1e, 0x1f934, 0x126f8, 0x1937e, 0x1fb74, 0x1f932,
  408. 0x16ef8, 0x1267c, 0x1fb72, 0x16e7c, 0x1263e, 0x16e3e, 0x1f274, 0x1277e,
  409. 0x1f6f4, 0x1f272, 0x16f7e, 0x1f6f2, 0x1e4f4, 0x1edf4, 0x1e4f2, 0x1edf2,
  410. 0x1c9f4, 0x1dbf4, 0x1c9f2, 0x1dbf2, 0x193f4, 0x193f2, 0x165c0, 0x1b2f0,
  411. 0x1d97c, 0x164e0, 0x1b278, 0x1d93e, 0x16470, 0x1b23c, 0x16438, 0x1b21e,
  412. 0x1641c, 0x1640e, 0x122f0, 0x1917c, 0x166f0, 0x12278, 0x1913e, 0x16678,
  413. 0x1b33e, 0x1663c, 0x1221e, 0x1661e, 0x1f91a, 0x1237c, 0x1fb3a, 0x1677c,
  414. 0x1233e, 0x1673e, 0x1f23a, 0x1f67a, 0x1e47a, 0x1ecfa, 0x1c8fa, 0x1d9fa,
  415. 0x191fa, 0x162e0, 0x1b178, 0x1d8be, 0x16270, 0x1b13c, 0x16238, 0x1b11e,
  416. 0x1621c, 0x1620e, 0x12178, 0x190be, 0x16378, 0x1213c, 0x1633c, 0x1211e,
  417. 0x1631e, 0x121be, 0x163be, 0x16170, 0x1b0bc, 0x16138, 0x1b09e, 0x1611c,
  418. 0x1610e, 0x120bc, 0x161bc, 0x1209e, 0x1619e, 0x160b8, 0x1b05e, 0x1609c,
  419. 0x1608e, 0x1205e, 0x160de, 0x1605c, 0x1604e, 0x115e0, 0x18af8, 0x1c57e,
  420. 0x114f0, 0x18a7c, 0x11478, 0x18a3e, 0x1143c, 0x1141e, 0x1f8b4, 0x116f8,
  421. 0x18b7e, 0x1f8b2, 0x1167c, 0x1163e, 0x1f174, 0x1177e, 0x1f172, 0x1e2f4,
  422. 0x1e2f2, 0x1c5f4, 0x1c5f2, 0x18bf4, 0x18bf2, 0x135c0, 0x19af0, 0x1cd7c,
  423. 0x134e0, 0x19a78, 0x1cd3e, 0x13470, 0x19a3c, 0x13438, 0x19a1e, 0x1341c,
  424. 0x1340e, 0x112f0, 0x1897c, 0x136f0, 0x11278, 0x1893e, 0x13678, 0x19b3e,
  425. 0x1363c, 0x1121e, 0x1361e, 0x1f89a, 0x1137c, 0x1f9ba, 0x1377c, 0x1133e,
  426. 0x1373e, 0x1f13a, 0x1f37a, 0x1e27a, 0x1e6fa, 0x1c4fa, 0x1cdfa, 0x189fa,
  427. 0x1bae0, 0x1dd78, 0x1eebe, 0x174c0, 0x1ba70, 0x1dd3c, 0x17460, 0x1ba38,
  428. 0x1dd1e, 0x17430, 0x1ba1c, 0x17418, 0x1ba0e, 0x1740c, 0x132e0, 0x19978,
  429. 0x1ccbe, 0x176e0, 0x13270, 0x1993c, 0x17670, 0x1bb3c, 0x1991e, 0x17638,
  430. 0x1321c, 0x1761c, 0x1320e, 0x1760e, 0x11178, 0x188be, 0x13378, 0x1113c,
  431. 0x17778, 0x1333c, 0x1111e, 0x1773c, 0x1331e, 0x1771e, 0x111be, 0x133be,
  432. 0x177be, 0x172c0, 0x1b970, 0x1dcbc, 0x17260, 0x1b938, 0x1dc9e, 0x17230,
  433. 0x1b91c, 0x17218, 0x1b90e, 0x1720c, 0x17206, 0x13170, 0x198bc, 0x17370,
  434. 0x13138, 0x1989e, 0x17338, 0x1b99e, 0x1731c, 0x1310e, 0x1730e, 0x110bc,
  435. 0x131bc, 0x1109e, 0x173bc, 0x1319e, 0x1739e, 0x17160, 0x1b8b8, 0x1dc5e,
  436. 0x17130, 0x1b89c, 0x17118, 0x1b88e, 0x1710c, 0x17106, 0x130b8, 0x1985e,
  437. 0x171b8, 0x1309c, 0x1719c, 0x1308e, 0x1718e, 0x1105e, 0x130de, 0x171de,
  438. 0x170b0, 0x1b85c, 0x17098, 0x1b84e, 0x1708c, 0x17086, 0x1305c, 0x170dc,
  439. 0x1304e, 0x170ce, 0x17058, 0x1b82e, 0x1704c, 0x17046, 0x1302e, 0x1706e,
  440. 0x1702c, 0x17026, 0x10af0, 0x1857c, 0x10a78, 0x1853e, 0x10a3c, 0x10a1e,
  441. 0x10b7c, 0x10b3e, 0x1f0ba, 0x1e17a, 0x1c2fa, 0x185fa, 0x11ae0, 0x18d78,
  442. 0x1c6be, 0x11a70, 0x18d3c, 0x11a38, 0x18d1e, 0x11a1c, 0x11a0e, 0x10978,
  443. 0x184be, 0x11b78, 0x1093c, 0x11b3c, 0x1091e, 0x11b1e, 0x109be, 0x11bbe,
  444. 0x13ac0, 0x19d70, 0x1cebc, 0x13a60, 0x19d38, 0x1ce9e, 0x13a30, 0x19d1c,
  445. 0x13a18, 0x19d0e, 0x13a0c, 0x13a06, 0x11970, 0x18cbc, 0x13b70, 0x11938,
  446. 0x18c9e, 0x13b38, 0x1191c, 0x13b1c, 0x1190e, 0x13b0e, 0x108bc, 0x119bc,
  447. 0x1089e, 0x13bbc, 0x1199e, 0x13b9e, 0x1bd60, 0x1deb8, 0x1ef5e, 0x17a40,
  448. 0x1bd30, 0x1de9c, 0x17a20, 0x1bd18, 0x1de8e, 0x17a10, 0x1bd0c, 0x17a08,
  449. 0x1bd06, 0x17a04, 0x13960, 0x19cb8, 0x1ce5e, 0x17b60, 0x13930, 0x19c9c,
  450. 0x17b30, 0x1bd9c, 0x19c8e, 0x17b18, 0x1390c, 0x17b0c, 0x13906, 0x17b06,
  451. 0x118b8, 0x18c5e, 0x139b8, 0x1189c, 0x17bb8, 0x1399c, 0x1188e, 0x17b9c,
  452. 0x1398e, 0x17b8e, 0x1085e, 0x118de, 0x139de, 0x17bde, 0x17940, 0x1bcb0,
  453. 0x1de5c, 0x17920, 0x1bc98, 0x1de4e, 0x17910, 0x1bc8c, 0x17908, 0x1bc86,
  454. 0x17904, 0x17902, 0x138b0, 0x19c5c, 0x179b0, 0x13898, 0x19c4e, 0x17998,
  455. 0x1bcce, 0x1798c, 0x13886, 0x17986, 0x1185c, 0x138dc, 0x1184e, 0x179dc,
  456. 0x138ce, 0x179ce, 0x178a0, 0x1bc58, 0x1de2e, 0x17890, 0x1bc4c, 0x17888,
  457. 0x1bc46, 0x17884, 0x17882, 0x13858, 0x19c2e, 0x178d8, 0x1384c, 0x178cc,
  458. 0x13846, 0x178c6, 0x1182e, 0x1386e, 0x178ee, 0x17850, 0x1bc2c, 0x17848,
  459. 0x1bc26, 0x17844, 0x17842, 0x1382c, 0x1786c, 0x13826, 0x17866, 0x17828,
  460. 0x1bc16, 0x17824, 0x17822, 0x13816, 0x17836, 0x10578, 0x182be, 0x1053c,
  461. 0x1051e, 0x105be, 0x10d70, 0x186bc, 0x10d38, 0x1869e, 0x10d1c, 0x10d0e,
  462. 0x104bc, 0x10dbc, 0x1049e, 0x10d9e, 0x11d60, 0x18eb8, 0x1c75e, 0x11d30,
  463. 0x18e9c, 0x11d18, 0x18e8e, 0x11d0c, 0x11d06, 0x10cb8, 0x1865e, 0x11db8,
  464. 0x10c9c, 0x11d9c, 0x10c8e, 0x11d8e, 0x1045e, 0x10cde, 0x11dde, 0x13d40,
  465. 0x19eb0, 0x1cf5c, 0x13d20, 0x19e98, 0x1cf4e, 0x13d10, 0x19e8c, 0x13d08,
  466. 0x19e86, 0x13d04, 0x13d02, 0x11cb0, 0x18e5c, 0x13db0, 0x11c98, 0x18e4e,
  467. 0x13d98, 0x19ece, 0x13d8c, 0x11c86, 0x13d86, 0x10c5c, 0x11cdc, 0x10c4e,
  468. 0x13ddc, 0x11cce, 0x13dce, 0x1bea0, 0x1df58, 0x1efae, 0x1be90, 0x1df4c,
  469. 0x1be88, 0x1df46, 0x1be84, 0x1be82, 0x13ca0, 0x19e58, 0x1cf2e, 0x17da0,
  470. 0x13c90, 0x19e4c, 0x17d90, 0x1becc, 0x19e46, 0x17d88, 0x13c84, 0x17d84,
  471. 0x13c82, 0x17d82, 0x11c58, 0x18e2e, 0x13cd8, 0x11c4c, 0x17dd8, 0x13ccc,
  472. 0x11c46, 0x17dcc, 0x13cc6, 0x17dc6, 0x10c2e, 0x11c6e, 0x13cee, 0x17dee,
  473. 0x1be50, 0x1df2c, 0x1be48, 0x1df26, 0x1be44, 0x1be42, 0x13c50, 0x19e2c,
  474. 0x17cd0, 0x13c48, 0x19e26, 0x17cc8, 0x1be66, 0x17cc4, 0x13c42, 0x17cc2,
  475. 0x11c2c, 0x13c6c, 0x11c26, 0x17cec, 0x13c66, 0x17ce6, 0x1be28, 0x1df16,
  476. 0x1be24, 0x1be22, 0x13c28, 0x19e16, 0x17c68, 0x13c24, 0x17c64, 0x13c22,
  477. 0x17c62, 0x11c16, 0x13c36, 0x17c76, 0x1be14, 0x1be12, 0x13c14, 0x17c34,
  478. 0x13c12, 0x17c32, 0x102bc, 0x1029e, 0x106b8, 0x1835e, 0x1069c, 0x1068e,
  479. 0x1025e, 0x106de, 0x10eb0, 0x1875c, 0x10e98, 0x1874e, 0x10e8c, 0x10e86,
  480. 0x1065c, 0x10edc, 0x1064e, 0x10ece, 0x11ea0, 0x18f58, 0x1c7ae, 0x11e90,
  481. 0x18f4c, 0x11e88, 0x18f46, 0x11e84, 0x11e82, 0x10e58, 0x1872e, 0x11ed8,
  482. 0x18f6e, 0x11ecc, 0x10e46, 0x11ec6, 0x1062e, 0x10e6e, 0x11eee, 0x19f50,
  483. 0x1cfac, 0x19f48, 0x1cfa6, 0x19f44, 0x19f42, 0x11e50, 0x18f2c, 0x13ed0,
  484. 0x19f6c, 0x18f26, 0x13ec8, 0x11e44, 0x13ec4, 0x11e42, 0x13ec2, 0x10e2c,
  485. 0x11e6c, 0x10e26, 0x13eec, 0x11e66, 0x13ee6, 0x1dfa8, 0x1efd6, 0x1dfa4,
  486. 0x1dfa2, 0x19f28, 0x1cf96, 0x1bf68, 0x19f24, 0x1bf64, 0x19f22, 0x1bf62,
  487. 0x11e28, 0x18f16, 0x13e68, 0x11e24, 0x17ee8, 0x13e64, 0x11e22, 0x17ee4,
  488. 0x13e62, 0x17ee2, 0x10e16, 0x11e36, 0x13e76, 0x17ef6, 0x1df94, 0x1df92,
  489. 0x19f14, 0x1bf34, 0x19f12, 0x1bf32, 0x11e14, 0x13e34, 0x11e12, 0x17e74,
  490. 0x13e32, 0x17e72, 0x1df8a, 0x19f0a, 0x1bf1a, 0x11e0a, 0x13e1a, 0x17e3a,
  491. 0x1035c, 0x1034e, 0x10758, 0x183ae, 0x1074c, 0x10746, 0x1032e, 0x1076e,
  492. 0x10f50, 0x187ac, 0x10f48, 0x187a6, 0x10f44, 0x10f42, 0x1072c, 0x10f6c,
  493. 0x10726, 0x10f66, 0x18fa8, 0x1c7d6, 0x18fa4, 0x18fa2, 0x10f28, 0x18796,
  494. 0x11f68, 0x18fb6, 0x11f64, 0x10f22, 0x11f62, 0x10716, 0x10f36, 0x11f76,
  495. 0x1cfd4, 0x1cfd2, 0x18f94, 0x19fb4, 0x18f92, 0x19fb2, 0x10f14, 0x11f34,
  496. 0x10f12, 0x13f74, 0x11f32, 0x13f72, 0x1cfca, 0x18f8a, 0x19f9a, 0x10f0a,
  497. 0x11f1a, 0x13f3a, 0x103ac, 0x103a6, 0x107a8, 0x183d6, 0x107a4, 0x107a2,
  498. 0x10396, 0x107b6, 0x187d4, 0x187d2, 0x10794, 0x10fb4, 0x10792, 0x10fb2,
  499. 0x1c7ea
  500. }};
  501. private static readonly int[][] ERROR_LEVEL = new int[][]
  502. {new int[]{
  503. 27, 917
  504. }, new int[]{
  505. 522, 568, 723, 809
  506. }, new int[]{
  507. 237, 308, 436, 284, 646, 653, 428, 379
  508. }, new int[]{
  509. 274, 562, 232, 755, 599, 524, 801, 132, 295, 116, 442, 428, 295, 42, 176, 65
  510. }, new int[]{
  511. 361, 575, 922, 525, 176, 586, 640, 321, 536, 742, 677, 742, 687, 284, 193, 517,
  512. 273, 494, 263, 147, 593, 800, 571, 320, 803, 133, 231, 390, 685, 330, 63, 410
  513. }, new int[]{
  514. 539, 422, 6, 93, 862, 771, 453, 106, 610, 287, 107, 505, 733, 877, 381, 612,
  515. 723, 476, 462, 172, 430, 609, 858, 822, 543, 376, 511, 400, 672, 762, 283, 184,
  516. 440, 35, 519, 31, 460, 594, 225, 535, 517, 352, 605, 158, 651, 201, 488, 502,
  517. 648, 733, 717, 83, 404, 97, 280, 771, 840, 629, 4, 381, 843, 623, 264, 543
  518. }, new int[]{
  519. 521, 310, 864, 547, 858, 580, 296, 379, 53, 779, 897, 444, 400, 925, 749, 415,
  520. 822, 93, 217, 208, 928, 244, 583, 620, 246, 148, 447, 631, 292, 908, 490, 704,
  521. 516, 258, 457, 907, 594, 723, 674, 292, 272, 96, 684, 432, 686, 606, 860, 569,
  522. 193, 219, 129, 186, 236, 287, 192, 775, 278, 173, 40, 379, 712, 463, 646, 776,
  523. 171, 491, 297, 763, 156, 732, 95, 270, 447, 90, 507, 48, 228, 821, 808, 898,
  524. 784, 663, 627, 378, 382, 262, 380, 602, 754, 336, 89, 614, 87, 432, 670, 616,
  525. 157, 374, 242, 726, 600, 269, 375, 898, 845, 454, 354, 130, 814, 587, 804, 34,
  526. 211, 330, 539, 297, 827, 865, 37, 517, 834, 315, 550, 86, 801, 4, 108, 539
  527. }, new int[]{
  528. 524, 894, 75, 766, 882, 857, 74, 204, 82, 586, 708, 250, 905, 786, 138, 720,
  529. 858, 194, 311, 913, 275, 190, 375, 850, 438, 733, 194, 280, 201, 280, 828, 757,
  530. 710, 814, 919, 89, 68, 569, 11, 204, 796, 605, 540, 913, 801, 700, 799, 137,
  531. 439, 418, 592, 668, 353, 859, 370, 694, 325, 240, 216, 257, 284, 549, 209, 884,
  532. 315, 70, 329, 793, 490, 274, 877, 162, 749, 812, 684, 461, 334, 376, 849, 521,
  533. 307, 291, 803, 712, 19, 358, 399, 908, 103, 511, 51, 8, 517, 225, 289, 470,
  534. 637, 731, 66, 255, 917, 269, 463, 830, 730, 433, 848, 585, 136, 538, 906, 90,
  535. 2, 290, 743, 199, 655, 903, 329, 49, 802, 580, 355, 588, 188, 462, 10, 134,
  536. 628, 320, 479, 130, 739, 71, 263, 318, 374, 601, 192, 605, 142, 673, 687, 234,
  537. 722, 384, 177, 752, 607, 640, 455, 193, 689, 707, 805, 641, 48, 60, 732, 621,
  538. 895, 544, 261, 852, 655, 309, 697, 755, 756, 60, 231, 773, 434, 421, 726, 528,
  539. 503, 118, 49, 795, 32, 144, 500, 238, 836, 394, 280, 566, 319, 9, 647, 550,
  540. 73, 914, 342, 126, 32, 681, 331, 792, 620, 60, 609, 441, 180, 791, 893, 754,
  541. 605, 383, 228, 749, 760, 213, 54, 297, 134, 54, 834, 299, 922, 191, 910, 532,
  542. 609, 829, 189, 20, 167, 29, 872, 449, 83, 402, 41, 656, 505, 579, 481, 173,
  543. 404, 251, 688, 95, 497, 555, 642, 543, 307, 159, 924, 558, 648, 55, 497, 10
  544. }, new int[]{
  545. 352, 77, 373, 504, 35, 599, 428, 207, 409, 574, 118, 498, 285, 380, 350, 492,
  546. 197, 265, 920, 155, 914, 299, 229, 643, 294, 871, 306, 88, 87, 193, 352, 781,
  547. 846, 75, 327, 520, 435, 543, 203, 666, 249, 346, 781, 621, 640, 268, 794, 534,
  548. 539, 781, 408, 390, 644, 102, 476, 499, 290, 632, 545, 37, 858, 916, 552, 41,
  549. 542, 289, 122, 272, 383, 800, 485, 98, 752, 472, 761, 107, 784, 860, 658, 741,
  550. 290, 204, 681, 407, 855, 85, 99, 62, 482, 180, 20, 297, 451, 593, 913, 142,
  551. 808, 684, 287, 536, 561, 76, 653, 899, 729, 567, 744, 390, 513, 192, 516, 258,
  552. 240, 518, 794, 395, 768, 848, 51, 610, 384, 168, 190, 826, 328, 596, 786, 303,
  553. 570, 381, 415, 641, 156, 237, 151, 429, 531, 207, 676, 710, 89, 168, 304, 402,
  554. 40, 708, 575, 162, 864, 229, 65, 861, 841, 512, 164, 477, 221, 92, 358, 785,
  555. 288, 357, 850, 836, 827, 736, 707, 94, 8, 494, 114, 521, 2, 499, 851, 543,
  556. 152, 729, 771, 95, 248, 361, 578, 323, 856, 797, 289, 51, 684, 466, 533, 820,
  557. 669, 45, 902, 452, 167, 342, 244, 173, 35, 463, 651, 51, 699, 591, 452, 578,
  558. 37, 124, 298, 332, 552, 43, 427, 119, 662, 777, 475, 850, 764, 364, 578, 911,
  559. 283, 711, 472, 420, 245, 288, 594, 394, 511, 327, 589, 777, 699, 688, 43, 408,
  560. 842, 383, 721, 521, 560, 644, 714, 559, 62, 145, 873, 663, 713, 159, 672, 729,
  561. 624, 59, 193, 417, 158, 209, 563, 564, 343, 693, 109, 608, 563, 365, 181, 772,
  562. 677, 310, 248, 353, 708, 410, 579, 870, 617, 841, 632, 860, 289, 536, 35, 777,
  563. 618, 586, 424, 833, 77, 597, 346, 269, 757, 632, 695, 751, 331, 247, 184, 45,
  564. 787, 680, 18, 66, 407, 369, 54, 492, 228, 613, 830, 922, 437, 519, 644, 905,
  565. 789, 420, 305, 441, 207, 300, 892, 827, 141, 537, 381, 662, 513, 56, 252, 341,
  566. 242, 797, 838, 837, 720, 224, 307, 631, 61, 87, 560, 310, 756, 665, 397, 808,
  567. 851, 309, 473, 795, 378, 31, 647, 915, 459, 806, 590, 731, 425, 216, 548, 249,
  568. 321, 881, 699, 535, 673, 782, 210, 815, 905, 303, 843, 922, 281, 73, 469, 791,
  569. 660, 162, 498, 308, 155, 422, 907, 817, 187, 62, 16, 425, 535, 336, 286, 437,
  570. 375, 273, 610, 296, 183, 923, 116, 667, 751, 353, 62, 366, 691, 379, 687, 842,
  571. 37, 357, 720, 742, 330, 5, 39, 923, 311, 424, 242, 749, 321, 54, 669, 316,
  572. 342, 299, 534, 105, 667, 488, 640, 672, 576, 540, 316, 486, 721, 610, 46, 656,
  573. 447, 171, 616, 464, 190, 531, 297, 321, 762, 752, 533, 175, 134, 14, 381, 433,
  574. 717, 45, 111, 20, 596, 284, 736, 138, 646, 411, 877, 669, 141, 919, 45, 780,
  575. 407, 164, 332, 899, 165, 726, 600, 325, 498, 655, 357, 752, 768, 223, 849, 647,
  576. 63, 310, 863, 251, 366, 304, 282, 738, 675, 410, 389, 244, 31, 121, 303, 263
  577. }};
  578. #endregion
  579. #region Properties
  580. /// <summary>
  581. /// Gets or sets the barcode aspect ratio.
  582. /// </summary>
  583. /// <remarks>
  584. /// A ratio or 0.5 will make the barcode width twice as large as the height.
  585. /// </remarks>
  586. [DefaultValue(0.5f)]
  587. public float AspectRatio
  588. {
  589. get { return aspectRatio; }
  590. set { aspectRatio = value; }
  591. }
  592. /// <summary>
  593. /// Gets or sets the number of barcode data columns.
  594. /// </summary>
  595. /// <remarks>
  596. /// To calculate the necessary number of columns and rows, set the <see cref="Columns"/>
  597. /// and <see cref="Rows"/> properties to 0. In this case, the <see cref="AspectRatio"/> property
  598. /// should be set to desired aspect ratio.
  599. /// </remarks>
  600. [DefaultValue(0)]
  601. public int Columns
  602. {
  603. get { return columns; }
  604. set { columns = value; }
  605. }
  606. /// <summary>
  607. /// Gets or sets the number of barcode data rows.
  608. /// </summary>
  609. /// <remarks>
  610. /// To calculate the necessary number of columns and rows, set the <see cref="Columns"/>
  611. /// and <see cref="Rows"/> properties to 0. In this case, the <see cref="AspectRatio"/> property
  612. /// should be set to desired aspect ratio.
  613. /// </remarks>
  614. [DefaultValue(0)]
  615. public int Rows
  616. {
  617. get { return rows; }
  618. set { rows = value; }
  619. }
  620. /// <summary>
  621. /// Gets or sets the error level correction used for the barcode.
  622. /// </summary>
  623. [DefaultValue(PDF417ErrorCorrection.Auto)]
  624. public PDF417ErrorCorrection ErrorCorrection
  625. {
  626. get { return errorCorrection; }
  627. set { errorCorrection = value; }
  628. }
  629. /// <summary>
  630. /// Gets or sets the code page used for text conversion.
  631. /// </summary>
  632. /// <remarks>
  633. /// Use this property to encode non-ASCII characters. For example, set this
  634. /// property to <b>1251</b> to use Window CP1251.
  635. /// </remarks>
  636. [DefaultValue(437)]
  637. public int CodePage
  638. {
  639. get { return codePage; }
  640. set { codePage = value; }
  641. }
  642. /// <summary>
  643. /// Gets or sets the compaction mode.
  644. /// </summary>
  645. [DefaultValue(PDF417CompactionMode.Auto)]
  646. public PDF417CompactionMode CompactionMode
  647. {
  648. get { return compactionMode; }
  649. set { compactionMode = value; }
  650. }
  651. /// <summary>
  652. /// Gets or sets the size of the pixel.
  653. /// </summary>
  654. public Size PixelSize
  655. {
  656. get { return pixelSize; }
  657. set { pixelSize = value; }
  658. }
  659. #endregion
  660. #region Private Methods
  661. private bool CheckSegmentType(Segment segment, char type)
  662. {
  663. if (segment == null)
  664. return false;
  665. return segment.type == type;
  666. }
  667. private int GetSegmentLength(Segment segment)
  668. {
  669. if (segment == null)
  670. return 0;
  671. return segment.end - segment.start;
  672. }
  673. private void OutCodeword17(int codeword)
  674. {
  675. unchecked
  676. {
  677. int bytePtr = bitPtr / 8;
  678. int bit = bitPtr - bytePtr * 8;
  679. outBits[bytePtr++] |= (byte)(codeword >> (9 + bit));
  680. outBits[bytePtr++] |= (byte)(codeword >> (1 + bit));
  681. codeword <<= 8;
  682. outBits[bytePtr] |= (byte)(codeword >> (1 + bit));
  683. bitPtr += 17;
  684. }
  685. }
  686. private void OutCodeword18(int codeword)
  687. {
  688. unchecked
  689. {
  690. int bytePtr = bitPtr / 8;
  691. int bit = bitPtr - bytePtr * 8;
  692. outBits[bytePtr++] |= (byte)(codeword >> (10 + bit));
  693. outBits[bytePtr++] |= (byte)(codeword >> (2 + bit));
  694. codeword <<= 8;
  695. outBits[bytePtr] |= (byte)(codeword >> (2 + bit));
  696. if (bit == 7)
  697. outBits[++bytePtr] |= 0x80;
  698. bitPtr += 18;
  699. }
  700. }
  701. private void OutCodeword(int codeword)
  702. {
  703. OutCodeword17(codeword);
  704. }
  705. private void OutStopPattern()
  706. {
  707. OutCodeword18(STOP_PATTERN);
  708. }
  709. private void OutStartPattern()
  710. {
  711. OutCodeword17(START_PATTERN);
  712. }
  713. private void OutPaintCode()
  714. {
  715. int codePtr = 0;
  716. bitColumns = START_CODE_SIZE * (codeColumns + 3) + STOP_SIZE;
  717. int lenBits = ((bitColumns - 1) / 8 + 1) * codeRows;
  718. outBits = new byte[lenBits];
  719. for (int row = 0; row < codeRows; ++row)
  720. {
  721. bitPtr = ((bitColumns - 1) / 8 + 1) * 8 * row;
  722. int rowMod = row % 3;
  723. int[] cluster = CLUSTERS[rowMod];
  724. OutStartPattern();
  725. int edge = 0;
  726. switch (rowMod)
  727. {
  728. case 0:
  729. edge = 30 * (row / 3) + ((codeRows - 1) / 3);
  730. break;
  731. case 1:
  732. edge = 30 * (row / 3) + errorLevel * 3 + ((codeRows - 1) % 3);
  733. break;
  734. default:
  735. edge = 30 * (row / 3) + codeColumns - 1;
  736. break;
  737. }
  738. OutCodeword(cluster[edge]);
  739. for (int column = 0; column < codeColumns; ++column)
  740. {
  741. OutCodeword(cluster[codewords[codePtr++]]);
  742. }
  743. switch (rowMod)
  744. {
  745. case 0:
  746. edge = 30 * (row / 3) + codeColumns - 1;
  747. break;
  748. case 1:
  749. edge = 30 * (row / 3) + ((codeRows - 1) / 3);
  750. break;
  751. default:
  752. edge = 30 * (row / 3) + errorLevel * 3 + ((codeRows - 1) % 3);
  753. break;
  754. }
  755. OutCodeword(cluster[edge]);
  756. OutStopPattern();
  757. }
  758. }
  759. private void CalculateErrorCorrection(int dest)
  760. {
  761. if (errorLevel < 0 || errorLevel > 8)
  762. errorLevel = 0;
  763. int[] A = ERROR_LEVEL[errorLevel];
  764. int Alength = 2 << errorLevel;
  765. for (int k = 0; k < Alength; ++k)
  766. codewords[dest + k] = 0;
  767. int lastE = Alength - 1;
  768. for (int k = 0; k < lenCodewords; ++k)
  769. {
  770. int t1 = codewords[k] + codewords[dest];
  771. for (int e = 0; e <= lastE; ++e)
  772. {
  773. int t2 = (t1 * A[lastE - e]) % MOD;
  774. int t3 = MOD - t2;
  775. codewords[dest + e] = ((e == lastE ? 0 : codewords[dest + e + 1]) + t3) % MOD;
  776. }
  777. }
  778. for (int k = 0; k < Alength; ++k)
  779. codewords[dest + k] = (MOD - codewords[dest + k]) % MOD;
  780. }
  781. private static int GetTextTypeAndValue(byte[] input, int maxLength, int idx)
  782. {
  783. if (idx >= maxLength)
  784. return 0;
  785. char c = (char)(input[idx] & 0xff);
  786. if (c >= 'A' && c <= 'Z')
  787. return (ALPHA + c - 'A');
  788. if (c >= 'a' && c <= 'z')
  789. return (LOWER + c - 'a');
  790. if (c == ' ')
  791. return (ALPHA + LOWER + MIXED + SPACE);
  792. int ms = MIXED_SET.IndexOf(c);
  793. int ps = PUNCTUATION_SET.IndexOf(c);
  794. if (ms < 0 && ps < 0)
  795. return (ISBYTE + c);
  796. if (ms == ps)
  797. return (MIXED + PUNCTUATION + ms);
  798. if (ms >= 0)
  799. return (MIXED + ms);
  800. return (PUNCTUATION + ps);
  801. }
  802. private int GetTextTypeAndValue(int maxLength, int idx)
  803. {
  804. return GetTextTypeAndValue(bytes, maxLength, idx);
  805. }
  806. private void TextCompaction(byte[] input, int start, int length)
  807. {
  808. int[] dest = new int[ABSOLUTE_MAX_TEXT_SIZE * 2];
  809. int mode = ALPHA;
  810. int ptr = 0;
  811. int fullBytes = 0;
  812. int v = 0;
  813. int k;
  814. int size;
  815. length += start;
  816. for (k = start; k < length; ++k)
  817. {
  818. v = GetTextTypeAndValue(input, length, k);
  819. if ((v & mode) != 0)
  820. {
  821. dest[ptr++] = v & 0xff;
  822. continue;
  823. }
  824. if ((v & ISBYTE) != 0)
  825. {
  826. if ((ptr & 1) != 0)
  827. {
  828. //add a padding word
  829. dest[ptr++] = PAL;
  830. mode = (mode & PUNCTUATION) != 0 ? ALPHA : mode;
  831. }
  832. dest[ptr++] = BYTESHIFT;
  833. dest[ptr++] = v & 0xff;
  834. fullBytes += 2;
  835. continue;
  836. }
  837. switch (mode)
  838. {
  839. case ALPHA:
  840. if ((v & LOWER) != 0)
  841. {
  842. dest[ptr++] = LL;
  843. dest[ptr++] = v & 0xff;
  844. mode = LOWER;
  845. }
  846. else if ((v & MIXED) != 0)
  847. {
  848. dest[ptr++] = ML;
  849. dest[ptr++] = v & 0xff;
  850. mode = MIXED;
  851. }
  852. else if ((GetTextTypeAndValue(input, length, k + 1) & GetTextTypeAndValue(input, length, k + 2) & PUNCTUATION) != 0)
  853. {
  854. dest[ptr++] = ML;
  855. dest[ptr++] = PL;
  856. dest[ptr++] = v & 0xff;
  857. mode = PUNCTUATION;
  858. }
  859. else
  860. {
  861. dest[ptr++] = PS;
  862. dest[ptr++] = v & 0xff;
  863. }
  864. break;
  865. case LOWER:
  866. if ((v & ALPHA) != 0)
  867. {
  868. if ((GetTextTypeAndValue(length, k + 1) & GetTextTypeAndValue(length, k + 2) & ALPHA) != 0)
  869. {
  870. dest[ptr++] = ML;
  871. dest[ptr++] = AL;
  872. mode = ALPHA;
  873. }
  874. else
  875. {
  876. dest[ptr++] = AS;
  877. }
  878. dest[ptr++] = v & 0xff;
  879. }
  880. else if ((v & MIXED) != 0)
  881. {
  882. dest[ptr++] = ML;
  883. dest[ptr++] = v & 0xff;
  884. mode = MIXED;
  885. }
  886. else if ((GetTextTypeAndValue(input, length, k + 1) & GetTextTypeAndValue(input, length, k + 2) & PUNCTUATION) != 0)
  887. {
  888. dest[ptr++] = ML;
  889. dest[ptr++] = PL;
  890. dest[ptr++] = v & 0xff;
  891. mode = PUNCTUATION;
  892. }
  893. else
  894. {
  895. dest[ptr++] = PS;
  896. dest[ptr++] = v & 0xff;
  897. }
  898. break;
  899. case MIXED:
  900. if ((v & LOWER) != 0)
  901. {
  902. dest[ptr++] = LL;
  903. dest[ptr++] = v & 0xff;
  904. mode = LOWER;
  905. }
  906. else if ((v & ALPHA) != 0)
  907. {
  908. dest[ptr++] = AL;
  909. dest[ptr++] = v & 0xff;
  910. mode = ALPHA;
  911. }
  912. else if ((GetTextTypeAndValue(input, length, k + 1) & GetTextTypeAndValue(input, length, k + 2) & PUNCTUATION) != 0)
  913. {
  914. dest[ptr++] = PL;
  915. dest[ptr++] = v & 0xff;
  916. mode = PUNCTUATION;
  917. }
  918. else
  919. {
  920. dest[ptr++] = PS;
  921. dest[ptr++] = v & 0xff;
  922. }
  923. break;
  924. case PUNCTUATION:
  925. dest[ptr++] = PAL;
  926. mode = ALPHA;
  927. --k;
  928. break;
  929. }
  930. }
  931. if ((ptr & 1) != 0)
  932. dest[ptr++] = PS;
  933. size = (ptr + fullBytes) / 2;
  934. if (size + cwPtr > MAX_DATA_CODEWORDS)
  935. {
  936. throw new ArgumentOutOfRangeException("The text is too big.");
  937. }
  938. length = ptr;
  939. ptr = 0;
  940. while (ptr < length)
  941. {
  942. v = dest[ptr++];
  943. if (v >= 30)
  944. {
  945. codewords[cwPtr++] = v;
  946. codewords[cwPtr++] = dest[ptr++];
  947. }
  948. else
  949. codewords[cwPtr++] = v * 30 + dest[ptr++];
  950. }
  951. }
  952. private void TextCompaction(int start, int length)
  953. {
  954. TextCompaction(bytes, start, length);
  955. }
  956. private void BasicNumberCompaction(int start, int length)
  957. {
  958. BasicNumberCompaction(bytes, start, length);
  959. }
  960. private void BasicNumberCompaction(byte[] input, int start, int length)
  961. {
  962. int ret = cwPtr;
  963. int retLast = length / 3;
  964. int ni, k;
  965. cwPtr += retLast + 1;
  966. for (k = 0; k <= retLast; ++k)
  967. codewords[ret + k] = 0;
  968. codewords[ret + retLast] = 1;
  969. length += start;
  970. for (ni = start; ni < length; ++ni)
  971. {
  972. // multiply by 10
  973. for (k = retLast; k >= 0; --k)
  974. codewords[ret + k] *= 10;
  975. // add the digit
  976. codewords[ret + retLast] += input[ni] - '0';
  977. // propagate carry
  978. for (k = retLast; k > 0; --k)
  979. {
  980. codewords[ret + k - 1] += codewords[ret + k] / 900;
  981. codewords[ret + k] %= 900;
  982. }
  983. }
  984. }
  985. private void NumberCompaction(byte[] input, int start, int length)
  986. {
  987. int full = (length / 44) * 15;
  988. int size = length % 44;
  989. int k;
  990. if (size == 0)
  991. size = full;
  992. else
  993. size = full + size / 3 + 1;
  994. if (size + cwPtr > MAX_DATA_CODEWORDS)
  995. {
  996. throw new ArgumentOutOfRangeException("The text is too big.");
  997. }
  998. length += start;
  999. for (k = start; k < length; k += 44)
  1000. {
  1001. size = length - k < 44 ? length - k : 44;
  1002. BasicNumberCompaction(input, k, size);
  1003. }
  1004. }
  1005. private void NumberCompaction(int start, int length)
  1006. {
  1007. NumberCompaction(bytes, start, length);
  1008. }
  1009. private void ByteCompaction6(int start)
  1010. {
  1011. int length = 6;
  1012. int ret = cwPtr;
  1013. int retLast = 4;
  1014. int ni, k;
  1015. cwPtr += retLast + 1;
  1016. for (k = 0; k <= retLast; ++k)
  1017. codewords[ret + k] = 0;
  1018. length += start;
  1019. for (ni = start; ni < length; ++ni)
  1020. {
  1021. // multiply by 256
  1022. for (k = retLast; k >= 0; --k)
  1023. codewords[ret + k] *= 256;
  1024. // add the digit
  1025. codewords[ret + retLast] += (int)bytes[ni] & 0xff;
  1026. // propagate carry
  1027. for (k = retLast; k > 0; --k)
  1028. {
  1029. codewords[ret + k - 1] += codewords[ret + k] / 900;
  1030. codewords[ret + k] %= 900;
  1031. }
  1032. }
  1033. }
  1034. private void ByteCompaction(int start, int length)
  1035. {
  1036. int k, j;
  1037. int size = (length / 6) * 5 + (length % 6);
  1038. if (size + cwPtr > MAX_DATA_CODEWORDS)
  1039. {
  1040. throw new ArgumentOutOfRangeException("The text is too big.");
  1041. }
  1042. length += start;
  1043. for (k = start; k < length; k += 6)
  1044. {
  1045. size = length - k < 44 ? length - k : 6;
  1046. if (size < 6)
  1047. {
  1048. for (j = 0; j < size; ++j)
  1049. codewords[cwPtr++] = (int)bytes[k + j] & 0xff;
  1050. }
  1051. else
  1052. {
  1053. ByteCompaction6(k);
  1054. }
  1055. }
  1056. }
  1057. private void BreakString()
  1058. {
  1059. int textLength = bytes.Length;
  1060. int lastP = 0;
  1061. int startN = 0;
  1062. int nd = 0;
  1063. char c = (char)0;
  1064. int k, j;
  1065. bool lastTxt, txt;
  1066. Segment v;
  1067. Segment vp;
  1068. Segment vn;
  1069. switch (CompactionMode)
  1070. {
  1071. case PDF417CompactionMode.Text:
  1072. segmentList.Add('T', 0, textLength);
  1073. return;
  1074. case PDF417CompactionMode.Numeric:
  1075. segmentList.Add('N', 0, textLength);
  1076. return;
  1077. case PDF417CompactionMode.Binary:
  1078. segmentList.Add('B', 0, textLength);
  1079. return;
  1080. }
  1081. for (k = 0; k < textLength; ++k)
  1082. {
  1083. c = (char)(bytes[k] & 0xff);
  1084. if (c >= '0' && c <= '9')
  1085. {
  1086. if (nd == 0)
  1087. startN = k;
  1088. ++nd;
  1089. continue;
  1090. }
  1091. if (nd >= 13)
  1092. {
  1093. if (lastP != startN)
  1094. {
  1095. c = (char)(bytes[lastP] & 0xff);
  1096. lastTxt = (c >= ' ' && c < 127) || c == '\r' || c == '\n' || c == '\t';
  1097. for (j = lastP; j < startN; ++j)
  1098. {
  1099. c = (char)(bytes[j] & 0xff);
  1100. txt = (c >= ' ' && c < 127) || c == '\r' || c == '\n' || c == '\t';
  1101. if (txt != lastTxt)
  1102. {
  1103. segmentList.Add(lastTxt ? 'T' : 'B', lastP, j);
  1104. lastP = j;
  1105. lastTxt = txt;
  1106. }
  1107. }
  1108. segmentList.Add(lastTxt ? 'T' : 'B', lastP, startN);
  1109. }
  1110. segmentList.Add('N', startN, k);
  1111. lastP = k;
  1112. }
  1113. nd = 0;
  1114. }
  1115. if (nd < 13)
  1116. startN = textLength;
  1117. if (lastP != startN)
  1118. {
  1119. c = (char)(bytes[lastP] & 0xff);
  1120. lastTxt = (c >= ' ' && c < 127) || c == '\r' || c == '\n' || c == '\t';
  1121. for (j = lastP; j < startN; ++j)
  1122. {
  1123. c = (char)(bytes[j] & 0xff);
  1124. txt = (c >= ' ' && c < 127) || c == '\r' || c == '\n' || c == '\t';
  1125. if (txt != lastTxt)
  1126. {
  1127. segmentList.Add(lastTxt ? 'T' : 'B', lastP, j);
  1128. lastP = j;
  1129. lastTxt = txt;
  1130. }
  1131. }
  1132. segmentList.Add(lastTxt ? 'T' : 'B', lastP, startN);
  1133. }
  1134. if (nd >= 13)
  1135. segmentList.Add('N', startN, textLength);
  1136. //optimize
  1137. //merge short binary
  1138. for (k = 0; k < segmentList.Size; ++k)
  1139. {
  1140. v = segmentList.Get(k);
  1141. vp = segmentList.Get(k - 1);
  1142. vn = segmentList.Get(k + 1);
  1143. if (CheckSegmentType(v, 'B') && GetSegmentLength(v) == 1)
  1144. {
  1145. if (CheckSegmentType(vp, 'T') && CheckSegmentType(vn, 'T')
  1146. && GetSegmentLength(vp) + GetSegmentLength(vn) >= 3)
  1147. {
  1148. vp.end = vn.end;
  1149. segmentList.Remove(k);
  1150. segmentList.Remove(k);
  1151. k = -1;
  1152. continue;
  1153. }
  1154. }
  1155. }
  1156. //merge text sections
  1157. for (k = 0; k < segmentList.Size; ++k)
  1158. {
  1159. v = segmentList.Get(k);
  1160. vp = segmentList.Get(k - 1);
  1161. vn = segmentList.Get(k + 1);
  1162. if (CheckSegmentType(v, 'T') && GetSegmentLength(v) >= 5)
  1163. {
  1164. bool redo = false;
  1165. if ((CheckSegmentType(vp, 'B') && GetSegmentLength(vp) == 1) || CheckSegmentType(vp, 'T'))
  1166. {
  1167. redo = true;
  1168. v.start = vp.start;
  1169. segmentList.Remove(k - 1);
  1170. --k;
  1171. }
  1172. if ((CheckSegmentType(vn, 'B') && GetSegmentLength(vn) == 1) || CheckSegmentType(vn, 'T'))
  1173. {
  1174. redo = true;
  1175. v.end = vn.end;
  1176. segmentList.Remove(k + 1);
  1177. }
  1178. if (redo)
  1179. {
  1180. k = -1;
  1181. continue;
  1182. }
  1183. }
  1184. }
  1185. //merge binary sections
  1186. for (k = 0; k < segmentList.Size; ++k)
  1187. {
  1188. v = segmentList.Get(k);
  1189. vp = segmentList.Get(k - 1);
  1190. vn = segmentList.Get(k + 1);
  1191. if (CheckSegmentType(v, 'B'))
  1192. {
  1193. bool redo = false;
  1194. if ((CheckSegmentType(vp, 'T') && GetSegmentLength(vp) < 5) || CheckSegmentType(vp, 'B'))
  1195. {
  1196. redo = true;
  1197. v.start = vp.start;
  1198. segmentList.Remove(k - 1);
  1199. --k;
  1200. }
  1201. if ((CheckSegmentType(vn, 'T') && GetSegmentLength(vn) < 5) || CheckSegmentType(vn, 'B'))
  1202. {
  1203. redo = true;
  1204. v.end = vn.end;
  1205. segmentList.Remove(k + 1);
  1206. }
  1207. if (redo)
  1208. {
  1209. k = -1;
  1210. continue;
  1211. }
  1212. }
  1213. }
  1214. // check if all numbers
  1215. if (segmentList.Size == 1 && (v = segmentList.Get(0)).type == 'T' && GetSegmentLength(v) >= 8)
  1216. {
  1217. for (k = v.start; k < v.end; ++k)
  1218. {
  1219. c = (char)(bytes[k] & 0xff);
  1220. if (c < '0' || c > '9')
  1221. break;
  1222. }
  1223. if (k == v.end)
  1224. v.type = 'N';
  1225. }
  1226. }
  1227. private void Assemble()
  1228. {
  1229. int k;
  1230. if (segmentList.Size == 0)
  1231. return;
  1232. cwPtr = 1;
  1233. for (k = 0; k < segmentList.Size; ++k)
  1234. {
  1235. Segment v = segmentList.Get(k);
  1236. switch (v.type)
  1237. {
  1238. case 'T':
  1239. if (k != 0)
  1240. codewords[cwPtr++] = TEXT_MODE;
  1241. TextCompaction(v.start, GetSegmentLength(v));
  1242. break;
  1243. case 'N':
  1244. codewords[cwPtr++] = NUMERIC_MODE;
  1245. NumberCompaction(v.start, GetSegmentLength(v));
  1246. break;
  1247. case 'B':
  1248. codewords[cwPtr++] = (GetSegmentLength(v) % 6) != 0 ? BYTE_MODE : BYTE_MODE_6;
  1249. ByteCompaction(v.start, GetSegmentLength(v));
  1250. break;
  1251. }
  1252. }
  1253. }
  1254. private static int MaxPossibleErrorLevel(int remain)
  1255. {
  1256. int level = 8;
  1257. int size = 512;
  1258. while (level > 0)
  1259. {
  1260. if (remain >= size)
  1261. return level;
  1262. --level;
  1263. size >>= 1;
  1264. }
  1265. return 0;
  1266. }
  1267. private int GetMaxSquare()
  1268. {
  1269. if (codeColumns > 21)
  1270. {
  1271. codeColumns = 29;
  1272. codeRows = 32;
  1273. }
  1274. else
  1275. {
  1276. codeColumns = 16;
  1277. codeRows = 58;
  1278. }
  1279. return MAX_DATA_CODEWORDS + 2;
  1280. }
  1281. /** Paints the barcode. If no exception was thrown a valid barcode is available. */
  1282. private void PaintCode()
  1283. {
  1284. int maxErr, lenErr, tot, pad;
  1285. if (bytes == null)
  1286. throw new ArgumentNullException("Text cannot be null.");
  1287. if (bytes.Length > ABSOLUTE_MAX_TEXT_SIZE)
  1288. {
  1289. throw new ArgumentOutOfRangeException("The text is too big.");
  1290. }
  1291. segmentList = new SegmentList();
  1292. BreakString();
  1293. Assemble();
  1294. segmentList = null;
  1295. codewords[0] = lenCodewords = cwPtr;
  1296. // error correction level
  1297. maxErr = MaxPossibleErrorLevel(MAX_DATA_CODEWORDS + 2 - lenCodewords);
  1298. if (ErrorCorrection == PDF417ErrorCorrection.Auto)
  1299. {
  1300. if (lenCodewords < 41)
  1301. errorLevel = 2;
  1302. else if (lenCodewords < 161)
  1303. errorLevel = 3;
  1304. else if (lenCodewords < 321)
  1305. errorLevel = 4;
  1306. else
  1307. errorLevel = 5;
  1308. }
  1309. else
  1310. errorLevel = (int)ErrorCorrection - 1;
  1311. if (errorLevel < 0)
  1312. errorLevel = 0;
  1313. else if (errorLevel > maxErr)
  1314. errorLevel = maxErr;
  1315. if (codeColumns < 1)
  1316. codeColumns = 1;
  1317. else if (codeColumns > 30)
  1318. codeColumns = 30;
  1319. if (codeRows < 3)
  1320. codeRows = 3;
  1321. else if (codeRows > 90)
  1322. codeRows = 90;
  1323. lenErr = 2 << errorLevel;
  1324. bool fixedColumn = rows == 0;
  1325. bool skipRowColAdjust = false;
  1326. tot = lenCodewords + lenErr;
  1327. if (columns != 0 && rows != 0)
  1328. {
  1329. tot = codeColumns * codeRows;
  1330. if (tot > MAX_DATA_CODEWORDS + 2)
  1331. {
  1332. tot = GetMaxSquare();
  1333. }
  1334. if (tot < lenCodewords + lenErr)
  1335. tot = lenCodewords + lenErr;
  1336. else
  1337. skipRowColAdjust = true;
  1338. }
  1339. else
  1340. {
  1341. double c, b;
  1342. fixedColumn = true;
  1343. if (aspectRatio < 0.001)
  1344. aspectRatio = 0.001f;
  1345. else if (aspectRatio > 1000)
  1346. aspectRatio = 1000;
  1347. float yHeight = PixelSize.Width == 0 ? 3 : (float)PixelSize.Height / (float)PixelSize.Width;
  1348. b = 73 * aspectRatio - 4;
  1349. c = (-b + Math.Sqrt(b * b + 4 * 17 * aspectRatio * (lenCodewords + lenErr) * yHeight)) / (2 * 17 * aspectRatio);
  1350. codeColumns = (int)(c + 0.5);
  1351. if (codeColumns < 1)
  1352. codeColumns = 1;
  1353. else if (codeColumns > 30)
  1354. codeColumns = 30;
  1355. }
  1356. if (!skipRowColAdjust)
  1357. {
  1358. if (fixedColumn)
  1359. {
  1360. codeRows = (tot - 1) / codeColumns + 1;
  1361. if (codeRows < 3)
  1362. codeRows = 3;
  1363. else if (codeRows > 90)
  1364. {
  1365. codeRows = 90;
  1366. codeColumns = (tot - 1) / 90 + 1;
  1367. }
  1368. }
  1369. else
  1370. {
  1371. codeColumns = (tot - 1) / codeRows + 1;
  1372. if (codeColumns > 30)
  1373. {
  1374. codeColumns = 30;
  1375. codeRows = (tot - 1) / 30 + 1;
  1376. }
  1377. }
  1378. tot = codeRows * codeColumns;
  1379. }
  1380. if (tot > MAX_DATA_CODEWORDS + 2)
  1381. {
  1382. tot = GetMaxSquare();
  1383. }
  1384. errorLevel = MaxPossibleErrorLevel(tot - lenCodewords);
  1385. lenErr = 2 << errorLevel;
  1386. pad = tot - lenErr - lenCodewords;
  1387. cwPtr = lenCodewords;
  1388. while (pad-- != 0)
  1389. codewords[cwPtr++] = TEXT_MODE;
  1390. codewords[0] = lenCodewords = cwPtr;
  1391. CalculateErrorCorrection(lenCodewords);
  1392. lenCodewords = tot;
  1393. OutPaintCode();
  1394. }
  1395. #endregion
  1396. #region Public Methods
  1397. /// <inheritdoc/>
  1398. public override void Assign(BarcodeBase source)
  1399. {
  1400. base.Assign(source);
  1401. BarcodePDF417 src = source as BarcodePDF417;
  1402. AspectRatio = src.AspectRatio;
  1403. Columns = src.Columns;
  1404. Rows = src.Rows;
  1405. CodePage = src.CodePage;
  1406. CompactionMode = src.CompactionMode;
  1407. ErrorCorrection = src.ErrorCorrection;
  1408. PixelSize = src.PixelSize;
  1409. }
  1410. internal override void Serialize(FastReport.Utils.FRWriter writer, string prefix, BarcodeBase diff)
  1411. {
  1412. base.Serialize(writer, prefix, diff);
  1413. BarcodePDF417 c = diff as BarcodePDF417;
  1414. if (c == null || AspectRatio != c.AspectRatio)
  1415. writer.WriteFloat(prefix + "AspectRatio", AspectRatio);
  1416. if (c == null || Columns != c.Columns)
  1417. writer.WriteInt(prefix + "Columns", Columns);
  1418. if (c == null || Rows != c.Rows)
  1419. writer.WriteInt(prefix + "Rows", Rows);
  1420. if (c == null || CodePage != c.CodePage)
  1421. writer.WriteInt(prefix + "CodePage", CodePage);
  1422. if (c == null || CompactionMode != c.CompactionMode)
  1423. writer.WriteValue(prefix + "CompactionMode", CompactionMode);
  1424. if (c == null || ErrorCorrection != c.ErrorCorrection)
  1425. writer.WriteValue(prefix + "ErrorCorrection", ErrorCorrection);
  1426. if (c == null || PixelSize != c.PixelSize)
  1427. writer.WriteValue(prefix + "PixelSize", PixelSize);
  1428. }
  1429. internal override void Initialize(string text, bool showText, int angle, float zoom)
  1430. {
  1431. base.Initialize(text, showText, angle, zoom);
  1432. if (String.IsNullOrEmpty(text))
  1433. bytes = new byte[0];
  1434. else
  1435. {
  1436. Encoding e = Encoding.GetEncoding(codePage);
  1437. bytes = e.GetBytes(text);
  1438. }
  1439. codeColumns = columns;
  1440. codeRows = rows;
  1441. PaintCode();
  1442. }
  1443. internal override SizeF CalcBounds()
  1444. {
  1445. int textAdd = showText ? 18 : 0;
  1446. return new SizeF(bitColumns * PixelSize.Width, codeRows * PixelSize.Height + textAdd);
  1447. }
  1448. internal override void Draw2DBarcode(IGraphics g, float kx, float ky)
  1449. {
  1450. Brush light = Brushes.White;
  1451. Brush dark = new SolidBrush(Color);
  1452. int stride = (bitColumns + 7) / 8;
  1453. for (int k = 0; k < codeRows; ++k)
  1454. {
  1455. int p = k * stride;
  1456. for (int j = 0; j < bitColumns; ++j)
  1457. {
  1458. int b = outBits[p + (j / 8)] & 0xff;
  1459. b <<= j % 8;
  1460. Brush brush = /*(b & 0x80) == 0 ? light :*/ dark;
  1461. if ((b & 0x80) != 0)
  1462. g.FillRectangle(brush, j * PixelSize.Width * kx, k * PixelSize.Height * ky,
  1463. PixelSize.Width * kx, PixelSize.Height * ky);
  1464. }
  1465. }
  1466. dark.Dispose();
  1467. }
  1468. #endregion
  1469. /// <summary>
  1470. /// Initializes a new instance of the <see cref="BarcodePDF417"/> class with default settings.
  1471. /// </summary>
  1472. public BarcodePDF417()
  1473. {
  1474. outBits = null;
  1475. bytes = new byte[0];
  1476. pixelSize = new Size(2, 8);
  1477. aspectRatio = 0.5f;
  1478. codePage = 437;
  1479. }
  1480. private class Segment
  1481. {
  1482. public char type;
  1483. public int start;
  1484. public int end;
  1485. public Segment(char type, int start, int end)
  1486. {
  1487. this.type = type;
  1488. this.start = start;
  1489. this.end = end;
  1490. }
  1491. }
  1492. private class SegmentList
  1493. {
  1494. protected ArrayList list = new ArrayList();
  1495. public void Add(char type, int start, int end)
  1496. {
  1497. list.Add(new Segment(type, start, end));
  1498. }
  1499. public Segment Get(int idx)
  1500. {
  1501. if (idx < 0 || idx >= list.Count)
  1502. return null;
  1503. return (Segment)list[idx];
  1504. }
  1505. public void Remove(int idx)
  1506. {
  1507. if (idx < 0 || idx >= list.Count)
  1508. return;
  1509. list.RemoveAt(idx);
  1510. }
  1511. public int Size
  1512. {
  1513. get
  1514. {
  1515. return list.Count;
  1516. }
  1517. }
  1518. }
  1519. }
  1520. }