168 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			168 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| /**
 | |
|  * @license
 | |
|  * Copyright Google Inc. 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';
 | |
| 
 | |
| import {createLanguageService} from '../src/language_service';
 | |
| import {Span} from '../src/types';
 | |
| import {TypeScriptServiceHost} from '../src/typescript_host';
 | |
| 
 | |
| import {toh} from './test_data';
 | |
| 
 | |
| import {MockTypescriptHost,} from './test_utils';
 | |
| 
 | |
| describe('definitions', () => {
 | |
|   let documentRegistry = ts.createDocumentRegistry();
 | |
|   let mockHost = new MockTypescriptHost(['/app/main.ts', '/app/parsing-cases.ts'], toh);
 | |
|   let service = ts.createLanguageService(mockHost, documentRegistry);
 | |
|   let ngHost = new TypeScriptServiceHost(mockHost, service);
 | |
|   let ngService = createLanguageService(ngHost);
 | |
|   ngHost.setSite(ngService);
 | |
| 
 | |
|   it('should be able to find field in an interpolation', () => {
 | |
|     localReference(
 | |
|         ` @Component({template: '{{«name»}}'}) export class MyComponent { «∆name∆: string;» }`);
 | |
|   });
 | |
| 
 | |
|   it('should be able to find a field in a attribute reference', () => {
 | |
|     localReference(
 | |
|         ` @Component({template: '<input [(ngModel)]="«name»">'}) export class MyComponent { «∆name∆: string;» }`);
 | |
|   });
 | |
| 
 | |
|   it('should be able to find a method from a call', () => {
 | |
|     localReference(
 | |
|         ` @Component({template: '<div (click)="«myClick»();"></div>'}) export class MyComponent { «∆myClick∆() { }»}`);
 | |
|   });
 | |
| 
 | |
|   it('should be able to find a field reference in an *ngIf', () => {
 | |
|     localReference(
 | |
|         ` @Component({template: '<div *ngIf="«include»"></div>'}) export class MyComponent { «∆include∆ = true;»}`);
 | |
|   });
 | |
| 
 | |
|   it('should be able to find a reference to a component', () => {
 | |
|     reference(
 | |
|         'parsing-cases.ts',
 | |
|         ` @Component({template: '<«test-comp»></test-comp>'}) export class MyComponent { }`);
 | |
|   });
 | |
| 
 | |
|   it('should be able to find an event provider', () => {
 | |
|     reference(
 | |
|         '/app/parsing-cases.ts', 'test',
 | |
|         ` @Component({template: '<test-comp («test»)="myHandler()"></div>'}) export class MyComponent { myHandler() {} }`);
 | |
|   });
 | |
| 
 | |
|   it('should be able to find an input provider', () => {
 | |
|     reference(
 | |
|         '/app/parsing-cases.ts', 'tcName',
 | |
|         ` @Component({template: '<test-comp [«tcName»]="name"></div>'}) export class MyComponent { name = 'my name'; }`);
 | |
|   });
 | |
| 
 | |
|   it('should be able to find a pipe', () => {
 | |
|     reference(
 | |
|         'async_pipe.d.ts',
 | |
|         ` @Component({template: '<div *ngIf="input | «async»"></div>'}) export class MyComponent { input: EventEmitter; }`);
 | |
|   });
 | |
| 
 | |
|   function localReference(code: string) {
 | |
|     addCode(code, fileName => {
 | |
|       const refResult = mockHost.getReferenceMarkers(fileName);
 | |
|       for (const name in refResult.references) {
 | |
|         const references = refResult.references[name];
 | |
|         const definitions = refResult.definitions[name];
 | |
|         expect(definitions).toBeDefined();  // If this fails the test data is wrong.
 | |
|         for (const reference of references) {
 | |
|           const definition = ngService.getDefinitionAt(fileName, reference.start);
 | |
|           if (definition) {
 | |
|             definition.forEach(d => expect(d.fileName).toEqual(fileName));
 | |
|             const match = matchingSpan(definition.map(d => d.span), definitions);
 | |
|             if (!match) {
 | |
|               throw new Error(
 | |
|                   `Expected one of ${stringifySpans(definition.map(d => d.span))} to match one of ${stringifySpans(definitions)}`);
 | |
|             }
 | |
|           } else {
 | |
|             throw new Error('Expected a definition');
 | |
|           }
 | |
|         }
 | |
|       }
 | |
|     });
 | |
|   }
 | |
| 
 | |
|   function reference(referencedFile: string, code: string): void;
 | |
|   function reference(referencedFile: string, span: Span, code: string): void;
 | |
|   function reference(referencedFile: string, definition: string, code: string): void;
 | |
|   function reference(referencedFile: string, p1?: any, p2?: any): void {
 | |
|     const code: string = p2 ? p2 : p1;
 | |
|     const definition: string = p2 ? p1 : undefined;
 | |
|     let span: Span = p2 && p1.start != null ? p1 : undefined;
 | |
|     if (definition && !span) {
 | |
|       const referencedFileMarkers = mockHost.getReferenceMarkers(referencedFile);
 | |
|       expect(referencedFileMarkers).toBeDefined();  // If this fails the test data is wrong.
 | |
|       const spans = referencedFileMarkers.definitions[definition];
 | |
|       expect(spans).toBeDefined();  // If this fails the test data is wrong.
 | |
|       span = spans[0];
 | |
|     }
 | |
|     addCode(code, fileName => {
 | |
|       const refResult = mockHost.getReferenceMarkers(fileName);
 | |
|       let tests = 0;
 | |
|       for (const name in refResult.references) {
 | |
|         const references = refResult.references[name];
 | |
|         expect(reference).toBeDefined();  // If this fails the test data is wrong.
 | |
|         for (const reference of references) {
 | |
|           tests++;
 | |
|           const definition = ngService.getDefinitionAt(fileName, reference.start);
 | |
|           if (definition) {
 | |
|             definition.forEach(d => {
 | |
|               if (d.fileName.indexOf(referencedFile) < 0) {
 | |
|                 throw new Error(
 | |
|                     `Expected reference to file ${referencedFile}, received ${d.fileName}`);
 | |
|               }
 | |
|               if (span) {
 | |
|                 expect(d.span).toEqual(span);
 | |
|               }
 | |
|             });
 | |
|           } else {
 | |
|             throw new Error('Expected a definition');
 | |
|           }
 | |
|         }
 | |
|       }
 | |
|       if (!tests) {
 | |
|         throw new Error('Expected at least one reference (test data error)');
 | |
|       }
 | |
|     });
 | |
|   }
 | |
| 
 | |
|   function addCode(code: string, cb: (fileName: string, content?: string) => void) {
 | |
|     const fileName = '/app/app.component.ts';
 | |
|     const originalContent = mockHost.getFileContent(fileName);
 | |
|     const newContent = originalContent + code;
 | |
|     mockHost.override(fileName, originalContent + code);
 | |
|     try {
 | |
|       cb(fileName, newContent);
 | |
|     } finally {
 | |
|       mockHost.override(fileName, undefined);
 | |
|     }
 | |
|   }
 | |
| });
 | |
| 
 | |
| function matchingSpan(aSpans: Span[], bSpans: Span[]): Span {
 | |
|   for (const a of aSpans) {
 | |
|     for (const b of bSpans) {
 | |
|       if (a.start == b.start && a.end == b.end) {
 | |
|         return a;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| function stringifySpan(span: Span) {
 | |
|   return span ? `(${span.start}-${span.end})` : '<undefined>';
 | |
| }
 | |
| 
 | |
| function stringifySpans(spans: Span[]) {
 | |
|   return spans ? `[${spans.map(stringifySpan).join(', ')}]` : '<empty>';
 | |
| } |