@angular/cdk
Version:
Angular Material Component Development Kit
98 lines • 13.3 kB
JavaScript
/**
* @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
*/
(function (factory) {
if (typeof module === "object" && typeof module.exports === "object") {
var v = factory(require, exports);
if (v !== undefined) module.exports = v;
}
else if (typeof define === "function" && define.amd) {
define("@angular/cdk/schematics/utils/html-manipulation", ["require", "exports", "@angular-devkit/schematics", "@angular/cdk/schematics/utils/parse5-element", "parse5"], factory);
}
})(function (require, exports) {
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
const schematics_1 = require("@angular-devkit/schematics");
const parse5_element_1 = require("@angular/cdk/schematics/utils/parse5-element");
const parse5_1 = require("parse5");
/** Appends the given element HTML fragment to the `<head>` element of the specified HTML file. */
function appendHtmlElementToHead(host, htmlFilePath, elementHtml) {
const htmlFileBuffer = host.read(htmlFilePath);
if (!htmlFileBuffer) {
throw new schematics_1.SchematicsException(`Could not read file for path: ${htmlFilePath}`);
}
const htmlContent = htmlFileBuffer.toString();
if (htmlContent.includes(elementHtml)) {
return;
}
const headTag = getHtmlHeadTagElement(htmlContent);
if (!headTag) {
throw Error(`Could not find '<head>' element in HTML file: ${htmlFileBuffer}`);
}
// We always have access to the source code location here because the `getHeadTagElement`
// function explicitly has the `sourceCodeLocationInfo` option enabled.
const endTagOffset = headTag.sourceCodeLocation.endTag.startOffset;
const indentationOffset = parse5_element_1.getChildElementIndentation(headTag);
const insertion = `${' '.repeat(indentationOffset)}${elementHtml}`;
const recordedChange = host
.beginUpdate(htmlFilePath)
.insertRight(endTagOffset, `${insertion}\n`);
host.commitUpdate(recordedChange);
}
exports.appendHtmlElementToHead = appendHtmlElementToHead;
/** Parses the given HTML file and returns the head element if available. */
function getHtmlHeadTagElement(htmlContent) {
return getElementByTagName('head', htmlContent);
}
exports.getHtmlHeadTagElement = getHtmlHeadTagElement;
/** Adds a class to the body of the document. */
function addBodyClass(host, htmlFilePath, className) {
const htmlFileBuffer = host.read(htmlFilePath);
if (!htmlFileBuffer) {
throw new schematics_1.SchematicsException(`Could not read file for path: ${htmlFilePath}`);
}
const htmlContent = htmlFileBuffer.toString();
const body = getElementByTagName('body', htmlContent);
if (!body) {
throw Error(`Could not find <body> element in HTML file: ${htmlFileBuffer}`);
}
const classAttribute = body.attrs.find(attribute => attribute.name === 'class');
if (classAttribute) {
const hasClass = classAttribute.value.split(' ').map(part => part.trim()).includes(className);
if (!hasClass) {
const classAttributeLocation = body.sourceCodeLocation.attrs.class;
const recordedChange = host
.beginUpdate(htmlFilePath)
.insertRight(classAttributeLocation.endOffset - 1, ` ${className}`);
host.commitUpdate(recordedChange);
}
}
else {
const recordedChange = host
.beginUpdate(htmlFilePath)
.insertRight(body.sourceCodeLocation.startTag.endOffset - 1, ` class="${className}"`);
host.commitUpdate(recordedChange);
}
}
exports.addBodyClass = addBodyClass;
/** Finds an element by its tag name. */
function getElementByTagName(tagName, htmlContent) {
const document = parse5_1.parse(htmlContent, { sourceCodeLocationInfo: true });
const nodeQueue = [...document.childNodes];
while (nodeQueue.length) {
const node = nodeQueue.shift();
if (node.nodeName.toLowerCase() === tagName) {
return node;
}
else if (node.childNodes) {
nodeQueue.push(...node.childNodes);
}
}
return null;
}
});
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"html-manipulation.js","sourceRoot":"","sources":["../../../../../../../src/cdk/schematics/utils/html-manipulation.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport {SchematicsException, Tree} from '@angular-devkit/schematics';\nimport {getChildElementIndentation} from './parse5-element';\nimport {DefaultTreeDocument, DefaultTreeElement, parse as parseHtml} from 'parse5';\n\n/** Appends the given element HTML fragment to the `<head>` element of the specified HTML file. */\nexport function appendHtmlElementToHead(host: Tree, htmlFilePath: string, elementHtml: string) {\n  const htmlFileBuffer = host.read(htmlFilePath);\n\n  if (!htmlFileBuffer) {\n    throw new SchematicsException(`Could not read file for path: ${htmlFilePath}`);\n  }\n\n  const htmlContent = htmlFileBuffer.toString();\n\n  if (htmlContent.includes(elementHtml)) {\n    return;\n  }\n\n  const headTag = getHtmlHeadTagElement(htmlContent);\n\n  if (!headTag) {\n    throw Error(`Could not find '<head>' element in HTML file: ${htmlFileBuffer}`);\n  }\n\n  // We always have access to the source code location here because the `getHeadTagElement`\n  // function explicitly has the `sourceCodeLocationInfo` option enabled.\n  const endTagOffset = headTag.sourceCodeLocation!.endTag.startOffset;\n  const indentationOffset = getChildElementIndentation(headTag);\n  const insertion = `${' '.repeat(indentationOffset)}${elementHtml}`;\n\n  const recordedChange = host\n    .beginUpdate(htmlFilePath)\n    .insertRight(endTagOffset, `${insertion}\\n`);\n\n  host.commitUpdate(recordedChange);\n}\n\n/** Parses the given HTML file and returns the head element if available. */\nexport function getHtmlHeadTagElement(htmlContent: string): DefaultTreeElement | null {\n  return getElementByTagName('head', htmlContent);\n}\n\n/** Adds a class to the body of the document. */\nexport function addBodyClass(host: Tree, htmlFilePath: string, className: string): void {\n  const htmlFileBuffer = host.read(htmlFilePath);\n\n  if (!htmlFileBuffer) {\n    throw new SchematicsException(`Could not read file for path: ${htmlFilePath}`);\n  }\n\n  const htmlContent = htmlFileBuffer.toString();\n  const body = getElementByTagName('body', htmlContent);\n\n  if (!body) {\n    throw Error(`Could not find <body> element in HTML file: ${htmlFileBuffer}`);\n  }\n\n  const classAttribute = body.attrs.find(attribute => attribute.name === 'class');\n\n  if (classAttribute) {\n    const hasClass = classAttribute.value.split(' ').map(part => part.trim()).includes(className);\n\n    if (!hasClass) {\n      const classAttributeLocation = body.sourceCodeLocation!.attrs.class;\n      const recordedChange = host\n        .beginUpdate(htmlFilePath)\n        .insertRight(classAttributeLocation.endOffset - 1, ` ${className}`);\n      host.commitUpdate(recordedChange);\n    }\n  } else {\n    const recordedChange = host\n      .beginUpdate(htmlFilePath)\n      .insertRight(body.sourceCodeLocation!.startTag.endOffset - 1, ` class=\"${className}\"`);\n    host.commitUpdate(recordedChange);\n  }\n}\n\n/** Finds an element by its tag name. */\nfunction getElementByTagName(tagName: string, htmlContent: string):\n  DefaultTreeElement | null {\n  const document = parseHtml(htmlContent, {sourceCodeLocationInfo: true}) as DefaultTreeDocument;\n  const nodeQueue = [...document.childNodes];\n\n  while (nodeQueue.length) {\n    const node = nodeQueue.shift() as DefaultTreeElement;\n\n    if (node.nodeName.toLowerCase() === tagName) {\n      return node;\n    } else if (node.childNodes) {\n      nodeQueue.push(...node.childNodes);\n    }\n  }\n\n  return null;\n}\n"]}