這篇文章將為大家詳細(xì)講解有關(guān)如何在canvas中使用環(huán)形倒計(jì)時(shí)組件,文章內(nèi)容質(zhì)量較高,因此小編分享給大家做個(gè)參考,希望大家閱讀完這篇文章后對相關(guān)知識有一定的了解。
創(chuàng)新互聯(lián)是一家集網(wǎng)站建設(shè),札達(dá)企業(yè)網(wǎng)站建設(shè),札達(dá)品牌網(wǎng)站建設(shè),網(wǎng)站定制,札達(dá)網(wǎng)站建設(shè)報(bào)價(jià),網(wǎng)絡(luò)營銷,網(wǎng)絡(luò)優(yōu)化,札達(dá)網(wǎng)站推廣為一體的創(chuàng)新建站企業(yè),幫助傳統(tǒng)企業(yè)提升企業(yè)形象加強(qiáng)企業(yè)競爭力。可充分滿足這一群體相比中小企業(yè)更為豐富、高端、多元的互聯(lián)網(wǎng)需求。同時(shí)我們時(shí)刻保持專業(yè)、時(shí)尚、前沿,時(shí)刻以成就客戶成長自我,堅(jiān)持不斷學(xué)習(xí)、思考、沉淀、凈化自己,讓我們?yōu)楦嗟钠髽I(yè)打造出實(shí)用型網(wǎng)站。
Canvas環(huán)形倒計(jì)時(shí)組件
Canvas環(huán)形倒計(jì)時(shí)是基于Canvas實(shí)現(xiàn)的倒計(jì)時(shí),建議于移動端使用
Canvas環(huán)形倒計(jì)時(shí) 下載地址
一、如何使用
1. html代碼
ID屬性可隨意取名
<canvas id="canvas"></canvas>
2. 引入process.js文件
頁面引用
<script src="js/process.js"></script>
3. 初始化參數(shù)
實(shí)例化即可
<script> window.onload = function () { let ctd = new Countdown(); ctd.init(); }; </script>
二、settings參數(shù)說明
以下參數(shù)非必選項(xiàng),可根據(jù)具體需求配置
window.onload = function () { let ctd = new Countdown(); ctd.init({ id: "canvas", // ID,canvas一定要有ID屬性 size: 130, // 繪制圓形的最大尺寸,寬=高 borderWidth: 4, // 邊框?qū)挾? borderColor:"#fff", // 邊框顏色 outerColor:"#fff", // 最外層底圓顏色 scheduleColor:"#fff", // 進(jìn)度條動畫顏色 fontColor: "#fff", // 字體顏色 ringColor: "#ffc720", // 進(jìn)度條環(huán)形顏色 innerColor: "#4e84e5",// 最內(nèi)圓底色 fontSize: 50, time: 5 }); };
三、示例代碼
html
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Title</title> <style> body { background: #c2c1ce; } .container { position: absolute; left: 50%; top: 50%; transform: translate(-50%, -50%); width: 130px; height: 130px; text-align: center; } </style> </head> <body> <div class="container"> <canvas class="canvas" id="canvas"></canvas> </div> <script src="js/process.js"></script> <script> window.onload = function () { let ctd = new Countdown(); ctd.init(); }; </script> </body> </html>
js
/** * Created by 譚瞎 on 2018/3/15. */ function Countdown() { // 設(shè)置默認(rèn)參數(shù) this.settings = { id: "canvas", // ID,canvas一定要有ID屬性 size: 130, // 繪制圓形的最大尺寸,寬=高 borderWidth: 4, // 邊框?qū)挾? borderColor:"#fff", // 邊框顏色 outerColor:"#fff", // 最外層底圓顏色 scheduleColor:"#fff", // 進(jìn)度條動畫顏色 fontColor: "#fff", // 字體顏色 ringColor: "#ffc720", // 進(jìn)度條環(huán)形顏色 innerColor: "#4e84e5",// 最內(nèi)圓底色 fontSize: 50, time: 5 } } Countdown.prototype.init = function (opt) { this.obj = document.getElementById(this.settings.id); this.obj.width = this.settings.size; this.obj.height = this.settings.size; this.ctx = this.obj.getContext("2d"); extend(this.settings, opt); this.countdown(); }; // 繪制底色 Countdown.prototype.drawBackground = function () { this.drawCircle(0, 360, 0, this.settings.outerColor); }; // 繪制進(jìn)度條動畫背景 Countdown.prototype.drawProcess = function () { this.drawCircle(0, 360, 4, this.settings.ringColor); }; // 繪制倒計(jì)時(shí) Countdown.prototype.drawInner = function () { this.drawCircle(0, 360, 23, this.settings.innerColor); this.strokeBorder(this.settings.borderWidth); }; // 繪制進(jìn)度條動畫 Countdown.prototype.drawAnimate = function () { // 旋轉(zhuǎn)的角度 let deg = Math.PI / 180; let v = schedule * 360, startAng = -90, endAng = -90 + v; this.ctx.beginPath(); this.ctx.moveTo(this.settings.size / 2, this.settings.size / 2); this.ctx.arc(this.settings.size / 2, this.settings.size / 2, this.settings.size / 2 -3, startAng * deg, endAng * deg, false); this.ctx.fillStyle = this.settings.scheduleColor; this.ctx.fill(); this.ctx.closePath(); }; // 繪制邊框 Countdown.prototype.strokeBorder = function (borderWidth) { this.ctx.lineWidth = borderWidth; this.ctx.strokeStyle = this.settings.borderColor; this.ctx.stroke(); }; // 繪制文字 Countdown.prototype.strokeText = function (text) { this.ctx.textAlign = "center"; this.ctx.textBaseline = "middle"; this.ctx.font = this.settings.fontSize+"px"+ " microsoft yahei"; this.ctx.fillStyle = this.settings.fontColor; this.ctx.fillText(text, this.settings.size / 2, this.settings.size / 2); }; // 繪制圓 Countdown.prototype.drawCircle = function (startAng, endAng, border, fillColor) { let deg = Math.PI / 180; this.ctx.beginPath(); this.ctx.arc(this.settings.size / 2, this.settings.size / 2, this.settings.size / 2 -border, startAng * deg, endAng * deg, false); this.ctx.fillStyle = fillColor; this.ctx.fill(); this.ctx.closePath(); }; // 進(jìn)度條動畫 Countdown.prototype.countdown = function () { let oldTime = +new Date(); timer = setInterval(() => { let allMs = this.settings.time * 1000,// 如30*1000=30 000ms currentTime = +new Date(); // 步長=(當(dāng)前的時(shí)間-過去的時(shí)間)/總秒數(shù) schedule = (currentTime - oldTime) / allMs; this.schedule = schedule; this.drawAll(schedule); if (currentTime - oldTime >= allMs) { // 重繪 this.drawBackground(); this.drawProcess(); this.drawAnimate(); this.drawInner(); this.strokeText(0); clearInterval(timer); } }, 100); }; // 繪制所有 Countdown.prototype.drawAll = function (schedule) { schedule = schedule >= 1 ? 1 : schedule; let text = parseInt(this.settings.time * (1 - schedule)) + 1; // 清除畫布 this.ctx.clearRect(0, 0, this.settings.size, this.settings.size); this.drawBackground(); this.drawProcess(); this.drawAnimate(); this.drawInner(); this.strokeText(text); }; // 對象拷貝 function extend(obj1,obj2){ for(let attr in obj2){ obj1[attr] = obj2[attr]; } }
四、附加——canvas準(zhǔn)備工作
canvas其實(shí)沒有那么玄乎,它不外乎是一個(gè)H5的標(biāo)簽,跟其它HTML標(biāo)簽如出一轍:
<canvas id="canvas"></canvas>
注意最好在一開始的時(shí)候就給canvas設(shè)置好其寬高(若不設(shè)定寬高,瀏覽器會默認(rèn)設(shè)置canvas大小為寬300、高100像素),而且不能使用css來設(shè)置(會被拉伸),建議直接寫于canvas標(biāo)簽內(nèi)部:
<canvas id="canvas" width="130" height="130"></canvas>
canvas本身沒有任何的繪圖能力,所有的繪圖工作都是通過js來實(shí)現(xiàn)的。通常我們在js通過getElementById來獲取要操作的canvas(這意味著得給canvas設(shè)個(gè)id):
var c = document.getElementById("canvas"); var ctx = c.getContext("2d");
1.準(zhǔn)備好畫筆之后就可以開始繪圖了,環(huán)形其實(shí)就是半徑不同的同心圓,圓心坐標(biāo)是(size/2,size/2), 先畫一個(gè)最大的白色背景底圓,半徑是size/2。
let deg = Math.PI / 180; // beginPath()可以做到隔離路徑繪制效果的作用,防止之前的效果被污染。 ctx.beginPath(); // tcx.arc(圓心X,圓心Y,半徑,起始角度,結(jié)束角度,順逆時(shí)針); ctx.arc(size / 2, size / 2, size / 2, 0* deg, 360 * deg, false); ctx.fillStyle = "#fff"; ctx.fill(); ctx.closePath();
2.開始畫第二個(gè)黃色打底圓,圓心也是(size/2,size/2),只是半徑比白色底圓小4px,所以黃色底圓的半徑是(size/2-4)
let deg = Math.PI / 180; // beginPath()可以做到隔離路徑繪制效果的作用,防止之前的效果被污染。 ctx.beginPath(); // tcx.arc(圓心X,圓心Y,半徑,起始角度,結(jié)束角度,順逆時(shí)針); ctx.arc(size / 2, size / 2, size / 2-4, 0* deg, 360 * deg, false); ctx.fillStyle = "#fff"; ctx.fill(); ctx.closePath();
3.開始畫藍(lán)色內(nèi)圓,同理圓心為(size/2,size/2),半徑為(size-23),再給它加上4px的白色邊框。
let deg = Math.PI / 180; // beginPath()可以做到隔離路徑繪制效果的作用,防止之前的效果被污染。 ctx.beginPath(); // tcx.arc(圓心X,圓心Y,半徑,起始角度,結(jié)束角度,順逆時(shí)針); ctx.arc(size / 2, size / 2, size / 2-23, 0* deg, 360 * deg, false); ctx.fillStyle = "#fff"; ctx.fill(); ctx.closePath(); // 白色邊框 ctx.lineWidth = 4; ctx.strokeStyle = #fff; ctx.stroke();
4.繪制文字,垂直居中
ctx.textAlign = "center"; ctx.textBaseline = "middle"; ctx.fillStyle = "#fff"; // ctx.fillText(文字,相對畫布的X坐標(biāo),相對畫布的Y坐標(biāo)) ctx.fillText(30, size / 2, size / 2);
5.如何制作動畫?其實(shí)也是畫白色圓的過程,慢慢的覆蓋黃色進(jìn)度條的過程,那么先把白色的圓畫出來出來,這個(gè)時(shí)候藍(lán)圓就會被白色的動畫圓給蓋住,這個(gè)時(shí)候最后畫藍(lán)圓就好了。
let deg = Math.PI / 180; ctx.beginPath(); // tcx.arc(圓心X,圓心Y,半徑,起始角度,結(jié)束角度,順逆時(shí)針); ctx.arc(size / 2, size / 2, size / 2-4, 0* deg, 360 * deg, false); ctx.fillStyle = "#fff"; ctx.fill(); ctx.closePath();
6.比較簡單的繪畫過程完成了,接下來要將動畫和數(shù)字關(guān)聯(lián)起來,利用當(dāng)前的最新時(shí)間-最開始的時(shí)間,再除總的時(shí)間可以得到一個(gè)關(guān)鍵的百分比,這個(gè)百分比決定數(shù)字的變化,以及白色動畫圓繪制的角度。
Countdown.prototype.countdown = function () { let oldTime = +new Date();// 過去的時(shí)間:1522136419291 timer = setInterval(() => { let currentTime = +new Date();// 現(xiàn)在的時(shí)間:1522136419393 let allMs = this.settings.time * 1000;// 總時(shí)間豪秒數(shù):如30*1000=30 000ms schedule = (currentTime - oldTime) / allMs;// 繪制百分比:(1522136419393-1522136419291)/30000=0.0204 this.schedule = schedule; this.drawAll(schedule); if (currentTime - oldTime >= allMs) { // 重繪 this.drawBackground(); this.drawProcess(); this.drawAnimate(); this.drawInner(); this.strokeText(0); clearInterval(timer); } }, 10); }; // 繪制所有 Countdown.prototype.drawAll = function (schedule) { schedule = schedule >= 1 ? 1 : schedule; let text = parseInt(this.settings.time * (1 - schedule)) + 1; // 清除畫布 this.ctx.clearRect(0, 0, this.settings.size, this.settings.size); this.drawBackground(); this.drawProcess(); this.drawAnimate(); this.drawInner(); this.strokeText(text); }; // 繪制進(jìn)度條動畫 Countdown.prototype.drawAnimate = function () { // 旋轉(zhuǎn)的角度 let deg = Math.PI / 180; let v = schedule * 360, startAng = -90,// 開始角度 endAng = -90 + v;// 結(jié)束角度 this.ctx.beginPath(); this.ctx.moveTo(this.settings.size / 2, this.settings.size / 2); this.ctx.arc(this.settings.size / 2, this.settings.size / 2, this.settings.size / 2 - 3, startAng * deg, endAng * deg, false); this.ctx.fillStyle = this.settings.scheduleColor; this.ctx.fill(); this.ctx.closePath(); };
面向過程版本
/** * 進(jìn)度條動畫 */ countdown: function () { this.getSystemInfo().then(v => { // 自適應(yīng) let width = v.windowWidth, size = width >= 414 ? 66 : 400 / 414 * 66; size = parseInt(size); size = size % 2 ? size + 1 : size; let maxtime =30, sTime = +new Date, temp = setInterval(() => { let time = maxtime * 1000, currentTime = +new Date, schedule = (currentTime - sTime) / time; this.drew(schedule, maxtime, size); if (currentTime - sTime >= time) { // 繪制文字 this.setData({ schedule: 0 }); clearInterval(temp); }; }, 100); }); }, /** * 繪制 */ drew: function (schedule, val, size) { size = size || 66; const _ts = this; schedule = schedule >= 1 ? 1 : schedule; let text = parseInt(val - val * schedule), r = size / 2, deg = Math.PI / 180; _ts.setData({ width: size, height: size, schedule: text + 1 }); // 清除畫布 ctx.clearRect(0, 0, size, size); // 繪制白色底 ctx.beginPath(); ctx.arc(r, r, r, 0 * deg, 360 * deg); ctx.fillStyle = 'rgba(255,255,255,1)'; ctx.closePath(); ctx.fill(); // 繪制橙色 ctx.beginPath(); ctx.arc(r, r, r - 2, 0 * deg, 360 * deg); ctx.fillStyle = 'rgba(248,200,80,1)'; ctx.closePath(); ctx.fill(); // 繪制白色進(jìn)度條 let v = schedule * 360; ctx.beginPath(); ctx.moveTo(r, r); ctx.arc(r, r, r, -90 * deg, (-90 + v) * deg); ctx.fillStyle = 'rgba(255,255,255,1)'; ctx.closePath(); ctx.fill(); // 中心藍(lán)色底 ctx.beginPath(); ctx.arc(r, r, r - 12, 0 * deg, 360 * deg); ctx.fillStyle = 'rgba(90,140,220,1)'; ctx.closePath(); ctx.fill(); // 繪制文字 ctx.strokeText(); // 統(tǒng)一畫 ctx.draw(); },
關(guān)于如何在canvas中使用環(huán)形倒計(jì)時(shí)組件就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,可以學(xué)到更多知識。如果覺得文章不錯(cuò),可以把它分享出去讓更多的人看到。
網(wǎng)站標(biāo)題:如何在canvas中使用環(huán)形倒計(jì)時(shí)組件
網(wǎng)站網(wǎng)址:http://www.rwnh.cn/article38/ippipp.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供全網(wǎng)營銷推廣、用戶體驗(yàn)、網(wǎng)站改版、響應(yīng)式網(wǎng)站、網(wǎng)站導(dǎo)航、微信小程序
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來源: 創(chuàng)新互聯(lián)