震驚!CSS 也能實現碰撞檢測?

来源:https://www.cnblogs.com/coco1s/archive/2023/08/22/17647903.html
-Advertisement-
Play Games

本文,我們將一起學習,使用純 CSS,實現如下所示的動畫效果: ![](https://img2023.cnblogs.com/blog/608782/202308/608782-20230822102547750-742841232.gif) 上面的動畫效果,非常有意思,核心有兩點: 1. 小球隨 ...


本文,我們將一起學習,使用純 CSS,實現如下所示的動畫效果:

上面的動畫效果,非常有意思,核心有兩點:

  1. 小球隨機做 X、Y 方向的直線運動,並且能夠實現碰撞到邊界的時候,實現反彈效果
  2. 小球在碰撞邊界的瞬間,顏色發生隨機的變化

嗯?很有意思的效果。看上去,我們好像使用 CSS 實現了碰撞檢測

然而,實際情況真的是這樣嗎?讓我們一起一探究竟!

實現 X 軸方向的運動

這裡其實我們並沒有實現碰撞檢測,因為小球和小球之間接觸時,並沒有發生碰撞效果。

我們只實現了,小球與邊界之間的碰撞反應。不過這裡,也並非碰撞檢測,我們只需要設置好單個方向的運動動畫,並且設置 animation-direction: alternate; 即可!

下麵,我們一起來實現單個方向上的運動動畫:

<div></div>
div {
    position: absolute;
    top: 0;
    left: 0;
    width: 100px;
    height: 100px;
    border-radius: 50%;
    background: #0cf;
    animation: horizontal 3s infinite linear alternate;
}

@keyframes horizontal {
    from { 
        left: 0;
    }
    to { 
        left: calc(100vw - 100px);
    }
}

簡單解讀一下:

  1. 元素設置為 position: absolute 絕對定位,利用 left 進行 X 軸方向的運動
  2. 我們讓元素 div 運動的距離為 left: calc(100vw - 100px),元素本身的高寬都是 100px,因此相當於運動到屏幕的最右側
  3. 動畫設置了 alternate 也就是 animation-direction: alternate; 的簡寫,表示動畫在每個迴圈中正反交替播放

這樣,我們就巧妙的實現了,在視覺上,小球元素移動到最右側邊界時,回彈的效果:

如法炮製 Y 軸方向的運動

好,有了上面的鋪墊,我們只需要再如法炮製 Y 軸方向的運動即可。

利用元素的 top 進行 Y 軸方向的運動:

div {
    position: absolute;
    top: 0;
    left: 0;
    width: 100px;
    height: 100px;
    border-radius: 50%;
    background: #0cf;
    animation: 
        horizontal 3s infinite linear alternate,
        vertical 3s infinite  linear alternate;
}

@keyframes horizontal {
    from { 
        left: 0;
    }
    to { 
        left: calc(100vw - 100px);
    }
}

@keyframes vertical {
    from { 
        top: 0;
    }
    to { 
        top: calc(100vh - 100px);
    }
}

我們增加了一個 vertical 3s infinite linear alternate Y 軸的運動動畫,實現小球從 top: 0top: calc(100vh - 100px); 的運動。

這樣,我們就成功的得到了 X、Y 兩個方向上的小球運動,它們疊加在一起的效果如下:

顏色的變化可以忽略,GIF 錄製問題。

當然,此時的問題在於,缺少了隨機性,小球的始終在左上和右下角之間來回運動。

為瞭解決這個問題,我們需要添加一定的隨機性,這個問題也要解決,我們只需要讓兩個方向上運動時間不一致即可。

我們修改一下代碼,讓 X、Y 軸的運動時長不一致即可:

div {
    position: absolute;
    // ...
    animation: 
        horizontal 2.6s infinite linear alternate,
        vertical 1.9s infinite  linear alternate;
}

如此一來,整體的效果就好上了不少,由於整個動畫是無限反覆進行的,隨著時間的推進,整個動畫呈現出來的就是無序、隨機的運動

使用 transform 替代 top、left

當然,上面的效果基本上沒有什麼太大的問題了,但是代碼層面不夠優雅,主要有兩點問題:

  1. 元素移動使用的是 topleft,性能相對較差,需要使用 transform 進行替代
  2. 代碼中 hardcode 了 100px,由於 DEMO 中小球的大小是 100px x 100px,並且在動畫的代碼中也使用了 100px 這個值進行了運動終態的計算,因此如果想修改小球的元素大小,需要改動地方較多

上述兩個問題,使用 transform: translate() 都可以解決,但是我們為什麼一開始不用 transform 呢?

我們來嘗試一下,使用 transform 替代 top、left:

div {
    position: absolute;
    top: 0;
    left: 0;
    width: 100px;
    height: 100px;
    border-radius: 50%;
    background: #0cf;
    animation: 
        horizontal 2.6s infinite linear alternate,
        vertical 1.9s infinite  linear alternate;
}
@keyframes horizontal {
    from { transform: translateX(0); }
    to { transform: translateX(calc(100vw - 100%)); }
}
@keyframes vertical {
    from { transform: translateY(0); }
    to { transform: translateY(calc(100vh - 100%)); }
}

上述代碼中,我們使用了 transform 替代 top、left 運動。並且,將動畫代碼中的 100px 替換成了 100%,這一點的好處是,在 transform: translate 中,100% 表示的是元素本身的高寬,這樣,當我們改變元素本身的大小時,就無需再改變 @keyframes 中的代碼,通用性更強。

我們來看看修改後的效果:

有點問題!預想中的效果並沒有出現,整個動畫只有 Y 軸方向上的動畫效果。

這是什麼原因呢?

其本質在於,定義的 vertical 1.9s infinite linear alternate 的垂直方向的動畫效果覆蓋了在其之前定義的 transform: translateX(calc(100vw - 100%)) 動畫效果。

說人話就是 X、Y 軸的動畫都使用了 transform 屬性,兩者之間造成了衝突

使用 animation-composition 進行動畫合成

在之前,這種情況基本是無解的,常見的解決方案就是:

  1. 解法一:使用 topleft 替代 transform
  2. 解法二:多一層嵌套,將一個方向的動畫拆解到元素的父元素上

不過,到今天,這個問題有了更好的解法!也就是 CSS animation 家族中的新屬性 —— animation-composition

這是一個非常新的屬性,表示動畫合成屬性,從 Chrome 112 版本開始支持。

有三種不同的取值:

{
    animation-composition: replace;        // 表示動畫值替換
    animation-composition: add;              // 表示動畫值追加
    animation-composition: accumulate; // 表示動畫值累加
}

本文不會詳細介紹 animation-composition,感興趣的可以看看 MDN 的屬性介紹或者 XBOXYAN 大佬的這篇文章 -- 瞭解一下全新的CSS動畫合成屬性animation-composition

這裡,基於上面的代碼,我們只需要再多設置一個 animation-composition: accumulate 即可解決問題:

div {
    animation: 
        horizontal 2.6s infinite linear alternate,
        vertical 1.9s infinite  linear alternate;
    animation-composition: accumulate;
}

此時,我們就能通過一個元素,利用 transform 得到 X、Y 兩個方向位移動畫的合成效果,也就是我們想要的效果:

使用 steps 實現顏色切換

解決了位移動畫的問題,我們就只剩下最後一個問題了,如何在碰撞的瞬間,實現顏色的切換?

這裡也非常好解決,由於我們是知道每一輪 X、Y 方向上的動畫時長的,那我們只需要在每次這個結點上,切換一次顏色即可。

並且,由於顏色不是過渡變換,而是直接的跳變,所以,我們需要用到 animation 中的 animation-timing-function: steps(),也就是步驟緩動函數。

animation-timing-function: steps() 還不太瞭解的,可能需要先補一補基礎,可以看看這一篇文章:深入淺出 CSS 動畫

舉個例子,假設 X 方向上,單次的動畫時長為 3s,那我們可以設置一個 steps(10) 的顏色動畫,總時長為 30s,這樣,每隔 3s 就會觸發一次 steps() 步驟動畫,顏色的變化就能夠和小球與邊界的碰撞動畫發生在同一時刻。

那如何快速實現顏色的變化呢?利用 filter: hue-rotate() 即可快速實現顏色的變化。

理解一下下麵的代碼:

<div class="normal"></div>
<div class="steps"></div>
div {
    width: 200px;
    height: 200px;
    background: #fc0;
}
.normal {
    animation: colorChange 10s linear infinite;
}
.steps {
    animation: colorChange 10s steps(5) infinite;
}
@keyframes colorChange {
    100% {
        filter: hue-rotate(360deg);
    }
}

這裡,我們用 filter: hue-rotate(360deg) 的改變,實現顏色的變化,觀察下麵的動圖,理解 steps(5) 的作用。

  1. animation: colorChange 10s linear infinite 表示背景動畫的過渡變化
  2. animation: colorChange 10s steps(5) infinite,這裡表示 10s 的動畫分成 5 步,每兩秒,會觸發一次動畫:

效果如下:

理解了這一步,我們就可以把顏色的變化,也一起疊加到上述的小球變化中:

div {
    animation: 
        horizontal 2.6s infinite linear alternate,
        vertical 2s infinite  linear alternate,
        colorX 26s infinite steps(10),
        colorY 14s infinite steps(7);
    animation-composition: accumulate;
}

@keyframes horizontal {
    from { transform: translateX(0); }
    to { transform: translateX(calc(100vw - 100%)); }
}
@keyframes vertical {
    from { transform: translateY(0); }
    to { transform: translateY(calc(100vh - 100%)); }
}
@keyframes colorX {
    to {
        filter: hue-rotate(360deg);
    }
}
@keyframes colorY {
    to {
        filter: hue-rotate(360deg);
    }
}

這樣,我們就成功的得到了題圖中的效果:

完整的代碼,你可以戳這裡:Random Circle Path

應用於圖片效果、應用與多粒子效果

OK,上面,我們就把整個效果的完整原理剖析了一遍。

掌握了整個原理之後,我們就可以把這個效果應用於不同場景中。

譬如,假設我們有這麼一張圖片:

基於上面的效果,稍加改造,我們就可以得到類似的如下效果:

<div></div>
div {
    width: 220px;
    height: 97px;
    background: linear-gradient(#f00, #f00), url(https://s1.ax1x.com/2023/08/15/pPQm9oT.jpg);
    background-blend-mode: lighten;
    background-size: contain; 
    animation: horizontal 3.7s infinite -1.4s linear alternate,
            vertical 4.1s infinite -2.1s linear alternate,
            colorX 37s infinite -1.4s steps(10),
            colorY 28.7s infinite -2.1s steps(7);
    animation-composition: accumulate;
}
@keyframes horizontal {
    from { transform: translateX(0); }
    to { transform: translateX(calc(100vw - 100%)); }
}
@keyframes vertical {
    from { transform: translateY(0); }
    to { transform: translateY(calc(100vh - 100%)); }
}
@keyframes colorX {
    to {
        filter: hue-rotate(2185deg);
    }
}
@keyframes colorY {
    to {
        filter: hue-rotate(1769deg);
    }
}

效果如下:

上面的 DEMO 是基於元素背景色的,本 DEMO 是基於圖片的,因此這裡多了一步,利用 mix-blend-mode,實現了圖片顏色的變化。

完整的代碼,你可以戳這裡:CodePen Demo -- Random DVD Path

實現多粒子碰撞

OK,我們再進一步,基於上面的效果,我們可以實現各種有趣的粒子效果,如果同時讓頁面存在 1000 個粒子呢?

下麵是我使用 CSS-Doodle 實現的純 CSS 的粒子效果,其核心原理與上面的保持一致,只是添加了更多的隨機性:

Amazing!是不是非常有趣,整個效果的代碼基於 CSS-doodle 的語法,不超過 40 行。完整的代碼,你可以戳這裡:CSS Doodle - CSS Particles Animation

最後

總結一下,本文介紹瞭如何巧妙的利用 CSS 中的各種高階技巧,組合實現類似於碰撞場景的動畫效果。創建出了非常有趣的 CSS 動畫,期間各種技巧的組合運用,值得好好琢磨學習。

OK,本文到此結束,希望本文對你有所幫助

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 當使用了多個資料庫來提供服務時,最為關鍵的點是如何讓每一個資料庫比較均勻的承擔壓力,而不至於其中的某些資料庫壓力過大,某些資料庫沒什麼壓力。這其中的關鍵點之一就是拆分鍵的設計 ...
  • 近日,第14屆中國資料庫技術大會(DTCC2023)在北京國際會議中心順利舉行。大會以“數智賦能 共築未來”為主題,邀請了上百位行業專家,一起探討新時代下各類型資料庫的最新動態和應用實踐,帶來一場資料庫領域的年度盛宴。在上午的主會場,華為雲資料庫服務產品部總經理蘇光牛圍繞“打造最可信資料庫,華為雲G ...
  • MySQL服務端配置對使用方來說是不可更改的,需要聯繫DBA進行操作。這些配置操作對我們來說是一個黑盒,但是瞭解核心配置可以幫助我們快速定位資料庫問題原因。 ...
  • 這裡給大家分享我在網上總結出來的一些知識,希望對大家有所幫助 在前段時間的一次面試中,被問到了一個如標題這樣的問題。要想好好地去回答這個問題,這裡牽扯到的知識點也是比較多的。 那麼接下來這篇文章我們就一點一點開始引出這個問題。 同源策略 在瀏覽器中,內容是很開放的,任何資源都可以接入其中,如 Jav ...
  • ##### 11 CSS盒子模型(重點) 盒模型是CSS的核心知識點之一,它指定元素如何顯示以及如何相互交互。HTML頁面上的每個元素都可以看成一個個方盒子,這些盒子由元素的content(內容)、padding(內邊距)、border(邊框)、margin(外邊距)組成。 ![image](htt ...
  • ##### 10 CSS邊框屬性 1. border-style(邊框風格) 定義邊框的風格,值可以有: ``` /* none:沒有邊框,當border的值為none的時候,系統將會忽略[border-color] hidden:隱藏邊框,低版本瀏覽器不支持。 dotted:點狀邊框。 dashe ...
  • ##### 15 JavaScript ES6中的箭頭函數 1. 什麼是箭頭函數 ES6中允許使用=>來定義函數。箭頭函數相當於匿名函數,並簡化了函數定義。 2. 基本語法 ```js // 箭頭函數 let fn = (name) => { // 函數體 return `Hello ${name} ...
  • 13 JavaScript關於prototype(超重點) prototype是js裡面給類增加功能擴展的一種模式. 寫個面向對象來看看. ```js function People(name, age){ this.name = name; this.age = age; this.run = f ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...