materialize-angular
Version:
Material UI Angular library
262 lines • 23.9 kB
JavaScript
/**
* @fileoverview added by tsickle
* Generated from: app/completed-components/html-visualizer/html-visualizer.component.ts
* @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
*/
/**
* @license
* Copyright Workylab. 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://raw.githubusercontent.com/workylab/materialize-angular/master/LICENSE
*/
import { Component, Input } from '@angular/core';
import { DomSanitizer } from '@angular/platform-browser';
import { config } from '../../config';
export class HTMLVisualizerComponent {
/**
* @param {?} sanitizer
*/
constructor(sanitizer) {
this.sanitizer = sanitizer;
this.className = HTMLVisualizerComponent.defaultProps.className;
this.html = HTMLVisualizerComponent.defaultProps.html;
this.prefix = config.components.prefix;
}
/**
* @return {?}
*/
ngOnInit() {
this.htmlContent = this.render(this.html);
}
/**
* @param {?} html
* @return {?}
*/
render(html) {
/** @type {?} */
const tags = this.splitByTags(html);
/** @type {?} */
const lines = [];
/** @type {?} */
let depth = 0;
for (let i = 0; i < tags.length; i++) {
/** @type {?} */
const currentTag = tags[i];
/** @type {?} */
const nextTag = tags[i + 1];
/** @type {?} */
const lastTag = tags[i + 2];
/** @type {?} */
const result = this.createLineLogic(currentTag, nextTag, lastTag, depth, i);
depth = result.depth;
i = result.cycle;
lines.push(result.line);
}
return lines.join('<br />');
}
/**
* @param {?} content
* @param {?} className
* @return {?}
*/
createNode(content, className) {
/** @type {?} */
let escapedContent = content.replace(/&/g, '&');
escapedContent = escapedContent.trim();
escapedContent = escapedContent.replace(/\[\[/g, '{{');
escapedContent = escapedContent.replace(/\]\]/g, '}}');
escapedContent = escapedContent.replace(/</g, '<');
escapedContent = escapedContent.replace(/>/g, '>');
escapedContent = escapedContent.replace(/(\S+)="([^']*?)"/g, (/**
* @param {?} match
* @return {?}
*/
match => {
/** @type {?} */
const value = match.match(/"([^']*?)"/g);
/** @type {?} */
const attribute = match.match(/(\S+)=/g);
/** @type {?} */
const attributeValidated = (attribute && attribute[0]) || '';
/** @type {?} */
const valueValidated = (value && value[0]) || '';
/** @type {?} */
const attributeTag = `<span class="${HTMLVisualizerComponent.ATTRIBUTE_CLASS}">${attributeValidated}</span>`;
/** @type {?} */
const valueTag = `<span class="${HTMLVisualizerComponent.VALUE_CLASS}">${valueValidated}</span>`;
return attributeTag + valueTag;
}));
escapedContent = escapedContent.replace(/(<\/|<)/g, (/**
* @param {?} match
* @return {?}
*/
match => `<span class="${HTMLVisualizerComponent.SYMBOL_CLASS}"><${match.includes('/') ? '/' : ''}</span>`));
escapedContent = escapedContent.replace(/(\/>|>)/g, (/**
* @param {?} match
* @return {?}
*/
match => `<span class="${HTMLVisualizerComponent.SYMBOL_CLASS}">${match.includes('/') ? '/' : ''}></span>`));
return `<span class="${className}">${escapedContent}</span>`;
}
/**
* @param {?} depth
* @param {?} content
* @return {?}
*/
indentNode(depth, content) {
/** @type {?} */
const indentationBase = '  ';
/** @type {?} */
const indentation = indentationBase.repeat(depth);
return `${indentation}${content}`;
}
/**
* @param {?} xml
* @return {?}
*/
splitByTags(xml) {
/** @type {?} */
const tags = xml.split(/(<\/?[^>]+>)/g);
/** @type {?} */
const cleanedTags = tags.filter((/**
* @param {?} line
* @return {?}
*/
line => line.trim() !== ''));
return cleanedTags;
}
/**
* @param {?} contentNode
* @return {?}
*/
isTag(contentNode) {
return (/<[^>!]+>/).test(contentNode);
}
/**
* @param {?} contentNode
* @return {?}
*/
isOpeningTag(contentNode) {
if (this.isTag(contentNode) && !this.isClosingTag(contentNode) && !this.isSelfClosingTag(contentNode)) {
return true;
}
return false;
}
/**
* @param {?} contentNode
* @return {?}
*/
isClosingTag(contentNode) {
return (/<\/+[^>]+>/).test(contentNode);
}
/**
* @param {?} contentNode
* @return {?}
*/
isSelfClosingTag(contentNode) {
return (/<[^>]+\/>/).test(contentNode);
}
/**
* @param {?} firstTag
* @param {?} middleTag
* @param {?} lastTag
* @param {?} depth
* @param {?} cycle
* @return {?}
*/
createLineLogic(firstTag, middleTag, lastTag, depth, cycle) {
if (this.isOpeningTag(firstTag) && this.isClosingTag(middleTag)) {
/** @type {?} */
const openNode = this.createNode(firstTag, 'tag');
/** @type {?} */
const closeNode = this.createNode(middleTag, 'tag');
return {
cycle: cycle + 1,
depth: depth,
line: this.indentNode(depth, `${openNode}${closeNode}`)
};
}
if (this.isOpeningTag(firstTag) && !this.isTag(middleTag)) {
/** @type {?} */
const openNode = this.createNode(firstTag, 'tag');
/** @type {?} */
const contentNode = this.createNode(middleTag, 'string');
/** @type {?} */
const closeNode = this.createNode(lastTag, 'tag');
return {
cycle: cycle + 2,
depth: depth,
line: this.indentNode(depth, `${openNode}${contentNode}${closeNode}`)
};
}
if (this.isSelfClosingTag(firstTag)) {
/** @type {?} */
const selfClosingNode = this.createNode(firstTag, 'tag');
return {
cycle: cycle,
depth: depth,
line: this.indentNode(depth, selfClosingNode)
};
}
if (this.isClosingTag(firstTag)) {
/** @type {?} */
const closeNode = this.createNode(firstTag, 'tag');
return {
cycle: cycle,
depth: depth - 1,
line: this.indentNode(depth - 1, closeNode)
};
}
return {
cycle: cycle,
depth: depth + 1,
line: this.indentNode(depth, this.createNode(firstTag, 'text'))
};
}
}
HTMLVisualizerComponent.ATTRIBUTE_CLASS = `${config.components.prefix}-html-visualizer-attribute`;
HTMLVisualizerComponent.VALUE_CLASS = `${config.components.prefix}-html-visualizer-value`;
HTMLVisualizerComponent.SYMBOL_CLASS = `${config.components.prefix}-html-visualizer-symbol`;
HTMLVisualizerComponent.defaultProps = {
className: '',
html: ''
};
HTMLVisualizerComponent.decorators = [
{ type: Component, args: [{
selector: `${config.components.prefix}-html-visualizer`,
template: "<div [ngClass]=\"prefix + '-html-visualizer'\">\n <div [ngClass]=\"prefix + '-html-visualizer-content'\" [innerHTML]=\"htmlContent\" ngNonBindable></div>\n</div>\n"
}] }
];
/** @nocollapse */
HTMLVisualizerComponent.ctorParameters = () => [
{ type: DomSanitizer }
];
HTMLVisualizerComponent.propDecorators = {
className: [{ type: Input }],
html: [{ type: Input }]
};
if (false) {
/** @type {?} */
HTMLVisualizerComponent.ATTRIBUTE_CLASS;
/** @type {?} */
HTMLVisualizerComponent.VALUE_CLASS;
/** @type {?} */
HTMLVisualizerComponent.SYMBOL_CLASS;
/** @type {?} */
HTMLVisualizerComponent.defaultProps;
/** @type {?} */
HTMLVisualizerComponent.prototype.className;
/** @type {?} */
HTMLVisualizerComponent.prototype.html;
/** @type {?} */
HTMLVisualizerComponent.prototype.htmlContent;
/** @type {?} */
HTMLVisualizerComponent.prototype.prefix;
/**
* @type {?}
* @private
*/
HTMLVisualizerComponent.prototype.sanitizer;
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"html-visualizer.component.js","sourceRoot":"ng://materialize-angular/","sources":["app/completed-components/html-visualizer/html-visualizer.component.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Workylab. 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://raw.githubusercontent.com/workylab/materialize-angular/master/LICENSE\n */\n\nimport { Component, Input, OnInit } from '@angular/core';\nimport { DomSanitizer, SafeHtml } from '@angular/platform-browser';\nimport { config } from '../../config';\nimport { HTMLVisualizerModel } from './html-visualizer.model';\n\n@Component({\n  selector: `${ config.components.prefix }-html-visualizer`,\n  templateUrl: './html-visualizer.component.html'\n})\nexport class HTMLVisualizerComponent implements HTMLVisualizerModel, OnInit {\n  static readonly ATTRIBUTE_CLASS = `${ config.components.prefix }-html-visualizer-attribute`;\n  static readonly VALUE_CLASS = `${ config.components.prefix }-html-visualizer-value`;\n  static readonly SYMBOL_CLASS = `${ config.components.prefix }-html-visualizer-symbol`;\n\n  static readonly defaultProps: HTMLVisualizerModel = {\n    className: '',\n    html: ''\n  };\n\n  @Input() className: string = HTMLVisualizerComponent.defaultProps.className;\n  @Input() html: string = HTMLVisualizerComponent.defaultProps.html;\n\n  public htmlContent: SafeHtml;\n  public prefix = config.components.prefix;\n\n  constructor(private sanitizer: DomSanitizer) {\n  }\n\n  ngOnInit() {\n    this.htmlContent = this.render(this.html);\n  }\n\n  render(html: string): string {\n    const tags = this.splitByTags(html);\n    const lines = [];\n\n    let depth = 0;\n\n    for (let i = 0; i < tags.length; i++) {\n      const currentTag = tags[i];\n      const nextTag = tags[i + 1];\n      const lastTag = tags[i + 2];\n\n      const result = this.createLineLogic(currentTag, nextTag, lastTag, depth, i);\n\n      depth = result.depth;\n      i = result.cycle;\n      lines.push(result.line);\n    }\n\n    return lines.join('<br />');\n  }\n\n  createNode(content: string, className: string) {\n    let escapedContent = content.replace(/&/g, '&amp;');\n\n    escapedContent = escapedContent.trim();\n    escapedContent = escapedContent.replace(/\\[\\[/g, '&#123&#123;');\n    escapedContent = escapedContent.replace(/\\]\\]/g, '&#125&#125;');\n\n    escapedContent = escapedContent.replace(/</g, '&lt;');\n    escapedContent = escapedContent.replace(/>/g, '&gt;');\n\n    escapedContent = escapedContent.replace(/(\\S+)=\"([^']*?)\"/g, match => {\n      const value = match.match(/\"([^']*?)\"/g);\n      const attribute = match.match(/(\\S+)=/g);\n      const attributeValidated = (attribute && attribute[0]) || '';\n      const valueValidated = (value && value[0]) || '';\n\n      const attributeTag = `<span class=\"${ HTMLVisualizerComponent.ATTRIBUTE_CLASS }\">${ attributeValidated }</span>`;\n      const valueTag = `<span class=\"${ HTMLVisualizerComponent.VALUE_CLASS }\">${ valueValidated }</span>`;\n\n      return attributeTag + valueTag;\n    });\n\n    escapedContent = escapedContent.replace(/(&lt;\\/|&lt;)/g, match =>\n      `<span class=\"${ HTMLVisualizerComponent.SYMBOL_CLASS }\">&lt;${ match.includes('/') ? '/' : '' }</span>`\n    );\n\n    escapedContent = escapedContent.replace(/(\\/&gt;|&gt;)/g, match =>\n      `<span class=\"${ HTMLVisualizerComponent.SYMBOL_CLASS }\">${ match.includes('/') ? '/' : '' }&gt;</span>`\n    );\n\n    return `<span class=\"${ className }\">${ escapedContent }</span>`;\n  }\n\n  indentNode(depth: number, content: string) {\n    const indentationBase = '&emsp;&emsp;';\n    const indentation = indentationBase.repeat(depth);\n\n    return `${ indentation }${ content }`;\n  }\n\n  splitByTags(xml: string): Array<string> {\n    const tags = xml.split(/(<\\/?[^>]+>)/g);\n    const cleanedTags = tags.filter(line => line.trim() !== '');\n\n    return cleanedTags;\n  }\n\n  isTag(contentNode: string): boolean {\n    return (/<[^>!]+>/).test(contentNode);\n  }\n\n  isOpeningTag(contentNode: string): boolean {\n    if (this.isTag(contentNode) && !this.isClosingTag(contentNode) && !this.isSelfClosingTag(contentNode)) {\n      return true;\n    }\n\n    return false;\n  }\n\n  isClosingTag(contentNode: string): boolean {\n    return (/<\\/+[^>]+>/).test(contentNode);\n  }\n\n  isSelfClosingTag(contentNode: string): boolean {\n    return (/<[^>]+\\/>/).test(contentNode);\n  }\n\n  createLineLogic(firstTag: string, middleTag: string, lastTag: string, depth: number, cycle: number): any {\n    if (this.isOpeningTag(firstTag) && this.isClosingTag(middleTag)) {\n      const openNode = this.createNode(firstTag, 'tag');\n      const closeNode = this.createNode(middleTag, 'tag');\n\n      return {\n        cycle: cycle + 1,\n        depth: depth,\n        line: this.indentNode(depth, `${ openNode }${ closeNode }`)\n      };\n    }\n\n    if (this.isOpeningTag(firstTag) && !this.isTag(middleTag)) {\n      const openNode = this.createNode(firstTag, 'tag');\n      const contentNode = this.createNode(middleTag, 'string');\n      const closeNode = this.createNode(lastTag, 'tag');\n\n      return {\n        cycle: cycle + 2,\n        depth: depth,\n        line: this.indentNode(depth, `${ openNode }${ contentNode }${ closeNode }`)\n      };\n    }\n\n    if (this.isSelfClosingTag(firstTag)) {\n      const selfClosingNode = this.createNode(firstTag, 'tag');\n\n      return {\n        cycle: cycle,\n        depth: depth,\n        line: this.indentNode(depth, selfClosingNode)\n      };\n    }\n\n    if (this.isClosingTag(firstTag)) {\n      const closeNode = this.createNode(firstTag, 'tag');\n\n      return {\n        cycle: cycle,\n        depth: depth - 1,\n        line: this.indentNode(depth - 1, closeNode)\n      };\n    }\n\n    return {\n      cycle: cycle,\n      depth: depth + 1,\n      line: this.indentNode(depth, this.createNode(firstTag, 'text'))\n    };\n  }\n}\n"]}