├── .gitignore
├── .npmignore
├── LICENSE
├── README.md
├── index-svg.html
├── index.html
├── package.json
├── qrcode.js
└── qrcode.min.js
/.gitignore:
--------------------------------------------------------------------------------
1 | .DS_Store
2 |
3 | .idea
4 | .project
5 |
--------------------------------------------------------------------------------
/.npmignore:
--------------------------------------------------------------------------------
1 | *.html
2 | qrcode.js
3 | LICENSE
4 |
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | The MIT License (MIT)
2 | ---------------------
3 | Copyright (c) 2012 davidshimjs
4 |
5 | Permission is hereby granted, free of charge,
6 | to any person obtaining a copy of this software and associated documentation files (the "Software"),
7 | to deal in the Software without restriction,
8 | including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense,
9 | and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so,
10 | subject to the following conditions:
11 |
12 | The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
13 |
14 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # QRCode.js
2 | QRCode.js is javascript library for making QRCode. QRCode.js supports Cross-browser with HTML5 Canvas and table tag in DOM.
3 | QRCode.js has no dependencies.
4 |
5 | Project forked by KeeeX Company from davidshimjs/qrcodejs, we fixed Code Length Overflow error, and cleaned up the repository.
6 |
7 | ## Basic Usages
8 | ```html
9 |
10 |
13 | ```
14 |
15 | or with some options
16 |
17 | ```html
18 |
19 |
29 | ```
30 |
31 | and you can use some methods
32 |
33 | ```javascript
34 | qrcode.clear(); // clear the code.
35 | qrcode.makeCode("https://github.com/KeeeX"); // make another code.
36 | ```
37 |
38 | ### Using with webpack
39 |
40 | ```javascript
41 | const QRCode = require("@keeex/qrcodejs-kx");
42 | // Use QRCode as usual
43 | ```
44 |
45 | ## Browser Compatibility
46 | IE6~10, Chrome, Firefox, Safari, Opera, Mobile Safari, Android, Windows Mobile, ETC.
47 |
48 | ## License
49 | MIT License
50 |
--------------------------------------------------------------------------------
/index-svg.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | Cross-Browser QRCode generator for Javascript
5 |
6 |
7 |
8 |
9 |
10 | Please type your input below, and press Enter.
11 |
12 |
13 |
14 |
15 |
42 |
43 |
--------------------------------------------------------------------------------
/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | Cross-Browser QRCode generator for Javascript
5 |
6 |
7 |
8 |
9 |
10 | Please type your input below, and press Enter.
11 |
12 |
13 |
14 |
40 |
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "@keeex/qrcodejs-kx",
3 | "version": "1.0.2",
4 | "description": "QRCodeJS is a javascript library for making QRCode. It has no dependencies, is cross-browser and using HTML5 Canvas. This porject is a fork by KeeeX Company from davidshimjs, with Code Length Overflow fixed.",
5 | "main": "qrcode.min.js",
6 | "scripts": {
7 | "test": "echo \"Error: no test specified\" && exit 1"
8 | },
9 | "repository": {
10 | "type": "git",
11 | "url": "git+https://github.com/KeeeX/qrcodejs.git"
12 | },
13 | "keywords": [
14 | "qr",
15 | "qrcode",
16 | "generator",
17 | "davidshimsjs",
18 | "keeex"
19 | ],
20 | "author": "KeeeX",
21 | "license": "MIT",
22 | "bugs": {
23 | "url": "https://github.com/KeeeX/qrcodejs/issues"
24 | },
25 | "homepage": "https://github.com/KeeeX/qrcodejs#readme"
26 | }
27 |
--------------------------------------------------------------------------------
/qrcode.js:
--------------------------------------------------------------------------------
1 | /**
2 | * @fileoverview
3 | * - Using the 'QRCode for Javascript library'
4 | * - Fixed dataset of 'QRCode for Javascript library' for support full-spec.
5 | * - this library has no dependencies.
6 | *
7 | * @author davidshimjs
8 | * @see http://www.d-project.com/
9 | * @see http://jeromeetienne.github.com/jquery-qrcode/
10 | */
11 | var QRCode;
12 |
13 | (function () {
14 | //---------------------------------------------------------------------
15 | // QRCode for JavaScript
16 | //
17 | // Copyright (c) 2009 Kazuhiko Arase
18 | //
19 | // URL: http://www.d-project.com/
20 | //
21 | // Licensed under the MIT license:
22 | // http://www.opensource.org/licenses/mit-license.php
23 | //
24 | // The word "QR Code" is registered trademark of
25 | // DENSO WAVE INCORPORATED
26 | // http://www.denso-wave.com/qrcode/faqpatent-e.html
27 | //
28 | //---------------------------------------------------------------------
29 | function QR8bitByte(data) {
30 | this.mode = QRMode.MODE_8BIT_BYTE;
31 | this.data = data;
32 | this.parsedData = [];
33 |
34 | // Added to support UTF-8 Characters
35 | for (var i = 0, l = this.data.length; i < l; i++) {
36 | var byteArray = [];
37 | var code = this.data.charCodeAt(i);
38 |
39 | if (code > 0x10000) {
40 | byteArray[0] = 0xF0 | ((code & 0x1C0000) >>> 18);
41 | byteArray[1] = 0x80 | ((code & 0x3F000) >>> 12);
42 | byteArray[2] = 0x80 | ((code & 0xFC0) >>> 6);
43 | byteArray[3] = 0x80 | (code & 0x3F);
44 | } else if (code > 0x800) {
45 | byteArray[0] = 0xE0 | ((code & 0xF000) >>> 12);
46 | byteArray[1] = 0x80 | ((code & 0xFC0) >>> 6);
47 | byteArray[2] = 0x80 | (code & 0x3F);
48 | } else if (code > 0x80) {
49 | byteArray[0] = 0xC0 | ((code & 0x7C0) >>> 6);
50 | byteArray[1] = 0x80 | (code & 0x3F);
51 | } else {
52 | byteArray[0] = code;
53 | }
54 |
55 | this.parsedData.push(byteArray);
56 | }
57 |
58 | this.parsedData = Array.prototype.concat.apply([], this.parsedData);
59 |
60 | if (this.parsedData.length != this.data.length) {
61 | this.parsedData.unshift(191);
62 | this.parsedData.unshift(187);
63 | this.parsedData.unshift(239);
64 | }
65 | }
66 |
67 | QR8bitByte.prototype = {
68 | getLength: function (buffer) {
69 | return this.parsedData.length;
70 | },
71 | write: function (buffer) {
72 | for (var i = 0, l = this.parsedData.length; i < l; i++) {
73 | buffer.put(this.parsedData[i], 8);
74 | }
75 | }
76 | };
77 |
78 | function QRCodeModel(typeNumber, errorCorrectLevel) {
79 | this.typeNumber = typeNumber;
80 | this.errorCorrectLevel = errorCorrectLevel;
81 | this.modules = null;
82 | this.moduleCount = 0;
83 | this.dataCache = null;
84 | this.dataList = [];
85 | }
86 |
87 | QRCodeModel.prototype={addData:function(data){var newData=new QR8bitByte(data);this.dataList.push(newData);this.dataCache=null;},isDark:function(row,col){if(row<0||this.moduleCount<=row||col<0||this.moduleCount<=col){throw new Error(row+","+col);}
88 | return this.modules[row][col];},getModuleCount:function(){return this.moduleCount;},make:function(){this.makeImpl(false,this.getBestMaskPattern());},makeImpl:function(test,maskPattern){this.moduleCount=this.typeNumber*4+17;this.modules=new Array(this.moduleCount);for(var row=0;row=7){this.setupTypeNumber(test);}
90 | if(this.dataCache==null){this.dataCache=QRCodeModel.createData(this.typeNumber,this.errorCorrectLevel,this.dataList);}
91 | this.mapData(this.dataCache,maskPattern);},setupPositionProbePattern:function(row,col){for(var r=-1;r<=7;r++){if(row+r<=-1||this.moduleCount<=row+r)continue;for(var c=-1;c<=7;c++){if(col+c<=-1||this.moduleCount<=col+c)continue;if((0<=r&&r<=6&&(c==0||c==6))||(0<=c&&c<=6&&(r==0||r==6))||(2<=r&&r<=4&&2<=c&&c<=4)){this.modules[row+r][col+c]=true;}else{this.modules[row+r][col+c]=false;}}}},getBestMaskPattern:function(){var minLostPoint=0;var pattern=0;for(var i=0;i<8;i++){this.makeImpl(true,i);var lostPoint=QRUtil.getLostPoint(this);if(i==0||minLostPoint>lostPoint){minLostPoint=lostPoint;pattern=i;}}
92 | return pattern;},createMovieClip:function(target_mc,instance_name,depth){var qr_mc=target_mc.createEmptyMovieClip(instance_name,depth);var cs=1;this.make();for(var row=0;row>i)&1)==1);this.modules[Math.floor(i/3)][i%3+this.moduleCount-8-3]=mod;}
98 | for(var i=0;i<18;i++){var mod=(!test&&((bits>>i)&1)==1);this.modules[i%3+this.moduleCount-8-3][Math.floor(i/3)]=mod;}},setupTypeInfo:function(test,maskPattern){var data=(this.errorCorrectLevel<<3)|maskPattern;var bits=QRUtil.getBCHTypeInfo(data);for(var i=0;i<15;i++){var mod=(!test&&((bits>>i)&1)==1);if(i<6){this.modules[i][8]=mod;}else if(i<8){this.modules[i+1][8]=mod;}else{this.modules[this.moduleCount-15+i][8]=mod;}}
99 | for(var i=0;i<15;i++){var mod=(!test&&((bits>>i)&1)==1);if(i<8){this.modules[8][this.moduleCount-i-1]=mod;}else if(i<9){this.modules[8][15-i-1+1]=mod;}else{this.modules[8][15-i-1]=mod;}}
100 | this.modules[this.moduleCount-8][8]=(!test);},mapData:function(data,maskPattern){var inc=-1;var row=this.moduleCount-1;var bitIndex=7;var byteIndex=0;for(var col=this.moduleCount-1;col>0;col-=2){if(col==6)col--;while(true){for(var c=0;c<2;c++){if(this.modules[row][col-c]==null){var dark=false;if(byteIndex>>bitIndex)&1)==1);}
101 | var mask=QRUtil.getMask(maskPattern,row,col-c);if(mask){dark=!dark;}
102 | this.modules[row][col-c]=dark;bitIndex--;if(bitIndex==-1){byteIndex++;bitIndex=7;}}}
103 | row+=inc;if(row<0||this.moduleCount<=row){row-=inc;inc=-inc;break;}}}}};QRCodeModel.PAD0=0xEC;QRCodeModel.PAD1=0x11;QRCodeModel.createData=function(typeNumber,errorCorrectLevel,dataList){var rsBlocks=QRRSBlock.getRSBlocks(typeNumber,errorCorrectLevel);var buffer=new QRBitBuffer();for(var i=0;itotalDataCount*8){throw new Error("code length overflow. ("
106 | +buffer.getLengthInBits()
107 | +">"
108 | +totalDataCount*8
109 | +")");}
110 | if(buffer.getLengthInBits()+4<=totalDataCount*8){buffer.put(0,4);}
111 | while(buffer.getLengthInBits()%8!=0){buffer.putBit(false);}
112 | while(true){if(buffer.getLengthInBits()>=totalDataCount*8){break;}
113 | buffer.put(QRCodeModel.PAD0,8);if(buffer.getLengthInBits()>=totalDataCount*8){break;}
114 | buffer.put(QRCodeModel.PAD1,8);}
115 | return QRCodeModel.createBytes(buffer,rsBlocks);};QRCodeModel.createBytes=function(buffer,rsBlocks){var offset=0;var maxDcCount=0;var maxEcCount=0;var dcdata=new Array(rsBlocks.length);var ecdata=new Array(rsBlocks.length);for(var r=0;r=0)?modPoly.get(modIndex):0;}}
117 | var totalCodeCount=0;for(var i=0;i=0){d^=(QRUtil.G15<<(QRUtil.getBCHDigit(d)-QRUtil.getBCHDigit(QRUtil.G15)));}
121 | return((data<<10)|d)^QRUtil.G15_MASK;},getBCHTypeNumber:function(data){var d=data<<12;while(QRUtil.getBCHDigit(d)-QRUtil.getBCHDigit(QRUtil.G18)>=0){d^=(QRUtil.G18<<(QRUtil.getBCHDigit(d)-QRUtil.getBCHDigit(QRUtil.G18)));}
122 | return(data<<12)|d;},getBCHDigit:function(data){var digit=0;while(data!=0){digit++;data>>>=1;}
123 | return digit;},getPatternPosition:function(typeNumber){return QRUtil.PATTERN_POSITION_TABLE[typeNumber-1];},getMask:function(maskPattern,i,j){switch(maskPattern){case QRMaskPattern.PATTERN000:return(i+j)%2==0;case QRMaskPattern.PATTERN001:return i%2==0;case QRMaskPattern.PATTERN010:return j%3==0;case QRMaskPattern.PATTERN011:return(i+j)%3==0;case QRMaskPattern.PATTERN100:return(Math.floor(i/2)+Math.floor(j/3))%2==0;case QRMaskPattern.PATTERN101:return(i*j)%2+(i*j)%3==0;case QRMaskPattern.PATTERN110:return((i*j)%2+(i*j)%3)%2==0;case QRMaskPattern.PATTERN111:return((i*j)%3+(i+j)%2)%2==0;default:throw new Error("bad maskPattern:"+maskPattern);}},getErrorCorrectPolynomial:function(errorCorrectLength){var a=new QRPolynomial([1],0);for(var i=0;i5){lostPoint+=(3+sameCount-5);}}}
129 | for(var row=0;row=256){n-=255;}
136 | return QRMath.EXP_TABLE[n];},EXP_TABLE:new Array(256),LOG_TABLE:new Array(256)};for(var i=0;i<8;i++){QRMath.EXP_TABLE[i]=1<>>(7-index%8))&1)==1;},put:function(num,length){for(var i=0;i>>(length-i-1))&1)==1);}},getLengthInBits:function(){return this.length;},putBit:function(bit){var bufIndex=Math.floor(this.length/8);if(this.buffer.length<=bufIndex){this.buffer.push(0);}
151 | if(bit){this.buffer[bufIndex]|=(0x80>>>(this.length%8));}
152 | this.length++;}};var QRCodeLimitLength=[[17,14,11,7],[32,26,20,14],[53,42,32,24],[78,62,46,34],[106,84,60,44],[134,106,74,58],[154,122,86,64],[192,152,108,84],[230,180,130,98],[271,213,151,119],[321,251,177,137],[367,287,203,155],[425,331,241,177],[458,362,258,194],[520,412,292,220],[586,450,322,250],[644,504,364,280],[718,560,394,310],[792,624,442,338],[858,666,482,382],[929,711,509,403],[1003,779,565,439],[1091,857,611,461],[1171,911,661,511],[1273,997,715,535],[1367,1059,751,593],[1465,1125,805,625],[1528,1190,868,658],[1628,1264,908,698],[1732,1370,982,742],[1840,1452,1030,790],[1952,1538,1112,842],[2068,1628,1168,898],[2188,1722,1228,958],[2303,1809,1283,983],[2431,1911,1351,1051],[2563,1989,1423,1093],[2699,2099,1499,1139],[2809,2213,1579,1219],[2953,2331,1663,1273]];
153 |
154 | function _isSupportCanvas() {
155 | return typeof CanvasRenderingContext2D != "undefined";
156 | }
157 |
158 | // android 2.x doesn't support Data-URI spec
159 | function _getAndroid() {
160 | var android = false;
161 | var sAgent = navigator.userAgent;
162 |
163 | if (/android/i.test(sAgent)) { // android
164 | android = true;
165 | var aMat = sAgent.toString().match(/android ([0-9]\.[0-9])/i);
166 |
167 | if (aMat && aMat[1]) {
168 | android = parseFloat(aMat[1]);
169 | }
170 | }
171 |
172 | return android;
173 | }
174 |
175 | var svgDrawer = (function() {
176 |
177 | var Drawing = function (el, htOption) {
178 | this._el = el;
179 | this._htOption = htOption;
180 | };
181 |
182 | Drawing.prototype.draw = function (oQRCode) {
183 | var _htOption = this._htOption;
184 | var _el = this._el;
185 | var nCount = oQRCode.getModuleCount();
186 | var nWidth = Math.floor(_htOption.width / nCount);
187 | var nHeight = Math.floor(_htOption.height / nCount);
188 |
189 | this.clear();
190 |
191 | function makeSVG(tag, attrs) {
192 | var el = document.createElementNS('http://www.w3.org/2000/svg', tag);
193 | for (var k in attrs)
194 | if (attrs.hasOwnProperty(k)) el.setAttribute(k, attrs[k]);
195 | return el;
196 | }
197 |
198 | var svg = makeSVG("svg" , {'viewBox': '0 0 ' + String(nCount) + " " + String(nCount), 'width': '100%', 'height': '100%', 'fill': _htOption.colorLight});
199 | svg.setAttributeNS("http://www.w3.org/2000/xmlns/", "xmlns:xlink", "http://www.w3.org/1999/xlink");
200 | _el.appendChild(svg);
201 |
202 | svg.appendChild(makeSVG("rect", {"fill": _htOption.colorLight, "width": "100%", "height": "100%"}));
203 | svg.appendChild(makeSVG("rect", {"fill": _htOption.colorDark, "width": "1", "height": "1", "id": "template"}));
204 |
205 | for (var row = 0; row < nCount; row++) {
206 | for (var col = 0; col < nCount; col++) {
207 | if (oQRCode.isDark(row, col)) {
208 | var child = makeSVG("use", {"x": String(col), "y": String(row)});
209 | child.setAttributeNS("http://www.w3.org/1999/xlink", "href", "#template")
210 | svg.appendChild(child);
211 | }
212 | }
213 | }
214 | };
215 | Drawing.prototype.clear = function () {
216 | while (this._el.hasChildNodes())
217 | this._el.removeChild(this._el.lastChild);
218 | };
219 | return Drawing;
220 | })();
221 |
222 | var useSVG = document.documentElement.tagName.toLowerCase() === "svg";
223 |
224 | // Drawing in DOM by using Table tag
225 | var Drawing = useSVG ? svgDrawer : !_isSupportCanvas() ? (function () {
226 | var Drawing = function (el, htOption) {
227 | this._el = el;
228 | this._htOption = htOption;
229 | };
230 |
231 | /**
232 | * Draw the QRCode
233 | *
234 | * @param {QRCode} oQRCode
235 | */
236 | Drawing.prototype.draw = function (oQRCode) {
237 | var _htOption = this._htOption;
238 | var _el = this._el;
239 | var nCount = oQRCode.getModuleCount();
240 | var nWidth = Math.floor(_htOption.width / nCount);
241 | var nHeight = Math.floor(_htOption.height / nCount);
242 | var aHTML = [''];
243 |
244 | for (var row = 0; row < nCount; row++) {
245 | aHTML.push('');
246 |
247 | for (var col = 0; col < nCount; col++) {
248 | aHTML.push(' ');
249 | }
250 |
251 | aHTML.push(' ');
252 | }
253 |
254 | aHTML.push('
');
255 | _el.innerHTML = aHTML.join('');
256 |
257 | // Fix the margin values as real size.
258 | var elTable = _el.childNodes[0];
259 | var nLeftMarginTable = (_htOption.width - elTable.offsetWidth) / 2;
260 | var nTopMarginTable = (_htOption.height - elTable.offsetHeight) / 2;
261 |
262 | if (nLeftMarginTable > 0 && nTopMarginTable > 0) {
263 | elTable.style.margin = nTopMarginTable + "px " + nLeftMarginTable + "px";
264 | }
265 | };
266 |
267 | /**
268 | * Clear the QRCode
269 | */
270 | Drawing.prototype.clear = function () {
271 | this._el.innerHTML = '';
272 | };
273 |
274 | return Drawing;
275 | })() : (function () { // Drawing in Canvas
276 | function _onMakeImage() {
277 | this._elImage.src = this._elCanvas.toDataURL("image/png");
278 | this._elImage.style.display = "block";
279 | this._elCanvas.style.display = "none";
280 | }
281 |
282 | // Android 2.1 bug workaround
283 | // http://code.google.com/p/android/issues/detail?id=5141
284 | if (this._android && this._android <= 2.1) {
285 | var factor = 1 / window.devicePixelRatio;
286 | var drawImage = CanvasRenderingContext2D.prototype.drawImage;
287 | CanvasRenderingContext2D.prototype.drawImage = function (image, sx, sy, sw, sh, dx, dy, dw, dh) {
288 | if (("nodeName" in image) && /img/i.test(image.nodeName)) {
289 | for (var i = arguments.length - 1; i >= 1; i--) {
290 | arguments[i] = arguments[i] * factor;
291 | }
292 | } else if (typeof dw == "undefined") {
293 | arguments[1] *= factor;
294 | arguments[2] *= factor;
295 | arguments[3] *= factor;
296 | arguments[4] *= factor;
297 | }
298 |
299 | drawImage.apply(this, arguments);
300 | };
301 | }
302 |
303 | /**
304 | * Check whether the user's browser supports Data URI or not
305 | *
306 | * @private
307 | * @param {Function} fSuccess Occurs if it supports Data URI
308 | * @param {Function} fFail Occurs if it doesn't support Data URI
309 | */
310 | function _safeSetDataURI(fSuccess, fFail) {
311 | var self = this;
312 | self._fFail = fFail;
313 | self._fSuccess = fSuccess;
314 |
315 | // Check it just once
316 | if (self._bSupportDataURI === null) {
317 | var el = document.createElement("img");
318 | var fOnError = function() {
319 | self._bSupportDataURI = false;
320 |
321 | if (self._fFail) {
322 | self._fFail.call(self);
323 | }
324 | };
325 | var fOnSuccess = function() {
326 | self._bSupportDataURI = true;
327 |
328 | if (self._fSuccess) {
329 | self._fSuccess.call(self);
330 | }
331 | };
332 |
333 | el.onabort = fOnError;
334 | el.onerror = fOnError;
335 | el.onload = fOnSuccess;
336 | el.src = "data:image/gif;base64,iVBORw0KGgoAAAANSUhEUgAAAAUAAAAFCAYAAACNbyblAAAAHElEQVQI12P4//8/w38GIAXDIBKE0DHxgljNBAAO9TXL0Y4OHwAAAABJRU5ErkJggg=="; // the Image contains 1px data.
337 | return;
338 | } else if (self._bSupportDataURI === true && self._fSuccess) {
339 | self._fSuccess.call(self);
340 | } else if (self._bSupportDataURI === false && self._fFail) {
341 | self._fFail.call(self);
342 | }
343 | };
344 |
345 | /**
346 | * Drawing QRCode by using canvas
347 | *
348 | * @constructor
349 | * @param {HTMLElement} el
350 | * @param {Object} htOption QRCode Options
351 | */
352 | var Drawing = function (el, htOption) {
353 | this._bIsPainted = false;
354 | this._android = _getAndroid();
355 |
356 | this._htOption = htOption;
357 | this._elCanvas = document.createElement("canvas");
358 | this._elCanvas.width = htOption.width;
359 | this._elCanvas.height = htOption.height;
360 | el.appendChild(this._elCanvas);
361 | this._el = el;
362 | this._oContext = this._elCanvas.getContext("2d");
363 | this._bIsPainted = false;
364 | this._elImage = document.createElement("img");
365 | this._elImage.alt = "Scan me!";
366 | this._elImage.style.display = "none";
367 | this._el.appendChild(this._elImage);
368 | this._bSupportDataURI = null;
369 | };
370 |
371 | /**
372 | * Draw the QRCode
373 | *
374 | * @param {QRCode} oQRCode
375 | */
376 | Drawing.prototype.draw = function (oQRCode) {
377 | var _elImage = this._elImage;
378 | var _oContext = this._oContext;
379 | var _htOption = this._htOption;
380 |
381 | var nCount = oQRCode.getModuleCount();
382 | var nWidth = _htOption.width / nCount;
383 | var nHeight = _htOption.height / nCount;
384 | var nRoundedWidth = Math.round(nWidth);
385 | var nRoundedHeight = Math.round(nHeight);
386 |
387 | _elImage.style.display = "none";
388 | this.clear();
389 |
390 | for (var row = 0; row < nCount; row++) {
391 | for (var col = 0; col < nCount; col++) {
392 | var bIsDark = oQRCode.isDark(row, col);
393 | var nLeft = col * nWidth;
394 | var nTop = row * nHeight;
395 | _oContext.strokeStyle = bIsDark ? _htOption.colorDark : _htOption.colorLight;
396 | _oContext.lineWidth = 1;
397 | _oContext.fillStyle = bIsDark ? _htOption.colorDark : _htOption.colorLight;
398 | _oContext.fillRect(nLeft, nTop, nWidth, nHeight);
399 |
400 | // 안티 앨리어싱 방지 처리
401 | _oContext.strokeRect(
402 | Math.floor(nLeft) + 0.5,
403 | Math.floor(nTop) + 0.5,
404 | nRoundedWidth,
405 | nRoundedHeight
406 | );
407 |
408 | _oContext.strokeRect(
409 | Math.ceil(nLeft) - 0.5,
410 | Math.ceil(nTop) - 0.5,
411 | nRoundedWidth,
412 | nRoundedHeight
413 | );
414 | }
415 | }
416 |
417 | this._bIsPainted = true;
418 | };
419 |
420 | /**
421 | * Make the image from Canvas if the browser supports Data URI.
422 | */
423 | Drawing.prototype.makeImage = function () {
424 | if (this._bIsPainted) {
425 | _safeSetDataURI.call(this, _onMakeImage);
426 | }
427 | };
428 |
429 | /**
430 | * Return whether the QRCode is painted or not
431 | *
432 | * @return {Boolean}
433 | */
434 | Drawing.prototype.isPainted = function () {
435 | return this._bIsPainted;
436 | };
437 |
438 | /**
439 | * Clear the QRCode
440 | */
441 | Drawing.prototype.clear = function () {
442 | this._oContext.clearRect(0, 0, this._elCanvas.width, this._elCanvas.height);
443 | this._bIsPainted = false;
444 | };
445 |
446 | /**
447 | * @private
448 | * @param {Number} nNumber
449 | */
450 | Drawing.prototype.round = function (nNumber) {
451 | if (!nNumber) {
452 | return nNumber;
453 | }
454 |
455 | return Math.floor(nNumber * 1000) / 1000;
456 | };
457 |
458 | return Drawing;
459 | })();
460 |
461 | /**
462 | * Get the type by string length
463 | *
464 | * @private
465 | * @param {String} sText
466 | * @param {Number} nCorrectLevel
467 | * @return {Number} type
468 | */
469 | function _getTypeNumber(sText, nCorrectLevel) {
470 | var nType = 1;
471 | var length = _getUTF8Length(sText);
472 |
473 | for (var i = 0, len = QRCodeLimitLength.length; i <= len; i++) {
474 | var nLimit = 0;
475 |
476 | switch (nCorrectLevel) {
477 | case QRErrorCorrectLevel.L :
478 | nLimit = QRCodeLimitLength[i][0];
479 | break;
480 | case QRErrorCorrectLevel.M :
481 | nLimit = QRCodeLimitLength[i][1];
482 | break;
483 | case QRErrorCorrectLevel.Q :
484 | nLimit = QRCodeLimitLength[i][2];
485 | break;
486 | case QRErrorCorrectLevel.H :
487 | nLimit = QRCodeLimitLength[i][3];
488 | break;
489 | }
490 |
491 | if (length <= nLimit) {
492 | break;
493 | } else {
494 | nType++;
495 | }
496 | }
497 |
498 | if (nType > QRCodeLimitLength.length) {
499 | throw new Error("Too long data");
500 | }
501 |
502 | return nType;
503 | }
504 |
505 | function _getUTF8Length(sText) {
506 | var replacedText = encodeURI(sText).toString().replace(/\%[0-9a-fA-F]{2}/g, 'a');
507 | return replacedText.length + (replacedText.length != sText ? 3 : 0);
508 | }
509 |
510 | /**
511 | * @class QRCode
512 | * @constructor
513 | * @example
514 | * new QRCode(document.getElementById("test"), "http://jindo.dev.naver.com/collie");
515 | *
516 | * @example
517 | * var oQRCode = new QRCode("test", {
518 | * text : "http://naver.com",
519 | * width : 128,
520 | * height : 128
521 | * });
522 | *
523 | * oQRCode.clear(); // Clear the QRCode.
524 | * oQRCode.makeCode("http://map.naver.com"); // Re-create the QRCode.
525 | *
526 | * @param {HTMLElement|String} el target element or 'id' attribute of element.
527 | * @param {Object|String} vOption
528 | * @param {String} vOption.text QRCode link data
529 | * @param {Number} [vOption.width=256]
530 | * @param {Number} [vOption.height=256]
531 | * @param {String} [vOption.colorDark="#000000"]
532 | * @param {String} [vOption.colorLight="#ffffff"]
533 | * @param {QRCode.CorrectLevel} [vOption.correctLevel=QRCode.CorrectLevel.H] [L|M|Q|H]
534 | */
535 | QRCode = function (el, vOption) {
536 | this._htOption = {
537 | width : 256,
538 | height : 256,
539 | typeNumber : 4,
540 | colorDark : "#000000",
541 | colorLight : "#ffffff",
542 | correctLevel : QRErrorCorrectLevel.H
543 | };
544 |
545 | if (typeof vOption === 'string') {
546 | vOption = {
547 | text : vOption
548 | };
549 | }
550 |
551 | // Overwrites options
552 | if (vOption) {
553 | for (var i in vOption) {
554 | this._htOption[i] = vOption[i];
555 | }
556 | }
557 |
558 | if (typeof el == "string") {
559 | el = document.getElementById(el);
560 | }
561 |
562 | if (this._htOption.useSVG) {
563 | Drawing = svgDrawer;
564 | }
565 |
566 | this._android = _getAndroid();
567 | this._el = el;
568 | this._oQRCode = null;
569 | this._oDrawing = new Drawing(this._el, this._htOption);
570 |
571 | if (this._htOption.text) {
572 | this.makeCode(this._htOption.text);
573 | }
574 | };
575 |
576 | /**
577 | * Make the QRCode
578 | *
579 | * @param {String} sText link data
580 | */
581 | QRCode.prototype.makeCode = function (sText) {
582 | this._oQRCode = new QRCodeModel(_getTypeNumber(sText, this._htOption.correctLevel), this._htOption.correctLevel);
583 | this._oQRCode.addData(sText);
584 | this._oQRCode.make();
585 | this._el.title = sText;
586 | this._oDrawing.draw(this._oQRCode);
587 | this.makeImage();
588 | };
589 |
590 | /**
591 | * Make the Image from Canvas element
592 | * - It occurs automatically
593 | * - Android below 3 doesn't support Data-URI spec.
594 | *
595 | * @private
596 | */
597 | QRCode.prototype.makeImage = function () {
598 | if (typeof this._oDrawing.makeImage == "function" && (!this._android || this._android >= 3)) {
599 | this._oDrawing.makeImage();
600 | }
601 | };
602 |
603 | /**
604 | * Clear the QRCode
605 | */
606 | QRCode.prototype.clear = function () {
607 | this._oDrawing.clear();
608 | };
609 |
610 | /**
611 | * @name QRCode.CorrectLevel
612 | */
613 | QRCode.CorrectLevel = QRErrorCorrectLevel;
614 | })();
615 |
616 | if (typeof module != "undefined") {
617 | module.exports = QRCode;
618 | }
619 |
--------------------------------------------------------------------------------
/qrcode.min.js:
--------------------------------------------------------------------------------
1 | var QRCode;!function(){function t(t){this.mode=r.MODE_8BIT_BYTE,this.data=t,this.parsedData=[];for(var e=0,o=this.data.length;e65536?(i[0]=240|(1835008&n)>>>18,i[1]=128|(258048&n)>>>12,i[2]=128|(4032&n)>>>6,i[3]=128|63&n):n>2048?(i[0]=224|(61440&n)>>>12,i[1]=128|(4032&n)>>>6,i[2]=128|63&n):n>128?(i[0]=192|(1984&n)>>>6,i[1]=128|63&n):i[0]=n,this.parsedData.push(i)}this.parsedData=Array.prototype.concat.apply([],this.parsedData),this.parsedData.length!=this.data.length&&(this.parsedData.unshift(191),this.parsedData.unshift(187),this.parsedData.unshift(239))}function e(t,e){this.typeNumber=t,this.errorCorrectLevel=e,this.modules=null,this.moduleCount=0,this.dataCache=null,this.dataList=[]}t.prototype={getLength:function(t){return this.parsedData.length},write:function(t){for(var e=0,r=this.parsedData.length;e=7&&this.setupTypeNumber(t),null==this.dataCache&&(this.dataCache=e.createData(this.typeNumber,this.errorCorrectLevel,this.dataList)),this.mapData(this.dataCache,r)},setupPositionProbePattern:function(t,e){for(var r=-1;r<=7;r++)if(!(t+r<=-1||this.moduleCount<=t+r))for(var o=-1;o<=7;o++)e+o<=-1||this.moduleCount<=e+o||(this.modules[t+r][e+o]=0<=r&&r<=6&&(0==o||6==o)||0<=o&&o<=6&&(0==r||6==r)||2<=r&&r<=4&&2<=o&&o<=4)},getBestMaskPattern:function(){for(var t=0,e=0,r=0;r<8;r++){this.makeImpl(!0,r);var o=g.getLostPoint(this);(0==r||t>o)&&(t=o,e=r)}return e},createMovieClip:function(t,e,r){var o=t.createEmptyMovieClip(e,r);this.make();for(var i=0;i>r&1);this.modules[Math.floor(r/3)][r%3+this.moduleCount-8-3]=o}for(r=0;r<18;r++){o=!t&&1==(e>>r&1);this.modules[r%3+this.moduleCount-8-3][Math.floor(r/3)]=o}},setupTypeInfo:function(t,e){for(var r=this.errorCorrectLevel<<3|e,o=g.getBCHTypeInfo(r),i=0;i<15;i++){var n=!t&&1==(o>>i&1);i<6?this.modules[i][8]=n:i<8?this.modules[i+1][8]=n:this.modules[this.moduleCount-15+i][8]=n}for(i=0;i<15;i++){n=!t&&1==(o>>i&1);i<8?this.modules[8][this.moduleCount-i-1]=n:i<9?this.modules[8][15-i-1+1]=n:this.modules[8][15-i-1]=n}this.modules[this.moduleCount-8][8]=!t},mapData:function(t,e){for(var r=-1,o=this.moduleCount-1,i=7,n=0,a=this.moduleCount-1;a>0;a-=2)for(6==a&&a--;;){for(var s=0;s<2;s++)if(null==this.modules[o][a-s]){var h=!1;n>>i&1)),g.getMask(e,o,a-s)&&(h=!h),this.modules[o][a-s]=h,-1==--i&&(n++,i=7)}if((o+=r)<0||this.moduleCount<=o){o-=r,r=-r;break}}}},e.PAD0=236,e.PAD1=17,e.createData=function(t,r,o){for(var i=m.getRSBlocks(t,r),n=new _,a=0;a8*h)throw new Error("code length overflow. ("+n.getLengthInBits()+">"+8*h+")");for(n.getLengthInBits()+4<=8*h&&n.put(0,4);n.getLengthInBits()%8!=0;)n.putBit(!1);for(;!(n.getLengthInBits()>=8*h||(n.put(e.PAD0,8),n.getLengthInBits()>=8*h));)n.put(e.PAD1,8);return e.createBytes(n,i)},e.createBytes=function(t,e){for(var r=0,o=0,i=0,n=new Array(e.length),a=new Array(e.length),s=0;s=0?d.get(c):0}}var m=0;for(u=0;u=0;)e^=g.G15<=0;)e^=g.G18<>>=1;return e},getPatternPosition:function(t){return g.PATTERN_POSITION_TABLE[t-1]},getMask:function(t,e,r){switch(t){case i:return(e+r)%2==0;case n:return e%2==0;case a:return r%3==0;case s:return(e+r)%3==0;case h:return(Math.floor(e/2)+Math.floor(r/3))%2==0;case l:return e*r%2+e*r%3==0;case u:return(e*r%2+e*r%3)%2==0;case f:return(e*r%3+(e+r)%2)%2==0;default:throw new Error("bad maskPattern:"+t)}},getErrorCorrectPolynomial:function(t){for(var e=new p([1],0),r=0;r5&&(r+=3+n-5)}for(o=0;o=256;)t-=255;return d.EXP_TABLE[t]},EXP_TABLE:new Array(256),LOG_TABLE:new Array(256)},c=0;c<8;c++)d.EXP_TABLE[c]=1<>>7-t%8&1)},put:function(t,e){for(var r=0;r>>e-r-1&1))},getLengthInBits:function(){return this.length},putBit:function(t){var e=Math.floor(this.length/8);this.buffer.length<=e&&this.buffer.push(0),t&&(this.buffer[e]|=128>>>this.length%8),this.length++}};var v=[[17,14,11,7],[32,26,20,14],[53,42,32,24],[78,62,46,34],[106,84,60,44],[134,106,74,58],[154,122,86,64],[192,152,108,84],[230,180,130,98],[271,213,151,119],[321,251,177,137],[367,287,203,155],[425,331,241,177],[458,362,258,194],[520,412,292,220],[586,450,322,250],[644,504,364,280],[718,560,394,310],[792,624,442,338],[858,666,482,382],[929,711,509,403],[1003,779,565,439],[1091,857,611,461],[1171,911,661,511],[1273,997,715,535],[1367,1059,751,593],[1465,1125,805,625],[1528,1190,868,658],[1628,1264,908,698],[1732,1370,982,742],[1840,1452,1030,790],[1952,1538,1112,842],[2068,1628,1168,898],[2188,1722,1228,958],[2303,1809,1283,983],[2431,1911,1351,1051],[2563,1989,1423,1093],[2699,2099,1499,1139],[2809,2213,1579,1219],[2953,2331,1663,1273]];function C(){var t=!1,e=navigator.userAgent;if(/android/i.test(e)){t=!0;var r=e.toString().match(/android ([0-9]\.[0-9])/i);r&&r[1]&&(t=parseFloat(r[1]))}return t}var w=function(){var t=function(t,e){this._el=t,this._htOption=e};return t.prototype.draw=function(t){var e=this._htOption,r=this._el,o=t.getModuleCount();Math.floor(e.width/o),Math.floor(e.height/o);function i(t,e){var r=document.createElementNS("http://www.w3.org/2000/svg",t);for(var o in e)e.hasOwnProperty(o)&&r.setAttribute(o,e[o]);return r}this.clear();var n=i("svg",{viewBox:"0 0 "+String(o)+" "+String(o),width:"100%",height:"100%",fill:e.colorLight});n.setAttributeNS("http://www.w3.org/2000/xmlns/","xmlns:xlink","http://www.w3.org/1999/xlink"),r.appendChild(n),n.appendChild(i("rect",{fill:e.colorLight,width:"100%",height:"100%"})),n.appendChild(i("rect",{fill:e.colorDark,width:"1",height:"1",id:"template"}));for(var a=0;a'],s=0;s");for(var h=0;h');a.push("")}a.push(""),r.innerHTML=a.join("");var l=r.childNodes[0],u=(e.width-l.offsetWidth)/2,f=(e.height-l.offsetHeight)/2;u>0&&f>0&&(l.style.margin=f+"px "+u+"px")},t.prototype.clear=function(){this._el.innerHTML=""},t}():function(){function t(){this._elImage.src=this._elCanvas.toDataURL("image/png"),this._elImage.style.display="block",this._elCanvas.style.display="none"}if(this._android&&this._android<=2.1){var e=1/window.devicePixelRatio,r=CanvasRenderingContext2D.prototype.drawImage;CanvasRenderingContext2D.prototype.drawImage=function(t,o,i,n,a,s,h,l,u){if("nodeName"in t&&/img/i.test(t.nodeName))for(var f=arguments.length-1;f>=1;f--)arguments[f]=arguments[f]*e;else void 0===l&&(arguments[1]*=e,arguments[2]*=e,arguments[3]*=e,arguments[4]*=e);r.apply(this,arguments)}}var o=function(t,e){this._bIsPainted=!1,this._android=C(),this._htOption=e,this._elCanvas=document.createElement("canvas"),this._elCanvas.width=e.width,this._elCanvas.height=e.height,t.appendChild(this._elCanvas),this._el=t,this._oContext=this._elCanvas.getContext("2d"),this._bIsPainted=!1,this._elImage=document.createElement("img"),this._elImage.alt="Scan me!",this._elImage.style.display="none",this._el.appendChild(this._elImage),this._bSupportDataURI=null};return o.prototype.draw=function(t){var e=this._elImage,r=this._oContext,o=this._htOption,i=t.getModuleCount(),n=o.width/i,a=o.height/i,s=Math.round(n),h=Math.round(a);e.style.display="none",this.clear();for(var l=0;lv.length)throw new Error("Too long data");return r}(QRCode=function(t,e){if(this._htOption={width:256,height:256,typeNumber:4,colorDark:"#000000",colorLight:"#ffffff",correctLevel:o.H},"string"==typeof e&&(e={text:e}),e)for(var r in e)this._htOption[r]=e[r];"string"==typeof t&&(t=document.getElementById(t)),this._htOption.useSVG&&(D=w),this._android=C(),this._el=t,this._oQRCode=null,this._oDrawing=new D(this._el,this._htOption),this._htOption.text&&this.makeCode(this._htOption.text)}).prototype.makeCode=function(t){this._oQRCode=new e(A(t,this._htOption.correctLevel),this._htOption.correctLevel),this._oQRCode.addData(t),this._oQRCode.make(),this._el.title=t,this._oDrawing.draw(this._oQRCode),this.makeImage()},QRCode.prototype.makeImage=function(){"function"==typeof this._oDrawing.makeImage&&(!this._android||this._android>=3)&&this._oDrawing.makeImage()},QRCode.prototype.clear=function(){this._oDrawing.clear()},QRCode.CorrectLevel=o}(),"undefined"!=typeof module&&(module.exports=QRCode);
2 |
--------------------------------------------------------------------------------