/* ::::::::::::::::::::::::::::::::::::::::::::::::::: 2019 July Edited for PayMe PayCode Design Guideline by tomyccc ::::::::::::::::::::::::::::::::::::::::::::::::::: */ (function( $ ){ $.fn.qrcode = function(options) { // if options is string, if( typeof options === 'string' ){ options = { text: options }; } // set default values // typeNumber < 1 for automatic calculation options = $.extend( {}, { render : "canvas", payme : false, payme_custom_logo_url : ETICKETS.payment.payme.guideline.qrIcon.url, //ETICKETS.payment.payme.guideline.paymeIcon.url width : 256, height : 256, typeNumber : -1, correctLevel : QRErrorCorrectLevel.H, background : "#ffffff", foreground : "#000000" }, options); // ------------- PayMe ------ var drawPayCode = function(qrcode, canvas, margin, logo, consumer = true) { var ctx = canvas.getContext('2d'); var width = ctx.canvas.width; var cellCount = qrcode.getModuleCount(); var cellRadius = ((width - (margin * 2)) / cellCount) / 2; var offset = cellRadius + margin var iconPercent = 32/260; var iconClipPercent = 35/260; var logoPercent = 67/260; var logoClipPercent = 72/260; // white background //ctx.fillStyle = '#ffffff'; //ctx.fillRect(0, 0, width, width); ctx.fillStyle = ((consumer) ? '#db0011' : '#c92a23'); // top left eye drawEye(ctx, width, margin, cellRadius, 0, 0, offset); // top right eye drawEye(ctx, width, margin, cellRadius, 2*(cellCount - 7)*cellRadius, 0, offset); // bottom left eye drawEye(ctx, width, margin, cellRadius, 0, 2*(cellCount - 7)*cellRadius, offset); var iconWidth = ((width - 2*margin) * iconPercent) var logoWidth = ((width - 2*margin) * logoPercent); var iconClip = width - cellRadius - margin - ((width - 2*margin) * iconClipPercent) var logoLeftClip = (width / 2) - (((width - 2*margin) * logoClipPercent) / 2); var logoRightClip = (width / 2) + (((width - 2*margin) * logoClipPercent) / 2); // PayMe icon in bottom right drawIcon(ctx, iconWidth, width, margin, consumer); // business logo in the middle drawLogo(ctx, logo, logoWidth, width, margin, consumer); for (var r = 0; r < cellCount; r += 1) { for (var c = 0; c < cellCount; c += 1) { var x = c*cellRadius*2 + offset; var y = r*cellRadius*2 + offset; if (r < 7 && (c < 7 || c > cellCount - 8) || r > cellCount - 8 && c < 7) { // don't draw cells over the "eyes" continue; } else if (x >= iconClip && y >= iconClip) { // don't draw cells over the PM4B icon continue; } else if (x >= logoLeftClip && x < logoRightClip && y >= logoLeftClip && y < logoRightClip ) { // don't draw cells over the logo continue; } if (qrcode.isDark(r, c)) { ctx.beginPath(); ctx.arc(x, y, cellRadius, 0, 2*Math.PI); ctx.fill(); } } } } var drawEye = function(ctx, width, margin, cellRadius, xOffset, yOffset, offset) { // outer edge ctx.beginPath(); ctx.moveTo(xOffset + 11*cellRadius + margin, yOffset + 0*cellRadius + margin); ctx.arcTo( xOffset + 14*cellRadius + margin, yOffset + 0*cellRadius + margin, xOffset + 14*cellRadius + margin, yOffset + 3*cellRadius + margin, 3*cellRadius); ctx.arcTo( xOffset + 14*cellRadius + margin, yOffset + 14*cellRadius + margin, xOffset + 12*cellRadius + margin, yOffset + 14*cellRadius + margin, 3*cellRadius); ctx.arcTo( xOffset + 0*cellRadius + margin, yOffset + 14*cellRadius + margin, xOffset + 0*cellRadius + margin, yOffset + 11*cellRadius + margin, 3*cellRadius); ctx.arcTo( xOffset + 0*cellRadius + margin, yOffset + 0*cellRadius + margin, xOffset + 3*cellRadius + margin, yOffset + 0*cellRadius + margin, 3*cellRadius); ctx.lineTo(xOffset + 11*cellRadius + margin, yOffset + 0*cellRadius + margin); // inner edge ctx.moveTo(xOffset + 4*cellRadius + margin, yOffset + 2*cellRadius + margin); ctx.arcTo( xOffset + 2*cellRadius + margin, yOffset + 2*cellRadius + margin, xOffset + 2*cellRadius + margin, yOffset + 4*cellRadius + margin, 2*cellRadius); ctx.arcTo( xOffset + 2*cellRadius + margin, yOffset + 12*cellRadius + margin, xOffset + 4*cellRadius + margin, yOffset + 12*cellRadius + margin, 2*cellRadius); ctx.arcTo( xOffset + 12*cellRadius + margin, yOffset + 12*cellRadius + margin, xOffset + 12*cellRadius + margin, yOffset + 10*cellRadius + margin, 2*cellRadius); ctx.arcTo( xOffset + 12*cellRadius + margin, yOffset + 2*cellRadius + margin, xOffset + 10*cellRadius + margin, yOffset + 2*cellRadius + margin, 2*cellRadius); ctx.lineTo(xOffset + 4*cellRadius + margin, yOffset + 2*cellRadius + margin); ctx.closePath(); ctx.fill(); // central rect ctx.beginPath(); ctx.moveTo(xOffset + 8*cellRadius + margin, yOffset + 4*cellRadius + margin); ctx.arcTo( xOffset + 10*cellRadius + margin, yOffset + 4*cellRadius + margin, xOffset + 10*cellRadius + margin, yOffset + 6*cellRadius + margin, cellRadius); ctx.arcTo( xOffset + 10*cellRadius + margin, yOffset + 10*cellRadius + margin, xOffset + 8*cellRadius + margin, yOffset + 10*cellRadius + margin, cellRadius); ctx.arcTo( xOffset + 4*cellRadius + margin, yOffset + 10*cellRadius + margin, xOffset + 4*cellRadius + margin, yOffset + 8*cellRadius + margin, cellRadius); ctx.arcTo( xOffset + 4*cellRadius + margin, yOffset + 4*cellRadius + margin, xOffset + 6*cellRadius + margin, yOffset + 4*cellRadius + margin, cellRadius); ctx.lineTo(xOffset + 8*cellRadius + margin, yOffset + 4*cellRadius + margin); ctx.closePath(); ctx.fill(); } var drawIcon = function(ctx, iconWidth, width, margin, consumer = true) { consumer = false; var icon = new Image(); var xOffset = width - margin - iconWidth; var yOffset = width - margin - iconWidth; // Apple design guidelines state that corner radius is 80px for a 512px icon var cornerRadius = iconWidth * 80/512; var edgeLength = iconWidth - cornerRadius; icon.onload = function(){ ctx.save(); ctx.beginPath(); ctx.moveTo(xOffset + edgeLength, yOffset); ctx.arcTo( xOffset + edgeLength + cornerRadius, yOffset, xOffset + edgeLength + cornerRadius, yOffset + cornerRadius, cornerRadius); ctx.arcTo( xOffset + edgeLength + cornerRadius, yOffset + edgeLength + cornerRadius, xOffset + edgeLength, yOffset + edgeLength + cornerRadius, cornerRadius); ctx.arcTo( xOffset, yOffset + edgeLength + cornerRadius, xOffset, yOffset + edgeLength, cornerRadius); ctx.arcTo( xOffset, yOffset, xOffset + cornerRadius, yOffset, cornerRadius); ctx.lineTo(xOffset + edgeLength, yOffset); ctx.closePath(); ctx.clip(); ctx.drawImage(icon, xOffset, yOffset, iconWidth, iconWidth); if (!consumer) { ctx.strokeStyle = "#767676"; ctx.lineWidth = 0.5; ctx.stroke(); } ctx.restore(); }; if (consumer) { icon.src = "" } else { icon.src = ''; } } var drawLogo = function(ctx, img, logoWidth, width, margin, consumer = true) { ctx.save(); ctx.beginPath(); ctx.arc(width / 2, width / 2, logoWidth / 2, 0, 2*Math.PI); ctx.closePath(); ctx.clip(); ctx.drawImage(img, (width - logoWidth) / 2, (width - logoWidth) / 2, logoWidth, logoWidth); if (!consumer) { // match PM4B app ctx.strokeStyle = "#e9e9e9"; ctx.lineWidth = width * 0.015; ctx.stroke(); } ctx.restore(); } var suggestTypeNumber = function(text){ var length = text.length; if (length <= 32) { return 3; } else if (length <= 46) { return 4; } else if (length <= 60) { return 5; } else if (length <= 74) { return 6; } else if (length <= 86) { return 7; } else if (length <= 108) { return 8; } else if (length <= 130) { return 9; } else if (length <= 151) { return 10; } else if (length <= 177) { return 11; } else if (length <= 203) { return 12; } else if (length <= 241) { return 13; } else if (length <= 258) { return 14; } else if (length <= 292) { return 15; } else { return 40; } } // var suggestTypeNumber(text) = function(){ // === END ----- PayMe ------ var createCanvas = function(){ // create the qrcode itself var qrcode if(options.payme){ options.text = options.text.replace(/^[\s\u3000]+|[\s\u3000]+$/g, ''); var typeNumber =((options.typeNumber == 0) ? suggestTypeNumber(options.text) : options.typeNumber); qrcode = new QRCode(typeNumber || 4, options.correctLevel || 'M'); } else { // if(options.payme){ qrcode = new QRCode(options.typeNumber, options.correctLevel); } // if(options.payme){ if(options.payme){ // payme qrcode.addData(options.text, 'Byte'); } else { // if(options.payme){ qrcode.addData(options.text); } // if(options.payme){ qrcode.make(); // create canvas element var canvas = document.createElement('canvas'); canvas.width = options.width; canvas.height = options.height; var ctx = canvas.getContext('2d'); if(options.payme){ // payme ctx.setTransform(1, 0, 0, 1, 0, 0); ctx.clearRect(0, 0, canvas.width, canvas.height); var icon_url = options.payme_custom_logo_url; var logo = new Image(); if(!icon_url){ drawPayCode(qrcode, canvas, 7, logo, true); } else { // if(!icon_url){ logo.onload = function() { drawPayCode(qrcode, canvas, 7, this, true); } // logo.onload = function() { logo.src = icon_url; } // if(!icon_url){ } else { // if(options.payme){ // compute tileW/tileH based on options.width/options.height var tileW = options.width / qrcode.getModuleCount(); var tileH = options.height / qrcode.getModuleCount(); // draw in the canvas for( var row = 0; row < qrcode.getModuleCount(); row++ ){ for( var col = 0; col < qrcode.getModuleCount(); col++ ){ ctx.fillStyle = qrcode.isDark(row, col) ? options.foreground : options.background; var w = (Math.ceil((col+1)*tileW) - Math.floor(col*tileW)); var h = (Math.ceil((row+1)*tileH) - Math.floor(row*tileH)); ctx.fillRect(Math.round(col*tileW),Math.round(row*tileH), w, h); } } } // if(options.payme){ // return just built canvas return canvas; } // from Jon-Carlos Rivera (https://github.com/imbcmdth) var createTable = function(){ // create the qrcode itself var qrcode = new QRCode(options.typeNumber, options.correctLevel); qrcode.addData(options.text); qrcode.make(); // create table element var $table = $('
') .css("width", options.width+"px") .css("height", options.height+"px") .css("border", "0px") .css("border-collapse", "collapse") .css('background-color', options.background); // compute tileS percentage var tileW = options.width / qrcode.getModuleCount(); var tileH = options.height / qrcode.getModuleCount(); // draw in the table for(var row = 0; row < qrcode.getModuleCount(); row++ ){ var $row = $('').css('height', tileH+"px").appendTo($table); for(var col = 0; col < qrcode.getModuleCount(); col++ ){ $('') .css('width', tileW+"px") .css('background-color', qrcode.isDark(row, col) ? options.foreground : options.background) .appendTo($row); } } // return just built canvas return $table; } return this.each(function(){ var element = options.render == "canvas" ? createCanvas() : createTable(); $(element).appendTo(this); }); }; })( jQuery );