summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorTrueDoctor <dennis@kobert.dev>2019-06-22 00:03:37 +0200
committerGitHub <noreply@github.com>2019-06-22 00:03:37 +0200
commit8b2040eb93e8e4355c8f357b775dedcffafff3dc (patch)
tree0b2c4724f0c75d09ee8b86f397bf6602a899e825
parent9e9c1c822a64c0a65033b7eed07ea661a385cecc (diff)
parent5522731ec4de8741f923b339ca942f8aaff2a338 (diff)
Merge pull request #39 from TrueDoctor/game_server_refactor
Game server refactor
-rw-r--r--webhogg/game_server/src/collide.rs28
-rw-r--r--webhogg/game_server/src/main.rs2
-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.rs2
-rw-r--r--webhogg/wasm/Cargo.toml14
-rw-r--r--webhogg/wasm/index.html11
-rw-r--r--webhogg/wasm/pkg/main.js38
-rw-r--r--webhogg/wasm/pkg/worker-graphics.js1
-rw-r--r--webhogg/wasm/pkg/worker.js17
-rw-r--r--webhogg/wasm/src/context/graphics.rs81
-rw-r--r--webhogg/wasm/src/context/logic.rs11
-rw-r--r--webhogg/wasm/src/context/main.fs9
-rw-r--r--webhogg/wasm/src/context/main.vs9
-rw-r--r--webhogg/wasm/src/context/mod.rs26
-rw-r--r--webhogg/wasm/src/context/shader.rs34
-rw-r--r--webhogg/wasm/src/context/webgl.rs134
-rw-r--r--webhogg/wasm/src/error.rs36
-rw-r--r--webhogg/wasm/src/graphics.rs25
-rw-r--r--webhogg/wasm/src/lib.rs13
-rw-r--r--webhogg/wasm/src/logger.rs33
-rw-r--r--webhogg/wasm/src/logic.rs19
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'");
+}