2020-02-09 16:30:51 +00:00
|
|
|
#![feature(impl_trait_in_bindings)]
|
|
|
|
|
|
|
|
mod camera;
|
2019-03-07 00:33:46 +00:00
|
|
|
mod demo;
|
2019-03-08 17:29:21 +00:00
|
|
|
mod demos;
|
2019-06-25 17:44:37 +00:00
|
|
|
mod hitable;
|
2020-02-09 16:30:51 +00:00
|
|
|
mod render;
|
2019-06-25 17:44:37 +00:00
|
|
|
mod shapes;
|
2020-02-09 16:30:51 +00:00
|
|
|
mod types;
|
2019-03-07 00:33:46 +00:00
|
|
|
|
|
|
|
use demo::Demo;
|
|
|
|
use sdl2::{
|
|
|
|
event::{Event, WindowEvent},
|
|
|
|
keyboard::Keycode,
|
|
|
|
pixels::PixelFormatEnum,
|
|
|
|
};
|
|
|
|
|
2020-02-09 16:30:51 +00:00
|
|
|
const NUM_SAMPLES: u8 = 10;
|
|
|
|
|
2019-03-07 00:33:46 +00:00
|
|
|
fn main() -> Result<(), String> {
|
|
|
|
let sdl_ctx = sdl2::init()?;
|
|
|
|
let video_subsys = sdl_ctx.video()?;
|
|
|
|
|
2019-06-25 17:44:37 +00:00
|
|
|
let (mut width, mut height): (usize, usize) = (1200, 600);
|
2019-03-07 00:33:46 +00:00
|
|
|
|
2019-06-25 17:44:37 +00:00
|
|
|
let window = video_subsys
|
2019-03-07 00:33:46 +00:00
|
|
|
.window("Ray tracing in a weekend", width as u32, height as u32)
|
|
|
|
.position_centered()
|
|
|
|
.build()
|
|
|
|
.map_err(|e| e.to_string())?;
|
|
|
|
|
|
|
|
let mut event_pump = sdl_ctx.event_pump()?;
|
|
|
|
|
|
|
|
let mut canvas = window
|
|
|
|
.into_canvas()
|
|
|
|
.target_texture()
|
|
|
|
.build()
|
|
|
|
.map_err(|e| e.to_string())?;
|
|
|
|
|
2019-06-24 16:40:45 +00:00
|
|
|
// RGBA framebuffer
|
2019-03-07 00:33:46 +00:00
|
|
|
let mut buffer = vec![0; height * width * 4];
|
|
|
|
|
|
|
|
let texture_creator = canvas.texture_creator();
|
|
|
|
let mut texture = texture_creator
|
2019-03-08 17:29:21 +00:00
|
|
|
.create_texture_static(PixelFormatEnum::BGR888, width as u32, height as u32)
|
2019-03-07 00:33:46 +00:00
|
|
|
.map_err(|e| e.to_string())?;
|
|
|
|
|
2019-03-08 17:29:21 +00:00
|
|
|
//println!("{:?} {:?} {:?}", texture.query(), texture.color_mod(), texture.alpha_mod());
|
2019-06-25 17:44:37 +00:00
|
|
|
|
2020-02-09 16:30:51 +00:00
|
|
|
let active_demo: impl Demo = demos::SurfaceNormalSphere;
|
2019-06-25 17:44:37 +00:00
|
|
|
// TODO: Should update when window is unfocus since the project window retains
|
|
|
|
// data from overlapped window
|
|
|
|
// TODO: Maybe consider using condition variable to make loop {} not run at full
|
|
|
|
// speed at all times pinning a core at 100%
|
2019-06-24 16:40:45 +00:00
|
|
|
let mut should_update = true;
|
2019-03-07 00:33:46 +00:00
|
|
|
loop {
|
|
|
|
for event in event_pump.poll_iter() {
|
|
|
|
match event {
|
|
|
|
Event::Quit { .. }
|
|
|
|
| Event::KeyDown {
|
|
|
|
keycode: Some(Keycode::Escape),
|
|
|
|
..
|
|
|
|
} => return Ok(()),
|
2019-06-25 17:44:37 +00:00
|
|
|
Event::KeyUp {
|
2019-03-07 00:33:46 +00:00
|
|
|
keycode: Some(Keycode::S),
|
|
|
|
..
|
2019-03-08 17:29:21 +00:00
|
|
|
} => active_demo.save_as_ppm(&buffer, width, height),
|
2019-03-07 00:33:46 +00:00
|
|
|
Event::Window {
|
|
|
|
win_event: WindowEvent::Resized(w, h),
|
|
|
|
..
|
|
|
|
} => {
|
|
|
|
width = w as usize;
|
|
|
|
height = h as usize;
|
|
|
|
buffer.resize(width * height * 4, 0);
|
|
|
|
texture = texture_creator
|
2019-03-08 17:29:21 +00:00
|
|
|
.create_texture_static(PixelFormatEnum::BGR888, width as u32, height as u32)
|
2019-03-07 00:33:46 +00:00
|
|
|
.expect("error in resizing texture");
|
2019-06-24 16:40:45 +00:00
|
|
|
should_update = true;
|
2019-03-07 00:33:46 +00:00
|
|
|
}
|
|
|
|
_ => {}
|
|
|
|
};
|
|
|
|
}
|
2019-06-24 16:40:45 +00:00
|
|
|
if should_update {
|
2020-02-09 16:30:51 +00:00
|
|
|
active_demo.render(&mut buffer, width, height, NUM_SAMPLES);
|
2019-06-24 16:40:45 +00:00
|
|
|
texture.update(None, &buffer, width * 4);
|
|
|
|
canvas.copy(&texture, None, None);
|
|
|
|
canvas.present();
|
|
|
|
should_update = false;
|
|
|
|
}
|
2019-03-07 00:33:46 +00:00
|
|
|
}
|
|
|
|
}
|