When both are present, the inlined styles are appended to the end of the styles PR Close #22688
		
			
				
	
	
		
			197 lines
		
	
	
		
			7.5 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			197 lines
		
	
	
		
			7.5 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 {MetadataCollector, isClassMetadata} from '../../src/metadata/index';
 | |
| import {InlineResourcesMetadataTransformer, getInlineResourcesTransformFactory} from '../../src/transformers/inline_resources';
 | |
| import {MetadataCache} from '../../src/transformers/metadata_cache';
 | |
| import {MockAotContext, MockCompilerHost} from '../mocks';
 | |
| 
 | |
| describe('inline resources transformer', () => {
 | |
|   describe('decorator input', () => {
 | |
|     describe('should not touch unrecognized decorators', () => {
 | |
|       it('Not from @angular/core', () => {
 | |
|         expect(convert(`declare const Component: Function;
 | |
|           @Component({templateUrl: './thing.html'}) class Foo {}`))
 | |
|             .toContain('templateUrl');
 | |
|       });
 | |
|       it('missing @ sign', () => {
 | |
|         expect(convert(`import {Component} from '@angular/core';
 | |
|           Component({templateUrl: './thing.html'}) class Foo {}`))
 | |
|             .toContain('templateUrl');
 | |
|       });
 | |
|       it('too many arguments to @Component', () => {
 | |
|         expect(convert(`import {Component} from '@angular/core';
 | |
|           @Component(1, {templateUrl: './thing.html'}) class Foo {}`))
 | |
|             .toContain('templateUrl');
 | |
|       });
 | |
|       it('wrong argument type to @Component', () => {
 | |
|         expect(convert(`import {Component} from '@angular/core';
 | |
|           @Component([{templateUrl: './thing.html'}]) class Foo {}`))
 | |
|             .toContain('templateUrl');
 | |
|       });
 | |
|     });
 | |
| 
 | |
|     it('should replace templateUrl', () => {
 | |
|       const actual = convert(`import {Component} from '@angular/core';
 | |
|         @Component({
 | |
|           templateUrl: './thing.html',
 | |
| 	        otherProp: 3,
 | |
| 	      }) export class Foo {}`);
 | |
|       expect(actual).not.toContain('templateUrl:');
 | |
|       expect(actual.replace(/\s+/g, ' '))
 | |
|           .toContain(
 | |
|               'Foo = __decorate([ core_1.Component({ template: "Some template", otherProp: 3 }) ], Foo)');
 | |
|     });
 | |
|     it('should allow different quotes', () => {
 | |
|       const actual = convert(`import {Component} from '@angular/core';
 | |
|         @Component({"templateUrl": \`./thing.html\`}) export class Foo {}`);
 | |
|       expect(actual).not.toContain('templateUrl:');
 | |
|       expect(actual).toContain('{ template: "Some template" }');
 | |
|     });
 | |
|     it('should replace styleUrls', () => {
 | |
|       const actual = convert(`import {Component} from '@angular/core';
 | |
|         @Component({
 | |
|           styleUrls: ['./thing1.css', './thing2.css'],
 | |
|         })
 | |
|         export class Foo {}`);
 | |
|       expect(actual).not.toContain('styleUrls:');
 | |
|       expect(actual).toContain('styles: [".some_style {}", ".some_other_style {}"]');
 | |
|     });
 | |
|     it('should preserve existing styles', () => {
 | |
|       const actual = convert(`import {Component} from '@angular/core';
 | |
|         @Component({
 | |
|           styles: ['h1 { color: blue }'],
 | |
|           styleUrls: ['./thing1.css'],
 | |
|         })
 | |
|         export class Foo {}`);
 | |
|       expect(actual).not.toContain('styleUrls:');
 | |
|       expect(actual).toContain(`styles: ['h1 { color: blue }', ".some_style {}"]`);
 | |
|     });
 | |
|     it('should handle empty styleUrls', () => {
 | |
|       const actual = convert(`import {Component} from '@angular/core';
 | |
|         @Component({styleUrls: [], styles: []}) export class Foo {}`);
 | |
|       expect(actual).not.toContain('styleUrls:');
 | |
|       expect(actual).not.toContain('styles:');
 | |
|     });
 | |
|   });
 | |
|   describe('annotation input', () => {
 | |
|     it('should replace templateUrl', () => {
 | |
|       const actual = convert(`import {Component} from '@angular/core';
 | |
|       declare const NotComponent: Function;
 | |
| 
 | |
|       export class Foo {
 | |
|         static decorators: {type: Function, args?: any[]}[] = [
 | |
|           {
 | |
|             type: NotComponent,
 | |
|             args: [],
 | |
|           },{
 | |
|             type: Component,
 | |
|             args: [{
 | |
|               templateUrl: './thing.html'
 | |
|           }],
 | |
|         }];
 | |
|       }
 | |
|     `);
 | |
|       expect(actual).not.toContain('templateUrl:');
 | |
|       expect(actual.replace(/\s+/g, ' '))
 | |
|           .toMatch(
 | |
|               /Foo\.decorators = [{ .*type: core_1\.Component, args: [{ template: "Some template" }]/);
 | |
|     });
 | |
|     it('should replace styleUrls', () => {
 | |
|       const actual = convert(`import {Component} from '@angular/core';
 | |
|       declare const NotComponent: Function;
 | |
| 
 | |
|       export class Foo {
 | |
|         static decorators: {type: Function, args?: any[]}[] = [{
 | |
|           type: Component,
 | |
|           args: [{
 | |
|             styleUrls: ['./thing1.css', './thing2.css'],
 | |
|           }],
 | |
|         }];
 | |
|       }
 | |
|     `);
 | |
|       expect(actual).not.toContain('styleUrls:');
 | |
|       expect(actual.replace(/\s+/g, ' '))
 | |
|           .toMatch(
 | |
|               /Foo\.decorators = [{ .*type: core_1\.Component, args: [{ style: "Some template" }]/);
 | |
|     });
 | |
|   });
 | |
| });
 | |
| 
 | |
| describe('metadata transformer', () => {
 | |
|   it('should transform decorators', () => {
 | |
|     const source = `import {Component} from '@angular/core';
 | |
|       @Component({
 | |
|         templateUrl: './thing.html',
 | |
|         styleUrls: ['./thing1.css', './thing2.css'],
 | |
|         styles: ['h1 { color: red }'],
 | |
|       })
 | |
|       export class Foo {}
 | |
|     `;
 | |
|     const sourceFile = ts.createSourceFile(
 | |
|         'someFile.ts', source, ts.ScriptTarget.Latest, /* setParentNodes */ true);
 | |
|     const cache = new MetadataCache(
 | |
|         new MetadataCollector(), /* strict */ true,
 | |
|         [new InlineResourcesMetadataTransformer(
 | |
|             {loadResource, resourceNameToFileName: (u: string) => u})]);
 | |
|     const metadata = cache.getMetadata(sourceFile);
 | |
|     expect(metadata).toBeDefined('Expected metadata from test source file');
 | |
|     if (metadata) {
 | |
|       const classData = metadata.metadata['Foo'];
 | |
|       expect(classData && isClassMetadata(classData))
 | |
|           .toBeDefined(`Expected metadata to contain data for Foo`);
 | |
|       if (classData && isClassMetadata(classData)) {
 | |
|         expect(JSON.stringify(classData)).not.toContain('templateUrl');
 | |
|         expect(JSON.stringify(classData)).toContain('"template":"Some template"');
 | |
|         expect(JSON.stringify(classData)).not.toContain('styleUrls');
 | |
|         expect(JSON.stringify(classData))
 | |
|             .toContain('"styles":["h1 { color: red }",".some_style {}",".some_other_style {}"]');
 | |
|       }
 | |
|     }
 | |
|   });
 | |
| });
 | |
| 
 | |
| function loadResource(path: string): Promise<string>|string {
 | |
|   if (path === './thing.html') return 'Some template';
 | |
|   if (path === './thing1.css') return '.some_style {}';
 | |
|   if (path === './thing2.css') return '.some_other_style {}';
 | |
|   throw new Error('No fake data for path ' + path);
 | |
| }
 | |
| 
 | |
| function convert(source: string) {
 | |
|   const baseFileName = 'someFile';
 | |
|   const moduleName = '/' + baseFileName;
 | |
|   const fileName = moduleName + '.ts';
 | |
|   const context = new MockAotContext('/', {[baseFileName + '.ts']: source});
 | |
|   const host = new MockCompilerHost(context);
 | |
| 
 | |
|   const sourceFile =
 | |
|       ts.createSourceFile(fileName, source, ts.ScriptTarget.Latest, /* setParentNodes */ true);
 | |
|   const program = ts.createProgram(
 | |
|       [fileName], {
 | |
|         module: ts.ModuleKind.CommonJS,
 | |
|         target: ts.ScriptTarget.ES2017,
 | |
|       },
 | |
|       host);
 | |
|   const moduleSourceFile = program.getSourceFile(fileName);
 | |
|   const transformers: ts.CustomTransformers = {
 | |
|     before: [getInlineResourcesTransformFactory(
 | |
|         program, {loadResource, resourceNameToFileName: (u: string) => u})]
 | |
|   };
 | |
|   let result = '';
 | |
|   const emitResult = program.emit(
 | |
|       moduleSourceFile, (emittedFileName, data, writeByteOrderMark, onError, sourceFiles) => {
 | |
|         if (fileName.startsWith(moduleName)) {
 | |
|           result = data;
 | |
|         }
 | |
|       }, undefined, undefined, transformers);
 | |
|   return result;
 | |
| }
 |