xref: /aosp_15_r20/external/pdfium/third_party/base/containers/contains.h (revision 3ac0a46f773bac49fa9476ec2b1cf3f8da5ec3a4)
1*3ac0a46fSAndroid Build Coastguard Worker // Copyright 2021 The Chromium Authors. All rights reserved.
2*3ac0a46fSAndroid Build Coastguard Worker // Use of this source code is governed by a BSD-style license that can be
3*3ac0a46fSAndroid Build Coastguard Worker // found in the LICENSE file.
4*3ac0a46fSAndroid Build Coastguard Worker 
5*3ac0a46fSAndroid Build Coastguard Worker #ifndef THIRD_PARTY_BASE_CONTAINERS_CONTAINS_H_
6*3ac0a46fSAndroid Build Coastguard Worker #define THIRD_PARTY_BASE_CONTAINERS_CONTAINS_H_
7*3ac0a46fSAndroid Build Coastguard Worker 
8*3ac0a46fSAndroid Build Coastguard Worker #include <algorithm>
9*3ac0a46fSAndroid Build Coastguard Worker #include <iterator>
10*3ac0a46fSAndroid Build Coastguard Worker #include <type_traits>
11*3ac0a46fSAndroid Build Coastguard Worker 
12*3ac0a46fSAndroid Build Coastguard Worker #include "third_party/base/template_util.h"
13*3ac0a46fSAndroid Build Coastguard Worker 
14*3ac0a46fSAndroid Build Coastguard Worker namespace pdfium {
15*3ac0a46fSAndroid Build Coastguard Worker 
16*3ac0a46fSAndroid Build Coastguard Worker namespace internal {
17*3ac0a46fSAndroid Build Coastguard Worker 
18*3ac0a46fSAndroid Build Coastguard Worker // Small helper to detect whether a given type has a nested `key_type` typedef.
19*3ac0a46fSAndroid Build Coastguard Worker // Used below to catch misuses of the API for associative containers.
20*3ac0a46fSAndroid Build Coastguard Worker template <typename T, typename SFINAE = void>
21*3ac0a46fSAndroid Build Coastguard Worker struct HasKeyType : std::false_type {};
22*3ac0a46fSAndroid Build Coastguard Worker 
23*3ac0a46fSAndroid Build Coastguard Worker template <typename T>
24*3ac0a46fSAndroid Build Coastguard Worker struct HasKeyType<T, std::void_t<typename T::key_type>> : std::true_type {};
25*3ac0a46fSAndroid Build Coastguard Worker 
26*3ac0a46fSAndroid Build Coastguard Worker // Utility type traits used for specializing base::Contains() below.
27*3ac0a46fSAndroid Build Coastguard Worker template <typename Container, typename Element, typename = void>
28*3ac0a46fSAndroid Build Coastguard Worker struct HasFindWithNpos : std::false_type {};
29*3ac0a46fSAndroid Build Coastguard Worker 
30*3ac0a46fSAndroid Build Coastguard Worker template <typename Container, typename Element>
31*3ac0a46fSAndroid Build Coastguard Worker struct HasFindWithNpos<
32*3ac0a46fSAndroid Build Coastguard Worker     Container,
33*3ac0a46fSAndroid Build Coastguard Worker     Element,
34*3ac0a46fSAndroid Build Coastguard Worker     std::void_t<decltype(std::declval<const Container&>().find(
35*3ac0a46fSAndroid Build Coastguard Worker                              std::declval<const Element&>()) !=
36*3ac0a46fSAndroid Build Coastguard Worker                          Container::npos)>> : std::true_type {};
37*3ac0a46fSAndroid Build Coastguard Worker 
38*3ac0a46fSAndroid Build Coastguard Worker template <typename Container, typename Element, typename = void>
39*3ac0a46fSAndroid Build Coastguard Worker struct HasFindWithEnd : std::false_type {};
40*3ac0a46fSAndroid Build Coastguard Worker 
41*3ac0a46fSAndroid Build Coastguard Worker template <typename Container, typename Element>
42*3ac0a46fSAndroid Build Coastguard Worker struct HasFindWithEnd<
43*3ac0a46fSAndroid Build Coastguard Worker     Container,
44*3ac0a46fSAndroid Build Coastguard Worker     Element,
45*3ac0a46fSAndroid Build Coastguard Worker     std::void_t<decltype(std::declval<const Container&>().find(
46*3ac0a46fSAndroid Build Coastguard Worker                              std::declval<const Element&>()) !=
47*3ac0a46fSAndroid Build Coastguard Worker                          std::declval<const Container&>().end())>>
48*3ac0a46fSAndroid Build Coastguard Worker     : std::true_type {};
49*3ac0a46fSAndroid Build Coastguard Worker 
50*3ac0a46fSAndroid Build Coastguard Worker template <typename Container, typename Element, typename = void>
51*3ac0a46fSAndroid Build Coastguard Worker struct HasContains : std::false_type {};
52*3ac0a46fSAndroid Build Coastguard Worker 
53*3ac0a46fSAndroid Build Coastguard Worker template <typename Container, typename Element>
54*3ac0a46fSAndroid Build Coastguard Worker struct HasContains<
55*3ac0a46fSAndroid Build Coastguard Worker     Container,
56*3ac0a46fSAndroid Build Coastguard Worker     Element,
57*3ac0a46fSAndroid Build Coastguard Worker     std::void_t<decltype(std::declval<const Container&>().contains(
58*3ac0a46fSAndroid Build Coastguard Worker         std::declval<const Element&>()))>> : std::true_type {};
59*3ac0a46fSAndroid Build Coastguard Worker 
60*3ac0a46fSAndroid Build Coastguard Worker }  // namespace internal
61*3ac0a46fSAndroid Build Coastguard Worker 
62*3ac0a46fSAndroid Build Coastguard Worker // General purpose implementation to check if |container| contains |value|.
63*3ac0a46fSAndroid Build Coastguard Worker template <typename Container,
64*3ac0a46fSAndroid Build Coastguard Worker           typename Value,
65*3ac0a46fSAndroid Build Coastguard Worker           std::enable_if_t<
66*3ac0a46fSAndroid Build Coastguard Worker               !internal::HasFindWithNpos<Container, Value>::value &&
67*3ac0a46fSAndroid Build Coastguard Worker               !internal::HasFindWithEnd<Container, Value>::value &&
68*3ac0a46fSAndroid Build Coastguard Worker               !internal::HasContains<Container, Value>::value>* = nullptr>
69*3ac0a46fSAndroid Build Coastguard Worker bool Contains(const Container& container, const Value& value) {
70*3ac0a46fSAndroid Build Coastguard Worker   static_assert(
71*3ac0a46fSAndroid Build Coastguard Worker       !internal::HasKeyType<Container>::value,
72*3ac0a46fSAndroid Build Coastguard Worker       "Error: About to perform linear search on an associative container. "
73*3ac0a46fSAndroid Build Coastguard Worker       "Either use a more generic comparator (e.g. std::less<>) or, if a linear "
74*3ac0a46fSAndroid Build Coastguard Worker       "search is desired, provide an explicit projection parameter.");
75*3ac0a46fSAndroid Build Coastguard Worker   using std::begin;
76*3ac0a46fSAndroid Build Coastguard Worker   using std::end;
77*3ac0a46fSAndroid Build Coastguard Worker   return std::find(begin(container), end(container), value) != end(container);
78*3ac0a46fSAndroid Build Coastguard Worker }
79*3ac0a46fSAndroid Build Coastguard Worker 
80*3ac0a46fSAndroid Build Coastguard Worker // Specialized Contains() implementation for when |container| has a find()
81*3ac0a46fSAndroid Build Coastguard Worker // member function and a static npos member, but no contains() member function.
82*3ac0a46fSAndroid Build Coastguard Worker template <typename Container,
83*3ac0a46fSAndroid Build Coastguard Worker           typename Value,
84*3ac0a46fSAndroid Build Coastguard Worker           std::enable_if_t<internal::HasFindWithNpos<Container, Value>::value &&
85*3ac0a46fSAndroid Build Coastguard Worker                            !internal::HasContains<Container, Value>::value>* =
86*3ac0a46fSAndroid Build Coastguard Worker               nullptr>
87*3ac0a46fSAndroid Build Coastguard Worker bool Contains(const Container& container, const Value& value) {
88*3ac0a46fSAndroid Build Coastguard Worker   return container.find(value) != Container::npos;
89*3ac0a46fSAndroid Build Coastguard Worker }
90*3ac0a46fSAndroid Build Coastguard Worker 
91*3ac0a46fSAndroid Build Coastguard Worker // Specialized Contains() implementation for when |container| has a find()
92*3ac0a46fSAndroid Build Coastguard Worker // and end() member function, but no contains() member function.
93*3ac0a46fSAndroid Build Coastguard Worker template <typename Container,
94*3ac0a46fSAndroid Build Coastguard Worker           typename Value,
95*3ac0a46fSAndroid Build Coastguard Worker           std::enable_if_t<internal::HasFindWithEnd<Container, Value>::value &&
96*3ac0a46fSAndroid Build Coastguard Worker                            !internal::HasContains<Container, Value>::value>* =
97*3ac0a46fSAndroid Build Coastguard Worker               nullptr>
98*3ac0a46fSAndroid Build Coastguard Worker bool Contains(const Container& container, const Value& value) {
99*3ac0a46fSAndroid Build Coastguard Worker   return container.find(value) != container.end();
100*3ac0a46fSAndroid Build Coastguard Worker }
101*3ac0a46fSAndroid Build Coastguard Worker 
102*3ac0a46fSAndroid Build Coastguard Worker // Specialized Contains() implementation for when |container| has a contains()
103*3ac0a46fSAndroid Build Coastguard Worker // member function.
104*3ac0a46fSAndroid Build Coastguard Worker template <
105*3ac0a46fSAndroid Build Coastguard Worker     typename Container,
106*3ac0a46fSAndroid Build Coastguard Worker     typename Value,
107*3ac0a46fSAndroid Build Coastguard Worker     std::enable_if_t<internal::HasContains<Container, Value>::value>* = nullptr>
108*3ac0a46fSAndroid Build Coastguard Worker bool Contains(const Container& container, const Value& value) {
109*3ac0a46fSAndroid Build Coastguard Worker   return container.contains(value);
110*3ac0a46fSAndroid Build Coastguard Worker }
111*3ac0a46fSAndroid Build Coastguard Worker 
112*3ac0a46fSAndroid Build Coastguard Worker }  // namespace pdfium
113*3ac0a46fSAndroid Build Coastguard Worker 
114*3ac0a46fSAndroid Build Coastguard Worker #endif  // THIRD_PARTY_BASE_CONTAINERS_CONTAINS_H_
115