摘要:現(xiàn)在又多了一種實(shí)現(xiàn)動畫的方案,那就是還在草案當(dāng)中的方法。這個(gè)方法就是傳遞給的回調(diào)函數(shù)。為回調(diào)函數(shù)一個(gè)簡單的例子模擬一個(gè)進(jìn)度條動畫,初始寬度為在函數(shù)中將進(jìn)度加然后再更新到寬度上,在進(jìn)度達(dá)到之前,一直重復(fù)這一過程。
HTML5/CSS3時(shí)代,我們要在web里做動畫選擇其實(shí)已經(jīng)很多了:
你可以用CSS3的animattion+keyframes;
你也可以用css3的transition;
你還可以用通過在canvas上作圖來實(shí)現(xiàn)動畫,也可以借助jQuery動畫相關(guān)的API方便地實(shí)現(xiàn);
當(dāng)然最原始的你還可以使用window.setTimout()或者window.setInterval()通過不斷更新元素的狀態(tài)位置等來實(shí)現(xiàn)動畫,前提是畫面的更新頻率要達(dá)到每秒60次才能讓肉眼看到流暢的動畫效果。
現(xiàn)在又多了一種實(shí)現(xiàn)動畫的方案,那就是還在草案當(dāng)中的window.requestAnimationFrame()方法。
初識requestAnimationFrame來看MDN上對其給出的詮釋:
The window.requestAnimationFrame() method tells the browser that you wish to perform an animation and requests that the browser call a specified function to update an animation before the next repaint. The method takes as an argument a callback to be invoked before the repaint.
window.requestAnimationFrame() 將告知瀏覽器你馬上要開始動畫效果了,后者需要在下次動畫前調(diào)用相應(yīng)方法來更新畫面。這個(gè)方法就是傳遞給window.requestAnimationFrame()的回調(diào)函數(shù)。
也可以說這個(gè)方法原理其實(shí)也就跟setTimeout/setInterval差不多,通過遞歸調(diào)用同一方法來不斷更新畫面以達(dá)到動起來的效果,但它優(yōu)于setTimeout/setInterval的地方在于它是由瀏覽器專門為動畫提供的API,在運(yùn)行時(shí)瀏覽器會自動優(yōu)化方法的調(diào)用,并且如果頁面不是激活狀態(tài)下的話,動畫會自動暫停,有效節(jié)省了CPU開銷。
基本語法可以直接調(diào)用,也可以通過window來調(diào)用,接收一個(gè)函數(shù)作為回調(diào),返回一個(gè)ID值,通過把這個(gè)ID值傳給window.cancelAnimationFrame()可以取消該次動畫。
requestAnimationFrame(callback)//callback為回調(diào)函數(shù)
?
一個(gè)簡單的例子模擬一個(gè)進(jìn)度條動畫,初始div寬度為1px,在step函數(shù)中將進(jìn)度加1然后再更新到div寬度上,在進(jìn)度達(dá)到100之前,一直重復(fù)這一過程。
為了演示方便加了一個(gè)運(yùn)行按鈕。
0%window.requestAnimationFrame = window.requestAnimationFrame || window.mozRequestAnimationFrame || window.webkitRequestAnimationFrame || window.msRequestAnimationFrame; var start = null; var ele = document.getElementById("test"); var progress = 0; function step(timestamp) { progress += 1; ele.style.width = progress + "%"; ele.innerHTML=progress + "%"; if (progress < 100) { requestAnimationFrame(step); } } requestAnimationFrame(step); document.getElementById("run").addEventListener("click", function() { ele.style.width = "1px"; progress = 0; requestAnimationFrame(step); }, false);
?
瀏覽器支持情況既然還是草案狀態(tài)下引入的一個(gè)功能,在使用全我們就需要關(guān)心一下各瀏覽器對它的支持情況了。就目前來說,主流現(xiàn)代瀏覽器都對它提供了支持。
更為具體的瀏覽器兼容性可以在這里看到。
PolyfillPolyfill就是墊片,按發(fā)明這個(gè)詞的人的原話來說,它就是一段這樣的代碼,讓瀏覽器原生地支持我們期望使用的一些API。
就比如這里的requestAnimationFrame,在看到了上面的瀏覽器支持情況后,你就知道了比上面列出的瀏覽器版本老的就不支持該方法,但為了讓代碼能夠有更好的瀏覽器兼容性在老機(jī)器上也能運(yùn)行不報(bào)錯(cuò),我們可以寫一些代碼讓瀏覽器在不支持requestAnimationFrame的情況下使用window.setTimeout(),這是一種回退(fallback)到過去的方法。
這樣一來,就可以通俗一點(diǎn)的理解polyfill了,它就是備胎。
下面是由Paul Irish及其他貢獻(xiàn)者放在GitHub Gist上的代碼片段,用于在瀏覽器不支持requestAnimationFrame情況下的回退,回退到使用setTmeout的情況。當(dāng)然,如果你確定代碼是工作在現(xiàn)代瀏覽器中,下面的代碼是不必的。
// http://paulirish.com/2011/requestanimationframe-for-smart-animating/ // http://my.opera.com/emoller/blog/2011/12/20/requestanimationframe-for-smart-er-animating // requestAnimationFrame polyfill by Erik M?ller. fixes from Paul Irish and Tino Zijdel // MIT license (function() { var lastTime = 0; var vendors = ["ms", "moz", "webkit", "o"]; for (var x = 0; x < vendors.length && !window.requestAnimationFrame; ++x) { window.requestAnimationFrame = window[vendors[x] + "RequestAnimationFrame"]; window.cancelAnimationFrame = window[vendors[x] + "CancelAnimationFrame"] || window[vendors[x] + "CancelRequestAnimationFrame"]; } if (!window.requestAnimationFrame) window.requestAnimationFrame = function(callback, element) { var currTime = new Date().getTime(); var timeToCall = Math.max(0, 16 - (currTime - lastTime)); var id = window.setTimeout(function() { callback(currTime + timeToCall); }, timeToCall); lastTime = currTime + timeToCall; return id; }; if (!window.cancelAnimationFrame) window.cancelAnimationFrame = function(id) { clearTimeout(id); }; }());
?
上面代碼作用有二,一是把各瀏覽器前綴進(jìn)行統(tǒng)一,二是在瀏覽器沒有requestAnimationFrame方法時(shí)將其指向setTimeout方法。
提到備胎代碼呢,這里多說一句,在CSS代碼中,我們也經(jīng)常使用這種回退的技巧,即對同一條CSS規(guī)則,編寫多條以不同瀏覽器前綴開頭代碼,或者編寫一條備用樣式。
下面是一個(gè)CSS中的備胎代碼的例子:
div { background: rgb(0, 0, 0); /* fallback */ background: rgba(0, 0, 0, 0.5); }
?
代碼中設(shè)置div背景為黑色帶50%的透明度,但I(xiàn)E9-的瀏覽器是不支持rbga格式的顏色的,所以瀏覽器會回退到上一條CSS規(guī)則應(yīng)用rgb顏色。
Referencearticle about rAF from css tricks
article about rAF from Paul Irish
what is polyfill
setInterval與requestAnimationFrame的時(shí)間間隔測試
求索:GSAP的動畫快于jQuery嗎?為何?
求索:GSAP的動畫快于jQuery嗎?/ 續(xù) V1.1
by 劉哇勇
編輯 SegmentFault
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/106990.html
摘要:現(xiàn)在又多了一種實(shí)現(xiàn)動畫的方案,那就是還在草案當(dāng)中的方法。這個(gè)方法就是傳遞給的回調(diào)函數(shù)。為回調(diào)函數(shù)一個(gè)簡單的例子模擬一個(gè)進(jìn)度條動畫,初始寬度為在函數(shù)中將進(jìn)度加然后再更新到寬度上,在進(jìn)度達(dá)到之前,一直重復(fù)這一過程。 HTML5/CSS3時(shí)代,我們要在web里做動畫選擇其實(shí)已經(jīng)很多了: 你可以用CSS3的animattion+keyframes; 你也可以用css3的transition...
摘要:現(xiàn)在又多了一種實(shí)現(xiàn)動畫的方案,那就是還在草案當(dāng)中的方法。這個(gè)方法就是傳遞給的回調(diào)函數(shù)。為回調(diào)函數(shù)一個(gè)簡單的例子模擬一個(gè)進(jìn)度條動畫,初始寬度為在函數(shù)中將進(jìn)度加然后再更新到寬度上,在進(jìn)度達(dá)到之前,一直重復(fù)這一過程。 HTML5/CSS3時(shí)代,我們要在web里做動畫選擇其實(shí)已經(jīng)很多了: 你可以用CSS3的animattion+keyframes; 你也可以用css3的transition...
摘要:首先,下載并在的中使用然后,我們需要準(zhǔn)備一個(gè)模型,在函數(shù)中,創(chuàng)建變量,用于導(dǎo)入模型導(dǎo)入模型的時(shí)候,接受兩個(gè)參數(shù),第一個(gè)表示模型路徑,第二個(gè)表示完成導(dǎo)入后的回調(diào)函數(shù),一般我們需要在這個(gè)回調(diào)函數(shù)中將導(dǎo)入的模型添加到場景中。 9. 動畫 在本章之前,所有畫面都是靜止的,本章將介紹如果使用Three.js進(jìn)行動態(tài)畫面的渲染。此外,將會介紹一個(gè)Three.js作者寫的另外一個(gè)庫,用來觀測每秒幀數(shù)...
摘要:對于廣大的前端開發(fā)人員來說,網(wǎng)站構(gòu)建本是家常便飯其中也不得不涉及到性能優(yōu)化的問題。將不影響首屏的資源和當(dāng)前屏幕資源不用的資源放到用戶需要時(shí)才加載,可以大大提升重要資源的顯示速度和降低總體流量按需加載會導(dǎo)致大量重繪,影響渲染性能。對于廣大的前端開發(fā)人員來說,網(wǎng)站構(gòu)建本是家常便飯;其中也不得不涉及到性能優(yōu)化的問題。之前也有接觸過,今天總結(jié)一下這方面的技巧,下面是我的一下認(rèn)知,歡迎探討: ? Nu...
閱讀 3583·2021-10-11 10:59
閱讀 1601·2021-09-29 09:35
閱讀 2269·2021-09-26 09:46
閱讀 3783·2021-09-10 10:50
閱讀 961·2019-08-29 12:17
閱讀 829·2019-08-26 13:40
閱讀 2443·2019-08-26 11:44
閱讀 2115·2019-08-26 11:22