blob: d27eb5a1a68313f7f222a5e39a9b3459dc43df8f [file] [log] [blame]
// Copyright (c) 2023, 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.
/// @assertion Assume that DV is an extension type declaration named Name, and
/// V1 occurs as one of the <type>s in the <interfaces> of DV. In this case we
/// say that V1 is a superinterface of DV.
/// ...
/// Assume that DV is an extension type declaration named Name, and the type V1,
/// declared by DV1, is a superinterface of DV (V1 could be an extension type or
/// a non-extension type). Let m be the name of a member of V1. If DV also
/// declares a member named m then the latter may be considered similar to a
/// declaration that "overrides" the former. However, it should be noted that
/// extension type method invocation is resolved statically, and hence there is
/// no override relationship among the two in the traditional object-oriented
/// sense (that is, it will never occur that the statically known declaration is
/// the member of V1, and the member invoked at run time is the one in DV). A
/// receiver with static type V1 will invoke the declaration in DV1, and a
/// receiver with a static type which is a reference to DV (like Name or
/// Name<...>) will invoke the one in DV.
///
/// Hence, we use a different word to describe the relationship between a member
/// named m of a superinterface, and a member named m which is declared by the
/// subinterface: We say that the latter redeclares the former.
///
/// In particular, if two different declarations of m are inherited from two
/// superinterfaces then the subinterface can resolve the conflict by
/// redeclaring m.
///
/// There is no notion of having a 'correct override relation' here. With
/// extension types, any member signature can redeclare any other member
/// signature with the same name, including the case where a method is
/// redeclared by a getter, or vice versa.
///
/// @description Checks that a `call` member can be redeclared as any other
/// member, and it can coexist with a member named `call` in the interface of
/// the representation type.
/// @author sgrekhov22@gmail.com
import '../../Utils/expect.dart';
class C1 {
String call() => "call from C1";
}
class C2 {
String get call => "call from C2";
}
class C3 {
void set call(String _) {}
}
extension type ET1(C1 c) {
String get call => "call from ET1";
}
extension type ET2(C2 c) {
String get call => "call from ET2";
}
extension type ET3(C3 c) {
String get call => "call from ET3";
}
extension type ET4(C1 c) implements C1 {
String get call => "call from ET4";
}
extension type ET5(C2 c) implements C2 {
String get call => "call from ET5";
}
extension type ET6(C3 c) implements C3 {
String get call => "call from ET6";
}
main() {
Expect.equals("call from ET1", ET1(C1()).call);
Expect.equals("call from ET2", ET2(C2()).call);
Expect.equals("call from ET3", ET3(C3()).call);
Expect.equals("call from ET4", ET4(C1()).call);
Expect.equals("call from ET5", ET5(C2()).call);
Expect.equals("call from ET6", ET6(C3()).call);
}