summaryrefslogtreecommitdiff
path: root/include
diff options
context:
space:
mode:
authorPhilipp David <pd@3b.pm>2022-03-24 08:29:40 +0100
committerPhilipp David <pd@3b.pm>2022-03-24 08:29:48 +0100
commit129be45a7f91920e76673af104534d215c497d85 (patch)
tree2d34d59bac71a7c922e77c6ca5533cd9abceda94 /include
parent5bbeb216d27f60bb8063e9ab01f7ea6976a2e0b4 (diff)
downloadProject-Tick-129be45a7f91920e76673af104534d215c497d85.tar.gz
Project-Tick-129be45a7f91920e76673af104534d215c497d85.zip
Move templated methods to header
Diffstat (limited to 'include')
-rw-r--r--include/tag_array.h104
1 files changed, 104 insertions, 0 deletions
diff --git a/include/tag_array.h b/include/tag_array.h
index bed84d8070..a12c22688b 100644
--- a/include/tag_array.h
+++ b/include/tag_array.h
@@ -21,8 +21,11 @@
#define TAG_ARRAY_H_INCLUDED
#include "crtp_tag.h"
+#include "io/stream_reader.h"
+#include "io/stream_writer.h"
#include <type_traits>
#include <vector>
+#include <istream>
namespace nbt
{
@@ -121,6 +124,107 @@ template<class T> bool operator==(const tag_array<T>& lhs, const tag_array<T>& r
template<class T> bool operator!=(const tag_array<T>& lhs, const tag_array<T>& rhs)
{ return !(lhs == rhs); }
+//Slightly different between byte_array and int_array
+//Reading
+template<>
+inline void tag_array<int8_t>::read_payload(io::stream_reader& reader)
+{
+ int32_t length;
+ reader.read_num(length);
+ if(length < 0)
+ reader.get_istr().setstate(std::ios::failbit);
+ if(!reader.get_istr())
+ throw io::input_error("Error reading length of tag_byte_array");
+
+ data.resize(length);
+ reader.get_istr().read(reinterpret_cast<char*>(data.data()), length);
+ if(!reader.get_istr())
+ throw io::input_error("Error reading contents of tag_byte_array");
+}
+
+template<typename T>
+inline void tag_array<T>::read_payload(io::stream_reader& reader)
+{
+ int32_t length;
+ reader.read_num(length);
+ if(length < 0)
+ reader.get_istr().setstate(std::ios::failbit);
+ if(!reader.get_istr())
+ throw io::input_error("Error reading length of generic array tag");
+
+ data.clear();
+ data.reserve(length);
+ for(T i = 0; i < length; ++i)
+ {
+ T val;
+ reader.read_num(val);
+ data.push_back(val);
+ }
+ if(!reader.get_istr())
+ throw io::input_error("Error reading contents of generic array tag");
+}
+
+template<>
+inline void tag_array<int64_t>::read_payload(io::stream_reader& reader)
+{
+ int32_t length;
+ reader.read_num(length);
+ if(length < 0)
+ reader.get_istr().setstate(std::ios::failbit);
+ if(!reader.get_istr())
+ throw io::input_error("Error reading length of tag_long_array");
+
+ data.clear();
+ data.reserve(length);
+ for(int32_t i = 0; i < length; ++i)
+ {
+ int64_t val;
+ reader.read_num(val);
+ data.push_back(val);
+ }
+ if(!reader.get_istr())
+ throw io::input_error("Error reading contents of tag_long_array");
+}
+
+//Writing
+template<>
+inline void tag_array<int8_t>::write_payload(io::stream_writer& writer) const
+{
+ if(size() > io::stream_writer::max_array_len)
+ {
+ writer.get_ostr().setstate(std::ios::failbit);
+ throw std::length_error("Byte array is too large for NBT");
+ }
+ writer.write_num(static_cast<int32_t>(size()));
+ writer.get_ostr().write(reinterpret_cast<const char*>(data.data()), data.size());
+}
+
+template<typename T>
+inline void tag_array<T>::write_payload(io::stream_writer& writer) const
+{
+ if(size() > io::stream_writer::max_array_len)
+ {
+ writer.get_ostr().setstate(std::ios::failbit);
+ throw std::length_error("Generic array is too large for NBT");
+ }
+ writer.write_num(static_cast<int32_t>(size()));
+ for(T i: data)
+ writer.write_num(i);
+}
+
+template<>
+inline void tag_array<int64_t>::write_payload(io::stream_writer& writer) const
+{
+ if(size() > io::stream_writer::max_array_len)
+ {
+ writer.get_ostr().setstate(std::ios::failbit);
+ throw std::length_error("Long array is too large for NBT");
+ }
+ writer.write_num(static_cast<int32_t>(size()));
+ for(int64_t i: data)
+ writer.write_num(i);
+}
+
//Typedefs that should be used instead of the template tag_array.
typedef tag_array<int8_t> tag_byte_array;
typedef tag_array<int32_t> tag_int_array;