@@ -1841,8 +1841,9 @@ class PointeeMatcher {
1841
1841
template <typename Pointer>
1842
1842
class Impl : public MatcherInterface <Pointer> {
1843
1843
public:
1844
- typedef typename PointeeOf<GTEST_REMOVE_REFERENCE_AND_CONST_(Pointer)>::type
1845
- Pointee;
1844
+ using Pointee =
1845
+ typename std::pointer_traits<GTEST_REMOVE_REFERENCE_AND_CONST_(
1846
+ Pointer)>::element_type;
1846
1847
1847
1848
explicit Impl (const InnerMatcher& matcher)
1848
1849
: matcher_(MatcherCast<const Pointee&>(matcher)) {}
@@ -1872,6 +1873,64 @@ class PointeeMatcher {
1872
1873
const InnerMatcher matcher_;
1873
1874
};
1874
1875
1876
+ // Implements the Pointer(m) matcher
1877
+ // Implements the Pointer(m) matcher for matching a pointer that matches matcher
1878
+ // m. The pointer can be either raw or smart, and will match `m` against the
1879
+ // raw pointer.
1880
+ template <typename InnerMatcher>
1881
+ class PointerMatcher {
1882
+ public:
1883
+ explicit PointerMatcher (const InnerMatcher& matcher) : matcher_(matcher) {}
1884
+
1885
+ // This type conversion operator template allows Pointer(m) to be
1886
+ // used as a matcher for any pointer type whose pointer type is
1887
+ // compatible with the inner matcher, where type PointerType can be
1888
+ // either a raw pointer or a smart pointer.
1889
+ //
1890
+ // The reason we do this instead of relying on
1891
+ // MakePolymorphicMatcher() is that the latter is not flexible
1892
+ // enough for implementing the DescribeTo() method of Pointer().
1893
+ template <typename PointerType>
1894
+ operator Matcher<PointerType>() const { // NOLINT
1895
+ return Matcher<PointerType>(new Impl<const PointerType&>(matcher_));
1896
+ }
1897
+
1898
+ private:
1899
+ // The monomorphic implementation that works for a particular pointer type.
1900
+ template <typename PointerType>
1901
+ class Impl : public MatcherInterface <PointerType> {
1902
+ public:
1903
+ using Pointer =
1904
+ const typename std::pointer_traits<GTEST_REMOVE_REFERENCE_AND_CONST_(
1905
+ PointerType)>::element_type*;
1906
+
1907
+ explicit Impl (const InnerMatcher& matcher)
1908
+ : matcher_(MatcherCast<Pointer>(matcher)) {}
1909
+
1910
+ void DescribeTo (::std::ostream* os) const override {
1911
+ *os << " is a pointer that " ;
1912
+ matcher_.DescribeTo (os);
1913
+ }
1914
+
1915
+ void DescribeNegationTo (::std::ostream* os) const override {
1916
+ *os << " is not a pointer that " ;
1917
+ matcher_.DescribeTo (os);
1918
+ }
1919
+
1920
+ bool MatchAndExplain (PointerType pointer,
1921
+ MatchResultListener* listener) const override {
1922
+ *listener << " which is a pointer that " ;
1923
+ Pointer p = GetRawPointer (pointer);
1924
+ return MatchPrintAndExplain (p, matcher_, listener);
1925
+ }
1926
+
1927
+ private:
1928
+ Matcher<Pointer> matcher_;
1929
+ };
1930
+
1931
+ const InnerMatcher matcher_;
1932
+ };
1933
+
1875
1934
#if GTEST_HAS_RTTI
1876
1935
// Implements the WhenDynamicCastTo<T>(m) matcher that matches a pointer or
1877
1936
// reference that matches inner_matcher when dynamic_cast<T> is applied.
@@ -4720,6 +4779,14 @@ internal::FieldsAreMatcher<typename std::decay<M>::type...> FieldsAre(
4720
4779
return internal::FieldsAreMatcher<typename std::decay<M>::type...>(
4721
4780
std::forward<M>(matchers)...);
4722
4781
}
4782
+
4783
+ // Creates a matcher that matches a pointer (raw or smart) that matches
4784
+ // inner_matcher.
4785
+ template <typename InnerMatcher>
4786
+ inline internal::PointerMatcher<InnerMatcher> Pointer (
4787
+ const InnerMatcher& inner_matcher) {
4788
+ return internal::PointerMatcher<InnerMatcher>(inner_matcher);
4789
+ }
4723
4790
} // namespace no_adl
4724
4791
4725
4792
// Returns a predicate that is satisfied by anything that matches the
0 commit comments