HTML5 Canvas 超逼真煙花綻放動畫

来源:https://www.cnblogs.com/frontworld/archive/2022/05/04/16220981.html
-Advertisement-
Play Games

js中關於原型和原型鏈有 __proto__ 、prototype、constructor 頻頻出現在面試題中,但是記得多了反而容易記混。 這裡簡單總結下每個屬性的使用場景,方便記憶。 對象和函數都有 __proto__,對象的 __proto__指向構造函數的prototype,構造函數的__pr ...


各位前端朋友們,大家好!五一假期即將結束,在開啟加班模式之前,我要給大家分享一個超酷超逼真的HTML5 Canvas煙花模擬動畫。這次升級版的煙花動畫有以下幾個特點:

  • 煙花綻放時,將展現不同的色彩,不像之前版本的一朵煙花只有一種色彩。
  • 夜空的顏色會自動適配當前綻放的煙花顏色,效果更為逼真。
  • 每一朵煙花綻放時的形狀不再是以前那樣單一,而是會隨機變換不同的花樣,這樣更符合實際的煙花場面。
  • 用戶可設置一些參數,例如開啟聲效、花樣選擇、畫質選擇和全屏設置等等。

先來看一張效果圖吧,非常壯觀!

效果預覽

image

代碼實現

HTML代碼

HTML代碼主要由兩部分組成,其一是設置面板表單,其二是動畫載體canvas元素。

其中設置面板按鈕是通過SVG實現的:

<div style="height: 0; width: 0; position: absolute; visibility: hidden;">
	<svg xmlns="http://www.w3.org/2000/svg">
		<symbol id="icon-play" viewBox="0 0 24 24">
			<path d="M8 5v14l11-7z"/>
		</symbol>
		<symbol id="icon-pause" viewBox="0 0 24 24">
			<path d="M6 19h4V5H6v14zm8-14v14h4V5h-4z"/>
		</symbol>
		<symbol id="icon-close" viewBox="0 0 24 24">
			<path d="M19 6.41L17.59 5 12 10.59 6.41 5 5 6.41 10.59 12 5 17.59 6.41 19 12 13.41 17.59 19 19 17.59 13.41 12z"/>
		</symbol>
		<symbol id="icon-settings" viewBox="0 0 24 24">
			<path d="M19.43 12.98c.04-.32.07-.64.07-.98s-.03-.66-.07-.98l2.11-1.65c.19-.15.24-.42.12-.64l-2-3.46c-.12-.22-.39-.3-.61-.22l-2.49 1c-.52-.4-1.08-.73-1.69-.98l-.38-2.65C14.46 2.18 14.25 2 14 2h-4c-.25 0-.46.18-.49.42l-.38 2.65c-.61.25-1.17.59-1.69.98l-2.49-1c-.23-.09-.49 0-.61.22l-2 3.46c-.13.22-.07.49.12.64l2.11 1.65c-.04.32-.07.65-.07.98s.03.66.07.98l-2.11 1.65c-.19.15-.24.42-.12.64l2 3.46c.12.22.39.3.61.22l2.49-1c.52.4 1.08.73 1.69.98l.38 2.65c.03.24.24.42.49.42h4c.25 0 .46-.18.49-.42l.38-2.65c.61-.25 1.17-.59 1.69-.98l2.49 1c.23.09.49 0 .61-.22l2-3.46c.12-.22.07-.49-.12-.64l-2.11-1.65zM12 15.5c-1.93 0-3.5-1.57-3.5-3.5s1.57-3.5 3.5-3.5 3.5 1.57 3.5 3.5-1.57 3.5-3.5 3.5z"/>
		</symbol>
		<symbol id="icon-sound-on" viewBox="0 0 24 24">
			<path d="M3 9v6h4l5 5V4L7 9H3zm13.5 3c0-1.77-1.02-3.29-2.5-4.03v8.05c1.48-.73 2.5-2.25 2.5-4.02zM14 3.23v2.06c2.89.86 5 3.54 5 6.71s-2.11 5.85-5 6.71v2.06c4.01-.91 7-4.49 7-8.77s-2.99-7.86-7-8.77z"/>
		</symbol>
		<symbol id="icon-sound-off" viewBox="0 0 24 24">
			<path d="M16.5 12c0-1.77-1.02-3.29-2.5-4.03v2.21l2.45 2.45c.03-.2.05-.41.05-.63zm2.5 0c0 .94-.2 1.82-.54 2.64l1.51 1.51C20.63 14.91 21 13.5 21 12c0-4.28-2.99-7.86-7-8.77v2.06c2.89.86 5 3.54 5 6.71zM4.27 3L3 4.27 7.73 9H3v6h4l5 5v-6.73l4.25 4.25c-.67.52-1.42.93-2.25 1.18v2.06c1.38-.31 2.63-.95 3.69-1.81L19.73 21 21 19.73l-9-9L4.27 3zM12 4L9.91 6.09 12 8.18V4z"/>
		</symbol>
	</svg>
</div>

對應界面上3個控制按鈕:

<div class="controls">
			<div class="btn pause-btn">
				<svg fill="white" width="24" height="24"><use href="#icon-pause" xlink:href="#icon-pause"></use></svg>
			</div>
			<div class="btn sound-btn">
				<svg fill="white" width="24" height="24"><use href="#icon-sound-off" xlink:href="#icon-sound-off"></use></svg>
			</div>
			<div class="btn settings-btn">
				<svg fill="white" width="24" height="24"><use href="#icon-settings" xlink:href="#icon-settings"></use></svg>
			</div>
</div>

設置菜單的HTML代碼並沒有什麼特別的地方,無非是由一些checkboxselect下拉框控制項組成,這裡就不展開了。

至於動畫載體畫布,我們直接在頁面上放置canvas元素即可:

<div class="canvas-container">
			<canvas id="trails-canvas"></canvas>
			<canvas id="main-canvas"></canvas>
</div>

CSS代碼主要用來控制設置面板的樣式,我們這裡也不再展開。

JavaScript代碼

頁面載入完成後,煙花動畫應用就開始初始化,主要應用當前的設置,以及根據設置啟動煙花綻放動畫:

function init() {
	// 移除頁面初始化時的載入動畫
	document.querySelector('.loading-init').remove();
	appNodes.stageContainer.classList.remove('remove');
	
	// 填充下拉列表
	function setOptionsForSelect(node, options) {
		node.innerHTML = options.reduce((acc, opt) => acc += `<option value="${opt.value}">${opt.label}</option>`, '');
	}

	// 煙花類型
	let options = '';
	shellNames.forEach(opt => options += `<option value="${opt}">${opt}</option>`);
	appNodes.shellType.innerHTML = options;
	// 煙花大小
	options = '';
	['3"', '4"', '6"', '8"', '12"', '16"'].forEach((opt, i) => options += `<option value="${i}">${opt}</option>`);
	appNodes.shellSize.innerHTML = options;
	
	setOptionsForSelect(appNodes.quality, [
		{ label: 'Low', value: QUALITY_LOW },
		{ label: 'Normal', value: QUALITY_NORMAL },
		{ label: 'High', value: QUALITY_HIGH }
	]);
	
	setOptionsForSelect(appNodes.skyLighting, [
		{ label: 'None', value: SKY_LIGHT_NONE },
		{ label: 'Dim', value: SKY_LIGHT_DIM },
		{ label: 'Normal', value: SKY_LIGHT_NORMAL }
	]);
	
	// 手機上應用0.9
	setOptionsForSelect(
		appNodes.scaleFactor,
		[0.5, 0.62, 0.75, 0.9, 1.0, 1.5, 2.0]
		.map(value => ({ value: value.toFixed(2), label: `${value*100}%` }))
	);
	
	// 開始播放
	togglePause(false);
	
	// 初始化動畫畫布
	renderApp(store.state);
	
	// 應用並更新配置
	configDidUpdate();
}

接下來是煙花動畫實現的核心方法update(frameTime, lag)render(speed)。主要運用了數學方法計算出煙花綻放時粒子的運動軌跡:

const burnRate = Math.pow(star.life / star.fullLife, 0.5);
const burnRateInverse = 1 - burnRate;

star.prevX = star.x;
star.prevY = star.y;
star.x += star.speedX * speed;
star.y += star.speedY * speed;
// Apply air drag if star isn't "heavy". The heavy property is used for the shell comets.
if (!star.heavy) {
	star.speedX *= starDrag;
	star.speedY *= starDrag;
}
else {
	star.speedX *= starDragHeavy;
	star.speedY *= starDragHeavy;
}
star.speedY += gAcc;

if (star.spinRadius) {
	star.spinAngle += star.spinSpeed * speed;
	star.x += Math.sin(star.spinAngle) * star.spinRadius * speed;
	star.y += Math.cos(star.spinAngle) * star.spinRadius * speed;
}

if (star.sparkFreq) {
	star.sparkTimer -= timeStep;
	while (star.sparkTimer < 0) {
		star.sparkTimer += star.sparkFreq * 0.75 + star.sparkFreq * burnRateInverse * 4;
		Spark.add(
			star.x,
			star.y,
			star.sparkColor,
			Math.random() * PI_2,
			Math.random() * star.sparkSpeed * burnRate,
			star.sparkLife * 0.8 + Math.random() * star.sparkLifeVariation * star.sparkLife
		);
	}
}

動畫渲染的代碼如下:

while (BurstFlash.active.length) {
	const bf = BurstFlash.active.pop();
	
	const burstGradient = trailsCtx.createRadialGradient(bf.x, bf.y, 0, bf.x, bf.y, bf.radius);
	burstGradient.addColorStop(0.024, 'rgba(255, 255, 255, 1)');
	burstGradient.addColorStop(0.125, 'rgba(255, 160, 20, 0.2)');
	burstGradient.addColorStop(0.32, 'rgba(255, 140, 20, 0.11)');
	burstGradient.addColorStop(1, 'rgba(255, 120, 20, 0)');
	trailsCtx.fillStyle = burstGradient;
	trailsCtx.fillRect(bf.x - bf.radius, bf.y - bf.radius, bf.radius * 2, bf.radius * 2);
	
	BurstFlash.returnInstance(bf);
}

以上就是這個HTML5煙花動畫的實現過程,文章最後也將全部源碼分享給大家。

源碼下載

完整的代碼我已經整理出了一個源碼包,供大家下載學習。

源碼下載地址:https://mp.weixin.qq.com/s/hfghyk1d-FjTPU5hj_ekZA

代碼僅供參考和學習,請不要用於商業用途。

最後總結

這款HTML5 Canvas煙花動畫應該算是網頁煙花模擬動畫的頂峰了,如果你有更好的想法和創意,也歡迎留言告訴我們。


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

-Advertisement-
Play Games
更多相關文章
  • 查看當前的內核版本: uname -r, 輸出示例: laggage@laplus:~$ uname -r 5.11.0-41-generic 下載鏡像安裝包 wget \ kernel.ubuntu.com/~kernel-ppa/mainline/v4.12/linux-headers-4.12 ...
  • 不管是在生活還是工作中,每個人都會逐漸養成一些小習慣。壞習慣一旦形成就很難改正,所在在系統學習 Linux之前,給大家一些建議,刻意去培養一些好的習慣,對自己是很有利的。 學習Linux,要習慣使用命令行 操作系統必須要有圖形界面,但早期的 Linux 並不完全支持圖形界面,操作起來也沒有 Wind ...
  • 本節旨在介紹對於初學者如何學習 Linux 的建議。如果你已經確定對 Linux 產生了興趣,那麼接下來我們介紹一下學習 Linux 的方法。 如何去學習 學習大多類似庖丁解牛,對事物的認識一般都是由淺入深、由表及裡的過程,循序才能漸進。學習 Linux 同樣要有一定的順序和方法,當然這也是你學習本 ...
  • 一、概述 Flink核心是一個流式的數據流執行引擎,並且能夠基於同一個Flink運行時,提供支持流處理和批處理兩種類型應用。其針對數據流的分散式計算提供了數據分佈,數據通信及容錯機制等功能。基於流執行引擎,Flink提供了跟多高抽象層的API便於用戶編寫分散式任務,下麵稍微介紹一下Flink的幾種A ...
  • Hive Hive將HiveQL(類sql語言)轉為MapReduce,完成數據的查詢與分析,減少了編寫MapReduce的複雜度。它有以下優點: 學習成本低:熟悉sql就能使用 良好的數據分析:底層基於MapReduce實現 同樣存在一些缺點: HiveDL表達能力有限 效率不高 Hive調優比較 ...
  • CentOS7.6搭建RAC 1.系統環境配置 1.1概述 ​ 搭建兩個節點的rac集群,其每個節點均有兩個網卡,public網卡和private網卡。兩個節點的主機名分別為rac1和rac2 1.2 參數設置(RAC1&RAC2) 編輯/etc/hosts文件 vim /etc/hosts 註釋掉 ...
  • 前言 一個開源庫,隨著不斷的迭代優化,難免會遇到一個很痛苦的問題 最初的設計並不是很合理:想添加的很多新功能都受此掣肘 想使得該庫更加的強大和健壯,必須要做一個重構 因為重構涉及到對外暴露的api,所以大家會遇到一個比較煩躁的問題:更新版本後,會大面積報錯 我考慮了很久,到底怎麼幫大家快速遷移呢?最 ...
  • Swift字元串追加 var str = "OC" str.append(" Swfit") print(str) // 輸出結果: OC Swift 輸出結果: Swift獲取字元串長度 let str = String(format: "數字%.2f", 333.333) // 獲取長度 pri ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...