基于Vue和Vuex实现俄罗斯方块小游戏

news/2025/1/18 6:17:43/

用 Vue、Vuex 做俄罗斯方块

本项目灵感来源于 React 版的俄罗斯方块,由于对其实现原理较感兴趣,而且相比于 React 更喜欢 Vue, 于是把 React 版的重构为了 Vue 版的,大致思路是把组件当成一个个函数,保证一个输入(props)能得到一个确定的输出(view),然后对不同方法也是做同样处理,对于 Redux 使用 Vuex 精简化

效果预览

正常速度的录制,体验流畅。

响应式

不仅指屏幕的自适应,而是 在PC使用键盘、在手机使用手指的响应式操作

数据持久化

此处为语雀视频卡片,点击链接查看:persistence.mp4

Vuex 状态预览

此处为语雀视频卡片,点击链接查看:vuex.mp4

Vuex 设计管理了所有应存的状态,这是上面持久化的保证。

游戏框架使用的是 Vue + Vuex

1、Web Audio API

游戏里有很多不同的音效,而实际上只引用了一个音效文件:/build/music.mp3。借助 Web Audio Api 能够以毫秒级精确、高频率的播放音效,这是 <audio> 标签所做不到的。在游戏进行中按住方向键移动方块,便可以听到高频率的音效。

WAA 是一套全新的相对独立的接口系统,对音频文件拥有更高的处理权限以及更专业的内置音频效果,是 W3C 的推荐接口,能专业处理“音速、音量、环境、音色可视化、高频、音向”等需求,下图介绍了 WAA 的使用流程。

其中 Source 代表一个音频源,Destination 代表最终的输出,多个 Source 合成出了 Destination。
源代码:/src/unit/music.js 实现了 AJAX 加载 mp3,并转为 WAA,控制播放的过程。

WAA 在各个浏览器的最新 2 个版本下的支持情况

可以看到 IE 阵营与大部分安卓机不能使用,其他 ok。

3、游戏在体验上的优化

  • 技术:
    • 按下方向键水平移动和竖直移动的触发频率是不同的,游戏可以定义触发频率,代替原生的事件频率,源代码:/src/unit/event.js ;
    • 左右移动可以 delay 掉落的速度,但在撞墙移动的时候 delay 的稍小;在速度为 6 级时 通过 delay 会保证在一行内水平完整移动一次;
    • 对按钮同时注册 touchstartmousedown 事件,以供响应式游戏。当 touchstart 发生时,不会触发 mousedown,而当 mousedown 发生时,由于鼠标移开事件元素可以不触发 mouseup,将同时监听 mouseout 模拟 mouseup。源代码:/src/components/keyboard/index.js;
    • 监听了 visibilitychange 事件,当页面被隐藏\切换的时候,游戏将不会进行,切换回来将继续,这个 focus 状态也被写进了 Vuex 中。所以当用手机玩来 电话 时,游戏进度将保存;PC 开着游戏干别的也不会听到 gameover,这有点像 ios 应用的切换。
    • 任意 时刻刷新网页,(比如消除方块时、游戏结束时)也能还原当前状态;
    • 游戏中唯一用到的图片是

      ,其他都是 CSS;
    • 游戏兼容 Chrome、Firefox、IE9+、Edge 等;
  • 玩法:
    • 可以在游戏未开始时制定初始的棋盘(十个级别)和速度(六个级别);
    • 一次消除 1 行得 100 分、2 行得 300 分、3 行得 700 分、4 行得 1500 分;
    • 方块掉落速度会随着消除的行数增加(每 20 行增加一个级别);

4、开发中的经验梳理,以及如何把 React 项目重构为 Vue 版本

Vue 版本和 React 版本核心代码基本相同,但在编写组件的时候遇到了几个问题,比如:

  1. 如何把 React 组件改写成 Vue 的,我的思路是把组件当成函数,保证一个输入(props)能得到一个确定的输出(view),然后对不同方法也是做同样处理, React 的 setState 会触发 render 方法,所以可以在 methods 自定义 render 方法再在 state 变化后手动触发 render 方法
  2. 生命周期,简单来说, React 的 componentWillMount 对应 Vue 的 beforeMount, React 的 componentDidMount 对应 Vue 的 mounted,React 的用来优化性能的 shouldComponentUpdate 在 Vue 里并不需要,不需要手动优化这也是我喜欢 Vue 的一点
  3. Vue 没有 React 的 componentWillReceiveProps 的生命周期,我的解决方法是使用 watch 配合 deep:true 来监听 props 的变化,如:
  watch: {$props: {deep: true,handler(nextProps) {//xxx}}}
  1. 在必要时候使用 jsx 和 'render' 函数,是的, vue 支持 jsx,在这个项目中 matrix 组件 的功能逻辑较复杂,使用 template 模版来渲染组件已经不合适了, React 每次 setState 会触发 'render' 方法,所以我们可以在 methods 自定义 'render' 方法再在 state 变化后手动触发 'render' 方法,但是这个方法对有复杂逻辑的组件来说会变得很繁琐,我的解决方法是通过 Vue 的 jsx 转换插件 babel-plugin-transform-vue-jsx 来使用 jsx 语法对页面进行渲染,当 props 或 state 变化了自动触发 'render' 方法,另外要注意的是 vue 的 jsx 和 React 的 jsx 书写上有一点的差异, 当 'render' 方法存在时,template 语法会失效. 'render' 函数一个比较实用的用处是在开发类似 React-log 之类的不需要渲染 HTML 只需要执行一些方法的组件时 template 会显得很多余,因为这时候并不需要渲染 dom 了,如果用了 'render' 函数,简单的在 'render' 函数里 return false 就行,如: react-log

5、架构差异

Redux 的数据流向是通过 mapStateToProps 把 store 的状态转化为 props 然后通过 connect 函数注入到 根组件,根组件再把这些 props 传入不同组件,当 store 的状态变化,根组件会重新 render, 更新子组件上的 props,子组件再 根据新 props 重新 render

而 vuex 的思路则不同,任何组件都随时可以通过 this.$store.state.xxx 获取 store 上的数据,更自由,从 store 实例中读取状态最简单的方法就是在计算属性中返回某个状态:

computed: {keyboard () {return this.store.state.keyboard}}

调用 store.commit 提交 payload 修改数据 或者 store.dispatch 提交 mutation 间接修改 store 上的数据, commit 和 dispatch 的区别在于 commit 用于同步修改状态, dispatch 用于异步修改状态,异步完成需要再调用 commit,一般简单的需求只需要 commit 一个 payload 就行,只要 store 上的数据变了,组件都会自动重新渲染

6、开发

安装

npm install

运行

npm run dev

浏览自动打开 http://localhost:8080

多语言

在 i18n.json 配置多语言环境,使用"lan"参数匹配语言如:https://Binaryify.github.io/vue-tetris/?lan=en

打包编译

npm run build

dist 文件夹下生成结果。


http://www.ppmy.cn/news/1564074.html

相关文章

西门子【Library of Basic Controls (LBC)基本控制库”(LBC) 提供基本控制功能】

AF架构中使用的库 文章目录 Table of contents Legal information ..............................................................................................................................2 1 Introduction ................................................…

dockerhub上一些镜像

K8s下网络排障工具 https://hub.docker.com/r/nicolaka/netshoot ex kubectl run tmp-shell --rm -i --tty --image nicolaka/netshoot -- /bin/bash # 主机的net ns下运行 kubectl run tmp-shell --rm -i --tty --overrides{"spec": {"hostNetwork": tru…

git 命令如何在本地新建分支,并push到远程仓库的新分支上

在 Git 中&#xff0c;新建本地分支并将其推送到远程仓库的新分支上&#xff0c;可以按照以下步骤进行&#xff1a; 确保工作目录是干净的&#xff1a; 在开始之前&#xff0c;最好确保你的工作目录中没有未提交的更改。你可以通过运行 git status 来检查。 新建本地分支&…

冯·诺依曼体系结构:计算机科学的奠基石

文章目录 前言&#x1f3b7;一、冯诺依曼体系结构&#xff08;Von Neumann Architecture&#xff09;&#x1f3b8;1.1 硬件介绍&#x1f941;1. 输入设备&#x1f941;2. 输出设备&#x1f941;3. 输入输出一体化设备&#x1f941;4. 存储器&#x1f941;5. 中央处理器CPU&…

ASP.Net Identity + IODC 解析ReturnUrl

Identity Ids4 配置 成认证服务 一、创建 Identity身份验证 项目 创建的项目结构中 没有 注册和登录的 控制器和视图 配置数据库地址 》》默认已经生成了Miagratin 直接update-database 二、在Identity项目 配置 IdentityServer4 Nuget 两个包 》》》配置Config 类 usin…

DX12 快速教程(3) —— 画矩形

快速导航 新建项目 "003-DrawRectangle"初识渲染管线什么是渲染管线可编程渲染管线与着色器渲染管线五大阶段Input Assembler (IA) Stage 输入装配阶段Vertex Shader (VS) Stage 顶点着色器阶段什么是齐次坐标透视与齐次坐标六大坐标空间顶点着色器的目标 Rasterizat…

《深度学习神经网络训练:数据集下载资源列表》

深度学习神经网络训练&#xff1a;数据集下载资源列表 一、数据集下载的重要性 在当今数字化时代&#xff0c;数据集下载对于各个领域的研究与发展都具有不可忽视的重要意义。尤其在机器学习、深度学习以及各类数据驱动的科研项目中&#xff0c;数据集更是起到了基础性的支撑…

JVM 触发类加载的条件有哪些?

目录 一、类加载生命周期 二、主动引用 2.1、创建类的实例 2.2、访问类的静态字段或静态方法 2.3、反射 2.4、初始化类的子类时&#xff0c;先初始化父类 2.5、虚拟机启动时&#xff0c;初始化 main 方法所在的类 2.6、动态语言支持 三、被动引用 3.1、通过子类引用父…