Commit 9c1a2a4cb1b322fe6146e66fa9e1170dfc2ebcf1

Authored by Igor Kulikov
1 parent 327607e8

Add markdown/HTML widget

... ... @@ -167,6 +167,24 @@
167 167 "dataKeySettingsSchema": "{}\n",
168 168 "defaultConfig": "{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"entityAliasId\":null,\"filterId\":null,\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Random\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.7036904308224163,\"funcBody\":\"var value = prevValue + Math.random() * 100 - 50;\\nvar multiplier = Math.pow(10, 2 || 0);\\nvar value = Math.round(value * multiplier) / multiplier;\\nif (value < -1000) {\\n\\tvalue = -1000;\\n} else if (value > 1000) {\\n\\tvalue = 1000;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"qrCodeTextPattern\":\"${entityName}\",\"useQrCodeTextFunction\":false,\"qrCodeTextFunction\":\"return data['entityName'];\"},\"title\":\"QR Code\"}"
169 169 }
  170 + },
  171 + {
  172 + "alias": "markdown_card",
  173 + "name": "Markdown Card",
  174 + "image": "",
  175 + "description": "Renders markdown/HTML using configurable pattern or function with applied attributes or timeseries values.",
  176 + "descriptor": {
  177 + "type": "latest",
  178 + "sizeX": 5,
  179 + "sizeY": 3.5,
  180 + "resources": [],
  181 + "templateHtml": "<tb-markdown-widget \n [ctx]=\"ctx\">\n</tb-markdown-widget>",
  182 + "templateCss": "#container {\n overflow: auto;\n}",
  183 + "controllerScript": "self.onInit = function() {\n}\n\nself.onDataUpdated = function() {\n self.ctx.$scope.markdownWidget.onDataUpdated();\n}\n\nself.actionSources = function() {\n return {\n 'elementClick': {\n name: 'widget-action.element-click',\n multiple: true\n }\n };\n}\n\nself.onDestroy = function() {\n}\n\n",
  184 + "settingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"Markdown card\",\n \"properties\": {\n \"markdownTextPattern\": {\n \"title\": \"Markdown pattern (markdown with variables, for ex. '${entityName} or ${keyName} - some text.')\",\n \"type\": \"string\",\n \"default\": \"# Markdown card \\n - **Current entity**: **${entityName}**. \\n - **Current value**: **${Random}**.\"\n },\n \"markdownCss\": {\n \"title\": \"Markdown CSS\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"useMarkdownTextFunction\": {\n \"title\": \"Use markdown text function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"markdownTextFunction\": {\n \"title\": \"Markdown text function: f(data)\",\n \"type\": \"string\",\n \"default\": \"return '# Some title\\\\n - Entity name: ' + data[0]['entityName'];\"\n }\n },\n \"required\": []\n },\n \"form\": [\n {\n \"key\": \"markdownTextPattern\",\n \"type\": \"markdown\"\n },\n {\n \"key\": \"markdownCss\",\n \"type\": \"css\"\n },\n \"useMarkdownTextFunction\",\n {\n \"key\": \"markdownTextFunction\",\n \"type\": \"javascript\"\n }\n ]\n}\n",
  185 + "dataKeySettingsSchema": "{}\n",
  186 + "defaultConfig": "{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Random\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.15479322438769105,\"funcBody\":\"var value = prevValue + Math.random() * 100 - 50;\\nvar multiplier = Math.pow(10, 2 || 0);\\nvar value = Math.round(value * multiplier) / multiplier;\\nif (value < -1000) {\\n\\tvalue = -1000;\\n} else if (value > 1000) {\\n\\tvalue = 1000;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"16px\",\"settings\":{\"markdownTextPattern\":\"### Markdown card\\n - **Current entity**: ${entityName}.\\n - **Current value**: ${Random}.\",\"markdownTextFunction\":\"return '# Some title\\\\n - Entity name: ' + data[0]['entityName'];\"},\"title\":\"Markdown Card\",\"showTitleIcon\":false,\"iconColor\":\"rgba(0, 0, 0, 0.87)\",\"iconSize\":\"24px\",\"titleTooltip\":\"\",\"dropShadow\":true,\"enableFullscreen\":true,\"widgetStyle\":{},\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"showLegend\":false}"
  187 + }
170 188 }
171 189 ]
172 190 }
... ...
... ... @@ -78,7 +78,8 @@
78 78 "node_modules/leaflet/dist/leaflet.css",
79 79 "src/app/modules/home/components/widget/lib/maps/markers.scss",
80 80 "node_modules/leaflet.markercluster/dist/MarkerCluster.css",
81   - "node_modules/leaflet.markercluster/dist/MarkerCluster.Default.css"
  81 + "node_modules/leaflet.markercluster/dist/MarkerCluster.Default.css",
  82 + "node_modules/prismjs/themes/prism.css"
82 83 ],
83 84 "stylePreprocessorOptions": {
84 85 "includePaths": [
... ... @@ -88,7 +89,11 @@
88 89 "scripts": [
89 90 "node_modules/tinycolor2/dist/tinycolor-min.js",
90 91 "node_modules/split.js/dist/split.min.js",
91   - "node_modules/systemjs/dist/system.js"
  92 + "node_modules/systemjs/dist/system.js",
  93 + "node_modules/marked/lib/marked.js",
  94 + "node_modules/prismjs/prism.js",
  95 + "node_modules/prismjs/components/prism-bash.min.js",
  96 + "node_modules/prismjs/components/prism-json.min.js"
92 97 ],
93 98 "customWebpackConfig": {
94 99 "path": "./extra-webpack.config.js"
... ...
... ... @@ -72,6 +72,7 @@
72 72 "ngx-drag-drop": "^2.0.0",
73 73 "ngx-flowchart": "git://github.com/thingsboard/ngx-flowchart.git#master",
74 74 "ngx-hm-carousel": "^2.0.0-rc.1",
  75 + "ngx-markdown": "^10.1.1",
75 76 "ngx-sharebuttons": "^8.0.5",
76 77 "ngx-translate-messageformat-compiler": "^4.9.0",
77 78 "objectpath": "^2.0.0",
... ...
... ... @@ -16,7 +16,7 @@
16 16
17 17 import { FormattedData, MapProviders, ReplaceInfo } from '@home/components/widget/lib/maps/map-models';
18 18 import {
19   - createLabelFromDatasource,
  19 + createLabelFromDatasource, deepClone,
20 20 hashCode,
21 21 isDefined,
22 22 isDefinedAndNotNull,
... ... @@ -343,6 +343,22 @@ export function parseData(input: DatasourceData[]): FormattedData[] {
343 343 });
344 344 }
345 345
  346 +export function flatData(input: FormattedData[]): FormattedData {
  347 + let result: FormattedData = {} as FormattedData;
  348 + if (input.length) {
  349 + for (const toMerge of input) {
  350 + result = {...result, ...toMerge};
  351 + }
  352 + result.entityName = input[0].entityName;
  353 + result.entityId = input[0].entityId;
  354 + result.entityType = input[0].entityType;
  355 + result.$datasource = input[0].$datasource;
  356 + result.dsIndex = input[0].dsIndex;
  357 + result.deviceType = input[0].deviceType;
  358 + }
  359 + return result;
  360 +}
  361 +
346 362 export function parseArray(input: DatasourceData[]): FormattedData[][] {
347 363 return _(input).groupBy(el => el?.datasource?.entityName)
348 364 .values().value().map((entityArray) =>
... ...
  1 +<!--
  2 +
  3 + Copyright © 2016-2021 The Thingsboard Authors
  4 +
  5 + Licensed under the Apache License, Version 2.0 (the "License");
  6 + you may not use this file except in compliance with the License.
  7 + You may obtain a copy of the License at
  8 +
  9 + http://www.apache.org/licenses/LICENSE-2.0
  10 +
  11 + Unless required by applicable law or agreed to in writing, software
  12 + distributed under the License is distributed on an "AS IS" BASIS,
  13 + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14 + See the License for the specific language governing permissions and
  15 + limitations under the License.
  16 +
  17 +-->
  18 +<markdown [data]="markdownText" class="tb-markdown-view" (click)="markdownClick($event)"></markdown>
... ...
  1 +/**
  2 + * Copyright © 2016-2021 The Thingsboard Authors
  3 + *
  4 + * Licensed under the Apache License, Version 2.0 (the "License");
  5 + * you may not use this file except in compliance with the License.
  6 + * You may obtain a copy of the License at
  7 + *
  8 + * http://www.apache.org/licenses/LICENSE-2.0
  9 + *
  10 + * Unless required by applicable law or agreed to in writing, software
  11 + * distributed under the License is distributed on an "AS IS" BASIS,
  12 + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13 + * See the License for the specific language governing permissions and
  14 + * limitations under the License.
  15 + */
  16 +
  17 +.tb-markdown-view {
  18 + display: block;
  19 + ::ng-deep {
  20 + h1 {
  21 + font-size: 32px;
  22 + padding-right: 60px;
  23 + }
  24 +
  25 + h1, h2, h3, h4, h5, h6 {
  26 + line-height: normal;
  27 + font-weight: 500;
  28 + margin-bottom: 30px;
  29 + padding-bottom: 10px;
  30 + border-bottom: 1px solid #ccc;
  31 + }
  32 +
  33 + p {
  34 + font-size: 16px;
  35 + font-weight: 400;
  36 + line-height: 1.25em;
  37 + margin: 0;
  38 + }
  39 +
  40 + p+p {
  41 + margin-top: 10px;
  42 + }
  43 +
  44 + table {
  45 + width: 100%;
  46 + border: 1px solid #ccc;
  47 + border-spacing: 0;
  48 + margin-top: 30px;
  49 + margin-bottom: 30px;
  50 + }
  51 +
  52 + thead {
  53 + background-color: #555;
  54 + color: #fff;
  55 + }
  56 +
  57 + th, td {
  58 + font-size: .85em;
  59 + padding: 8px;
  60 + margin: 0;
  61 + text-align: left;
  62 + }
  63 +
  64 + td[align=center], th[align=center] {
  65 + text-align: center;
  66 + }
  67 +
  68 + td[align=right], th[align=right] {
  69 + text-align: right;
  70 + }
  71 +
  72 + tr:nth-child(even) {
  73 + background-color: #f7f7f7;
  74 + }
  75 +
  76 + code:not([class*=language-]) {
  77 + background: #f5f5f5;
  78 + border-radius: 2px;
  79 + color: #dd4a68;
  80 + padding: 2px 4px;
  81 + }
  82 +
  83 + div.code-wrapper {
  84 + position: relative;
  85 + button.clipboard-btn {
  86 + cursor: pointer;
  87 + margin: 0;
  88 + border: 0;
  89 + outline: none;
  90 + position: absolute;
  91 + top: 5px;
  92 + right: 5px;
  93 + background: #fff;
  94 + box-shadow: 0 1px 8px 0 rgba(0,0,0,0.2), 0 3px 4px 0 rgba(0,0,0,0.14), 0 3px 3px -2px rgba(0,0,0,0.12);
  95 + border-radius: 5px;
  96 + opacity: 0;
  97 + transition: opacity .3s;
  98 + padding: 3px 6px;
  99 + line-height: 16px;
  100 + img {
  101 + width: 18px;
  102 + }
  103 + &:hover {
  104 + background: #f9f9f9;
  105 + }
  106 + &:active {
  107 + background-color: #ececec;
  108 + box-shadow: inset 1px -1px 4px 0px rgba(0,0,0,0.4);
  109 + }
  110 + }
  111 + &:hover {
  112 + button.clipboard-btn {
  113 + opacity: .85;
  114 + }
  115 + }
  116 + }
  117 +
  118 + th, td {
  119 + div.code-wrapper {
  120 + display: inline-block;
  121 + button.clipboard-btn {
  122 + top: -5px;
  123 + right: -30px;
  124 + padding: 0 3px;
  125 + }
  126 + }
  127 + }
  128 +
  129 + }
  130 +}
... ...
  1 +///
  2 +/// Copyright © 2016-2021 The Thingsboard Authors
  3 +///
  4 +/// Licensed under the Apache License, Version 2.0 (the "License");
  5 +/// you may not use this file except in compliance with the License.
  6 +/// You may obtain a copy of the License at
  7 +///
  8 +/// http://www.apache.org/licenses/LICENSE-2.0
  9 +///
  10 +/// Unless required by applicable law or agreed to in writing, software
  11 +/// distributed under the License is distributed on an "AS IS" BASIS,
  12 +/// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13 +/// See the License for the specific language governing permissions and
  14 +/// limitations under the License.
  15 +///
  16 +
  17 +import { ChangeDetectorRef, Component, ElementRef, Input, OnInit } from '@angular/core';
  18 +import { PageComponent } from '@shared/components/page.component';
  19 +import { WidgetContext } from '@home/models/widget-component.models';
  20 +import { Store } from '@ngrx/store';
  21 +import { AppState } from '@core/core.state';
  22 +import { DatasourceData } from '@shared/models/widget.models';
  23 +import { DataKeyType } from '@shared/models/telemetry/telemetry.models';
  24 +import {
  25 + fillPattern, flatData,
  26 + parseData,
  27 + parseFunction,
  28 + processPattern,
  29 + safeExecute
  30 +} from '@home/components/widget/lib/maps/common-maps-utils';
  31 +import { FormattedData } from '@home/components/widget/lib/maps/map-models';
  32 +import { hashCode, isNotEmptyStr } from '@core/utils';
  33 +import cssjs from '@core/css/css';
  34 +
  35 +interface MarkdownWidgetSettings {
  36 + markdownTextPattern: string;
  37 + useMarkdownTextFunction: boolean;
  38 + markdownTextFunction: string;
  39 + markdownCss: string;
  40 +}
  41 +
  42 +type MarkdownTextFunction = (data: FormattedData[]) => string;
  43 +
  44 +@Component({
  45 + selector: 'tb-markdown-widget ',
  46 + templateUrl: './markdown-widget.component.html',
  47 + styleUrls: ['./markdown-widget.component.scss']
  48 +})
  49 +export class MarkdownWidgetComponent extends PageComponent implements OnInit {
  50 +
  51 + settings: MarkdownWidgetSettings;
  52 + markdownTextFunction: MarkdownTextFunction;
  53 +
  54 + @Input()
  55 + ctx: WidgetContext;
  56 +
  57 + markdownText: string;
  58 +
  59 + constructor(protected store: Store<AppState>,
  60 + private elementRef: ElementRef,
  61 + private cd: ChangeDetectorRef) {
  62 + super(store);
  63 + }
  64 +
  65 + ngOnInit(): void {
  66 + this.ctx.$scope.markdownWidget = this;
  67 + this.settings = this.ctx.settings;
  68 + this.markdownTextFunction = this.settings.useMarkdownTextFunction ? parseFunction(this.settings.markdownTextFunction, ['data']) : null;
  69 +
  70 + const cssString = this.settings.markdownCss;
  71 + if (isNotEmptyStr(cssString)) {
  72 + const cssParser = new cssjs();
  73 + cssParser.testMode = false;
  74 + const namespace = 'entities-hierarchy-' + hashCode(cssString);
  75 + cssParser.cssPreviewNamespace = namespace;
  76 + cssParser.createStyleElement(namespace, cssString);
  77 + $(this.elementRef.nativeElement).addClass(namespace);
  78 + }
  79 + }
  80 +
  81 + public onDataUpdated() {
  82 + let initialData: DatasourceData[];
  83 + if (this.ctx.data?.length) {
  84 + initialData = this.ctx.data;
  85 + } else if (this.ctx.datasources?.length) {
  86 + initialData = [
  87 + {
  88 + datasource: this.ctx.datasources[0],
  89 + dataKey: {
  90 + type: DataKeyType.attribute,
  91 + name: 'empty'
  92 + },
  93 + data: []
  94 + }
  95 + ];
  96 + }
  97 + let markdownText: string;
  98 + if (initialData) {
  99 + const data = parseData(initialData);
  100 + markdownText = this.settings.useMarkdownTextFunction ?
  101 + safeExecute(this.markdownTextFunction, [data]) : this.settings.markdownTextPattern;
  102 + const allData = flatData(data);
  103 + const replaceInfo = processPattern(markdownText, allData);
  104 + markdownText = fillPattern(markdownText, replaceInfo, allData);
  105 + }
  106 + if (this.markdownText !== markdownText) {
  107 + this.markdownText = markdownText;
  108 + this.cd.detectChanges();
  109 + }
  110 + }
  111 +
  112 + markdownClick($event: MouseEvent) {
  113 + this.ctx.actionsApi.elementClick($event);
  114 + }
  115 +
  116 +}
... ...
... ... @@ -101,7 +101,7 @@ export class QrCodeWidgetComponent extends PageComponent implements OnInit, Afte
101 101 const dataSourceData = data[0];
102 102 const pattern = this.settings.useQrCodeTextFunction ?
103 103 safeExecute(this.qrCodeTextFunction, [dataSourceData]) : this.settings.qrCodeTextPattern;
104   - const replaceInfo = processPattern(pattern, data);
  104 + const replaceInfo = processPattern(pattern, dataSourceData);
105 105 qrCodeText = fillPattern(pattern, replaceInfo, dataSourceData);
106 106 }
107 107 this.updateQrCodeText(qrCodeText);
... ...
... ... @@ -40,6 +40,7 @@ import { NavigationCardWidgetComponent } from '@home/components/widget/lib/navig
40 40 import { EdgesOverviewWidgetComponent } from '@home/components/widget/lib/edges-overview-widget.component';
41 41 import { JsonInputWidgetComponent } from '@home/components/widget/lib/json-input-widget.component';
42 42 import { QrCodeWidgetComponent } from '@home/components/widget/lib/qrcode-widget.component';
  43 +import { MarkdownWidgetComponent } from '@home/components/widget/lib/markdown-widget.component';
43 44
44 45 @NgModule({
45 46 declarations:
... ... @@ -60,7 +61,8 @@ import { QrCodeWidgetComponent } from '@home/components/widget/lib/qrcode-widget
60 61 GatewayFormComponent,
61 62 NavigationCardsWidgetComponent,
62 63 NavigationCardWidgetComponent,
63   - QrCodeWidgetComponent
  64 + QrCodeWidgetComponent,
  65 + MarkdownWidgetComponent
64 66 ],
65 67 imports: [
66 68 CommonModule,
... ... @@ -83,7 +85,8 @@ import { QrCodeWidgetComponent } from '@home/components/widget/lib/qrcode-widget
83 85 GatewayFormComponent,
84 86 NavigationCardsWidgetComponent,
85 87 NavigationCardWidgetComponent,
86   - QrCodeWidgetComponent
  88 + QrCodeWidgetComponent,
  89 + MarkdownWidgetComponent
87 90 ],
88 91 providers: [
89 92 CustomDialogService,
... ...
... ... @@ -159,8 +159,8 @@ class ThingsboardAceEditor extends React.Component<ThingsboardAceEditorProps, Th
159 159 <div className='json-form-ace-editor'>
160 160 <div className='title-panel'>
161 161 <label>{this.props.mode}</label>
162   - <Button style={ styles.tidyButtonStyle }
163   - className='tidy-button' onClick={this.onTidy}>Tidy</Button>
  162 + { this.props.onTidy ? <Button style={ styles.tidyButtonStyle }
  163 + className='tidy-button' onClick={this.onTidy}>Tidy</Button> : null }
164 164 <Button style={ styles.tidyButtonStyle }
165 165 className='tidy-button' onClick={this.onToggleFull}>
166 166 {this.state.isFull ?
... ...
  1 +/*
  2 + * Copyright © 2016-2021 The Thingsboard Authors
  3 + *
  4 + * Licensed under the Apache License, Version 2.0 (the "License");
  5 + * you may not use this file except in compliance with the License.
  6 + * You may obtain a copy of the License at
  7 + *
  8 + * http://www.apache.org/licenses/LICENSE-2.0
  9 + *
  10 + * Unless required by applicable law or agreed to in writing, software
  11 + * distributed under the License is distributed on an "AS IS" BASIS,
  12 + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13 + * See the License for the specific language governing permissions and
  14 + * limitations under the License.
  15 + */
  16 +import * as React from 'react';
  17 +import ThingsboardAceEditor from './json-form-ace-editor';
  18 +import { JsonFormFieldProps, JsonFormFieldState } from '@shared/components/json-form/react/json-form.models';
  19 +
  20 +class ThingsboardMarkdown extends React.Component<JsonFormFieldProps, JsonFormFieldState> {
  21 +
  22 + constructor(props) {
  23 + super(props);
  24 + }
  25 +
  26 + render() {
  27 + return (
  28 + <ThingsboardAceEditor {...this.props} mode='markdown' {...this.state}></ThingsboardAceEditor>
  29 + );
  30 + }
  31 +}
  32 +
  33 +export default ThingsboardMarkdown;
... ...
... ... @@ -38,6 +38,7 @@ import { JsonFormData, JsonFormProps, onChangeFn, OnColorClickFn, OnIconClickFn
38 38 import _ from 'lodash';
39 39 import * as tinycolor_ from 'tinycolor2';
40 40 import { GroupInfo } from '@shared/models/widget.models';
  41 +import ThingsboardMarkdown from '@shared/components/json-form/react/json-form-markdown';
41 42
42 43 const tinycolor = tinycolor_;
43 44
... ... @@ -65,6 +66,7 @@ class ThingsboardSchemaForm extends React.Component<JsonFormProps, any> {
65 66 json: ThingsboardJson,
66 67 html: ThingsboardHtml,
67 68 css: ThingsboardCss,
  69 + markdown: ThingsboardMarkdown,
68 70 color: ThingsboardColor,
69 71 'rc-select': ThingsboardRcSelect,
70 72 fieldset: ThingsboardFieldSet,
... ... @@ -91,7 +93,7 @@ class ThingsboardSchemaForm extends React.Component<JsonFormProps, any> {
91 93 }
92 94
93 95 onIconClick(key: (string | number)[], val: string,
94   - iconSelectedFn: (icon: string) => void) {
  96 + iconSelectedFn: (icon: string) => void) {
95 97 this.props.onIconClick(key, val, iconSelectedFn);
96 98 }
97 99
... ...
  1 +///
  2 +/// Copyright © 2016-2021 The Thingsboard Authors
  3 +///
  4 +/// Licensed under the Apache License, Version 2.0 (the "License");
  5 +/// you may not use this file except in compliance with the License.
  6 +/// You may obtain a copy of the License at
  7 +///
  8 +/// http://www.apache.org/licenses/LICENSE-2.0
  9 +///
  10 +/// Unless required by applicable law or agreed to in writing, software
  11 +/// distributed under the License is distributed on an "AS IS" BASIS,
  12 +/// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13 +/// See the License for the specific language governing permissions and
  14 +/// limitations under the License.
  15 +///
  16 +
  17 +
  18 +import { MarkedOptions, MarkedRenderer } from 'ngx-markdown';
  19 +
  20 +export function markedOptionsFactory(): MarkedOptions {
  21 + const renderer = new MarkedRenderer();
  22 + const renderer2 = new MarkedRenderer();
  23 +
  24 + const copyCodeBlock = '{:copy-code}';
  25 +
  26 + let id = 1;
  27 +
  28 + renderer.code = (code: string, language: string | undefined, isEscaped: boolean) => {
  29 + if (code.endsWith(copyCodeBlock)) {
  30 + code = code.substring(0, code.length - copyCodeBlock.length);
  31 + const content = renderer2.code(code, language, isEscaped);
  32 + id++;
  33 + return wrapCopyCode(id, content, code);
  34 + } else {
  35 + return renderer2.code(code, language, isEscaped);
  36 + }
  37 + };
  38 +
  39 + renderer.tablecell = (content: string, flags: {
  40 + header: boolean;
  41 + align: 'center' | 'left' | 'right' | null;
  42 + }) => {
  43 + if (content.endsWith(copyCodeBlock)) {
  44 + content = content.substring(0, content.length - copyCodeBlock.length);
  45 + id++;
  46 + content = wrapCopyCode(id, content, content);
  47 + }
  48 + return renderer2.tablecell(content, flags);
  49 + };
  50 +
  51 + return {
  52 + renderer,
  53 + headerIds: true,
  54 + gfm: true,
  55 + breaks: false,
  56 + pedantic: false,
  57 + smartLists: true,
  58 + smartypants: false,
  59 + };
  60 +}
  61 +
  62 +function wrapCopyCode(id: number, content: string, code: string): string {
  63 + return '<div class="code-wrapper">' + content + '<span id="copyCodeId' + id + '" style="display: none;">' + code + '</span>' +
  64 + '<button id="copyCodeBtn' + id + '" onClick="markdownCopyCode(' + id + ')" ' +
  65 + 'class="clipboard-btn"><img src="https://clipboardjs.com/assets/images/clippy.svg" alt="Copy to clipboard">' +
  66 + '</button></div>';
  67 +}
  68 +
  69 +(window as any).markdownCopyCode = (id: number) => {
  70 + const text = $('#copyCodeId' + id).text();
  71 + navigator.clipboard.writeText(text).then(() => {
  72 + import('tooltipster').then(
  73 + () => {
  74 + const copyBtn = $('#copyCodeBtn' + id);
  75 + if (!copyBtn.hasClass('tooltipstered')) {
  76 + copyBtn.tooltipster(
  77 + {
  78 + content: 'Copied!',
  79 + theme: 'tooltipster-shadow',
  80 + delay: 0,
  81 + trigger: 'custom',
  82 + triggerClose: {
  83 + click: true,
  84 + tap: true,
  85 + scroll: true,
  86 + mouseleave: true
  87 + },
  88 + side: 'bottom',
  89 + distance: 12,
  90 + trackOrigin: true
  91 + }
  92 + );
  93 + }
  94 + const tooltip = copyBtn.tooltipster('instance');
  95 + tooltip.open();
  96 + }
  97 + );
  98 + });
  99 +};
... ...
... ... @@ -14,7 +14,7 @@
14 14 /// limitations under the License.
15 15 ///
16 16
17   -import { NgModule } from '@angular/core';
  17 +import { NgModule, SecurityContext } from '@angular/core';
18 18 import { CommonModule, DatePipe } from '@angular/common';
19 19 import { FooterComponent } from '@shared/components/footer.component';
20 20 import { LogoComponent } from '@shared/components/logo.component';
... ... @@ -78,6 +78,7 @@ import { DatetimePeriodComponent } from '@shared/components/time/datetime-period
78 78 import { EnumToArrayPipe } from '@shared/pipe/enum-to-array.pipe';
79 79 import { ClipboardModule } from 'ngx-clipboard';
80 80 import { ValueInputComponent } from '@shared/components/value-input.component';
  81 +import { MarkdownModule, MarkedOptions } from 'ngx-markdown';
81 82 import { FullscreenDirective } from '@shared/components/fullscreen.directive';
82 83 import { HighlightPipe } from '@shared/pipe/highlight.pipe';
83 84 import { DashboardAutocompleteComponent } from '@shared/components/dashboard-autocomplete.component';
... ... @@ -145,6 +146,7 @@ import { OtaPackageAutocompleteComponent } from '@shared/components/ota-package/
145 146 import { MAT_DATE_LOCALE } from '@angular/material/core';
146 147 import { CopyButtonComponent } from '@shared/components/button/copy-button.component';
147 148 import { TogglePasswordComponent } from '@shared/components/button/toggle-password.component';
  149 +import { markedOptionsFactory } from '@shared/components/markdown.factory';
148 150
149 151 @NgModule({
150 152 providers: [
... ... @@ -294,7 +296,15 @@ import { TogglePasswordComponent } from '@shared/components/button/toggle-passwo
294 296 NgxHmCarouselModule,
295 297 DndModule,
296 298 NgxFlowModule,
297   - NgxFlowchartModule
  299 + NgxFlowchartModule,
  300 + // ngx-markdown
  301 + MarkdownModule.forRoot({
  302 + sanitize: SecurityContext.NONE,
  303 + markedOptions: {
  304 + provide: MarkedOptions,
  305 + useFactory: markedOptionsFactory
  306 + }
  307 + })
298 308 ],
299 309 exports: [
300 310 FooterComponent,
... ... @@ -386,6 +396,7 @@ import { TogglePasswordComponent } from '@shared/components/button/toggle-passwo
386 396 NgxHmCarouselModule,
387 397 DndModule,
388 398 NgxFlowchartModule,
  399 + MarkdownModule,
389 400 ConfirmDialogComponent,
390 401 AlertDialogComponent,
391 402 TodoDialogComponent,
... ...
... ... @@ -1817,6 +1817,11 @@
1817 1817 resolved "https://registry.yarnpkg.com/@types/lodash/-/lodash-4.14.170.tgz#0d67711d4bf7f4ca5147e9091b847479b87925d6"
1818 1818 integrity sha512-bpcvu/MKHHeYX+qeEN8GE7DIravODWdACVA1ctevD8CN24RhPZIKMn9ntfAsrvLfSX3cR5RrBKAbYm9bGs0A+Q==
1819 1819
  1820 +"@types/marked@^1.1.0":
  1821 + version "1.2.2"
  1822 + resolved "https://registry.yarnpkg.com/@types/marked/-/marked-1.2.2.tgz#1f858a0e690247ecf3b2eef576f98f86e8d960d4"
  1823 + integrity sha512-wLfw1hnuuDYrFz97IzJja0pdVsC0oedtS4QsKH1/inyW9qkLQbXgMUqEQT0MVtUBx3twjWeInUfjQbhBVLECXw==
  1824 +
1820 1825 "@types/minimatch@*":
1821 1826 version "3.0.3"
1822 1827 resolved "https://registry.yarnpkg.com/@types/minimatch/-/minimatch-3.0.3.tgz#3dca0e3f33b200fc7d1139c0cd96c1268cadfd9d"
... ... @@ -4137,6 +4142,11 @@ emoji-regex@^8.0.0:
4137 4142 resolved "https://registry.yarnpkg.com/emoji-regex/-/emoji-regex-8.0.0.tgz#e818fd69ce5ccfcb404594f842963bf53164cc37"
4138 4143 integrity sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==
4139 4144
  4145 +emoji-toolkit@^6.0.1:
  4146 + version "6.6.0"
  4147 + resolved "https://registry.yarnpkg.com/emoji-toolkit/-/emoji-toolkit-6.6.0.tgz#e7287c43a96f940ec4c5428cd7100a40e57518f1"
  4148 + integrity sha512-pEu0kow2p1N8zCKnn/L6H0F3rWUBB3P3hVjr/O5yl1fK7N9jU4vO4G7EFapC5Y3XwZLUCY0FZbOPyTkH+4V2eQ==
  4149 +
4140 4150 emojis-list@^3.0.0:
4141 4151 version "3.0.0"
4142 4152 resolved "https://registry.yarnpkg.com/emojis-list/-/emojis-list-3.0.0.tgz#5570662046ad29e2e916e71aae260abdff4f6a78"
... ... @@ -6126,6 +6136,13 @@ karma@~6.3.2:
6126 6136 ua-parser-js "^0.7.23"
6127 6137 yargs "^16.1.1"
6128 6138
  6139 +katex@^0.12.0:
  6140 + version "0.12.0"
  6141 + resolved "https://registry.yarnpkg.com/katex/-/katex-0.12.0.tgz#2fb1c665dbd2b043edcf8a1f5c555f46beaa0cb9"
  6142 + integrity sha512-y+8btoc/CK70XqcHqjxiGWBOeIL8upbS0peTPXTvgrh21n1RiWWcIpSWM+4uXq+IAgNh9YYQWdc7LVDPDAEEAg==
  6143 + dependencies:
  6144 + commander "^2.19.0"
  6145 +
6129 6146 killable@^1.0.1:
6130 6147 version "1.0.1"
6131 6148 resolved "https://registry.yarnpkg.com/killable/-/killable-1.0.1.tgz#4c8ce441187a061c7474fb87ca08e2a638194892"
... ... @@ -6425,6 +6442,11 @@ map-visit@^1.0.0:
6425 6442 dependencies:
6426 6443 object-visit "^1.0.0"
6427 6444
  6445 +marked@^1.1.0:
  6446 + version "1.2.9"
  6447 + resolved "https://registry.yarnpkg.com/marked/-/marked-1.2.9.tgz#53786f8b05d4c01a2a5a76b7d1ec9943d29d72dc"
  6448 + integrity sha512-H8lIX2SvyitGX+TRdtS06m1jHMijKN/XjfH6Ooii9fvxMlh8QdqBfBDkGUpMWH2kQNrtixjzYUa3SH8ROTgRRw==
  6449 +
6428 6450 material-design-icons@^3.0.1:
6429 6451 version "3.0.1"
6430 6452 resolved "https://registry.yarnpkg.com/material-design-icons/-/material-design-icons-3.0.1.tgz#9a71c48747218ebca51e51a66da682038cdcb7bf"
... ... @@ -6877,6 +6899,18 @@ ngx-hm-carousel@^2.0.0-rc.1:
6877 6899 hammerjs "^2.0.8"
6878 6900 resize-observer-polyfill "^1.5.1"
6879 6901
  6902 +ngx-markdown@^10.1.1:
  6903 + version "10.1.1"
  6904 + resolved "https://registry.yarnpkg.com/ngx-markdown/-/ngx-markdown-10.1.1.tgz#17840c773db7ced4b18ccbf2e8cb06182e422de3"
  6905 + integrity sha512-bUVgN6asb35d5U4xM5CNfo7pSpuwqJSdTgK0PhNZzLiaiyPIK2owtLF6sWGhxTThJu+LngJPjj4MQ+AFe/s8XQ==
  6906 + dependencies:
  6907 + "@types/marked" "^1.1.0"
  6908 + emoji-toolkit "^6.0.1"
  6909 + katex "^0.12.0"
  6910 + marked "^1.1.0"
  6911 + prismjs "^1.20.0"
  6912 + tslib "^2.0.0"
  6913 +
6880 6914 ngx-sharebuttons@^8.0.5:
6881 6915 version "8.0.5"
6882 6916 resolved "https://registry.yarnpkg.com/ngx-sharebuttons/-/ngx-sharebuttons-8.0.5.tgz#49481fcb8bf9541747fd72093eca6f4777c1d371"
... ... @@ -7877,6 +7911,11 @@ pretty-bytes@^5.3.0:
7877 7911 resolved "https://registry.yarnpkg.com/pretty-bytes/-/pretty-bytes-5.6.0.tgz#356256f643804773c82f64723fe78c92c62beaeb"
7878 7912 integrity sha512-FFw039TmrBqFK8ma/7OL3sDz/VytdtJr044/QUJtH0wK9lb9jLq9tJyIxUwtQJHwar2BqtiA4iCWSwo9JLkzFg==
7879 7913
  7914 +prismjs@^1.20.0:
  7915 + version "1.24.1"
  7916 + resolved "https://registry.yarnpkg.com/prismjs/-/prismjs-1.24.1.tgz#c4d7895c4d6500289482fa8936d9cdd192684036"
  7917 + integrity sha512-mNPsedLuk90RVJioIky8ANZEwYm5w9LcvCXrxHlwf4fNVSn8jEipMybMkWUyyF0JhnC+C4VcOVSBuHRKs1L5Ow==
  7918 +
7880 7919 prismjs@^1.23.0:
7881 7920 version "1.23.0"
7882 7921 resolved "https://registry.yarnpkg.com/prismjs/-/prismjs-1.23.0.tgz#d3b3967f7d72440690497652a9d40ff046067f33"
... ...