简介
HiDPI 屏幕非常棒,可让所有内容看起来更流畅、更清晰。但它们也给开发者带来了一系列新的挑战。在本文中,我们将探讨在高 DPI 屏幕环境中在画布中绘制图片时遇到的独特挑战。
devicePixelRatio 属性
我们从头再来一次。在我们还没有高 DPI 屏幕之前,一个像素就是一个像素(如果我们暂时忽略缩放和放大),就这样,您实际上不需要更改任何内容。如果您将某个元素的宽度设置为 100px,就只需要这样做。然后,首批高 DPI 移动手机开始出现,其中 window 对象上有一个略显神秘的 devicePixelRatio 属性,可用于媒体查询。借助此属性,我们可以了解 CSS 中的像素值(我们称之为逻辑像素值)如何转换为设备在渲染时使用的实际像素数。对于 devicePixelRatio 为 2 的 iPhone 4S,您会发现 100px 的逻辑值相当于 200px 的设备值。
这很有趣,但这对我们开发者来说意味着什么?在早期,我们就发现这些设备会放大图片。我们在元素的逻辑像素宽度下创建图片,在绘制时,这些图片会按 devicePixelRatio 放大,并且会变得模糊。
事实上,解决此问题的方法是创建按 devicePixelRatio 放大的图片,然后使用 CSS 将其缩小相同的比例,对于画布也是如此!
function setupCanvas(canvas) {
// Get the device pixel ratio, falling back to 1.
var dpr = window.devicePixelRatio || 1;
// Get the size of the canvas in CSS pixels.
var rect = canvas.getBoundingClientRect();
// Give the canvas pixel dimensions of their CSS
// size * the device pixel ratio.
canvas.width = rect.width * dpr;
canvas.height = rect.height * dpr;
var ctx = canvas.getContext('2d');
// Scale all drawing operations by the dpr, so you
// don't have to worry about the difference.
ctx.scale(dpr, dpr);
return ctx;
}
// Now this line will be the same size on the page
// but will look sharper on high-DPI devices!
var ctx = setupCanvas(document.querySelector('.my-canvas'));
ctx.lineWidth = 5;
ctx.beginPath();
ctx.moveTo(100, 100);
ctx.lineTo(200, 200);
ctx.stroke();