mirror of
https://github.com/angular/angular-cli.git
synced 2025-05-17 19:13:34 +08:00
Updates for all angular.io links to the new angular.dev domain. Additionally, adjustment to new resources where the equivalent does not exist on the new site (e.g. Tour of Heroes tutorial)
213 lines
7.8 KiB
TypeScript
213 lines
7.8 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.dev/license
|
|
*/
|
|
|
|
import { SchematicTestRunner, UnitTestTree } from '@angular-devkit/schematics/testing';
|
|
import { Schema as ApplicationOptions } from '../application/schema';
|
|
import { Schema as WorkspaceOptions } from '../workspace/schema';
|
|
import { Schema as DirectiveOptions } from './schema';
|
|
|
|
describe('Directive Schematic', () => {
|
|
const schematicRunner = new SchematicTestRunner(
|
|
'@schematics/angular',
|
|
require.resolve('../collection.json'),
|
|
);
|
|
const defaultOptions: DirectiveOptions = {
|
|
name: 'foo',
|
|
module: undefined,
|
|
export: false,
|
|
prefix: 'app',
|
|
flat: true,
|
|
project: 'bar',
|
|
};
|
|
|
|
const workspaceOptions: WorkspaceOptions = {
|
|
name: 'workspace',
|
|
newProjectRoot: 'projects',
|
|
version: '6.0.0',
|
|
};
|
|
|
|
const appOptions: ApplicationOptions = {
|
|
name: 'bar',
|
|
inlineStyle: false,
|
|
inlineTemplate: false,
|
|
routing: false,
|
|
skipTests: false,
|
|
skipPackageJson: false,
|
|
};
|
|
let appTree: UnitTestTree;
|
|
beforeEach(async () => {
|
|
appTree = await schematicRunner.runSchematic('workspace', workspaceOptions);
|
|
appTree = await schematicRunner.runSchematic('application', appOptions, appTree);
|
|
});
|
|
|
|
it('should create respect the flat flag', async () => {
|
|
const options = { ...defaultOptions, flat: false };
|
|
|
|
const tree = await schematicRunner.runSchematic('directive', options, appTree);
|
|
const files = tree.files;
|
|
expect(files).toContain('/projects/bar/src/app/foo/foo.directive.spec.ts');
|
|
expect(files).toContain('/projects/bar/src/app/foo/foo.directive.ts');
|
|
});
|
|
|
|
it('should converts dash-cased-name to a camelCasedSelector', async () => {
|
|
const options = { ...defaultOptions, name: 'my-dir' };
|
|
|
|
const tree = await schematicRunner.runSchematic('directive', options, appTree);
|
|
const content = tree.readContent('/projects/bar/src/app/my-dir.directive.ts');
|
|
expect(content).toMatch(/selector: '\[appMyDir\]'/);
|
|
});
|
|
|
|
it('should create the right selector with a path in the name', async () => {
|
|
const options = { ...defaultOptions, name: 'sub/test' };
|
|
appTree = await schematicRunner.runSchematic('directive', options, appTree);
|
|
|
|
const content = appTree.readContent('/projects/bar/src/app/sub/test.directive.ts');
|
|
expect(content).toMatch(/selector: '\[appTest\]'/);
|
|
});
|
|
|
|
it('should use the prefix', async () => {
|
|
const options = { ...defaultOptions, prefix: 'pre' };
|
|
const tree = await schematicRunner.runSchematic('directive', options, appTree);
|
|
|
|
const content = tree.readContent('/projects/bar/src/app/foo.directive.ts');
|
|
expect(content).toMatch(/selector: '\[preFoo\]'/);
|
|
});
|
|
|
|
it('should use the default project prefix if none is passed', async () => {
|
|
const options = { ...defaultOptions, prefix: undefined };
|
|
const tree = await schematicRunner.runSchematic('directive', options, appTree);
|
|
|
|
const content = tree.readContent('/projects/bar/src/app/foo.directive.ts');
|
|
expect(content).toMatch(/selector: '\[appFoo\]'/);
|
|
});
|
|
|
|
it('should use the supplied prefix if it is ""', async () => {
|
|
const options = { ...defaultOptions, prefix: '' };
|
|
const tree = await schematicRunner.runSchematic('directive', options, appTree);
|
|
|
|
const content = tree.readContent('/projects/bar/src/app/foo.directive.ts');
|
|
expect(content).toMatch(/selector: '\[foo\]'/);
|
|
});
|
|
|
|
it('should respect skipTests flag', async () => {
|
|
const options = { ...defaultOptions, skipTests: true };
|
|
|
|
const tree = await schematicRunner.runSchematic('directive', options, appTree);
|
|
const files = tree.files;
|
|
expect(files).toContain('/projects/bar/src/app/foo.directive.ts');
|
|
expect(files).not.toContain('/projects/bar/src/app/foo.directive.spec.ts');
|
|
});
|
|
|
|
it('should create a standalone directive', async () => {
|
|
const options = { ...defaultOptions, standalone: true };
|
|
const tree = await schematicRunner.runSchematic('directive', options, appTree);
|
|
const directiveContent = tree.readContent('/projects/bar/src/app/foo.directive.ts');
|
|
expect(directiveContent).toContain('standalone: true');
|
|
expect(directiveContent).toContain('class FooDirective');
|
|
});
|
|
|
|
describe('standalone=false', () => {
|
|
const defaultNonStandaloneOptions: DirectiveOptions = {
|
|
...defaultOptions,
|
|
standalone: false,
|
|
project: 'baz',
|
|
};
|
|
|
|
beforeEach(async () => {
|
|
appTree = await schematicRunner.runSchematic(
|
|
'application',
|
|
{ ...appOptions, standalone: false, name: 'baz' },
|
|
appTree,
|
|
);
|
|
});
|
|
|
|
it('should create a directive', async () => {
|
|
const options = { ...defaultNonStandaloneOptions };
|
|
|
|
const tree = await schematicRunner.runSchematic('directive', options, appTree);
|
|
const files = tree.files;
|
|
expect(files).toContain('/projects/baz/src/app/foo.directive.spec.ts');
|
|
expect(files).toContain('/projects/baz/src/app/foo.directive.ts');
|
|
const moduleContent = tree.readContent('/projects/baz/src/app/app.module.ts');
|
|
expect(moduleContent).toMatch(/import.*Foo.*from '.\/foo.directive'/);
|
|
expect(moduleContent).toMatch(/declarations:\s*\[[^\]]+?,\r?\n\s+FooDirective\r?\n/m);
|
|
});
|
|
|
|
it('should respect the sourceRoot value', async () => {
|
|
const config = JSON.parse(appTree.readContent('/angular.json'));
|
|
config.projects.baz.sourceRoot = 'projects/baz/custom';
|
|
appTree.overwrite('/angular.json', JSON.stringify(config, null, 2));
|
|
|
|
// should fail without a module in that dir
|
|
await expectAsync(
|
|
schematicRunner.runSchematic('directive', defaultNonStandaloneOptions, appTree),
|
|
).toBeRejected();
|
|
|
|
// move the module
|
|
appTree.rename(
|
|
'/projects/baz/src/app/app.module.ts',
|
|
'/projects/baz/custom/app/app.module.ts',
|
|
);
|
|
appTree = await schematicRunner.runSchematic(
|
|
'directive',
|
|
defaultNonStandaloneOptions,
|
|
appTree,
|
|
);
|
|
|
|
expect(appTree.files).toContain('/projects/baz/custom/app/foo.directive.ts');
|
|
});
|
|
|
|
it('should find the closest module', async () => {
|
|
const options = { ...defaultNonStandaloneOptions, flat: false };
|
|
const fooModule = '/projects/baz/src/app/foo/foo.module.ts';
|
|
appTree.create(
|
|
fooModule,
|
|
`
|
|
import { NgModule } from '@angular/core';
|
|
|
|
@NgModule({
|
|
imports: [],
|
|
declarations: []
|
|
})
|
|
export class FooModule { }
|
|
`,
|
|
);
|
|
|
|
const tree = await schematicRunner.runSchematic('directive', options, appTree);
|
|
const fooModuleContent = tree.readContent(fooModule);
|
|
expect(fooModuleContent).toMatch(/import { FooDirective } from '.\/foo.directive'/);
|
|
});
|
|
|
|
it('should export the directive', async () => {
|
|
const options = { ...defaultNonStandaloneOptions, export: true };
|
|
|
|
const tree = await schematicRunner.runSchematic('directive', options, appTree);
|
|
const appModuleContent = tree.readContent('/projects/baz/src/app/app.module.ts');
|
|
expect(appModuleContent).toMatch(/exports: \[\n(\s*) {2}FooDirective\n\1\]/);
|
|
});
|
|
|
|
it('should import into a specified module', async () => {
|
|
const options = { ...defaultNonStandaloneOptions, module: 'app.module.ts' };
|
|
|
|
const tree = await schematicRunner.runSchematic('directive', options, appTree);
|
|
const appModule = tree.readContent('/projects/baz/src/app/app.module.ts');
|
|
|
|
expect(appModule).toMatch(/import { FooDirective } from '.\/foo.directive'/);
|
|
});
|
|
|
|
it('should fail if specified module does not exist', async () => {
|
|
const options = {
|
|
...defaultNonStandaloneOptions,
|
|
module: '/projects/baz/src/app/app.moduleXXX.ts',
|
|
};
|
|
|
|
await expectAsync(schematicRunner.runSchematic('directive', options, appTree)).toBeRejected();
|
|
});
|
|
});
|
|
});
|