qrcode.js

The original author jeromeetienne https://github.com/jeromeetienne/jquery-qrcode

此脚本不应直接安装,它是供其他脚本使用的外部库。如果你需要使用该库,请在脚本元属性加入:// @require https://update.cn-greasyfork.org/scripts/31884/208862/qrcodejs.js

  1. // ==UserScript==
  2. // @license MIT License
  3. // ==/UserScript==
  4.  
  5. //---------------------------------------------------------------------
  6. // QRCode for JavaScript
  7. //
  8. // Copyright (c) 2009 Kazuhiko Arase
  9. //
  10. // URL: http://www.d-project.com/
  11. //
  12. // Licensed under the MIT license:
  13. // http://www.opensource.org/licenses/mit-license.php
  14. //
  15. // The word "QR Code" is registered trademark of
  16. // DENSO WAVE INCORPORATED
  17. // http://www.denso-wave.com/qrcode/faqpatent-e.html
  18. //
  19. //---------------------------------------------------------------------
  20.  
  21. //---------------------------------------------------------------------
  22. // QR8bitByte
  23. //---------------------------------------------------------------------
  24.  
  25. function QR8bitByte(data) {
  26. this.mode = QRMode.MODE_8BIT_BYTE;
  27. this.data = data;
  28. }
  29.  
  30. QR8bitByte.prototype = {
  31.  
  32. getLength : function(buffer) {
  33. return this.data.length;
  34. },
  35. write : function(buffer) {
  36. for (var i = 0; i < this.data.length; i++) {
  37. // not JIS ...
  38. buffer.put(this.data.charCodeAt(i), 8);
  39. }
  40. }
  41. };
  42.  
  43. //---------------------------------------------------------------------
  44. // QRCode
  45. //---------------------------------------------------------------------
  46.  
  47. function QRCode(typeNumber, errorCorrectLevel) {
  48. this.typeNumber = typeNumber;
  49. this.errorCorrectLevel = errorCorrectLevel;
  50. this.modules = null;
  51. this.moduleCount = 0;
  52. this.dataCache = null;
  53. this.dataList = new Array();
  54. }
  55.  
  56. QRCode.prototype = {
  57. addData : function(data) {
  58. var newData = new QR8bitByte(data);
  59. this.dataList.push(newData);
  60. this.dataCache = null;
  61. },
  62. isDark : function(row, col) {
  63. if (row < 0 || this.moduleCount <= row || col < 0 || this.moduleCount <= col) {
  64. throw new Error(row + "," + col);
  65. }
  66. return this.modules[row][col];
  67. },
  68.  
  69. getModuleCount : function() {
  70. return this.moduleCount;
  71. },
  72. make : function() {
  73. // Calculate automatically typeNumber if provided is < 1
  74. if (this.typeNumber < 1 ){
  75. var typeNumber = 1;
  76. for (typeNumber = 1; typeNumber < 40; typeNumber++) {
  77. var rsBlocks = QRRSBlock.getRSBlocks(typeNumber, this.errorCorrectLevel);
  78.  
  79. var buffer = new QRBitBuffer();
  80. var totalDataCount = 0;
  81. for (var i = 0; i < rsBlocks.length; i++) {
  82. totalDataCount += rsBlocks[i].dataCount;
  83. }
  84.  
  85. for (var i = 0; i < this.dataList.length; i++) {
  86. var data = this.dataList[i];
  87. buffer.put(data.mode, 4);
  88. buffer.put(data.getLength(), QRUtil.getLengthInBits(data.mode, typeNumber) );
  89. data.write(buffer);
  90. }
  91. if (buffer.getLengthInBits() <= totalDataCount * 8)
  92. break;
  93. }
  94. this.typeNumber = typeNumber;
  95. }
  96. this.makeImpl(false, this.getBestMaskPattern() );
  97. },
  98. makeImpl : function(test, maskPattern) {
  99. this.moduleCount = this.typeNumber * 4 + 17;
  100. this.modules = new Array(this.moduleCount);
  101. for (var row = 0; row < this.moduleCount; row++) {
  102. this.modules[row] = new Array(this.moduleCount);
  103. for (var col = 0; col < this.moduleCount; col++) {
  104. this.modules[row][col] = null;//(col + row) % 3;
  105. }
  106. }
  107. this.setupPositionProbePattern(0, 0);
  108. this.setupPositionProbePattern(this.moduleCount - 7, 0);
  109. this.setupPositionProbePattern(0, this.moduleCount - 7);
  110. this.setupPositionAdjustPattern();
  111. this.setupTimingPattern();
  112. this.setupTypeInfo(test, maskPattern);
  113. if (this.typeNumber >= 7) {
  114. this.setupTypeNumber(test);
  115. }
  116. if (this.dataCache == null) {
  117. this.dataCache = QRCode.createData(this.typeNumber, this.errorCorrectLevel, this.dataList);
  118. }
  119. this.mapData(this.dataCache, maskPattern);
  120. },
  121.  
  122. setupPositionProbePattern : function(row, col) {
  123. for (var r = -1; r <= 7; r++) {
  124. if (row + r <= -1 || this.moduleCount <= row + r) continue;
  125. for (var c = -1; c <= 7; c++) {
  126. if (col + c <= -1 || this.moduleCount <= col + c) continue;
  127. if ( (0 <= r && r <= 6 && (c == 0 || c == 6) )
  128. || (0 <= c && c <= 6 && (r == 0 || r == 6) )
  129. || (2 <= r && r <= 4 && 2 <= c && c <= 4) ) {
  130. this.modules[row + r][col + c] = true;
  131. } else {
  132. this.modules[row + r][col + c] = false;
  133. }
  134. }
  135. }
  136. },
  137. getBestMaskPattern : function() {
  138. var minLostPoint = 0;
  139. var pattern = 0;
  140. for (var i = 0; i < 8; i++) {
  141. this.makeImpl(true, i);
  142. var lostPoint = QRUtil.getLostPoint(this);
  143. if (i == 0 || minLostPoint > lostPoint) {
  144. minLostPoint = lostPoint;
  145. pattern = i;
  146. }
  147. }
  148. return pattern;
  149. },
  150. createMovieClip : function(target_mc, instance_name, depth) {
  151. var qr_mc = target_mc.createEmptyMovieClip(instance_name, depth);
  152. var cs = 1;
  153. this.make();
  154.  
  155. for (var row = 0; row < this.modules.length; row++) {
  156. var y = row * cs;
  157. for (var col = 0; col < this.modules[row].length; col++) {
  158. var x = col * cs;
  159. var dark = this.modules[row][col];
  160. if (dark) {
  161. qr_mc.beginFill(0, 100);
  162. qr_mc.moveTo(x, y);
  163. qr_mc.lineTo(x + cs, y);
  164. qr_mc.lineTo(x + cs, y + cs);
  165. qr_mc.lineTo(x, y + cs);
  166. qr_mc.endFill();
  167. }
  168. }
  169. }
  170. return qr_mc;
  171. },
  172.  
  173. setupTimingPattern : function() {
  174. for (var r = 8; r < this.moduleCount - 8; r++) {
  175. if (this.modules[r][6] != null) {
  176. continue;
  177. }
  178. this.modules[r][6] = (r % 2 == 0);
  179. }
  180. for (var c = 8; c < this.moduleCount - 8; c++) {
  181. if (this.modules[6][c] != null) {
  182. continue;
  183. }
  184. this.modules[6][c] = (c % 2 == 0);
  185. }
  186. },
  187. setupPositionAdjustPattern : function() {
  188. var pos = QRUtil.getPatternPosition(this.typeNumber);
  189. for (var i = 0; i < pos.length; i++) {
  190. for (var j = 0; j < pos.length; j++) {
  191. var row = pos[i];
  192. var col = pos[j];
  193. if (this.modules[row][col] != null) {
  194. continue;
  195. }
  196. for (var r = -2; r <= 2; r++) {
  197. for (var c = -2; c <= 2; c++) {
  198. if (r == -2 || r == 2 || c == -2 || c == 2
  199. || (r == 0 && c == 0) ) {
  200. this.modules[row + r][col + c] = true;
  201. } else {
  202. this.modules[row + r][col + c] = false;
  203. }
  204. }
  205. }
  206. }
  207. }
  208. },
  209. setupTypeNumber : function(test) {
  210. var bits = QRUtil.getBCHTypeNumber(this.typeNumber);
  211. for (var i = 0; i < 18; i++) {
  212. var mod = (!test && ( (bits >> i) & 1) == 1);
  213. this.modules[Math.floor(i / 3)][i % 3 + this.moduleCount - 8 - 3] = mod;
  214. }
  215. for (var i = 0; i < 18; i++) {
  216. var mod = (!test && ( (bits >> i) & 1) == 1);
  217. this.modules[i % 3 + this.moduleCount - 8 - 3][Math.floor(i / 3)] = mod;
  218. }
  219. },
  220. setupTypeInfo : function(test, maskPattern) {
  221. var data = (this.errorCorrectLevel << 3) | maskPattern;
  222. var bits = QRUtil.getBCHTypeInfo(data);
  223. // vertical
  224. for (var i = 0; i < 15; i++) {
  225. var mod = (!test && ( (bits >> i) & 1) == 1);
  226. if (i < 6) {
  227. this.modules[i][8] = mod;
  228. } else if (i < 8) {
  229. this.modules[i + 1][8] = mod;
  230. } else {
  231. this.modules[this.moduleCount - 15 + i][8] = mod;
  232. }
  233. }
  234. // horizontal
  235. for (var i = 0; i < 15; i++) {
  236. var mod = (!test && ( (bits >> i) & 1) == 1);
  237. if (i < 8) {
  238. this.modules[8][this.moduleCount - i - 1] = mod;
  239. } else if (i < 9) {
  240. this.modules[8][15 - i - 1 + 1] = mod;
  241. } else {
  242. this.modules[8][15 - i - 1] = mod;
  243. }
  244. }
  245. // fixed module
  246. this.modules[this.moduleCount - 8][8] = (!test);
  247. },
  248. mapData : function(data, maskPattern) {
  249. var inc = -1;
  250. var row = this.moduleCount - 1;
  251. var bitIndex = 7;
  252. var byteIndex = 0;
  253. for (var col = this.moduleCount - 1; col > 0; col -= 2) {
  254. if (col == 6) col--;
  255. while (true) {
  256. for (var c = 0; c < 2; c++) {
  257. if (this.modules[row][col - c] == null) {
  258. var dark = false;
  259. if (byteIndex < data.length) {
  260. dark = ( ( (data[byteIndex] >>> bitIndex) & 1) == 1);
  261. }
  262. var mask = QRUtil.getMask(maskPattern, row, col - c);
  263. if (mask) {
  264. dark = !dark;
  265. }
  266. this.modules[row][col - c] = dark;
  267. bitIndex--;
  268. if (bitIndex == -1) {
  269. byteIndex++;
  270. bitIndex = 7;
  271. }
  272. }
  273. }
  274. row += inc;
  275. if (row < 0 || this.moduleCount <= row) {
  276. row -= inc;
  277. inc = -inc;
  278. break;
  279. }
  280. }
  281. }
  282. }
  283.  
  284. };
  285.  
  286. QRCode.PAD0 = 0xEC;
  287. QRCode.PAD1 = 0x11;
  288.  
  289. QRCode.createData = function(typeNumber, errorCorrectLevel, dataList) {
  290. var rsBlocks = QRRSBlock.getRSBlocks(typeNumber, errorCorrectLevel);
  291. var buffer = new QRBitBuffer();
  292. for (var i = 0; i < dataList.length; i++) {
  293. var data = dataList[i];
  294. buffer.put(data.mode, 4);
  295. buffer.put(data.getLength(), QRUtil.getLengthInBits(data.mode, typeNumber) );
  296. data.write(buffer);
  297. }
  298.  
  299. // calc num max data.
  300. var totalDataCount = 0;
  301. for (var i = 0; i < rsBlocks.length; i++) {
  302. totalDataCount += rsBlocks[i].dataCount;
  303. }
  304.  
  305. if (buffer.getLengthInBits() > totalDataCount * 8) {
  306. throw new Error("code length overflow. ("
  307. + buffer.getLengthInBits()
  308. + ">"
  309. + totalDataCount * 8
  310. + ")");
  311. }
  312.  
  313. // end code
  314. if (buffer.getLengthInBits() + 4 <= totalDataCount * 8) {
  315. buffer.put(0, 4);
  316. }
  317.  
  318. // padding
  319. while (buffer.getLengthInBits() % 8 != 0) {
  320. buffer.putBit(false);
  321. }
  322.  
  323. // padding
  324. while (true) {
  325. if (buffer.getLengthInBits() >= totalDataCount * 8) {
  326. break;
  327. }
  328. buffer.put(QRCode.PAD0, 8);
  329. if (buffer.getLengthInBits() >= totalDataCount * 8) {
  330. break;
  331. }
  332. buffer.put(QRCode.PAD1, 8);
  333. }
  334.  
  335. return QRCode.createBytes(buffer, rsBlocks);
  336. }
  337.  
  338. QRCode.createBytes = function(buffer, rsBlocks) {
  339.  
  340. var offset = 0;
  341. var maxDcCount = 0;
  342. var maxEcCount = 0;
  343. var dcdata = new Array(rsBlocks.length);
  344. var ecdata = new Array(rsBlocks.length);
  345. for (var r = 0; r < rsBlocks.length; r++) {
  346.  
  347. var dcCount = rsBlocks[r].dataCount;
  348. var ecCount = rsBlocks[r].totalCount - dcCount;
  349.  
  350. maxDcCount = Math.max(maxDcCount, dcCount);
  351. maxEcCount = Math.max(maxEcCount, ecCount);
  352. dcdata[r] = new Array(dcCount);
  353. for (var i = 0; i < dcdata[r].length; i++) {
  354. dcdata[r][i] = 0xff & buffer.buffer[i + offset];
  355. }
  356. offset += dcCount;
  357. var rsPoly = QRUtil.getErrorCorrectPolynomial(ecCount);
  358. var rawPoly = new QRPolynomial(dcdata[r], rsPoly.getLength() - 1);
  359.  
  360. var modPoly = rawPoly.mod(rsPoly);
  361. ecdata[r] = new Array(rsPoly.getLength() - 1);
  362. for (var i = 0; i < ecdata[r].length; i++) {
  363. var modIndex = i + modPoly.getLength() - ecdata[r].length;
  364. ecdata[r][i] = (modIndex >= 0)? modPoly.get(modIndex) : 0;
  365. }
  366.  
  367. }
  368. var totalCodeCount = 0;
  369. for (var i = 0; i < rsBlocks.length; i++) {
  370. totalCodeCount += rsBlocks[i].totalCount;
  371. }
  372.  
  373. var data = new Array(totalCodeCount);
  374. var index = 0;
  375.  
  376. for (var i = 0; i < maxDcCount; i++) {
  377. for (var r = 0; r < rsBlocks.length; r++) {
  378. if (i < dcdata[r].length) {
  379. data[index++] = dcdata[r][i];
  380. }
  381. }
  382. }
  383.  
  384. for (var i = 0; i < maxEcCount; i++) {
  385. for (var r = 0; r < rsBlocks.length; r++) {
  386. if (i < ecdata[r].length) {
  387. data[index++] = ecdata[r][i];
  388. }
  389. }
  390. }
  391.  
  392. return data;
  393.  
  394. }
  395.  
  396. //---------------------------------------------------------------------
  397. // QRMode
  398. //---------------------------------------------------------------------
  399.  
  400. var QRMode = {
  401. MODE_NUMBER : 1 << 0,
  402. MODE_ALPHA_NUM : 1 << 1,
  403. MODE_8BIT_BYTE : 1 << 2,
  404. MODE_KANJI : 1 << 3
  405. };
  406.  
  407. //---------------------------------------------------------------------
  408. // QRErrorCorrectLevel
  409. //---------------------------------------------------------------------
  410. var QRErrorCorrectLevel = {
  411. L : 1,
  412. M : 0,
  413. Q : 3,
  414. H : 2
  415. };
  416.  
  417. //---------------------------------------------------------------------
  418. // QRMaskPattern
  419. //---------------------------------------------------------------------
  420.  
  421. var QRMaskPattern = {
  422. PATTERN000 : 0,
  423. PATTERN001 : 1,
  424. PATTERN010 : 2,
  425. PATTERN011 : 3,
  426. PATTERN100 : 4,
  427. PATTERN101 : 5,
  428. PATTERN110 : 6,
  429. PATTERN111 : 7
  430. };
  431.  
  432. //---------------------------------------------------------------------
  433. // QRUtil
  434. //---------------------------------------------------------------------
  435. var QRUtil = {
  436.  
  437. PATTERN_POSITION_TABLE : [
  438. [],
  439. [6, 18],
  440. [6, 22],
  441. [6, 26],
  442. [6, 30],
  443. [6, 34],
  444. [6, 22, 38],
  445. [6, 24, 42],
  446. [6, 26, 46],
  447. [6, 28, 50],
  448. [6, 30, 54],
  449. [6, 32, 58],
  450. [6, 34, 62],
  451. [6, 26, 46, 66],
  452. [6, 26, 48, 70],
  453. [6, 26, 50, 74],
  454. [6, 30, 54, 78],
  455. [6, 30, 56, 82],
  456. [6, 30, 58, 86],
  457. [6, 34, 62, 90],
  458. [6, 28, 50, 72, 94],
  459. [6, 26, 50, 74, 98],
  460. [6, 30, 54, 78, 102],
  461. [6, 28, 54, 80, 106],
  462. [6, 32, 58, 84, 110],
  463. [6, 30, 58, 86, 114],
  464. [6, 34, 62, 90, 118],
  465. [6, 26, 50, 74, 98, 122],
  466. [6, 30, 54, 78, 102, 126],
  467. [6, 26, 52, 78, 104, 130],
  468. [6, 30, 56, 82, 108, 134],
  469. [6, 34, 60, 86, 112, 138],
  470. [6, 30, 58, 86, 114, 142],
  471. [6, 34, 62, 90, 118, 146],
  472. [6, 30, 54, 78, 102, 126, 150],
  473. [6, 24, 50, 76, 102, 128, 154],
  474. [6, 28, 54, 80, 106, 132, 158],
  475. [6, 32, 58, 84, 110, 136, 162],
  476. [6, 26, 54, 82, 110, 138, 166],
  477. [6, 30, 58, 86, 114, 142, 170]
  478. ],
  479.  
  480. G15 : (1 << 10) | (1 << 8) | (1 << 5) | (1 << 4) | (1 << 2) | (1 << 1) | (1 << 0),
  481. G18 : (1 << 12) | (1 << 11) | (1 << 10) | (1 << 9) | (1 << 8) | (1 << 5) | (1 << 2) | (1 << 0),
  482. G15_MASK : (1 << 14) | (1 << 12) | (1 << 10) | (1 << 4) | (1 << 1),
  483.  
  484. getBCHTypeInfo : function(data) {
  485. var d = data << 10;
  486. while (QRUtil.getBCHDigit(d) - QRUtil.getBCHDigit(QRUtil.G15) >= 0) {
  487. d ^= (QRUtil.G15 << (QRUtil.getBCHDigit(d) - QRUtil.getBCHDigit(QRUtil.G15) ) );
  488. }
  489. return ( (data << 10) | d) ^ QRUtil.G15_MASK;
  490. },
  491.  
  492. getBCHTypeNumber : function(data) {
  493. var d = data << 12;
  494. while (QRUtil.getBCHDigit(d) - QRUtil.getBCHDigit(QRUtil.G18) >= 0) {
  495. d ^= (QRUtil.G18 << (QRUtil.getBCHDigit(d) - QRUtil.getBCHDigit(QRUtil.G18) ) );
  496. }
  497. return (data << 12) | d;
  498. },
  499.  
  500. getBCHDigit : function(data) {
  501.  
  502. var digit = 0;
  503.  
  504. while (data != 0) {
  505. digit++;
  506. data >>>= 1;
  507. }
  508.  
  509. return digit;
  510. },
  511.  
  512. getPatternPosition : function(typeNumber) {
  513. return QRUtil.PATTERN_POSITION_TABLE[typeNumber - 1];
  514. },
  515.  
  516. getMask : function(maskPattern, i, j) {
  517. switch (maskPattern) {
  518. case QRMaskPattern.PATTERN000 : return (i + j) % 2 == 0;
  519. case QRMaskPattern.PATTERN001 : return i % 2 == 0;
  520. case QRMaskPattern.PATTERN010 : return j % 3 == 0;
  521. case QRMaskPattern.PATTERN011 : return (i + j) % 3 == 0;
  522. case QRMaskPattern.PATTERN100 : return (Math.floor(i / 2) + Math.floor(j / 3) ) % 2 == 0;
  523. case QRMaskPattern.PATTERN101 : return (i * j) % 2 + (i * j) % 3 == 0;
  524. case QRMaskPattern.PATTERN110 : return ( (i * j) % 2 + (i * j) % 3) % 2 == 0;
  525. case QRMaskPattern.PATTERN111 : return ( (i * j) % 3 + (i + j) % 2) % 2 == 0;
  526.  
  527. default :
  528. throw new Error("bad maskPattern:" + maskPattern);
  529. }
  530. },
  531.  
  532. getErrorCorrectPolynomial : function(errorCorrectLength) {
  533.  
  534. var a = new QRPolynomial([1], 0);
  535.  
  536. for (var i = 0; i < errorCorrectLength; i++) {
  537. a = a.multiply(new QRPolynomial([1, QRMath.gexp(i)], 0) );
  538. }
  539.  
  540. return a;
  541. },
  542.  
  543. getLengthInBits : function(mode, type) {
  544.  
  545. if (1 <= type && type < 10) {
  546.  
  547. // 1 - 9
  548.  
  549. switch(mode) {
  550. case QRMode.MODE_NUMBER : return 10;
  551. case QRMode.MODE_ALPHA_NUM : return 9;
  552. case QRMode.MODE_8BIT_BYTE : return 8;
  553. case QRMode.MODE_KANJI : return 8;
  554. default :
  555. throw new Error("mode:" + mode);
  556. }
  557.  
  558. } else if (type < 27) {
  559.  
  560. // 10 - 26
  561.  
  562. switch(mode) {
  563. case QRMode.MODE_NUMBER : return 12;
  564. case QRMode.MODE_ALPHA_NUM : return 11;
  565. case QRMode.MODE_8BIT_BYTE : return 16;
  566. case QRMode.MODE_KANJI : return 10;
  567. default :
  568. throw new Error("mode:" + mode);
  569. }
  570.  
  571. } else if (type < 41) {
  572.  
  573. // 27 - 40
  574.  
  575. switch(mode) {
  576. case QRMode.MODE_NUMBER : return 14;
  577. case QRMode.MODE_ALPHA_NUM : return 13;
  578. case QRMode.MODE_8BIT_BYTE : return 16;
  579. case QRMode.MODE_KANJI : return 12;
  580. default :
  581. throw new Error("mode:" + mode);
  582. }
  583.  
  584. } else {
  585. throw new Error("type:" + type);
  586. }
  587. },
  588.  
  589. getLostPoint : function(qrCode) {
  590. var moduleCount = qrCode.getModuleCount();
  591. var lostPoint = 0;
  592. // LEVEL1
  593. for (var row = 0; row < moduleCount; row++) {
  594.  
  595. for (var col = 0; col < moduleCount; col++) {
  596.  
  597. var sameCount = 0;
  598. var dark = qrCode.isDark(row, col);
  599.  
  600. for (var r = -1; r <= 1; r++) {
  601.  
  602. if (row + r < 0 || moduleCount <= row + r) {
  603. continue;
  604. }
  605.  
  606. for (var c = -1; c <= 1; c++) {
  607.  
  608. if (col + c < 0 || moduleCount <= col + c) {
  609. continue;
  610. }
  611.  
  612. if (r == 0 && c == 0) {
  613. continue;
  614. }
  615.  
  616. if (dark == qrCode.isDark(row + r, col + c) ) {
  617. sameCount++;
  618. }
  619. }
  620. }
  621.  
  622. if (sameCount > 5) {
  623. lostPoint += (3 + sameCount - 5);
  624. }
  625. }
  626. }
  627.  
  628. // LEVEL2
  629.  
  630. for (var row = 0; row < moduleCount - 1; row++) {
  631. for (var col = 0; col < moduleCount - 1; col++) {
  632. var count = 0;
  633. if (qrCode.isDark(row, col ) ) count++;
  634. if (qrCode.isDark(row + 1, col ) ) count++;
  635. if (qrCode.isDark(row, col + 1) ) count++;
  636. if (qrCode.isDark(row + 1, col + 1) ) count++;
  637. if (count == 0 || count == 4) {
  638. lostPoint += 3;
  639. }
  640. }
  641. }
  642.  
  643. // LEVEL3
  644.  
  645. for (var row = 0; row < moduleCount; row++) {
  646. for (var col = 0; col < moduleCount - 6; col++) {
  647. if (qrCode.isDark(row, col)
  648. && !qrCode.isDark(row, col + 1)
  649. && qrCode.isDark(row, col + 2)
  650. && qrCode.isDark(row, col + 3)
  651. && qrCode.isDark(row, col + 4)
  652. && !qrCode.isDark(row, col + 5)
  653. && qrCode.isDark(row, col + 6) ) {
  654. lostPoint += 40;
  655. }
  656. }
  657. }
  658.  
  659. for (var col = 0; col < moduleCount; col++) {
  660. for (var row = 0; row < moduleCount - 6; row++) {
  661. if (qrCode.isDark(row, col)
  662. && !qrCode.isDark(row + 1, col)
  663. && qrCode.isDark(row + 2, col)
  664. && qrCode.isDark(row + 3, col)
  665. && qrCode.isDark(row + 4, col)
  666. && !qrCode.isDark(row + 5, col)
  667. && qrCode.isDark(row + 6, col) ) {
  668. lostPoint += 40;
  669. }
  670. }
  671. }
  672.  
  673. // LEVEL4
  674. var darkCount = 0;
  675.  
  676. for (var col = 0; col < moduleCount; col++) {
  677. for (var row = 0; row < moduleCount; row++) {
  678. if (qrCode.isDark(row, col) ) {
  679. darkCount++;
  680. }
  681. }
  682. }
  683. var ratio = Math.abs(100 * darkCount / moduleCount / moduleCount - 50) / 5;
  684. lostPoint += ratio * 10;
  685.  
  686. return lostPoint;
  687. }
  688.  
  689. };
  690.  
  691.  
  692. //---------------------------------------------------------------------
  693. // QRMath
  694. //---------------------------------------------------------------------
  695.  
  696. var QRMath = {
  697.  
  698. glog : function(n) {
  699. if (n < 1) {
  700. throw new Error("glog(" + n + ")");
  701. }
  702. return QRMath.LOG_TABLE[n];
  703. },
  704. gexp : function(n) {
  705. while (n < 0) {
  706. n += 255;
  707. }
  708. while (n >= 256) {
  709. n -= 255;
  710. }
  711. return QRMath.EXP_TABLE[n];
  712. },
  713. EXP_TABLE : new Array(256),
  714. LOG_TABLE : new Array(256)
  715.  
  716. };
  717. for (var i = 0; i < 8; i++) {
  718. QRMath.EXP_TABLE[i] = 1 << i;
  719. }
  720. for (var i = 8; i < 256; i++) {
  721. QRMath.EXP_TABLE[i] = QRMath.EXP_TABLE[i - 4]
  722. ^ QRMath.EXP_TABLE[i - 5]
  723. ^ QRMath.EXP_TABLE[i - 6]
  724. ^ QRMath.EXP_TABLE[i - 8];
  725. }
  726. for (var i = 0; i < 255; i++) {
  727. QRMath.LOG_TABLE[QRMath.EXP_TABLE[i] ] = i;
  728. }
  729.  
  730. //---------------------------------------------------------------------
  731. // QRPolynomial
  732. //---------------------------------------------------------------------
  733.  
  734. function QRPolynomial(num, shift) {
  735.  
  736. if (num.length == undefined) {
  737. throw new Error(num.length + "/" + shift);
  738. }
  739.  
  740. var offset = 0;
  741.  
  742. while (offset < num.length && num[offset] == 0) {
  743. offset++;
  744. }
  745.  
  746. this.num = new Array(num.length - offset + shift);
  747. for (var i = 0; i < num.length - offset; i++) {
  748. this.num[i] = num[i + offset];
  749. }
  750. }
  751.  
  752. QRPolynomial.prototype = {
  753.  
  754. get : function(index) {
  755. return this.num[index];
  756. },
  757. getLength : function() {
  758. return this.num.length;
  759. },
  760. multiply : function(e) {
  761. var num = new Array(this.getLength() + e.getLength() - 1);
  762. for (var i = 0; i < this.getLength(); i++) {
  763. for (var j = 0; j < e.getLength(); j++) {
  764. num[i + j] ^= QRMath.gexp(QRMath.glog(this.get(i) ) + QRMath.glog(e.get(j) ) );
  765. }
  766. }
  767. return new QRPolynomial(num, 0);
  768. },
  769. mod : function(e) {
  770. if (this.getLength() - e.getLength() < 0) {
  771. return this;
  772. }
  773. var ratio = QRMath.glog(this.get(0) ) - QRMath.glog(e.get(0) );
  774. var num = new Array(this.getLength() );
  775. for (var i = 0; i < this.getLength(); i++) {
  776. num[i] = this.get(i);
  777. }
  778. for (var i = 0; i < e.getLength(); i++) {
  779. num[i] ^= QRMath.gexp(QRMath.glog(e.get(i) ) + ratio);
  780. }
  781. // recursive call
  782. return new QRPolynomial(num, 0).mod(e);
  783. }
  784. };
  785.  
  786. //---------------------------------------------------------------------
  787. // QRRSBlock
  788. //---------------------------------------------------------------------
  789.  
  790. function QRRSBlock(totalCount, dataCount) {
  791. this.totalCount = totalCount;
  792. this.dataCount = dataCount;
  793. }
  794.  
  795. QRRSBlock.RS_BLOCK_TABLE = [
  796.  
  797. // L
  798. // M
  799. // Q
  800. // H
  801.  
  802. // 1
  803. [1, 26, 19],
  804. [1, 26, 16],
  805. [1, 26, 13],
  806. [1, 26, 9],
  807. // 2
  808. [1, 44, 34],
  809. [1, 44, 28],
  810. [1, 44, 22],
  811. [1, 44, 16],
  812.  
  813. // 3
  814. [1, 70, 55],
  815. [1, 70, 44],
  816. [2, 35, 17],
  817. [2, 35, 13],
  818.  
  819. // 4
  820. [1, 100, 80],
  821. [2, 50, 32],
  822. [2, 50, 24],
  823. [4, 25, 9],
  824. // 5
  825. [1, 134, 108],
  826. [2, 67, 43],
  827. [2, 33, 15, 2, 34, 16],
  828. [2, 33, 11, 2, 34, 12],
  829. // 6
  830. [2, 86, 68],
  831. [4, 43, 27],
  832. [4, 43, 19],
  833. [4, 43, 15],
  834. // 7
  835. [2, 98, 78],
  836. [4, 49, 31],
  837. [2, 32, 14, 4, 33, 15],
  838. [4, 39, 13, 1, 40, 14],
  839. // 8
  840. [2, 121, 97],
  841. [2, 60, 38, 2, 61, 39],
  842. [4, 40, 18, 2, 41, 19],
  843. [4, 40, 14, 2, 41, 15],
  844. // 9
  845. [2, 146, 116],
  846. [3, 58, 36, 2, 59, 37],
  847. [4, 36, 16, 4, 37, 17],
  848. [4, 36, 12, 4, 37, 13],
  849. // 10
  850. [2, 86, 68, 2, 87, 69],
  851. [4, 69, 43, 1, 70, 44],
  852. [6, 43, 19, 2, 44, 20],
  853. [6, 43, 15, 2, 44, 16],
  854.  
  855. // 11
  856. [4, 101, 81],
  857. [1, 80, 50, 4, 81, 51],
  858. [4, 50, 22, 4, 51, 23],
  859. [3, 36, 12, 8, 37, 13],
  860.  
  861. // 12
  862. [2, 116, 92, 2, 117, 93],
  863. [6, 58, 36, 2, 59, 37],
  864. [4, 46, 20, 6, 47, 21],
  865. [7, 42, 14, 4, 43, 15],
  866.  
  867. // 13
  868. [4, 133, 107],
  869. [8, 59, 37, 1, 60, 38],
  870. [8, 44, 20, 4, 45, 21],
  871. [12, 33, 11, 4, 34, 12],
  872.  
  873. // 14
  874. [3, 145, 115, 1, 146, 116],
  875. [4, 64, 40, 5, 65, 41],
  876. [11, 36, 16, 5, 37, 17],
  877. [11, 36, 12, 5, 37, 13],
  878.  
  879. // 15
  880. [5, 109, 87, 1, 110, 88],
  881. [5, 65, 41, 5, 66, 42],
  882. [5, 54, 24, 7, 55, 25],
  883. [11, 36, 12],
  884.  
  885. // 16
  886. [5, 122, 98, 1, 123, 99],
  887. [7, 73, 45, 3, 74, 46],
  888. [15, 43, 19, 2, 44, 20],
  889. [3, 45, 15, 13, 46, 16],
  890.  
  891. // 17
  892. [1, 135, 107, 5, 136, 108],
  893. [10, 74, 46, 1, 75, 47],
  894. [1, 50, 22, 15, 51, 23],
  895. [2, 42, 14, 17, 43, 15],
  896.  
  897. // 18
  898. [5, 150, 120, 1, 151, 121],
  899. [9, 69, 43, 4, 70, 44],
  900. [17, 50, 22, 1, 51, 23],
  901. [2, 42, 14, 19, 43, 15],
  902.  
  903. // 19
  904. [3, 141, 113, 4, 142, 114],
  905. [3, 70, 44, 11, 71, 45],
  906. [17, 47, 21, 4, 48, 22],
  907. [9, 39, 13, 16, 40, 14],
  908.  
  909. // 20
  910. [3, 135, 107, 5, 136, 108],
  911. [3, 67, 41, 13, 68, 42],
  912. [15, 54, 24, 5, 55, 25],
  913. [15, 43, 15, 10, 44, 16],
  914.  
  915. // 21
  916. [4, 144, 116, 4, 145, 117],
  917. [17, 68, 42],
  918. [17, 50, 22, 6, 51, 23],
  919. [19, 46, 16, 6, 47, 17],
  920.  
  921. // 22
  922. [2, 139, 111, 7, 140, 112],
  923. [17, 74, 46],
  924. [7, 54, 24, 16, 55, 25],
  925. [34, 37, 13],
  926.  
  927. // 23
  928. [4, 151, 121, 5, 152, 122],
  929. [4, 75, 47, 14, 76, 48],
  930. [11, 54, 24, 14, 55, 25],
  931. [16, 45, 15, 14, 46, 16],
  932.  
  933. // 24
  934. [6, 147, 117, 4, 148, 118],
  935. [6, 73, 45, 14, 74, 46],
  936. [11, 54, 24, 16, 55, 25],
  937. [30, 46, 16, 2, 47, 17],
  938.  
  939. // 25
  940. [8, 132, 106, 4, 133, 107],
  941. [8, 75, 47, 13, 76, 48],
  942. [7, 54, 24, 22, 55, 25],
  943. [22, 45, 15, 13, 46, 16],
  944.  
  945. // 26
  946. [10, 142, 114, 2, 143, 115],
  947. [19, 74, 46, 4, 75, 47],
  948. [28, 50, 22, 6, 51, 23],
  949. [33, 46, 16, 4, 47, 17],
  950.  
  951. // 27
  952. [8, 152, 122, 4, 153, 123],
  953. [22, 73, 45, 3, 74, 46],
  954. [8, 53, 23, 26, 54, 24],
  955. [12, 45, 15, 28, 46, 16],
  956.  
  957. // 28
  958. [3, 147, 117, 10, 148, 118],
  959. [3, 73, 45, 23, 74, 46],
  960. [4, 54, 24, 31, 55, 25],
  961. [11, 45, 15, 31, 46, 16],
  962.  
  963. // 29
  964. [7, 146, 116, 7, 147, 117],
  965. [21, 73, 45, 7, 74, 46],
  966. [1, 53, 23, 37, 54, 24],
  967. [19, 45, 15, 26, 46, 16],
  968.  
  969. // 30
  970. [5, 145, 115, 10, 146, 116],
  971. [19, 75, 47, 10, 76, 48],
  972. [15, 54, 24, 25, 55, 25],
  973. [23, 45, 15, 25, 46, 16],
  974.  
  975. // 31
  976. [13, 145, 115, 3, 146, 116],
  977. [2, 74, 46, 29, 75, 47],
  978. [42, 54, 24, 1, 55, 25],
  979. [23, 45, 15, 28, 46, 16],
  980.  
  981. // 32
  982. [17, 145, 115],
  983. [10, 74, 46, 23, 75, 47],
  984. [10, 54, 24, 35, 55, 25],
  985. [19, 45, 15, 35, 46, 16],
  986.  
  987. // 33
  988. [17, 145, 115, 1, 146, 116],
  989. [14, 74, 46, 21, 75, 47],
  990. [29, 54, 24, 19, 55, 25],
  991. [11, 45, 15, 46, 46, 16],
  992.  
  993. // 34
  994. [13, 145, 115, 6, 146, 116],
  995. [14, 74, 46, 23, 75, 47],
  996. [44, 54, 24, 7, 55, 25],
  997. [59, 46, 16, 1, 47, 17],
  998.  
  999. // 35
  1000. [12, 151, 121, 7, 152, 122],
  1001. [12, 75, 47, 26, 76, 48],
  1002. [39, 54, 24, 14, 55, 25],
  1003. [22, 45, 15, 41, 46, 16],
  1004.  
  1005. // 36
  1006. [6, 151, 121, 14, 152, 122],
  1007. [6, 75, 47, 34, 76, 48],
  1008. [46, 54, 24, 10, 55, 25],
  1009. [2, 45, 15, 64, 46, 16],
  1010.  
  1011. // 37
  1012. [17, 152, 122, 4, 153, 123],
  1013. [29, 74, 46, 14, 75, 47],
  1014. [49, 54, 24, 10, 55, 25],
  1015. [24, 45, 15, 46, 46, 16],
  1016.  
  1017. // 38
  1018. [4, 152, 122, 18, 153, 123],
  1019. [13, 74, 46, 32, 75, 47],
  1020. [48, 54, 24, 14, 55, 25],
  1021. [42, 45, 15, 32, 46, 16],
  1022.  
  1023. // 39
  1024. [20, 147, 117, 4, 148, 118],
  1025. [40, 75, 47, 7, 76, 48],
  1026. [43, 54, 24, 22, 55, 25],
  1027. [10, 45, 15, 67, 46, 16],
  1028.  
  1029. // 40
  1030. [19, 148, 118, 6, 149, 119],
  1031. [18, 75, 47, 31, 76, 48],
  1032. [34, 54, 24, 34, 55, 25],
  1033. [20, 45, 15, 61, 46, 16]
  1034. ];
  1035.  
  1036. QRRSBlock.getRSBlocks = function(typeNumber, errorCorrectLevel) {
  1037. var rsBlock = QRRSBlock.getRsBlockTable(typeNumber, errorCorrectLevel);
  1038. if (rsBlock == undefined) {
  1039. throw new Error("bad rs block @ typeNumber:" + typeNumber + "/errorCorrectLevel:" + errorCorrectLevel);
  1040. }
  1041.  
  1042. var length = rsBlock.length / 3;
  1043. var list = new Array();
  1044. for (var i = 0; i < length; i++) {
  1045.  
  1046. var count = rsBlock[i * 3 + 0];
  1047. var totalCount = rsBlock[i * 3 + 1];
  1048. var dataCount = rsBlock[i * 3 + 2];
  1049.  
  1050. for (var j = 0; j < count; j++) {
  1051. list.push(new QRRSBlock(totalCount, dataCount) );
  1052. }
  1053. }
  1054. return list;
  1055. }
  1056.  
  1057. QRRSBlock.getRsBlockTable = function(typeNumber, errorCorrectLevel) {
  1058.  
  1059. switch(errorCorrectLevel) {
  1060. case QRErrorCorrectLevel.L :
  1061. return QRRSBlock.RS_BLOCK_TABLE[(typeNumber - 1) * 4 + 0];
  1062. case QRErrorCorrectLevel.M :
  1063. return QRRSBlock.RS_BLOCK_TABLE[(typeNumber - 1) * 4 + 1];
  1064. case QRErrorCorrectLevel.Q :
  1065. return QRRSBlock.RS_BLOCK_TABLE[(typeNumber - 1) * 4 + 2];
  1066. case QRErrorCorrectLevel.H :
  1067. return QRRSBlock.RS_BLOCK_TABLE[(typeNumber - 1) * 4 + 3];
  1068. default :
  1069. return undefined;
  1070. }
  1071. }
  1072.  
  1073. //---------------------------------------------------------------------
  1074. // QRBitBuffer
  1075. //---------------------------------------------------------------------
  1076.  
  1077. function QRBitBuffer() {
  1078. this.buffer = new Array();
  1079. this.length = 0;
  1080. }
  1081.  
  1082. QRBitBuffer.prototype = {
  1083.  
  1084. get : function(index) {
  1085. var bufIndex = Math.floor(index / 8);
  1086. return ( (this.buffer[bufIndex] >>> (7 - index % 8) ) & 1) == 1;
  1087. },
  1088. put : function(num, length) {
  1089. for (var i = 0; i < length; i++) {
  1090. this.putBit( ( (num >>> (length - i - 1) ) & 1) == 1);
  1091. }
  1092. },
  1093. getLengthInBits : function() {
  1094. return this.length;
  1095. },
  1096. putBit : function(bit) {
  1097. var bufIndex = Math.floor(this.length / 8);
  1098. if (this.buffer.length <= bufIndex) {
  1099. this.buffer.push(0);
  1100. }
  1101. if (bit) {
  1102. this.buffer[bufIndex] |= (0x80 >>> (this.length % 8) );
  1103. }
  1104. this.length++;
  1105. }
  1106. };