纯 canvas 实现漂亮的仪表盘

在这里插入图片描述
经验:

  1. save-restore-restore-restore 是不行的。用 save 保存的状态只能用一次,最后两个 restore 是回不到第一次 save 的状态的。
  2. oninput 是变了就触发,onchange 是变了且 失去焦点 才触发

<!DOCTYPE html>
<html>

<head>
	<meta charset="UTF-8">
	<style>
		canvas {
			background: linear-gradient(20deg, rgba(255, 152, 0, 0.5) 0, rgba(255, 87, 34, 0.9) 45%, rgba(255, 87, 34, 0.8) 50%, rgba(255, 152, 0, 0.6) 100%);
		}
	</style>
</head>

<body>

	<canvas id="canvas"></canvas>
	<!-- 要用oninput,用onchange不好 -->
	<input type="range" value="70" oninput="handleInput()" id="inputRange">
	<img src="" style="display:none;" id="beautiful-icon">
	<img src="" style="display:none;" id="beautiful-icon-light">


	<script>
		const width = 500;
		const height = 300;
		let canvas;
		let beautiful_icon_base64;
		let beautiful_icon_light_base64;

		window.onload = function() {
			// 给图片设置src(base64)
			document.getElementById("beautiful-icon").setAttribute("src", beautiful_icon_base64)
			document.getElementById("beautiful-icon-light").setAttribute("src", beautiful_icon_light_base64)
			// 给画布设置宽高
			canvas = document.getElementById("canvas");
			canvas.setAttribute("width", width);
			canvas.setAttribute("height", height);
			// 画一个
			handleInput(60);
		}

		function handleInput() {
			const val = parseInt(document.getElementById("inputRange").value);
			draw(val);
		}

		function draw(value) {
			let startAngle = 17 / 16 * Math.PI, // 和数学上的保持一致
				endAngle = -1 / 16 * Math.PI,
				splitNumber = 25,
				diamondNumber = 10,
				diamondSize = 20,
				center = [width / 2, height / 2 + 32];
			var ctx = canvas.getContext('2d');
			ctx.save();
			ctx.clearRect(0, 0, width, height);
			ctx.translate(center[0], center[1]);
			// 
			ctx.beginPath();
			ctx.strokeStyle = 'white';
			ctx.lineWidth = 5;
			ctx.shadowBlur = 10;
			ctx.shadowColor = "rgba(255, 255, 255, 0.8)";
			ctx.arc(0, 0, 35, 2 * Math.PI - startAngle, 2 * Math.PI - endAngle, false);
			ctx.stroke();
			// 画刻度
			ctx.lineWidth = 2;
			let r1 = 90;
			let r2 = 105;
			ctx.shadowBlur = 0;
			for (let i = 0; i <= splitNumber; i++) {
				let angle = startAngle - (startAngle - endAngle) / splitNumber * i;
				let startX = Math.cos(angle) * r1;
				let startY = -Math.sin(angle) * r1;
				let endX = Math.cos(angle) * r2;
				let endY = -Math.sin(angle) * r2;
				if (i <= value * splitNumber / 100) {
					ctx.strokeStyle = 'white';
				} else {
					ctx.strokeStyle = '#ffa96d';
				}
				ctx.beginPath();
				ctx.moveTo(startX, startY);
				ctx.lineTo(endX, endY);
				ctx.stroke();
			}
			// 画指针
			ctx.strokeStyle = '#fff';
			let r3 = 115;
			let angle = startAngle - (startAngle - endAngle) / splitNumber * value * splitNumber / 100;
			let startX = Math.cos(Math.PI + angle) * 12;
			let startY = -Math.sin(Math.PI + angle) * 12;
			let endX = Math.cos(angle) * r3;
			let endY = -Math.sin(angle) * r3;
			ctx.lineWidth = 3;
			ctx.beginPath();
			ctx.moveTo(startX, startY);
			ctx.lineTo(endX, endY);
			ctx.stroke();
			// 
			// 画指针中心的圈圈
			ctx.beginPath();
			ctx.lineWidth = 2;
			ctx.fillStyle = "pink";
			ctx.arc(0, 0, 4, 0, 2 * Math.PI);
			ctx.fill();
			ctx.stroke();
			// 画钻石
			for (let i = 0; i <= diamondNumber; i++) {
				ctx.restore();
				// 不能连着用 restore(),没有用!
				ctx.save();
				let angle = startAngle - (startAngle - endAngle) / diamondNumber * i;
				let startX = Math.cos(angle) * 130;
				let startY = -Math.sin(angle) * 130;
				ctx.translate(center[0] + startX, center[1] + startY);
				ctx.rotate(Math.PI / 2 - angle);
				let scale = 50;
				if (i < diamondNumber * 0.5) {
					scale = 0.5
				} else if (i < diamondNumber * 0.8) {
					scale = i / diamondNumber
				} else {
					scale = 1;
				}
				ctx.scale(scale, scale);
				let img;
				if (i <= value * diamondNumber / 100) {
					img = document.getElementById('beautiful-icon');
				} else {
					img = document.getElementById('beautiful-icon-light');
				}
				if (value === 0) {
					img = document.getElementById('beautiful-icon-light');
				}
				ctx.drawImage(img, 0, 0, diamondSize * 2, diamondSize * 2, -diamondSize, -diamondSize, diamondSize * 2, diamondSize * 2);
			}
			ctx.restore();
		}
	</script>
	<script>
		beautiful_icon_base64 = '';
		beautiful_icon_light_base64 = ''
	</script>

</body>

</html>

猜你喜欢

转载自blog.csdn.net/tangran0526/article/details/105298037
今日推荐