canvas 图片、文字模糊问题

注:[n]标识为遗留问题,在文章末尾遗留问题部分有详细解释说明。

之前做了一个在线给图片添加文本框的工具,大体思路是先把图片加载到一个 DOM 结构中,然后通过 html2canvas 导出到一个canvas,最后通过 canvas 自带的 toDataURL 方法导出成图片。

这个思路并不复杂,但是中间遇到几个小问题:

  1. 跨域图片的导出问题:你可以把图片绘制到 canvas 中,但是不能做任何有关导出数据的操作(比如 toDataURL ),因为 canvas 认为它自己是被污染(tainted)的。(当然本地上传的图片是不存在这个问题的)

    This protects users from having private data exposed by using images to pull information from remote web sites without permission.

    ——出自 canvas-todataurl-securityerror

    大概意思是说,这样可以保护用户隐私数据不被暴露。

  2. 在 retina 屏幕上canvas 的内容显示变模糊。

  3. 图片模糊就算了,为什么fillText输入的文字也会模糊?而且导出来会清晰一点(但是还是模糊)

解决过程:

  1. 第一个问题其实就是解决我们熟悉的跨域问题。这个工具的主要使用场景是在海外的 i8n 项目,图片一般放在海外的图片服务器上。我给图片添加 crossorigin:anonymous 不生效,所以决定换条路。

    既然传统的跨域用法是失败的,但是我们知道 <img>src 属性可以用 base64编码后的数据表示图片的内容,这样不会存在跨域问题。所以我想用 FileReader 转换图片格式。但是后来才发现 FileReader 同样不允许处理跨域资源…计划泡汤。

    然后发现这么个工具CORS Anywhere,是给你的请求头部加 CORS header 的。这样一来应该可以解决跨域问题。(未具体尝试)

  2. 这个问题才是今天想讲的主题。

    先把网上的解决方法贴出来:

    devicePixelRatio = window.devicePixelRatio || 1,
    backingStoreRatio = context.webkitBackingStorePixelRatio || 1,
    ratio = devicePixelRatio / backingStoreRatio;
    
    var w = $("#code").width();
    var h = $("#code").height();
    
    //要将 canvas 的宽高设置成容器宽高的 2 倍
    var canvas = document.createElement("canvas");
    canvas.width = w * ratio;
    canvas.height = h * ratio;
    canvas.style.width = w + "px";
    canvas.style.height = h + "px";
    var context = canvas.getContext("2d");
    //然后将画布缩放,将图像放大两倍画到画布上
    context.scale(ratio,ratio);
    

    上面的代码我们分两部分看,先忽略上面定义 ratio 值的部分,往下看。
    说明一下,canvas 的属性 width/height和样式表里指定的宽高不同,前者确定了这个画布的内容大小,而后者只是显示上的大小。所以上面代码就不难理解了,其实是把画布的内容高宽放大二倍,而样式上不变,视觉上就会变得精细很多,和二倍图的原理基本上是类似的。

    道理我都懂,但是代码开头那一大堆在算什么?

    按照上面的逻辑来说,我们只需要通过 devicePixelRatio 判断设备是不是 retina 屏幕(不严格地说)就可以了。为什么要算他和backingStoreRatio的比值,这又是个什么东西?

    我们在往 canvas 里画任何东西的时候,实际上浏览器都在把这些写到了一个后备存储空间里。浏览器在重新绘制到屏幕时候,数据就是来自这里。webkitBackingStorePixelRatio这个值告诉我们的是后备空间相对 canvas 本身容量的大小。

    现在我们知道了这个值的作用,它是如何控制展示的?

    1x1x1

    上图展示的是 dpr:bk === 1 的情况,就像没有出现 retina 屏幕这件事一样,导出和汇入两不相干。
    关键是两者值都为2的时候也是如此。所以即使是在 retina 屏幕上,也有可能不做多余的代码处理图片也可以很清楚。这也是为什么我们说计算 ratio 的值时我们要算二者的比值而不是单纯用 dpr。
    而且这两个更多时候确实没有任何关系,并不是 dpr 为2 bk 的值就也一定高。

    1x1x1

    dpr:bk === 2问题出现了。我们原样把图片放进来,canvas 因为 bk 值为1所以没有对图片做其他处理,再展示到页面上的时候就会模糊。这其实跟一般的图片在 retina 屏幕上模糊的原因相同。

    比如我们有一个长宽都为30px的图,放到 retina 屏幕上占有 30 csspx 的宽度,但是实际上填充他宽度的有60个物理像素。我们的图片只提供了30个已知的像素值,其余的30个只能靠浏览器根据周围的像素点去计算。所以会模糊。

  3. 下面来讨论为什么文字模糊的问题。
    刚开始看到文字模糊的时候觉得没什么难理解的,明显是和图片一个套路。但是细想觉得不对,图片是因为在 dpr 为2的情况下,图片内容宽和图片样式宽却是相等的所以模糊。但是文字在我打到页面上到画到 canvas 的过程中,实际像素数是足够的,为什么会模糊?

    在查了部分资料之后发现,在页面上字体的展示和在 Canvas 里 用fillText 去绘制文字是不一样的,后者其实是在 canvas 里「画」字,而这个画的结果的展示单元和上面图片是一样的,到现在为止我们可以把这个过程和图片展示想成相同的了。

    至于为什么下载后会清楚一些但是却不「那么清楚」,我们当做两个问题来解答。
    为什么会清楚一些?因为模糊实际上是浏览器渲染时候的行为,下载之后查看图片是没有这个像素估算的过程的。
    为什么却不那么清楚?详细的我不想讲了,具体的可以看这个回答

遗留问题:
[1]: 发送的 file 协议的请求到服务器端判断跨域的时候和 http 是一样的标准吗?我个人觉得其实应该是的,因为同源策略本身的目的就是出于安全,这一点和你客户端的协议其实是没关系的。

参考文章:

High DPI Canvas

设备像素,设备独立像素,CSS像素

Canvas text rendering (blurry)

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,463评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,868评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,213评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,666评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,759评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,725评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,716评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,484评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,928评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,233评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,393评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,073评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,718评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,308评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,538评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,338评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,260评论 2 352

推荐阅读更多精彩内容