介绍
<canvas>
最早由 Apple 引入 WebKit,用于 Mac OS X 的 Dashboard,随后被各个浏览器实现。现在,所有支流的浏览器都反对它。Canvas API 提供了一个通过 JavaScript 和 HTML 的 <canvas>
元素来绘制图形的形式。它能够用于动画、游戏画面、数据可视化、图片编辑以及实时视频解决等方面。Canvas 适宜绘制大数据量图形元素的图表(如热力求、天文坐标系或平行坐标系上的大规模线图或散点图等),也适宜实现某些视觉特效。它还能可能以 png、jpg 或 webp 格局保留图像。Canvas 提供了弱小的 Web 绘图能力,所以咱们要学会应用它。
成果:
创立一个画布
默认状况下 <canvas>
元素没有边框和内容。默认大小为 300px × 150px(宽 × 高),咱们能够应用 width 和 height 属性指定。
<canvas id="canvas"></canvas>
获取画布和半径
为了在 <canvas>
上绘制图形,咱们须要应用一个 JavaScript 上下文对象,它能动态创建图像。这里建设了一个 CanvasRenderingContext2D 二维渲染上下文。
const ctx = canvas.getContext('2d');
let radius = canvas.height / 2;
绘制圆周和时钟核心
- beginPath() 用来起始一条门路,或重置以后门路。
- arc() 用于创立圆形或弧形。
- fill() 用来填充图形。
-
stroke() 用来绘制门路。
function drawFace(context, radius) {context.beginPath(); context.arc(0, 0, radius, 0, 2 * Math.PI); context.fillStyle = 'white'; context.fill(); // 重置门路 画时钟核心圆点 context.beginPath(); context.arc(0, 0, radius * 0.06, 0, 2 * Math.PI); context.fillStyle = 'green'; context.fill();}
绘制表盘数字
- rotate() 用来旋转以后绘图。360 度角等于 Math.PI * 2,Math.PI / 6 就是 30 度角。
- translate() 用来将原点挪动到新地位。
-
fillText() 用来绘制文本。
function drawNumbers(context, radius) { context.fillStyle = 'green'; context.font = radius * 0.15 + 'px arial'; context.textBaseline = 'middle'; context.textAlign = 'center'; for (let num = 1; num <= 12; num++) { const angle = num * Math.PI / 6; context.rotate(angle); context.translate(0, -radius * 0.82); context.rotate(-angle); context.fillText(num.toString(), 0, 0); context.rotate(angle); context.translate(0, radius * 0.82); context.rotate(-angle); } }
绘制表盘刻度
- moveTo() 挪动门路到指定点。
-
lineTo() 增加一个新点,并创立从最初指定点到该点的线条。
function drawLine(context, radius) { context.lineCap = 'butt'; for (let i = 1; i <= 60; i++) { context.strokeStyle = i % 5 === 0 ? 'yellowgreen' : 'lightgray'; context.beginPath(); context.lineWidth = i % 5 === 0 ? radius * 0.03 : radius * 0.02; context.rotate(i * Math.PI / 30); context.moveTo(0, -radius); context.lineTo(0, i % 5 === 0 ? -radius * 0.93 : -radius * 0.96); context.stroke(); context.rotate(-i * Math.PI / 30); } }
绘制表盘时、分、秒针
function drawTime(context, radius) {const date = new Date();
const h = date.getHours();
const m = date.getMinutes();
const s = date.getSeconds();
const hourAngle = (h * Math.PI / 6) + (m * Math.PI / (6 * 60)) + (s * Math.PI / (6 * 60 * 60));
const minuteAngle = (m * Math.PI / 30) + (s * Math.PI / (30 * 60));
const secondAngle = (s * Math.PI / 30);
drawHand(context, hourAngle, radius * 0.5, radius * 0.05, 'green');
drawHand(context, minuteAngle, radius * 0.7, radius * 0.04, 'dodgerblue');
drawHand(context, secondAngle, radius * 0.9, radius * 0.02, 'orange');
}
function drawHand(context, angle, length, width, color) {context.beginPath();
context.lineWidth = width;
context.lineCap = 'round'; // 圆形末端
context.moveTo(0, 0);
context.rotate(angle);
context.lineTo(0, -length);
context.strokeStyle = color;
context.stroke();
context.rotate(-angle);
}
增加款式
因为 canvas 画圆会有锯齿,这里应用 CSS 款式会更圆润。
canvas {
background: green;
border-radius: 50%;
}
启动时钟
translate(radius, radius) 将地位挪动到画布核心。
ctx.translate(radius, radius);
radius = radius * 0.9; // 时钟半径
function draw() {drawFace(ctx, radius);
drawNumbers(ctx, radius);
drawLine(ctx, radius);
drawTime(ctx, radius);
requestAnimationFrame(draw);
}
draw();
预览地址