mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-26 05:12:35 +00:00 
			
		
		
		
	 08a303172d
			
		
	
	
		08a303172d
		
	
	
	
	
		
			
			Instead of only parsing a primary expression, we should also allow member expressions, call expressions, and tagged template literals (and optional chains, which we don't have yet). In the spec, all of this is covered by `LeftHandSideExpression` (https://tc39.es/ecma262/#prod-LeftHandSideExpression).
		
			
				
	
	
		
			172 lines
		
	
	
	
		
			3.5 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			172 lines
		
	
	
	
		
			3.5 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| test("method inheritance", () => {
 | |
|     class Parent {
 | |
|         method() {
 | |
|             return 3;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     class Child extends Parent {}
 | |
| 
 | |
|     const p = new Parent();
 | |
|     const c = new Child();
 | |
|     expect(p.method()).toBe(3);
 | |
|     expect(c.method()).toBe(3);
 | |
| });
 | |
| 
 | |
| test("method overriding", () => {
 | |
|     class Parent {
 | |
|         method() {
 | |
|             return 3;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     class Child extends Parent {
 | |
|         method() {
 | |
|             return 10;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     const p = new Parent();
 | |
|     const c = new Child();
 | |
|     expect(p.method()).toBe(3);
 | |
|     expect(c.method()).toBe(10);
 | |
| });
 | |
| 
 | |
| test("parent method reference with super", () => {
 | |
|     class Parent {
 | |
|         method() {
 | |
|             return 3;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     class Child extends Parent {
 | |
|         method() {
 | |
|             return super.method() * 2;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     const p = new Parent();
 | |
|     const c = new Child();
 | |
|     expect(p.method()).toBe(3);
 | |
|     expect(c.method()).toBe(6);
 | |
| });
 | |
| 
 | |
| test("child class access to parent class initialized properties", () => {
 | |
|     class Parent {
 | |
|         constructor() {
 | |
|             this.x = 3;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     class Child extends Parent {}
 | |
| 
 | |
|     const p = new Parent();
 | |
|     const c = new Child();
 | |
|     expect(p.x).toBe(3);
 | |
|     expect(c.x).toBe(3);
 | |
| });
 | |
| 
 | |
| test("child class modification of parent class properties", () => {
 | |
|     class Parent {
 | |
|         constructor() {
 | |
|             this.x = 3;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     class Child extends Parent {
 | |
|         change() {
 | |
|             this.x = 10;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     const p = new Parent();
 | |
|     const c = new Child();
 | |
|     expect(p.x).toBe(3);
 | |
|     expect(c.x).toBe(3);
 | |
| 
 | |
|     c.change();
 | |
|     expect(c.x).toBe(10);
 | |
| });
 | |
| 
 | |
| test("inheritance and hasOwnProperty", () => {
 | |
|     class Parent {
 | |
|         constructor() {
 | |
|             this.x = 3;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     class Child extends Parent {
 | |
|         method() {
 | |
|             this.y = 10;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     const p = new Parent();
 | |
|     const c = new Child();
 | |
|     expect(p.hasOwnProperty("x")).toBeTrue();
 | |
|     expect(p.hasOwnProperty("y")).toBeFalse();
 | |
|     expect(c.hasOwnProperty("x")).toBeTrue();
 | |
|     expect(c.hasOwnProperty("y")).toBeFalse();
 | |
| 
 | |
|     c.method();
 | |
|     expect(c.hasOwnProperty("x")).toBeTrue();
 | |
|     expect(c.hasOwnProperty("y")).toBeTrue();
 | |
| });
 | |
| 
 | |
| test("super constructor call from child class with argument", () => {
 | |
|     class Parent {
 | |
|         constructor(x) {
 | |
|             this.x = x;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     class Child extends Parent {
 | |
|         constructor() {
 | |
|             super(10);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     const p = new Parent(3);
 | |
|     const c = new Child(3);
 | |
|     expect(p.x).toBe(3);
 | |
|     expect(c.x).toBe(10);
 | |
| });
 | |
| 
 | |
| test("advanced 'extends' RHS", () => {
 | |
|     const foo = {
 | |
|         bar() {
 | |
|             return {
 | |
|                 baz() {
 | |
|                     return function () {
 | |
|                         return function () {
 | |
|                             return { quux: Number };
 | |
|                         };
 | |
|                     };
 | |
|                 },
 | |
|             };
 | |
|         },
 | |
|     };
 | |
|     class Foo extends foo.bar()["baz"]()`qux`().quux {}
 | |
|     expect(new Foo()).toBeInstanceOf(Number);
 | |
| });
 | |
| 
 | |
| test("issue #7045, super constructor call from child class in catch {}", () => {
 | |
|     class Parent {
 | |
|         constructor(x) {
 | |
|             this.x = x;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     class Child extends Parent {
 | |
|         constructor() {
 | |
|             try {
 | |
|                 throw new Error("Error in Child constructor");
 | |
|             } catch (e) {
 | |
|                 super(e.message);
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     const c = new Child();
 | |
|     expect(c.x).toBe("Error in Child constructor");
 | |
| });
 |