Skip to content

Commit 07375ce

Browse files
committed
✨ fix: Add a temp solution for canvas
1 parent 8e38563 commit 07375ce

File tree

4 files changed

+109
-1
lines changed

4 files changed

+109
-1
lines changed

docs/e2e/basic/canvas.md

Lines changed: 63 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,63 @@
1+
---
2+
title: Canvas 解析
3+
order: 30
4+
---
5+
6+
## 基本 Canvas 解析
7+
8+
<code src="./demos/CanvasBasic.tsx" />
9+
10+
## Canvas 跨域图片解析
11+
12+
<code src="./demos/CanvasCors.tsx" />
13+
14+
### 跨域问题说明
15+
16+
某些 Canvas 由于绘制了跨域图片,因此无法使用 `toDataURL` 方法 ,会报错:
17+
18+
```
19+
Tainted canvases may not be exported
20+
```
21+
22+
### 解决方案
23+
24+
这个时候解决方案有两种:
25+
26+
#### 你可以控制 `img`
27+
28+
如果你可以控制提供 canvas 渲染的 img 节点,那一切都行相对容易,给 img 节点添加上 `crossOrigin` 属性即可
29+
30+
```html
31+
<img src="otherdomain.com" crossorigin="Anonymous" />
32+
33+
<!-- Or with Javascript -->
34+
<script>
35+
var image = new Image();
36+
image.crossOrigin = "Anonymous";
37+
...
38+
</script>
39+
```
40+
41+
不过这样的前提是你的图片服务器请求头中必须带有 `Access-Control-Allow-Origin "*"` ,否则就会提示跨域问题
42+
43+
PS: 如果无法处理图片服务器,可以利用 `cors-anywhere` 进行中转,或自行搭建代理
44+
45+
```
46+
https://cors-anywhere.herokuapp.com/${url}
47+
```
48+
49+
#### 你不能控制 `img`
50+
51+
如果你没法拿到 image 的 src 只能通过禁用 chrome 的同源策略才能解决
52+
53+
禁用方法(以 macOS 为例):
54+
55+
macOS:
56+
57+
```bash
58+
open -a "/Applications/Google Chrome.app" --args \
59+
--disable-web-security \
60+
--user-data-dir=/Users/{your account name}/chromeTempData/
61+
```
62+
63+
第一次打开如下图,就说明已经禁用同源策略了: ![](https://gw.alipayobjects.com/zos/antfincdn/hokkCFfJB9/1a018628-ae13-4fbc-82c4-eac5626e33c6.png)
File renamed without changes.

docs/e2e/basic/demos/CanvasCors.tsx

Lines changed: 33 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,33 @@
1+
import React, { useEffect } from 'react';
2+
import { useElements, TestLayout } from '@e2e-utils';
3+
4+
/**
5+
*
6+
*/
7+
export default () => {
8+
const { elements, ref } = useElements();
9+
10+
useEffect(() => {
11+
const canvas = ref.current;
12+
const ctx = canvas.getContext('2d');
13+
const image = document.getElementById('source');
14+
15+
image!.onload = function () {
16+
ctx.drawImage(image, 0, 0);
17+
};
18+
}, []);
19+
return (
20+
<TestLayout elements={elements}>
21+
<div>
22+
<canvas ref={ref} id="canvas" width="500" height="500" />
23+
</div>
24+
<div style={{ display: 'none' }}>
25+
<img
26+
id="source"
27+
src="https://wx2.sinaimg.cn/large/4176adebgy1ge52j9bku3j20dl7pse2q.jpg"
28+
alt={'image'}
29+
/>
30+
</div>
31+
</TestLayout>
32+
);
33+
};

src/parser/canvas.ts

Lines changed: 13 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
import { Bitmap } from '../model';
22
import { isCanvasNode } from '../utils/nodeType';
3+
import { errorBase64Url } from '../utils/image';
34

45
/**
56
* 将Canvas 解析为图片
@@ -8,8 +9,19 @@ import { isCanvasNode } from '../utils/nodeType';
89
export const parseCanvasToBitmap = (canvas: HTMLCanvasElement) => {
910
if (!isCanvasNode(canvas)) return;
1011
const { width, height, y, x } = canvas.getBoundingClientRect();
12+
let url;
13+
try {
14+
url = canvas.toDataURL();
15+
} catch (e) {
16+
const errMsg = e.toString();
1117

12-
const url = canvas.toDataURL();
18+
// 存在跨域问题
19+
if (errMsg.includes('Tainted canvases may not be exported.')) {
20+
// 跨域问题的显示图片
21+
url =
22+
'';
23+
} else url = errorBase64Url;
24+
}
1325

1426
const bitmap = new Bitmap({ url, width, height, y, x });
1527
bitmap.mapBasicInfo(canvas);

0 commit comments

Comments
 (0)