❤️创意网页:抖音汉字鬼抓人小游戏复刻——附带外挂(“鬼鬼定身术”和“鬼鬼消失术”)坚持60秒轻轻松松(●‘◡‘●)

博主:命运之光 

🌸专栏:Python星辰秘典

🐳专栏:web开发(简单好用又好看)

❤️专栏:Java经典程序设计

☀️博主的其他文章:点击进入博主的主页

前言:欢迎踏入我的Web项目专栏,一段神奇而令人陶醉的数字世界!

🌌在这里,我将带您穿越时空,揭开属于Web的奥秘。通过HTML、CSS和JavaScript的魔力,我创造了一系列令人惊叹的Web项目,它们仿佛是从梦境中涌现而出。

🌌在这个专栏中,您将遇到华丽的界面,如流星划过夜空般迷人;您将感受到动态的交互,如魔法般让您沉浸其中;您将探索响应式设计的玄妙,让您的屏幕变幻出不同的绚丽景象。

🌌无论您是一个探险家还是一位嗜血的代码巫师,这个专栏将成为您的魔法书。我将分享每个项目的秘密,解开编码的谜题,让您也能够拥有制作奇迹的力量。

🌌准备好了吗?拿起您的键盘,跟随我的指引,一起进入这个神秘而充满惊喜的数字王国。在这里,您将找到灵感的源泉,为自己创造出一段奇幻的Web之旅!

目录

简介

动态图展示

静态图展示

视频展示

开始编写追逐游戏

初始化游戏设置和变量

绘制角色和敌人

绘制游戏边界

生成敌人鬼

更新游戏状态和碰撞检测

实时计时器

键盘控制角色移动

开始游戏和生成敌人

项目完整代码 

代码的使用方法(超简单什么都不用下载)

🍓1.打开记事本 

🍓2.将上面的源代码复制粘贴到记事本里面将文件另存为HTML文件点击保存即可

🍓3.打开html文件(大功告成(●'◡'●))

结语


简介

本篇博客将介绍如何使用HTML5的Canvas元素和JavaScript编写一个简单的追逐游戏。在这个游戏中,玩家可以通过键盘控制一个角色“我”,并且需要躲避不断增加并追逐“我”的敌人“鬼”。同时,我们还会加入实时计时功能,记录玩家坚持游戏的时间,并在游戏结束时显示游戏时长。最后,我们会为游戏添加一个漂亮的背景图。


动态图展示


静态图展示


视频展示

抖音汉字鬼抓人小游戏


开始编写追逐游戏

首先,我们创建一个基本的HTML框架,包含一个Canvas元素用于绘制游戏界面,以及一个JavaScript脚本来实现游戏逻辑。以下是完整的HTML代码:

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>追逐游戏</title><style>body {display: flex;justify-content: center;align-items: center;height: 100vh;margin: 0;background-image: url('background.jpg'); /* 替换成背景图的URL */background-size: cover;background-position: center;background-repeat: no-repeat;}canvas {border: none;}</style>
</head>
<body><canvas id="gameCanvas" width="600" height="600"></canvas><script>// JavaScript游戏逻辑将在这里编写</script>
</body>
</html>

初始化游戏设置和变量

在JavaScript脚本中,我们首先定义一些游戏所需的设置和变量。这些设置包括游戏画布的大小、角色的移动速度以及游戏是否结束的标志等。同时,我们还需要定义一个玩家角色对象和敌人鬼对象数组,用于存储玩家和敌人的位置和状态。

const canvas = document.getElementById('gameCanvas');
const ctx = canvas.getContext('2d');const boxSize = 30; // 角色和格子的大小
const canvasSize = 600; // 画布大小// 玩家对象
const player = {x: canvasSize / 2, // 初始位置居中y: canvasSize / 2,speed: 3, // 移动速度dx: 0, // 横向移动方向dy: 0, // 纵向移动方向
};const ghosts = []; // 敌人鬼的数组let gameOver = false; // 游戏结束标志
let startTime = 0; // 游戏开始时间

绘制角色和敌人

接下来,我们需要编写绘制角色和敌人的函数。在Canvas中,我们使用ctx.fillText()来绘制文本,用于显示角色和敌人的图形。

function drawPlayer() {ctx.font = '24px Arial';ctx.fillStyle = 'blue';ctx.fillText('我', player.x, player.y);
}function drawGhost(x, y) {ctx.font = '24px Arial';ctx.fillStyle = 'red';ctx.fillText('鬼', x, y);
}

绘制游戏边界

为了限制角色和敌人的移动范围,我们需要绘制游戏边界。这里我们使用汉字“墙”来代表游戏的边界。

function drawWall(x, y) {ctx.font = '24px Arial';ctx.fillStyle = 'black';ctx.fillText('墙', x, y + boxSize - 6); // 调整绘制位置
}

生成敌人鬼

我们还需要一个函数来生成敌人鬼。我们随机生成敌人的位置,并将其加入敌人数组中。

function generateGhost() {ghosts.push({x: Math.floor(Math.random() * (canvasSize / boxSize)) * boxSize,y: Math.floor(Math.random() * (canvasSize / boxSize)) * boxSize,});
}

更新游戏状态和碰撞检测

在游戏的主循环中,我们需要更新角色和敌人的状态,检测碰撞,并处理游戏结束的情况。

function draw() {if (gameOver) {return;}ctx.clearRect(0, 0, canvasSize, canvasSize);// 绘制游戏边界for (let x = 0; x < canvasSize; x += boxSize) {drawWall(x, 0);drawWall(x, canvasSize - boxSize);}for (let y = boxSize; y < canvasSize - boxSize; y += boxSize) {drawWall(0, y);drawWall(canvasSize - boxSize, y);}drawPlayer();// 更新角色位置player.x += player.dx * player.speed;player.y += player.dy * player.speed;// 角色的位置循环在画布上if (player.x < boxSize) player.x = canvasSize - boxSize * 2;if (player.x >= canvasSize - boxSize) player.x = boxSize;if (player.y < boxSize) player.y = canvasSize - boxSize * 2;if (player.y >= canvasSize - boxSize) player.y = boxSize;// 绘制敌人鬼for (const ghost of ghosts) {drawGhost(ghost.x, ghost.y);// 敌人鬼向角色移动if (ghost.x < player.x) {ghost.x += 1;} else if (ghost.x > player.x) {ghost.x -= 1;}if (ghost.y < player.y) {ghost.y += 1;} else if (ghost.y > player.y) {ghost.y -= 1;}// 碰撞检测if (player.x < ghost.x + boxSize &&player.x + boxSize > ghost.x &&player.y < ghost.y + boxSize &&player.y + boxSize > ghost.y) {// 游戏结束gameOver = true;const currentTime = Math.floor((Date.now() - startTime) / 1000);ctx.font = '24px Arial';ctx.fillStyle = 'red';ctx.fillText(`游戏结束!已经坚持${currentTime}秒`, 150, canvasSize / 2);return;}}// 更新计时器updateTimer();requestAnimationFrame(draw);
}

实时计时器

我们还需要一个函数来更新实时计时器,显示玩家坚持游戏的时间。

function updateTimer() {const currentTime = Math.floor((Date.now() - startTime) / 1000);ctx.font = '18px Arial';ctx.fillStyle = 'green';ctx.fillText(`已经坚持 ${currentTime} 秒`, 30, 50);
}

键盘控制角色移动

最后,我们需要监听键盘按键事件,控制角色的移动方向。

document.addEventListener('keydown', (event) => {if (event.key === 'ArrowUp') {player.dx = 0;player.dy = -1;} else if (event.key === 'ArrowDown') {player.dx = 0;player.dy = 1;} else if (event.key === 'ArrowLeft') {player.dx = -1;player.dy = 0;} else if (event.key === 'ArrowRight') {player.dx = 1;player.dy = 0;}
});

开始游戏和生成敌人

最后,在游戏开始时,我们需要设置游戏开始时间,并在一定时间间隔内生成敌人。

// 生成一个新的敌人鬼,间隔为1秒
setInterval(generateGhost, 1000);// 开始游戏计时
startTime = Date.now();// 运行游戏循环
draw();

项目完整代码 

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>追逐游戏</title><style>body {display: flex;justify-content: center;align-items: center;height: 100vh;margin: 0;background-image: url('background.jpg'); /* 替换成背景图的URL */background-size: cover;background-position: center;background-repeat: no-repeat;}canvas {border: none;}.button-container {position: absolute;top: 20px;right: 20px;display: flex;flex-direction: column;gap: 10px; /* 两个按钮之间的距离 */}.game-button {padding: 10px 20px;font-size: 16px;background-color: #f0f0f0;border: 2px solid #888;cursor: pointer;transition: background-color 0.3s, border-color 0.3s;}.game-button:hover {background-color: #ddd;border-color: #666;}</style>
</head>
<body><canvas id="gameCanvas" width="600" height="600"></canvas><div class="button-container"><button id="invincibleButton" class="game-button">急急如意令,是鬼就不要动</button><button id="clearGhostsButton" class="game-button">妈咪妈咪哄,鬼鬼消失术</button></div><script>const canvas = document.getElementById('gameCanvas');const ctx = canvas.getContext('2d');const boxSize = 30;const canvasSize = 600;const player = {x: canvasSize / 2,y: canvasSize / 2,speed: 3,dx: 0,dy: 0,};const ghosts = [];let gameOver = false;let startTime = 0;let invincibleMode = false;function drawPlayer() {ctx.font = '24px Arial';ctx.fillStyle = 'white';ctx.fillText('我', player.x, player.y);}function drawGhost(x, y) {ctx.font = '24px Arial';ctx.fillStyle = 'red';ctx.fillText('鬼', x, y);}function drawWall(x, y) {ctx.font = '24px Arial';ctx.fillStyle = 'green';ctx.fillText('墙', x, y + boxSize - 6); // 调整绘制位置}function generateGhost() {ghosts.push({x: Math.floor(Math.random() * (canvasSize / boxSize)) * boxSize,y: Math.floor(Math.random() * (canvasSize / boxSize)) * boxSize,});}function draw() {if (gameOver) {return;}ctx.clearRect(0, 0, canvasSize, canvasSize);// 绘制游戏边界for (let x = 0; x < canvasSize; x += boxSize) {drawWall(x, 0);drawWall(x, canvasSize - boxSize);}for (let y = boxSize; y < canvasSize - boxSize; y += boxSize) {drawWall(0, y);drawWall(canvasSize - boxSize, y);}drawPlayer();// 更新角色位置player.x += player.dx * player.speed;player.y += player.dy * player.speed;// 角色的位置循环在画布上if (player.x < boxSize) player.x = canvasSize - boxSize * 2;if (player.x >= canvasSize - boxSize) player.x = boxSize;if (player.y < boxSize) player.y = canvasSize - boxSize * 2;if (player.y >= canvasSize - boxSize) player.y = boxSize;// 绘制敌人鬼for (const ghost of ghosts) {if (ghosts.length > 0) {drawGhost(ghost.x, ghost.y);// 敌人鬼向角色移动if (!invincibleMode) {if (ghost.x < player.x) {ghost.x += 1;} else if (ghost.x > player.x) {ghost.x -= 1;}if (ghost.y < player.y) {ghost.y += 1;} else if (ghost.y > player.y) {ghost.y -= 1;}}// 碰撞检测if (player.x < ghost.x + boxSize &&player.x + boxSize > ghost.x &&player.y < ghost.y + boxSize &&player.y + boxSize > ghost.y) {// 游戏结束gameOver = true;const currentTime = Math.floor((Date.now() - startTime) / 1000);ctx.font = '24px Arial';ctx.fillStyle = 'crimson';ctx.fillText(`游戏结束!已经坚持${currentTime}秒`, 150, canvasSize / 2);return;}}}// 更新计时器updateTimer();requestAnimationFrame(draw);}function updateTimer() {const currentTime = Math.floor((Date.now() - startTime) / 1000);ctx.font = '18px Arial';ctx.fillStyle = 'white';ctx.fillText(`已经坚持 ${currentTime} 秒`, 30, 50);}// 监听按钮点击事件const invincibleButton = document.getElementById('invincibleButton');invincibleButton.addEventListener('click', () => {invincibleMode = true;});const clearGhostsButton = document.getElementById('clearGhostsButton');clearGhostsButton.addEventListener('click', () => {ghosts.length = 0; // 移除所有的鬼});// 监听键盘控制角色移动document.addEventListener('keydown', (event) => {if (event.key === 'ArrowUp') {player.dx = 0;player.dy = -1;} else if (event.key === 'ArrowDown') {player.dx = 0;player.dy = 1;} else if (event.key === 'ArrowLeft') {player.dx = -1;player.dy = 0;} else if (event.key === 'ArrowRight') {player.dx = 1;player.dy = 0;}});// 生成一个新的敌人鬼,间隔为1秒setInterval(generateGhost, 1000);// 开始游戏计时startTime = Date.now();// 运行游戏循环draw();</script>
</body>
</html>

代码的使用方法(超简单什么都不用下载)

🍓1.打开记事本 

🍓2.将上面的源代码复制粘贴到记事本里面将文件另存为HTML文件点击保存即可

🍓3.打开html文件(大功告成(●'◡'●))


结语

本章的内容就到这里了,觉得对你有帮助的话就支持一下博主把~

🌌点击下方个人名片,交流会更方便哦~
↓ ↓ ↓ ↓ ↓ ↓ ↓ ↓ ↓ ↓ ↓ ↓ ↓ ↓ ↓

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/6302.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

OpenCv之图像形态学

目录 一、形态学 二、图像全局二值化 三、自适应阈值二值化 四、腐蚀操作 五、获取形态学卷积核 六、膨胀操作 七、开运算 八、闭运算 一、形态学 定义: 指一系列处理图像形状特征的图像处理技术形态学的基本思想是利用一种特殊的结构元(本质上就是卷积核)来测量或提取输…

Flink简介及部署模式

文章目录 1、Flink简介2、Flink部署2.1 本地模式2.1 Standalone模式部署2.2 Standalone模式下的高可用2.3 Yarn模式Yarn模式的高可用配置&#xff1a;yarn模式中三种子模式的区别&#xff1a; 3、并行度4、提交命令执行指定任务Application Mode VS yarn per-job 5、注意事项5、…

4.1 Bootstrap UI 编辑器

文章目录 1. Bootstrap Magic2. BootSwatchr3. Bootstrap Live Editor4. Fancy Boot5. Style Bootstrap6. Lavish7. Bootstrap ThemeRoller8. LayoutIt!9. Pingendo10. Kickstrap11. Bootply12. X-editable13. Jetstrap14. DivShot15. PaintStrap 以下是 15 款最好的 Bootstrap…

ffplay播放器剖析(5)----视频输出剖析

文章目录 1.视频输出模块1.1 视频输出初始化1.1.1 视频输出初始化主要流程1.1.2 calculate_display_rect初始化显示窗口大小 1.2 视频输出逻辑1.2.1 event_loop开始处理SDL事件1.2.2 video_refresh1.2.2.1 计算上一帧显示时长,判断是否还要继续上一帧1.2.2.2 估算当前帧显示时长…

笔记本电脑的电池健康:确保长时间使用和优异性能的关键

笔记本电脑已经成为我们日常生活中不可或缺的工具&#xff0c;无论是办公、学习还是娱乐&#xff0c;我们都依赖着它的便携性和高效性能。而在所有的硬件组件中&#xff0c;电池健康被认为是确保长时间使用和良好性能的关键因素之一。一块健康的电池不仅能提供持久的续航时间&a…

CHI协议保序之Compack保序

一致性系统中&#xff0c;使用三种保序方式&#xff1b; Completion ack response ⭕Completion acknowledgment&#xff1a; □ 该域段主要是用来&#xff0c; □ 决定 RN 发送的 trans&#xff0c;与其他 RN 发送的命令产生的 SNP 之间的顺序&#xff1b; …

MySQL存储过程——系统变量

1.存储过程中的变量 1.1 查看系统变量 查看所有的系统变量 show variables;查看会话级别的系统变量 show session variables&#xff1b;查看会话和auto相关的变量 show session variables like auto%;查看全局的和auto相关变量 show global variables like auto%;查看某一…

R语言贝叶斯METROPOLIS-HASTINGS GIBBS 吉布斯采样器估计变点指数分布分析泊松过程车站等待时间...

原文链接&#xff1a;http://tecdat.cn/?p26578 指数分布是泊松过程中事件之间时间的概率分布&#xff0c;因此它用于预测到下一个事件的等待时间&#xff0c;例如&#xff0c;您需要在公共汽车站等待的时间&#xff0c;直到下一班车到了&#xff08;点击文末“阅读原文”获取…

ext4 - delay allocation数据结构

概述 延迟分配delay allocation是ext4非常重要的特性&#xff0c;启用该特性write系统将用户空间buffer写入内存page cache中即返回&#xff0c;此时也不会真正进行磁盘block分配&#xff0c;而是延迟到磁盘回写时&#xff08;比如dirty ratio达到一定值&#xff0c;定时刷新&…

【华为c# OD机考参考答案】01---IPv4地址转换成整数

题目 1、题目 01---IPv4地址转换成整数2、解图思路 1、IP地址转为二进制 2、二进制转十进制 3、注意事项 1、IP地址的范围判断 2、空字符串判断 3、非法字符判断 4、考点 1、string的split 、convert等相关用法 2、正则表达式 3、进制转换 4、理解32位整数的意思 5、代码 判…

【NOSQL】MongoDB

MongoDB MongoDB简介体系结构Linux系统中的安装启动和连接&#xff08;1&#xff09;先到官网下载压缩包——>解压——>重命名新建几个目录&#xff0c;分别用来存储数据和日志&#xff1a;新建并修改配置文件官网下载MongoDB Compass MongoDB简介 MongoDB是一个开源、高…

C# List 详解二

目录 5.Clear() 6.Contains(T) 7.ConvertAll(Converter) ,toutput> 8.CopyTo(Int32, T[], Int32, Int32) 9.CopyTo(T[]) 10.CopyTo(T[], Int32) C# List 详解一 1.Add(T)&#xff0c;2.AddRange(IEnumerable)&#xff0c;3.AsReadOnly()&…

Matlab的GUI设计

文章目录 AppDesigner各个版本的特点mlapp文件基本格式AppDesigner的回调函数常见控件的属性MVC模式MVC模式设计GUIMVC简单使用 其他让app designer置顶将Guide的GUI导出为m文件将app编译为exe将app中的多个控件组合在一起 AppDesigner 20200328 各个版本的特点 在2017b版本中…

【JavaEE】Spring中注解的方式去获取Bean对象

【JavaEE】Spring的开发要点总结&#xff08;3&#xff09; 文章目录 【JavaEE】Spring的开发要点总结&#xff08;3&#xff09;1. 属性注入1.1 Autowired注解1.2 依赖查找 VS 依赖注入1.3 配合Qualifier 筛选Bean对象1.4 属性注入的优缺点 2. Setter注入2.1 Autowired注解2.2…

21matlab数据分析牛顿插值(matlab程序)

1.简述 一、牛顿插值法原理 1.牛顿插值多项式   定义牛顿插值多项式为&#xff1a; N n ( x ) a 0 a 1 ( x − x 0 ) a 2 ( x − x 0 ) ( x − x 1 ) ⋯ a n ( x − x 0 ) ( x − x 1 ) ⋯ ( x − x n − 1 ) N_n\left(x\right)a_0a_1\left(x-x_0\right)a_2\left(x-x_0\…

SpringBoot 如何使用 EmbeddedDatabaseBuilder 进行数据库集成测试

SpringBoot 如何使用 EmbeddedDatabaseBuilder 进行数据库集成测试 在开发 SpringBoot 应用程序时&#xff0c;我们通常需要与数据库进行交互。为了确保我们的应用程序在生产环境中可以正常工作&#xff0c;我们需要进行数据库集成测试&#xff0c;以测试我们的应用程序是否能…

剑指offer61.扑克牌中的顺子

我的想法非常简单&#xff0c;就是先给数组排序&#xff0c;然后统计里面有几个0&#xff0c;然后遍历数组&#xff0c;如果是0或者比后面一个数小1就直接进入下一次循环&#xff0c;如果比后面一个数小2&#xff0c;就用掉一个0&#xff0c;0的数量减1&#xff0c;如果比后面的…

Pycharm----导入库文件夹不在py文件的目录下

问题描述&#xff1a; 想在不同目录下导入根目录的包&#xff0c;直接写会报错。如下边object_detect.py在function文件夹下&#xff0c;导入包默认在这个文件下&#xff0c;但我想导入根目录models和utils下的包 解决方法&#xff1a; 将根目录设置为源代码根目录&#xff0…

【OC总结 面向对象 + 内存管理 + runtime】

文章目录 前言面向对象1.1 一个NSObject对象占用多少内存&#xff1f;1.2 iOS的继承链 & 对象的指针指向了哪里&#xff1f;1.3 OC的类的信息存放在哪里&#xff1f;-isa指针1.4 isMemberOfClass & isKindOfClass Runtime1.4 讲一下OC的消息机制1.5 消息转发机制流程1.…

【指针和数组笔试题(1)】详解指针、数组笔试题

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言整型数组字符数组第一组题第二组题第三组题 总结 前言 在计算之前要了解基本概念&#xff1a; 数组名的理解 数组名是数组首元素的地址 有两个例外 1.sizeof(…