File: transform_view.h

package info (click to toggle)
chromium 138.0.7204.183-1
  • links: PTS, VCS
  • area: main
  • in suites: trixie
  • size: 6,071,908 kB
  • sloc: cpp: 34,937,088; ansic: 7,176,967; javascript: 4,110,704; python: 1,419,953; asm: 946,768; xml: 739,971; pascal: 187,324; sh: 89,623; perl: 88,663; objc: 79,944; sql: 50,304; cs: 41,786; fortran: 24,137; makefile: 21,806; php: 13,980; tcl: 13,166; yacc: 8,925; ruby: 7,485; awk: 3,720; lisp: 3,096; lex: 1,327; ada: 727; jsp: 228; sed: 36
file content (114 lines) | stat: -rw-r--r-- 3,885 bytes parent folder | download | duplicates (6)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
// Copyright 2024 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

#ifndef THIRD_PARTY_BLINK_RENDERER_PLATFORM_BINDINGS_TRANSFORM_VIEW_H_
#define THIRD_PARTY_BLINK_RENDERER_PLATFORM_BINDINGS_TRANSFORM_VIEW_H_

#include <stddef.h>

#include <concepts>
#include <iterator>
#include <utility>

#include "base/compiler_specific.h"

namespace blink::bindings {

// This is somewhat similar in spirit to std::range::views, but really lean,
// mean and specifically tailored to exposing ranges of blink data to the
// generated bindings where those expect an IDLSequence<>.
// Some limitations include:
// - TransformedView only offers a forward iterator, even if
//     unedrlying collection can offer better iterators;
// - No sentinel support
// - Transform instance is not accepted as an argument or preserved
//     across calls.
// All of these may be re-evaluated if new usages arise.

template <typename Range, typename Transform>
  requires(std::forward_iterator<decltype(std::begin(
               std::declval<const Range&>()))> &&
           std::regular_invocable<Transform,
                                  decltype(*std::begin(
                                      std::declval<const Range&>()))>)
class TransformedView {
 public:
  class TransformingIterator {
   public:
    using inner_iterator = decltype(std::begin(std::declval<const Range&>()));
    using inner_value_t = std::iter_value_t<inner_iterator>;

    // std::iterator interface
    // TODO(caseq): It may be nice to follow the category of the
    // underlying iterator instead of just offering a forward one.
    // We don't want to implement the entire bunch of operations required
    // by random_access_iterator so far, but we need to be able to compute
    // size for current use cases, so we offer size if underlying iterator
    // is a random access one.
    using iterator_category = std::forward_iterator_tag;
    using value_type = decltype(std::declval<Transform>()(
        std::declval<const inner_value_t&>()));

    TransformingIterator() = default;
    explicit TransformingIterator(inner_iterator it) : it_(it) {}
    TransformingIterator(const TransformingIterator& r) = default;
    TransformingIterator& operator=(const TransformingIterator& r) = default;

    bool operator==(const TransformingIterator& r) const {
      return it_ == r.it_;
    }
    bool operator!=(const TransformingIterator& r) const {
      return it_ != r.it_;
    }
    TransformingIterator& operator++() {
      UNSAFE_TODO(++it_);
      return *this;
    }
    TransformingIterator operator++(int) {
      return UNSAFE_TODO(TransformingIterator(it_++));
    }

    value_type operator*() const { return Transform()(*it_); }

    size_t operator-(const TransformingIterator& r) const
      requires(std::random_access_iterator<inner_iterator>)
    {
      return it_ - r.it_;
    }

   private:
    inner_iterator it_;
  };

  using inner_iterator = decltype(std::begin(std::declval<const Range&>()));
  using iterator_t = TransformingIterator;
  using value_type = iterator_t::value_type;

  TransformedView(inner_iterator begin, inner_iterator end)
      : begin_(begin), end_(end) {}
  explicit TransformedView(const Range& r)
      : begin_(std::begin(r)), end_(std::end(r)) {}

  iterator_t begin() const { return begin_; }
  iterator_t end() const { return end_; }

  size_t size() const
    requires(std::random_access_iterator<inner_iterator>)
  {
    return end_ - begin_;
  }
  bool empty() const { return begin_ == end_; }

  iterator_t begin_;
  iterator_t end_;
};

template <typename TransformFunc, typename Range>
auto Transform(const Range& range) {
  return TransformedView<Range, TransformFunc>(range);
}

}  // namespace blink::bindings

#endif  // THIRD_PARTY_BLINK_RENDERER_PLATFORM_BINDINGS_TRANSFORM_VIEW_H_