背景:项目现场提出将一个html做的图形页面导出为一张图片的需求,在网上搜了一下,发现都不是很全面,所以综合了很多大神的帖子,自己再次封装,以适用项目需求。

所需js库:html2canvas.js(https://github.com/niklasvh/html2canvas); Export2Image.js(自己封装),其中new Export2Image(htmlDom,[opts]).export()是执行导出命令。opts = {width:400,height:400,type:"png",fname:"downloadName"}为可选参数。

注意:在使用过程中,发现一个问题,dom背景图片千万不能设置background-size:contain/conver,否则背景图片导不出来。

代码:

 /**
* Created by tengri on 2016-5-9.
*/ /**
* 导出类
* @param content 要导出的内容
* @constructor
*/
function Export2Image(content,opts){
this.exportObj = typeof(content) == "string" ? document.getElementById(content) : content;
if(!this.exportObj) throw new Error("导出内容对象只能传递ID和DOM对象");
this.opts = opts || {};
if(this.exportObj.nodeName !="CANVAS"){
this.exportType = "html2Image";
this.canvas = document.createElement("canvas");
this.canvas.style.display = "none";
//如果没有设置宽度和高度,实际是多大就导出多大
this.canvas.width = this.opts.width || this.exportObj.scrollWidth + 10;
this.canvas.height = this.opts.height || this.exportObj.scrollHeight + 10;
}else{
this.exportType = "canvas2Image";
this.canvas = this.exportObj;
}
if(this.opts.width && this.opts.height){
this.actualWidth = this.opts.width;
this.actualHeight = this.opts.height;
}
this.type = this.opts.type || "png";
this.fileName = (this.opts.name || new Date().getTime()) + "." + this.type;
this.init();
return this;
} /**
* 初始化
*/
Export2Image.prototype.init = function(){
this._fixType();
} Export2Image.prototype._encodeData = function(data){
if(!window.btoa) throw "btoa undefined";
var strDtata = "";
if(typeof(data) !="string"){
for(var i = 0 ; i < data.length;i++){
strDtata +=String.fromCharCode(data[i]);
}
}else{
strDtata = data;
}
return window.btoa(strDtata);
}; /**
* 根据配置生成固定大小图片
* @param width
* @param height
*/
Export2Image.prototype._scaleCanvas = function(width,height){
var w = this.canvas.width;
var h = this.canvas.height;
width = width || w;
height = height || h; var newCanvas = document.createElement("canvas");
newCanvas.width = width;
newCanvas.height = height;
var ctx = newCanvas.getContext("2d");
ctx.drawImage(this.canvas,0,0,w,h,0,0,width,height);
this.canvas = newCanvas;
}; /**
* 获取canvas的Dataurl
*/
Export2Image.prototype._getDataURL = function(){
return this.canvas.toDataURL(this.type);
};
/**
* 获取导出图片类型
* @private
*/
Export2Image.prototype._fixType = function(){
var type = this.type.toLocaleLowerCase().replace(/jpg/i,"jpeg");
var res = type.match(/png|jpeg|bmp|gif/)[0];
this.type = "image/" + res;
}; /**
* 获取数据
*/
Export2Image.prototype.getData = function(){
if(this.actualWidth && this.actualHeight){
this._scaleCanvas(this.actualWidth,this.actualHeight);
}
var strData = "";
if (/bmp/.test(this.type)) {
var data = this._getImageData();
strData = this._getBitmapImage(data);
} else {
strData = this._getDataURL();
}
return strData;
} /**
* 普通图片获取
* @private
*/
Export2Image.prototype._getImageData = function(){
var w = this.canvas.width, h = this.canvas.height;
return this.canvas.getContext('2d').getImageData(0, 0, w, h);
}; /**
* 位图获取
* @private
*/
Export2Image.prototype._getBitmapImage = function(oData){
var aHeader = []; var iWidth = oData.width;
var iHeight = oData.height; aHeader.push(0x42); // magic 1
aHeader.push(0x4D); var iFileSize = iWidth*iHeight*3 + 54; // total header size = 54 bytes
aHeader.push(iFileSize % 256); iFileSize = Math.floor(iFileSize / 256);
aHeader.push(iFileSize % 256); iFileSize = Math.floor(iFileSize / 256);
aHeader.push(iFileSize % 256); iFileSize = Math.floor(iFileSize / 256);
aHeader.push(iFileSize % 256); aHeader.push(0); // reserved
aHeader.push(0);
aHeader.push(0); // reserved
aHeader.push(0); aHeader.push(54); // dataoffset
aHeader.push(0);
aHeader.push(0);
aHeader.push(0); var aInfoHeader = [];
aInfoHeader.push(40); // info header size
aInfoHeader.push(0);
aInfoHeader.push(0);
aInfoHeader.push(0); var iImageWidth = iWidth;
aInfoHeader.push(iImageWidth % 256); iImageWidth = Math.floor(iImageWidth / 256);
aInfoHeader.push(iImageWidth % 256); iImageWidth = Math.floor(iImageWidth / 256);
aInfoHeader.push(iImageWidth % 256); iImageWidth = Math.floor(iImageWidth / 256);
aInfoHeader.push(iImageWidth % 256); var iImageHeight = iHeight;
aInfoHeader.push(iImageHeight % 256); iImageHeight = Math.floor(iImageHeight / 256);
aInfoHeader.push(iImageHeight % 256); iImageHeight = Math.floor(iImageHeight / 256);
aInfoHeader.push(iImageHeight % 256); iImageHeight = Math.floor(iImageHeight / 256);
aInfoHeader.push(iImageHeight % 256); aInfoHeader.push(1); // num of planes
aInfoHeader.push(0); aInfoHeader.push(24); // num of bits per pixel
aInfoHeader.push(0); aInfoHeader.push(0); // compression = none
aInfoHeader.push(0);
aInfoHeader.push(0);
aInfoHeader.push(0); var iDataSize = iWidth*iHeight*3;
aInfoHeader.push(iDataSize % 256); iDataSize = Math.floor(iDataSize / 256);
aInfoHeader.push(iDataSize % 256); iDataSize = Math.floor(iDataSize / 256);
aInfoHeader.push(iDataSize % 256); iDataSize = Math.floor(iDataSize / 256);
aInfoHeader.push(iDataSize % 256); for (var i=0;i<16;i++) {
aInfoHeader.push(0); // these bytes not used
} var iPadding = (4 - ((iWidth * 3) % 4)) % 4; var aImgData = oData.data; var strPixelData = "";
var y = iHeight;
do {
var iOffsetY = iWidth*(y-1)*4;
var strPixelRow = "";
for (var x=0;x<iWidth;x++) {
var iOffsetX = 4*x; strPixelRow += String.fromCharCode(aImgData[iOffsetY+iOffsetX+2]);
strPixelRow += String.fromCharCode(aImgData[iOffsetY+iOffsetX+1]);
strPixelRow += String.fromCharCode(aImgData[iOffsetY+iOffsetX]);
}
for (var c=0;c<iPadding;c++) {
strPixelRow += String.fromCharCode(0);
}
strPixelData += strPixelRow;
} while (--y); var strEncoded = this._encodeData(aHeader.concat(aInfoHeader)) + this._encodeData(strPixelData); return this._makeDataURI(strEncoded);
}; Export2Image.prototype._makeDataURI = function(strData){
return "data:" + this.type + ";base64," + strData;
} /**
* 保存
* @param data
* @param fileName
* @private
*/
Export2Image.prototype._saveFile = function(data,fileName){
try{
//TODO:IE浏览器
new ActiveXObject("Microsoft.XMLHTTP");
var oImg = document.createElement("img");
oImg.src = data;
oImg.onload = function(){
myWindow=window.open('','_blank','width=800,height=600');
myWindow.document.write("<img src='"+data+"'>")
myWindow.focus()
}
}catch(e){
var saveLink = document.createElementNS("http://www.w3.org/1999/xhtml","a");
saveLink.href = data;
saveLink.download = fileName;
var event = document.createEvent("MouseEvents");
event.initMouseEvent('click', true, false, window, 0, 0, 0, 0, 0, false, false, false, false, 0, null);
saveLink.dispatchEvent(event);
} }; Export2Image.prototype.exportPic = function(){
if(this.exportType == "html2Image"){
if(typeof(html2canvas) !="function"){
alert("需要引入html2canvas.js库文件");
return;
}
var that = this;
html2canvas(this.exportObj, {
onrendered: function(canvas) {
that.canvas = canvas;
var data = that.getData();
var imgData = data.replace(that.type,'image/octet-stream');
that._saveFile(imgData,that.fileName);
}
});
}else{
var data = this.getData();
var imgData = data.replace(this.type,'image/octet-stream');
this._saveFile(imgData,this.fileName);
} };
 <html>
<meta http-equiv="X-UA-Compatible" content="chrome=1">
<meta charset="utf-8">
<head>
<title>dom页面导出为图片</title>
<style type="text/css">
#wrap{
width:400px;
height:300px;
line-height:300px;
font-family: "Courier New", Courier, monospace;
font-size: 18px;
text-align: center;
border: 1px solid #ccc;
background-image: url("img/a.jpg");
overflow: auto;
} </style>
</head>
<body>
<div id="wrap">这是DOM元素,相当于项目中的图形所在的div
</div>
<input type="button" value="html导出图片" onclick="htmlExportHandler();"/><br><hr/>
<canvas id="myCanvas" width="600" height="400" style="background:yellowgreen"></canvas>
<input type="button" value="canvas导出图片" onclick="canvasExportHandler();"/><br><hr/> </body>
</html>
<script type="text/javascript" src="js/html5.js"></script>
<script type="text/javascript" src="js/html2canvas.js"></script>
<script type="text/javascript" src="js/Export2Image.js"></script>
<script> window.onload = function() {
var canvas = document.getElementById("myCanvas");
var ctx=canvas.getContext('2d');
ctx.fillStyle='#FF0000';
ctx.fillRect(10,10,200,100); }; function htmlExportHandler(){
var oContent = document.getElementById("wrap");
new Export2Image(oContent,{type:"bmp",name:"html导出文件"}).export();
} function canvasExportHandler(){
new Export2Image("myCanvas").export();
} </script>                                                                                                                          

html页面、canvas导出图片的更多相关文章

  1. 利用canvas 导出图片

    1.使用canvas绘制图片,并将图片导出. 在本地直接访问静态网页时,无法使用toDataURL(),需要将网页发布后,canvas才能使用toDataURL获取画布上的内容.因为canvas不允许 ...

  2. 微信小程序 canvas导出图片模糊

    //保存到手机相册save:function () { wx.canvasToTempFilePath({ x: , y: , width: , //导出图片的宽 height: , //导出图片的高 ...

  3. canvas导出图片方法总结

    html代码 <canvas id="canvas" width="100" height="100" ></canvas ...

  4. php保存canvas导出的base64图片

    代码如下: <?php $img='data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAEYAAABxCAYAAABoUdWRAAAAAXNSR0IAr ...

  5. canvas画布导出图片并下载

    近期在学习关于画布知识,关于 画布导出图片, 在导出jpeg格式的图片时,会发现图片背景色变成了黑色,原因是画布透明的地方 默认转成了黑色,可以在绘制画布前设置透明处背景色为白色. // 背景色转换成 ...

  6. JN_0011:改变PPT的页面尺寸,并导出图片

    1,改变尺寸 设计 --  幻灯片大小 --  自定义大小 2,导出图片 另存为 JPG 图片

  7. Activiti开发案例之activiti-app工作流导出图片

    前言 自从 Activiti 和 JBPM4 分家以后,Activiti 目前已经发展到了版本7,本着稳定性原则我们最终选择了6,之前还有一个版本5. 问题 在开发使用的过程中发现 Activiti ...

  8. html导出图片

    有一个神奇的库:html2canvas. 这个库简洁优美,使用方便. 下面先看一个小demo,它需要用到一张图片:haha.jpg. <html> <head> <scr ...

  9. vue组件:canvas实现图片涂鸦功能

    方案背景 需求 需要对图片进行标注,导出图片. 需要标注N多图片最后同时保存. 需要根据多边形区域数据(区域.颜色.名称)标注. 对应方案 用canvas实现涂鸦.圆形.矩形的绘制,最终生成图片bas ...

随机推荐

  1. (转)基于MVC4+EasyUI的Web开发框架经验总结(13)--DataGrid控件实现自动适应宽带高度

    http://www.cnblogs.com/wuhuacong/p/4085725.html 在默认情况下,EasyUI的DataGrid好像都没有具备自动宽度的适应功能,一般是指定像素宽度的,但是 ...

  2. vue遇到的大坑,h5在ios10版本下不能打开页面

    无论是谁,在做事情的过程中总是会遇到学坑,才能成为最后的大神 这个坑不说了,找了半天.希望能帮助到你们 进入build文件夹: 找到webpack.prod.conf.js文件: 在UglifyPlu ...

  3. linux 下取进程占用 cpu/内存 最高的前10个进程

    linux下获取占用CPU资源最多的10个进程,可以使用如下命令组合: ;|head linux下获取占用内存资源最多的10个进程,可以使用如下命令组合: ;|head 命令组合解析(针对CPU的,M ...

  4. day37-3 异常处理

    目录 异常处理 捕捉异常 raise assert 异常处理 捕捉异常 语法错误无法通过try检测,就像函数一样 try: 1/0 except Exception as e: # Exception ...

  5. 【转载】IDEA的这八条配置你一定要改!

    引言 坦白说,我很少写这种操作类型的文章.因为这种文章没啥新意,大家操作步骤肯定是一样的.然而,我答应了我的同事小阳,给她出一篇!毕竟人家打算从Eclipse转IDEA了,于是以示鼓励,写一篇给她!那 ...

  6. 【转载】InputStreamReader和OutputStreamWriter 的区别和用法

    一.InputStreamReader 用于将一个字节流中的字节解码成字符 , 用法如下: @Test public void Test19() throws Exception { InputStr ...

  7. PAT_A1015#Reversible Primes

    Source: PAT A1015 Reversible Primes (20 分) Description: A reversible prime in any number system is a ...

  8. Spring 属性输入和实例化操作_总结

    Spring 之 applicationContext.xml  配置 默认实例化(无参构造) //被实例化对象 package cn.ibbidream.Bean; public class Bea ...

  9. 通过Sqoop实现Mysql / Oracle 与HDFS / Hbase互导数据

    通过Sqoop实现Mysql / Oracle 与HDFS / Hbase互导数据\ 下文将重点说明通过Sqoop实现Mysql与HDFS互导数据,Mysql与Hbase,Oracle与Hbase的互 ...

  10. 洛谷——P1094 纪念品分组

    https://www.luogu.org/problem/show?pid=1094#sub 题目描述 元旦快到了,校学生会让乐乐负责新年晚会的纪念品发放工作.为使得参加晚会的同学所获得 的纪念品价 ...