基于 WebAssembly 的 Game of Life 交互实现

一、前言

在前期的实现中,我们使用 Rust 编写核心逻辑,并通过 WebAssembly 将其引入到 Web 环境中,再利用 JavaScript 进行渲染。接下来,我们将在这一基础上增加用户交互功能,使模拟过程不仅能够自动演化,还能支持用户对细胞状态进行手动修改。

二、实现暂停与恢复游戏

1. 在 HTML 中添加按钮

首先,在 wasm-game-of-life/www/index.html 文件中,在 <canvas> 上方添加一个按钮,用于控制游戏的暂停与恢复:

<button id="play-pause"></button>

2. 在 JavaScript 中管理动画

为了控制动画,我们需要保存 requestAnimationFrame 返回的标识符,以便后续通过 cancelAnimationFrame 取消动画调用。在 wasm-game-of-life/www/index.js 中,新增如下代码:

let animationId = null;// renderLoop 函数负责绘制网格、细胞并更新状态
const renderLoop = () => {drawGrid();drawCells();universe.tick();animationId = requestAnimationFrame(renderLoop);
};// 判断游戏是否暂停
const isPaused = () => {return animationId === null;
};const playPauseButton = document.getElementById("play-pause");// 开始游戏,设置按钮图标并启动动画
const play = () => {playPauseButton.textContent = "⏸";renderLoop();
};// 暂停游戏,更新按钮图标并取消下一帧动画
const pause = () => {playPauseButton.textContent = "▶";cancelAnimationFrame(animationId);animationId = null;
};// 根据当前状态切换播放和暂停
playPauseButton.addEventListener("click", event => {if (isPaused()) {play();} else {pause();}
});// 初始化时调用 play() 启动动画
play();

通过这段代码,我们可以根据 animationId 的状态判断游戏是否在播放,并相应地启动或暂停动画。同时,按钮的图标也会显示对应的状态。

三、实现点击切换细胞状态

为了让用户能直接在画布上点击修改细胞状态,我们需要在 Rust 和 JavaScript 端分别进行处理。

1. Rust 端的修改

wasm-game-of-life/src/lib.rs 文件中,为 Cell 添加一个 toggle 方法,用于切换细胞状态:

impl Cell {fn toggle(&mut self) {*self = match *self {Cell::Dead => Cell::Alive,Cell::Alive => Cell::Dead,};}
}

接着,在 Universe 的公开接口中添加 toggle_cell 方法,这样 JavaScript 就能调用它来切换指定行列上的细胞状态:

#[wasm_bindgen]
impl Universe {pub fn toggle_cell(&mut self, row: u32, column: u32) {let idx = self.get_index(row, column);self.cells[idx].toggle();}
}

通过 #[wasm_bindgen] 标注,该方法会被导出至 JavaScript 环境中。

2. JavaScript 端的处理

wasm-game-of-life/www/index.js 中,为 <canvas> 添加点击事件监听器,完成以下步骤:

  1. 获取点击事件相对于页面的坐标;
  2. 将页面坐标转换为 <canvas> 内部的坐标;
  3. 根据坐标计算出对应的行和列;
  4. 调用 universe.toggle_cell 切换细胞状态;
  5. 重新绘制网格和细胞以更新界面显示。

具体实现如下:

canvas.addEventListener("click", event => {const boundingRect = canvas.getBoundingClientRect();const scaleX = canvas.width / boundingRect.width;const scaleY = canvas.height / boundingRect.height;const canvasLeft = (event.clientX - boundingRect.left) * scaleX;const canvasTop = (event.clientY - boundingRect.top) * scaleY;const row = Math.min(Math.floor(canvasTop / (CELL_SIZE + 1)), height - 1);const col = Math.min(Math.floor(canvasLeft / (CELL_SIZE + 1)), width - 1);universe.toggle_cell(row, col);drawGrid();drawCells();
});

通过以上代码,当用户点击 <canvas> 时,就能实现对对应位置细胞状态的即时切换,从而自由绘制自定义图案。

四、效果预览

完成上述步骤后,使用 wasm-pack build 重建项目,然后刷新 http://localhost:8080/。你将会看到:

  • 点击页面上的按钮可以暂停或恢复游戏的动画;
  • 在暂停状态下,点击画布即可改变相应位置细胞的状态,实现自定义图案的绘制。

五、总结

通过本文,我们实现了基于 WebAssembly 的 Game of Life 模拟的交互功能:

  • 利用 JavaScript 控制动画的暂停和恢复,使得用户能够在需要时停下演化过程进行编辑;
  • 通过点击 <canvas> 切换细胞状态,让自定义图案的绘制变得直观便捷。

这种前后端技术的紧密结合展示了现代 Web 开发中多技术协作的魅力,为进一步扩展和优化应用提供了坚实基础。希望这篇博客能为你提供有价值的参考,激发你探索更多有趣的功能扩展和优化方案。

Happy coding!

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

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

相关文章

VLAN综合实验二

一.实验拓扑&#xff1a; 二.实验需求&#xff1a; 1.内网Ip地址使用172.16.0.0/分配 2.sw1和SW2之间互为备份 3.VRRP/STP/VLAN/Eth-trunk均使用 4.所有Pc均通过DHCP获取IP地址 5.ISP只能配置IP地址 6.所有…

ABAP FPM

1.效果 2.查询条件的feed class SE11创建feed class数据的结构 ZCL_FPM_FIFO_SEARCH GET_DEFINITION方法代码 METHOD if_fpm_guibb_search~get_definition.eo_field_catalog_attr ? cl_abap_structdescr>describe_by_name( ZSS_FIFO_DATA ).ENDMETHOD. PROCESS_EVENT代码…

某大麦手机端-抢票

引言 仅供学习研究&#xff0c;欢迎交流 抢票难&#xff0c;难于上青天&#xff01;无论是演唱会、话剧还是体育赛事&#xff0c;大麦网的票总是秒光。作为一名技术爱好者&#xff0c;你是否想过用技术手段提高抢票成功率&#xff1f;本文将为你揭秘大麦手机端抢票的核心技术…

【免费】2007-2019年各省地方财政文化体育与传媒支出数据

2007-2019年各省地方财政文化体育与传媒支出数据 1、时间&#xff1a;2007-2019年 2、来源&#xff1a;国家统计局、统计年鉴 3、指标&#xff1a;行政区划代码、地区、年份、地方财政文化体育与传媒支出 4、范围&#xff1a;31省 5、指标说明&#xff1a;地方财政在文化、…

Playwright从入门到实战:比Selenium更快的数据爬取案例实战

摘要 Playwright 是微软开源的下一代浏览器自动化工具&#xff0c;凭借其高性能、跨浏览器支持和现代化设计&#xff0c;迅速成为 Web 自动化领域的热门选择。本文将从 安装配置 开始&#xff0c;通过 实战演练 展示其核心功能&#xff0c;并与 Selenium 深度对比&#xff0c;…

音频知识 参数分析

通道布局 参考 通过pcm音频数据计算分贝 理解FFT和信号加窗原理及意义 dts音效大师教程

《STL 六大组件之容器探秘:深入剖析 vector》

目录 一、vector 简介二、vector 的常用接口1. 构造函数&#xff08;constructor function&#xff09;2. 与迭代器相关的接口&#xff08;iterator&#xff09;3. 与容量相关的接口&#xff08;capacity&#xff09;4. 与访问和修改有关的接口&#xff08;access、modify&…

基于深度学习的手势识别系统设计

目录 背景与分析&#xff1a; 基于深度学习的手势识别系统设计 课题要求 技术分析&#xff1a; 技术点概述&#xff1a; 过程&#xff08;主体展示为主&#xff0c;部分功能不一一展示&#xff09;&#xff1a; 数据集&#xff08;17个G&#xff09;&#xff1a; 源码、…

【SpringCloud】Eureka的使用

3. Eureka 3.1 Eureka 介绍 Eureka主要分为两个部分&#xff1a; EurekaServer: 作为注册中心Server端&#xff0c;向微服务应用程序提供服务注册&#xff0c;发现&#xff0c;健康检查等能力。 EurekaClient: 服务提供者&#xff0c;服务启动时&#xff0c;会向 EurekaS…

构建大语言模型应用:简介(第一部分)

本专栏聚焦大语言模型&#xff08;LLM&#xff09;相关内容的解析&#xff0c;通过检索增强生成&#xff08;RAG&#xff09;应用的视角来进行。 本系列文章 简介&#xff08;本文&#xff09;数据准备句子转换器向量数据库搜索与检索大语言模型开源检索增强生成评估大语言模…

Python:日志模块操作及基本配置,日志格式化输出

logger(日志管理器)&#xff1a;用户默认是root&#xff0c;其配置步骤为&#xff1a; 1.初始化对象 2.设置级别 3.定义handler(控制台或文件显示) 4.格式化输出 import logging #创建log文件并设置级别 logging.basicConfig(filenametext.log, levellogging.DEBUG) #在文…

The selected directory is not a valid home for Go SDK

安装完毕Go和GoLand之后&#xff0c;重新配置GoLand的GOROOT时&#xff0c;可能会报错 The selected directory is not a valid home for Go SDK 需要找到安装Go的目录&#xff0c;我的目录如下 D:\SoftFolder\Go\src\runtime\internal\sys 进入到sys文件下&#xff0c;找到…

pycharm找不到conda可执行文件解决办法

配置一个新项目&#xff0c;需要使用远程服务器环境的时候&#xff0c;找不到anaconda里面的python可执行文件&#xff0c;怎么都解决不了&#xff0c;很尴尬&#xff0c;看了很多&#xff0c;终于找到一个博客有用&#xff1a; https://blog.csdn.net/weixin_43912188/articl…

Blender多摄像机怎么指定相机渲染图像

如题目所说&#xff0c;当blender的场景里面有摄像机的时候&#xff0c;按F12可以预览渲染结果&#xff0c;但是当有多个摄像机的时候就不知道使用哪个进行渲染了。 之前在网上没有找到方法&#xff0c;就用笨方法&#xff0c;把所有的摄像机删除&#xff0c;然后设置自己需要…

org.apache.maven.surefire:surefire-junit-platform:jar:2.22.2 Maven打包失败

org.apache.maven.surefire:surefire-junit-platform:jar:2.22.2 解决办法 勾上这个&#xff0c;打包时跳过测试代码

CKS认证 | Day3 K8s容器运行环境安全加固

一、最小特权原则&#xff08;POLP&#xff09; 1&#xff09;最小特权原则 (Principle of least privilege&#xff0c;POLP) &#xff1a; 是一种信息安全概念&#xff0c;即为用户提供执行其工作职责所需的最 小权限等级或许可。 最小特权原则被广泛认为是网络安全的最佳实…

项目-苍穹外卖(十七) Apache POI+导出数据

一、介绍 二、入门案例 package com.sky.test;import org.apache.poi.xssf.usermodel.XSSFRow; import org.apache.poi.xssf.usermodel.XSSFSheet; import org.apache.poi.xssf.usermodel.XSSFWorkbook;import java.io.File; import java.io.FileNotFoundException; import jav…

2023第十四届蓝桥杯大赛软件赛国赛C/C++ 大学 B 组(真题题解)(C++/Java题解)

本来想刷省赛题呢&#xff0c;结果一不小心刷成国赛了 真是个小迷糊〒▽〒 但&#xff0c;又如何( •̀ ω •́ )✧ 记录刷题的过程、感悟、题解。 希望能帮到&#xff0c;那些与我一同前行的&#xff0c;来自远方的朋友&#x1f609; 大纲&#xff1a; 一、子2023-&#xff…

Cannot find a valid baseurl for repo: centos-sclo-sclo/x86_64

​ rpm -Uvh https://repo.zabbix.com/zabbix/5.0/rhel/7/x86_64/zabbix-release-latest-5.0.el7.noarch.rpmyum clean allyum macache fast​ 编辑配置文件 /etc/yum.repos.d/zabbix.repo and enable zabbix-frontend repository. [zabbix-frontend]...enabled1... 下载相关…

【STM32】WDG看门狗(学习笔记)

学习来源----->江协科技STM32 WDG简介 WDG&#xff08;Watchdog&#xff09;看门狗看门狗可以监控程序的运行状态&#xff0c;当程序因为设计漏洞、硬件故障、电磁干扰等原因&#xff0c;出现卡死或跑飞现象时&#xff0c;看门狗能及时复位程序&#xff0c;避免程序陷入长…