SVG实现 进度条 效果
最近需求中,要实现一个进度条效果,如下图:
加上一些其他的显示节点信息就是这样(随便画画演示一下,请包涵~)
实现的方法呢,个人感觉用css不是很好实现,因为我没想到什么好办法,哈哈哈。
主要在于这个进度条是一个渐变颜色,而其中的节点呢,也需要同步动态位置的颜色。
最后选择了使用svg去实现这个效果。
构思层级
我的想法就是:首先拆分两层,背景层 和 颜色层,颜色层压着背景层,实现进度条效果
一、容器设定
你需要先设置一个容器,这个容器主要是定义进度条的宽度
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>Document</title>
<style>
.container {
margin: 50px auto;
width: 400px;
height: 100px;
}
</style>
</head>
<body>
<div class="container">
<svg ref="progressBar" width="100%" height="100%" xmlns="http://www.w3.org/2000/svg">
...插入内容
</svg>
</div>
</body>
</html>
二、背景层的实现
2.1、灰色背景条实现
通过在 <defs>
标签中定义一个名为progressBg的<rect>
裁剪路径,并填上背景色以及宽width
、高height
、圆角rx
属性,最后得到一个灰色背景
<svg width="100%" height="100%" xmlns="http://www.w3.org/2000/svg">
<defs>
<clipPath id="progressBg">
<rect x="0" y="24" width="100%" height="8" rx="8"></rect>
</clipPath>
</defs>
<rect clip-path="url(#progressBg)" fill="#f3f4f6" x="0" y="0" height="100%" width="100%"></rect>
</svg>
2.2、灰色背景节点实现
另外对于这个节点实现,需要进行下偏移计算
设定一个数据哈,比如:
target
代表进度条满条件数量,我们可以用来当计算几点位置的分母
doneNum
呢,代表的是我们已经完成的数量,我们可以等下用来计算彩条进度
nodeList
就是代表节点的个数还有表示到达阶段条件,amount
可以用来做背景节点的偏移量位置计算
const data = {
target: 150,
doneNum: 45,
nodeList: [
{
amount:50
},
{
amount:90
},
{
amount:150
}
]
}
我们现在需要将灰色背景节点偏移量计算,背景节点的半径我设置是7px哈,
在计算偏移量的时候,需要再向左偏移一个半径单位,保证原心在中间
最后将计算好的节点dom,插入到svg的实例中
// 容器的宽度400px
const grayNodeList = nodeList.map((item, index) => {
const positionX = (item.amount / target) * 400 - 7
return `<circle key="${index}" cx="${positionX}" cy="28" r="7"></circle>`
})
const result = `<svg width="100%" height="100%" xmlns="http://www.w3.org/2000/svg">
<defs>
<clipPath id="progressBg">
${grayNodeList.join('')}
<rect x="0" y="24" width="100%" height="8" rx="8"></rect>
</clipPath>
</defs>
<rect clip-path="url(#progressBg)" fill="#f3f4f6" x="0" y="0" height="100%" width="100%"></rect>
</svg>`
this.$refs.progressBar.innerHTML = result
得到如下背景层
三、颜色层以及整体的实现
3.1、 定义彩条渐变颜色
通过在 <defs>
标签中定义一个id名为subtraction的<linearGradient>
渐变背景色
<svg width="100%" height="100%" xmlns="http://www.w3.org/2000/svg">
<defs>
<linearGradient id="subtraction" x1="0" y1="0" x2="100%" y2="0">
<stop offset="10.89%" stop-color="#ff8c28"></stop>
<stop offset="88.45%" stop-color="#ffd74a"></stop>
<stop offset="100.3%" stop-color="#ffdc99"></stop>
</linearGradient>
<clipPath id="progressBg">
<rect x="0" y="24" width="100%" height="8" rx="8"></rect>
</clipPath>
</defs>
<rect clip-path="url(#progressBg)" fill="#f3f4f6" x="0" y="0" height="100%" width="100%"></rect>
</svg>
它主要定义的css属性是
background-color: linear-gradient(to right, #ff8c28 10.89%, #ffd74a 88.45%, #ffdc99 100.3%);
3.2、彩色背景条 & 整体实现
通过在 <defs>
标签中定义一个名为progressBar的<rect>
裁剪路径,并填上背景色以及宽width
、高height
、圆角rx
属性,最后得到一个彩背景,宽度需要计算一下
参照 灰色背景条和节点的实现,注意渲染的点
彩条宽度需要根据数据中的doneNum
已完成数 和 target
目标数以及容器宽度400px来计算
const colorBarWidth = doneNum/target * 400
彩条的节点计算注意 达到条件的节点才需要渲染,这里彩色节点我是设置的比灰色节点要小1px,半径为6px
// 先过滤达到条件的节点才需要渲染
const canGotAwardList = await nodeList.filter((item: any) => doneNum >= item.amount)
// 计算彩色节点偏移量
const colorNodeList = nodeList.map((item, index) => {
const positionX = (item.amount / target) * 400 - 6
return `<circle key="${index}" cx="${positionX}" cy="28" r="6"></circle>`
})
const result = `<svg width="100%" height="100%" xmlns="http://www.w3.org/2000/svg">
<defs>
<linearGradient id="subtraction" x1="0" y1="0" x2="100%" y2="0">
<stop offset="10.89%" stop-color="#ff8c28"></stop>
<stop offset="88.45%" stop-color="#ffd74a"></stop>
<stop offset="100.3%" stop-color="#ffdc99"></stop>
</linearGradient>
<clipPath id="progressBg">
${grayNodeList.join('')}
<rect x="0" y="24" width="100%" height="8" rx="8"></rect>
</clipPath>
<clipPath id="progressBar">
${colorNodeList.join('')}
<rect x="1" y="25" width=${colorBarWidth} height="6" rx="6" ry="6"></rect>
</clipPath>
</defs>
<rect clip-path="url(#progressBg)" fill="#f3f4f6" x="0" y="0" height="100%" width="100%"></rect>
<rect clip-path="url(#progressBar)" fill="url(#subtraction)" x="0" y="0" height="100%" width="100%"></rect>
</svg>`
this.$refs.progressBar.innerHTML = result
finally
最后的话,实现效果还可以,达到验收要求咯。
换你们的话,会用什么实现呢?css
镂空 或者是 canvas
,还是其他,
欢迎JY大佬们在评论区建议发言!
转载自:https://juejin.cn/post/7374683456716537895