录屏实现方案:rrweb.js
官方文档:github.com/rrweb-io/rr…
截屏实现方案:html2canvas.js
rrweb录屏实现的原理是记录dom的变化,播放的时候重新执行dom操作,所以不支持生成mp4,只可以在web端重放
html2canvas截图实现的原理是生成一个canvas画布,画布的内容是插件帮我们实现好的,我们需要做的是将canvas画布保存成base64图片,保存到本地或者上传
服务端例子:express
需要引入的文件:
rrweb:
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/rrweb@latest/dist/rrweb.min.css"/>
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/rrweb-player@latest/dist/style.css"/>
<script src="https://cdn.jsdelivr.net/npm/rrweb@latest/dist/rrweb.min.js"></script>
<script src="https://cdn.jsdelivr.net/npm/rrweb-player@latest/dist/index.js"></script>
html2canvas(从官网下载到本地的文件)
<script src="/javascripts/html2canvas.js"></script>
下面用一个例子演示一下跨页面录屏和截屏的效果
前端完整代码如下:
页面1:
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width,initial-scale=1.0">
<link rel="icon" href="<%= BASE_URL %>favicon.ico">
<link
rel="stylesheet"
href="https://cdn.jsdelivr.net/npm/rrweb@latest/dist/rrweb.min.css"
/>
<link
rel="stylesheet"
href="https://cdn.jsdelivr.net/npm/rrweb-player@latest/dist/style.css"
/>
</head>
<body>
<!-- 不支持 IE11 以下的浏览器 -->
<script src="https://cdn.jsdelivr.net/npm/rrweb@latest/dist/rrweb.min.js"></script>
<script src="https://cdn.jsdelivr.net/npm/rrweb-player@latest/dist/index.js"></script>
<script src="/javascripts/html2canvas.js"></script>
<button class="btn">开始录制</button>
<button class="btn2">结束并播放</button>
<button class="btn3" id="capture">截图</button>
<button class="btn4">下一步</button>
<button class="btn5">清空服务器数据</button>
<br>
<br>
<div>this is page1</div>
<div>
<input type="text" placeholder="请输入信息">
</div>
<script>
window.onload = function(){
let el = document.getElementsByClassName("btn")[0];
let el2 = document.getElementsByClassName("btn2")[0];
let el3 = document.getElementById("capture")
let el4 = document.getElementsByClassName("btn4")[0];
let el5 = document.getElementsByClassName("btn5")[0];
let events = []
let r
el.onclick = function(){
alert("开始录制")
r = rrweb.record({
emit(event) {
// 将 event 存入 events 数组中
events.push(event);
console.log(events)
},
});
}
el2.onclick = function(){
alert("播放")
r()
new rrwebPlayer({
target: document.body, // 可以自定义 DOM 元素
data: {
events,
},
});
}
el3.onclick = function(){
html2canvas(document.querySelector("body")).then(canvas => {
// document.body.appendChild(canvas)
let pic = canvas.toDataURL("image/png");
downloadFileByBase64(pic,'金融测试下载')
// window.location.href = pic;
});
}
el4.onclick = function(){
// console.log(JSON.stringify(events))
// console.log(events.length)
uploadEvents(1,events)
return false
}
el5.onclick = function(){
// console.log(JSON.stringify(events))
// console.log(events.length)
ajax({
url:'/clearEvents',
type:'POST',
dataType:'json',
data:{},
// data:{name:1},
success:function(response,xml){
//请求成功后执行的代码
alert(response.msg)
// console.log(events)
window.open("/page2.html")
},
error:function(status){
alert(response.msg)
}
});
return false
}
}
// 分批上传
function uploadEvents(m,events){
console.log(JSON.stringify(events).length)
ajax({
url:'/saveEvents',
type:'POST',
dataType:'json',
data:{events:JSON.stringify(events),page:1},
// data:{name:1},
success:function(response,xml){
//请求成功后执行的代码
response = JSON.parse(response);
// console.log(events)
window.open("/page2.html")
},
error:function(status){
alert(response.msg)
}
});
}
// base64生成图片方法
function dataURLtoBlob(dataurl) {
var arr = dataurl.split(','), mime = arr[0].match(/:(.*?);/)[1],
bstr = atob(arr[1]), n = bstr.length, u8arr = new Uint8Array(n);
while (n--) {
u8arr[n] = bstr.charCodeAt(n);
}
return new Blob([u8arr], { type: mime });
}
function downloadFile(url,name='What\'s the fuvk'){
var a = document.createElement("a")
a.setAttribute("href",url)
a.setAttribute("download",name)
a.setAttribute("target","_blank")
let clickEvent = document.createEvent("MouseEvents");
clickEvent.initEvent("click", true, true);
a.dispatchEvent(clickEvent);
}
function downloadFileByBase64(base64,name){
var myBlob = dataURLtoBlob(base64)
var myUrl = URL.createObjectURL(myBlob)
downloadFile(myUrl,name)
}
// 原生ajax方法
function ajax(options){
var options=options||{};
options.type=(options.type||'GET').toUpperCase();
options.dataType=options.dataType||'json';
var params=formatParams(options.data);
//创建-第一步
var xhr;
//非IE6
if(window.XMLHttpRequest){
xhr=new XMLHttpRequest();
}else{
//ie6及其以下版本浏览器
xhr=ActiveXObject('Microsoft.XMLHTTP');
}
//接收-第三步
xhr.onreadystatechange=function(){
if(xhr.readyState==4){
var status=xhr.status;
if(status>=200&&status<300){
options.success&&options.success(xhr.responseText,xhr.responseXML);
}else{
options.error&&options.error(status);
}
}
}
//连接和发送-第二步
if(options.type=='GET'){
xhr.open('GET',options.url+'?'+params,true);
xhr.send(null);
}else if(options.type=='POST'){
xhr.open('POST',options.url,true);
//设置表单提交时的内容类型
xhr.setRequestHeader("Content-Type", "application/x-www-form-urlencoded");
xhr.send(params);
}
}
//格式化参数
function formatParams(data){
var arr=[];
for(var name in data){
arr.push(encodeURIComponent(name)+'='+encodeURIComponent(data[name]));
}
// arr.push(('v='+Math.random()).replace('.',''));
return arr.join('&');
}
// save 函数用于将 events 发送至后端存入,并重置 events 数组
// function save() {
// const body = JSON.stringify({ events });
// events = [];
// fetch('http://YOUR_BACKEND_API', {
// method: 'POST',
// headers: {
// 'Content-Type': 'application/json',
// },
// body,
// });
// 每 10 秒调用一次 save 方法,避免请求过多
// setInterval(save, 10 * 1000);
</script>
</body>
</html>
页面2:
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width,initial-scale=1.0">
<link rel="icon" href="<%= BASE_URL %>favicon.ico">
<link
rel="stylesheet"
href="https://cdn.jsdelivr.net/npm/rrweb@latest/dist/rrweb.min.css"
/>
<link
rel="stylesheet"
href="https://cdn.jsdelivr.net/npm/rrweb-player@latest/dist/style.css"
/>
</head>
<body>
<!-- 不支持 IE11 以下的浏览器 -->
<script src="https://cdn.jsdelivr.net/npm/rrweb@latest/dist/rrweb.min.js"></script>
<script src="https://cdn.jsdelivr.net/npm/rrweb-player@latest/dist/index.js"></script>
<script src="/javascripts/html2canvas.js"></script>
<button class="btn">开始录制2</button>
<button class="btn2">结束并播放1和2</button>
<button class="btn3" id="capture">截图2</button>
<br>
<br>
<div>this is page2</div>
<div>
<input type="text" placeholder="请输入信息">
</div>
<script>
window.onload = function(){
let el = document.getElementsByClassName("btn")[0];
let el2 = document.getElementsByClassName("btn2")[0];
let el3 = document.getElementById("capture")
console.log(el)
let events = []
let r
el.onclick = function(){
alert("开始录制")
r = rrweb.record({
emit(event) {
// 将 event 存入 events 数组中
events.push(event);
console.log(events)
},
});
}
el2.onclick = function(){
alert("播放")
if(r){
r()
}
ajax({
url:'/saveEvents',
type:'POST',
dataType:'json',
data:{events:JSON.stringify(events),finished:false},
// data:{name:1},
success:function(response,xml){
//请求成功后执行的代码
// console.log(events)
ajax({
url:'/getEvents',
type:'POST',
dataType:'json',
data:{},
success:function(response,xml){
//请求成功后执行的代码
response = JSON.parse(response);
events = response;
console.log(response)
new rrwebPlayer({
target: document.body, // 可以自定义 DOM 元素
data: {
events,
},
});
},
error:function(status){
alert(response.msg)
}
});
},
error:function(status){
alert(response.msg)
}
});
}
el3.onclick = function(){
html2canvas(document.querySelector("body")).then(canvas => {
// document.body.appendChild(canvas)
let pic = canvas.toDataURL("image/png");
downloadFileByBase64(pic,'金融测试下载')
// window.location.href = pic;
});
}
}
function dataURLtoBlob(dataurl) {
var arr = dataurl.split(','), mime = arr[0].match(/:(.*?);/)[1],
bstr = atob(arr[1]), n = bstr.length, u8arr = new Uint8Array(n);
while (n--) {
u8arr[n] = bstr.charCodeAt(n);
}
return new Blob([u8arr], { type: mime });
}
function downloadFile(url,name='What\'s the fuvk'){
var a = document.createElement("a")
a.setAttribute("href",url)
a.setAttribute("download",name)
a.setAttribute("target","_blank")
let clickEvent = document.createEvent("MouseEvents");
clickEvent.initEvent("click", true, true);
a.dispatchEvent(clickEvent);
}
function downloadFileByBase64(base64,name){
var myBlob = dataURLtoBlob(base64)
var myUrl = URL.createObjectURL(myBlob)
downloadFile(myUrl,name)
}
// 原生ajax方法
function ajax(options){
var options=options||{};
options.type=(options.type||'GET').toUpperCase();
options.dataType=options.dataType||'json';
var params=formatParams(options.data);
//创建-第一步
var xhr;
//非IE6
if(window.XMLHttpRequest){
xhr=new XMLHttpRequest();
}else{
//ie6及其以下版本浏览器
xhr=ActiveXObject('Microsoft.XMLHTTP');
}
//接收-第三步
xhr.onreadystatechange=function(){
if(xhr.readyState==4){
var status=xhr.status;
if(status>=200&&status<300){
options.success&&options.success(xhr.responseText,xhr.responseXML);
}else{
options.error&&options.error(status);
}
}
}
//连接和发送-第二步
if(options.type=='GET'){
xhr.open('GET',options.url+'?'+params,true);
xhr.send(null);
}else if(options.type=='POST'){
xhr.open('POST',options.url,true);
//设置表单提交时的内容类型
xhr.setRequestHeader("Content-Type", "application/x-www-form-urlencoded");
xhr.send(params);
}
}
//格式化参数
function formatParams(data){
var arr=[];
for(var name in data){
arr.push(encodeURIComponent(name)+'='+encodeURIComponent(data[name]));
}
// arr.push(('v='+Math.random()).replace('.',''));
return arr.join('&');
}
// save 函数用于将 events 发送至后端存入,并重置 events 数组
// function save() {
// const body = JSON.stringify({ events });
// events = [];
// fetch('http://YOUR_BACKEND_API', {
// method: 'POST',
// headers: {
// 'Content-Type': 'application/json',
// },
// body,
// });
// 每 10 秒调用一次 save 方法,避免请求过多
// setInterval(save, 10 * 1000);
</script>
</body>
</html>
服务端代码:
/*
* @Author: your name
* @Date: 2020-07-18 19:09:47
* @LastEditTime: 2020-07-19 12:24:16
* @LastEditors: Please set LastEditors
* @Description: In User Settings Edit
* @FilePath: /record-2/app.js
*/
var createError = require('http-errors');
var express = require('express');
var path = require('path');
var cookieParser = require('cookie-parser');
var logger = require('morgan');
var fs = require('fs');
var indexRouter = require('./routes/index');
var usersRouter = require('./routes/users');
var bodyParser = require('body-parser');
var app = express();
app.use(bodyParser.urlencoded({
extended:true
}));
let events = [];
// view engine setup
app.use(express.static(path.join(__dirname, 'public')));
// 服务端页面
app.get('/', function(req, res, next) {
// res.writeHead(200,{'Content-Type':'text/html'})
fs.readFile('./html/index.html','utf-8',function(err,data){
if(err){
throw err ;
}
res.send(data);
});
});
app.get('/page2.html', function(req, res, next) {
// res.writeHead(200,{'Content-Type':'text/html'})
fs.readFile('./html/page2.html','utf-8',function(err,data){
if(err){
throw err ;
}
res.send(data);
});
});
app.post('/clearEvents',function(req, res, next) {
// res.writeHead(200,{'Content-Type':'text/html'})
console.log(req.body)
events = [];
res.send({msg:"操作成功"})
// if(req.query.events.length>0){
// res.send({state:1,msg:"上传成功"})
// }else{
// res.send({state:0,msg:"参数错误,未获取到录屏信息"})
// }
});
// 存
app.post('/saveEvents',function(req, res, next) {
// res.writeHead(200,{'Content-Type':'text/html'})
console.log(req.body)
res.send({state:1,msg:"上传成功"})
// res.send(req.body.events)
if(req.body.page==1){
events = []
}
events = events.concat(JSON.parse(req.body.events))
fs.writeFile("./data.txt",JSON.stringify(req.body.events),'utf-8',function(){
})
// if(req.query.events.length>0){
// res.send({state:1,msg:"上传成功"})
// }else{
// res.send({state:0,msg:"参数错误,未获取到录屏信息"})
// }
});
// 取
app.post('/getEvents',function(req, res, next) {
res.send(events);
// fs.readFile('./data.txt','utf-8',function(err,data){
// if(err){
// throw err ;
// }
// console.log(JSON.parse(data))
// res.send(data);
// });
// if(req.query.events.length>0){
// res.send({state:1,msg:"上传成功"})
// }else{
// res.send({state:0,msg:"参数错误,未获取到录屏信息"})
// }
});
module.exports = app;
上面的例子如果只想在前端看效果的话,只运行第一个页面的前端代码就能查看。