2023-01-13 21:28:03 +00:00
|
|
|
#pragma once
|
|
|
|
#include <luna/Alloc.h>
|
|
|
|
#include <luna/CString.h>
|
|
|
|
#include <luna/Result.h>
|
|
|
|
#include <luna/Types.h>
|
|
|
|
|
|
|
|
template <typename T> class Vector
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
typedef T* Iterator;
|
|
|
|
typedef const T* ConstIterator;
|
|
|
|
|
|
|
|
Vector()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2023-01-14 09:50:28 +00:00
|
|
|
Vector(const Vector<T>& other)
|
|
|
|
{
|
|
|
|
reserve(other.capacity());
|
|
|
|
memcpy(m_data, other.data(), other.size());
|
|
|
|
m_size = other.size();
|
|
|
|
}
|
|
|
|
|
|
|
|
Vector(Vector<T>&& other)
|
|
|
|
{
|
|
|
|
m_data = other.data();
|
|
|
|
m_capacity = other.capacity();
|
|
|
|
m_size = other.size();
|
|
|
|
|
|
|
|
other.m_capacity = other.m_size = 0;
|
|
|
|
other.m_data = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
Vector<T>& operator=(const Vector<T>& other)
|
|
|
|
{
|
|
|
|
if (&other == this) return *this;
|
|
|
|
|
|
|
|
if (m_data) free_impl(m_data);
|
|
|
|
|
|
|
|
m_data = nullptr;
|
|
|
|
m_capacity = m_size = 0;
|
|
|
|
|
|
|
|
reserve(other.capacity());
|
|
|
|
memcpy(m_data, other.data(), other.size());
|
|
|
|
m_size = other.size();
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
Vector<T>& operator=(Vector<T>&& other)
|
|
|
|
{
|
|
|
|
if (&other == this) return *this;
|
|
|
|
|
|
|
|
if (m_data) free_impl(m_data);
|
|
|
|
|
|
|
|
m_data = other.data();
|
|
|
|
m_capacity = other.capacity();
|
|
|
|
m_size = other.size();
|
|
|
|
|
|
|
|
other.m_capacity = other.m_size = 0;
|
|
|
|
other.m_data = nullptr;
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
~Vector()
|
|
|
|
{
|
2023-04-08 12:46:58 +00:00
|
|
|
if (m_data)
|
|
|
|
{
|
|
|
|
for (const T& item : *this) { item.~T(); }
|
2023-04-01 08:56:21 +00:00
|
|
|
free_impl(m_data);
|
2023-04-08 12:46:58 +00:00
|
|
|
}
|
2023-01-14 09:50:28 +00:00
|
|
|
}
|
|
|
|
|
2023-01-13 21:28:03 +00:00
|
|
|
Result<void> try_reserve(usize capacity)
|
|
|
|
{
|
|
|
|
return resize(capacity);
|
|
|
|
}
|
|
|
|
|
|
|
|
void reserve(usize capacity)
|
|
|
|
{
|
|
|
|
resize(capacity).release_value();
|
|
|
|
}
|
|
|
|
|
2023-02-27 14:02:55 +00:00
|
|
|
Result<void> try_append(T&& item)
|
2023-01-13 21:28:03 +00:00
|
|
|
{
|
|
|
|
if (m_capacity == m_size) TRY(resize(m_capacity + 8));
|
|
|
|
|
|
|
|
new (&m_data[m_size]) T(move(item));
|
|
|
|
|
|
|
|
m_size++;
|
|
|
|
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2023-02-27 14:02:55 +00:00
|
|
|
Result<void> try_append(const T& item)
|
|
|
|
{
|
|
|
|
return try_append(T(item));
|
|
|
|
}
|
|
|
|
|
2023-01-13 21:28:03 +00:00
|
|
|
Option<T> try_pop()
|
|
|
|
{
|
|
|
|
if (m_size == 0) return {};
|
|
|
|
|
|
|
|
m_size--;
|
|
|
|
|
|
|
|
return move(m_data[m_size]);
|
|
|
|
}
|
|
|
|
|
2023-03-29 16:27:02 +00:00
|
|
|
Option<T> try_dequeue()
|
|
|
|
{
|
|
|
|
if (m_size == 0) return {};
|
|
|
|
|
2023-04-12 16:11:36 +00:00
|
|
|
return remove_at(0);
|
2023-03-29 16:27:02 +00:00
|
|
|
}
|
|
|
|
|
2023-01-13 21:28:03 +00:00
|
|
|
const T& operator[](usize index) const
|
|
|
|
{
|
|
|
|
check(index < m_size);
|
2023-03-28 23:05:30 +00:00
|
|
|
return m_data[index];
|
2023-01-13 21:28:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
T& operator[](usize index)
|
|
|
|
{
|
|
|
|
check(index < m_size);
|
2023-03-28 23:05:30 +00:00
|
|
|
return m_data[index];
|
2023-01-13 21:28:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Iterator begin()
|
|
|
|
{
|
|
|
|
return m_data;
|
|
|
|
}
|
|
|
|
|
|
|
|
ConstIterator begin() const
|
|
|
|
{
|
|
|
|
return m_data;
|
|
|
|
}
|
|
|
|
|
|
|
|
Iterator end()
|
|
|
|
{
|
|
|
|
return m_data + m_size;
|
|
|
|
}
|
|
|
|
|
|
|
|
ConstIterator end() const
|
|
|
|
{
|
|
|
|
return m_data + m_size;
|
|
|
|
}
|
|
|
|
|
|
|
|
const T* data() const
|
|
|
|
{
|
|
|
|
return m_data;
|
|
|
|
}
|
|
|
|
|
|
|
|
T* data()
|
|
|
|
{
|
|
|
|
return m_data;
|
|
|
|
}
|
|
|
|
|
|
|
|
usize capacity() const
|
|
|
|
{
|
|
|
|
return m_capacity;
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:07:08 +00:00
|
|
|
usize byte_capacity() const
|
|
|
|
{
|
|
|
|
return m_capacity * sizeof(T);
|
|
|
|
}
|
|
|
|
|
2023-01-13 21:28:03 +00:00
|
|
|
usize size() const
|
|
|
|
{
|
|
|
|
return m_size;
|
|
|
|
}
|
|
|
|
|
2023-04-12 16:11:36 +00:00
|
|
|
T remove_at(usize index)
|
|
|
|
{
|
|
|
|
check(index < m_size);
|
|
|
|
|
|
|
|
T item = move(m_data[index]);
|
|
|
|
|
|
|
|
memmove(m_data + index, m_data + (index + 1), (m_size - (index + 1)) * sizeof(T));
|
|
|
|
|
|
|
|
m_size--;
|
|
|
|
|
|
|
|
return move(item);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename Callback> Option<T> remove_first_matching(Callback callback)
|
|
|
|
{
|
|
|
|
Option<usize> index = {};
|
|
|
|
|
|
|
|
for (usize i = 0; i < m_size; i++)
|
|
|
|
{
|
|
|
|
if (callback(m_data[i]))
|
|
|
|
{
|
|
|
|
index = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!index.has_value()) return {};
|
|
|
|
|
|
|
|
return remove_at(index.value());
|
|
|
|
}
|
|
|
|
|
2023-03-23 20:20:41 +00:00
|
|
|
void clear()
|
|
|
|
{
|
2023-04-12 16:11:36 +00:00
|
|
|
for (usize i = 0; i < m_size; i++) { m_data[i].~T(); }
|
|
|
|
|
2023-03-23 20:20:41 +00:00
|
|
|
m_size = m_capacity = 0;
|
|
|
|
free_impl(m_data);
|
2023-03-23 21:19:43 +00:00
|
|
|
m_data = nullptr;
|
2023-03-23 20:20:41 +00:00
|
|
|
}
|
|
|
|
|
2023-01-13 21:28:03 +00:00
|
|
|
private:
|
|
|
|
T* m_data { nullptr };
|
|
|
|
usize m_capacity { 0 };
|
|
|
|
usize m_size { 0 };
|
|
|
|
|
|
|
|
Result<void> resize(usize new_capacity)
|
|
|
|
{
|
2023-03-10 20:07:08 +00:00
|
|
|
const usize new_byte_capacity = new_capacity * sizeof(T);
|
|
|
|
|
|
|
|
void* const ptr = TRY(realloc_impl(m_data, new_byte_capacity));
|
|
|
|
|
2023-01-13 21:28:03 +00:00
|
|
|
m_capacity = new_capacity;
|
|
|
|
m_data = (T*)ptr;
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
};
|