因爲本人最近在作一些 growth hacking 的工做,業務上之後可能也會涉及去作一些可以在朋友圈火爆分享的 H5 頁面,忽然想到去年看到一個網易娛樂年度新聞盤點的 H5 頁面很是的新穎,採用畫中畫的形式依次串聯十多個手繪娛樂圖片,加上洗腦的「好運來」音樂,讓人有很大的分享的慾望。css
手機掃碼體驗網易年度娛樂盤點:前端
接下來咱們來一步步實現這樣的一個 H5 頁面,首先,咱們須要搞懂這個頁面用到了那些前端的知識點。css3
首屏有不少動畫,其中大多數是用雪碧圖+animation 的 step 動畫函數實現的,包括底部的鼓,右上角的鑔,中間人物飄動的頭髮。腳下來回滾動的浪花就是普通的 animation 動畫。除了首屏的這些動畫,後面切換到某些場景的時候也會有動畫,這些動畫也是用的雪碧圖動畫。git
這個就是使用 audio 元素便可,設置 audio 爲循環播放,當點擊右上角鑔的動圖的時候,調用 audio.pause()便可。github
在首屏,當長按鼓的時候,頁面的 animation 動畫會中止,靜態畫面一點點的縮小,直至出現第一個完整的畫中畫。此時過渡動畫中止,頁面 animation 動畫(白百何一指禪)開始出現。canvas
咱們先來分析這一小段,咱們代碼上要作哪些工做。瀏覽器
首先,咱們須要兩個圖層,一個 canvas 圖層用來展現場景過渡動畫,z-index 較低;一個展現場景動畫的圖層,咱們叫作 gif 圖層,z-index 較高;微信
在 canvas 圖層裏,咱們使用 drawImage()這個方法來繪製每一幀的過渡圖片,咱們先來看看這個方法的使用方式:app
context.drawImage(img,sx,sy,swidth,sheight,x,y,width,height);dom
參數值
參數 | 描述 |
---|---|
img | 規定要使用的圖像、畫布或視頻。 |
sx | 可選。開始剪切的 x 座標位置。 |
sy | 可選。開始剪切的 y 座標位置。 |
swidth | 可選。被剪切圖像的寬度。 |
sHeight | 可選。被剪切圖像的高度。 |
x | 在畫布上放置圖像的 x 座標位置。 |
y | 在畫布上放置圖像的 y 座標位置。 |
width | 可選。要使用的圖像的寬度。(伸展或縮小圖像) |
height | 可選。要使用的圖像的高度。(伸展或縮小圖像)。 |
過渡動畫的每一幀,咱們都要在 canvas 上面使用 drawImage 繪製兩張圖片,一張是大圖,一張是畫中畫裏的小圖,以第一個過渡動畫爲例,大圖是 P2,小圖是 P1,
如圖:(原諒我不知道使用什麼工具畫圖,只好動手了)
咱們假設大圖 P2 是長方形 ABCD,小圖 P1 是長方形 IJKL,動畫過程當中某一時刻的手機屏幕是長方形 EFGH,咱們有個前提條件就是這三個長方形都是寬高比爲 750:1206 的長方形,並且,全部的圖片寬高像素大小是相等的(網易的場景圖片大小統一爲:1875*3015)這也意味着 iPhone X 等全面屏手機的適配會有問題,在 iPhone678 手機上表現良好。(看看今年網易會不會解決這個問題,畢竟全面屏手機愈來愈多)。
那麼,在這樣的一個時刻,咱們須要在 canvas 上面畫兩張圖片,
drawImage(P2,ME,NE,EF,EH,0,0,750,1206)
drwaImage(P1,0,0,AB,AD,OI,PI,IJ,IL)
複製代碼
那咱們知道了某一時刻的狀況,可是如何將畫面動起來,有一個收縮畫面的效果呢?
如今開始寫咱們的 render 函數:
const render = () => {
this.radio = this.radio * this.scale;
this.timer = requestAnimationFrame(render);
this.draw();// 繪製兩個圖片
};
draw() {
if (this.index + 1 != this.imgList.length) {
if (
this.radio <
this.imgList[this.index + 1].areaW / this.imgList[this.index + 1].imgW
) {
if (this.willPause) {
this.radio =
this.imgList[this.index + 1].areaW / this.imgList[this.index + 1].imgW;
cancelAnimationFrame(this.timer);
}
this.index++;
this.radio = 1;
if (!this.imgList[this.index + 1]) {
this.showEnd();
}
}
this.imgNext = this.imgList[this.index + 1];
this.imgCur = this.imgList[this.index];
this.containerImage = this.domList[this.index + 1];
this.innerImage = this.domList[this.index];
this.drawImgOversize(
this.containerImage,
this.imgNext.imgW,
this.imgNext.imgH,
this.imgNext.areaW,
this.imgNext.areaH,
this.imgNext.areaL,
this.imgNext.areaT,
this.radio,
),
this.drawImgMinisize(
this.innerImage,
this.imgCur.imgW,
this.imgCur.imgH,
this.imgNext.imgW,
this.imgNext.imgH,
this.imgNext.areaW,
this.imgNext.areaH,
this.imgNext.areaL,
this.imgNext.areaT,
this.radio,
);
}
}
複製代碼
render 函數裏面有兩個變量 radio 和 scale,radio = IJ/EF,因此在一個場景切換動畫中,咱們只須要改變 radio 的值,使其從 1 逐漸變小到等於 IJ/AB 便可。scale 就是這樣一個用來表示 radio 變化速率的常量。這裏咱們能夠定義爲 0.99,由於 requestAnimationFrame 的回調在瀏覽器裏面大約一秒會執行 60 次, 而 o.99^240 = 0.08 因此大約 4s 左右,咱們就能夠完成一個場景切換,這個速度仍是比較適中的。
從而,在動畫中的任一時刻,EF 的大小能夠表示爲 IJ/this.radio,另外,由於全部的圖片都是咱們的畫師製做的,因此,每張圖的像素大小(imgW、imgH)、小圖在大圖中的偏移位置 SI(areaL)、TI(areaT)、小圖的寬高 IJ(areaW)、IL(areaH),都是已知的,根據這些已知的數據,咱們能夠輕鬆的(對於數學好的同窗)將 drawImage 中未知變量用 this.radio 表示。
這樣,咱們一個切換動畫算是搞定了,可是咱們如何將多個切換動畫串聯起來呢,很簡單,看看 draw()的代碼,咱們只須要在 this.radio 達到臨界值時候,將 index++,從新給 imgNext 和 imgCur 賦值。
最後將 render 函數寫到 touchHandler 裏面便可。
touchHandler(e) {
e.stopPropagation();
// e.preventDefault();
const render = () => {
this.radio = this.radio * this.scale;
this.timer = requestAnimationFrame(render);
this.draw();
};
cancelAnimationFrame(this.timer);
this.willPause = false;
// clearInterval(this.gif_timer);
this.timer = requestAnimationFrame(render);
}
複製代碼
說是 gif 動畫,可是實現上仍是用雪碧圖+step 實現的。若是某一場景中有動畫展現的環節,那麼在過渡動畫結束時,我就能夠將 gif 圖層展現出來,gif 圖層有兩部分構成,一個是背景圖片,一個是動畫區域。背景圖片將動畫區域留白,動畫區域採用雪碧圖+step 的方式,實現動畫。這樣作是爲了減小圖片資源大小,加快加載速度。
這個 H5 頁面須要加載大量的圖片,而這些圖片必定要保證在用戶交互以前加載完成,因此咱們要給頁面初始化時候一個加載態,當全部圖片加載完成後,咱們才展現可交互的頁面。因此,咱們須要知道何時圖片已經加載好了,上代碼:
loadGifImg() {
const loadPromises = this.gifImgs.map(
item =>
new Promise((resolve, reject) => {
const img = new Image();
img.src = item;
img.onload = () => resolve(img);
img.onerror = () => reject();
}),
);
return Promise.all(loadPromises);
}
loadPageImg() {
const loadPromises = this.imgList.map(
(item, index) =>
new Promise((resolve, reject) => {
const img = new Image();
img.src = item.link;
img.i = index;
img.name = index;
img.className = 'item';
item.image = img;
img.onload = () => {
$('.collection').append(item.image);
resolve();
};
img.onerror = () => reject();
}),
);
return Promise.all(loadPromises);
}
複製代碼
因此,咱們只須要等這兩個 Promise resolve 了就加載完成了。
完整的代碼 github 歡迎 star
微信掃碼體驗