diff --git a/application/src/main/data/json/system/widget_bundles/cards.json b/application/src/main/data/json/system/widget_bundles/cards.json
index 3cd26a5624..c4cea769f1 100644
--- a/application/src/main/data/json/system/widget_bundles/cards.json
+++ b/application/src/main/data/json/system/widget_bundles/cards.json
@@ -167,6 +167,24 @@
         "dataKeySettingsSchema": "{}\n",
         "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\"}"
       }
+    },
+    {
+      "alias": "markdown_card",
+      "name": "Markdown Card",
+      "image": "",
+      "description": "Renders markdown/HTML using configurable pattern or function with applied attributes or timeseries values.",
+      "descriptor": {
+        "type": "latest",
+        "sizeX": 5,
+        "sizeY": 3.5,
+        "resources": [],
+        "templateHtml": "\n",
+        "templateCss": "#container {\n    overflow: auto;\n}",
+        "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",
+        "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",
+        "dataKeySettingsSchema": "{}\n",
+        "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}"
+      }
     }
   ]
 }
diff --git a/ui-ngx/angular.json b/ui-ngx/angular.json
index 1eff5453a2..90585f0ac0 100644
--- a/ui-ngx/angular.json
+++ b/ui-ngx/angular.json
@@ -78,7 +78,8 @@
               "node_modules/leaflet/dist/leaflet.css",
               "src/app/modules/home/components/widget/lib/maps/markers.scss",
               "node_modules/leaflet.markercluster/dist/MarkerCluster.css",
-              "node_modules/leaflet.markercluster/dist/MarkerCluster.Default.css"
+              "node_modules/leaflet.markercluster/dist/MarkerCluster.Default.css",
+              "node_modules/prismjs/themes/prism.css"
             ],
             "stylePreprocessorOptions": {
               "includePaths": [
@@ -88,7 +89,11 @@
             "scripts": [
               "node_modules/tinycolor2/dist/tinycolor-min.js",
               "node_modules/split.js/dist/split.min.js",
-              "node_modules/systemjs/dist/system.js"
+              "node_modules/systemjs/dist/system.js",
+              "node_modules/marked/lib/marked.js",
+              "node_modules/prismjs/prism.js",
+              "node_modules/prismjs/components/prism-bash.min.js",
+              "node_modules/prismjs/components/prism-json.min.js"
             ],
             "customWebpackConfig": {
               "path": "./extra-webpack.config.js"
diff --git a/ui-ngx/package.json b/ui-ngx/package.json
index 9d2be6658f..ba0ff16c5b 100644
--- a/ui-ngx/package.json
+++ b/ui-ngx/package.json
@@ -72,6 +72,7 @@
     "ngx-drag-drop": "^2.0.0",
     "ngx-flowchart": "git://github.com/thingsboard/ngx-flowchart.git#master",
     "ngx-hm-carousel": "^2.0.0-rc.1",
+    "ngx-markdown": "^10.1.1",
     "ngx-sharebuttons": "^8.0.5",
     "ngx-translate-messageformat-compiler": "^4.9.0",
     "objectpath": "^2.0.0",
diff --git a/ui-ngx/src/app/modules/home/components/widget/lib/maps/common-maps-utils.ts b/ui-ngx/src/app/modules/home/components/widget/lib/maps/common-maps-utils.ts
index 9b517bfab7..6b2bb5fafe 100644
--- a/ui-ngx/src/app/modules/home/components/widget/lib/maps/common-maps-utils.ts
+++ b/ui-ngx/src/app/modules/home/components/widget/lib/maps/common-maps-utils.ts
@@ -16,7 +16,7 @@
 
 import { FormattedData, MapProviders, ReplaceInfo } from '@home/components/widget/lib/maps/map-models';
 import {
-  createLabelFromDatasource,
+  createLabelFromDatasource, deepClone,
   hashCode,
   isDefined,
   isDefinedAndNotNull,
@@ -343,6 +343,22 @@ export function parseData(input: DatasourceData[]): FormattedData[] {
     });
 }
 
+export function flatData(input: FormattedData[]): FormattedData {
+  let result: FormattedData = {} as FormattedData;
+  if (input.length) {
+    for (const toMerge of input) {
+      result = {...result, ...toMerge};
+    }
+    result.entityName =  input[0].entityName;
+    result.entityId =  input[0].entityId;
+    result.entityType =  input[0].entityType;
+    result.$datasource =  input[0].$datasource;
+    result.dsIndex =  input[0].dsIndex;
+    result.deviceType =  input[0].deviceType;
+  }
+  return result;
+}
+
 export function parseArray(input: DatasourceData[]): FormattedData[][] {
   return _(input).groupBy(el => el?.datasource?.entityName)
     .values().value().map((entityArray) =>
diff --git a/ui-ngx/src/app/modules/home/components/widget/lib/markdown-widget.component.html b/ui-ngx/src/app/modules/home/components/widget/lib/markdown-widget.component.html
new file mode 100644
index 0000000000..df6c55762d
--- /dev/null
+++ b/ui-ngx/src/app/modules/home/components/widget/lib/markdown-widget.component.html
@@ -0,0 +1,18 @@
+
+
diff --git a/ui-ngx/src/app/modules/home/components/widget/lib/markdown-widget.component.scss b/ui-ngx/src/app/modules/home/components/widget/lib/markdown-widget.component.scss
new file mode 100644
index 0000000000..bea3ee1509
--- /dev/null
+++ b/ui-ngx/src/app/modules/home/components/widget/lib/markdown-widget.component.scss
@@ -0,0 +1,130 @@
+/**
+ * Copyright © 2016-2021 The Thingsboard Authors
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ *     http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+.tb-markdown-view {
+  display: block;
+  ::ng-deep {
+    h1 {
+      font-size: 32px;
+      padding-right: 60px;
+    }
+
+    h1, h2, h3, h4, h5, h6 {
+      line-height: normal;
+      font-weight: 500;
+      margin-bottom: 30px;
+      padding-bottom: 10px;
+      border-bottom: 1px solid #ccc;
+    }
+
+    p {
+      font-size: 16px;
+      font-weight: 400;
+      line-height: 1.25em;
+      margin: 0;
+    }
+
+    p+p {
+      margin-top: 10px;
+    }
+
+    table {
+      width: 100%;
+      border: 1px solid #ccc;
+      border-spacing: 0;
+      margin-top: 30px;
+      margin-bottom: 30px;
+    }
+
+    thead {
+      background-color: #555;
+      color: #fff;
+    }
+
+    th, td {
+      font-size: .85em;
+      padding: 8px;
+      margin: 0;
+      text-align: left;
+    }
+
+    td[align=center], th[align=center] {
+      text-align: center;
+    }
+
+    td[align=right], th[align=right] {
+      text-align: right;
+    }
+
+    tr:nth-child(even) {
+      background-color: #f7f7f7;
+    }
+
+    code:not([class*=language-]) {
+      background: #f5f5f5;
+      border-radius: 2px;
+      color: #dd4a68;
+      padding: 2px 4px;
+    }
+
+    div.code-wrapper {
+      position: relative;
+      button.clipboard-btn {
+        cursor: pointer;
+        margin: 0;
+        border: 0;
+        outline: none;
+        position: absolute;
+        top: 5px;
+        right: 5px;
+        background: #fff;
+        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);
+        border-radius: 5px;
+        opacity: 0;
+        transition: opacity .3s;
+        padding: 3px 6px;
+        line-height: 16px;
+        img {
+          width: 18px;
+        }
+        &:hover {
+          background: #f9f9f9;
+        }
+        &:active {
+          background-color: #ececec;
+          box-shadow: inset 1px -1px 4px 0px rgba(0,0,0,0.4);
+        }
+      }
+      &:hover {
+        button.clipboard-btn {
+          opacity: .85;
+        }
+      }
+    }
+
+    th, td {
+      div.code-wrapper {
+        display: inline-block;
+        button.clipboard-btn {
+          top: -5px;
+          right: -30px;
+          padding: 0 3px;
+        }
+      }
+    }
+
+  }
+}
diff --git a/ui-ngx/src/app/modules/home/components/widget/lib/markdown-widget.component.ts b/ui-ngx/src/app/modules/home/components/widget/lib/markdown-widget.component.ts
new file mode 100644
index 0000000000..111b74eed4
--- /dev/null
+++ b/ui-ngx/src/app/modules/home/components/widget/lib/markdown-widget.component.ts
@@ -0,0 +1,116 @@
+///
+/// Copyright © 2016-2021 The Thingsboard Authors
+///
+/// Licensed under the Apache License, Version 2.0 (the "License");
+/// you may not use this file except in compliance with the License.
+/// You may obtain a copy of the License at
+///
+///     http://www.apache.org/licenses/LICENSE-2.0
+///
+/// Unless required by applicable law or agreed to in writing, software
+/// distributed under the License is distributed on an "AS IS" BASIS,
+/// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+/// See the License for the specific language governing permissions and
+/// limitations under the License.
+///
+
+import { ChangeDetectorRef, Component, ElementRef, Input, OnInit } from '@angular/core';
+import { PageComponent } from '@shared/components/page.component';
+import { WidgetContext } from '@home/models/widget-component.models';
+import { Store } from '@ngrx/store';
+import { AppState } from '@core/core.state';
+import { DatasourceData } from '@shared/models/widget.models';
+import { DataKeyType } from '@shared/models/telemetry/telemetry.models';
+import {
+  fillPattern, flatData,
+  parseData,
+  parseFunction,
+  processPattern,
+  safeExecute
+} from '@home/components/widget/lib/maps/common-maps-utils';
+import { FormattedData } from '@home/components/widget/lib/maps/map-models';
+import { hashCode, isNotEmptyStr } from '@core/utils';
+import cssjs from '@core/css/css';
+
+interface MarkdownWidgetSettings {
+  markdownTextPattern: string;
+  useMarkdownTextFunction: boolean;
+  markdownTextFunction: string;
+  markdownCss: string;
+}
+
+type MarkdownTextFunction = (data: FormattedData[]) => string;
+
+@Component({
+  selector: 'tb-markdown-widget ',
+  templateUrl: './markdown-widget.component.html',
+  styleUrls: ['./markdown-widget.component.scss']
+})
+export class MarkdownWidgetComponent extends PageComponent implements OnInit {
+
+  settings: MarkdownWidgetSettings;
+  markdownTextFunction: MarkdownTextFunction;
+
+  @Input()
+  ctx: WidgetContext;
+
+  markdownText: string;
+
+  constructor(protected store: Store,
+              private elementRef: ElementRef,
+              private cd: ChangeDetectorRef) {
+    super(store);
+  }
+
+  ngOnInit(): void {
+    this.ctx.$scope.markdownWidget = this;
+    this.settings = this.ctx.settings;
+    this.markdownTextFunction = this.settings.useMarkdownTextFunction ? parseFunction(this.settings.markdownTextFunction, ['data']) : null;
+
+    const cssString = this.settings.markdownCss;
+    if (isNotEmptyStr(cssString)) {
+      const cssParser = new cssjs();
+      cssParser.testMode = false;
+      const namespace = 'entities-hierarchy-' + hashCode(cssString);
+      cssParser.cssPreviewNamespace = namespace;
+      cssParser.createStyleElement(namespace, cssString);
+      $(this.elementRef.nativeElement).addClass(namespace);
+    }
+  }
+
+  public onDataUpdated() {
+    let initialData: DatasourceData[];
+    if (this.ctx.data?.length) {
+      initialData = this.ctx.data;
+    } else if (this.ctx.datasources?.length) {
+      initialData = [
+        {
+          datasource: this.ctx.datasources[0],
+          dataKey: {
+            type: DataKeyType.attribute,
+            name: 'empty'
+          },
+          data: []
+        }
+      ];
+    }
+    let markdownText: string;
+    if (initialData) {
+      const data = parseData(initialData);
+      markdownText = this.settings.useMarkdownTextFunction ?
+        safeExecute(this.markdownTextFunction, [data]) : this.settings.markdownTextPattern;
+      const allData = flatData(data);
+      const replaceInfo = processPattern(markdownText, allData);
+      markdownText = fillPattern(markdownText, replaceInfo, allData);
+    }
+    if (this.markdownText !== markdownText) {
+      this.markdownText = markdownText;
+      this.cd.detectChanges();
+    }
+  }
+
+  markdownClick($event: MouseEvent) {
+    this.ctx.actionsApi.elementClick($event);
+  }
+
+}
diff --git a/ui-ngx/src/app/modules/home/components/widget/lib/qrcode-widget.component.ts b/ui-ngx/src/app/modules/home/components/widget/lib/qrcode-widget.component.ts
index 6c1f148de5..dff51a1378 100644
--- a/ui-ngx/src/app/modules/home/components/widget/lib/qrcode-widget.component.ts
+++ b/ui-ngx/src/app/modules/home/components/widget/lib/qrcode-widget.component.ts
@@ -101,7 +101,7 @@ export class QrCodeWidgetComponent extends PageComponent implements OnInit, Afte
       const dataSourceData = data[0];
       const pattern = this.settings.useQrCodeTextFunction ?
         safeExecute(this.qrCodeTextFunction, [dataSourceData]) : this.settings.qrCodeTextPattern;
-      const replaceInfo = processPattern(pattern, data);
+      const replaceInfo = processPattern(pattern, dataSourceData);
       qrCodeText = fillPattern(pattern, replaceInfo, dataSourceData);
     }
     this.updateQrCodeText(qrCodeText);
diff --git a/ui-ngx/src/app/modules/home/components/widget/widget-components.module.ts b/ui-ngx/src/app/modules/home/components/widget/widget-components.module.ts
index 475d2c7170..ed9ef23f82 100644
--- a/ui-ngx/src/app/modules/home/components/widget/widget-components.module.ts
+++ b/ui-ngx/src/app/modules/home/components/widget/widget-components.module.ts
@@ -40,6 +40,7 @@ import { NavigationCardWidgetComponent } from '@home/components/widget/lib/navig
 import { EdgesOverviewWidgetComponent } from '@home/components/widget/lib/edges-overview-widget.component';
 import { JsonInputWidgetComponent } from '@home/components/widget/lib/json-input-widget.component';
 import { QrCodeWidgetComponent } from '@home/components/widget/lib/qrcode-widget.component';
+import { MarkdownWidgetComponent } from '@home/components/widget/lib/markdown-widget.component';
 
 @NgModule({
   declarations:
@@ -60,7 +61,8 @@ import { QrCodeWidgetComponent } from '@home/components/widget/lib/qrcode-widget
       GatewayFormComponent,
       NavigationCardsWidgetComponent,
       NavigationCardWidgetComponent,
-      QrCodeWidgetComponent
+      QrCodeWidgetComponent,
+      MarkdownWidgetComponent
     ],
   imports: [
     CommonModule,
@@ -83,7 +85,8 @@ import { QrCodeWidgetComponent } from '@home/components/widget/lib/qrcode-widget
     GatewayFormComponent,
     NavigationCardsWidgetComponent,
     NavigationCardWidgetComponent,
-    QrCodeWidgetComponent
+    QrCodeWidgetComponent,
+    MarkdownWidgetComponent
   ],
   providers: [
     CustomDialogService,
diff --git a/ui-ngx/src/app/shared/components/json-form/react/json-form-ace-editor.tsx b/ui-ngx/src/app/shared/components/json-form/react/json-form-ace-editor.tsx
index fdcf8484b5..823ef25921 100644
--- a/ui-ngx/src/app/shared/components/json-form/react/json-form-ace-editor.tsx
+++ b/ui-ngx/src/app/shared/components/json-form/react/json-form-ace-editor.tsx
@@ -159,8 +159,8 @@ class ThingsboardAceEditor extends React.Component
                       
                           
-                          
+                          { this.props.onTidy ? 
 : null }