diff options
author | TrueDoctor <dennis@kobert.dev> | 2019-06-22 00:03:37 +0200 |
---|---|---|
committer | GitHub <noreply@github.com> | 2019-06-22 00:03:37 +0200 |
commit | 8b2040eb93e8e4355c8f357b775dedcffafff3dc (patch) | |
tree | 0b2c4724f0c75d09ee8b86f397bf6602a899e825 | |
parent | 9e9c1c822a64c0a65033b7eed07ea661a385cecc (diff) | |
parent | 5522731ec4de8741f923b339ca942f8aaff2a338 (diff) |
Merge pull request #39 from TrueDoctor/game_server_refactor
Game server refactor
-rw-r--r-- | webhogg/game_server/src/collide.rs | 28 | ||||
-rw-r--r-- | webhogg/game_server/src/main.rs | 2 | ||||
-rw-r--r-- | webhogg/game_server/src/math.rs (renamed from webhogg/game_server/src/maths.rs) | 39 | ||||
-rw-r--r-- | webhogg/game_server/src/webhogg_game.rs | 2 | ||||
-rw-r--r-- | webhogg/wasm/Cargo.toml | 14 | ||||
-rw-r--r-- | webhogg/wasm/index.html | 11 | ||||
-rw-r--r-- | webhogg/wasm/pkg/main.js | 38 | ||||
-rw-r--r-- | webhogg/wasm/pkg/worker-graphics.js | 1 | ||||
-rw-r--r-- | webhogg/wasm/pkg/worker.js | 17 | ||||
-rw-r--r-- | webhogg/wasm/src/context/graphics.rs | 81 | ||||
-rw-r--r-- | webhogg/wasm/src/context/logic.rs | 11 | ||||
-rw-r--r-- | webhogg/wasm/src/context/main.fs | 9 | ||||
-rw-r--r-- | webhogg/wasm/src/context/main.vs | 9 | ||||
-rw-r--r-- | webhogg/wasm/src/context/mod.rs | 26 | ||||
-rw-r--r-- | webhogg/wasm/src/context/shader.rs | 34 | ||||
-rw-r--r-- | webhogg/wasm/src/context/webgl.rs | 134 | ||||
-rw-r--r-- | webhogg/wasm/src/error.rs | 36 | ||||
-rw-r--r-- | webhogg/wasm/src/graphics.rs | 25 | ||||
-rw-r--r-- | webhogg/wasm/src/lib.rs | 13 | ||||
-rw-r--r-- | webhogg/wasm/src/logger.rs | 33 | ||||
-rw-r--r-- | webhogg/wasm/src/logic.rs | 19 |
21 files changed, 533 insertions, 49 deletions
diff --git a/webhogg/game_server/src/collide.rs b/webhogg/game_server/src/collide.rs index 16b5357..940cffd 100644 --- a/webhogg/game_server/src/collide.rs +++ b/webhogg/game_server/src/collide.rs @@ -1,4 +1,4 @@ -use crate::maths::{Vec2, AABox, RBox}; +use crate::math::{Vec2, AABox, RBox}; pub trait Collide<Rhs> { fn collides(&self, other: &Rhs) -> bool; @@ -12,7 +12,7 @@ impl Collide<Vec2> for Vec2 { impl Collide<Vec2> for AABox { fn collides(&self, other: &Vec2) -> bool { - self.pos < *other && other < &(self.pos + self.size) + self.pos < *other && other < &(self.pos + self.size) } } @@ -25,8 +25,8 @@ impl Collide<AABox> for AABox { impl Collide<Vec2> for RBox { fn collides(&self, other: &Vec2) -> bool { - let v1_diff = *other + self.v1 * (-self.v1.scalar(&(*other - self.pos)) / self.v1.distance2()); - let v2_diff = *other + self.v2 * (-self.v2.scalar(&(*other - self.pos)) / self.v2.distance2()); + let v1_diff = *other + self.v1 * (-self.v1.dot(&(*other - self.pos)) / self.v1.norm2()); + let v2_diff = *other + self.v2 * (-self.v2.dot(&(*other - self.pos)) / self.v2.norm2()); let v1_dist = ((v1_diff - self.pos) / self.v2).x; let v2_dist = ((v2_diff - self.pos) / self.v1).x; @@ -42,16 +42,16 @@ impl Collide<AABox> for RBox { let other_size = other.pos + other.size; // project points onto a orthogonal line - let v1_diff = other.pos + self.v1 * (-self.v1.scalar(&(other.pos - self.pos)) / self.v1.distance2()); - let v2_diff = other.pos + self.v2 * (-self.v2.scalar(&other.pos) / self.v2.distance2()); - let v1_diff_size = other_size + self.v1 * (-self.v1.scalar(&(other_size - self.pos)) / self.v1.distance2()); - let v2_diff_size = other_size + self.v2 * (-self.v2.scalar(&(other_size - self.pos)) / self.v2.distance2()); + let v1_diff = other.pos + self.v1 * (-self.v1.dot(&(other.pos - self.pos)) / self.v1.norm2()); + let v2_diff = other.pos + self.v2 * (-self.v2.dot(&other.pos) / self.v2.norm2()); + let v1_diff_size = other_size + self.v1 * (-self.v1.dot(&(other_size - self.pos)) / self.v1.norm2()); + let v2_diff_size = other_size + self.v2 * (-self.v2.dot(&(other_size - self.pos)) / self.v2.norm2()); - // calculate the distance - let v1_dist = ((v1_diff - self.pos) / self.v2); - let v2_dist = ((v2_diff - self.pos) / self.v1); - let v1_dist_size = ((v1_diff_size - self.pos) / self.v2); - let v2_dist_size = ((v2_diff_size - self.pos) / self.v1); + // calculate the norm + let v1_dist = (v1_diff - self.pos) / self.v2; + let v2_dist = (v2_diff - self.pos) / self.v1; + let v1_dist_size = (v1_diff_size - self.pos) / self.v2; + let v2_dist_size = (v2_diff_size - self.pos) / self.v1; let v1_dist = if v1_dist.x.is_finite() {v1_dist.x} else {v1_dist.y}; let v2_dist = if v2_dist.x.is_finite() {v2_dist.x} else {v2_dist.y}; @@ -150,7 +150,7 @@ impl<S, T: Collide<S>> Collide<S> for Vec<T> { } #[test] - fn test_collide_Rbox_dot() { + fn test_collide_rbox_dot() { let a = Vec2{x: 1.0, y: 1.0}; let b = Vec2{x: 1.0, y: 1.0}; let c = Vec2{x: 1.0, y: -1.0}; diff --git a/webhogg/game_server/src/main.rs b/webhogg/game_server/src/main.rs index cfd9787..2db757b 100644 --- a/webhogg/game_server/src/main.rs +++ b/webhogg/game_server/src/main.rs @@ -1,5 +1,5 @@ mod group; -mod maths; +mod math; mod scribble_group; mod webhogg_group; mod webhogg_game; diff --git a/webhogg/game_server/src/maths.rs b/webhogg/game_server/src/math.rs index b9303af..6d9f1e1 100644 --- a/webhogg/game_server/src/maths.rs +++ b/webhogg/game_server/src/math.rs @@ -101,20 +101,20 @@ impl std::cmp::PartialEq for Vec2 { impl std::cmp::Eq for Vec2 {} impl Vec2 { - pub fn distance(&self) -> f32 { - f32::sqrt(self.distance2()) + pub fn norm(&self) -> f32 { + f32::hypot(self.x, self.y) } - pub fn distance2(&self) -> f32 { - self.scalar(self) + pub fn norm2(&self) -> f32 { + self.dot(self) } - pub fn scalar(&self, other: &Vec2) -> f32 { + pub fn dot(&self, other: &Vec2) -> f32 { self.x * other.x + self.y * other.y } - pub fn norm(&self) -> Vec2 { - let len = self.distance(); + pub fn normalized(&self) -> Vec2 { + let len = self.norm(); Vec2 { x: self.x / len, y: self.y / len, @@ -149,7 +149,7 @@ impl std::ops::Sub<Vec2> for AABox { type Output = Self; fn sub(self, other: Vec2) -> Self { Self { - pos: self.pos + other, + pos: self.pos - other, size: self.size } } @@ -174,7 +174,7 @@ impl std::cmp::Eq for AABox {} pub struct RBox { /// origin pub pos: Vec2, - /// Vwctor1 + /// Vector1 pub v1: Vec2, /// Vector2 pub v2: Vec2, @@ -182,7 +182,7 @@ pub struct RBox { impl RBox { pub fn new(pos: Vec2, orientation: Vec2, width: f32) -> Self { - let scale = width / orientation.distance(); + let scale = width / orientation.norm(); let orth = Vec2 {x: orientation.x / scale, y: -orientation.y / scale}; Self { pos: pos, @@ -213,8 +213,8 @@ impl std::ops::Sub<Vec2> for RBox { type Output = Self; fn sub(self, other: Vec2) -> Self { Self { - pos: self.pos + other, - v1: self.v1 + other, + pos: self.pos - other, + v1: self.v1, v2: self.v2, } } @@ -230,7 +230,7 @@ impl std::cmp::PartialEq for RBox { fn eq(&self, other: &Self) -> bool { self.pos == other.pos && self.v1 == other.v1 - && self.v1 == self.v2 + && self.v2 == other.v2 } } @@ -273,7 +273,6 @@ mod tests { assert!(!(a > b)); } - #[test] fn test_add_vec2() { let a = Vec2{x: 1.0, y: 7.5}; @@ -301,25 +300,25 @@ mod tests { } #[test] - fn test_distance_vec2() { + fn test_norm_vec2() { let a = Vec2{x: 2.0, y: 2.0}; - assert!(f32::abs(a.distance() - 2.0) < 1e8); + assert!(f32::abs(a.norm() - 2.0) < 1e8); } #[test] - fn test_distance2_vec2() { + fn test_norm2_vec2() { let a = Vec2{x: 1.0, y: 2.0}; - assert!(f32::abs(a.distance2() - 5.0) < 1e8); + assert!(f32::abs(a.norm2() - 5.0) < 1e8); } #[test] - fn test_norm_vec2() { + fn test_normalized_vec2() { let a = Vec2{x: 2.0, y: -2.0}; let b = Vec2{x: std::f32::consts::FRAC_1_SQRT_2, y: -std::f32::consts::FRAC_1_SQRT_2}; - assert_eq!(a.norm(), b); + assert_eq!(a.normalized(), b); } #[test] diff --git a/webhogg/game_server/src/webhogg_game.rs b/webhogg/game_server/src/webhogg_game.rs index 7b94fcb..b3f35cc 100644 --- a/webhogg/game_server/src/webhogg_game.rs +++ b/webhogg/game_server/src/webhogg_game.rs @@ -1,4 +1,4 @@ -use crate::maths::Vec2; +use crate::math::Vec2; pub struct WebhoggPlayer { pos: Vec2, diff --git a/webhogg/wasm/Cargo.toml b/webhogg/wasm/Cargo.toml index 7978928..8f44389 100644 --- a/webhogg/wasm/Cargo.toml +++ b/webhogg/wasm/Cargo.toml @@ -16,3 +16,17 @@ lto = true [dependencies] wasm-bindgen = "0.2" +log = "0.4" +fern = "0.5" +js-sys = "0.3" + +[dependencies.web-sys] +version = "0.3" +features = [ + "OffscreenCanvas", + "WebGl2RenderingContext", + "WebGlShader", + "WebGlProgram", + "WebGlBuffer", + "WebGlVertexArrayObject" +] diff --git a/webhogg/wasm/index.html b/webhogg/wasm/index.html index eae1cc2..a992390 100644 --- a/webhogg/wasm/index.html +++ b/webhogg/wasm/index.html @@ -4,8 +4,15 @@ <meta charset='utf-8'/> <title>webhogg</title> </head> - <body> - <canvas id='c'>your browser is incompetent</canvas> + <body style='margin: 0; width: 100%; height: 100%; posisiton: fixed; display: block;'> + <canvas id='c' style='width: 100%; height: 100%; position: fixed; display: block; overflow: hidden;'> + your browser is incompetent + </canvas> + <script> + let canvas = document.getElementById('c'); + canvas.width = window.innerWidth; + canvas.height = window.innerHeight; + </script> <script src='pkg/main.js'></script> </body> </html> diff --git a/webhogg/wasm/pkg/main.js b/webhogg/wasm/pkg/main.js index eec740d..90b3956 100644 --- a/webhogg/wasm/pkg/main.js +++ b/webhogg/wasm/pkg/main.js @@ -1,9 +1,39 @@ +workers = []; + +function exit() { + for (var worker of workers) { + worker.terminate(); + } + console.clear(); +} + async function main() { let fetchingSource = fetch('bin/webhogg-wasm.wasm'); + + let canvasElement = document.getElementById('c'); + let offCanvas = canvasElement.transferControlToOffscreen(); + let fetchedSource = await fetchingSource; - let source = await fetchedSource.text(); - //alert(source) - let workerGraphics = new Worker('pkg/worker-graphics.js'); -} + source = await fetchedSource.arrayBuffer(); + const modules = [ + { type: 'graphics', + source: source, + canvas: offCanvas, + dt: 16 }, + { type: 'logic', + source: source, + canvas: [], + dt: 10000 }, + ]; + for (var module of modules) { + let worker = new Worker('pkg/worker.js'); + if (module.type === 'graphics') { + worker.postMessage(module, [module.canvas]); + } else { + worker.postMessage(module); + } + workers.push(worker); + } +} main(); diff --git a/webhogg/wasm/pkg/worker-graphics.js b/webhogg/wasm/pkg/worker-graphics.js deleted file mode 100644 index 8360014..0000000 --- a/webhogg/wasm/pkg/worker-graphics.js +++ /dev/null @@ -1 +0,0 @@ -console.log('lelel'); diff --git a/webhogg/wasm/pkg/worker.js b/webhogg/wasm/pkg/worker.js new file mode 100644 index 0000000..780ea6f --- /dev/null +++ b/webhogg/wasm/pkg/worker.js @@ -0,0 +1,17 @@ +let data = null; + +onmessage = function (e) { + data = e.data; + + importScripts('../bin/webhogg-wasm.js'); + wasm_bindgen(data.source).then(ctx => { + if (data.type === 'graphics') { + wasm_bindgen.start_graphics(data.canvas); + setInterval(wasm_bindgen.loop_graphics, data.dt); + } else if (data.type === 'logic') { + wasm_bindgen.start_logic(); + setInterval(wasm_bindgen.loop_logic, data.dt); + } + + }); +} diff --git a/webhogg/wasm/src/context/graphics.rs b/webhogg/wasm/src/context/graphics.rs new file mode 100644 index 0000000..6af35a2 --- /dev/null +++ b/webhogg/wasm/src/context/graphics.rs @@ -0,0 +1,81 @@ +use crate::error::WasmError; +use wasm_bindgen::JsCast; +use web_sys::WebGl2RenderingContext as GlContext; + +use super::webgl; +use super::webgl::{Color4, ShaderType, WebGl2}; +use super::shader::{MAIN_VERTEX_SHADER, MAIN_FRAGMENT_SHADER}; +use super::shader::ShaderProgram; + +pub struct GraphicsContext { + gl: WebGl2, + frame_nr: u64, + shader: ShaderProgram, + vao: webgl::WebGlVertexArrayObject, + buffer: webgl::WebGlBuffer, +} + +impl GraphicsContext { + pub fn from_canvas(canvas: web_sys::OffscreenCanvas) -> Result<Self, WasmError> { + let context = canvas.get_context("webgl2") + .map_err(|_| WasmError::WebGl2ContextCreation( + format!("context cration failed: getContext returned an exception")))? + .ok_or_else(|| WasmError::WebGl2ContextCreation( + format!("context cration failed: getContext returned nothing")))?; + let context = context + .dyn_into::<GlContext>() + .map_err(|_| WasmError::WebGl2ContextCreation( + format!("context object is not a context")))?; + + let gl = WebGl2::from_context(context); + let shader = ShaderProgram::from_sources(&gl, &[ + (ShaderType::Vertex, MAIN_VERTEX_SHADER.to_string()), + (ShaderType::Fragment, MAIN_FRAGMENT_SHADER.to_string()), + ])?; + + let vao = gl.create_vertex_array() + .map_err(|_| WasmError::WebGlBuffer( + format!("glGenVertexArrays failed")))?; + gl.bind_vertex_array(&vao); + + let buffer = gl.create_buffer() + .map_err(|_| WasmError::WebGlBuffer( + format!("glCreateBuffer failed")))?; + gl.bind_array_buffer(&buffer); + gl.array_buffer_data_f32(&[ + 0.0, 0.0, + 1.0, 0.0, + 0.0, 1.0, + 1.0, 1.0, + 0.0, 1.0, + 1.0, 0.0, + ]); + gl.enable_vertex_attrib_array(0); + + Ok(Self { + gl, frame_nr: 0, + shader, vao, buffer + }) + } + + pub fn update(&mut self) -> Result<(), WasmError> { + let light = 0.5; + let speed = 60.0; + + let a = (self.frame_nr as f32) / speed; + let a = f32::abs(f32::sin(a)); + let b = f32::abs(f32::cos(a)); + let (a, b) = (a * light, b * light); + + self.gl.set_viewport(); + self.gl.clear(&Color4::new(a, light - a, b, 1.0)); + + self.shader.run(&self.gl); + self.gl.vertex_attrib_f32_pointer(0, 2); + self.gl.draw_triangle_arrays(6); + + self.frame_nr += 1; + + Ok(()) + } +} diff --git a/webhogg/wasm/src/context/logic.rs b/webhogg/wasm/src/context/logic.rs new file mode 100644 index 0000000..71dfea4 --- /dev/null +++ b/webhogg/wasm/src/context/logic.rs @@ -0,0 +1,11 @@ +use crate::error::WasmError; + +pub struct LogicContext { +} + +impl LogicContext { + pub fn new() -> Result<Self, WasmError> { + Ok(Self { + }) + } +} diff --git a/webhogg/wasm/src/context/main.fs b/webhogg/wasm/src/context/main.fs new file mode 100644 index 0000000..971d2db --- /dev/null +++ b/webhogg/wasm/src/context/main.fs @@ -0,0 +1,9 @@ +#version 300 es + +precision highp float; + +out vec4 color; + +void main() { + color = vec4(0.0, 1.0, 1.0, 1.0); +} diff --git a/webhogg/wasm/src/context/main.vs b/webhogg/wasm/src/context/main.vs new file mode 100644 index 0000000..18df0a0 --- /dev/null +++ b/webhogg/wasm/src/context/main.vs @@ -0,0 +1,9 @@ +#version 300 es + +precision highp float; + +layout(location=0) in vec2 position; + +void main() { + gl_Position = vec4(position, 0.0, 0.0); +} diff --git a/webhogg/wasm/src/context/mod.rs b/webhogg/wasm/src/context/mod.rs new file mode 100644 index 0000000..09902ce --- /dev/null +++ b/webhogg/wasm/src/context/mod.rs @@ -0,0 +1,26 @@ +mod shader; +mod webgl; +pub mod graphics; +pub mod logic; + +use graphics::GraphicsContext; +use logic::LogicContext; + +static mut GTX: Option<GraphicsContext> = None; +static mut LTX: Option<LogicContext> = None; + +pub fn get_graphics() -> &'static mut GraphicsContext { + unsafe { GTX.as_mut().unwrap() } +} + +pub fn get_logic() -> &'static mut LogicContext { + unsafe { LTX.as_mut().unwrap() } +} + +pub fn set_graphics(gtx: GraphicsContext) { + unsafe { GTX = Some(gtx) } +} + +pub fn set_logic(ltx: LogicContext) { + unsafe { LTX = Some(ltx) } +} diff --git a/webhogg/wasm/src/context/shader.rs b/webhogg/wasm/src/context/shader.rs new file mode 100644 index 0000000..9ccb9fc --- /dev/null +++ b/webhogg/wasm/src/context/shader.rs @@ -0,0 +1,34 @@ +use crate::error::WasmError; +use super::webgl; +use super::webgl::{WebGl2, ShaderType}; + +pub const MAIN_VERTEX_SHADER: &str = include_str!("main.vs"); +pub const MAIN_FRAGMENT_SHADER: &str = include_str!("main.fs"); + +pub struct ShaderProgram { + program: webgl::WebGlProgram, +} + +impl ShaderProgram { + pub fn from_sources(gl: &WebGl2, sources: &[(ShaderType, String)]) -> Result<Self, WasmError> { + let program = gl.create_program() + .map_err(|_| WasmError::Shader(format!("glCreateProgram failed ({})", gl.get_error())))?; + for (shader_type, source) in sources { + let shader = gl.create_shader(shader_type) + .map_err(|_| WasmError::Shader(format!("glCreateShader failed ({})", gl.get_error())))?; + gl.shader_source(&shader, source); + gl.compile_shader(&shader) + .map_err(|e| WasmError::Shader(format!("compile error in {} shader: {}", shader_type, e)))?; + gl.attach_shader(&program, &shader) + } + gl.link_program(&program) + .map_err(|e| WasmError::Shader(format!("linker error in program: {}", e)))?; + Ok(Self { + program + }) + } + + pub fn run(&self, gl: &WebGl2) { + gl.use_program(&self.program) + } +} diff --git a/webhogg/wasm/src/context/webgl.rs b/webhogg/wasm/src/context/webgl.rs new file mode 100644 index 0000000..8883835 --- /dev/null +++ b/webhogg/wasm/src/context/webgl.rs @@ -0,0 +1,134 @@ +pub use web_sys::{ + WebGl2RenderingContext as GlContext, + WebGlProgram, WebGlShader, + WebGlBuffer, WebGlVertexArrayObject, +}; +use wasm_bindgen::prelude::*; +use std::fmt::Display; + +pub struct Color4(f32, f32, f32, f32); + +#[wasm_bindgen] +extern "C" { + #[wasm_bindgen(js_namespace=Float32Array, js_name=of, variadic)] + fn _create_f32_buffer(args: &[f32]) -> js_sys::Float32Array; +} + +#[derive(Debug)] +pub enum ShaderType { + Vertex, + Fragment, +} + +impl Display for ShaderType { + fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result { + write!(f, "{:?}", self) + } +} + +impl ShaderType { + pub fn to_id(&self) -> u32 { + match self { + ShaderType::Vertex => GlContext::VERTEX_SHADER, + ShaderType::Fragment => GlContext::FRAGMENT_SHADER, + } + } +} + +impl Color4 { + pub fn new(r: f32, g: f32, b: f32, a: f32) -> Color4 { + Color4(r, g, b, a) + } +} + +pub fn create_f32_buffer(buffer: &[f32]) -> js_sys::Float32Array { + _create_f32_buffer(buffer) +} + +pub struct WebGl2 { + gl: GlContext, +} + +impl WebGl2 { + pub fn from_context(context: GlContext) -> Self { + WebGl2 { + gl: context, + } + } + + pub fn create_program(&self) -> Result<WebGlProgram, ()> { + self.gl.create_program().ok_or(()) + } + + pub fn create_shader(&self, shader_type: &ShaderType) -> Result<WebGlShader, ()> { + self.gl.create_shader(shader_type.to_id()).ok_or(()) + } + + pub fn get_error(&self) -> u32 { self.gl.get_error() } + pub fn shader_source(&self, id: &WebGlShader, source: &str) { self.gl.shader_source(id, source) } + pub fn compile_shader(&self, id: &WebGlShader) -> Result<(), String> { + self.gl.compile_shader(id); + if self.gl.get_shader_parameter(id, GlContext::COMPILE_STATUS) == JsValue::FALSE { + Err(self.gl.get_shader_info_log(id) + .unwrap_or("/could not retrieve program information/".to_string())) + } else { Ok(()) } + } + pub fn link_program(&self, id: &WebGlProgram) -> Result<(), String> { + self.gl.link_program(id); + if self.gl.get_program_parameter(id, GlContext::LINK_STATUS) == JsValue::FALSE { + Err(self.gl.get_program_info_log(id) + .unwrap_or("/could not retrieve program information/".to_string())) + } else { Ok(()) } + } + pub fn attach_shader(&self, program: &WebGlProgram, shader: &WebGlShader) { + self.gl.attach_shader(program, shader) + } + + pub fn clear(&self, color: &Color4) { + self.gl.clear_color(color.0, color.1, color.2, color.3); + self.gl.clear(GlContext::COLOR_BUFFER_BIT); + } + + pub fn set_viewport(&self) { + self.gl.viewport(0, 0, self.gl.drawing_buffer_width(), self.gl.drawing_buffer_height()); + } + + pub fn create_buffer(&self) -> Result<WebGlBuffer, ()> { + self.gl.create_buffer().ok_or(()) + } + + pub fn bind_array_buffer(&self, buffer: &WebGlBuffer) { + self.gl.bind_buffer(GlContext::ARRAY_BUFFER, Some(buffer)) + } + pub fn unbind_array_buffer(&self) { self.gl.bind_buffer(GlContext::ARRAY_BUFFER, None) } + + pub fn array_buffer_data_f32(&self, data: &[f32]) { + self.gl.buffer_data_with_opt_array_buffer( + GlContext::ARRAY_BUFFER, + Some(&create_f32_buffer(data).buffer()), + GlContext::STATIC_DRAW) + } + + pub fn create_vertex_array(&self) -> Result<WebGlVertexArrayObject, ()> { + self.gl.create_vertex_array().ok_or(()) + } + pub fn bind_vertex_array(&self, array: &WebGlVertexArrayObject) { + self.gl.bind_vertex_array(Some(array)) + } + pub fn unbind_vertex_array(&self) { self.gl.bind_vertex_array(None) } + pub fn vertex_attrib_f32_pointer(&self, location: u32, dim: i32) { + self.gl.vertex_attrib_pointer_with_i32(location, dim, GlContext::FLOAT, false, 0, 0) + } + + pub fn draw_triangle_arrays(&self, count: i32) { + self.gl.draw_arrays(GlContext::TRIANGLES, 0, count) + } + + pub fn enable_vertex_attrib_array(&self, location: u32) { + self.gl.enable_vertex_attrib_array(location) + } + + pub fn use_program(&self, program: &WebGlProgram) { + self.gl.use_program(Some(program)) + } +} diff --git a/webhogg/wasm/src/error.rs b/webhogg/wasm/src/error.rs new file mode 100644 index 0000000..fbb6bf8 --- /dev/null +++ b/webhogg/wasm/src/error.rs @@ -0,0 +1,36 @@ +use std::error::Error; + +#[derive(Debug)] +pub enum WasmError { + WebGl2ContextCreation(String), + Shader(String), + WebGlBuffer(String), +} + +impl std::fmt::Display for WasmError { + fn fmt(&self, f: &mut std::fmt::Formatter) -> Result<(), std::fmt::Error> { + write!(f, "{}: {}", self.name(), self.description()) + } +} + +impl Error for WasmError { + fn description(&self) -> &str { + match self { + WasmError::WebGl2ContextCreation(msg) => msg, + WasmError::Shader(msg) => msg, + WasmError::WebGlBuffer(msg) => msg, + } + } + + fn source(&self) -> Option<&'static dyn Error> { None } +} + +impl WasmError { + pub fn name(&self) -> &str { + match self { + WasmError::WebGl2ContextCreation(_) => "WebGl2ContextCreationError", + WasmError::Shader(_) => "ShaderError", + WasmError::WebGlBuffer(_) => "WebGlBufferError", + } + } +} diff --git a/webhogg/wasm/src/graphics.rs b/webhogg/wasm/src/graphics.rs new file mode 100644 index 0000000..219c652 --- /dev/null +++ b/webhogg/wasm/src/graphics.rs @@ -0,0 +1,25 @@ +use wasm_bindgen::prelude::*; +use log::*; +use crate::*; + +#[wasm_bindgen] +pub fn start_graphics(canvas: web_sys::OffscreenCanvas) { + logger::init_logger(); + info!("graphics: wasm entry-point reached"); + //debug!("js value is?: undefined: {}", canvas.is_undefined()); + + match context::graphics::GraphicsContext::from_canvas(canvas) { + Ok(ctx) => context::set_graphics(ctx), + Err(e) => { + error!("graphics {}", e); + panic!() + } + } +} + +#[wasm_bindgen] +pub fn loop_graphics() { + context::get_graphics().update() + .map_err(|e| error!("gaphics loop {}", e)) + .unwrap(); +} diff --git a/webhogg/wasm/src/lib.rs b/webhogg/wasm/src/lib.rs index cb334fb..7aa4e86 100644 --- a/webhogg/wasm/src/lib.rs +++ b/webhogg/wasm/src/lib.rs @@ -1,8 +1,9 @@ -use wasm_bindgen::prelude::*; +mod logger; +pub mod error; +pub mod context; -#[wasm_bindgen] -pub fn enrty() { -} +pub mod logic; +pub mod graphics; -fn main() { -} +pub use logic::*; +pub use graphics::*; diff --git a/webhogg/wasm/src/logger.rs b/webhogg/wasm/src/logger.rs new file mode 100644 index 0000000..96c4cbf --- /dev/null +++ b/webhogg/wasm/src/logger.rs @@ -0,0 +1,33 @@ +use wasm_bindgen::prelude::*; + +#[wasm_bindgen] +extern "C" { + #[wasm_bindgen(js_namespace=console, js_name=log)] + fn __console_log_colored2(f: &str, c1: &str, c2: &str); +} + +fn log(rec: &log::Record) { + __console_log_colored2(&format!("{}", rec.args()), + &format!("color: {}", match rec.level() { + log::Level::Trace => "violet", + log::Level::Debug => "blue", + log::Level::Info => "green", + log::Level::Warn => "orange", + log::Level::Error => "red" + }), ""); +} + +pub fn init_logger() { + fern::Dispatch::new().format(|out, message, record|{ + out.finish(format_args!( + "%c{}%c {} > {}", + record.level(), + record.target(), + message + ) + ) + }) + .level(log::LevelFilter::Debug) + .chain(fern::Output::call(log)) + .apply().unwrap(); +} diff --git a/webhogg/wasm/src/logic.rs b/webhogg/wasm/src/logic.rs new file mode 100644 index 0000000..14272d9 --- /dev/null +++ b/webhogg/wasm/src/logic.rs @@ -0,0 +1,19 @@ +use wasm_bindgen::prelude::*; +use log::*; +use crate::*; + +#[wasm_bindgen] +pub fn start_logic() { + logger::init_logger(); + info!("logic: wasm entry-point reached"); + + match context::logic::LogicContext::new() { + Ok(ctx) => context::set_logic(ctx), + Err(e) => error!("logic {}", e) + } +} + +#[wasm_bindgen] +pub fn loop_logic() { + debug!("logic: loopin'"); +} |