AmCharts.AmExport = AmCharts.Class({ construct: function (chart,cfg) { var _this = this; _this.DEBUG = false; _this.chart = chart; _this.canvas = null; _this.svgs = []; _this.cfg = { menuTop : 'auto', menuLeft : 'auto', menuRight : '0px', menuBottom : '0px', menuItems : [{ textAlign : 'center', icon : _this.chart.pathToImages + 'export.png', iconTitle : 'Save chart as an image', format : 'png' }], menuItemStyle : { backgroundColor : 'transparent', rollOverBackgroundColor : '#EFEFEF', color : '#000000', rollOverColor : '#CC0000', paddingTop : '6px', paddingRight : '6px', paddingBottom : '6px', paddingLeft : '6px', marginTop : '0px', marginRight : '0px', marginBottom : '0px', marginLeft : '0px', textAlign : 'left', textDecoration : 'none', fontFamily : _this.chart.fontFamily, fontSize : _this.chart.fontSize + 'px' }, menuItemOutput : { backgroundColor : '#FFFFFF', fileName : 'amChart', format : 'png', output : 'dataurlnewwindow', render : 'browser', dpi : 90, onclick : function(instance,config,event) { instance.output(config); } }, removeImagery : true }; _this.processing = { buffer : [], drawn : 0, timer : 0 } // Config dependency adaption if ( typeof(window.canvg) != 'undefined' && typeof(window.RGBColor) != 'undefined' ) { _this.cfg.menuItemOutput.render = 'canvg'; } if ( typeof(window.saveAs) != 'undefined' ) { _this.cfg.menuItemOutput.output = 'save'; } if ( AmCharts.isIE && AmCharts.IEversion < 10 ) { _this.cfg.menuItemOutput.output = 'dataurlnewwindow'; } // Merge given configs if ( cfg ) { cfg.menuItemOutput = AmCharts.extend(_this.cfg.menuItemOutput,cfg.menuItemOutput || {}); cfg.menuItemStyle = AmCharts.extend(_this.cfg.menuItemStyle,cfg.menuItemStyle || {}); _this.cfg = AmCharts.extend(_this.cfg,cfg); } // Add reference to chart _this.chart.AmExport = _this; // Listen to the drawer _this.chart.addListener('rendered',function() { _this.setup(); }); // DEBUG; Public reference if ( _this.DEBUG ) { window.AmExport = _this }; }, /* Simple log function for internal purpose @param **args */ log: function() { console.log('AmExport: ',arguments); }, /* PUBLIC Prepares everything to get exported @param none */ setup: function() { var _this = this; if ( _this.DEBUG == 10 ) { _this.log('SETUP START'); } // DEBUG if ( !AmCharts.isIE || ( AmCharts.isIE && AmCharts.IEversion > 9 ) ) { window.clearTimeout(_this.processing.timer); _this.processing.timer = setTimeout(function() { // Polify SVG; needs to wait _this.polifySVG(); // Build Buttons _this.generateButtons(); if ( _this.DEBUG == 10 ) { _this.log('SETUP END'); } // DEBUG },1000); } else { if ( _this.DEBUG == 10 ) { _this.log('< IE10 NOT SUPPORTED'); } // DEBUG } }, /* PUBLIC Decodes base64 string to binary array @param base64_string @copyright Eli Grey, http://eligrey.com and Devin Samarin, https://github.com/eboyjr */ generateBinaryArray: function(base64_string) { var len = base64_string.length , buffer = new Uint8Array(len / 4 * 3 | 0) , i = 0 , outptr = 0 , last = [0, 0] , state = 0 , save = 0 , rank , code , undef , base64_ranks = new Uint8Array([ 62, -1, -1, -1, 63, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, -1 , -1, -1, 0, -1, -1, -1, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9 , 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25 , -1, -1, -1, -1, -1, -1, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35 , 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51 ]); while (len--) { code = base64_string.charCodeAt(i++); rank = base64_ranks[code-43]; if (rank !== 255 && rank !== undef) { last[1] = last[0]; last[0] = code; save = (save << 6) | rank; state++; if (state === 4) { buffer[outptr++] = save >>> 16; if (last[1] !== 61 /* padding character */) { buffer[outptr++] = save >>> 8; } if (last[0] !== 61 /* padding character */) { buffer[outptr++] = save; } state = 0; } } } // 2/3 chance there's going to be some null bytes at the end, but that // doesn't really matter with most image formats. // If it somehow matters for you, truncate the buffer up outptr. return buffer; }, /* Creates blob object @param base64_datastring string @param type string */ generateBlob: function(datastring,type) { var _this = this, header_end = datastring.indexOf(',') + 1, header = datastring.substring(0,header_end), data = datastring, blob = new Blob(); if ( header.indexOf('base64') != -1 ) { data = _this.generateBinaryArray(datastring.substring(header_end)); } // Fake blob for IE if ( AmCharts.isIE && AmCharts.IEversion < 10 ) { blob.data = data; blob.size = data.length; blob.type = type; blob.encoding = 'base64'; } else { blob = new Blob([data],{type: type}); } return blob }, /* Creates PDF object @param config object */ generatePDF: function(cfg) { var _this = this, pdf = {output: function() {return ''}}, data = _this.canvas.toDataURL('image/jpeg'), // JSPDF ONLY SUPPORTS JPG width = (_this.canvas.width * 25.4) / cfg.dpi, height = (_this.canvas.height * 25.4) / cfg.dpi; // Check if ( window.jsPDF ) { pdf = new jsPDF(); if ( pdf.addImage ) { pdf.addImage(data, 'JPEG', 0, 0, width, height); } else { alert("Missing jsPDF plugin; Please add the 'addImage' plugin."); } } else { alert("Missing jsPDF lib; Don't forget to add the addImage plugin."); } return pdf; }, /* Creates the CANVAS to receive the image data @param format void() @param callback; given callback function which returns the blob or datastring of the configured ouput type */ output: function(cfg,externalCallback) { var _this = this, cfg = AmCharts.extend(AmCharts.extend({},_this.cfg.menuItemOutput),cfg||{}); /* PRIVATE Callback function which gets called after the drawing process is done @param none */ function internalCallback() { var data = null; if ( _this.DEBUG == 10 ) { _this.log('OUTPUT',format); } // DEBUG // SVG if ( cfg.format == 'image/svg+xml' || cfg.format == 'svg' ) { for ( var i = 0; i < _this.processing.buffer.length; i++ ) { data = new XMLSerializer().serializeToString(_this.processing.buffer[i][0]), blob = _this.generateBlob(data,'image/svg+xml'); if ( cfg.output == 'save' ) { saveAs(blob, cfg.fileName + '.svg'); } else if ( cfg.output == 'datastring' || cfg.output == 'datauristring' || cfg.output == 'dataurlstring' ) { blob = 'data:image/svg+xml;base64,' + btoa(data); } else if ( cfg.output == 'dataurlnewwindow' ) { window.open('data:image/svg+xml;base64,' + btoa(data)); } else if ( cfg.output == 'datauri' || cfg.output == 'dataurl' ) { location.href = 'data:image/svg+xml;base64,' + btoa(data); } else if ( cfg.output == 'datastream' ) { location.href = 'data:image/octet-stream;base64,' + btoa(data); } if ( externalCallback ) externalCallback.apply(_this,[blob]); } // PDF } else if ( cfg.format == 'application/pdf' || cfg.format == 'pdf' ) { data = _this.generatePDF(cfg).output('dataurlstring'), blob = _this.generateBlob(data,'application/pdf'); if ( cfg.output == 'save' ) { saveAs(blob, cfg.fileName + '.pdf'); } else if ( cfg.output == 'datastring' || cfg.output == 'datauristring' || cfg.output == 'dataurlstring' ) { blob = data; } else if ( cfg.output == 'dataurlnewwindow' ) { window.open(data); } else if ( cfg.output == 'datauri' || cfg.output == 'dataurl' ) { location.href = data; } else if ( cfg.output == 'datastream' ) { location.href = data.replace('application/pdf','application/octet-stream'); } if ( externalCallback ) externalCallback.apply(_this,[blob]); // PNG } else if ( cfg.format == 'image/png' || cfg.format == 'png' ) { data = _this.canvas.toDataURL('image/png'), blob = _this.generateBlob(data,'image/png'); if ( cfg.output == 'save' ) { saveAs(blob, cfg.fileName + '.png'); } else if ( cfg.output == 'datastring' || cfg.output == 'datauristring' || cfg.output == 'dataurlstring' ) { blob = data; } else if ( cfg.output == 'dataurlnewwindow' ) { window.open(data); } else if ( cfg.output == 'datauri' || cfg.output == 'dataurl' ) { location.href = data; } else if ( cfg.output == 'datastream' ) { location.href = data.replace('image/png','image/octet-stream'); } if ( externalCallback ) externalCallback.apply(_this,[blob]); // JPG } else if ( cfg.format == 'image/jpeg' || cfg.format == 'jpeg' || cfg.format == 'jpg' ) { data = _this.canvas.toDataURL('image/jpeg'), blob = _this.generateBlob(data,'image/jpeg'); if ( cfg.output == 'save' ) { saveAs(blob, cfg.fileName + '.jpg'); } else if ( cfg.output == 'datastring' || cfg.output == 'datauristring' || cfg.output == 'dataurlstring' ) { blob = data; } else if ( cfg.output == 'dataurlnewwindow' ) { window.open(data); } else if ( cfg.output == 'datauri' || cfg.output == 'dataurl' ) { location.href = data; } else if ( cfg.output == 'datastream' ) { location.href = data.replace('image/jpeg','image/octet-stream'); } if ( externalCallback ) externalCallback.apply(_this,[blob]); } } return _this.generateOutput(cfg,internalCallback); }, /* PUBLIC Polifies missing attributes to the SVG and replaces images to embedded base64 images @param none */ polifySVG: function() { var _this = this; var svgs = _this.chart.div.getElementsByTagName('svg'); // Recursive function to force the attributes function recursiveChange(svg,tag) { var items = svg.getElementsByTagName(tag); for ( var i = 0; i < items.length; i++) { if ( _this.cfg.removeImagery ) { items[i].parentNode.removeChild(items[i]); } else { var image = document.createElement('img'); var canvas = document.createElement('canvas'); var ctx = canvas.getContext('2d'); canvas.width = items[i].getAttribute('width'); canvas.height = items[i].getAttribute('height'); image.src = items[i].getAttribute('xlink:href'); image.width = items[i].getAttribute('width'); image.height = items[i].getAttribute('height'); try { ctx.drawImage(image,0,0,image.width,image.height); datastring = canvas.toDataURL(); // image.src; // canvas.toDataURL(); // } catch(err) { datastring = image.src; // image.src; // canvas.toDataURL(); // _this.log('Tainted canvas, reached browser CORS security; origin from imagery must be equal to the server!'); throw new Error(err); } items[i].setAttribute('xlink:href',datastring); } if ( _this.DEBUG == 10 ) { _this.log('POLIFIED',items[i]); } // DEBUG } } // Loop through svgs to add some standardization for ( var i = 0; i < svgs.length; i++ ) { var parent = svgs[i].parentNode; // Put some attrs to it /* if ( !AmCharts.isIE ) { svgs[i].setAttribute('xmlns','http://www.w3.org/2000/svg'); svgs[i].setAttribute('xmlns:xlink','http://www.w3.org/1999/xlink'); } svgs[i].setAttribute('width',parent.style.width); svgs[i].setAttribute('height',parent.style.height); */ if ( _this.DEBUG == 10 ) { _this.log('POLIFIED',svgs[i]); } // DEBUG // Force link adaption recursiveChange(svgs[i],'pattern'); recursiveChange(svgs[i],'image'); _this.svgs.push(svgs[i]); } return svgs; }, /* PUBLIC Generates the canvas with the given SVGs and configured renderer @param callback; function(); gets called after drawing process on the canvas has been finished */ generateOutput: function(cfg,callback) { var _this = this, svgs = _this.chart.div.getElementsByTagName('svg'), canvas = document.createElement('canvas'), context = canvas.getContext('2d'), offset = { y: 0, x: 0 }, tmp = {}; // Reset _this.processing.buffer = []; _this.processing.drawn = 0; _this.canvas = canvas; // Walkthroug SVGs if ( _this.DEBUG == 10 ) { _this.log('START EXPORT'); } // DEBUG if ( _this.DEBUG == 10 ) { _this.log('START BUFFERING'); } // DEBUG for ( var i = 0; i < svgs.length; i++ ) { var parent = svgs[i].parentNode, svgX = Number(parent.style.left.slice(0,-2)), svgY = Number(parent.style.top.slice(0,-2)); tmp = AmCharts.extend({},offset); // Overtake parent position if givwn offset.x = svgX?svgX:offset.x; offset.y = svgY?svgY:offset.y; _this.processing.buffer.push([svgs[i],AmCharts.extend({},offset)]); // Put back from "cache" if ( svgY&&svgX ) { offset = tmp; // New offset for next one } else { offset.y += svgY?0:parent.offsetHeight; } if ( _this.DEBUG == 10 ) { _this.log('BUFFERED',svgs[i],offset); } // DEBUG } if ( _this.DEBUG == 10 ) { _this.log('END BUFFERING'); } // DEBUG // Apply background if ( _this.DEBUG == 10 ) { _this.log('START DRAWING',cfg.render); } // DEBUG if ( _this.DEBUG == 10 ) { _this.log('FILL BACKGROUND'); } // DEBUG canvas.id = AmCharts.getUniqueId(); canvas.width = _this.chart.divRealWidth; canvas.height = _this.chart.divRealHeight; // Set given background; jpeg default if ( cfg.backgroundColor || format == 'image/jpeg' ) { context.fillStyle = cfg.backgroundColor || '#FFFFFF'; context.fillRect(0,0,canvas.width,canvas.height); } /* PRIVATE Recursive function to draw the images to the canvas; @param none; */ function drawItWhenItsLoaded() { var img,buffer,offset,source; // DRAWING PROCESS DONE if ( _this.processing.buffer.length == _this.processing.drawn ) { if ( _this.DEBUG == 10 ) { _this.log('END DRAWING'); } // DEBUG return callback(); // LOOPING LUI } else { if ( _this.DEBUG == 10 ) { _this.log('DRAW',_this.processing.drawn + 1,'OF',_this.processing.buffer.length); } // DEBUG buffer = _this.processing.buffer[_this.processing.drawn]; source = new XMLSerializer().serializeToString(buffer[0]); //source = 'data:image/svg+xml;base64,' + btoa(); offset = buffer[1]; if ( _this.DEBUG == 10 ) { _this.log('SOURCE',source); } // DEBUG // NATIVE if ( cfg.render == 'browser' ) { img = new Image(); img.id = AmCharts.getUniqueId(); source = 'data:image/svg+xml;base64,' + btoa(source); //img.crossOrigin = "Anonymous"; img.onload = function() { context.drawImage(this,buffer[1].x,buffer[1].y); _this.processing.drawn++; if ( _this.DEBUG == 10 ) { _this.log('ONLOAD',this); } // DEBUG drawItWhenItsLoaded(); } img.onerror = function() { if ( _this.DEBUG == 10 ) { _this.log('ONERROR',this); } // DEBUG context.drawImage(this,buffer[1].x,buffer[1].y); _this.processing.drawn++; drawItWhenItsLoaded(); } img.src = source; if ( _this.DEBUG == 10 ) { _this.log('ADD',img); } // DEBUG if ( img.complete || typeof(img.complete) == 'undefined' || img.complete === undefined ) { if ( _this.DEBUG == 10 ) { _this.log('FORCE ONLOAD',img); } // DEBUG img.src = ""; img.src = source; } // CANVG } else if ( cfg.render == 'canvg' ) { canvg(canvas,source,{ offsetX : offset.x, offsetY : offset.y, ignoreMouse : true, ignoreAnimation : true, ignoreDimensions : true, ignoreClear : true, renderCallback : function() { _this.processing.drawn++; drawItWhenItsLoaded(); } }); } } } return drawItWhenItsLoaded(); }, /* Generates the export menu to trigger the exportation @param none; */ generateButtons: function() { var _this = this, div = document.createElement('div'), lvl = 0; // Push sublings function createList(items) { var ul = document.createElement('ul'); ul.setAttribute('style','list-style: none; margin: 0; padding: 0;'); // Walkthrough items for ( var i = 0; i < items.length; i++ ) { var li = document.createElement('li'), img = document.createElement('img'), a = document.createElement('a'), item = items[i], children = null, itemStyle = AmCharts.extend(AmCharts.extend({},_this.cfg.menuItemStyle),items[i]); // MERGE CFG item = AmCharts.extend(AmCharts.extend({},_this.cfg.menuItemOutput),item); // ICON if ( item['icon'] ) { img.alt = ''; img.src = item['icon']; img.setAttribute('style','margin: 0 auto;border: none;outline: none'); if ( item['iconTitle'] ) { img.title=item['iconTitle']; } a.appendChild(img); } // TITLE; STYLING a.href = '#'; if ( item['title'] ) { img.setAttribute('style','margin-right: 5px;'); a.innerHTML += item.title; } a.setAttribute('style','display: block;'); AmCharts.extend(a.style,itemStyle) // ONCLICK a.onclick = item.onclick.bind(a,_this,item); li.appendChild(a); // APPEND SIBLINGS if ( item.items ) { children = createList(item.items); li.appendChild(children); li.onmouseover = function() { children.style.display = 'block'; } li.onmouseout = function() { children.style.display = 'none'; } children.style.display = 'none'; } // Append to parent ul.appendChild(li); // Apply hover a.onmouseover = function() { this.style.backgroundColor = itemStyle.rollOverBackgroundColor; this.style.color = itemStyle.rollOverColor; this.style.borderColor = itemStyle.rollOverBorderColor; } a.onmouseout = function() { this.style.backgroundColor = itemStyle.backgroundColor; this.style.color = itemStyle.color; this.style.borderColor = itemStyle.borderColor; } } lvl++; if ( _this.DEBUG == 10 ) { _this.log('MENU',ul); } // DEBUG return ul; } // Style wrapper; Push into chart div div.setAttribute('style','position: absolute;top:'+ _this.cfg.menuTop +';right:'+ _this.cfg.menuRight +';bottom:'+ _this.cfg.menuBottom +';left:'+ _this.cfg.menuLeft +';box-shadow:0px 0px 1px 0px rgba(0,0,0,0);'); div.appendChild(createList(_this.cfg.menuItems)); _this.chart.div.style.position = 'relative'; _this.chart.div.appendChild(div); } });