aboutsummaryrefslogtreecommitdiff
path: root/Minecraft.Client/Camera.cpp
blob: 4de9634919fcab45ea27e093cb03f1f4e8062ec3 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
#include "stdafx.h"
#include "Camera.h"
#include "MemoryTracker.h"
#include "..\Minecraft.World\net.minecraft.world.entity.player.h"
#include "..\Minecraft.World\net.minecraft.world.level.h"
#include "..\Minecraft.World\net.minecraft.world.level.tile.h"
#include "..\Minecraft.World\TilePos.h"

float Camera::xPlayerOffs = 0.0f;
float Camera::yPlayerOffs = 0.0f;
float Camera::zPlayerOffs = 0.0f;

//IntBuffer *Camera::viewport		= MemoryTracker::createIntBuffer(16);
FloatBuffer *Camera::modelview	= MemoryTracker::createFloatBuffer(16);
FloatBuffer *Camera::projection = MemoryTracker::createFloatBuffer(16);
//FloatBuffer *Camera::position	= MemoryTracker::createFloatBuffer(3);

float Camera::xa = 0.0f;
float Camera::ya = 0.0f;
float Camera::za = 0.0f;
float Camera::xa2 = 0.0f;
float Camera::za2 = 0.0f;

void Camera::prepare(shared_ptr<Player> player, bool mirror)
{
    glGetFloat(GL_MODELVIEW_MATRIX, modelview);
    glGetFloat(GL_PROJECTION_MATRIX, projection);

	/* Original java code for reference
    glGetInteger(GL_VIEWPORT, viewport);

    float x = (viewport.get(0) + viewport.get(2)) / 2;
    float y = (viewport.get(1) + viewport.get(3)) / 2;
    gluUnProject(x, y, 0, modelview, projection, viewport, position);

    xPlayerOffs = position->get(0);
    yPlayerOffs = position->get(1);
    zPlayerOffs = position->get(2);
	*/

	// Xbox conversion here... note that we don't bother getting the viewport as this is just working out how to get a (0,0,0) point in clip space to pass into the inverted
	// combined model/view/projection matrix, so we just need to get this matrix and get its translation as an equivalent.
	XMMATRIX _modelview, _proj, _final, _invert;
	XMVECTOR _det;
	XMFLOAT4 trans;

	memcpy( &_modelview, modelview->_getDataPointer(), 64 );
	memcpy( &_proj, projection->_getDataPointer(), 64 );

#if ( defined __ORBIS__ ) || ( defined __PSVITA__ )
	_modelview = transpose(_modelview);
	_proj = transpose(_proj);
	_final = _modelview * _proj;
	_invert = sce::Vectormath::Simd::Aos::inverse(_final);
	xPlayerOffs = _invert.getElem(0,3) / _invert.getElem(3,3);
	yPlayerOffs = _invert.getElem(1,3) / _invert.getElem(3,3);
	zPlayerOffs = _invert.getElem(2,3) / _invert.getElem(3,3);
#elif defined  __PS3__
	_modelview = transpose(_modelview);
	_proj = transpose(_proj);
	_final = _modelview * _proj;
	_invert = Vectormath::Aos::inverse(_final);
	xPlayerOffs = _invert.getElem(0,3) / _invert.getElem(3,3);
	yPlayerOffs = _invert.getElem(1,3) / _invert.getElem(3,3);
	zPlayerOffs = _invert.getElem(2,3) / _invert.getElem(3,3);
#else
	_final = XMMatrixMultiply( _modelview, _proj );
	_det = XMMatrixDeterminant(_final);
	_invert = XMMatrixInverse(&_det, _final);

	XMStoreFloat4(&trans,_invert.r[3]);

	xPlayerOffs = trans.x / trans.w;
	yPlayerOffs = trans.y / trans.w;
	zPlayerOffs = trans.z / trans.w;
#endif

    int flipCamera = mirror ? 1 : 0;

    float xRot = player->xRot;
    float yRot = player->yRot;

    xa = cosf(yRot * PI / 180.0f) * (1 - flipCamera * 2);
    za = sinf(yRot * PI / 180.0f) * (1 - flipCamera * 2);

    xa2 = -za * sinf(xRot * PI / 180.0f) * (1 - flipCamera * 2);
    za2 = xa * sinf(xRot * PI / 180.0f) * (1 - flipCamera * 2);
    ya = cosf(xRot * PI / 180.0f);
}

TilePos *Camera::getCameraTilePos(shared_ptr<LivingEntity> player, double alpha)
{
	return new TilePos(getCameraPos(player, alpha));
}

Vec3 *Camera::getCameraPos(shared_ptr<LivingEntity> player, double alpha)
{
    double xx = player->xo + (player->x - player->xo) * alpha;
    double yy = player->yo + (player->y - player->yo) * alpha + player->getHeadHeight();
    double zz = player->zo + (player->z - player->zo) * alpha;

    double xt = xx + Camera::xPlayerOffs * 1;
    double yt = yy + Camera::yPlayerOffs * 1;
    double zt = zz + Camera::zPlayerOffs * 1;

    return Vec3::newTemp(xt, yt, zt);
}

int Camera::getBlockAt(Level *level, shared_ptr<LivingEntity> player, float alpha)
{
    Vec3 *p = Camera::getCameraPos(player, alpha);
    TilePos tp = TilePos(p);
    int t = level->getTile(tp.x, tp.y, tp.z);
    if (t != 0 && Tile::tiles[t]->material->isLiquid())
	{
        float hh = LiquidTile::getHeight(level->getData(tp.x, tp.y, tp.z)) - 1 / 9.0f;
        float h = tp.y + 1 - hh;
        if (p->y >= h)
		{
            t = level->getTile(tp.x, tp.y + 1, tp.z);
        }
    }
    return t;
}