llvm/libcxx/test/std/ranges/range.adaptors/range.elements/adaptor.pass.cpp

//===----------------------------------------------------------------------===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//

// UNSUPPORTED: c++03, c++11, c++14, c++17

// std::views::elements<N>
// std::views::keys
// std::views::values

#include <algorithm>
#include <cassert>
#include <ranges>
#include <tuple>
#include <type_traits>
#include <utility>

#include "test_range.h"

template <class T>
struct View : std::ranges::view_base {
  T* begin() const;
  T* end() const;
};

static_assert(!std::is_invocable_v<decltype((std::views::elements<0>))>);
static_assert(!std::is_invocable_v<decltype((std::views::elements<0>)), View<int>>);
static_assert(std::is_invocable_v<decltype((std::views::elements<0>)), View<std::pair<int, int>>>);
static_assert(std::is_invocable_v<decltype((std::views::elements<0>)), View<std::tuple<int>>>);
static_assert(!std::is_invocable_v<decltype((std::views::elements<5>)), View<std::tuple<int>>>);

static_assert(!std::is_invocable_v<decltype((std::views::keys))>);
static_assert(!std::is_invocable_v<decltype((std::views::keys)), View<int>>);
static_assert(std::is_invocable_v<decltype((std::views::keys)), View<std::pair<int, int>>>);
static_assert(std::is_invocable_v<decltype((std::views::keys)), View<std::tuple<int>>>);

static_assert(!std::is_invocable_v<decltype((std::views::values))>);
static_assert(!std::is_invocable_v<decltype((std::views::values)), View<int>>);
static_assert(std::is_invocable_v<decltype((std::views::values)), View<std::pair<int, int>>>);
static_assert(!std::is_invocable_v<decltype((std::views::values)), View<std::tuple<int>>>);

static_assert(!CanBePiped<View<int>, decltype((std::views::elements<0>))>);
static_assert(CanBePiped<View<std::pair<int, int>>, decltype((std::views::elements<0>))>);
static_assert(CanBePiped<View<std::tuple<int>>, decltype((std::views::elements<0>))>);
static_assert(!CanBePiped<View<std::tuple<int>>, decltype((std::views::elements<5>))>);

static_assert(!CanBePiped<View<int>, decltype((std::views::keys))>);
static_assert(CanBePiped<View<std::pair<int, int>>, decltype((std::views::keys))>);
static_assert(CanBePiped<View<std::tuple<int>>, decltype((std::views::keys))>);

static_assert(!CanBePiped<View<int>, decltype((std::views::values))>);
static_assert(CanBePiped<View<std::pair<int, int>>, decltype((std::views::values))>);
static_assert(!CanBePiped<View<std::tuple<int>>, decltype((std::views::values))>);

constexpr bool test() {
  std::pair<int, int> buff[] = {{1, 2}, {3, 4}, {5, 6}};

  // Test `views::elements<N>(v)`
  {
    using Result = std::ranges::elements_view<std::ranges::ref_view<std::pair<int, int>[3]>, 0>;
    std::same_as<Result> decltype(auto) result = std::views::elements<0>(buff);
    auto expected                              = {1, 3, 5};
    assert(std::ranges::equal(result, expected));
  }

  // Test `views::keys(v)`
  {
    using Result = std::ranges::elements_view<std::ranges::ref_view<std::pair<int, int>[3]>, 0>;
    std::same_as<Result> decltype(auto) result = std::views::keys(buff);
    auto expected                              = {1, 3, 5};
    assert(std::ranges::equal(result, expected));
  }

  // Test `views::values(v)`
  {
    using Result = std::ranges::elements_view<std::ranges::ref_view<std::pair<int, int>[3]>, 1>;
    std::same_as<Result> decltype(auto) result = std::views::values(buff);
    auto expected                              = {2, 4, 6};
    assert(std::ranges::equal(result, expected));
  }

  // Test `v | views::elements<N>`
  {
    using Result = std::ranges::elements_view<std::ranges::ref_view<std::pair<int, int>[3]>, 1>;
    std::same_as<Result> decltype(auto) result = buff | std::views::elements<1>;
    auto expected                              = {2, 4, 6};
    assert(std::ranges::equal(result, expected));
  }

  // Test `v | views::keys`
  {
    using Result = std::ranges::elements_view<std::ranges::ref_view<std::pair<int, int>[3]>, 0>;
    std::same_as<Result> decltype(auto) result = buff | std::views::keys;
    auto expected                              = {1, 3, 5};
    assert(std::ranges::equal(result, expected));
  }

  // Test `v | views::values`
  {
    using Result = std::ranges::elements_view<std::ranges::ref_view<std::pair<int, int>[3]>, 1>;
    std::same_as<Result> decltype(auto) result = buff | std::views::values;
    auto expected                              = {2, 4, 6};
    assert(std::ranges::equal(result, expected));
  }

  // Test views::elements<0> | views::elements<0>
  {
    std::pair<std::tuple<int>, std::tuple<int>> nested[] = {{{1}, {2}}, {{3}, {4}}, {{5}, {6}}};
    using Result                                         = std::ranges::elements_view<
        std::ranges::elements_view<std::ranges::ref_view<std::pair<std::tuple<int>, std::tuple<int>>[3]>, 0>,
        0>;
    auto const partial                         = std::views::elements<0> | std::views::elements<0>;
    std::same_as<Result> decltype(auto) result = nested | partial;
    auto expected                              = {1, 3, 5};
    assert(std::ranges::equal(result, expected));
  }

  // Test views::keys | views::keys
  {
    std::pair<std::tuple<int>, std::tuple<int>> nested[] = {{{1}, {2}}, {{3}, {4}}, {{5}, {6}}};
    using Result                                         = std::ranges::elements_view<
        std::ranges::elements_view<std::ranges::ref_view<std::pair<std::tuple<int>, std::tuple<int>>[3]>, 0>,
        0>;
    auto const partial                         = std::views::keys | std::views::keys;
    std::same_as<Result> decltype(auto) result = nested | partial;
    auto expected                              = {1, 3, 5};
    assert(std::ranges::equal(result, expected));
  }

  // Test views::values | views::values
  {
    std::pair<std::tuple<int>, std::tuple<int, int>> nested[] = {{{1}, {2, 3}}, {{4}, {5, 6}}, {{7}, {8, 9}}};
    using Result                                              = std::ranges::elements_view<
        std::ranges::elements_view<std::ranges::ref_view<std::pair<std::tuple<int>, std::tuple<int, int>>[3]>, 1>,
        1>;
    auto const partial                         = std::views::values | std::views::values;
    std::same_as<Result> decltype(auto) result = nested | partial;
    auto expected                              = {3, 6, 9};
    assert(std::ranges::equal(result, expected));
  }

  // Test views::keys | views::values
  {
    std::pair<std::tuple<int, int>, std::tuple<int>> nested[] = {{{1, 2}, {3}}, {{4, 5}, {6}}, {{7, 8}, {9}}};
    using Result                                              = std::ranges::elements_view<
        std::ranges::elements_view<std::ranges::ref_view<std::pair<std::tuple<int, int>, std::tuple<int>>[3]>, 0>,
        1>;
    auto const partial                         = std::views::keys | std::views::values;
    std::same_as<Result> decltype(auto) result = nested | partial;
    auto expected                              = {2, 5, 8};
    assert(std::ranges::equal(result, expected));
  }

  // Test views::values | views::keys
  {
    std::pair<std::tuple<int>, std::tuple<int, int>> nested[] = {{{1}, {2, 3}}, {{4}, {5, 6}}, {{7}, {8, 9}}};
    using Result                                              = std::ranges::elements_view<
        std::ranges::elements_view<std::ranges::ref_view<std::pair<std::tuple<int>, std::tuple<int, int>>[3]>, 1>,
        0>;
    auto const partial                         = std::views::values | std::views::keys;
    std::same_as<Result> decltype(auto) result = nested | partial;
    auto expected                              = {2, 5, 8};
    assert(std::ranges::equal(result, expected));
  }

  return true;
}

int main(int, char**) {
  test();
  static_assert(test());

  return 0;
}