321 lines
9.8 KiB
C++
321 lines
9.8 KiB
C++
#pragma once
|
|
|
|
#include "StarNetElement.hpp"
|
|
#include "StarIdMap.hpp"
|
|
#include "StarStrongTypedef.hpp"
|
|
#include "StarDataStreamExtra.hpp"
|
|
|
|
namespace Star {
|
|
|
|
// A dynamic group of NetElements that manages creation and destruction of
|
|
// individual elements, that is itself a NetElement. Element changes are not
|
|
// delayed by the interpolation delay, they will always happen immediately, but
|
|
// this does not inhibit the Elements themselves from handling their own delta
|
|
// update delays normally.
|
|
template <typename Element>
|
|
class NetElementDynamicGroup : public NetElement {
|
|
public:
|
|
typedef shared_ptr<Element> ElementPtr;
|
|
typedef uint32_t ElementId;
|
|
static ElementId const NullElementId = 0;
|
|
|
|
NetElementDynamicGroup() = default;
|
|
|
|
NetElementDynamicGroup(NetElementDynamicGroup const&) = delete;
|
|
NetElementDynamicGroup& operator=(NetElementDynamicGroup const&) = delete;
|
|
|
|
// Must not call addNetElement / removeNetElement when being used as a slave,
|
|
// id errors will result.
|
|
ElementId addNetElement(ElementPtr element);
|
|
void removeNetElement(ElementId id);
|
|
|
|
// Remove all elements
|
|
void clearNetElements();
|
|
|
|
List<ElementId> netElementIds() const;
|
|
ElementPtr getNetElement(ElementId id) const;
|
|
|
|
List<ElementPtr> netElements() const;
|
|
|
|
void initNetVersion(NetElementVersion const* version = nullptr) override;
|
|
|
|
// Values are never interpolated, but they will be delayed for the given
|
|
// interpolationTime.
|
|
void enableNetInterpolation(float extrapolationHint = 0.0f) override;
|
|
void disableNetInterpolation() override;
|
|
void tickNetInterpolation(float dt) override;
|
|
|
|
void netStore(DataStream& ds, NetCompatibilityRules rules = {}) const override;
|
|
void netLoad(DataStream& ds, NetCompatibilityRules rules) override;
|
|
|
|
bool writeNetDelta(DataStream& ds, uint64_t fromVersion, NetCompatibilityRules rules = {}) const override;
|
|
void readNetDelta(DataStream& ds, float interpolationTime = 0.0f, NetCompatibilityRules rules = {}) override;
|
|
void blankNetDelta(float interpolationTime = 0.0f) override;
|
|
|
|
private:
|
|
// If a delta is written from further back than this many versions, the delta
|
|
// will fall back to a full serialization of the entire state.
|
|
static int64_t const MaxChangeDataVersions = 100;
|
|
|
|
typedef ElementId ElementRemovalType;
|
|
typedef pair<ElementId, ByteArray> ElementAdditionType;
|
|
|
|
strong_typedef(Empty, ElementReset);
|
|
strong_typedef_builtin(ElementRemovalType, ElementRemoval);
|
|
strong_typedef(ElementAdditionType, ElementAddition);
|
|
|
|
typedef Variant<ElementReset, ElementRemoval, ElementAddition> ElementChange;
|
|
|
|
typedef IdMap<ElementId, ElementPtr> ElementMap;
|
|
|
|
void addChangeData(ElementChange change);
|
|
|
|
void readyElement(ElementPtr const& element);
|
|
|
|
NetElementVersion const* m_netVersion = nullptr;
|
|
bool m_interpolationEnabled = false;
|
|
float m_extrapolationHint = 0.0f;
|
|
|
|
ElementMap m_idMap = ElementMap(1, highest<ElementId>());
|
|
|
|
Deque<pair<uint64_t, ElementChange>> m_changeData;
|
|
uint64_t m_changeDataLastVersion = 0;
|
|
|
|
mutable DataStreamBuffer m_buffer;
|
|
mutable HashSet<ElementId> m_receivedDeltaIds;
|
|
};
|
|
|
|
template <typename Element>
|
|
auto NetElementDynamicGroup<Element>::addNetElement(ElementPtr element) -> ElementId {
|
|
readyElement(element);
|
|
DataStreamBuffer storeBuffer;
|
|
element->netStore(storeBuffer, {});
|
|
auto id = m_idMap.add(std::move(element));
|
|
|
|
addChangeData(ElementAddition(id, storeBuffer.takeData()));
|
|
|
|
return id;
|
|
}
|
|
|
|
template <typename Element>
|
|
void NetElementDynamicGroup<Element>::removeNetElement(ElementId id) {
|
|
m_idMap.remove(id);
|
|
addChangeData(ElementRemoval{id});
|
|
}
|
|
|
|
template <typename Element>
|
|
void NetElementDynamicGroup<Element>::clearNetElements() {
|
|
for (auto const& id : netElementIds())
|
|
removeNetElement(id);
|
|
}
|
|
|
|
template <typename Element>
|
|
auto NetElementDynamicGroup<Element>::netElementIds() const -> List<ElementId> {
|
|
return m_idMap.keys();
|
|
}
|
|
|
|
template <typename Element>
|
|
auto NetElementDynamicGroup<Element>::getNetElement(ElementId id) const -> ElementPtr {
|
|
return m_idMap.get(id);
|
|
}
|
|
|
|
template <typename Element>
|
|
auto NetElementDynamicGroup<Element>::netElements() const -> List<ElementPtr> {
|
|
return m_idMap.values();
|
|
}
|
|
|
|
template <typename Element>
|
|
void NetElementDynamicGroup<Element>::initNetVersion(NetElementVersion const* version) {
|
|
m_netVersion = version;
|
|
m_changeData.clear();
|
|
m_changeDataLastVersion = 0;
|
|
|
|
addChangeData(ElementReset());
|
|
for (auto& pair : m_idMap) {
|
|
pair.second->initNetVersion(m_netVersion);
|
|
DataStreamBuffer storeBuffer;
|
|
pair.second->netStore(storeBuffer, {});
|
|
addChangeData(ElementAddition(pair.first, storeBuffer.takeData()));
|
|
}
|
|
}
|
|
|
|
template <typename Element>
|
|
void NetElementDynamicGroup<Element>::enableNetInterpolation(float extrapolationHint) {
|
|
m_interpolationEnabled = true;
|
|
m_extrapolationHint = extrapolationHint;
|
|
for (auto& p : m_idMap)
|
|
p.second->enableNetInterpolation(extrapolationHint);
|
|
}
|
|
|
|
template <typename Element>
|
|
void NetElementDynamicGroup<Element>::disableNetInterpolation() {
|
|
m_interpolationEnabled = false;
|
|
m_extrapolationHint = 0.0f;
|
|
for (auto& p : m_idMap)
|
|
p.second->disableNetInterpolation();
|
|
}
|
|
|
|
template <typename Element>
|
|
void NetElementDynamicGroup<Element>::tickNetInterpolation(float dt) {
|
|
for (auto& p : m_idMap)
|
|
p.second->tickNetInterpolation(dt);
|
|
}
|
|
|
|
template <typename Element>
|
|
void NetElementDynamicGroup<Element>::netStore(DataStream& ds, NetCompatibilityRules rules) const {
|
|
if (!checkWithRules(rules)) return;
|
|
ds.writeVlqU(m_idMap.size());
|
|
|
|
m_buffer.setStreamCompatibilityVersion(rules);
|
|
for (auto& pair : m_idMap) {
|
|
ds.writeVlqU(pair.first);
|
|
pair.second->netStore(m_buffer, rules);
|
|
ds.write(m_buffer.data());
|
|
m_buffer.clear();
|
|
}
|
|
}
|
|
|
|
template <typename Element>
|
|
void NetElementDynamicGroup<Element>::netLoad(DataStream& ds, NetCompatibilityRules rules) {
|
|
if (!checkWithRules(rules)) return;
|
|
m_changeData.clear();
|
|
m_changeDataLastVersion = m_netVersion ? m_netVersion->current() : 0;
|
|
m_idMap.clear();
|
|
|
|
addChangeData(ElementReset());
|
|
|
|
uint64_t count = ds.readVlqU();
|
|
|
|
for (uint64_t i = 0; i < count; ++i) {
|
|
ElementId id = ds.readVlqU();
|
|
DataStreamBuffer storeBuffer(ds.read<ByteArray>());
|
|
|
|
ElementPtr element = make_shared<Element>();
|
|
element->netLoad(storeBuffer, rules);
|
|
readyElement(element);
|
|
|
|
m_idMap.add(id, std::move(element));
|
|
addChangeData(ElementAddition(id, storeBuffer.takeData()));
|
|
}
|
|
}
|
|
|
|
template <typename Element>
|
|
bool NetElementDynamicGroup<Element>::writeNetDelta(DataStream& ds, uint64_t fromVersion, NetCompatibilityRules rules) const {
|
|
if (!checkWithRules(rules)) return false;
|
|
if (fromVersion < m_changeDataLastVersion) {
|
|
ds.write<bool>(true);
|
|
netStore(ds, rules);
|
|
return true;
|
|
|
|
} else {
|
|
bool deltaWritten = false;
|
|
auto willWrite = [&]() {
|
|
if (!deltaWritten) {
|
|
deltaWritten = true;
|
|
ds.write<bool>(false);
|
|
}
|
|
};
|
|
|
|
for (auto const& p : m_changeData) {
|
|
if (p.first >= fromVersion) {
|
|
willWrite();
|
|
ds.writeVlqU(1);
|
|
ds.write(p.second);
|
|
}
|
|
}
|
|
|
|
m_buffer.setStreamCompatibilityVersion(rules);
|
|
for (auto& p : m_idMap) {
|
|
if (p.second->writeNetDelta(m_buffer, fromVersion, rules)) {
|
|
willWrite();
|
|
ds.writeVlqU(p.first + 1);
|
|
ds.writeBytes(m_buffer.data());
|
|
m_buffer.clear();
|
|
}
|
|
}
|
|
|
|
if (deltaWritten)
|
|
ds.writeVlqU(0);
|
|
|
|
return deltaWritten;
|
|
}
|
|
}
|
|
|
|
template <typename Element>
|
|
void NetElementDynamicGroup<Element>::readNetDelta(DataStream& ds, float interpolationTime, NetCompatibilityRules rules) {
|
|
if (!checkWithRules(rules)) return;
|
|
bool isFull = ds.read<bool>();
|
|
if (isFull) {
|
|
netLoad(ds, rules);
|
|
} else {
|
|
while (true) {
|
|
uint64_t code = ds.readVlqU();
|
|
if (code == 0) {
|
|
break;
|
|
}
|
|
if (code == 1) {
|
|
auto changeUpdate = ds.read<ElementChange>();
|
|
addChangeData(changeUpdate);
|
|
|
|
if (changeUpdate.template is<ElementReset>()) {
|
|
m_idMap.clear();
|
|
} else if (auto addition = changeUpdate.template ptr<ElementAddition>()) {
|
|
ElementPtr element = make_shared<Element>();
|
|
DataStreamBuffer storeBuffer(std::move(get<1>(*addition)));
|
|
element->netLoad(storeBuffer, rules);
|
|
readyElement(element);
|
|
m_idMap.add(get<0>(*addition), std::move(element));
|
|
} else if (auto removal = changeUpdate.template ptr<ElementRemoval>()) {
|
|
m_idMap.remove(*removal);
|
|
}
|
|
} else {
|
|
ElementId elementId = code - 1;
|
|
auto const& element = m_idMap.get(elementId);
|
|
element->readNetDelta(ds, interpolationTime, rules);
|
|
if (m_interpolationEnabled)
|
|
m_receivedDeltaIds.add(elementId);
|
|
}
|
|
}
|
|
|
|
if (m_interpolationEnabled) {
|
|
for (auto& p : m_idMap) {
|
|
if (!m_receivedDeltaIds.contains(p.first))
|
|
p.second->blankNetDelta(interpolationTime);
|
|
}
|
|
|
|
m_receivedDeltaIds.clear();
|
|
}
|
|
}
|
|
}
|
|
|
|
template <typename Element>
|
|
void NetElementDynamicGroup<Element>::blankNetDelta(float interpolationTime) {
|
|
if (m_interpolationEnabled) {
|
|
for (auto& p : m_idMap)
|
|
p.second->blankNetDelta(interpolationTime);
|
|
}
|
|
}
|
|
|
|
template <typename Element>
|
|
void NetElementDynamicGroup<Element>::addChangeData(ElementChange change) {
|
|
uint64_t currentVersion = m_netVersion ? m_netVersion->current() : 0;
|
|
starAssert(m_changeData.empty() || m_changeData.last().first <= currentVersion);
|
|
|
|
m_changeData.append({currentVersion, std::move(change)});
|
|
|
|
m_changeDataLastVersion = max<int64_t>((int64_t)currentVersion - MaxChangeDataVersions, 0);
|
|
while (!m_changeData.empty() && m_changeData.first().first < m_changeDataLastVersion)
|
|
m_changeData.removeFirst();
|
|
}
|
|
|
|
template <typename Element>
|
|
void NetElementDynamicGroup<Element>::readyElement(ElementPtr const& element) {
|
|
element->initNetVersion(m_netVersion);
|
|
if (m_interpolationEnabled)
|
|
element->enableNetInterpolation(m_extrapolationHint);
|
|
else
|
|
element->disableNetInterpolation();
|
|
}
|
|
|
|
}
|