#pragma once #include #include "StarNetElement.hpp" #include "StarString.hpp" #include "StarByteArray.hpp" namespace Star { template class NetElementBasicField : public NetElement { public: virtual ~NetElementBasicField() = default; T const& get() const; // Updates the value if the value is different than the existing value, // requires T have operator== void set(T const& value); // Always updates the value and marks it as updated. void push(T value); // Has this field been updated since the last call to pullUpdated? bool pullUpdated(); // Update the value in place. The mutator will be called as bool // mutator(T&), return true to signal that the value was updated. template void update(Mutator&& mutator); 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; protected: virtual void readData(DataStream& ds, T& t) const = 0; virtual void writeData(DataStream& ds, T const& t) const = 0; virtual void updated(); private: NetElementVersion const* m_netVersion = nullptr; uint64_t m_latestUpdateVersion = 0; T m_value = T(); bool m_updated = false; Maybe>> m_pendingInterpolatedValues; }; template class NetElementIntegral : public NetElementBasicField { protected: void readData(DataStream& ds, T& v) const override; void writeData(DataStream& ds, T const& v) const override; }; typedef NetElementIntegral NetElementInt; typedef NetElementIntegral NetElementUInt; // Properly encodes NPos no matter the platform width of size_t NetElement // size_t values are NOT clamped when setting. class NetElementSize : public NetElementBasicField { protected: void readData(DataStream& ds, size_t& v) const override; void writeData(DataStream& ds, size_t const& v) const override; }; class NetElementBool : public NetElementBasicField { protected: void readData(DataStream& ds, bool& v) const override; void writeData(DataStream& ds, bool const& v) const override; }; template class NetElementEnum : public NetElementBasicField { protected: void readData(DataStream& ds, Enum& v) const override; void writeData(DataStream& ds, Enum const& v) const override; }; // Wraps a uint64_t to give a simple event stream. Every trigger is an // increment to a held uint64_t value, and slaves can see how many triggers // have occurred since the last check. class NetElementEvent : public NetElementUInt { public: void trigger(); // Returns the number of times this event has been triggered since the last // pullOccurrences call. uint64_t pullOccurrences(); // Pulls whether this event occurred at all, ignoring the number bool pullOccurred(); // Ignore all the existing ocurrences void ignoreOccurrences(); void setIgnoreOccurrencesOnNetLoad(bool ignoreOccurrencesOnNetLoad); void netLoad(DataStream& ds, NetCompatibilityRules rules) override; protected: void updated() override; private: using NetElementUInt::get; using NetElementUInt::set; using NetElementUInt::push; using NetElementUInt::update; uint64_t m_pulledOccurrences = 0; bool m_ignoreOccurrencesOnNetLoad = false; }; // Holds an arbitrary serializable value template class NetElementData : public NetElementBasicField { public: NetElementData(); NetElementData(function reader, function writer); protected: void readData(DataStream& ds, T& v) const override; void writeData(DataStream& ds, T const& v) const override; private: function m_reader; function m_writer; }; typedef NetElementData NetElementString; typedef NetElementData NetElementBytes; template T const& NetElementBasicField::get() const { return m_value; } template void NetElementBasicField::set(T const& value) { if (!(m_value == value)) push(value); } template void NetElementBasicField::push(T value) { m_value = std::move(value); updated(); m_latestUpdateVersion = m_netVersion ? m_netVersion->current() : 0; if (m_pendingInterpolatedValues) m_pendingInterpolatedValues->clear(); } template bool NetElementBasicField::pullUpdated() { return take(m_updated); } template template void NetElementBasicField::update(Mutator&& mutator) { if (mutator(m_value)) { updated(); m_latestUpdateVersion = m_netVersion ? m_netVersion->current() : 0; if (m_pendingInterpolatedValues) m_pendingInterpolatedValues->clear(); } } template void NetElementBasicField::initNetVersion(NetElementVersion const* version) { m_netVersion = version; m_latestUpdateVersion = 0; } template void NetElementBasicField::enableNetInterpolation(float) { if (!m_pendingInterpolatedValues) m_pendingInterpolatedValues.emplace(); } template void NetElementBasicField::disableNetInterpolation() { if (m_pendingInterpolatedValues) { if (!m_pendingInterpolatedValues->empty()) m_value = m_pendingInterpolatedValues->takeLast().second; m_pendingInterpolatedValues.reset(); } } template void NetElementBasicField::tickNetInterpolation(float dt) { if (m_pendingInterpolatedValues) { for (auto& p : *m_pendingInterpolatedValues) p.first -= dt; while (!m_pendingInterpolatedValues->empty() && m_pendingInterpolatedValues->first().first <= 0.0f) { m_value = m_pendingInterpolatedValues->takeFirst().second; updated(); } } } template void NetElementBasicField::netStore(DataStream& ds, NetCompatibilityRules rules) const { if (!checkWithRules(rules)) return; if (m_pendingInterpolatedValues && !m_pendingInterpolatedValues->empty()) writeData(ds, m_pendingInterpolatedValues->last().second); else writeData(ds, m_value); } template void NetElementBasicField::netLoad(DataStream& ds, NetCompatibilityRules rules) { if (!checkWithRules(rules)) return; readData(ds, m_value); m_latestUpdateVersion = m_netVersion ? m_netVersion->current() : 0; updated(); if (m_pendingInterpolatedValues) m_pendingInterpolatedValues->clear(); } template bool NetElementBasicField::writeNetDelta(DataStream& ds, uint64_t fromVersion, NetCompatibilityRules rules) const { if (!checkWithRules(rules)) return false; if (m_latestUpdateVersion < fromVersion) return false; if (m_pendingInterpolatedValues && !m_pendingInterpolatedValues->empty()) writeData(ds, m_pendingInterpolatedValues->last().second); else writeData(ds, m_value); return true; } template void NetElementBasicField::readNetDelta(DataStream& ds, float interpolationTime, NetCompatibilityRules rules) { if (!checkWithRules(rules)) return; T t; readData(ds, t); m_latestUpdateVersion = m_netVersion ? m_netVersion->current() : 0; if (m_pendingInterpolatedValues) { // Only append an incoming delta to our pending value list if the incoming // step is forward in time of every other pending value. In any other // case, this is an error or the step tracking is wildly off, so just clear // any other incoming values. if (interpolationTime > 0.0f && (m_pendingInterpolatedValues->empty() || interpolationTime >= m_pendingInterpolatedValues->last().first)) { m_pendingInterpolatedValues->append({interpolationTime, std::move(t)}); } else { m_value = std::move(t); m_pendingInterpolatedValues->clear(); updated(); } } else { m_value = std::move(t); updated(); } } template void NetElementBasicField::updated() { m_updated = true; } template void NetElementIntegral::readData(DataStream& ds, T& v) const { if (sizeof(T) == 1) { ds.read(v); } else { if (std::is_unsigned::value) v = ds.readVlqU(); else v = ds.readVlqI(); } } template void NetElementIntegral::writeData(DataStream& ds, T const& v) const { if (sizeof(T) == 1) { ds.write(v); } else { if (std::is_unsigned::value) ds.writeVlqU(v); else ds.writeVlqI(v); } } template void NetElementEnum::readData(DataStream& ds, Enum& v) const { if (sizeof(Enum) == 1) ds.read(v); else v = (Enum)ds.readVlqI(); } template void NetElementEnum::writeData(DataStream& ds, Enum const& v) const { if (sizeof(Enum) == 1) ds.write(v); else ds.writeVlqI((int64_t)v); } template NetElementData::NetElementData() : NetElementData([](DataStream& ds, T & t) { ds >> t; }, [](DataStream& ds, T const& t) { ds << t; }) {} template NetElementData::NetElementData(function reader, function writer) : m_reader(std::move(reader)), m_writer(std::move(writer)) {} template void NetElementData::readData(DataStream& ds, T& v) const { m_reader(ds, v); } template void NetElementData::writeData(DataStream& ds, T const& v) const { m_writer(ds, v); } }