about summary refs log tree commit diff
path: root/immer/experimental/dvektor.hpp
blob: e809e749d5df4293857165ac26f844cbb6ad1055 (plain) (blame)
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
//
// immer: immutable data structures for C++
// Copyright (C) 2016, 2017, 2018 Juan Pedro Bolivar Puente
//
// This software is distributed under the Boost Software License, Version 1.0.
// See accompanying file LICENSE or copy at http://boost.org/LICENSE_1_0.txt
//

#pragma once

#include <immer/experimental/detail/dvektor_impl.hpp>

#include <immer/memory_policy.hpp>

namespace immer {

template <typename T, int B = 5, typename MemoryPolicy = default_memory_policy>
class dvektor
{
    using impl_t = detail::dvektor::impl<T, B, MemoryPolicy>;

public:
    using value_type      = T;
    using reference       = const T&;
    using size_type       = std::size_t;
    using difference_type = std::ptrdiff_t;
    using const_reference = const T&;

    using iterator         = detail::dvektor::iterator<T, B, MemoryPolicy>;
    using const_iterator   = iterator;
    using reverse_iterator = std::reverse_iterator<iterator>;

    dvektor() = default;

    iterator begin() const { return {impl_}; }
    iterator end() const { return {impl_, typename iterator::end_t{}}; }

    reverse_iterator rbegin() const { return reverse_iterator{end()}; }
    reverse_iterator rend() const { return reverse_iterator{begin()}; }

    std::size_t size() const { return impl_.size; }
    bool empty() const { return impl_.size == 0; }

    reference operator[](size_type index) const { return impl_.get(index); }

    dvektor push_back(value_type value) const
    {
        return {impl_.push_back(std::move(value))};
    }

    dvektor assoc(std::size_t idx, value_type value) const
    {
        return {impl_.assoc(idx, std::move(value))};
    }

    template <typename FnT>
    dvektor update(std::size_t idx, FnT&& fn) const
    {
        return {impl_.update(idx, std::forward<FnT>(fn))};
    }

private:
    dvektor(impl_t impl)
        : impl_(std::move(impl))
    {}
    impl_t impl_ = detail::dvektor::empty<T, B, MemoryPolicy>;
};

} // namespace immer