Add new widget 'multiple-widget' to bundle 'input_widgets'
This commit is contained in:
parent
a568564ed3
commit
6091f1a8d8
@ -196,6 +196,22 @@
|
|||||||
"dataKeySettingsSchema": "{}\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\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{},\"title\":\"Update integer timeseries\",\"dropShadow\":true,\"enableFullscreen\":false,\"widgetStyle\":{},\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"actions\":{}}"
|
"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\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{},\"title\":\"Update integer timeseries\",\"dropShadow\":true,\"enableFullscreen\":false,\"widgetStyle\":{},\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"actions\":{}}"
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"alias": "update_multiple_attributes",
|
||||||
|
"name": "Update Multiple Attributes",
|
||||||
|
"descriptor": {
|
||||||
|
"type": "latest",
|
||||||
|
"sizeX": 7.5,
|
||||||
|
"sizeY": 3.5,
|
||||||
|
"resources": [],
|
||||||
|
"templateHtml": "<tb-multiple-input-widget \n form-id=\"formId\"\n ctx=\"ctx\">\n</tb-multiple-input-widget>",
|
||||||
|
"templateCss": "",
|
||||||
|
"controllerScript": "let $scope;\r\nlet settings;\r\nlet attributeService;\r\nlet toast;\r\nlet utils;\r\nlet types;\r\n\r\nself.onInit = function() {\r\n var scope = self.ctx.$scope;\r\n var id = self.ctx.$scope.$injector.get('utils').guid();\r\n scope.formId = \"form-\"+id;\r\n scope.ctx = self.ctx;\r\n}\r\n\r\nself.onDataUpdated = function() {\r\n self.ctx.$scope.$broadcast('multiple-input-data-updated', self.ctx.$scope.formId);\r\n}\r\n",
|
||||||
|
"settingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"MultipleInput\",\n \"properties\": {\n \"widgetTitle\": {\n \"title\": \"Multiple input title\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"attributesShared\": {\n \"title\": \"Attributes are 'shared' (default value is 'server')\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"showResultMessage\":{\n \"title\":\"Show result message\",\n \"type\":\"boolean\",\n \"default\":true\n }\n },\n \"required\": []\n },\n \"form\": [\n \"widgetTitle\",\n \"attributesShared\",\n \"showResultMessage\"\n ]\n}",
|
||||||
|
"dataKeySettingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"DataKeySettings\",\n \"properties\": {\n \"inputTypeNumber\": {\n \"title\": \"Datakey is a number\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"step\": {\n \"title\": \"Step interval between valid values (only for numbers)\",\n \"type\": \"number\",\n \"default\": \"1\"\n },\n \"icon\": {\n \"title\": \"Icon to show before input cell\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"useCellStyleFunction\": {\n \"title\": \"Use cell style function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"cellStyleFunction\": {\n \"title\": \"Cell style function: f(value)\",\n \"type\": \"string\",\n \"default\": \"\"\n }\n },\n \"required\": []\n },\n \"form\": [\n \"inputTypeNumber\",\n \"step\",\n\t\t{\n \t\t\"key\": \"icon\",\n\t\t\t\"type\": \"icon\"\n\t\t},\n \"useCellStyleFunction\",\n {\n \"key\": \"cellStyleFunction\",\n \"type\": \"javascript\"\n }\n ]\n}\n",
|
||||||
|
"defaultConfig": "{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Sin\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.23592248334107624,\"funcBody\":\"return Math.round(1000*Math.sin(time/5000));\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{},\"title\":\"Update Multiple Attributes\",\"dropShadow\":true,\"enableFullscreen\":false,\"enableDataExport\":false,\"widgetStyle\":{},\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false,\"actions\":{}}"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|||||||
@ -24,6 +24,7 @@ import thingsboardEntitiesTableWidget from '../widget/lib/entities-table-widget'
|
|||||||
import thingsboardEntitiesHierarchyWidget from '../widget/lib/entities-hierarchy-widget';
|
import thingsboardEntitiesHierarchyWidget from '../widget/lib/entities-hierarchy-widget';
|
||||||
import thingsboardExtensionsTableWidget from '../widget/lib/extensions-table-widget';
|
import thingsboardExtensionsTableWidget from '../widget/lib/extensions-table-widget';
|
||||||
import thingsboardDateRangeNavigatorWidget from '../widget/lib/date-range-navigator/date-range-navigator';
|
import thingsboardDateRangeNavigatorWidget from '../widget/lib/date-range-navigator/date-range-navigator';
|
||||||
|
import thingsboardMultipleInputWidget from '../widget/lib/multiple-input-widget';
|
||||||
|
|
||||||
import thingsboardRpcWidgets from '../widget/lib/rpc';
|
import thingsboardRpcWidgets from '../widget/lib/rpc';
|
||||||
|
|
||||||
@ -49,7 +50,7 @@ import thingsboardUtils from '../common/utils.service';
|
|||||||
export default angular.module('thingsboard.api.widget', ['oc.lazyLoad', thingsboardLedLight,
|
export default angular.module('thingsboard.api.widget', ['oc.lazyLoad', thingsboardLedLight,
|
||||||
thingsboardTimeseriesTableWidget, thingsboardAlarmsTableWidget, thingsboardEntitiesTableWidget,
|
thingsboardTimeseriesTableWidget, thingsboardAlarmsTableWidget, thingsboardEntitiesTableWidget,
|
||||||
thingsboardEntitiesHierarchyWidget, thingsboardExtensionsTableWidget, thingsboardDateRangeNavigatorWidget,
|
thingsboardEntitiesHierarchyWidget, thingsboardExtensionsTableWidget, thingsboardDateRangeNavigatorWidget,
|
||||||
thingsboardRpcWidgets, thingsboardTypes, thingsboardUtils, TripAnimationWidget])
|
thingsboardMultipleInputWidget, thingsboardRpcWidgets, thingsboardTypes, thingsboardUtils, TripAnimationWidget])
|
||||||
.factory('widgetService', WidgetService)
|
.factory('widgetService', WidgetService)
|
||||||
.name;
|
.name;
|
||||||
|
|
||||||
|
|||||||
@ -49,7 +49,8 @@
|
|||||||
"import": "Import",
|
"import": "Import",
|
||||||
"export": "Export",
|
"export": "Export",
|
||||||
"share-via": "Share via {{provider}}",
|
"share-via": "Share via {{provider}}",
|
||||||
"continue": "Continue"
|
"continue": "Continue",
|
||||||
|
"discard-changes": "Discard Changes"
|
||||||
},
|
},
|
||||||
"aggregation": {
|
"aggregation": {
|
||||||
"aggregation": "Aggregation",
|
"aggregation": "Aggregation",
|
||||||
@ -1694,4 +1695,4 @@
|
|||||||
"cs_CZ": "Czech"
|
"cs_CZ": "Czech"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -48,7 +48,8 @@
|
|||||||
"paste-reference": "Pegar referencia",
|
"paste-reference": "Pegar referencia",
|
||||||
"import": "Importar",
|
"import": "Importar",
|
||||||
"export": "Exportar",
|
"export": "Exportar",
|
||||||
"share-via": "Compartir via {{provider}}"
|
"share-via": "Compartir via {{provider}}",
|
||||||
|
"discard-changes": "Cancelar los cambios"
|
||||||
},
|
},
|
||||||
"aggregation": {
|
"aggregation": {
|
||||||
"aggregation": "Agregación",
|
"aggregation": "Agregación",
|
||||||
|
|||||||
@ -48,7 +48,8 @@
|
|||||||
"undo": "Annuler",
|
"undo": "Annuler",
|
||||||
"update": "mise à jour",
|
"update": "mise à jour",
|
||||||
"view": "Afficher",
|
"view": "Afficher",
|
||||||
"yes": "Oui"
|
"yes": "Oui",
|
||||||
|
"discard-changes": "Annuler les modifications"
|
||||||
},
|
},
|
||||||
"admin": {
|
"admin": {
|
||||||
"base-url": "URL de base",
|
"base-url": "URL de base",
|
||||||
|
|||||||
@ -48,7 +48,8 @@
|
|||||||
"paste-reference": "Incolla riferimento",
|
"paste-reference": "Incolla riferimento",
|
||||||
"import": "Importa",
|
"import": "Importa",
|
||||||
"export": "Esporta",
|
"export": "Esporta",
|
||||||
"share-via": "Condividi con {{provider}}"
|
"share-via": "Condividi con {{provider}}",
|
||||||
|
"discard-changes": "Annulla le modifiche"
|
||||||
},
|
},
|
||||||
"aggregation": {
|
"aggregation": {
|
||||||
"aggregation": "Aggregazione",
|
"aggregation": "Aggregazione",
|
||||||
|
|||||||
259
ui/src/app/widget/lib/multiple-input-widget.js
Normal file
259
ui/src/app/widget/lib/multiple-input-widget.js
Normal file
@ -0,0 +1,259 @@
|
|||||||
|
/*
|
||||||
|
* Copyright © 2016-2019 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 './multiple-input-widget.scss';
|
||||||
|
|
||||||
|
/* eslint-disable import/no-unresolved, import/default */
|
||||||
|
|
||||||
|
import multipleInputWidgetTemplate from './multiple-input-widget.tpl.html';
|
||||||
|
|
||||||
|
/* eslint-enable import/no-unresolved, import/default */
|
||||||
|
|
||||||
|
export default angular.module('thingsboard.widgets.multipleInputWidget', [])
|
||||||
|
.directive('tbMultipleInputWidget', MultipleInputWidget)
|
||||||
|
.name;
|
||||||
|
|
||||||
|
/*@ngInject*/
|
||||||
|
function MultipleInputWidget() {
|
||||||
|
return {
|
||||||
|
restrict: "E",
|
||||||
|
scope: true,
|
||||||
|
bindToController: {
|
||||||
|
formId: '=',
|
||||||
|
ctx: '='
|
||||||
|
},
|
||||||
|
controller: MultipleInputWidgetController,
|
||||||
|
controllerAs: 'vm',
|
||||||
|
templateUrl: multipleInputWidgetTemplate
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/*@ngInject*/
|
||||||
|
function MultipleInputWidgetController($q, $scope, attributeService, toast, types, utils) {
|
||||||
|
var vm = this;
|
||||||
|
|
||||||
|
vm.dataKeyDetected = false;
|
||||||
|
vm.hasAnyChange = false;
|
||||||
|
vm.entityDetected = false;
|
||||||
|
vm.isValidParameter = true;
|
||||||
|
vm.message = 'No entity selected';
|
||||||
|
|
||||||
|
vm.rows = [];
|
||||||
|
vm.rowIndex = 0;
|
||||||
|
|
||||||
|
vm.datasources = null;
|
||||||
|
|
||||||
|
vm.cellStyle = cellStyle;
|
||||||
|
vm.discardAll = discardAll;
|
||||||
|
vm.inputChanged = inputChanged;
|
||||||
|
vm.postData = postData;
|
||||||
|
|
||||||
|
$scope.$watch('vm.ctx', function() {
|
||||||
|
if (vm.ctx && vm.ctx.defaultSubscription) {
|
||||||
|
vm.settings = vm.ctx.settings;
|
||||||
|
vm.widgetConfig = vm.ctx.widgetConfig;
|
||||||
|
vm.subscription = vm.ctx.defaultSubscription;
|
||||||
|
vm.datasources = vm.subscription.datasources;
|
||||||
|
initializeConfig();
|
||||||
|
updateDatasources();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
$scope.$on('multiple-input-data-updated', function(event, formId) {
|
||||||
|
if (vm.formId == formId) {
|
||||||
|
updateRowData(vm.subscription.data);
|
||||||
|
$scope.$digest();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
function defaultStyle() {
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
function cellStyle(key) {
|
||||||
|
var style = {};
|
||||||
|
if (key) {
|
||||||
|
var styleInfo = vm.stylesInfo[key.label];
|
||||||
|
var value = key.currentValue;
|
||||||
|
if (styleInfo.useCellStyleFunction && styleInfo.cellStyleFunction) {
|
||||||
|
try {
|
||||||
|
style = styleInfo.cellStyleFunction(value);
|
||||||
|
} catch (e) {
|
||||||
|
style = {};
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
style = defaultStyle();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return style;
|
||||||
|
}
|
||||||
|
|
||||||
|
function discardAll() {
|
||||||
|
for (var r = 0; r < vm.rows.length; r++) {
|
||||||
|
var row = vm.rows[r];
|
||||||
|
for (var d = 0; d < row.data.length; d++ ) {
|
||||||
|
row.data[d].currentValue = row.data[d].originalValue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
vm.hasAnyChange = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
function inputChanged() {
|
||||||
|
var newValue = false;
|
||||||
|
for (var r = 0; r < vm.rows.length; r++) {
|
||||||
|
var row = vm.rows[r];
|
||||||
|
for (var d = 0; d < row.data.length; d++ ) {
|
||||||
|
if (!row.data[d].currentValue) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (row.data[d].currentValue !== row.data[d].originalValue) {
|
||||||
|
newValue = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
vm.hasAnyChange = newValue;
|
||||||
|
}
|
||||||
|
|
||||||
|
function postData() {
|
||||||
|
var promises = [];
|
||||||
|
for (var r = 0; r < vm.rows.length; r++) {
|
||||||
|
var row = vm.rows[r];
|
||||||
|
var datasource = row.datasource;
|
||||||
|
var attributes = [];
|
||||||
|
var newValues = false;
|
||||||
|
|
||||||
|
for (var d = 0; d < row.data.length; d++ ) {
|
||||||
|
if (row.data[d].currentValue !== row.data[d].originalValue) {
|
||||||
|
attributes.push({
|
||||||
|
key : row.data[d].name,
|
||||||
|
value : row.data[d].currentValue,
|
||||||
|
});
|
||||||
|
newValues = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (newValues) {
|
||||||
|
promises.push(attributeService.saveEntityAttributes(
|
||||||
|
datasource.entityType,
|
||||||
|
datasource.entityId,
|
||||||
|
vm.attributeScope,
|
||||||
|
attributes));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (promises.length) {
|
||||||
|
$q.all(promises).then(
|
||||||
|
function success() {
|
||||||
|
for (var d = 0; d < row.data.length; d++ ) {
|
||||||
|
row.data[d].originalValue = row.data[d].currentValue;
|
||||||
|
}
|
||||||
|
vm.hasAnyChange = false;
|
||||||
|
if (vm.settings.showResultMessage) {
|
||||||
|
toast.showSuccess('Update successful', 1000, angular.element(vm.ctx.$container), 'bottom left');
|
||||||
|
}
|
||||||
|
},
|
||||||
|
function fail() {
|
||||||
|
if (vm.settings.showResultMessage) {
|
||||||
|
toast.showError('Update failed', angular.element(vm.ctx.$container), 'bottom left');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function initializeConfig() {
|
||||||
|
|
||||||
|
if (vm.settings.widgetTitle && vm.settings.widgetTitle.length) {
|
||||||
|
vm.widgetTitle = utils.customTranslation(vm.settings.widgetTitle, vm.settings.widgetTitle);
|
||||||
|
} else {
|
||||||
|
vm.widgetTitle = vm.ctx.widgetConfig.title;
|
||||||
|
}
|
||||||
|
|
||||||
|
vm.ctx.widgetTitle = vm.widgetTitle;
|
||||||
|
|
||||||
|
vm.attributeScope = vm.settings.attributesShared ? types.attributesScope.shared.value : types.attributesScope.server.value;
|
||||||
|
}
|
||||||
|
|
||||||
|
function updateDatasources() {
|
||||||
|
|
||||||
|
vm.stylesInfo = {};
|
||||||
|
vm.rows = [];
|
||||||
|
vm.rowIndex = 0;
|
||||||
|
|
||||||
|
if (vm.datasources) {
|
||||||
|
vm.entityDetected = true;
|
||||||
|
for (var ds = 0; ds < vm.datasources.length; ds++) {
|
||||||
|
var row = {};
|
||||||
|
var datasource = vm.datasources[ds];
|
||||||
|
row.datasource = datasource;
|
||||||
|
row.data = [];
|
||||||
|
if (datasource.dataKeys) {
|
||||||
|
vm.dataKeyDetected = true;
|
||||||
|
for (var a = 0; a < datasource.dataKeys.length; a++ ) {
|
||||||
|
var dataKey = datasource.dataKeys[a];
|
||||||
|
|
||||||
|
if (dataKey.units) {
|
||||||
|
dataKey.label += ' (' + dataKey.units + ')';
|
||||||
|
}
|
||||||
|
|
||||||
|
var keySettings = dataKey.settings;
|
||||||
|
if (keySettings.inputTypeNumber) {
|
||||||
|
keySettings.inputType = 'number';
|
||||||
|
} else {
|
||||||
|
keySettings.inputType = 'text';
|
||||||
|
}
|
||||||
|
|
||||||
|
var cellStyleFunction = null;
|
||||||
|
var useCellStyleFunction = false;
|
||||||
|
|
||||||
|
if (keySettings.useCellStyleFunction === true) {
|
||||||
|
if (angular.isDefined(keySettings.cellStyleFunction) && keySettings.cellStyleFunction.length > 0) {
|
||||||
|
try {
|
||||||
|
cellStyleFunction = new Function('value', keySettings.cellStyleFunction);
|
||||||
|
useCellStyleFunction = true;
|
||||||
|
} catch (e) {
|
||||||
|
cellStyleFunction = null;
|
||||||
|
useCellStyleFunction = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
vm.stylesInfo[dataKey.label] = {
|
||||||
|
useCellStyleFunction: useCellStyleFunction,
|
||||||
|
cellStyleFunction: cellStyleFunction
|
||||||
|
};
|
||||||
|
|
||||||
|
row.data.push(dataKey);
|
||||||
|
}
|
||||||
|
vm.rows.push(row);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function updateRowData(data) {
|
||||||
|
var dataIndex = 0;
|
||||||
|
for (var r = 0; r < vm.rows.length; r++) {
|
||||||
|
var row = vm.rows[r];
|
||||||
|
for (var d = 0; d < row.data.length; d++ ) {
|
||||||
|
var keyData = data[dataIndex++].data;
|
||||||
|
if (keyData && keyData.length && keyData[0].length > 1) {
|
||||||
|
row.data[d].currentValue = row.data[d].originalValue = keyData[0][1];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
55
ui/src/app/widget/lib/multiple-input-widget.scss
Normal file
55
ui/src/app/widget/lib/multiple-input-widget.scss
Normal file
@ -0,0 +1,55 @@
|
|||||||
|
/**
|
||||||
|
* Copyright © 2016-2019 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-multiple-input {
|
||||||
|
height: 100%;
|
||||||
|
|
||||||
|
.md-button.md-icon-button {
|
||||||
|
width: 32px;
|
||||||
|
min-width: 32px;
|
||||||
|
height: 32px;
|
||||||
|
min-height: 32px;
|
||||||
|
padding: 0 !important;
|
||||||
|
margin: 0;
|
||||||
|
line-height: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.md-icon-button md-icon {
|
||||||
|
width: 20px;
|
||||||
|
min-width: 20px;
|
||||||
|
height: 20px;
|
||||||
|
min-height: 20px;
|
||||||
|
font-size: 20px;
|
||||||
|
|
||||||
|
&:not([disabled]) {
|
||||||
|
color: #f66;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
md-toast {
|
||||||
|
min-width: 0;
|
||||||
|
|
||||||
|
.md-toast-content {
|
||||||
|
font-size: 14px !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.footer {
|
||||||
|
position: absolute;
|
||||||
|
bottom: 0;
|
||||||
|
left: 0;
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
55
ui/src/app/widget/lib/multiple-input-widget.tpl.html
Normal file
55
ui/src/app/widget/lib/multiple-input-widget.tpl.html
Normal file
@ -0,0 +1,55 @@
|
|||||||
|
<!--
|
||||||
|
|
||||||
|
Copyright © 2016-2019 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.
|
||||||
|
|
||||||
|
-->
|
||||||
|
<form class="tb-multiple-input" name="multipleInputForm" ng-submit="vm.postData($event)">
|
||||||
|
<div style="padding: 0 8px; margin: auto 0;">
|
||||||
|
<div ng-show="vm.entityDetected" layout="row" flex ng-repeat="row in vm.rows track by $index">
|
||||||
|
<div layout="column" flex ng-repeat="key in row.data track by $index">
|
||||||
|
<md-input-container class="md-icon-float" ng-style="vm.cellStyle(key)">
|
||||||
|
<label>{{key.label}}</label>
|
||||||
|
<md-icon class="material-icons" ng-if="key.settings.icon">
|
||||||
|
{{key.settings.icon}}
|
||||||
|
</md-icon>
|
||||||
|
<input name="key.name"
|
||||||
|
ng-model="key.currentValue"
|
||||||
|
type="{{key.settings.inputType}}"
|
||||||
|
step="{{key.settings.step}}"
|
||||||
|
md-select-on-focus
|
||||||
|
ng-change="vm.inputChanged()">
|
||||||
|
</md-input-container>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div style="text-align: center; font-size: 18px; color: #a0a0a0;" ng-hide="vm.entityDetected" ng-bind="vm.message"
|
||||||
|
></div>
|
||||||
|
<div style="text-align: center; font-size: 18px; color: #a0a0a0;" ng-show="vm.entityDetected && !vm.dataKeyDetected">
|
||||||
|
No attribute is selected
|
||||||
|
</div>
|
||||||
|
<div style="text-align: center; font-size: 18px; color: #a0a0a0;" ng-show="vm.entityDetected && !vm.isValidParameter">
|
||||||
|
Timeseries parameter cannot be used in this widget
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="footer md-padding" layout="row" layout-align="end center" ng-show="vm.entityDetected && vm.dataKeyDetected">
|
||||||
|
<md-button class="md-primary" ng-click="vm.discardAll()" style="max-height: 50px;margin-right:20px;" ng-disabled="!vm.hasAnyChange">
|
||||||
|
{{ 'action.discard-changes' | translate }}
|
||||||
|
</md-button>
|
||||||
|
<md-button class="md-raised md-primary" type="submit" value="Submit" style="max-height: 50px;margin-right:20px;"
|
||||||
|
ng-disabled="!vm.hasAnyChange" ng-click="vm.isFocused = false">
|
||||||
|
{{ 'action.save' | translate }}
|
||||||
|
</md-button>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
Loading…
x
Reference in New Issue
Block a user