基于 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.mzph.cn/bicheng/75008.shtml

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

相关文章

【keil】单步调试

一、步骤 1、打开stc-isp软件 2.打开keil仿真设置&#xff0c;选择对应的单片机型号 3.点击将所选目标单片机设置为仿真芯片&#xff0c;点击下载&#xff0c;按一下单片机打下载按钮 4.此时已经将仿真程序下载到单片机 5.此时点击options,找到debug选择STC Montor 51 Driv…

c++弱指针实现原理

在 C 中&#xff0c;弱指针&#xff08;std::weak_ptr&#xff09;是一种特殊的智能指针&#xff0c;其核心目标是‌解决 std::shared_ptr 的循环引用问题‌&#xff0c;同时不增加对象的引用计数。它的实现原理基于与 std::shared_ptr 共享的 ‌控制块&#xff08;Control Blo…

【ManiSkill】环境success条件和reward函数学习笔记

1. “PickCube-v1” info["success"]&#xff1a;用于指示任务是否成功完成 布尔型张量&#xff0c;在环境的evaluate()方法中计算并返回&#xff1a; "success": is_obj_placed & is_robot_static这确保了机器人不仅能将物体准确放置在目标位置&am…

用空闲时间做了一个小程序-二维码生成器

一直在摸鱼中赚钱的大家好呀~ 先向各位鱼友们汇报一下情况&#xff0c;目前小程序已经有900的鱼友注册使用过。虽然每天都有新的鱼友注册&#xff0c;但是鱼友增长的还很缓慢。自从国庆前的文字转语音的工具上线到现在已经将近有1个月没有更新小程序了。但是今天终终终终终于又…

31天Python入门——第14天:异常处理

你好&#xff0c;我是安然无虞。 文章目录 异常处理1. Python异常2. 异常捕获try-except语句捕获所有的异常信息获取异常对象finally块 3. raise语句4. 自定义异常5. 函数调用里面产生的异常补充练习 异常处理 1. Python异常 Python异常指的是在程序执行过程中发生的错误或异…

PyQt6实例_批量下载pdf工具_使用pyinstaller与installForge打包成exe文件

目录 前置&#xff1a; 步骤&#xff1a; step one 准备好已开发完毕的项目代码 step two 安装pyinstaller step three 执行pyinstaller pdfdownload.py&#xff0c;获取初始.spec文件 step four 修改.spec文件&#xff0c;将data文件夹加入到打包程序中 step five 增加…

Axure项目实战:智慧城市APP(完整交互汇总版)

亲爱的小伙伴&#xff0c;在您浏览之前&#xff0c;烦请关注一下&#xff0c;在此深表感谢&#xff01; 课程主题&#xff1a;智慧城市APP 主要内容&#xff1a;主功能&#xff08;社保查询、医疗信息、公交查询等&#xff09;、活动、消息、我的页面汇总 应用场景&#xff…

Appium Inspector使用教程

1.下载最新版本 https://github.com/appium/appium-inspector/releases 2.本地启动一个Appium服务 若Android SDK已安装Appium服务&#xff0c;则在任意terminal使用appium启动服务即可 3.Appium Inspector客户端配置连接到Appium服务 Configuring and Starting a Session…

Pycharm(七):几个简单案例

一.剪刀石头布 需求&#xff1a;和电脑玩剪刀石头布游戏 考察点&#xff1a;1.随机数&#xff1b;2.判断语句 import random # numrandom.randint(1,3) # print(num) # print(**30) #1.录入玩家手势 playerint(input(请输入手势&#xff1a;&#xff08;1.剪刀 2.石头 3&…

Python Cookbook-4.13 获取字典的一个子集

任务 你有一个巨大的字典&#xff0c;字典中的一些键属于一个特定的集合&#xff0c;而你想创建一个包含这个键集合及其对应值的新字典。 解决方案 如果你不想改动原字典: def sub_dict(somedict,somekeys,default None):return dict([(k, somedict.get(k,default)) for k…

VMware Ubuntu 网络配置全攻略:从断网到畅通无阻

一、网络连接模式选择&#xff08;先搞懂原理&#xff09; VMware提供三种网络模式&#xff0c;就像手机的不同网络套餐&#xff1a; 模式适用场景特点类比NAT个人上网/新手首选虚拟机共享主机IP&#xff0c;能上网但隐身家用WiFi桥接服务器/需要被局域网访问虚拟机会获得独立…

链表(C++)

这是本人第二次学习链表&#xff0c;第一次学习链表是在大一上的C语言课上&#xff0c;首次接触&#xff0c;感到有些难&#xff1b;第二次是在大一下学习数据结构时&#xff08;就是这次&#xff09;&#xff0c;使用C再次理解链表。同时&#xff0c;这也是开启数据结构学习写…

【SPP】蓝牙串口协议应用层深度解析:从连接建立到实战开发

目录 一、SPP应用层协议框架与角色模型 1.1 分层协议栈模型 1.2 设备角色模型&#xff08;DevA 与 DevB 交互&#xff09; 二、连接建立流程&#xff1a;从 SDP 到 RFCOMM 2.1 服务发现&#xff08;SDP&#xff09;流程&#xff08;SDP 记录关键参数&#xff09; 2.2 连接…

Giteki 认证:无线产品进入日本市场的关键保障

目录 适用产品认证范围 认证项目及技术要求 认证流程 认证周期 与其他认证的对比 常见问题 注意事项 Giteki 认证&#xff0c;其名称来源于日本语 “技適マーク”&#xff0c;罗马字拼写为 “GITEKI” &#xff0c;在行业内也常被称为 Telec 认证、MIC 认证、RF 认证或技…

Ubuntu24.04 配置远程桌面服务

一&#xff1a;安装 sudo apt update sudo apt install vino 二&#xff1a;设置 gsettings set org.gnome.Vino require-encryption false # 关闭加密&#xff08;某些 VNC 客户端不支持加密&#xff09; gsettings set org.gnome.Vino prompt-enabled false # 关闭连接…

人工智能与软件工程结合的发展趋势

AI与软件工程的结合正在深刻改变软件开发的流程、工具和方法&#xff0c;其发展方向涵盖了从代码生成到系统维护的整个生命周期。以下是主要的发展方向和技术趋势&#xff1a; 1. 软件架构体系的重构 从“面向过程”到“面向目标”的架构转型&#xff1a; AI驱动软件设计以目标…

转发和重定向的区别详解

转发&#xff08;Forward&#xff09;和重定向&#xff08;Redirect&#xff09;是 Web 开发中两种常用的请求处理方式&#xff0c;主要用于将客户端请求从一个资源转移到另一个资源。它们在实现机制、行为表现和应用场景上有显著区别&#xff0c;以下是对两者的详细解析&#…

python专题1-----判断一个变量是否是字符串类型

在 Python 中&#xff0c;可以使用 isinstance() 函数来判断一个变量是否是字符串类型。字符串在 Python 中是以 str 类型表示的。下面是一些示例代码&#xff0c;展示如何判断一个变量是否是字符串类型&#xff1a; # 示例变量 var1 "Hello, World!" var2 12345 …

软件工程之需求工程(需求获取、分析、验证)

一、需求获取&#xff08;Requirements Elicitation&#xff09; 1. 定义与目标 需求获取是通过与用户、利益相关者等交互&#xff0c;识别并捕获系统需求的过程&#xff0c;目标是明确用户意图与业务目标&#xff0c;避免后期因需求偏差导致返工。 2. 主要方法 问卷法&…

Java简单生成pdf

生成这样的PDF 直接上代码 public static void main(String[] args) {String logoPath "Q:\\IdeaWork\\Demo\\src\\main\\webapp\\images\\logo.jpg"; // 替换为实际路径String baseDir "E:/Demo/TEST/problem/Generate"; // 基础目录int year 2025; //…