
经验:
save-restore-restore-restore
是不行的。用 save
保存的状态只能用一次,最后两个 restore 是回不到第一次 save 的状态的。
oninput
是变了就触发,onchange
是变了且 失去焦点 才触发
<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>
<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() {
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();
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>