2010-11-20 10:37:14 +00:00
|
|
|
/* (c) Magnus Auvinen. See licence.txt in the root of the distribution for more information. */
|
|
|
|
/* If you are missing that file, acquire a complete release at teeworlds.com. */
|
2011-08-31 11:56:04 +00:00
|
|
|
|
|
|
|
#include <base/tl/string.h>
|
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
#include <engine/shared/config.h>
|
2008-08-27 15:48:50 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
#include <base/math.h>
|
|
|
|
#include <game/client/component.h>
|
2020-06-30 22:19:17 +00:00
|
|
|
#include <game/client/gameclient.h>
|
|
|
|
#include <game/collision.h>
|
2008-08-27 15:48:50 +00:00
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
#include "camera.h"
|
|
|
|
#include "controls.h"
|
2008-08-27 15:48:50 +00:00
|
|
|
|
2011-04-09 06:41:31 +00:00
|
|
|
#include <engine/serverbrowser.h>
|
2020-12-20 17:19:53 +00:00
|
|
|
#include <limits>
|
2011-04-09 06:41:31 +00:00
|
|
|
|
2020-06-30 22:19:17 +00:00
|
|
|
const float ZoomStep = 0.866025f;
|
|
|
|
|
2010-05-29 07:25:38 +00:00
|
|
|
CCamera::CCamera()
|
2008-08-27 15:48:50 +00:00
|
|
|
{
|
2011-03-12 17:07:57 +00:00
|
|
|
m_CamType = CAMTYPE_UNDEFINED;
|
2015-01-10 00:10:51 +00:00
|
|
|
m_ZoomSet = false;
|
2019-07-08 21:08:42 +00:00
|
|
|
m_Zoom = 1.0f;
|
2020-06-30 22:19:17 +00:00
|
|
|
m_Zooming = false;
|
2021-01-22 17:16:58 +00:00
|
|
|
m_ForceFreeviewPos = vec2(-1, -1);
|
2008-08-27 15:48:50 +00:00
|
|
|
}
|
|
|
|
|
2020-06-30 22:19:17 +00:00
|
|
|
float CCamera::ZoomProgress(float CurrentTime) const
|
2008-08-27 15:48:50 +00:00
|
|
|
{
|
2020-06-30 22:19:17 +00:00
|
|
|
return (CurrentTime - m_ZoomSmoothingStart) / (m_ZoomSmoothingEnd - m_ZoomSmoothingStart);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CCamera::ScaleZoom(float Factor)
|
|
|
|
{
|
|
|
|
float CurrentTarget = m_Zooming ? m_ZoomSmoothingTarget : m_Zoom;
|
|
|
|
ChangeZoom(CurrentTarget * Factor);
|
|
|
|
}
|
|
|
|
|
2020-09-06 22:19:38 +00:00
|
|
|
float CCamera::MaxZoomLevel()
|
|
|
|
{
|
2020-12-20 17:19:53 +00:00
|
|
|
return (g_Config.m_ClLimitMaxZoomLevel) ? ((Graphics()->IsTileBufferingEnabled() ? 60 : 30)) : std::numeric_limits<float>::max();
|
2020-09-06 22:19:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
float CCamera::MinZoomLevel()
|
|
|
|
{
|
|
|
|
return 0.01f;
|
|
|
|
}
|
|
|
|
|
2020-06-30 22:19:17 +00:00
|
|
|
void CCamera::ChangeZoom(float Target)
|
|
|
|
{
|
2020-09-06 22:19:38 +00:00
|
|
|
if(Target > MaxZoomLevel() || Target < MinZoomLevel())
|
2020-06-30 14:25:54 +00:00
|
|
|
{
|
2020-06-30 22:19:17 +00:00
|
|
|
return;
|
|
|
|
}
|
2020-06-30 17:06:29 +00:00
|
|
|
|
2020-06-30 22:19:17 +00:00
|
|
|
float Now = Client()->LocalTime();
|
|
|
|
float Current = m_Zoom;
|
|
|
|
float Derivative = 0.0f;
|
|
|
|
if(m_Zooming)
|
|
|
|
{
|
|
|
|
float Progress = ZoomProgress(Now);
|
|
|
|
Current = m_ZoomSmoothing.Evaluate(Progress);
|
|
|
|
Derivative = m_ZoomSmoothing.Derivative(Progress);
|
|
|
|
}
|
2020-06-30 17:06:29 +00:00
|
|
|
|
2020-06-30 22:19:17 +00:00
|
|
|
m_ZoomSmoothingTarget = Target;
|
|
|
|
m_ZoomSmoothing = CCubicBezier::With(Current, Derivative, 0, m_ZoomSmoothingTarget);
|
|
|
|
m_ZoomSmoothingStart = Now;
|
2020-07-07 20:57:35 +00:00
|
|
|
m_ZoomSmoothingEnd = Now + (float)g_Config.m_ClSmoothZoomTime / 1000;
|
2020-06-30 22:19:17 +00:00
|
|
|
|
|
|
|
m_Zooming = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CCamera::OnRender()
|
|
|
|
{
|
|
|
|
if(m_Zooming)
|
|
|
|
{
|
|
|
|
float Time = Client()->LocalTime();
|
|
|
|
if(Time >= m_ZoomSmoothingEnd)
|
|
|
|
{
|
|
|
|
m_Zoom = m_ZoomSmoothingTarget;
|
|
|
|
m_Zooming = false;
|
|
|
|
}
|
2020-06-30 14:25:54 +00:00
|
|
|
else
|
2020-06-30 22:19:17 +00:00
|
|
|
{
|
|
|
|
m_Zoom = m_ZoomSmoothing.Evaluate(ZoomProgress(Time));
|
|
|
|
}
|
2020-09-06 22:19:38 +00:00
|
|
|
m_Zoom = clamp(m_Zoom, MinZoomLevel(), MaxZoomLevel());
|
2020-06-30 14:25:54 +00:00
|
|
|
}
|
|
|
|
|
2019-06-03 19:52:14 +00:00
|
|
|
if(!(m_pClient->m_Snap.m_SpecInfo.m_Active || GameClient()->m_GameInfo.m_AllowZoom || Client()->State() == IClient::STATE_DEMOPLAYBACK))
|
2015-01-03 02:13:21 +00:00
|
|
|
{
|
2019-06-15 10:23:50 +00:00
|
|
|
m_ZoomSet = false;
|
2019-07-08 21:08:42 +00:00
|
|
|
m_Zoom = 1.0f;
|
2020-06-30 22:19:17 +00:00
|
|
|
m_Zooming = false;
|
2015-01-03 02:13:21 +00:00
|
|
|
}
|
2015-01-10 00:10:51 +00:00
|
|
|
else if(!m_ZoomSet && g_Config.m_ClDefaultZoom != 10)
|
2015-01-03 02:13:21 +00:00
|
|
|
{
|
2015-01-10 00:10:51 +00:00
|
|
|
m_ZoomSet = true;
|
2015-01-03 02:13:21 +00:00
|
|
|
OnReset();
|
|
|
|
}
|
2010-08-25 14:15:59 +00:00
|
|
|
|
2011-04-13 18:37:12 +00:00
|
|
|
// update camera center
|
2011-03-12 17:07:57 +00:00
|
|
|
if(m_pClient->m_Snap.m_SpecInfo.m_Active && !m_pClient->m_Snap.m_SpecInfo.m_UsePosition)
|
2010-09-19 14:00:46 +00:00
|
|
|
{
|
2011-03-12 17:07:57 +00:00
|
|
|
if(m_CamType != CAMTYPE_SPEC)
|
2010-09-19 14:12:18 +00:00
|
|
|
{
|
2021-07-12 09:43:56 +00:00
|
|
|
m_LastPos[g_Config.m_ClDummy] = m_pClient->m_Controls.m_MousePos[g_Config.m_ClDummy];
|
|
|
|
m_pClient->m_Controls.m_MousePos[g_Config.m_ClDummy] = m_PrevCenter;
|
|
|
|
m_pClient->m_Controls.ClampMousePos();
|
2011-03-12 17:07:57 +00:00
|
|
|
m_CamType = CAMTYPE_SPEC;
|
2010-09-19 14:12:18 +00:00
|
|
|
}
|
2021-07-12 09:43:56 +00:00
|
|
|
m_Center = m_pClient->m_Controls.m_MousePos[g_Config.m_ClDummy];
|
2010-09-19 14:00:46 +00:00
|
|
|
}
|
2008-08-27 15:48:50 +00:00
|
|
|
else
|
|
|
|
{
|
2011-03-12 17:07:57 +00:00
|
|
|
if(m_CamType != CAMTYPE_PLAYER)
|
2010-09-19 14:00:46 +00:00
|
|
|
{
|
2019-07-16 10:56:43 +00:00
|
|
|
if((m_LastPos[g_Config.m_ClDummy].x < g_Config.m_ClMouseMinDistance) || (m_LastPos[g_Config.m_ClDummy].x < g_Config.m_ClDyncamMinDistance))
|
2021-07-12 09:43:56 +00:00
|
|
|
m_pClient->m_Controls.m_MousePos[g_Config.m_ClDummy].x = m_LastPos[g_Config.m_ClDummy].x + g_Config.m_ClMouseMinDistance + g_Config.m_ClDyncamMinDistance;
|
2019-04-13 12:33:25 +00:00
|
|
|
else
|
2021-07-12 09:43:56 +00:00
|
|
|
m_pClient->m_Controls.m_MousePos[g_Config.m_ClDummy] = m_LastPos[g_Config.m_ClDummy];
|
|
|
|
m_pClient->m_Controls.ClampMousePos();
|
2011-03-12 17:07:57 +00:00
|
|
|
m_CamType = CAMTYPE_PLAYER;
|
2010-09-19 14:00:46 +00:00
|
|
|
}
|
2011-03-10 16:57:15 +00:00
|
|
|
|
2020-10-14 00:43:30 +00:00
|
|
|
float DeltaTime = Client()->RenderFrameTime();
|
|
|
|
static vec2 s_LastMousePos(0, 0);
|
|
|
|
static vec2 s_CurrentCameraOffset[2] = {vec2(0, 0), vec2(0, 0)};
|
|
|
|
static float s_SpeedBias = 0.5f;
|
2010-09-27 19:41:41 +00:00
|
|
|
|
2020-11-07 10:29:23 +00:00
|
|
|
if(g_Config.m_ClDyncamSmoothness > 0)
|
2020-10-14 00:43:30 +00:00
|
|
|
{
|
2020-11-07 10:29:23 +00:00
|
|
|
float CameraSpeed = (1.0f - (g_Config.m_ClDyncamSmoothness / 100.0f)) * 9.5f + 0.5f;
|
|
|
|
float CameraStabilizingFactor = 1 + g_Config.m_ClDyncamStabilizing / 100.0f;
|
2020-10-14 00:43:30 +00:00
|
|
|
|
|
|
|
s_SpeedBias += CameraSpeed * DeltaTime;
|
|
|
|
if(g_Config.m_ClDyncam)
|
|
|
|
{
|
2021-07-12 09:43:56 +00:00
|
|
|
s_SpeedBias -= length(m_pClient->m_Controls.m_MousePos[g_Config.m_ClDummy] - s_LastMousePos) * log10f(CameraStabilizingFactor) * 0.02f;
|
2020-10-14 00:43:30 +00:00
|
|
|
s_SpeedBias = clamp(s_SpeedBias, 0.5f, CameraSpeed);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
s_SpeedBias = maximum(5.0f, CameraSpeed); // make sure toggle back is fast
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
vec2 TargetCameraOffset(0, 0);
|
2021-07-12 09:43:56 +00:00
|
|
|
s_LastMousePos = m_pClient->m_Controls.m_MousePos[g_Config.m_ClDummy];
|
2020-10-14 00:43:30 +00:00
|
|
|
float l = length(s_LastMousePos);
|
2008-08-27 15:48:50 +00:00
|
|
|
if(l > 0.0001f) // make sure that this isn't 0
|
2011-03-10 16:57:15 +00:00
|
|
|
{
|
2015-08-27 18:26:05 +00:00
|
|
|
float DeadZone = g_Config.m_ClDyncam ? g_Config.m_ClDyncamDeadzone : g_Config.m_ClMouseDeadzone;
|
|
|
|
float FollowFactor = (g_Config.m_ClDyncam ? g_Config.m_ClDyncamFollowFactor : g_Config.m_ClMouseFollowfactor) / 100.0f;
|
2020-09-26 19:41:58 +00:00
|
|
|
float OffsetAmount = maximum(l - DeadZone, 0.0f) * FollowFactor;
|
2011-03-10 16:57:15 +00:00
|
|
|
|
2021-07-12 09:43:56 +00:00
|
|
|
TargetCameraOffset = normalize(m_pClient->m_Controls.m_MousePos[g_Config.m_ClDummy]) * OffsetAmount;
|
2011-03-10 16:57:15 +00:00
|
|
|
}
|
2011-04-13 18:37:12 +00:00
|
|
|
|
2020-11-07 10:29:23 +00:00
|
|
|
if(g_Config.m_ClDyncamSmoothness > 0)
|
2020-10-14 00:43:30 +00:00
|
|
|
s_CurrentCameraOffset[g_Config.m_ClDummy] += (TargetCameraOffset - s_CurrentCameraOffset[g_Config.m_ClDummy]) * minimum(DeltaTime * s_SpeedBias, 1.0f);
|
|
|
|
else
|
|
|
|
s_CurrentCameraOffset[g_Config.m_ClDummy] = TargetCameraOffset;
|
|
|
|
|
2011-03-12 17:07:57 +00:00
|
|
|
if(m_pClient->m_Snap.m_SpecInfo.m_Active)
|
2020-10-14 00:43:30 +00:00
|
|
|
m_Center = m_pClient->m_Snap.m_SpecInfo.m_Position + s_CurrentCameraOffset[g_Config.m_ClDummy];
|
2011-03-10 16:57:15 +00:00
|
|
|
else
|
2020-10-14 00:43:30 +00:00
|
|
|
m_Center = m_pClient->m_LocalCharacterPos + s_CurrentCameraOffset[g_Config.m_ClDummy];
|
2008-08-27 15:48:50 +00:00
|
|
|
}
|
2011-03-12 17:07:57 +00:00
|
|
|
|
2021-01-22 17:16:58 +00:00
|
|
|
if(m_ForceFreeviewPos != vec2(-1, -1) && m_CamType == CAMTYPE_SPEC)
|
|
|
|
{
|
2021-07-12 09:43:56 +00:00
|
|
|
m_Center = m_pClient->m_Controls.m_MousePos[g_Config.m_ClDummy] = m_ForceFreeviewPos;
|
2021-01-22 17:16:58 +00:00
|
|
|
m_ForceFreeviewPos = vec2(-1, -1);
|
|
|
|
}
|
2011-03-12 17:07:57 +00:00
|
|
|
m_PrevCenter = m_Center;
|
2008-08-27 15:48:50 +00:00
|
|
|
}
|
2010-10-17 08:43:27 +00:00
|
|
|
|
|
|
|
void CCamera::OnConsoleInit()
|
|
|
|
{
|
2011-08-13 00:11:06 +00:00
|
|
|
Console()->Register("zoom+", "", CFGFLAG_CLIENT, ConZoomPlus, this, "Zoom increase");
|
|
|
|
Console()->Register("zoom-", "", CFGFLAG_CLIENT, ConZoomMinus, this, "Zoom decrease");
|
2021-07-10 13:53:23 +00:00
|
|
|
Console()->Register("zoom", "?i", CFGFLAG_CLIENT, ConZoom, this, "Change zoom");
|
2021-01-22 17:16:58 +00:00
|
|
|
Console()->Register("set_view", "i[x]i[y]", CFGFLAG_CLIENT, ConSetView, this, "Set camera position to x and y in the map");
|
2010-10-17 08:43:27 +00:00
|
|
|
}
|
2011-01-06 03:46:10 +00:00
|
|
|
|
2011-12-20 20:45:07 +00:00
|
|
|
void CCamera::OnReset()
|
|
|
|
{
|
2020-06-30 22:19:17 +00:00
|
|
|
m_Zoom = pow(ZoomStep, g_Config.m_ClDefaultZoom - 10);
|
|
|
|
m_Zooming = false;
|
2011-12-20 20:45:07 +00:00
|
|
|
}
|
|
|
|
|
2011-08-13 00:11:06 +00:00
|
|
|
void CCamera::ConZoomPlus(IConsole::IResult *pResult, void *pUserData)
|
2011-02-16 09:20:27 +00:00
|
|
|
{
|
|
|
|
CCamera *pSelf = (CCamera *)pUserData;
|
2019-06-03 19:52:14 +00:00
|
|
|
if(pSelf->m_pClient->m_Snap.m_SpecInfo.m_Active || pSelf->GameClient()->m_GameInfo.m_AllowZoom || pSelf->Client()->State() == IClient::STATE_DEMOPLAYBACK)
|
2020-06-30 14:25:54 +00:00
|
|
|
{
|
2020-06-30 22:19:17 +00:00
|
|
|
pSelf->ScaleZoom(ZoomStep);
|
2020-06-30 14:25:54 +00:00
|
|
|
}
|
2011-01-06 03:46:10 +00:00
|
|
|
}
|
2011-08-13 00:11:06 +00:00
|
|
|
void CCamera::ConZoomMinus(IConsole::IResult *pResult, void *pUserData)
|
2011-02-16 09:20:27 +00:00
|
|
|
{
|
|
|
|
CCamera *pSelf = (CCamera *)pUserData;
|
2019-06-03 19:52:14 +00:00
|
|
|
if(pSelf->m_pClient->m_Snap.m_SpecInfo.m_Active || pSelf->GameClient()->m_GameInfo.m_AllowZoom || pSelf->Client()->State() == IClient::STATE_DEMOPLAYBACK)
|
2017-09-13 18:33:58 +00:00
|
|
|
{
|
2020-06-30 22:19:17 +00:00
|
|
|
pSelf->ScaleZoom(1 / ZoomStep);
|
2017-09-12 18:14:34 +00:00
|
|
|
}
|
2011-02-16 09:20:27 +00:00
|
|
|
}
|
2021-07-10 13:53:23 +00:00
|
|
|
void CCamera::ConZoom(IConsole::IResult *pResult, void *pUserData)
|
2011-02-16 09:20:27 +00:00
|
|
|
{
|
2021-08-19 20:13:04 +00:00
|
|
|
float TargetLevel = pResult->NumArguments() ? clamp<float>(pResult->GetFloat(0), 0, 20) : g_Config.m_ClDefaultZoom;
|
2021-07-10 13:53:23 +00:00
|
|
|
((CCamera *)pUserData)->ChangeZoom(pow(ZoomStep, TargetLevel - 10));
|
2020-06-30 14:30:40 +00:00
|
|
|
}
|
2021-01-22 17:16:58 +00:00
|
|
|
void CCamera::ConSetView(IConsole::IResult *pResult, void *pUserData)
|
|
|
|
{
|
|
|
|
CCamera *pSelf = (CCamera *)pUserData;
|
|
|
|
// wait until free view camera type to update the position
|
|
|
|
pSelf->m_ForceFreeviewPos = vec2(
|
|
|
|
clamp(pResult->GetInteger(0) * 32.0f, 200.0f, pSelf->Collision()->GetWidth() * 32 - 200.0f),
|
|
|
|
clamp(pResult->GetInteger(1) * 32.0f, 200.0f, pSelf->Collision()->GetWidth() * 32 - 200.0f));
|
|
|
|
}
|