亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > HTML > 正文

canvas環形倒計時組件的示例代碼

2024-08-26 00:20:45
字體:
來源:轉載
供稿:網友

本文介紹了canvas環形倒計時組件的示例代碼,分享給大家,具體如下:

效果如下圖一:

canvas,環形,倒計時,組件

Canvas環形倒計時組件

Canvas環形倒計時是基于Canvas實現的倒計時,建議于移動端使用

Canvas環形倒計時 下載地址

一、如何使用

1. html代碼

ID屬性可隨意取名

<canvas id="canvas"></canvas>

2. 引入process.js文件

頁面引用

<script src="js/process.js"></script>

3. 初始化參數

實例化即可

<script>    window.onload = function () {        let ctd = new Countdown();        ctd.init();    };</script>

二、settings參數說明

以下參數非必選項,可根據具體需求配置

window.onload = function () {        let ctd = new Countdown();        ctd.init({            id: "canvas",         // ID,canvas一定要有ID屬性            size: 130,            // 繪制圓形的最大尺寸,寬=高            borderWidth: 4,       // 邊框寬度            borderColor:"#fff",   // 邊框顏色            outerColor:"#fff",    // 最外層底圓顏色            scheduleColor:"#fff", // 進度條動畫顏色            fontColor: "#fff",    // 字體顏色            ringColor: "#ffc720", // 進度條環形顏色            innerColor: "#4e84e5",// 最內圓底色            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() {    // 設置默認參數    this.settings = {        id: "canvas",         // ID,canvas一定要有ID屬性        size: 130,            // 繪制圓形的最大尺寸,寬=高        borderWidth: 4,       // 邊框寬度        borderColor:"#fff",   // 邊框顏色        outerColor:"#fff",    // 最外層底圓顏色        scheduleColor:"#fff", // 進度條動畫顏色        fontColor: "#fff",    // 字體顏色        ringColor: "#ffc720", // 進度條環形顏色        innerColor: "#4e84e5",// 最內圓底色        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);};// 繪制進度條動畫背景Countdown.prototype.drawProcess = function () {    this.drawCircle(0, 360, 4, this.settings.ringColor);};// 繪制倒計時Countdown.prototype.drawInner = function () {    this.drawCircle(0, 360, 23, this.settings.innerColor);    this.strokeBorder(this.settings.borderWidth);};// 繪制進度條動畫Countdown.prototype.drawAnimate = function () {    // 旋轉的角度    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();};// 進度條動畫Countdown.prototype.countdown = function () {    let oldTime = +new Date();    timer = setInterval(() => {        let allMs = this.settings.time * 1000,// 如30*1000=30 000ms            currentTime = +new Date();        // 步長=(當前的時間-過去的時間)/總秒數        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準備工作

canvas其實沒有那么玄乎,它不外乎是一個H5的標簽,跟其它HTML標簽如出一轍:

<canvas id="canvas"></canvas>  

注意最好在一開始的時候就給canvas設置好其寬高(若不設定寬高,瀏覽器會默認設置canvas大小為寬300、高100像素),而且不能使用css來設置(會被拉伸),建議直接寫于canvas標簽內部:

<canvas id="canvas" width="130" height="130"></canvas>

canvas本身沒有任何的繪圖能力,所有的繪圖工作都是通過js來實現的。通常我們在js通過getElementById來獲取要操作的canvas(這意味著得給canvas設個id):

var c = document.getElementById("canvas");var ctx = c.getContext("2d");

1.準備好畫筆之后就可以開始繪圖了,環形其實就是半徑不同的同心圓,圓心坐標是(size/2,size/2), 先畫一個最大的白色背景底圓,半徑是size/2。

let deg = Math.PI / 180;// beginPath()可以做到隔離路徑繪制效果的作用,防止之前的效果被污染。ctx.beginPath();// tcx.arc(圓心X,圓心Y,半徑,起始角度,結束角度,順逆時針);ctx.arc(size / 2, size / 2, size / 2, 0* deg, 360 * deg, false);ctx.fillStyle = "#fff";ctx.fill();ctx.closePath();

canvas,環形,倒計時,組件

2.開始畫第二個黃色打底圓,圓心也是(size/2,size/2),只是半徑比白色底圓小4px,所以黃色底圓的半徑是(size/2-4)

let deg = Math.PI / 180;// beginPath()可以做到隔離路徑繪制效果的作用,防止之前的效果被污染。ctx.beginPath();// tcx.arc(圓心X,圓心Y,半徑,起始角度,結束角度,順逆時針);ctx.arc(size / 2, size / 2, size / 2-4, 0* deg, 360 * deg, false);ctx.fillStyle = "#fff";ctx.fill();ctx.closePath();

canvas,環形,倒計時,組件

3.開始畫藍色內圓,同理圓心為(size/2,size/2),半徑為(size-23),再給它加上4px的白色邊框。

let deg = Math.PI / 180;// beginPath()可以做到隔離路徑繪制效果的作用,防止之前的效果被污染。ctx.beginPath();// tcx.arc(圓心X,圓心Y,半徑,起始角度,結束角度,順逆時針);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();

canvas,環形,倒計時,組件

4.繪制文字,垂直居中

ctx.textAlign = "center";ctx.textBaseline = "middle";ctx.fillStyle = "#fff";// ctx.fillText(文字,相對畫布的X坐標,相對畫布的Y坐標)ctx.fillText(30, size / 2, size / 2);

canvas,環形,倒計時,組件

5.如何制作動畫?其實也是畫白色圓的過程,慢慢的覆蓋黃色進度條的過程,那么先把白色的圓畫出來出來,這個時候藍圓就會被白色的動畫圓給蓋住,這個時候最后畫藍圓就好了。

let deg = Math.PI / 180;ctx.beginPath();// tcx.arc(圓心X,圓心Y,半徑,起始角度,結束角度,順逆時針);ctx.arc(size / 2, size / 2, size / 2-4, 0* deg, 360 * deg, false);ctx.fillStyle = "#fff";ctx.fill();ctx.closePath();

canvas,環形,倒計時,組件

6.比較簡單的繪畫過程完成了,接下來要將動畫和數字關聯起來,利用當前的最新時間-最開始的時間,再除總的時間可以得到一個關鍵的百分比,這個百分比決定數字的變化,以及白色動畫圓繪制的角度。

Countdown.prototype.countdown = function () {    let oldTime = +new Date();// 過去的時間:1522136419291    timer = setInterval(() => {        let currentTime = +new Date();// 現在的時間:1522136419393        let allMs = this.settings.time * 1000;// 總時間豪秒數:如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);};// 繪制進度條動畫Countdown.prototype.drawAnimate = function () {    // 旋轉的角度    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();};

canvas,環形,倒計時,組件

面向過程版本

/**    * 進度條動畫    */    countdown: function () {        this.getSystemInfo().then(v => {            // 自適應            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();        // 繪制白色進度條        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();        // 中心藍色底        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();                // 統一畫        ctx.draw();                },

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持VeVb武林網。


注:相關教程知識閱讀請移步到HTML教程頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
日韩中文字幕国产| 亚洲社区在线观看| 午夜美女久久久久爽久久| 欧美人与性动交| 日本久久亚洲电影| 国产一区二区日韩精品欧美精品| 亚洲欧洲免费视频| 国产精品国产三级国产专播精品人| 成人在线视频网| 亚洲爱爱爱爱爱| 欧美性jizz18性欧美| 丝袜亚洲另类欧美重口| 久久精品影视伊人网| 狠狠色狠色综合曰曰| 欧美日韩另类字幕中文| 欧美富婆性猛交| 亚洲iv一区二区三区| 久久免费国产视频| 久久久久久欧美| 在线电影中文日韩| 日韩精品电影网| 国产精品免费在线免费| 国产精品久久久久国产a级| 亚洲性猛交xxxxwww| 欧美有码在线观看视频| 亚洲第一免费播放区| 久久九九有精品国产23| 亚洲成av人乱码色午夜| 色综合久久久888| 欧美一级高清免费| 国产suv精品一区二区| 欧美日韩一区二区三区| 亚洲无av在线中文字幕| 日本不卡免费高清视频| 国产精品黄页免费高清在线观看| 国产精品99久久久久久www| 欧美一级在线亚洲天堂| 国产精品爱啪在线线免费观看| 狠狠做深爱婷婷久久综合一区| 美女扒开尿口让男人操亚洲视频网站| 色噜噜狠狠狠综合曰曰曰| 国产精品日韩在线观看| 久久视频在线观看免费| 日韩在线激情视频| 国产精品video| 亚洲系列中文字幕| 日韩专区在线播放| 日韩成人av网| 亚洲成人久久久| 亚洲欧美国产一区二区三区| 亚洲黄色av女优在线观看| 亚洲加勒比久久88色综合| 国内精品在线一区| 色婷婷成人综合| 色yeye香蕉凹凸一区二区av| 青青草国产精品一区二区| 亚洲欧美日韩天堂| 日本免费在线精品| 欧美在线观看一区二区三区| 高跟丝袜一区二区三区| 51视频国产精品一区二区| 成人乱色短篇合集| 欧美国产乱视频| 亚洲小视频在线| 日本一欧美一欧美一亚洲视频| 青草青草久热精品视频在线观看| 欧美色欧美亚洲高清在线视频| 黑人巨大精品欧美一区二区| 欧美日韩免费在线观看| 91久久精品日日躁夜夜躁国产| 秋霞成人午夜鲁丝一区二区三区| 久久综合伊人77777蜜臀| 91在线视频九色| 中文字幕亚洲天堂| 国产精品免费久久久久影院| 亚洲精品98久久久久久中文字幕| 日本中文字幕成人| 国产成人在线亚洲欧美| 亚洲精品ady| 亚洲激情小视频| 亚洲欧美精品在线| 欧美一级黄色网| 精品国产欧美一区二区三区成人| 狠狠操狠狠色综合网| 国产精品专区h在线观看| 欧美成人在线免费视频| 日韩欧美成人网| 欧美美女18p| 欧美精品九九久久| 欧美夫妻性视频| 青青久久av北条麻妃海外网| 最近2019年手机中文字幕| 国产精品九九久久久久久久| 欧美大尺度激情区在线播放| 日韩视频在线免费观看| 55夜色66夜色国产精品视频| 成人免费大片黄在线播放| 国产精品久久久| 久久久国产成人精品| 久久久久久久久中文字幕| 欧美怡红院视频一区二区三区| 亚洲永久免费观看| 日本午夜精品理论片a级appf发布| 77777亚洲午夜久久多人| 91久热免费在线视频| 91嫩草在线视频| 欧美激情视频在线免费观看 欧美视频免费一| 国产精品久久一| 欧美日本高清一区| 国产99久久精品一区二区永久免费| 精品久久在线播放| 欧美性jizz18性欧美| 成人网页在线免费观看| 精品久久久一区| 国产精品香蕉国产| 日韩极品精品视频免费观看| 亚洲人成在线观看| 国产精品激情av电影在线观看| 久久久免费观看| 国产精品精品视频| 日韩综合视频在线观看| 国产免费一区二区三区香蕉精| 久久视频精品在线| 欧美日韩另类视频| 亚洲黄在线观看| 91影视免费在线观看| 久久久www成人免费精品张筱雨| 欧美久久久精品| 92裸体在线视频网站| 亚洲欧洲免费视频| 亚洲激情久久久| 亚洲精品视频网上网址在线观看| 成人黄色在线免费| 欧美一性一乱一交一视频| 亚洲欧美中文字幕在线一区| 亚洲视频欧洲视频| 欧美丰满少妇xxxx| 欧美久久精品午夜青青大伊人| 日韩av日韩在线观看| 国产日韩精品在线播放| 欧美日韩激情网| 中文字幕亚洲情99在线| 懂色av一区二区三区| 国产精品亚洲网站| 亚洲精品自在久久| …久久精品99久久香蕉国产| 久久综合久久88| 欧美在线影院在线视频| 伊人成人开心激情综合网| 国产精品美女免费视频| 日韩欧美成人区| 亚洲va久久久噜噜噜| 日韩高清a**址| 久久视频在线视频| 中文国产成人精品| 国产欧美最新羞羞视频在线观看| 日韩在线观看免费高清完整版| 超碰精品一区二区三区乱码| 亚洲精品乱码久久久久久金桔影视| 18久久久久久| 亚洲伊人一本大道中文字幕| 国产美女主播一区| 欧美亚洲国产视频| 国产97在线|亚洲|