blob: d1916b63ff3aa43da982e2731d2d773fab5b7a3a [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 semantics of the ?. operator when it appears in a postincrement or
// preincrement expression (or a postdecrement or predecrement expression).
import "package:expect/expect.dart";
import "conditional_access_helper.dart" as h;
class C {
int v;
C(this.v);
static late int staticInt;
}
class D {
E v;
D(this.v);
static late E staticE;
}
class E {
G operator +(int i) => new I();
G operator -(int i) => new I();
}
class F {}
class G extends E implements F {}
class H {}
class I extends G implements H {}
C? nullC() => null;
main() {
// Make sure the "none" test fails if assignment to "?." is not implemented.
// This makes status files easier to maintain.
nullC()?.v = 1;
// e1?.v++ is equivalent to ((x) => x == null ? null : x.v++)(e1).
Expect.equals(null, nullC()?.v++);
{
var c = new C(1) as C?;
Expect.equals(1, c?.v++);
Expect.equals(2, c!.v);
}
// C?.v++ is equivalent to C.v++.
{
C.staticInt = 1;
Expect.equals(1, C?.staticInt++);
// ^
// [cfe] The class 'C' cannot be null.
Expect.equals(2, C.staticInt);
}
{
h.C.staticInt = 1;
Expect.equals(1, h.C?.staticInt++);
// ^
// [cfe] The class 'C' cannot be null.
Expect.equals(2, h.C.staticInt);
}
// The static type of e1?.v++ is the same as the static type of e1.v.
{
E e1 = new E();
var d = new D(e1) as D?;
E? e2 = d?.v++;
Expect.identical(e1, e2);
}
{
G g = new G();
var d = new D(g) as D?;
F? f = d?.v++;
// ^^^^^^
// [analyzer] COMPILE_TIME_ERROR.INVALID_ASSIGNMENT
// [cfe] A value of type 'E?' can't be assigned to a variable of type 'F?'.
Expect.identical(f, g);
}
{
E e1 = new E();
D.staticE = e1;
E? e2 = D?.staticE++;
// ^
// [cfe] The class 'D' cannot be null.
Expect.identical(e1, e2);
}
{
h.E e1 = new h.E();
h.D.staticE = e1;
h.E? e2 = h.D?.staticE++;
// ^
// [cfe] The class 'D' cannot be null.
Expect.identical(e1, e2);
}
{
G g = new G();
D.staticE = g;
F? f = D?.staticE++;
// ^^^^^^^^^^^^
// [analyzer] COMPILE_TIME_ERROR.INVALID_ASSIGNMENT
// [cfe] The class 'D' cannot be null.
// ^
// [cfe] A value of type 'E' can't be assigned to a variable of type 'F?'.
Expect.identical(f, g);
}
{
h.G g = new h.G();
h.D.staticE = g;
h.F? f = h.D?.staticE++;
// ^^^^^^^^^^^^^^
// [analyzer] COMPILE_TIME_ERROR.INVALID_ASSIGNMENT
// ^
// [cfe] The class 'D' cannot be null.
// ^
// [cfe] A value of type 'E' can't be assigned to a variable of type 'F?'.
Expect.identical(f, g);
}
// e1?.v-- is equivalent to ((x) => x == null ? null : x.v--)(e1).
Expect.equals(null, nullC()?.v--);
{
var c = new C(1) as C?;
Expect.equals(1, c?.v--);
Expect.equals(0, c!.v);
}
// C?.v-- is equivalent to C.v--.
{
C.staticInt = 1;
Expect.equals(1, C?.staticInt--);
// ^
// [cfe] The class 'C' cannot be null.
Expect.equals(0, C.staticInt);
}
{
h.C.staticInt = 1;
Expect.equals(1, h.C?.staticInt--);
// ^
// [cfe] The class 'C' cannot be null.
Expect.equals(0, h.C.staticInt);
}
// The static type of e1?.v-- is the same as the static type of e1.v.
{
E e1 = new E();
var d = new D(e1) as D?;
E? e2 = d?.v--;
Expect.identical(e1, e2);
}
{
G g = new G();
var d = new D(g) as D?;
F? f = d?.v--;
// ^^^^^^
// [analyzer] COMPILE_TIME_ERROR.INVALID_ASSIGNMENT
// [cfe] A value of type 'E?' can't be assigned to a variable of type 'F?'.
Expect.identical(f, g);
}
{
E e1 = new E();
D.staticE = e1;
E? e2 = D?.staticE--;
// ^
// [cfe] The class 'D' cannot be null.
Expect.identical(e1, e2);
}
{
h.E e1 = new h.E();
h.D.staticE = e1;
h.E? e2 = h.D?.staticE--;
// ^
// [cfe] The class 'D' cannot be null.
Expect.identical(e1, e2);
}
{
G g = new G();
D.staticE = g;
F? f = D?.staticE--;
// ^^^^^^^^^^^^
// [analyzer] COMPILE_TIME_ERROR.INVALID_ASSIGNMENT
// [cfe] The class 'D' cannot be null.
// ^
// [cfe] A value of type 'E' can't be assigned to a variable of type 'F?'.
Expect.identical(f, g);
}
{
h.G g = new h.G();
h.D.staticE = g;
h.F? f = h.D?.staticE--;
// ^^^^^^^^^^^^^^
// [analyzer] COMPILE_TIME_ERROR.INVALID_ASSIGNMENT
// ^
// [cfe] The class 'D' cannot be null.
// ^
// [cfe] A value of type 'E' can't be assigned to a variable of type 'F?'.
Expect.identical(f, g);
}
// ++e1?.v is equivalent to e1?.v += 1.
Expect.equals(null, ++nullC()?.v);
{
var c = new C(1) as C?;
Expect.equals(2, ++c?.v);
Expect.equals(2, c!.v);
}
// ++C?.v is equivalent to C?.v += 1.
{
C.staticInt = 1;
Expect.equals(2, ++C?.staticInt);
// ^
// [cfe] The class 'C' cannot be null.
Expect.equals(2, C.staticInt);
}
{
h.C.staticInt = 1;
Expect.equals(2, ++h.C?.staticInt);
// ^
// [cfe] The class 'C' cannot be null.
Expect.equals(2, h.C.staticInt);
}
// The static type of ++e1?.v is the same as the static type of e1.v + 1.
{
var d = new D(new E()) as D?;
F? f = ++d?.v;
Expect.identical(d!.v, f);
}
{
var d = new D(new E()) as D?;
H? h = ++d?.v;
// ^^^^^^
// [analyzer] COMPILE_TIME_ERROR.INVALID_ASSIGNMENT
// ^
// [cfe] A value of type 'G?' can't be assigned to a variable of type 'H?'.
Expect.identical(d!.v, h);
}
{
D.staticE = new E();
F? f = ++D?.staticE;
// ^
// [cfe] The class 'D' cannot be null.
Expect.identical(D.staticE, f);
}
{
h.D.staticE = new h.E();
h.F? f = ++h.D?.staticE;
// ^
// [cfe] The class 'D' cannot be null.
Expect.identical(h.D.staticE, f);
}
{
D.staticE = new E();
H? h = ++D?.staticE;
// ^^^^^^^^^^^^
// [analyzer] COMPILE_TIME_ERROR.INVALID_ASSIGNMENT
// ^
// [cfe] The class 'D' cannot be null.
// ^
// [cfe] A value of type 'G' can't be assigned to a variable of type 'H?'.
Expect.identical(D.staticE, h);
}
{
h.D.staticE = new h.E();
h.H? hh = ++h.D?.staticE;
// ^^^^^^^^^^^^^^
// [analyzer] COMPILE_TIME_ERROR.INVALID_ASSIGNMENT
// ^
// [cfe] The class 'D' cannot be null.
// ^
// [cfe] A value of type 'G' can't be assigned to a variable of type 'H?'.
Expect.identical(h.D.staticE, hh);
}
// --e1?.v is equivalent to e1?.v -= 1.
Expect.equals(null, --nullC()?.v);
{
var c = new C(1) as C?;
Expect.equals(0, --c?.v);
Expect.equals(0, c!.v);
}
// --C?.v is equivalent to C?.v -= 1.
{
C.staticInt = 1;
Expect.equals(0, --C?.staticInt);
// ^
// [cfe] The class 'C' cannot be null.
Expect.equals(0, C.staticInt);
}
{
h.C.staticInt = 1;
Expect.equals(0, --h.C?.staticInt);
// ^
// [cfe] The class 'C' cannot be null.
Expect.equals(0, h.C.staticInt);
}
// The static type of --e1?.v is the same as the static type of e1.v - 1.
{
var d = new D(new E()) as D?;
F? f = --d?.v;
Expect.identical(d!.v, f);
}
{
var d = new D(new E()) as D?;
H? h = --d?.v;
// ^^^^^^
// [analyzer] COMPILE_TIME_ERROR.INVALID_ASSIGNMENT
// ^
// [cfe] A value of type 'G?' can't be assigned to a variable of type 'H?'.
Expect.identical(d!.v, h);
}
{
D.staticE = new E();
F? f = --D?.staticE;
// ^
// [cfe] The class 'D' cannot be null.
Expect.identical(D.staticE, f);
}
{
h.D.staticE = new h.E();
h.F? f = --h.D?.staticE;
// ^
// [cfe] The class 'D' cannot be null.
Expect.identical(h.D.staticE, f);
}
{
D.staticE = new E();
H? h = --D?.staticE;
// ^^^^^^^^^^^^
// [analyzer] COMPILE_TIME_ERROR.INVALID_ASSIGNMENT
// ^
// [cfe] The class 'D' cannot be null.
// ^
// [cfe] A value of type 'G' can't be assigned to a variable of type 'H?'.
Expect.identical(D.staticE, h);
}
{
h.D.staticE = new h.E();
h.H? hh = --h.D?.staticE;
// ^^^^^^^^^^^^^^
// [analyzer] COMPILE_TIME_ERROR.INVALID_ASSIGNMENT
// ^
// [cfe] The class 'D' cannot be null.
// ^
// [cfe] A value of type 'G' can't be assigned to a variable of type 'H?'.
Expect.identical(h.D.staticE, hh);
}
}