Showing
15 changed files
with
502 additions
and
11 deletions
... | ... | @@ -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" | ... | ... |