JavaScript 禁止用户保存图片
本文最后更新于:2020年4月27日 下午
场景
在业务需求中不希望用户保存图片,因为是一些供内部使用的图片。
思路
- 添加事件禁止选择、拖拽、右键(简单的禁止用户保存图片,但无法阻止用户打开控制台查看,或是直接抓包)
- 将之转换为 canvas(让浏览器认为不是图片以此禁止用户对之进行图片的操作,但无法阻止抓包)
- 禁止用户使用控制台查看源码(阻止浏览器打开控制台,但无法阻止抓包)
- 传输图片使用自定义格式(可以阻止抓包,但需要后台配合)
注:以下内容使用 react+ts 实现
添加事件禁止选择、拖拽、右键
简而言之,这是一种简单有效的方式,能够在用户不打开控制台的情况下阻止用户保存图片。
1 |
|
将之转换为 canvas
另一种思路是将图片转换为 canvas 避免用户使用 img
相关的操作。
将图片转成 canvas
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22export async function imageToCanvas(url: string, canvas: HTMLCanvasElement) {
return new Promise((resolve, reject) => {
//新建Image对象,引入当前目录下的图片
const img = new Image()
img.src = url
const c = canvas.getContext('2d')!
//图片初始化完成后调用
img.onload = function () {
//将canvas的宽高设置为图像的宽高
canvas.width = img.width
canvas.height = img.height
//canvas画图片
c.drawImage(img, 0, 0, img.width, img.height)
resolve()
}
img.addEventListener('error', (e) => {
reject(e)
})
})
}禁用 canvas 事件
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21const throwFn = () => {
throw new Error(
"Uncaught DOMException: Failed to execute 'toDataURL' on 'HTMLCanvasElement': Tainted canvases may not be exported.",
)
}
const $canvasRef = useRef<HTMLCanvasElement>(null)
useEffect(() => {
;(async () => {
await imageToCanvas(props.url, $canvasRef.current!)
$canvasRef.current!.toBlob = throwFn
$canvasRef.current!.toDataURL = throwFn
})()
}, [])
return (
<canvas
ref={$canvasRef}
onTouchStart={preventDefaultListener}
onContextMenu={preventDefaultListener}
/>
)
禁止用户使用控制台查看源码
如果能禁止用户操作控制台,那么自然能够避免用户查看源码了,下面是一个简单的实现。
1 |
|
1 |
|
传输图片使用自定义格式
该功能需要服务端配合,故而此处赞不实现,可以参考 微信读书,就是将文本转为 canvas,数据传输也进行了加密,可以在很大程度上防止普通用户想要复制/下载的行为了。
总结
如同所有的前端限制用户的技术一样,这是一个没有终点的斗争。。。
参考广告屏蔽和屏蔽复制粘贴的发展。。。
JavaScript 禁止用户保存图片
https://blog.rxliuli.com/p/452eb79b3bd74cbbbc95636b450b52f3/