fix: disable injectable-pipe migration (#30180)
Disables the injectable pipe migration until we can decide whether this is the right solution for Ivy. Rolling it out properly will involve a more detailed plan and more changes like updating the styleguide, scaffolding schematics etc. Context for the new `test-migrations.json`: since we use the `migrations.json` both for the real migrations and for tests, it doesn't allow us to disable a schematic, but continue running its tests. This change adds the test-specific file so that we can continue running the `injectable-pipe` tests, even though the schematic itself is disabled. PR Close #30180
This commit is contained in:
parent
6f433887e0
commit
572b54967c
|
@ -3,6 +3,7 @@ load("//tools:defaults.bzl", "npm_package")
|
||||||
exports_files([
|
exports_files([
|
||||||
"tsconfig.json",
|
"tsconfig.json",
|
||||||
"migrations.json",
|
"migrations.json",
|
||||||
|
"test-migrations.json",
|
||||||
])
|
])
|
||||||
|
|
||||||
npm_package(
|
npm_package(
|
||||||
|
|
|
@ -14,11 +14,6 @@
|
||||||
"version": "8-beta",
|
"version": "8-beta",
|
||||||
"description": "Warns developers if values are assigned to template variables",
|
"description": "Warns developers if values are assigned to template variables",
|
||||||
"factory": "./migrations/template-var-assignment/index"
|
"factory": "./migrations/template-var-assignment/index"
|
||||||
},
|
|
||||||
"migration-v8-injectable-pipe": {
|
|
||||||
"version": "8-beta",
|
|
||||||
"description": "Migrates all Pipe classes so that they have an Injectable annotation",
|
|
||||||
"factory": "./migrations/injectable-pipe/index"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,20 @@
|
||||||
|
{
|
||||||
|
"schematics": {
|
||||||
|
"migration-move-document": {
|
||||||
|
"description": "Migrates DOCUMENT Injection token from platform-browser imports to common import",
|
||||||
|
"factory": "./migrations/move-document/index"
|
||||||
|
},
|
||||||
|
"migration-static-queries": {
|
||||||
|
"description": "Migrates ViewChild and ContentChild to explicit query timing",
|
||||||
|
"factory": "./migrations/static-queries/index"
|
||||||
|
},
|
||||||
|
"migration-template-local-variables": {
|
||||||
|
"description": "Warns developers if values are assigned to template variables",
|
||||||
|
"factory": "./migrations/template-var-assignment/index"
|
||||||
|
},
|
||||||
|
"migration-injectable-pipe": {
|
||||||
|
"description": "Migrates all Pipe classes so that they have an Injectable annotation",
|
||||||
|
"factory": "./migrations/injectable-pipe/index"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -5,7 +5,7 @@ ts_library(
|
||||||
testonly = True,
|
testonly = True,
|
||||||
srcs = glob(["**/*.ts"]),
|
srcs = glob(["**/*.ts"]),
|
||||||
data = [
|
data = [
|
||||||
"//packages/core/schematics:migrations.json",
|
"//packages/core/schematics:test-migrations.json",
|
||||||
],
|
],
|
||||||
deps = [
|
deps = [
|
||||||
"//packages/core/schematics/migrations/injectable-pipe",
|
"//packages/core/schematics/migrations/injectable-pipe",
|
||||||
|
|
|
@ -20,7 +20,7 @@ describe('injectable pipe migration', () => {
|
||||||
let previousWorkingDir: string;
|
let previousWorkingDir: string;
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
runner = new SchematicTestRunner('test', require.resolve('../migrations.json'));
|
runner = new SchematicTestRunner('test', require.resolve('../test-migrations.json'));
|
||||||
host = new TempScopedNodeJsSyncHost();
|
host = new TempScopedNodeJsSyncHost();
|
||||||
tree = new UnitTestTree(new HostTree(host));
|
tree = new UnitTestTree(new HostTree(host));
|
||||||
|
|
||||||
|
@ -123,5 +123,5 @@ describe('injectable pipe migration', () => {
|
||||||
host.sync.write(normalize(filePath), virtualFs.stringToFileBuffer(contents));
|
host.sync.write(normalize(filePath), virtualFs.stringToFileBuffer(contents));
|
||||||
}
|
}
|
||||||
|
|
||||||
function runMigration() { runner.runSchematic('migration-v8-injectable-pipe', {}, tree); }
|
function runMigration() { runner.runSchematic('migration-injectable-pipe', {}, tree); }
|
||||||
});
|
});
|
||||||
|
|
|
@ -20,7 +20,7 @@ describe('move-document migration', () => {
|
||||||
let previousWorkingDir: string;
|
let previousWorkingDir: string;
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
runner = new SchematicTestRunner('test', require.resolve('../migrations.json'));
|
runner = new SchematicTestRunner('test', require.resolve('../test-migrations.json'));
|
||||||
host = new TempScopedNodeJsSyncHost();
|
host = new TempScopedNodeJsSyncHost();
|
||||||
tree = new UnitTestTree(new HostTree(host));
|
tree = new UnitTestTree(new HostTree(host));
|
||||||
|
|
||||||
|
@ -151,5 +151,5 @@ describe('move-document migration', () => {
|
||||||
host.sync.write(normalize(filePath), virtualFs.stringToFileBuffer(contents));
|
host.sync.write(normalize(filePath), virtualFs.stringToFileBuffer(contents));
|
||||||
}
|
}
|
||||||
|
|
||||||
function runMigration() { runner.runSchematic('migration-v8-move-document', {}, tree); }
|
function runMigration() { runner.runSchematic('migration-move-document', {}, tree); }
|
||||||
});
|
});
|
||||||
|
|
|
@ -21,7 +21,7 @@ describe('static-queries migration with template strategy', () => {
|
||||||
let warnOutput: string[];
|
let warnOutput: string[];
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
runner = new SchematicTestRunner('test', require.resolve('../migrations.json'));
|
runner = new SchematicTestRunner('test', require.resolve('../test-migrations.json'));
|
||||||
host = new TempScopedNodeJsSyncHost();
|
host = new TempScopedNodeJsSyncHost();
|
||||||
tree = new UnitTestTree(new HostTree(host));
|
tree = new UnitTestTree(new HostTree(host));
|
||||||
|
|
||||||
|
@ -97,7 +97,7 @@ describe('static-queries migration with template strategy', () => {
|
||||||
}
|
}
|
||||||
|
|
||||||
async function runMigration() {
|
async function runMigration() {
|
||||||
await runner.runSchematicAsync('migration-v8-static-queries', {}, tree).toPromise();
|
await runner.runSchematicAsync('migration-static-queries', {}, tree).toPromise();
|
||||||
}
|
}
|
||||||
|
|
||||||
describe('ViewChild', () => {
|
describe('ViewChild', () => {
|
||||||
|
@ -105,7 +105,7 @@ describe('static-queries migration with template strategy', () => {
|
||||||
it('should detect queries selecting elements through template reference', async() => {
|
it('should detect queries selecting elements through template reference', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, NgModule, ViewChild} from '@angular/core';
|
import {Component, NgModule, ViewChild} from '@angular/core';
|
||||||
|
|
||||||
@Component({template: \`
|
@Component({template: \`
|
||||||
<ng-template>
|
<ng-template>
|
||||||
<button #myButton>My Button</button>
|
<button #myButton>My Button</button>
|
||||||
|
@ -118,7 +118,7 @@ describe('static-queries migration with template strategy', () => {
|
||||||
private @ViewChild('myButton') query: any;
|
private @ViewChild('myButton') query: any;
|
||||||
private @ViewChild('myStaticButton') query2: any;
|
private @ViewChild('myStaticButton') query2: any;
|
||||||
}
|
}
|
||||||
|
|
||||||
@NgModule({declarations: [MyComp]})
|
@NgModule({declarations: [MyComp]})
|
||||||
export class MyModule {}
|
export class MyModule {}
|
||||||
`);
|
`);
|
||||||
|
@ -134,7 +134,7 @@ describe('static-queries migration with template strategy', () => {
|
||||||
it('should detect queries selecting ng-template as static', async() => {
|
it('should detect queries selecting ng-template as static', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, NgModule, ViewChild} from '@angular/core';
|
import {Component, NgModule, ViewChild} from '@angular/core';
|
||||||
|
|
||||||
@Component({template: \`
|
@Component({template: \`
|
||||||
<ng-template #myTmpl>
|
<ng-template #myTmpl>
|
||||||
My template
|
My template
|
||||||
|
@ -143,7 +143,7 @@ describe('static-queries migration with template strategy', () => {
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
private @ViewChild('myTmpl') query: any;
|
private @ViewChild('myTmpl') query: any;
|
||||||
}
|
}
|
||||||
|
|
||||||
@NgModule({declarations: [MyComp]})
|
@NgModule({declarations: [MyComp]})
|
||||||
export class MyModule {}
|
export class MyModule {}
|
||||||
`);
|
`);
|
||||||
|
@ -157,7 +157,7 @@ describe('static-queries migration with template strategy', () => {
|
||||||
it('should detect queries selecting component view providers through string token', async() => {
|
it('should detect queries selecting component view providers through string token', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, Directive, NgModule, ViewChild} from '@angular/core';
|
import {Component, Directive, NgModule, ViewChild} from '@angular/core';
|
||||||
|
|
||||||
@Directive({
|
@Directive({
|
||||||
selector: '[myDirective]',
|
selector: '[myDirective]',
|
||||||
providers: [
|
providers: [
|
||||||
|
@ -165,7 +165,7 @@ describe('static-queries migration with template strategy', () => {
|
||||||
]
|
]
|
||||||
})
|
})
|
||||||
export class MyDirective {}
|
export class MyDirective {}
|
||||||
|
|
||||||
@Directive({
|
@Directive({
|
||||||
selector: '[myDirective2]',
|
selector: '[myDirective2]',
|
||||||
providers: [
|
providers: [
|
||||||
|
@ -173,13 +173,13 @@ describe('static-queries migration with template strategy', () => {
|
||||||
]
|
]
|
||||||
})
|
})
|
||||||
export class MyDirective2 {}
|
export class MyDirective2 {}
|
||||||
|
|
||||||
@Component({templateUrl: './my-tmpl.html'})
|
@Component({templateUrl: './my-tmpl.html'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
private @ViewChild('my-token') query: any;
|
private @ViewChild('my-token') query: any;
|
||||||
private @ViewChild('my-token-2') query2: any;
|
private @ViewChild('my-token-2') query2: any;
|
||||||
}
|
}
|
||||||
|
|
||||||
@NgModule({declarations: [MyComp, MyDirective, MyDirective2]})
|
@NgModule({declarations: [MyComp, MyDirective, MyDirective2]})
|
||||||
export class MyModule {}
|
export class MyModule {}
|
||||||
`);
|
`);
|
||||||
|
@ -202,28 +202,28 @@ describe('static-queries migration with template strategy', () => {
|
||||||
it('should detect queries selecting component view providers using class token', async() => {
|
it('should detect queries selecting component view providers using class token', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, Directive, NgModule, ViewChild} from '@angular/core';
|
import {Component, Directive, NgModule, ViewChild} from '@angular/core';
|
||||||
|
|
||||||
export class MyService {}
|
export class MyService {}
|
||||||
export class MyService2 {}
|
export class MyService2 {}
|
||||||
|
|
||||||
@Directive({
|
@Directive({
|
||||||
selector: '[myDirective]',
|
selector: '[myDirective]',
|
||||||
providers: [MyService]
|
providers: [MyService]
|
||||||
})
|
})
|
||||||
export class MyDirective {}
|
export class MyDirective {}
|
||||||
|
|
||||||
@Directive({
|
@Directive({
|
||||||
selector: '[myDirective2]',
|
selector: '[myDirective2]',
|
||||||
providers: [MyService2]
|
providers: [MyService2]
|
||||||
})
|
})
|
||||||
export class MyDirective2 {}
|
export class MyDirective2 {}
|
||||||
|
|
||||||
@Component({templateUrl: './my-tmpl.html'})
|
@Component({templateUrl: './my-tmpl.html'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
private @ViewChild(MyService) query: any;
|
private @ViewChild(MyService) query: any;
|
||||||
private @ViewChild(MyService2) query2: any;
|
private @ViewChild(MyService2) query2: any;
|
||||||
}
|
}
|
||||||
|
|
||||||
@NgModule({declarations: [MyComp, MyDirective, MyDirective2]})
|
@NgModule({declarations: [MyComp, MyDirective, MyDirective2]})
|
||||||
export class MyModule {}
|
export class MyModule {}
|
||||||
`);
|
`);
|
||||||
|
@ -247,7 +247,7 @@ describe('static-queries migration with template strategy', () => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, NgModule, ViewChild} from '@angular/core';
|
import {Component, NgModule, ViewChild} from '@angular/core';
|
||||||
import {HomeComponent, HomeComponent2} from './home-comp';
|
import {HomeComponent, HomeComponent2} from './home-comp';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
template: \`
|
template: \`
|
||||||
<home-comp></home-comp>
|
<home-comp></home-comp>
|
||||||
|
@ -260,20 +260,20 @@ describe('static-queries migration with template strategy', () => {
|
||||||
private @ViewChild(HomeComponent) query: any;
|
private @ViewChild(HomeComponent) query: any;
|
||||||
private @ViewChild(HomeComponent2) query2: any;
|
private @ViewChild(HomeComponent2) query2: any;
|
||||||
}
|
}
|
||||||
|
|
||||||
@NgModule({declarations: [MyComp, HomeComponent, HomeComponent2]})
|
@NgModule({declarations: [MyComp, HomeComponent, HomeComponent2]})
|
||||||
export class MyModule {}
|
export class MyModule {}
|
||||||
`);
|
`);
|
||||||
|
|
||||||
writeFile(`/home-comp.ts`, `
|
writeFile(`/home-comp.ts`, `
|
||||||
import {Component} from '@angular/core';
|
import {Component} from '@angular/core';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'home-comp',
|
selector: 'home-comp',
|
||||||
template: '<span>Home</span>'
|
template: '<span>Home</span>'
|
||||||
})
|
})
|
||||||
export class HomeComponent {}
|
export class HomeComponent {}
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'home-comp2',
|
selector: 'home-comp2',
|
||||||
template: '<span>Home 2</span>'
|
template: '<span>Home 2</span>'
|
||||||
|
@ -294,12 +294,12 @@ describe('static-queries migration with template strategy', () => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, NgModule, ViewChild} from '@angular/core';
|
import {Component, NgModule, ViewChild} from '@angular/core';
|
||||||
import {MyLibComponent} from 'my-lib';
|
import {MyLibComponent} from 'my-lib';
|
||||||
|
|
||||||
@Component({templateUrl: './my-tmpl.html'})
|
@Component({templateUrl: './my-tmpl.html'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
private @ViewChild(MyLibComponent) query: any;
|
private @ViewChild(MyLibComponent) query: any;
|
||||||
}
|
}
|
||||||
|
|
||||||
@NgModule({declarations: [MyComp, MyLibComponent]})
|
@NgModule({declarations: [MyComp, MyLibComponent]})
|
||||||
export class MyModule {}
|
export class MyModule {}
|
||||||
`);
|
`);
|
||||||
|
@ -319,12 +319,12 @@ describe('static-queries migration with template strategy', () => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, NgModule, ViewChild} from '@angular/core';
|
import {Component, NgModule, ViewChild} from '@angular/core';
|
||||||
import {MyLibComponent} from 'my-lib';
|
import {MyLibComponent} from 'my-lib';
|
||||||
|
|
||||||
@Component({templateUrl: './my-tmpl.html'})
|
@Component({templateUrl: './my-tmpl.html'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
private @ViewChild(MyLibComponent) query: any;
|
private @ViewChild(MyLibComponent) query: any;
|
||||||
}
|
}
|
||||||
|
|
||||||
@NgModule({declarations: [MyComp, MyLibComponent]})
|
@NgModule({declarations: [MyComp, MyLibComponent]})
|
||||||
export class MyModule {}
|
export class MyModule {}
|
||||||
`);
|
`);
|
||||||
|
@ -345,16 +345,16 @@ describe('static-queries migration with template strategy', () => {
|
||||||
it('should detect queries within structural directive', async() => {
|
it('should detect queries within structural directive', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, Directive, NgModule, ViewChild} from '@angular/core';
|
import {Component, Directive, NgModule, ViewChild} from '@angular/core';
|
||||||
|
|
||||||
@Directive({selector: '[ngIf]'})
|
@Directive({selector: '[ngIf]'})
|
||||||
export class FakeNgIf {}
|
export class FakeNgIf {}
|
||||||
|
|
||||||
@Component({templateUrl: 'my-tmpl.html'})
|
@Component({templateUrl: 'my-tmpl.html'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
private @ViewChild('myRef') query: any;
|
private @ViewChild('myRef') query: any;
|
||||||
private @ViewChild('myRef2') query2: any;
|
private @ViewChild('myRef2') query2: any;
|
||||||
}
|
}
|
||||||
|
|
||||||
@NgModule({declarations: [MyComp, FakeNgIf]})
|
@NgModule({declarations: [MyComp, FakeNgIf]})
|
||||||
export class MyModule {}
|
export class MyModule {}
|
||||||
`);
|
`);
|
||||||
|
@ -375,14 +375,14 @@ describe('static-queries migration with template strategy', () => {
|
||||||
it('should detect inherited queries', async() => {
|
it('should detect inherited queries', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, NgModule, ViewChild} from '@angular/core';
|
import {Component, NgModule, ViewChild} from '@angular/core';
|
||||||
|
|
||||||
export class BaseClass {
|
export class BaseClass {
|
||||||
@ViewChild('myRef') query: any;
|
@ViewChild('myRef') query: any;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Component({templateUrl: 'my-tmpl.html'})
|
@Component({templateUrl: 'my-tmpl.html'})
|
||||||
export class MyComp extends BaseClass {}
|
export class MyComp extends BaseClass {}
|
||||||
|
|
||||||
@NgModule({declarations: [MyComp]})
|
@NgModule({declarations: [MyComp]})
|
||||||
export class MyModule {}
|
export class MyModule {}
|
||||||
`);
|
`);
|
||||||
|
@ -400,7 +400,7 @@ describe('static-queries migration with template strategy', () => {
|
||||||
it('should add a todo if a query is not declared in any component', async() => {
|
it('should add a todo if a query is not declared in any component', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, NgModule, ViewChild, SomeToken} from '@angular/core';
|
import {Component, NgModule, ViewChild, SomeToken} from '@angular/core';
|
||||||
|
|
||||||
export class NotAComponent {
|
export class NotAComponent {
|
||||||
@ViewChild('myRef', {read: SomeToken}) query: any;
|
@ViewChild('myRef', {read: SomeToken}) query: any;
|
||||||
}
|
}
|
||||||
|
@ -420,17 +420,17 @@ describe('static-queries migration with template strategy', () => {
|
||||||
it('should add a todo if a query is used multiple times with different timing', async() => {
|
it('should add a todo if a query is used multiple times with different timing', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, NgModule, ViewChild} from '@angular/core';
|
import {Component, NgModule, ViewChild} from '@angular/core';
|
||||||
|
|
||||||
export class BaseClass {
|
export class BaseClass {
|
||||||
@ViewChild('myRef') query: any;
|
@ViewChild('myRef') query: any;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Component({template: '<ng-template><p #myRef></p></ng-template>'})
|
@Component({template: '<ng-template><p #myRef></p></ng-template>'})
|
||||||
export class FirstComp extends BaseClass {}
|
export class FirstComp extends BaseClass {}
|
||||||
|
|
||||||
@Component({template: '<span #myRef></span>'})
|
@Component({template: '<span #myRef></span>'})
|
||||||
export class SecondComp extends BaseClass {}
|
export class SecondComp extends BaseClass {}
|
||||||
|
|
||||||
@NgModule({declarations: [FirstComp, SecondComp]})
|
@NgModule({declarations: [FirstComp, SecondComp]})
|
||||||
export class MyModule {}
|
export class MyModule {}
|
||||||
`);
|
`);
|
||||||
|
@ -448,12 +448,12 @@ describe('static-queries migration with template strategy', () => {
|
||||||
it('should gracefully exit migration if queries could not be analyzed', async() => {
|
it('should gracefully exit migration if queries could not be analyzed', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ViewChild} from '@angular/core';
|
import {Component, ViewChild} from '@angular/core';
|
||||||
|
|
||||||
@Component({template: '<ng-template><p #myRef></p></ng-template>'})
|
@Component({template: '<ng-template><p #myRef></p></ng-template>'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
@ViewChild('myRef') query: any;
|
@ViewChild('myRef') query: any;
|
||||||
}
|
}
|
||||||
|
|
||||||
// **NOTE**: Analysis will fail as there is no "NgModule" that declares the component.
|
// **NOTE**: Analysis will fail as there is no "NgModule" that declares the component.
|
||||||
`);
|
`);
|
||||||
|
|
||||||
|
@ -533,7 +533,7 @@ describe('static-queries migration with template strategy', () => {
|
||||||
writeFile('/src/test.ts', `
|
writeFile('/src/test.ts', `
|
||||||
import {ViewChild} from '@angular/core';
|
import {ViewChild} from '@angular/core';
|
||||||
import {AppComponent} from './app.component';
|
import {AppComponent} from './app.component';
|
||||||
|
|
||||||
@Component({template: '<span #test>Test</span>'})
|
@Component({template: '<span #test>Test</span>'})
|
||||||
class MyTestComponent {
|
class MyTestComponent {
|
||||||
@ViewChild('test') query: any;
|
@ViewChild('test') query: any;
|
||||||
|
@ -542,7 +542,7 @@ describe('static-queries migration with template strategy', () => {
|
||||||
|
|
||||||
writeFile('/src/app.component.ts', `
|
writeFile('/src/app.component.ts', `
|
||||||
import {Component, ViewChild} from '@angular/core';
|
import {Component, ViewChild} from '@angular/core';
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class AppComponent {
|
export class AppComponent {
|
||||||
@ViewChild('test') query: any;
|
@ViewChild('test') query: any;
|
||||||
|
@ -552,7 +552,7 @@ describe('static-queries migration with template strategy', () => {
|
||||||
writeFile('/src/app.module.ts', `
|
writeFile('/src/app.module.ts', `
|
||||||
import {NgModule} from '@angular/core';
|
import {NgModule} from '@angular/core';
|
||||||
import {AppComponent} from './app.component';
|
import {AppComponent} from './app.component';
|
||||||
|
|
||||||
@NgModule({declarations: [AppComponent]})
|
@NgModule({declarations: [AppComponent]})
|
||||||
export class MyModule {}
|
export class MyModule {}
|
||||||
`);
|
`);
|
||||||
|
|
|
@ -26,7 +26,7 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
afterAll(() => process.env['NG_STATIC_QUERY_USAGE_STRATEGY'] = '');
|
afterAll(() => process.env['NG_STATIC_QUERY_USAGE_STRATEGY'] = '');
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
runner = new SchematicTestRunner('test', require.resolve('../migrations.json'));
|
runner = new SchematicTestRunner('test', require.resolve('../test-migrations.json'));
|
||||||
host = new TempScopedNodeJsSyncHost();
|
host = new TempScopedNodeJsSyncHost();
|
||||||
tree = new UnitTestTree(new HostTree(host));
|
tree = new UnitTestTree(new HostTree(host));
|
||||||
|
|
||||||
|
@ -58,11 +58,11 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should mark view queries used in "ngAfterContentInit" as static', async() => {
|
it('should mark view queries used in "ngAfterContentInit" as static', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ViewChild} from '@angular/core';
|
import {Component, ViewChild} from '@angular/core';
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
@ViewChild('test') query: any;
|
@ViewChild('test') query: any;
|
||||||
|
|
||||||
ngAfterContentInit() {
|
ngAfterContentInit() {
|
||||||
this.query.classList.add('test');
|
this.query.classList.add('test');
|
||||||
}
|
}
|
||||||
|
@ -78,11 +78,11 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should mark view queries used in "ngAfterContentChecked" as static', async() => {
|
it('should mark view queries used in "ngAfterContentChecked" as static', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ViewChild} from '@angular/core';
|
import {Component, ViewChild} from '@angular/core';
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
@ViewChild('test') query: any;
|
@ViewChild('test') query: any;
|
||||||
|
|
||||||
ngAfterContentChecked() {
|
ngAfterContentChecked() {
|
||||||
this.query.classList.add('test');
|
this.query.classList.add('test');
|
||||||
}
|
}
|
||||||
|
@ -102,11 +102,11 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should not mark content queries used in "ngAfterContentInit" as static', async() => {
|
it('should not mark content queries used in "ngAfterContentInit" as static', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ContentChild} from '@angular/core';
|
import {Component, ContentChild} from '@angular/core';
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
@ContentChild('test') query: any;
|
@ContentChild('test') query: any;
|
||||||
|
|
||||||
ngAfterContentInit() {
|
ngAfterContentInit() {
|
||||||
this.query.classList.add('test');
|
this.query.classList.add('test');
|
||||||
}
|
}
|
||||||
|
@ -122,11 +122,11 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should not mark content queries used in "ngAfterContentChecked" as static', async() => {
|
it('should not mark content queries used in "ngAfterContentChecked" as static', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ContentChild} from '@angular/core';
|
import {Component, ContentChild} from '@angular/core';
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
@ContentChild('test') query: any;
|
@ContentChild('test') query: any;
|
||||||
|
|
||||||
ngAfterContentChecked() {
|
ngAfterContentChecked() {
|
||||||
this.query.classList.add('test');
|
this.query.classList.add('test');
|
||||||
}
|
}
|
||||||
|
@ -145,19 +145,19 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
}
|
}
|
||||||
|
|
||||||
async function runMigration() {
|
async function runMigration() {
|
||||||
await runner.runSchematicAsync('migration-v8-static-queries', {}, tree).toPromise();
|
await runner.runSchematicAsync('migration-static-queries', {}, tree).toPromise();
|
||||||
}
|
}
|
||||||
|
|
||||||
function createQueryTests(queryType: 'ViewChild' | 'ContentChild') {
|
function createQueryTests(queryType: 'ViewChild' | 'ContentChild') {
|
||||||
it('should mark queries as dynamic', async() => {
|
it('should mark queries as dynamic', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
@${queryType}('test') unused: any;
|
@${queryType}('test') unused: any;
|
||||||
@${queryType}('dynamic') dynamic: any;
|
@${queryType}('dynamic') dynamic: any;
|
||||||
|
|
||||||
onClick() {
|
onClick() {
|
||||||
this.dynamicQuery.classList.add('test');
|
this.dynamicQuery.classList.add('test');
|
||||||
}
|
}
|
||||||
|
@ -175,13 +175,13 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should mark queries used in "ngOnChanges" as static', async() => {
|
it('should mark queries used in "ngOnChanges" as static', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
@${queryType}('test') query: any;
|
@${queryType}('test') query: any;
|
||||||
|
|
||||||
ngOnChanges() {
|
ngOnChanges() {
|
||||||
this.query.classList.add('test');
|
this.query.classList.add('test');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
`);
|
`);
|
||||||
|
@ -195,13 +195,13 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should mark queries used in "ngOnInit" as static', async() => {
|
it('should mark queries used in "ngOnInit" as static', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
@${queryType}('test') query: any;
|
@${queryType}('test') query: any;
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
this.query.classList.add('test');
|
this.query.classList.add('test');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
`);
|
`);
|
||||||
|
@ -215,13 +215,13 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should mark queries used in "ngDoCheck" as static', async() => {
|
it('should mark queries used in "ngDoCheck" as static', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
@${queryType}('test') query: any;
|
@${queryType}('test') query: any;
|
||||||
|
|
||||||
ngDoCheck() {
|
ngDoCheck() {
|
||||||
this.query.classList.add('test');
|
this.query.classList.add('test');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
`);
|
`);
|
||||||
|
@ -235,11 +235,11 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should keep existing query options when updating timing', async() => {
|
it('should keep existing query options when updating timing', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
@${queryType}('test', { /* test */ read: null }) query: any;
|
@${queryType}('test', { /* test */ read: null }) query: any;
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
this.query.classList.add('test');
|
this.query.classList.add('test');
|
||||||
}
|
}
|
||||||
|
@ -255,7 +255,7 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should not overwrite existing explicit query timing', async() => {
|
it('should not overwrite existing explicit query timing', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
@${queryType}('test', {static: /* untouched */ someVal}) query: any;
|
@${queryType}('test', {static: /* untouched */ someVal}) query: any;
|
||||||
|
@ -271,26 +271,26 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should detect queries used in deep method chain', async() => {
|
it('should detect queries used in deep method chain', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
// We intentionally add this comma for the second parameter in order
|
// We intentionally add this comma for the second parameter in order
|
||||||
// to ensure that the migration does not incorrectly create an invalid
|
// to ensure that the migration does not incorrectly create an invalid
|
||||||
// decorator call with three parameters. e.g. "ViewQuery('test', {...}, )"
|
// decorator call with three parameters. e.g. "ViewQuery('test', {...}, )"
|
||||||
@${queryType}('test', ) query: any;
|
@${queryType}('test', ) query: any;
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
this.a();
|
this.a();
|
||||||
}
|
}
|
||||||
|
|
||||||
a() {
|
a() {
|
||||||
this.b();
|
this.b();
|
||||||
}
|
}
|
||||||
|
|
||||||
b() {
|
b() {
|
||||||
this.c();
|
this.c();
|
||||||
}
|
}
|
||||||
|
|
||||||
c() {
|
c() {
|
||||||
console.log(this.query);
|
console.log(this.query);
|
||||||
}
|
}
|
||||||
|
@ -306,16 +306,16 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should properly exit if recursive function is analyzed', async() => {
|
it('should properly exit if recursive function is analyzed', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
@${queryType}('test') query: any;
|
@${queryType}('test') query: any;
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
this.recursive();
|
this.recursive();
|
||||||
}
|
}
|
||||||
|
|
||||||
recursive() {
|
recursive() {
|
||||||
this.recursive();
|
this.recursive();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -330,27 +330,27 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should detect queries used in newly instantiated classes', async() => {
|
it('should detect queries used in newly instantiated classes', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
@${queryType}('test') query: any;
|
@${queryType}('test') query: any;
|
||||||
@${queryType}('test') query2: any;
|
@${queryType}('test') query2: any;
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
new A(this);
|
new A(this);
|
||||||
|
|
||||||
new class Inline {
|
new class Inline {
|
||||||
constructor(private ctx: MyComp) {
|
constructor(private ctx: MyComp) {
|
||||||
this.a();
|
this.a();
|
||||||
}
|
}
|
||||||
|
|
||||||
a() {
|
a() {
|
||||||
this.ctx.query2.useStatically();
|
this.ctx.query2.useStatically();
|
||||||
}
|
}
|
||||||
}(this);
|
}(this);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export class A {
|
export class A {
|
||||||
constructor(ctx: MyComp) {
|
constructor(ctx: MyComp) {
|
||||||
ctx.query.test();
|
ctx.query.test();
|
||||||
|
@ -369,16 +369,16 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should detect queries used in parenthesized new expressions', async() => {
|
it('should detect queries used in parenthesized new expressions', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
@${queryType}('test') query: any;
|
@${queryType}('test') query: any;
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
new ((A))(this);
|
new ((A))(this);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export class A {
|
export class A {
|
||||||
constructor(ctx: MyComp) {
|
constructor(ctx: MyComp) {
|
||||||
ctx.query.test();
|
ctx.query.test();
|
||||||
|
@ -395,11 +395,11 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should detect queries in lifecycle hook with string literal name', async() => {
|
it('should detect queries in lifecycle hook with string literal name', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
@${queryType}('test') query: any;
|
@${queryType}('test') query: any;
|
||||||
|
|
||||||
'ngOnInit'() {
|
'ngOnInit'() {
|
||||||
this.query.test();
|
this.query.test();
|
||||||
}
|
}
|
||||||
|
@ -415,19 +415,19 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should detect static queries within nested inheritance', async() => {
|
it('should detect static queries within nested inheritance', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
@${queryType}('test') query: any;
|
@${queryType}('test') query: any;
|
||||||
}
|
}
|
||||||
|
|
||||||
export class A extends MyComp {}
|
export class A extends MyComp {}
|
||||||
export class B extends A {
|
export class B extends A {
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
this.query.testFn();
|
this.query.testFn();
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
`);
|
`);
|
||||||
|
|
||||||
|
@ -440,11 +440,11 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should detect static queries used within input setters', async() => {
|
it('should detect static queries used within input setters', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, Input, ${queryType}} from '@angular/core';
|
import {Component, Input, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
@${queryType}('test') query: any;
|
@${queryType}('test') query: any;
|
||||||
|
|
||||||
@Input()
|
@Input()
|
||||||
get myVal() { return null; }
|
get myVal() { return null; }
|
||||||
set myVal(newVal: any) {
|
set myVal(newVal: any) {
|
||||||
|
@ -462,14 +462,14 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should detect inputs defined in metadata', async() => {
|
it('should detect inputs defined in metadata', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
template: '<span #test></span>',
|
template: '<span #test></span>',
|
||||||
inputs: ["myVal"],
|
inputs: ["myVal"],
|
||||||
})
|
})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
@${queryType}('test') query: any;
|
@${queryType}('test') query: any;
|
||||||
|
|
||||||
// We don't use the input decorator here as we want to verify
|
// We don't use the input decorator here as we want to verify
|
||||||
// that it properly detects the input through the component metadata.
|
// that it properly detects the input through the component metadata.
|
||||||
get myVal() { return null; }
|
get myVal() { return null; }
|
||||||
|
@ -488,14 +488,14 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should detect aliased inputs declared in metadata', async() => {
|
it('should detect aliased inputs declared in metadata', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
template: '<span #test></span>',
|
template: '<span #test></span>',
|
||||||
inputs: ['prop: publicName'],
|
inputs: ['prop: publicName'],
|
||||||
})
|
})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
@${queryType}('test') query: any;
|
@${queryType}('test') query: any;
|
||||||
|
|
||||||
set prop(val: any) {
|
set prop(val: any) {
|
||||||
this.query.test();
|
this.query.test();
|
||||||
}
|
}
|
||||||
|
@ -511,11 +511,11 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should not mark query as static if query is used in non-input setter', async() => {
|
it('should not mark query as static if query is used in non-input setter', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
@${queryType}('test') query: any;
|
@${queryType}('test') query: any;
|
||||||
|
|
||||||
set myProperty(val: any) {
|
set myProperty(val: any) {
|
||||||
this.query.test();
|
this.query.test();
|
||||||
}
|
}
|
||||||
|
@ -531,13 +531,13 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should detect input decorator on setter', async() => {
|
it('should detect input decorator on setter', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Input, Component, ${queryType}} from '@angular/core';
|
import {Input, Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
@${queryType}('test') query: any;
|
@${queryType}('test') query: any;
|
||||||
|
|
||||||
get myProperty() { return null; }
|
get myProperty() { return null; }
|
||||||
|
|
||||||
// Usually the decorator is set on the get accessor, but it's also possible
|
// Usually the decorator is set on the get accessor, but it's also possible
|
||||||
// to declare the input on the setter. This ensures that it is handled properly.
|
// to declare the input on the setter. This ensures that it is handled properly.
|
||||||
@Input()
|
@Input()
|
||||||
|
@ -556,7 +556,7 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should detect setter inputs in derived classes', async() => {
|
it('should detect setter inputs in derived classes', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
template: '<span #test></span>',
|
template: '<span #test></span>',
|
||||||
inputs: ['childSetter'],
|
inputs: ['childSetter'],
|
||||||
|
@ -564,7 +564,7 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
protected @${queryType}('test') query: any;
|
protected @${queryType}('test') query: any;
|
||||||
}
|
}
|
||||||
|
|
||||||
export class B extends MyComp {
|
export class B extends MyComp {
|
||||||
set childSetter(newVal: any) {
|
set childSetter(newVal: any) {
|
||||||
this.query.test();
|
this.query.test();
|
||||||
|
@ -581,7 +581,7 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should properly detect static query in external derived class', async() => {
|
it('should properly detect static query in external derived class', async() => {
|
||||||
writeFile('/src/index.ts', `
|
writeFile('/src/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
@${queryType}('test') query: any;
|
@${queryType}('test') query: any;
|
||||||
|
@ -590,7 +590,7 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
|
|
||||||
writeFile('/src/external.ts', `
|
writeFile('/src/external.ts', `
|
||||||
import {MyComp} from './index';
|
import {MyComp} from './index';
|
||||||
|
|
||||||
export class ExternalComp extends MyComp {
|
export class ExternalComp extends MyComp {
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
this.query.test();
|
this.query.test();
|
||||||
|
@ -614,30 +614,30 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should not mark queries used in promises as static', async() => {
|
it('should not mark queries used in promises as static', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
private @${queryType}('test') query: any;
|
private @${queryType}('test') query: any;
|
||||||
private @${queryType}('test') query2: any;
|
private @${queryType}('test') query2: any;
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
const a = Promise.resolve();
|
const a = Promise.resolve();
|
||||||
|
|
||||||
Promise.resolve().then(() => {
|
Promise.resolve().then(() => {
|
||||||
this.query.doSomething();
|
this.query.doSomething();
|
||||||
});
|
});
|
||||||
|
|
||||||
Promise.reject().catch(() => {
|
Promise.reject().catch(() => {
|
||||||
this.query.doSomething();
|
this.query.doSomething();
|
||||||
});
|
});
|
||||||
|
|
||||||
a.then(() => {}).then(() => {
|
a.then(() => {}).then(() => {
|
||||||
this.query.doSomething();
|
this.query.doSomething();
|
||||||
});
|
});
|
||||||
|
|
||||||
Promise.resolve().then(this.createPromiseCb());
|
Promise.resolve().then(this.createPromiseCb());
|
||||||
}
|
}
|
||||||
|
|
||||||
createPromiseCb() {
|
createPromiseCb() {
|
||||||
this.query2.doSomething();
|
this.query2.doSomething();
|
||||||
return () => { /* empty callback */}
|
return () => { /* empty callback */}
|
||||||
|
@ -656,19 +656,19 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should handle function callbacks which statically access queries', async() => {
|
it('should handle function callbacks which statically access queries', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
private @${queryType}('test') query: any;
|
private @${queryType}('test') query: any;
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
this.callSync(() => this.query.doSomething());
|
this.callSync(() => this.query.doSomething());
|
||||||
}
|
}
|
||||||
|
|
||||||
callSync(cb: Function) {
|
callSync(cb: Function) {
|
||||||
this.callSync2(cb);
|
this.callSync2(cb);
|
||||||
}
|
}
|
||||||
|
|
||||||
callSync2(cb: Function) {
|
callSync2(cb: Function) {
|
||||||
cb();
|
cb();
|
||||||
}
|
}
|
||||||
|
@ -686,12 +686,12 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
import {External} from './external';
|
import {External} from './external';
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
private @${queryType}('test') query: any;
|
private @${queryType}('test') query: any;
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
new External(() => this.query.doSomething());
|
new External(() => this.query.doSomething());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -700,7 +700,7 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
writeFile('/external.ts', `
|
writeFile('/external.ts', `
|
||||||
export class External {
|
export class External {
|
||||||
constructor(cb: () => void) {
|
constructor(cb: () => void) {
|
||||||
// Add extra parentheses to ensure that expression is unwrapped.
|
// Add extra parentheses to ensure that expression is unwrapped.
|
||||||
((cb))();
|
((cb))();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -715,21 +715,21 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should handle nested functions with arguments from parent closure', async() => {
|
it('should handle nested functions with arguments from parent closure', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
private @${queryType}('test') query: any;
|
private @${queryType}('test') query: any;
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
this.callSync(() => this.query.doSomething());
|
this.callSync(() => this.query.doSomething());
|
||||||
}
|
}
|
||||||
|
|
||||||
callSync(cb: Function) {
|
callSync(cb: Function) {
|
||||||
function callSyncNested() {
|
function callSyncNested() {
|
||||||
// The "cb" identifier comes from the "callSync" function.
|
// The "cb" identifier comes from the "callSync" function.
|
||||||
cb();
|
cb();
|
||||||
}
|
}
|
||||||
|
|
||||||
callSyncNested();
|
callSyncNested();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -744,22 +744,22 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should not mark queries used in setTimeout as static', async() => {
|
it('should not mark queries used in setTimeout as static', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
private @${queryType}('test') query: any;
|
private @${queryType}('test') query: any;
|
||||||
private @${queryType}('test') query2: any;
|
private @${queryType}('test') query2: any;
|
||||||
private @${queryType}('test') query3: any;
|
private @${queryType}('test') query3: any;
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
setTimeout(function() {
|
setTimeout(function() {
|
||||||
this.query.doSomething();
|
this.query.doSomething();
|
||||||
});
|
});
|
||||||
|
|
||||||
setTimeout(createCallback(this));
|
setTimeout(createCallback(this));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function createCallback(instance: MyComp) {
|
function createCallback(instance: MyComp) {
|
||||||
instance.query2.doSomething();
|
instance.query2.doSomething();
|
||||||
return () => instance.query3.doSomething();
|
return () => instance.query3.doSomething();
|
||||||
|
@ -779,13 +779,13 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should not mark queries used in "addEventListener" as static', async() => {
|
it('should not mark queries used in "addEventListener" as static', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ElementRef, ${queryType}} from '@angular/core';
|
import {Component, ElementRef, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
private @${queryType}('test') query: any;
|
private @${queryType}('test') query: any;
|
||||||
|
|
||||||
constructor(private elementRef: ElementRef) {}
|
constructor(private elementRef: ElementRef) {}
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
this.elementRef.addEventListener(() => {
|
this.elementRef.addEventListener(() => {
|
||||||
this.query.classList.add('test');
|
this.query.classList.add('test');
|
||||||
|
@ -803,13 +803,13 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should not mark queries used in "requestAnimationFrame" as static', async() => {
|
it('should not mark queries used in "requestAnimationFrame" as static', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ElementRef, ${queryType}} from '@angular/core';
|
import {Component, ElementRef, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
private @${queryType}('test') query: any;
|
private @${queryType}('test') query: any;
|
||||||
|
|
||||||
constructor(private elementRef: ElementRef) {}
|
constructor(private elementRef: ElementRef) {}
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
requestAnimationFrame(() => {
|
requestAnimationFrame(() => {
|
||||||
this.query.classList.add('test');
|
this.query.classList.add('test');
|
||||||
|
@ -827,17 +827,17 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should mark queries used in immediately-invoked function expression as static', async() => {
|
it('should mark queries used in immediately-invoked function expression as static', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
private @${queryType}('test') query: any;
|
private @${queryType}('test') query: any;
|
||||||
private @${queryType}('test') query2: any;
|
private @${queryType}('test') query2: any;
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
(() => {
|
(() => {
|
||||||
this.query.usedStatically();
|
this.query.usedStatically();
|
||||||
})();
|
})();
|
||||||
|
|
||||||
(function(ctx) {
|
(function(ctx) {
|
||||||
ctx.query2.useStatically();
|
ctx.query2.useStatically();
|
||||||
})(this);
|
})(this);
|
||||||
|
@ -857,11 +857,11 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
import {externalFn} from './external';
|
import {externalFn} from './external';
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
private @${queryType}('test') query: any;
|
private @${queryType}('test') query: any;
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
externalFn(this);
|
externalFn(this);
|
||||||
}
|
}
|
||||||
|
@ -870,7 +870,7 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
|
|
||||||
writeFile('/external.ts', `
|
writeFile('/external.ts', `
|
||||||
import {MyComp} from './index';
|
import {MyComp} from './index';
|
||||||
|
|
||||||
export function externalFn(ctx: MyComp) {
|
export function externalFn(ctx: MyComp) {
|
||||||
ctx.query.usedStatically();
|
ctx.query.usedStatically();
|
||||||
}
|
}
|
||||||
|
@ -885,15 +885,15 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should detect static queries used through getter property access', async() => {
|
it('should detect static queries used through getter property access', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
private @${queryType}('test') query: any;
|
private @${queryType}('test') query: any;
|
||||||
|
|
||||||
get myProp() {
|
get myProp() {
|
||||||
return this.query.myValue;
|
return this.query.myValue;
|
||||||
}
|
}
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
this.myProp.test();
|
this.myProp.test();
|
||||||
}
|
}
|
||||||
|
@ -910,17 +910,17 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
import {External} from './external';
|
import {External} from './external';
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
@${queryType}('test') query: any;
|
@${queryType}('test') query: any;
|
||||||
|
|
||||||
private external = new External(this);
|
private external = new External(this);
|
||||||
|
|
||||||
get myProp() {
|
get myProp() {
|
||||||
return this.query.myValue;
|
return this.query.myValue;
|
||||||
}
|
}
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
console.log(this.external.query);
|
console.log(this.external.query);
|
||||||
}
|
}
|
||||||
|
@ -929,10 +929,10 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
|
|
||||||
writeFile('/external.ts', `
|
writeFile('/external.ts', `
|
||||||
import {MyComp} from './index';
|
import {MyComp} from './index';
|
||||||
|
|
||||||
export class External {
|
export class External {
|
||||||
constructor(private comp: MyComp) {}
|
constructor(private comp: MyComp) {}
|
||||||
|
|
||||||
set query() { /** noop */ }
|
set query() { /** noop */ }
|
||||||
get query() { return this.comp.query; }
|
get query() { return this.comp.query; }
|
||||||
}
|
}
|
||||||
|
@ -947,16 +947,16 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should not mark queries as static if a value is assigned to accessor property', async() => {
|
it('should not mark queries as static if a value is assigned to accessor property', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
private @${queryType}('test') query: any;
|
private @${queryType}('test') query: any;
|
||||||
|
|
||||||
set myProp(value: any) { /* noop */}
|
set myProp(value: any) { /* noop */}
|
||||||
get myProp() {
|
get myProp() {
|
||||||
return this.query.myValue;
|
return this.query.myValue;
|
||||||
}
|
}
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
this.myProp = true;
|
this.myProp = true;
|
||||||
}
|
}
|
||||||
|
@ -972,16 +972,16 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should mark queries as static if non-input setter uses query', async() => {
|
it('should mark queries as static if non-input setter uses query', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
private @${queryType}('test') query: any;
|
private @${queryType}('test') query: any;
|
||||||
|
|
||||||
get myProp() { return null; }
|
get myProp() { return null; }
|
||||||
set myProp(value: any) {
|
set myProp(value: any) {
|
||||||
this.query.doSomething();
|
this.query.doSomething();
|
||||||
}
|
}
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
this.myProp = 'newValue';
|
this.myProp = 'newValue';
|
||||||
}
|
}
|
||||||
|
@ -997,17 +997,17 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should check setter and getter when using compound assignment', async() => {
|
it('should check setter and getter when using compound assignment', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
private @${queryType}('test') query: any;
|
private @${queryType}('test') query: any;
|
||||||
private @${queryType}('test') query2: any;
|
private @${queryType}('test') query2: any;
|
||||||
|
|
||||||
get myProp() { return this.query2 }
|
get myProp() { return this.query2 }
|
||||||
set myProp(value: any) {
|
set myProp(value: any) {
|
||||||
this.query.doSomething();
|
this.query.doSomething();
|
||||||
}
|
}
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
this.myProp *= 5;
|
this.myProp *= 5;
|
||||||
}
|
}
|
||||||
|
@ -1025,14 +1025,14 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should check getters when using comparison operator in binary expression', async() => {
|
it('should check getters when using comparison operator in binary expression', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
private @${queryType}('test') query: any;
|
private @${queryType}('test') query: any;
|
||||||
|
|
||||||
get myProp() { return this.query }
|
get myProp() { return this.query }
|
||||||
set myProp(value: any) { /* noop */ }
|
set myProp(value: any) { /* noop */ }
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
if (this.myProp === 3) {
|
if (this.myProp === 3) {
|
||||||
// noop
|
// noop
|
||||||
|
@ -1050,29 +1050,29 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should check derived abstract class methods', async() => {
|
it('should check derived abstract class methods', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
export abstract class RootBaseClass {
|
export abstract class RootBaseClass {
|
||||||
abstract getQuery(): any;
|
abstract getQuery(): any;
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
this.getQuery().doSomething();
|
this.getQuery().doSomething();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export abstract class BaseClass extends RootBaseClass {
|
export abstract class BaseClass extends RootBaseClass {
|
||||||
abstract getQuery2(): any;
|
abstract getQuery2(): any;
|
||||||
|
|
||||||
getQuery() {
|
getQuery() {
|
||||||
this.getQuery2();
|
this.getQuery2();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class Subclass extends BaseClass {
|
export class Subclass extends BaseClass {
|
||||||
@${queryType}('test') query: any;
|
@${queryType}('test') query: any;
|
||||||
|
|
||||||
getQuery2(): any {
|
getQuery2(): any {
|
||||||
return this.query;
|
return this.query;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
`);
|
`);
|
||||||
|
@ -1086,25 +1086,25 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should detect queries accessed through deep abstract class method', async() => {
|
it('should detect queries accessed through deep abstract class method', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
export abstract class RootBaseClass {
|
export abstract class RootBaseClass {
|
||||||
abstract getQuery(): any;
|
abstract getQuery(): any;
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
this.getQuery().doSomething();
|
this.getQuery().doSomething();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export abstract class BaseClass extends RootBaseClass {
|
export abstract class BaseClass extends RootBaseClass {
|
||||||
/* additional layer of indirection */
|
/* additional layer of indirection */
|
||||||
}
|
}
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class Subclass extends BaseClass {
|
export class Subclass extends BaseClass {
|
||||||
@${queryType}('test') query: any;
|
@${queryType}('test') query: any;
|
||||||
|
|
||||||
getQuery(): any {
|
getQuery(): any {
|
||||||
return this.query;
|
return this.query;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
`);
|
`);
|
||||||
|
@ -1118,19 +1118,19 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should detect queries accessed through abstract property getter', async() => {
|
it('should detect queries accessed through abstract property getter', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
export abstract class BaseClass {
|
export abstract class BaseClass {
|
||||||
abstract myQuery: any;
|
abstract myQuery: any;
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
this.myQuery.doSomething();
|
this.myQuery.doSomething();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class Subclass extends BaseClass {
|
export class Subclass extends BaseClass {
|
||||||
@${queryType}('test') query: any;
|
@${queryType}('test') query: any;
|
||||||
|
|
||||||
get myQuery() { return this.query; }
|
get myQuery() { return this.query; }
|
||||||
}
|
}
|
||||||
`);
|
`);
|
||||||
|
@ -1144,19 +1144,19 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should detect queries accessed through abstract property setter', async() => {
|
it('should detect queries accessed through abstract property setter', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
export abstract class BaseClass {
|
export abstract class BaseClass {
|
||||||
abstract myQuery: any;
|
abstract myQuery: any;
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
this.myQuery = "trigger";
|
this.myQuery = "trigger";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class Subclass extends BaseClass {
|
export class Subclass extends BaseClass {
|
||||||
@${queryType}('test') query: any;
|
@${queryType}('test') query: any;
|
||||||
|
|
||||||
set myQuery(val: any) { this.query.doSomething() }
|
set myQuery(val: any) { this.query.doSomething() }
|
||||||
get myQuery() { /* noop */ }
|
get myQuery() { /* noop */ }
|
||||||
}
|
}
|
||||||
|
@ -1171,27 +1171,27 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should detect query usage in abstract class methods accessing inherited query', async() => {
|
it('should detect query usage in abstract class methods accessing inherited query', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
export abstract class RootBaseClass {
|
export abstract class RootBaseClass {
|
||||||
abstract getQuery(): any;
|
abstract getQuery(): any;
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
this.getQuery().doSomething();
|
this.getQuery().doSomething();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export abstract class BaseClass extends RootBaseClass {
|
export abstract class BaseClass extends RootBaseClass {
|
||||||
@${queryType}('test') query: any;
|
@${queryType}('test') query: any;
|
||||||
abstract getQuery2(): any;
|
abstract getQuery2(): any;
|
||||||
|
|
||||||
getQuery() {
|
getQuery() {
|
||||||
this.getQuery2();
|
this.getQuery2();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class Subclass extends BaseClass {
|
export class Subclass extends BaseClass {
|
||||||
|
|
||||||
getQuery2(): any {
|
getQuery2(): any {
|
||||||
return this.query;
|
return this.query;
|
||||||
}
|
}
|
||||||
|
@ -1207,7 +1207,7 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should detect query usage within component template', async() => {
|
it('should detect query usage within component template', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
@Component({templateUrl: 'my-template.html'})
|
@Component({templateUrl: 'my-template.html'})
|
||||||
export class MyComponent {
|
export class MyComponent {
|
||||||
@${queryType}('test') query: any;
|
@${queryType}('test') query: any;
|
||||||
|
@ -1228,7 +1228,7 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should detect query usage with nested property read within component template', async() => {
|
it('should detect query usage with nested property read within component template', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
@Component({templateUrl: 'my-template.html'})
|
@Component({templateUrl: 'my-template.html'})
|
||||||
export class MyComponent {
|
export class MyComponent {
|
||||||
@${queryType}('test') query: any;
|
@${queryType}('test') query: any;
|
||||||
|
@ -1250,7 +1250,7 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
async() => {
|
async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
@Component({templateUrl: 'my-template.html'})
|
@Component({templateUrl: 'my-template.html'})
|
||||||
export class MyComponent {
|
export class MyComponent {
|
||||||
@${queryType}('test') query: any;
|
@${queryType}('test') query: any;
|
||||||
|
@ -1274,7 +1274,7 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
async() => {
|
async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
@Component({templateUrl: 'my-template.html'})
|
@Component({templateUrl: 'my-template.html'})
|
||||||
export class MyComponent {
|
export class MyComponent {
|
||||||
myObject: {someProp: any};
|
myObject: {someProp: any};
|
||||||
|
@ -1299,7 +1299,7 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should ignore queries accessed within <ng-template> element', async() => {
|
it('should ignore queries accessed within <ng-template> element', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
@Component({templateUrl: 'my-template.html'})
|
@Component({templateUrl: 'my-template.html'})
|
||||||
export class MyComponent {
|
export class MyComponent {
|
||||||
@${queryType}('test') query: any;
|
@${queryType}('test') query: any;
|
||||||
|
@ -1308,7 +1308,7 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
|
|
||||||
writeFile(`/my-template.html`, `
|
writeFile(`/my-template.html`, `
|
||||||
<foo #test></foo>
|
<foo #test></foo>
|
||||||
|
|
||||||
<ng-template>
|
<ng-template>
|
||||||
<my-comp [myInput]="query"></my-comp>
|
<my-comp [myInput]="query"></my-comp>
|
||||||
</ng-template>
|
</ng-template>
|
||||||
|
@ -1323,11 +1323,11 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should detect inherited queries used in templates', async() => {
|
it('should detect inherited queries used in templates', async() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
export class ParentClass {
|
export class ParentClass {
|
||||||
@${queryType}('test') query: any;
|
@${queryType}('test') query: any;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Component({templateUrl: 'my-template.html'})
|
@Component({templateUrl: 'my-template.html'})
|
||||||
export class MyComponent extends ParentClass {}
|
export class MyComponent extends ParentClass {}
|
||||||
`);
|
`);
|
||||||
|
@ -1346,7 +1346,7 @@ describe('static-queries migration with usage strategy', () => {
|
||||||
it('should properly handle multiple tsconfig files', async() => {
|
it('should properly handle multiple tsconfig files', async() => {
|
||||||
writeFile('/src/index.ts', `
|
writeFile('/src/index.ts', `
|
||||||
import {Component, ${queryType}} from '@angular/core';
|
import {Component, ${queryType}} from '@angular/core';
|
||||||
|
|
||||||
@Component({template: '<span #test></span>'})
|
@Component({template: '<span #test></span>'})
|
||||||
export class MyComp {
|
export class MyComp {
|
||||||
private @${queryType}('test') query: any;
|
private @${queryType}('test') query: any;
|
||||||
|
|
|
@ -21,7 +21,7 @@ describe('template variable assignment migration', () => {
|
||||||
let warnOutput: string[];
|
let warnOutput: string[];
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
runner = new SchematicTestRunner('test', require.resolve('../migrations.json'));
|
runner = new SchematicTestRunner('test', require.resolve('../test-migrations.json'));
|
||||||
host = new TempScopedNodeJsSyncHost();
|
host = new TempScopedNodeJsSyncHost();
|
||||||
tree = new UnitTestTree(new HostTree(host));
|
tree = new UnitTestTree(new HostTree(host));
|
||||||
|
|
||||||
|
@ -58,14 +58,12 @@ describe('template variable assignment migration', () => {
|
||||||
host.sync.write(normalize(filePath), virtualFs.stringToFileBuffer(contents));
|
host.sync.write(normalize(filePath), virtualFs.stringToFileBuffer(contents));
|
||||||
}
|
}
|
||||||
|
|
||||||
function runMigration() {
|
function runMigration() { runner.runSchematic('migration-template-local-variables', {}, tree); }
|
||||||
runner.runSchematic('migration-v8-template-local-variables', {}, tree);
|
|
||||||
}
|
|
||||||
|
|
||||||
it('should warn for two-way data binding variable assignment', () => {
|
it('should warn for two-way data binding variable assignment', () => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component} from '@angular/core';
|
import {Component} from '@angular/core';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
template: '<cmp *ngFor="let optionName of options" [(opt)]="optionName"></cmp>',
|
template: '<cmp *ngFor="let optionName of options" [(opt)]="optionName"></cmp>',
|
||||||
})
|
})
|
||||||
|
@ -81,7 +79,7 @@ describe('template variable assignment migration', () => {
|
||||||
it('should warn for two-way data binding assigning to "as" variable', () => {
|
it('should warn for two-way data binding assigning to "as" variable', () => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component} from '@angular/core';
|
import {Component} from '@angular/core';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
templateUrl: './tmpl.html',
|
templateUrl: './tmpl.html',
|
||||||
})
|
})
|
||||||
|
@ -103,7 +101,7 @@ describe('template variable assignment migration', () => {
|
||||||
it('should warn for bound event assignments to "as" variable', () => {
|
it('should warn for bound event assignments to "as" variable', () => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component} from '@angular/core';
|
import {Component} from '@angular/core';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
templateUrl: './sub_dir/tmpl.html',
|
templateUrl: './sub_dir/tmpl.html',
|
||||||
})
|
})
|
||||||
|
@ -127,7 +125,7 @@ describe('template variable assignment migration', () => {
|
||||||
it('should warn for bound event assignments to template "let" variables', () => {
|
it('should warn for bound event assignments to template "let" variables', () => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component} from '@angular/core';
|
import {Component} from '@angular/core';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
templateUrl: './sub_dir/tmpl.html',
|
templateUrl: './sub_dir/tmpl.html',
|
||||||
})
|
})
|
||||||
|
@ -151,7 +149,7 @@ describe('template variable assignment migration', () => {
|
||||||
it('should not warn for bound event assignments to component property', () => {
|
it('should not warn for bound event assignments to component property', () => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component} from '@angular/core';
|
import {Component} from '@angular/core';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
templateUrl: './sub_dir/tmpl.html',
|
templateUrl: './sub_dir/tmpl.html',
|
||||||
})
|
})
|
||||||
|
@ -168,7 +166,7 @@ describe('template variable assignment migration', () => {
|
||||||
it('should not warn for bound event assignments to template variable object property', () => {
|
it('should not warn for bound event assignments to template variable object property', () => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component} from '@angular/core';
|
import {Component} from '@angular/core';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
templateUrl: './sub_dir/tmpl.html',
|
templateUrl: './sub_dir/tmpl.html',
|
||||||
})
|
})
|
||||||
|
@ -188,7 +186,7 @@ describe('template variable assignment migration', () => {
|
||||||
() => {
|
() => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component} from '@angular/core';
|
import {Component} from '@angular/core';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
templateUrl: './sub_dir/tmpl.html',
|
templateUrl: './sub_dir/tmpl.html',
|
||||||
})
|
})
|
||||||
|
@ -213,7 +211,7 @@ describe('template variable assignment migration', () => {
|
||||||
it('should not warn for property writes with template variable name but different scope', () => {
|
it('should not warn for property writes with template variable name but different scope', () => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component} from '@angular/core';
|
import {Component} from '@angular/core';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
templateUrl: './sub_dir/tmpl.html',
|
templateUrl: './sub_dir/tmpl.html',
|
||||||
})
|
})
|
||||||
|
@ -236,7 +234,7 @@ describe('template variable assignment migration', () => {
|
||||||
it('should not throw an error if a detected template fails parsing', () => {
|
it('should not throw an error if a detected template fails parsing', () => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component} from '@angular/core';
|
import {Component} from '@angular/core';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
templateUrl: './sub_dir/tmpl.html',
|
templateUrl: './sub_dir/tmpl.html',
|
||||||
})
|
})
|
||||||
|
@ -253,12 +251,12 @@ describe('template variable assignment migration', () => {
|
||||||
it('should be able to report multiple templates within the same source file', () => {
|
it('should be able to report multiple templates within the same source file', () => {
|
||||||
writeFile('/index.ts', `
|
writeFile('/index.ts', `
|
||||||
import {Component} from '@angular/core';
|
import {Component} from '@angular/core';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
template: '<ng-template let-one><a (sayHello)="one=true"></a></ng-template>',
|
template: '<ng-template let-one><a (sayHello)="one=true"></a></ng-template>',
|
||||||
})
|
})
|
||||||
export class MyComp {}
|
export class MyComp {}
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
template: '<ng-template let-two><b (greet)="two=true"></b></ng-template>',
|
template: '<ng-template let-two><b (greet)="two=true"></b></ng-template>',
|
||||||
})
|
})
|
||||||
|
|
Loading…
Reference in New Issue