Compare commits
6 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 361ff95f7f | |||
| f3fe1b783b | |||
| 570cee0f62 | |||
| 5b6fd34a6f | |||
| 973fa08ac9 | |||
| 4a1f565f72 |
@@ -1,28 +1,4 @@
|
|||||||
[target.wasm32-unknown-emscripten]
|
[target.wasm32-unknown-emscripten]
|
||||||
rustflags = [
|
rustflags = [
|
||||||
"-C",
|
"--use-preload-plugins --preload-file assets -s USE_SDL=2 -s USE_SDL_IMAGE=2 -s ASSERTIONS=1",
|
||||||
"link-arg=-s",
|
|
||||||
"-C",
|
|
||||||
"link-arg=FULL_ES2",
|
|
||||||
"-C",
|
|
||||||
"link-arg=-s",
|
|
||||||
"-C",
|
|
||||||
"link-arg=FULL_ES3",
|
|
||||||
"-C",
|
|
||||||
"link-arg=-s",
|
|
||||||
"-C",
|
|
||||||
"link-arg=USE_SDL=2",
|
|
||||||
"-C",
|
|
||||||
"link-arg=-s",
|
|
||||||
"-C",
|
|
||||||
"link-arg=MAX_WEBGL_VERSION=2 ",
|
|
||||||
"-C",
|
|
||||||
"link-arg=-s",
|
|
||||||
"-C",
|
|
||||||
"link-arg=MIN_WEBGL_VERSION=2",
|
|
||||||
"-C",
|
|
||||||
"link-arg=-s",
|
|
||||||
"-C",
|
|
||||||
"link-arg=ERROR_ON_UNDEFINED_SYMBOLS=0", # for ignoring some egl symbols. needed for wgpu
|
|
||||||
|
|
||||||
]
|
]
|
||||||
58
README.md
@@ -2,11 +2,65 @@
|
|||||||
|
|
||||||
If the title doesn't clue you in, I'm remaking Pac-Man with SDL and Rust.
|
If the title doesn't clue you in, I'm remaking Pac-Man with SDL and Rust.
|
||||||
|
|
||||||
The project is *extremely* early in development, but check back in a week, and maybe I'll have something cool to look
|
The project is _extremely_ early in development, but check back in a week, and maybe I'll have something cool to look
|
||||||
at.
|
at.
|
||||||
|
|
||||||
## Feature Targets
|
## Feature Targets
|
||||||
|
|
||||||
- Near-perfect replication of logic, scoring, graphics, sound, and behaviors.
|
- Near-perfect replication of logic, scoring, graphics, sound, and behaviors.
|
||||||
- Written in Rust, buildable on Windows, Linux, Mac and WebAssembly.
|
- Written in Rust, buildable on Windows, Linux, Mac and WebAssembly.
|
||||||
- Online demo, playable in a browser.
|
- Online demo, playable in a browser.
|
||||||
|
- Automatic build system, with releases for Windows, Linux, and Mac & Web-Assembly.
|
||||||
|
- Debug tooling
|
||||||
|
- Game state visualization
|
||||||
|
- Game speed controls + pausing
|
||||||
|
- Log tracing
|
||||||
|
- Performance details
|
||||||
|
|
||||||
|
## Experimental Ideas
|
||||||
|
|
||||||
|
- Perfected Ghost Algorithms
|
||||||
|
- More than 4 ghosts
|
||||||
|
- Custom Level Generation
|
||||||
|
- Multi-map tunnelling
|
||||||
|
|
||||||
|
## Installation
|
||||||
|
|
||||||
|
Besides SDL2, the following extensions are required: Image, Mixer, and TTF.
|
||||||
|
|
||||||
|
### Ubuntu
|
||||||
|
|
||||||
|
On Ubuntu, you can install the required packages with the following command:
|
||||||
|
|
||||||
|
```
|
||||||
|
sudo apt install libsdl2-dev libsdl2-image-dev libsdl2-mixer-dev libsdl2-ttf-dev
|
||||||
|
```
|
||||||
|
|
||||||
|
### Windows
|
||||||
|
|
||||||
|
On Windows, installation requires either building from source (not covered), or downloading the pre-built binaries.
|
||||||
|
|
||||||
|
The latest releases can be found here:
|
||||||
|
|
||||||
|
- [SDL2](https://github.com/libsdl-org/SDL/releases/latest/)
|
||||||
|
- [SDL2_image](https://github.com/libsdl-org/SDL_image/releases/latest/)
|
||||||
|
- [SDL2_mixer](https://github.com/libsdl-org/SDL_mixer/releases/latest/)
|
||||||
|
- [SDL2_ttf](https://github.com/libsdl-org/SDL_ttf/releases/latest/)
|
||||||
|
|
||||||
|
Download each for your architecture, and locate the appropriately named DLL within. Move said DLL to root of this project.
|
||||||
|
|
||||||
|
## Building
|
||||||
|
|
||||||
|
To build the project, run the following command:
|
||||||
|
|
||||||
|
```
|
||||||
|
cargo build
|
||||||
|
```
|
||||||
|
|
||||||
|
During development, you can easily run the project with:
|
||||||
|
|
||||||
|
```
|
||||||
|
cargo run
|
||||||
|
cargo run -q # Quiet mode, no logging
|
||||||
|
cargo run --release # Release mode, optimized
|
||||||
|
```
|
||||||
BIN
SDL2_image.dll
BIN
SDL2_mixer.dll
BIN
SDL2_ttf.dll
BIN
assets/24/energizer.png
Normal file
|
After Width: | Height: | Size: 174 B |
BIN
assets/24/pellet.png
Normal file
|
After Width: | Height: | Size: 158 B |
BIN
assets/32/fruit.png
Normal file
|
After Width: | Height: | Size: 1.5 KiB |
BIN
assets/32/game_over.png
Normal file
|
After Width: | Height: | Size: 1.2 KiB |
BIN
assets/32/ghost_body.png
Normal file
|
After Width: | Height: | Size: 528 B |
BIN
assets/32/ghost_eyes.png
Normal file
|
After Width: | Height: | Size: 394 B |
BIN
assets/32/life.png
Normal file
|
After Width: | Height: | Size: 228 B |
BIN
assets/32/pacman.png
Normal file
|
After Width: | Height: | Size: 370 B |
BIN
assets/door.png
Normal file
|
After Width: | Height: | Size: 90 B |
7
build.ps1
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
& cargo build --target=wasm32-unknown-emscripten --release
|
||||||
|
|
||||||
|
mkdir -p dist -Force
|
||||||
|
|
||||||
|
cp ./target/wasm32-unknown-emscripten/release/Pac_Man.wasm ./dist
|
||||||
|
cp ./target/wasm32-unknown-emscripten/release/Pac-Man.js ./dist
|
||||||
|
cp index.html dist
|
||||||
4
build.sh
@@ -5,6 +5,6 @@ cargo build --target=wasm32-unknown-emscripten --release
|
|||||||
|
|
||||||
mkdir -p dist
|
mkdir -p dist
|
||||||
|
|
||||||
cp target/wasm32-unknown-emscripten/release/rust_sdl2_wasm.wasm dist
|
cp target/wasm32-unknown-emscripten/release/Pac_Man.wasm dist
|
||||||
cp target/wasm32-unknown-emscripten/release/rust-sdl2-wasm.js dist
|
cp target/wasm32-unknown-emscripten/release/Pac-Man.js dist
|
||||||
cp index.html dist
|
cp index.html dist
|
||||||
21
index.html
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1.0"/>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<canvas id="canvas"></canvas>
|
||||||
|
<script type="text/javascript">
|
||||||
|
let Module = {
|
||||||
|
canvas: (function () {
|
||||||
|
// this is how we provide a canvas to our sdl2
|
||||||
|
return document.getElementById("canvas");
|
||||||
|
})(),
|
||||||
|
preRun: [function () {
|
||||||
|
ENV.RUST_LOG = "info,wgpu=warn"
|
||||||
|
}]
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
<script src="Pac-Man.js"></script>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
44
src/emscripten.rs
Normal file
@@ -0,0 +1,44 @@
|
|||||||
|
// taken from https://github.com/Gigoteur/PX8/blob/master/src/px8/emscripten.rs
|
||||||
|
#[cfg(target_os = "emscripten")]
|
||||||
|
pub mod emscripten {
|
||||||
|
use std::cell::RefCell;
|
||||||
|
use std::ptr::null_mut;
|
||||||
|
use std::os::raw::{c_int, c_void, c_char, c_float};
|
||||||
|
use std::ffi::{CStr, CString};
|
||||||
|
|
||||||
|
#[allow(non_camel_case_types)]
|
||||||
|
type em_callback_func = unsafe extern "C" fn();
|
||||||
|
|
||||||
|
extern "C" {
|
||||||
|
// void emscripten_set_main_loop(em_callback_func func, int fps, int simulate_infinite_loop)
|
||||||
|
pub fn emscripten_set_main_loop(func: em_callback_func,
|
||||||
|
fps: c_int,
|
||||||
|
simulate_infinite_loop: c_int);
|
||||||
|
|
||||||
|
pub fn emscripten_cancel_main_loop();
|
||||||
|
pub fn emscripten_pause_main_loop();
|
||||||
|
pub fn emscripten_get_now() -> c_float;
|
||||||
|
}
|
||||||
|
|
||||||
|
thread_local!(static MAIN_LOOP_CALLBACK: RefCell<*mut c_void> = RefCell::new(null_mut()));
|
||||||
|
|
||||||
|
pub fn set_main_loop_callback<F>(callback: F)
|
||||||
|
where F: FnMut()
|
||||||
|
{
|
||||||
|
MAIN_LOOP_CALLBACK
|
||||||
|
.with(|log| { *log.borrow_mut() = &callback as *const _ as *mut c_void; });
|
||||||
|
|
||||||
|
unsafe {
|
||||||
|
emscripten_set_main_loop(wrapper::<F>, -1, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
unsafe extern "C" fn wrapper<F>()
|
||||||
|
where F: FnMut()
|
||||||
|
{
|
||||||
|
MAIN_LOOP_CALLBACK.with(|z| {
|
||||||
|
let closure = *z.borrow_mut() as *mut F;
|
||||||
|
(*closure)();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
87
src/main.rs
@@ -1,45 +1,96 @@
|
|||||||
use sdl2::pixels::Color;
|
use crate::constants::{WINDOW_HEIGHT, WINDOW_WIDTH};
|
||||||
use sdl2::event::Event;
|
use sdl2::event::{Event, WindowEvent};
|
||||||
use sdl2::image::LoadTexture;
|
use sdl2::image::LoadTexture;
|
||||||
use sdl2::keyboard::{Keycode, Mod};
|
use sdl2::keyboard::Keycode;
|
||||||
|
use sdl2::pixels::Color;
|
||||||
|
use sdl2::render::{Texture, Canvas};
|
||||||
use std::time::Duration;
|
use std::time::Duration;
|
||||||
use crate::constants::{WINDOW_WIDTH, WINDOW_HEIGHT};
|
|
||||||
|
|
||||||
mod constants;
|
#[cfg(target_os = "emscripten")]
|
||||||
|
pub mod emscripten;
|
||||||
|
|
||||||
mod board;
|
mod board;
|
||||||
|
mod constants;
|
||||||
mod game;
|
mod game;
|
||||||
|
|
||||||
|
fn redraw(canvas: &mut Canvas<sdl2::video::Window>, tex: &Texture, i: u8) {
|
||||||
|
canvas.set_draw_color(Color::RGB(i, i, i));
|
||||||
|
canvas.clear();
|
||||||
|
canvas
|
||||||
|
.copy(tex, None, None)
|
||||||
|
.expect("Could not render texture on canvas");
|
||||||
|
}
|
||||||
|
|
||||||
pub fn main() {
|
pub fn main() {
|
||||||
let sdl_context = sdl2::init().unwrap();
|
let sdl_context = sdl2::init().unwrap();
|
||||||
let video_subsystem = sdl_context.video().unwrap();
|
let video_subsystem = sdl_context.video().unwrap();
|
||||||
|
|
||||||
let window = video_subsystem.window("Pac-Man", WINDOW_WIDTH, WINDOW_HEIGHT)
|
let window = video_subsystem
|
||||||
|
.window("Pac-Man", WINDOW_WIDTH, WINDOW_HEIGHT)
|
||||||
.position_centered()
|
.position_centered()
|
||||||
|
.resizable()
|
||||||
.build()
|
.build()
|
||||||
.expect("Could not initialize window");
|
.expect("Could not initialize window");
|
||||||
|
|
||||||
let mut canvas = window.into_canvas().build().expect("Could not build canvas");
|
let mut canvas = window
|
||||||
let texture_creator= canvas.texture_creator();
|
.into_canvas()
|
||||||
|
.build()
|
||||||
|
.expect("Could not build canvas");
|
||||||
|
let texture_creator = canvas.texture_creator();
|
||||||
|
|
||||||
let map_texture = texture_creator.load_texture("assets/map.png").expect("Could not load pacman texture");
|
let map_texture = texture_creator
|
||||||
canvas.copy(&map_texture, None, None).expect("Could not render texture on canvas");
|
.load_texture("assets/map.png")
|
||||||
|
.expect("Could not load pacman texture");
|
||||||
|
|
||||||
let mut event_pump = sdl_context.event_pump().expect("Could not get SDL EventPump");
|
canvas
|
||||||
'main: loop {
|
.copy(&map_texture, None, None)
|
||||||
|
.expect("Could not render texture on canvas");
|
||||||
|
|
||||||
|
let mut i = 0u8;
|
||||||
|
|
||||||
|
let mut event_pump = sdl_context
|
||||||
|
.event_pump()
|
||||||
|
.expect("Could not get SDL EventPump");
|
||||||
|
|
||||||
|
let mut main_loop = || {
|
||||||
for event in event_pump.poll_iter() {
|
for event in event_pump.poll_iter() {
|
||||||
match event {
|
match event {
|
||||||
Event::Quit { .. } |
|
// Handle quitting keys or window close
|
||||||
Event::KeyDown { keycode: Some(Keycode::Q), .. } => {
|
Event::Quit { .. }
|
||||||
break 'main;
|
| Event::KeyDown {
|
||||||
}
|
keycode: Some(Keycode::Escape) | Some(Keycode::Q),
|
||||||
event @ Event::KeyDown { .. } => {
|
..
|
||||||
|
} => return false,
|
||||||
|
event @ Event::KeyDown { .. } => {
|
||||||
println!("{:?}", event);
|
println!("{:?}", event);
|
||||||
},
|
},
|
||||||
|
Event::Window { win_event, .. } => {
|
||||||
|
if let WindowEvent::Resized(width, height) = win_event {
|
||||||
|
i = i.wrapping_add(1);
|
||||||
|
|
||||||
|
canvas.set_logical_size(width as u32, height as u32).unwrap();
|
||||||
|
redraw(&mut canvas, &map_texture, i);
|
||||||
|
}
|
||||||
|
},
|
||||||
_ => {}
|
_ => {}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
canvas.present();
|
canvas.present();
|
||||||
::std::thread::sleep(Duration::from_millis(10));
|
::std::thread::sleep(Duration::from_millis(10));
|
||||||
|
true
|
||||||
|
};
|
||||||
|
|
||||||
|
#[cfg(target_os = "emscripten")]
|
||||||
|
use emscripten::emscripten;
|
||||||
|
|
||||||
|
#[cfg(target_os = "emscripten")]
|
||||||
|
emscripten::set_main_loop_callback(main_loop);
|
||||||
|
|
||||||
|
#[cfg(not(target_os = "emscripten"))]
|
||||||
|
loop {
|
||||||
|
if !main_loop() {
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||