Compare commits

..

No commits in common. 'master' and 'v0.15.0' have entirely different histories.

@ -12,7 +12,7 @@ insert_final_newline = true
trim_trailing_whitespace = true trim_trailing_whitespace = true
[{*.json,*.yml}] [{package.json,.eslintrc,.babelrc,.travis.yml}]
indent_size = 2 indent_size = 2

@ -1,4 +1,5 @@
/build/ /build/
/coverage/ /coverage/
/dist/ /dist/
/vendor/
/node_modules/ /node_modules/

@ -5,8 +5,29 @@
es6: true es6: true
node: true node: true
parserOptions: ecmaFeatures:
ecmaVersion: 2020 arrowFunctions: true
binaryLiterals: true
blockBindings: true
classes: false
defaultParams: true
destructuring: true
forOf: true
generators: true
globalReturn: true
jsx: false
modules: true
objectLiteralComputedProperties: true
objectLiteralDuplicateProperties: true
objectLiteralShorthandMethods: true
objectLiteralShorthandProperties: true
octalLiterals: true
regexUFlag: true
regexYFlag: true
spread: true
superInFunctions: false
templateStrings: true
unicodeCodePointEscapes: true
rules: rules:
array-bracket-spacing: [2, never] array-bracket-spacing: [2, never]
@ -41,7 +62,7 @@
max-depth: [1, 4] max-depth: [1, 4]
max-len: [0, 80, 4] max-len: [0, 80, 4]
max-nested-callbacks: [1, 3] max-nested-callbacks: [1, 3]
max-params: [1, 16] ### max-params: [1, 5] ###
max-statements: [1, 32] ### max-statements: [1, 32] ###
new-cap: 0 new-cap: 0
new-parens: 2 new-parens: 2

@ -3,14 +3,12 @@
[![license][license-img]][github] [![web][web-img]][web] [![github][github-img]][github] [![license][license-img]][github] [![web][web-img]][web] [![github][github-img]][github]
jQuery plugin to dynamically generate QR codes. Uses [QR Code Generator][qrcode] (MIT). jQuery plugin to dynamically generate QR codes. Uses [QR Code Generator][qrcode] (MIT).
There is a newer lib named [kjua][kjua] that works in all modern browsers
with crisp codes on all devices.
## License ## License
The MIT License (MIT) The MIT License (MIT)
Copyright (c) 2020 Lars Jung (https://larsjung.de) Copyright (c) 2019 Lars Jung (https://larsjung.de)
Permission is hereby granted, free of charge, to any person obtaining a copy Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal of this software and associated documentation files (the "Software"), to deal
@ -35,8 +33,7 @@ THE SOFTWARE.
[github]: https://github.com/lrsjng/jquery-qrcode [github]: https://github.com/lrsjng/jquery-qrcode
[license-img]: https://img.shields.io/badge/license-MIT-a0a060.svg?style=flat-square [license-img]: https://img.shields.io/badge/license-MIT-a0a060.svg?style=flat-square
[web-img]: https://img.shields.io/badge/web-larsjung.de/jquery--qrcode-a0a060.svg?style=flat-square [web-img]: https://img.shields.io/badge/web-larsjung.de/qrcode-a0a060.svg?style=flat-square
[github-img]: https://img.shields.io/badge/github-lrsjng/jquery--qrcode-a0a060.svg?style=flat-square [github-img]: https://img.shields.io/badge/github-lrsjng/jquery--qrcode-a0a060.svg?style=flat-square
[qrcode]: https://github.com/kazuhikoarase/qrcode-generator [qrcode]: https://github.com/kazuhikoarase/qrcode-generator
[kjua]: https://larsjung.de/kjua/

1201
dist/jquery-qrcode.js vendored

File diff suppressed because it is too large Load Diff

File diff suppressed because one or more lines are too long

@ -1,11 +1,11 @@
const {resolve, join} = require('path'); const {resolve, join} = require('path');
const {ghu, pug, jszip, mapfn, read, remove, webpack, uglify, wrap, write} = require('ghu'); const {ghu, includeit, pug, jszip, mapfn, read, remove, uglify, wrap, write} = require('ghu');
const overwrite = arg => write(arg, {overwrite: true});
const NAME = 'jquery-qrcode'; const NAME = 'jquery-qrcode';
const ROOT = resolve(__dirname); const ROOT = resolve(__dirname);
const SRC = join(ROOT, 'src'); const SRC = join(ROOT, 'src');
const VENDOR = join(ROOT, 'vendor');
const BUILD = join(ROOT, 'build'); const BUILD = join(ROOT, 'build');
const DIST = join(ROOT, 'dist'); const DIST = join(ROOT, 'dist');
@ -15,52 +15,54 @@ ghu.before(runtime => {
runtime.pkg = Object.assign({}, require('./package.json')); runtime.pkg = Object.assign({}, require('./package.json'));
runtime.comment = `${runtime.pkg.name} v${runtime.pkg.version} - ${runtime.pkg.homepage}`; runtime.comment = `${runtime.pkg.name} v${runtime.pkg.version} - ${runtime.pkg.homepage}`;
runtime.commentJs = `/*! ${runtime.comment} */\n`; runtime.commentJs = `/*! ${runtime.comment} */\n`;
console.log(runtime.comment); console.log(runtime.comment);
}); });
ghu.task('clean', () => { ghu.task('clean', 'delete build folder', () => {
return remove(`${BUILD}, ${DIST}`); return remove(`${BUILD}, ${DIST}`);
}); });
ghu.task('build:scripts', runtime => { ghu.task('build:scripts', runtime => {
return read(`${SRC}/${NAME}.js`) return read(`${SRC}/${NAME}.js`)
.then(webpack(webpack.cfg_umd(NAME, [SRC]))) .then(includeit())
.then(wrap(runtime.commentJs)) .then(wrap(runtime.commentJs))
.then(overwrite(`${DIST}/${NAME}.js`)) .then(write(`${DIST}/${NAME}.js`, {overwrite: true}))
.then(overwrite(`${BUILD}/${NAME}-${runtime.pkg.version}.js`)) .then(write(`${BUILD}/${NAME}-${runtime.pkg.version}.js`, {overwrite: true}))
.then(uglify()) .then(uglify({compressor: {warnings: false}}))
.then(wrap(runtime.commentJs)) .then(wrap(runtime.commentJs))
.then(overwrite(`${DIST}/${NAME}.min.js`)) .then(write(`${DIST}/${NAME}.min.js`, {overwrite: true}))
.then(overwrite(`${BUILD}/${NAME}-${runtime.pkg.version}.min.js`)); .then(write(`${BUILD}/${NAME}-${runtime.pkg.version}.min.js`, {overwrite: true}));
}); });
ghu.task('build:other', runtime => { ghu.task('build:demo', runtime => {
return Promise.all([ return Promise.all([
read(`${ROOT}/*.md`)
.then(overwrite(mapfn.p(ROOT, BUILD))),
read(`${SRC}/demo/*.pug`) read(`${SRC}/demo/*.pug`)
.then(pug({pkg: runtime.pkg})) .then(pug({pkg: runtime.pkg}))
.then(overwrite(mapfn.p(SRC, BUILD).s('.pug', ''))), .then(write(mapfn.p(SRC, BUILD).s('.pug', ''), {overwrite: true})),
read(`${SRC}/demo/*.js`) read(`${SRC}/demo/*, !**/*.pug`)
.then(webpack(webpack.cfg([SRC]))) .then(write(mapfn.p(SRC, BUILD), {overwrite: true})),
.then(uglify())
.then(wrap(runtime.commentJs))
.then(overwrite(mapfn.p(SRC, BUILD))),
read(`${SRC}/demo/*, !**/*.pug, !**/*.js`)
.then(overwrite(mapfn.p(SRC, BUILD))),
read(`${ROOT}/node_modules/jquery/dist/jquery.min.js`) read(`${ROOT}/node_modules/jquery/dist/jquery.min.js`)
.then(overwrite(`${BUILD}/demo/jquery.min.js`)) .then(write(`${BUILD}/demo/jquery.min.js`, {overwrite: true}))
]);
});
ghu.task('build:copy', () => {
return Promise.all([
read(`${VENDOR}/demo/*`)
.then(write(mapfn.p(VENDOR, BUILD), {overwrite: true})),
read(`${ROOT}/*.md`)
.then(write(mapfn.p(ROOT, BUILD), {overwrite: true}))
]); ]);
}); });
ghu.task('build', ['build:scripts', 'build:other']); ghu.task('build', ['build:scripts', 'build:demo', 'build:copy']);
ghu.task('zip', ['build'], runtime => { ghu.task('zip', ['build'], runtime => {
return read(`${BUILD}/**/*`) return read(`${BUILD}/**/*`)
.then(jszip({dir: BUILD, level: 9})) .then(jszip({dir: BUILD, level: 9}))
.then(overwrite(`${BUILD}/${NAME}-${runtime.pkg.version}.zip`)); .then(write(`${BUILD}/${NAME}-${runtime.pkg.version}.zip`, {overwrite: true}));
}); });
ghu.task('release', ['clean', 'zip']); ghu.task('release', ['clean', 'zip']);

7584
package-lock.json generated

File diff suppressed because it is too large Load Diff

@ -1,14 +1,15 @@
{ {
"name": "jquery-qrcode", "name": "jquery-qrcode",
"title": "jQuery.qrcode", "title": "jQuery.qrcode",
"version": "0.18.0", "version": "0.15.0",
"description": "Generate QR codes dynamically.", "description": "Generate QR codes dynamically.",
"homepage": "https://larsjung.de/jquery-qrcode/", "homepage": "https://larsjung.de/jquery-qrcode/",
"bugs": "https://github.com/lrsjng/jquery-qrcode/issues",
"author": "Lars Jung <lrsjng@gmail.com> (https://larsjung.de)", "author": "Lars Jung <lrsjng@gmail.com> (https://larsjung.de)",
"license": "MIT", "license": "MIT",
"scripts": { "scripts": {
"lint": "eslint .", "lint": "eslint .",
"build": "node ghu release", "build": "node ghu.js release",
"precommit": "npm run -s lint && npm run -s build" "precommit": "npm run -s lint && npm run -s build"
}, },
"repository": { "repository": {
@ -16,11 +17,11 @@
"url": "https://github.com/lrsjng/jquery-qrcode.git" "url": "https://github.com/lrsjng/jquery-qrcode.git"
}, },
"devDependencies": { "devDependencies": {
"@babel/core": "7.10.5", "eslint": "5.15.1",
"@babel/preset-env": "7.10.4", "ghu": "0.13.0",
"eslint": "7.5.0", "jquery": "3.3.1"
"ghu": "0.26.0", },
"jquery": "3.5.1", "engines": {
"qrcode-generator": "1.4.4" "node": ">=8.0.0"
} }
} }

@ -0,0 +1,9 @@
---
env:
browser: true
rules:
func-names: 0
no-var: 0
prefer-arrow-callback: 0
strict: 0

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.8 KiB

After

Width:  |  Height:  |  Size: 2.9 KiB

@ -2,11 +2,14 @@ doctype html
html(lang='en') html(lang='en')
head head
meta(charset='utf-8') meta(charset='utf-8')
title #{pkg.title} #{pkg.version} Demo meta(http-equiv='x-ua-compatible', content='ie=edge')
meta(name='description', content=`Demo for ${pkg.title} (${pkg.homepage}).`) title #{pkg.title} #{pkg.version} · Demo
meta(name='description', content=`demo for ${pkg.title} (${pkg.homepage})`)
meta(name='viewport', content='width=device-width, initial-scale=1') meta(name='viewport', content='width=device-width, initial-scale=1')
link(rel='shortcut icon', type='image/png', href='dummy.png')
link(rel='apple-touch-icon-precomposed', type='image/png', href='dummy.png')
link(href='//fonts.googleapis.com/css?family=Ubuntu:300,400,700', rel='stylesheet')
link(href='styles.css', rel='stylesheet') link(href='styles.css', rel='stylesheet')
link(href='//fonts.googleapis.com/css2?family=Ubuntu+Mono:wght@400;700&display=swap' rel='stylesheet')
script(src='jquery.min.js') script(src='jquery.min.js')
script(src=`../jquery-qrcode-${pkg.version}.js`) script(src=`../jquery-qrcode-${pkg.version}.js`)
script(src='scripts.js') script(src='scripts.js')
@ -69,7 +72,7 @@ html(lang='en')
label(for='label') LABEL label(for='label') LABEL
input#label(type='text', value=pkg.title) input#label(type='text', value=pkg.title)
label(for='font') FONT NAME label(for='font') FONT NAME
input#font(type='text', value='Ubuntu Mono') input#font(type='text', value='Ubuntu')
label(for='fontcolor') FONT COLOR label(for='fontcolor') FONT COLOR
input#fontcolor(type='color', value='#ff9818') input#fontcolor(type='color', value='#ff9818')

@ -1,7 +1,8 @@
const WIN = global.window; (function () {
const JQ = WIN.jQuery; 'use strict';
const GUI_VALUE_PAIRS = [ var jq = window.jQuery;
var guiValuePairs = [
['size', 'px'], ['size', 'px'],
['minversion', ''], ['minversion', ''],
['quiet', ' modules'], ['quiet', ' modules'],
@ -11,71 +12,73 @@ const GUI_VALUE_PAIRS = [
['mposy', '%'] ['mposy', '%']
]; ];
const update_gui = () => { function updateGui() {
JQ.each(GUI_VALUE_PAIRS, (idx, pair) => { jq.each(guiValuePairs, function (idx, pair) {
const $label = JQ('label[for="' + pair[0] + '"]'); var $label = jq('label[for="' + pair[0] + '"]');
$label.text($label.text().replace(/:.*/, ': ' + JQ('#' + pair[0]).val() + pair[1])); $label.text($label.text().replace(/:.*/, ': ' + jq('#' + pair[0]).val() + pair[1]));
}); });
}; }
const update_qrcode = () => { function updateQrCode() {
const options = { var options = {
render: JQ('#render').val(), render: jq('#render').val(),
ecLevel: JQ('#eclevel').val(), ecLevel: jq('#eclevel').val(),
minVersion: parseInt(JQ('#minversion').val(), 10), minVersion: parseInt(jq('#minversion').val(), 10),
fill: JQ('#fill').val(), fill: jq('#fill').val(),
background: JQ('#background').val(), background: jq('#background').val(),
// fill: jq('#img-buffer')[0],
text: JQ('#text').val(), text: jq('#text').val(),
size: parseInt(JQ('#size').val(), 10), size: parseInt(jq('#size').val(), 10),
radius: parseInt(JQ('#radius').val(), 10) * 0.01, radius: parseInt(jq('#radius').val(), 10) * 0.01,
quiet: parseInt(JQ('#quiet').val(), 10), quiet: parseInt(jq('#quiet').val(), 10),
mode: parseInt(JQ('#mode').val(), 10), mode: parseInt(jq('#mode').val(), 10),
mSize: parseInt(JQ('#msize').val(), 10) * 0.01, mSize: parseInt(jq('#msize').val(), 10) * 0.01,
mPosX: parseInt(JQ('#mposx').val(), 10) * 0.01, mPosX: parseInt(jq('#mposx').val(), 10) * 0.01,
mPosY: parseInt(JQ('#mposy').val(), 10) * 0.01, mPosY: parseInt(jq('#mposy').val(), 10) * 0.01,
label: JQ('#label').val(), label: jq('#label').val(),
fontname: JQ('#font').val(), fontname: jq('#font').val(),
fontcolor: JQ('#fontcolor').val(), fontcolor: jq('#fontcolor').val(),
image: JQ('#img-buffer')[0] image: jq('#img-buffer')[0]
}; };
// options.fill = JQ('#img-buffer')[0]; jq('#container').empty().qrcode(options);
// options.fill = 'rgba(255,0,0,0.5)'; }
// options.background = JQ('#img-buffer')[0];
// options.background = 'rgba(255,0,0,0.5)';
JQ('#container').empty().qrcode(options);
};
const update = () => { function update() {
update_gui(); updateGui();
update_qrcode(); updateQrCode();
}; }
const on_img_input = () => { function onImageInput() {
const input = JQ('#image')[0]; var input = jq('#image')[0];
if (input.files && input.files[0]) { if (input.files && input.files[0]) {
const reader = new WIN.FileReader(); var reader = new FileReader();
reader.onload = event => { reader.onload = function (event) {
JQ('#img-buffer').attr('src', event.target.result); jq('#img-buffer').attr('src', event.target.result);
JQ('#mode').val('4'); jq('#mode').val('4');
setTimeout(update, 250); setTimeout(update, 250);
}; };
reader.readAsDataURL(input.files[0]); reader.readAsDataURL(input.files[0]);
} }
}; }
function download() {
jq('#download').attr('href', jq('#container canvas')[0].toDataURL('image/png'));
}
const init = () => { function init() {
JQ('#image').on('change', on_img_input); jq('#download').on('click', download);
JQ('input, textarea, select').on('input change', update); jq('#image').on('change', onImageInput);
JQ(WIN).on('load', update); jq('input, textarea, select').on('input change', update);
jq(window).on('load', update);
update(); update();
}; }
JQ(init); jq(init);
}());

@ -1,10 +1,7 @@
* { body {
font-family: 'Ubuntu', 'Arial', 'sans';
margin: 0; margin: 0;
padding: 0; padding: 0;
}
body {
font-family: 'Ubuntu Mono', 'monospace';
text-align: center; text-align: center;
background: repeat url('back.png'); background: repeat url('back.png');
} }
@ -25,7 +22,6 @@ body {
display: inline-block; display: inline-block;
margin: 20px auto; margin: 20px auto;
box-shadow: 0 0 16px rgba(0,0,0,0.5); box-shadow: 0 0 16px rgba(0,0,0,0.5);
-webkit-print-color-adjust: exact;
} }
#container > * { #container > * {
@ -59,14 +55,26 @@ hr {
} }
label { label {
font-size: 10px;
color: #aaa;
padding: 12px 4px 0 4px;
}
#download {
text-align: center;
font-weight: bold;
text-decoration: none;
display: block; display: block;
font-size: 12px;
color: #555; color: #555;
padding: 12px 4px 4px 4px; background-color: #ddd;
margin: 4px;
padding: 8px 0;
border: 1px solid #ddd;
cursor: pointer;
} }
input, textarea, select { input, textarea, select {
font-family: 'Ubuntu Mono', 'monospace'; font-family: 'Ubuntu', 'Arial', 'sans';
display: block; display: block;
background-color: #fff; background-color: #fff;
margin: 2px; margin: 2px;
@ -96,13 +104,3 @@ input::-webkit-slider-thumb {
border-radius: 3px; border-radius: 3px;
background-image: -webkit-gradient(linear, left top, left bottom, color-stop(0, #aaa), color-stop(1, #aaa)); background-image: -webkit-gradient(linear, left top, left bottom, color-stop(0, #aaa), color-stop(1, #aaa));
} }
@media print {
#container {
box-shadow: none;
}
.control {
display: none;
}
}

452
src/jquery-qrcode.js vendored

@ -1,305 +1,326 @@
const WIN = window; // eslint-disable-line (function (vendor_qrcode) {
const JQ = WIN.jQuery; 'use strict';
const is_img_el = x => x && typeof x.tagName === 'string' && x.tagName.toUpperCase() === 'IMG'; var jq = window.jQuery;
// Check if canvas is available in the browser (as Modernizr does)
var hasCanvas = (function () {
var elem = document.createElement('canvas');
return !!(elem.getContext && elem.getContext('2d'));
}());
// Wrapper for the original QR code generator. // Wrapper for the original QR code generator.
const create_qrcode = (text, level, version, quiet) => { function createQRCode(text, level, version, quiet) {
const qr = {}; var qr = {};
const qr_gen = require('qrcode-generator'); var vqr = vendor_qrcode(version, level);
qr_gen.stringToBytes = qr_gen.stringToBytesFuncs['UTF-8'];
const vqr = qr_gen(version, level);
vqr.addData(text); vqr.addData(text);
vqr.make(); vqr.make();
quiet = quiet || 0; quiet = quiet || 0;
const module_count = vqr.getModuleCount(); var qrModuleCount = vqr.getModuleCount();
const quiet_module_count = module_count + 2 * quiet; var quietModuleCount = vqr.getModuleCount() + 2 * quiet;
const is_dark = (row, col) => { function isDark(row, col) {
row -= quiet; row -= quiet;
col -= quiet; col -= quiet;
return row >= 0 && row < module_count && col >= 0 && col < module_count && vqr.isDark(row, col);
};
const add_blank = (l, t, r, b) => { if (row < 0 || row >= qrModuleCount || col < 0 || col >= qrModuleCount) {
const prev_is_dark = qr.is_dark; return false;
const module_size = 1 / quiet_module_count; }
return vqr.isDark(row, col);
}
function addBlank(l, t, r, b) {
var prevIsDark = qr.isDark;
var moduleSize = 1 / quietModuleCount;
qr.is_dark = (row, col) => { qr.isDark = function (row, col) {
const ml = col * module_size; var ml = col * moduleSize;
const mt = row * module_size; var mt = row * moduleSize;
const mr = ml + module_size; var mr = ml + moduleSize;
const mb = mt + module_size; var mb = mt + moduleSize;
return prev_is_dark(row, col) && (l > mr || ml > r || t > mb || mt > b);
}; return prevIsDark(row, col) && (l > mr || ml > r || t > mb || mt > b);
}; };
}
qr.text = text; qr.text = text;
qr.level = level; qr.level = level;
qr.version = version; qr.version = version;
qr.module_count = quiet_module_count; qr.moduleCount = quietModuleCount;
qr.is_dark = is_dark; qr.isDark = isDark;
qr.add_blank = add_blank; qr.addBlank = addBlank;
return qr; return qr;
}; }
// Returns a minimal QR code for the given text starting with version `min_ver`. // Returns a minimal QR code for the given text starting with version `minVersion`.
// Returns `undefined` if `text` is too long to be encoded in `max_ver`. // Returns `undefined` if `text` is too long to be encoded in `maxVersion`.
const create_min_qrcode = (text, level, min_ver, max_ver, quiet) => { function createMinQRCode(text, level, minVersion, maxVersion, quiet) {
min_ver = Math.max(1, min_ver || 1); minVersion = Math.max(1, minVersion || 1);
max_ver = Math.min(40, max_ver || 40); maxVersion = Math.min(40, maxVersion || 40);
for (let ver = min_ver; ver <= max_ver; ver += 1) { for (var version = minVersion; version <= maxVersion; version += 1) {
try { try {
return create_qrcode(text, level, ver, quiet); return createQRCode(text, level, version, quiet);
} catch (err) {/* empty */} } catch (err) {/* empty */}
} }
return undefined; return undefined;
}; }
const draw_background_label = (qr, ctx, settings) => { function drawBackgroundLabel(qr, context, settings) {
const size = settings.size; var size = settings.size;
const font = 'bold ' + settings.mSize * size + 'px ' + settings.fontname; var font = 'bold ' + settings.mSize * size + 'px ' + settings.fontname;
const tmp_ctx = JQ('<canvas/>')[0].getContext('2d'); var ctx = jq('<canvas/>')[0].getContext('2d');
tmp_ctx.font = font; ctx.font = font;
const w = tmp_ctx.measureText(settings.label).width; var w = ctx.measureText(settings.label).width;
const sh = settings.mSize; var sh = settings.mSize;
const sw = w / size; var sw = w / size;
const sl = (1 - sw) * settings.mPosX; var sl = (1 - sw) * settings.mPosX;
const st = (1 - sh) * settings.mPosY; var st = (1 - sh) * settings.mPosY;
const sr = sl + sw; var sr = sl + sw;
const sb = st + sh; var sb = st + sh;
const pad = 0.01; var pad = 0.01;
if (settings.mode === 1) { if (settings.mode === 1) {
// Strip // Strip
qr.add_blank(0, st - pad, size, sb + pad); qr.addBlank(0, st - pad, size, sb + pad);
} else { } else {
// Box // Box
qr.add_blank(sl - pad, st - pad, sr + pad, sb + pad); qr.addBlank(sl - pad, st - pad, sr + pad, sb + pad);
} }
ctx.fillStyle = settings.fontcolor; context.fillStyle = settings.fontcolor;
ctx.font = font; context.font = font;
ctx.fillText(settings.label, sl * size, st * size + 0.75 * settings.mSize * size); context.fillText(settings.label, sl * size, st * size + 0.75 * settings.mSize * size);
}; }
const draw_background_img = (qr, ctx, settings) => { function drawBackgroundImage(qr, context, settings) {
const size = settings.size; var size = settings.size;
const w = settings.image.naturalWidth || 1; var w = settings.image.naturalWidth || 1;
const h = settings.image.naturalHeight || 1; var h = settings.image.naturalHeight || 1;
const sh = settings.mSize; var sh = settings.mSize;
const sw = sh * w / h; var sw = sh * w / h;
const sl = (1 - sw) * settings.mPosX; var sl = (1 - sw) * settings.mPosX;
const st = (1 - sh) * settings.mPosY; var st = (1 - sh) * settings.mPosY;
const sr = sl + sw; var sr = sl + sw;
const sb = st + sh; var sb = st + sh;
const pad = 0.01; var pad = 0.01;
if (settings.mode === 3) { if (settings.mode === 3) {
// Strip // Strip
qr.add_blank(0, st - pad, size, sb + pad); qr.addBlank(0, st - pad, size, sb + pad);
} else { } else {
// Box // Box
qr.add_blank(sl - pad, st - pad, sr + pad, sb + pad); qr.addBlank(sl - pad, st - pad, sr + pad, sb + pad);
} }
ctx.drawImage(settings.image, sl * size, st * size, sw * size, sh * size); context.drawImage(settings.image, sl * size, st * size, sw * size, sh * size);
}; }
const draw_background = (qr, ctx, settings) => { function drawBackground(qr, context, settings) {
if (is_img_el(settings.background)) { if (jq(settings.background).is('img')) {
ctx.drawImage(settings.background, 0, 0, settings.size, settings.size); context.drawImage(settings.background, 0, 0, settings.size, settings.size);
} else if (settings.background) { } else if (settings.background) {
ctx.fillStyle = settings.background; context.fillStyle = settings.background;
ctx.fillRect(settings.left, settings.top, settings.size, settings.size); context.fillRect(settings.left, settings.top, settings.size, settings.size);
} }
const mode = settings.mode; var mode = settings.mode;
if (mode === 1 || mode === 2) { if (mode === 1 || mode === 2) {
draw_background_label(qr, ctx, settings); drawBackgroundLabel(qr, context, settings);
} else if (is_img_el(settings.image) && (mode === 3 || mode === 4)) { } else if (mode === 3 || mode === 4) {
draw_background_img(qr, ctx, settings); drawBackgroundImage(qr, context, settings);
}
} }
};
const draw_modules_default = (qr, ctx, settings, left, top, width, row, col) => { function drawModuleDefault(qr, context, settings, left, top, width, row, col) {
if (qr.is_dark(row, col)) { if (qr.isDark(row, col)) {
ctx.r(left, top, width, width); context.rect(left, top, width, width);
}
} }
};
const draw_modules_rounded_dark = (ctx, l, t, r, b, rad, nw, ne, se, sw) => { function drawModuleRoundedDark(ctx, l, t, r, b, rad, nw, ne, se, sw) {
if (nw) { if (nw) {
ctx.m(l + rad, t); ctx.moveTo(l + rad, t);
} else { } else {
ctx.m(l, t); ctx.moveTo(l, t);
} }
if (ne) { if (ne) {
ctx.l(r - rad, t).a(r, t, r, b, rad); ctx.lineTo(r - rad, t);
ctx.arcTo(r, t, r, b, rad);
} else { } else {
ctx.l(r, t); ctx.lineTo(r, t);
} }
if (se) { if (se) {
ctx.l(r, b - rad).a(r, b, l, b, rad); ctx.lineTo(r, b - rad);
ctx.arcTo(r, b, l, b, rad);
} else { } else {
ctx.l(r, b); ctx.lineTo(r, b);
} }
if (sw) { if (sw) {
ctx.l(l + rad, b).a(l, b, l, t, rad); ctx.lineTo(l + rad, b);
ctx.arcTo(l, b, l, t, rad);
} else { } else {
ctx.l(l, b); ctx.lineTo(l, b);
} }
if (nw) { if (nw) {
ctx.l(l, t + rad).a(l, t, r, t, rad); ctx.lineTo(l, t + rad);
ctx.arcTo(l, t, r, t, rad);
} else { } else {
ctx.l(l, t); ctx.lineTo(l, t);
}
} }
};
const draw_modules_rounded_light = (ctx, l, t, r, b, rad, nw, ne, se, sw) => { function drawModuleRoundendLight(ctx, l, t, r, b, rad, nw, ne, se, sw) {
if (nw) { if (nw) {
ctx.m(l + rad, t).l(l, t).l(l, t + rad).a(l, t, l + rad, t, rad); ctx.moveTo(l + rad, t);
ctx.lineTo(l, t);
ctx.lineTo(l, t + rad);
ctx.arcTo(l, t, l + rad, t, rad);
} }
if (ne) { if (ne) {
ctx.m(r - rad, t).l(r, t).l(r, t + rad).a(r, t, r - rad, t, rad); ctx.moveTo(r - rad, t);
ctx.lineTo(r, t);
ctx.lineTo(r, t + rad);
ctx.arcTo(r, t, r - rad, t, rad);
} }
if (se) { if (se) {
ctx.m(r - rad, b).l(r, b).l(r, b - rad).a(r, b, r - rad, b, rad); ctx.moveTo(r - rad, b);
ctx.lineTo(r, b);
ctx.lineTo(r, b - rad);
ctx.arcTo(r, b, r - rad, b, rad);
} }
if (sw) { if (sw) {
ctx.m(l + rad, b).l(l, b).l(l, b - rad).a(l, b, l + rad, b, rad); ctx.moveTo(l + rad, b);
ctx.lineTo(l, b);
ctx.lineTo(l, b - rad);
ctx.arcTo(l, b, l + rad, b, rad);
}
} }
};
const draw_modules_rounded = (qr, ctx, settings, left, top, width, row, col) => { function drawModuleRounded(qr, context, settings, left, top, width, row, col) {
const right = left + width; var isDark = qr.isDark;
const bottom = top + width; var right = left + width;
const rad = settings.radius * width; var bottom = top + width;
var radius = settings.radius * width;
const row_n = row - 1; var rowT = row - 1;
const row_s = row + 1; var rowB = row + 1;
const col_w = col - 1; var colL = col - 1;
const col_e = col + 1; var colR = col + 1;
var center = isDark(row, col);
const is_dark = qr.is_dark; var northwest = isDark(rowT, colL);
const d_center = is_dark(row, col); var north = isDark(rowT, col);
const d_nw = is_dark(row_n, col_w); var northeast = isDark(rowT, colR);
const d_n = is_dark(row_n, col); var east = isDark(row, colR);
const d_ne = is_dark(row_n, col_e); var southeast = isDark(rowB, colR);
const d_e = is_dark(row, col_e); var south = isDark(rowB, col);
const d_se = is_dark(row_s, col_e); var southwest = isDark(rowB, colL);
const d_s = is_dark(row_s, col); var west = isDark(row, colL);
const d_sw = is_dark(row_s, col_w);
const d_w = is_dark(row, col_w); if (center) {
drawModuleRoundedDark(context, left, top, right, bottom, radius, !north && !west, !north && !east, !south && !east, !south && !west);
if (d_center) {
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);
} else { } else {
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); drawModuleRoundendLight(context, left, top, right, bottom, radius, north && west && northwest, north && east && northeast, south && east && southeast, south && west && southwest);
}
} }
};
const draw_modules = (qr, ctx, settings) => { function drawModules(qr, context, settings) {
const module_count = qr.module_count; var moduleCount = qr.moduleCount;
const module_size = settings.size / module_count; var moduleSize = settings.size / moduleCount;
var fn = drawModuleDefault;
var row;
var col;
let fn = draw_modules_default;
if (settings.radius > 0 && settings.radius <= 0.5) { if (settings.radius > 0 && settings.radius <= 0.5) {
fn = draw_modules_rounded; fn = drawModuleRounded;
} }
const draw_ctx = { context.beginPath();
m(x, y) {ctx.moveTo(x, y); return draw_ctx;}, for (row = 0; row < moduleCount; row += 1) {
l(x, y) {ctx.lineTo(x, y); return draw_ctx;}, for (col = 0; col < moduleCount; col += 1) {
a(...args) {ctx.arcTo(...args); return draw_ctx;}, var l = settings.left + col * moduleSize;
r(...args) {ctx.rect(...args); return draw_ctx;} var t = settings.top + row * moduleSize;
}; var w = moduleSize;
ctx.beginPath(); fn(qr, context, settings, l, t, w, row, col);
for (let row = 0; row < module_count; row += 1) {
for (let col = 0; col < module_count; col += 1) {
const l = settings.left + col * module_size;
const t = settings.top + row * module_size;
const w = module_size;
fn(qr, draw_ctx, settings, l, t, w, row, col);
} }
} }
if (is_img_el(settings.fill)) { if (jq(settings.fill).is('img')) {
ctx.strokeStyle = 'rgba(0,0,0,0.5)'; context.strokeStyle = 'rgba(0,0,0,0.5)';
ctx.lineWidth = 2; context.lineWidth = 2;
ctx.stroke(); context.stroke();
const prev = ctx.globalCompositeOperation; var prev = context.globalCompositeOperation;
ctx.globalCompositeOperation = 'destination-out'; context.globalCompositeOperation = 'destination-out';
ctx.fill(); context.fill();
ctx.globalCompositeOperation = prev; context.globalCompositeOperation = prev;
ctx.clip(); context.clip();
ctx.drawImage(settings.fill, 0, 0, settings.size, settings.size); context.drawImage(settings.fill, 0, 0, settings.size, settings.size);
ctx.restore(); context.restore();
} else { } else {
ctx.fillStyle = settings.fill; context.fillStyle = settings.fill;
ctx.fill(); context.fill();
}
} }
};
// Draws QR code to the given `canvas` and returns it. // Draws QR code to the given `canvas` and returns it.
const draw_on_canvas = (canvas, settings) => { function drawOnCanvas(canvas, settings) {
const qr = create_min_qrcode(settings.text, settings.ecLevel, settings.minVersion, settings.maxVersion, settings.quiet); var qr = createMinQRCode(settings.text, settings.ecLevel, settings.minVersion, settings.maxVersion, settings.quiet);
if (!qr) { if (!qr) {
return null; return null;
} }
const $canvas = JQ(canvas).data('qrcode', qr); var $canvas = jq(canvas).data('qrcode', qr);
const ctx = $canvas[0].getContext('2d'); var context = $canvas[0].getContext('2d');
draw_background(qr, ctx, settings); drawBackground(qr, context, settings);
draw_modules(qr, ctx, settings); drawModules(qr, context, settings);
return $canvas; return $canvas;
}; }
// Returns a `canvas` element representing the QR code for the given settings. // Returns a `canvas` element representing the QR code for the given settings.
const create_canvas = settings => { function createCanvas(settings) {
const $canvas = JQ('<canvas/>').attr('width', settings.size).attr('height', settings.size); var $canvas = jq('<canvas/>').attr('width', settings.size).attr('height', settings.size);
return draw_on_canvas($canvas, settings); return drawOnCanvas($canvas, settings);
}; }
// Returns an `image` element representing the QR code for the given settings. // Returns an `image` element representing the QR code for the given settings.
const create_img = settings => { function createImage(settings) {
return JQ('<img/>').attr('src', create_canvas(settings)[0].toDataURL('image/png')); return jq('<img/>').attr('src', createCanvas(settings)[0].toDataURL('image/png'));
}; }
// Returns a `div` element representing the QR code for the given settings. // Returns a `div` element representing the QR code for the given settings.
const create_div = settings => { function createDiv(settings) {
const qr = create_min_qrcode(settings.text, settings.ecLevel, settings.minVersion, settings.maxVersion, settings.quiet); var qr = createMinQRCode(settings.text, settings.ecLevel, settings.minVersion, settings.maxVersion, settings.quiet);
if (!qr) { if (!qr) {
return null; return null;
} }
// some shortcuts to improve compression // some shortcuts to improve compression
const settings_size = settings.size; var settings_size = settings.size;
const settings_bgColor = settings.background; var settings_bgColor = settings.background;
const math_floor = Math.floor; var math_floor = Math.floor;
var moduleCount = qr.moduleCount;
var moduleSize = math_floor(settings_size / moduleCount);
var offset = math_floor(0.5 * (settings_size - moduleSize * moduleCount));
const module_count = qr.module_count; var row;
const module_size = math_floor(settings_size / module_count); var col;
const offset = math_floor(0.5 * (settings_size - module_size * module_count));
const container_css = { var containerCSS = {
position: 'relative', position: 'relative',
left: 0, left: 0,
top: 0, top: 0,
@ -308,29 +329,29 @@ const create_div = settings => {
width: settings_size, width: settings_size,
height: settings_size height: settings_size
}; };
const dark_css = { var darkCSS = {
position: 'absolute', position: 'absolute',
padding: 0, padding: 0,
margin: 0, margin: 0,
width: module_size, width: moduleSize,
height: module_size, height: moduleSize,
'background-color': settings.fill 'background-color': settings.fill
}; };
const $div = JQ('<div/>').data('qrcode', qr).css(container_css); var $div = jq('<div/>').data('qrcode', qr).css(containerCSS);
if (settings_bgColor) { if (settings_bgColor) {
$div.css('background-color', settings_bgColor); $div.css('background-color', settings_bgColor);
} }
for (let row = 0; row < module_count; row += 1) { for (row = 0; row < moduleCount; row += 1) {
for (let col = 0; col < module_count; col += 1) { for (col = 0; col < moduleCount; col += 1) {
if (qr.is_dark(row, col)) { if (qr.isDark(row, col)) {
JQ('<div/>') jq('<div/>')
.css(dark_css) .css(darkCSS)
.css({ .css({
left: offset + col * module_size, left: offset + col * moduleSize,
top: offset + row * module_size top: offset + row * moduleSize
}) })
.appendTo($div); .appendTo($div);
} }
@ -338,19 +359,24 @@ const create_div = settings => {
} }
return $div; return $div;
}; }
const create_html = settings => { function createHTML(settings) {
if (settings.render === 'canvas') { if (hasCanvas && settings.render === 'canvas') {
return create_canvas(settings); return createCanvas(settings);
} else if (settings.render === 'image') { } else if (hasCanvas && settings.render === 'image') {
return create_img(settings); return createImage(settings);
} }
return create_div(settings); return createDiv(settings);
}; }
// Plugin
// ======
const DEFAULTS = { // Default settings
// ----------------
var defaults = {
// render method: `'canvas'`, `'image'` or `'div'` // render method: `'canvas'`, `'image'` or `'div'`
render: 'canvas', render: 'canvas',
@ -372,7 +398,7 @@ const DEFAULTS = {
fill: '#000', fill: '#000',
// background color or image element, `null` for transparent background // background color or image element, `null` for transparent background
background: '#fff', background: null,
// content // content
text: 'no text', text: 'no text',
@ -402,14 +428,22 @@ const DEFAULTS = {
image: null image: null
}; };
JQ.fn.qrcode = module.exports = function main(options) { // Register the plugin
const settings = JQ.extend({}, DEFAULTS, options); // -------------------
jq.fn.qrcode = function (options) {
var settings = jq.extend({}, defaults, options);
return this.each((idx, el) => { return this.each(function (idx, el) {
if (el.nodeName.toLowerCase() === 'canvas') { if (el.nodeName.toLowerCase() === 'canvas') {
draw_on_canvas(el, settings); drawOnCanvas(el, settings);
} else { } else {
JQ(el).append(create_html(settings)); jq(el).append(createHTML(settings));
} }
}); });
}; };
}(function () {
// `qrcode` is the single public function defined by the `QR Code Generator`
// @include "../vendor/qrcode.js"
// @include "../vendor/qrcode_UTF8.js"
return qrcode; // eslint-disable-line no-undef
}()));

1824
vendor/qrcode.js vendored

File diff suppressed because it is too large Load Diff

@ -0,0 +1,59 @@
//---------------------------------------------------------------------
//
// QR Code Generator for JavaScript UTF8 Support (optional)
//
// Copyright (c) 2011 Kazuhiko Arase
//
// URL: http://www.d-project.com/
//
// Licensed under the MIT license:
// http://www.opensource.org/licenses/mit-license.php
//
// The word 'QR Code' is registered trademark of
// DENSO WAVE INCORPORATED
// http://www.denso-wave.com/qrcode/faqpatent-e.html
//
//---------------------------------------------------------------------
!function(qrcode) {
//---------------------------------------------------------------------
// overwrite qrcode.stringToBytes
//---------------------------------------------------------------------
qrcode.stringToBytes = function(s) {
// http://stackoverflow.com/questions/18729405/how-to-convert-utf8-string-to-byte-array
function toUTF8Array(str) {
var utf8 = [];
for (var i=0; i < str.length; i++) {
var charcode = str.charCodeAt(i);
if (charcode < 0x80) utf8.push(charcode);
else if (charcode < 0x800) {
utf8.push(0xc0 | (charcode >> 6),
0x80 | (charcode & 0x3f));
}
else if (charcode < 0xd800 || charcode >= 0xe000) {
utf8.push(0xe0 | (charcode >> 12),
0x80 | ((charcode>>6) & 0x3f),
0x80 | (charcode & 0x3f));
}
// surrogate pair
else {
i++;
// UTF-16 encodes 0x10000-0x10FFFF by
// subtracting 0x10000 and splitting the
// 20 bits of 0x0-0xFFFFF into two halves
charcode = 0x10000 + (((charcode & 0x3ff)<<10)
| (str.charCodeAt(i) & 0x3ff));
utf8.push(0xf0 | (charcode >>18),
0x80 | ((charcode>>12) & 0x3f),
0x80 | ((charcode>>6) & 0x3f),
0x80 | (charcode & 0x3f));
}
}
return utf8;
}
return toUTF8Array(s);
};
}(qrcode);
Loading…
Cancel
Save