用HTML5的<canvas>元素實現刮刮樂游戲

用HTML5的<canvas>元素實現刮刮樂

用HTML5的<canvas>元素實現刮刮樂,要求:將上面的“圖層”的圖像可用鼠標刮去,露出下面的“圖層”的圖像。

示例從簡單到復雜。

簡單示例

準備兩張圖像,我這里上面的圖像top_image.png,下面的圖像bottom_image.png,如下圖:

?

?

我這里為方便 ,經圖片和源碼文件放在同一個文件夾中。

先看用一個canvas元素實現刮刮樂,源碼如下:

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>刮刮樂(Scratch Card)</title><style>* {margin: 0;padding: 0;box-sizing: border-box;}body {height: 100vh;display: flex;justify-content: center;align-items: center;background-color: #f0f0f0; /* 背景色 */}canvas {background-image: url('bottom_image.png'); /* 底層圖片 */background-size: cover;}</style>
</head>
<body><canvas id="canvas" width="356" height="358"></canvas><script>var canvas = document.getElementById("canvas");var ctx = canvas.getContext("2d");// 加載上層圖片(可被刮去的圖層)var img = new Image();img.src = "top_image.png"; // 上層圖片路徑img.onload = function() {ctx.drawImage(img, 0, 0, canvas.width, canvas.height);};// 標記是否按下鼠標(開始刮卡)var isDown = false;// 鼠標按下事件canvas.addEventListener('mousedown', function() {isDown = true;// 切換到“擦除”模式ctx.globalCompositeOperation = 'destination-out';});// 鼠標松開事件canvas.addEventListener('mouseup', function() {isDown = false;});// 鼠標移動事件canvas.addEventListener('mousemove', function(event) {if (isDown) {let x = event.offsetX;let y = event.offsetY;// 繪制擦除效果ctx.beginPath();ctx.arc(x, y, 20, 0, Math.PI * 2, false); // 使用圓形筆觸ctx.fill();ctx.closePath();}});</script>
</body>
</html>

下面用兩個canvas元素實現刮刮樂,底層圖片和上層圖片各用一個canvas元素,效果和上面的一樣。實現的源碼如下:

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>刮刮樂(Scratch Card)2</title><style>* {margin: 0;padding: 0;box-sizing: border-box;}body {height: 100vh;display: flex;justify-content: center;align-items: center;background-color: #f0f0f0;}#container {position: relative;width: 356px;height: 358px;}canvas {position: absolute;top: 0;left: 0;}</style>
</head>
<body><div id="container"><canvas id="bottomCanvas" width="356" height="358"></canvas> <!-- 底層Canvas --><canvas id="topCanvas" width="356" height="358"></canvas> <!-- 上層Canvas --></div><script>document.addEventListener('DOMContentLoaded', function() {var bottomCanvas = document.getElementById('bottomCanvas');var topCanvas = document.getElementById('topCanvas');var bottomCtx = bottomCanvas.getContext('2d');var topCtx = topCanvas.getContext('2d');// 加載底層圖片var bottomImage = new Image();bottomImage.src = 'bottom_image.png'; // 底層圖片路徑bottomImage.onload = function() {bottomCtx.drawImage(bottomImage, 0, 0, bottomCanvas.width, bottomCanvas.height);};// 加載上層圖片var topImage = new Image();topImage.src = 'top_image.png'; // 上層圖片路徑topImage.onload = function() {topCtx.drawImage(topImage, 0, 0, topCanvas.width, topCanvas.height);};var isDown = false;// 鼠標按下事件topCanvas.addEventListener('mousedown', function() {isDown = true;topCtx.globalCompositeOperation = 'destination-out';});// 鼠標松開事件topCanvas.addEventListener('mouseup', function() {isDown = false;});// 鼠標移動事件topCanvas.addEventListener('mousemove', function(event) {if (!isDown) return;var x = event.offsetX;var y = event.offsetY;// 繪制擦除效果topCtx.beginPath();topCtx.arc(x, y, 20, 0, Math.PI * 2, false); // 使用圓形筆觸topCtx.fill();topCtx.closePath();});});</script>
</body>
</html>

復雜示例

下面是改進,從列表框(下拉框)選擇圖片刮刮樂,增加了游戲的趣味性。

先給出效果

我這里游戲圖片:

源碼如下:

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>刮刮樂(Scratch Card)3</title><style>div{ margin:20px;text-align:center;}* {margin: 0;padding: 0;box-sizing: border-box;}body {height: 100vh;display: flex;justify-content: center;align-items: center;background-color: #f0f0f0;}#container {position: relative;width: 356px;height: 358px;}canvas {position: absolute;top: 0;left: 0;}</style>
</head>
<body><div> 選擇游戲圖片<select id="mySelect" onchange="loadImages()"> <!-- 添加 onchange 事件 --><option value="1">1</option> <!-- 更改 value 以匹配圖像名稱 --><option value="2">2</option><option value="3">3</option></select><div><div id="container"><canvas id="bottomCanvas" width="356" height="358"></canvas> <!-- 底層Canvas --><canvas id="topCanvas" width="356" height="358"></canvas> <!-- 上層Canvas --></div><script>function loadImages() {var selectElement = document.getElementById('mySelect');var selectedValue = selectElement.options[selectElement.selectedIndex].value;var bottomCanvas = document.getElementById('bottomCanvas');var topCanvas = document.getElementById('topCanvas');var bottomCtx = bottomCanvas.getContext('2d');var topCtx = topCanvas.getContext('2d');// 清除畫布bottomCtx.clearRect(0, 0, bottomCanvas.width, bottomCanvas.height);topCtx.clearRect(0, 0, topCanvas.width, topCanvas.height);// 加載底層圖片var bottomImage = new Image();bottomImage.src = 'img/bottom' + selectedValue + '.png';bottomImage.onload = function() {bottomCtx.drawImage(bottomImage, 0, 0, bottomCanvas.width, bottomCanvas.height);};// 重新加載并繪制上層圖片var topImage = new Image();topImage.src = 'img/top' + selectedValue + '.png'; // 確保這里的路徑正確匹配你的圖片路徑和命名topImage.onload = function() {topCtx.globalCompositeOperation = 'source-over'; // 重置合成操作為默認值topCtx.drawImage(topImage, 0, 0, topCanvas.width, topCanvas.height);// 確保刮刮效果重新應用addScratchEffect(topCanvas, topCtx);};}function addScratchEffect(canvas, ctx) {var isDown = false;// 移除之前可能添加的事件監聽器canvas.onmousedown = null;canvas.onmouseup = null;canvas.onmousemove = null;// 鼠標按下事件canvas.onmousedown = function() {isDown = true;ctx.globalCompositeOperation = 'destination-out'; // 設置合成操作以實現刮效果};// 鼠標松開事件canvas.onmouseup = function() {isDown = false;};// 鼠標移動事件canvas.onmousemove = function(event) {if (!isDown) return;var x = event.offsetX;var y = event.offsetY;// 繪制擦除效果ctx.beginPath();ctx.arc(x, y, 20, 0, Math.PI * 2); // 使用圓形筆觸ctx.fill();};}// 頁面加載完畢后初始化畫布document.addEventListener('DOMContentLoaded', function() {loadImages(); // 頁面加載時也加載圖片});</script>
</body>
</html>

本文是對https://blog.csdn.net/cnds123/article/details/112392014 例子的補充

關于HTML5中,使用<select>元素創建一個列表框(下拉框),并使用JavaScript來操作,可參見https://blog.csdn.net/cnds123/article/details/128353007

?

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/712167.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/712167.shtml
英文地址,請注明出處:http://en.pswp.cn/news/712167.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

node express實現Excel文檔轉json文件

有些場景我們需要將Excel文檔中的內容抽取出來生成別的文件&#xff0c;作為一個前端&#xff0c;服務框架最應該熟悉的就是node了&#xff0c;以下是基于多語言轉換實現代碼&#xff0c;看明白原理自己改一改就能用了 1.安裝node環境 2.創建一個文件夾&#xff0c;文件夾中創建…

7、Redis-事務、持久化、內存淘汰機制和過期key處理

目錄 一、事務 二、持久化 三、內存淘汰機制 四、過期key處理 一、事務 Redis的事務本質上就是一個批量執行命令的操作。分為三個步驟&#xff1a; 開始事務&#xff1a;multi命令入隊&#xff1a;正常輸入命令即可執行事務&#xff08;依次執行命令&#xff09;&#xf…

掌握java模板方法模式,提升代碼復用與擴展的藝術

Java 模板方法模式是一種行為型設計模式&#xff0c;它定義了一個算法的骨架&#xff0c;并將一些步驟延遲到子類中實現。模板方法模式使得子類可以在不改變算法結構的情況下重定義算法中的某些步驟。 使用場景 算法骨架固定&#xff1a;如果一個算法的基本結構已經固定&#…

跨專業考研難度大嗎?聽聽過來人的真實經歷

在考研的大潮中&#xff0c;跨專業考研成為了一個不可忽視的現象。許多考生因為對原專業失去興趣、追求職業夢想或其他原因&#xff0c;選擇了跨專業報考。那么&#xff0c;跨專業考研的難度究竟有多大呢&#xff1f;今天&#xff0c;我們就來聊聊這個話題&#xff0c;聽聽過來…

不是我吹,這8道HashMap面試題讓你面試時對答如流

前言 又到了一年一度的金三銀四面試季&#xff0c;我們拿著自己的面試秘籍去面試&#xff0c;但是面試官的問題五花八門&#xff0c;讓我們摸不清他們的套路。今天我就總結了面試時必問的hashmap面試題&#xff0c;無論面試官怎么問&#xff0c;我們都對答如流。 另外本人整理了…

java小記(2)

IS-A&#xff1a;類的父子繼承關系。 default&#xff1a;關鍵字&#xff0c;與Java中的public&#xff0c;private等關鍵字一樣&#xff0c;都屬于修飾符關鍵字&#xff0c;可以用來修飾屬性、方法以及類&#xff0c;但是default一般用來修飾接口中的方法。 接口與抽象類的區…

代碼隨想錄算法訓練營第二十四天 | 77. 組合

回溯算法理論基礎 https://programmercarl.com/%E5%9B%9E%E6%BA%AF%E7%AE%97%E6%B3%95%E7%90%86%E8%AE%BA%E5%9F%BA%E7%A1%80.html 回溯法也可以叫做回溯搜索法&#xff0c;它是一種搜索的方式。 回溯是遞歸的副產品&#xff0c;只要有遞歸就會有回溯。 回溯法并不是什么高效的…

馬斯克正式起訴OpenAI和奧特曼!

就在剛剛&#xff0c;馬斯克鬧出來一件大事——正式起訴OpenAI和Sam Altman&#xff0c;并要求OpenAI 恢復開源GPT-4等模型&#xff01; 眾所周知&#xff0c;馬斯克這兩年一只在推特上指責 OpenAI是CloseAI(不開源)&#xff0c;但都只是停留在口頭上。 而這次馬斯克動了真格。…

nginx if 指令

目錄 nginx if 指令直接判斷變量判斷是否等于字符串判斷變量是否匹配正則表達式文件及目錄判斷示例1&#xff1a;判斷index.html是否存在示例2&#xff1a;判斷URL中是否存在某個參數Parameter示例3&#xff1a;判斷URI中是否為某個特定路徑示例4&#xff1a;開放白名單內的功能…

從0開始python學習-53.python中flask創建簡單接口

目錄 1. 創建一個簡單的請求,沒有寫方法時默認為get 2. 創建一個get請求 3. 創建一個post請求&#xff0c;默認可以使用params和表單傳參 4. 帶有參數的post請求 1. 創建一個簡單的請求,沒有寫方法時默認為get from flask import Flask, request# 初始化一個flask的對象 ap…

RK3566 linux iperf網絡測試

一、開發環境 系統:buildroot&#xff1b; 在Linux目標板和Windows PC上運行iperf進行測試&#xff1b; 二、調試 1、查詢目標板上的iperf 使用終端助手連接目標板&#xff0c;然后輸入命令查詢iperf的版本&#xff1a; rootrk3566-buildroot:~# iperf -v iperf version …

圖數據庫 之 Neo4j - 應用場景3 - 知識圖譜(8)

背景 知識圖譜的復雜性:知識圖譜通常包含大量的實體、關系和屬性,以及它們之間的復雜關聯。傳統的關系型數據庫在處理這種復雜性時可能面臨性能和靈活性的挑戰。 圖數據庫的優勢:圖數據庫是一種專門用于存儲和處理圖結構數據的數據庫。它們使用節點和邊來表示實體和關系,并…

USB - Battery Charing

Getting to the bottom of USB Battery Charging (了解 USB 電池充電的真相) 如今&#xff0c;幾乎所有帶電池的產品都被期望支持 BC1.2 USB 充電標準。 Today, almost every product with a battery is expected to support the BC1.2 standard for USB charging. 這對消費者來…

詳解字符串函數<string.h>(上)

1. strlen函數的使用和模擬實現 size_t strlen(const char* str); 1.1 函數功能以及用法 字符串長度 strlen函數的功能是計算字符串的長度。在使用時&#xff0c;要求用戶傳入需要計算長度的字符串的起始位置&#xff0c;并返回字符串的長度。 #include <stdio.h> #…

基于SSM醫院電子病歷管理系統的設計與實現(源代碼+數據庫腳本+萬字文檔+PPT)

系統介紹 醫院電子病歷管理系統主要是借助計算機&#xff0c;通過對醫院電子病歷管理系統所需的信息管理&#xff0c;增加用戶的選擇&#xff0c;同時也方便對廣大用戶信息的及時查詢、修改以及對用戶信息的及時了解。醫院電子病歷管理系統 對用戶帶來了更多的便利&#xff0c…

Python GUI自動化定位代碼參考

一、pyautogui原始邏輯 import pyautogui # 獲取指定圖片在屏幕上的位置 image_path path/to/image.png target_position pyautogui.locateCenterOnScreen(image_path) if target_position is not None: # 獲取偏移量 offset_x 10 offset_y 10 # 計算實際點…

一文讀懂ZKFair PFP-CyberArmy的參與價值與潛力

3月2日&#xff0c;ZKFair PFP-CyberArmy 將在 Element 上正式開始Public Sale。

文件基礎和文件fd

文章目錄 預備知識C語言的文件接口系統調用文件fd 正文開始前給大家推薦個網站&#xff0c;前些天發現了一個巨牛的 人工智能學習網站&#xff0c; 通俗易懂&#xff0c;風趣幽默&#xff0c;忍不住分享一下給大家。 點擊跳轉到網站。 預備知識 我們平時說文件就是說文件里…

1_Springboot(一)入門

Springboot&#xff08;一&#xff09;——入門 本章重點&#xff1a; 1.什么是Springboot; 2.使用Springboot搭建web項目&#xff1b; 一、Springboot 1.Springboot產生的背景 Servlet->Struts2->Spring->SpringMVC&#xff0c;技術發展過程中&#xff0c;對使…

大模型量化技術原理-SmoothQuant

近年來&#xff0c;隨著Transformer、MOE架構的提出&#xff0c;使得深度學習模型輕松突破上萬億規模參數&#xff0c;從而導致模型變得越來越大&#xff0c;因此&#xff0c;我們需要一些大模型壓縮技術來降低模型部署的成本&#xff0c;并提升模型的推理性能。 模型壓縮主要分…