jquery-qrcode.js 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415
  1. const WIN = window; // eslint-disable-line
  2. const JQ = WIN.jQuery;
  3. const is_img_el = x => x && typeof x.tagName === 'string' && x.tagName.toUpperCase() === 'IMG';
  4. // Wrapper for the original QR code generator.
  5. const create_qrcode = (text, level, version, quiet) => {
  6. const qr = {};
  7. const qr_gen = require('qrcode-generator');
  8. qr_gen.stringToBytes = qr_gen.stringToBytesFuncs['UTF-8'];
  9. const vqr = qr_gen(version, level);
  10. vqr.addData(text);
  11. vqr.make();
  12. quiet = quiet || 0;
  13. const module_count = vqr.getModuleCount();
  14. const quiet_module_count = module_count + 2 * quiet;
  15. const is_dark = (row, col) => {
  16. row -= quiet;
  17. col -= quiet;
  18. return row >= 0 && row < module_count && col >= 0 && col < module_count && vqr.isDark(row, col);
  19. };
  20. const add_blank = (l, t, r, b) => {
  21. const prev_is_dark = qr.is_dark;
  22. const module_size = 1 / quiet_module_count;
  23. qr.is_dark = (row, col) => {
  24. const ml = col * module_size;
  25. const mt = row * module_size;
  26. const mr = ml + module_size;
  27. const mb = mt + module_size;
  28. return prev_is_dark(row, col) && (l > mr || ml > r || t > mb || mt > b);
  29. };
  30. };
  31. qr.text = text;
  32. qr.level = level;
  33. qr.version = version;
  34. qr.module_count = quiet_module_count;
  35. qr.is_dark = is_dark;
  36. qr.add_blank = add_blank;
  37. return qr;
  38. };
  39. // Returns a minimal QR code for the given text starting with version `min_ver`.
  40. // Returns `undefined` if `text` is too long to be encoded in `max_ver`.
  41. const create_min_qrcode = (text, level, min_ver, max_ver, quiet) => {
  42. min_ver = Math.max(1, min_ver || 1);
  43. max_ver = Math.min(40, max_ver || 40);
  44. for (let ver = min_ver; ver <= max_ver; ver += 1) {
  45. try {
  46. return create_qrcode(text, level, ver, quiet);
  47. } catch (err) {/* empty */}
  48. }
  49. return undefined;
  50. };
  51. const draw_background_label = (qr, ctx, settings) => {
  52. const size = settings.size;
  53. const font = 'bold ' + settings.mSize * size + 'px ' + settings.fontname;
  54. const tmp_ctx = JQ('<canvas/>')[0].getContext('2d');
  55. tmp_ctx.font = font;
  56. const w = tmp_ctx.measureText(settings.label).width;
  57. const sh = settings.mSize;
  58. const sw = w / size;
  59. const sl = (1 - sw) * settings.mPosX;
  60. const st = (1 - sh) * settings.mPosY;
  61. const sr = sl + sw;
  62. const sb = st + sh;
  63. const pad = 0.01;
  64. if (settings.mode === 1) {
  65. // Strip
  66. qr.add_blank(0, st - pad, size, sb + pad);
  67. } else {
  68. // Box
  69. qr.add_blank(sl - pad, st - pad, sr + pad, sb + pad);
  70. }
  71. ctx.fillStyle = settings.fontcolor;
  72. ctx.font = font;
  73. ctx.fillText(settings.label, sl * size, st * size + 0.75 * settings.mSize * size);
  74. };
  75. const draw_background_img = (qr, ctx, settings) => {
  76. const size = settings.size;
  77. const w = settings.image.naturalWidth || 1;
  78. const h = settings.image.naturalHeight || 1;
  79. const sh = settings.mSize;
  80. const sw = sh * w / h;
  81. const sl = (1 - sw) * settings.mPosX;
  82. const st = (1 - sh) * settings.mPosY;
  83. const sr = sl + sw;
  84. const sb = st + sh;
  85. const pad = 0.01;
  86. if (settings.mode === 3) {
  87. // Strip
  88. qr.add_blank(0, st - pad, size, sb + pad);
  89. } else {
  90. // Box
  91. qr.add_blank(sl - pad, st - pad, sr + pad, sb + pad);
  92. }
  93. ctx.drawImage(settings.image, sl * size, st * size, sw * size, sh * size);
  94. };
  95. const draw_background = (qr, ctx, settings) => {
  96. if (is_img_el(settings.background)) {
  97. ctx.drawImage(settings.background, 0, 0, settings.size, settings.size);
  98. } else if (settings.background) {
  99. ctx.fillStyle = settings.background;
  100. ctx.fillRect(settings.left, settings.top, settings.size, settings.size);
  101. }
  102. const mode = settings.mode;
  103. if (mode === 1 || mode === 2) {
  104. draw_background_label(qr, ctx, settings);
  105. } else if (is_img_el(settings.image) && (mode === 3 || mode === 4)) {
  106. draw_background_img(qr, ctx, settings);
  107. }
  108. };
  109. const draw_modules_default = (qr, ctx, settings, left, top, width, row, col) => {
  110. if (qr.is_dark(row, col)) {
  111. ctx.r(left, top, width, width);
  112. }
  113. };
  114. const draw_modules_rounded_dark = (ctx, l, t, r, b, rad, nw, ne, se, sw) => {
  115. if (nw) {
  116. ctx.m(l + rad, t);
  117. } else {
  118. ctx.m(l, t);
  119. }
  120. if (ne) {
  121. ctx.l(r - rad, t).a(r, t, r, b, rad);
  122. } else {
  123. ctx.l(r, t);
  124. }
  125. if (se) {
  126. ctx.l(r, b - rad).a(r, b, l, b, rad);
  127. } else {
  128. ctx.l(r, b);
  129. }
  130. if (sw) {
  131. ctx.l(l + rad, b).a(l, b, l, t, rad);
  132. } else {
  133. ctx.l(l, b);
  134. }
  135. if (nw) {
  136. ctx.l(l, t + rad).a(l, t, r, t, rad);
  137. } else {
  138. ctx.l(l, t);
  139. }
  140. };
  141. const draw_modules_rounded_light = (ctx, l, t, r, b, rad, nw, ne, se, sw) => {
  142. if (nw) {
  143. ctx.m(l + rad, t).l(l, t).l(l, t + rad).a(l, t, l + rad, t, rad);
  144. }
  145. if (ne) {
  146. ctx.m(r - rad, t).l(r, t).l(r, t + rad).a(r, t, r - rad, t, rad);
  147. }
  148. if (se) {
  149. ctx.m(r - rad, b).l(r, b).l(r, b - rad).a(r, b, r - rad, b, rad);
  150. }
  151. if (sw) {
  152. ctx.m(l + rad, b).l(l, b).l(l, b - rad).a(l, b, l + rad, b, rad);
  153. }
  154. };
  155. const draw_modules_rounded = (qr, ctx, settings, left, top, width, row, col) => {
  156. const right = left + width;
  157. const bottom = top + width;
  158. const rad = settings.radius * width;
  159. const row_n = row - 1;
  160. const row_s = row + 1;
  161. const col_w = col - 1;
  162. const col_e = col + 1;
  163. const is_dark = qr.is_dark;
  164. const d_center = is_dark(row, col);
  165. const d_nw = is_dark(row_n, col_w);
  166. const d_n = is_dark(row_n, col);
  167. const d_ne = is_dark(row_n, col_e);
  168. const d_e = is_dark(row, col_e);
  169. const d_se = is_dark(row_s, col_e);
  170. const d_s = is_dark(row_s, col);
  171. const d_sw = is_dark(row_s, col_w);
  172. const d_w = is_dark(row, col_w);
  173. if (d_center) {
  174. draw_modules_rounded_dark(ctx, left, top, right, bottom, rad, !d_n && !d_w, !d_n && !d_e, !d_s && !d_e, !d_s && !d_w);
  175. } else {
  176. draw_modules_rounded_light(ctx, left, top, right, bottom, rad, d_n && d_w && d_nw, d_n && d_e && d_ne, d_s && d_e && d_se, d_s && d_w && d_sw);
  177. }
  178. };
  179. const draw_modules = (qr, ctx, settings) => {
  180. const module_count = qr.module_count;
  181. const module_size = settings.size / module_count;
  182. let fn = draw_modules_default;
  183. if (settings.radius > 0 && settings.radius <= 0.5) {
  184. fn = draw_modules_rounded;
  185. }
  186. const draw_ctx = {
  187. m(x, y) {ctx.moveTo(x, y); return draw_ctx;},
  188. l(x, y) {ctx.lineTo(x, y); return draw_ctx;},
  189. a(...args) {ctx.arcTo(...args); return draw_ctx;},
  190. r(...args) {ctx.rect(...args); return draw_ctx;}
  191. };
  192. ctx.beginPath();
  193. for (let row = 0; row < module_count; row += 1) {
  194. for (let col = 0; col < module_count; col += 1) {
  195. const l = settings.left + col * module_size;
  196. const t = settings.top + row * module_size;
  197. const w = module_size;
  198. fn(qr, draw_ctx, settings, l, t, w, row, col);
  199. }
  200. }
  201. if (is_img_el(settings.fill)) {
  202. ctx.strokeStyle = 'rgba(0,0,0,0.5)';
  203. ctx.lineWidth = 2;
  204. ctx.stroke();
  205. const prev = ctx.globalCompositeOperation;
  206. ctx.globalCompositeOperation = 'destination-out';
  207. ctx.fill();
  208. ctx.globalCompositeOperation = prev;
  209. ctx.clip();
  210. ctx.drawImage(settings.fill, 0, 0, settings.size, settings.size);
  211. ctx.restore();
  212. } else {
  213. ctx.fillStyle = settings.fill;
  214. ctx.fill();
  215. }
  216. };
  217. // Draws QR code to the given `canvas` and returns it.
  218. const draw_on_canvas = (canvas, settings) => {
  219. const qr = create_min_qrcode(settings.text, settings.ecLevel, settings.minVersion, settings.maxVersion, settings.quiet);
  220. if (!qr) {
  221. return null;
  222. }
  223. const $canvas = JQ(canvas).data('qrcode', qr);
  224. const ctx = $canvas[0].getContext('2d');
  225. draw_background(qr, ctx, settings);
  226. draw_modules(qr, ctx, settings);
  227. return $canvas;
  228. };
  229. // Returns a `canvas` element representing the QR code for the given settings.
  230. const create_canvas = settings => {
  231. const $canvas = JQ('<canvas/>').attr('width', settings.size).attr('height', settings.size);
  232. return draw_on_canvas($canvas, settings);
  233. };
  234. // Returns an `image` element representing the QR code for the given settings.
  235. const create_img = settings => {
  236. return JQ('<img/>').attr('src', create_canvas(settings)[0].toDataURL('image/png'));
  237. };
  238. // Returns a `div` element representing the QR code for the given settings.
  239. const create_div = settings => {
  240. const qr = create_min_qrcode(settings.text, settings.ecLevel, settings.minVersion, settings.maxVersion, settings.quiet);
  241. if (!qr) {
  242. return null;
  243. }
  244. // some shortcuts to improve compression
  245. const settings_size = settings.size;
  246. const settings_bgColor = settings.background;
  247. const math_floor = Math.floor;
  248. const module_count = qr.module_count;
  249. const module_size = math_floor(settings_size / module_count);
  250. const offset = math_floor(0.5 * (settings_size - module_size * module_count));
  251. const container_css = {
  252. position: 'relative',
  253. left: 0,
  254. top: 0,
  255. padding: 0,
  256. margin: 0,
  257. width: settings_size,
  258. height: settings_size
  259. };
  260. const dark_css = {
  261. position: 'absolute',
  262. padding: 0,
  263. margin: 0,
  264. width: module_size,
  265. height: module_size,
  266. 'background-color': settings.fill
  267. };
  268. const $div = JQ('<div/>').data('qrcode', qr).css(container_css);
  269. if (settings_bgColor) {
  270. $div.css('background-color', settings_bgColor);
  271. }
  272. for (let row = 0; row < module_count; row += 1) {
  273. for (let col = 0; col < module_count; col += 1) {
  274. if (qr.is_dark(row, col)) {
  275. JQ('<div/>')
  276. .css(dark_css)
  277. .css({
  278. left: offset + col * module_size,
  279. top: offset + row * module_size
  280. })
  281. .appendTo($div);
  282. }
  283. }
  284. }
  285. return $div;
  286. };
  287. const create_html = settings => {
  288. if (settings.render === 'canvas') {
  289. return create_canvas(settings);
  290. } else if (settings.render === 'image') {
  291. return create_img(settings);
  292. }
  293. return create_div(settings);
  294. };
  295. const DEFAULTS = {
  296. // render method: `'canvas'`, `'image'` or `'div'`
  297. render: 'canvas',
  298. // version range somewhere in 1 .. 40
  299. minVersion: 1,
  300. maxVersion: 40,
  301. // error correction level: `'L'`, `'M'`, `'Q'` or `'H'`
  302. ecLevel: 'L',
  303. // offset in pixel if drawn onto existing canvas
  304. left: 0,
  305. top: 0,
  306. // size in pixel
  307. size: 200,
  308. // code color or image element
  309. fill: '#000',
  310. // background color or image element, `null` for transparent background
  311. background: '#fff',
  312. // content
  313. text: 'no text',
  314. // corner radius relative to module width: 0.0 .. 0.5
  315. radius: 0,
  316. // quiet zone in modules
  317. quiet: 0,
  318. // modes
  319. // 0: normal
  320. // 1: label strip
  321. // 2: label box
  322. // 3: image strip
  323. // 4: image box
  324. mode: 0,
  325. mSize: 0.1,
  326. mPosX: 0.5,
  327. mPosY: 0.5,
  328. label: 'no label',
  329. fontname: 'sans',
  330. fontcolor: '#000',
  331. image: null
  332. };
  333. JQ.fn.qrcode = module.exports = function main(options) {
  334. const settings = JQ.extend({}, DEFAULTS, options);
  335. return this.each((idx, el) => {
  336. if (el.nodeName.toLowerCase() === 'canvas') {
  337. draw_on_canvas(el, settings);
  338. } else {
  339. JQ(el).append(create_html(settings));
  340. }
  341. });
  342. };