394 lines
14 KiB
TypeScript
394 lines
14 KiB
TypeScript
|
/**
|
||
|
* @license
|
||
|
* Copyright Google LLC All Rights Reserved.
|
||
|
*
|
||
|
* Use of this source code is governed by an MIT-style license that can be
|
||
|
* found in the LICENSE file at https://angular.io/license
|
||
|
*/
|
||
|
|
||
|
import * as ts from 'typescript/lib/tsserverlibrary';
|
||
|
|
||
|
import {LanguageService} from '../language_service';
|
||
|
|
||
|
import {APP_COMPONENT, setup, TEST_TEMPLATE} from './mock_host';
|
||
|
|
||
|
describe('quick info', () => {
|
||
|
const {project, service, tsLS} = setup();
|
||
|
const ngLS = new LanguageService(project, tsLS);
|
||
|
|
||
|
beforeEach(() => {
|
||
|
service.reset();
|
||
|
});
|
||
|
|
||
|
describe('elements', () => {
|
||
|
it('should work for native elements', () => {
|
||
|
expectQuickInfo({
|
||
|
templateOverride: `<butt¦on></button>`,
|
||
|
expectedSpanText: '<button></button>',
|
||
|
expectedDisplayString: '(element) button: HTMLButtonElement'
|
||
|
});
|
||
|
});
|
||
|
});
|
||
|
|
||
|
describe('templates', () => {
|
||
|
it('should return undefined for ng-templates', () => {
|
||
|
const {documentation} = expectQuickInfo({
|
||
|
templateOverride: `<ng-templ¦ate></ng-template>`,
|
||
|
expectedSpanText: '<ng-template></ng-template>',
|
||
|
expectedDisplayString: '(template) ng-template'
|
||
|
});
|
||
|
expect(toText(documentation))
|
||
|
.toContain('The `<ng-template>` is an Angular element for rendering HTML.');
|
||
|
});
|
||
|
});
|
||
|
|
||
|
describe('directives', () => {
|
||
|
it('should work for directives', () => {
|
||
|
expectQuickInfo({
|
||
|
templateOverride: `<div string-model¦></div>`,
|
||
|
expectedSpanText: 'string-model',
|
||
|
// TODO(atscott): Find a way to include the module
|
||
|
// expectedDisplayParts: '(directive) AppModule.StringModel'
|
||
|
expectedDisplayString: '(directive) StringModel'
|
||
|
});
|
||
|
});
|
||
|
|
||
|
it('should work for components', () => {
|
||
|
const {documentation} = expectQuickInfo({
|
||
|
templateOverride: `<t¦est-comp></test-comp>`,
|
||
|
expectedSpanText: '<test-comp></test-comp>',
|
||
|
// TODO(atscott): Find a way to include the module
|
||
|
// expectedDisplayParts: '(component) AppModule.TestComponent'
|
||
|
expectedDisplayString: '(component) TestComponent'
|
||
|
});
|
||
|
expect(toText(documentation)).toBe('This Component provides the `test-comp` selector.');
|
||
|
});
|
||
|
|
||
|
it('should work for structural directives', () => {
|
||
|
const {documentation} = expectQuickInfo({
|
||
|
templateOverride: `<div *¦ngFor="let item of heroes"></div>`,
|
||
|
expectedSpanText: 'ngFor',
|
||
|
expectedDisplayString: '(directive) NgForOf<Hero, Array<Hero>>'
|
||
|
});
|
||
|
expect(toText(documentation))
|
||
|
.toContain('A [structural directive](guide/structural-directives) that renders');
|
||
|
});
|
||
|
|
||
|
it('should work for directives with compound selectors, some of which are bindings', () => {
|
||
|
expectQuickInfo({
|
||
|
templateOverride: `<ng-template ngF¦or let-hero [ngForOf]="heroes">{{item}}</ng-template>`,
|
||
|
expectedSpanText: 'ngFor',
|
||
|
expectedDisplayString: '(directive) NgForOf<Hero, Array<Hero>>'
|
||
|
});
|
||
|
});
|
||
|
|
||
|
it('should work for data-let- syntax', () => {
|
||
|
expectQuickInfo({
|
||
|
templateOverride:
|
||
|
`<ng-template ngFor data-let-he¦ro [ngForOf]="heroes">{{item}}</ng-template>`,
|
||
|
expectedSpanText: 'hero',
|
||
|
expectedDisplayString: '(variable) hero: Hero'
|
||
|
});
|
||
|
});
|
||
|
});
|
||
|
|
||
|
describe('bindings', () => {
|
||
|
describe('inputs', () => {
|
||
|
it('should work for input providers', () => {
|
||
|
expectQuickInfo({
|
||
|
templateOverride: `<test-comp [tcN¦ame]="name"></test-comp>`,
|
||
|
expectedSpanText: 'tcName',
|
||
|
expectedDisplayString: '(property) TestComponent.name: string'
|
||
|
});
|
||
|
});
|
||
|
|
||
|
it('should work for bind- syntax', () => {
|
||
|
expectQuickInfo({
|
||
|
templateOverride: `<test-comp bind-tcN¦ame="name"></test-comp>`,
|
||
|
expectedSpanText: 'tcName',
|
||
|
expectedDisplayString: '(property) TestComponent.name: string'
|
||
|
});
|
||
|
expectQuickInfo({
|
||
|
templateOverride: `<test-comp data-bind-tcN¦ame="name"></test-comp>`,
|
||
|
expectedSpanText: 'tcName',
|
||
|
expectedDisplayString: '(property) TestComponent.name: string'
|
||
|
});
|
||
|
});
|
||
|
|
||
|
it('should work for structural directive inputs ngForTrackBy', () => {
|
||
|
expectQuickInfo({
|
||
|
templateOverride: `<div *ngFor="let item of heroes; tr¦ackBy: test;"></div>`,
|
||
|
expectedSpanText: 'trackBy',
|
||
|
expectedDisplayString:
|
||
|
'(property) NgForOf<Hero, Hero[]>.ngForTrackBy: TrackByFunction<Hero>'
|
||
|
});
|
||
|
});
|
||
|
|
||
|
it('should work for structural directive inputs ngForOf', () => {
|
||
|
expectQuickInfo({
|
||
|
templateOverride: `<div *ngFor="let item o¦f heroes; trackBy: test;"></div>`,
|
||
|
expectedSpanText: 'of',
|
||
|
expectedDisplayString:
|
||
|
'(property) NgForOf<Hero, Hero[]>.ngForOf: Hero[] | (Hero[] & Iterable<Hero>) | null | undefined'
|
||
|
});
|
||
|
});
|
||
|
|
||
|
it('should work for two-way binding providers', () => {
|
||
|
expectQuickInfo({
|
||
|
templateOverride: `<test-comp string-model [(mo¦del)]="title"></test-comp>`,
|
||
|
expectedSpanText: 'model',
|
||
|
expectedDisplayString: '(property) StringModel.model: string'
|
||
|
});
|
||
|
});
|
||
|
});
|
||
|
|
||
|
describe('outputs', () => {
|
||
|
it('should work for event providers', () => {
|
||
|
expectQuickInfo({
|
||
|
templateOverride: `<test-comp (te¦st)="myClick($event)"></test-comp>`,
|
||
|
expectedSpanText: '(test)="myClick($event)"',
|
||
|
expectedDisplayString: '(event) TestComponent.testEvent: EventEmitter<any>'
|
||
|
});
|
||
|
});
|
||
|
|
||
|
it('should work for on- syntax binding', () => {
|
||
|
expectQuickInfo({
|
||
|
templateOverride: `<test-comp on-te¦st="myClick($event)"></test-comp>`,
|
||
|
expectedSpanText: 'on-test="myClick($event)"',
|
||
|
expectedDisplayString: '(event) TestComponent.testEvent: EventEmitter<any>'
|
||
|
});
|
||
|
expectQuickInfo({
|
||
|
templateOverride: `<test-comp data-on-te¦st="myClick($event)"></test-comp>`,
|
||
|
expectedSpanText: 'data-on-test="myClick($event)"',
|
||
|
expectedDisplayString: '(event) TestComponent.testEvent: EventEmitter<any>'
|
||
|
});
|
||
|
});
|
||
|
|
||
|
it('should work for $event from EventEmitter', () => {
|
||
|
expectQuickInfo({
|
||
|
templateOverride: `<div string-model (modelChange)="myClick($e¦vent)"></div>`,
|
||
|
expectedSpanText: '$event',
|
||
|
expectedDisplayString: '(parameter) $event: string'
|
||
|
});
|
||
|
});
|
||
|
|
||
|
it('should work for $event from native element', () => {
|
||
|
expectQuickInfo({
|
||
|
templateOverride: `<div (click)="myClick($e¦vent)"></div>`,
|
||
|
expectedSpanText: '$event',
|
||
|
expectedDisplayString: '(parameter) $event: MouseEvent'
|
||
|
});
|
||
|
});
|
||
|
});
|
||
|
});
|
||
|
|
||
|
describe('references', () => {
|
||
|
it('should work for element reference declarations', () => {
|
||
|
const {documentation} = expectQuickInfo({
|
||
|
templateOverride: `<div #¦chart></div>`,
|
||
|
expectedSpanText: '#chart',
|
||
|
expectedDisplayString: '(reference) chart: HTMLDivElement'
|
||
|
});
|
||
|
expect(toText(documentation))
|
||
|
.toEqual(
|
||
|
'Provides special properties (beyond the regular HTMLElement ' +
|
||
|
'interface it also has available to it by inheritance) for manipulating <div> elements.');
|
||
|
});
|
||
|
|
||
|
it('should work for ref- syntax', () => {
|
||
|
expectQuickInfo({
|
||
|
templateOverride: `<div ref-ch¦art></div>`,
|
||
|
expectedSpanText: 'ref-chart',
|
||
|
expectedDisplayString: '(reference) chart: HTMLDivElement'
|
||
|
});
|
||
|
expectQuickInfo({
|
||
|
templateOverride: `<div data-ref-ch¦art></div>`,
|
||
|
expectedSpanText: 'data-ref-chart',
|
||
|
expectedDisplayString: '(reference) chart: HTMLDivElement'
|
||
|
});
|
||
|
});
|
||
|
});
|
||
|
|
||
|
describe('variables', () => {
|
||
|
it('should work for array members', () => {
|
||
|
const {documentation} = expectQuickInfo({
|
||
|
templateOverride: `<div *ngFor="let hero of heroes">{{her¦o}}</div>`,
|
||
|
expectedSpanText: 'hero',
|
||
|
expectedDisplayString: '(variable) hero: Hero'
|
||
|
});
|
||
|
expect(toText(documentation)).toEqual('The most heroic being.');
|
||
|
});
|
||
|
|
||
|
it('should work for ReadonlyArray members (#36191)', () => {
|
||
|
expectQuickInfo({
|
||
|
templateOverride: `<div *ngFor="let hero of readonlyHeroes">{{her¦o}}</div>`,
|
||
|
expectedSpanText: 'hero',
|
||
|
expectedDisplayString: '(variable) hero: Readonly<Hero>'
|
||
|
});
|
||
|
});
|
||
|
|
||
|
it('should work for const array members (#36191)', () => {
|
||
|
expectQuickInfo({
|
||
|
templateOverride: `<div *ngFor="let name of constNames">{{na¦me}}</div>`,
|
||
|
expectedSpanText: 'name',
|
||
|
expectedDisplayString: '(variable) name: { readonly name: "name"; }'
|
||
|
});
|
||
|
});
|
||
|
});
|
||
|
|
||
|
describe('pipes', () => {
|
||
|
it('should work for pipes', () => {
|
||
|
const templateOverride = `<p>The hero's birthday is {{birthday | da¦te: "MM/dd/yy"}}</p>`;
|
||
|
expectQuickInfo({
|
||
|
templateOverride,
|
||
|
expectedSpanText: 'date',
|
||
|
expectedDisplayString:
|
||
|
'(pipe) DatePipe.transform(value: string | number | Date, format?: string | undefined, timezone?: ' +
|
||
|
'string | undefined, locale?: string | undefined): string | null (+2 overloads)'
|
||
|
});
|
||
|
});
|
||
|
});
|
||
|
|
||
|
describe('expressions', () => {
|
||
|
it('should find members in a text interpolation', () => {
|
||
|
expectQuickInfo({
|
||
|
templateOverride: `<div>{{ tit¦le }}</div>`,
|
||
|
expectedSpanText: 'title',
|
||
|
expectedDisplayString: '(property) AppComponent.title: string'
|
||
|
});
|
||
|
});
|
||
|
|
||
|
it('should work for accessed property reads', () => {
|
||
|
expectQuickInfo({
|
||
|
templateOverride: `<div>{{title.len¦gth}}</div>`,
|
||
|
expectedSpanText: 'length',
|
||
|
expectedDisplayString: '(property) String.length: number'
|
||
|
});
|
||
|
});
|
||
|
|
||
|
it('should find members in an attribute interpolation', () => {
|
||
|
expectQuickInfo({
|
||
|
templateOverride: `<div string-model model="{{tit¦le}}"></div>`,
|
||
|
expectedSpanText: 'title',
|
||
|
expectedDisplayString: '(property) AppComponent.title: string'
|
||
|
});
|
||
|
});
|
||
|
|
||
|
it('should find members of input binding', () => {
|
||
|
expectQuickInfo({
|
||
|
templateOverride: `<test-comp [tcName]="ti¦tle"></test-comp>`,
|
||
|
expectedSpanText: 'title',
|
||
|
expectedDisplayString: '(property) AppComponent.title: string'
|
||
|
});
|
||
|
});
|
||
|
|
||
|
it('should find input binding on text attribute', () => {
|
||
|
expectQuickInfo({
|
||
|
templateOverride: `<test-comp tcN¦ame="title"></test-comp>`,
|
||
|
expectedSpanText: 'tcName="title"',
|
||
|
expectedDisplayString: '(property) TestComponent.name: string'
|
||
|
});
|
||
|
});
|
||
|
|
||
|
it('should find members of event binding', () => {
|
||
|
expectQuickInfo({
|
||
|
templateOverride: `<test-comp (test)="ti¦tle=$event"></test-comp>`,
|
||
|
expectedSpanText: 'title',
|
||
|
expectedDisplayString: '(property) AppComponent.title: string'
|
||
|
});
|
||
|
});
|
||
|
|
||
|
it('should work for method calls', () => {
|
||
|
expectQuickInfo({
|
||
|
templateOverride: `<div (click)="setT¦itle('title')"></div>`,
|
||
|
expectedSpanText: 'setTitle',
|
||
|
expectedDisplayString: '(method) AppComponent.setTitle(newTitle: string): void'
|
||
|
});
|
||
|
});
|
||
|
|
||
|
it('should work for accessed properties in writes', () => {
|
||
|
expectQuickInfo({
|
||
|
templateOverride: `<div (click)="hero.i¦d = 2"></div>`,
|
||
|
expectedSpanText: 'id',
|
||
|
expectedDisplayString: '(property) Hero.id: number'
|
||
|
});
|
||
|
});
|
||
|
|
||
|
it('should work for method call arguments', () => {
|
||
|
expectQuickInfo({
|
||
|
templateOverride: `<div (click)="setTitle(hero.nam¦e)"></div>`,
|
||
|
expectedSpanText: 'name',
|
||
|
expectedDisplayString: '(property) Hero.name: string'
|
||
|
});
|
||
|
});
|
||
|
|
||
|
it('should find members of two-way binding', () => {
|
||
|
expectQuickInfo({
|
||
|
templateOverride: `<input [(ngModel)]="ti¦tle" />`,
|
||
|
expectedSpanText: 'title',
|
||
|
expectedDisplayString: '(property) AppComponent.title: string'
|
||
|
});
|
||
|
});
|
||
|
|
||
|
it('should find members in a structural directive', () => {
|
||
|
expectQuickInfo({
|
||
|
templateOverride: `<div *ngIf="anyV¦alue"></div>`,
|
||
|
expectedSpanText: 'anyValue',
|
||
|
expectedDisplayString: '(property) AppComponent.anyValue: any'
|
||
|
});
|
||
|
});
|
||
|
|
||
|
it('should work for members in structural directives', () => {
|
||
|
expectQuickInfo({
|
||
|
templateOverride: `<div *ngFor="let item of her¦oes; trackBy: test;"></div>`,
|
||
|
expectedSpanText: 'heroes',
|
||
|
expectedDisplayString: '(property) AppComponent.heroes: Hero[]'
|
||
|
});
|
||
|
});
|
||
|
|
||
|
it('should work for the $any() cast function', () => {
|
||
|
expectQuickInfo({
|
||
|
templateOverride: `<div>{{$an¦y(title)}}</div>`,
|
||
|
expectedSpanText: '$any',
|
||
|
expectedDisplayString: '(method) $any: any'
|
||
|
});
|
||
|
});
|
||
|
|
||
|
it('should provide documentation', () => {
|
||
|
const {position} = service.overwriteInlineTemplate(APP_COMPONENT, `<div>{{¦title}}</div>`);
|
||
|
const quickInfo = ngLS.getQuickInfoAtPosition(APP_COMPONENT, position);
|
||
|
const documentation = toText(quickInfo!.documentation);
|
||
|
expect(documentation).toBe('This is the title of the `AppComponent` Component.');
|
||
|
});
|
||
|
|
||
|
// TODO(atscott): Enable once #39065 is merged
|
||
|
xit('works with external template', () => {
|
||
|
const {position, text} = service.overwrite(TEST_TEMPLATE, '<butt¦on></button>');
|
||
|
const quickInfo = ngLS.getQuickInfoAtPosition(TEST_TEMPLATE, position);
|
||
|
expect(quickInfo).toBeTruthy();
|
||
|
const {textSpan, displayParts} = quickInfo!;
|
||
|
expect(text.substring(textSpan.start, textSpan.start + textSpan.length))
|
||
|
.toEqual('<button></button>');
|
||
|
expect(toText(displayParts)).toEqual('(element) button: HTMLButtonElement');
|
||
|
});
|
||
|
});
|
||
|
|
||
|
function expectQuickInfo(
|
||
|
{templateOverride, expectedSpanText, expectedDisplayString}:
|
||
|
{templateOverride: string, expectedSpanText: string, expectedDisplayString: string}):
|
||
|
ts.QuickInfo {
|
||
|
const {position, text} = service.overwriteInlineTemplate(APP_COMPONENT, templateOverride);
|
||
|
const quickInfo = ngLS.getQuickInfoAtPosition(APP_COMPONENT, position);
|
||
|
expect(quickInfo).toBeTruthy();
|
||
|
const {textSpan, displayParts} = quickInfo!;
|
||
|
expect(text.substring(textSpan.start, textSpan.start + textSpan.length))
|
||
|
.toEqual(expectedSpanText);
|
||
|
expect(toText(displayParts)).toEqual(expectedDisplayString);
|
||
|
return quickInfo!;
|
||
|
}
|
||
|
});
|
||
|
|
||
|
function toText(displayParts?: ts.SymbolDisplayPart[]): string {
|
||
|
return (displayParts || []).map(p => p.text).join('');
|
||
|
}
|