4 vg_tex2d tex_norwey
= { .path
= "textures/norway_foliage.qoi" };
5 vg_tex2d tex_grid
= { .path
= "textures/grid.qoi" };
6 vg_tex2d tex_sky
= { .path
= "textures/sky.qoi" };
7 vg_tex2d tex_gradients
= { .path
= "textures/gradients.qoi",
8 .flags
= VG_TEXTURE_CLAMP
};
9 vg_tex2d tex_cement
= { .path
= "textures/cement512.qoi" };
10 vg_tex2d tex_water
= { .path
= "textures/water.qoi" };
13 static int debugview
= 0;
14 static int sv_debugcam
= 0;
21 #include "character.h"
23 #include "rigidbody.h"
30 #include "shaders/blit.h"
31 #include "shaders/standard.h"
32 #include "shaders/unlit.h"
34 void vg_register(void)
36 shader_blit_register();
37 shader_standard_register();
38 shader_unlit_register();
46 vg_tex2d
*texture_list
[] =
57 int main( int argc
, char *argv
[] )
59 vg_init( argc
, argv
, "Voyager Game Engine" );
64 .bbx
= {{ -1.0f
, -0.25f
, -0.25f
}, { 1.0f
, 0.25f
, 0.25f
}}
68 .bbx
= {{ -0.5f
, -0.25f
, -0.25f
}, { 0.5f
, 0.25f
, 0.25f
}}
71 teleport_gate gate_a
= {
72 .co
= { 0.0f
, -3.0f
, -15.0f
},
73 .q
= { 0.0f
, 0.0f
, 0.0f
, 1.0f
}
76 .co
= { -8.0f
, -3.0f
, -17.0f
},
77 .q
= { 0.0f
, 0.0f
, 0.0f
, 1.0f
}
81 static int playermodel( int argc
, char const *argv
[] )
83 if( argc
< 1 ) return 0;
85 glmesh old_mesh
= player
.mdl
.mesh
;
87 if( character_load( &player
.mdl
, argv
[0] ) )
88 mesh_free( &old_mesh
);
95 vg_convar_push( (struct vg_convar
){
98 .data_type
= k_convar_dtype_i32
,
99 .opt_i32
= { .min
=0, .max
=1, .clamp
=1 },
103 vg_convar_push( (struct vg_convar
){
105 .data
= &walk_grid_iterations
,
106 .data_type
= k_convar_dtype_i32
,
107 .opt_i32
= { .min
=0, .max
=1, .clamp
=0 },
111 vg_convar_push( (struct vg_convar
){
112 .name
= "walk_speed",
113 .data
= &k_walkspeed
,
114 .data_type
= k_convar_dtype_f32
,
115 .opt_f32
= { .clamp
= 0 },
119 vg_convar_push( (struct vg_convar
){
121 .data
= &sv_debugcam
,
122 .data_type
= k_convar_dtype_i32
,
123 .opt_i32
= { .min
=0, .max
=1, .clamp
=0 },
127 vg_convar_push( (struct vg_convar
){
130 .data_type
= k_convar_dtype_i32
,
131 .opt_i32
= { .min
=0, .max
=1, .clamp
=0 },
135 vg_function_push( (struct vg_cmd
){
137 .function
= reset_player
140 vg_tex2d_init( texture_list
, vg_list_size( texture_list
) );
147 character_load( &player
.mdl
, "ch_default" );
148 character_init_ragdoll( &player
.mdl
);
152 reset_player( 1, (const char *[]){ "tutorial" } );
153 player_transform_update();
161 static void vg_framebuffer_resize( int w
, int h
)
170 glBindFramebuffer( GL_FRAMEBUFFER
, 0 );
171 glViewport( 0,0, vg_window_x
, vg_window_y
);
173 glDisable( GL_DEPTH_TEST
);
174 glClearColor( 0.11f
, 0.35f
, 0.37f
, 1.0f
);
175 glClear( GL_COLOR_BUFFER_BIT
| GL_DEPTH_BUFFER_BIT
);
177 float speed
= freecam
? 0.0f
: v3_length( player
.v
);
178 v3f shake
= { vg_randf()-0.5f
, vg_randf()-0.5f
, vg_randf()-0.5f
};
179 v3_muls( shake
, speed
*0.01f
, shake
);
185 v3_sub( player
.mdl
.ragdoll
[k_chpart_head
].co
, player
.follow
, delta
);
189 v3_muladds( player
.mdl
.ragdoll
[k_chpart_head
].co
, delta
,
191 v3_lerp( player
.follow
, follow_pos
, 0.1f
, player
.follow
);
192 v3_negate( player
.follow
, final
);
195 float yaw
= atan2f( delta
[0], -delta
[2] );
196 float pitch
= asinf( delta
[1] );
197 m4x3_rotate_x( world_matrix
, -pitch
);
198 m4x3_rotate_y( world_matrix
, yaw
);
206 m4x3_expand( player
.camera_inverse
, world_4x4
);
208 gpipeline
.fov
= freecam
? 60.0f
: 100.0f
; /* 120 */
209 m4x4_projection( vg_pv
, gpipeline
.fov
,
210 (float)vg_window_x
/ (float)vg_window_y
,
213 m4x4_mul( vg_pv
, world_4x4
, vg_pv
);
215 vg_line( (v3f
){ 0.0f
, 0.0f
, 0.0f
}, (v3f
){ 1.0f
, 0.0f
, 0.0f
}, 0xffff0000 );
216 vg_line( (v3f
){ 0.0f
, 0.0f
, 0.0f
}, (v3f
){ 0.0f
, 1.0f
, 0.0f
}, 0xff00ff00 );
217 vg_line( (v3f
){ 0.0f
, 0.0f
, 0.0f
}, (v3f
){ 0.0f
, 0.0f
, 1.0f
}, 0xff0000ff );
219 glEnable( GL_DEPTH_TEST
);
225 render_world( vg_pv
, player
.camera
);
226 render_water_texture( player
.camera
);
228 glBindFramebuffer( GL_FRAMEBUFFER
, 0 );
229 render_water_surface( vg_pv
);
232 vg_tex2d_bind( &tex_water
, 1 );
233 render_gate( &gate_a
, cam_transform
);
237 /* Copy the RGB of what we have into the background buffer */
238 glBindFramebuffer( GL_READ_FRAMEBUFFER
, 0 );
239 glBindFramebuffer( GL_DRAW_FRAMEBUFFER
, gpipeline
.fb_background
);
240 glBlitFramebuffer( 0,0, vg_window_x
, vg_window_y
,
241 0,0, vg_window_x
, vg_window_y
,
245 /* Clear out the colour buffer, but keep depth */
246 glBindFramebuffer( GL_FRAMEBUFFER
, 0 );
247 glClearColor( 0.0f
, 0.0f
, 0.0f
, 0.0f
);
249 if( !player
.is_dead
)
250 glClear( GL_COLOR_BUFFER_BIT
|GL_DEPTH_BUFFER_BIT
);
252 glClear( GL_COLOR_BUFFER_BIT
);
256 /* Draw back in the background */
258 glDisable(GL_DEPTH_TEST
);
259 glBlendFunc(GL_ONE_MINUS_DST_ALPHA
, GL_DST_ALPHA
);
260 glBlendEquation(GL_FUNC_ADD
);
263 shader_blit_uTexMain( 0 );
264 glActiveTexture(GL_TEXTURE0
);
265 glBindTexture( GL_TEXTURE_2D
, gpipeline
.rgb_background
);
271 glDisable( GL_DEPTH_TEST
);
272 vg_lines_drawall( (float *)vg_pv
);
274 /* Debugger camera */
276 glViewport( 0,0, 800, 800 );
277 glClearColor( 0.1f
, 0.0f
, 0.2f
, 1.0f
);
278 glClear( GL_DEPTH_BUFFER_BIT
);
280 m4x3_identity( world_matrix
);
283 v3_negate( player
.co
, debugcam
);
287 m4x3_translate( world_matrix
, debugcam
);
288 m4x3_expand( world_matrix
, world_4x4
);
290 m4x4_projection( vg_pv
,
292 (float)128.0f
/ (float)128.0f
,
294 m4x4_mul( vg_pv
, world_4x4
, vg_pv
);
298 glEnable( GL_DEPTH_TEST
);
303 glDisable( GL_DEPTH_TEST
);
304 vg_lines_drawall( (float *)vg_pv
);
306 glViewport( 0,0, vg_window_x
, vg_window_y
);
313 snprintf( buf
, 20, "%.2fm/s", v3_length( player
.v
) );
314 gui_text( (ui_px
[2]){ 0, 0 }, buf
, 1, k_text_align_left
);
316 snprintf( buf
, 20, "%.2f %.2f %.2f m/s",
317 player
.a
[0], player
.a
[1], player
.a
[2] );
318 gui_text( (ui_px
[2]){ 0, 20 }, buf
, 1, k_text_align_left
);
320 snprintf( buf
, 20, "pos %.2f %.2f %.2f",
321 player
.co
[0], player
.co
[1], player
.co
[2] );
322 gui_text( (ui_px
[2]){ 0, 40 }, buf
, 1, k_text_align_left
);
324 if( vg_gamepad_ready
)
326 for( int i
=0; i
<6; i
++ )
328 snprintf( buf
, 20, "%.2f", vg_gamepad
.axes
[i
] );
329 gui_text( (ui_px
[2]){ 0, (i
+3)*20 }, buf
, 1, k_text_align_left
);
334 gui_text( (ui_px
[2]){ 0, 60 },
335 "Gamepad not ready", 1, k_text_align_left
);