QRCode.js 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129
  1. // 整改自 https://github.com/davidshimjs/qrcodejs
  2. import QRErrorCorrectLevel from "./QRErrorCorrectLevel.js";
  3. import QRCodeLimitLength from "./QRCodeLimitLength.js";
  4. import QRCodeModel from "./QRCodeModel.js";
  5. export default class QRCode {
  6. _htOption = {
  7. x: 0,
  8. y: 0,
  9. width: 256,
  10. height: 256,
  11. typeNumber: 4,
  12. colorDark: "#000000",
  13. colorLight: "#ffffff",
  14. correctLevel: QRErrorCorrectLevel.M
  15. };
  16. _oQRCode = null;
  17. _context = null;
  18. constructor(context, vOption) {
  19. if (typeof vOption === 'string') {
  20. vOption = {
  21. text: vOption
  22. };
  23. }
  24. if (vOption) {
  25. for (var i in vOption) {
  26. this._htOption[i] = vOption[i];
  27. }
  28. }
  29. this._context = context
  30. }
  31. async makeCode(sText) {
  32. this._oQRCode = new QRCodeModel(this._getTypeNumber(sText, this._htOption.correctLevel), this._htOption
  33. .correctLevel);
  34. this._oQRCode.addData(sText);
  35. this._oQRCode.make();
  36. return await this.makeImage(true);
  37. }
  38. async calcCode(sText) {
  39. this._oQRCode = new QRCodeModel(this._getTypeNumber(sText, this._htOption.correctLevel), this._htOption
  40. .correctLevel);
  41. this._oQRCode.addData(sText);
  42. this._oQRCode.make();
  43. return await this.makeImage(false);
  44. }
  45. /**
  46. * Get the type by string length
  47. *
  48. * @private
  49. * @param {String} sText
  50. * @param {Number} nCorrectLevel
  51. * @return {Number} type
  52. */
  53. _getTypeNumber(sText, nCorrectLevel) {
  54. var nType = 1;
  55. var length = QRCode._getUTF8Length(sText);
  56. for (var i = 0, len = QRCodeLimitLength.length; i <= len; i++) {
  57. var nLimit = 0;
  58. switch (nCorrectLevel) {
  59. case QRErrorCorrectLevel.L:
  60. nLimit = QRCodeLimitLength[i][0];
  61. break;
  62. case QRErrorCorrectLevel.M:
  63. nLimit = QRCodeLimitLength[i][1];
  64. break;
  65. case QRErrorCorrectLevel.Q:
  66. nLimit = QRCodeLimitLength[i][2];
  67. break;
  68. case QRErrorCorrectLevel.H:
  69. nLimit = QRCodeLimitLength[i][3];
  70. break;
  71. }
  72. if (length <= nLimit) {
  73. break;
  74. } else {
  75. nType++;
  76. }
  77. }
  78. if (nType > QRCodeLimitLength.length) {
  79. throw new Error("Too long data");
  80. }
  81. return nType;
  82. }
  83. static _getUTF8Length(sText) {
  84. var replacedText = encodeURI(sText).toString().replace(/\%[0-9a-fA-F]{2}/g, 'a');
  85. return replacedText.length + (replacedText.length != sText ? 3 : 0);
  86. }
  87. async makeImage(needDraw) {
  88. var _oContext = this._context
  89. var _htOption = this._htOption;
  90. var oQRCode = this._oQRCode
  91. var padding = 0;
  92. if (_htOption.padding)
  93. padding = _htOption.padding;
  94. var nCount = oQRCode.getModuleCount();
  95. var nWidth = Math.floor((_htOption.width - 2 * padding) / nCount);
  96. var nHeight = Math.floor((_htOption.height - 2 * padding) / nCount);
  97. if (needDraw) {
  98. let list = [];
  99. for (var row = 0; row < nCount; row++) {
  100. for (var col = 0; col < nCount; col++) {
  101. var nLeft = col * nWidth + padding + _htOption.x;
  102. var nTop = row * nHeight + padding + _htOption.y;
  103. list.push({
  104. fillStyle: oQRCode.isDark(row, col) ? _htOption.colorDark : _htOption.colorLight,
  105. x: nLeft,
  106. y: nTop,
  107. w: nWidth,
  108. h: nHeight
  109. });
  110. }
  111. }
  112. await _oContext.fillRectList(list);
  113. }
  114. return {
  115. width: nWidth * nCount + 2 * padding,
  116. height: nHeight * nCount + 2 * padding
  117. }
  118. }
  119. }