Canvas 實現畫中畫動畫效果--網易娛樂年度盤點H5動畫解密

前言

因爲本人最近在作一些 growth hacking 的工做,業務上之後可能也會涉及去作一些可以在朋友圈火爆分享的 H5 頁面,忽然想到去年看到一個網易娛樂年度新聞盤點的 H5 頁面很是的新穎,採用畫中畫的形式依次串聯十多個手繪娛樂圖片,加上洗腦的「好運來」音樂,讓人有很大的分享的慾望。css

手機掃碼體驗網易年度娛樂盤點:前端

一步步實現

接下來咱們來一步步實現這樣的一個 H5 頁面,首先,咱們須要搞懂這個頁面用到了那些前端的知識點。css3

css3 動畫 animation

首屏有不少動畫,其中大多數是用雪碧圖+animation 的 step 動畫函數實現的,包括底部的鼓,右上角的鑔,中間人物飄動的頭髮。腳下來回滾動的浪花就是普通的 animation 動畫。除了首屏的這些動畫,後面切換到某些場景的時候也會有動畫,這些動畫也是用的雪碧圖動畫。git

背景音樂

這個就是使用 audio 元素便可,設置 audio 爲循環播放,當點擊右上角鑔的動圖的時候,調用 audio.pause()便可。github

場景切換

在首屏,當長按鼓的時候,頁面的 animation 動畫會中止,靜態畫面一點點的縮小,直至出現第一個完整的畫中畫。此時過渡動畫中止,頁面 animation 動畫(白百何一指禪)開始出現。canvas

咱們先來分析這一小段,咱們代碼上要作哪些工做。瀏覽器

首先,咱們須要兩個圖層,一個 canvas 圖層用來展現場景過渡動畫,z-index 較低;一個展現場景動畫的圖層,咱們叫作 gif 圖層,z-index 較高;微信

drawImage

在 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 動畫

說是 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

微信掃碼體驗

若是以爲文章不錯話,不妨點個關注再走啊~

相關文章
相關標籤/搜索