canvas 简易的加载进度条
做一个web app,想在第一次或者更新的时候,有一个更新进度条,我个人比较喜欢圆的那种。
canvas + svg高低配,应该还不错的。顺便一提,canvas用来压缩图片也是么么哒的。
先看下效果图,我咋觉得边有点虚。基本的样子有了,但是美观,美观,我要美观,下次再来。

原理就是绘制圆和文本绘制。
特别说明:
1. 会自动计算canvas的宽高,取最大值,并重设宽高为最大值
2. 有简单的计算逻辑,让 百分比的文本居中
3. 超大的边宽会被限制
上代码。
class CanvasProgress {
constructor(cv, options = {
bgColor: '#123456',
cBgColor: 'green',
edgeWidth: 20
}) {
if (!cv || !cv.getContext) {
throw new Error('参数cv为空或者getContext方法未定义')
}
this._ctx = cv.getContext('2d')
this._diameter = this._getDiameter(cv) //直径
this._radius = Math.ceil(this._diameter / 2) //半径
this._options = Object.assign({
bgColor: '#123456', //未加载的背景色
cBgColor: 'green', //已加载的背景色
edgeWidth: 20, //边款
textMaxWidth: this._radius //进度文本最大长度
}, options)
this._options.edgeWidth = Math.min(this._radius * 0.28, this._options.edgeWidth) //重新计算
this._circleParams = {
x: this._radius,//圆心的x轴坐标值
y: this._radius,//圆心的y轴坐标值
r: Math.floor(this._radius - this._options.edgeWidth / 2 - 1) //圆的半径
}
this._resizeCanvas(cv) //调整canvas宽高一致
this._initialize() //绘制背景圆
}
/**
* 通过canvas获得直径
* @param {*canvas对象} cv
*/
_getDiameter(cv) {
return Math.max(cv.height || 0, cv.width || 0) || 200
}
/**
* 进度转换角度
* @param {*进度} progress
*/
_getAngle(progress) {
return (progress / 100) * Math.PI * 2
}
_resizeCanvas(cvProgress) {
cvProgress.width = cvProgress.height = this._diameter
}
/**
* 调整canvas宽高一致
*/
_initialize() {
let ctx = this._ctx
// 开始一个新的绘制路径
ctx.beginPath()
//设置弧线的颜色为蓝色
ctx.strokeStyle = this._options.bgColor
ctx.lineWidth = this._options.edgeWidth
//以canvas中的坐标点(100,100)为圆心,绘制一个半径为50px的圆形
ctx.arc(this._circleParams.x, this._circleParams.y, this._circleParams.r, 0, Math.PI * 2, false)
//按照指定的路径绘制弧线
ctx.stroke()
}
/**
* 更近进度
* @param {*进度 0-100} progressValue
* @param {*绘制圆的设置} circleSettings
* @param {*绘制文本的设置} progressTextSettings
*/
updateProgress(progressValue, circleSettings, progressTextSettings) {
if (progressValue <= 0) {
return
}
this._updateCircle(progressValue, circleSettings)
this._updateProgressText(progressValue, progressTextSettings)
}
/**
* 绘制圆
* @param {*进度} progressValue
* @param {*设置} settings
*/
_updateCircle(progressValue, settings) {
let ctx = this._ctx
ctx.beginPath()
let angle = this._getAngle(progressValue)
ctx.strokeStyle = this._options.cBgColor
ctx.lineWidth = this._options.edgeWidth
if (settings) {
Object.keys(settings).forEach(k => {
ctx[k] = settings[k]
})
}
ctx.arc(this._circleParams.x, this._circleParams.y, this._circleParams.r, 0 + Math.PI * 1.5, angle + Math.PI * 1.5, false)
ctx.stroke()
}
/**
* 绘制进度文本
* @param {*进度} progressValue
* @param {*设置} settings
*/
_updateProgressText(progressValue, settings) {
if (progressValue < 0 || progressValue > 100) {
return
}
let ctx = this._ctx,
r = this._radius,
fontSize = this._getFontSize(settings)
ctx.clearRect(r * 0.5, r * 0.5, r, r)
ctx.font = `${fontSize}px sans-serif`
ctx.fillStyle = this._getTextGradient()
if (settings) {
Object.keys(settings).forEach(k => {
ctx[k] = settings[k]
})
}
ctx.fillText(progressValue + '%', this._circleParams.x - this._options.textMaxWidth / 2, this._circleParams.y + fontSize / 2, this._options.textMaxWidth)
}
/**
* 渐变设置
*/
_getTextGradient() {
var gradient = this._ctx.createLinearGradient(0, 0, 100, 0);
gradient.addColorStop("0", "red");
gradient.addColorStop("0.5", "blue");
gradient.addColorStop("1.0", "green");
return gradient
}
/**
* 获得当前文字大小
* @param {*设置} settings
*/
_getFontSize(settings) {
if (settings && settings.font) {
let matchItem = settings.font.match(/\d{1,2}/)
if (matchItem) {
return Number.parseInt(matchItem[0])
}
}
return this._radius * 0.6
}
}
调用代码:
<!DOCTYPE>
<html> <head>
<title> canvas 原型进度条</title>
<meta name="viewport" content="width=device-width,initial-scale=1.0, minimum-scale=1.0, maximum-scale=1.0, user-scalable=no">
<meta charset="utf-8">
</head> <body style="margin:5% 10% 0 10%">
<div>
<canvas id="cvProgress" height="250" width="15">
</div>
<script src="CanvasProgress.js"></script>
</body>
<script>
let ps = new CanvasProgress(cvProgress, {
edgeWidth: 50
}), progress = 0
let tickets = setInterval(function () {
progress += 5
ps.updateProgress(progress,null,{
})
if (progress > 100) {
clearInterval(tickets)
}
}, 100) </script> </html>
源码地址:https://github.com/xiangwenhu/BlogCodes/tree/master/client/canvas
canvas 简易的加载进度条的更多相关文章
- Unity3D 场景切换加载进度条实现
需要三个场景,场景A,场景B,场景C: 场景A:一个按钮,点击加载场景B: 场景B:从A切换到C过度场景,加载进度条: 场景C:目标场景: 创建OnProgress.cs脚本: using Syste ...
- css3 linear-gradient实现页面加载进度条效果
最终效果图: html结构: <div> <p class="p1"> <span></span> < ...
- ajax页面加载进度条插件
下面两个都是youtube视频的加载进度条效果的ajax插件 一.官网:http://ricostacruz.com/nprogress/官网 github:https://github.com/rs ...
- pace.js – 加载进度条插件
这儿只是简单介绍一下这个插件pace.js. 在页面中引入Pace.js,页面就会自动监测你的请求(包括Ajax请求),在事件循环滞后,会在页面记录加载的状态以及进度情况.此插件的兼容性很好,可以兼容 ...
- 仿UC浏览器图片加载进度条
前几天用UC浏览器看新闻(无意中给UC打了广告),看到它的图片加载进度条,正好最近有时间,所以就自己写了一个. 效果图如下 进度条的底色和填充颜色都可以调整. 首先中间的笑脸作为一个整体,其实现代码如 ...
- 【Web前沿技术】纯 CSS3 打造的10个精美加载进度条动画
之前向大家介绍8款优秀的 jQuery 加载动画和进度条插件,今天这篇文章向大家推荐10个纯 CSS3 代码实现精美加载进度条动画效果的方案.加载动画和进度条在网站和 Web 应用中的使用非常流行,特 ...
- jQuery模拟页面加载进度条
因为我们无法通过任何方法获取整个页面的大小和当前加载了多少,所以想制作一个加载进度条的唯一办法就是模拟.那要怎么模拟呢? 我们知道,页面是从上往下执行的,也就是说我们可以大致估算出在页面的某个位置加载 ...
- js页面加载进度条
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...
- iOS UIWebView 加载进度条的使用-WKWebView的使用,更新2017.6.26
1.由于项目中加载网络插件,直接使用了webview加载.使用了三方NJKWebViewProgress进度条的使用,近期在测试时发现,网络缓慢时出现白屏,有卡顿现象. 于是采用了WKWebView进 ...
随机推荐
- N卡全部历史驱动
记录一下寻找驱动方法 打开链接 http://www.geforce.cn/drivers/beta-legacy 默认搜索出来是10个,之后打开控制台输入如下内容回车显示全部(100,可以修改数字来 ...
- (hdu step 8.1.6)士兵队列训练问题(数据结构,简单模拟——第一次每2个去掉1个,第二次每3个去掉1个.知道队伍中的人数<=3,输出剩下的人 )
题目: 士兵队列训练问题 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) Total ...
- 用HTML5实现的各种排序算法的动画比較
用HTML5实现的各种排序算法的动画比較 非常有意思,详见: http://www.webhek.com/misc/comparison-sort/
- linux下分析Java程序内存汇总
使用pmap查看进程内存 执行命令 使用pmap能够查看某一个进程(非java的也能够)的内存使用使用情况, 命令格式: pmap 进程id 演示样例说明 比如执行: pmap 12358 显示结果例 ...
- 11g使用非duplicate方式创建物理standby要注意的问题总结
在上篇博文中,使用了duplicate方式来创建物理standby http://blog.csdn.net/aaron8219/article/details/38434579 今天来说说在11g中 ...
- 基于lucene.net 和ICTCLAS2014的站内搜索的实现1
Lucene.net是一个搜索引擎的框架,它自身并不能实现搜索.须要我们自己在当中实现索引的建立,索引的查找.全部这些都是依据它自身提供的API来实现.Lucene.net本身是基于java的,可是经 ...
- PMP杂谈--PMP中一些easy忽视的地方
识别干系人:这个过程是持续的,在整个项目的生命周期中都要持续识别干系人. 组织过程资产和事业环境因素:这两个东西在过程的输入中似乎常常看到,但有时候又看不到,不要纠结了 ,不要浪费脑细胞去背诵哪个有, ...
- poj:4091:The Closest M Points
poj:4091:The Closest M Points 题目 描写叙述 每到饭点,就又到了一日几度的小L纠结去哪吃饭的时候了.由于有太多太多好吃的地方能够去吃,而小L又比較懒不想走太远,所以小L会 ...
- 掀起Azure AD的盖头来——深入理解Microsoft Graph应用程序和服务权限声明
作者:陈希章 发表于 2017年7月12日 引子 这是一篇计划外的文章.我们都知道要进行Microsoft Graph的开发的话,需要进行应用程序注册.这个在此前我已经有专门的文章写过了.但这里存在一 ...
- 微服务架构之RPC-client序列化细节
通过上篇文章的介绍,知道了要实施微服务,首先要搞定RPC框架,RPC框架的职责要向[调用方]和[服务提供方]屏蔽各种复杂性: (1)让调用方感觉就像调用本地函数一样 (2)让服务提供方感觉就像实现一个 ...