blob: 7eee56b13234d7960dc262d3ae7ca366b6a74fc9 [file] [log] [blame]
// Copyright (c) 2015, 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.
// Verify that the static type of a ??= b is the least upper bound of the
// static types of a and b.
import "package:expect/expect.dart";
bad() {
Expect.fail('Should not be executed');
}
/// Actually of type B so that the implicit downcasts below succeed at runtime.
A theA = new B();
B theB = new B();
class A {
String? a;
}
class B extends A {
String? b;
}
class C extends A {
String? c;
}
A? get a => null;
void set a(A? value) {}
B? get b => null;
void set b(B? value) {}
class ClassWithStaticGetters {
static A? get a => null;
static void set a(A? value) {}
static B? get b => null;
static void set b(B? value) {}
}
class ClassWithInstanceGetters {
A? get a => null;
void set a(A? value) {}
B? get b => null;
void set b(B? value) {}
}
class DerivedClass extends ClassWithInstanceGetters {
A? get a => bad();
void set a(A? value) {
bad();
}
B? get b => bad();
void set b(B? value) {
bad();
}
void derivedTest() {
// The static type of super.v ??= e is the LUB of NonNull(typeof(super.v))
// and typeof(e).
(super.a ??= theA).a; //# 01: ok
(super.a ??= theA).b; //# 02: compile-time error
(super.a ??= theB).a; //# 03: ok
(super.a ??= theB).b; //# 04: compile-time error
(super.b ??= theA).a; //# 05: compile-time error
(super.b ??= theA).b; //# 06: compile-time error
// Exactly the same static errors that would be caused by super.v = e
// are also generated in the case of super.v ??= e.
super.b ??= new C(); //# 07: compile-time error
}
}
main() {
new DerivedClass().derivedTest();
// The static type of v ??= e is the LUB of NonNull(typeof(v)) and typeof(e).
(a ??= theA).a; //# 08: ok
(a ??= theA).b; //# 09: compile-time error
(a ??= theB).a; //# 10: ok
(a ??= theB).b; //# 11: compile-time error
(b ??= theA).a; //# 12: compile-time error
(b ??= theA).b; //# 13: compile-time error
// Exactly the same static errors that would be caused by v = e are also
// generated in the case of v ??= e.
b ??= new C(); //# 14: compile-time error
// The static type of C.v ??= e is the LUB of NonNull(typeof(C.v)) and
// typeof(e).
(ClassWithStaticGetters.a ??= theA).a; //# 15: ok
(ClassWithStaticGetters.a ??= theA).b; //# 16: compile-time error
(ClassWithStaticGetters.a ??= theB).a; //# 17: ok
(ClassWithStaticGetters.a ??= theB).b; //# 18: compile-time error
(ClassWithStaticGetters.b ??= theA).a; //# 19: compile-time error
(ClassWithStaticGetters.b ??= theA).b; //# 20: compile-time error
// Exactly the same static errors that would be caused by C.v = e are
// also generated in the case of C.v ??= e.
ClassWithStaticGetters.b ??= new C(); //# 21: compile-time error
// The static type of e1.v ??= e2 is the LUB of NonNull(typeof(e1.v)) and
// typeof(e2).
(new ClassWithInstanceGetters().a ??= theA).a; //# 22: ok
(new ClassWithInstanceGetters().a ??= theA).b; //# 23: compile-time error
(new ClassWithInstanceGetters().a ??= theB).a; //# 24: ok
(new ClassWithInstanceGetters().a ??= theB).b; //# 25: compile-time error
(new ClassWithInstanceGetters().b ??= theA).a; //# 26: compile-time error
(new ClassWithInstanceGetters().b ??= theA).b; //# 27: compile-time error
// Exactly the same static errors that would be caused by e1.v = e2 are
// also generated in the case of e1.v ??= e2.
new ClassWithInstanceGetters().b ??= new C(); //# 28: compile-time error
// The static type of e1[e2] ??= e3 is the LUB of NonNull(typeof(e1[e2])) and
// typeof(e3).
((<A?>[null])[0] ??= theA).a; //# 29: ok
((<A?>[null])[0] ??= theA).b; //# 30: compile-time error
((<A?>[null])[0] ??= theB).a; //# 31: ok
((<A?>[null])[0] ??= theB).b; //# 32: compile-time error
((<B?>[null])[0] ??= theA).a; //# 33: compile-time error
((<B?>[null])[0] ??= theA).b; //# 34: compile-time error
// Exactly the same static errors that would be caused by e1[e2] = e3 are
// also generated in the case of e1[e2] ??= e3.
(<B?>[null])[0] ??= new C(); //# 35: compile-time error
// The static type of e1?.v op= e2 is the static type of e1.v op e2,
// therefore the static type of e1?.v ??= e2 is the static type of
// e1.v ?? e2, which is the LUB of NonNull(typeof(e1?.v)) and typeof(e2).
var c = new ClassWithInstanceGetters() as ClassWithInstanceGetters?;
(c?.a ??= theA)!.a; //# 36: ok
(c?.a ??= theA)!.b; //# 37: compile-time error
(c?.a ??= theB)!.a; //# 38: ok
(c?.a ??= theB)!.b; //# 39: compile-time error
(c?.b ??= theA)!.a; //# 40: compile-time error
(c?.b ??= theA)!.b; //# 41: compile-time error
// Exactly the same static errors that would be caused by e1.v ??= e2 are
// also generated in the case of e1?.v ??= e2.
c?.b ??= new C(); //# 42: compile-time error
}