mirror of
https://github.com/angular/angular-cli.git
synced 2025-05-18 03:23:57 +08:00
With the APF changes in version 14, the dts entrypoint file has been changed to `index.d.ts`, this results in TypeScript being able to resolve the type definition file without the need of the additional path to the flat module file dts.
196 lines
5.6 KiB
TypeScript
196 lines
5.6 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 { join, normalize } from '@angular-devkit/core';
|
|
import {
|
|
Rule,
|
|
SchematicContext,
|
|
Tree,
|
|
apply,
|
|
applyTemplates,
|
|
chain,
|
|
mergeWith,
|
|
move,
|
|
noop,
|
|
schematic,
|
|
strings,
|
|
url,
|
|
} from '@angular-devkit/schematics';
|
|
import { NodePackageInstallTask } from '@angular-devkit/schematics/tasks';
|
|
import { NodeDependencyType, addPackageJsonDependency } from '../utility/dependencies';
|
|
import { JSONFile } from '../utility/json-file';
|
|
import { latestVersions } from '../utility/latest-versions';
|
|
import { relativePathToWorkspaceRoot } from '../utility/paths';
|
|
import { getWorkspace, updateWorkspace } from '../utility/workspace';
|
|
import { Builders, ProjectType } from '../utility/workspace-models';
|
|
import { Schema as LibraryOptions } from './schema';
|
|
|
|
function updateTsConfig(packageName: string, ...paths: string[]) {
|
|
return (host: Tree) => {
|
|
if (!host.exists('tsconfig.json')) {
|
|
return host;
|
|
}
|
|
|
|
const file = new JSONFile(host, 'tsconfig.json');
|
|
const jsonPath = ['compilerOptions', 'paths', packageName];
|
|
const value = file.get(jsonPath);
|
|
file.modify(jsonPath, Array.isArray(value) ? [...value, ...paths] : paths);
|
|
};
|
|
}
|
|
|
|
function addDependenciesToPackageJson() {
|
|
return (host: Tree) => {
|
|
[
|
|
{
|
|
type: NodeDependencyType.Dev,
|
|
name: '@angular/compiler-cli',
|
|
version: latestVersions.Angular,
|
|
},
|
|
{
|
|
type: NodeDependencyType.Dev,
|
|
name: '@angular-devkit/build-angular',
|
|
version: latestVersions.DevkitBuildAngular,
|
|
},
|
|
{
|
|
type: NodeDependencyType.Dev,
|
|
name: 'ng-packagr',
|
|
version: latestVersions['ng-packagr'],
|
|
},
|
|
{
|
|
type: NodeDependencyType.Default,
|
|
name: 'tslib',
|
|
version: latestVersions['tslib'],
|
|
},
|
|
{
|
|
type: NodeDependencyType.Dev,
|
|
name: 'typescript',
|
|
version: latestVersions['typescript'],
|
|
},
|
|
].forEach((dependency) => addPackageJsonDependency(host, dependency));
|
|
|
|
return host;
|
|
};
|
|
}
|
|
|
|
function addLibToWorkspaceFile(
|
|
options: LibraryOptions,
|
|
projectRoot: string,
|
|
projectName: string,
|
|
): Rule {
|
|
return updateWorkspace((workspace) => {
|
|
workspace.projects.add({
|
|
name: projectName,
|
|
root: projectRoot,
|
|
sourceRoot: `${projectRoot}/src`,
|
|
projectType: ProjectType.Library,
|
|
prefix: options.prefix,
|
|
targets: {
|
|
build: {
|
|
builder: Builders.NgPackagr,
|
|
defaultConfiguration: 'production',
|
|
options: {
|
|
project: `${projectRoot}/ng-package.json`,
|
|
},
|
|
configurations: {
|
|
production: {
|
|
tsConfig: `${projectRoot}/tsconfig.lib.prod.json`,
|
|
},
|
|
development: {
|
|
tsConfig: `${projectRoot}/tsconfig.lib.json`,
|
|
},
|
|
},
|
|
},
|
|
test: {
|
|
builder: Builders.Karma,
|
|
options: {
|
|
main: `${projectRoot}/src/test.ts`,
|
|
tsConfig: `${projectRoot}/tsconfig.spec.json`,
|
|
karmaConfig: `${projectRoot}/karma.conf.js`,
|
|
},
|
|
},
|
|
},
|
|
});
|
|
});
|
|
}
|
|
|
|
export default function (options: LibraryOptions): Rule {
|
|
return async (host: Tree) => {
|
|
const prefix = options.prefix;
|
|
|
|
// If scoped project (i.e. "@foo/bar"), convert projectDir to "foo/bar".
|
|
const packageName = options.name;
|
|
if (/^@.*\/.*/.test(options.name)) {
|
|
const [, name] = options.name.split('/');
|
|
options.name = name;
|
|
}
|
|
|
|
const workspace = await getWorkspace(host);
|
|
const newProjectRoot = (workspace.extensions.newProjectRoot as string | undefined) || '';
|
|
|
|
let folderName = packageName.startsWith('@') ? packageName.slice(1) : packageName;
|
|
if (/[A-Z]/.test(folderName)) {
|
|
folderName = strings.dasherize(folderName);
|
|
}
|
|
|
|
const projectRoot = join(normalize(newProjectRoot), folderName);
|
|
const distRoot = `dist/${folderName}`;
|
|
const sourceDir = `${projectRoot}/src/lib`;
|
|
|
|
const templateSource = apply(url('./files'), [
|
|
applyTemplates({
|
|
...strings,
|
|
...options,
|
|
packageName,
|
|
projectRoot,
|
|
distRoot,
|
|
relativePathToWorkspaceRoot: relativePathToWorkspaceRoot(projectRoot),
|
|
prefix,
|
|
angularLatestVersion: latestVersions.Angular.replace(/~|\^/, ''),
|
|
tsLibLatestVersion: latestVersions['tslib'].replace(/~|\^/, ''),
|
|
folderName,
|
|
}),
|
|
move(projectRoot),
|
|
]);
|
|
|
|
return chain([
|
|
mergeWith(templateSource),
|
|
addLibToWorkspaceFile(options, projectRoot, packageName),
|
|
options.skipPackageJson ? noop() : addDependenciesToPackageJson(),
|
|
options.skipTsConfig ? noop() : updateTsConfig(packageName, distRoot),
|
|
schematic('module', {
|
|
name: options.name,
|
|
commonModule: false,
|
|
flat: true,
|
|
path: sourceDir,
|
|
project: packageName,
|
|
}),
|
|
schematic('component', {
|
|
name: options.name,
|
|
selector: `${prefix}-${options.name}`,
|
|
inlineStyle: true,
|
|
inlineTemplate: true,
|
|
flat: true,
|
|
path: sourceDir,
|
|
export: true,
|
|
project: packageName,
|
|
}),
|
|
schematic('service', {
|
|
name: options.name,
|
|
flat: true,
|
|
path: sourceDir,
|
|
project: packageName,
|
|
}),
|
|
(_tree: Tree, context: SchematicContext) => {
|
|
if (!options.skipPackageJson && !options.skipInstall) {
|
|
context.addTask(new NodePackageInstallTask());
|
|
}
|
|
},
|
|
]);
|
|
};
|
|
}
|