summaryrefslogtreecommitdiffstats
path: root/src/gamestate/GameState.cpp
blob: 5c5b9c4220b3325cc5aa5ea41fcee05d2fbe470e (plain) (blame)
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
125
126
127
128
129
130
131
#include "GameState.hpp"

GameState::GameState(NetworkClient *Net) : nc(Net) {
    Packet *response = nc->GetPacket();
    if (response->GetId() != 0x02) {
        LOG(ERROR) << "Response id is " << response->GetId();
        throw std::runtime_error("Response id is not 0x02");
    }
    PacketParser::Parse(*response, Login);
    g_PlayerUuid = response->GetField(0).GetString();
    g_PlayerName = response->GetField(1).GetString();
    delete response;
    m_networkState = ConnectionState::Play;
    LOG(INFO) << g_PlayerName << "'s UUID is " << g_PlayerUuid;
}

void GameState::Update() {
    Packet *packetPtr;

    try {
        packetPtr = nc->GetPacket();
        if (packetPtr == nullptr)
            return;
        PacketParser::Parse(*packetPtr, m_networkState);
    } catch (std::exception &e) {
        LOG(ERROR) << "Catched exception during packet pulling: " << e.what();
        return;
    }
    Packet &packet = *packetPtr;
    nlohmann::json json;

    switch (packet.GetId()) {
        case 0x23:
            g_PlayerEid = packet.GetField(0).GetInt();
            g_Gamemode = (packet.GetField(1).GetUByte() & 0b11111011);
            g_Dimension = packet.GetField(2).GetInt();
            g_Difficulty = packet.GetField(3).GetUByte();
            g_MaxPlayers = packet.GetField(4).GetUByte();
            g_LevelType = packet.GetField(5).GetString();
            g_ReducedDebugInfo = packet.GetField(6).GetBool();
            LOG(INFO) << "Gamemode is " << g_Gamemode << ", Difficulty is " << (int) g_Difficulty
                      << ", Level Type is " << g_LevelType;
            break;
        case 0x0D:
            g_Difficulty = packet.GetField(0).GetUByte();
            LOG(INFO) << "Difficulty now is " << (int) g_Difficulty;
            break;
        case 0x43:
            g_SpawnPosition = packet.GetField(0).GetPosition();
            LOG(INFO) << "Spawn position is " << g_SpawnPosition.GetX() << "," << g_SpawnPosition.GetY() << ","
                      << g_SpawnPosition.GetZ();
            break;
        case 0x2B:
            g_PlayerInvulnerable = (packet.GetField(0).GetByte() & 0x01) != 0;
            g_PlayerFlying = (packet.GetField(0).GetByte() & 0x02) != 0;
            g_PlayerAllowFlying = (packet.GetField(0).GetByte() & 0x04) != 0;
            g_PlayerCreativeMode = (packet.GetField(0).GetByte() & 0x08) != 0;
            g_PlayerFlyingSpeed = packet.GetField(1).GetFloat();
            g_PlayerFovModifier = packet.GetField(2).GetFloat();
            LOG(INFO) << "FOV modifier is " << g_PlayerFovModifier;
            break;
        case 0x2E:
            if ((packet.GetField(5).GetByte() & 0x10) != 0) {
                g_PlayerPitch += packet.GetField(4).GetFloat();
            } else {
                g_PlayerPitch = packet.GetField(4).GetFloat();
            };

            if ((packet.GetField(5).GetByte() & 0x08) != 0) {
                g_PlayerYaw += packet.GetField(3).GetFloat();
            } else {
                g_PlayerYaw = packet.GetField(3).GetFloat();
            }

            if ((packet.GetField(5).GetByte() & 0x01) != 0) {
                g_PlayerX += packet.GetField(0).GetDouble();
            } else {
                g_PlayerX = packet.GetField(0).GetDouble();
            }

            if ((packet.GetField(5).GetByte() & 0x02) != 0) {
                g_PlayerY += packet.GetField(1).GetDouble();
            } else {
                g_PlayerY = packet.GetField(1).GetDouble();
            }

            if ((packet.GetField(5).GetByte() & 0x04) != 0) {
                g_PlayerZ += packet.GetField(2).GetDouble();
            } else {
                g_PlayerZ = packet.GetField(2).GetDouble();
            }

            g_IsGameStarted = true;
            nc->AddPacketToQueue(PacketBuilder::CPlay0x03(0));
            nc->AddPacketToQueue(PacketBuilder::CPlay0x00(packet.GetField(6).GetVarInt()));
            LOG(INFO) << "Game is started!";
            LOG(INFO) << "PlayerPos is " << g_PlayerX << ", " << g_PlayerY << ", " << g_PlayerZ << "\tAngle: "
                      << g_PlayerYaw
                      << "," << g_PlayerPitch;
            break;
        case 0x1A:
            json = nlohmann::json::parse(packet.GetField(0).GetString());
            LOG(INFO) << "Disconnect reason: " << json["text"].get<std::string>();
            throw 119;
            break;
        case 0x20:
            world.ParseChunkData(packet);
            break;
        case 0x07:
            LOG(INFO) << "Statistics: ";
            for (int i = 0; i < packet.GetField(0).GetVarInt(); i++) {
                LOG(INFO) << "\t" << packet.GetField(1).GetArray()[0].GetString() << ": "
                          << packet.GetField(1).GetArray()[1].GetVarInt();
            }
            break;
        default:
            break;
    }
    if (g_IsGameStarted) {
        std::chrono::steady_clock clock;
        static auto timeOfPreviousSendedPpalPacket(clock.now());
        std::chrono::duration<double, std::milli> delta = clock.now() - timeOfPreviousSendedPpalPacket;
        if (delta.count() >= 50) {
            nc->AddPacketToQueue(
                    PacketBuilder::CPlay0x0D(g_PlayerX, g_PlayerY, g_PlayerZ, g_PlayerYaw, g_PlayerPitch, true));
            timeOfPreviousSendedPpalPacket = clock.now();
        }
    }

    delete packetPtr;
}