2023-10-19 14:56:39 -07:00
use std ::time ::Instant ;
2023-10-04 15:58:02 -07:00
use physics ::{ InputInstruction , PhysicsInstruction } ;
2023-09-19 17:53:29 -07:00
use instruction ::{ TimedInstruction , InstructionConsumer } ;
2023-08-29 18:20:58 -07:00
2023-10-05 20:32:25 -07:00
mod bvh ;
2023-10-05 20:32:02 -07:00
mod aabb ;
2023-09-22 15:21:13 -07:00
mod model ;
2023-09-27 02:12:20 -07:00
mod model_graphics ;
2023-09-22 15:19:44 -07:00
mod zeroes ;
2023-10-04 15:58:02 -07:00
mod worker ;
2023-10-18 17:17:21 -07:00
mod integer ;
2023-10-04 15:58:02 -07:00
mod physics ;
2023-10-18 17:17:21 -07:00
mod graphics ;
2023-10-09 17:09:24 -07:00
mod settings ;
2023-09-22 15:19:44 -07:00
mod framework ;
2023-09-22 19:41:27 -07:00
mod primitives ;
2023-09-22 15:19:44 -07:00
mod instruction ;
mod load_roblox ;
2023-10-19 18:42:00 -07:00
mod render_thread ;
2023-10-04 20:04:04 -07:00
2023-10-03 19:43:41 -07:00
pub struct GlobalState {
start_time : std ::time ::Instant ,
manual_mouse_lock :bool ,
2023-10-19 14:56:39 -07:00
mouse :std ::sync ::Arc < std ::sync ::Mutex < physics ::MouseState > > ,
2023-10-09 19:44:49 -07:00
user_settings :settings ::UserSettings ,
2023-10-19 14:56:39 -07:00
//Ideally the graphics thread worker description is:
/*
WorkerDescription {
input :Immediate ,
output :Realtime ( PoolOrdering ::Ordered ( 3 ) ) ,
}
* /
//up to three frames in flight, dropping new frame requests when all three are busy, and dropping output frames when one renders out of order
graphics_thread :worker ::INWorker < graphics ::GraphicsInstruction > ,
physics_thread :worker ::QNWorker < TimedInstruction < InputInstruction > > ,
2023-10-03 19:43:41 -07:00
}
impl framework ::Example for GlobalState {
2023-09-06 14:39:44 -07:00
fn optional_features ( ) -> wgpu ::Features {
wgpu ::Features ::TEXTURE_COMPRESSION_ASTC
| wgpu ::Features ::TEXTURE_COMPRESSION_ETC2
2023-09-28 10:58:51 -07:00
}
fn required_features ( ) -> wgpu ::Features {
wgpu ::Features ::TEXTURE_COMPRESSION_BC
2023-09-06 14:39:44 -07:00
}
2023-09-21 11:55:31 -07:00
fn required_limits ( ) -> wgpu ::Limits {
wgpu ::Limits ::default ( ) //framework.rs was using goofy limits that caused me a multi-day headache
}
2023-09-06 14:39:44 -07:00
fn init (
config : & wgpu ::SurfaceConfiguration ,
_adapter : & wgpu ::Adapter ,
device : & wgpu ::Device ,
queue : & wgpu ::Queue ,
) -> Self {
2023-10-09 19:44:49 -07:00
//wee
let user_settings = settings ::read_user_settings ( ) ;
2023-09-28 19:10:22 -07:00
let mut indexed_models = Vec ::new ( ) ;
2023-09-27 02:12:20 -07:00
indexed_models . append ( & mut model ::generate_indexed_model_list_from_obj ( obj ::ObjData ::load_buf ( & include_bytes! ( " ../models/teslacyberv3.0.obj " ) [ .. ] ) . unwrap ( ) , glam ::Vec4 ::ONE ) ) ;
2023-09-30 16:18:23 -07:00
indexed_models . push ( primitives ::unit_sphere ( ) ) ;
indexed_models . push ( primitives ::unit_cylinder ( ) ) ;
indexed_models . push ( primitives ::unit_cube ( ) ) ;
2023-09-28 19:10:22 -07:00
println! ( " models.len = {:?} " , indexed_models . len ( ) ) ;
2023-09-27 02:12:20 -07:00
indexed_models [ 0 ] . instances . push ( model ::ModelInstance {
transform :integer ::Planar64Affine3 ::try_from ( glam ::Affine3A ::from_translation ( glam ::vec3 ( 10. , 0. , - 10. ) ) ) . unwrap ( ) ,
2023-10-03 19:45:01 -07:00
.. Default ::default ( )
2023-09-21 13:02:01 -07:00
} ) ;
2023-09-20 22:37:03 -07:00
//quad monkeys
2023-09-27 02:12:20 -07:00
indexed_models [ 1 ] . instances . push ( model ::ModelInstance {
transform :integer ::Planar64Affine3 ::try_from ( glam ::Affine3A ::from_translation ( glam ::vec3 ( 10. , 5. , 10. ) ) ) . unwrap ( ) ,
2023-10-03 19:45:01 -07:00
.. Default ::default ( )
2023-09-21 13:02:01 -07:00
} ) ;
2023-09-27 02:12:20 -07:00
indexed_models [ 1 ] . instances . push ( model ::ModelInstance {
transform :integer ::Planar64Affine3 ::try_from ( glam ::Affine3A ::from_translation ( glam ::vec3 ( 20. , 5. , 10. ) ) ) . unwrap ( ) ,
2023-09-21 13:02:01 -07:00
color :glam ::vec4 ( 1.0 , 0.0 , 0.0 , 1.0 ) ,
2023-10-03 19:45:01 -07:00
.. Default ::default ( )
2023-09-21 13:02:01 -07:00
} ) ;
2023-09-27 02:12:20 -07:00
indexed_models [ 1 ] . instances . push ( model ::ModelInstance {
transform :integer ::Planar64Affine3 ::try_from ( glam ::Affine3A ::from_translation ( glam ::vec3 ( 10. , 5. , 20. ) ) ) . unwrap ( ) ,
2023-09-21 13:02:01 -07:00
color :glam ::vec4 ( 0.0 , 1.0 , 0.0 , 1.0 ) ,
2023-10-03 19:45:01 -07:00
.. Default ::default ( )
2023-09-21 13:02:01 -07:00
} ) ;
2023-09-27 02:12:20 -07:00
indexed_models [ 1 ] . instances . push ( model ::ModelInstance {
transform :integer ::Planar64Affine3 ::try_from ( glam ::Affine3A ::from_translation ( glam ::vec3 ( 20. , 5. , 20. ) ) ) . unwrap ( ) ,
2023-09-21 13:02:01 -07:00
color :glam ::vec4 ( 0.0 , 0.0 , 1.0 , 1.0 ) ,
2023-10-03 19:45:01 -07:00
.. Default ::default ( )
2023-10-03 16:34:54 -07:00
} ) ;
//decorative monkey
2023-09-27 02:12:20 -07:00
indexed_models [ 1 ] . instances . push ( model ::ModelInstance {
transform :integer ::Planar64Affine3 ::try_from ( glam ::Affine3A ::from_translation ( glam ::vec3 ( 15. , 10. , 15. ) ) ) . unwrap ( ) ,
2023-10-03 16:34:54 -07:00
color :glam ::vec4 ( 0.5 , 0.5 , 0.5 , 0.5 ) ,
attributes :model ::CollisionAttributes ::Decoration ,
2023-10-03 19:45:01 -07:00
.. Default ::default ( )
2023-09-21 13:02:01 -07:00
} ) ;
2023-09-20 22:37:03 -07:00
//teapot
2023-09-27 02:12:20 -07:00
indexed_models [ 2 ] . instances . push ( model ::ModelInstance {
transform :integer ::Planar64Affine3 ::try_from ( glam ::Affine3A ::from_scale_rotation_translation ( glam ::vec3 ( 0.5 , 1.0 , 0.2 ) , glam ::quat ( - 0.22248298016985793 , - 0.839457167990537 , - 0.05603504040830783 , - 0.49261857546227916 ) , glam ::vec3 ( - 10. , 7. , 10. ) ) ) . unwrap ( ) ,
2023-10-03 19:45:01 -07:00
.. Default ::default ( )
2023-09-21 13:02:01 -07:00
} ) ;
2023-09-20 22:37:03 -07:00
//ground
2023-09-27 02:12:20 -07:00
indexed_models [ 3 ] . instances . push ( model ::ModelInstance {
transform :integer ::Planar64Affine3 ::try_from ( glam ::Affine3A ::from_translation ( glam ::vec3 ( 0. , 0. , 0. ) ) * glam ::Affine3A ::from_scale ( glam ::vec3 ( 160.0 , 1.0 , 160.0 ) ) ) . unwrap ( ) ,
2023-10-03 19:45:01 -07:00
.. Default ::default ( )
2023-09-21 13:02:01 -07:00
} ) ;
2023-09-06 14:39:44 -07:00
2023-10-18 17:17:21 -07:00
let mut graphics = GraphicsState ::new ( ) ;
2023-09-20 22:36:42 -07:00
2023-10-09 19:44:49 -07:00
graphics . load_user_settings ( & user_settings ) ;
2023-09-28 19:10:22 -07:00
let indexed_model_instances = model ::IndexedModelInstances {
textures :Vec ::new ( ) ,
models :indexed_models ,
2023-09-27 02:12:20 -07:00
spawn_point :integer ::Planar64Vec3 ::Y * 50 ,
2023-10-03 19:42:07 -07:00
modes :Vec ::new ( ) ,
2023-09-28 19:10:22 -07:00
} ;
2023-10-04 20:04:04 -07:00
//how to multithread
//1. build
physics . generate_models ( & indexed_model_instances ) ;
//2. move
let physics_thread = physics . into_worker ( ) ;
//3. forget
let mut state = GlobalState {
start_time :Instant ::now ( ) ,
manual_mouse_lock :false ,
mouse :physics ::MouseState ::default ( ) ,
2023-10-09 19:44:49 -07:00
user_settings ,
2023-10-04 20:04:04 -07:00
graphics ,
physics_thread ,
} ;
2023-10-03 19:43:41 -07:00
state . generate_model_graphics ( & device , & queue , indexed_model_instances ) ;
2023-09-20 22:36:42 -07:00
2023-10-01 17:18:50 -07:00
let args :Vec < String > = std ::env ::args ( ) . collect ( ) ;
if args . len ( ) = = 2 {
2023-10-03 19:43:41 -07:00
state . load_file ( std ::path ::PathBuf ::from ( & args [ 1 ] ) , device , queue ) ;
2023-10-01 17:18:50 -07:00
}
2023-10-03 19:43:41 -07:00
return state ;
2023-09-06 14:39:44 -07:00
}
2023-10-01 17:18:29 -07:00
fn load_file ( & mut self , path : std ::path ::PathBuf , device : & wgpu ::Device , queue : & wgpu ::Queue ) {
println! ( " Loading file: {:?} " , & path ) ;
//oh boy! let's load the map!
if let Ok ( file ) = std ::fs ::File ::open ( path ) {
let mut input = std ::io ::BufReader ::new ( file ) ;
let mut first_8 = [ 0 u8 ; 8 ] ;
//.rbxm roblox binary = "<roblox!"
//.rbxmx roblox xml = "<roblox "
//.bsp = "VBSP"
//.vmf =
//.snf = "SNMF"
//.snf = "SNBF"
if let ( Ok ( ( ) ) , Ok ( ( ) ) ) = ( std ::io ::Read ::read_exact ( & mut input , & mut first_8 ) , std ::io ::Seek ::rewind ( & mut input ) ) {
2023-10-02 15:27:41 -07:00
if let Some ( indexed_model_instances ) = {
2023-10-01 17:18:29 -07:00
match & first_8 [ 0 .. 4 ] {
b " <rob " = > {
match match & first_8 [ 4 .. 8 ] {
b " lox! " = > rbx_binary ::from_reader ( input ) . map_err ( | e | format! ( " {:?} " , e ) ) ,
b " lox " = > rbx_xml ::from_reader ( input , rbx_xml ::DecodeOptions ::default ( ) ) . map_err ( | e | format! ( " {:?} " , e ) ) ,
other = > Err ( format! ( " Unknown Roblox file type {:?} " , other ) ) ,
} {
2023-10-02 15:27:41 -07:00
Ok ( dom ) = > Some ( load_roblox ::generate_indexed_models ( dom ) ) ,
2023-10-01 17:18:29 -07:00
Err ( e ) = > {
println! ( " Error loading roblox file: {:?} " , e ) ;
None
2023-09-30 02:53:25 -07:00
} ,
2023-09-26 14:25:44 -07:00
}
2023-10-01 17:18:29 -07:00
} ,
2023-10-02 15:27:41 -07:00
//b"VBSP"=>Some(load_bsp::generate_indexed_models(input)),
//b"SNFM"=>Some(sniffer::generate_indexed_models(input)),
//b"SNFB"=>Some(sniffer::load_bot(input)),
other = > {
println! ( " loser file {:?} " , other ) ;
None
} ,
2023-09-26 14:25:44 -07:00
}
2023-10-01 17:18:29 -07:00
} {
2023-10-02 15:27:41 -07:00
let spawn_point = indexed_model_instances . spawn_point ;
2023-10-01 17:18:29 -07:00
//if generate_indexed_models succeeds, clear the previous ones
2023-10-03 19:43:41 -07:00
self . graphics . clear ( ) ;
2023-10-04 20:04:04 -07:00
let mut physics = physics ::PhysicsState ::default ( ) ;
2023-10-16 18:57:37 -07:00
//physics.spawn()
2023-10-04 20:04:04 -07:00
physics . game . stage_id = 0 ;
physics . spawn_point = spawn_point ;
physics . process_instruction ( instruction ::TimedInstruction {
time :physics . time ,
instruction : PhysicsInstruction ::Input ( physics ::PhysicsInputInstruction ::Reset ) ,
} ) ;
2023-10-09 19:44:49 -07:00
physics . load_user_settings ( & self . user_settings ) ;
2023-10-04 20:04:04 -07:00
physics . generate_models ( & indexed_model_instances ) ;
self . physics_thread = physics . into_worker ( ) ;
2023-10-09 19:44:49 -07:00
//graphics.load_user_settings(&self.user_settings);
2023-10-01 17:18:29 -07:00
self . generate_model_graphics ( device , queue , indexed_model_instances ) ;
//manual reset
2023-09-20 22:36:42 -07:00
} else {
2023-10-01 17:18:29 -07:00
println! ( " No modeldatas were generated " ) ;
2023-09-20 22:36:42 -07:00
}
2023-10-01 17:18:29 -07:00
} else {
println! ( " Failed to read first 8 bytes and seek back to beginning of file. " ) ;
}
} else {
println! ( " Could not open file " ) ;
}
}
#[ allow(clippy::single_match) ]
fn update ( & mut self , window : & winit ::window ::Window , device : & wgpu ::Device , queue : & wgpu ::Queue , event : winit ::event ::WindowEvent ) {
2023-09-27 02:12:20 -07:00
let time = integer ::Time ::from_nanos ( self . start_time . elapsed ( ) . as_nanos ( ) as i64 ) ;
2023-10-01 17:18:29 -07:00
match event {
winit ::event ::WindowEvent ::DroppedFile ( path ) = > self . load_file ( path , device , queue ) ,
2023-10-01 19:29:29 -07:00
winit ::event ::WindowEvent ::Focused ( state ) = > {
//pause unpause
//recalculate pressed keys on focus
2023-10-01 15:54:50 -07:00
} ,
winit ::event ::WindowEvent ::KeyboardInput {
input :winit ::event ::KeyboardInput { state , virtual_keycode , .. } ,
2023-09-06 14:39:44 -07:00
..
2023-10-01 15:54:50 -07:00
} = > {
2023-09-19 17:53:29 -07:00
let s = match state {
winit ::event ::ElementState ::Pressed = > true ,
winit ::event ::ElementState ::Released = > false ,
} ;
2023-10-01 15:54:50 -07:00
match virtual_keycode {
Some ( winit ::event ::VirtualKeyCode ::Tab ) = > {
2023-10-02 02:40:36 -07:00
if s {
self . manual_mouse_lock = false ;
2023-10-04 20:04:04 -07:00
match window . set_cursor_position ( winit ::dpi ::PhysicalPosition ::new ( self . graphics . camera . screen_size . x as f32 / 2.0 , self . graphics . camera . screen_size . y as f32 / 2.0 ) ) {
2023-10-02 02:40:36 -07:00
Ok ( ( ) ) = > ( ) ,
Err ( e ) = > println! ( " Could not set cursor position: {:?} " , e ) ,
}
match window . set_cursor_grab ( winit ::window ::CursorGrabMode ::None ) {
Ok ( ( ) ) = > ( ) ,
Err ( e ) = > println! ( " Could not release cursor: {:?} " , e ) ,
}
} else {
//if cursor is outside window don't lock but apparently there's no get pos function
//let pos=window.get_cursor_pos();
match window . set_cursor_grab ( winit ::window ::CursorGrabMode ::Locked ) {
Ok ( ( ) ) = > ( ) ,
Err ( _ ) = > {
match window . set_cursor_grab ( winit ::window ::CursorGrabMode ::Confined ) {
Ok ( ( ) ) = > ( ) ,
Err ( e ) = > {
self . manual_mouse_lock = true ;
println! ( " Could not confine cursor: {:?} " , e )
} ,
}
}
}
}
window . set_cursor_visible ( s ) ;
} ,
2023-10-01 15:54:50 -07:00
Some ( winit ::event ::VirtualKeyCode ::F11 ) = > {
if s {
if window . fullscreen ( ) . is_some ( ) {
window . set_fullscreen ( None ) ;
} else {
window . set_fullscreen ( Some ( winit ::window ::Fullscreen ::Borderless ( None ) ) ) ;
}
}
} ,
Some ( winit ::event ::VirtualKeyCode ::Escape ) = > {
if s {
self . manual_mouse_lock = false ;
match window . set_cursor_grab ( winit ::window ::CursorGrabMode ::None ) {
Ok ( ( ) ) = > ( ) ,
Err ( e ) = > println! ( " Could not release cursor: {:?} " , e ) ,
}
window . set_cursor_visible ( true ) ;
}
} ,
Some ( keycode ) = > {
if let Some ( input_instruction ) = match keycode {
winit ::event ::VirtualKeyCode ::W = > Some ( InputInstruction ::MoveForward ( s ) ) ,
winit ::event ::VirtualKeyCode ::A = > Some ( InputInstruction ::MoveLeft ( s ) ) ,
winit ::event ::VirtualKeyCode ::S = > Some ( InputInstruction ::MoveBack ( s ) ) ,
winit ::event ::VirtualKeyCode ::D = > Some ( InputInstruction ::MoveRight ( s ) ) ,
winit ::event ::VirtualKeyCode ::E = > Some ( InputInstruction ::MoveUp ( s ) ) ,
winit ::event ::VirtualKeyCode ::Q = > Some ( InputInstruction ::MoveDown ( s ) ) ,
winit ::event ::VirtualKeyCode ::Space = > Some ( InputInstruction ::Jump ( s ) ) ,
winit ::event ::VirtualKeyCode ::Z = > Some ( InputInstruction ::Zoom ( s ) ) ,
winit ::event ::VirtualKeyCode ::R = > if s { Some ( InputInstruction ::Reset ) } else { None } ,
_ = > None ,
} {
self . physics_thread . send ( TimedInstruction {
time ,
instruction :input_instruction ,
} ) . unwrap ( ) ;
}
} ,
_ = > ( ) ,
2023-09-19 17:53:29 -07:00
}
} ,
2023-10-01 15:54:50 -07:00
_ = > ( ) ,
}
}
fn device_event ( & mut self , window : & winit ::window ::Window , event : winit ::event ::DeviceEvent ) {
//there's no way this is the best way get a timestamp.
2023-09-27 02:12:20 -07:00
let time = integer ::Time ::from_nanos ( self . start_time . elapsed ( ) . as_nanos ( ) as i64 ) ;
2023-10-01 15:54:50 -07:00
match event {
2023-09-19 17:53:29 -07:00
winit ::event ::DeviceEvent ::MouseMotion {
delta , //these (f64,f64) are integers on my machine
2023-09-06 14:39:44 -07:00
} = > {
2023-10-02 02:40:36 -07:00
if self . manual_mouse_lock {
2023-10-04 20:04:04 -07:00
match window . set_cursor_position ( winit ::dpi ::PhysicalPosition ::new ( self . graphics . camera . screen_size . x as f32 / 2.0 , self . graphics . camera . screen_size . y as f32 / 2.0 ) ) {
2023-10-02 02:40:36 -07:00
Ok ( ( ) ) = > ( ) ,
Err ( e ) = > println! ( " Could not set cursor position: {:?} " , e ) ,
}
}
2023-09-19 17:53:29 -07:00
//do not step the physics because the mouse polling rate is higher than the physics can run.
//essentially the previous input will be overwritten until a true step runs
//which is fine because they run all the time.
2023-10-04 20:04:04 -07:00
let delta = glam ::ivec2 ( delta . 0 as i32 , delta . 1 as i32 ) ;
self . mouse . pos + = delta ;
self . physics_thread . send ( TimedInstruction {
2023-09-19 17:53:29 -07:00
time ,
2023-10-04 20:04:04 -07:00
instruction :InputInstruction ::MoveMouse ( self . mouse . pos ) ,
} ) . unwrap ( ) ;
2023-09-19 17:53:29 -07:00
} ,
winit ::event ::DeviceEvent ::MouseWheel {
delta ,
} = > {
2023-10-02 03:03:07 -07:00
println! ( " mousewheel {:?} " , delta ) ;
2023-10-01 15:54:24 -07:00
if false { //self.physics.style.use_scroll{
2023-10-04 20:04:04 -07:00
self . physics_thread . send ( TimedInstruction {
2023-09-19 17:53:29 -07:00
time ,
2023-10-04 20:04:04 -07:00
instruction :InputInstruction ::Jump ( true ) , //activates the immediate jump path, but the style modifier prevents controls&CONTROL_JUMP bit from being set to auto jump
} ) . unwrap ( ) ;
2023-09-06 14:39:44 -07:00
}
}
2023-09-19 17:53:29 -07:00
_ = > ( ) ,
2023-09-06 14:39:44 -07:00
}
}
2023-08-29 18:20:58 -07:00
}
2023-07-14 16:49:01 -07:00
fn main ( ) {
2023-10-03 19:43:41 -07:00
framework ::run ::< GlobalState > (
2023-09-06 14:39:44 -07:00
format! ( " Strafe Client v {} " ,
env! ( " CARGO_PKG_VERSION " )
) . as_str ( )
) ;
2023-08-29 18:20:58 -07:00
}