forked from dart-lang/co19
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathstatic_type_helper.dart
90 lines (80 loc) · 2.87 KB
/
static_type_helper.dart
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
// Copyright (c) 2020, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
/// Helper to create [Type] values.
Type typeOf<T>() => T;
/// Ensures a context type of [T] for the operand.
Object? context<T>(T x) => x;
/// Captures the context type of the call and returns the same type.
///
/// Can be used to check the context type as:
/// ```dart
/// int x = contextType(1 /* valid value */)..expectStaticType<Exactly<int>>;
/// ```
T contextType<T>(Object result) => result as T;
extension StaticType<T> on T {
/// Check the static type.
///
/// Use as follows (assuming `e` has static type `num`):
/// ```dart
/// e.expectStaticType<Exactly<num>>() // No context type.
/// e.expectStaticType<SubtypeOf<Object>>() // No context type.
/// e.expectStaticType<SupertypeOf<int>>() // No context type.
/// ```
/// or
/// ```dart
/// e..expectStaticType<Exactly<num>>() // Preserve context type.
/// e..expectStaticType<SubtypeOf<Object>>() // Preserve context type.
/// e..expectStaticType<SupertypeOf<int>>() // Preserve context type.
/// ```
/// This will be a *compile-time error* if the static type is not
/// as required by the constraints type (the one passed to [Exactly],
/// [SubtypeOf] or [SupertypeOf].)
T expectStaticType<R extends Exactly<T>>() {
return this;
}
/// Invokes [callback] with the static type of `this`.
///
/// Allows any operation on the type.
T captureStaticType(void Function<X>() callback) {
callback<T>();
return this;
}
}
/// Invokes [callback] with the static type of [value].
///
/// Similar to [StaticType.captureStaticType], but works
/// for types like `void` and `dynamic` which do not allow
/// extension methods.
void captureStaticType<T>(T value, void Function<X>(X value) callback) {
callback<T>(value);
}
/// Use with [StaticType.expectStaticType] to expect precisely the type [T].
///
/// Example use:
/// ```dart
/// "abc".expectStaticType<Exactly<String>>();
/// ```
typedef Exactly<T> = T Function(T);
/// Use with [StaticType.expectStaticType] to expect a subtype of [T].
///
/// Example use:
/// ```dart
/// num x = 1;
/// x.expectStaticType<SubtypeOf<Object>>();
/// ```
typedef SubtypeOf<T> = Never Function(T);
/// Use with [StaticType.expectStaticType] to expect a supertype of [T].
///
/// Example use:
/// ```dart
/// num x = 1;
/// x.expectStaticType<SupertypeOf<int>>();
/// ```
typedef SupertypeOf<T> = T Function(Object?);
/// Checks that an expression is assignable to [T1], [T2] and [Object].
///
/// This ensures that the static type of the expression is either dynamic,
/// Never, or a type assignable to both [T1] and [T2], and if those are
/// unrelated, it must be an intersection type.
void checkIntersectionType<T1, T2>(T1 v1, T2 v2, Object v3) {}