网站首页 > 技术文章 正文
DOM to Image
dom-to-image是一个js库,可以将任意dom节点转换为矢量(SVG)或光栅(PNG或JPEG)图像。
安装
npm install dom-to-image -S
加载
/* in ES 6 */
import domtoimage from 'dom-to-image';
/* in ES 5 */
var domtoimage = require('dom-to-image');
用法
所有高阶函数都接受DOM节点和渲染选项options ,并返回promises。
- 获取PNG图像base64编码的data URL:
<div id="my-node"></div>
var node = document.getElementById('my-node');
// options 可不传
var options = {}
domtoimage.toPng(node, options)
.then(function (dataUrl) {
var img = new Image();
img.src = dataUrl;
document.body.appendChild(img);
})
.catch(function (error) {
console.error('oops, something went wrong!', error);
});
- 获取图像blob:
domtoimage.toBlob(document.getElementById('my-node'))
.then(function (blob) {
console.log('blob', blob)
});
- 获取JPEG图像base64编码的data URL并下载:
domtoimage.toJpeg(document.getElementById('my-node'), { quality: 0.95 })
.then(function (dataUrl) {
var link = document.createElement('a');
link.download = 'my-image-name.jpeg';
link.href = dataUrl;
link.click();
});
- 获取SVGdata URL,但筛选出所有元素:
function filter (node) {
return (node.tagName !== 'i');
}
domtoimage.toSvg(document.getElementById('my-node'), {filter: filter})
.then(function (dataUrl) {
/* do something */
});
- 以uint8数组的形式获取原始像素数据,每4个数组元素表示一个像素的RGBA数据:
var node = document.getElementById('my-node');
domtoimage.toPixelData(node)
.then(function (pixels) {
for (var y = 0; y < node.scrollHeight; ++y) {
for (var x = 0; x < node.scrollWidth; ++x) {
pixelAtXYOffset = (4 * y * node.scrollHeight) + (4 * x);
/* pixelAtXY is a Uint8Array[4] containing RGBA values of the pixel at (x, y) in the range 0..255 */
pixelAtXY = pixels.slice(pixelAtXYOffset, pixelAtXYOffset + 4);
}
}
});
options参数
Name | 类型 | Default | Description |
filter | Function | —— | 以DOM节点为参数的函数。如果传递的节点应包含在输出中,则应返回true(排除节点意味着也排除其子节点) |
bgcolor | String | —— | 背景色的字符串值,任何有效的CSS颜色值。 |
height | Number | —— | 渲染前应用于节点的高度(以像素为单位)。 |
width | Number | —— | 渲染前应用于节点的宽度(以像素为单位)。 |
style | Object | —— | object对象,其属性在渲染之前要复制到节点的样式中。 |
quality | Number | 1.0 | 介于0和1之间的数字,表示JPEG图像的图像质量(例如0.92=>92%)。默认值为1.0(100%) |
cacheBust | Boolean | false | 设置为true可将当前时间作为查询字符串附加到URL请求以启用清除缓存。 |
imagePlaceholder | Boolean | undefined | 获取图片失败时使用图片的数据URL作为占位符。默认为未定义,并将在失败的图像上引发错误。 |
原理
dom-to-image使用SVG的一个特性,它允许在标记中包含任意HTML内容。
- 递归地克隆原始DOM节点
- 计算节点和每个子节点的样式,并将其复制到相应的克隆 创建伪元素,因为它们不是以任何方式克隆的
- 嵌入web字体 查找所有@font face声明的web字体 解析文件URL,下载相应文件 base64编码的内联作为data:URLs 将所有已处理的CSS放入中,然后将其附加到克隆
- 嵌入图片 在嵌入图片URL 使用backgroundCSS属性的图片,方法类似于字体
- 将克隆的节点序列化为XML
- 将XML包装到标记中,然后包装到SVG中,然后使其成为data URL
- 或者,要以Uint8Array的形式获取PNG内容或原始像素数据,可以创建一个以SVG为源的图像元素,并将其呈现在已经创建的canvas上,从canvas读取内容
部分源码分析
dom-to-image.js
// Default impl options
var defaultOptions = {
// Default is to fail on error, no placeholder
imagePlaceholder: undefined,
// Default cache bust is false, it will use the cache
cacheBust: false
};
var domtoimage = {
toSvg: toSvg,
toPng: toPng,
toJpeg: toJpeg,
toBlob: toBlob,
toPixelData: toPixelData,
impl: {
fontFaces: fontFaces,
images: images,
util: util,
inliner: inliner,
options: {}
}
};
if (typeof module !== 'undefined')
module.exports = domtoimage;
else
global.domtoimage = domtoimage;
- defaultOptions设置默认options选项
- domtoimage的核心api:
- toSvg
- toPng
- toJpeg
- toBlob
- toPixelData
- 例:toJpeg:将draw函数返回的canvas实例,使用canvas的toDataURL方法生成jpeg图片。toSvg函数将递归地克隆原始DOM节点, 将克隆的节点序列化为XML,将XML包装到标记中,然后包装到SVG中,然后使其转成dataURL。
function toJpeg(node, options) {
options = options || {};
return draw(node, options)
.then(function (canvas) {
return canvas.toDataURL('image/jpeg', options.quality || 1.0);
});
}
复制代码
function draw(domNode, options) {
return toSvg(domNode, options)
.then(util.makeImage)
.then(util.delay(100))
.then(function (image) {
var canvas = newCanvas(domNode);
canvas.getContext('2d').drawImage(image, 0, 0);
return canvas;
});
function newCanvas(domNode) {
var canvas = document.createElement('canvas');
canvas.width = options.width || util.width(domNode);
canvas.height = options.height || util.height(domNode);
if (options.bgcolor) {
var ctx = canvas.getContext('2d');
ctx.fillStyle = options.bgcolor;
ctx.fillRect(0, 0, canvas.width, canvas.height);
}
return canvas;
}
}
function toSvg(node, options) {
options = options || {};
copyOptions(options);
return Promise.resolve(node)
.then(function (node) {
return cloneNode(node, options.filter, true);
})
.then(embedFonts)
.then(inlineImages)
.then(applyOptions)
.then(function (clone) {
return makeSvgDataUri(clone,
options.width || util.width(node),
options.height || util.height(node)
);
});
function applyOptions(clone) {
if (options.bgcolor) clone.style.backgroundColor = options.bgcolor;
if (options.width) clone.style.width = options.width + 'px';
if (options.height) clone.style.height = options.height + 'px';
if (options.style)
Object.keys(options.style).forEach(function (property) {
clone.style[property] = options.style[property];
});
return clone;
}
}
作者:知其
https://juejin.cn/post/6988045156473634852
- 上一篇: 花10分钟学一项技能 ImageJ 图像分析
- 下一篇: 十款代码表白特效,一个比一个浪漫
猜你喜欢
- 2024-11-23 轻量图片查看器-imageGlass
- 2024-11-23 「正点原子FPGA连载」第十八章Linux内核移植
- 2024-11-23 OpenShift 4 概念 - ImageStream,Image和Image Registry
- 2024-11-23 图片处理工具,FocusOn Image Viewer软件体验
- 2024-11-23 深入分析ImageMagick的Shell注入漏洞
- 2024-11-23 一键部署!markdown-to-image:将 Markdown 转为海报的编辑器!
- 2024-11-23 手机误删照片怎么恢复?这款免费照片恢复App你值得拥有
- 2024-11-23 Image Enhance Pro for Mac(图像处理工具)中文版
- 2024-11-23 WiseImage Pro——为AutoCAD开发的插件工具推荐
- 2024-11-23 使用ImageJ进行运动粒子分析
- 标签列表
-
- content-disposition (47)
- nth-child (56)
- math.pow (44)
- 原型和原型链 (63)
- canvas mdn (36)
- css @media (49)
- promise mdn (39)
- readasdataurl (52)
- if-modified-since (49)
- css ::after (50)
- border-image-slice (40)
- flex mdn (37)
- .join (41)
- function.apply (60)
- input type number (64)
- weakmap (62)
- js arguments (45)
- js delete方法 (61)
- blob type (44)
- math.max.apply (51)
- js (44)
- firefox 3 (47)
- cssbox-sizing (52)
- js删除 (49)
- js for continue (56)
- 最新留言
-