FTXUI 6.1.9
C++ functional terminal UI.
Loading...
Searching...
No Matches
ref.hpp
Go to the documentation of this file.
1// Copyright 2020 Arthur Sonzogni. All rights reserved.
2// Use of this source code is governed by the MIT license that can be found in
3// the LICENSE file.
4#ifndef FTXUI_UTIL_REF_HPP
5#define FTXUI_UTIL_REF_HPP
6
8#include <memory>
9#include <string>
10#include <variant>
11#include <vector>
12
13namespace ftxui {
14
15/// @brief 一個適配器。擁有或引用一個不可變的物件。
16template <typename T>
17class ConstRef {
18 public:
19 ConstRef() = default;
20 ConstRef(T t) : variant_(std::move(t)) {} // NOLINT
21 ConstRef(const T* t) : variant_(t) {} // NOLINT
22 ConstRef& operator=(ConstRef&&) noexcept = default;
23 ConstRef(const ConstRef<T>&) = default;
24 ConstRef(ConstRef<T>&&) noexcept = default;
25 ~ConstRef() = default;
26
27 // 建立一個「可重新設定」的引用
28 ConstRef<T>& operator=(const ConstRef<T>&) = default;
29
30 // 存取器:
31 const T& operator()() const { return *Address(); }
32 const T& operator*() const { return *Address(); }
33 const T* operator->() const { return Address(); }
34
35 private:
36 std::variant<T, const T*> variant_ = T{};
37
38 const T* Address() const {
39 return std::holds_alternative<T>(variant_) ? &std::get<T>(variant_)
40 : std::get<const T*>(variant_);
41 }
42};
43
44/// @brief 一個適配器。擁有或引用一個可變的物件。
45template <typename T>
46class Ref {
47 public:
48 Ref() = default;
49 Ref(T t) : variant_(std::move(t)) {} // NOLINT
50 Ref(T* t) : variant_(t) {} // NOLINT
51 ~Ref() = default;
52 Ref& operator=(Ref&&) noexcept = default;
53 Ref(const Ref<T>&) = default;
54 Ref(Ref<T>&&) noexcept = default;
55
56 // 建立一個「可重新設定」的引用。
57 Ref<T>& operator=(const Ref<T>&) = default;
58
59 // 存取器:
60 T& operator()() { return *Address(); }
61 T& operator*() { return *Address(); }
62 T* operator->() { return Address(); }
63 const T& operator()() const { return *Address(); }
64 const T& operator*() const { return *Address(); }
65 const T* operator->() const { return Address(); }
66
67 private:
68 std::variant<T, T*> variant_ = T{};
69
70 const T* Address() const {
71 return std::holds_alternative<T>(variant_) ? &std::get<T>(variant_)
72 : std::get<T*>(variant_);
73 }
74 T* Address() {
75 return std::holds_alternative<T>(variant_) ? &std::get<T>(variant_)
76 : std::get<T*>(variant_);
77 }
78};
79
80/// @brief 一個適配器。擁有或引用一個常數字串。為方便起見,此類別將多個可變字串轉換為共享表示。
81class StringRef : public Ref<std::string> {
82 public:
83 using Ref<std::string>::Ref;
84
85 StringRef(const wchar_t* ref) // NOLINT
86 : StringRef(to_string(std::wstring(ref))) {}
87 StringRef(const char* ref) // NOLINT
88 : StringRef(std::string(ref)) {}
89};
90
91/// @brief 一個適配器。擁有或引用一個常數字串。為方便起見,此類別將多個不可變字串轉換為共享表示。
92class ConstStringRef : public ConstRef<std::string> {
93 public:
94 using ConstRef<std::string>::ConstRef;
95
96 ConstStringRef(const std::wstring* ref) // NOLINT
97 : ConstStringRef(to_string(*ref)) {}
98 ConstStringRef(const std::wstring ref) // NOLINT
99 : ConstStringRef(to_string(ref)) {}
100 ConstStringRef(const wchar_t* ref) // NOLINT
101 : ConstStringRef(to_string(std::wstring(ref))) {}
102 ConstStringRef(const char* ref) // NOLINT
103 : ConstStringRef(std::string(ref)) {}
104};
105
106/// @brief 一個適配器。引用一個字串列表。
107///
108/// 支援的輸入:
109/// - `std::vector<std::string>`
110/// - `std::vector<std::string>*`
111/// - `std::vector<std::wstring>*`
112/// - `Adapter*`
113/// - `std::unique_ptr<Adapter>`
115 public:
116 // 使用您自己的適配器:
117 class Adapter {
118 public:
119 Adapter() = default;
120 Adapter(const Adapter&) = default;
121 Adapter& operator=(const Adapter&) = default;
122 Adapter(Adapter&&) = default;
123 Adapter& operator=(Adapter&&) = default;
124 virtual ~Adapter() = default;
125 virtual size_t size() const = 0;
126 virtual std::string operator[](size_t i) const = 0;
127 };
128 using Variant = std::variant<const std::vector<std::string>, //
129 const std::vector<std::string>*, //
130 const std::vector<std::wstring>*, //
131 Adapter*, //
132 std::unique_ptr<Adapter> //
133 >;
134
141
142 ConstStringListRef(std::vector<std::string> value) // NOLINT
143 {
144 variant_ = std::make_shared<Variant>(value);
145 }
146 ConstStringListRef(const std::vector<std::string>* value) // NOLINT
147 {
148 variant_ = std::make_shared<Variant>(value);
149 }
150 ConstStringListRef(const std::vector<std::wstring>* value) // NOLINT
151 {
152 variant_ = std::make_shared<Variant>(value);
153 }
154 ConstStringListRef(Adapter* adapter) // NOLINT
155 {
156 variant_ = std::make_shared<Variant>(adapter);
157 }
158 template <typename AdapterType>
159 ConstStringListRef(std::unique_ptr<AdapterType> adapter) // NOLINT
160 {
161 variant_ = std::make_shared<Variant>(
162 static_cast<std::unique_ptr<Adapter>>(std::move(adapter)));
163 }
164
165 size_t size() const {
166 return variant_ ? std::visit(SizeVisitor(), *variant_) : 0;
167 }
168
169 std::string operator[](size_t i) const {
170 return variant_ ? std::visit(IndexedGetter(i), *variant_) : "";
171 }
172
173 private:
174 struct SizeVisitor {
175 size_t operator()(const std::vector<std::string>& v) const {
176 return v.size();
177 }
178 size_t operator()(const std::vector<std::string>* v) const {
179 return v->size();
180 }
181 size_t operator()(const std::vector<std::wstring>* v) const {
182 return v->size();
183 }
184 size_t operator()(const Adapter* v) const { return v->size(); }
185 size_t operator()(const std::unique_ptr<Adapter>& v) const {
186 return v->size();
187 }
188 };
189
190 struct IndexedGetter {
191 IndexedGetter(size_t index) // NOLINT
192 : index_(index) {}
193 size_t index_;
194 std::string operator()(const std::vector<std::string>& v) const {
195 return v[index_];
196 }
197 std::string operator()(const std::vector<std::string>* v) const {
198 return (*v)[index_];
199 }
200 std::string operator()(const std::vector<std::wstring>* v) const {
201 return to_string((*v)[index_]);
202 }
203 std::string operator()(const Adapter* v) const { return (*v)[index_]; }
204 std::string operator()(const std::unique_ptr<Adapter>& v) const {
205 return (*v)[index_];
206 }
207 };
208
209 std::shared_ptr<Variant> variant_;
210};
211
212} // namespace ftxui
213
214#endif /* end of include guard: FTXUI_UTIL_REF_HPP */
一個適配器。擁有或引用一個不可變的物件。
Definition ref.hpp:17
const T & operator*() const
Definition ref.hpp:32
ConstRef()=default
ConstRef(const T *t)
Definition ref.hpp:21
ConstRef(T t)
Definition ref.hpp:20
const T * operator->() const
Definition ref.hpp:33
ConstRef & operator=(ConstRef &&) noexcept=default
Adapter & operator=(Adapter &&)=default
Adapter & operator=(const Adapter &)=default
virtual size_t size() const =0
virtual std::string operator[](size_t i) const =0
Adapter(const Adapter &)=default
一個適配器。引用一個字串列表。
Definition ref.hpp:114
ConstStringListRef(std::vector< std::string > value)
Definition ref.hpp:142
ConstStringListRef(const std::vector< std::string > *value)
Definition ref.hpp:146
size_t size() const
Definition ref.hpp:165
ConstStringListRef(ConstStringListRef &&)=default
ConstStringListRef & operator=(ConstStringListRef &&)=default
std::variant< const std::vector< std::string >, const std::vector< std::string > *, const std::vector< std::wstring > *, Adapter *, std::unique_ptr< Adapter > > Variant
Definition ref.hpp:128
ConstStringListRef & operator=(const ConstStringListRef &)=default
ConstStringListRef(std::unique_ptr< AdapterType > adapter)
Definition ref.hpp:159
ConstStringListRef(const std::vector< std::wstring > *value)
Definition ref.hpp:150
ConstStringListRef(const ConstStringListRef &)=default
ConstStringListRef(Adapter *adapter)
Definition ref.hpp:154
std::string operator[](size_t i) const
Definition ref.hpp:169
一個適配器。擁有或引用一個常數字串。為方便起見,此類別將多個不可變字串轉換為共享表示。
Definition ref.hpp:92
ConstStringRef(const char *ref)
Definition ref.hpp:102
ConstStringRef(const std::wstring ref)
Definition ref.hpp:98
ConstStringRef(const std::wstring *ref)
Definition ref.hpp:96
ConstStringRef(const wchar_t *ref)
Definition ref.hpp:100
一個適配器。擁有或引用一個可變的物件。
Definition ref.hpp:46
const T & operator*() const
Definition ref.hpp:64
~Ref()=default
T & operator*()
Definition ref.hpp:61
T * operator->()
Definition ref.hpp:62
Ref()=default
Ref(T t)
Definition ref.hpp:49
const T * operator->() const
Definition ref.hpp:65
Ref & operator=(Ref &&) noexcept=default
Ref(T *t)
Definition ref.hpp:50
const T & operator()() const
Definition ref.hpp:63
一個適配器。擁有或引用一個常數字串。為方便起見,此類別將多個可變字串轉換為共享表示。
Definition ref.hpp:81
StringRef(const char *ref)
Definition ref.hpp:87
StringRef(const wchar_t *ref)
Definition ref.hpp:85
FTXUI 的 ftxui:: 命名空間
Definition animation.hpp:10
std::string to_string(const std::wstring &s)
將 std::wstring 轉換為 UTF8 std::string。
Definition string.cpp:1542