HTML+CSS+JS实现2048经典小游戏(附完整源码)

2024-06-05 18:20

本文主要是介绍HTML+CSS+JS实现2048经典小游戏(附完整源码),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

2048 小游戏的目标是通过合并数字单元格,最终在 4x4 的棋盘上创建一个值为 2048 的单元格。

一、预览效果

在这里插入图片描述

二、程序源码

html代码

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>2048 Game</title><link rel="stylesheet" href="game.css">
</head>
<body><div class="game-container"><div class="score-container">分数: <span id="score">0</span></div><div class="game-board" id="game-board"><!-- 这里将动态生成游戏单元格 --></div></div><script src="game.js"></script>
</body>
</html>

css代码

/* styles.css */
body {display: flex;justify-content: center;align-items: center;height: 100vh;margin: 0;background-color: #faf8ef;font-family: 'Arial', sans-serif;
}.game-container {display: flex;justify-content: center;align-items: center;flex-direction: column;
}.score-container {margin-bottom: 20px;font-size: 24px;font-weight: bold;color: #776e65;
}.game-board {display: grid;grid-template-columns: repeat(4, 100px);grid-template-rows: repeat(4, 100px);gap: 10px;background-color: #bbada0;padding: 10px;border-radius: 10px;
}.cell {width: 100px;height: 100px;display: flex;justify-content: center;align-items: center;background-color: #cdc1b4;border-radius: 5px;font-size: 24px;font-weight: bold;color: #776e65;
}.cell-2 { background-color: #eee4da; }
.cell-4 { background-color: #ede0c8; }
.cell-8 { background-color: #f2b179; color: #f9f6f2; }
.cell-16 { background-color: #f59563; color: #f9f6f2; }
.cell-32 { background-color: #f67c5f; color: #f9f6f2; }
.cell-64 { background-color: #f65e3b; color: #f9f6f2; }
.cell-128 { background-color: #edcf72; color: #f9f6f2; }
.cell-256 { background-color: #edcc61; color: #f9f6f2; }
.cell-512 { background-color: #edc850; color: #f9f6f2; }
.cell-1024 { background-color: #edc53f; color: #f9f6f2; }
.cell-2048 { background-color: #edc22e; color: #f9f6f2; }

js代码

// game.js
document.addEventListener('DOMContentLoaded', () => {const gameBoard = document.getElementById('game-board');const scoreElement = document.getElementById('score');const size = 4;let cells = [];let score = 0;function createBoard() {for (let i = 0; i < size * size; i++) {const cell = document.createElement('div');cell.classList.add('cell');gameBoard.appendChild(cell);cells.push(cell);}addNumber();addNumber();}function addNumber() {let emptyCells = cells.filter(cell => cell.innerText === '');if (emptyCells.length === 0) return;let randomCell = emptyCells[Math.floor(Math.random() * emptyCells.length)];randomCell.innerText = Math.random() > 0.1 ? 2 : 4;randomCell.classList.add(`cell-${randomCell.innerText}`);}function move(direction) {let hasMoved = false;for (let i = 0; i < size; i++) {let rowOrCol = [];for (let j = 0; j < size; j++) {let index = direction === 'left' || direction === 'right' ? i * size + j : j * size + i;rowOrCol.push(cells[index]);}if (direction === 'right' || direction === 'down') rowOrCol.reverse();let newRowOrCol = slide(rowOrCol);if (direction === 'right' || direction === 'down') newRowOrCol.reverse();for (let j = 0; j < size; j++) {let index = direction === 'left' || direction === 'right' ? i * size + j : j * size + i;if (cells[index].innerText !== newRowOrCol[j].innerText) hasMoved = true;cells[index].innerText = newRowOrCol[j].innerText;cells[index].className = 'cell';if (cells[index].innerText !== '') cells[index].classList.add(`cell-${cells[index].innerText}`);}}if (hasMoved) {addNumber();updateScore();if (checkGameOver()) {setTimeout(() => alert('游戏结束!'), 100);}}}function slide(rowOrCol) {let arr = rowOrCol.filter(cell => cell.innerText !== '').map(cell => parseInt(cell.innerText));for (let i = 0; i < arr.length - 1; i++) {if (arr[i] === arr[i + 1]) {arr[i] *= 2;score += arr[i];  // 更新分数arr.splice(i + 1, 1);}}while (arr.length < size) arr.push('');return arr.map(num => {let cell = document.createElement('div');cell.classList.add('cell');cell.innerText = num;return cell;});}function handleKey(e) {switch (e.key) {case 'ArrowUp':move('up');break;case 'ArrowDown':move('down');break;case 'ArrowLeft':move('left');break;case 'ArrowRight':move('right');break;}}function updateScore() {scoreElement.innerText = score;}function checkGameOver() {for (let i = 0; i < size; i++) {for (let j = 0; j < size; j++) {let index = i * size + j;if (cells[index].innerText === '') return false;if (j < size - 1 && cells[index].innerText === cells[index + 1].innerText) return false;if (i < size - 1 && cells[index].innerText === cells[index + size].innerText) return false;}}return true;}createBoard();document.addEventListener('keydown', handleKey);
});

这篇关于HTML+CSS+JS实现2048经典小游戏(附完整源码)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1033824

相关文章

MySQL双主搭建+keepalived高可用的实现

《MySQL双主搭建+keepalived高可用的实现》本文主要介绍了MySQL双主搭建+keepalived高可用的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,... 目录一、测试环境准备二、主从搭建1.创建复制用户2.创建复制关系3.开启复制,确认复制是否成功4.同

Java实现文件图片的预览和下载功能

《Java实现文件图片的预览和下载功能》这篇文章主要为大家详细介绍了如何使用Java实现文件图片的预览和下载功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... Java实现文件(图片)的预览和下载 @ApiOperation("访问文件") @GetMapping("

用js控制视频播放进度基本示例代码

《用js控制视频播放进度基本示例代码》写前端的时候,很多的时候是需要支持要网页视频播放的功能,下面这篇文章主要给大家介绍了关于用js控制视频播放进度的相关资料,文中通过代码介绍的非常详细,需要的朋友可... 目录前言html部分:JavaScript部分:注意:总结前言在javascript中控制视频播放

使用Sentinel自定义返回和实现区分来源方式

《使用Sentinel自定义返回和实现区分来源方式》:本文主要介绍使用Sentinel自定义返回和实现区分来源方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Sentinel自定义返回和实现区分来源1. 自定义错误返回2. 实现区分来源总结Sentinel自定

Java实现时间与字符串互相转换详解

《Java实现时间与字符串互相转换详解》这篇文章主要为大家详细介绍了Java中实现时间与字符串互相转换的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、日期格式化为字符串(一)使用预定义格式(二)自定义格式二、字符串解析为日期(一)解析ISO格式字符串(二)解析自定义

opencv图像处理之指纹验证的实现

《opencv图像处理之指纹验证的实现》本文主要介绍了opencv图像处理之指纹验证的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学... 目录一、简介二、具体案例实现1. 图像显示函数2. 指纹验证函数3. 主函数4、运行结果三、总结一、

Springboot处理跨域的实现方式(附Demo)

《Springboot处理跨域的实现方式(附Demo)》:本文主要介绍Springboot处理跨域的实现方式(附Demo),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不... 目录Springboot处理跨域的方式1. 基本知识2. @CrossOrigin3. 全局跨域设置4.

Spring Boot 3.4.3 基于 Spring WebFlux 实现 SSE 功能(代码示例)

《SpringBoot3.4.3基于SpringWebFlux实现SSE功能(代码示例)》SpringBoot3.4.3结合SpringWebFlux实现SSE功能,为实时数据推送提供... 目录1. SSE 简介1.1 什么是 SSE?1.2 SSE 的优点1.3 适用场景2. Spring WebFlu

基于SpringBoot实现文件秒传功能

《基于SpringBoot实现文件秒传功能》在开发Web应用时,文件上传是一个常见需求,然而,当用户需要上传大文件或相同文件多次时,会造成带宽浪费和服务器存储冗余,此时可以使用文件秒传技术通过识别重复... 目录前言文件秒传原理代码实现1. 创建项目基础结构2. 创建上传存储代码3. 创建Result类4.

SpringBoot日志配置SLF4J和Logback的方法实现

《SpringBoot日志配置SLF4J和Logback的方法实现》日志记录是不可或缺的一部分,本文主要介绍了SpringBoot日志配置SLF4J和Logback的方法实现,文中通过示例代码介绍的非... 目录一、前言二、案例一:初识日志三、案例二:使用Lombok输出日志四、案例三:配置Logback一