Showing
7 changed files
with
225 additions
and
6 deletions
@@ -149,6 +149,24 @@ | @@ -149,6 +149,24 @@ | ||
149 | "dataKeySettingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"DataKeySettings\",\n \"properties\": {},\n \"required\": []\n },\n \"form\": []\n}", | 149 | "dataKeySettingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"DataKeySettings\",\n \"properties\": {},\n \"required\": []\n },\n \"form\": []\n}", |
150 | "defaultConfig": "{\"timewindow\":{\"realtime\":{\"interval\":1000,\"timewindowMs\":86400000},\"aggregation\":{\"type\":\"NONE\",\"limit\":200}},\"showTitle\":true,\"backgroundColor\":\"rgb(255, 255, 255)\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"4px\",\"settings\":{\"nodeRelationQueryFunction\":\"/**\\n\\n// Function should return relations query object for current node used to fetch entity children.\\n// Function can return 'default' string value. In this case default relations query will be used.\\n\\n// The following example code will construct simple relations query that will fetch relations of type 'Contains'\\n// from the current entity.\\n\\nvar entity = nodeCtx.entity;\\nvar query = {\\n parameters: {\\n rootId: entity.id.id,\\n rootType: entity.id.entityType,\\n direction: \\\"FROM\\\",\\n maxLevel: 1\\n },\\n filters: [{\\n relationType: \\\"Contains\\\",\\n entityTypes: []\\n }]\\n};\\nreturn query;\\n\\n**/\\n\",\"nodeHasChildrenFunction\":\"/**\\n\\n// Function should return boolean value indicating whether current node has children (whether it can be expanded).\\n\\n// The following example code will restrict entities hierarchy expansion up to third level.\\n\\nreturn nodeCtx.level <= 2;\\n\\n// The next example code will restrict entities expansion according to the value of example 'nodeHasChildren' attribute.\\n\\nvar data = nodeCtx.data;\\nif (data.hasOwnProperty('nodeHasChildren') && data['nodeHasChildren'] !== null) {\\n return data['nodeHasChildren'] === 'true';\\n} else {\\n return true;\\n}\\n \\n**/\\n \",\"nodeTextFunction\":\"/**\\n\\n// Function should return text (can be HTML code) for the current node.\\n\\n// The following example code will generate node text consisting of entity name and temperature if temperature value is present in entity attributes/timeseries.\\n\\nvar data = nodeCtx.data;\\nvar entity = nodeCtx.entity;\\nvar text = entity.name;\\nif (data.hasOwnProperty('temperature') && data['temperature'] !== null) {\\n text += \\\" <b>\\\"+ data['temperature'] +\\\" °C</b>\\\";\\n}\\nreturn text;\\n\\n**/\",\"nodeIconFunction\":\"/** \\n\\n// Function should return node icon info object.\\n// Resulting object should contain either 'materialIcon' or 'iconUrl' property. \\n// Where:\\n - 'materialIcon' - name of the material icon to be used from the Material Icons Library (https://material.io/tools/icons);\\n - 'iconUrl' - url of the external image to be used as node icon.\\n// Function can return 'default' string value. In this case default icons according to entity type will be used.\\n\\n// The following example code shows how to use external image for devices which name starts with 'Test' and use \\n// default icons for the rest of entities.\\n\\nvar entity = nodeCtx.entity;\\nif (entity.id.entityType === 'DEVICE' && entity.name.startsWith('Test')) {\\n return {iconUrl: 'https://avatars1.githubusercontent.com/u/14793288?v=4&s=117'};\\n} else {\\n return 'default';\\n}\\n \\n**/\",\"nodeDisabledFunction\":\"/**\\n\\n// Function should return boolean value indicating whether current node should be disabled (not selectable).\\n\\n// The following example code will disable current node according to the value of example 'nodeDisabled' attribute.\\n\\nvar data = nodeCtx.data;\\nif (data.hasOwnProperty('nodeDisabled') && data['nodeDisabled'] !== null) {\\n return data['nodeDisabled'] === 'true';\\n} else {\\n return false;\\n}\\n \\n**/\\n\",\"nodesSortFunction\":\"/**\\n\\n// This function is used to sort nodes of the same level. Function should compare two nodes and return \\n// integer value: \\n// - less than 0 - sort nodeCtx1 to an index lower than nodeCtx2\\n// - 0 - leave nodeCtx1 and nodeCtx2 unchanged with respect to each other\\n// - greater than 0 - sort nodeCtx2 to an index lower than nodeCtx1\\n\\n// The following example code will sort entities first by entity type in alphabetical order then\\n// by entity name in alphabetical order.\\n\\nvar result = nodeCtx1.entity.id.entityType.localeCompare(nodeCtx2.entity.id.entityType);\\nif (result === 0) {\\n result = nodeCtx1.entity.name.localeCompare(nodeCtx2.entity.name);\\n}\\nreturn result;\\n \\n**/\",\"nodeOpenedFunction\":\"/**\\n\\n// Function should return boolean value indicating whether current node should be opened (expanded) when it first loaded.\\n\\n// The following example code will open by default nodes up to third level.\\n\\nreturn nodeCtx.level <= 2;\\n\\n**/\\n \"},\"title\":\"Entities hierarchy\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400,\"padding\":\"5px 10px 5px 10px\"},\"useDashboardTimewindow\":false,\"showLegend\":false,\"datasources\":[{\"type\":\"function\",\"name\":\"Simulated\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Sin\",\"color\":\"#2196f3\",\"settings\":{\"columnWidth\":\"0px\",\"useCellStyleFunction\":false,\"cellStyleFunction\":\"\",\"useCellContentFunction\":false,\"cellContentFunction\":\"\"},\"_hash\":0.472295003170325,\"funcBody\":\"return Math.round(1000*Math.sin(time/5000));\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Cos\",\"color\":\"#4caf50\",\"settings\":{\"columnWidth\":\"0px\",\"useCellStyleFunction\":false,\"cellStyleFunction\":\"\",\"useCellContentFunction\":false,\"cellContentFunction\":\"\"},\"_hash\":0.8926244886945558,\"funcBody\":\"return Math.round(1000*Math.cos(time/5000));\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Random\",\"color\":\"#f44336\",\"settings\":{\"columnWidth\":\"0px\",\"useCellStyleFunction\":false,\"cellStyleFunction\":\"\",\"useCellContentFunction\":false,\"cellContentFunction\":\"\"},\"_hash\":0.6401141393938932,\"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;\"}]}],\"widgetStyle\":{},\"actions\":{}}" | 150 | "defaultConfig": "{\"timewindow\":{\"realtime\":{\"interval\":1000,\"timewindowMs\":86400000},\"aggregation\":{\"type\":\"NONE\",\"limit\":200}},\"showTitle\":true,\"backgroundColor\":\"rgb(255, 255, 255)\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"4px\",\"settings\":{\"nodeRelationQueryFunction\":\"/**\\n\\n// Function should return relations query object for current node used to fetch entity children.\\n// Function can return 'default' string value. In this case default relations query will be used.\\n\\n// The following example code will construct simple relations query that will fetch relations of type 'Contains'\\n// from the current entity.\\n\\nvar entity = nodeCtx.entity;\\nvar query = {\\n parameters: {\\n rootId: entity.id.id,\\n rootType: entity.id.entityType,\\n direction: \\\"FROM\\\",\\n maxLevel: 1\\n },\\n filters: [{\\n relationType: \\\"Contains\\\",\\n entityTypes: []\\n }]\\n};\\nreturn query;\\n\\n**/\\n\",\"nodeHasChildrenFunction\":\"/**\\n\\n// Function should return boolean value indicating whether current node has children (whether it can be expanded).\\n\\n// The following example code will restrict entities hierarchy expansion up to third level.\\n\\nreturn nodeCtx.level <= 2;\\n\\n// The next example code will restrict entities expansion according to the value of example 'nodeHasChildren' attribute.\\n\\nvar data = nodeCtx.data;\\nif (data.hasOwnProperty('nodeHasChildren') && data['nodeHasChildren'] !== null) {\\n return data['nodeHasChildren'] === 'true';\\n} else {\\n return true;\\n}\\n \\n**/\\n \",\"nodeTextFunction\":\"/**\\n\\n// Function should return text (can be HTML code) for the current node.\\n\\n// The following example code will generate node text consisting of entity name and temperature if temperature value is present in entity attributes/timeseries.\\n\\nvar data = nodeCtx.data;\\nvar entity = nodeCtx.entity;\\nvar text = entity.name;\\nif (data.hasOwnProperty('temperature') && data['temperature'] !== null) {\\n text += \\\" <b>\\\"+ data['temperature'] +\\\" °C</b>\\\";\\n}\\nreturn text;\\n\\n**/\",\"nodeIconFunction\":\"/** \\n\\n// Function should return node icon info object.\\n// Resulting object should contain either 'materialIcon' or 'iconUrl' property. \\n// Where:\\n - 'materialIcon' - name of the material icon to be used from the Material Icons Library (https://material.io/tools/icons);\\n - 'iconUrl' - url of the external image to be used as node icon.\\n// Function can return 'default' string value. In this case default icons according to entity type will be used.\\n\\n// The following example code shows how to use external image for devices which name starts with 'Test' and use \\n// default icons for the rest of entities.\\n\\nvar entity = nodeCtx.entity;\\nif (entity.id.entityType === 'DEVICE' && entity.name.startsWith('Test')) {\\n return {iconUrl: 'https://avatars1.githubusercontent.com/u/14793288?v=4&s=117'};\\n} else {\\n return 'default';\\n}\\n \\n**/\",\"nodeDisabledFunction\":\"/**\\n\\n// Function should return boolean value indicating whether current node should be disabled (not selectable).\\n\\n// The following example code will disable current node according to the value of example 'nodeDisabled' attribute.\\n\\nvar data = nodeCtx.data;\\nif (data.hasOwnProperty('nodeDisabled') && data['nodeDisabled'] !== null) {\\n return data['nodeDisabled'] === 'true';\\n} else {\\n return false;\\n}\\n \\n**/\\n\",\"nodesSortFunction\":\"/**\\n\\n// This function is used to sort nodes of the same level. Function should compare two nodes and return \\n// integer value: \\n// - less than 0 - sort nodeCtx1 to an index lower than nodeCtx2\\n// - 0 - leave nodeCtx1 and nodeCtx2 unchanged with respect to each other\\n// - greater than 0 - sort nodeCtx2 to an index lower than nodeCtx1\\n\\n// The following example code will sort entities first by entity type in alphabetical order then\\n// by entity name in alphabetical order.\\n\\nvar result = nodeCtx1.entity.id.entityType.localeCompare(nodeCtx2.entity.id.entityType);\\nif (result === 0) {\\n result = nodeCtx1.entity.name.localeCompare(nodeCtx2.entity.name);\\n}\\nreturn result;\\n \\n**/\",\"nodeOpenedFunction\":\"/**\\n\\n// Function should return boolean value indicating whether current node should be opened (expanded) when it first loaded.\\n\\n// The following example code will open by default nodes up to third level.\\n\\nreturn nodeCtx.level <= 2;\\n\\n**/\\n \"},\"title\":\"Entities hierarchy\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400,\"padding\":\"5px 10px 5px 10px\"},\"useDashboardTimewindow\":false,\"showLegend\":false,\"datasources\":[{\"type\":\"function\",\"name\":\"Simulated\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Sin\",\"color\":\"#2196f3\",\"settings\":{\"columnWidth\":\"0px\",\"useCellStyleFunction\":false,\"cellStyleFunction\":\"\",\"useCellContentFunction\":false,\"cellContentFunction\":\"\"},\"_hash\":0.472295003170325,\"funcBody\":\"return Math.round(1000*Math.sin(time/5000));\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Cos\",\"color\":\"#4caf50\",\"settings\":{\"columnWidth\":\"0px\",\"useCellStyleFunction\":false,\"cellStyleFunction\":\"\",\"useCellContentFunction\":false,\"cellContentFunction\":\"\"},\"_hash\":0.8926244886945558,\"funcBody\":\"return Math.round(1000*Math.cos(time/5000));\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Random\",\"color\":\"#f44336\",\"settings\":{\"columnWidth\":\"0px\",\"useCellStyleFunction\":false,\"cellStyleFunction\":\"\",\"useCellContentFunction\":false,\"cellContentFunction\":\"\"},\"_hash\":0.6401141393938932,\"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;\"}]}],\"widgetStyle\":{},\"actions\":{}}" |
151 | } | 151 | } |
152 | + }, | ||
153 | + { | ||
154 | + "alias": "qr_code", | ||
155 | + "name": "QR Code", | ||
156 | + "image": "", | ||
157 | + "description": "Displays QR code of calculated text from configured pattern or function with applied attributes or timeseries values.", | ||
158 | + "descriptor": { | ||
159 | + "type": "latest", | ||
160 | + "sizeX": 4, | ||
161 | + "sizeY": 3.5, | ||
162 | + "resources": [], | ||
163 | + "templateHtml": "<tb-qrcode-widget \n [ctx]=\"ctx\">\n</tb-qrcode-widget>", | ||
164 | + "templateCss": "#container {\n overflow: auto;\n}\n\n.tbDatasource-container {\n margin: 5px;\n padding: 8px;\n}\n\n.tbDatasource-title {\n font-size: 1.200rem;\n font-weight: 500;\n padding-bottom: 10px;\n}\n\n.tbDatasource-table {\n width: 100%;\n box-shadow: 0 0 10px #ccc;\n border-collapse: collapse;\n white-space: nowrap;\n font-size: 1.000rem;\n color: #757575;\n}\n\n.tbDatasource-table td {\n position: relative;\n border-top: 1px solid rgba(0, 0, 0, 0.12);\n border-bottom: 1px solid rgba(0, 0, 0, 0.12);\n padding: 0px 18px;\n box-sizing: border-box;\n}", | ||
165 | + "controllerScript": "self.onInit = function() {\n}\n\nself.onDataUpdated = function() {\n self.ctx.$scope.qrCodeWidget.onDataUpdated();\n}\n\nself.typeParameters = function() {\n return {\n maxDatasources: 1,\n dataKeysOptional: true,\n singleEntity: true\n };\n}\n\nself.onDestroy = function() {\n}\n\n", | ||
166 | + "settingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"QR Code\",\n \"properties\": {\n \"qrCodeTextPattern\": {\n \"title\": \"QR code text pattern (for ex. '${entityName} | ${keyName} - some text.')\",\n \"type\": \"string\",\n \"default\": \"${entityName}\"\n },\n \"useQrCodeTextFunction\": {\n \"title\": \"Use QR code text function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"qrCodeTextFunction\": {\n \"title\": \"QR code text function: f(data)\",\n \"type\": \"string\",\n \"default\": \"return data['entityName'];\"\n }\n },\n \"required\": []\n },\n \"form\": [\n \"qrCodeTextPattern\",\n \"useQrCodeTextFunction\",\n {\n \"key\": \"qrCodeTextFunction\",\n \"type\": \"javascript\"\n }\n ]\n}\n", | ||
167 | + "dataKeySettingsSchema": "{}\n", | ||
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 | + } | ||
152 | } | 170 | } |
153 | ] | 171 | ] |
154 | } | 172 | } |
@@ -77,6 +77,7 @@ | @@ -77,6 +77,7 @@ | ||
77 | "objectpath": "^2.0.0", | 77 | "objectpath": "^2.0.0", |
78 | "prettier": "^2.1.2", | 78 | "prettier": "^2.1.2", |
79 | "prop-types": "^15.7.2", | 79 | "prop-types": "^15.7.2", |
80 | + "qrcode": "^1.4.4", | ||
80 | "raphael": "^2.3.0", | 81 | "raphael": "^2.3.0", |
81 | "rc-select": "~10.5.1", | 82 | "rc-select": "~10.5.1", |
82 | "react": "~16.14.0", | 83 | "react": "~16.14.0", |
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 | +<div fxLayout="column" fxLayoutAlign="center center" style="width: 100%; height: 100%;"> | ||
19 | + <canvas fxFlex #canvas [ngStyle]="{display: qrCodeText ? 'block' : 'none'}"></canvas> | ||
20 | + <div *ngIf="!qrCodeText" translate>entity.no-data</div> | ||
21 | +</div> |
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 { AfterViewInit, ChangeDetectorRef, Component, ElementRef, Input, OnInit, ViewChild } 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 QRCode from 'qrcode'; | ||
23 | +import { | ||
24 | + fillPattern, | ||
25 | + parseData, | ||
26 | + parseFunction, | ||
27 | + processPattern, | ||
28 | + safeExecute | ||
29 | +} from '@home/components/widget/lib/maps/common-maps-utils'; | ||
30 | +import { FormattedData } from '@home/components/widget/lib/maps/map-models'; | ||
31 | +import { DatasourceData } from '@shared/models/widget.models'; | ||
32 | +import { DataKeyType } from '@shared/models/telemetry/telemetry.models'; | ||
33 | + | ||
34 | +interface QrCodeWidgetSettings { | ||
35 | + qrCodeTextPattern: string; | ||
36 | + useQrCodeTextFunction: boolean; | ||
37 | + qrCodeTextFunction: string; | ||
38 | +} | ||
39 | + | ||
40 | +type QrCodeTextFunction = (data: FormattedData) => string; | ||
41 | + | ||
42 | +@Component({ | ||
43 | + selector: 'tb-qrcode-widget', | ||
44 | + templateUrl: './qrcode-widget.component.html', | ||
45 | + styleUrls: [] | ||
46 | +}) | ||
47 | +export class QrCodeWidgetComponent extends PageComponent implements OnInit, AfterViewInit { | ||
48 | + | ||
49 | + settings: QrCodeWidgetSettings; | ||
50 | + qrCodeTextFunction: QrCodeTextFunction; | ||
51 | + | ||
52 | + @Input() | ||
53 | + ctx: WidgetContext; | ||
54 | + | ||
55 | + qrCodeText: string; | ||
56 | + | ||
57 | + private viewInited: boolean; | ||
58 | + private scheduleUpdateCanvas: boolean; | ||
59 | + | ||
60 | + @ViewChild('canvas', {static: false}) canvasRef: ElementRef<HTMLCanvasElement>; | ||
61 | + | ||
62 | + constructor(protected store: Store<AppState>, | ||
63 | + protected cd: ChangeDetectorRef) { | ||
64 | + super(store); | ||
65 | + } | ||
66 | + | ||
67 | + ngOnInit(): void { | ||
68 | + this.ctx.$scope.qrCodeWidget = this; | ||
69 | + this.settings = this.ctx.settings; | ||
70 | + this.qrCodeTextFunction = this.settings.useQrCodeTextFunction ? parseFunction(this.settings.qrCodeTextFunction, ['data']) : null; | ||
71 | + } | ||
72 | + | ||
73 | + ngAfterViewInit(): void { | ||
74 | + this.viewInited = true; | ||
75 | + if (this.scheduleUpdateCanvas) { | ||
76 | + this.scheduleUpdateCanvas = false; | ||
77 | + this.updateCanvas(); | ||
78 | + } | ||
79 | + } | ||
80 | + | ||
81 | + public onDataUpdated() { | ||
82 | + let initialData: DatasourceData[]; | ||
83 | + let qrCodeText: string; | ||
84 | + if (this.ctx.data?.length) { | ||
85 | + initialData = this.ctx.data; | ||
86 | + } else if (this.ctx.datasources?.length) { | ||
87 | + initialData = [ | ||
88 | + { | ||
89 | + datasource: this.ctx.datasources[0], | ||
90 | + dataKey: { | ||
91 | + type: DataKeyType.attribute, | ||
92 | + name: 'empty' | ||
93 | + }, | ||
94 | + data: [] | ||
95 | + } | ||
96 | + ]; | ||
97 | + } | ||
98 | + if (initialData) { | ||
99 | + const data = parseData(initialData); | ||
100 | + const dataSourceData = data[0]; | ||
101 | + const pattern = this.settings.useQrCodeTextFunction ? | ||
102 | + safeExecute(this.qrCodeTextFunction, [dataSourceData]) : this.settings.qrCodeTextPattern; | ||
103 | + const replaceInfo = processPattern(pattern, data); | ||
104 | + qrCodeText = fillPattern(pattern, replaceInfo, dataSourceData); | ||
105 | + } | ||
106 | + this.updateQrCodeText(qrCodeText); | ||
107 | + } | ||
108 | + | ||
109 | + private updateQrCodeText(newQrCodeText: string): void { | ||
110 | + if (this.qrCodeText !== newQrCodeText) { | ||
111 | + this.qrCodeText = newQrCodeText; | ||
112 | + if (this.qrCodeText) { | ||
113 | + this.updateCanvas(); | ||
114 | + } | ||
115 | + this.cd.detectChanges(); | ||
116 | + } | ||
117 | + } | ||
118 | + | ||
119 | + private updateCanvas() { | ||
120 | + if (this.viewInited) { | ||
121 | + QRCode.toCanvas(this.canvasRef.nativeElement, this.qrCodeText); | ||
122 | + this.canvasRef.nativeElement.style.width = 'auto'; | ||
123 | + this.canvasRef.nativeElement.style.height = 'auto'; | ||
124 | + } else { | ||
125 | + this.scheduleUpdateCanvas = true; | ||
126 | + } | ||
127 | + } | ||
128 | + | ||
129 | +} |
@@ -39,6 +39,7 @@ import { NavigationCardsWidgetComponent } from '@home/components/widget/lib/navi | @@ -39,6 +39,7 @@ import { NavigationCardsWidgetComponent } from '@home/components/widget/lib/navi | ||
39 | import { NavigationCardWidgetComponent } from '@home/components/widget/lib/navigation-card-widget.component'; | 39 | import { NavigationCardWidgetComponent } from '@home/components/widget/lib/navigation-card-widget.component'; |
40 | import { EdgesOverviewWidgetComponent } from '@home/components/widget/lib/edges-overview-widget.component'; | 40 | import { EdgesOverviewWidgetComponent } from '@home/components/widget/lib/edges-overview-widget.component'; |
41 | import { JsonInputWidgetComponent } from '@home/components/widget/lib/json-input-widget.component'; | 41 | import { JsonInputWidgetComponent } from '@home/components/widget/lib/json-input-widget.component'; |
42 | +import { QrCodeWidgetComponent } from '@home/components/widget/lib/qrcode-widget.component'; | ||
42 | 43 | ||
43 | @NgModule({ | 44 | @NgModule({ |
44 | declarations: | 45 | declarations: |
@@ -58,7 +59,8 @@ import { JsonInputWidgetComponent } from '@home/components/widget/lib/json-input | @@ -58,7 +59,8 @@ import { JsonInputWidgetComponent } from '@home/components/widget/lib/json-input | ||
58 | PhotoCameraInputWidgetComponent, | 59 | PhotoCameraInputWidgetComponent, |
59 | GatewayFormComponent, | 60 | GatewayFormComponent, |
60 | NavigationCardsWidgetComponent, | 61 | NavigationCardsWidgetComponent, |
61 | - NavigationCardWidgetComponent | 62 | + NavigationCardWidgetComponent, |
63 | + QrCodeWidgetComponent | ||
62 | ], | 64 | ], |
63 | imports: [ | 65 | imports: [ |
64 | CommonModule, | 66 | CommonModule, |
@@ -80,7 +82,8 @@ import { JsonInputWidgetComponent } from '@home/components/widget/lib/json-input | @@ -80,7 +82,8 @@ import { JsonInputWidgetComponent } from '@home/components/widget/lib/json-input | ||
80 | PhotoCameraInputWidgetComponent, | 82 | PhotoCameraInputWidgetComponent, |
81 | GatewayFormComponent, | 83 | GatewayFormComponent, |
82 | NavigationCardsWidgetComponent, | 84 | NavigationCardsWidgetComponent, |
83 | - NavigationCardWidgetComponent | 85 | + NavigationCardWidgetComponent, |
86 | + QrCodeWidgetComponent | ||
84 | ], | 87 | ], |
85 | providers: [ | 88 | providers: [ |
86 | CustomDialogService, | 89 | CustomDialogService, |
@@ -2774,7 +2774,25 @@ browserstack@^1.5.1: | @@ -2774,7 +2774,25 @@ browserstack@^1.5.1: | ||
2774 | dependencies: | 2774 | dependencies: |
2775 | https-proxy-agent "^2.2.1" | 2775 | https-proxy-agent "^2.2.1" |
2776 | 2776 | ||
2777 | -buffer-from@^1.0.0: | 2777 | +buffer-alloc-unsafe@^1.1.0: |
2778 | + version "1.1.0" | ||
2779 | + resolved "https://registry.yarnpkg.com/buffer-alloc-unsafe/-/buffer-alloc-unsafe-1.1.0.tgz#bd7dc26ae2972d0eda253be061dba992349c19f0" | ||
2780 | + integrity sha512-TEM2iMIEQdJ2yjPJoSIsldnleVaAk1oW3DBVUykyOLsEsFmEc9kn+SFFPz+gl54KQNxlDnAwCXosOS9Okx2xAg== | ||
2781 | + | ||
2782 | +buffer-alloc@^1.2.0: | ||
2783 | + version "1.2.0" | ||
2784 | + resolved "https://registry.yarnpkg.com/buffer-alloc/-/buffer-alloc-1.2.0.tgz#890dd90d923a873e08e10e5fd51a57e5b7cce0ec" | ||
2785 | + integrity sha512-CFsHQgjtW1UChdXgbyJGtnm+O/uLQeZdtbDo8mfUgYXCHSM1wgrVxXm6bSyrUuErEb+4sYVGCzASBRot7zyrow== | ||
2786 | + dependencies: | ||
2787 | + buffer-alloc-unsafe "^1.1.0" | ||
2788 | + buffer-fill "^1.0.0" | ||
2789 | + | ||
2790 | +buffer-fill@^1.0.0: | ||
2791 | + version "1.0.0" | ||
2792 | + resolved "https://registry.yarnpkg.com/buffer-fill/-/buffer-fill-1.0.0.tgz#f8f78b76789888ef39f205cd637f68e702122b2c" | ||
2793 | + integrity sha1-+PeLdniYiO858gXNY39o5wISKyw= | ||
2794 | + | ||
2795 | +buffer-from@^1.0.0, buffer-from@^1.1.1: | ||
2778 | version "1.1.1" | 2796 | version "1.1.1" |
2779 | resolved "https://registry.yarnpkg.com/buffer-from/-/buffer-from-1.1.1.tgz#32713bc028f75c02fdb710d7c7bcec1f2c6070ef" | 2797 | resolved "https://registry.yarnpkg.com/buffer-from/-/buffer-from-1.1.1.tgz#32713bc028f75c02fdb710d7c7bcec1f2c6070ef" |
2780 | integrity sha512-MQcXEUbCKtEo7bhqEs6560Hyd4XaovZlO/k9V3hjVUF/zwW7KBVdSK4gIt/bzwS9MbR5qob+F5jusZsb0YQK2A== | 2798 | integrity sha512-MQcXEUbCKtEo7bhqEs6560Hyd4XaovZlO/k9V3hjVUF/zwW7KBVdSK4gIt/bzwS9MbR5qob+F5jusZsb0YQK2A== |
@@ -2798,7 +2816,7 @@ buffer@^4.3.0: | @@ -2798,7 +2816,7 @@ buffer@^4.3.0: | ||
2798 | ieee754 "^1.1.4" | 2816 | ieee754 "^1.1.4" |
2799 | isarray "^1.0.0" | 2817 | isarray "^1.0.0" |
2800 | 2818 | ||
2801 | -buffer@^5.5.0: | 2819 | +buffer@^5.4.3, buffer@^5.5.0: |
2802 | version "5.7.1" | 2820 | version "5.7.1" |
2803 | resolved "https://registry.yarnpkg.com/buffer/-/buffer-5.7.1.tgz#ba62e7c13133053582197160851a8f648e99eed0" | 2821 | resolved "https://registry.yarnpkg.com/buffer/-/buffer-5.7.1.tgz#ba62e7c13133053582197160851a8f648e99eed0" |
2804 | integrity sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ== | 2822 | integrity sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ== |
@@ -3953,6 +3971,11 @@ diffie-hellman@^5.0.0: | @@ -3953,6 +3971,11 @@ diffie-hellman@^5.0.0: | ||
3953 | miller-rabin "^4.0.0" | 3971 | miller-rabin "^4.0.0" |
3954 | randombytes "^2.0.0" | 3972 | randombytes "^2.0.0" |
3955 | 3973 | ||
3974 | +dijkstrajs@^1.0.1: | ||
3975 | + version "1.0.2" | ||
3976 | + resolved "https://registry.yarnpkg.com/dijkstrajs/-/dijkstrajs-1.0.2.tgz#2e48c0d3b825462afe75ab4ad5e829c8ece36257" | ||
3977 | + integrity sha512-QV6PMaHTCNmKSeP6QoXhVTw9snc9VD8MulTT0Bd99Pacp4SS1cjcrYPgBPmibqKVtMJJfqC6XvOXgPMEEPH/fg== | ||
3978 | + | ||
3956 | dir-glob@^3.0.1: | 3979 | dir-glob@^3.0.1: |
3957 | version "3.0.1" | 3980 | version "3.0.1" |
3958 | resolved "https://registry.yarnpkg.com/dir-glob/-/dir-glob-3.0.1.tgz#56dbf73d992a4a93ba1584f4534063fd2e41717f" | 3981 | resolved "https://registry.yarnpkg.com/dir-glob/-/dir-glob-3.0.1.tgz#56dbf73d992a4a93ba1584f4534063fd2e41717f" |
@@ -5670,6 +5693,11 @@ isarray@1.0.0, isarray@^1.0.0, isarray@~1.0.0: | @@ -5670,6 +5693,11 @@ isarray@1.0.0, isarray@^1.0.0, isarray@~1.0.0: | ||
5670 | resolved "https://registry.yarnpkg.com/isarray/-/isarray-1.0.0.tgz#bb935d48582cba168c06834957a54a3e07124f11" | 5693 | resolved "https://registry.yarnpkg.com/isarray/-/isarray-1.0.0.tgz#bb935d48582cba168c06834957a54a3e07124f11" |
5671 | integrity sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE= | 5694 | integrity sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE= |
5672 | 5695 | ||
5696 | +isarray@^2.0.1: | ||
5697 | + version "2.0.5" | ||
5698 | + resolved "https://registry.yarnpkg.com/isarray/-/isarray-2.0.5.tgz#8af1e4c1221244cc62459faf38940d4e644a5723" | ||
5699 | + integrity sha512-xHjhDr3cNBK0BzdUJSPXZntQUx/mwMS5Rw4A7lPJ90XGAO6ISP/ePDNuo0vhqOZU+UD5JoodwCAAoZQd3FeAKw== | ||
5700 | + | ||
5673 | isbinaryfile@^4.0.6: | 5701 | isbinaryfile@^4.0.6: |
5674 | version "4.0.6" | 5702 | version "4.0.6" |
5675 | resolved "https://registry.yarnpkg.com/isbinaryfile/-/isbinaryfile-4.0.6.tgz#edcb62b224e2b4710830b67498c8e4e5a4d2610b" | 5703 | resolved "https://registry.yarnpkg.com/isbinaryfile/-/isbinaryfile-4.0.6.tgz#edcb62b224e2b4710830b67498c8e4e5a4d2610b" |
@@ -7503,6 +7531,11 @@ pkg-dir@^4.1.0: | @@ -7503,6 +7531,11 @@ pkg-dir@^4.1.0: | ||
7503 | dependencies: | 7531 | dependencies: |
7504 | find-up "^4.0.0" | 7532 | find-up "^4.0.0" |
7505 | 7533 | ||
7534 | +pngjs@^3.3.0: | ||
7535 | + version "3.4.0" | ||
7536 | + resolved "https://registry.yarnpkg.com/pngjs/-/pngjs-3.4.0.tgz#99ca7d725965fb655814eaf65f38f12bbdbf555f" | ||
7537 | + integrity sha512-NCrCHhWmnQklfH4MtJMRjZ2a8c80qXeMlQMv2uVp9ISJMTt562SbGd6n2oq0PaPgKm7Z6pL9E2UlLIhC+SHL3w== | ||
7538 | + | ||
7506 | pnp-webpack-plugin@1.6.4: | 7539 | pnp-webpack-plugin@1.6.4: |
7507 | version "1.6.4" | 7540 | version "1.6.4" |
7508 | resolved "https://registry.yarnpkg.com/pnp-webpack-plugin/-/pnp-webpack-plugin-1.6.4.tgz#c9711ac4dc48a685dabafc86f8b6dd9f8df84149" | 7541 | resolved "https://registry.yarnpkg.com/pnp-webpack-plugin/-/pnp-webpack-plugin-1.6.4.tgz#c9711ac4dc48a685dabafc86f8b6dd9f8df84149" |
@@ -8007,6 +8040,19 @@ qjobs@^1.2.0: | @@ -8007,6 +8040,19 @@ qjobs@^1.2.0: | ||
8007 | resolved "https://registry.yarnpkg.com/qjobs/-/qjobs-1.2.0.tgz#c45e9c61800bd087ef88d7e256423bdd49e5d071" | 8040 | resolved "https://registry.yarnpkg.com/qjobs/-/qjobs-1.2.0.tgz#c45e9c61800bd087ef88d7e256423bdd49e5d071" |
8008 | integrity sha512-8YOJEHtxpySA3fFDyCRxA+UUV+fA+rTWnuWvylOK/NCjhY+b4ocCtmu8TtsWb+mYeU+GCHf/S66KZF/AsteKHg== | 8041 | integrity sha512-8YOJEHtxpySA3fFDyCRxA+UUV+fA+rTWnuWvylOK/NCjhY+b4ocCtmu8TtsWb+mYeU+GCHf/S66KZF/AsteKHg== |
8009 | 8042 | ||
8043 | +qrcode@^1.4.4: | ||
8044 | + version "1.4.4" | ||
8045 | + resolved "https://registry.yarnpkg.com/qrcode/-/qrcode-1.4.4.tgz#f0c43568a7e7510a55efc3b88d9602f71963ea83" | ||
8046 | + integrity sha512-oLzEC5+NKFou9P0bMj5+v6Z40evexeE29Z9cummZXZ9QXyMr3lphkURzxjXgPJC5azpxcshoDWV1xE46z+/c3Q== | ||
8047 | + dependencies: | ||
8048 | + buffer "^5.4.3" | ||
8049 | + buffer-alloc "^1.2.0" | ||
8050 | + buffer-from "^1.1.1" | ||
8051 | + dijkstrajs "^1.0.1" | ||
8052 | + isarray "^2.0.1" | ||
8053 | + pngjs "^3.3.0" | ||
8054 | + yargs "^13.2.4" | ||
8055 | + | ||
8010 | qs@6.7.0: | 8056 | qs@6.7.0: |
8011 | version "6.7.0" | 8057 | version "6.7.0" |
8012 | resolved "https://registry.yarnpkg.com/qs/-/qs-6.7.0.tgz#41dc1a015e3d581f1621776be31afb2876a9b1bc" | 8058 | resolved "https://registry.yarnpkg.com/qs/-/qs-6.7.0.tgz#41dc1a015e3d581f1621776be31afb2876a9b1bc" |
@@ -10389,7 +10435,7 @@ yargs-parser@^20.2.2: | @@ -10389,7 +10435,7 @@ yargs-parser@^20.2.2: | ||
10389 | resolved "https://registry.yarnpkg.com/yargs-parser/-/yargs-parser-20.2.7.tgz#61df85c113edfb5a7a4e36eb8aa60ef423cbc90a" | 10435 | resolved "https://registry.yarnpkg.com/yargs-parser/-/yargs-parser-20.2.7.tgz#61df85c113edfb5a7a4e36eb8aa60ef423cbc90a" |
10390 | integrity sha512-FiNkvbeHzB/syOjIUxFDCnhSfzAL8R5vs40MgLFBorXACCOAEaWu0gRZl14vG8MR9AOJIZbmkjhusqBYZ3HTHw== | 10436 | integrity sha512-FiNkvbeHzB/syOjIUxFDCnhSfzAL8R5vs40MgLFBorXACCOAEaWu0gRZl14vG8MR9AOJIZbmkjhusqBYZ3HTHw== |
10391 | 10437 | ||
10392 | -yargs@^13.3.2: | 10438 | +yargs@^13.2.4, yargs@^13.3.2: |
10393 | version "13.3.2" | 10439 | version "13.3.2" |
10394 | resolved "https://registry.yarnpkg.com/yargs/-/yargs-13.3.2.tgz#ad7ffefec1aa59565ac915f82dccb38a9c31a2dd" | 10440 | resolved "https://registry.yarnpkg.com/yargs/-/yargs-13.3.2.tgz#ad7ffefec1aa59565ac915f82dccb38a9c31a2dd" |
10395 | integrity sha512-AX3Zw5iPruN5ie6xGRIDgqkT+ZhnRlZMLMHAs8tg7nRruy2Nb+i5o9bwghAogtM08q1dpr2LVoS8KSTMYpWXUw== | 10441 | integrity sha512-AX3Zw5iPruN5ie6xGRIDgqkT+ZhnRlZMLMHAs8tg7nRruy2Nb+i5o9bwghAogtM08q1dpr2LVoS8KSTMYpWXUw== |