aboutsummaryrefslogtreecommitdiff
path: root/toolsrc/include/vcpkg/base/stringview.h
blob: 9238990fef737b3b657e5c62f986a971bfbc7a56 (plain)
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
#pragma once

#include <vcpkg/base/fwd/stringview.h>

#include <vcpkg/base/optional.h>

#include <limits>
#include <string>
#include <vector>

namespace vcpkg
{
    struct StringView
    {
        static std::vector<StringView> find_all_enclosed(const StringView& input,
                                                         const std::string& left_delim,
                                                         const std::string& right_delim);

        static StringView find_exactly_one_enclosed(const StringView& input,
                                                    const std::string& left_tag,
                                                    const std::string& right_tag);

        static Optional<StringView> find_at_most_one_enclosed(const StringView& input,
                                                              const std::string& left_tag,
                                                              const std::string& right_tag);

        constexpr StringView() = default;
        StringView(const std::string& s); // Implicit by design

        // NOTE: we do this instead of the delegating constructor since delegating ctors are a perf footgun
        template<size_t Sz>
        constexpr StringView(const char (&arr)[Sz]) : m_ptr(arr), m_size(Sz - 1)
        {
        }

        constexpr StringView(const char* ptr, size_t size) : m_ptr(ptr), m_size(size) { }
        constexpr StringView(const char* b, const char* e) : m_ptr(b), m_size(static_cast<size_t>(e - b)) { }

        constexpr const char* begin() const { return m_ptr; }
        constexpr const char* end() const { return m_ptr + m_size; }

        std::reverse_iterator<const char*> rbegin() const { return std::make_reverse_iterator(end()); }
        std::reverse_iterator<const char*> rend() const { return std::make_reverse_iterator(begin()); }

        constexpr const char* data() const { return m_ptr; }
        constexpr size_t size() const { return m_size; }

        std::string to_string() const;
        void to_string(std::string& out) const;

        StringView substr(size_t pos, size_t count = std::numeric_limits<size_t>::max()) const;

        constexpr char byte_at_index(size_t pos) const { return m_ptr[pos]; }

    private:
        const char* m_ptr = 0;
        size_t m_size = 0;
    };

    bool operator==(StringView lhs, StringView rhs) noexcept;
    bool operator!=(StringView lhs, StringView rhs) noexcept;
    bool operator<(StringView lhs, StringView rhs) noexcept;
    bool operator>(StringView lhs, StringView rhs) noexcept;
    bool operator<=(StringView lhs, StringView rhs) noexcept;
    bool operator>=(StringView lhs, StringView rhs) noexcept;
}