1个不稳定版本

0.0.0 2023年1月1日

#38#local-storage

MIT/Apache

9KB
224

docs.rs docs

让使用Rust WebAssembly编写Web应用程序变得容易

我想有一个库,人们可以在一个下午学会使用并开始制作交互式浏览器体验。这个项目不支持所有浏览器功能。虽然你可以很容易地使用这个库中使用的JavaScript调用机制添加你自己的功能。请随意提交一个PR以添加更多功能。

  • 异步 & 协程支持
  • 元素操作
  • 鼠标、键盘和更改事件监听器
  • canvas2d
  • 本地存储
  • fetch & XMLHttpRequest
  • 样式 & 类
  • 历史 & 位置信息
  • WebGPU
  • 其他实用工具

查看文档这里

cargo add web

你好世界

让我们看看如何在控制台放置一些基本示例

use web::*;

fn main() {
    console_log("Hello, world!");
}
<html>
    <head>
        <meta charset="utf-8">
        <script src="https://unpkg.com/js-wasm/js-wasm.js"></script>
        <script type="application/wasm" src="helloworld.wasm"></script>
    </head>
    <body>
        Open my console.
    </body>
</html>

请记住为WebAssembly配置您的库 Cargo.toml

# add these lines for WebAssembly to end of your Cargo.toml

[lib]
crate-type =["cdylib"]

[profile.release]
lto = true
cargo build --target wasm32-unknown-unknown --release
cp target/wasm32-unknown-unknown/release/helloworld.wasm .
python3 -m http.server

# open https://127.0.0.1:8000 in browser
# right click, inspect, look at message in console

完整示例这里

更高级的东西?

让我们看看我们的蛇示例和它的关键功能使用

Screen Shot 2022-12-21 at 12 35 48 PM

在这里玩演示这里

canvas

此示例使用canvas

//get an element and get the 2D context for canvas
let screen = query_selector("#screen");
let width: f64 = get_property_f64(&screen, "width");
let height: f64 = get_property_f64(&screen, "height");
let ctx = CanvasContext::from_element(&screen);

...

//clear screen
self.ctx.clear_rect(
    0.0,
    0.0,
    self.canvas_width as f64,
    self.canvas_height as f64,
);

// iterate through all the cells of the screen and draw a rectangle
for (_id, (pos, color)) in &mut self.world.query::<(&Position, &Color)>() {
    self.ctx.set_fill_style(&color.0);
    self.ctx.fill_rect(
        (pos.0 * (self.canvas_width / MAP_WIDTH)) as f64,
        (pos.1 * (self.canvas_height / MAP_HEIGHT)) as f64,
        (self.canvas_width / MAP_WIDTH) as f64,
        (self.canvas_height / MAP_HEIGHT) as f64,
    );
}

请求动画帧

让我们看看如何运行游戏循环

fn game_loop() {
    // run game loop assuming 15 ms has passed
    match Game::instance().run(15.0) {
        Err(e) => console_error(&e.to_string()),
        _ => (),
    };
    // request next animation frame
    request_animation_frame(game_loop);
}

... 

// start the loop
request_animation_frame(game_loop);

事件

let body = query_selector("body");
element_add_key_down_listener(&body, |e| {
    Game::instance().key_down(e.key_code as u32);
});

异步 & 协程

此库支持异步和生成协程。考虑这个程序,它启动一个循环控制台日志并在屏幕上绘制随机方块。

use web::*;

// easily make your first function async
#[web::main]
async fn main() {
    let canvas = query_selector("#canvas");
    let ctx = CanvasContext::from_element(&canvas);

    // we can spawn concurrent operations
    coroutine(async {
        loop {
            console_log("⏰ tik");
            // hand async set_timeout
            sleep(1000).await;
            console_log("⏰ tok");
            sleep(1000).await;
        }
    });

    loop {
        // draw a random color rect
        ctx.set_fill_style(&format!(
            "rgb({}, {}, {})",
            random() * 255.0,
            random() * 255.0,
            random() * 255.0
        ));
        ctx.fill_rect(
            random() * 500.0,
            random() * 500.0,
            random() * 500.0,
            random() * 500.0,
        );
        // a more async way to write graphics code
        wait_til_animation_frame().await;
    }
}

许可证

本项目采用Apache许可证2.0版或MIT许可证。

任选其一。

贡献

除非你明确声明,否则你提交给 web 的任何有意贡献,根据Apache-2.0许可证的定义,都应以上述方式双重许可,不附加任何额外条款或条件。

依赖关系

~1MB
~20K SLoC