igniteui-angular-sovn
Version:
Ignite UI for Angular is a dependency-free Angular toolkit for building modern web apps
238 lines (207 loc) • 10.9 kB
text/typescript
import { workspaces } from '@angular-devkit/core';
import { SchematicContext, Rule, Tree } from '@angular-devkit/schematics';
import { Options } from '../interfaces/options';
import { createHost, ProjectType } from './util';
export enum PackageTarget {
DEV = 'devDependencies',
REGULAR = 'dependencies',
NONE = 'none'
}
export interface PackageEntry {
name: string;
target: PackageTarget;
}
const schematicsPackage = '@igniteui/angular-schematics';
/**
* Dependencies are explicitly defined here, so we avoid adding
* unnecessary packages to the consuming project's deps
*/
export const DEPENDENCIES_MAP: PackageEntry[] = [
// dependencies
{ name: 'hammerjs', target: PackageTarget.REGULAR },
{ name: 'fflate', target: PackageTarget.REGULAR },
{ name: 'tslib', target: PackageTarget.NONE },
{ name: '@types/hammerjs', target: PackageTarget.DEV },
{ name: 'igniteui-trial-watermark', target: PackageTarget.NONE },
{ name: 'lodash-es', target: PackageTarget.NONE },
{ name: 'uuid', target: PackageTarget.NONE },
{ name: '@igniteui/material-icons-extended', target: PackageTarget.REGULAR },
// peerDependencies
{ name: '@angular/forms', target: PackageTarget.NONE },
{ name: '@angular/common', target: PackageTarget.NONE },
{ name: '@angular/core', target: PackageTarget.NONE },
{ name: '@angular/animations', target: PackageTarget.NONE },
{ name: 'igniteui-theming', target: PackageTarget.NONE },
// igxDevDependencies
{ name: '@igniteui/angular-schematics', target: PackageTarget.DEV }
];
export const getWorkspacePath = (host: Tree): string => {
const targetFiles = ['/angular.json', '/.angular.json'];
return targetFiles.filter(p => host.exists(p))[0];
};
const logIncludingDependency = (context: SchematicContext, pkg: string, version: string): void =>
context.logger.info(`Including ${pkg} - Version: ${version}`);
const getTargetedProjectOptions = (project: workspaces.ProjectDefinition, target: string, context: SchematicContext) => {
if (project.targets &&
project.targets[target] &&
project.targets[target].options) {
return project.targets[target].options;
}
const projectTarget = project.targets?.get(target);
if (projectTarget) {
return projectTarget.options;
}
context.logger.warn(`Could not find matching ${target} options ` +
`in Angular workspace ${project.sourceRoot}. ` +
`It could require you to manually add and update the ${target} section.`);
};
export const getConfigFile =
(project: workspaces.ProjectDefinition, option: string, context: SchematicContext, configSection = 'build'): string => {
const options = getTargetedProjectOptions(project, configSection, context);
if (!options) {
context.logger.warn(`Could not find matching ${configSection} options in Angular workspace. ` +
`It could require you to manually add and update the ${configSection} options.`);
}
if (options) {
if (!options[option]) {
context.logger.warn(`Could not find a matching ${option} property under ${configSection} options in Angular workspace. ` +
`Some updates may not execute correctly.`);
} else {
return options[option];
}
}
};
export const overwriteJsonFile = (tree: Tree, targetFile: string, data: any) =>
tree.overwrite(targetFile, JSON.stringify(data, null, 2) + '\n');
// eslint-disable-next-line @typescript-eslint/no-unused-vars
export const logSuccess = (options: Options): Rule => (tree: Tree, context: SchematicContext) => {
context.logger.info('');
context.logger.warn('Ignite UI for Angular installed');
context.logger.info('Learn more: https://www.infragistics.com/products/ignite-ui-angular');
context.logger.info('');
};
// eslint-disable-next-line @typescript-eslint/no-unused-vars
export const addDependencies = (options: Options) => async (tree: Tree, context: SchematicContext): Promise<void> => {
const pkgJson = require('../../package.json');
const workspaceHost = createHost(tree);
const { workspace } = await workspaces.readWorkspace(tree.root.path, workspaceHost);
await includeDependencies(workspaceHost, workspace, pkgJson, context, tree);
await includeStylePreprocessorOptions(workspaceHost, workspace, context, tree);
addPackageToPkgJson(tree, schematicsPackage, pkgJson.igxDevDependencies[schematicsPackage], PackageTarget.DEV);
};
/** Checks whether a property exists in the angular workspace. */
export const propertyExistsInWorkspace = (targetProp: string, workspace: workspaces.WorkspaceDefinition): boolean => {
const foundProp = getPropertyFromWorkspace(targetProp, workspace);
return foundProp !== null && foundProp.key === targetProp;
};
/** Recursively search for the first property that matches targetProp within a json file. */
export const getPropertyFromWorkspace = (targetProp: string, workspace: any, curKey = ''): { key: string; value: any } => {
if (workspace.hasOwnProperty(targetProp)) {
return { key: targetProp, value: workspace[targetProp] };
}
const workspaceKeys = Object.keys(workspace);
for (const key of workspaceKeys) {
// If the target property is an array, return its key and its contents.
if (Array.isArray(workspace[key])) {
return {
key: curKey,
value: workspace[key]
};
} else if (workspace[key] instanceof Object) {
// If the target property is an object, go one level in.
if (workspace.hasOwnProperty(key)) {
const newValue = getPropertyFromWorkspace(targetProp, workspace[key], key);
if (newValue) {
return newValue;
}
}
}
}
return null;
};
const addHammerToConfig =
async (project: workspaces.ProjectDefinition, tree: Tree, config: string, context: SchematicContext): Promise<void> => {
const projectOptions = getTargetedProjectOptions(project, config, context);
const tsPath = getConfigFile(project, 'main', context, config);
const hammerImport = 'import \'hammerjs\';\n';
const tsContent = tree.read(tsPath)?.toString();
// if there are no elements in the architect[config]options.scripts array that contain hammerjs
// and the "main" file does not contain an import with hammerjs
if (!projectOptions?.scripts?.some(el => el.includes('hammerjs')) && !tsContent?.includes(hammerImport)) {
const hammerjsFilePath = './node_modules/hammerjs/hammer.min.js';
if (projectOptions?.scripts) {
projectOptions.scripts.push(hammerjsFilePath);
return;
}
context.logger.warn(`Could not find a matching scripts array property under ${config} options. ` +
`It could require you to manually update it to 'scripts': [ ${hammerjsFilePath}] `);
}
};
export const includeStylePreprocessorOptions = async (workspaceHost: workspaces.WorkspaceHost, workspace: workspaces.WorkspaceDefinition, context: SchematicContext, tree: Tree): Promise<void> => {
await Promise.all(Array.from(workspace.projects.values()).map(async (project: workspaces.ProjectDefinition) => {
if (project.extensions['projectType'] === ProjectType.Library) return;
await addStylePreprocessorOptions(project, tree, "build", context);
await addStylePreprocessorOptions(project, tree, "server", context);
await addStylePreprocessorOptions(project, tree, "test", context);
}));
await workspaces.writeWorkspace(workspace, workspaceHost);
};
const addStylePreprocessorOptions =
async (project: workspaces.ProjectDefinition, tree: Tree, config: string, context: SchematicContext): Promise<void> => {
const projectOptions = getTargetedProjectOptions(project, config, context);
const warn = `Could not find a matching stylePreprocessorOptions includePaths array property under ${config} options. ` +
`It could require you to manually update it to "stylePreprocessorOptions": { "includePaths": ["node_modules"] }`;
if (!projectOptions) {
context.logger.warn(warn);
return;
}
// if there are no elements in the architect[config]options.stylePreprocessorOptions.includePaths that contain node_modules
const stylePrepropPath = 'node_modules';
if (!projectOptions?.stylePreprocessorOptions?.includePaths?.some(el => el.includes(stylePrepropPath))) {
if (projectOptions?.stylePreprocessorOptions?.includePaths) {
projectOptions?.stylePreprocessorOptions?.includePaths.push(stylePrepropPath);
} else if (!projectOptions?.stylePreprocessorOptions) {
projectOptions["stylePreprocessorOptions"] = { includePaths: [stylePrepropPath]};
} else {
context.logger.warn(warn);
}
}
};
const includeDependencies = async (workspaceHost: workspaces.WorkspaceHost, workspace: workspaces.WorkspaceDefinition, pkgJson: any, context: SchematicContext, tree: Tree): Promise<void> => {
for (const pkg of Object.keys(pkgJson.dependencies)) {
const version = pkgJson.dependencies[pkg];
const entry = DEPENDENCIES_MAP.find(e => e.name === pkg);
if (!entry || entry.target === PackageTarget.NONE) {
continue;
}
logIncludingDependency(context, pkg, version);
addPackageToPkgJson(tree, pkg, version, entry.target);
if (pkg === 'hammerjs') {
await Promise.all(Array.from(workspace.projects.values()).map(async (project) => {
await addHammerToConfig(project, tree, 'build', context);
await addHammerToConfig(project, tree, 'test', context);
}));
}
}
await workspaces.writeWorkspace(workspace, workspaceHost);
};
export const addPackageToPkgJson = (tree: Tree, pkg: string, version: string, target: string): boolean => {
const targetFile = 'package.json';
if (tree.exists(targetFile)) {
const sourceText = tree.read(targetFile).toString();
const json = JSON.parse(sourceText);
if (!json[target]) {
json[target] = {};
}
if (!json.dependencies[pkg]) {
json[target][pkg] = version;
json[target] =
Object.keys(json[target])
.sort()
.reduce((result, key) => (result[key] = json[target][key]) && result, {});
tree.overwrite(targetFile, JSON.stringify(json, null, 2) + '\n');
}
return true;
}
return false;
};