Index: mojo/public/cpp/bindings/wtf_map.h |
diff --git a/mojo/public/cpp/bindings/wtf_map.h b/mojo/public/cpp/bindings/wtf_map.h |
new file mode 100644 |
index 0000000000000000000000000000000000000000..319c44e6ec781e8de1652cbac2063cc93182a158 |
--- /dev/null |
+++ b/mojo/public/cpp/bindings/wtf_map.h |
@@ -0,0 +1,221 @@ |
+// Copyright 2016 The Chromium Authors. All rights reserved. |
+// Use of this source code is governed by a BSD-style license that can be |
+// found in the LICENSE file. |
+ |
+#ifndef MOJO_PUBLIC_CPP_BINDINGS_WTF_MAP_H_ |
+#define MOJO_PUBLIC_CPP_BINDINGS_WTF_MAP_H_ |
+ |
+#include <stddef.h> |
+#include <utility> |
+ |
+#include "base/move.h" |
+#include "mojo/public/cpp/bindings/lib/template_util.h" |
+#include "mojo/public/cpp/bindings/type_converter.h" |
+#include "third_party/WebKit/Source/wtf/HashMap.h" |
+#include "third_party/WebKit/Source/wtf/text/StringHash.h" |
+ |
+namespace mojo { |
+ |
+// Represents a map backed by WTF::HashMap. Comparing with WTF::HashMap, |
+// mojo::WTFMap is move-only and can be null. |
+// |
+// It is easy to convert between WTF::HashMap<K, V> and mojo::WTFMap<K, V>: |
+// - constructor WTFMap(WTF::HashMap<K, V>&&) takes the contents of a |
+// WTF::HashMap<K, V>; |
+// - method PassStorage() passes the underlying WTF::HashMap. |
+template <typename Key, typename Value> |
+class WTFMap { |
+ MOVE_ONLY_TYPE_FOR_CPP_03(WTFMap); |
+ |
+ public: |
+ using Iterator = typename WTF::HashMap<Key, Value>::iterator; |
+ using ConstIterator = typename WTF::HashMap<Key, Value>::const_iterator; |
+ |
+ // Constructs an empty map. |
+ WTFMap() : is_null_(false) {} |
+ // Constructs a null map. |
+ WTFMap(std::nullptr_t null_pointer) : is_null_(true) {} |
+ |
+ ~WTFMap() {} |
+ |
+ WTFMap(WTF::HashMap<Key, Value>&& other) |
+ : map_(std::move(other)), is_null_(false) {} |
+ WTFMap(WTFMap&& other) : is_null_(true) { Take(&other); } |
+ |
+ WTFMap& operator=(WTF::HashMap<Key, Value>&& other) { |
+ is_null_ = false; |
+ map_ = std::move(other); |
+ return *this; |
+ } |
+ WTFMap& operator=(WTFMap&& other) { |
+ Take(&other); |
+ return *this; |
+ } |
+ |
+ WTFMap& operator=(std::nullptr_t null_pointer) { |
+ is_null_ = true; |
+ map_.clear(); |
+ return *this; |
+ } |
+ |
+ // Copies the contents of some other type of map into a new WTFMap using a |
+ // TypeConverter. |
+ template <typename U> |
+ static WTFMap From(const U& other) { |
+ return TypeConverter<WTFMap, U>::Convert(other); |
+ } |
+ |
+ // Copies the contents of the WTFMap into some other type of map. |
+ template <typename U> |
+ U To() const { |
+ return TypeConverter<U, WTFMap>::Convert(*this); |
+ } |
+ |
+ // Indicates whether the map is null (which is distinct from empty). |
+ bool is_null() const { return is_null_; } |
+ |
+ // Indicates whether the map is empty (which is distinct from null). |
+ bool empty() const { return map_.isEmpty() && !is_null_; } |
+ |
+ // Indicates the number of keys in the map, which will be zero if the map is |
+ // null. |
+ size_t size() const { return map_.size(); } |
+ |
+ // Inserts a key-value pair into the map. Like WTF::HashMap::add(), this does |
+ // not insert |value| if |key| is already a member of the map. |
+ void insert(const Key& key, const Value& value) { |
+ is_null_ = false; |
+ map_.add(key, value); |
+ } |
+ void insert(const Key& key, Value&& value) { |
+ is_null_ = false; |
+ map_.add(key, std::move(value)); |
+ } |
+ |
+ // Returns a reference to the value associated with the specified key, |
+ // crashing the process if the key is not present in the map. |
+ Value& at(const Key& key) { return map_.find(key)->value; } |
+ const Value& at(const Key& key) const { return map_.find(key)->value; } |
+ |
+ // Returns a reference to the value associated with the specified key, |
+ // creating a new entry if the key is not already present in the map. A |
+ // newly-created value will be value-initialized (meaning that it will be |
+ // initialized by the default constructor of the value type, if any, or else |
+ // will be zero-initialized). |
+ Value& operator[](const Key& key) { |
+ is_null_ = false; |
+ if (!map_.contains(key)) |
+ map_.add(key, Value()); |
+ return at(key); |
+ } |
+ |
+ // Sets the map to empty (even if previously it was null). |
+ void SetToEmpty() { |
+ is_null_ = false; |
+ map_.clear(); |
+ } |
+ |
+ // Returns a const reference to the WTF::HashMap managed by this class. If |
+ // this object is null, the return value will be an empty map. |
+ const WTF::HashMap<Key, Value>& storage() const { return map_; } |
+ |
+ // Passes the underlying storage and resets this map to null. |
+ WTF::HashMap<Key, Value> PassStorage() { |
+ is_null_ = true; |
+ return std::move(map_); |
+ } |
+ |
+ // Swaps the contents of this WTFMap with another WTFMap of the same type |
+ // (including nullness). |
+ void Swap(WTFMap<Key, Value>* other) { |
+ std::swap(is_null_, other->is_null_); |
+ map_.swap(other->map_); |
+ } |
+ |
+ // Swaps the contents of this WTFMap with an WTF::HashMap containing keys and |
+ // values of the same type. Since WTF::HashMap cannot represent the null |
+ // state, the WTF::HashMap will be empty if WTFMap is null. The WTFMap will |
+ // always be left in a non-null state. |
+ void Swap(WTF::HashMap<Key, Value>* other) { |
+ is_null_ = false; |
+ map_.swap(*other); |
+ } |
+ |
+ // Returns a new WTFMap that contains a copy of the contents of this map. If |
+ // the key/value type defines a Clone() method, it will be used; otherwise |
+ // copy constructor/assignment will be used. |
+ // |
+ // Please note that calling this method will fail compilation if the key/value |
+ // type cannot be cloned (which usually means that it is a Mojo handle type or |
+ // a type containing Mojo handles). |
+ WTFMap Clone() const { |
+ WTFMap result; |
+ result.is_null_ = is_null_; |
+ auto map_end = map_.end(); |
+ for (auto it = map_.begin(); it != map_end; ++it) |
+ result.map_.add(internal::Clone(it->key), internal::Clone(it->value)); |
+ return result; |
+ } |
+ |
+ // Indicates whether the contents of this map are equal to those of another |
+ // WTFMap (including nullness). If the key/value type defines an Equals() |
+ // method, it will be used; otherwise == operator will be used. |
+ bool Equals(const WTFMap& other) const { |
+ if (is_null() != other.is_null()) |
+ return false; |
+ if (size() != other.size()) |
+ return false; |
+ |
+ auto this_end = map_.end(); |
+ auto other_end = other.map_.end(); |
+ |
+ for (auto iter = map_.begin(); iter != this_end; ++iter) { |
+ auto other_iter = other.map_.find(iter->key); |
+ if (other_iter == other_end || |
+ !internal::Equals(iter->value, other_iter->value)) { |
+ return false; |
+ } |
+ } |
+ return true; |
+ } |
+ |
+ ConstIterator begin() const { return map_.begin(); } |
+ Iterator begin() { return map_.begin(); } |
+ |
+ ConstIterator end() const { return map_.end(); } |
+ Iterator end() { return map_.end(); } |
+ |
+ // Returns the iterator pointing to the entry for |key|, if present, or else |
+ // returns end(). |
+ ConstIterator find(const Key& key) const { return map_.find(key); } |
+ Iterator find(const Key& key) { return map_.find(key); } |
+ |
+ private: |
+ typedef WTF::HashMap<Key, Value> WTFMap::*Testable; |
+ |
+ public: |
+ // The WTFMap may be used in boolean expressions to determine if it is |
+ // non-null, but is not implicitly convertible to an actual bool value (which |
+ // would be dangerous). |
+ operator Testable() const { return is_null_ ? 0 : &WTFMap::map_; } |
Yuta Kitamura
2016/06/06 05:29:42
Now you can safely use "explicit operator bool()".
yzshen1
2016/06/07 23:49:11
Done. Good point!
|
+ |
+ private: |
+ // Forbid the == and != operators explicitly, otherwise WTFMap will be |
+ // converted to Testable to do == or != comparison. |
+ template <typename T, typename U> |
+ bool operator==(const WTFMap<T, U>& other) const = delete; |
+ template <typename T, typename U> |
+ bool operator!=(const WTFMap<T, U>& other) const = delete; |
+ |
+ void Take(WTFMap* other) { |
+ operator=(nullptr); |
+ Swap(other); |
+ } |
+ |
+ WTF::HashMap<Key, Value> map_; |
Yuta Kitamura
2016/06/06 05:29:42
What's a bit tricky is that WTF::HashMap does not
yzshen1
2016/06/07 23:49:11
I have added comments. And also added code to reje
|
+ bool is_null_; |
+}; |
+ |
+} // namespace mojo |
+ |
+#endif // MOJO_PUBLIC_CPP_BINDINGS_WTF_MAP_H_ |