// 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_STL_CONVERTERS_H_ #define MOJO_PUBLIC_CPP_BINDINGS_STL_CONVERTERS_H_ #include <map> #include <string> #include <type_traits> #include <vector> #include "mojo/public/cpp/bindings/array.h" #include "mojo/public/cpp/bindings/map.h" #include "mojo/public/cpp/bindings/string.h" // Two functions are defined to facilitate conversion between // mojo::Array/Map/String and std::vector/map/string: mojo::UnwrapToSTLType() // recursively convert mojo types to STL types; mojo::WrapSTLType() does the // opposite. For example: // mojo::Array<mojo::Map<mojo::String, mojo::Array<int32_t>>> mojo_obj; // // std::vector<std::map<std::string, std::vector<int32_t>>> stl_obj = // mojo::UnwrapToSTLType(std::move(mojo_obj)); // // mojo_obj = mojo::WrapSTLType(std::move(stl_obj)); // // Notes: // - The conversion moves as much contents as possible. The two functions both // take an rvalue ref as input in order to avoid accidental copies. // - Because std::vector/map/string cannot express null, UnwrapToSTLType() // converts null mojo::Array/Map/String to empty. // - The recursive conversion stops at any types that are not the types listed // above. For example, unwrapping mojo::Array<StructContainingMojoMap> will // result in std::vector<StructContainingMojoMap>. It won't convert // mojo::Map inside the struct. namespace mojo { namespace internal { template <typename T> struct UnwrapTraits; template <typename T> struct UnwrapShouldGoDeeper { public: static const bool value = !std::is_same<T, typename UnwrapTraits<T>::Type>::value; }; template <typename T> struct UnwrapTraits { public: using Type = T; static Type Unwrap(T input) { return input; } }; template <typename T> struct UnwrapTraits<Array<T>> { public: using Type = std::vector<typename UnwrapTraits<T>::Type>; static Type Unwrap(Array<T> input) { return Helper<T>::Run(std::move(input)); } private: template <typename U, bool should_go_deeper = UnwrapShouldGoDeeper<U>::value> struct Helper {}; template <typename U> struct Helper<U, true> { public: static Type Run(Array<T> input) { Type output; output.reserve(input.size()); for (size_t i = 0; i < input.size(); ++i) output.push_back(UnwrapTraits<T>::Unwrap(std::move(input[i]))); return output; } }; template <typename U> struct Helper<U, false> { public: static Type Run(Array<T> input) { return input.PassStorage(); } }; }; template <typename K, typename V> struct UnwrapTraits<Map<K, V>> { public: using Type = std::map<typename UnwrapTraits<K>::Type, typename UnwrapTraits<V>::Type>; static Type Unwrap(Map<K, V> input) { return Helper<K, V>::Run(std::move(input)); } private: template <typename X, typename Y, bool should_go_deeper = UnwrapShouldGoDeeper<X>::value || UnwrapShouldGoDeeper<Y>::value> struct Helper {}; template <typename X, typename Y> struct Helper<X, Y, true> { public: static Type Run(Map<K, V> input) { std::map<K, V> input_storage = input.PassStorage(); Type output; for (auto& pair : input_storage) { output.insert( std::make_pair(UnwrapTraits<K>::Unwrap(pair.first), UnwrapTraits<V>::Unwrap(std::move(pair.second)))); } return output; } }; template <typename X, typename Y> struct Helper<X, Y, false> { public: static Type Run(Map<K, V> input) { return input.PassStorage(); } }; }; template <> struct UnwrapTraits<String> { public: using Type = std::string; static std::string Unwrap(const String& input) { return input; } }; template <typename T> struct WrapTraits; template <typename T> struct WrapShouldGoDeeper { public: static const bool value = !std::is_same<T, typename WrapTraits<T>::Type>::value; }; template <typename T> struct WrapTraits { public: using Type = T; static T Wrap(T input) { return input; } }; template <typename T> struct WrapTraits<std::vector<T>> { public: using Type = Array<typename WrapTraits<T>::Type>; static Type Wrap(std::vector<T> input) { return Helper<T>::Run(std::move(input)); } private: template <typename U, bool should_go_deeper = WrapShouldGoDeeper<U>::value> struct Helper {}; template <typename U> struct Helper<U, true> { public: static Type Run(std::vector<T> input) { std::vector<typename WrapTraits<T>::Type> output_storage; output_storage.reserve(input.size()); for (auto& element : input) output_storage.push_back(WrapTraits<T>::Wrap(std::move(element))); return Type(std::move(output_storage)); } }; template <typename U> struct Helper<U, false> { public: static Type Run(std::vector<T> input) { return Type(std::move(input)); } }; }; template <typename K, typename V> struct WrapTraits<std::map<K, V>> { public: using Type = Map<typename WrapTraits<K>::Type, typename WrapTraits<V>::Type>; static Type Wrap(std::map<K, V> input) { return Helper<K, V>::Run(std::move(input)); } private: template <typename X, typename Y, bool should_go_deeper = WrapShouldGoDeeper<X>::value || WrapShouldGoDeeper<Y>::value> struct Helper {}; template <typename X, typename Y> struct Helper<X, Y, true> { public: static Type Run(std::map<K, V> input) { Type output; for (auto& pair : input) { output.insert(WrapTraits<K>::Wrap(pair.first), WrapTraits<V>::Wrap(std::move(pair.second))); } return output; } }; template <typename X, typename Y> struct Helper<X, Y, false> { public: static Type Run(std::map<K, V> input) { return Type(std::move(input)); } }; }; template <> struct WrapTraits<std::string> { public: using Type = String; static String Wrap(const std::string& input) { return input; } }; } // namespace internal template <typename T> typename internal::UnwrapTraits<T>::Type UnwrapToSTLType(T&& input) { return internal::UnwrapTraits<T>::Unwrap(std::move(input)); } template <typename T> typename internal::WrapTraits<T>::Type WrapSTLType(T&& input) { return internal::WrapTraits<T>::Wrap(std::move(input)); } } // namespace mojo #endif // MOJO_PUBLIC_CPP_BINDINGS_STL_CONVERTERS_H_