中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

css3如何實現彈幕效果

發布時間:2020-10-23 16:33:39 來源:億速云 閱讀:185 作者:小新 欄目:web開發

這篇文章將為大家詳細講解有關css3如何實現彈幕效果,小編覺得挺實用的,因此分享給大家做個參考,希望大家閱讀完這篇文章后可以有所收獲。

1. css3實現乞丐版的彈幕

(1)如何通過css3實現彈幕

首先來看如何通過css的方法實現一個最簡單的彈幕:

首先在html中定義一條彈幕的dom結構:

<p class="block">我是彈幕</p>

彈幕的移動可以通過移動這個block來實現,以從右向左移動的彈幕為例,彈幕的初始位置在容器的最左側且貼邊隱藏(彈幕的最左邊與容器的最右貼合),可以通過絕對定位加transform來實現:

.block{
   position:absolute;
}

初始位置:

from{
    left:100%;
    transform:translateX(0)
}

移動到最左邊的結束位置為(彈幕的最右邊與容器的最左邊貼合):

to{
   left:0;
   transform:translateX(-100%)
}

起始位置和結束位置的具體圖示如下所示:

css3如何實現彈幕效果

根據起始位置和結束位置可以定義完整的兩幀彈幕動畫:

@keyframes barrage{
   from{
     left:100%;
     transform:translateX(0);
   }
   to{
     left:0;
     transform:translateX(-100%);
   }
}

給彈幕元素引入這個動畫:

.block{
  position:absolute;
  /* other decorate style */
  animation:barrage 5s linear 0s;
}

這樣就可以實現一個乞丐版的彈幕效果:

css3如何實現彈幕效果

(2)通過絕對定位和left實現彈幕的缺陷

首先明確一下css的渲染過程

  • I)根據HTML的結構生成DOM樹(DOM樹中包含了display:none的節點)

  • II)在DOM樹的基礎上,根據節點的幾何屬性(margin/padding/width/height/left等)生成render樹

  • III)在render樹的基礎上繼續渲染color,font等屬性

其中如果I)中和II)中的屬性發生變化會發生reflow(回流),如果僅僅III)中的屬性發生改變,只會發生repaint(重繪)。顯然從css的渲染過程我們也可以看出來:reflow(回流)必伴隨著重繪。

reflow(回流):當render樹中的一部分或者全部因為大小邊距等問題發生改變而需要重建的過程叫做回流
repaint(重繪):當元素的一部分屬性發生變化,如外觀背景色不會引起布局變化而需要重新渲染的過程叫做重繪

reflow(回流)會影響瀏覽器css的渲染速度,因此在做網頁性能優化的時候要減少回流的發生。

在第一節,我們通過left屬性,實現了彈幕的效果,left會改變元素的布局,因此會發生reflow(回流),表現在移動端頁面上會造成彈幕動畫的卡頓。

2. css3彈幕性能優化

我們直到了第一節中的彈幕動畫存在卡頓的問題,下面我們看看如何解決動畫的卡頓。

(1)CSS開啟硬件加速

在瀏覽器中用css開啟硬件加速,使用GPU(Graphics Processing Unit)可以提升網頁性能。鑒于此,我們可以發揮GPU的力量,從而使我們的網站或應用表現的更為流暢。

CSS animations, transforms 以及 transitions 不會自動開啟GPU加速,而是由瀏覽器的緩慢的軟件渲染引擎來執行。那我們怎樣才可以切換到GPU模式呢,很多瀏覽器提供了某些觸發的CSS規則。

比較常見的方式是,我們可以通過3d變化(translate3d屬性)來開啟硬件加速,鑒于此,我們修改動畫為:

@keyframes barrage{
   from{
     left:100%;
     transform:translate3d(0,0,0);
   }
   to{
     left:0;
     transform:translate3d(-100%,0,0);
   }
}

這樣就可以通過開啟硬件加速的方式,優化網頁性能。但是這種方式沒有從根本上解決問題,同時使用GPU增加了內存的使用,會減少移動設備的電池壽命等等。

(2)不改變left屬性

第二種方法,就是想辦法在彈幕動畫的前后不改變left屬性的值,這樣就不會發生reflow。

我們想僅僅通過translateX來確定彈幕節點的初始位置,但是translateX(-100%)是相對于彈幕節點本身的,而不是相對于父元素,因此我們耦合js和css,在js中獲取彈幕節點所在的父元素的寬度,接著根據寬度來定義彈幕節點的初始位置。

以父元素為body時為例:

//css
 .block{
  position:absolute;
  left:0;
  visibility:hidden;
  /* other decorate style */
  animation:barrage 5s linear 0s;
}
//js
let style = document.createElement('style');
document.head.appendChild(style);
let width = window.innerWidth;
let from = `from { visibility: visible; -webkit-transform: translateX(${width}px); }`;
let to = `to { visibility: visible; -webkit-transform: translateX(-100%); }`;
style.sheet.insertRule(`@-webkit-keyframes barrage { ${from} ${to} }`, 0);

除了耦合js計算了父元素的寬度,從而確定彈幕節點的初始位置之外,這里在彈幕節點中我們為了防止初始位置就有顯示,增加了visibility:hidden屬性。防止彈幕節點在未確定初始位置時就顯示在父容器內。只有彈幕開始從初始位置滾動,才會變得可見。

但是這種css的實現方式,在實現彈幕的擴展功能方面比較麻煩,比如如何控制彈幕暫停等等。

3. canvas實現彈幕

除了通過css實現彈幕的方法之外,通過canvas也可以實現彈幕。

通過canvas實現彈幕的原理就是時時的重繪文字,下面來一步步的實現。

  • 獲取畫布

    let canvas = document.getElementById('canvas');
      let ctx = canvas.getContext('2d');

  • 繪制文字

ctx.font = '20px Microsoft YaHei';          
ctx.fillStyle = '#000000';                
ctx.fillText('canvas 繪制文字', x, y);

上面的fillText就是實現彈幕效果的主要api,其中x表示橫方向的坐標,y表示縱方向的坐標,只要時時的改變x,y進行重繪,就可以實現動態的彈幕效果。
  • 清除繪制內容

    ctx.clearRect(0, 0, width, height);

  • 具體實現

通過定時器,定時改變x,y,每次改變之前先進性清屏,然后根據改變后的x,y進行重繪。當存在多條彈幕的情況下,定義:

let colorArr=_this.getColor(color);  彈幕數組多對應的顏色數組
let numArrL=_this.getLeft();  彈幕數組所對應的x坐標位置數組
let numArrT=_this.getTop();  彈幕數組所對應的y坐標位置數組
let speedArr=_this.getSpeed(); 彈幕數組所對應的彈幕移動速度數組

定時的重繪彈幕函數為:

_this.timer=setInterval(function(){
    ctx.clearRect(0,0,canvas.width,canvas.height);
    ctx.save();
    for(let j=0;j<barrageList.length;j++){
      numArrL[j]-=speedArr[j];
      ctx.fillStyle = colorArr[j]
      ctx.fillText(barrageList[j],numArrL[j],numArrT[j]);
      ctx.restore();
 },16.7);

實現的效果為:

css3如何實現彈幕效果

4. canva彈幕的擴展功能

通過canvas實現彈幕的方式,很方便做比如暫停彈幕滾動等擴展功能,此外,也可以給彈幕增加頭像,給每條彈幕增加邊框等等功能,以后再補充。

關于css3如何實現彈幕效果就分享到這里了,希望以上內容可以對大家有一定的幫助,可以學到更多知識。如果覺得文章不錯,可以把它分享出去讓更多的人看到。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

临颍县| 凤城市| 翁牛特旗| 怀来县| 天峨县| 辽源市| 镇赉县| 瑞昌市| 武鸣县| 固始县| 无锡市| 灯塔市| 呼图壁县| 南平市| 社会| 邳州市| 谢通门县| 会昌县| 饶平县| 于都县| 綦江县| 大城县| 宿松县| 内江市| 敦煌市| 册亨县| 博湖县| 保德县| 枞阳县| 高邮市| 盘山县| 新河县| 习水县| 阿克陶县| 阿图什市| 迭部县| 长宁区| 海原县| 新蔡县| 六安市| 广安市|