blob: 1a19dcac57eb55c0b573337f072ae3a092116fa3 [file] [log] [blame]
// Copyright (c) 2017, 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.
import "package:expect/expect.dart";
class A {}
abstract class B<T> {
// x will be marked genericCovariantInterface, since x's type is covariant in
// the type parameter T.
void set s2(T x);
// x will be marked genericCovariantInterface, since x's type is covariant in
// the type parameter T.
void set s3(T x);
void set s4(Object x);
void set s5(Object x) {
s4 = x;
}
}
class C extends B<A> {
void set s1(A x) {}
// x will be marked genericCovariantImpl, since it might be called via
// e.g. B<Object>.
void set s2(A x) {}
// x will be marked genericCovariantImpl, since it might be called via
// e.g. B<Object>.
void set s3(covariant A x) {}
void set s4(covariant A x) {}
}
main() {
// Dynamic method calls should always have their arguments type checked.
dynamic d = new C();
Expect.throwsTypeError(() => d.s1 = new Object()); //# 01: ok
// Interface calls should have any arguments marked "genericCovariantImpl"
// type checked provided that the corresponding argument on the interface
// target is marked "genericCovariantInterface".
B<Object> b = new C();
Expect.throwsTypeError(() => b.s2 = new Object()); //# 02: ok
// Interface calls should have any arguments marked "covariant" type checked,
// regardless of whether the corresponding argument on the interface target is
// marked "genericCovariantInterface".
Expect.throwsTypeError(() => b.s3 = new Object()); //# 03: ok
Expect.throwsTypeError(() => b.s4 = new Object()); //# 04: ok
// This calls should have any arguments marked "covariant" type checked.
Expect.throwsTypeError(() => b.s5 = new Object()); //# 05: ok
testMixin(); //# 06: ok
}
abstract class D<T> {
void set m1(T x);
}
class E {
void set m1(A x) {}
}
class F = Object with E implements D<A>;
class G = C with E implements D<A>;
class H extends Object with E implements D<A> {}
class I extends Object with F {}
void testMixin() {
D<Object> f = new F();
f.m1 = new A();
Expect.throwsTypeError(() => f.m1 = new Object());
f = new G();
f.m1 = new A();
Expect.throwsTypeError(() => f.m1 = new Object());
f = new H();
f.m1 = new A();
Expect.throwsTypeError(() => f.m1 = new Object());
f = new I();
f.m1 = new A();
Expect.throwsTypeError(() => f.m1 = new Object());
}