#include "camera.h" #include "vk_common.h" #include "vk_math.h" #include "vk_textures.h" #include "ref_params.h" #include "pm_movevars.h" #include "pm_defs.h" #include "pmtrace.h" vk_global_camera_t g_camera; #define WORLDMODEL (gEngine.pfnGetModelByIndex( 1 )) #define MOVEVARS (gEngine.pfnGetMoveVars()) static float R_GetFarClip( void ) { if( WORLDMODEL /* FIXME VK && RI.drawWorld */ ) return MOVEVARS->zmax * 1.73f; return 2048.0f; } static void R_SetupModelviewMatrix( matrix4x4 m ) { Matrix4x4_CreateModelview( m ); Matrix4x4_ConcatRotate( m, -g_camera.viewangles[2], 1, 0, 0 ); Matrix4x4_ConcatRotate( m, -g_camera.viewangles[0], 0, 1, 0 ); Matrix4x4_ConcatRotate( m, -g_camera.viewangles[1], 0, 0, 1 ); Matrix4x4_ConcatTranslate( m, -g_camera.vieworg[0], -g_camera.vieworg[1], -g_camera.vieworg[2] ); } static void R_SetupProjectionMatrix( matrix4x4 m ) { float xMin, xMax, yMin, yMax, zNear, zFar; /* if( RI.drawOrtho ) { const ref_overview_t *ov = gEngfuncs.GetOverviewParms(); Matrix4x4_CreateOrtho( m, ov->xLeft, ov->xRight, ov->yTop, ov->yBottom, ov->zNear, ov->zFar ); return; } */ const float farClip = R_GetFarClip(); zNear = 4.0f; zFar = Q_max( 256.0f, farClip ); yMax = zNear * tan( g_camera.fov_y * M_PI_F / 360.0f ); yMin = -yMax; xMax = zNear * tan( g_camera.fov_x * M_PI_F / 360.0f ); xMin = -xMax; Matrix4x4_CreateProjection( m, xMax, xMin, yMax, yMin, zNear, zFar ); } // Analagous to R_SetupRefParams, R_SetupFrustum in GL/Soft renderers void R_SetupCamera( const ref_viewpass_t *rvp ) { /* FIXME VK unused? RI.params = RP_NONE; RI.drawWorld = FBitSet( rvp->flags, RF_DRAW_WORLD ); RI.onlyClientDraw = FBitSet( rvp->flags, RF_ONLY_CLIENTDRAW ); RI.farClip = 0; if( !FBitSet( rvp->flags, RF_DRAW_CUBEMAP )) RI.drawOrtho = FBitSet( rvp->flags, RF_DRAW_OVERVIEW ); else RI.drawOrtho = false; */ // setup viewport g_camera.viewport[0] = rvp->viewport[0]; g_camera.viewport[1] = rvp->viewport[1]; g_camera.viewport[2] = rvp->viewport[2]; g_camera.viewport[3] = rvp->viewport[3]; // calc FOV g_camera.fov_x = rvp->fov_x; g_camera.fov_y = rvp->fov_y; VectorCopy( rvp->vieworigin, g_camera.vieworg ); VectorCopy( rvp->viewangles, g_camera.viewangles ); // FIXME VK unused? VectorCopy( rvp->vieworigin, g_camera.pvsorigin ); #define RP_NORMALPASS() true // FIXME ??? if( RP_NORMALPASS() && ( gEngine.EngineGetParm( PARM_WATER_LEVEL, 0 ) >= 3 )) { g_camera.fov_x = atan( tan( DEG2RAD( g_camera.fov_x ) / 2 ) * ( 0.97f + sin( gpGlobals->time * 1.5f ) * 0.03f )) * 2 / (M_PI_F / 180.0f); g_camera.fov_y = atan( tan( DEG2RAD( g_camera.fov_y ) / 2 ) * ( 1.03f - sin( gpGlobals->time * 1.5f ) * 0.03f )) * 2 / (M_PI_F / 180.0f); } // build the transformation matrix for the given view angles AngleVectors( g_camera.viewangles, g_camera.vforward, g_camera.vright, g_camera.vup ); /* FIXME VK unused? if( !r_lockfrustum->value ) { VectorCopy( RI.vieworg, RI.cullorigin ); VectorCopy( RI.vforward, RI.cull_vforward ); VectorCopy( RI.vright, RI.cull_vright ); VectorCopy( RI.vup, RI.cull_vup ); } */ /* FIXME VK unused? if( RI.drawOrtho ) GL_FrustumInitOrtho( &RI.frustum, ov->xLeft, ov->xRight, ov->yTop, ov->yBottom, ov->zNear, ov->zFar ); else GL_FrustumInitProj( &g_camera.frustum, 0.0f, R_GetFarClip(), g_camera.fov_x, g_camera.fov_y ); // NOTE: we ignore nearplane here (mirrors only) */ R_SetupProjectionMatrix( g_camera.projectionMatrix ); R_SetupModelviewMatrix( g_camera.modelviewMatrix ); Matrix4x4_Concat( g_camera.worldviewProjectionMatrix, g_camera.projectionMatrix, g_camera.modelviewMatrix ); } int R_WorldToScreen( const vec3_t point, vec3_t screen ) { matrix4x4 worldToScreen; qboolean behind; float w; if( !point || !screen ) return true; Matrix4x4_Copy( worldToScreen, g_camera.worldviewProjectionMatrix ); screen[0] = worldToScreen[0][0] * point[0] + worldToScreen[0][1] * point[1] + worldToScreen[0][2] * point[2] + worldToScreen[0][3]; screen[1] = worldToScreen[1][0] * point[0] + worldToScreen[1][1] * point[1] + worldToScreen[1][2] * point[2] + worldToScreen[1][3]; w = worldToScreen[3][0] * point[0] + worldToScreen[3][1] * point[1] + worldToScreen[3][2] * point[2] + worldToScreen[3][3]; screen[2] = 0.0f; // just so we have something valid here if( w < 0.001f ) { screen[0] *= 100000; screen[1] *= 100000; behind = true; } else { float invw = 1.0f / w; screen[0] *= invw; screen[1] *= invw; behind = false; } return behind; } int TriWorldToScreen( const float *world, float *screen ) { int retval; retval = R_WorldToScreen( world, screen ); screen[0] = 0.5f * screen[0] * (float)g_camera.viewport[2]; screen[1] = -0.5f * screen[1] * (float)g_camera.viewport[3]; screen[0] += 0.5f * (float)g_camera.viewport[2]; screen[1] += 0.5f * (float)g_camera.viewport[3]; return retval; } static const char *renderModeName( int rendermode ) { switch (rendermode) { case kRenderNormal: return "kRenderNormal"; case kRenderTransColor: return "kRenderTransColor"; case kRenderTransTexture: return "kRenderTransTexture"; case kRenderGlow: return "kRenderGlow"; case kRenderTransAlpha: return "kRenderTransAlpha"; case kRenderTransAdd: return "kRenderTransAdd"; default: return "UNKNOWN"; } } void XVK_CameraDebugPrintCenterEntity( void ) { vec3_t vec_end; pmtrace_t trace; const msurface_t *surf; char buf[1024], *p = buf, *end = buf + sizeof(buf); const physent_t *physent = NULL; const cl_entity_t *ent = NULL; VectorMA(g_camera.vieworg, 1e6, g_camera.vforward, vec_end); trace = gEngine.CL_TraceLine( g_camera.vieworg, vec_end, PM_NORMAL ); surf = gEngine.EV_TraceSurface( Q_max(trace.ent, 0), g_camera.vieworg, vec_end ); if (trace.ent > 0) { physent = gEngine.EV_GetPhysent( trace.ent ); } ent = gEngine.GetEntityByIndex( (physent && physent->info > 0) ? physent->info : 0 ); p += Q_snprintf(p, end - p, "o\n" "cam.origin: %.03f %.03f %.03f\n", // TODO cam dir // TODO hit pos g_camera.vieworg[0], g_camera.vieworg[1], g_camera.vieworg[2] ); p += Q_snprintf(p, end - p, "entity index: %d, name: %s\n", (physent && physent->info > 0) ? physent->info : -1, (ent && ent->model) ? ent->model->name : "N/A"); if (ent) { p += Q_snprintf(p, end - p, "ent type: %d, rendermode: %d(%s)\n", ent->curstate.entityType, ent->curstate.rendermode, renderModeName(ent->curstate.rendermode)); } if (surf && ent && ent->model && ent->model->surfaces) { const int surface_index = surf - ent->model->surfaces; const int tex_id = surf->texinfo->texture->gl_texturenum; const vk_texture_t* const texture = findTexture( tex_id ); p += Q_snprintf(p, end - p, "surface index: %d; texture: %s\n", surface_index, texture ? texture->name : "NONE" ); } gEngine.CL_CenterPrint(buf, 0.5f); }