[JavaScript]canvas绘制圆环进度条出现模糊效果解决方案
问题
近期用canvas绘制了圆环进度条,但是进度条出现周围模糊的现象,针对这种现象,网上搜了搜,有人提问,但是貌似没有很好的解决方案,针对这种情况,提出几种解决方案,仅供参考! 模糊效果如下:
解决方案
针对这种情况,我提出几种解决思路。
一、运用hidpi-canvas-polyfill 的js进行解决
HiDPI Canvas Polyfill 是针对设备提出的canvas高清解决方案,首先引入hidpi-canvas.js。
这个js会自动识别你的canvas,会把你的canvas变小,虽然不模糊了,但是不是我们想要的效果。(可以结合后面的方法进行改进)
关于hidpi-canvas-polyfill 地址:https://github.com/jondavidjohn/hidpi-canvas-polyfill
具体使用方法大家可以看他的描述。在这里就不展开讲解了。
缺点: 这种方式虽然可以解决,但是感觉毕竟要引入一些js还有,进行自动化识别中,canvas可能会变小,还有,他会自动给canvas加了一个宽高,这些在一定情况下不是我们想要的。
二、指定默认宽高法
这种方式在一定程度上,可以解决我说的模糊问题。将上面模糊的代码进行如下改进
<canvas id="pczren" data-process="70" width="250" height="250"></canvas>
canvas指定一个宽高,然后半径只要小于250/2就可以。中心点坐标直接是canvas的宽高除以2。
代码如下:
var pczren = document.getElementById('pczren');
var mprocess = pczren.getAttribute('data-process');
var mctx = pczren.getContext('2d');
var Wc = pczren.width;
var Hc = pczren.height;
function draw(ctx, process, colors, fco) {
// 画灰色的圆
ctx.beginPath();
ctx.arc(Wc/2, Hc/2,100, 0, Math.PI * 2);
ctx.closePath();
ctx.fillStyle = fco;
ctx.fill();
// 画进度环
ctx.beginPath();
ctx.moveTo(Wc/2, Hc/2);
ctx.arc(Wc/2, Hc/2, 100, Math.PI * 2.5, Math.PI * (2.5 + 2 * process / 100));
ctx.closePath();
ctx.fillStyle = colors;
ctx.fill();
// 画内填充圆
ctx.beginPath();
ctx.arc(Wc/2, Hc/2, 80, 0, Math.PI * 2);
ctx.closePath();
ctx.fillStyle = '#fff';
ctx.fill();
}
draw(mctx, mprocess, '#53b48d', '#edecec');
效果图如下:
三、canvas替代法
当然,这种方法不是我们想要的,在没有办法的情况下,我们可以选择替代方案来解决这个问题,我们可以想象,除了用canvas绘制圆环进度条之外,我们有没有其他方式呢?例如css3+jquery方案,css3方案等等。
3.1css3+jquery方案
html如下:
<div class="circle" style="left:220px">
<div class="pie_left"><div class="left"></div></div>
<div class="pie_right"><div class="right"></div></div>
<div class="mask"><span>15</span>%</div>
</div>
css如下:
body {
font-family: "微软雅黑";
}
.circle {
width: 200px;
height: 200px;
position: absolute;
border-radius: 50%;
background: #0cc;
}
.pie_left, .pie_right {
width:200px;
height:200px;
position: absolute;
top: 0;left: 0;
}
.left, .right {
width:200px;
height:200px;
background:#00aacc;
border-radius: 50%;
position: absolute;
top: 0;
left: 0;
}
.pie_right, .right {
clip:rect(0,auto,auto,100px);
}
.pie_left, .left {
clip:rect(0,100px,auto,0);
}
.mask {
width: 150px;
height: 150px;
border-radius: 50%;
left: 25px;
top: 25px;
background: #FFF;
position: absolute;
text-align: center;
line-height: 150px;
font-size: 20px;
font-weight: bold;
color: #00aacc;
}
jquery如下:
$(function() {
$('.circle').each(function(index, el) {
var num = $(this).find('span').text() * 3.6;
if (num<=180) {
$(this).find('.right').css('transform', "rotate(" + num + "deg)");
} else {
$(this).find('.right').css('transform', "rotate(180deg)");
$(this).find('.left').css('transform', "rotate(" + (num - 180) + "deg)");
};
});
});
上述代码可以实现圆环进度条!
3.2 css方案
**方法一:**用图片方式,n张图片,不停的background-position位置变化,模拟1%到100%的情况!
方法二:
圆环css写法如下:
.circleprogress{
width: 160px;
height: 160px;
border:20px solid red;
border-radius: 50%;
}
不完整的圆如下写法:
.circleprogress{
width: 160px;
height: 160px;
border:20px solid red;
border-left:20px solid transparent;
border-bottom:20px solid transparent;
border-radius: 50%;
}
但是不是45度角的倍数怎么办呢?如下代码可以用css动画实现进度条效果!
.circleProgress_wrapper{
width: 200px;
height: 200px;
margin: 50px auto;
position: relative;
border:1px solid #ddd;
}
.wrapper{
width: 100px;
height: 200px;
position: absolute;
top:0;
overflow: hidden;
}
.right{
right:0;
}
.left{
left:0;
}
.circleProgress{
width: 160px;
height: 160px;
border:20px solid rgb(232, 232, 12);
border-radius: 50%;
position: absolute;
top:0;
-webkit-transform: rotate(45deg);
}
.rightcircle{
border-top:20px solid green;
border-right:20px solid green;
right:0;
-webkit-animation: circleProgressLoad_right 5s linear infinite;
}
.leftcircle{
border-bottom:20px solid green;
border-left:20px solid green;
left:0;
-webkit-animation: circleProgressLoad_left 5s linear infinite;
}
@-webkit-keyframes circleProgressLoad_right{
0%{
border-top:20px solid #ED1A1A;
border-right:20px solid #ED1A1A;
-webkit-transform: rotate(45deg);
}
50%{
border-top:20px solid rgb(232, 232, 12);
border-right:20px solid rgb(232, 232, 12);
border-left:20px solid rgb(81, 197, 81);
border-bottom:20px solid rgb(81, 197, 81);
-webkit-transform: rotate(225deg);
}
100%{
border-left:20px solid green;
border-bottom:20px solid green;
-webkit-transform: rotate(225deg);
}
}
@-webkit-keyframes circleProgressLoad_left{
0%{
border-bottom:20px solid #ED1A1A;
border-left:20px solid #ED1A1A;
-webkit-transform: rotate(45deg);
}
50%{
border-bottom:20px solid rgb(232, 232, 12);
border-left:20px solid rgb(232, 232, 12);
border-top:20px solid rgb(81, 197, 81);
border-right:20px solid rgb(81, 197, 81);
-webkit-transform: rotate(45deg);
}
100%{
border-top:20px solid green;
border-right:20px solid green;
border-bottom:20px solid green;
border-left:20px solid green;
-webkit-transform: rotate(225deg);
}
}
发表评论