aboutsummaryrefslogtreecommitdiff
path: root/src/server/game/Entities/Object/Updates/WowCSEntityDefinitions.cpp
blob: b89d7aab275900d21bb330da7aaba7c4e34a2cdd (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
/*
 * This file is part of the TrinityCore Project. See AUTHORS file for Copyright information
 *
 * This program is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License as published by the
 * Free Software Foundation; either version 2 of the License, or (at your
 * option) any later version.
 *
 * This program is distributed in the hope that it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
 * more details.
 *
 * You should have received a copy of the GNU General Public License along
 * with this program. If not, see <http://www.gnu.org/licenses/>.
 */

#include "WowCSEntityDefinitions.h"
#include "Errors.h"
#include <algorithm>

namespace WowCS
{
void EntityFragmentsHolder::Add(EntityFragment fragment, bool update)
{
    ASSERT(Count < Ids.size());

    auto insertSorted = []<size_t N>(std::array<EntityFragment, N>& arr, uint8& count, EntityFragment f)
    {
        auto end = arr.begin() + count;
        auto where = std::ranges::lower_bound(arr.begin(), end, f);
        if (*where == f)
            return std::pair(where, false);

        std::ranges::move_backward(where, end, end + 1);
        ++count;
        *where = f;
        return std::pair(where, true);
    };

    if (!insertSorted(Ids, Count, fragment).second)
        return;

    if (IsUpdateableFragment(fragment))
    {
        ASSERT(UpdateableCount < UpdateableIds.size());

        auto insertedItr = insertSorted(UpdateableIds, UpdateableCount, fragment).first;
        std::ptrdiff_t index = std::ranges::distance(UpdateableIds.begin(), insertedItr);
        uint8 maskLowPart = ContentsChangedMask & ((1 << index) - 1);
        uint8 maskHighPart = (ContentsChangedMask & ~((1 << index) - 1)) << (1 + IsIndirectFragment(fragment));
        ContentsChangedMask = maskLowPart | maskHighPart;
        for (uint8 i = 0, maskIndex = 0; i < UpdateableCount; ++i)
        {
            UpdateableMasks[i] = 1 << maskIndex++;
            if (IsIndirectFragment(UpdateableIds[i]))
            {
                ContentsChangedMask |= UpdateableMasks[i]; // set the first bit to true to activate fragment
                ++maskIndex;
                UpdateableMasks[i] <<= 1;
            }
        }
    }

    if (update)
        IdsChanged = true;
}

void EntityFragmentsHolder::Remove(EntityFragment fragment)
{
    auto removeSorted = []<size_t N>(std::array<EntityFragment, N>& arr, uint8& count, EntityFragment f)
    {
        auto end = arr.begin() + count;
        auto where = std::ranges::find(arr.begin(), end, f);
        if (where != end)
        {
            *std::ranges::move(where + 1, end, where).out = EntityFragment::End;
            --count;
            return std::pair(where, true);
        }
        return std::pair(where, false);
    };

    if (!removeSorted(Ids, Count, fragment).second)
        return;

    if (IsUpdateableFragment(fragment))
    {
        auto [removedItr, removed] = removeSorted(UpdateableIds, UpdateableCount, fragment);
        if (removed)
        {
            std::ptrdiff_t index = std::ranges::distance(UpdateableIds.begin(), removedItr);
            uint8 maskLowPart = ContentsChangedMask & ((1 << index) - 1);
            uint8 maskHighPart = (ContentsChangedMask & ~((1 << index) - 1)) >> (1 + IsIndirectFragment(fragment));
            ContentsChangedMask = maskLowPart | maskHighPart;
            for (uint8 i = 0, maskIndex = 0; i < UpdateableCount; ++i)
            {
                UpdateableMasks[i] = 1 << maskIndex++;
                if (IsIndirectFragment(UpdateableIds[i]))
                {
                    ++maskIndex;
                    UpdateableMasks[i] <<= 1;
                }
            }
        }
    }

    IdsChanged = true;
}
}