Commit 8c9a6c3e4742dff95e751b91bb17247738bdcf3f

Authored by Andrew Shvayka
2 parents 2590f04a 0f14a36c

Merge with master

Showing 73 changed files with 2793 additions and 463 deletions

Too many changes to show.

To preserve performance only 73 of 82 files are displayed.

  1 +--
  2 +-- Copyright © 2016-2018 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 +DROP MATERIALIZED VIEW IF EXISTS thingsboard.entity_view_by_tenant_and_name;
  18 +DROP MATERIALIZED VIEW IF EXISTS thingsboard.entity_view_by_tenant_and_search_text;
  19 +DROP MATERIALIZED VIEW IF EXISTS thingsboard.entity_view_by_tenant_and_customer;
  20 +DROP MATERIALIZED VIEW IF EXISTS thingsboard.entity_view_by_tenant_and_entity_id;
  21 +
  22 +DROP TABLE IF EXISTS thingsboard.entity_views;
  23 +
  24 +CREATE TABLE IF NOT EXISTS thingsboard.entity_view (
  25 + id timeuuid,
  26 + entity_id timeuuid,
  27 + entity_type text,
  28 + tenant_id timeuuid,
  29 + customer_id timeuuid,
  30 + name text,
  31 + type text,
  32 + keys text,
  33 + start_ts bigint,
  34 + end_ts bigint,
  35 + search_text text,
  36 + additional_info text,
  37 + PRIMARY KEY (id, entity_id, tenant_id, customer_id, type)
  38 +);
  39 +
  40 +CREATE MATERIALIZED VIEW IF NOT EXISTS thingsboard.entity_view_by_tenant_and_name AS
  41 + SELECT *
  42 + from thingsboard.entity_view
  43 + WHERE tenant_id IS NOT NULL
  44 + AND entity_id IS NOT NULL
  45 + AND customer_id IS NOT NULL
  46 + AND type IS NOT NULL
  47 + AND name IS NOT NULL
  48 + AND id IS NOT NULL
  49 + PRIMARY KEY (tenant_id, name, id, customer_id, entity_id, type)
  50 + WITH CLUSTERING ORDER BY (name ASC, id DESC, customer_id DESC);
  51 +
  52 +CREATE MATERIALIZED VIEW IF NOT EXISTS thingsboard.entity_view_by_tenant_and_search_text AS
  53 + SELECT *
  54 + from thingsboard.entity_view
  55 + WHERE tenant_id IS NOT NULL
  56 + AND entity_id IS NOT NULL
  57 + AND customer_id IS NOT NULL
  58 + AND type IS NOT NULL
  59 + AND search_text IS NOT NULL
  60 + AND id IS NOT NULL
  61 + PRIMARY KEY (tenant_id, search_text, id, customer_id, entity_id, type)
  62 + WITH CLUSTERING ORDER BY (search_text ASC, id DESC, customer_id DESC);
  63 +
  64 +CREATE MATERIALIZED VIEW IF NOT EXISTS thingsboard.entity_view_by_tenant_by_type_and_search_text AS
  65 + SELECT *
  66 + from thingsboard.entity_view
  67 + WHERE tenant_id IS NOT NULL
  68 + AND entity_id IS NOT NULL
  69 + AND customer_id IS NOT NULL
  70 + AND type IS NOT NULL
  71 + AND search_text IS NOT NULL
  72 + AND id IS NOT NULL
  73 + PRIMARY KEY (tenant_id, type, search_text, id, customer_id, entity_id)
  74 + WITH CLUSTERING ORDER BY (type ASC, search_text ASC, id DESC, customer_id DESC);
  75 +
  76 +CREATE MATERIALIZED VIEW IF NOT EXISTS thingsboard.entity_view_by_tenant_and_customer AS
  77 + SELECT *
  78 + from thingsboard.entity_view
  79 + WHERE tenant_id IS NOT NULL
  80 + AND customer_id IS NOT NULL
  81 + AND entity_id IS NOT NULL
  82 + AND type IS NOT NULL
  83 + AND search_text IS NOT NULL
  84 + AND id IS NOT NULL
  85 + PRIMARY KEY (tenant_id, customer_id, search_text, id, entity_id, type)
  86 + WITH CLUSTERING ORDER BY (customer_id DESC, search_text ASC, id DESC);
  87 +
  88 +CREATE MATERIALIZED VIEW IF NOT EXISTS thingsboard.entity_view_by_tenant_and_customer_and_type AS
  89 + SELECT *
  90 + from thingsboard.entity_view
  91 + WHERE tenant_id IS NOT NULL
  92 + AND customer_id IS NOT NULL
  93 + AND entity_id IS NOT NULL
  94 + AND type IS NOT NULL
  95 + AND search_text IS NOT NULL
  96 + AND id IS NOT NULL
  97 + PRIMARY KEY (tenant_id, type, customer_id, search_text, id, entity_id)
  98 + WITH CLUSTERING ORDER BY (type ASC, customer_id DESC, search_text ASC, id DESC);
  99 +
  100 +CREATE MATERIALIZED VIEW IF NOT EXISTS thingsboard.entity_view_by_tenant_and_entity_id AS
  101 + SELECT *
  102 + from thingsboard.entity_view
  103 + WHERE tenant_id IS NOT NULL
  104 + AND customer_id IS NOT NULL
  105 + AND entity_id IS NOT NULL
  106 + AND type IS NOT NULL
  107 + AND search_text IS NOT NULL
  108 + AND id IS NOT NULL
  109 + PRIMARY KEY (tenant_id, entity_id, customer_id, search_text, id, type)
  110 + WITH CLUSTERING ORDER BY (entity_id DESC, customer_id DESC, search_text ASC, id DESC);
\ No newline at end of file
... ...
  1 +--
  2 +-- Copyright © 2016-2018 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 +DROP TABLE IF EXISTS entity_views;
  18 +
  19 +CREATE TABLE IF NOT EXISTS entity_view (
  20 + id varchar(31) NOT NULL CONSTRAINT entity_view_pkey PRIMARY KEY,
  21 + entity_id varchar(31),
  22 + entity_type varchar(255),
  23 + tenant_id varchar(31),
  24 + customer_id varchar(31),
  25 + type varchar(255),
  26 + name varchar(255),
  27 + keys varchar(255),
  28 + start_ts bigint,
  29 + end_ts bigint,
  30 + search_text varchar(255),
  31 + additional_info varchar
  32 +);
... ...
... ... @@ -15,6 +15,8 @@
15 15 */
16 16 package org.thingsboard.server.controller;
17 17
  18 +import lombok.Getter;
  19 +import org.springframework.beans.factory.annotation.Value;
18 20 import org.springframework.http.HttpStatus;
19 21 import org.springframework.security.access.prepost.PreAuthorize;
20 22 import org.springframework.web.bind.annotation.PathVariable;
... ... @@ -49,6 +51,11 @@ public class DashboardController extends BaseController {
49 51
50 52 public static final String DASHBOARD_ID = "dashboardId";
51 53
  54 + @Value("${dashboard.max_datapoints_limit}")
  55 + @Getter
  56 + private long maxDatapointsLimit;
  57 +
  58 +
52 59 @PreAuthorize("hasAnyAuthority('SYS_ADMIN', 'TENANT_ADMIN', 'CUSTOMER_USER')")
53 60 @RequestMapping(value = "/dashboard/serverTime", method = RequestMethod.GET)
54 61 @ResponseBody
... ... @@ -57,6 +64,13 @@ public class DashboardController extends BaseController {
57 64 }
58 65
59 66 @PreAuthorize("hasAnyAuthority('SYS_ADMIN', 'TENANT_ADMIN', 'CUSTOMER_USER')")
  67 + @RequestMapping(value = "/dashboard/maxDatapointsLimit", method = RequestMethod.GET)
  68 + @ResponseBody
  69 + public long getMaxDatapointsLimit() throws ThingsboardException {
  70 + return maxDatapointsLimit;
  71 + }
  72 +
  73 + @PreAuthorize("hasAnyAuthority('SYS_ADMIN', 'TENANT_ADMIN', 'CUSTOMER_USER')")
60 74 @RequestMapping(value = "/dashboard/info/{dashboardId}", method = RequestMethod.GET)
61 75 @ResponseBody
62 76 public DashboardInfo getDashboardInfoById(@PathVariable(DASHBOARD_ID) String strDashboardId) throws ThingsboardException {
... ...
... ... @@ -15,6 +15,7 @@
15 15 */
16 16 package org.thingsboard.server.controller;
17 17
  18 +import com.google.common.util.concurrent.ListenableFuture;
18 19 import org.springframework.http.HttpStatus;
19 20 import org.springframework.security.access.prepost.PreAuthorize;
20 21 import org.springframework.web.bind.annotation.PathVariable;
... ... @@ -26,6 +27,7 @@ import org.springframework.web.bind.annotation.ResponseBody;
26 27 import org.springframework.web.bind.annotation.ResponseStatus;
27 28 import org.springframework.web.bind.annotation.RestController;
28 29 import org.thingsboard.server.common.data.Customer;
  30 +import org.thingsboard.server.common.data.EntitySubtype;
29 31 import org.thingsboard.server.common.data.EntityType;
30 32 import org.thingsboard.server.common.data.EntityView;
31 33 import org.thingsboard.server.common.data.audit.ActionType;
... ... @@ -38,6 +40,7 @@ import org.thingsboard.server.common.data.page.TextPageData;
38 40 import org.thingsboard.server.common.data.page.TextPageLink;
39 41 import org.thingsboard.server.dao.exception.IncorrectParameterException;
40 42 import org.thingsboard.server.dao.model.ModelConstants;
  43 +import org.thingsboard.server.service.security.model.SecurityUser;
41 44
42 45 import java.util.List;
43 46 import java.util.stream.Collectors;
... ... @@ -161,6 +164,7 @@ public class EntityViewController extends BaseController {
161 164 public TextPageData<EntityView> getCustomerEntityViews(
162 165 @PathVariable("customerId") String strCustomerId,
163 166 @RequestParam int limit,
  167 + @RequestParam(required = false) String type,
164 168 @RequestParam(required = false) String textSearch,
165 169 @RequestParam(required = false) String idOffset,
166 170 @RequestParam(required = false) String textOffset) throws ThingsboardException {
... ... @@ -170,7 +174,11 @@ public class EntityViewController extends BaseController {
170 174 CustomerId customerId = new CustomerId(toUUID(strCustomerId));
171 175 checkCustomerId(customerId);
172 176 TextPageLink pageLink = createPageLink(limit, textSearch, idOffset, textOffset);
173   - return checkNotNull(entityViewService.findEntityViewsByTenantIdAndCustomerId(tenantId, customerId, pageLink));
  177 + if (type != null && type.trim().length() > 0) {
  178 + return checkNotNull(entityViewService.findEntityViewsByTenantIdAndCustomerIdAndType(tenantId, customerId, pageLink, type));
  179 + } else {
  180 + return checkNotNull(entityViewService.findEntityViewsByTenantIdAndCustomerId(tenantId, customerId, pageLink));
  181 + }
174 182 } catch (Exception e) {
175 183 throw handleException(e);
176 184 }
... ... @@ -181,13 +189,19 @@ public class EntityViewController extends BaseController {
181 189 @ResponseBody
182 190 public TextPageData<EntityView> getTenantEntityViews(
183 191 @RequestParam int limit,
  192 + @RequestParam(required = false) String type,
184 193 @RequestParam(required = false) String textSearch,
185 194 @RequestParam(required = false) String idOffset,
186 195 @RequestParam(required = false) String textOffset) throws ThingsboardException {
187 196 try {
188 197 TenantId tenantId = getCurrentUser().getTenantId();
189 198 TextPageLink pageLink = createPageLink(limit, textSearch, idOffset, textOffset);
190   - return checkNotNull(entityViewService.findEntityViewByTenantId(tenantId, pageLink));
  199 +
  200 + if (type != null && type.trim().length() > 0) {
  201 + return checkNotNull(entityViewService.findEntityViewByTenantIdAndType(tenantId, pageLink, type));
  202 + } else {
  203 + return checkNotNull(entityViewService.findEntityViewByTenantId(tenantId, pageLink));
  204 + }
191 205 } catch (Exception e) {
192 206 throw handleException(e);
193 207 }
... ... @@ -199,6 +213,7 @@ public class EntityViewController extends BaseController {
199 213 public List<EntityView> findByQuery(@RequestBody EntityViewSearchQuery query) throws ThingsboardException {
200 214 checkNotNull(query);
201 215 checkNotNull(query.getParameters());
  216 + checkNotNull(query.getEntityViewTypes());
202 217 checkEntityId(query.getParameters().getEntityId());
203 218 try {
204 219 List<EntityView> entityViews = checkNotNull(entityViewService.findEntityViewsByQuery(query).get());
... ... @@ -215,4 +230,18 @@ public class EntityViewController extends BaseController {
215 230 throw handleException(e);
216 231 }
217 232 }
  233 +
  234 + @PreAuthorize("hasAnyAuthority('TENANT_ADMIN', 'CUSTOMER_USER')")
  235 + @RequestMapping(value = "/entityView/types", method = RequestMethod.GET)
  236 + @ResponseBody
  237 + public List<EntitySubtype> getEntityViewTypes() throws ThingsboardException {
  238 + try {
  239 + SecurityUser user = getCurrentUser();
  240 + TenantId tenantId = user.getTenantId();
  241 + ListenableFuture<List<EntitySubtype>> entityViewTypes = entityViewService.findEntityViewTypesByTenantId(tenantId);
  242 + return checkNotNull(entityViewTypes.get());
  243 + } catch (Exception e) {
  244 + throw handleException(e);
  245 + }
  246 + }
218 247 }
... ...
... ... @@ -97,6 +97,11 @@ public class ThingsboardInstallService {
97 97
98 98 databaseUpgradeService.upgradeDatabase("2.0.0");
99 99
  100 + case "2.1.1":
  101 + log.info("Upgrading ThingsBoard from version 2.1.1 to 2.1.2 ...");
  102 +
  103 + databaseUpgradeService.upgradeDatabase("2.1.1");
  104 +
100 105 log.info("Updating system data...");
101 106
102 107 systemDataLoaderService.deleteSystemWidgetBundle("charts");
... ...
... ... @@ -39,10 +39,19 @@ import static org.thingsboard.server.service.install.DatabaseHelper.CONFIGURATIO
39 39 import static org.thingsboard.server.service.install.DatabaseHelper.CUSTOMER_ID;
40 40 import static org.thingsboard.server.service.install.DatabaseHelper.DASHBOARD;
41 41 import static org.thingsboard.server.service.install.DatabaseHelper.DEVICE;
  42 +import static org.thingsboard.server.service.install.DatabaseHelper.END_TS;
  43 +import static org.thingsboard.server.service.install.DatabaseHelper.ENTITY_ID;
  44 +import static org.thingsboard.server.service.install.DatabaseHelper.ENTITY_TYPE;
  45 +import static org.thingsboard.server.service.install.DatabaseHelper.ENTITY_VIEW;
  46 +import static org.thingsboard.server.service.install.DatabaseHelper.ENTITY_VIEWS;
42 47 import static org.thingsboard.server.service.install.DatabaseHelper.ID;
  48 +import static org.thingsboard.server.service.install.DatabaseHelper.KEYS;
  49 +import static org.thingsboard.server.service.install.DatabaseHelper.NAME;
43 50 import static org.thingsboard.server.service.install.DatabaseHelper.SEARCH_TEXT;
  51 +import static org.thingsboard.server.service.install.DatabaseHelper.START_TS;
44 52 import static org.thingsboard.server.service.install.DatabaseHelper.TENANT_ID;
45 53 import static org.thingsboard.server.service.install.DatabaseHelper.TITLE;
  54 +import static org.thingsboard.server.service.install.DatabaseHelper.TYPE;
46 55
47 56 @Service
48 57 @NoSqlDao
... ... @@ -203,6 +212,46 @@ public class CassandraDatabaseUpgradeService implements DatabaseUpgradeService {
203 212 log.info("Schema updated.");
204 213
205 214 break;
  215 +
  216 + case "2.0.0":
  217 +
  218 + log.info("Updating schema ...");
  219 + schemaUpdateFile = Paths.get(installScripts.getDataDir(), "upgrade", "2.1.1", SCHEMA_UPDATE_CQL);
  220 + loadCql(schemaUpdateFile);
  221 + log.info("Schema updated.");
  222 +
  223 + break;
  224 +
  225 + case "2.1.1":
  226 +
  227 + log.info("Upgrading Cassandra DataBase from version {} to 2.1.2 ...", fromVersion);
  228 +
  229 + cluster.getSession();
  230 +
  231 + ks = cluster.getCluster().getMetadata().getKeyspace(cluster.getKeyspaceName());
  232 +
  233 + log.info("Dumping entity views ...");
  234 + Path entityViewsDump = CassandraDbHelper.dumpCfIfExists(ks, cluster.getSession(), ENTITY_VIEWS,
  235 + new String[]{ID, ENTITY_ID, ENTITY_TYPE, TENANT_ID, CUSTOMER_ID, NAME, TYPE, KEYS, START_TS, END_TS, SEARCH_TEXT, ADDITIONAL_INFO},
  236 + new String[]{"", "", "", "", "", "", "default", "", "0", "0", "", ""},
  237 + "tb-entity-views");
  238 + log.info("Entity views dumped.");
  239 +
  240 + log.info("Updating schema ...");
  241 + schemaUpdateFile = Paths.get(installScripts.getDataDir(), "upgrade", "2.1.2", SCHEMA_UPDATE_CQL);
  242 + loadCql(schemaUpdateFile);
  243 + log.info("Schema updated.");
  244 +
  245 + log.info("Restoring entity views ...");
  246 + if (entityViewsDump != null) {
  247 + CassandraDbHelper.loadCf(ks, cluster.getSession(), ENTITY_VIEW,
  248 + new String[]{ID, ENTITY_ID, ENTITY_TYPE, TENANT_ID, CUSTOMER_ID, NAME, TYPE, KEYS, START_TS, END_TS, SEARCH_TEXT, ADDITIONAL_INFO}, entityViewsDump);
  249 + Files.deleteIfExists(entityViewsDump);
  250 + }
  251 + log.info("Entity views restored.");
  252 +
  253 + break;
  254 +
206 255 default:
207 256 throw new RuntimeException("Unable to upgrade Cassandra database, unsupported fromVersion: " + fromVersion);
208 257 }
... ...
... ... @@ -45,14 +45,23 @@ public class DatabaseHelper {
45 45 public static final CSVFormat CSV_DUMP_FORMAT = CSVFormat.DEFAULT.withNullString("\\N");
46 46
47 47 public static final String DEVICE = "device";
  48 + public static final String ENTITY_ID = "entity_id";
48 49 public static final String TENANT_ID = "tenant_id";
  50 + public static final String ENTITY_TYPE = "entity_type";
49 51 public static final String CUSTOMER_ID = "customer_id";
50 52 public static final String SEARCH_TEXT = "search_text";
51 53 public static final String ADDITIONAL_INFO = "additional_info";
52 54 public static final String ASSET = "asset";
53 55 public static final String DASHBOARD = "dashboard";
  56 + public static final String ENTITY_VIEWS = "entity_views";
  57 + public static final String ENTITY_VIEW = "entity_view";
54 58 public static final String ID = "id";
55 59 public static final String TITLE = "title";
  60 + public static final String TYPE = "type";
  61 + public static final String NAME = "name";
  62 + public static final String KEYS = "keys";
  63 + public static final String START_TS = "start_ts";
  64 + public static final String END_TS = "end_ts";
56 65 public static final String ASSIGNED_CUSTOMERS = "assigned_customers";
57 66 public static final String CONFIGURATION = "configuration";
58 67
... ...
... ... @@ -31,14 +31,24 @@ import java.nio.file.Paths;
31 31 import java.sql.Connection;
32 32 import java.sql.DriverManager;
33 33
  34 +import static org.thingsboard.server.service.install.DatabaseHelper.ADDITIONAL_INFO;
34 35 import static org.thingsboard.server.service.install.DatabaseHelper.ASSIGNED_CUSTOMERS;
35 36 import static org.thingsboard.server.service.install.DatabaseHelper.CONFIGURATION;
36 37 import static org.thingsboard.server.service.install.DatabaseHelper.CUSTOMER_ID;
37 38 import static org.thingsboard.server.service.install.DatabaseHelper.DASHBOARD;
  39 +import static org.thingsboard.server.service.install.DatabaseHelper.END_TS;
  40 +import static org.thingsboard.server.service.install.DatabaseHelper.ENTITY_ID;
  41 +import static org.thingsboard.server.service.install.DatabaseHelper.ENTITY_TYPE;
  42 +import static org.thingsboard.server.service.install.DatabaseHelper.ENTITY_VIEW;
  43 +import static org.thingsboard.server.service.install.DatabaseHelper.ENTITY_VIEWS;
38 44 import static org.thingsboard.server.service.install.DatabaseHelper.ID;
  45 +import static org.thingsboard.server.service.install.DatabaseHelper.KEYS;
  46 +import static org.thingsboard.server.service.install.DatabaseHelper.NAME;
39 47 import static org.thingsboard.server.service.install.DatabaseHelper.SEARCH_TEXT;
  48 +import static org.thingsboard.server.service.install.DatabaseHelper.START_TS;
40 49 import static org.thingsboard.server.service.install.DatabaseHelper.TENANT_ID;
41 50 import static org.thingsboard.server.service.install.DatabaseHelper.TITLE;
  51 +import static org.thingsboard.server.service.install.DatabaseHelper.TYPE;
42 52
43 53 @Service
44 54 @Profile("install")
... ... @@ -115,6 +125,30 @@ public class SqlDatabaseUpgradeService implements DatabaseUpgradeService {
115 125 log.info("Schema updated.");
116 126 }
117 127 break;
  128 + case "2.1.1":
  129 + try (Connection conn = DriverManager.getConnection(dbUrl, dbUserName, dbPassword)) {
  130 +
  131 + log.info("Dumping entity views ...");
  132 + Path entityViewsDump = SqlDbHelper.dumpTableIfExists(conn, ENTITY_VIEWS,
  133 + new String[]{ID, ENTITY_ID, ENTITY_TYPE, TENANT_ID, CUSTOMER_ID, TYPE, NAME, KEYS, START_TS, END_TS, SEARCH_TEXT, ADDITIONAL_INFO},
  134 + new String[]{"", "", "", "", "", "default", "", "", "0", "0", "", ""},
  135 + "tb-entity-views", true);
  136 + log.info("Entity views dumped.");
  137 +
  138 + log.info("Updating schema ...");
  139 + schemaUpdateFile = Paths.get(installScripts.getDataDir(), "upgrade", "2.1.2", SCHEMA_UPDATE_SQL);
  140 + loadSql(schemaUpdateFile, conn);
  141 + log.info("Schema updated.");
  142 +
  143 + log.info("Restoring entity views ...");
  144 + if (entityViewsDump != null) {
  145 + SqlDbHelper.loadTable(conn, ENTITY_VIEW,
  146 + new String[]{ID, ENTITY_ID, ENTITY_TYPE, TENANT_ID, CUSTOMER_ID, TYPE, NAME, KEYS, START_TS, END_TS, SEARCH_TEXT, ADDITIONAL_INFO}, entityViewsDump, true);
  147 + Files.deleteIfExists(entityViewsDump);
  148 + }
  149 + log.info("Entity views restored.");
  150 + }
  151 + break;
118 152
119 153 default:
120 154 throw new RuntimeException("Unable to upgrade SQL database, unsupported fromVersion: " + fromVersion);
... ...
... ... @@ -147,6 +147,8 @@ public class CassandraDbHelper {
147 147 str = new Double(row.getDouble(index)).toString();
148 148 } else if (type == DataType.cint()) {
149 149 str = new Integer(row.getInt(index)).toString();
  150 + } else if (type == DataType.bigint()) {
  151 + str = new Long(row.getLong(index)).toString();
150 152 } else if (type == DataType.uuid()) {
151 153 str = row.getUUID(index).toString();
152 154 } else if (type == DataType.timeuuid()) {
... ... @@ -193,6 +195,8 @@ public class CassandraDbHelper {
193 195 boundStatement.setDouble(column, Double.valueOf(value));
194 196 } else if (type == DataType.cint()) {
195 197 boundStatement.setInt(column, Integer.valueOf(value));
  198 + } else if (type == DataType.bigint()) {
  199 + boundStatement.setLong(column, Long.valueOf(value));
196 200 } else if (type == DataType.uuid()) {
197 201 boundStatement.setUUID(column, UUID.fromString(value));
198 202 } else if (type == DataType.timeuuid()) {
... ...
... ... @@ -77,46 +77,10 @@ security:
77 77 # Enable/disable access to Tenant Administrators JWT token by System Administrator or Customer Users JWT token by Tenant Administrator
78 78 user_token_access_enabled: "${SECURITY_USER_TOKEN_ACCESS_ENABLED:true}"
79 79
80   -# Device communication protocol parameters
81   -http:
82   - request_timeout: "${HTTP_REQUEST_TIMEOUT:60000}"
83   -
84   -# MQTT server parameters
85   -mqtt:
86   - # Enable/disable mqtt transport protocol.
87   - enabled: "${MQTT_ENABLED:false}"
88   - bind_address: "${MQTT_BIND_ADDRESS:0.0.0.0}"
89   - bind_port: "${MQTT_BIND_PORT:1883}"
90   - adaptor: "${MQTT_ADAPTOR_NAME:JsonMqttAdaptor}"
91   - timeout: "${MQTT_TIMEOUT:10000}"
92   - netty:
93   - leak_detector_level: "${NETTY_LEASK_DETECTOR_LVL:DISABLED}"
94   - boss_group_thread_count: "${NETTY_BOSS_GROUP_THREADS:1}"
95   - worker_group_thread_count: "${NETTY_WORKER_GROUP_THREADS:12}"
96   - max_payload_size: "${NETTY_MAX_PAYLOAD_SIZE:65536}"
97   - # MQTT SSL configuration
98   - ssl:
99   - # Enable/disable SSL support
100   - enabled: "${MQTT_SSL_ENABLED:false}"
101   - # SSL protocol: See http://docs.oracle.com/javase/8/docs/technotes/guides/security/StandardNames.html#SSLContext
102   - protocol: "${MQTT_SSL_PROTOCOL:TLSv1.2}"
103   - # Path to the key store that holds the SSL certificate
104   - key_store: "${MQTT_SSL_KEY_STORE:mqttserver.jks}"
105   - # Password used to access the key store
106   - key_store_password: "${MQTT_SSL_KEY_STORE_PASSWORD:server_ks_password}"
107   - # Password used to access the key
108   - key_password: "${MQTT_SSL_KEY_PASSWORD:server_key_password}"
109   - # Type of the key store
110   - key_store_type: "${MQTT_SSL_KEY_STORE_TYPE:JKS}"
111   -
112   -# CoAP server parameters
113   -coap:
114   - # Enable/disable coap transport protocol.
115   - enabled: "${COAP_ENABLED:false}"
116   - bind_address: "${COAP_BIND_ADDRESS:0.0.0.0}"
117   - bind_port: "${COAP_BIND_PORT:5683}"
118   - adaptor: "${COAP_ADAPTOR_NAME:JsonCoapAdaptor}"
119   - timeout: "${COAP_TIMEOUT:10000}"
  80 +# Dashboard parameters
  81 +dashboard:
  82 + # Maximum allowed datapoints fetched by widgets
  83 + max_datapoints_limit: "${DASHBOARD_MAX_DATAPOINTS_LIMIT:50000}"
120 84
121 85 #Quota parameters
122 86 quota:
... ... @@ -136,8 +100,8 @@ quota:
136 100 # Array of blacklist hosts
137 101 blacklist: "${QUOTA_HOST_BLACKLIST:}"
138 102 log:
139   - topSize: 10
140   - intervalMin: 2
  103 + topSize: "${QUOTA_HOST_LOG_TOP_SIZE:10}"
  104 + intervalMin: "${QUOTA_HOST_LOG_INTERVAL_MIN:2}"
141 105 rule:
142 106 tenant:
143 107 # Max allowed number of API requests in interval for single tenant
... ... @@ -153,10 +117,10 @@ quota:
153 117 # Array of whitelist tenants
154 118 whitelist: "${QUOTA_TENANT_WHITELIST:}"
155 119 # Array of blacklist tenants
156   - blacklist: "${QUOTA_HOST_BLACKLIST:}"
  120 + blacklist: "${QUOTA_HOST_TENANT_BLACKLIST:}"
157 121 log:
158   - topSize: 10
159   - intervalMin: 2
  122 + topSize: "${QUOTA_TENANT_LOG_TOP_SIZE:10}"
  123 + intervalMin: "${QUOTA_TENANT_LOG_INTERVAL_MIN:2}"
160 124
161 125 database:
162 126 entities:
... ... @@ -185,7 +149,8 @@ cassandra:
185 149 init_timeout_ms: "${CASSANDRA_CLUSTER_INIT_TIMEOUT_MS:300000}"
186 150 # Specify cassandra claster initialization retry interval (if no hosts available during startup)
187 151 init_retry_interval_ms: "${CASSANDRA_CLUSTER_INIT_RETRY_INTERVAL_MS:3000}"
188   -
  152 + max_requests_per_connection_local: "${CASSANDRA_MAX_REQUESTS_PER_CONNECTION_LOCAL:32768}"
  153 + max_requests_per_connection_remote: "${CASSANDRA_MAX_REQUESTS_PER_CONNECTION_REMOTE:32768}"
189 154 # Credential parameters #
190 155 credentials: "${CASSANDRA_USE_CREDENTIALS:false}"
191 156 # Specify your username
... ... @@ -466,4 +431,42 @@ transport:
466 431 poll_interval: "${TB_RULE_ENGINE_POLL_INTERVAL_MS:25}"
467 432 auto_commit_interval: "${TB_RULE_ENGINE_AUTO_COMMIT_INTERVAL_MS:100}"
468 433 notifications:
469   - topic: "${TB_TRANSPORT_NOTIFICATIONS_TOPIC:tb.transport.notifications}"
\ No newline at end of file
  434 + topic: "${TB_TRANSPORT_NOTIFICATIONS_TOPIC:tb.transport.notifications}"
  435 + # Local HTTP transport parameters
  436 + http:
  437 + request_timeout: "${HTTP_REQUEST_TIMEOUT:60000}"
  438 + # Local MQTT transport parameters
  439 + mqtt:
  440 + # Enable/disable mqtt transport protocol.
  441 + enabled: "${MQTT_ENABLED:false}"
  442 + bind_address: "${MQTT_BIND_ADDRESS:0.0.0.0}"
  443 + bind_port: "${MQTT_BIND_PORT:1883}"
  444 + adaptor: "${MQTT_ADAPTOR_NAME:JsonMqttAdaptor}"
  445 + timeout: "${MQTT_TIMEOUT:10000}"
  446 + netty:
  447 + leak_detector_level: "${NETTY_LEASK_DETECTOR_LVL:DISABLED}"
  448 + boss_group_thread_count: "${NETTY_BOSS_GROUP_THREADS:1}"
  449 + worker_group_thread_count: "${NETTY_WORKER_GROUP_THREADS:12}"
  450 + max_payload_size: "${NETTY_MAX_PAYLOAD_SIZE:65536}"
  451 + # MQTT SSL configuration
  452 + ssl:
  453 + # Enable/disable SSL support
  454 + enabled: "${MQTT_SSL_ENABLED:false}"
  455 + # SSL protocol: See http://docs.oracle.com/javase/8/docs/technotes/guides/security/StandardNames.html#SSLContext
  456 + protocol: "${MQTT_SSL_PROTOCOL:TLSv1.2}"
  457 + # Path to the key store that holds the SSL certificate
  458 + key_store: "${MQTT_SSL_KEY_STORE:mqttserver.jks}"
  459 + # Password used to access the key store
  460 + key_store_password: "${MQTT_SSL_KEY_STORE_PASSWORD:server_ks_password}"
  461 + # Password used to access the key
  462 + key_password: "${MQTT_SSL_KEY_PASSWORD:server_key_password}"
  463 + # Type of the key store
  464 + key_store_type: "${MQTT_SSL_KEY_STORE_TYPE:JKS}"
  465 + # Local CoAP transport parameters
  466 + coap:
  467 + # Enable/disable coap transport protocol.
  468 + enabled: "${COAP_ENABLED:false}"
  469 + bind_address: "${COAP_BIND_ADDRESS:0.0.0.0}"
  470 + bind_port: "${COAP_BIND_PORT:5683}"
  471 + adaptor: "${COAP_ADAPTOR_NAME:JsonCoapAdaptor}"
  472 + timeout: "${COAP_TIMEOUT:10000}"
... ...
... ... @@ -85,11 +85,11 @@ public abstract class BaseEntityViewControllerTest extends AbstractControllerTes
85 85 testDevice = doPost("/api/device", device, Device.class);
86 86
87 87 telemetry = new TelemetryEntityView(
88   - Arrays.asList("109", "209", "309"),
  88 + Arrays.asList("tsKey1", "tsKey2", "tsKey3"),
89 89 new AttributesEntityView(
90   - Arrays.asList("caValue1", "caValue2", "caValue3", "caValue4"),
91   - Arrays.asList("saValue1", "saValue2", "saValue3", "saValue4"),
92   - Arrays.asList("shValue1", "shValue2", "shValue3", "shValue4")));
  90 + Arrays.asList("caKey1", "caKey2", "caKey3", "caKey4"),
  91 + Arrays.asList("saKey1", "saKey2", "saKey3", "saKey4"),
  92 + Arrays.asList("shKey1", "shKey2", "shKey3", "shKey4")));
93 93 }
94 94
95 95 @After
... ... @@ -144,7 +144,9 @@ public abstract class BaseEntityViewControllerTest extends AbstractControllerTes
144 144
145 145 @Test
146 146 public void testSaveEntityViewWithEmptyName() throws Exception {
147   - doPost("/api/entityView", new EntityView())
  147 + EntityView entityView = new EntityView();
  148 + entityView.setType("default");
  149 + doPost("/api/entityView", entityView)
148 150 .andExpect(status().isBadRequest())
149 151 .andExpect(statusReason(containsString("Entity view name should be specified!")));
150 152 }
... ... @@ -322,10 +324,10 @@ public abstract class BaseEntityViewControllerTest extends AbstractControllerTes
322 324 @Test
323 325 public void testTheCopyOfAttrsIntoTSForTheView() throws Exception {
324 326 Set<String> actualAttributesSet =
325   - getAttributesByKeys("{\"caValue1\":\"value1\", \"caValue2\":true, \"caValue3\":42.0, \"caValue4\":73}");
  327 + getAttributesByKeys("{\"caKey1\":\"value1\", \"caKey2\":true, \"caKey3\":42.0, \"caKey4\":73}");
326 328
327 329 Set<String> expectedActualAttributesSet =
328   - new HashSet<>(Arrays.asList("caValue1", "caValue2", "caValue3", "caValue4"));
  330 + new HashSet<>(Arrays.asList("caKey1", "caKey2", "caKey3", "caKey4"));
329 331 assertTrue(actualAttributesSet.containsAll(expectedActualAttributesSet));
330 332 Thread.sleep(1000);
331 333
... ... @@ -333,18 +335,18 @@ public abstract class BaseEntityViewControllerTest extends AbstractControllerTes
333 335 List<Map<String, Object>> values = doGetAsync("/api/plugins/telemetry/ENTITY_VIEW/" + savedView.getId().getId().toString() +
334 336 "/values/attributes?keys=" + String.join(",", actualAttributesSet), List.class);
335 337
336   - assertEquals("value1", getValue(values, "caValue1"));
337   - assertEquals(true, getValue(values, "caValue2"));
338   - assertEquals(42.0, getValue(values, "caValue3"));
339   - assertEquals(73, getValue(values, "caValue4"));
  338 + assertEquals("value1", getValue(values, "caKey1"));
  339 + assertEquals(true, getValue(values, "caKey2"));
  340 + assertEquals(42.0, getValue(values, "caKey3"));
  341 + assertEquals(73, getValue(values, "caKey4"));
340 342 }
341 343
342 344 @Test
343 345 public void testTheCopyOfAttrsOutOfTSForTheView() throws Exception {
344 346 Set<String> actualAttributesSet =
345   - getAttributesByKeys("{\"caValue1\":\"value1\", \"caValue2\":true, \"caValue3\":42.0, \"caValue4\":73}");
  347 + getAttributesByKeys("{\"caKey1\":\"value1\", \"caKey2\":true, \"caKey3\":42.0, \"caKey4\":73}");
346 348
347   - Set<String> expectedActualAttributesSet = new HashSet<>(Arrays.asList("caValue1", "caValue2", "caValue3", "caValue4"));
  349 + Set<String> expectedActualAttributesSet = new HashSet<>(Arrays.asList("caKey1", "caKey2", "caKey3", "caKey4"));
348 350 assertTrue(actualAttributesSet.containsAll(expectedActualAttributesSet));
349 351 Thread.sleep(1000);
350 352
... ... @@ -355,6 +357,7 @@ public abstract class BaseEntityViewControllerTest extends AbstractControllerTes
355 357 view.setEntityId(testDevice.getId());
356 358 view.setTenantId(savedTenant.getId());
357 359 view.setName("Test entity view");
  360 + view.setType("default");
358 361 view.setKeys(telemetry);
359 362 view.setStartTimeMs((long) getValue(valueTelemetryOfDevices, "lastActivityTime") * 10);
360 363 view.setEndTimeMs((long) getValue(valueTelemetryOfDevices, "lastActivityTime") / 10);
... ... @@ -365,6 +368,69 @@ public abstract class BaseEntityViewControllerTest extends AbstractControllerTes
365 368 assertEquals(0, values.size());
366 369 }
367 370
  371 +
  372 + @Test
  373 + public void testGetTelemetryWhenEntityViewTimeRangeInsideTimestampRange() throws Exception {
  374 + uploadTelemetry("{\"tsKey1\":\"value1\", \"tsKey2\":true, \"tsKey3\":40.0}");
  375 + Thread.sleep(1000);
  376 + long startTimeMs = System.currentTimeMillis();
  377 + uploadTelemetry("{\"tsKey1\":\"value2\", \"tsKey2\":false, \"tsKey3\":80.0}");
  378 + Thread.sleep(1000);
  379 + uploadTelemetry("{\"tsKey1\":\"value3\", \"tsKey2\":false, \"tsKey3\":120.0}");
  380 + long endTimeMs = System.currentTimeMillis();
  381 + uploadTelemetry("{\"tsKey1\":\"value4\", \"tsKey2\":true, \"tsKey3\":160.0}");
  382 +
  383 + String deviceId = testDevice.getId().getId().toString();
  384 + Set<String> keys = getTelemetryKeys("DEVICE", deviceId);
  385 + Thread.sleep(1000);
  386 +
  387 + EntityView view = createEntityView("Test entity view", startTimeMs, endTimeMs);
  388 + EntityView savedView = doPost("/api/entityView", view, EntityView.class);
  389 + String entityViewId = savedView.getId().getId().toString();
  390 +
  391 + Map<String, List<Map<String, String>>> expectedValues = getTelemetryValues("DEVICE", deviceId, keys, 0L, (startTimeMs + endTimeMs) / 2);
  392 + Assert.assertEquals(2, expectedValues.get("tsKey1").size());
  393 + Assert.assertEquals(2, expectedValues.get("tsKey2").size());
  394 + Assert.assertEquals(2, expectedValues.get("tsKey3").size());
  395 +
  396 + Map<String, List<Map<String, String>>> actualValues = getTelemetryValues("ENTITY_VIEW", entityViewId, keys, 0L, (startTimeMs + endTimeMs) / 2);
  397 + Assert.assertEquals(1, actualValues.get("tsKey1").size());
  398 + Assert.assertEquals(1, actualValues.get("tsKey2").size());
  399 + Assert.assertEquals(1, actualValues.get("tsKey3").size());
  400 + }
  401 +
  402 + private void uploadTelemetry(String strKvs) throws Exception {
  403 + String viewDeviceId = testDevice.getId().getId().toString();
  404 + DeviceCredentials deviceCredentials =
  405 + doGet("/api/device/" + viewDeviceId + "/credentials", DeviceCredentials.class);
  406 + assertEquals(testDevice.getId(), deviceCredentials.getDeviceId());
  407 +
  408 + String accessToken = deviceCredentials.getCredentialsId();
  409 + assertNotNull(accessToken);
  410 +
  411 + String clientId = MqttAsyncClient.generateClientId();
  412 + MqttAsyncClient client = new MqttAsyncClient("tcp://localhost:1883", clientId);
  413 +
  414 + MqttConnectOptions options = new MqttConnectOptions();
  415 + options.setUserName(accessToken);
  416 + client.connect(options);
  417 + Thread.sleep(3000);
  418 +
  419 + MqttMessage message = new MqttMessage();
  420 + message.setPayload(strKvs.getBytes());
  421 + client.publish("v1/devices/me/telemetry", message);
  422 + Thread.sleep(1000);
  423 + }
  424 +
  425 + private Set<String> getTelemetryKeys(String type, String id) throws Exception {
  426 + return new HashSet<>(doGetAsync("/api/plugins/telemetry/" + type + "/" + id + "/keys/timeseries", List.class));
  427 + }
  428 +
  429 + private Map<String, List<Map<String, String>>> getTelemetryValues(String type, String id, Set<String> keys, Long startTs, Long endTs) throws Exception {
  430 + return doGetAsync("/api/plugins/telemetry/" + type + "/" + id +
  431 + "/values/timeseries?keys=" + String.join(",", keys) + "&startTs=" + startTs + "&endTs=" + endTs, Map.class);
  432 + }
  433 +
368 434 private Set<String> getAttributesByKeys(String stringKV) throws Exception {
369 435 String viewDeviceId = testDevice.getId().getId().toString();
370 436 DeviceCredentials deviceCredentials =
... ... @@ -387,7 +453,7 @@ public abstract class BaseEntityViewControllerTest extends AbstractControllerTes
387 453 client.publish("v1/devices/me/attributes", message);
388 454 Thread.sleep(1000);
389 455
390   - return new HashSet<>(doGetAsync("/api/plugins/telemetry/DEVICE/" + viewDeviceId + "/keys/attributes", List.class));
  456 + return new HashSet<>(doGetAsync("/api/plugins/telemetry/DEVICE/" + viewDeviceId + "/keys/attributes", List.class));
391 457 }
392 458
393 459 private Object getValue(List<Map<String, Object>> values, String stringValue) {
... ... @@ -398,12 +464,20 @@ public abstract class BaseEntityViewControllerTest extends AbstractControllerTes
398 464 }
399 465
400 466 private EntityView getNewSavedEntityView(String name) throws Exception {
  467 + EntityView view = createEntityView(name, 0, 0);
  468 + return doPost("/api/entityView", view, EntityView.class);
  469 + }
  470 +
  471 + private EntityView createEntityView(String name, long startTimeMs, long endTimeMs) {
401 472 EntityView view = new EntityView();
402 473 view.setEntityId(testDevice.getId());
403 474 view.setTenantId(savedTenant.getId());
404 475 view.setName(name);
  476 + view.setType("default");
405 477 view.setKeys(telemetry);
406   - return doPost("/api/entityView", view, EntityView.class);
  478 + view.setStartTimeMs(startTimeMs);
  479 + view.setEndTimeMs(endTimeMs);
  480 + return view;
407 481 }
408 482
409 483 private Customer getNewCustomer(String title) {
... ...
... ... @@ -16,6 +16,7 @@
16 16 package org.thingsboard.server.mqtt.rpc;
17 17
18 18 import com.datastax.driver.core.utils.UUIDs;
  19 +import io.netty.handler.codec.mqtt.MqttQoS;
19 20 import lombok.extern.slf4j.Slf4j;
20 21 import org.apache.commons.lang3.StringUtils;
21 22 import org.eclipse.paho.client.mqttv3.IMqttDeliveryToken;
... ... @@ -23,19 +24,19 @@ import org.eclipse.paho.client.mqttv3.MqttAsyncClient;
23 24 import org.eclipse.paho.client.mqttv3.MqttCallback;
24 25 import org.eclipse.paho.client.mqttv3.MqttConnectOptions;
25 26 import org.eclipse.paho.client.mqttv3.MqttMessage;
26   -import org.junit.After;
27   -import org.junit.Assert;
28   -import org.junit.Before;
29   -import org.junit.Test;
  27 +import org.junit.*;
30 28 import org.thingsboard.server.common.data.Device;
31 29 import org.thingsboard.server.common.data.Tenant;
32 30 import org.thingsboard.server.common.data.User;
33 31 import org.thingsboard.server.common.data.security.Authority;
34 32 import org.thingsboard.server.common.data.security.DeviceCredentials;
35 33 import org.thingsboard.server.controller.AbstractControllerTest;
  34 +import org.thingsboard.server.mqtt.telemetry.AbstractMqttTelemetryIntegrationTest;
36 35 import org.thingsboard.server.service.security.AccessValidator;
37 36
38 37 import java.util.Arrays;
  38 +import java.util.concurrent.CountDownLatch;
  39 +import java.util.concurrent.TimeUnit;
39 40
40 41 import static org.junit.Assert.assertEquals;
41 42 import static org.junit.Assert.assertNotNull;
... ... @@ -101,13 +102,19 @@ public abstract class AbstractMqttServerSideRpcIntegrationTest extends AbstractC
101 102 MqttConnectOptions options = new MqttConnectOptions();
102 103 options.setUserName(accessToken);
103 104 client.connect(options).waitForCompletion();
104   - client.subscribe("v1/devices/me/rpc/request/+", 1);
105   - client.setCallback(new TestMqttCallback(client));
  105 +
  106 + CountDownLatch latch = new CountDownLatch(1);
  107 + TestMqttCallback callback = new TestMqttCallback(client, latch);
  108 + client.setCallback(callback);
  109 +
  110 + client.subscribe("v1/devices/me/rpc/request/+", MqttQoS.AT_MOST_ONCE.value());
106 111
107 112 String setGpioRequest = "{\"method\":\"setGpio\",\"params\":{\"pin\": \"23\",\"value\": 1}}";
108 113 String deviceId = savedDevice.getId().getId().toString();
109 114 String result = doPostAsync("/api/plugins/rpc/oneway/" + deviceId, setGpioRequest, String.class, status().isOk());
110 115 Assert.assertTrue(StringUtils.isEmpty(result));
  116 + latch.await(3, TimeUnit.SECONDS);
  117 + assertEquals(MqttQoS.AT_MOST_ONCE.value(), callback.getQoS());
111 118 }
112 119
113 120 @Test
... ... @@ -156,7 +163,7 @@ public abstract class AbstractMqttServerSideRpcIntegrationTest extends AbstractC
156 163 options.setUserName(accessToken);
157 164 client.connect(options).waitForCompletion();
158 165 client.subscribe("v1/devices/me/rpc/request/+", 1);
159   - client.setCallback(new TestMqttCallback(client));
  166 + client.setCallback(new TestMqttCallback(client, new CountDownLatch(1)));
160 167
161 168 String setGpioRequest = "{\"method\":\"setGpio\",\"params\":{\"pin\": \"23\",\"value\": 1}}";
162 169 String deviceId = savedDevice.getId().getId().toString();
... ... @@ -204,9 +211,16 @@ public abstract class AbstractMqttServerSideRpcIntegrationTest extends AbstractC
204 211 private static class TestMqttCallback implements MqttCallback {
205 212
206 213 private final MqttAsyncClient client;
  214 + private final CountDownLatch latch;
  215 + private Integer qoS;
207 216
208   - TestMqttCallback(MqttAsyncClient client) {
  217 + TestMqttCallback(MqttAsyncClient client, CountDownLatch latch) {
209 218 this.client = client;
  219 + this.latch = latch;
  220 + }
  221 +
  222 + int getQoS() {
  223 + return qoS;
210 224 }
211 225
212 226 @Override
... ... @@ -219,7 +233,9 @@ public abstract class AbstractMqttServerSideRpcIntegrationTest extends AbstractC
219 233 MqttMessage message = new MqttMessage();
220 234 String responseTopic = requestTopic.replace("request", "response");
221 235 message.setPayload("{\"value1\":\"A\", \"value2\":\"B\"}".getBytes("UTF-8"));
  236 + qoS = mqttMessage.getQos();
222 237 client.publish(responseTopic, message);
  238 + latch.countDown();
223 239 }
224 240
225 241 @Override
... ...
... ... @@ -15,10 +15,9 @@
15 15 */
16 16 package org.thingsboard.server.mqtt.telemetry;
17 17
  18 +import io.netty.handler.codec.mqtt.MqttQoS;
18 19 import lombok.extern.slf4j.Slf4j;
19   -import org.eclipse.paho.client.mqttv3.MqttAsyncClient;
20   -import org.eclipse.paho.client.mqttv3.MqttConnectOptions;
21   -import org.eclipse.paho.client.mqttv3.MqttMessage;
  20 +import org.eclipse.paho.client.mqttv3.*;
22 21 import org.junit.Before;
23 22 import org.junit.Ignore;
24 23 import org.junit.Test;
... ... @@ -30,9 +29,12 @@ import org.thingsboard.server.dao.service.DaoNoSqlTest;
30 29
31 30 import java.net.URI;
32 31 import java.util.*;
  32 +import java.util.concurrent.CountDownLatch;
  33 +import java.util.concurrent.TimeUnit;
33 34
34 35 import static org.junit.Assert.assertEquals;
35 36 import static org.junit.Assert.assertNotNull;
  37 +import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.status;
36 38
37 39 /**
38 40 * @author Valerii Sosliuk
... ... @@ -94,4 +96,62 @@ public abstract class AbstractMqttTelemetryIntegrationTest extends AbstractContr
94 96 assertEquals("3.0", values.get("key3").get(0).get("value"));
95 97 assertEquals("4", values.get("key4").get(0).get("value"));
96 98 }
  99 +
  100 + @Test
  101 + public void testMqttQoSLevel() throws Exception {
  102 + String clientId = MqttAsyncClient.generateClientId();
  103 + MqttAsyncClient client = new MqttAsyncClient(MQTT_URL, clientId);
  104 +
  105 + MqttConnectOptions options = new MqttConnectOptions();
  106 + options.setUserName(accessToken);
  107 + client.connect(options).waitForCompletion(3000);
  108 + CountDownLatch latch = new CountDownLatch(1);
  109 + TestMqttCallback callback = new TestMqttCallback(client, latch);
  110 + client.setCallback(callback);
  111 + client.subscribe("v1/devices/me/attributes", MqttQoS.AT_MOST_ONCE.value());
  112 + String payload = "{\"key\":\"value\"}";
  113 + String result = doPostAsync("/api/plugins/telemetry/" + savedDevice.getId() + "/SHARED_SCOPE", payload, String.class, status().isOk());
  114 + latch.await(3, TimeUnit.SECONDS);
  115 + assertEquals(payload, callback.getPayload());
  116 + assertEquals(MqttQoS.AT_MOST_ONCE.value(), callback.getQoS());
  117 + }
  118 +
  119 + private static class TestMqttCallback implements MqttCallback {
  120 +
  121 + private final MqttAsyncClient client;
  122 + private final CountDownLatch latch;
  123 + private Integer qoS;
  124 + private String payload;
  125 +
  126 + String getPayload() {
  127 + return payload;
  128 + }
  129 +
  130 + TestMqttCallback(MqttAsyncClient client, CountDownLatch latch) {
  131 + this.client = client;
  132 + this.latch = latch;
  133 + }
  134 +
  135 + int getQoS() {
  136 + return qoS;
  137 + }
  138 +
  139 + @Override
  140 + public void connectionLost(Throwable throwable) {
  141 + }
  142 +
  143 + @Override
  144 + public void messageArrived(String requestTopic, MqttMessage mqttMessage) {
  145 + payload = new String(mqttMessage.getPayload());
  146 + qoS = mqttMessage.getQos();
  147 + latch.countDown();
  148 + }
  149 +
  150 + @Override
  151 + public void deliveryComplete(IMqttDeliveryToken iMqttDeliveryToken) {
  152 +
  153 + }
  154 + }
  155 +
  156 +
97 157 }
... ...
... ... @@ -40,6 +40,7 @@ public class EntityView extends SearchTextBasedWithAdditionalInfo<EntityViewId>
40 40 private TenantId tenantId;
41 41 private CustomerId customerId;
42 42 private String name;
  43 + private String type;
43 44 private TelemetryEntityView keys;
44 45 private long startTimeMs;
45 46 private long endTimeMs;
... ...
... ... @@ -30,6 +30,7 @@ public class EntityViewSearchQuery {
30 30
31 31 private RelationsSearchParameters parameters;
32 32 private String relationType;
  33 + private List<String> entityViewTypes;
33 34
34 35 public EntityRelationsQuery toEntitySearchQuery() {
35 36 EntityRelationsQuery query = new EntityRelationsQuery();
... ...
... ... @@ -15,8 +15,13 @@
15 15 */
16 16 package org.thingsboard.server.dao.entityview;
17 17
  18 +import com.datastax.driver.core.ResultSet;
  19 +import com.datastax.driver.core.ResultSetFuture;
18 20 import com.datastax.driver.core.Statement;
19 21 import com.datastax.driver.core.querybuilder.Select;
  22 +import com.datastax.driver.mapping.Result;
  23 +import com.google.common.base.Function;
  24 +import com.google.common.util.concurrent.Futures;
20 25 import com.google.common.util.concurrent.ListenableFuture;
21 26 import lombok.extern.slf4j.Slf4j;
22 27 import org.springframework.stereotype.Component;
... ... @@ -30,6 +35,8 @@ import org.thingsboard.server.dao.model.nosql.EntityViewEntity;
30 35 import org.thingsboard.server.dao.nosql.CassandraAbstractSearchTextDao;
31 36 import org.thingsboard.server.dao.util.NoSqlDao;
32 37
  38 +import javax.annotation.Nullable;
  39 +import java.util.ArrayList;
33 40 import java.util.Arrays;
34 41 import java.util.Collections;
35 42 import java.util.List;
... ... @@ -39,14 +46,21 @@ import java.util.UUID;
39 46 import static com.datastax.driver.core.querybuilder.QueryBuilder.eq;
40 47 import static com.datastax.driver.core.querybuilder.QueryBuilder.select;
41 48 import static org.thingsboard.server.dao.model.ModelConstants.CUSTOMER_ID_PROPERTY;
  49 +import static org.thingsboard.server.dao.model.ModelConstants.DEVICE_TYPE_PROPERTY;
42 50 import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_ID_COLUMN;
  51 +import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_SUBTYPE_COLUMN_FAMILY_NAME;
  52 +import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_SUBTYPE_ENTITY_TYPE_PROPERTY;
  53 +import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_SUBTYPE_TENANT_ID_PROPERTY;
  54 +import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_VIEW_BY_TENANT_AND_CUSTOMER_AND_TYPE_CF;
43 55 import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_VIEW_BY_TENANT_AND_CUSTOMER_CF;
44 56 import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_VIEW_BY_TENANT_AND_ENTITY_ID_CF;
45 57 import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_VIEW_BY_TENANT_AND_NAME;
46   -import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_VIEW_BY_TENANT_AND_SEARCH_TEXT_COLUMN_FAMILY_NAME;
  58 +import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_VIEW_BY_TENANT_AND_SEARCH_TEXT_CF;
  59 +import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_VIEW_BY_TENANT_BY_TYPE_AND_SEARCH_TEXT_CF;
47 60 import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_VIEW_NAME_PROPERTY;
48 61 import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_VIEW_TABLE_FAMILY_NAME;
49 62 import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_VIEW_TENANT_ID_PROPERTY;
  63 +import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_VIEW_TYPE_PROPERTY;
50 64 import static org.thingsboard.server.dao.model.ModelConstants.TENANT_ID_PROPERTY;
51 65
52 66 /**
... ... @@ -82,7 +96,7 @@ public class CassandraEntityViewDao extends CassandraAbstractSearchTextDao<Entit
82 96 public List<EntityView> findEntityViewsByTenantId(UUID tenantId, TextPageLink pageLink) {
83 97 log.debug("Try to find entity views by tenantId [{}] and pageLink [{}]", tenantId, pageLink);
84 98 List<EntityViewEntity> entityViewEntities =
85   - findPageWithTextSearch(ENTITY_VIEW_BY_TENANT_AND_SEARCH_TEXT_COLUMN_FAMILY_NAME,
  99 + findPageWithTextSearch(ENTITY_VIEW_BY_TENANT_AND_SEARCH_TEXT_CF,
86 100 Collections.singletonList(eq(TENANT_ID_PROPERTY, tenantId)), pageLink);
87 101 log.trace("Found entity views [{}] by tenantId [{}] and pageLink [{}]",
88 102 entityViewEntities, tenantId, pageLink);
... ... @@ -90,6 +104,18 @@ public class CassandraEntityViewDao extends CassandraAbstractSearchTextDao<Entit
90 104 }
91 105
92 106 @Override
  107 + public List<EntityView> findEntityViewsByTenantIdAndType(UUID tenantId, String type, TextPageLink pageLink) {
  108 + log.debug("Try to find entity views by tenantId [{}], type [{}] and pageLink [{}]", tenantId, type, pageLink);
  109 + List<EntityViewEntity> entityViewEntities =
  110 + findPageWithTextSearch(ENTITY_VIEW_BY_TENANT_BY_TYPE_AND_SEARCH_TEXT_CF,
  111 + Arrays.asList(eq(ENTITY_VIEW_TYPE_PROPERTY, type),
  112 + eq(TENANT_ID_PROPERTY, tenantId)), pageLink);
  113 + log.trace("Found entity views [{}] by tenantId [{}], type [{}] and pageLink [{}]",
  114 + entityViewEntities, tenantId, type, pageLink);
  115 + return DaoUtil.convertDataList(entityViewEntities);
  116 + }
  117 +
  118 + @Override
93 119 public Optional<EntityView> findEntityViewByTenantIdAndName(UUID tenantId, String name) {
94 120 Select.Where query = select().from(ENTITY_VIEW_BY_TENANT_AND_NAME).where();
95 121 query.and(eq(ENTITY_VIEW_TENANT_ID_PROPERTY, tenantId));
... ... @@ -111,6 +137,19 @@ public class CassandraEntityViewDao extends CassandraAbstractSearchTextDao<Entit
111 137 }
112 138
113 139 @Override
  140 + public List<EntityView> findEntityViewsByTenantIdAndCustomerIdAndType(UUID tenantId, UUID customerId, String type, TextPageLink pageLink) {
  141 + log.debug("Try to find entity views by tenantId [{}], customerId[{}], type [{}] and pageLink [{}]",
  142 + tenantId, customerId, type, pageLink);
  143 + List<EntityViewEntity> entityViewEntities = findPageWithTextSearch(
  144 + ENTITY_VIEW_BY_TENANT_AND_CUSTOMER_AND_TYPE_CF,
  145 + Arrays.asList(eq(DEVICE_TYPE_PROPERTY, type), eq(CUSTOMER_ID_PROPERTY, customerId), eq(TENANT_ID_PROPERTY, tenantId)),
  146 + pageLink);
  147 + log.trace("Found find entity views [{}] by tenantId [{}], customerId [{}], type [{}] and pageLink [{}]",
  148 + entityViewEntities, tenantId, customerId, type, pageLink);
  149 + return DaoUtil.convertDataList(entityViewEntities);
  150 + }
  151 +
  152 + @Override
114 153 public ListenableFuture<List<EntityView>> findEntityViewsByTenantIdAndEntityIdAsync(UUID tenantId, UUID entityId) {
115 154 log.debug("Try to find entity views by tenantId [{}] and entityId [{}]", tenantId, entityId);
116 155 Select.Where query = select().from(ENTITY_VIEW_BY_TENANT_AND_ENTITY_ID_CF).where();
... ... @@ -118,4 +157,30 @@ public class CassandraEntityViewDao extends CassandraAbstractSearchTextDao<Entit
118 157 query.and(eq(ENTITY_ID_COLUMN, entityId));
119 158 return findListByStatementAsync(query);
120 159 }
  160 +
  161 + @Override
  162 + public ListenableFuture<List<EntitySubtype>> findTenantEntityViewTypesAsync(UUID tenantId) {
  163 + Select select = select().from(ENTITY_SUBTYPE_COLUMN_FAMILY_NAME);
  164 + Select.Where query = select.where();
  165 + query.and(eq(ENTITY_SUBTYPE_TENANT_ID_PROPERTY, tenantId));
  166 + query.and(eq(ENTITY_SUBTYPE_ENTITY_TYPE_PROPERTY, EntityType.ENTITY_VIEW));
  167 + query.setConsistencyLevel(cluster.getDefaultReadConsistencyLevel());
  168 + ResultSetFuture resultSetFuture = executeAsyncRead(query);
  169 + return Futures.transform(resultSetFuture, new Function<ResultSet, List<EntitySubtype>>() {
  170 + @Nullable
  171 + @Override
  172 + public List<EntitySubtype> apply(@Nullable ResultSet resultSet) {
  173 + Result<EntitySubtypeEntity> result = cluster.getMapper(EntitySubtypeEntity.class).map(resultSet);
  174 + if (result != null) {
  175 + List<EntitySubtype> entitySubtypes = new ArrayList<>();
  176 + result.all().forEach((entitySubtypeEntity) ->
  177 + entitySubtypes.add(entitySubtypeEntity.toEntitySubtype())
  178 + );
  179 + return entitySubtypes;
  180 + } else {
  181 + return Collections.emptyList();
  182 + }
  183 + }
  184 + });
  185 + }
121 186 }
... ...
... ... @@ -17,6 +17,7 @@ package org.thingsboard.server.dao.entityview;
17 17
18 18 import com.google.common.util.concurrent.ListenableFuture;
19 19 import org.thingsboard.server.common.data.Device;
  20 +import org.thingsboard.server.common.data.EntitySubtype;
20 21 import org.thingsboard.server.common.data.EntityView;
21 22 import org.thingsboard.server.common.data.page.TextPageLink;
22 23 import org.thingsboard.server.dao.Dao;
... ... @@ -48,6 +49,16 @@ public interface EntityViewDao extends Dao<EntityView> {
48 49 List<EntityView> findEntityViewsByTenantId(UUID tenantId, TextPageLink pageLink);
49 50
50 51 /**
  52 + * Find entity views by tenantId, type and page link.
  53 + *
  54 + * @param tenantId the tenantId
  55 + * @param type the type
  56 + * @param pageLink the page link
  57 + * @return the list of entity view objects
  58 + */
  59 + List<EntityView> findEntityViewsByTenantIdAndType(UUID tenantId, String type, TextPageLink pageLink);
  60 +
  61 + /**
51 62 * Find entity views by tenantId and entity view name.
52 63 *
53 64 * @param tenantId the tenantId
... ... @@ -68,6 +79,27 @@ public interface EntityViewDao extends Dao<EntityView> {
68 79 UUID customerId,
69 80 TextPageLink pageLink);
70 81
  82 + /**
  83 + * Find entity views by tenantId, customerId, type and page link.
  84 + *
  85 + * @param tenantId the tenantId
  86 + * @param customerId the customerId
  87 + * @param type the type
  88 + * @param pageLink the page link
  89 + * @return the list of entity view objects
  90 + */
  91 + List<EntityView> findEntityViewsByTenantIdAndCustomerIdAndType(UUID tenantId,
  92 + UUID customerId,
  93 + String type,
  94 + TextPageLink pageLink);
71 95
72 96 ListenableFuture<List<EntityView>> findEntityViewsByTenantIdAndEntityIdAsync(UUID tenantId, UUID entityId);
  97 +
  98 + /**
  99 + * Find tenants entity view types.
  100 + *
  101 + * @return the list of tenant entity view type objects
  102 + */
  103 + ListenableFuture<List<EntitySubtype>> findTenantEntityViewTypesAsync(UUID tenantId);
  104 +
73 105 }
... ...
... ... @@ -16,6 +16,7 @@
16 16 package org.thingsboard.server.dao.entityview;
17 17
18 18 import com.google.common.util.concurrent.ListenableFuture;
  19 +import org.thingsboard.server.common.data.EntitySubtype;
19 20 import org.thingsboard.server.common.data.EntityView;
20 21 import org.thingsboard.server.common.data.entityview.EntityViewSearchQuery;
21 22 import org.thingsboard.server.common.data.id.CustomerId;
... ... @@ -44,8 +45,12 @@ public interface EntityViewService {
44 45
45 46 TextPageData<EntityView> findEntityViewByTenantId(TenantId tenantId, TextPageLink pageLink);
46 47
  48 + TextPageData<EntityView> findEntityViewByTenantIdAndType(TenantId tenantId, TextPageLink pageLink, String type);
  49 +
47 50 TextPageData<EntityView> findEntityViewsByTenantIdAndCustomerId(TenantId tenantId, CustomerId customerId, TextPageLink pageLink);
48 51
  52 + TextPageData<EntityView> findEntityViewsByTenantIdAndCustomerIdAndType(TenantId tenantId, CustomerId customerId, TextPageLink pageLink, String type);
  53 +
49 54 ListenableFuture<List<EntityView>> findEntityViewsByQuery(EntityViewSearchQuery query);
50 55
51 56 ListenableFuture<EntityView> findEntityViewByIdAsync(EntityViewId entityViewId);
... ... @@ -55,4 +60,6 @@ public interface EntityViewService {
55 60 void deleteEntityView(EntityViewId entityViewId);
56 61
57 62 void deleteEntityViewsByTenantId(TenantId tenantId);
  63 +
  64 + ListenableFuture<List<EntitySubtype>> findEntityViewTypesByTenantId(TenantId tenantId);
58 65 }
... ...
... ... @@ -15,6 +15,7 @@
15 15 */
16 16 package org.thingsboard.server.dao.entityview;
17 17
  18 +import com.google.common.base.Function;
18 19 import com.google.common.util.concurrent.FutureCallback;
19 20 import com.google.common.util.concurrent.Futures;
20 21 import com.google.common.util.concurrent.ListenableFuture;
... ... @@ -29,6 +30,8 @@ import org.springframework.cache.annotation.Caching;
29 30 import org.springframework.stereotype.Service;
30 31 import org.thingsboard.server.common.data.Customer;
31 32 import org.thingsboard.server.common.data.DataConstants;
  33 +import org.thingsboard.server.common.data.Device;
  34 +import org.thingsboard.server.common.data.EntitySubtype;
32 35 import org.thingsboard.server.common.data.EntityType;
33 36 import org.thingsboard.server.common.data.EntityView;
34 37 import org.thingsboard.server.common.data.Tenant;
... ... @@ -54,6 +57,8 @@ import javax.annotation.Nullable;
54 57 import java.util.ArrayList;
55 58 import java.util.Arrays;
56 59 import java.util.Collection;
  60 +import java.util.Collections;
  61 +import java.util.Comparator;
57 62 import java.util.List;
58 63 import java.util.concurrent.ExecutionException;
59 64 import java.util.stream.Collectors;
... ... @@ -63,6 +68,7 @@ import static org.thingsboard.server.common.data.CacheConstants.RELATIONS_CACHE;
63 68 import static org.thingsboard.server.dao.model.ModelConstants.NULL_UUID;
64 69 import static org.thingsboard.server.dao.service.Validator.validateId;
65 70 import static org.thingsboard.server.dao.service.Validator.validatePageLink;
  71 +import static org.thingsboard.server.dao.service.Validator.validateString;
66 72
67 73 /**
68 74 * Created by Victor Basanets on 8/28/2017.
... ... @@ -158,6 +164,16 @@ public class EntityViewServiceImpl extends AbstractEntityService implements Enti
158 164 }
159 165
160 166 @Override
  167 + public TextPageData<EntityView> findEntityViewByTenantIdAndType(TenantId tenantId, TextPageLink pageLink, String type) {
  168 + log.trace("Executing findEntityViewByTenantIdAndType, tenantId [{}], pageLink [{}], type [{}]", tenantId, pageLink, type);
  169 + validateId(tenantId, INCORRECT_TENANT_ID + tenantId);
  170 + validatePageLink(pageLink, INCORRECT_PAGE_LINK + pageLink);
  171 + validateString(type, "Incorrect type " + type);
  172 + List<EntityView> entityViews = entityViewDao.findEntityViewsByTenantIdAndType(tenantId.getId(), type, pageLink);
  173 + return new TextPageData<>(entityViews, pageLink);
  174 + }
  175 +
  176 + @Override
161 177 public TextPageData<EntityView> findEntityViewsByTenantIdAndCustomerId(TenantId tenantId, CustomerId customerId,
162 178 TextPageLink pageLink) {
163 179 log.trace("Executing findEntityViewByTenantIdAndCustomerId, tenantId [{}], customerId [{}]," +
... ... @@ -171,6 +187,19 @@ public class EntityViewServiceImpl extends AbstractEntityService implements Enti
171 187 }
172 188
173 189 @Override
  190 + public TextPageData<EntityView> findEntityViewsByTenantIdAndCustomerIdAndType(TenantId tenantId, CustomerId customerId, TextPageLink pageLink, String type) {
  191 + log.trace("Executing findEntityViewsByTenantIdAndCustomerIdAndType, tenantId [{}], customerId [{}]," +
  192 + " pageLink [{}], type [{}]", tenantId, customerId, pageLink, type);
  193 + validateId(tenantId, INCORRECT_TENANT_ID + tenantId);
  194 + validateId(customerId, INCORRECT_CUSTOMER_ID + customerId);
  195 + validatePageLink(pageLink, INCORRECT_PAGE_LINK + pageLink);
  196 + validateString(type, "Incorrect type " + type);
  197 + List<EntityView> entityViews = entityViewDao.findEntityViewsByTenantIdAndCustomerIdAndType(tenantId.getId(),
  198 + customerId.getId(), type, pageLink);
  199 + return new TextPageData<>(entityViews, pageLink);
  200 + }
  201 +
  202 + @Override
174 203 public ListenableFuture<List<EntityView>> findEntityViewsByQuery(EntityViewSearchQuery query) {
175 204 ListenableFuture<List<EntityRelation>> relations = relationService.findByQuery(query.toEntitySearchQuery());
176 205 ListenableFuture<List<EntityView>> entityViews = Futures.transformAsync(relations, r -> {
... ... @@ -184,6 +213,15 @@ public class EntityViewServiceImpl extends AbstractEntityService implements Enti
184 213 }
185 214 return Futures.successfulAsList(futures);
186 215 });
  216 +
  217 + entityViews = Futures.transform(entityViews, new Function<List<EntityView>, List<EntityView>>() {
  218 + @Nullable
  219 + @Override
  220 + public List<EntityView> apply(@Nullable List<EntityView> entityViewList) {
  221 + return entityViewList == null ? Collections.emptyList() : entityViewList.stream().filter(entityView -> query.getEntityViewTypes().contains(entityView.getType())).collect(Collectors.toList());
  222 + }
  223 + });
  224 +
187 225 return entityViews;
188 226 }
189 227
... ... @@ -216,6 +254,7 @@ public class EntityViewServiceImpl extends AbstractEntityService implements Enti
216 254 public void onSuccess(@Nullable List<EntityView> result) {
217 255 cache.putIfAbsent(tenantIdAndEntityId, result);
218 256 }
  257 +
219 258 @Override
220 259 public void onFailure(Throwable t) {
221 260 log.error("Error while finding entity views by tenantId and entityId", t);
... ... @@ -243,6 +282,18 @@ public class EntityViewServiceImpl extends AbstractEntityService implements Enti
243 282 tenantEntityViewRemover.removeEntities(tenantId);
244 283 }
245 284
  285 + @Override
  286 + public ListenableFuture<List<EntitySubtype>> findEntityViewTypesByTenantId(TenantId tenantId) {
  287 + log.trace("Executing findEntityViewTypesByTenantId, tenantId [{}]", tenantId);
  288 + validateId(tenantId, INCORRECT_TENANT_ID + tenantId);
  289 + ListenableFuture<List<EntitySubtype>> tenantEntityViewTypes = entityViewDao.findTenantEntityViewTypesAsync(tenantId.getId());
  290 + return Futures.transform(tenantEntityViewTypes,
  291 + entityViewTypes -> {
  292 + entityViewTypes.sort(Comparator.comparing(EntitySubtype::getType));
  293 + return entityViewTypes;
  294 + });
  295 + }
  296 +
246 297 private ListenableFuture<List<Void>> copyAttributesFromEntityToEntityView(EntityView entityView, String scope, Collection<String> keys) {
247 298 if (keys != null && !keys.isEmpty()) {
248 299 ListenableFuture<List<AttributeKvEntry>> getAttrFuture = attributesService.find(entityView.getEntityId(), scope, keys);
... ... @@ -296,6 +347,9 @@ public class EntityViewServiceImpl extends AbstractEntityService implements Enti
296 347
297 348 @Override
298 349 protected void validateDataImpl(EntityView entityView) {
  350 + if (StringUtils.isEmpty(entityView.getType())) {
  351 + throw new DataValidationException("Entity View type should be specified!");
  352 + }
299 353 if (StringUtils.isEmpty(entityView.getName())) {
300 354 throw new DataValidationException("Entity view name should be specified!");
301 355 }
... ...
... ... @@ -145,18 +145,21 @@ public class ModelConstants {
145 145 /**
146 146 * Cassandra entityView constants.
147 147 */
148   - public static final String ENTITY_VIEW_TABLE_FAMILY_NAME = "entity_views";
  148 + public static final String ENTITY_VIEW_TABLE_FAMILY_NAME = "entity_view";
149 149 public static final String ENTITY_VIEW_ENTITY_ID_PROPERTY = ENTITY_ID_COLUMN;
150 150 public static final String ENTITY_VIEW_TENANT_ID_PROPERTY = TENANT_ID_PROPERTY;
151 151 public static final String ENTITY_VIEW_CUSTOMER_ID_PROPERTY = CUSTOMER_ID_PROPERTY;
152 152 public static final String ENTITY_VIEW_NAME_PROPERTY = DEVICE_NAME_PROPERTY;
153 153 public static final String ENTITY_VIEW_BY_TENANT_AND_CUSTOMER_CF = "entity_view_by_tenant_and_customer";
  154 + public static final String ENTITY_VIEW_BY_TENANT_AND_CUSTOMER_AND_TYPE_CF = "entity_view_by_tenant_and_customer_and_type";
154 155 public static final String ENTITY_VIEW_BY_TENANT_AND_ENTITY_ID_CF = "entity_view_by_tenant_and_entity_id";
155 156 public static final String ENTITY_VIEW_KEYS_PROPERTY = "keys";
  157 + public static final String ENTITY_VIEW_TYPE_PROPERTY = "type";
156 158 public static final String ENTITY_VIEW_START_TS_PROPERTY = "start_ts";
157 159 public static final String ENTITY_VIEW_END_TS_PROPERTY = "end_ts";
158 160 public static final String ENTITY_VIEW_ADDITIONAL_INFO_PROPERTY = ADDITIONAL_INFO_PROPERTY;
159   - public static final String ENTITY_VIEW_BY_TENANT_AND_SEARCH_TEXT_COLUMN_FAMILY_NAME = "entity_view_by_tenant_and_search_text";
  161 + public static final String ENTITY_VIEW_BY_TENANT_AND_SEARCH_TEXT_CF = "entity_view_by_tenant_and_search_text";
  162 + public static final String ENTITY_VIEW_BY_TENANT_BY_TYPE_AND_SEARCH_TEXT_CF = "entity_view_by_tenant_by_type_and_search_text";
160 163 public static final String ENTITY_VIEW_BY_TENANT_AND_NAME = "entity_view_by_tenant_and_name";
161 164
162 165 /**
... ...
... ... @@ -41,6 +41,7 @@ import javax.persistence.Enumerated;
41 41 import java.io.IOException;
42 42 import java.util.UUID;
43 43
  44 +import static org.thingsboard.server.dao.model.ModelConstants.DEVICE_TYPE_PROPERTY;
44 45 import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_TYPE_PROPERTY;
45 46 import static org.thingsboard.server.dao.model.ModelConstants.ENTITY_VIEW_TABLE_FAMILY_NAME;
46 47 import static org.thingsboard.server.dao.model.ModelConstants.ID_PROPERTY;
... ... @@ -71,6 +72,10 @@ public class EntityViewEntity implements SearchTextEntity<EntityView> {
71 72 @Column(name = ModelConstants.ENTITY_VIEW_CUSTOMER_ID_PROPERTY)
72 73 private UUID customerId;
73 74
  75 + @PartitionKey(value = 3)
  76 + @Column(name = DEVICE_TYPE_PROPERTY)
  77 + private String type;
  78 +
74 79 @Column(name = ModelConstants.ENTITY_VIEW_ENTITY_ID_PROPERTY)
75 80 private UUID entityId;
76 81
... ... @@ -113,6 +118,7 @@ public class EntityViewEntity implements SearchTextEntity<EntityView> {
113 118 if (entityView.getCustomerId() != null) {
114 119 this.customerId = entityView.getCustomerId().getId();
115 120 }
  121 + this.type = entityView.getType();
116 122 this.name = entityView.getName();
117 123 try {
118 124 this.keys = mapper.writeValueAsString(entityView.getKeys());
... ... @@ -143,6 +149,7 @@ public class EntityViewEntity implements SearchTextEntity<EntityView> {
143 149 if (customerId != null) {
144 150 entityView.setCustomerId(new CustomerId(customerId));
145 151 }
  152 + entityView.setType(type);
146 153 entityView.setName(name);
147 154 try {
148 155 entityView.setKeys(mapper.readValue(keys, TelemetryEntityView.class));
... ...
... ... @@ -69,6 +69,9 @@ public class EntityViewEntity extends BaseSqlEntity<EntityView> implements Searc
69 69 @Column(name = ModelConstants.ENTITY_VIEW_CUSTOMER_ID_PROPERTY)
70 70 private String customerId;
71 71
  72 + @Column(name = ModelConstants.DEVICE_TYPE_PROPERTY)
  73 + private String type;
  74 +
72 75 @Column(name = ModelConstants.ENTITY_VIEW_NAME_PROPERTY)
73 76 private String name;
74 77
... ... @@ -108,6 +111,7 @@ public class EntityViewEntity extends BaseSqlEntity<EntityView> implements Searc
108 111 if (entityView.getCustomerId() != null) {
109 112 this.customerId = toString(entityView.getCustomerId().getId());
110 113 }
  114 + this.type = entityView.getType();
111 115 this.name = entityView.getName();
112 116 try {
113 117 this.keys = mapper.writeValueAsString(entityView.getKeys());
... ... @@ -144,6 +148,7 @@ public class EntityViewEntity extends BaseSqlEntity<EntityView> implements Searc
144 148 if (customerId != null) {
145 149 entityView.setCustomerId(new CustomerId(toUUID(customerId)));
146 150 }
  151 + entityView.setType(type);
147 152 entityView.setName(name);
148 153 try {
149 154 entityView.setKeys(mapper.readValue(keys, TelemetryEntityView.class));
... ...
... ... @@ -19,8 +19,6 @@ import org.springframework.data.domain.Pageable;
19 19 import org.springframework.data.jpa.repository.Query;
20 20 import org.springframework.data.repository.CrudRepository;
21 21 import org.springframework.data.repository.query.Param;
22   -import org.thingsboard.server.common.data.EntityView;
23   -import org.thingsboard.server.common.data.id.EntityId;
24 22 import org.thingsboard.server.dao.model.sql.EntityViewEntity;
25 23 import org.thingsboard.server.dao.util.SqlDao;
26 24
... ... @@ -36,21 +34,46 @@ public interface EntityViewRepository extends CrudRepository<EntityViewEntity, S
36 34 "AND LOWER(e.searchText) LIKE LOWER(CONCAT(:textSearch, '%')) " +
37 35 "AND e.id > :idOffset ORDER BY e.id")
38 36 List<EntityViewEntity> findByTenantId(@Param("tenantId") String tenantId,
39   - @Param("textSearch") String textSearch,
40   - @Param("idOffset") String idOffset,
41   - Pageable pageable);
  37 + @Param("textSearch") String textSearch,
  38 + @Param("idOffset") String idOffset,
  39 + Pageable pageable);
  40 +
  41 + @Query("SELECT e FROM EntityViewEntity e WHERE e.tenantId = :tenantId " +
  42 + "AND e.type = :type " +
  43 + "AND LOWER(e.searchText) LIKE LOWER(CONCAT(:textSearch, '%')) " +
  44 + "AND e.id > :idOffset ORDER BY e.id")
  45 + List<EntityViewEntity> findByTenantIdAndType(@Param("tenantId") String tenantId,
  46 + @Param("type") String type,
  47 + @Param("textSearch") String textSearch,
  48 + @Param("idOffset") String idOffset,
  49 + Pageable pageable);
42 50
43 51 @Query("SELECT e FROM EntityViewEntity e WHERE e.tenantId = :tenantId " +
44 52 "AND e.customerId = :customerId " +
45 53 "AND LOWER(e.searchText) LIKE LOWER(CONCAT(:searchText, '%')) " +
46 54 "AND e.id > :idOffset ORDER BY e.id")
47 55 List<EntityViewEntity> findByTenantIdAndCustomerId(@Param("tenantId") String tenantId,
48   - @Param("customerId") String customerId,
49   - @Param("searchText") String searchText,
50   - @Param("idOffset") String idOffset,
51   - Pageable pageable);
  56 + @Param("customerId") String customerId,
  57 + @Param("searchText") String searchText,
  58 + @Param("idOffset") String idOffset,
  59 + Pageable pageable);
  60 +
  61 + @Query("SELECT e FROM EntityViewEntity e WHERE e.tenantId = :tenantId " +
  62 + "AND e.customerId = :customerId " +
  63 + "AND e.type = :type " +
  64 + "AND LOWER(e.searchText) LIKE LOWER(CONCAT(:searchText, '%')) " +
  65 + "AND e.id > :idOffset ORDER BY e.id")
  66 + List<EntityViewEntity> findByTenantIdAndCustomerIdAndType(@Param("tenantId") String tenantId,
  67 + @Param("customerId") String customerId,
  68 + @Param("type") String type,
  69 + @Param("searchText") String searchText,
  70 + @Param("idOffset") String idOffset,
  71 + Pageable pageable);
52 72
53 73 EntityViewEntity findByTenantIdAndName(String tenantId, String name);
54 74
55 75 List<EntityViewEntity> findAllByTenantIdAndEntityId(String tenantId, String entityId);
  76 +
  77 + @Query("SELECT DISTINCT ev.type FROM EntityViewEntity ev WHERE ev.tenantId = :tenantId")
  78 + List<String> findTenantEntityViewTypes(@Param("tenantId") String tenantId);
56 79 }
... ...
... ... @@ -24,7 +24,6 @@ import org.thingsboard.server.common.data.EntitySubtype;
24 24 import org.thingsboard.server.common.data.EntityType;
25 25 import org.thingsboard.server.common.data.EntityView;
26 26 import org.thingsboard.server.common.data.UUIDConverter;
27   -import org.thingsboard.server.common.data.id.EntityId;
28 27 import org.thingsboard.server.common.data.id.TenantId;
29 28 import org.thingsboard.server.common.data.page.TextPageLink;
30 29 import org.thingsboard.server.dao.DaoUtil;
... ... @@ -41,7 +40,6 @@ import java.util.Optional;
41 40 import java.util.UUID;
42 41
43 42 import static org.thingsboard.server.common.data.UUIDConverter.fromTimeUUID;
44   -import static org.thingsboard.server.common.data.UUIDConverter.fromTimeUUIDs;
45 43 import static org.thingsboard.server.dao.model.ModelConstants.NULL_UUID_STR;
46 44
47 45 /**
... ... @@ -76,6 +74,17 @@ public class JpaEntityViewDao extends JpaAbstractSearchTextDao<EntityViewEntity,
76 74 }
77 75
78 76 @Override
  77 + public List<EntityView> findEntityViewsByTenantIdAndType(UUID tenantId, String type, TextPageLink pageLink) {
  78 + return DaoUtil.convertDataList(
  79 + entityViewRepository.findByTenantIdAndType(
  80 + fromTimeUUID(tenantId),
  81 + type,
  82 + Objects.toString(pageLink.getTextSearch(), ""),
  83 + pageLink.getIdOffset() == null ? NULL_UUID_STR : fromTimeUUID(pageLink.getIdOffset()),
  84 + new PageRequest(0, pageLink.getLimit())));
  85 + }
  86 +
  87 + @Override
79 88 public Optional<EntityView> findEntityViewByTenantIdAndName(UUID tenantId, String name) {
80 89 return Optional.ofNullable(
81 90 DaoUtil.getData(entityViewRepository.findByTenantIdAndName(fromTimeUUID(tenantId), name)));
... ... @@ -96,8 +105,37 @@ public class JpaEntityViewDao extends JpaAbstractSearchTextDao<EntityViewEntity,
96 105 }
97 106
98 107 @Override
  108 + public List<EntityView> findEntityViewsByTenantIdAndCustomerIdAndType(UUID tenantId, UUID customerId, String type, TextPageLink pageLink) {
  109 + return DaoUtil.convertDataList(
  110 + entityViewRepository.findByTenantIdAndCustomerIdAndType(
  111 + fromTimeUUID(tenantId),
  112 + fromTimeUUID(customerId),
  113 + type,
  114 + Objects.toString(pageLink.getTextSearch(), ""),
  115 + pageLink.getIdOffset() == null ? NULL_UUID_STR : fromTimeUUID(pageLink.getIdOffset()),
  116 + new PageRequest(0, pageLink.getLimit())
  117 + ));
  118 + }
  119 +
  120 + @Override
99 121 public ListenableFuture<List<EntityView>> findEntityViewsByTenantIdAndEntityIdAsync(UUID tenantId, UUID entityId) {
100 122 return service.submit(() -> DaoUtil.convertDataList(
101 123 entityViewRepository.findAllByTenantIdAndEntityId(UUIDConverter.fromTimeUUID(tenantId), UUIDConverter.fromTimeUUID(entityId))));
102 124 }
  125 +
  126 + @Override
  127 + public ListenableFuture<List<EntitySubtype>> findTenantEntityViewTypesAsync(UUID tenantId) {
  128 + return service.submit(() -> convertTenantEntityViewTypesToDto(tenantId, entityViewRepository.findTenantEntityViewTypes(fromTimeUUID(tenantId))));
  129 + }
  130 +
  131 + private List<EntitySubtype> convertTenantEntityViewTypesToDto(UUID tenantId, List<String> types) {
  132 + List<EntitySubtype> list = Collections.emptyList();
  133 + if (types != null && !types.isEmpty()) {
  134 + list = new ArrayList<>();
  135 + for (String type : types) {
  136 + list.add(new EntitySubtype(new TenantId(tenantId), EntityType.ENTITY_VIEW, type));
  137 + }
  138 + }
  139 + return list;
  140 + }
103 141 }
... ...
... ... @@ -87,7 +87,11 @@ public class BaseTimeseriesService implements TimeseriesService {
87 87 .map(key -> new BaseReadTsKvQuery(key, entityView.getStartTimeMs(), entityView.getEndTimeMs(), 1, "ASC"))
88 88 .collect(Collectors.toList());
89 89
90   - return timeseriesDao.findAllAsync(entityView.getEntityId(), updateQueriesForEntityView(entityView, queries));
  90 + if (queries.size() > 0) {
  91 + return timeseriesDao.findAllAsync(entityView.getEntityId(), queries);
  92 + } else {
  93 + return Futures.immediateFuture(new ArrayList<>());
  94 + }
91 95 }
92 96 keys.forEach(key -> futures.add(timeseriesDao.findLatest(entityId, key)));
93 97 return Futures.allAsList(futures);
... ... @@ -133,11 +137,20 @@ public class BaseTimeseriesService implements TimeseriesService {
133 137
134 138 private List<ReadTsKvQuery> updateQueriesForEntityView(EntityView entityView, List<ReadTsKvQuery> queries) {
135 139 return queries.stream().map(query -> {
136   - long startTs = entityView.getStartTimeMs() == 0 ? query.getStartTs() : entityView.getStartTimeMs();
137   - long endTs = entityView.getEndTimeMs() == 0 ? query.getEndTs() : entityView.getEndTimeMs();
  140 + long startTs;
  141 + if (entityView.getStartTimeMs() != 0 && entityView.getStartTimeMs() > query.getStartTs()) {
  142 + startTs = entityView.getStartTimeMs();
  143 + } else {
  144 + startTs = query.getStartTs();
  145 + }
138 146
139   - return startTs <= query.getStartTs() && endTs >= query.getEndTs() ? query :
140   - new BaseReadTsKvQuery(query.getKey(), startTs, endTs, query.getInterval(), query.getLimit(), query.getAggregation());
  147 + long endTs;
  148 + if (entityView.getEndTimeMs() != 0 && entityView.getEndTimeMs() < query.getEndTs()) {
  149 + endTs = entityView.getEndTimeMs();
  150 + } else {
  151 + endTs = query.getEndTs();
  152 + }
  153 + return new BaseReadTsKvQuery(query.getKey(), startTs, endTs, query.getInterval(), query.getLimit(), query.getAggregation());
141 154 }).collect(Collectors.toList());
142 155 }
143 156
... ...
... ... @@ -624,61 +624,90 @@ CREATE TABLE IF NOT EXISTS thingsboard.rule_node (
624 624 PRIMARY KEY (id)
625 625 );
626 626
627   -CREATE TABLE IF NOT EXISTS thingsboard.entity_views (
  627 +CREATE TABLE IF NOT EXISTS thingsboard.entity_view (
628 628 id timeuuid,
629 629 entity_id timeuuid,
630 630 entity_type text,
631 631 tenant_id timeuuid,
632 632 customer_id timeuuid,
633 633 name text,
  634 + type text,
634 635 keys text,
635 636 start_ts bigint,
636 637 end_ts bigint,
637 638 search_text text,
638 639 additional_info text,
639   - PRIMARY KEY (id, entity_id, tenant_id, customer_id)
  640 + PRIMARY KEY (id, entity_id, tenant_id, customer_id, type)
640 641 );
641 642
642 643 CREATE MATERIALIZED VIEW IF NOT EXISTS thingsboard.entity_view_by_tenant_and_name AS
643 644 SELECT *
644   - from thingsboard.entity_views
  645 + from thingsboard.entity_view
645 646 WHERE tenant_id IS NOT NULL
646 647 AND entity_id IS NOT NULL
647 648 AND customer_id IS NOT NULL
  649 + AND type IS NOT NULL
648 650 AND name IS NOT NULL
649 651 AND id IS NOT NULL
650   - PRIMARY KEY (tenant_id, name, id, customer_id, entity_id)
  652 + PRIMARY KEY (tenant_id, name, id, customer_id, entity_id, type)
651 653 WITH CLUSTERING ORDER BY (name ASC, id DESC, customer_id DESC);
652 654
653 655 CREATE MATERIALIZED VIEW IF NOT EXISTS thingsboard.entity_view_by_tenant_and_search_text AS
654 656 SELECT *
655   - from thingsboard.entity_views
  657 + from thingsboard.entity_view
656 658 WHERE tenant_id IS NOT NULL
657 659 AND entity_id IS NOT NULL
658 660 AND customer_id IS NOT NULL
  661 + AND type IS NOT NULL
659 662 AND search_text IS NOT NULL
660 663 AND id IS NOT NULL
661   - PRIMARY KEY (tenant_id, search_text, id, customer_id, entity_id)
  664 + PRIMARY KEY (tenant_id, search_text, id, customer_id, entity_id, type)
662 665 WITH CLUSTERING ORDER BY (search_text ASC, id DESC, customer_id DESC);
663 666
  667 +CREATE MATERIALIZED VIEW IF NOT EXISTS thingsboard.entity_view_by_tenant_by_type_and_search_text AS
  668 + SELECT *
  669 + from thingsboard.entity_view
  670 + WHERE tenant_id IS NOT NULL
  671 + AND entity_id IS NOT NULL
  672 + AND customer_id IS NOT NULL
  673 + AND type IS NOT NULL
  674 + AND search_text IS NOT NULL
  675 + AND id IS NOT NULL
  676 + PRIMARY KEY (tenant_id, type, search_text, id, customer_id, entity_id)
  677 + WITH CLUSTERING ORDER BY (type ASC, search_text ASC, id DESC, customer_id DESC);
  678 +
664 679 CREATE MATERIALIZED VIEW IF NOT EXISTS thingsboard.entity_view_by_tenant_and_customer AS
665 680 SELECT *
666   - from thingsboard.entity_views
  681 + from thingsboard.entity_view
667 682 WHERE tenant_id IS NOT NULL
668 683 AND customer_id IS NOT NULL
669 684 AND entity_id IS NOT NULL
  685 + AND type IS NOT NULL
670 686 AND search_text IS NOT NULL
671 687 AND id IS NOT NULL
672   - PRIMARY KEY (tenant_id, customer_id, search_text, id, entity_id)
  688 + PRIMARY KEY (tenant_id, customer_id, search_text, id, entity_id, type)
673 689 WITH CLUSTERING ORDER BY (customer_id DESC, search_text ASC, id DESC);
674 690
  691 +CREATE MATERIALIZED VIEW IF NOT EXISTS thingsboard.entity_view_by_tenant_and_customer_and_type AS
  692 + SELECT *
  693 + from thingsboard.entity_view
  694 + WHERE tenant_id IS NOT NULL
  695 + AND customer_id IS NOT NULL
  696 + AND entity_id IS NOT NULL
  697 + AND type IS NOT NULL
  698 + AND search_text IS NOT NULL
  699 + AND id IS NOT NULL
  700 + PRIMARY KEY (tenant_id, type, customer_id, search_text, id, entity_id)
  701 + WITH CLUSTERING ORDER BY (type ASC, customer_id DESC, search_text ASC, id DESC);
  702 +
675 703 CREATE MATERIALIZED VIEW IF NOT EXISTS thingsboard.entity_view_by_tenant_and_entity_id AS
676 704 SELECT *
677   - from thingsboard.entity_views
  705 + from thingsboard.entity_view
678 706 WHERE tenant_id IS NOT NULL
679 707 AND customer_id IS NOT NULL
680 708 AND entity_id IS NOT NULL
  709 + AND type IS NOT NULL
681 710 AND search_text IS NOT NULL
682 711 AND id IS NOT NULL
683   - PRIMARY KEY (tenant_id, entity_id, customer_id, search_text, id)
  712 + PRIMARY KEY (tenant_id, entity_id, customer_id, search_text, id, type)
684 713 WITH CLUSTERING ORDER BY (entity_id DESC, customer_id DESC, search_text ASC, id DESC);
\ No newline at end of file
... ...
... ... @@ -41,242 +41,4 @@ VALUES ( now ( ), 'mail', '{
41 41 "enableTls": "false",
42 42 "username": "",
43 43 "password": ""
44   -}' );
45   -
46   -/** System widgets library **/
47   -INSERT INTO "thingsboard"."widgets_bundle" ( "id", "tenant_id", "alias", "search_text", "title" )
48   -VALUES ( now ( ), minTimeuuid ( 0 ), 'gpio_widgets', 'gpio widgets', 'GPIO widgets' );
49   -
50   -INSERT INTO "thingsboard"."widgets_bundle" ( "id", "tenant_id", "alias", "search_text", "title" )
51   -VALUES ( now ( ), minTimeuuid ( 0 ), 'maps', 'maps', 'Maps' );
52   -
53   -INSERT INTO "thingsboard"."widgets_bundle" ( "id", "tenant_id", "alias", "search_text", "title" )
54   -VALUES ( now ( ), minTimeuuid ( 0 ), 'digital_gauges', 'digital gauges', 'Digital gauges' );
55   -
56   -INSERT INTO "thingsboard"."widgets_bundle" ( "id", "tenant_id", "alias", "search_text", "title" )
57   -VALUES ( now ( ), minTimeuuid ( 0 ), 'charts', 'charts', 'Charts' );
58   -
59   -INSERT INTO "thingsboard"."widgets_bundle" ( "id", "tenant_id", "alias", "search_text", "title" )
60   -VALUES ( now ( ), minTimeuuid ( 0 ), 'cards', 'cards', 'Cards' );
61   -
62   -INSERT INTO "thingsboard"."widgets_bundle" ( "id", "tenant_id", "alias", "search_text", "title" )
63   -VALUES ( now ( ), minTimeuuid ( 0 ), 'analogue_gauges', 'analogue gauges', 'Analogue gauges' );
64   -
65   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
66   -VALUES ( now ( ), minTimeuuid ( 0 ), 'charts', 'timeseries',
67   -'{"type":"timeseries","sizeX":8,"sizeY":5,"resources":[{"url":"https://cdnjs.cloudflare.com/ajax/libs/Chart.js/2.4.0/Chart.min.js"}],"templateHtml":"<canvas id=\"lineChart\"></canvas>\n","templateCss":"","controllerScript":"self.onInit = function() {\n \n var lineData = {\n labels: [],\n datasets: []\n };\n \n for (var i=0; i < self.ctx.data.length; i++) {\n var dataKey = self.ctx.data[i].dataKey;\n var keySettings = dataKey.settings;\n var backgroundColor = tinycolor(dataKey.color);\n backgroundColor.setAlpha(0.4);\n var dataset = {\n label: dataKey.label,\n data: [],\n borderColor: dataKey.color,\n borderWidth: 2,\n backgroundColor: backgroundColor.toRgbString(),\n pointRadius: keySettings.showPoints ? 1 : 0,\n fill: keySettings.fillLines || false,\n showLine: keySettings.showLines || true,\n spanGaps: false,\n lineTension: angular.isDefined(keySettings.tension) ? keySettings.tension : 0.2\n }\n lineData.datasets.push(dataset);\n }\n\n var ctx = $(''#lineChart'', self.ctx.$container);\n self.ctx.chart = new Chart(ctx, {\n type: ''line'',\n data: lineData,\n options: {\n responsive: false,\n tooltips: {\n mode: ''index'',\n inersect: true\n },\n hover: {\n mode: ''index'',\n inersect: true\n },\n maintainAspectRatio: false,\n /*animation: {\n duration: 200,\n easing: ''linear''\n },*/\n elements: {\n line: {\n tension: 0.2\n } \n },\n scales: {\n xAxes: [{\n type: ''time'',\n ticks: {\n maxRotation: 20,\n autoSkip: true\n },\n time: {\n displayFormats: {\n second: ''hh:mm:ss'',\n minute: ''hh:mm:ss''\n }\n }\n }]\n },\n zoom: {\n onSelect: function(startTimeMs, endTimeMs) {\n self.ctx.timewindowFunctions.onUpdateTimewindow(startTimeMs, endTimeMs);\n },\n onResetSelect: function() {\n self.ctx.timewindowFunctions.onResetTimewindow();\n }\n }\n }\n });\n \n self.onResize();\n \n}\n\nself.onDataUpdated = function() {\n \n if (self.ctx.chart.zoom.isMouseInteraction) {\n return;\n }\n if (!self.ctx.tickUpdate) {\n for (var i = 0; i < self.ctx.data.length; i++) {\n var dataSetData = [];\n var dataKeyData = self.ctx.data[i].data;\n for (var i2 = 0; i2 < dataKeyData.length; i2 ++) {\n dataSetData.push({x: moment(dataKeyData[i2][0]), y: dataKeyData[i2][1]});\n \n }\n self.ctx.chart.data.datasets[i].data = dataSetData; \n }\n }\n \n self.ctx.chart.options.scales.xAxes[0].time.min = moment(self.ctx.timeWindow.minTime);\n self.ctx.chart.options.scales.xAxes[0].time.max = moment(self.ctx.timeWindow.maxTime);\n \n self.ctx.chart.update(0, true);\n\n}\n\nself.onResize = function() {\n self.ctx.chart.resize();\n self.ctx.chart.update(0, true);\n}\n\nself.onDestroy = function() {\n}\n\nfunction getYAxis(chartInstance) {\n var scales = chartInstance.scales;\n for (var scaleId in scales) {\n\t var scale = scales[scaleId];\n\n\t if (!scale.isHorizontal()) {\n\t\t return scale;\n\t }\n }\n}\n\nfunction getXAxis(chartInstance) {\n var scales = chartInstance.scales;\n for (var scaleId in scales) {\n\t var scale = scales[scaleId];\n\n\t if (scale.isHorizontal()) {\n\t\t return scale;\n\t }\n }\n}\n\nfunction eventPointer (e) {\n if (angular.isDefined(e.touches) && e.touches.length > 0) {\n return {\n x : e.touches[0].pageX,\n y : e.touches[0].pageY\n };\n } else if (angular.isDefined(e.changedTouches) && e.changedTouches.length > 0) {\n return {\n x : e.changedTouches[0].pageX,\n y : e.changedTouches[0].pageY\n };\n } else if (e.pageX || e.pageY) {\n return {\n x : e.pageX,\n y : e.pageY\n };\n } else if (e.clientX || e.clientY) {\n var\n d = document,\n b = d.body,\n de = d.documentElement;\n return {\n x: e.clientX + b.scrollLeft + de.scrollLeft,\n y: e.clientY + b.scrollTop + de.scrollTop\n };\n }\n}\n\nvar zoomPlugin = {\n beforeInit: function(chartInstance) {\n chartInstance.zoom = {};\n var node = chartInstance.zoom.node = chartInstance.chart.ctx.canvas;\n \n chartInstance.zoom.mouseDownHandler = function(event) {\n chartInstance.zoom.dragZoomStart = event;\n chartInstance.zoom.dragZoomStartPointer = eventPointer(event);\n chartInstance.zoom.isMouseInteraction = true;\n };\n\n node.addEventListener(''mousedown'', chartInstance.zoom.mouseDownHandler);\n \n chartInstance.zoom.mouseMoveHandler = function(event) {\n if (chartInstance.zoom.dragZoomStart) {\n chartInstance.update(0);\n chartInstance.zoom.dragZoomEnd = event;\n chartInstance.zoom.dragZoomEndPointer = eventPointer(event);\n }\n };\n \n node.addEventListener(''mousemove'', chartInstance.zoom.mouseMoveHandler);\n \n chartInstance.zoom.mouseUpHandler = function(event) {\n if (chartInstance.zoom.dragZoomStart) {\n \n var chartArea = chartInstance.chartArea;\n var yAxis = getYAxis(chartInstance);\n\t\t\t\t\tvar beginPoint = chartInstance.zoom.dragZoomStart;\n\t\t\t\t\tvar beginPointer = chartInstance.zoom.dragZoomStartPointer;\n\t\t\t\t\tvar upEventPointer = eventPointer(event);\n\t\t\t\t\tvar offsetX = beginPoint.target.getBoundingClientRect().left;\n\t\t\t\t\tvar startX = Math.min(beginPointer.x, upEventPointer.x) - offsetX;\n\t\t\t\t\tvar endX = Math.max(beginPointer.x, upEventPointer.x) - offsetX;\n\t\t\t\t\tvar dragDistance = endX - startX;\n\t\t\t\t\t\n\t\t\t\t\tif (dragDistance > 0) {\n \t\t\t\t\tvar xAxis = getXAxis(chartInstance);\n \t\t\t\t\tvar options = chartInstance.options;\n \t\t\t\t\tif (options.scales.xAxes[0].time) {\n \t\t\t\t\t startX = Math.max(startX, xAxis.left);\n \t\t\t\t\t endX = Math.min(endX, xAxis.right);\n \t\t\t\t\t if (endX - startX > 0) {\n \t\t\t\t\t startX = startX - xAxis.left;\n \t\t\t\t\t endX = endX - xAxis.left;\n \t\t\t\t\t var time = options.scales.xAxes[0].time;\n \t\t\t\t\t var min = time.min.valueOf();\n \t\t\t\t\t var max = time.max.valueOf();\n \t\t\t\t\t var axisDistance = xAxis.right - xAxis.left;\n \t\t\t\t\t var timeDistance = max - min;\n \t\t\t\t\t \n \t\t\t\t\t var zoomStartTime = min + startX / axisDistance * timeDistance;\n \t\t\t\t\t var zoomEndTime = min + endX / axisDistance * timeDistance;\n\n \t\t\t\t\t if (options.zoom && options.zoom.onSelect) {\n \t\t\t\t\t options.zoom.onSelect(zoomStartTime, zoomEndTime);\n \t\t\t\t\t }\n \t\t\t\t\t }\n \t\t\t\t\t}\n\t\t\t\t\t}\n \t\t\tchartInstance.zoom.dragZoomStart = null;\n \t\t\tchartInstance.zoom.dragZoomEnd = null; \n }\n chartInstance.zoom.isMouseInteraction = false;\n };\n \n node.addEventListener(''mouseup'', chartInstance.zoom.mouseUpHandler);\n \n chartInstance.zoom.mouseLeaveHandler = function(event) {\n if (chartInstance.zoom.dragZoomStart) {\n \t\t\tchartInstance.zoom.dragZoomStart = null;\n \t\t\tchartInstance.zoom.dragZoomEnd = null; \n }\n chartInstance.zoom.isMouseInteraction = false;\n };\n \n node.addEventListener(''mouseleave'', chartInstance.zoom.mouseLeaveHandler);\n \n chartInstance.zoom.dblClickHandler = function(event) {\n if (chartInstance.zoom.dragZoomStart) {\n \t\t\tchartInstance.zoom.dragZoomStart = null;\n \t\t\tchartInstance.zoom.dragZoomEnd = null; \n }\n var options = chartInstance.options;\n if (options.zoom && options.zoom.onResetSelect) {\n options.zoom.onResetSelect();\n }\n };\n \n node.addEventListener(''dblclick'', chartInstance.zoom.dblClickHandler);\n },\n beforeDatasetsDraw: function(chartInstance, easing) {\n \t\tvar ctx = chartInstance.chart.ctx;\n \t\tvar chartArea = chartInstance.chartArea;\n \t\tctx.save();\n \t\tctx.beginPath();\n \t\tif (chartInstance.zoom && chartInstance.zoom.dragZoomEnd) {\n \t\t\tvar yAxis = getYAxis(chartInstance);\n \t\t\tvar beginPoint = chartInstance.zoom.dragZoomStart;\n \t\t\tvar beginPointer = chartInstance.zoom.dragZoomStartPointer;\n \t\t\tvar endPoint = chartInstance.zoom.dragZoomEnd;\n \t\t\tvar endPointer = chartInstance.zoom.dragZoomEndPointer;\n \t\t\t\n \t\t\tvar offsetX = beginPoint.target.getBoundingClientRect().left;\n \t\t\tvar startX = Math.min(beginPointer.x, endPointer.x) - offsetX;\n \t\t\tvar endX = Math.max(beginPointer.x, endPointer.x) - offsetX;\n \t\t\tvar rectWidth = endX - startX;\n \t\t\tctx.fillStyle = ''rgba(157,203,255,0.1)'';\n \t\t\tctx.lineWidth = 1;\n \t\t\tctx.strokeRect(startX, yAxis.top, rectWidth, yAxis.bottom - yAxis.top);\n \t\t\tctx.fillRect(startX, yAxis.top, rectWidth, yAxis.bottom - yAxis.top);\n \t\t}\n \t\tif (chartArea) {\n \t\t ctx.rect(chartArea.left, chartArea.top, chartArea.right - chartArea.left, chartArea.bottom - chartArea.top);\n \t\t}\n\t\t ctx.clip(); \n },\n \n afterDatasetsDraw: function(chartInstance) {\n\t chartInstance.chart.ctx.restore();\n },\n \n destroy: function(chartInstance) {\n if (chartInstance.zoom) {\n var node = chartInstance.zoom.node;\n\t\t\t\tnode.removeEventListener(''mousedown'', chartInstance.zoom.mouseDownHandler);\n\t\t\t\tnode.removeEventListener(''mousemove'', chartInstance.zoom.mouseMoveHandler);\n\t\t\t\tnode.removeEventListener(''mouseup'', chartInstance.zoom.mouseUpHandler);\n\t\t\t\tnode.removeEventListener(''mouseleave'', chartInstance.zoom.mouseLeaveHandler);\t \n\t\t\t\tnode.removeEventListener(''dblclick'', chartInstance.zoom.dblClickHandler);\n\t\t\t\tdelete chartInstance.zoom;\n }\n }\n };\n\nChart.pluginService.register(zoomPlugin);\n","settingsSchema":"{}","dataKeySettingsSchema":"{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"DataKeySettings\",\n \"properties\": {\n \"showLines\": {\n \"title\": \"Show lines\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"fillLines\": {\n \"title\": \"Fill lines\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"showPoints\": {\n \"title\": \"Show points\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"tension\": {\n \"title\": \"Line tension\",\n \"type\": \"number\",\n \"default\": 0.2\n }\n },\n \"required\": [\"showLines\", \"fillLines\", \"showPoints\"]\n },\n \"form\": [\n \"showLines\",\n \"fillLines\",\n \"showPoints\",\n \"tension\"\n ]\n}","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"First\",\"color\":\"#2196f3\",\"settings\":{\"showLines\":true,\"fillLines\":false,\"showPoints\":false},\"_hash\":0.5644745944820795,\"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;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Second\",\"color\":\"#ffc107\",\"settings\":{\"showLines\":true,\"fillLines\":true,\"showPoints\":false},\"_hash\":0.18379294198604845,\"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\":\"Timeseries - Chart.js (Deprecated)\"}"}',
68   -'Timeseries - Chart.js (Deprecated)' );
69   -
70   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
71   -VALUES ( now ( ), minTimeuuid ( 0 ), 'digital_gauges',
72   -'digital_vertical_bar',
73   -'{"type":"latest","sizeX":2.5,"sizeY":4.5,"resources":[],"templateHtml":"<canvas id=\"digitalGauge\"></canvas>","templateCss":"#gauge {\n text-align: center;\n /* margin-left: -100px;\n margin-right: -100px;*/\n /*margin-top: -50px;*/\n \n}\n","controllerScript":"self.onInit = function() {\n self.ctx.gauge = new TbCanvasDigitalGauge(self.ctx, ''digitalGauge''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.gauge.update();\n}\n\nself.onResize = function() {\n self.ctx.gauge.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbCanvasDigitalGauge.settingsSchema;\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.gauge.mobileModeChanged();\n}\n\nself.onDestroy = function() {\n}\n\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Temp\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.7282710489093589,\"funcBody\":\"var value = prevValue + Math.random() * 40 - 20;\\nif (value < -60) {\\n\\tvalue = -60;\\n} else if (value > 60) {\\n\\tvalue = 60;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"#000000\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"0px\",\"settings\":{\"maxValue\":60,\"donutStartAngle\":90,\"showValue\":true,\"showMinMax\":true,\"gaugeWidthScale\":0.75,\"levelColors\":[\"#3d5afe\",\"#f44336\"],\"refreshAnimationType\":\"<>\",\"refreshAnimationTime\":700,\"startAnimationType\":\"<>\",\"startAnimationTime\":700,\"titleFont\":{\"family\":\"Roboto\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\"},\"labelFont\":{\"family\":\"Roboto\",\"size\":8,\"style\":\"normal\",\"weight\":\"500\"},\"valueFont\":{\"family\":\"Segment7Standard\",\"style\":\"normal\",\"weight\":\"500\",\"size\":14},\"minMaxFont\":{\"family\":\"Segment7Standard\",\"size\":8,\"style\":\"normal\",\"weight\":\"normal\",\"color\":\"#cccccc\"},\"neonGlowBrightness\":20,\"decimals\":0,\"showUnitTitle\":true,\"gaugeColor\":\"#171a1c\",\"gaugeType\":\"verticalBar\",\"showTitle\":false,\"units\":\"°C\",\"minValue\":-60,\"dashThickness\":1.2},\"title\":\"Digital vertical bar\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',
74   -'Digital vertical bar' );
75   -
76   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
77   -VALUES ( now ( ), minTimeuuid ( 0 ), 'gpio_widgets', 'gpio_panel',
78   -'{"type":"latest","sizeX":5,"sizeY":2,"resources":[],"templateHtml":"<div class=\"gpio-panel\" style=\"height: 100%;\">\n <section layout=\"row\" ng-repeat=\"row in rows\">\n <section flex layout=\"row\" ng-repeat=\"cell in row\">\n <section layout=\"row\" flex ng-if=\"cell\" layout-align=\"{{$index===0 ? ''end center'' : ''start center''}}\">\n <span class=\"gpio-left-label\" ng-show=\"$index===0\">{{ cell.label }}</span>\n <section layout=\"row\" class=\"led-panel\" ng-class=\"$index===0 ? ''col-0'' : ''col-1''\"\n ng-style=\"{backgroundColor: ledPanelBackgroundColor }\">\n <span class=\"pin\" ng-show=\"$index===0\">{{cell.pin}}</span>\n <span class=\"led-container\">\n <tb-led-light size=\"prefferedRowHeight\"\n color-on=\"cell.colorOn\"\n color-off=\"cell.colorOff\"\n off-opacity=\"''0.9''\"\n tb-enabled=\"cell.enabled\">\n </tb-led-light>\n </span>\n <span class=\"pin\" ng-show=\"$index===1\">{{cell.pin}}</span>\n </section>\n <span class=\"gpio-right-label\" ng-show=\"$index===1\">{{ cell.label }}</span>\n </section>\n <section layout=\"row\" flex ng-if=\"!cell\">\n <span flex ng-show=\"$index===0\"></span>\n <span class=\"led-panel\"\n ng-style=\"{backgroundColor: ledPanelBackgroundColor }\"></span>\n <span flex ng-show=\"$index===1\"></span>\n </section>\n </section>\n </section> \n</div>","templateCss":".error {\n font-size: 14px !important;\n color: maroon;/*rgb(250,250,250);*/\n background-color: transparent;\n padding: 6px;\n}\n\n.error span {\n margin: auto;\n}\n\n.gpio-panel {\n padding-top: 10px;\n white-space: nowrap;\n}\n\n.gpio-panel tb-led-light > div {\n margin: auto;\n}\n\n.led-panel {\n margin: 0;\n width: 66px;\n min-width: 66px;\n}\n\n.led-container {\n width: 48px;\n min-width: 48px;\n}\n\n.pin {\n margin-top: auto;\n margin-bottom: auto;\n color: white;\n font-size: 12px;\n width: 16px;\n min-width: 16px;\n}\n\n.led-panel.col-0 .pin {\n margin-left: auto;\n padding-left: 2px;\n text-align: right;\n}\n\n.led-panel.col-1 .pin {\n margin-right: auto;\n \n text-align: left;\n}\n\n.gpio-left-label {\n margin-right: 8px;\n}\n\n.gpio-right-label {\n margin-left: 8px;\n}","controllerScript":"self.onInit = function() {\n var i, gpio;\n \n var scope = self.ctx.$scope;\n var settings = self.ctx.settings;\n \n scope.gpioList = [];\n scope.gpioByPin = {};\n for (var g = 0; g < settings.gpioList.length; g++) {\n gpio = settings.gpioList[g];\n scope.gpioList.push(\n {\n row: gpio.row,\n col: gpio.col,\n pin: gpio.pin,\n label: gpio.label,\n enabled: false,\n colorOn: tinycolor(gpio.color).lighten(20).toHexString(),\n colorOff: tinycolor(gpio.color).darken().toHexString()\n }\n );\n scope.gpioByPin[gpio.pin] = scope.gpioList[scope.gpioList.length-1];\n }\n\n scope.ledPanelBackgroundColor = settings.ledPanelBackgroundColor || tinycolor(''green'').lighten(2).toRgbString();\n\n scope.gpioCells = {};\n var rowCount = 0;\n for (i = 0; i < scope.gpioList.length; i++) {\n gpio = scope.gpioList[i];\n scope.gpioCells[gpio.row+''_''+gpio.col] = gpio;\n rowCount = Math.max(rowCount, gpio.row+1);\n }\n \n scope.prefferedRowHeight = 32;\n scope.rows = [];\n for (i = 0; i < rowCount; i++) {\n var row = [];\n for (var c =0; c<2;c++) {\n if (scope.gpioCells[i+''_''+c]) {\n row[c] = scope.gpioCells[i+''_''+c];\n } else {\n row[c] = null;\n }\n }\n scope.rows.push(row);\n } \n \n self.onResize();\n}\n\nself.onDataUpdated = function() {\n var changed = false;\n for (var d = 0; d < self.ctx.data.length; d++) {\n var cellData = self.ctx.data[d];\n var dataKey = cellData.dataKey;\n var gpio = self.ctx.$scope.gpioByPin[dataKey.label];\n if (gpio) {\n var enabled = false;\n if (cellData.data.length > 0) {\n var tvPair = cellData.data[cellData.data.length - 1];\n enabled = (tvPair[1] === true || tvPair[1] === ''true'');\n }\n if (gpio.enabled != enabled) {\n changed = true;\n gpio.enabled = enabled;\n }\n }\n }\n if (changed) {\n self.ctx.$scope.$digest();\n } \n}\n\nself.onResize = function() {\n var rowCount = self.ctx.$scope.rows.length;\n var prefferedRowHeight = (self.ctx.height - 35)/rowCount;\n prefferedRowHeight = Math.min(32, prefferedRowHeight);\n prefferedRowHeight = Math.max(12, prefferedRowHeight);\n self.ctx.$scope.prefferedRowHeight = prefferedRowHeight;\n \n var ratio = prefferedRowHeight/32;\n \n var leftLabels = $(''.gpio-left-label'', self.ctx.$container);\n leftLabels.css(''font-size'', 16*ratio+''px'');\n var rightLabels = $(''.gpio-right-label'', self.ctx.$container);\n rightLabels.css(''font-size'', 16*ratio+''px'');\n var pins = $(''.pin'', self.ctx.$container);\n var pinsFontSize = Math.max(9, 12*ratio);\n pins.css(''font-size'', pinsFontSize+''px''); \n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"Settings\",\n \"properties\": {\n \"gpioList\": {\n \"title\": \"Gpio leds\",\n \"type\": \"array\",\n \"minItems\" : 1,\n \"items\": {\n \"title\": \"Gpio led\",\n \"type\": \"object\",\n \"properties\": {\n \"pin\": {\n \"title\": \"Pin\",\n \"type\": \"number\"\n },\n \"label\": {\n \"title\": \"Label\",\n \"type\": \"string\"\n },\n \"row\": {\n \"title\": \"Row\",\n \"type\": \"number\"\n },\n \"col\": {\n \"title\": \"Column\",\n \"type\": \"number\"\n },\n \"color\": {\n \"title\": \"Color\",\n \"type\": \"string\",\n \"default\": \"red\"\n }\n },\n \"required\": [\"pin\", \"label\", \"row\", \"col\", \"color\"]\n }\n },\n \"ledPanelBackgroundColor\": {\n \"title\": \"LED panel background color\",\n \"type\": \"string\",\n \"default\": \"#008a00\"\n } \n },\n \"required\": [\"gpioList\", \n \"ledPanelBackgroundColor\"]\n },\n \"form\": [\n {\n \"key\": \"gpioList\",\n \"items\": [\n \"gpioList[].pin\",\n \"gpioList[].label\",\n \"gpioList[].row\",\n \"gpioList[].col\",\n {\n \"key\": \"gpioList[].color\",\n \"type\": \"color\"\n }\n ]\n },\n {\n \"key\": \"ledPanelBackgroundColor\",\n \"type\": \"color\"\n }\n ]\n}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"0px\",\"settings\":{\"gpioList\":[{\"pin\":1,\"label\":\"GPIO 1\",\"row\":0,\"col\":0,\"color\":\"#008000\",\"_uniqueKey\":0},{\"pin\":2,\"label\":\"GPIO 2\",\"row\":0,\"col\":1,\"color\":\"#ffff00\",\"_uniqueKey\":1},{\"pin\":3,\"label\":\"GPIO 3\",\"row\":1,\"col\":0,\"color\":\"#cf006f\",\"_uniqueKey\":2}],\"ledPanelBackgroundColor\":\"#b71c1c\"},\"title\":\"Basic GPIO Panel\",\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"1\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.22518255793320163,\"funcBody\":\"var period = time % 1500;\\nreturn period < 500;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"2\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.7008206860666621,\"funcBody\":\"var period = time % 1500;\\nreturn period >= 500 && period < 1000;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"3\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.42600325102193426,\"funcBody\":\"var period = time % 1500;\\nreturn period >= 1000;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}}}"}',
79   -'Basic GPIO Panel' );
80   -
81   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
82   -VALUES ( now ( ), minTimeuuid ( 0 ), 'cards', 'timeseries_table',
83   -'{"type":"timeseries","sizeX":8,"sizeY":6.5,"resources":[],"templateHtml":"<tb-timeseries-table-widget \n config=\"config\"\n table-id=\"tableId\"\n datasources=\"datasources\"\n data=\"data\">\n</tb-timeseries-table-widget>","templateCss":"","controllerScript":"self.onInit = function() {\n \n var scope = self.ctx.$scope;\n var id = self.ctx.$scope.$injector.get(''utils'').guid();\n\n scope.config = {\n settings: self.ctx.settings,\n widgetConfig: self.ctx.widgetConfig\n }\n\n scope.datasources = self.ctx.datasources;\n scope.data = self.ctx.data;\n scope.tableId = \"table-\"+id;\n \n}\n\nself.onDataUpdated = function() {\n self.ctx.$scope.data = self.ctx.data;\n self.ctx.$scope.$broadcast(''timeseries-table-data-updated'', self.ctx.$scope.tableId);\n}\n\nself.onDestroy = function() {\n}","settingsSchema":"{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"TimeseriesTableSettings\",\n \"properties\": {\n \"showTimestamp\": {\n \"title\": \"Display timestamp column\",\n \"type\": \"boolean\",\n \"default\": true\n }\n },\n \"required\": []\n },\n \"form\": [\n \"showTimestamp\"\n ]\n}","dataKeySettingsSchema":"{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"DataKeySettings\",\n \"properties\": {\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 \"useCellContentFunction\": {\n \"title\": \"Use cell content function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"cellContentFunction\": {\n \"title\": \"Cell content function: f(value, rowData, filter)\",\n \"type\": \"string\",\n \"default\": \"\"\n }\n },\n \"required\": []\n },\n \"form\": [\n \"useCellStyleFunction\",\n {\n \"key\": \"cellStyleFunction\",\n \"type\": \"javascript\"\n },\n \"useCellContentFunction\",\n {\n \"key\": \"cellContentFunction\",\n \"type\": \"javascript\"\n }\n ]\n}","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Temperature °C\",\"color\":\"#2196f3\",\"settings\":{\"useCellStyleFunction\":true,\"cellStyleFunction\":\"if (value) {\\n var percent = (value + 60)/120 * 100;\\n var color = tinycolor.mix(''blue'', ''red'', amount = percent);\\n color.setAlpha(.5);\\n return {\\n paddingLeft: ''20px'',\\n color: ''#ffffff'',\\n background: color.toRgbString(),\\n fontSize: ''18px''\\n };\\n} else {\\n return {};\\n}\"},\"_hash\":0.8587686344902596,\"funcBody\":\"var value = prevValue + Math.random() * 40 - 20;\\nvar multiplier = Math.pow(10, 1 || 0);\\nvar value = Math.round(value * multiplier) / multiplier;\\nif (value < -60) {\\n\\tvalue = -60;\\n} else if (value > 60) {\\n\\tvalue = 60;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Humidity, %\",\"color\":\"#ffc107\",\"settings\":{\"useCellStyleFunction\":true,\"cellStyleFunction\":\"if (value) {\\n var percent = value;\\n var backgroundColor = tinycolor(''blue'');\\n backgroundColor.setAlpha(value/100);\\n var color = ''blue'';\\n if (value > 50) {\\n color = ''white'';\\n }\\n \\n return {\\n paddingLeft: ''20px'',\\n color: color,\\n background: backgroundColor.toRgbString(),\\n fontSize: ''18px''\\n };\\n} else {\\n return {};\\n}\",\"useCellContentFunction\":false},\"_hash\":0.12775350966079668,\"funcBody\":\"var value = prevValue + Math.random() * 20 - 10;\\nvar multiplier = Math.pow(10, 1 || 0);\\nvar value = Math.round(value * multiplier) / multiplier;\\nif (value < 5) {\\n\\tvalue = 5;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"interval\":1000,\"timewindowMs\":60000},\"aggregation\":{\"type\":\"NONE\",\"limit\":200}},\"showTitle\":true,\"backgroundColor\":\"rgb(255, 255, 255)\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"showTimestamp\":true},\"title\":\"Timeseries table\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":false,\"showLegend\":false}"}',
84   -'Timeseries table' );
85   -
86   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
87   -VALUES ( now ( ), minTimeuuid ( 0 ), 'charts', 'polar_area_chart_js',
88   -'{"type":"latest","sizeX":7,"sizeY":5,"resources":[{"url":"https://cdnjs.cloudflare.com/ajax/libs/Chart.js/2.3.0/Chart.min.js"}],"templateHtml":"<canvas id=\"pieChart\"></canvas>\n","templateCss":"","controllerScript":"self.onInit = function() {\n var pieData = {\n labels: [],\n datasets: []\n };\n\n var dataset = {\n data: [],\n backgroundColor: [],\n borderColor: [],\n borderWidth: [],\n hoverBackgroundColor: []\n }\n \n pieData.datasets.push(dataset);\n \n for (var i = 0; i < self.ctx.data.length; i++) {\n var dataKey = self.ctx.data[i].dataKey;\n pieData.labels.push(dataKey.label);\n dataset.data.push(0);\n var hoverBackgroundColor = tinycolor(dataKey.color).lighten(15);\n var borderColor = tinycolor(dataKey.color).darken();\n dataset.backgroundColor.push(dataKey.color);\n dataset.borderColor.push(''#fff'');\n dataset.borderWidth.push(5);\n dataset.hoverBackgroundColor.push(hoverBackgroundColor.toRgbString());\n }\n\n var ctx = $(''#pieChart'', self.ctx.$container);\n self.ctx.chart = new Chart(ctx, {\n type: ''polarArea'',\n data: pieData,\n options: {\n responsive: false,\n maintainAspectRatio: false\n }\n });\n \n self.onResize();\n}\n\nself.onDataUpdated = function() {\n for (var i = 0; i < self.ctx.data.length; i++) {\n var cellData = self.ctx.data[i];\n if (cellData.data.length > 0) {\n var tvPair = cellData.data[cellData.data.length - 1];\n var value = tvPair[1];\n self.ctx.chart.data.datasets[0].data[i] = parseFloat(value);\n }\n }\n \n self.ctx.chart.update();\n}\n\nself.onResize = function() {\n if (self.ctx.height >= 70) {\n try {\n self.ctx.chart.resize();\n } catch (e) {}\n }\n}\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"First\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.15479322438769105,\"funcBody\":\"var value = (prevValue-50) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+50;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Second\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.545701115289893,\"funcBody\":\"var value = (prevValue-20) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+20;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Third\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.2592906835158064,\"funcBody\":\"var value = (prevValue-40) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+40;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Fourth\",\"color\":\"#ffc107\",\"settings\":{},\"_hash\":0.12880275585455747,\"funcBody\":\"var value = (prevValue-50) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+50;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Fifth\",\"color\":\"#607d8b\",\"settings\":{},\"_hash\":0.2074391823443591,\"funcBody\":\"var value = (prevValue-50) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+50;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{},\"title\":\"Polar Area - Chart.js\"}"}',
89   -'Polar Area - Chart.js' );
90   -
91   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
92   -VALUES ( now ( ), minTimeuuid ( 0 ), 'cards', 'html_card',
93   -'{"type":"static","sizeX":7.5,"sizeY":3,"resources":[],"templateHtml":"","templateCss":"","controllerScript":"self.onInit = function() {\n\n var cssParser = new cssjs();\n cssParser.testMode = false;\n var namespace = ''html-card-'' + hashCode(self.ctx.settings.cardCss);\n cssParser.cssPreviewNamespace = namespace;\n cssParser.createStyleElement(namespace, self.ctx.settings.cardCss);\n self.ctx.$container.addClass(namespace);\n cardHtml = self.ctx.settings.cardHtml;\n self.ctx.$container.html(cardHtml);\n \n function hashCode(str) {\n var hash = 0;\n var i, char;\n if (str.length === 0) return hash;\n for (i = 0; i < str.length; i++) {\n char = str.charCodeAt(i);\n hash = ((hash << 5) - hash) + char;\n hash = hash & hash;\n }\n return hash;\n }\n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"Settings\",\n \"required\": [\"cardHtml\"],\n \"properties\": {\n \"cardCss\": {\n \"title\": \"CSS\",\n \"type\": \"string\",\n \"default\": \".card {\\n font-weight: bold; \\n}\"\n },\n \"cardHtml\": {\n \"title\": \"HTML\",\n \"type\": \"string\",\n \"default\": \"<div class=''card''>HTML code here</div>\"\n }\n }\n },\n \"form\": [\n {\n \"key\": \"cardCss\",\n \"type\": \"css\"\n }, \n {\n \"key\": \"cardHtml\",\n \"type\": \"html\"\n } \n ]\n}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"static\",\"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\":\"rgb(255, 255, 255)\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"cardHtml\":\"<div class=''card''>HTML code here</div>\",\"cardCss\":\".card {\\n font-weight: bold;\\n font-size: 32px;\\n color: #999;\\n width: 100%;\\n height: 100%;\\n display: flex;\\n align-items: center;\\n justify-content: center;\\n}\"},\"title\":\"HTML Card\",\"dropShadow\":true}"}',
94   -'HTML Card' );
95   -
96   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
97   -VALUES ( now ( ), minTimeuuid ( 0 ), 'digital_gauges',
98   -'simple_neon_gauge_justgage',
99   -'{"type":"latest","sizeX":3,"sizeY":3,"resources":[],"templateHtml":"<canvas id=\"digitalGauge\"></canvas>","templateCss":"#gauge {\n text-align: center;\n /* margin-left: -100px;\n margin-right: -100px;*/\n /*margin-top: -50px;*/\n \n}\n","controllerScript":"self.onInit = function() {\n self.ctx.gauge = new TbCanvasDigitalGauge(self.ctx, ''digitalGauge''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.gauge.update();\n}\n\nself.onResize = function() {\n self.ctx.gauge.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbCanvasDigitalGauge.settingsSchema;\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.gauge.mobileModeChanged();\n}\n\nself.onDestroy = function() {\n}\n\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Temp\",\"color\":\"#388e3c\",\"settings\":{},\"_hash\":0.7282710489093589,\"funcBody\":\"var value = prevValue + Math.random() * 20 - 10;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"#000000\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"0px\",\"settings\":{\"maxValue\":100,\"minValue\":0,\"donutStartAngle\":90,\"showValue\":true,\"showMinMax\":true,\"gaugeWidthScale\":1,\"levelColors\":[],\"refreshAnimationType\":\">\",\"refreshAnimationTime\":700,\"startAnimationType\":\">\",\"startAnimationTime\":700,\"titleFont\":{\"family\":\"Roboto\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\"},\"labelFont\":{\"family\":\"Roboto\",\"size\":8,\"style\":\"normal\",\"weight\":\"500\"},\"valueFont\":{\"family\":\"Segment7Standard\",\"style\":\"normal\",\"weight\":\"500\",\"size\":32},\"minMaxFont\":{\"family\":\"Segment7Standard\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\"},\"neonGlowBrightness\":40,\"dashThickness\":1.5,\"decimals\":0,\"gaugeType\":\"donut\"},\"title\":\"Simple neon gauge - justGage\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',
100   -'Simple neon gauge - justGage' );
101   -
102   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
103   -VALUES ( now ( ), minTimeuuid ( 0 ), 'analogue_gauges',
104   -'radial_gauge_canvas_gauges',
105   -'{"type":"latest","sizeX":6,"sizeY":5,"resources":[],"templateHtml":"<canvas id=\"radialGauge\"></canvas>\n","templateCss":"","controllerScript":"self.onInit = function() {\n self.ctx.gauge = new TbAnalogueRadialGauge(self.ctx, ''radialGauge''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.gauge.update();\n}\n\nself.onResize = function() {\n self.ctx.gauge.resize();\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.gauge.mobileModeChanged();\n}\n\nself.getSettingsSchema = function() {\n return TbAnalogueRadialGauge.settingsSchema;\n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Temp\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.7282710489093589,\"funcBody\":\"var value = prevValue + Math.random() * 50 - 25;\\nif (value < -100) {\\n\\tvalue = -100;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"rgb(255, 255, 255)\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"maxValue\":100,\"startAngle\":45,\"ticksAngle\":270,\"showBorder\":true,\"defaultColor\":\"#e65100\",\"needleCircleSize\":10,\"highlights\":[],\"showUnitTitle\":true,\"colorPlate\":\"#fff\",\"colorMajorTicks\":\"#444\",\"colorMinorTicks\":\"#666\",\"minorTicks\":10,\"valueInt\":3,\"valueDec\":0,\"highlightsWidth\":15,\"valueBox\":true,\"animation\":true,\"animationDuration\":500,\"animationRule\":\"cycle\",\"colorNeedleShadowUp\":\"rgba(2, 255, 255, 0)\",\"numbersFont\":{\"family\":\"Roboto\",\"size\":18,\"style\":\"normal\",\"weight\":\"500\",\"color\":\"#616161\"},\"titleFont\":{\"family\":\"Roboto\",\"size\":24,\"style\":\"normal\",\"weight\":\"500\",\"color\":\"#888\"},\"unitsFont\":{\"family\":\"Roboto\",\"size\":22,\"style\":\"normal\",\"weight\":\"500\",\"color\":\"#616161\"},\"valueFont\":{\"family\":\"Segment7Standard\",\"size\":36,\"style\":\"normal\",\"weight\":\"normal\",\"shadowColor\":\"rgba(0, 0, 0, 0.49)\",\"color\":\"#444\"},\"minValue\":-100,\"colorNeedleShadowDown\":\"rgba(188,143,143,0.45)\",\"colorValueBoxRect\":\"#888\",\"colorValueBoxRectEnd\":\"#666\",\"colorValueBoxBackground\":\"#babab2\",\"colorValueBoxShadow\":\"rgba(0,0,0,1)\"},\"title\":\"Radial gauge - Canvas Gauges\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',
106   -'Radial gauge - Canvas Gauges' );
107   -
108   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
109   -VALUES ( now ( ), minTimeuuid ( 0 ), 'digital_gauges',
110   -'digital_speedometer',
111   -'{"type":"latest","sizeX":5,"sizeY":3,"resources":[],"templateHtml":"<canvas id=\"digitalGauge\"></canvas>","templateCss":"#gauge {\n text-align: center;\n /* margin-left: -100px;\n margin-right: -100px;*/\n /*margin-top: -50px;*/\n \n}\n","controllerScript":"self.onInit = function() {\n self.ctx.gauge = new TbCanvasDigitalGauge(self.ctx, ''digitalGauge''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.gauge.update();\n}\n\nself.onResize = function() {\n self.ctx.gauge.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbCanvasDigitalGauge.settingsSchema;\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.gauge.mobileModeChanged();\n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Speed\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.7282710489093589,\"funcBody\":\"var value = prevValue + Math.random() * 40 - 20;\\nif (value < 45) {\\n\\tvalue = 45;\\n} else if (value > 130) {\\n\\tvalue = 130;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"#000000\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"0px\",\"settings\":{\"maxValue\":180,\"minValue\":0,\"donutStartAngle\":90,\"showValue\":true,\"showMinMax\":true,\"gaugeWidthScale\":0.75,\"levelColors\":[\"#008000\",\"#fbc02d\",\"#f44336\"],\"refreshAnimationType\":\"linear\",\"refreshAnimationTime\":700,\"startAnimationType\":\"linear\",\"startAnimationTime\":700,\"titleFont\":{\"family\":\"Roboto\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\"},\"labelFont\":{\"family\":\"Roboto\",\"size\":8,\"style\":\"normal\",\"weight\":\"500\"},\"valueFont\":{\"family\":\"Segment7Standard\",\"style\":\"normal\",\"weight\":\"500\",\"size\":32},\"minMaxFont\":{\"family\":\"Segment7Standard\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\",\"color\":\"#ffffff\"},\"neonGlowBrightness\":40,\"dashThickness\":1.5,\"decimals\":0,\"unitTitle\":\"MPH\",\"showUnitTitle\":true,\"gaugeColor\":\"#171a1c\",\"gaugeType\":\"arc\"},\"title\":\"Digital speedometer\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',
112   -'Digital speedometer' );
113   -
114   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
115   -VALUES ( now ( ), minTimeuuid ( 0 ), 'charts', 'doughnut_chart_js',
116   -'{"type":"latest","sizeX":7,"sizeY":5,"resources":[{"url":"https://cdnjs.cloudflare.com/ajax/libs/Chart.js/2.3.0/Chart.min.js"}],"templateHtml":"<canvas id=\"pieChart\"></canvas>\n","templateCss":"","controllerScript":"self.onInit = function() {\n var pieData = {\n labels: [],\n datasets: []\n };\n\n var dataset = {\n data: [],\n backgroundColor: [],\n borderColor: [],\n borderWidth: [],\n hoverBackgroundColor: []\n }\n \n var borderColor = self.ctx.settings.borderColor || ''#fff'';\n var borderWidth = angular.isDefined(self.ctx.settings.borderWidth) ? self.ctx.settings.borderWidth : 5;\n \n pieData.datasets.push(dataset);\n \n for (var i=0; i < self.ctx.data.length; i++) {\n var dataKey = self.ctx.data[i].dataKey;\n pieData.labels.push(dataKey.label);\n dataset.data.push(0);\n var hoverBackgroundColor = tinycolor(dataKey.color).lighten(15);\n dataset.backgroundColor.push(dataKey.color);\n dataset.borderColor.push(borderColor);\n dataset.borderWidth.push(borderWidth);\n dataset.hoverBackgroundColor.push(hoverBackgroundColor.toRgbString());\n }\n\n var options = {\n responsive: false,\n maintainAspectRatio: false,\n legend: {\n display: true,\n labels: {\n fontColor: ''#666''\n }\n },\n tooltips: {\n callbacks: {\n label: function(tooltipItem, data) {\n var label = data.labels[tooltipItem.index];\n var value = data.datasets[tooltipItem.datasetIndex].data[tooltipItem.index];\n var content = label + '': '' + value;\n var units = self.ctx.settings.units ? self.ctx.settings.units : self.ctx.units;\n if (units) {\n content += '' '' + units;\n } \n return content;\n }\n }\n }\n };\n\n if (self.ctx.settings.legend) {\n options.legend.display = self.ctx.settings.legend.display !== false;\n options.legend.labels.fontColor = self.ctx.settings.legend.labelsFontColor || ''#666'';\n }\n\n var ctx = $(''#pieChart'', self.ctx.$container);\n self.ctx.chart = new Chart(ctx, {\n type: ''doughnut'',\n data: pieData,\n options: options\n });\n \n self.onResize();\n}\n\nself.onDataUpdated = function() {\n for (var i = 0; i < self.ctx.data.length; i++) {\n var cellData = self.ctx.data[i];\n if (cellData.data.length > 0) {\n var tvPair = cellData.data[cellData.data.length - 1];\n var value = tvPair[1];\n self.ctx.chart.data.datasets[0].data[i] = parseFloat(value);\n }\n }\n self.ctx.chart.update();\n}\n\nself.onResize = function() {\n self.ctx.chart.resize();\n}\n\n","settingsSchema":"{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"Settings\",\n \"properties\": {\n \"borderWidth\": {\n \"title\": \"Border width\",\n \"type\": \"number\",\n \"default\": 5\n },\n \"borderColor\": {\n \"title\": \"Border color\",\n \"type\": \"string\",\n \"default\": \"#fff\"\n },\n \"legend\": {\n \"title\": \"Legend settings\",\n \"type\": \"object\",\n \"properties\": {\n \"display\": {\n \"title\": \"Display legend\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"labelsFontColor\": {\n \"title\": \"Labels font color\",\n \"type\": \"string\",\n \"default\": \"#666\"\n }\n }\n }\n },\n \"required\": []\n },\n \"form\": [\n \"borderWidth\", \n {\n \"key\": \"borderColor\",\n \"type\": \"color\"\n }, \n {\n \"key\": \"legend\",\n \"items\": [\n \"legend.display\",\n {\n \"key\": \"legend.labelsFontColor\",\n \"type\": \"color\"\n }\n ]\n }\n ]\n}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"First\",\"color\":\"#26a69a\",\"settings\":{},\"_hash\":0.15479322438769105,\"funcBody\":\"var value = (prevValue-50) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+50;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Second\",\"color\":\"#f57c00\",\"settings\":{},\"_hash\":0.545701115289893,\"funcBody\":\"var value = (prevValue-20) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+20;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Third\",\"color\":\"#afb42b\",\"settings\":{},\"_hash\":0.2592906835158064,\"funcBody\":\"var value = (prevValue-40) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+40;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Fourth\",\"color\":\"#673ab7\",\"settings\":{},\"_hash\":0.12880275585455747,\"funcBody\":\"var value = (prevValue-50) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+50;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"borderWidth\":5,\"borderColor\":\"#fff\",\"legend\":{\"display\":true,\"labelsFontColor\":\"#666666\"}},\"title\":\"Doughnut - Chart.js\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',
117   -'Doughnut - Chart.js' );
118   -
119   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
120   -VALUES ( now ( ), minTimeuuid ( 0 ), 'charts', 'pie_chart_js',
121   -'{"type":"latest","sizeX":8,"sizeY":5,"resources":[{"url":"https://cdnjs.cloudflare.com/ajax/libs/Chart.js/2.3.0/Chart.min.js"}],"templateHtml":"<canvas id=\"pieChart\"></canvas>\n","templateCss":"","controllerScript":"self.onInit = function() {\n \n var pieData = {\n labels: [],\n datasets: []\n };\n\n var dataset = {\n data: [],\n backgroundColor: [],\n borderColor: [],\n borderWidth: [],\n hoverBackgroundColor: []\n }\n \n pieData.datasets.push(dataset);\n \n for (var i=0; i < self.ctx.data.length; i++) {\n var dataKey = self.ctx.data[i].dataKey;\n pieData.labels.push(dataKey.label);\n dataset.data.push(0);\n var hoverBackgroundColor = tinycolor(dataKey.color).lighten(15);\n var borderColor = tinycolor(dataKey.color).darken();\n dataset.backgroundColor.push(dataKey.color);\n dataset.borderColor.push(''#fff'');\n dataset.borderWidth.push(5);\n dataset.hoverBackgroundColor.push(hoverBackgroundColor.toRgbString());\n }\n\n var ctx = $(''#pieChart'', self.ctx.$container);\n self.ctx.chart = new Chart(ctx, {\n type: ''pie'',\n data: pieData,\n options: {\n responsive: false,\n maintainAspectRatio: false\n }\n }); \n \n self.onResize();\n}\n\nself.onDataUpdated = function() {\n for (var i = 0; i < self.ctx.data.length; i++) {\n var cellData = self.ctx.data[i];\n if (cellData.data.length > 0) {\n var tvPair = cellData.data[cellData.data.length - 1];\n var value = tvPair[1];\n self.ctx.chart.data.datasets[0].data[i] = parseFloat(value);\n }\n }\n self.ctx.chart.update();\n}\n\nself.onResize = function() {\n self.ctx.chart.resize();\n}\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"First\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.15479322438769105,\"funcBody\":\"var value = (prevValue-50) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+50;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Second\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.545701115289893,\"funcBody\":\"var value = (prevValue-20) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+20;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Third\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.2592906835158064,\"funcBody\":\"var value = (prevValue-40) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+40;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Fourth\",\"color\":\"#ffc107\",\"settings\":{},\"_hash\":0.12880275585455747,\"funcBody\":\"var value = (prevValue-50) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+50;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{},\"title\":\"Pie - Chart.js\"}"}',
122   -'Pie - Chart.js' );
123   -
124   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
125   -VALUES ( now ( ), minTimeuuid ( 0 ), 'charts', 'bars',
126   -'{"type":"latest","sizeX":7,"sizeY":5,"resources":[{"url":"https://cdnjs.cloudflare.com/ajax/libs/Chart.js/2.3.0/Chart.min.js"}],"templateHtml":"<canvas id=\"barChart\"></canvas>\n","templateCss":"","controllerScript":"self.onInit = function() {\n var barData = {\n labels: [],\n datasets: []\n };\n \n for (var i = 0; i < self.ctx.datasources.length; i++) {\n var datasource = self.ctx.datasources[i];\n for (var d = 0; d < datasource.dataKeys.length; d++) {\n var dataset = {\n label: datasource.dataKeys[d].label,\n data: [0],\n backgroundColor: [datasource.dataKeys[d].color],\n borderColor: [datasource.dataKeys[d].color],\n borderWidth: 1\n }\n barData.datasets.push(dataset);\n }\n }\n\n var ctx = $(''#barChart'', self.ctx.$container);\n self.ctx.chart = new Chart(ctx, {\n type: ''bar'',\n data: barData,\n options: {\n responsive: false,\n maintainAspectRatio: false,\n scales: {\n yAxes: [{\n ticks: {\n beginAtZero:true\n }\n }]\n }\n }\n });\n \n self.onResize();\n}\n\nself.onDataUpdated = function() {\n var c = 0;\n for (var i = 0; i < self.ctx.chart.data.datasets.length; i++) {\n var dataset = self.ctx.chart.data.datasets[i];\n var cellData = self.ctx.data[i]; \n if (cellData.data.length > 0) {\n var tvPair = cellData.data[cellData.data.length - 1];\n var value = tvPair[1];\n dataset.data[0] = parseFloat(value);\n }\n }\n self.ctx.chart.update();\n}\n\nself.onResize = function() {\n self.ctx.chart.resize();\n}\n\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"First\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.15479322438769105,\"funcBody\":\"var value = (prevValue-50) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+50;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Second\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.545701115289893,\"funcBody\":\"var value = (prevValue-20) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+20;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Third\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.2592906835158064,\"funcBody\":\"var value = (prevValue-40) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+40;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Fourth\",\"color\":\"#ffc107\",\"settings\":{},\"_hash\":0.12880275585455747,\"funcBody\":\"var value = (prevValue-50) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+50;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{},\"title\":\"Bars - Chart.js\"}"}',
127   -'Bars - Chart.js' );
128   -
129   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
130   -VALUES ( now ( ), minTimeuuid ( 0 ), 'digital_gauges', 'digital_bar',
131   -'{"type":"latest","sizeX":6,"sizeY":2.5,"resources":[],"templateHtml":"<canvas id=\"digitalGauge\"></canvas>","templateCss":"#gauge {\n text-align: center;\n /* margin-left: -100px;\n margin-right: -100px;*/\n /*margin-top: -50px;*/\n \n}\n","controllerScript":"self.onInit = function() {\n self.ctx.gauge = new TbCanvasDigitalGauge(self.ctx, ''digitalGauge''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.gauge.update();\n}\n\nself.onResize = function() {\n self.ctx.gauge.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbCanvasDigitalGauge.settingsSchema;\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.gauge.mobileModeChanged();\n}\n\nself.onDestroy = function() {\n}","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Speed\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.7282710489093589,\"funcBody\":\"var value = prevValue + Math.random() * 40 - 20;\\nif (value < 80) {\\n\\tvalue = 80;\\n} else if (value > 160) {\\n\\tvalue = 160;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"#000000\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"0px\",\"settings\":{\"maxValue\":180,\"minValue\":0,\"donutStartAngle\":90,\"showValue\":true,\"showMinMax\":true,\"gaugeWidthScale\":0.75,\"levelColors\":[\"#008000\",\"#fbc02d\",\"#f44336\"],\"refreshAnimationType\":\"linear\",\"refreshAnimationTime\":700,\"startAnimationType\":\"linear\",\"startAnimationTime\":700,\"titleFont\":{\"family\":\"Roboto\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\"},\"labelFont\":{\"family\":\"Roboto\",\"size\":8,\"style\":\"normal\",\"weight\":\"500\"},\"valueFont\":{\"family\":\"Segment7Standard\",\"style\":\"normal\",\"weight\":\"500\",\"size\":18},\"minMaxFont\":{\"family\":\"Segment7Standard\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\",\"color\":\"#ffffff\"},\"neonGlowBrightness\":40,\"dashThickness\":1.5,\"decimals\":0,\"unitTitle\":\"MPH\",\"showUnitTitle\":true,\"gaugeColor\":\"#171a1c\",\"gaugeType\":\"horizontalBar\",\"showTitle\":false},\"title\":\"Digital horizontal bar\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',
132   -'Digital horizontal bar' );
133   -
134   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
135   -VALUES ( now ( ), minTimeuuid ( 0 ), 'digital_gauges',
136   -'mini_gauge_justgage',
137   -'{"type":"latest","sizeX":2,"sizeY":2,"resources":[],"templateHtml":"<canvas id=\"digitalGauge\"></canvas>","templateCss":"#gauge {\n text-align: center;\n /* margin-left: -100px;\n margin-right: -100px;*/\n /*margin-top: -50px;*/\n \n}\n","controllerScript":"self.onInit = function() {\n self.ctx.gauge = new TbCanvasDigitalGauge(self.ctx, ''digitalGauge''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.gauge.update();\n}\n\nself.onResize = function() {\n self.ctx.gauge.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbCanvasDigitalGauge.settingsSchema;\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.gauge.mobileModeChanged();\n}\n\nself.onDestroy = function() {\n}\n\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Temp\",\"color\":\"#7cb342\",\"settings\":{},\"_hash\":0.7282710489093589,\"funcBody\":\"var value = prevValue + Math.random() * 20 - 10;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"#ffffff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"0px\",\"settings\":{\"maxValue\":100,\"minValue\":0,\"donutStartAngle\":90,\"showValue\":true,\"showMinMax\":true,\"gaugeWidthScale\":0.75,\"levelColors\":[],\"refreshAnimationType\":\">\",\"refreshAnimationTime\":700,\"startAnimationType\":\">\",\"startAnimationTime\":700,\"titleFont\":{\"family\":\"Roboto\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\"},\"labelFont\":{\"family\":\"Roboto\",\"size\":8,\"style\":\"normal\",\"weight\":\"500\"},\"valueFont\":{\"family\":\"Roboto\",\"style\":\"normal\",\"weight\":\"500\",\"size\":32},\"minMaxFont\":{\"family\":\"Segment7Standard\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\"},\"neonGlowBrightness\":0,\"dashThickness\":0,\"decimals\":0,\"roundedLineCap\":true,\"gaugeType\":\"donut\"},\"title\":\"Mini gauge - justGage\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',
138   -'Mini gauge - justGage' );
139   -
140   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
141   -VALUES ( now ( ), minTimeuuid ( 0 ), 'maps', 'route_map_openstreetmap',
142   -'{"type":"timeseries","sizeX":8.5,"sizeY":6,"resources":[],"templateHtml":"","templateCss":".leaflet-zoom-box {\n\tz-index: 9;\n}\n\n.leaflet-pane { z-index: 4; }\n\n.leaflet-tile-pane { z-index: 2; }\n.leaflet-overlay-pane { z-index: 4; }\n.leaflet-shadow-pane { z-index: 5; }\n.leaflet-marker-pane { z-index: 6; }\n.leaflet-tooltip-pane { z-index: 7; }\n.leaflet-popup-pane { z-index: 8; }\n\n.leaflet-map-pane canvas { z-index: 1; }\n.leaflet-map-pane svg { z-index: 2; }\n\n.leaflet-control {\n\tz-index: 9;\n}\n.leaflet-top,\n.leaflet-bottom {\n\tz-index: 11;\n}\n\n.tb-marker-label {\n border: none;\n background: none;\n box-shadow: none;\n}\n\n.tb-marker-label:before {\n border: none;\n background: none;\n}\n","controllerScript":"self.onInit = function() {\n self.ctx.map = new TbMapWidget(''openstreet-map'', true, self.ctx);\n}\n\nself.onDataUpdated = function() {\n self.ctx.map.update();\n}\n\nself.onResize = function() {\n self.ctx.map.resize();\n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{\n \"schema\": {\n \"title\": \"Route Map Configuration\",\n \"type\": \"object\",\n \"properties\": {\n \"defaultZoomLevel\": {\n \"title\": \"Default map zoom level (1 - 20)\",\n \"type\": \"number\"\n },\n \"fitMapBounds\": {\n \"title\": \"Fit map bounds to cover all markers\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"routesSettings\": {\n \"title\": \"Routes\",\n \"type\": \"array\",\n \"items\": {\n \"title\": \"Route settings\",\n \"type\": \"object\",\n \"properties\": {\n \"latKeyName\": {\n \"title\": \"Latitude key name\",\n \"type\": \"string\",\n \"default\": \"lat\"\n },\n \"lngKeyName\": {\n \"title\": \"Longitude key name\",\n \"type\": \"string\",\n \"default\": \"lng\"\n },\n \"showLabel\": {\n \"title\": \"Show label\",\n \"type\": \"boolean\",\n \"default\": true\n }, \n \"label\": {\n \"title\": \"Label\",\n \"type\": \"string\"\n },\n \"tooltipPattern\": {\n \"title\": \"Pattern ( for ex. ''Text ${keyName} units.'' or ''${#<key index>} units'' )\",\n \"type\": \"string\",\n \"default\": \"<b>Latitude:</b> ${lat:7}<br/><b>Longitude:</b> ${lng:7}\"\n },\n \"color\": {\n \"title\": \"Color\",\n \"type\": \"string\"\n },\n \"useColorFunction\": {\n \"title\": \"Use color function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"colorFunction\": {\n \"title\": \"Color function: f(data)\",\n \"type\": \"string\"\n },\n \"markerImage\": {\n \"title\": \"Custom marker image\",\n \"type\": \"string\"\n },\n \"markerImageSize\": {\n \"title\": \"Custom marker image size (px)\",\n \"type\": \"number\",\n \"default\": 34\n },\n \"useMarkerImageFunction\": {\n \"title\": \"Use marker image function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"markerImageFunction\": {\n \"title\": \"Marker image function: f(data, images)\",\n \"type\": \"string\"\n },\n \"markerImages\": {\n \"title\": \"Marker images\",\n \"type\": \"array\",\n \"items\": {\n \"title\": \"Marker image\",\n \"type\": \"string\"\n }\n },\n \"strokeWeight\": {\n \"title\": \"Stroke weight\",\n \"type\": \"number\",\n \"default\": 2\n },\n \"strokeOpacity\": {\n \"title\": \"Stroke opacity\",\n \"type\": \"number\",\n \"default\": 1.0\n }\n }\n }\n }\n },\n \"required\": [\n ]\n },\n \"form\": [\n \"defaultZoomLevel\",\n \"fitMapBounds\",\n {\n \"key\": \"routesSettings\",\n \"items\": [\n \"routesSettings[].latKeyName\",\n \"routesSettings[].lngKeyName\",\n \"routesSettings[].showLabel\",\n \"routesSettings[].label\",\n \"routesSettings[].tooltipPattern\",\n {\n \"key\": \"routesSettings[].color\",\n \"type\": \"color\"\n },\n \"routesSettings[].useColorFunction\",\n {\n \"key\": \"routesSettings[].colorFunction\",\n \"type\": \"javascript\"\n },\n {\n \"key\": \"routesSettings[].markerImage\",\n \"type\": \"image\"\n },\n \"routesSettings[].markerImageSize\",\n \"routesSettings[].useMarkerImageFunction\",\n {\n \"key\": \"routesSettings[].markerImageFunction\",\n \"type\": \"javascript\"\n },\n {\n \"key\": \"routesSettings[].markerImages\",\n \"items\": [\n {\n \"key\": \"routesSettings[].markerImages[]\",\n \"type\": \"image\"\n }\n ]\n },\n \"routesSettings[].strokeWeight\",\n \"routesSettings[].strokeOpacity\"\n ]\n }\n ]\n}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"latitude\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.8950926999078694,\"funcBody\":\"var lats = [37.7696499,\\n37.7699074,\\n37.7699536,\\n37.7697242,\\n37.7695189,\\n37.7696889,\\n37.7697153,\\n37.7701244,\\n37.7700604,\\n37.7705491,\\n37.7715705,\\n37.771752,\\n37.7707533,\\n37.769866];\\n\\nvar i = Math.floor((time/3 % 14000) / 1000);\\n\\nreturn lats[i];\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"longitude\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.2757675428823283,\"funcBody\":\"var lons = [-122.4261215,\\n-122.4219157,\\n-122.4199623,\\n-122.4179074,\\n-122.4155876,\\n-122.4155521,\\n-122.4163203,\\n-122.4193876,\\n-122.4210496,\\n-122.422284,\\n-122.4232717,\\n-122.4235138,\\n-122.4247605,\\n-122.4258812];\\n\\nvar i = Math.floor((time/3 % 14000) / 1000);\\n\\nreturn lons[i];\"}],\"intervalSec\":60},{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Speed\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.14481354591724638,\"funcBody\":\"var value = prevValue;\\nif (time % 500 < 100) {\\n value = value + Math.random() * 40 - 20;\\n if (value < 45) {\\n \\tvalue = 45;\\n } else if (value > 130) {\\n \\tvalue = 130;\\n }\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":30000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"fitMapBounds\":true,\"routesSettings\":[{\"latKeyName\":\"latitude\",\"lngKeyName\":\"longitude\",\"showLabel\":true,\"tooltipPattern\":\"<b>Latitude:</b> ${latitude:7}<br/><b>Longitude:</b> ${longitude:7}<br/><b>Speed:</b> ${Speed} MPH<br/><small>See advanced settings for details</small>\",\"strokeWeight\":4,\"label\":\"First route\",\"color\":\"#3d5afe\",\"strokeOpacity\":1,\"useColorFunction\":true,\"useMarkerImageFunction\":true,\"markerImages\":[\"\",\"\",\"\"],\"colorFunction\":\"var speed = data[''Speed''];\\nif (typeof speed !== undefined) {\\n var percent = (speed - 45)/85;\\n if (percent < 0.5) {\\n percent *=2*100; \\n return tinycolor.mix(''green'', ''yellow'', amount = percent).toHexString();\\n } else {\\n percent = (percent - 0.5)*2*100;\\n return tinycolor.mix(''yellow'', ''red'', amount = percent).toHexString();\\n }\\n}\\nreturn ''green'';\",\"markerImageFunction\":\"var speed = data[''Speed''];\\nvar res = {\\n url: images[0],\\n size: 55\\n};\\nif (typeof speed !== undefined) {\\n var percent = (speed - 45)/85;\\n var index = Math.floor(3 * percent);\\n res.url = images[index];\\n}\\nreturn res;\"}]},\"title\":\"Route Map - OpenStreetMap\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false}"}',
143   -'Route Map - OpenStreetMap' );
144   -
145   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
146   -VALUES ( now ( ), minTimeuuid ( 0 ), 'cards', 'label_widget',
147   -'{"type":"latest","sizeX":4.5,"sizeY":5,"resources":[],"templateHtml":"","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}","controllerScript":"self.onInit = function() {\n self.ctx.varsRegex = /\\$\\{([^\\}]*)\\}/g;\n \n var imageUrl = self.ctx.settings.backgroundImageUrl ? self.ctx.settings.backgroundImageUrl :\n '''';\n\n self.ctx.$container.css(''background'', ''url(\"''+imageUrl+''\") no-repeat'');\n self.ctx.$container.css(''backgroundSize'', ''contain'');\n self.ctx.$container.css(''backgroundPosition'', ''50% 50%'');\n \n function processLabelPattern(pattern, data) {\n var match = self.ctx.varsRegex.exec(pattern);\n var replaceInfo = {};\n replaceInfo.variables = [];\n while (match !== null) {\n var variableInfo = {};\n variableInfo.dataKeyIndex = -1;\n var variable = match[0];\n var label = match[1];\n var valDec = 2;\n var splitVals = label.split('':'');\n if (splitVals.length > 1) {\n label = splitVals[0];\n valDec = parseFloat(splitVals[1]);\n }\n variableInfo.variable = variable;\n variableInfo.valDec = valDec;\n \n if (label.startsWith(''#'')) {\n var keyIndexStr = label.substring(1);\n var n = Math.floor(Number(keyIndexStr));\n if (String(n) === keyIndexStr && n >= 0) {\n variableInfo.dataKeyIndex = n;\n }\n }\n if (variableInfo.dataKeyIndex === -1) {\n for (var i = 0; i < data.length; i++) {\n var datasourceData = data[i];\n var dataKey = datasourceData.dataKey;\n if (dataKey.label === label) {\n variableInfo.dataKeyIndex = i;\n break;\n }\n }\n }\n replaceInfo.variables.push(variableInfo);\n match = self.ctx.varsRegex.exec(pattern);\n }\n return replaceInfo;\n }\n\n var configuredLabels = self.ctx.settings.labels;\n if (!configuredLabels) {\n configuredLabels = [];\n }\n \n self.ctx.labels = [];\n\n for (var l = 0; l < configuredLabels.length; l++) {\n var labelConfig = configuredLabels[l];\n var localConfig = {};\n localConfig.font = {};\n \n localConfig.pattern = labelConfig.pattern ? labelConfig.pattern : ''${#0}'';\n localConfig.x = labelConfig.x ? labelConfig.x : 0;\n localConfig.y = labelConfig.y ? labelConfig.y : 0;\n localConfig.backgroundColor = labelConfig.backgroundColor ? labelConfig.backgroundColor : ''rgba(0,0,0,0)'';\n \n var settingsFont = labelConfig.font;\n if (!settingsFont) {\n settingsFont = {};\n }\n \n localConfig.font.family = settingsFont.family || ''RobotoDraft'';\n localConfig.font.size = settingsFont.size ? settingsFont.size : 6;\n localConfig.font.style = settingsFont.style ? settingsFont.style : ''normal'';\n localConfig.font.weight = settingsFont.weight ? settingsFont.weight : ''500'';\n localConfig.font.color = settingsFont.color ? settingsFont.color : ''#fff'';\n \n localConfig.replaceInfo = processLabelPattern(localConfig.pattern, self.ctx.data);\n \n var label = {};\n var labelElement = $(''<div/>'');\n labelElement.css(''position'', ''absolute'');\n labelElement.css(''display'', ''none'');\n labelElement.css(''top'', ''0'');\n labelElement.css(''left'', ''0'');\n labelElement.css(''backgroundColor'', localConfig.backgroundColor);\n labelElement.css(''color'', localConfig.font.color);\n labelElement.css(''fontFamily'', localConfig.font.family);\n labelElement.css(''fontStyle'', localConfig.font.style);\n labelElement.css(''fontWeight'', localConfig.font.weight);\n \n labelElement.html(localConfig.pattern);\n self.ctx.$container.append(labelElement);\n label.element = labelElement;\n label.config = localConfig;\n label.htmlSet = false;\n label.visible = false;\n self.ctx.labels.push(label);\n }\n\n var bgImg = $(''<img />'');\n bgImg.hide();\n bgImg.bind(''load'', function()\n {\n self.ctx.bImageHeight = $(this).height();\n self.ctx.bImageWidth = $(this).width();\n self.onResize();\n });\n self.ctx.$container.append(bgImg);\n bgImg.attr(''src'', imageUrl);\n \n self.onDataUpdated();\n}\n\nself.onDataUpdated = function() {\n updateLabels();\n}\n\nself.onResize = function() {\n if (self.ctx.bImageHeight && self.ctx.bImageWidth) {\n var backgroundRect = {};\n var imageRatio = self.ctx.bImageWidth / self.ctx.bImageHeight;\n var componentRatio = self.ctx.width / self.ctx.height;\n if (componentRatio >= imageRatio) {\n backgroundRect.top = 0;\n backgroundRect.bottom = 1.0;\n backgroundRect.xRatio = imageRatio / componentRatio;\n backgroundRect.yRatio = 1;\n var offset = (1 - backgroundRect.xRatio) / 2;\n backgroundRect.left = offset;\n backgroundRect.right = 1 - offset;\n } else {\n backgroundRect.left = 0;\n backgroundRect.right = 1.0;\n backgroundRect.xRatio = 1;\n backgroundRect.yRatio = componentRatio / imageRatio;\n var offset = (1 - backgroundRect.yRatio) / 2;\n backgroundRect.top = offset;\n backgroundRect.bottom = 1 - offset;\n }\n for (var l = 0; l < self.ctx.labels.length; l++) {\n var label = self.ctx.labels[l];\n var labelLeft = backgroundRect.left*100 + (label.config.x*backgroundRect.xRatio);\n var labelTop = backgroundRect.top*100 + (label.config.y*backgroundRect.yRatio);\n var fontSize = self.ctx.height * backgroundRect.yRatio * label.config.font.size / 100;\n label.element.css(''top'', labelTop + ''%'');\n label.element.css(''left'', labelLeft + ''%'');\n label.element.css(''fontSize'', fontSize + ''px'');\n if (!label.visible) {\n label.element.css(''display'', ''block'');\n label.visible = true;\n }\n }\n } \n}\n\n\nfunction isNumber(n) {\n return !isNaN(parseFloat(n)) && isFinite(n);\n}\n\nfunction padValue(val, dec, int) {\n var i = 0;\n var s, strVal, n;\n\n val = parseFloat(val);\n n = (val < 0);\n val = Math.abs(val);\n\n if (dec > 0) {\n strVal = val.toFixed(dec).toString().split(''.'');\n s = int - strVal[0].length;\n\n for (; i < s; ++i) {\n strVal[0] = ''0'' + strVal[0];\n }\n\n strVal = (n ? ''-'' : '''') + strVal[0] + ''.'' + strVal[1];\n }\n\n else {\n strVal = Math.round(val).toString();\n s = int - strVal.length;\n\n for (; i < s; ++i) {\n strVal = ''0'' + strVal;\n }\n\n strVal = (n ? ''-'' : '''') + strVal;\n }\n\n return strVal;\n}\n\nfunction updateLabels() {\n for (var l = 0; l < self.ctx.labels.length; l++) {\n var label = self.ctx.labels[l];\n var text = label.config.pattern;\n var replaceInfo = label.config.replaceInfo;\n var updated = false;\n for (var v = 0; v < replaceInfo.variables.length; v++) {\n var variableInfo = replaceInfo.variables[v];\n var txtVal = '''';\n if (variableInfo.dataKeyIndex > -1) {\n var varData = self.ctx.data[variableInfo.dataKeyIndex].data;\n if (varData.length > 0) {\n var val = varData[varData.length-1][1];\n if (isNumber(val)) {\n txtVal = padValue(val, variableInfo.valDec, 0);\n updated = true;\n } else {\n txtVal = val;\n updated = true;\n }\n }\n }\n text = text.split(variableInfo.variable).join(txtVal);\n }\n if (updated || !label.htmlSet) {\n label.element.html(text);\n if (!label.htmlSet) {\n label.htmlSet = true;\n }\n }\n }\n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"Settings\",\n \"required\": [\"backgroundImageUrl\"],\n \"properties\": {\n \"backgroundImageUrl\": {\n \"title\": \"Background image\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"labels\": {\n \"title\": \"Labels\",\n \"type\": \"array\",\n \"items\": {\n \"title\": \"Label\",\n \"type\": \"object\",\n \"required\": [\"pattern\"],\n \"properties\": {\n \"pattern\": {\n \"title\": \"Pattern ( for ex. ''Text ${keyName} units.'' or ''${#<key index>} units'' )\",\n \"type\": \"string\",\n \"default\": \"${#0}\"\n },\n \"x\": {\n \"title\": \"X (Percentage relative to background)\",\n \"type\": \"number\",\n \"default\": 50\n },\n \"y\": {\n \"title\": \"Y (Percentage relative to background)\",\n \"type\": \"number\",\n \"default\": 50\n },\n \"backgroundColor\": {\n \"title\": \"Backround color\",\n \"type\": \"string\",\n \"default\": \"rgba(0,0,0,0)\"\n },\n \"font\": {\n \"type\": \"object\",\n \"properties\": {\n \"family\": {\n \"title\": \"Font family\",\n \"type\": \"string\",\n \"default\": \"RobotoDraft\"\n },\n \"size\": {\n \"title\": \"Relative font size (percents)\",\n \"type\": \"number\",\n \"default\": 6\n },\n \"style\": {\n \"title\": \"Style\",\n \"type\": \"string\",\n \"default\": \"normal\"\n },\n \"weight\": {\n \"title\": \"Weight\",\n \"type\": \"string\",\n \"default\": \"500\"\n },\n \"color\": {\n \"title\": \"color\",\n \"type\": \"string\",\n \"default\": \"#fff\"\n }\n }\n }\n }\n }\n }\n }\n },\n \"form\": [\n {\n \"key\": \"backgroundImageUrl\",\n \"type\": \"image\"\n },\n {\n \"key\": \"labels\",\n \"items\": [\n \"labels[].pattern\",\n \"labels[].x\",\n \"labels[].y\",\n {\n \"key\": \"labels[].backgroundColor\",\n \"type\": \"color\"\n },\n \"labels[].font.family\",\n \"labels[].font.size\",\n {\n \"key\": \"labels[].font.style\",\n \"type\": \"rc-select\",\n \"multiple\": false,\n \"items\": [\n {\n \"value\": \"normal\",\n \"label\": \"Normal\"\n },\n {\n \"value\": \"italic\",\n \"label\": \"Italic\"\n },\n {\n \"value\": \"oblique\",\n \"label\": \"Oblique\"\n }\n ]\n\n },\n {\n \"key\": \"labels[].font.weight\",\n \"type\": \"rc-select\",\n \"multiple\": false,\n \"items\": [\n {\n \"value\": \"normal\",\n \"label\": \"Normal\"\n },\n {\n \"value\": \"bold\",\n \"label\": \"Bold\"\n },\n {\n \"value\": \"bolder\",\n \"label\": \"Bolder\"\n },\n {\n \"value\": \"lighter\",\n \"label\": \"Lighter\"\n },\n {\n \"value\": \"100\",\n \"label\": \"100\"\n },\n {\n \"value\": \"200\",\n \"label\": \"200\"\n },\n {\n \"value\": \"300\",\n \"label\": \"300\"\n },\n {\n \"value\": \"400\",\n \"label\": \"400\"\n },\n {\n \"value\": \"500\",\n \"label\": \"500\"\n },\n {\n \"value\": \"600\",\n \"label\": \"600\"\n },\n {\n \"value\": \"700\",\n \"label\": \"800\"\n },\n {\n \"value\": \"800\",\n \"label\": \"800\"\n },\n {\n \"value\": \"900\",\n \"label\": \"900\"\n }\n ]\n },\n {\n \"key\": \"labels[].font.color\",\n \"type\": \"color\"\n }\n ]\n }\n ]\n}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"var\",\"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\":{\"backgroundImageUrl\":\"\",\"labels\":[{\"pattern\":\"Value: ${#0:2} units.\",\"x\":20,\"y\":47,\"font\":{\"color\":\"#515151\",\"family\":\"Roboto\",\"size\":6,\"style\":\"normal\",\"weight\":\"500\"}}]},\"title\":\"Label widget\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',
148   -'Label widget' );
149   -
150   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
151   -VALUES ( now ( ), minTimeuuid ( 0 ), 'cards', 'simple_card',
152   -'{"type":"latest","sizeX":5,"sizeY":3,"resources":[],"templateHtml":"","templateCss":"#container {\n overflow: auto;\n}\n\n.tbDatasource-container {\n width: 100%;\n height: 100%;\n overflow: hidden;\n}\n\n.tbDatasource-table {\n width: 100%;\n height: 100%;\n border-collapse: collapse;\n white-space: nowrap;\n font-weight: 100;\n text-align: right;\n}\n\n.tbDatasource-table td {\n padding: 12px;\n position: relative;\n box-sizing: border-box;\n}\n\n.tbDatasource-data-key {\n opacity: 0.7;\n font-weight: 400;\n font-size: 3.500rem;\n}\n\n.tbDatasource-value {\n font-size: 5.000rem;\n}","controllerScript":"self.onInit = function() {\n self.ctx.units = self.ctx.settings.units || self.ctx.units;\n self.ctx.valueDec = (typeof self.ctx.settings.valueDec !== ''undefined'' && self.ctx.settings.valueDec !== null)\n ? self.ctx.settings.valueDec : self.ctx.decimals;\n \n self.ctx.labelPosition = self.ctx.settings.labelPosition || ''left'';\n \n if (self.ctx.datasources.length > 0) {\n var tbDatasource = self.ctx.datasources[0];\n var datasourceId = ''tbDatasource'' + 0;\n self.ctx.$container.append(\n \"<div id=''\" + datasourceId +\n \"'' class=''tbDatasource-container''></div>\"\n );\n \n self.ctx.datasourceContainer = $(''#'' + datasourceId,\n self.ctx.$container);\n \n var tableId = ''table'' + 0;\n self.ctx.datasourceContainer.append(\n \"<table id=''\" + tableId +\n \"'' class=''tbDatasource-table''><col width=''30%''><col width=''70%''></table>\"\n );\n var table = $(''#'' + tableId, self.ctx.$container);\n if (self.ctx.labelPosition === ''top'') {\n table.css(''text-align'', ''left'');\n }\n \n if (tbDatasource.dataKeys.length > 0) {\n var dataKey = tbDatasource.dataKeys[0];\n var labelCellId = ''labelCell'' + 0;\n var cellId = ''cell'' + 0;\n if (self.ctx.labelPosition === ''left'') {\n table.append(\n \"<tr><td class=''tbDatasource-data-key'' id=''\" + labelCellId +\"''>\" +\n dataKey.label +\n \"</td><td class=''tbDatasource-value'' id=''\" +\n cellId +\n \"''></td></tr>\");\n } else {\n table.append(\n \"<tr style=''vertical-align: bottom;''><td class=''tbDatasource-data-key'' id=''\" + labelCellId +\"''>\" +\n dataKey.label +\n \"</td></tr><tr><td class=''tbDatasource-value'' id=''\" +\n cellId +\n \"''></td></tr>\");\n }\n self.ctx.labelCell = $(''#'' + labelCellId, table);\n self.ctx.valueCell = $(''#'' + cellId, table);\n self.ctx.valueCell.html(0 + '' '' + self.ctx.units);\n }\n }\n \n $.fn.textWidth = function(){\n var html_org = $(this).html();\n var html_calc = ''<span>'' + html_org + ''</span>'';\n $(this).html(html_calc);\n var width = $(this).find(''span:first'').width();\n $(this).html(html_org);\n return width;\n }; \n \n self.onResize();\n}\n\nself.onDataUpdated = function() {\n \n function isNumber(n) {\n return !isNaN(parseFloat(n)) && isFinite(n);\n }\n \n function padValue(val, dec, int) {\n var i = 0;\n var s, strVal, n;\n \n val = parseFloat(val);\n n = (val < 0);\n val = Math.abs(val);\n \n if (dec > 0) {\n strVal = val.toFixed(dec).toString().split(''.'');\n s = int - strVal[0].length;\n \n for (; i < s; ++i) {\n strVal[0] = ''0'' + strVal[0];\n }\n \n strVal = (n ? ''-'' : '''') + strVal[0] + ''.'' + strVal[1];\n }\n \n else {\n strVal = Math.round(val).toString();\n s = int - strVal.length;\n \n for (; i < s; ++i) {\n strVal = ''0'' + strVal;\n }\n \n strVal = (n ? ''-'' : '''') + strVal;\n }\n \n return strVal;\n }\n \n if (self.ctx.valueCell && self.ctx.data.length > 0) {\n var cellData = self.ctx.data[0];\n if (cellData.data.length > 0) {\n var tvPair = cellData.data[cellData.data.length -\n 1];\n var value = tvPair[1];\n var txtValue;\n if (isNumber(value)) {\n txtValue = padValue(value, self.ctx.alueDec, 0) + '' '' + self.ctx.units;\n } else {\n txtValue = value;\n }\n self.ctx.valueCell.html(txtValue);\n var targetWidth;\n var minDelta;\n if (self.ctx.labelPosition === ''left'') {\n targetWidth = self.ctx.datasourceContainer.width() - self.ctx.labelCell.width();\n minDelta = self.ctx.width/16 + self.ctx.padding;\n } else {\n targetWidth = self.ctx.datasourceContainer.width();\n minDelta = self.ctx.padding;\n }\n var delta = targetWidth - self.ctx.valueCell.textWidth();\n var fontSize = self.ctx.valueFontSize;\n if (targetWidth > minDelta) {\n while (delta < minDelta && fontSize > 6) {\n fontSize--;\n self.ctx.valueCell.css(''font-size'', fontSize+''px'');\n delta = targetWidth - self.ctx.valueCell.textWidth();\n }\n }\n }\n } \n \n}\n\nself.onResize = function() {\n var labelFontSize;\n if (self.ctx.labelPosition === ''top'') {\n self.ctx.padding = self.ctx.height/20;\n labelFontSize = self.ctx.height/4;\n self.ctx.valueFontSize = self.ctx.height/2;\n } else {\n self.ctx.padding = self.ctx.width/50;\n labelFontSize = self.ctx.height/2.5;\n self.ctx.valueFontSize = height/2;\n if (self.ctx.width/self.ctx.height <= 2.7) {\n labelFontSize = self.ctx.width/7;\n self.ctx.valueFontSize = self.ctx.width/6;\n }\n }\n self.ctx.padding = Math.min(12, self.ctx.padding);\n \n if (self.ctx.labelCell) {\n self.ctx.labelCell.css(''font-size'', labelFontSize+''px'');\n self.ctx.labelCell.css(''padding'', self.ctx.padding+''px'');\n }\n if (self.ctx.valueCell) {\n self.ctx.valueCell.css(''font-size'', self.ctx.valueFontSize+''px'');\n self.ctx.valueCell.css(''padding'', self.ctx.padding+''px'');\n } \n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"Settings\",\n \"properties\": {\n \"labelPosition\": {\n \"title\": \"Label position\",\n \"type\": \"string\",\n \"default\": \"left\"\n }\n },\n \"required\": []\n },\n \"form\": [\n {\n \"key\": \"labelPosition\",\n \"type\": \"rc-select\",\n \"multiple\": false,\n \"items\": [\n {\n \"value\": \"left\",\n \"label\": \"Left\"\n },\n {\n \"value\": \"top\",\n \"label\": \"Top\"\n }\n ]\n }\n ]\n}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Temp\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.2392660816082064,\"funcBody\":\"var value = prevValue + Math.random() * 40 - 20;\\nif (value < -60) {\\n\\tvalue = -60;\\n} else if (value > 60) {\\n\\tvalue = 60;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"#ff5722\",\"color\":\"rgba(255, 255, 255, 0.87)\",\"padding\":\"16px\",\"settings\":{\"labelPosition\":\"top\"},\"title\":\"Simple card\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"units\":\"°C\",\"decimals\":0,\"useDashboardTimewindow\":true,\"showLegend\":false}"}',
153   -'Simple card' );
154   -
155   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
156   -VALUES ( now ( ), minTimeuuid ( 0 ), 'digital_gauges', 'lcd_bar_gauge',
157   -'{"type":"latest","sizeX":2,"sizeY":3.5,"resources":[],"templateHtml":"<canvas id=\"digitalGauge\"></canvas>","templateCss":"#gauge {\n text-align: center;\n /* margin-left: -100px;\n margin-right: -100px;*/\n /*margin-top: -50px;*/\n \n}\n","controllerScript":"self.onInit = function() {\n self.ctx.gauge = new TbCanvasDigitalGauge(self.ctx, ''digitalGauge''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.gauge.update();\n}\n\nself.onResize = function() {\n self.ctx.gauge.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbCanvasDigitalGauge.settingsSchema;\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.gauge.mobileModeChanged();\n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Humidity\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.7282710489093589,\"funcBody\":\"var value = prevValue + Math.random() * 40 - 20;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"#babab2\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"0px\",\"settings\":{\"maxValue\":100,\"minValue\":0,\"donutStartAngle\":90,\"showValue\":true,\"showMinMax\":true,\"gaugeWidthScale\":0.75,\"levelColors\":[],\"refreshAnimationType\":\"linear\",\"refreshAnimationTime\":700,\"startAnimationType\":\"linear\",\"startAnimationTime\":700,\"titleFont\":{\"family\":\"Roboto\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\"},\"labelFont\":{\"family\":\"Roboto\",\"size\":8,\"style\":\"normal\",\"weight\":\"500\"},\"valueFont\":{\"family\":\"Roboto\",\"style\":\"normal\",\"weight\":\"400\",\"size\":16},\"minMaxFont\":{\"family\":\"Segment7Standard\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\"},\"neonGlowBrightness\":0,\"dashThickness\":1.5,\"decimals\":0,\"showUnitTitle\":true,\"defaultColor\":\"#444444\",\"gaugeType\":\"verticalBar\",\"units\":\"%\"},\"title\":\"LCD bar gauge\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',
158   -'LCD bar gauge' );
159   -
160   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
161   -VALUES ( now ( ), minTimeuuid ( 0 ), 'digital_gauges',
162   -'vertical_bar_justgage',
163   -'{"type":"latest","sizeX":2,"sizeY":3.5,"resources":[],"templateHtml":"<canvas id=\"digitalGauge\"></canvas>","templateCss":"#gauge {\n text-align: center;\n /* margin-left: -100px;\n margin-right: -100px;*/\n /*margin-top: -50px;*/\n \n}\n","controllerScript":"self.onInit = function() {\n self.ctx.gauge = new TbCanvasDigitalGauge(self.ctx, ''digitalGauge''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.gauge.update();\n}\n\nself.onResize = function() {\n self.ctx.gauge.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbCanvasDigitalGauge.settingsSchema;\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.gauge.mobileModeChanged();\n}\n\nself.onDestroy = function() {\n}\n\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Temp\",\"color\":\"#f57c00\",\"settings\":{},\"_hash\":0.7282710489093589,\"funcBody\":\"var value = prevValue + Math.random() * 20 - 10;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"#ffffff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"0px\",\"settings\":{\"maxValue\":100,\"minValue\":0,\"donutStartAngle\":90,\"showValue\":true,\"showMinMax\":true,\"gaugeWidthScale\":0.75,\"levelColors\":[],\"refreshAnimationType\":\">\",\"refreshAnimationTime\":700,\"startAnimationType\":\">\",\"startAnimationTime\":700,\"titleFont\":{\"family\":\"Roboto\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\",\"color\":\"#999999\"},\"labelFont\":{\"family\":\"Roboto\",\"size\":8,\"style\":\"normal\",\"weight\":\"500\"},\"valueFont\":{\"family\":\"Roboto\",\"style\":\"normal\",\"weight\":\"500\",\"size\":12,\"color\":\"#666666\"},\"minMaxFont\":{\"family\":\"Roboto\",\"size\":8,\"style\":\"normal\",\"weight\":\"500\",\"color\":\"#666666\"},\"neonGlowBrightness\":0,\"decimals\":0,\"dashThickness\":1.5,\"gaugeColor\":\"#eeeeee\",\"showTitle\":false,\"gaugeType\":\"verticalBar\"},\"title\":\"Vertical bar - justGage\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',
164   -'Vertical bar - justGage' );
165   -
166   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
167   -VALUES ( now ( ), minTimeuuid ( 0 ), 'cards', 'html_value_card',
168   -'{"type":"latest","sizeX":7.5,"sizeY":3,"resources":[],"templateHtml":"","templateCss":"","controllerScript":"self.onInit = function() {\n self.ctx.varsRegex = /\\$\\{([^\\}]*)\\}/g;\n self.ctx.htmlSet = false;\n \n var cssParser = new cssjs();\n cssParser.testMode = false;\n var namespace = ''html-value-card-'' + hashCode(self.ctx.settings.cardCss);\n cssParser.cssPreviewNamespace = namespace;\n cssParser.createStyleElement(namespace, self.ctx.settings.cardCss);\n self.ctx.$container.addClass(namespace);\n self.ctx.html = self.ctx.settings.cardHtml;\n self.ctx.replaceInfo = processHtmlPattern(self.ctx.html, self.ctx.data);\n \n updateHtml();\n \n function hashCode(str) {\n var hash = 0;\n var i, char;\n if (str.length === 0) return hash;\n for (i = 0; i < str.length; i++) {\n char = str.charCodeAt(i);\n hash = ((hash << 5) - hash) + char;\n hash = hash & hash;\n }\n return hash;\n }\n \n function processHtmlPattern(pattern, data) {\n var match = self.ctx.varsRegex.exec(pattern);\n var replaceInfo = {};\n replaceInfo.variables = [];\n while (match !== null) {\n var variableInfo = {};\n variableInfo.dataKeyIndex = -1;\n var variable = match[0];\n var label = match[1];\n var valDec = 2;\n var splitVals = label.split('':'');\n if (splitVals.length > 1) {\n label = splitVals[0];\n valDec = parseFloat(splitVals[1]);\n }\n variableInfo.variable = variable;\n variableInfo.valDec = valDec;\n \n if (label.startsWith(''#'')) {\n var keyIndexStr = label.substring(1);\n var n = Math.floor(Number(keyIndexStr));\n if (String(n) === keyIndexStr && n >= 0) {\n variableInfo.dataKeyIndex = n;\n }\n }\n if (variableInfo.dataKeyIndex === -1) {\n for (var i = 0; i < data.length; i++) {\n var datasourceData = data[i];\n var dataKey = datasourceData.dataKey;\n if (dataKey.label === label) {\n variableInfo.dataKeyIndex = i;\n break;\n }\n }\n }\n replaceInfo.variables.push(variableInfo);\n match = self.ctx.varsRegex.exec(pattern);\n }\n return replaceInfo;\n } \n}\n\nself.onDataUpdated = function() {\n updateHtml();\n}\n\nself.onDestroy = function() {\n}\n\nfunction isNumber(n) {\n return !isNaN(parseFloat(n)) && isFinite(n);\n}\n\nfunction padValue(val, dec, int) {\n var i = 0;\n var s, strVal, n;\n\n val = parseFloat(val);\n n = (val < 0);\n val = Math.abs(val);\n\n if (dec > 0) {\n strVal = val.toFixed(dec).toString().split(''.'');\n s = int - strVal[0].length;\n\n for (; i < s; ++i) {\n strVal[0] = ''0'' + strVal[0];\n }\n\n strVal = (n ? ''-'' : '''') + strVal[0] + ''.'' + strVal[1];\n }\n\n else {\n strVal = Math.round(val).toString();\n s = int - strVal.length;\n\n for (; i < s; ++i) {\n strVal = ''0'' + strVal;\n }\n\n strVal = (n ? ''-'' : '''') + strVal;\n }\n\n return strVal;\n}\n\nfunction updateHtml() {\n var text = self.ctx.html;\n var updated = false;\n for (var v in self.ctx.replaceInfo.variables) {\n var variableInfo = self.ctx.replaceInfo.variables[v];\n var txtVal = '''';\n if (variableInfo.dataKeyIndex > -1) {\n var varData = self.ctx.data[variableInfo.dataKeyIndex].data;\n if (varData.length > 0) {\n var val = varData[varData.length-1][1];\n if (isNumber(val)) {\n txtVal = padValue(val, variableInfo.valDec, 0);\n } else {\n txtVal = val;\n }\n }\n }\n if (typeof variableInfo.lastVal === undefined ||\n variableInfo.lastVal !== txtVal) {\n updated = true;\n variableInfo.lastVal = txtVal;\n }\n text = text.split(variableInfo.variable).join(txtVal);\n }\n if (updated || !self.ctx.htmlSet) {\n self.ctx.$container.html(text);\n if (!self.ctx.htmlSet) {\n self.ctx.htmlSet = true;\n }\n }\n}\n\n","settingsSchema":"{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"Settings\",\n \"required\": [\"cardHtml\"],\n \"properties\": {\n \"cardCss\": {\n \"title\": \"CSS\",\n \"type\": \"string\",\n \"default\": \".card {\\n font-weight: bold; \\n}\"\n },\n \"cardHtml\": {\n \"title\": \"HTML\",\n \"type\": \"string\",\n \"default\": \"<div class=''card''>HTML code here</div>\"\n }\n }\n },\n \"form\": [\n {\n \"key\": \"cardCss\",\n \"type\": \"css\"\n }, \n {\n \"key\": \"cardHtml\",\n \"type\": \"html\"\n } \n ]\n}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"My value\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.15479322438769105,\"funcBody\":\"return Math.random() * 5.45;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"0px\",\"settings\":{\"cardCss\":\".card {\\n width: 100%;\\n height: 100%;\\n border: 2px solid #ccc;\\n box-sizing: border-box;\\n}\\n\\n.card .content {\\n padding: 20px;\\n display: flex;\\n flex-direction: row;\\n align-items: center;\\n justify-content: space-around;\\n height: 100%;\\n box-sizing: border-box;\\n}\\n\\n.card .content .column {\\n display: flex;\\n flex-direction: column; \\n justify-content: space-around;\\n height: 100%;\\n}\\n\\n.card h1 {\\n text-transform: uppercase;\\n color: #999;\\n font-size: 20px;\\n font-weight: bold;\\n margin: 0;\\n padding-bottom: 10px;\\n line-height: 32px;\\n}\\n\\n.card .value {\\n font-size: 38px;\\n font-weight: 200;\\n}\\n\\n.card .description {\\n font-size: 20px;\\n color: #999;\\n}\\n\",\"cardHtml\":\"<div class=''card''>\\n <div class=''content''>\\n <div class=''column''>\\n <h1>Value title</h1>\\n <div class=''value''>\\n ${My value:2} units.\\n </div> \\n <div class=''description''>\\n Value description text\\n </div>\\n </div>\\n <img height=\\\"80px\\\" src=\\\"https://thingsboard.io/images/logo_small.png\\\" />\\n </div>\\n</div>\"},\"title\":\"HTML Value Card\",\"dropShadow\":false}"}',
169   -'HTML Value Card' );
170   -
171   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
172   -VALUES ( now ( ), minTimeuuid ( 0 ), 'gpio_widgets',
173   -'raspberry_pi_gpio_panel',
174   -'{"type":"latest","sizeX":7,"sizeY":10.5,"resources":[],"templateHtml":"<div class=\"gpio-panel\" style=\"height: 100%;\">\n <section layout=\"row\" ng-repeat=\"row in rows\">\n <section flex layout=\"row\" ng-repeat=\"cell in row\">\n <section layout=\"row\" flex ng-if=\"cell\" layout-align=\"{{$index===0 ? ''end center'' : ''start center''}}\">\n <span class=\"gpio-left-label\" ng-show=\"$index===0\">{{ cell.label }}</span>\n <section layout=\"row\" class=\"led-panel\" ng-class=\"$index===0 ? ''col-0'' : ''col-1''\"\n ng-style=\"{backgroundColor: ledPanelBackgroundColor}\">\n <span class=\"pin\" ng-show=\"$index===0\">{{cell.pin}}</span>\n <span class=\"led-container\">\n <tb-led-light size=\"prefferedRowHeight\"\n color-on=\"cell.colorOn\"\n color-off=\"cell.colorOff\"\n off-opacity=\"''0.9''\"\n tb-enabled=\"cell.enabled\">\n </tb-led-light>\n </span>\n <span class=\"pin\" ng-show=\"$index===1\">{{cell.pin}}</span>\n </section>\n <span class=\"gpio-right-label\" ng-show=\"$index===1\">{{ cell.label }}</span>\n </section>\n <section layout=\"row\" flex ng-if=\"!cell\">\n <span flex ng-show=\"$index===0\"></span>\n <span class=\"led-panel\"\n ng-style=\"{backgroundColor: ledPanelBackgroundColor}\"></span>\n <span flex ng-show=\"$index===1\"></span>\n </section>\n </section>\n </section> \n</div>","templateCss":".error {\n font-size: 14px !important;\n color: maroon;/*rgb(250,250,250);*/\n background-color: transparent;\n padding: 6px;\n}\n\n.error span {\n margin: auto;\n}\n\n.gpio-panel {\n padding-top: 10px;\n white-space: nowrap;\n}\n\n.gpio-panel tb-led-light > div {\n margin: auto;\n}\n\n.led-panel {\n margin: 0;\n width: 66px;\n min-width: 66px;\n}\n\n.led-container {\n width: 48px;\n min-width: 48px;\n}\n\n.pin {\n margin-top: auto;\n margin-bottom: auto;\n color: white;\n font-size: 12px;\n width: 16px;\n min-width: 16px;\n}\n\n.led-panel.col-0 .pin {\n margin-left: auto;\n padding-left: 2px;\n text-align: right;\n}\n\n.led-panel.col-1 .pin {\n margin-right: auto;\n \n text-align: left;\n}\n\n.gpio-left-label {\n margin-right: 8px;\n}\n\n.gpio-right-label {\n margin-left: 8px;\n}","controllerScript":"self.onInit = function() {\n var i, gpio;\n \n var scope = self.ctx.$scope;\n var settings = self.ctx.settings;\n \n scope.gpioList = [];\n scope.gpioByPin = {};\n for (var g = 0; g < settings.gpioList.length; g++) {\n gpio = settings.gpioList[g];\n scope.gpioList.push(\n {\n row: gpio.row,\n col: gpio.col,\n pin: gpio.pin,\n label: gpio.label,\n enabled: false,\n colorOn: tinycolor(gpio.color).lighten(20).toHexString(),\n colorOff: tinycolor(gpio.color).darken().toHexString()\n }\n );\n scope.gpioByPin[gpio.pin] = scope.gpioList[scope.gpioList.length-1];\n }\n\n scope.ledPanelBackgroundColor = settings.ledPanelBackgroundColor || tinycolor(''green'').lighten(2).toRgbString();\n\n scope.gpioCells = {};\n var rowCount = 0;\n for (i = 0; i < scope.gpioList.length; i++) {\n gpio = scope.gpioList[i];\n scope.gpioCells[gpio.row+''_''+gpio.col] = gpio;\n rowCount = Math.max(rowCount, gpio.row+1);\n }\n \n scope.prefferedRowHeight = 32;\n scope.rows = [];\n for (i = 0; i < rowCount; i++) {\n var row = [];\n for (var c =0; c<2;c++) {\n if (scope.gpioCells[i+''_''+c]) {\n row[c] = scope.gpioCells[i+''_''+c];\n } else {\n row[c] = null;\n }\n }\n scope.rows.push(row);\n } \n \n self.onResize();\n}\n\nself.onDataUpdated = function() {\n var changed = false;\n for (var d = 0; d < self.ctx.data.length; d++) {\n var cellData = self.ctx.data[d];\n var dataKey = cellData.dataKey;\n var gpio = self.ctx.$scope.gpioByPin[dataKey.label];\n if (gpio) {\n var enabled = false;\n if (cellData.data.length > 0) {\n var tvPair = cellData.data[cellData.data.length - 1];\n enabled = (tvPair[1] === true || tvPair[1] === ''true'');\n }\n if (gpio.enabled != enabled) {\n changed = true;\n gpio.enabled = enabled;\n }\n }\n }\n if (changed) {\n self.ctx.$scope.$digest();\n } \n}\n\nself.onResize = function() {\n var rowCount = self.ctx.$scope.rows.length;\n var prefferedRowHeight = (self.ctx.height - 35)/rowCount;\n prefferedRowHeight = Math.min(32, prefferedRowHeight);\n prefferedRowHeight = Math.max(12, prefferedRowHeight);\n self.ctx.$scope.prefferedRowHeight = prefferedRowHeight;\n \n var ratio = prefferedRowHeight/32;\n \n var leftLabels = $(''.gpio-left-label'', self.ctx.$container);\n leftLabels.css(''font-size'', 16*ratio+''px'');\n var rightLabels = $(''.gpio-right-label'', self.ctx.$container);\n rightLabels.css(''font-size'', 16*ratio+''px'');\n var pins = $(''.pin'', self.ctx.$container);\n var pinsFontSize = Math.max(9, 12*ratio);\n pins.css(''font-size'', pinsFontSize+''px''); \n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"Settings\",\n \"properties\": {\n \"gpioList\": {\n \"title\": \"Gpio leds\",\n \"type\": \"array\",\n \"minItems\" : 1,\n \"items\": {\n \"title\": \"Gpio led\",\n \"type\": \"object\",\n \"properties\": {\n \"pin\": {\n \"title\": \"Pin\",\n \"type\": \"number\"\n },\n \"label\": {\n \"title\": \"Label\",\n \"type\": \"string\"\n },\n \"row\": {\n \"title\": \"Row\",\n \"type\": \"number\"\n },\n \"col\": {\n \"title\": \"Column\",\n \"type\": \"number\"\n },\n \"color\": {\n \"title\": \"Color\",\n \"type\": \"string\",\n \"default\": \"red\"\n }\n },\n \"required\": [\"pin\", \"label\", \"row\", \"col\", \"color\"]\n }\n },\n \"ledPanelBackgroundColor\": {\n \"title\": \"LED panel background color\",\n \"type\": \"string\",\n \"default\": \"#008a00\"\n } \n },\n \"required\": [\"gpioList\", \n \"ledPanelBackgroundColor\"]\n },\n \"form\": [\n {\n \"key\": \"gpioList\",\n \"items\": [\n \"gpioList[].pin\",\n \"gpioList[].label\",\n \"gpioList[].row\",\n \"gpioList[].col\",\n {\n \"key\": \"gpioList[].color\",\n \"type\": \"color\"\n }\n ]\n },\n {\n \"key\": \"ledPanelBackgroundColor\",\n \"type\": \"color\"\n }\n ]\n}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"0px\",\"settings\":{\"gpioList\":[{\"pin\":1,\"label\":\"3.3V\",\"row\":0,\"col\":0,\"color\":\"#fc9700\",\"_uniqueKey\":0},{\"pin\":2,\"label\":\"5V\",\"row\":0,\"col\":1,\"color\":\"#fb0000\",\"_uniqueKey\":1},{\"pin\":3,\"label\":\"GPIO 2 (I2C1_SDA)\",\"row\":1,\"col\":0,\"color\":\"#02fefb\",\"_uniqueKey\":2},{\"color\":\"#fb0000\",\"pin\":4,\"label\":\"5V\",\"row\":1,\"col\":1},{\"color\":\"#02fefb\",\"pin\":5,\"label\":\"GPIO 3 (I2C1_SCL)\",\"row\":2,\"col\":0},{\"color\":\"#000000\",\"pin\":6,\"label\":\"GND\",\"row\":2,\"col\":1},{\"color\":\"#00fd00\",\"pin\":7,\"label\":\"GPIO 4 (GPCLK0)\",\"row\":3,\"col\":0},{\"color\":\"#fdfb00\",\"pin\":8,\"label\":\"GPIO 14 (UART_TXD)\",\"row\":3,\"col\":1},{\"color\":\"#000000\",\"pin\":9,\"label\":\"GND\",\"row\":4,\"col\":0},{\"color\":\"#fdfb00\",\"pin\":10,\"label\":\"GPIO 15 (UART_RXD)\",\"row\":4,\"col\":1},{\"color\":\"#00fd00\",\"pin\":11,\"label\":\"GPIO 17\",\"row\":5,\"col\":0},{\"color\":\"#00fd00\",\"pin\":12,\"label\":\"GPIO 18\",\"row\":5,\"col\":1},{\"color\":\"#00fd00\",\"pin\":13,\"label\":\"GPIO 27\",\"row\":6,\"col\":0},{\"color\":\"#000000\",\"pin\":14,\"label\":\"GND\",\"row\":6,\"col\":1},{\"color\":\"#00fd00\",\"pin\":15,\"label\":\"GPIO 22\",\"row\":7,\"col\":0},{\"color\":\"#00fd00\",\"pin\":16,\"label\":\"GPIO 23\",\"row\":7,\"col\":1},{\"color\":\"#fc9700\",\"pin\":17,\"label\":\"3.3V\",\"row\":8,\"col\":0},{\"color\":\"#00fd00\",\"pin\":18,\"label\":\"GPIO 24\",\"row\":8,\"col\":1},{\"color\":\"#fd01fd\",\"pin\":19,\"label\":\"GPIO 10 (SPI_MOSI)\",\"row\":9,\"col\":0},{\"color\":\"#000000\",\"pin\":20,\"label\":\"GND\",\"row\":9,\"col\":1},{\"color\":\"#fd01fd\",\"pin\":21,\"label\":\"GPIO 9 (SPI_MISO)\",\"row\":10,\"col\":0},{\"color\":\"#00fd00\",\"pin\":22,\"label\":\"GPIO 25\",\"row\":10,\"col\":1},{\"color\":\"#fd01fd\",\"pin\":23,\"label\":\"GPIO 11 (SPI_SCLK)\",\"row\":11,\"col\":0},{\"color\":\"#fd01fd\",\"pin\":24,\"label\":\"GPIO 8 (SPI_CE0)\",\"row\":11,\"col\":1},{\"color\":\"#000000\",\"pin\":25,\"label\":\"GND\",\"row\":12,\"col\":0},{\"color\":\"#fd01fd\",\"pin\":26,\"label\":\"GPIO 7 (SPI_CE1)\",\"row\":12,\"col\":1},{\"color\":\"#ffffff\",\"pin\":27,\"label\":\"ID_SD\",\"row\":13,\"col\":0},{\"color\":\"#ffffff\",\"pin\":28,\"label\":\"ID_SC\",\"row\":13,\"col\":1},{\"color\":\"#00fd00\",\"pin\":29,\"label\":\"GPIO 5\",\"row\":14,\"col\":0},{\"color\":\"#000000\",\"pin\":30,\"label\":\"GND\",\"row\":14,\"col\":1},{\"color\":\"#00fd00\",\"pin\":31,\"label\":\"GPIO 6\",\"row\":15,\"col\":0},{\"color\":\"#00fd00\",\"pin\":32,\"label\":\"GPIO 12\",\"row\":15,\"col\":1},{\"color\":\"#00fd00\",\"pin\":33,\"label\":\"GPIO 13\",\"row\":16,\"col\":0},{\"color\":\"#000000\",\"pin\":34,\"label\":\"GND\",\"row\":16,\"col\":1},{\"color\":\"#00fd00\",\"pin\":35,\"label\":\"GPIO 19\",\"row\":17,\"col\":0},{\"color\":\"#00fd00\",\"pin\":36,\"label\":\"GPIO 16\",\"row\":17,\"col\":1},{\"color\":\"#00fd00\",\"pin\":37,\"label\":\"GPIO 26\",\"row\":18,\"col\":0},{\"color\":\"#00fd00\",\"pin\":38,\"label\":\"GPIO 20\",\"row\":18,\"col\":1},{\"color\":\"#000000\",\"pin\":39,\"label\":\"GND\",\"row\":19,\"col\":0},{\"color\":\"#00fd00\",\"pin\":40,\"label\":\"GPIO 21\",\"row\":19,\"col\":1}],\"ledPanelBackgroundColor\":\"#008a00\"},\"title\":\"Raspberry Pi GPIO Panel\",\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"7\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.22518255793320163,\"funcBody\":\"var period = time % 1500;\\nreturn period < 500;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"11\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.7008206860666621,\"funcBody\":\"var period = time % 1500;\\nreturn period >= 500 && period < 1000;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"12\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.42600325102193426,\"funcBody\":\"var period = time % 1500;\\nreturn period >= 1000;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"13\",\"color\":\"#ffc107\",\"settings\":{},\"_hash\":0.48362241571415243,\"funcBody\":\"var period = time % 1500;\\nreturn period < 500;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"29\",\"color\":\"#607d8b\",\"settings\":{},\"_hash\":0.7217670147518815,\"funcBody\":\"var period = time % 1500;\\nreturn period >= 500 && period < 1000;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}}}"}',
175   -'Raspberry Pi GPIO Panel' );
176   -
177   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
178   -VALUES ( now ( ), minTimeuuid ( 0 ), 'charts', 'radar_chart_js',
179   -'{"type":"latest","sizeX":7,"sizeY":5,"resources":[{"url":"https://cdnjs.cloudflare.com/ajax/libs/Chart.js/2.3.0/Chart.min.js"}],"templateHtml":"<canvas id=\"radarChart\"></canvas>\n","templateCss":"","controllerScript":"self.onInit = function() {\n var barData = {\n labels: [],\n datasets: []\n };\n\n var backgroundColor = tinycolor(self.ctx.data[0].dataKey.color);\n backgroundColor.setAlpha(0.2);\n var borderColor = tinycolor(self.ctx.data[0].dataKey.color);\n borderColor.setAlpha(1);\n var dataset = {\n label: self.ctx.datasources[0].name,\n data: [],\n backgroundColor: backgroundColor.toRgbString(),\n borderColor: borderColor.toRgbString(),\n pointBackgroundColor: borderColor.toRgbString(),\n pointBorderColor: borderColor.darken().toRgbString(),\n borderWidth: 1\n }\n \n barData.datasets.push(dataset);\n \n for (var i = 0; i < self.ctx.data.length; i++) {\n var dataKey = self.ctx.data[i].dataKey;\n barData.labels.push(dataKey.label);\n dataset.data.push(0);\n }\n\n var ctx = $(''#radarChart'', self.ctx.$container);\n self.ctx.chart = new Chart(ctx, {\n type: ''radar'',\n data: barData,\n options: {\n responsive: false,\n maintainAspectRatio: false\n }\n });\n \n self.onResize();\n}\n\nself.onDataUpdated = function() {\n for (var i = 0; i < self.ctx.data.length; i++) {\n var cellData = self.ctx.data[i];\n if (cellData.data.length > 0) {\n var tvPair = cellData.data[cellData.data.length - 1];\n var value = tvPair[1];\n self.ctx.chart.data.datasets[0].data[i] = parseFloat(value);\n }\n } \n self.ctx.chart.update();\n}\n\nself.onResize = function() {\n if (self.ctx.height >= 70) {\n self.ctx.chart.resize();\n }\n}\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"First\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.15479322438769105,\"funcBody\":\"var value = (prevValue-50) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+50;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Second\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.545701115289893,\"funcBody\":\"var value = (prevValue-20) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+20;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Third\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.2592906835158064,\"funcBody\":\"var value = (prevValue-40) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+40;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Fourth\",\"color\":\"#ffc107\",\"settings\":{},\"_hash\":0.12880275585455747,\"funcBody\":\"var value = (prevValue-50) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+50;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{},\"title\":\"Radar - Chart.js\"}"}',
180   -'Radar - Chart.js' );
181   -
182   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
183   -VALUES ( now ( ), minTimeuuid ( 0 ), 'digital_gauges',
184   -'neon_gauge_justgage',
185   -'{"type":"latest","sizeX":5,"sizeY":3,"resources":[],"templateHtml":"<canvas id=\"digitalGauge\"></canvas>","templateCss":"#gauge {\n text-align: center;\n /* margin-left: -100px;\n margin-right: -100px;*/\n /*margin-top: -50px;*/\n \n}\n","controllerScript":"self.onInit = function() {\n self.ctx.gauge = new TbCanvasDigitalGauge(self.ctx, ''digitalGauge''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.gauge.update();\n}\n\nself.onResize = function() {\n self.ctx.gauge.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbCanvasDigitalGauge.settingsSchema;\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.gauge.mobileModeChanged();\n}\n\nself.onDestroy = function() {\n}\n\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Temp\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.7282710489093589,\"funcBody\":\"var value = prevValue + Math.random() * 20 - 10;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"#000000\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"0px\",\"settings\":{\"maxValue\":100,\"minValue\":0,\"donutStartAngle\":90,\"showValue\":true,\"showMinMax\":true,\"gaugeWidthScale\":0.75,\"levelColors\":[],\"refreshAnimationType\":\">\",\"refreshAnimationTime\":700,\"startAnimationType\":\">\",\"startAnimationTime\":700,\"titleFont\":{\"family\":\"Roboto\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\"},\"labelFont\":{\"family\":\"Roboto\",\"size\":8,\"style\":\"normal\",\"weight\":\"500\"},\"valueFont\":{\"family\":\"Segment7Standard\",\"style\":\"normal\",\"weight\":\"500\",\"size\":32},\"minMaxFont\":{\"family\":\"Segment7Standard\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\"},\"neonGlowBrightness\":70,\"dashThickness\":1,\"decimals\":1,\"gaugeType\":\"arc\"},\"title\":\"Neon gauge - justGage\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',
186   -'Neon gauge - justGage' );
187   -
188   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
189   -VALUES ( now ( ), minTimeuuid ( 0 ), 'digital_gauges',
190   -'simple_gauge_justgage',
191   -'{"type":"latest","sizeX":2,"sizeY":2,"resources":[],"templateHtml":"<canvas id=\"digitalGauge\"></canvas>\n","templateCss":"#gauge {\n text-align: center;\n /* margin-left: -100px;\n margin-right: -100px;*/\n /*margin-top: -50px;*/\n \n}\n","controllerScript":"\nself.onInit = function() {\n self.ctx.gauge = new TbCanvasDigitalGauge(self.ctx, ''digitalGauge''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.gauge.update();\n}\n\nself.onResize = function() {\n self.ctx.gauge.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbCanvasDigitalGauge.settingsSchema;\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.gauge.mobileModeChanged();\n}\n\nself.onDestroy = function() {\n}\n\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Temp\",\"color\":\"#ef6c00\",\"settings\":{},\"_hash\":0.7282710489093589,\"funcBody\":\"var value = prevValue + Math.random() * 20 - 10;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"#ffffff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"0px\",\"settings\":{\"maxValue\":100,\"minValue\":0,\"donutStartAngle\":90,\"showValue\":true,\"showMinMax\":true,\"gaugeWidthScale\":0.75,\"levelColors\":[],\"refreshAnimationType\":\">\",\"refreshAnimationTime\":700,\"startAnimationType\":\">\",\"startAnimationTime\":700,\"titleFont\":{\"family\":\"Roboto\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\"},\"labelFont\":{\"family\":\"Roboto\",\"size\":8,\"style\":\"normal\",\"weight\":\"500\"},\"valueFont\":{\"family\":\"Roboto\",\"style\":\"normal\",\"weight\":\"500\",\"size\":32,\"color\":\"#666666\"},\"minMaxFont\":{\"family\":\"Segment7Standard\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\"},\"neonGlowBrightness\":0,\"dashThickness\":0,\"decimals\":0,\"gaugeColor\":\"#eeeeee\",\"gaugeType\":\"donut\"},\"title\":\"Simple gauge - justGage\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',
192   -'Simple gauge - justGage' );
193   -
194   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
195   -VALUES ( now ( ), minTimeuuid ( 0 ), 'maps', 'route_map',
196   -'{"type":"timeseries","sizeX":8.5,"sizeY":6,"resources":[],"templateHtml":"","templateCss":".error {\n color: red;\n}\n.tb-labels {\n color: #222;\n font: 12px/1.5 \"Helvetica Neue\", Arial, Helvetica, sans-serif;\n text-align: center;\n width: 100px;\n white-space: nowrap;\n}","controllerScript":"self.onInit = function() {\n self.ctx.map = new TbMapWidget(''google-map'', true, self.ctx);\n}\n\nself.onDataUpdated = function() {\n self.ctx.map.update();\n}\n\nself.onResize = function() {\n self.ctx.map.resize();\n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{\n \"schema\": {\n \"title\": \"Route Map Configuration\",\n \"type\": \"object\",\n \"properties\": {\n \"gmApiKey\": {\n \"title\": \"Google Maps API Key\",\n \"type\": \"string\"\n },\n \"gmDefaultMapType\": {\n \"title\": \"Default map type\",\n \"type\": \"string\",\n \"default\": \"roadmap\"\n },\n \"defaultZoomLevel\": {\n \"title\": \"Default map zoom level (1 - 20)\",\n \"type\": \"number\"\n },\n \"fitMapBounds\": {\n \"title\": \"Fit map bounds to cover all routes\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"routesSettings\": {\n \"title\": \"Routes\",\n \"type\": \"array\",\n \"items\": {\n \"title\": \"Route settings\",\n \"type\": \"object\",\n \"properties\": {\n \"latKeyName\": {\n \"title\": \"Latitude key name\",\n \"type\": \"string\",\n \"default\": \"lat\"\n },\n \"lngKeyName\": {\n \"title\": \"Longitude key name\",\n \"type\": \"string\",\n \"default\": \"lng\"\n },\n \"showLabel\": {\n \"title\": \"Show label\",\n \"type\": \"boolean\",\n \"default\": true\n }, \n \"label\": {\n \"title\": \"Label\",\n \"type\": \"string\"\n },\n \"tooltipPattern\": {\n \"title\": \"Pattern ( for ex. ''Text ${keyName} units.'' or ''${#<key index>} units'' )\",\n \"type\": \"string\",\n \"default\": \"<b>Latitude:</b> ${lat:7}<br/><b>Longitude:</b> ${lng:7}\"\n },\n \"color\": {\n \"title\": \"Color\",\n \"type\": \"string\"\n },\n \"useColorFunction\": {\n \"title\": \"Use color function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"colorFunction\": {\n \"title\": \"Color function: f(data)\",\n \"type\": \"string\"\n },\n \"markerImage\": {\n \"title\": \"Custom marker image\",\n \"type\": \"string\"\n },\n \"markerImageSize\": {\n \"title\": \"Custom marker image size (px)\",\n \"type\": \"number\",\n \"default\": 34\n },\n \"useMarkerImageFunction\": {\n \"title\": \"Use marker image function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"markerImageFunction\": {\n \"title\": \"Marker image function: f(data, images)\",\n \"type\": \"string\"\n },\n \"markerImages\": {\n \"title\": \"Marker images\",\n \"type\": \"array\",\n \"items\": {\n \"title\": \"Marker image\",\n \"type\": \"string\"\n }\n },\n \"strokeWeight\": {\n \"title\": \"Stroke weight\",\n \"type\": \"number\",\n \"default\": 2\n },\n \"strokeOpacity\": {\n \"title\": \"Stroke opacity\",\n \"type\": \"number\",\n \"default\": 1.0\n }\n }\n }\n }\n },\n \"required\": [\n \"gmApiKey\"\n ]\n },\n \"form\": [\n \"gmApiKey\",\n {\n \"key\": \"gmDefaultMapType\",\n \"type\": \"rc-select\",\n \"multiple\": false,\n \"items\": [\n {\n \"value\": \"roadmap\",\n \"label\": \"Roadmap\"\n },\n {\n \"value\": \"satellite\",\n \"label\": \"Satellite\"\n },\n {\n \"value\": \"hybrid\",\n \"label\": \"Hybrid\"\n },\n {\n \"value\": \"terrain\",\n \"label\": \"Terrain\"\n }\n ]\n }, \n \"defaultZoomLevel\",\n \"fitMapBounds\",\n {\n \"key\": \"routesSettings\",\n \"items\": [\n \"routesSettings[].latKeyName\",\n \"routesSettings[].lngKeyName\",\n \"routesSettings[].showLabel\",\n \"routesSettings[].label\",\n \"routesSettings[].tooltipPattern\",\n {\n \"key\": \"routesSettings[].color\",\n \"type\": \"color\"\n },\n \"routesSettings[].useColorFunction\",\n {\n \"key\": \"routesSettings[].colorFunction\",\n \"type\": \"javascript\"\n },\n {\n \"key\": \"routesSettings[].markerImage\",\n \"type\": \"image\"\n },\n \"routesSettings[].markerImageSize\",\n \"routesSettings[].useMarkerImageFunction\",\n {\n \"key\": \"routesSettings[].markerImageFunction\",\n \"type\": \"javascript\"\n },\n {\n \"key\": \"routesSettings[].markerImages\",\n \"items\": [\n {\n \"key\": \"routesSettings[].markerImages[]\",\n \"type\": \"image\"\n }\n ]\n },\n \"routesSettings[].strokeWeight\",\n \"routesSettings[].strokeOpacity\"\n ]\n }\n ]\n}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"latitude\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.3467277073670627,\"funcBody\":\"var lats = [37.7696499,\\n37.7699074,\\n37.7699536,\\n37.7697242,\\n37.7695189,\\n37.7696889,\\n37.7697153,\\n37.7701244,\\n37.7700604,\\n37.7705491,\\n37.7715705,\\n37.771752,\\n37.7707533,\\n37.769866];\\n\\nvar i = Math.floor((time/3 % 14000) / 1000);\\n\\nreturn lats[i];\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"longitude\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.058309787276281666,\"funcBody\":\"var lons = [-122.4261215,\\n-122.4219157,\\n-122.4199623,\\n-122.4179074,\\n-122.4155876,\\n-122.4155521,\\n-122.4163203,\\n-122.4193876,\\n-122.4210496,\\n-122.422284,\\n-122.4232717,\\n-122.4235138,\\n-122.4247605,\\n-122.4258812];\\n\\nvar i = Math.floor((time/3 % 14000) / 1000);\\n\\nreturn lons[i];\"}],\"intervalSec\":60},{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Speed\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.14288960550237473,\"funcBody\":\"var value = prevValue;\\nif (time % 500 < 100) {\\n value = value + Math.random() * 40 - 20;\\n if (value < 45) {\\n \\tvalue = 45;\\n } else if (value > 130) {\\n \\tvalue = 130;\\n }\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":30000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"gmApiKey\":\"AIzaSyDoEx2kaGz3PxwbI9T7ccTSg5xjdw8Nw8Q\",\"fitMapBounds\":true,\"routesSettings\":[{\"latKeyName\":\"latitude\",\"lngKeyName\":\"longitude\",\"showLabel\":true,\"color\":\"#1976d2\",\"strokeWeight\":4,\"strokeOpacity\":0.65,\"label\":\"First route\",\"tooltipPattern\":\"<b>Latitude:</b> ${latitude:7}<br/><b>Longitude:</b> ${longitude:7}<br/><b>Speed:</b> ${Speed} MPH<br/><small>See advanced settings for details</small>\",\"useColorFunction\":true,\"markerImageSize\":34,\"useMarkerImageFunction\":true,\"markerImageFunction\":\"var speed = data[''Speed''];\\nvar res = {\\n url: images[0],\\n size: 55\\n};\\nif (typeof speed !== undefined) {\\n var percent = (speed - 45)/85;\\n var index = Math.floor(3 * percent);\\n res.url = images[index];\\n}\\nreturn res;\",\"markerImages\":[\"\",\"\",\"\"],\"colorFunction\":\"var speed = data[''Speed''];\\nif (typeof speed !== undefined) {\\n var percent = (speed - 45)/85;\\n if (percent < 0.5) {\\n percent *=2*100; \\n return tinycolor.mix(''green'', ''yellow'', amount = percent).toHexString();\\n } else {\\n percent = (percent - 0.5)*2*100;\\n return tinycolor.mix(''yellow'', ''red'', amount = percent).toHexString();\\n }\\n}\\nreturn ''green'';\"}]},\"title\":\"Route Map\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false}"}',
197   -'Route Map' );
198   -
199   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
200   -VALUES ( now ( ), minTimeuuid ( 0 ), 'analogue_gauges',
201   -'temperature_gauge_canvas_gauges',
202   -'{"type":"latest","sizeX":7,"sizeY":3,"resources":[],"templateHtml":"<canvas id=\"linearGauge\"></canvas>\n","templateCss":"","controllerScript":"self.onInit = function() {\n self.ctx.gauge = new TbAnalogueLinearGauge(self.ctx, ''linearGauge''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.gauge.update();\n}\n\nself.onResize = function() {\n self.ctx.gauge.resize();\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.gauge.mobileModeChanged();\n}\n\nself.getSettingsSchema = function() {\n return TbAnalogueLinearGauge.settingsSchema;\n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Temp\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.7282710489093589,\"funcBody\":\"var value = prevValue + Math.random() * 30 - 15;\\nif (value < -60) {\\n\\tvalue = -60;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"rgb(255, 255, 255)\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"maxValue\":100,\"defaultColor\":\"#e64a19\",\"barStrokeWidth\":2.5,\"colorBar\":\"rgba(255, 255, 255, 0.4)\",\"colorBarEnd\":\"rgba(221, 221, 221, 0.38)\",\"showUnitTitle\":true,\"minorTicks\":2,\"valueBox\":true,\"valueInt\":3,\"colorPlate\":\"#fff\",\"colorMajorTicks\":\"#444\",\"colorMinorTicks\":\"#666\",\"colorNeedleShadowUp\":\"rgba(2,255,255,0.2)\",\"colorNeedleShadowDown\":\"rgba(188,143,143,0.45)\",\"colorValueBoxRect\":\"#888\",\"colorValueBoxRectEnd\":\"#666\",\"colorValueBoxBackground\":\"#babab2\",\"colorValueBoxShadow\":\"rgba(0,0,0,1)\",\"highlightsWidth\":10,\"animation\":true,\"animationDuration\":1500,\"animationRule\":\"linear\",\"showBorder\":false,\"majorTicksCount\":8,\"numbersFont\":{\"family\":\"Arial\",\"size\":18,\"style\":\"normal\",\"weight\":\"normal\",\"color\":\"#263238\"},\"titleFont\":{\"family\":\"Roboto\",\"size\":24,\"style\":\"normal\",\"weight\":\"normal\",\"color\":\"#78909c\"},\"unitsFont\":{\"family\":\"Roboto\",\"size\":26,\"style\":\"normal\",\"weight\":\"500\",\"color\":\"#37474f\"},\"valueFont\":{\"family\":\"Roboto\",\"size\":40,\"style\":\"normal\",\"weight\":\"500\",\"color\":\"#444\",\"shadowColor\":\"rgba(0,0,0,0.3)\"},\"minValue\":-60,\"highlights\":[{\"from\":-60,\"to\":-40,\"color\":\"#90caf9\"},{\"from\":-40,\"to\":-20,\"color\":\"rgba(144, 202, 249, 0.66)\"},{\"from\":-20,\"to\":0,\"color\":\"rgba(144, 202, 249, 0.33)\"},{\"from\":0,\"to\":20,\"color\":\"rgba(244, 67, 54, 0.2)\"},{\"from\":20,\"to\":40,\"color\":\"rgba(244, 67, 54, 0.4)\"},{\"from\":40,\"to\":60,\"color\":\"rgba(244, 67, 54, 0.6)\"},{\"from\":60,\"to\":80,\"color\":\"rgba(244, 67, 54, 0.8)\"},{\"from\":80,\"to\":100,\"color\":\"#f44336\"}],\"unitTitle\":\"Temperature\",\"units\":\"°C\",\"colorBarProgress\":\"#90caf9\",\"colorBarProgressEnd\":\"#f44336\",\"colorBarStroke\":\"#b0bec5\",\"valueDec\":1},\"title\":\"Temperature gauge - Canvas Gauges\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',
203   -'Temperature gauge - Canvas Gauges' );
204   -
205   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
206   -VALUES ( now ( ), minTimeuuid ( 0 ), 'maps', 'google_maps',
207   -'{"type":"latest","sizeX":8.5,"sizeY":6,"resources":[],"templateHtml":"","templateCss":".error {\n color: red;\n}\n.tb-labels {\n color: #222;\n font: 12px/1.5 \"Helvetica Neue\", Arial, Helvetica, sans-serif;\n text-align: center;\n width: 100px;\n white-space: nowrap;\n}","controllerScript":"self.onInit = function() {\n self.ctx.map = new TbMapWidget(''google-map'', false, self.ctx);\n}\n\nself.onDataUpdated = function() {\n self.ctx.map.update();\n}\n\nself.onResize = function() {\n self.ctx.map.resize();\n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{\n \"schema\": {\n \"title\": \"Google Map Configuration\",\n \"type\": \"object\",\n \"properties\": {\n \"gmApiKey\": {\n \"title\": \"Google Maps API Key\",\n \"type\": \"string\"\n },\n \"gmDefaultMapType\": {\n \"title\": \"Default map type\",\n \"type\": \"string\",\n \"default\": \"roadmap\"\n },\n \"defaultZoomLevel\": {\n \"title\": \"Default map zoom level (1 - 20)\",\n \"type\": \"number\"\n },\n \"fitMapBounds\": {\n \"title\": \"Fit map bounds to cover all markers\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"markersSettings\": {\n \"title\": \"Markers\",\n \"type\": \"array\",\n \"items\": {\n \"title\": \"Marker settings\",\n \"type\": \"object\",\n \"properties\": {\n \"latKeyName\": {\n \"title\": \"Latitude key name\",\n \"type\": \"string\",\n \"default\": \"lat\"\n },\n \"lngKeyName\": {\n \"title\": \"Longitude key name\",\n \"type\": \"string\",\n \"default\": \"lng\"\n }, \n \"showLabel\": {\n \"title\": \"Show label\",\n \"type\": \"boolean\",\n \"default\": true\n }, \n \"label\": {\n \"title\": \"Label\",\n \"type\": \"string\"\n },\n \"tooltipPattern\": {\n \"title\": \"Pattern ( for ex. ''Text ${keyName} units.'' or ''${#<key index>} units'' )\",\n \"type\": \"string\",\n \"default\": \"<b>Latitude:</b> ${lat:7}<br/><b>Longitude:</b> ${lng:7}\"\n },\n \"color\": {\n \"title\": \"Color\",\n \"type\": \"string\"\n },\n \"useColorFunction\": {\n \"title\": \"Use color function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"colorFunction\": {\n \"title\": \"Color function: f(data)\",\n \"type\": \"string\"\n },\n \"markerImage\": {\n \"title\": \"Custom marker image\",\n \"type\": \"string\"\n },\n \"markerImageSize\": {\n \"title\": \"Custom marker image size (px)\",\n \"type\": \"number\",\n \"default\": 34\n },\n \"useMarkerImageFunction\": {\n \"title\": \"Use marker image function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"markerImageFunction\": {\n \"title\": \"Marker image function: f(data, images)\",\n \"type\": \"string\"\n },\n \"markerImages\": {\n \"title\": \"Marker images\",\n \"type\": \"array\",\n \"items\": {\n \"title\": \"Marker image\",\n \"type\": \"string\"\n }\n }\n }\n }\n }\n },\n \"required\": [\n \"gmApiKey\"\n ]\n },\n \"form\": [\n \"gmApiKey\",\n {\n \"key\": \"gmDefaultMapType\",\n \"type\": \"rc-select\",\n \"multiple\": false,\n \"items\": [\n {\n \"value\": \"roadmap\",\n \"label\": \"Roadmap\"\n },\n {\n \"value\": \"satellite\",\n \"label\": \"Satellite\"\n },\n {\n \"value\": \"hybrid\",\n \"label\": \"Hybrid\"\n },\n {\n \"value\": \"terrain\",\n \"label\": \"Terrain\"\n }\n ]\n },\n \"defaultZoomLevel\",\n \"fitMapBounds\",\n {\n \"key\": \"markersSettings\",\n \"items\": [\n \"markersSettings[].latKeyName\",\n \"markersSettings[].lngKeyName\",\n \"markersSettings[].showLabel\",\n \"markersSettings[].label\",\n \"markersSettings[].tooltipPattern\",\n {\n \"key\": \"markersSettings[].color\",\n \"type\": \"color\"\n },\n \"markersSettings[].useColorFunction\",\n {\n \"key\": \"markersSettings[].colorFunction\",\n \"type\": \"javascript\"\n },\n {\n \"key\": \"markersSettings[].markerImage\",\n \"type\": \"image\"\n },\n \"markersSettings[].markerImageSize\",\n \"markersSettings[].useMarkerImageFunction\",\n {\n \"key\": \"markersSettings[].markerImageFunction\",\n \"type\": \"javascript\"\n },\n {\n \"key\": \"markersSettings[].markerImages\",\n \"items\": [\n {\n \"key\": \"markersSettings[].markerImages[]\",\n \"type\": \"image\"\n }\n ]\n }\n ]\n }\n ]\n}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"latitude\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.05427416942713381,\"funcBody\":\"var value = prevValue || 15.833293;\\nif (time % 5000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"longitude\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.680594833308841,\"funcBody\":\"var value = prevValue || -90.454350;\\nif (time % 5000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"}]},{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"lat\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.05012157428742059,\"funcBody\":\"var value = prevValue || 14.450463;\\nif (time % 4000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"lng\",\"color\":\"#ffc107\",\"settings\":{},\"_hash\":0.6742359401617628,\"funcBody\":\"var value = prevValue || -84.845334;\\nif (time % 4000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"}]},{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"temperature\",\"color\":\"#607d8b\",\"settings\":{},\"_hash\":0.799863043034289,\"funcBody\":\"var value = prevValue + Math.random() * 40 - 20;\\nif (value < -60) {\\n\\tvalue = -60;\\n} else if (value > 60) {\\n\\tvalue = 60;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"gmApiKey\":\"AIzaSyDoEx2kaGz3PxwbI9T7ccTSg5xjdw8Nw8Q\",\"markersSettings\":[{\"label\":\"First point\",\"color\":\"#1e88e5\",\"latKeyName\":\"latitude\",\"lngKeyName\":\"longitude\",\"showLabel\":true,\"tooltipPattern\":\"<b>Latitude:</b> ${latitude:7}<br/><b>Longitude:</b> ${longitude:7}<br/><b>Temperature:</b> ${temperature} °C<br/><small>See advanced settings for details</small>\",\"useColorFunction\":true,\"colorFunction\":\"var temperature = data[''temperature''];\\nif (typeof temperature !== undefined) {\\n var percent = (temperature + 60)/120 * 100;\\n return tinycolor.mix(''blue'', ''red'', amount = percent).toHexString();\\n}\\nreturn ''blue'';\",\"markerImages\":[],\"useMarkerImageFunction\":false},{\"label\":\"Second point\",\"color\":\"#fdd835\",\"latKeyName\":\"lat\",\"lngKeyName\":\"lng\",\"showLabel\":true,\"tooltipPattern\":\"<b>Latitude:</b> ${lat:7}<br/><b>Longitude:</b> ${lng:7}<br/><b>Temperature:</b> ${temperature} °C<br/><small>See advanced settings for details</small>\",\"markerImageSize\":34,\"markerImages\":[\"\",\"\",\"\",\"\"],\"useMarkerImageFunction\":true,\"markerImageFunction\":\"var res = {\\n url: images[0],\\n size: 40\\n}\\nvar temperature = data[''temperature''];\\nif (typeof temperature !== undefined) {\\n var percent = (temperature + 60)/120;\\n var index = Math.floor(4 * percent);\\n res.url = images[index];\\n}\\nreturn res;\",\"useColorFunction\":false}],\"fitMapBounds\":true,\"gmDefaultMapType\":\"roadmap\"},\"title\":\"Google Maps\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"showLegend\":false}"}',
208   -'Google Maps' );
209   -
210   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
211   -VALUES ( now ( ), minTimeuuid ( 0 ), 'analogue_gauges',
212   -'temperature_radial_gauge_canvas_gauges',
213   -'{"type":"latest","sizeX":6,"sizeY":5,"resources":[],"templateHtml":"<canvas id=\"radialGauge\"></canvas>\n","templateCss":"","controllerScript":"self.onInit = function() {\n self.ctx.gauge = new TbAnalogueRadialGauge(self.ctx, ''radialGauge''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.gauge.update();\n}\n\nself.onResize = function() {\n self.ctx.gauge.resize();\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.gauge.mobileModeChanged();\n}\n\nself.getSettingsSchema = function() {\n return TbAnalogueRadialGauge.settingsSchema;\n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Temperature\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.7282710489093589,\"funcBody\":\"var value = prevValue + Math.random() * 40 - 20;\\nif (value < -60) {\\n\\tvalue = -60;\\n} else if (value > 60) {\\n\\tvalue = 60;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"rgb(255, 255, 255)\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"maxValue\":60,\"startAngle\":67.5,\"ticksAngle\":225,\"showBorder\":true,\"defaultColor\":\"#e65100\",\"needleCircleSize\":7,\"highlights\":[{\"from\":-60,\"to\":-50,\"color\":\"#42a5f5\"},{\"from\":-50,\"to\":-40,\"color\":\"rgba(66, 165, 245, 0.83)\"},{\"from\":-40,\"to\":-30,\"color\":\"rgba(66, 165, 245, 0.66)\"},{\"from\":-30,\"to\":-20,\"color\":\"rgba(66, 165, 245, 0.5)\"},{\"from\":-20,\"to\":-10,\"color\":\"rgba(66, 165, 245, 0.33)\"},{\"from\":-10,\"to\":0,\"color\":\"rgba(66, 165, 245, 0.16)\"},{\"from\":0,\"to\":10,\"color\":\"rgba(229, 115, 115, 0.16)\"},{\"from\":10,\"to\":20,\"color\":\"rgba(229, 115, 115, 0.33)\"},{\"from\":20,\"to\":30,\"color\":\"rgba(229, 115, 115, 0.5)\"},{\"from\":30,\"to\":40,\"color\":\"rgba(229, 115, 115, 0.66)\"},{\"from\":40,\"to\":50,\"color\":\"rgba(229, 115, 115, 0.83)\"},{\"from\":50,\"to\":60,\"color\":\"#e57373\"}],\"showUnitTitle\":true,\"colorPlate\":\"#cfd8dc\",\"colorMajorTicks\":\"#444\",\"colorMinorTicks\":\"#666\",\"minorTicks\":2,\"valueInt\":3,\"valueDec\":1,\"highlightsWidth\":15,\"valueBox\":true,\"animation\":true,\"animationDuration\":1000,\"animationRule\":\"bounce\",\"colorNeedleShadowUp\":\"rgba(2, 255, 255, 0)\",\"colorNeedleShadowDown\":\"rgba(188, 143, 143, 0.78)\",\"units\":\"°C\",\"majorTicksCount\":12,\"numbersFont\":{\"family\":\"Roboto\",\"size\":20,\"style\":\"normal\",\"weight\":\"normal\",\"color\":\"#263238\"},\"titleFont\":{\"family\":\"Roboto\",\"size\":24,\"style\":\"normal\",\"weight\":\"normal\",\"color\":\"#263238\"},\"unitsFont\":{\"family\":\"Roboto\",\"size\":28,\"style\":\"normal\",\"weight\":\"500\",\"color\":\"#616161\"},\"valueFont\":{\"family\":\"Segment7Standard\",\"size\":30,\"style\":\"normal\",\"weight\":\"normal\",\"shadowColor\":\"rgba(0, 0, 0, 0.49)\",\"color\":\"#444\"},\"colorValueBoxRect\":\"#888\",\"colorValueBoxRectEnd\":\"#666\",\"colorValueBoxBackground\":\"#babab2\",\"colorValueBoxShadow\":\"rgba(0,0,0,1)\",\"unitTitle\":\"Temperature\",\"minValue\":-60},\"title\":\"Temperature radial gauge - Canvas Gauges\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',
214   -'Temperature radial gauge - Canvas Gauges' );
215   -
216   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
217   -VALUES ( now ( ), minTimeuuid ( 0 ), 'cards', 'attributes_card',
218   -'{"type":"latest","sizeX":7.5,"sizeY":3,"resources":[],"templateHtml":"","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}","controllerScript":"self.onInit = function() {\n \n self.ctx.datasourceTitleCells = [];\n self.ctx.valueCells = [];\n self.ctx.labelCells = [];\n \n for (var i=0; i < self.ctx.datasources.length; i++) {\n var tbDatasource = self.ctx.datasources[i];\n\n var datasourceId = ''tbDatasource'' + i;\n self.ctx.$container.append(\n \"<div id=''\" + datasourceId +\n \"'' class=''tbDatasource-container''></div>\"\n );\n\n var datasourceContainer = $(''#'' + datasourceId,\n self.ctx.$container);\n\n datasourceContainer.append(\n \"<div class=''tbDatasource-title''>\" +\n tbDatasource.name + \"</div>\"\n );\n \n var datasourceTitleCell = $(''.tbDatasource-title'', datasourceContainer);\n self.ctx.datasourceTitleCells.push(datasourceTitleCell);\n \n var tableId = ''table'' + i;\n datasourceContainer.append(\n \"<table id=''\" + tableId +\n \"'' class=''tbDatasource-table''><col width=''30%''><col width=''70%''></table>\"\n );\n var table = $(''#'' + tableId, self.ctx.$container);\n\n for (var a = 0; a < tbDatasource.dataKeys.length; a++) {\n var dataKey = tbDatasource.dataKeys[a];\n var labelCellId = ''labelCell'' + a;\n var cellId = ''cell'' + a;\n table.append(\"<tr><td id=''\" + labelCellId + \"''>\" + dataKey.label +\n \"</td><td id=''\" + cellId +\n \"''></td></tr>\");\n var labelCell = $(''#'' + labelCellId, table);\n self.ctx.labelCells.push(labelCell);\n var valueCell = $(''#'' + cellId, table);\n self.ctx.valueCells.push(valueCell);\n }\n } \n \n self.onResize();\n}\n\nself.onDataUpdated = function() {\n for (var i = 0; i < self.ctx.valueCells.length; i++) {\n var cellData = self.ctx.data[i];\n if (cellData && cellData.data && cellData.data.length > 0) {\n var tvPair = cellData.data[cellData.data.length -\n 1];\n var value = tvPair[1];\n self.ctx.valueCells[i].html(value);\n }\n } \n}\n\nself.onResize = function() {\n var datasoirceTitleFontSize = self.ctx.height/8;\n if (self.ctx.width/self.ctx.height <= 1.5) {\n datasoirceTitleFontSize = self.ctx.width/12;\n }\n datasoirceTitleFontSize = Math.min(datasoirceTitleFontSize, 20);\n for (var i = 0; i < self.ctx.datasourceTitleCells.length; i++) {\n self.ctx.datasourceTitleCells[i].css(''font-size'', datasoirceTitleFontSize+''px'');\n }\n var valueFontSize = self.ctx.height/9;\n var labelFontSize = self.ctx.height/9;\n if (self.ctx.width/self.ctx.height <= 1.5) {\n valueFontSize = self.ctx.width/15;\n labelFontSize = self.ctx.width/15;\n }\n valueFontSize = Math.min(valueFontSize, 18);\n labelFontSize = Math.min(labelFontSize, 18);\n\n for (i = 0; i < self.ctx.valueCells; i++) {\n self.ctx.valueCells[i].css(''font-size'', valueFontSize+''px'');\n self.ctx.valueCells[i].css(''height'', valueFontSize*2.5+''px'');\n self.ctx.valueCells[i].css(''padding'', ''0px '' + valueFontSize + ''px'');\n self.ctx.labelCells[i].css(''font-size'', labelFontSize+''px'');\n self.ctx.labelCells[i].css(''height'', labelFontSize*2.5+''px'');\n self.ctx.labelCells[i].css(''padding'', ''0px '' + labelFontSize + ''px'');\n } \n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{}","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\":\"Attributes card\"}"}',
219   -'Attributes card' );
220   -
221   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
222   -VALUES ( now ( ), minTimeuuid ( 0 ), 'digital_gauges',
223   -'digital_thermometer',
224   -'{"type":"latest","sizeX":3,"sizeY":3,"resources":[],"templateHtml":"<canvas id=\"digitalGauge\"></canvas>","templateCss":"#gauge {\n text-align: center;\n /* margin-left: -100px;\n margin-right: -100px;*/\n /*margin-top: -50px;*/\n \n}\n","controllerScript":"self.onInit = function() {\n self.ctx.gauge = new TbCanvasDigitalGauge(self.ctx, ''digitalGauge''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.gauge.update();\n}\n\nself.onResize = function() {\n self.ctx.gauge.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbCanvasDigitalGauge.settingsSchema;\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.gauge.mobileModeChanged();\n}\n\nself.onDestroy = function() {\n}\n\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Temp\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.7282710489093589,\"funcBody\":\"var value = prevValue + Math.random() * 20 - 10;\\nif (value < -60) {\\n\\tvalue = 60;\\n} else if (value > 60) {\\n\\tvalue = 60;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"#000000\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"0px\",\"settings\":{\"maxValue\":60,\"donutStartAngle\":90,\"showValue\":true,\"showMinMax\":true,\"gaugeWidthScale\":1,\"levelColors\":[\"#304ffe\",\"#7e57c2\",\"#ff4081\",\"#d32f2f\"],\"refreshAnimationType\":\"<>\",\"refreshAnimationTime\":700,\"startAnimationType\":\"<>\",\"startAnimationTime\":700,\"titleFont\":{\"family\":\"Roboto\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\"},\"labelFont\":{\"family\":\"Roboto\",\"size\":8,\"style\":\"normal\",\"weight\":\"500\"},\"valueFont\":{\"family\":\"Segment7Standard\",\"style\":\"normal\",\"weight\":\"500\",\"size\":18},\"minMaxFont\":{\"family\":\"Segment7Standard\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\"},\"dashThickness\":1.5,\"decimals\":0,\"minValue\":-60,\"units\":\"°C\",\"gaugeColor\":\"#333333\",\"neonGlowBrightness\":35,\"gaugeType\":\"donut\"},\"title\":\"Digital thermometer\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',
225   -'Digital thermometer' );
226   -
227   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
228   -VALUES ( now ( ), minTimeuuid ( 0 ), 'gpio_widgets',
229   -'raspberry_pi_gpio_control',
230   -'{"type":"rpc","sizeX":6,"sizeY":10.5,"resources":[],"templateHtml":"<fieldset class=\"gpio-panel\" ng-disabled=\"!rpcEnabled || executingRpcRequest\" style=\"height: 100%;\">\n <section class=\"gpio-row\" layout=\"row\" ng-repeat=\"row in rows track by $index\" \n ng-style=\"{''height'': prefferedRowHeight+''px''}\">\n <section flex layout=\"row\" ng-repeat=\"cell in row track by $index\">\n <section layout=\"row\" flex ng-if=\"cell\" layout-align=\"{{$index===0 ? ''end center'' : ''start center''}}\">\n <span class=\"gpio-left-label\" ng-show=\"$index===0\">{{ cell.label }}</span>\n <section layout=\"row\" class=\"switch-panel\" layout-align=\"start center\" ng-class=\"$index===0 ? ''col-0'' : ''col-1''\"\n ng-style=\"{''height'': prefferedRowHeight+''px'', ''backgroundColor'': ''{{ switchPanelBackgroundColor }}''}\">\n <span class=\"pin\" ng-show=\"$index===0\">{{cell.pin}}</span>\n <span flex ng-show=\"$index===1\"></span>\n <md-switch\n aria-label=\"{{ cell.label }}\"\n ng-disabled=\"!rpcEnabled || executingRpcRequest\"\n ng-model=\"cell.enabled\" \n ng-change=\"cell.enabled = !cell.enabled\" \n ng-click=\"gpioClick($event, cell)\">\n </md-switch>\n <span flex ng-show=\"$index===0\"></span>\n <span class=\"pin\" ng-show=\"$index===1\">{{cell.pin}}</span>\n </section>\n <span class=\"gpio-right-label\" ng-show=\"$index===1\">{{ cell.label }}</span>\n </section>\n <section layout=\"row\" flex ng-if=\"!cell\">\n <span flex ng-show=\"$index===0\"></span>\n <span class=\"switch-panel\"\n ng-style=\"{''height'': prefferedRowHeight+''px'', ''backgroundColor'': ''{{ switchPanelBackgroundColor }}''}\"></span>\n <span flex ng-show=\"$index===1\"></span>\n </section>\n </section>\n </section> \n <span class=\"error\" style=\"position: absolute; bottom: 5px;\" ng-show=\"rpcErrorText\">{{rpcErrorText}}</span>\n <md-progress-linear ng-show=\"executingRpcRequest\" style=\"position: absolute; bottom: 0;\" md-mode=\"indeterminate\"></md-progress-linear> \n</fieldset>","templateCss":".error {\n font-size: 14px !important;\n color: maroon;/*rgb(250,250,250);*/\n background-color: transparent;\n padding: 6px;\n}\n\n.error span {\n margin: auto;\n}\n\n.gpio-panel {\n padding-top: 10px;\n white-space: nowrap;\n}\n\n.switch-panel {\n margin: 0;\n height: 32px;\n width: 66px;\n min-width: 66px;\n}\n\n.switch-panel md-switch {\n margin: 0;\n width: 36px;\n min-width: 36px;\n}\n\n.switch-panel md-switch > div.md-container {\n margin: 0;\n}\n\n.switch-panel.col-0 md-switch {\n margin-left: 8px;\n margin-right: 4px;\n}\n\n.switch-panel.col-1 md-switch {\n margin-left: 4px;\n margin-right: 8px;\n}\n\n.gpio-row {\n height: 32px;\n}\n\n.pin {\n margin-top: auto;\n margin-bottom: auto;\n color: white;\n font-size: 12px;\n width: 16px;\n min-width: 16px;\n}\n\n.switch-panel.col-0 .pin {\n margin-left: auto;\n padding-left: 2px;\n text-align: right;\n}\n\n.switch-panel.col-1 .pin {\n margin-right: auto;\n \n text-align: left;\n}\n\n.gpio-left-label {\n margin-right: 8px;\n}\n\n.gpio-right-label {\n margin-left: 8px;\n}","controllerScript":"self.onInit = function() {\n \n var i, gpio;\n var scope = self.ctx.$scope;\n var settings = self.ctx.settings;\n scope.gpioList = [];\n for (var g = 0; g < settings.gpioList.length; g++) {\n gpio = settings.gpioList[g];\n scope.gpioList.push(\n {\n row: gpio.row,\n col: gpio.col,\n pin: gpio.pin,\n label: gpio.label,\n enabled: false\n }\n );\n }\n\n scope.requestTimeout = settings.requestTimeout || 1000;\n\n scope.switchPanelBackgroundColor = settings.switchPanelBackgroundColor || tinycolor(''green'').lighten(2).toRgbString();\n\n scope.gpioStatusRequest = {\n method: \"getGpioStatus\",\n paramsBody: \"{}\"\n };\n \n if (settings.gpioStatusRequest) {\n scope.gpioStatusRequest.method = settings.gpioStatusRequest.method || scope.gpioStatusRequest.method;\n scope.gpioStatusRequest.paramsBody = settings.gpioStatusRequest.paramsBody || scope.gpioStatusRequest.paramsBody;\n }\n \n scope.gpioStatusChangeRequest = {\n method: \"setGpioStatus\",\n paramsBody: \"{\\n \\\"pin\\\": \\\"{$pin}\\\",\\n \\\"enabled\\\": \\\"{$enabled}\\\"\\n}\"\n };\n \n if (settings.gpioStatusChangeRequest) {\n scope.gpioStatusChangeRequest.method = settings.gpioStatusChangeRequest.method || scope.gpioStatusChangeRequest.method;\n scope.gpioStatusChangeRequest.paramsBody = settings.gpioStatusChangeRequest.paramsBody || scope.gpioStatusChangeRequest.paramsBody;\n }\n \n scope.parseGpioStatusFunction = \"return body[pin] === true;\";\n \n if (settings.parseGpioStatusFunction && settings.parseGpioStatusFunction.length > 0) {\n scope.parseGpioStatusFunction = settings.parseGpioStatusFunction;\n }\n \n scope.parseGpioStatusFunction = new Function(\"body, pin\", scope.parseGpioStatusFunction);\n \n function requestGpioStatus() {\n self.ctx.controlApi.sendTwoWayCommand(scope.gpioStatusRequest.method, \n scope.gpioStatusRequest.paramsBody, \n scope.requestTimeout)\n .then(\n function success(responseBody) {\n for (var g = 0; g < scope.gpioList.length; g++) {\n var gpio = scope.gpioList[g];\n var enabled = scope.parseGpioStatusFunction.apply(this, [responseBody, gpio.pin]);\n gpio.enabled = enabled; \n }\n }\n );\n }\n \n function changeGpioStatus(gpio) {\n var pin = gpio.pin + '''';\n var enabled = !gpio.enabled;\n enabled = enabled === true ? ''true'' : ''false'';\n var paramsBody = scope.gpioStatusChangeRequest.paramsBody;\n var requestBody = JSON.parse(paramsBody.replace(\"\\\"{$pin}\\\"\", pin).replace(\"\\\"{$enabled}\\\"\", enabled));\n self.ctx.controlApi.sendTwoWayCommand(scope.gpioStatusChangeRequest.method, \n requestBody, scope.requestTimeout)\n .then(\n function success(responseBody) {\n var enabled = scope.parseGpioStatusFunction.apply(this, [responseBody, gpio.pin]);\n gpio.enabled = enabled;\n }\n );\n }\n \n scope.gpioCells = {};\n var rowCount = 0;\n for (i = 0; i < scope.gpioList.length; i++) {\n gpio = scope.gpioList[i];\n scope.gpioCells[gpio.row+''_''+gpio.col] = gpio;\n rowCount = Math.max(rowCount, gpio.row+1);\n }\n \n scope.prefferedRowHeight = 32;\n scope.rows = [];\n for (i = 0; i < rowCount; i++) {\n var row = [];\n for (var c =0; c<2;c++) {\n if (scope.gpioCells[i+''_''+c]) {\n row[c] = scope.gpioCells[i+''_''+c];\n } else {\n row[c] = null;\n }\n }\n scope.rows.push(row);\n }\n\n scope.gpioClick = function($event, gpio) {\n changeGpioStatus(gpio);\n };\n\n requestGpioStatus(); \n \n self.onResize();\n}\n\nself.onResize = function() {\n var scope = self.ctx.$scope;\n var rowCount = scope.rows.length;\n var prefferedRowHeight = (self.ctx.height - 35)/rowCount;\n prefferedRowHeight = Math.min(32, prefferedRowHeight);\n prefferedRowHeight = Math.max(12, prefferedRowHeight);\n scope.prefferedRowHeight = prefferedRowHeight;\n var ratio = prefferedRowHeight/32;\n var switches = $(''md-switch'', self.ctx.$container);\n switches.css(''height'', 30*ratio+''px'');\n switches.css(''width'', 36*ratio+''px'');\n switches.css(''min-width'', 36*ratio+''px'');\n $(''.md-container'', switches).css(''height'', 24*ratio+''px'');\n $(''.md-container'', switches).css(''width'', 36*ratio+''px'');\n var bars = $(''.md-bar'', self.ctx.$container);\n bars.css(''height'', 14*ratio+''px'');\n bars.css(''width'', 34*ratio+''px'');\n var thumbs = $(''.md-thumb'', self.ctx.$container);\n thumbs.css(''height'', 20*ratio+''px'');\n thumbs.css(''width'', 20*ratio+''px'');\n \n var leftLabels = $(''.gpio-left-label'', self.ctx.$container);\n leftLabels.css(''font-size'', 16*ratio+''px'');\n var rightLabels = $(''.gpio-right-label'', self.ctx.$container);\n rightLabels.css(''font-size'', 16*ratio+''px'');\n var pins = $(''.pin'', self.ctx.$container);\n var pinsFontSize = Math.max(9, 12*ratio);\n pins.css(''font-size'', pinsFontSize+''px''); \n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"Settings\",\n \"properties\": {\n \"gpioList\": {\n \"title\": \"Gpio switches\",\n \"type\": \"array\",\n \"minItems\" : 1,\n \"items\": {\n \"title\": \"Gpio switch\",\n \"type\": \"object\",\n \"properties\": {\n \"pin\": {\n \"title\": \"Pin\",\n \"type\": \"number\"\n },\n \"label\": {\n \"title\": \"Label\",\n \"type\": \"string\"\n },\n \"row\": {\n \"title\": \"Row\",\n \"type\": \"number\"\n },\n \"col\": {\n \"title\": \"Column\",\n \"type\": \"number\"\n }\n },\n \"required\": [\"pin\", \"label\", \"row\", \"col\"]\n }\n },\n \"requestTimeout\": {\n \"title\": \"RPC request timeout\",\n \"type\": \"number\",\n \"default\": 500\n },\n \"switchPanelBackgroundColor\": {\n \"title\": \"Switches panel background color\",\n \"type\": \"string\",\n \"default\": \"#008a00\"\n },\n \"gpioStatusRequest\": {\n \"title\": \"GPIO status request\",\n \"type\": \"object\",\n \"properties\": {\n \"method\": {\n \"title\": \"Method name\",\n \"type\": \"string\",\n \"default\": \"getGpioStatus\"\n },\n \"paramsBody\": {\n \"title\": \"Method body\",\n \"type\": \"string\",\n \"default\": \"{}\"\n }\n },\n \"required\": [\"method\", \"paramsBody\"]\n },\n \"gpioStatusChangeRequest\": {\n \"title\": \"GPIO status change request\",\n \"type\": \"object\",\n \"properties\": {\n \"method\": {\n \"title\": \"Method name\",\n \"type\": \"string\",\n \"default\": \"setGpioStatus\"\n },\n \"paramsBody\": {\n \"title\": \"Method body\",\n \"type\": \"string\",\n \"default\": \"{\\n \\\"pin\\\": \\\"{$pin}\\\",\\n \\\"enabled\\\": \\\"{$enabled}\\\"\\n}\"\n }\n },\n \"required\": [\"method\", \"paramsBody\"]\n },\n \"parseGpioStatusFunction\": {\n \"title\": \"Parse gpio status function\",\n \"type\": \"string\",\n \"default\": \"return body[pin] === true;\"\n } \n },\n \"required\": [\"gpioList\", \n \"requestTimeout\",\n \"switchPanelBackgroundColor\",\n \"gpioStatusRequest\",\n \"gpioStatusChangeRequest\",\n \"parseGpioStatusFunction\"]\n },\n \"form\": [\n \"gpioList\",\n \"requestTimeout\",\n {\n \"key\": \"switchPanelBackgroundColor\",\n \"type\": \"color\"\n },\n {\n \"key\": \"gpioStatusRequest\",\n \"items\": [\n \"gpioStatusRequest.method\",\n {\n \"key\": \"gpioStatusRequest.paramsBody\",\n \"type\": \"json\"\n }\n ]\n },\n {\n \"key\": \"gpioStatusChangeRequest\",\n \"items\": [\n \"gpioStatusChangeRequest.method\",\n {\n \"key\": \"gpioStatusChangeRequest.paramsBody\",\n \"type\": \"json\"\n }\n ]\n },\n {\n \"key\": \"parseGpioStatusFunction\",\n \"type\": \"javascript\"\n }\n ]\n}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"targetDeviceAliases\":[],\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"0px\",\"settings\":{\"parseGpioStatusFunction\":\"return body[pin] === true;\",\"gpioStatusChangeRequest\":{\"method\":\"setGpioStatus\",\"paramsBody\":\"{\\n \\\"pin\\\": \\\"{$pin}\\\",\\n \\\"enabled\\\": \\\"{$enabled}\\\"\\n}\"},\"requestTimeout\":500,\"switchPanelBackgroundColor\":\"#008a00\",\"gpioStatusRequest\":{\"method\":\"getGpioStatus\",\"paramsBody\":\"{}\"},\"gpioList\":[{\"pin\":7,\"label\":\"GPIO 4 (GPCLK0)\",\"row\":3,\"col\":0,\"_uniqueKey\":0},{\"pin\":11,\"label\":\"GPIO 17\",\"row\":5,\"col\":0,\"_uniqueKey\":1},{\"pin\":12,\"label\":\"GPIO 18\",\"row\":5,\"col\":1,\"_uniqueKey\":2},{\"_uniqueKey\":3,\"pin\":13,\"label\":\"GPIO 27\",\"row\":6,\"col\":0},{\"_uniqueKey\":4,\"pin\":15,\"label\":\"GPIO 22\",\"row\":7,\"col\":0},{\"_uniqueKey\":5,\"pin\":16,\"label\":\"GPIO 23\",\"row\":7,\"col\":1},{\"_uniqueKey\":6,\"pin\":18,\"label\":\"GPIO 24\",\"row\":8,\"col\":1},{\"_uniqueKey\":7,\"pin\":22,\"label\":\"GPIO 25\",\"row\":10,\"col\":1},{\"_uniqueKey\":8,\"pin\":29,\"label\":\"GPIO 5\",\"row\":14,\"col\":0},{\"_uniqueKey\":9,\"pin\":31,\"label\":\"GPIO 6\",\"row\":15,\"col\":0},{\"_uniqueKey\":10,\"pin\":32,\"label\":\"GPIO 12\",\"row\":15,\"col\":1},{\"_uniqueKey\":11,\"pin\":33,\"label\":\"GPIO 13\",\"row\":16,\"col\":0},{\"_uniqueKey\":12,\"pin\":35,\"label\":\"GPIO 19\",\"row\":17,\"col\":0},{\"_uniqueKey\":13,\"pin\":36,\"label\":\"GPIO 16\",\"row\":17,\"col\":1},{\"_uniqueKey\":14,\"pin\":37,\"label\":\"GPIO 26\",\"row\":18,\"col\":0},{\"_uniqueKey\":15,\"pin\":38,\"label\":\"GPIO 20\",\"row\":18,\"col\":1},{\"_uniqueKey\":16,\"pin\":40,\"label\":\"GPIO 21\",\"row\":19,\"col\":1}]},\"title\":\"Raspberry Pi GPIO Control\"}"}',
231   -'Raspberry Pi GPIO Control' );
232   -
233   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
234   -VALUES ( now ( ), minTimeuuid ( 0 ), 'gpio_widgets',
235   -'basic_gpio_control',
236   -'{"type":"rpc","sizeX":4,"sizeY":2,"resources":[],"templateHtml":"<fieldset class=\"gpio-panel\" ng-disabled=\"!rpcEnabled || executingRpcRequest\" style=\"height: 100%;\">\n <section class=\"gpio-row\" layout=\"row\" ng-repeat=\"row in rows track by $index\" \n ng-style=\"{''height'': prefferedRowHeight+''px''}\">\n <section flex layout=\"row\" ng-repeat=\"cell in row track by $index\">\n <section layout=\"row\" flex ng-if=\"cell\" layout-align=\"{{$index===0 ? ''end center'' : ''start center''}}\">\n <span class=\"gpio-left-label\" ng-show=\"$index===0\">{{ cell.label }}</span>\n <section layout=\"row\" class=\"switch-panel\" layout-align=\"start center\" ng-class=\"$index===0 ? ''col-0'' : ''col-1''\"\n ng-style=\"{''height'': prefferedRowHeight+''px'', ''backgroundColor'': ''{{ switchPanelBackgroundColor }}''}\">\n <span class=\"pin\" ng-show=\"$index===0\">{{cell.pin}}</span>\n <span flex ng-show=\"$index===1\"></span>\n <md-switch\n aria-label=\"{{ cell.label }}\"\n ng-disabled=\"!rpcEnabled || executingRpcRequest\"\n ng-model=\"cell.enabled\" \n ng-change=\"cell.enabled = !cell.enabled\" \n ng-click=\"gpioClick($event, cell)\">\n </md-switch>\n <span flex ng-show=\"$index===0\"></span>\n <span class=\"pin\" ng-show=\"$index===1\">{{cell.pin}}</span>\n </section>\n <span class=\"gpio-right-label\" ng-show=\"$index===1\">{{ cell.label }}</span>\n </section>\n <section layout=\"row\" flex ng-if=\"!cell\">\n <span flex ng-show=\"$index===0\"></span>\n <span class=\"switch-panel\"\n ng-style=\"{''height'': prefferedRowHeight+''px'', ''backgroundColor'': ''{{ switchPanelBackgroundColor }}''}\"></span>\n <span flex ng-show=\"$index===1\"></span>\n </section>\n </section>\n </section> \n <span class=\"error\" style=\"position: absolute; bottom: 5px;\" ng-show=\"rpcErrorText\">{{rpcErrorText}}</span>\n <md-progress-linear ng-show=\"executingRpcRequest\" style=\"position: absolute; bottom: 0;\" md-mode=\"indeterminate\"></md-progress-linear> \n</fieldset>","templateCss":".error {\n font-size: 14px !important;\n color: maroon;/*rgb(250,250,250);*/\n background-color: transparent;\n padding: 6px;\n}\n\n.error span {\n margin: auto;\n}\n\n.gpio-panel {\n padding-top: 10px;\n white-space: nowrap;\n}\n\n.switch-panel {\n margin: 0;\n height: 32px;\n width: 66px;\n min-width: 66px;\n}\n\n.switch-panel md-switch {\n margin: 0;\n width: 36px;\n min-width: 36px;\n}\n\n.switch-panel md-switch > div.md-container {\n margin: 0;\n}\n\n.switch-panel.col-0 md-switch {\n padding-left: 8px;\n padding-right: 4px;\n}\n\n.switch-panel.col-1 md-switch {\n padding-left: 4px;\n padding-right: 8px;\n}\n\n.gpio-row {\n height: 32px;\n}\n\n.pin {\n margin-top: auto;\n margin-bottom: auto;\n color: white;\n font-size: 12px;\n width: 16px;\n min-width: 16px;\n}\n\n.switch-panel.col-0 .pin {\n margin-left: auto;\n padding-left: 2px;\n text-align: right;\n}\n\n.switch-panel.col-1 .pin {\n margin-right: auto;\n \n text-align: left;\n}\n\n.gpio-left-label {\n margin-right: 8px;\n}\n\n.gpio-right-label {\n margin-left: 8px;\n}","controllerScript":"self.onInit = function() {\n \n var i, gpio;\n var scope = self.ctx.$scope;\n var settings = self.ctx.settings;\n scope.gpioList = [];\n for (var g = 0; g < settings.gpioList.length; g++) {\n gpio = settings.gpioList[g];\n scope.gpioList.push(\n {\n row: gpio.row,\n col: gpio.col,\n pin: gpio.pin,\n label: gpio.label,\n enabled: false\n }\n );\n }\n\n scope.requestTimeout = settings.requestTimeout || 1000;\n\n scope.switchPanelBackgroundColor = settings.switchPanelBackgroundColor || tinycolor(''green'').lighten(2).toRgbString();\n\n scope.gpioStatusRequest = {\n method: \"getGpioStatus\",\n paramsBody: \"{}\"\n };\n \n if (settings.gpioStatusRequest) {\n scope.gpioStatusRequest.method = settings.gpioStatusRequest.method || scope.gpioStatusRequest.method;\n scope.gpioStatusRequest.paramsBody = settings.gpioStatusRequest.paramsBody || scope.gpioStatusRequest.paramsBody;\n }\n \n scope.gpioStatusChangeRequest = {\n method: \"setGpioStatus\",\n paramsBody: \"{\\n \\\"pin\\\": \\\"{$pin}\\\",\\n \\\"enabled\\\": \\\"{$enabled}\\\"\\n}\"\n };\n \n if (settings.gpioStatusChangeRequest) {\n scope.gpioStatusChangeRequest.method = settings.gpioStatusChangeRequest.method || scope.gpioStatusChangeRequest.method;\n scope.gpioStatusChangeRequest.paramsBody = settings.gpioStatusChangeRequest.paramsBody || scope.gpioStatusChangeRequest.paramsBody;\n }\n \n scope.parseGpioStatusFunction = \"return body[pin] === true;\";\n \n if (settings.parseGpioStatusFunction && settings.parseGpioStatusFunction.length > 0) {\n scope.parseGpioStatusFunction = settings.parseGpioStatusFunction;\n }\n \n scope.parseGpioStatusFunction = new Function(\"body, pin\", scope.parseGpioStatusFunction);\n \n function requestGpioStatus() {\n self.ctx.controlApi.sendTwoWayCommand(scope.gpioStatusRequest.method, \n scope.gpioStatusRequest.paramsBody, \n scope.requestTimeout)\n .then(\n function success(responseBody) {\n for (var g = 0; g < scope.gpioList.length; g++) {\n var gpio = scope.gpioList[g];\n var enabled = scope.parseGpioStatusFunction.apply(this, [responseBody, gpio.pin]);\n gpio.enabled = enabled; \n }\n }\n );\n }\n \n function changeGpioStatus(gpio) {\n var pin = gpio.pin + '''';\n var enabled = !gpio.enabled;\n enabled = enabled === true ? ''true'' : ''false'';\n var paramsBody = scope.gpioStatusChangeRequest.paramsBody;\n var requestBody = JSON.parse(paramsBody.replace(\"\\\"{$pin}\\\"\", pin).replace(\"\\\"{$enabled}\\\"\", enabled));\n self.ctx.controlApi.sendTwoWayCommand(scope.gpioStatusChangeRequest.method, \n requestBody, scope.requestTimeout)\n .then(\n function success(responseBody) {\n var enabled = scope.parseGpioStatusFunction.apply(this, [responseBody, gpio.pin]);\n gpio.enabled = enabled;\n }\n );\n }\n \n scope.gpioCells = {};\n var rowCount = 0;\n for (i = 0; i < scope.gpioList.length; i++) {\n gpio = scope.gpioList[i];\n scope.gpioCells[gpio.row+''_''+gpio.col] = gpio;\n rowCount = Math.max(rowCount, gpio.row+1);\n }\n \n scope.prefferedRowHeight = 32;\n scope.rows = [];\n for (i = 0; i < rowCount; i++) {\n var row = [];\n for (var c =0; c<2;c++) {\n if (scope.gpioCells[i+''_''+c]) {\n row[c] = scope.gpioCells[i+''_''+c];\n } else {\n row[c] = null;\n }\n }\n scope.rows.push(row);\n }\n\n scope.gpioClick = function($event, gpio) {\n changeGpioStatus(gpio);\n };\n\n requestGpioStatus(); \n \n self.onResize();\n}\n\nself.onResize = function() {\n var scope = self.ctx.$scope;\n var rowCount = scope.rows.length;\n var prefferedRowHeight = (self.ctx.height - 35)/rowCount;\n prefferedRowHeight = Math.min(32, prefferedRowHeight);\n prefferedRowHeight = Math.max(12, prefferedRowHeight);\n scope.prefferedRowHeight = prefferedRowHeight;\n var ratio = prefferedRowHeight/32;\n var switches = $(''md-switch'', self.ctx.$container);\n switches.css(''height'', 30*ratio+''px'');\n switches.css(''width'', 36*ratio+''px'');\n switches.css(''min-width'', 36*ratio+''px'');\n $(''.md-container'', switches).css(''height'', 24*ratio+''px'');\n $(''.md-container'', switches).css(''width'', 36*ratio+''px'');\n var bars = $(''.md-bar'', self.ctx.$container);\n bars.css(''height'', 14*ratio+''px'');\n bars.css(''width'', 34*ratio+''px'');\n var thumbs = $(''.md-thumb'', self.ctx.$container);\n thumbs.css(''height'', 20*ratio+''px'');\n thumbs.css(''width'', 20*ratio+''px'');\n \n var leftLabels = $(''.gpio-left-label'', self.ctx.$container);\n leftLabels.css(''font-size'', 16*ratio+''px'');\n var rightLabels = $(''.gpio-right-label'', self.ctx.$container);\n rightLabels.css(''font-size'', 16*ratio+''px'');\n var pins = $(''.pin'', self.ctx.$container);\n var pinsFontSize = Math.max(9, 12*ratio);\n pins.css(''font-size'', pinsFontSize+''px''); \n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"Settings\",\n \"properties\": {\n \"gpioList\": {\n \"title\": \"Gpio switches\",\n \"type\": \"array\",\n \"minItems\" : 1,\n \"items\": {\n \"title\": \"Gpio switch\",\n \"type\": \"object\",\n \"properties\": {\n \"pin\": {\n \"title\": \"Pin\",\n \"type\": \"number\"\n },\n \"label\": {\n \"title\": \"Label\",\n \"type\": \"string\"\n },\n \"row\": {\n \"title\": \"Row\",\n \"type\": \"number\"\n },\n \"col\": {\n \"title\": \"Column\",\n \"type\": \"number\"\n }\n },\n \"required\": [\"pin\", \"label\", \"row\", \"col\"]\n }\n },\n \"requestTimeout\": {\n \"title\": \"RPC request timeout\",\n \"type\": \"number\",\n \"default\": 500\n },\n \"switchPanelBackgroundColor\": {\n \"title\": \"Switches panel background color\",\n \"type\": \"string\",\n \"default\": \"#008a00\"\n },\n \"gpioStatusRequest\": {\n \"title\": \"GPIO status request\",\n \"type\": \"object\",\n \"properties\": {\n \"method\": {\n \"title\": \"Method name\",\n \"type\": \"string\",\n \"default\": \"getGpioStatus\"\n },\n \"paramsBody\": {\n \"title\": \"Method body\",\n \"type\": \"string\",\n \"default\": \"{}\"\n }\n },\n \"required\": [\"method\", \"paramsBody\"]\n },\n \"gpioStatusChangeRequest\": {\n \"title\": \"GPIO status change request\",\n \"type\": \"object\",\n \"properties\": {\n \"method\": {\n \"title\": \"Method name\",\n \"type\": \"string\",\n \"default\": \"setGpioStatus\"\n },\n \"paramsBody\": {\n \"title\": \"Method body\",\n \"type\": \"string\",\n \"default\": \"{\\n \\\"pin\\\": \\\"{$pin}\\\",\\n \\\"enabled\\\": \\\"{$enabled}\\\"\\n}\"\n }\n },\n \"required\": [\"method\", \"paramsBody\"]\n },\n \"parseGpioStatusFunction\": {\n \"title\": \"Parse gpio status function\",\n \"type\": \"string\",\n \"default\": \"return body[pin] === true;\"\n } \n },\n \"required\": [\"gpioList\", \n \"requestTimeout\",\n \"switchPanelBackgroundColor\",\n \"gpioStatusRequest\",\n \"gpioStatusChangeRequest\",\n \"parseGpioStatusFunction\"]\n },\n \"form\": [\n \"gpioList\",\n \"requestTimeout\",\n {\n \"key\": \"switchPanelBackgroundColor\",\n \"type\": \"color\"\n },\n {\n \"key\": \"gpioStatusRequest\",\n \"items\": [\n \"gpioStatusRequest.method\",\n {\n \"key\": \"gpioStatusRequest.paramsBody\",\n \"type\": \"json\"\n }\n ]\n },\n {\n \"key\": \"gpioStatusChangeRequest\",\n \"items\": [\n \"gpioStatusChangeRequest.method\",\n {\n \"key\": \"gpioStatusChangeRequest.paramsBody\",\n \"type\": \"json\"\n }\n ]\n },\n {\n \"key\": \"parseGpioStatusFunction\",\n \"type\": \"javascript\"\n }\n ]\n}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"targetDeviceAliases\":[],\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"0px\",\"settings\":{\"parseGpioStatusFunction\":\"return body[pin] === true;\",\"gpioStatusChangeRequest\":{\"method\":\"setGpioStatus\",\"paramsBody\":\"{\\n \\\"pin\\\": \\\"{$pin}\\\",\\n \\\"enabled\\\": \\\"{$enabled}\\\"\\n}\"},\"requestTimeout\":500,\"switchPanelBackgroundColor\":\"#b71c1c\",\"gpioStatusRequest\":{\"method\":\"getGpioStatus\",\"paramsBody\":\"{}\"},\"gpioList\":[{\"pin\":1,\"label\":\"GPIO 1\",\"row\":0,\"col\":0,\"_uniqueKey\":0},{\"pin\":2,\"label\":\"GPIO 2\",\"row\":0,\"col\":1,\"_uniqueKey\":1},{\"pin\":3,\"label\":\"GPIO 3\",\"row\":1,\"col\":0,\"_uniqueKey\":2}]},\"title\":\"Basic GPIO Control\"}"}',
237   -'Basic GPIO Control' );
238   -
239   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
240   -VALUES ( now ( ), minTimeuuid ( 0 ), 'digital_gauges',
241   -'horizontal_bar_justgage',
242   -'{"type":"latest","sizeX":7,"sizeY":3,"resources":[],"templateHtml":"<canvas id=\"digitalGauge\"></canvas>\n","templateCss":"#gauge {\n text-align: center;\n /* margin-left: -100px;\n margin-right: -100px;*/\n /*margin-top: -50px;*/\n \n}\n","controllerScript":"self.onInit = function() {\n self.ctx.gauge = new TbCanvasDigitalGauge(self.ctx, ''digitalGauge''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.gauge.update();\n}\n\nself.onResize = function() {\n self.ctx.gauge.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbCanvasDigitalGauge.settingsSchema;\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.gauge.mobileModeChanged();\n}\n\nself.onDestroy = function() {\n}\n\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Temp\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.7282710489093589,\"funcBody\":\"var value = prevValue + Math.random() * 20 - 10;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"#ffffff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"0px\",\"settings\":{\"maxValue\":100,\"minValue\":0,\"donutStartAngle\":90,\"showValue\":true,\"showMinMax\":true,\"gaugeWidthScale\":0.75,\"levelColors\":[],\"refreshAnimationType\":\">\",\"refreshAnimationTime\":700,\"startAnimationType\":\">\",\"startAnimationTime\":700,\"titleFont\":{\"family\":\"Roboto\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\",\"color\":\"#999999\"},\"labelFont\":{\"family\":\"Roboto\",\"size\":8,\"style\":\"normal\",\"weight\":\"500\"},\"valueFont\":{\"family\":\"Roboto\",\"style\":\"normal\",\"weight\":\"500\",\"size\":18,\"color\":\"#666666\"},\"minMaxFont\":{\"family\":\"Roboto\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\",\"color\":\"#666666\"},\"neonGlowBrightness\":0,\"decimals\":0,\"dashThickness\":0,\"gaugeColor\":\"#eeeeee\",\"showTitle\":true,\"gaugeType\":\"horizontalBar\"},\"title\":\"Horizontal bar - justGage\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',
243   -'Horizontal bar - justGage' );
244   -
245   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
246   -VALUES ( now ( ), minTimeuuid ( 0 ), 'digital_gauges',
247   -'gauge_justgage',
248   -'{"type":"latest","sizeX":4,"sizeY":3,"resources":[],"templateHtml":"<canvas id=\"digitalGauge\"></canvas>","templateCss":"#gauge {\n text-align: center;\n /* margin-left: -100px;\n margin-right: -100px;*/\n /*margin-top: -50px;*/\n \n}\n","controllerScript":"self.onInit = function() {\n self.ctx.gauge = new TbCanvasDigitalGauge(self.ctx, ''digitalGauge''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.gauge.update();\n}\n\nself.onResize = function() {\n self.ctx.gauge.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbCanvasDigitalGauge.settingsSchema;\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.gauge.mobileModeChanged();\n}\n\nself.onDestroy = function() {\n}\n\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Temp\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.7282710489093589,\"funcBody\":\"var value = prevValue + Math.random() * 20 - 10;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"#ffffff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"0px\",\"settings\":{\"maxValue\":100,\"minValue\":0,\"donutStartAngle\":90,\"showValue\":true,\"showMinMax\":true,\"gaugeWidthScale\":0.75,\"levelColors\":[],\"refreshAnimationType\":\">\",\"refreshAnimationTime\":700,\"startAnimationType\":\">\",\"startAnimationTime\":700,\"titleFont\":{\"family\":\"Roboto\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\",\"color\":\"#999999\"},\"labelFont\":{\"family\":\"Roboto\",\"size\":8,\"style\":\"normal\",\"weight\":\"500\"},\"valueFont\":{\"family\":\"Roboto\",\"style\":\"normal\",\"weight\":\"500\",\"size\":36,\"color\":\"#666666\"},\"minMaxFont\":{\"family\":\"Roboto\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\",\"color\":\"#666666\"},\"neonGlowBrightness\":0,\"decimals\":0,\"dashThickness\":0,\"gaugeColor\":\"#eeeeee\",\"showTitle\":true,\"gaugeType\":\"arc\"},\"title\":\"Gauge - justGage\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',
249   -'Gauge - justGage' );
250   -
251   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
252   -VALUES ( now ( ), minTimeuuid ( 0 ), 'digital_gauges', 'lcd_gauge',
253   -'{"type":"latest","sizeX":5,"sizeY":3,"resources":[],"templateHtml":"<canvas id=\"digitalGauge\"></canvas>","templateCss":"#gauge {\n text-align: center;\n /* margin-left: -100px;\n margin-right: -100px;*/\n /*margin-top: -50px;*/\n \n}\n","controllerScript":"self.onInit = function() {\n self.ctx.gauge = new TbCanvasDigitalGauge(self.ctx, ''digitalGauge''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.gauge.update();\n}\n\nself.onResize = function() {\n self.ctx.gauge.resize();\n}\n\nself.getSettingsSchema = function() {\n return TbCanvasDigitalGauge.settingsSchema;\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.gauge.mobileModeChanged();\n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Speed\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.7282710489093589,\"funcBody\":\"var value = prevValue + Math.random() * 40 - 20;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 180) {\\n\\tvalue = 180;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"#babab2\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"0px\",\"settings\":{\"maxValue\":180,\"minValue\":0,\"donutStartAngle\":90,\"showValue\":true,\"showMinMax\":true,\"gaugeWidthScale\":0.75,\"levelColors\":[],\"refreshAnimationType\":\"linear\",\"refreshAnimationTime\":700,\"startAnimationType\":\"linear\",\"startAnimationTime\":700,\"titleFont\":{\"family\":\"Roboto\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\"},\"labelFont\":{\"family\":\"Roboto\",\"size\":8,\"style\":\"normal\",\"weight\":\"500\"},\"valueFont\":{\"family\":\"Segment7Standard\",\"style\":\"normal\",\"weight\":\"500\",\"size\":32},\"minMaxFont\":{\"family\":\"Segment7Standard\",\"size\":12,\"style\":\"normal\",\"weight\":\"500\"},\"neonGlowBrightness\":0,\"dashThickness\":1.5,\"decimals\":0,\"unitTitle\":\"MPH\",\"showUnitTitle\":true,\"defaultColor\":\"#444444\",\"gaugeType\":\"arc\"},\"title\":\"LCD gauge\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',
254   -'LCD gauge' );
255   -
256   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
257   -VALUES ( now ( ), minTimeuuid ( 0 ), 'maps', 'openstreetmap',
258   -'{"type":"latest","sizeX":8.5,"sizeY":6,"resources":[],"templateHtml":"","templateCss":".leaflet-zoom-box {\n\tz-index: 9;\n}\n\n.leaflet-pane { z-index: 4; }\n\n.leaflet-tile-pane { z-index: 2; }\n.leaflet-overlay-pane { z-index: 4; }\n.leaflet-shadow-pane { z-index: 5; }\n.leaflet-marker-pane { z-index: 6; }\n.leaflet-tooltip-pane { z-index: 7; }\n.leaflet-popup-pane { z-index: 8; }\n\n.leaflet-map-pane canvas { z-index: 1; }\n.leaflet-map-pane svg { z-index: 2; }\n\n.leaflet-control {\n\tz-index: 9;\n}\n.leaflet-top,\n.leaflet-bottom {\n\tz-index: 11;\n}\n\n.tb-marker-label {\n border: none;\n background: none;\n box-shadow: none;\n}\n\n.tb-marker-label:before {\n border: none;\n background: none;\n}\n","controllerScript":"self.onInit = function() {\n self.ctx.map = new TbMapWidget(''openstreet-map'', false, self.ctx);\n}\n\nself.onDataUpdated = function() {\n self.ctx.map.update();\n}\n\nself.onResize = function() {\n self.ctx.map.resize();\n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{\n \"schema\": {\n \"title\": \"Google Map Configuration\",\n \"type\": \"object\",\n \"properties\": {\n \"defaultZoomLevel\": {\n \"title\": \"Default map zoom level (1 - 20)\",\n \"type\": \"number\"\n },\n \"fitMapBounds\": {\n \"title\": \"Fit map bounds to cover all markers\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"markersSettings\": {\n \"title\": \"Markers\",\n \"type\": \"array\",\n \"items\": {\n \"title\": \"Marker settings\",\n \"type\": \"object\",\n \"properties\": {\n \"latKeyName\": {\n \"title\": \"Latitude key name\",\n \"type\": \"string\",\n \"default\": \"lat\"\n },\n \"lngKeyName\": {\n \"title\": \"Longitude key name\",\n \"type\": \"string\",\n \"default\": \"lng\"\n }, \n \"showLabel\": {\n \"title\": \"Show label\",\n \"type\": \"boolean\",\n \"default\": true\n }, \n \"label\": {\n \"title\": \"Label\",\n \"type\": \"string\"\n },\n \"tooltipPattern\": {\n \"title\": \"Pattern ( for ex. ''Text ${keyName} units.'' or ''${#<key index>} units'' )\",\n \"type\": \"string\",\n \"default\": \"<b>Latitude:</b> ${lat:7}<br/><b>Longitude:</b> ${lng:7}\"\n },\n \"color\": {\n \"title\": \"Color\",\n \"type\": \"string\"\n },\n \"useColorFunction\": {\n \"title\": \"Use color function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"colorFunction\": {\n \"title\": \"Color function: f(data)\",\n \"type\": \"string\"\n },\n \"markerImage\": {\n \"title\": \"Custom marker image\",\n \"type\": \"string\"\n },\n \"markerImageSize\": {\n \"title\": \"Custom marker image size (px)\",\n \"type\": \"number\",\n \"default\": 34\n },\n \"useMarkerImageFunction\": {\n \"title\": \"Use marker image function\",\n \"type\": \"boolean\",\n \"default\": false\n },\n \"markerImageFunction\": {\n \"title\": \"Marker image function: f(data, images)\",\n \"type\": \"string\"\n },\n \"markerImages\": {\n \"title\": \"Marker images\",\n \"type\": \"array\",\n \"items\": {\n \"title\": \"Marker image\",\n \"type\": \"string\"\n }\n }\n }\n }\n }\n },\n \"required\": [\n ]\n },\n \"form\": [\n \"defaultZoomLevel\",\n \"fitMapBounds\",\n {\n \"key\": \"markersSettings\",\n \"items\": [\n \"markersSettings[].latKeyName\",\n \"markersSettings[].lngKeyName\",\n \"markersSettings[].showLabel\",\n \"markersSettings[].label\",\n \"markersSettings[].tooltipPattern\",\n {\n \"key\": \"markersSettings[].color\",\n \"type\": \"color\"\n },\n \"markersSettings[].useColorFunction\",\n {\n \"key\": \"markersSettings[].colorFunction\",\n \"type\": \"javascript\"\n },\n {\n \"key\": \"markersSettings[].markerImage\",\n \"type\": \"image\"\n },\n \"markersSettings[].markerImageSize\",\n \"markersSettings[].useMarkerImageFunction\",\n {\n \"key\": \"markersSettings[].markerImageFunction\",\n \"type\": \"javascript\"\n },\n {\n \"key\": \"markersSettings[].markerImages\",\n \"items\": [\n {\n \"key\": \"markersSettings[].markerImages[]\",\n \"type\": \"image\"\n }\n ]\n }\n ]\n }\n ]\n}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"latitude\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.05427416942713381,\"funcBody\":\"var value = prevValue || 15.833293;\\nif (time % 5000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"longitude\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.680594833308841,\"funcBody\":\"var value = prevValue || -90.454350;\\nif (time % 5000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"}]},{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"lat\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.05012157428742059,\"funcBody\":\"var value = prevValue || 14.450463;\\nif (time % 4000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"lng\",\"color\":\"#ffc107\",\"settings\":{},\"_hash\":0.6742359401617628,\"funcBody\":\"var value = prevValue || -84.845334;\\nif (time % 4000 < 500) {\\n value += Math.random() * 0.05 - 0.025;\\n}\\nreturn value;\"}]},{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"temperature\",\"color\":\"#607d8b\",\"settings\":{},\"_hash\":0.21553274887887564,\"funcBody\":\"var value = prevValue + Math.random() * 40 - 20;\\nif (value < -60) {\\n\\tvalue = -60;\\n} else if (value > 60) {\\n\\tvalue = 60;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"markersSettings\":[{\"label\":\"First point\",\"color\":\"#1e88e5\",\"latKeyName\":\"latitude\",\"lngKeyName\":\"longitude\",\"showLabel\":true,\"tooltipPattern\":\"<b>Latitude:</b> ${latitude:7}<br/><b>Longitude:</b> ${longitude:7}<br/><b>Temperature:</b> ${temperature} °C<br/><small>See advanced settings for details</small>\",\"useColorFunction\":true,\"colorFunction\":\"var temperature = data[''temperature''];\\nif (typeof temperature !== undefined) {\\n var percent = (temperature + 60)/120 * 100;\\n return tinycolor.mix(''blue'', ''red'', amount = percent).toHexString();\\n}\\nreturn ''blue'';\"},{\"label\":\"Second point\",\"color\":\"#fdd835\",\"latKeyName\":\"lat\",\"lngKeyName\":\"lng\",\"showLabel\":true,\"tooltipPattern\":\"<b>Latitude:</b> ${lat:7}<br/><b>Longitude:</b> ${lng:7}<br/><b>Temperature:</b> ${temperature} °C<br/><small>See advanced settings for details</small>\",\"markerImageSize\":34,\"useMarkerImageFunction\":true,\"markerImages\":[\"\",\"\",\"\",\"\"],\"markerImageFunction\":\"var res = {\\n url: images[0],\\n size: 40\\n}\\nvar temperature = data[''temperature''];\\nif (typeof temperature !== undefined) {\\n var percent = (temperature + 60)/120;\\n var index = Math.floor(4 * percent);\\n res.url = images[index];\\n}\\nreturn res;\"}],\"fitMapBounds\":true},\"title\":\"OpenStreetMap\"}"}',
259   -'OpenStreetMap' );
260   -
261   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
262   -VALUES ( now ( ), minTimeuuid ( 0 ), 'analogue_gauges',
263   -'speed_gauge_canvas_gauges',
264   -'{"type":"latest","sizeX":7,"sizeY":5,"resources":[],"templateHtml":"<canvas id=\"radialGauge\"></canvas>\n","templateCss":"","controllerScript":"self.onInit = function() {\n self.ctx.gauge = new TbAnalogueRadialGauge(self.ctx, ''radialGauge''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.gauge.update();\n}\n\nself.onResize = function() {\n self.ctx.gauge.resize();\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.gauge.mobileModeChanged();\n}\n\nself.getSettingsSchema = function() {\n return TbAnalogueRadialGauge.settingsSchema;\n}\n\nself.onDestroy = function() {\n}\n","settingsSchema":"{}","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Speed\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.7282710489093589,\"funcBody\":\"var value = prevValue + Math.random() * 50 - 25;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 220) {\\n\\tvalue = 220;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":false,\"backgroundColor\":\"rgb(255, 255, 255)\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"maxValue\":180,\"startAngle\":45,\"ticksAngle\":270,\"showBorder\":false,\"defaultColor\":\"#e65100\",\"needleCircleSize\":7,\"highlights\":[{\"from\":80,\"to\":120,\"color\":\"#fdd835\"},{\"color\":\"#e57373\",\"from\":120,\"to\":180}],\"showUnitTitle\":false,\"colorPlate\":\"#fff\",\"colorMajorTicks\":\"#444\",\"colorMinorTicks\":\"#666\",\"minorTicks\":2,\"valueInt\":3,\"minValue\":0,\"valueDec\":0,\"highlightsWidth\":15,\"valueBox\":true,\"animation\":true,\"animationDuration\":1500,\"animationRule\":\"linear\",\"colorNeedleShadowUp\":\"rgba(2, 255, 255, 0)\",\"colorNeedleShadowDown\":\"rgba(188, 143, 143, 0.78)\",\"units\":\"MPH\",\"majorTicksCount\":9,\"numbersFont\":{\"family\":\"Roboto\",\"size\":22,\"style\":\"normal\",\"weight\":\"500\",\"color\":\"#616161\"},\"titleFont\":{\"family\":\"Roboto\",\"size\":24,\"style\":\"normal\",\"weight\":\"500\",\"color\":\"#888\"},\"unitsFont\":{\"family\":\"Roboto\",\"size\":28,\"style\":\"normal\",\"weight\":\"500\",\"color\":\"#616161\"},\"valueFont\":{\"size\":32,\"style\":\"normal\",\"weight\":\"normal\",\"shadowColor\":\"rgba(0, 0, 0, 0.49)\",\"color\":\"#444\",\"family\":\"Segment7Standard\"},\"colorValueBoxRect\":\"#888\",\"colorValueBoxRectEnd\":\"#666\",\"colorValueBoxBackground\":\"#babab2\",\"colorValueBoxShadow\":\"rgba(0,0,0,1)\"},\"title\":\"Speed gauge - Canvas Gauges\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',
265   -'Speed gauge - Canvas Gauges' );
266   -
267   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
268   -VALUES ( now ( ), minTimeuuid ( 0 ), 'charts', 'pie',
269   -'{"type":"latest","sizeX":8,"sizeY":5,"resources":[],"templateHtml":"","templateCss":".legend {\n font-size: 13px;\n line-height: 10px;\n}\n\n.legend table { \n border-spacing: 0px;\n border-collapse: separate;\n}\n\n.pie-label {\n font-size: 12px;\n font-family: ''Roboto'';\n font-weight: bold;\n text-align: center;\n padding: 2px;\n color: white;\n}\n","controllerScript":"self.onInit = function() {\n self.ctx.flot = new TbFlot(self.ctx, ''pie''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.flot.update();\n}\n\nself.onResize = function() {\n self.ctx.flot.resize();\n}\n\nself.onEditModeChanged = function() {\n self.ctx.flot.checkMouseEvents();\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.flot.checkMouseEvents();\n}\n\nself.getSettingsSchema = function() {\n return TbFlot.pieSettingsSchema;\n}\n\nself.getDataKeySettingsSchema = function() {\n return TbFlot.pieDatakeySettingsSchema;\n}\n\nself.onDestroy = function() {\n self.ctx.flot.destroy();\n}\n","settingsSchema":"{}\n","dataKeySettingsSchema":"{}\n","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"First\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.15479322438769105,\"funcBody\":\"var value = (prevValue-50) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+50;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Second\",\"color\":\"#4caf50\",\"settings\":{},\"_hash\":0.6114638304362894,\"funcBody\":\"var value = (prevValue-20) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+20;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Third\",\"color\":\"#f44336\",\"settings\":{},\"_hash\":0.9955906536344441,\"funcBody\":\"var value = (prevValue-40) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+40;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Fourth\",\"color\":\"#ffc107\",\"settings\":{},\"_hash\":0.9430835931647599,\"funcBody\":\"var value = (prevValue-50) + Math.random() * 2 - 1;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 100) {\\n\\tvalue = 100;\\n}\\nreturn value+50;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"radius\":1,\"fontColor\":\"#545454\",\"fontSize\":10,\"decimals\":1,\"legend\":{\"show\":true,\"position\":\"nw\",\"labelBoxBorderColor\":\"#CCCCCC\",\"backgroundColor\":\"#F0F0F0\",\"backgroundOpacity\":0.85},\"innerRadius\":0,\"showLabels\":true,\"stroke\":{\"width\":5},\"tilt\":1,\"animatedPie\":false},\"title\":\"Pie - Flot\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400}}"}',
270   -'Pie - Flot' );
271   -
272   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
273   -VALUES ( now ( ), minTimeuuid ( 0 ), 'charts', 'timeseries_bars_flot',
274   -'{"type":"timeseries","sizeX":8,"sizeY":5,"resources":[],"templateHtml":"","templateCss":".legend {\n font-size: 13px;\n line-height: 10px;\n}\n\n.legend table { \n border-spacing: 0px;\n border-collapse: separate;\n}\n\n.mouse-events .flot-overlay {\n cursor: crosshair; \n}\n\n","controllerScript":"self.onInit = function() {\n self.ctx.flot = new TbFlot(self.ctx, ''bar''); \n}\n\nself.onDataUpdated = function() {\n self.ctx.flot.update();\n}\n\nself.onResize = function() {\n self.ctx.flot.resize();\n}\n\nself.onEditModeChanged = function() {\n self.ctx.flot.checkMouseEvents();\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.flot.checkMouseEvents();\n}\n\nself.getSettingsSchema = function() {\n return TbFlot.settingsSchema;\n}\n\nself.getDataKeySettingsSchema = function() {\n return TbFlot.datakeySettingsSchema(false);\n}\n\nself.onDestroy = function() {\n self.ctx.flot.destroy();\n}\n","settingsSchema":"{}","dataKeySettingsSchema":"{}","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"First\",\"color\":\"#2196f3\",\"settings\":{\"showLines\":false,\"fillLines\":false,\"showPoints\":false},\"_hash\":0.8587686344902596,\"funcBody\":\"var value = prevValue + Math.random() * 100 - 50;\\nvar multiplier = Math.pow(10, 2 || 0);\\nvar value = Math.round(value * multiplier) / multiplier;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 1000) {\\n\\tvalue = 1000;\\n}\\nreturn value;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Second\",\"color\":\"#ffc107\",\"settings\":{\"showLines\":false,\"fillLines\":false,\"showPoints\":false},\"_hash\":0.12775350966079668,\"funcBody\":\"var value = prevValue + Math.random() * 100 - 50;\\nvar multiplier = Math.pow(10, 2 || 0);\\nvar value = Math.round(value * multiplier) / multiplier;\\nif (value < 0) {\\n\\tvalue = 0;\\n} else if (value > 1000) {\\n\\tvalue = 1000;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000},\"aggregation\":{\"limit\":200,\"type\":\"AVG\"}},\"showTitle\":true,\"backgroundColor\":\"#fff\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"8px\",\"settings\":{\"shadowSize\":4,\"fontColor\":\"#545454\",\"fontSize\":10,\"xaxis\":{\"showLabels\":true,\"color\":\"#545454\"},\"yaxis\":{\"showLabels\":true,\"color\":\"#545454\"},\"grid\":{\"color\":\"#545454\",\"tickColor\":\"#DDDDDD\",\"verticalLines\":true,\"horizontalLines\":true,\"outlineWidth\":1},\"legend\":{\"show\":true,\"position\":\"nw\",\"backgroundColor\":\"#f0f0f0\",\"backgroundOpacity\":0.85,\"labelBoxBorderColor\":\"rgba(1, 1, 1, 0.45)\"},\"decimals\":1,\"stack\":true,\"tooltipIndividual\":false},\"title\":\"Timeseries Bars - Flot\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"mobileHeight\":null}"}',
275   -'Timeseries Bars - Flot' );
276   -
277   -INSERT INTO "thingsboard"."widget_type" ( "id", "tenant_id", "bundle_alias", "alias", "descriptor", "name" )
278   -VALUES ( now ( ), minTimeuuid ( 0 ), 'charts', 'basic_timeseries',
279   -'{"type":"timeseries","sizeX":8,"sizeY":5,"resources":[],"templateHtml":"","templateCss":".legend {\n font-size: 13px;\n line-height: 10px;\n}\n\n.legend table { \n border-spacing: 0px;\n border-collapse: separate;\n}\n\n.mouse-events .flot-overlay {\n cursor: crosshair; \n}\n\n","controllerScript":"self.onInit = function() {\n self.ctx.flot = new TbFlot(self.ctx); \n}\n\nself.onDataUpdated = function() {\n self.ctx.flot.update();\n}\n\nself.onResize = function() {\n self.ctx.flot.resize();\n}\n\nself.onEditModeChanged = function() {\n self.ctx.flot.checkMouseEvents();\n}\n\nself.onMobileModeChanged = function() {\n self.ctx.flot.checkMouseEvents();\n}\n\nself.getSettingsSchema = function() {\n return TbFlot.settingsSchema;\n}\n\nself.getDataKeySettingsSchema = function() {\n return TbFlot.datakeySettingsSchema(true);\n}\n\nself.onDestroy = function() {\n self.ctx.flot.destroy();\n}\n","settingsSchema":"{}","dataKeySettingsSchema":"{}","defaultConfig":"{\"datasources\":[{\"type\":\"function\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"First\",\"color\":\"#2196f3\",\"settings\":{\"showLines\":true,\"fillLines\":true,\"showPoints\":false},\"_hash\":0.8587686344902596,\"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;\"},{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Second\",\"color\":\"#ffc107\",\"settings\":{\"showLines\":true,\"fillLines\":false,\"showPoints\":false},\"_hash\":0.12775350966079668,\"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\":{\"shadowSize\":4,\"fontColor\":\"#545454\",\"fontSize\":10,\"xaxis\":{\"showLabels\":true,\"color\":\"#545454\"},\"yaxis\":{\"showLabels\":true,\"color\":\"#545454\"},\"grid\":{\"color\":\"#545454\",\"tickColor\":\"#DDDDDD\",\"verticalLines\":true,\"horizontalLines\":true,\"outlineWidth\":1},\"legend\":{\"show\":true,\"position\":\"nw\",\"backgroundColor\":\"#f0f0f0\",\"backgroundOpacity\":0.85,\"labelBoxBorderColor\":\"rgba(1, 1, 1, 0.45)\"},\"decimals\":1,\"stack\":false,\"tooltipIndividual\":false},\"title\":\"Timeseries - Flot\",\"dropShadow\":true,\"enableFullscreen\":true,\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"mobileHeight\":null}"}',
280   -'Timeseries - Flot' );
281   -
282   -/** SYSTEM **/
  44 +}' );
\ No newline at end of file
... ...
... ... @@ -228,12 +228,13 @@ CREATE TABLE IF NOT EXISTS rule_node (
228 228 search_text varchar(255)
229 229 );
230 230
231   -CREATE TABLE IF NOT EXISTS entity_views (
  231 +CREATE TABLE IF NOT EXISTS entity_view (
232 232 id varchar(31) NOT NULL CONSTRAINT entity_view_pkey PRIMARY KEY,
233 233 entity_id varchar(31),
234 234 entity_type varchar(255),
235 235 tenant_id varchar(31),
236 236 customer_id varchar(31),
  237 + type varchar(255),
237 238 name varchar(255),
238 239 keys varchar(255),
239 240 start_ts bigint,
... ...
... ... @@ -19,4 +19,4 @@ DROP TABLE IF EXISTS widget_type;
19 19 DROP TABLE IF EXISTS widgets_bundle;
20 20 DROP TABLE IF EXISTS rule_node;
21 21 DROP TABLE IF EXISTS rule_chain;
22   -DROP TABLE IF EXISTS entity_views;
  22 +DROP TABLE IF EXISTS entity_view;
... ...
1 1
2 2 DOCKER_REPO=local-maven-build
  3 +
  4 +JS_EXECUTOR_DOCKER_NAME=tb-js-executor
  5 +TB_NODE_DOCKER_NAME=tb-node
  6 +WEB_UI_DOCKER_NAME=tb-web-ui
  7 +
3 8 TB_VERSION=2.2.0-SNAPSHOT
4 9
5 10 KAFKA_TOPICS=js.eval.requests:100:1
... ...
... ... @@ -15,7 +15,7 @@
15 15 #
16 16
17 17
18   -version: '2'
  18 +version: '2.2'
19 19
20 20 services:
21 21 zookeeper:
... ... @@ -40,7 +40,8 @@ services:
40 40 - zookeeper
41 41 tb-js-executor:
42 42 restart: always
43   - image: "${DOCKER_REPO}/tb-js-executor:${TB_VERSION}"
  43 + image: "${DOCKER_REPO}/${JS_EXECUTOR_DOCKER_NAME}:${TB_VERSION}"
  44 + scale: 20
44 45 environment:
45 46 TB_KAFKA_SERVERS: kafka:9092
46 47 env_file:
... ... @@ -49,11 +50,16 @@ services:
49 50 - kafka
50 51 tb:
51 52 restart: always
52   - image: "${DOCKER_REPO}/tb-node:${TB_VERSION}"
  53 + image: "${DOCKER_REPO}/${TB_NODE_DOCKER_NAME}:${TB_VERSION}"
53 54 ports:
54 55 - "8080"
55 56 - "1883:1883"
56 57 - "5683:5683/udp"
  58 + logging:
  59 + driver: "json-file"
  60 + options:
  61 + max-size: "200m"
  62 + max-file: "30"
57 63 env_file:
58 64 - tb-node.env
59 65 environment:
... ... @@ -68,7 +74,7 @@ services:
68 74 - kafka
69 75 tb-web-ui1:
70 76 restart: always
71   - image: "${DOCKER_REPO}/tb-web-ui:${TB_VERSION}"
  77 + image: "${DOCKER_REPO}/${WEB_UI_DOCKER_NAME}:${TB_VERSION}"
72 78 ports:
73 79 - "8080"
74 80 environment:
... ... @@ -78,7 +84,7 @@ services:
78 84 - tb-web-ui.env
79 85 tb-web-ui2:
80 86 restart: always
81   - image: "${DOCKER_REPO}/tb-web-ui:${TB_VERSION}"
  87 + image: "${DOCKER_REPO}/${WEB_UI_DOCKER_NAME}:${TB_VERSION}"
82 88 ports:
83 89 - "8080"
84 90 environment:
... ...
... ... @@ -17,4 +17,4 @@
17 17
18 18 ./check-dirs.sh
19 19
20   -docker-compose up -d --scale tb-js-executor=20
  20 +docker-compose up -d
... ...
... ... @@ -15,4 +15,5 @@
15 15 # limitations under the License.
16 16 #
17 17
  18 +docker-compose pull $@
18 19 docker-compose up -d --no-deps --build $@
... ...
... ... @@ -35,8 +35,9 @@ frontend http-in
35 35
36 36 reqadd X-Forwarded-Proto:\ http
37 37
  38 + acl transport_http_acl path_beg /api/v1/
38 39 acl letsencrypt_http_acl path_beg /.well-known/acme-challenge/
39   - redirect scheme https if !letsencrypt_http_acl
  40 + redirect scheme https if !letsencrypt_http_acl !transport_http_acl
40 41 use_backend letsencrypt_http if letsencrypt_http_acl
41 42
42 43 default_backend tb-web-backend
... ...
... ... @@ -19,6 +19,6 @@ export JAVA_OPTS="$JAVA_OPTS -Xloggc:/var/log/thingsboard/gc.log -XX:+IgnoreUnre
19 19 export JAVA_OPTS="$JAVA_OPTS -XX:+PrintHeapAtGC -XX:+PrintTenuringDistribution -XX:+PrintGCApplicationStoppedTime -XX:+UseGCLogFileRotation -XX:NumberOfGCLogFiles=10"
20 20 export JAVA_OPTS="$JAVA_OPTS -XX:GCLogFileSize=10M -XX:-UseBiasedLocking -XX:+UseTLAB -XX:+ResizeTLAB -XX:+PerfDisableSharedMem -XX:+UseCondCardMark"
21 21 export JAVA_OPTS="$JAVA_OPTS -XX:CMSWaitDuration=10000 -XX:+UseParNewGC -XX:+UseConcMarkSweepGC -XX:+CMSParallelRemarkEnabled -XX:+CMSParallelInitialMarkEnabled"
22   -export JAVA_OPTS="$JAVA_OPTS -XX:+CMSEdenChunksRecordAlways -XX:CMSInitiatingOccupancyFraction=75 -XX:+UseCMSInitiatingOccupancyOnly"
  22 +export JAVA_OPTS="$JAVA_OPTS -XX:+CMSEdenChunksRecordAlways -XX:CMSInitiatingOccupancyFraction=75 -XX:+UseCMSInitiatingOccupancyOnly -XX:+ExitOnOutOfMemoryError"
23 23 export LOG_FILENAME=thingsboard.out
24 24 export LOADER_PATH=/usr/share/thingsboard/conf,/usr/share/thingsboard/extensions
... ...
... ... @@ -23,4 +23,6 @@ RUN chmod a+x /tmp/*.sh \
23 23
24 24 RUN dpkg -i /tmp/${pkg.name}.deb
25 25
  26 +RUN update-rc.d ${pkg.name} disable
  27 +
26 28 CMD ["start-js-executor.sh"]
... ...
... ... @@ -35,6 +35,7 @@
35 35 <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
36 36 <main.dir>${basedir}/../..</main.dir>
37 37 <pkg.name>tb-js-executor</pkg.name>
  38 + <docker.name>tb-js-executor</docker.name>
38 39 <pkg.user>thingsboard</pkg.user>
39 40 <pkg.unixLogFolder>/var/log/${pkg.name}</pkg.unixLogFolder>
40 41 <pkg.installFolder>/usr/share/${pkg.name}</pkg.installFolder>
... ... @@ -290,7 +291,7 @@
290 291 </executions>
291 292 <configuration>
292 293 <skip>${dockerfile.skip}</skip>
293   - <repository>${docker.repo}/${pkg.name}</repository>
  294 + <repository>${docker.repo}/${docker.name}</repository>
294 295 <tag>${project.version}</tag>
295 296 <verbose>true</verbose>
296 297 <googleContainerRegistryEnabled>false</googleContainerRegistryEnabled>
... ...
... ... @@ -2,5 +2,5 @@
2 2
3 3 chown -R ${pkg.user}: ${pkg.logFolder}
4 4 chown -R ${pkg.user}: ${pkg.installFolder}
5   -# update-rc.d ${pkg.name} defaults
  5 +update-rc.d ${pkg.name} defaults
6 6
... ...
... ... @@ -35,6 +35,7 @@
35 35 <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
36 36 <main.dir>${basedir}/../..</main.dir>
37 37 <pkg.name>thingsboard</pkg.name>
  38 + <docker.name>tb-node</docker.name>
38 39 <pkg.user>thingsboard</pkg.user>
39 40 <pkg.unixLogFolder>/var/log/${pkg.name}</pkg.unixLogFolder>
40 41 <pkg.installFolder>/usr/share/${pkg.name}</pkg.installFolder>
... ... @@ -114,7 +115,7 @@
114 115 </executions>
115 116 <configuration>
116 117 <skip>${dockerfile.skip}</skip>
117   - <repository>${docker.repo}/tb-node</repository>
  118 + <repository>${docker.repo}/${docker.name}</repository>
118 119 <tag>${project.version}</tag>
119 120 <verbose>true</verbose>
120 121 <googleContainerRegistryEnabled>false</googleContainerRegistryEnabled>
... ...
... ... @@ -23,4 +23,6 @@ RUN chmod a+x /tmp/*.sh \
23 23
24 24 RUN dpkg -i /tmp/${pkg.name}.deb
25 25
  26 +RUN update-rc.d ${pkg.name} disable
  27 +
26 28 CMD ["start-web-ui.sh"]
... ...
... ... @@ -35,6 +35,7 @@
35 35 <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
36 36 <main.dir>${basedir}/../..</main.dir>
37 37 <pkg.name>tb-web-ui</pkg.name>
  38 + <docker.name>tb-web-ui</docker.name>
38 39 <pkg.user>thingsboard</pkg.user>
39 40 <pkg.unixLogFolder>/var/log/${pkg.name}</pkg.unixLogFolder>
40 41 <pkg.installFolder>/usr/share/${pkg.name}</pkg.installFolder>
... ... @@ -314,7 +315,7 @@
314 315 </executions>
315 316 <configuration>
316 317 <skip>${dockerfile.skip}</skip>
317   - <repository>${docker.repo}/${pkg.name}</repository>
  318 + <repository>${docker.repo}/${docker.name}</repository>
318 319 <tag>${project.version}</tag>
319 320 <verbose>true</verbose>
320 321 <googleContainerRegistryEnabled>false</googleContainerRegistryEnabled>
... ...
... ... @@ -2,5 +2,5 @@
2 2
3 3 chown -R ${pkg.user}: ${pkg.logFolder}
4 4 chown -R ${pkg.user}: ${pkg.installFolder}
5   -# update-rc.d ${pkg.name} defaults
  5 +update-rc.d ${pkg.name} defaults
6 6
... ...
... ... @@ -48,7 +48,7 @@ import static org.thingsboard.rule.engine.api.TbRelationTypes.SUCCESS;
48 48 @Slf4j
49 49 @RuleNode(
50 50 type = ComponentType.ACTION,
51   - name = "copy attributes",
  51 + name = "copy to view",
52 52 configClazz = EmptyNodeConfiguration.class,
53 53 nodeDescription = "Copy attributes from asset/device to entity view and changes message originator to related entity view",
54 54 nodeDetails = "Copy attributes from asset/device to related entity view according to entity view configuration. \n " +
... ...
... ... @@ -37,7 +37,7 @@ import java.net.InetSocketAddress;
37 37 import java.net.UnknownHostException;
38 38
39 39 @Service("CoapTransportService")
40   -@ConditionalOnProperty(prefix = "coap", value = "enabled", havingValue = "true", matchIfMissing = true)
  40 +@ConditionalOnProperty(prefix = "transport.coap", value = "enabled", havingValue = "true")
41 41 @Slf4j
42 42 public class CoapTransportService {
43 43
... ...
... ... @@ -18,6 +18,7 @@ package org.thingsboard.server.transport.http;
18 18 import lombok.extern.slf4j.Slf4j;
19 19 import org.springframework.beans.factory.annotation.Autowired;
20 20 import org.springframework.beans.factory.annotation.Value;
  21 +import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
21 22 import org.springframework.http.HttpStatus;
22 23 import org.springframework.http.ResponseEntity;
23 24 import org.springframework.web.bind.annotation.*;
... ... @@ -34,6 +35,7 @@ import javax.servlet.http.HttpServletRequest;
34 35 * @author Andrew Shvayka
35 36 */
36 37 @RestController
  38 +@ConditionalOnProperty(prefix = "transport.http", value = "enabled", havingValue = "true")
37 39 @RequestMapping("/api/v1")
38 40 @Slf4j
39 41 public class DeviceApiController {
... ...
... ... @@ -23,6 +23,7 @@ import lombok.extern.slf4j.Slf4j;
23 23 import org.apache.commons.lang3.RandomStringUtils;
24 24 import org.springframework.beans.factory.annotation.Autowired;
25 25 import org.springframework.beans.factory.annotation.Value;
  26 +import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
26 27 import org.springframework.context.annotation.Lazy;
27 28 import org.springframework.stereotype.Component;
28 29 import org.springframework.util.StringUtils;
... ... @@ -41,6 +42,7 @@ import java.util.concurrent.Executors;
41 42 * Created by ashvayka on 04.10.18.
42 43 */
43 44 @Slf4j
  45 +@ConditionalOnProperty(prefix = "transport.mqtt", value = "enabled", havingValue = "true", matchIfMissing = true)
44 46 @Component
45 47 @Data
46 48 public class MqttTransportContext {
... ... @@ -59,7 +61,7 @@ public class MqttTransportContext {
59 61 @Autowired
60 62 private MqttTransportAdaptor adaptor;
61 63
62   - @Value("${mqtt.netty.max_payload_size}")
  64 + @Value("${transport.mqtt.netty.max_payload_size}")
63 65 private Integer maxPayloadSize;
64 66
65 67 @Value("${cluster.node_id:#{null}}")
... ...
... ... @@ -55,6 +55,7 @@ import org.thingsboard.server.gen.transport.TransportProtos.ValidateDeviceX509Ce
55 55 import org.thingsboard.server.transport.mqtt.adaptors.MqttTransportAdaptor;
56 56 import org.thingsboard.server.transport.mqtt.session.DeviceSessionCtx;
57 57 import org.thingsboard.server.transport.mqtt.session.GatewaySessionHandler;
  58 +import org.thingsboard.server.transport.mqtt.session.MqttTopicMatcher;
58 59 import org.thingsboard.server.transport.mqtt.util.SslUtil;
59 60
60 61 import javax.net.ssl.SSLPeerUnverifiedException;
... ... @@ -93,7 +94,7 @@ public class MqttTransportHandler extends ChannelInboundHandlerAdapter implement
93 94 private final TransportService transportService;
94 95 private final QuotaService quotaService;
95 96 private final SslHandler sslHandler;
96   - private final ConcurrentMap<String, Integer> mqttQoSMap;
  97 + private final ConcurrentMap<MqttTopicMatcher, Integer> mqttQoSMap;
97 98
98 99 private volatile SessionInfoProto sessionInfo;
99 100 private volatile InetSocketAddress address;
... ... @@ -295,7 +296,7 @@ public class MqttTransportHandler extends ChannelInboundHandlerAdapter implement
295 296
296 297 private void registerSubQoS(String topic, List<Integer> grantedQoSList, MqttQoS reqQoS) {
297 298 grantedQoSList.add(getMinSupportedQos(reqQoS));
298   - mqttQoSMap.put(topic, getMinSupportedQos(reqQoS));
  299 + mqttQoSMap.put(new MqttTopicMatcher(topic), getMinSupportedQos(reqQoS));
299 300 }
300 301
301 302 private void processUnsubscribe(ChannelHandlerContext ctx, MqttUnsubscribeMessage mqttMsg) {
... ... @@ -304,7 +305,7 @@ public class MqttTransportHandler extends ChannelInboundHandlerAdapter implement
304 305 }
305 306 log.trace("[{}] Processing subscription [{}]!", sessionId, mqttMsg.variableHeader().messageId());
306 307 for (String topicName : mqttMsg.payload().topics()) {
307   - mqttQoSMap.remove(topicName);
  308 + mqttQoSMap.remove(new MqttTopicMatcher(topicName));
308 309 try {
309 310 switch (topicName) {
310 311 case MqttTopics.DEVICE_ATTRIBUTES_TOPIC: {
... ...
... ... @@ -44,25 +44,22 @@ import javax.annotation.PreDestroy;
44 44 * @author Andrew Shvayka
45 45 */
46 46 @Service("MqttTransportService")
47   -@ConditionalOnProperty(prefix = "mqtt", value = "enabled", havingValue = "true", matchIfMissing = false)
  47 +@ConditionalOnProperty(prefix = "transport.mqtt", value = "enabled", havingValue = "true", matchIfMissing = true)
48 48 @Slf4j
49 49 public class MqttTransportService {
50 50
51   - private static final String V1 = "v1";
52   - private static final String DEVICE = "device";
53   -
54   - @Value("${mqtt.bind_address}")
  51 + @Value("${transport.mqtt.bind_address}")
55 52 private String host;
56   - @Value("${mqtt.bind_port}")
  53 + @Value("${transport.mqtt.bind_port}")
57 54 private Integer port;
58   - @Value("${mqtt.adaptor}")
  55 + @Value("${transport.mqtt.adaptor}")
59 56 private String adaptorName;
60 57
61   - @Value("${mqtt.netty.leak_detector_level}")
  58 + @Value("${transport.mqtt.netty.leak_detector_level}")
62 59 private String leakDetectorLevel;
63   - @Value("${mqtt.netty.boss_group_thread_count}")
  60 + @Value("${transport.mqtt.netty.boss_group_thread_count}")
64 61 private Integer bossGroupThreadCount;
65   - @Value("${mqtt.netty.worker_group_thread_count}")
  62 + @Value("${transport.mqtt.netty.worker_group_thread_count}")
66 63 private Integer workerGroupThreadCount;
67 64
68 65 @Autowired
... ...
... ... @@ -36,19 +36,10 @@ public class DeviceSessionCtx extends MqttDeviceAwareSessionContext {
36 36 private ChannelHandlerContext channel;
37 37 private AtomicInteger msgIdSeq = new AtomicInteger(0);
38 38
39   - public DeviceSessionCtx(UUID sessionId, ConcurrentMap<String, Integer> mqttQoSMap) {
  39 + public DeviceSessionCtx(UUID sessionId, ConcurrentMap<MqttTopicMatcher, Integer> mqttQoSMap) {
40 40 super(sessionId, mqttQoSMap);
41 41 }
42 42
43   - private void logAndWrap(AdaptorException e) throws SessionException {
44   - log.warn("Failed to convert msg: {}", e.getMessage(), e);
45   - throw new SessionException(e);
46   - }
47   -
48   - private void pushToNetwork(MqttMessage msg) {
49   - channel.writeAndFlush(msg);
50   - }
51   -
52 43 public void setChannel(ChannelHandlerContext channel) {
53 44 this.channel = channel;
54 45 }
... ...
... ... @@ -33,7 +33,7 @@ public class GatewayDeviceSessionCtx extends MqttDeviceAwareSessionContext imple
33 33 private final GatewaySessionHandler parent;
34 34 private final SessionInfoProto sessionInfo;
35 35
36   - public GatewayDeviceSessionCtx(GatewaySessionHandler parent, DeviceInfoProto deviceInfo, ConcurrentMap<String, Integer> mqttQoSMap) {
  36 + public GatewayDeviceSessionCtx(GatewaySessionHandler parent, DeviceInfoProto deviceInfo, ConcurrentMap<MqttTopicMatcher, Integer> mqttQoSMap) {
37 37 super(UUID.randomUUID(), mqttQoSMap);
38 38 this.parent = parent;
39 39 this.sessionInfo = SessionInfoProto.newBuilder()
... ...
... ... @@ -68,7 +68,7 @@ public class GatewaySessionHandler {
68 68 private final DeviceInfoProto gateway;
69 69 private final UUID sessionId;
70 70 private final Map<String, GatewayDeviceSessionCtx> devices;
71   - private final ConcurrentMap<String, Integer> mqttQoSMap;
  71 + private final ConcurrentMap<MqttTopicMatcher, Integer> mqttQoSMap;
72 72 private final ChannelHandlerContext channel;
73 73 private final DeviceSessionCtx deviceSessionCtx;
74 74
... ...
... ... @@ -16,35 +16,38 @@
16 16 package org.thingsboard.server.transport.mqtt.session;
17 17
18 18 import io.netty.handler.codec.mqtt.MqttQoS;
19   -import org.thingsboard.server.common.data.Device;
20   -import org.thingsboard.server.common.transport.SessionMsgProcessor;
21   -import org.thingsboard.server.common.transport.auth.DeviceAuthService;
22 19 import org.thingsboard.server.common.transport.session.DeviceAwareSessionContext;
23 20
  21 +import java.util.List;
24 22 import java.util.Map;
25 23 import java.util.UUID;
26 24 import java.util.concurrent.ConcurrentMap;
  25 +import java.util.stream.Collectors;
27 26
28 27 /**
29 28 * Created by ashvayka on 30.08.18.
30 29 */
31 30 public abstract class MqttDeviceAwareSessionContext extends DeviceAwareSessionContext {
32 31
33   - private final ConcurrentMap<String, Integer> mqttQoSMap;
  32 + private final ConcurrentMap<MqttTopicMatcher, Integer> mqttQoSMap;
34 33
35   - public MqttDeviceAwareSessionContext(UUID sessionId, ConcurrentMap<String, Integer> mqttQoSMap) {
  34 + public MqttDeviceAwareSessionContext(UUID sessionId, ConcurrentMap<MqttTopicMatcher, Integer> mqttQoSMap) {
36 35 super(sessionId);
37 36 this.mqttQoSMap = mqttQoSMap;
38 37 }
39 38
40   - public ConcurrentMap<String, Integer> getMqttQoSMap() {
  39 + public ConcurrentMap<MqttTopicMatcher, Integer> getMqttQoSMap() {
41 40 return mqttQoSMap;
42 41 }
43 42
44 43 public MqttQoS getQoSForTopic(String topic) {
45   - Integer qos = mqttQoSMap.get(topic);
46   - if (qos != null) {
47   - return MqttQoS.valueOf(qos);
  44 + List<Integer> qosList = mqttQoSMap.entrySet()
  45 + .stream()
  46 + .filter(entry -> entry.getKey().matches(topic))
  47 + .map(Map.Entry::getValue)
  48 + .collect(Collectors.toList());
  49 + if (!qosList.isEmpty()) {
  50 + return MqttQoS.valueOf(qosList.get(0));
48 51 } else {
49 52 return MqttQoS.AT_LEAST_ONCE;
50 53 }
... ...
  1 +/**
  2 + * Copyright © 2016-2018 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 +package org.thingsboard.server.transport.mqtt.session;
  17 +
  18 +import java.util.regex.Pattern;
  19 +
  20 +public class MqttTopicMatcher {
  21 +
  22 + private final String topic;
  23 + private final Pattern topicRegex;
  24 +
  25 + public MqttTopicMatcher(String topic) {
  26 + if(topic == null){
  27 + throw new NullPointerException("topic");
  28 + }
  29 + this.topic = topic;
  30 + this.topicRegex = Pattern.compile(topic.replace("+", "[^/]+").replace("#", ".+") + "$");
  31 + }
  32 +
  33 + public String getTopic() {
  34 + return topic;
  35 + }
  36 +
  37 + public boolean matches(String topic){
  38 + return this.topicRegex.matcher(topic).matches();
  39 + }
  40 +
  41 + @Override
  42 + public boolean equals(Object o) {
  43 + if (this == o) return true;
  44 + if (o == null || getClass() != o.getClass()) return false;
  45 +
  46 + MqttTopicMatcher that = (MqttTopicMatcher) o;
  47 +
  48 + return topic.equals(that.topic);
  49 + }
  50 +
  51 + @Override
  52 + public int hashCode() {
  53 + return topic.hashCode();
  54 + }
  55 +}
... ...
... ... @@ -17,32 +17,31 @@
17 17 spring.main.web-application-type: none
18 18
19 19 # MQTT server parameters
20   -mqtt:
21   - # Enable/disable mqtt transport protocol.
22   - enabled: "${MQTT_ENABLED:true}"
23   - bind_address: "${MQTT_BIND_ADDRESS:0.0.0.0}"
24   - bind_port: "${MQTT_BIND_PORT:1883}"
25   - adaptor: "${MQTT_ADAPTOR_NAME:JsonMqttAdaptor}"
26   - timeout: "${MQTT_TIMEOUT:10000}"
27   - netty:
28   - leak_detector_level: "${NETTY_LEASK_DETECTOR_LVL:DISABLED}"
29   - boss_group_thread_count: "${NETTY_BOSS_GROUP_THREADS:1}"
30   - worker_group_thread_count: "${NETTY_WORKER_GROUP_THREADS:12}"
31   - max_payload_size: "${NETTY_MAX_PAYLOAD_SIZE:65536}"
32   - # MQTT SSL configuration
33   - ssl:
34   - # Enable/disable SSL support
35   - enabled: "${MQTT_SSL_ENABLED:false}"
36   - # SSL protocol: See http://docs.oracle.com/javase/8/docs/technotes/guides/security/StandardNames.html#SSLContext
37   - protocol: "${MQTT_SSL_PROTOCOL:TLSv1.2}"
38   - # Path to the key store that holds the SSL certificate
39   - key_store: "${MQTT_SSL_KEY_STORE:mqttserver.jks}"
40   - # Password used to access the key store
41   - key_store_password: "${MQTT_SSL_KEY_STORE_PASSWORD:server_ks_password}"
42   - # Password used to access the key
43   - key_password: "${MQTT_SSL_KEY_PASSWORD:server_key_password}"
44   - # Type of the key store
45   - key_store_type: "${MQTT_SSL_KEY_STORE_TYPE:JKS}"
  20 +transport:
  21 + mqtt:
  22 + bind_address: "${MQTT_BIND_ADDRESS:0.0.0.0}"
  23 + bind_port: "${MQTT_BIND_PORT:1883}"
  24 + adaptor: "${MQTT_ADAPTOR_NAME:JsonMqttAdaptor}"
  25 + timeout: "${MQTT_TIMEOUT:10000}"
  26 + netty:
  27 + leak_detector_level: "${NETTY_LEASK_DETECTOR_LVL:DISABLED}"
  28 + boss_group_thread_count: "${NETTY_BOSS_GROUP_THREADS:1}"
  29 + worker_group_thread_count: "${NETTY_WORKER_GROUP_THREADS:12}"
  30 + max_payload_size: "${NETTY_MAX_PAYLOAD_SIZE:65536}"
  31 + # MQTT SSL configuration
  32 + ssl:
  33 + # Enable/disable SSL support
  34 + enabled: "${MQTT_SSL_ENABLED:false}"
  35 + # SSL protocol: See http://docs.oracle.com/javase/8/docs/technotes/guides/security/StandardNames.html#SSLContext
  36 + protocol: "${MQTT_SSL_PROTOCOL:TLSv1.2}"
  37 + # Path to the key store that holds the SSL certificate
  38 + key_store: "${MQTT_SSL_KEY_STORE:mqttserver.jks}"
  39 + # Password used to access the key store
  40 + key_store_password: "${MQTT_SSL_KEY_STORE_PASSWORD:server_ks_password}"
  41 + # Password used to access the key
  42 + key_password: "${MQTT_SSL_KEY_PASSWORD:server_key_password}"
  43 + # Type of the key store
  44 + key_store_type: "${MQTT_SSL_KEY_STORE_TYPE:JKS}"
46 45
47 46 #Quota parameters
48 47 quota:
... ... @@ -56,7 +55,7 @@ quota:
56 55 # Interval for scheduled task that cleans expired records. TTL is used for expiring
57 56 cleanPeriodMs: "${QUOTA_HOST_CLEAN_PERIOD_MS:300000}"
58 57 # Enable Host API Limits
59   - enabled: "${QUOTA_HOST_ENABLED:false}"
  58 + enabled: "${QUOTA_HOST_ENABLED:true}"
60 59 # Array of whitelist hosts
61 60 whitelist: "${QUOTA_HOST_WHITELIST:localhost,127.0.0.1}"
62 61 # Array of blacklist hosts
... ...
... ... @@ -533,6 +533,21 @@ function EntityService($http, $q, $filter, $translate, $log, userService, device
533 533 }
534 534 );
535 535 break;
  536 + case types.aliasFilterType.entityViewType.value:
  537 + getEntitiesByNameFilter(types.entityType.entityView, filter.entityViewNameFilter, maxItems, {ignoreLoading: true}, filter.entityViewType).then(
  538 + function success(entities) {
  539 + if (entities && entities.length || !failOnEmpty) {
  540 + result.entities = entitiesToEntitiesInfo(entities);
  541 + deferred.resolve(result);
  542 + } else {
  543 + deferred.reject();
  544 + }
  545 + },
  546 + function fail() {
  547 + deferred.reject();
  548 + }
  549 + );
  550 + break;
536 551 case types.aliasFilterType.relationsQuery.value:
537 552 result.stateEntity = filter.rootStateEntity;
538 553 var rootEntityType;
... ... @@ -578,6 +593,7 @@ function EntityService($http, $q, $filter, $translate, $log, userService, device
578 593 break;
579 594 case types.aliasFilterType.assetSearchQuery.value:
580 595 case types.aliasFilterType.deviceSearchQuery.value:
  596 + case types.aliasFilterType.entityViewSearchQuery.value:
581 597 result.stateEntity = filter.rootStateEntity;
582 598 if (result.stateEntity && stateEntityId) {
583 599 rootEntityType = stateEntityId.entityType;
... ... @@ -604,6 +620,9 @@ function EntityService($http, $q, $filter, $translate, $log, userService, device
604 620 } else if (filter.type == types.aliasFilterType.deviceSearchQuery.value) {
605 621 searchQuery.deviceTypes = filter.deviceTypes;
606 622 findByQueryPromise = deviceService.findByQuery(searchQuery, false, {ignoreLoading: true});
  623 + } else if (filter.type == types.aliasFilterType.entityViewSearchQuery.value) {
  624 + searchQuery.entityViewTypes = filter.entityViewTypes;
  625 + findByQueryPromise = entityViewService.findByQuery(searchQuery, false, {ignoreLoading: true});
607 626 }
608 627 findByQueryPromise.then(
609 628 function success(entities) {
... ... @@ -646,6 +665,8 @@ function EntityService($http, $q, $filter, $translate, $log, userService, device
646 665 return entityTypes.indexOf(types.entityType.asset) > -1 ? true : false;
647 666 case types.aliasFilterType.deviceType.value:
648 667 return entityTypes.indexOf(types.entityType.device) > -1 ? true : false;
  668 + case types.aliasFilterType.entityViewType.value:
  669 + return entityTypes.indexOf(types.entityType.entityView) > -1 ? true : false;
649 670 case types.aliasFilterType.relationsQuery.value:
650 671 if (filter.filters && filter.filters.length) {
651 672 var match = false;
... ... @@ -671,6 +692,8 @@ function EntityService($http, $q, $filter, $translate, $log, userService, device
671 692 return entityTypes.indexOf(types.entityType.asset) > -1 ? true : false;
672 693 case types.aliasFilterType.deviceSearchQuery.value:
673 694 return entityTypes.indexOf(types.entityType.device) > -1 ? true : false;
  695 + case types.aliasFilterType.entityViewSearchQuery.value:
  696 + return entityTypes.indexOf(types.entityType.entityView) > -1 ? true : false;
674 697 }
675 698 }
676 699 return false;
... ... @@ -690,12 +713,16 @@ function EntityService($http, $q, $filter, $translate, $log, userService, device
690 713 return entityType === types.entityType.asset;
691 714 case types.aliasFilterType.deviceType.value:
692 715 return entityType === types.entityType.device;
  716 + case types.aliasFilterType.entityViewType.value:
  717 + return entityType === types.entityType.entityView;
693 718 case types.aliasFilterType.relationsQuery.value:
694 719 return true;
695 720 case types.aliasFilterType.assetSearchQuery.value:
696 721 return entityType === types.entityType.asset;
697 722 case types.aliasFilterType.deviceSearchQuery.value:
698 723 return entityType === types.entityType.device;
  724 + case types.aliasFilterType.entityViewSearchQuery.value:
  725 + return entityType === types.entityType.entityView;
699 726 }
700 727 return false;
701 728 }
... ... @@ -1046,6 +1073,8 @@ function EntityService($http, $q, $filter, $translate, $log, userService, device
1046 1073 return assetService.deleteAsset(entityId.id);
1047 1074 } else if (entityId.entityType == types.entityType.device) {
1048 1075 return deviceService.deleteDevice(entityId.id);
  1076 + } else if (entityId.entityType == types.entityType.entityView) {
  1077 + return entityViewService.deleteEntityView(entityId.id);
1049 1078 }
1050 1079 }
1051 1080
... ... @@ -1151,6 +1180,8 @@ function EntityService($http, $q, $filter, $translate, $log, userService, device
1151 1180 return assetService.saveAsset(entity);
1152 1181 } else if (entityType == types.entityType.device) {
1153 1182 return deviceService.saveDevice(entity);
  1183 + } else if (entityType == types.entityType.entityView) {
  1184 + return entityViewService.saveEntityView(entity);
1154 1185 }
1155 1186 }
1156 1187
... ... @@ -1279,6 +1310,8 @@ function EntityService($http, $q, $filter, $translate, $log, userService, device
1279 1310 searchQuery.assetTypes = entitySubTypes;
1280 1311 } else if (entityType == types.entityType.device) {
1281 1312 searchQuery.deviceTypes = entitySubTypes;
  1313 + } else if (entityType == types.entityType.entityView) {
  1314 + searchQuery.entityViewTypes = entitySubTypes;
1282 1315 } else {
1283 1316 return null; //Not supported
1284 1317 }
... ...
... ... @@ -172,9 +172,9 @@ export default class Subscription {
172 172 if (this.type === this.ctx.types.widgetType.rpc.value) {
173 173 if (this.targetDeviceId) {
174 174 entityId = {
175   - entityType: this.ctx.entityType.device,
  175 + entityType: this.ctx.types.entityType.device,
176 176 id: this.targetDeviceId
177   - }
  177 + };
178 178 entityName = this.targetDeviceName;
179 179 }
180 180 } else if (this.type == this.ctx.types.widgetType.alarm.value) {
... ... @@ -182,7 +182,7 @@ export default class Subscription {
182 182 entityId = {
183 183 entityType: this.alarmSource.entityType,
184 184 id: this.alarmSource.entityId
185   - }
  185 + };
186 186 entityName = this.alarmSource.entityName;
187 187 }
188 188 } else {
... ... @@ -192,7 +192,7 @@ export default class Subscription {
192 192 entityId = {
193 193 entityType: datasource.entityType,
194 194 id: datasource.entityId
195   - }
  195 + };
196 196 entityName = datasource.entityName;
197 197 break;
198 198 }
... ...
... ... @@ -26,15 +26,17 @@ const MIN_INTERVAL = SECOND;
26 26 const MAX_INTERVAL = 365 * 20 * DAY;
27 27
28 28 const MIN_LIMIT = 10;
29   -const AVG_LIMIT = 200;
30   -const MAX_LIMIT = 500;
  29 +//const AVG_LIMIT = 200;
  30 +//const MAX_LIMIT = 500;
31 31
32 32 /*@ngInject*/
33   -function TimeService($translate, types) {
  33 +function TimeService($translate, $http, $q, types) {
34 34
35 35 var predefIntervals;
  36 + var maxDatapointsLimit;
36 37
37 38 var service = {
  39 + loadMaxDatapointsLimit: loadMaxDatapointsLimit,
38 40 minIntervalLimit: minIntervalLimit,
39 41 maxIntervalLimit: maxIntervalLimit,
40 42 boundMinInterval: boundMinInterval,
... ... @@ -45,20 +47,38 @@ function TimeService($translate, types) {
45 47 defaultTimewindow: defaultTimewindow,
46 48 toHistoryTimewindow: toHistoryTimewindow,
47 49 createSubscriptionTimewindow: createSubscriptionTimewindow,
48   - avgAggregationLimit: function () {
49   - return AVG_LIMIT;
  50 + getMaxDatapointsLimit: function () {
  51 + return maxDatapointsLimit;
  52 + },
  53 + getMinDatapointsLimit: function () {
  54 + return MIN_LIMIT;
50 55 }
51 56 }
52 57
53 58 return service;
54 59
  60 + function loadMaxDatapointsLimit() {
  61 + var deferred = $q.defer();
  62 + var url = '/api/dashboard/maxDatapointsLimit';
  63 + $http.get(url, {ignoreLoading: true}).then(function success(response) {
  64 + maxDatapointsLimit = response.data;
  65 + if (!maxDatapointsLimit || maxDatapointsLimit <= MIN_LIMIT) {
  66 + maxDatapointsLimit = MIN_LIMIT + 1;
  67 + }
  68 + deferred.resolve();
  69 + }, function fail() {
  70 + deferred.reject();
  71 + });
  72 + return deferred.promise;
  73 + }
  74 +
55 75 function minIntervalLimit(timewindow) {
56   - var min = timewindow / MAX_LIMIT;
  76 + var min = timewindow / 500;
57 77 return boundMinInterval(min);
58 78 }
59 79
60 80 function avgInterval(timewindow) {
61   - var avg = timewindow / AVG_LIMIT;
  81 + var avg = timewindow / 200;
62 82 return boundMinInterval(avg);
63 83 }
64 84
... ... @@ -230,7 +250,7 @@ function TimeService($translate, types) {
230 250 },
231 251 aggregation: {
232 252 type: types.aggregation.avg.value,
233   - limit: AVG_LIMIT
  253 + limit: Math.floor(maxDatapointsLimit / 2)
234 254 }
235 255 }
236 256 return timewindow;
... ... @@ -246,22 +266,27 @@ function TimeService($translate, types) {
246 266 }
247 267
248 268 var aggType;
  269 + var limit;
249 270 if (timewindow.aggregation) {
250 271 aggType = timewindow.aggregation.type || types.aggregation.avg.value;
  272 + limit = timewindow.aggregation.limit || maxDatapointsLimit;
251 273 } else {
252 274 aggType = types.aggregation.avg.value;
  275 + limit = maxDatapointsLimit;
253 276 }
254 277
  278 +
255 279 var historyTimewindow = {
256 280 history: {
257 281 fixedTimewindow: {
258 282 startTimeMs: startTimeMs,
259 283 endTimeMs: endTimeMs
260 284 },
261   - interval: boundIntervalToTimewindow(endTimeMs - startTimeMs, interval, aggType)
  285 + interval: boundIntervalToTimewindow(endTimeMs - startTimeMs, interval, types.aggregation.avg.value)
262 286 },
263 287 aggregation: {
264   - type: aggType
  288 + type: aggType,
  289 + limit: limit
265 290 }
266 291 }
267 292
... ... @@ -275,7 +300,7 @@ function TimeService($translate, types) {
275 300 realtimeWindowMs: null,
276 301 aggregation: {
277 302 interval: SECOND,
278   - limit: AVG_LIMIT,
  303 + limit: maxDatapointsLimit,
279 304 type: types.aggregation.avg.value
280 305 }
281 306 };
... ... @@ -283,14 +308,14 @@ function TimeService($translate, types) {
283 308 if (stateData) {
284 309 subscriptionTimewindow.aggregation = {
285 310 interval: SECOND,
286   - limit: MAX_LIMIT,
  311 + limit: maxDatapointsLimit,
287 312 type: types.aggregation.none.value,
288 313 stateData: true
289 314 };
290 315 } else {
291 316 subscriptionTimewindow.aggregation = {
292 317 interval: SECOND,
293   - limit: AVG_LIMIT,
  318 + limit: maxDatapointsLimit,
294 319 type: types.aggregation.avg.value
295 320 };
296 321 }
... ... @@ -298,7 +323,7 @@ function TimeService($translate, types) {
298 323 if (angular.isDefined(timewindow.aggregation) && !stateData) {
299 324 subscriptionTimewindow.aggregation = {
300 325 type: timewindow.aggregation.type || types.aggregation.avg.value,
301   - limit: timewindow.aggregation.limit || AVG_LIMIT
  326 + limit: timewindow.aggregation.limit || maxDatapointsLimit
302 327 };
303 328 }
304 329 if (angular.isDefined(timewindow.realtime)) {
... ...
... ... @@ -22,7 +22,7 @@ export default angular.module('thingsboard.api.user', [thingsboardApiLogin,
22 22 .name;
23 23
24 24 /*@ngInject*/
25   -function UserService($http, $q, $rootScope, adminService, dashboardService, loginService, toast, store, jwtHelper, $translate, $state, $location) {
  25 +function UserService($http, $q, $rootScope, adminService, dashboardService, timeService, loginService, toast, store, jwtHelper, $translate, $state, $location) {
26 26 var currentUser = null,
27 27 currentUserDetails = null,
28 28 lastPublicDashboardId = null,
... ... @@ -390,6 +390,7 @@ function UserService($http, $q, $rootScope, adminService, dashboardService, logi
390 390 function loadSystemParams() {
391 391 var promises = [];
392 392 promises.push(loadIsUserTokenAccessEnabled());
  393 + promises.push(timeService.loadMaxDatapointsLimit());
393 394 return $q.all(promises);
394 395 }
395 396
... ...
... ... @@ -253,6 +253,10 @@ export default angular.module('thingsboard.types', [])
253 253 value: 'deviceType',
254 254 name: 'alias.filter-type-device-type'
255 255 },
  256 + entityViewType: {
  257 + value: 'entityViewType',
  258 + name: 'alias.filter-type-entity-view-type'
  259 + },
256 260 relationsQuery: {
257 261 value: 'relationsQuery',
258 262 name: 'alias.filter-type-relations-query'
... ... @@ -264,6 +268,10 @@ export default angular.module('thingsboard.types', [])
264 268 deviceSearchQuery: {
265 269 value: 'deviceSearchQuery',
266 270 name: 'alias.filter-type-device-search-query'
  271 + },
  272 + entityViewSearchQuery: {
  273 + value: 'entityViewSearchQuery',
  274 + name: 'alias.filter-type-entity-view-search-query'
267 275 }
268 276 },
269 277 position: {
... ...
... ... @@ -31,6 +31,8 @@ export default function TimewindowPanelController(mdPanelRef, $scope, timeServic
31 31 vm.maxRealtimeAggInterval = maxRealtimeAggInterval;
32 32 vm.minHistoryAggInterval = minHistoryAggInterval;
33 33 vm.maxHistoryAggInterval = maxHistoryAggInterval;
  34 + vm.minDatapointsLimit = minDatapointsLimit;
  35 + vm.maxDatapointsLimit = maxDatapointsLimit;
34 36
35 37 if (vm.historyOnly) {
36 38 vm.timewindow.selectedTab = 1;
... ... @@ -86,6 +88,14 @@ export default function TimewindowPanelController(mdPanelRef, $scope, timeServic
86 88 return timeService.maxIntervalLimit(currentHistoryTimewindow());
87 89 }
88 90
  91 + function minDatapointsLimit () {
  92 + return timeService.getMinDatapointsLimit();
  93 + }
  94 +
  95 + function maxDatapointsLimit () {
  96 + return timeService.getMaxDatapointsLimit();
  97 + }
  98 +
89 99 function currentHistoryTimewindow() {
90 100 if (vm.timewindow.history.historyType === 0) {
91 101 return vm.timewindow.history.timewindowMs;
... ...
... ... @@ -60,19 +60,22 @@
60 60 </md-option>
61 61 </md-select>
62 62 </md-input-container>
63   - <md-slider-container ng-show="vm.showLimit()">
  63 + <md-slider-container ng-if="vm.showLimit()">
64 64 <span translate>aggregation.limit</span>
65   - <md-slider flex min="10" max="500" ng-model="vm.timewindow.aggregation.limit" aria-label="limit" id="limit-slider">
  65 + <md-slider flex min="{{vm.minDatapointsLimit()}}" max="{{vm.maxDatapointsLimit()}}" step="1"
  66 + ng-model="vm.timewindow.aggregation.limit" aria-label="limit" id="limit-slider">
66 67 </md-slider>
67   - <md-input-container>
68   - <input flex type="number" ng-model="vm.timewindow.aggregation.limit" aria-label="limit" aria-controls="limit-slider">
  68 + <md-input-container style="max-width: 80px;">
  69 + <input flex type="number" step="1"
  70 + min="{{vm.minDatapointsLimit()}}" max="{{vm.maxDatapointsLimit()}}"
  71 + ng-model="vm.timewindow.aggregation.limit" aria-label="limit" aria-controls="limit-slider">
69 72 </md-input-container>
70 73 </md-slider-container>
71   - <tb-timeinterval ng-show="vm.showRealtimeAggInterval()" min="vm.minRealtimeAggInterval()" max="vm.maxRealtimeAggInterval()"
  74 + <tb-timeinterval ng-if="vm.showRealtimeAggInterval()" min="vm.minRealtimeAggInterval()" max="vm.maxRealtimeAggInterval()"
72 75 predefined-name="'aggregation.group-interval'"
73 76 ng-model="vm.timewindow.realtime.interval">
74 77 </tb-timeinterval>
75   - <tb-timeinterval ng-show="vm.showHistoryAggInterval()" min="vm.minHistoryAggInterval()" max="vm.maxHistoryAggInterval()"
  78 + <tb-timeinterval ng-if="vm.showHistoryAggInterval()" min="vm.minHistoryAggInterval()" max="vm.maxHistoryAggInterval()"
76 79 predefined-name="'aggregation.group-interval'"
77 80 ng-model="vm.timewindow.history.interval">
78 81 </tb-timeinterval>
... ...
... ... @@ -228,7 +228,7 @@ function Timewindow($compile, $templateCache, $filter, $mdPanel, $document, $mdM
228 228 if (angular.isDefined(value.aggregation.type) && value.aggregation.type.length > 0) {
229 229 model.aggregation.type = value.aggregation.type;
230 230 }
231   - model.aggregation.limit = value.aggregation.limit || timeService.avgAggregationLimit();
  231 + model.aggregation.limit = value.aggregation.limit || Math.floor(timeService.getMaxDatapointsLimit() / 2);
232 232 }
233 233 }
234 234 scope.updateDisplayValue();
... ...
... ... @@ -52,12 +52,22 @@
52 52 <div translate ng-message="required">entity-view.name-required</div>
53 53 </div>
54 54 </md-input-container>
55   - <tb-entity-select flex ng-disabled="!isEdit"
56   - the-form="theForm"
57   - tb-required="true"
58   - allowed-entity-types="allowedEntityTypes"
59   - ng-model="entityView.entityId">
60   - </tb-entity-select>
  55 + <tb-entity-subtype-autocomplete
  56 + ng-disabled="$root.loading || !isEdit"
  57 + tb-required="true"
  58 + the-form="theForm"
  59 + ng-model="entityView.type"
  60 + entity-type="types.entityType.entityView">
  61 + </tb-entity-subtype-autocomplete>
  62 + <section layout="column">
  63 + <label translate class="tb-title no-padding">entity-view.related-entity</label>
  64 + <tb-entity-select flex ng-disabled="!isEdit"
  65 + the-form="theForm"
  66 + tb-required="true"
  67 + allowed-entity-types="allowedEntityTypes"
  68 + ng-model="entityView.entityId">
  69 + </tb-entity-select>
  70 + </section>
61 71 <md-input-container class="md-block">
62 72 <label translate>entity-view.description</label>
63 73 <textarea ng-model="entityView.additionalInfo.description" rows="2"></textarea>
... ...
... ... @@ -42,7 +42,7 @@ export default function EntityViewRoutes($stateProvider, types) {
42 42 pageTitle: 'entity-view.entity-views'
43 43 },
44 44 ncyBreadcrumb: {
45   - label: '{"icon": "view_stream", "label": "entity-view.entity-views"}'
  45 + label: '{"icon": "view_quilt", "label": "entity-view.entity-views"}'
46 46 }
47 47 })
48 48 .state('home.customers.entityViews', {
... ... @@ -65,7 +65,7 @@ export default function EntityViewRoutes($stateProvider, types) {
65 65 pageTitle: 'customer.entity-views'
66 66 },
67 67 ncyBreadcrumb: {
68   - label: '{"icon": "view_stream", "label": "{{ vm.customerEntityViewsTitle }}", "translate": "false"}'
  68 + label: '{"icon": "view_quilt", "label": "{{ vm.customerEntityViewsTitle }}", "translate": "false"}'
69 69 }
70 70 });
71 71
... ...
... ... @@ -77,6 +77,15 @@ export default function EntityFilterViewDirective($compile, $templateCache, $q,
77 77 scope.filterDisplayValue = $translate.instant('alias.filter-type-device-type-description', {deviceType: deviceType});
78 78 }
79 79 break;
  80 + case types.aliasFilterType.entityViewType.value:
  81 + var entityViewType = scope.filter.entityViewType;
  82 + prefix = scope.filter.entityViewNameFilter;
  83 + if (prefix && prefix.length) {
  84 + scope.filterDisplayValue = $translate.instant('alias.filter-type-entity-view-type-and-name-description', {entityViewType: entityViewType, prefix: prefix});
  85 + } else {
  86 + scope.filterDisplayValue = $translate.instant('alias.filter-type-entity-view-type-description', {entityViewType: entityViewType});
  87 + }
  88 + break;
80 89 case types.aliasFilterType.relationsQuery.value:
81 90 var rootEntityText;
82 91 var directionText;
... ... @@ -134,6 +143,7 @@ export default function EntityFilterViewDirective($compile, $templateCache, $q,
134 143 break;
135 144 case types.aliasFilterType.assetSearchQuery.value:
136 145 case types.aliasFilterType.deviceSearchQuery.value:
  146 + case types.aliasFilterType.entityViewSearchQuery.value:
137 147 allEntitiesText = $translate.instant('alias.all-entities');
138 148 anyRelationText = $translate.instant('alias.any-relation');
139 149 if (scope.filter.rootStateEntity) {
... ... @@ -165,7 +175,7 @@ export default function EntityFilterViewDirective($compile, $templateCache, $q,
165 175 scope.filterDisplayValue = $translate.instant('alias.filter-type-asset-search-query-description',
166 176 translationValues
167 177 );
168   - } else {
  178 + } else if (scope.filter.type == types.aliasFilterType.deviceSearchQuery.value) {
169 179 var deviceTypesQuoted = [];
170 180 scope.filter.deviceTypes.forEach(function(deviceType) {
171 181 deviceTypesQuoted.push("'"+deviceType+"'");
... ... @@ -175,6 +185,16 @@ export default function EntityFilterViewDirective($compile, $templateCache, $q,
175 185 scope.filterDisplayValue = $translate.instant('alias.filter-type-device-search-query-description',
176 186 translationValues
177 187 );
  188 + } else if (scope.filter.type == types.aliasFilterType.entityViewSearchQuery.value) {
  189 + var entityViewTypesQuoted = [];
  190 + scope.filter.entityViewTypes.forEach(function(entityViewType) {
  191 + entityViewTypesQuoted.push("'"+entityViewType+"'");
  192 + });
  193 + var entityViewTypesText = entityViewTypesQuoted.join(', ');
  194 + translationValues.entityViewTypes = entityViewTypesText;
  195 + scope.filterDisplayValue = $translate.instant('alias.filter-type-entity-view-search-query-description',
  196 + translationValues
  197 + );
178 198 }
179 199 break;
180 200 default:
... ...
... ... @@ -69,9 +69,14 @@ export default function EntityFilterDirective($compile, $templateCache, $q, $doc
69 69 filter.deviceType = null;
70 70 filter.deviceNameFilter = '';
71 71 break;
  72 + case types.aliasFilterType.entityViewType.value:
  73 + filter.entityViewType = null;
  74 + filter.entityViewNameFilter = '';
  75 + break;
72 76 case types.aliasFilterType.relationsQuery.value:
73 77 case types.aliasFilterType.assetSearchQuery.value:
74 78 case types.aliasFilterType.deviceSearchQuery.value:
  79 + case types.aliasFilterType.entityViewSearchQuery.value:
75 80 filter.rootStateEntity = false;
76 81 filter.stateEntityParamName = null;
77 82 filter.defaultStateEntity = null;
... ... @@ -86,6 +91,9 @@ export default function EntityFilterDirective($compile, $templateCache, $q, $doc
86 91 } else if (filter.type === types.aliasFilterType.deviceSearchQuery.value) {
87 92 filter.relationType = null;
88 93 filter.deviceTypes = [];
  94 + } else if (filter.type === types.aliasFilterType.entityViewSearchQuery.value) {
  95 + filter.relationType = null;
  96 + filter.entityViewTypes = [];
89 97 }
90 98 break;
91 99 }
... ...
... ... @@ -112,6 +112,20 @@
112 112 aria-label="{{ 'device.name-starts-with' | translate }}">
113 113 </md-input-container>
114 114 </section>
  115 + <section layout="column" ng-if="filter.type == types.aliasFilterType.entityViewType.value" id="entityViewTypeFilter">
  116 + <tb-entity-subtype-autocomplete
  117 + tb-required="true"
  118 + the-form="theForm"
  119 + ng-model="filter.entityViewType"
  120 + entity-type="types.entityType.entityView">
  121 + </tb-entity-subtype-autocomplete>
  122 + <md-input-container class="md-block">
  123 + <label translate>entity-view.name-starts-with</label>
  124 + <input name="entityViewNameFilter"
  125 + ng-model="filter.entityViewNameFilter"
  126 + aria-label="{{ 'entity-view.name-starts-with' | translate }}">
  127 + </md-input-container>
  128 + </section>
115 129 <section layout="column" ng-if="filter.type == types.aliasFilterType.relationsQuery.value" id="relationsQueryFilter">
116 130 <label class="tb-small">{{ 'alias.root-entity' | translate }}</label>
117 131 <section class="tb-root-state-entity-switch" layout="row" layout-align="start center" style="padding-left: 0px;">
... ... @@ -311,4 +325,73 @@
311 325 ng-model="filter.deviceTypes">
312 326 </tb-entity-subtype-list>
313 327 </section>
  328 + <section layout="column" ng-if="filter.type == types.aliasFilterType.entityViewSearchQuery.value" id="entityViewSearchQueryFilter">
  329 + <label class="tb-small">{{ 'alias.root-entity' | translate }}</label>
  330 + <section class="tb-root-state-entity-switch" layout="row" layout-align="start center" style="padding-left: 0px;">
  331 + <md-switch class="root-state-entity-switch" ng-model="filter.rootStateEntity"
  332 + aria-label="{{ 'alias.root-state-entity' | translate }}">
  333 + </md-switch>
  334 + <label class="tb-small root-state-entity-label" translate>alias.root-state-entity</label>
  335 + </section>
  336 + <div flex layout="row" ng-if="!filter.rootStateEntity">
  337 + <tb-entity-select flex
  338 + the-form="theForm"
  339 + tb-required="!filter.rootStateEntity"
  340 + ng-disabled="filter.rootStateEntity"
  341 + use-alias-entity-types="true"
  342 + ng-model="filter.rootEntity">
  343 + </tb-entity-select>
  344 + </div>
  345 + <div flex layout="row" ng-if="filter.rootStateEntity">
  346 + <md-input-container class="md-block" style="margin-top: 32px;">
  347 + <label translate>alias.state-entity-parameter-name</label>
  348 + <input name="stateEntityParamName"
  349 + placeholder="{{ 'alias.default-entity-parameter-name' | translate }}"
  350 + ng-model="filter.stateEntityParamName"
  351 + aria-label="{{ 'alias.state-entity-parameter-name' | translate }}">
  352 + </md-input-container>
  353 + <div flex layout="column">
  354 + <label class="tb-small">{{ 'alias.default-state-entity' | translate }}</label>
  355 + <tb-entity-select flex
  356 + the-form="theForm"
  357 + tb-required="false"
  358 + use-alias-entity-types="true"
  359 + ng-model="filter.defaultStateEntity">
  360 + </tb-entity-select>
  361 + </div>
  362 + </div>
  363 + <div flex layout="row">
  364 + <md-input-container class="md-block" style="min-width: 100px;">
  365 + <label translate>relation.direction</label>
  366 + <md-select required ng-model="filter.direction">
  367 + <md-option ng-repeat="direction in types.entitySearchDirection" ng-value="direction">
  368 + {{ ('relation.search-direction.' + direction) | translate}}
  369 + </md-option>
  370 + </md-select>
  371 + </md-input-container>
  372 + <md-input-container flex class="md-block">
  373 + <label translate>alias.max-relation-level</label>
  374 + <input name="maxRelationLevel"
  375 + type="number"
  376 + min="1"
  377 + step="1"
  378 + placeholder="{{ 'alias.unlimited-level' | translate }}"
  379 + ng-model="filter.maxLevel"
  380 + aria-label="{{ 'alias.max-relation-level' | translate }}">
  381 + </md-input-container>
  382 + </div>
  383 + <div class="md-caption" style="color: rgba(0,0,0,0.57);" translate>relation.relation-type</div>
  384 + <tb-relation-type-autocomplete flex
  385 + hide-label
  386 + the-form="theForm"
  387 + ng-model="filter.relationType"
  388 + tb-required="false">
  389 + </tb-relation-type-autocomplete>
  390 + <div class="md-caption tb-required" style="color: rgba(0,0,0,0.57);" translate>entity-view.entity-view-types</div>
  391 + <tb-entity-subtype-list
  392 + tb-required="true"
  393 + entity-type="types.entityType.entityView"
  394 + ng-model="filter.entityViewTypes">
  395 + </tb-entity-subtype-list>
  396 + </section>
314 397 </div>
... ...
... ... @@ -22,7 +22,7 @@ import entitySubtypeAutocompleteTemplate from './entity-subtype-autocomplete.tpl
22 22 /* eslint-enable import/no-unresolved, import/default */
23 23
24 24 /*@ngInject*/
25   -export default function EntitySubtypeAutocomplete($compile, $templateCache, $q, $filter, assetService, deviceService, types) {
  25 +export default function EntitySubtypeAutocomplete($compile, $templateCache, $q, $filter, assetService, deviceService, entityViewService, types) {
26 26
27 27 var linker = function (scope, element, attrs, ngModelCtrl) {
28 28 var template = $templateCache.get(entitySubtypeAutocompleteTemplate);
... ... @@ -96,6 +96,8 @@ export default function EntitySubtypeAutocomplete($compile, $templateCache, $q,
96 96 entitySubtypesPromise = assetService.getAssetTypes({ignoreLoading: true});
97 97 } else if (scope.entityType == types.entityType.device) {
98 98 entitySubtypesPromise = deviceService.getDeviceTypes({ignoreLoading: true});
  99 + } else if (scope.entityType == types.entityType.entityView) {
  100 + entitySubtypesPromise = entityViewService.getEntityViewTypes({ignoreLoading: true});
99 101 }
100 102 if (entitySubtypesPromise) {
101 103 entitySubtypesPromise.then(
... ... @@ -134,6 +136,13 @@ export default function EntitySubtypeAutocomplete($compile, $templateCache, $q,
134 136 scope.$on('deviceSaved', function() {
135 137 scope.entitySubtypes = null;
136 138 });
  139 + } else if (scope.entityType == types.entityType.entityView) {
  140 + scope.selectEntitySubtypeText = 'entity-view.select-entity-view-type';
  141 + scope.entitySubtypeText = 'entity-view.entity-view-type';
  142 + scope.entitySubtypeRequiredText = 'entity-view.entity-view-type-required';
  143 + scope.$on('entityViewSaved', function() {
  144 + scope.entitySubtypes = null;
  145 + });
137 146 }
138 147 }
139 148
... ...
... ... @@ -22,7 +22,7 @@ import entitySubtypeListTemplate from './entity-subtype-list.tpl.html';
22 22 import './entity-subtype-list.scss';
23 23
24 24 /*@ngInject*/
25   -export default function EntitySubtypeListDirective($compile, $templateCache, $q, $mdUtil, $translate, $filter, types, assetService, deviceService) {
  25 +export default function EntitySubtypeListDirective($compile, $templateCache, $q, $mdUtil, $translate, $filter, types, assetService, deviceService, entityViewService) {
26 26
27 27 var linker = function (scope, element, attrs, ngModelCtrl) {
28 28
... ... @@ -47,6 +47,12 @@ export default function EntitySubtypeListDirective($compile, $templateCache, $q,
47 47 scope.secondaryPlaceholder = '+' + $translate.instant('device.device-type');
48 48 scope.noSubtypesMathingText = 'device.no-device-types-matching';
49 49 scope.subtypeListEmptyText = 'device.device-type-list-empty';
  50 + } else if (scope.entityType == types.entityType.entityView) {
  51 + scope.placeholder = scope.tbRequired ? $translate.instant('entity-view.enter-entity-view-type')
  52 + : $translate.instant('entity-view.any-entity-view');
  53 + scope.secondaryPlaceholder = '+' + $translate.instant('entity-view.entity-view-type');
  54 + scope.noSubtypesMathingText = 'entity-view.no-entity-view-types-matching';
  55 + scope.subtypeListEmptyText = 'entity-view.entity-view-type-list-empty';
50 56 }
51 57
52 58 scope.$watch('tbRequired', function () {
... ... @@ -97,6 +103,8 @@ export default function EntitySubtypeListDirective($compile, $templateCache, $q,
97 103 entitySubtypesPromise = assetService.getAssetTypes({ignoreLoading: true});
98 104 } else if (scope.entityType == types.entityType.device) {
99 105 entitySubtypesPromise = deviceService.getDeviceTypes({ignoreLoading: true});
  106 + } else if (scope.entityType == types.entityType.entityView) {
  107 + entitySubtypesPromise = entityViewService.getEntityViewTypes({ignoreLoading: true});
100 108 }
101 109 if (entitySubtypesPromise) {
102 110 entitySubtypesPromise.then(
... ...
... ... @@ -22,7 +22,7 @@ import entitySubtypeSelectTemplate from './entity-subtype-select.tpl.html';
22 22 /* eslint-enable import/no-unresolved, import/default */
23 23
24 24 /*@ngInject*/
25   -export default function EntitySubtypeSelect($compile, $templateCache, $translate, assetService, deviceService, types) {
  25 +export default function EntitySubtypeSelect($compile, $templateCache, $translate, assetService, deviceService, entityViewService, types) {
26 26
27 27 var linker = function (scope, element, attrs, ngModelCtrl) {
28 28 var template = $templateCache.get(entitySubtypeSelectTemplate);
... ... @@ -75,6 +75,8 @@ export default function EntitySubtypeSelect($compile, $templateCache, $translate
75 75 entitySubtypesPromise = assetService.getAssetTypes({ignoreLoading: true});
76 76 } else if (scope.entityType == types.entityType.device) {
77 77 entitySubtypesPromise = deviceService.getDeviceTypes({ignoreLoading: true});
  78 + } else if (scope.entityType == types.entityType.entityView) {
  79 + entitySubtypesPromise = entityViewService.getEntityViewTypes({ignoreLoading: true});
78 80 }
79 81 if (entitySubtypesPromise) {
80 82 entitySubtypesPromise.then(
... ... @@ -100,6 +102,9 @@ export default function EntitySubtypeSelect($compile, $templateCache, $translate
100 102 } else if (scope.entityType == types.entityType.device) {
101 103 scope.entitySubtypeTitle = 'device.device-type';
102 104 scope.entitySubtypeRequiredText = 'device.device-type-required';
  105 + } else if (scope.entityType == types.entityType.entityView) {
  106 + scope.entitySubtypeTitle = 'entity-view.entity-view-type';
  107 + scope.entitySubtypeRequiredText = 'entity-view.entity-view-type-required';
103 108 }
104 109 scope.entitySubtypes.length = 0;
105 110 if (scope.entitySubtypesList && scope.entitySubtypesList.length) {
... ... @@ -116,6 +121,10 @@ export default function EntitySubtypeSelect($compile, $templateCache, $translate
116 121 scope.$on('deviceSaved', function() {
117 122 loadSubTypes();
118 123 });
  124 + } else if (scope.entityType == types.entityType.entityView) {
  125 + scope.$on('entityViewSaved', function() {
  126 + loadSubTypes();
  127 + });
119 128 }
120 129 }
121 130 }
... ...
... ... @@ -96,6 +96,7 @@ export default angular.module('thingsboard.help', [])
96 96 customers: helpBaseUrl + "/docs/user-guide/ui/customers",
97 97 assets: helpBaseUrl + "/docs/user-guide/ui/assets",
98 98 devices: helpBaseUrl + "/docs/user-guide/ui/devices",
  99 + entityViews: helpBaseUrl + "/docs/user-guide/ui/entity-views",
99 100 dashboards: helpBaseUrl + "/docs/user-guide/ui/dashboards",
100 101 users: helpBaseUrl + "/docs/user-guide/ui/users",
101 102 widgetsBundles: helpBaseUrl + "/docs/user-guide/ui/widget-library#bundles",
... ...
  1 +{
  2 + "access": {
  3 + "unauthorized": "Yetkisiz",
  4 + "unauthorized-access": "Yetkisiz Eriþim",
  5 + "unauthorized-access-text": "Bu kaynaða eriþmek için giriþ yapmalýsýnýz!",
  6 + "access-forbidden": "Eriþim Yasaklandý",
  7 + "access-forbidden-text": "Bu yere eriþim haklarýna sahip deðilsiniz! <br/> Bu konuma eriþmek istiyorsanýz yine de farklý kullanýcýlarla giriþ yapmayý deneyin.",
  8 + "refresh-token-expired": "Oturum süresi doldu",
  9 + "refresh-token-failed": "Seans yenilenemiyor"
  10 + },
  11 + "action": {
  12 + "activate": "Aktive Et",
  13 + "suspend": "Askýya Alma",
  14 + "save": "Kayýt etmek",
  15 + "saveAs": "Farklý kaydet",
  16 + "cancel": "Ýptal etmek",
  17 + "ok": "TAMAM",
  18 + "delete": "Sil",
  19 + "add": "Eklemek",
  20 + "yes": "Evet",
  21 + "no": "Yok hayýr",
  22 + "update": "Güncelleþtirme",
  23 + "remove": "Kaldýr",
  24 + "search": "Arama",
  25 + "clear-search": "Aramayý Temizle",
  26 + "assign": "Atamak",
  27 + "unassign": "Atamayý kaldýrma",
  28 + "share": "Pay",
  29 + "make-private": "Özel yap",
  30 + "apply": "Uygulamak",
  31 + "apply-changes": "Deðiþiklikleri uygula",
  32 + "edit-mode": "Düzenleme modu",
  33 + "enter-edit-mode": "Düzenleme moduna girin",
  34 + "decline-changes": "Deðiþiklikleri reddet",
  35 + "close": "Kapat",
  36 + "back": "Geri",
  37 + "run": "Koþmak",
  38 + "sign-in": "Oturum aç!",
  39 + "edit": "Düzenle",
  40 + "view": "Görünüm",
  41 + "create": "Yaratmak",
  42 + "drag": "Sürüklemek",
  43 + "refresh": "Yenile",
  44 + "undo": "Geri alma",
  45 + "copy": "Kopyala",
  46 + "paste": "Yapýþtýrmak",
  47 + "copy-reference": "Referansý kopyala",
  48 + "paste-reference": "Referansý yapýþtýr",
  49 + "import": "Ýthalat",
  50 + "export": "Ýhracat",
  51 + "share-via": "{{Provider}} aracýlýðýyla paylaþ"
  52 + },
  53 + "aggregation": {
  54 + "aggregation": "Toplama",
  55 + "function": "Veri toplama iþlevi",
  56 + "limit": "Maksimum deðerler",
  57 + "group-interval": "Gruplama aralýðý",
  58 + "min": "Min",
  59 + "max": "Max",
  60 + "avg": "Ortalama",
  61 + "sum": "Sum",
  62 + "count": "Sayý",
  63 + "none": "Yok"
  64 + },
  65 + "admin": {
  66 + "general": "Genel",
  67 + "general-settings": "Genel Ayarlar",
  68 + "outgoing-mail": "Giden posta",
  69 + "outgoing-mail-settings": "Giden Posta Ayarlarý",
  70 + "system-settings": "Sistem ayarlarý",
  71 + "test-mail-sent": "Test postasý baþarýyla gönderildi!",
  72 + "base-url": "Temel URL",
  73 + "base-url-required": "Temel URL gerekli.",
  74 + "mail-from": "Mail þu kiþiden geldi",
  75 + "mail-from-required": "Mail From gereklidir.",
  76 + "smtp-protocol": "SMTP protokolü",
  77 + "smtp-host": "SMTP ana bilgisayarý",
  78 + "smtp-host-required": "SMTP ana bilgisayarý gerekli.",
  79 + "smtp-port": "SMTP portu",
  80 + "smtp-port-required": "Bir smtp portu saðlamalýsýnýz.",
  81 + "smtp-port-invalid": "Bu geçerli bir smtp portuna benzemiyor.",
  82 + "timeout-msec": "Zaman aþýmý (msn)",
  83 + "timeout-required": "Zaman aþýmý gerekli.",
  84 + "timeout-invalid": "Bu geçerli bir zaman aþýmý gibi görünmüyor.",
  85 + "enable-tls": "TLS'yi etkinleþtir",
  86 + "send-test-mail": "Test postasý gönder"
  87 + },
  88 + "alarm": {
  89 + "alarm": "Alarm",
  90 + "alarms": "Alarmlar",
  91 + "select-alarm": "Alarmý seç",
  92 + "no-alarms-matching": " '{{Entity}}' ile eþleþen hiçbir alarm bulunamadý. ",
  93 + "alarm-required": "Alarm gerekli",
  94 + "alarm-status": "Alarm durumu",
  95 + "search-status": {
  96 + "ANY": "Herhangi biri",
  97 + "ACTIVE": "Aktif",
  98 + "CLEARED": "Temizlendi",
  99 + "ACK": "Kabul edilen",
  100 + "UNACK": "Unacknowledged"
  101 + },
  102 + "display-status": {
  103 + "ACTIVE_UNACK": "Aktif Olmamýþ",
  104 + "ACTIVE_ACK": "Aktif Kabul",
  105 + "CLEARED_UNACK": "Onaylanmamýþ Onaylandý",
  106 + "CLEARED_ACK": "Onaylandý onaylandý"
  107 + },
  108 + "no-alarms-prompt": "Alarm bulunamadý",
  109 + "created-time": "Zaman yaratýldý",
  110 + "type": "Tür",
  111 + "severity": "Önem",
  112 + "originator": "Originator",
  113 + "originator-type": "Gönderen türü",
  114 + "details": "Ayrýntýlar",
  115 + "status": "Durum",
  116 + "alarm-details": "Alarm detaylarý",
  117 + "start-time": "Baþlama zamaný",
  118 + "end-time": "Bitiþ zamaný",
  119 + "ack-time": "Onaylanmýþ zaman",
  120 + "clear-time": "Temizlenmiþ zaman",
  121 + "severity-critical": "Kritik",
  122 + "severity-major": "Majör",
  123 + "severity-minor": "Minör",
  124 + "severity-warning": "Uyarý",
  125 + "severity-indeterminate": "Belirsiz",
  126 + "acknowledge": "Onay",
  127 + "clear": "Açýk",
  128 + "search": "Arama alarmlarý",
  129 + "selected-alarms": "{count, çoðul, 1 {1 alarm} diðer {# alarm}} seçildi",
  130 + "no-data": "Gösterilecek bilgi yok",
  131 + "polling-interval": "Alarmlar sorgulama aralýðý (sn)",
  132 + "polling-interval-required": "Alarmlar sorgulama aralýðý gerekli.",
  133 + "min-polling-interval-message": "En az 1 san yoklama aralýðýna izin verilir.",
  134 + "aknowledge-alarms-title": "Kabul et { count, plural, 1 {1 alarm} other {# alarms}} selected ",
  135 + "aknowledge-alarms-text": "{Count, plural, 1 {1 alarm} dother {# alarm}} kimliðini kabul etmek istediðinizden emin misiniz?",
  136 + "clear-alarms-title": "Temizle {count, plural, 1 {1 alarm} other {# alarm}}",
  137 + "clear-alarms-text": "count, plural, 1 {1 alarm} diðer {# alarms}} silmek istediðinizden emin misiniz?"
  138 + },
  139 + "alias": {
  140 + "add": "Takma ad ekle",
  141 + "edit": "Takma adý düzenle",
  142 + "name": "Takma ad",
  143 + "name-required": "Takma ad gerekli",
  144 + "duplicate-alias": "Ayný ada sahip diðer ad zaten var.",
  145 + "filter-type-single-entity": "Tek varlýk",
  146 + "filter-type-entity-list": "Varlýk listesi",
  147 + "filter-type-entity-name": "Varlýk adý",
  148 + "filter-type-state-entity": "Gösterge panosu durumundan varlýk",
  149 + "filter-type-state-entity-description": "Gösterge panosu durum parametrelerinden alýnan varlýk",
  150 + "filter-type-asset-type": "Varlýk türü",
  151 + "filter-type-asset-type-description": "Öðe varlýklarý {{assetType}} '",
  152 + "filter-type-asset-type-and-name-description": "{{AssetType}} türündeki varlýklar ve {{prefix}} ile baþlayan adla ",
  153 + "filter-type-device-type": "Cihaz tipi",
  154 + "filter-type-device-type-description": "Türlü cihazlar {{deviceType}} ",
  155 + "filter-type-device-type-and-name-description": "{{DeviceType}} türündeki cihazlar ve {{prefix}} ile baþlayan adla ",
  156 + "filter-type-relations-query": "Ýliþkiler sorgusu",
  157 + "filter-type-relations-query-description": "{{relationType}} iliþkisine sahip {{direction}} {{rootEntity}} olan {{entities}}",
  158 + "filter-type-asset-search-query": "Öðe arama sorgusu",
  159 + "filter-type-asset-search-query-description": "{{RelationType}} iliþkisine sahip {{direction}}} türlerine sahip öðeler {{direction}} {{rootEntity}} ",
  160 + "filter-type-device-search-query": "Cihaz arama sorgusu",
  161 + "filter-type-device-search-query-description": "{{RelationType}} iliþkisi olan {{direction}} {{rootEntity}} içeren {{deviceTypes}} türüne sahip cihazlar",
  162 + "entity-filter": "Varlýk filtresi",
  163 + "resolve-multiple": "Birden çok varlýk olarak çöz",
  164 + "filter-type": "Filtre tipi",
  165 + "filter-type-required": "Filtre türü gerekli.",
  166 + "entity-filter-no-entity-matched": "Belirtilen filtreyle eþleþen varlýk bulunamadý.",
  167 + "no-entity-filter-specified": "Hiçbir varlýk filtresi belirtilmemiþ",
  168 + "root-state-entity": "Gösterge panosu durum varlýðýný root olarak kullan",
  169 + "root-entity": "Kök varlýk",
  170 + "state-entity-parameter-name": "Durum öðesi parametre adý",
  171 + "default-state-entity": "Varsayýlan durum varlýðý",
  172 + "default-entity-parameter-name": "Varsayýlan olarak",
  173 + "max-relation-level": "Maksimum iliþki seviyesi",
  174 + "unlimited-level": "Sýnýrsýz seviye",
  175 + "state-entity": "Gösterge panosu durumu",
  176 + "all-entities": "Tüm varlýklar",
  177 + "any-relation": "Herhangi"
  178 + },
  179 + "asset": {
  180 + "asset": "Varlýk",
  181 + "assets": "Varlýklar",
  182 + "management": "Varlýk Yönetimi",
  183 + "view-assets": "Varlýklarý Görüntüle",
  184 + "add": "Öðe Ekle",
  185 + "assign-to-customer": "Müþteriye atama",
  186 + "assign-asset-to-customer": "Müþteriye Varlýk (lar) Atama",
  187 + "assign-asset-to-customer-text": "Lütfen müþteriye atamak için varlýklarý seçin",
  188 + "no-assets-text": "Öðe bulunamadý",
  189 + "assign-to-customer-text": "Lütfen varlýk (lar) a atamak için müþteriyi seçin",
  190 + "public": "Halka açýk",
  191 + "assignedToCustomer": "Müþteriye atandý",
  192 + "make-public": "Varlýðý herkese açýk yap",
  193 + "make-private": "Varlýk özel yap",
  194 + "unassign-from-customer": "Müþteriden atama",
  195 + "delete": "Öðeyi sil",
  196 + "asset-public": "Varlýk halka açýk",
  197 + "asset-type": "Varlýk türü",
  198 + "asset-type-required": "Öðe türü gerekli.",
  199 + "select-asset-type": "Varlýk türünü seç",
  200 + "enter-asset-type": "Varlýk türünü girin",
  201 + "any-asset": "Herhangi bir varlýk",
  202 + "no-asset-types-matching": "{{EntitySubtype}} ile eþleþen öðe türü bulunamadý. ",
  203 + "asset-type-list-empty": "Hiçbir öðe türü seçilmedi.",
  204 + "asset-types": "Varlýk türleri",
  205 + "name": "Ad",
  206 + "name-required": "Ýsim gerekli.",
  207 + "description": "Açýklama",
  208 + "type": "Tür",
  209 + "type-required": "Tür gerekli.",
  210 + "details": "Ayrýntýlar",
  211 + "events": "Etkinlikler",
  212 + "add-asset-text": "Yeni varlýk ekle",
  213 + "asset-details": "Varlýk ayrýntýlarý",
  214 + "assign-assets": "Öðeleri atama",
  215 + "assign-assets-text": "Müþteriye {count, çoðul, 1 {1 asset}} diðer {# asset}} atayýn ",
  216 + "delete-assets": "Öðeleri sil",
  217 + "unassign-assets": "Varlýklarý atama",
  218 + "unassign-assets-action-title": "Müþteriden tekil, sayým, çoðul, 1 {1 öðe} diðer {# varlýk}} atama ",
  219 + "assign-new-asset": "Yeni varlýk atama",
  220 + "delete-asset-title": "Öðeyi {{assetName}} silmek istediðinizden emin misiniz?",
  221 + "delete-asset-text": "Dikkatli olun, onaylandýktan sonra varlýk ve ilgili tüm veriler kurtarýlamaz.",
  222 + "delete-assets-title": "{Count, çoðul, 1 {1 asset} diðer {# asset}} silmek istediðinizden emin misiniz?",
  223 + "delete-assets-action-title": "Sil {count, çoðul, 1 {1 asset} diðer {# asset}}",
  224 + "delete-assets-text": "Dikkatli olun, onaylandýktan sonra tüm seçilmiþ varlýklar kaldýrýlacak ve ilgili tüm veriler kurtarýlamayacaktýr.",
  225 + "make-public-asset-title": "Varlýðý {{assetName}} herkese açýk yapmak istediðinizden emin misiniz? ",
  226 + "make-public-asset-text": "Onaydan sonra varlýk ve tüm verileri kamuya açýk ve baþkalarý tarafýndan eriþilebilir olacak.",
  227 + "make-private-asset-title": "Öðeyi {{assetName}} özel yapmak istediðinizden emin misiniz? ",
  228 + "make-private-asset-text": "Onaydan sonra varlýk ve tüm verileri gizli tutulacak ve baþkalarý tarafýndan eriþilemeyecektir.",
  229 + "unassign-asset-title": "Öðeyi {{assetName}} öðesinin atamasýný kaldýrmak istediðinizden emin misiniz?",
  230 + "unassign-asset-text": "Onaydan sonra varlýk atama yapýlmayacak ve müþteri tarafýndan eriþilemeyecektir.",
  231 + "unassign-asset": "Atama atamayý kaldýr",
  232 + "unassign-assets-title": "count, plural, 1 {1 öðe} other {# asset}} atama atamak istediðinizden emin misiniz?",
  233 + "unassign-assets-text": "Onaylandýktan sonra, seçilen tüm öðeler atamadan kaldýrýlacak ve müþteri tarafýndan eriþilemeyecektir.",
  234 + "copyId": "Öðe Kimliði Kopyala",
  235 + "idCopiedMessage": "Öðe Kimliði panoya kopyalandý",
  236 + "select-asset": "Varlýk seç",
  237 + "no-assets-matching": "{{Entity}} ile eþleþen hiçbir öðe bulunamadý. ",
  238 + "asset-required": "Varlýk gerekli",
  239 + "name-starts-with": "Öðe adý ile baþlar"
  240 + },
  241 + "attribute": {
  242 + "attributes": "Öznitellikler",
  243 + "latest-telemetry": "Son telemetri",
  244 + "attributes-scope": "Varlýk öznitelikleri kapsamý",
  245 + "scope-latest-telemetry": "Son telemetri",
  246 + "scope-client": "Müþteri özellikleri",
  247 + "scope-server": "Sunucu öznitelikleri",
  248 + "scope-shared": "Paylaþýlan özellikler",
  249 + "add": "Özellik ekle",
  250 + "key": "Anahtar",
  251 + "last-update-time": "Son güncelleme zamaný",
  252 + "key-required": "Özellik anahtarý gerekli.",
  253 + "value": "Deðer",
  254 + "value-required": "Özellik deðeri gerekli.",
  255 + "delete-attributes-title": "{Count, çoðul, 1 {1 özellik} diðer {# attributes}} silmek istediðinizden emin misiniz?",
  256 + "delete-attributes-text": "Dikkatli olun, onaylandýktan sonra tüm seçilen özellikler kaldýrýlacak.",
  257 + "delete-attributes": "Nitelikleri sil",
  258 + "enter-attribute-value": "Özellik deðerini girin",
  259 + "show-on-widget": "Widget’ta göster",
  260 + "widget-mode": "Widget modu",
  261 + "next-widget": "Sonraki pencere öðesi",
  262 + "prev-widget": "Önceki pencere öðesi",
  263 + "add-to-dashboard": "Gösterge tablosuna ekle",
  264 + "add-widget-to-dashboard": "Gösterge tablosuna widget ekle",
  265 + "selected-attributes": "{count, çoðul, 1 {1 özellik} diðer {# attributes}} seçildi",
  266 + "selected-telemetry": "{count, çoðul, 1 {1 telemetri birimi} diðer {# telemetri birimleri}} seçildi"
  267 + },
  268 + "audit-log": {
  269 + "audit": "Denetim",
  270 + "audit-logs": "Denetim Kayýtlarý",
  271 + "timestamp": "Zaman Damgasý",
  272 + "entity-type": "Varlýk Türü",
  273 + "entity-name": "Varlýk adý",
  274 + "user": "Kullanýcý",
  275 + "type": "Tür",
  276 + "status": "Durum",
  277 + "details": "Ayrýntýlar",
  278 + "type-added": "Katma",
  279 + "type-deleted": "Silindi",
  280 + "type-updated": "Güncellenmiþ",
  281 + "type-attributes-updated": "Öznitelikler güncellendi",
  282 + "type-attributes-deleted": "Öznitelikler silindi",
  283 + "type-rpc-call": "RPC çaðrýsý",
  284 + "type-credentials-updated": "Kimlik bilgileri güncellendi",
  285 + "type-assigned-to-customer": "Müþteriye Atanan",
  286 + "type-unassigned-from-customer": "Müþteriden Atanmamýþ",
  287 + "type-activated": "Aktif",
  288 + "type-suspended": "Askýya alýndý",
  289 + "type-credentials-read": "Kimlik bilgileri okundu",
  290 + "type-attributes-read": "Nitelikler oku",
  291 + "type-relation-add-or-update": "Ýliþki güncellendi",
  292 + "type-relation-delete": "Ýliþki silindi",
  293 + "type-relations-delete": "Tüm iliþki silindi",
  294 + "type-alarm-ack": "Kabul edilen",
  295 + "type-alarm-clear": "Temizlendi",
  296 + "status-success": "Baþarý",
  297 + "status-failure": "Baþarýsýzlýk",
  298 + "audit-log-details": "Denetim kaydý ayrýntýlarý",
  299 + "no-audit-logs-prompt": "Kayýt bulunamadý",
  300 + "action-data": "Eylem verileri",
  301 + "failure-details": "Arýza detaylarý",
  302 + "search": "Denetim günlüklerini ara",
  303 + "clear-search": "Aramayý Temizle"
  304 + },
  305 + "confirm-on-exit": {
  306 + "message": "Kaydedilmemiþ deðiþiklikleriniz var. Bu sayfadan ayrýlmak istediðinizden emin misiniz?",
  307 + "html-message": "Kaydedilmemiþ deðiþiklikleriniz var. <br/> Bu sayfadan ayrýlmak istediðinizden emin misiniz?",
  308 + "title": "Kaydedilmemiþ deðiþiklikler"
  309 + },
  310 + "contact": {
  311 + "country": "Ülke",
  312 + "city": "Kent",
  313 + "state": "Eyalet / Ýl",
  314 + "postal-code": "Posta Kodu",
  315 + "postal-code-invalid": "Geçersiz Posta Kodu / Posta Kodu biçimi.",
  316 + "address": "Adres",
  317 + "address2": "Adres 2",
  318 + "phone": "Telefon",
  319 + "email": "E-posta",
  320 + "no-address": "Adres yok"
  321 + },
  322 + "common": {
  323 + "username": "Kullanýcý adý",
  324 + "password": "Parola",
  325 + "enter-username": "Kullanýcý adý girin",
  326 + "enter-password": "Parolaný Gir",
  327 + "enter-search": "Arama girin"
  328 + },
  329 + "content-type": {
  330 + "json": "Json",
  331 + "text": "Metin",
  332 + "binary": "Ýkili (Base64)"
  333 + },
  334 + "customer": {
  335 + "customer": "Müþteri",
  336 + "customers": "Müþteriler",
  337 + "management": "Müþteri yönetimi",
  338 + "dashboard": "Müþteri Kontrol Paneli",
  339 + "dashboards": "Müþteri Kontrol Panelleri",
  340 + "devices": "Müþteri Cihazlarý",
  341 + "entity-views": "Müþteri Varlýðý Görüntüleme Sayýsý",
  342 + "assets": "Müþteri Varlýklarý",
  343 + "public-dashboards": "Genel Panolar",
  344 + "public-devices": "Kamu Aygýtlarý",
  345 + "public-assets": "Kamu Varlýklarý",
  346 + "public-entity-views": "Kamu Varlýk Görüntüleme Sayýsý",
  347 + "add": "Müþteri Ekle",
  348 + "delete": "Müþteriyi sil",
  349 + "manage-customer-users": "Müþteri kullanýcýlarýný yönet",
  350 + "manage-customer-devices": "Müþteri cihazlarýný yönet",
  351 + "manage-customer-dashboards": "Müþteri panolarýný yönet",
  352 + "manage-public-devices": "Genel cihazlarý yönet",
  353 + "manage-public-dashboards": "Genel panolarý yönet",
  354 + "manage-customer-assets": "Müþteri varlýklarýný yönet",
  355 + "manage-public-assets": "Kamu varlýklarýný yönet",
  356 + "add-customer-text": "Yeni müþteri ekle",
  357 + "no-customers-text": "Müþteri bulunamadý",
  358 + "customer-details": "Müþteri detaylarý",
  359 + "delete-customer-title": "Müþteriyi silmek istediðinizden emin misiniz? {{CustomerTitle}} ? ",
  360 + "delete-customer-text": "Dikkatli olun, onaylandýktan sonra müþteri ve ilgili tüm veriler kurtarýlamaz.",
  361 + "delete-customers-title": "{Count, çoðul, 1 {1 müþteri} diðer {# customers}} silmek istediðinizden emin misiniz?",
  362 + "delete-customers-action-title": "Sil {count, çoðul, 1 {1 müþteri} diðer {# customers}}",
  363 + "delete-customers-text": "Dikkatli olun, onaylandýktan sonra tüm seçilen müþteriler kaldýrýlacak ve ilgili tüm veriler kurtarýlamayacaktýr.",
  364 + "manage-users": "Kullanýcýlarý Yönet",
  365 + "manage-assets": "Varlýklarý yönet",
  366 + "manage-devices": "Cihazlarý yönet",
  367 + "manage-dashboards": "Gösterge tablolarýný yönet",
  368 + "title": "Baþlýk",
  369 + "title-required": "Baþlýk gerekli.",
  370 + "description": "Açýklama",
  371 + "details": "Ayrýntýlar",
  372 + "events": "Etkinlikler",
  373 + "copyId": "Müþteri kimliðini kopyala",
  374 + "idCopiedMessage": "Müþteri Kimliði panoya kopyalandý",
  375 + "select-customer": "Müþteriyi seç",
  376 + "no-customers-matching": "{{Entity}} ile eþleþen hiçbir müþteri bulunamadý. ",
  377 + "customer-required": "Müþteri gerekli",
  378 + "select-default-customer": "Varsayýlan müþteriyi seç",
  379 + "default-customer": "Varsayýlan müþteri",
  380 + "default-customer-required": "Kiracý düzeyinde gösterge tablosunda hata ayýklamak için varsayýlan müþteri gerekiyor"
  381 + },
  382 + "datetime": {
  383 + "date-from": "Tarih",
  384 + "time-from": "Zaman",
  385 + "date-to": "Tarih",
  386 + "time-to": "Zaman"
  387 + },
  388 + "dashboard": {
  389 + "dashboard": "Pano",
  390 + "dashboards": "Gösterge tablolarý",
  391 + "management": "Gösterge tablosu yönetimi",
  392 + "view-dashboards": "Gösterge Panolarý",
  393 + "add": "Gösterge Tablosu Ekle",
  394 + "assign-dashboard-to-customer": "Gösterge Tablosunu / Müþterilerini Müþteriye Atama",
  395 + "assign-dashboard-to-customer-text": "Lütfen müþteriye atamak için kontrol panellerini seçin",
  396 + "assign-to-customer-text": "Gösterge panellerini atamak için lütfen müþteriyi seçiniz",
  397 + "assign-to-customer": "Müþteriye atama",
  398 + "unassign-from-customer": "Müþteriden atama",
  399 + "make-public": "Gösterge panosunu herkese açýk yap",
  400 + "make-private": "Gösterge panosunu özel yap",
  401 + "manage-assigned-customers": "Atanan müþterileri yönet",
  402 + "assigned-customers": "Atanan müþteriler",
  403 + "assign-to-customers": "Gösterge Tablosunu / Müþterilerini Müþterilere Atama",
  404 + "assign-to-customers-text": "Lütfen gösterge panosunu atamak için müþterileri seçin",
  405 + "unassign-from-customers": "Müþterilerden Gösterge Tablosunu (Notlarýný) Atama",
  406 + "unassign-from-customers-text": "Lütfen gösterge tablosundan atamak için müþterileri seçin",
  407 + "no-dashboards-text": "Gösterge panelleri bulunamadý",
  408 + "no-widgets": "Hiçbir widget yapýlandýrýlmamýþ",
  409 + "add-widget": "Yeni pencere öðesi ekle",
  410 + "title": "Baþlýk",
  411 + "select-widget-title": "Widget seç",
  412 + "select-widget-subtitle": "Kullanýlabilir pencere öðesi türleri",
  413 + "delete": "Gösterge tablosunu sil",
  414 + "title-required": "Baþlýk gerekli.",
  415 + "description": "Açýklama",
  416 + "details": "Ayrýntýlar",
  417 + "dashboard-details": "Gösterge tablosu ayrýntýlarý",
  418 + "add-dashboard-text": "Yeni gösterge tablosu ekle",
  419 + "assign-dashboards": "Gösterge tablolarý atama",
  420 + "assign-new-dashboard": "Yeni kontrol paneli atama",
  421 + "assign-dashboards-text": "Müþterilere {count, plural, 1 {1 gösterge tablosu} other {# dashboard}}} atayýn ",
  422 + "unassign-dashboards-action-text": "Müþterilerden atama sayýsý, çoðul, 1 {1 gösterge tablosu} diðer {# panolar}}",
  423 + "delete-dashboards": "Gösterge tablolarýný sil",
  424 + "unassign-dashboards": "Atanmamýþ gösterge panolarý",
  425 + "unassign-dashboards-action-title": "Müþteriden atama sayým, çoðul, 1 {1 gösterge tablosu} diðer {# pano}}",
  426 + "delete-dashboard-title": "'{{DashboardTitle}}' gösterge panosunu silmek istediðinizden emin misiniz?",
  427 + "delete-dashboard-text": "Doðrulamadan sonra, kontrol panelinden ve ilgili tüm veriler kurtarýlamayacak.",
  428 + "delete-dashboards-title": "{Count, çoðul, 1 {1 dashboard} diðer {# dashboards}} silmek istediðinizden emin misiniz?",
  429 + "delete-dashboards-action-title": "Sil {count, çoðul, 1 {1 gösterge tablosu} diðer {# dashboard}}",
  430 + "delete-dashboards-text": "Dikkatli olun, onaylandýktan sonra tüm seçilen panolar silinecek ve ilgili tüm veriler kurtarýlamayacaktýr.",
  431 + "unassign-dashboard-title": "{{DashboardTitle}} gösterge tablosunun atamasýný kaldýrmak istediðinizden emin misiniz? ",
  432 + "unassign-dashboard-text": "Onaydan sonra gösterge panosu atanmamýþ olacak ve müþteri tarafýndan eriþilemeyecektir.",
  433 + "unassign-dashboard": "Atanmamýþ gösterge panosu",
  434 + "unassign-dashboards-title": "Sayým, çoðul, 1 {1 gösterge tablosu} diðer {# panolar}} atama atamak istediðinizden emin misiniz?",
  435 + "unassign-dashboards-text": "Onaydan sonra tüm seçilen gösterge tablolarý atanmamýþ olacak ve müþteri tarafýndan eriþilemeyecektir.",
  436 + "public-dashboard-title": "Gösterge panosu artýk herkese açýk",
  437 + "public-dashboard-text": "Gösterge tablonuz <b> {{dashboardTitle}} </ b> artýk herkese açýk ve bir sonraki herkese açýk <a href='{{publicLink}} 'target='_blank'> link </a> üzerinden eriþilebilir:",
  438 + "public-dashboard-notice": "<b> Not: </ b> Verilerine eriþmek için ilgili cihazlarý herkese açýk yapmayý unutmayýn.",
  439 + "make-private-dashboard-title": "Gösterge panosunu {{dashboardTitle}} özel yapmak istediðinizden emin misiniz?",
  440 + "make-private-dashboard-text": "Onaydan sonra gösterge panosu özel hale getirilecek ve baþkalarý tarafýndan eriþilemeyecektir.",
  441 + "make-private-dashboard": "Gösterge panosunu özel yap",
  442 + "socialshare-text": "'{{dashboardTitle}} ThingsBoard tarafýndan desteklenmektedir ",
  443 + "socialshare-title": "'{{dashboardTitle}} ThingsBoard tarafýndan desteklenmektedir ",
  444 + "select-dashboard": "Gösterge tablosu seç",
  445 + "no-dashboards-matching": "{{Entity}} ile eþleþen hiçbir gösterge paneli bulunamadý. ",
  446 + "dashboard-required": "Gösterge tablosu gerekli.",
  447 + "select-existing": "Mevcut kontrol panelini seç",
  448 + "create-new": "Yeni gösterge tablosu oluþtur",
  449 + "new-dashboard-title": "Yeni kontrol paneli baþlýðý",
  450 + "open-dashboard": "Açýk kontrol paneli",
  451 + "set-background": "Arka planý ayarla",
  452 + "background-color": "Arka plan rengi",
  453 + "background-image": "Arka plan görüntüsü",
  454 + "background-size-mode": "Arka plan boyut modu",
  455 + "no-image": "Görüntü seçilmedi",
  456 + "drop-image": "Bir resim býrakýn veya yüklenecek bir dosya seçmek için týklayýn.",
  457 + "settings": "Ayarlar",
  458 + "columns-count": "Sütun sayýsý",
  459 + "columns-count-required": "Sütun sayýsý gerekli.",
  460 + "min-columns-count-message": "Sadece 10 minimum sütun sayýsýna izin verilir.",
  461 + "max-columns-count-message": "Sadece maksimum 1000 sütun sayýsýna izin verilir.",
  462 + "widgets-margins": "Widget arasýnda marj",
  463 + "horizontal-margin": "Yatay kenar boþluðu",
  464 + "horizontal-margin-required": "Yatay kenar boþluðu deðeri gerekiyor.",
  465 + "min-horizontal-margin-message": "Minimum yatay kenar boþluðu deðeri olarak yalnýzca 0'a izin verilir.",
  466 + "max-horizontal-margin-message": "Maksimum yatay kenar boþluðu deðeri olarak yalnýzca 50'ye izin verilir.",
  467 + "vertical-margin": "Dikey kenar boþluðu",
  468 + "vertical-margin-required": "Dikey kenar boþluðu deðeri gereklidir.",
  469 + "min-vertical-margin-message": "En az dikey kenar boþluðu deðeri olarak yalnýzca 0'a izin verilir.",
  470 + "max-vertical-margin-message": "Maksimum dikey kenar boþluðu deðeri olarak yalnýzca 50'ye izin verilir.",
  471 + "autofill-height": "Otomatik dolgu düzeni yüksekliði",
  472 + "mobile-layout": "Mobil düzen ayarlarý",
  473 + "mobile-row-height": "Mobil satýr yüksekliði, px",
  474 + "mobile-row-height-required": "Mobil satýr yüksekliði deðeri gerekli.",
  475 + "min-mobile-row-height-message": "Minimum mobil satýr yüksekliði deðeri olarak yalnýzca 5 piksele izin verilir.",
  476 + "max-mobile-row-height-message": "Maksimum mobil satýr yüksekliði deðeri olarak yalnýzca 200 piksele izin verilir.",
  477 + "display-title": "Gösterge tablosu baþlýðý",
  478 + "toolbar-always-open": "Araç çubuðunu açýk tut",
  479 + "title-color": "Baþlýk rengi",
  480 + "display-dashboards-selection": "Gösterge panolarý seçimi",
  481 + "display-entities-selection": "Öðe varlýklarý seçimi",
  482 + "display-dashboard-timewindow": "Zaman penceresi göster",
  483 + "display-dashboard-export": "Görüntülü dýþa aktarma",
  484 + "import": "Gösterge paneli",
  485 + "export": "Dýþ gösterge panosu",
  486 + "export-failed-error": "Gösterge tablosu gönderilemiyor: {{error}}",
  487 + "create-new-dashboard": "Yeni gösterge tablosu oluþtur",
  488 + "dashboard-file": "Gösterge tablosu dosyasý",
  489 + "invalid-dashboard-file-error": "Gösterge tablosu alýnamadý: Geçersiz pano veri yapýsý.",
  490 + "dashboard-import-missing-aliases-title": "Alýnan pano tarafýndan kullanýlan takma adlarý yapýlandýr",
  491 + "create-new-widget": "Yeni pencere öðesi oluþtur",
  492 + "import-widget": "Widget'ý içe aktar",
  493 + "widget-file": "Widget dosyasý",
  494 + "invalid-widget-file-error": "Widget içe aktarýlamýyor: Geçersiz pencere öðesi yapýsý.",
  495 + "widget-import-missing-aliases-title": "Ýçe aktarýlan pencere aracý tarafýndan kullanýlan takma adlarý yapýlandýr",
  496 + "open-toolbar": "Gösterge tablosu araç çubuðunu aç",
  497 + "close-toolbar": "Araç çubuðunu kapat",
  498 + "configuration-error": "Yapýlandýrma hatasý",
  499 + "alias-resolution-error-title": "Gösterge tablosu takma ad hatasý",
  500 + "invalid-aliases-config": "Diðer ad filtrelerinin biriyle eþleþen herhangi bir cihaz bulunamadý. <br/> Lütfen bu sorunu çözmek için yöneticinize baþvurun.",
  501 + "select-devices": "Cihaz seç",
  502 + "assignedToCustomer": "Müþteriye atandý",
  503 + "assignedToCustomers": "Müþterilere atandý",
  504 + "public": "Halka açýk",
  505 + "public-link": "Genel baðlantý",
  506 + "copy-public-link": "Genel baðlantýyý kopyala",
  507 + "public-link-copied-message": "Pano genel baðlantýsý panoya kopyalandý",
  508 + "manage-states": "Gösterge panosu durumlarýný yönet",
  509 + "states": "Gösterge durumu",
  510 + "search-states": "Gösterge panosu durumlarý",
  511 + "selected-states": "{count, çoðul, 1 {1 gösterge tablosu durumu} diðer {# dashboard durumlarý}} seçildi",
  512 + "edit-state": "Gösterge panosu durumunu düzenle",
  513 + "delete-state": "Gösterge panosu durumunu sil",
  514 + "add-state": "Gösterge paneli durumu",
  515 + "state": "Gösterge paneli durumu",
  516 + "state-name": "Ad",
  517 + "state-name-required": "Gösterge panosu durum adý gerekli.",
  518 + "state-id": "Durum Kimliði",
  519 + "state-id-required": "Gösterge durumu kimliði gerekiyor.",
  520 + "state-id-exists": "Ayný kimliðe sahip gösterge tablosu zaten var.",
  521 + "is-root-state": "Kök devlet",
  522 + "delete-state-title": "Gösterge panosu durumunu sil",
  523 + "delete-state-text": "Gösterge panosu durumunu '{{stateName}}' adýyla silmek istediðinizden emin misiniz?",
  524 + "show-details": "Detaylarý göster",
  525 + "hide-details": "Detaylarý gizle",
  526 + "select-state": "Hedef durumu seç",
  527 + "state-controller": "Durum kontrolörü"
  528 + },
  529 + "datakey": {
  530 + "settings": "Ayarlar",
  531 + "advanced": "Ýleri",
  532 + "label": "Etiket",
  533 + "color": "Renk",
  534 + "units": "Deðerin yanýnda gösterilecek özel sembol",
  535 + "decimals": "Kayan noktadan sonraki basamak sayýsý",
  536 + "data-generation-func": "Veri oluþturma iþlevi",
  537 + "use-data-post-processing-func": "Veri iþleme sonrasý iþlevini kullan",
  538 + "configuration": "Veri anahtarý yapýlandýrmasý",
  539 + "timeseries": "Zaman serisi",
  540 + "attributes": "Öznitellikler",
  541 + "alarm": "Alarm alanlarý",
  542 + "timeseries-required": "Varlýk zamanlamalarý gerekli.",
  543 + "timeseries-or-attributes-required": "Varlýk zaman çizelgeleri / öznitelikler gereklidir.",
  544 + "maximum-timeseries-or-attributes": "Maksimum {count, çoðul, 1 {1 timeseries / attribute.} Diðer {# timeseries / attributes {} izin verilir}}",
  545 + "alarm-fields-required": "Alarm alanlarý gerekli.",
  546 + "function-types": "Ýþlev türleri",
  547 + "function-types-required": "Ýþlev tipleri gereklidir.",
  548 + "maximum-function-types": "Maksimum {sayým, çoðul, 1 {1 iþlev türüne izin verilir.} Diðer {# iþlev türlerine izin verilir}}"
  549 + },
  550 + "datasource": {
  551 + "type": "Veri kaynaðý türü",
  552 + "name": "Ad",
  553 + "add-datasource-prompt": "Lütfen veri kaynaðý ekle"
  554 + },
  555 + "details": {
  556 + "edit-mode": "Düzenleme modu",
  557 + "toggle-edit-mode": "Düzenleme modunu deðiþtir"
  558 + },
  559 + "device": {
  560 + "device": "Cihaz",
  561 + "device-required": "Cihaz gerekli.",
  562 + "devices": "Cihazlar",
  563 + "management": "Cihaz yönetimi",
  564 + "view-devices": "Cihazlarý Görüntüle",
  565 + "device-alias": "Cihaz takma adý",
  566 + "aliases": "Cihaz takma adlarý",
  567 + "no-alias-matching": "'{{alias}} bulunamadý. ",
  568 + "no-aliases-found": "Takma ad bulunamadý",
  569 + "no-key-matching": "'{{anahtar bulunamadý.",
  570 + "no-keys-found": "Anahtar bulunamadý.",
  571 + "create-new-alias": "Yeni bir tane oluþtur!",
  572 + "create-new-key": "Yeni bir tane oluþtur!",
  573 + "duplicate-alias-error": "Yinelenen takma ad bulundu {{alias}}.. <br> Cihaz takma adlarý, kontrol panelinde benzersiz olmalýdýr. ",
  574 + "configure-alias": "Yapýlandýrma {{alias}} takma ad",
  575 + "no-devices-matching": "{{Entity}} ile eþleþen hiçbir cihaz bulunamadý. ",
  576 + "alias": "Alias",
  577 + "alias-required": "Cihaz takma adý gerekiyor.",
  578 + "remove-alias": "Cihaz takma adýný kaldýr",
  579 + "add-alias": "Cihaz takma adý ekle",
  580 + "name-starts-with": "Cihaz adý ile baþlýyor",
  581 + "device-list": "Aygýt listesi",
  582 + "use-device-name-filter": "Filtre kullan",
  583 + "device-list-empty": "Cihaz seçilmedi.",
  584 + "device-name-filter-required": "Cihaz adý filtresi gerekli.",
  585 + "device-name-filter-no-device-matched": "{{Device}} ile baþlayan hiçbir cihaz bulunamadý. ",
  586 + "add": "Cihaz ekle",
  587 + "assign-to-customer": "Müþteriye atama",
  588 + "assign-device-to-customer": "Aygýtý / Aygýtlarý Müþteriye Atama",
  589 + "assign-device-to-customer-text": "Lütfen müþteriye atamak istediðiniz cihazlarý seçin",
  590 + "make-public": "Cihazý herkese açýk yap",
  591 + "make-private": "Cihazý özel yap",
  592 + "no-devices-text": "Hiçbir cihaz bulunamadý",
  593 + "assign-to-customer-text": "Lütfen cihazý atamak için müþteriyi seçin",
  594 + "device-details": "Cihaz detaylarý",
  595 + "add-device-text": "Yeni cihaz ekle",
  596 + "credentials": "Kimlik bilgileri",
  597 + "manage-credentials": "Kimlik bilgilerini yönet",
  598 + "delete": "Cihazý sil",
  599 + "assign-devices": "Aygýtlarý atama",
  600 + "assign-devices-text": "Müþteriye {count, çoðul, 1 {1 cihaz} diðer {# devices}} atayýn ",
  601 + "delete-devices": "Cihazlarý sil",
  602 + "unassign-from-customer": "Müþteriden atama",
  603 + "unassign-devices": "Atanmamýþ cihazlarý",
  604 + "unassign-devices-action-title": "Müþteriden atayýn, çoðul, 1 {1 cihaz} diðer {# devices}} atama ",
  605 + "assign-new-device": "Yeni cihaz atama",
  606 + "make-public-device-title": "Cihazý {{deviceName}} herkese açýk yapmak istediðinizden emin misiniz?",
  607 + "make-public-device-text": "Onaydan sonra cihaz ve tüm verileri kamuya açýk ve baþkalarý tarafýndan eriþilebilir olacak.",
  608 + "make-private-device-title": "Cihazý {{deviceName}} özel yapmak istediðinizden emin misiniz?",
  609 + "make-private-device-text": "Onaylandýktan sonra cihaz ve tüm verileri gizli tutulacak ve baþkalarý tarafýndan eriþilemeyecektir.",
  610 + "view-credentials": "Kimlik bilgilerini görüntüle",
  611 + "delete-device-title": "Cihazý {{deviceName}} silmek istediðinizden emin misiniz? ",
  612 + "delete-device-text": "Dikkatli olun, onaylamadan sonra cihaz ve tüm ilgili veriler kurtarýlamaz.",
  613 + "delete-devices-title": "{Count, çoðul, 1 {1 cihaz} diðer {# devices}} silmek istediðinizden emin misiniz?",
  614 + "delete-devices-action-title": "Sil {count, çoðul, 1 {1 cihaz} diðer {# devices}}",
  615 + "delete-devices-text": "Dikkatli olun, onaylandýktan sonra tüm seçilen cihazlar silinecek ve ilgili tüm veriler kurtarýlamayacaktýr.",
  616 + "unassign-device-title": "Cihazýn atamasýný kaldýrmak istediðinizden emin misiniz? {{DeviceName}} ? ",
  617 + "unassign-device-text": "Onaydan sonra cihaz atanmamýþ olacak ve müþteri tarafýndan eriþilemeyecektir.",
  618 + "unassign-device": "Atanmamýþ cihaz",
  619 + "unassign-devices-title": "Sayým, çoðul, 1 {1 cihaz} diðer {# devices}} atama atamak istediðinizden emin misiniz?",
  620 + "unassign-devices-text": "Onaylandýktan sonra tüm seçilen cihazlar atanmamýþ olacak ve müþteri tarafýndan eriþilemeyecektir.",
  621 + "device-credentials": "Cihaz Kimlik Bilgileri",
  622 + "credentials-type": "Kimlik bilgileri türü",
  623 + "access-token": "Eriþim belirteci",
  624 + "access-token-required": "Eriþim belirteci gerekli.",
  625 + "access-token-invalid": "Eriþim belirteci uzunluðu 1 ile 20 karakter arasýnda olmalýdýr.",
  626 + "rsa-key": "RSA ortak anahtarý",
  627 + "rsa-key-required": "RSA ortak anahtarý gerekli.",
  628 + "secret": "Gizli",
  629 + "secret-required": "Gizli gerekli",
  630 + "device-type": "Cihaz tipi",
  631 + "device-type-required": "Cihaz tipi gerekli.",
  632 + "select-device-type": "Cihaz türünü seç",
  633 + "enter-device-type": "Cihaz türünü girin",
  634 + "any-device": "Herhangi bir cihaz",
  635 + "no-device-types-matching": "{{EntitySubtype}} ile eþleþen cihaz türü bulunamadý. ",
  636 + "device-type-list-empty": "Hiçbir cihaz türü seçilmedi.",
  637 + "device-types": "Cihaz türleri",
  638 + "name": "Ad",
  639 + "name-required": "Ýsim gerekli.",
  640 + "description": "Açýklama",
  641 + "events": "Etkinlikler",
  642 + "details": "Ayrýntýlar",
  643 + "copyId": "Cihaz kimliðini kopyala",
  644 + "copyAccessToken": "Eriþim belirteci kopyala",
  645 + "idCopiedMessage": "Cihaz Kimliði panoya kopyalandý",
  646 + "accessTokenCopiedMessage": "Cihaz eriþim belirteci panoya kopyalandý",
  647 + "assignedToCustomer": "Müþteriye atandý",
  648 + "unable-delete-device-alias-title": "Cihaz takma adý silinemiyor",
  649 + "unable-delete-device-alias-text": "Cihaz takma adý {{deviceAlias}} ', þu widget (lar) tarafýndan kullanýldýðý þekliyle silinemiyor: <br/> {{widgetsList}} ",
  650 + "is-gateway": "Að geçidi",
  651 + "public": "Halka açýk",
  652 + "device-public": "Cihaz herkese açýk",
  653 + "select-device": "Cihaz seç"
  654 + },
  655 + "dialog": {
  656 + "close": "Ýletiþim kutusunu kapat"
  657 + },
  658 + "error": {
  659 + "unable-to-connect": "Sunucuya baðlanýlamýyor! Lütfen Ýnternet baðlantýnýzý kontrol edin.",
  660 + "unhandled-error-code": "Ýþlenmemiþ hata kodu: {{errorCode}}",
  661 + "unknown-error": "Bilinmeyen hata"
  662 + },
  663 + "entity": {
  664 + "entity": "Varlýk",
  665 + "entities": "Varlýklarý",
  666 + "aliases": "Varlýk takma adlarý",
  667 + "entity-alias": "Varlýk takma adý",
  668 + "unable-delete-entity-alias-title": "Varlýk takma adý silinemiyor",
  669 + "unable-delete-entity-alias-text": "Varlýk takma adý {{entityAlias}} ', þu widget (lar) tarafýndan kullanýldýðý þekliyle silinemez: <br/> {{widgetsList}} ",
  670 + "duplicate-alias-error": "Yinelenen takma ad bulundu {{alias}} '... Entity takma adlar, gösterge panosunda benzersiz olmalýdýr. ",
  671 + "missing-entity-filter-error": "Diðer adlar için filtre eksik {{alias}}. ",
  672 + "configure-alias": "Yapýlandýrma {{alias}} takma ad",
  673 + "alias": "Alias",
  674 + "alias-required": "Varlýk takma adý gerekiyor.",
  675 + "remove-alias": "Varlýk takma adýný kaldýrma",
  676 + "add-alias": "Varlýk takma adý ekle",
  677 + "entity-list": "Varlýk listesi",
  678 + "entity-type": "Varlýk türü",
  679 + "entity-types": "Varlýk türleri",
  680 + "entity-type-list": "Varlýk türü listesi",
  681 + "any-entity": "Herhangi bir varlýk",
  682 + "enter-entity-type": "Varlýk türü girin",
  683 + "no-entities-matching": "{{Entity}} ile eþleþen hiçbir varlýk bulunamadý. ",
  684 + "no-entity-types-matching": "{{EntityType}} ile eþleþen hiçbir varlýk türü bulunamadý. ",
  685 + "name-starts-with": "Ýsim ile baþlar",
  686 + "use-entity-name-filter": "Filtre kullan",
  687 + "entity-list-empty": "Hiçbir varlýk seçilmedi.",
  688 + "entity-type-list-empty": "Hiçbir varlýk türü seçilmedi.",
  689 + "entity-name-filter-required": "Varlýk adý filtresi gerekli.",
  690 + "entity-name-filter-no-entity-matched": "{{Entity}} ile baþlayan hiçbir varlýk bulunamadý. ",
  691 + "all-subtypes": "Herþey",
  692 + "select-entities": "Öðeleri seç",
  693 + "no-aliases-found": "Takma ad bulunamadý",
  694 + "no-alias-matching": "'{{alias}} bulunamadý. ",
  695 + "create-new-alias": "Yeni bir tane oluþtur!",
  696 + "key": "Anahtar",
  697 + "key-name": "Anahtar adý",
  698 + "no-keys-found": "Anahtar bulunamadý.",
  699 + "no-key-matching": "'{{anahtar bulunamadý.",
  700 + "create-new-key": "Yeni bir tane oluþtur!",
  701 + "type": "Tür",
  702 + "type-required": "Varlýk türü gerekli.",
  703 + "type-device": "Cihaz",
  704 + "type-devices": "Cihazlar",
  705 + "list-of-devices": "{sayým, çoðul, 1 {Bir cihaz} diðer {# cihazlarýn listesi}}",
  706 + "device-name-starts-with": "Ýsimleri '{{prefix}} ile baþlayan cihazlar ",
  707 + "type-asset": "Varlýk",
  708 + "type-assets": "Varlýklar",
  709 + "list-of-assets": "{count, plural, 1 {One asset} diðer {# asset}}",
  710 + "asset-name-starts-with": "Adlarý {{prefix}} ile baþlayan varlýklar ",
  711 + "type-entity-view": "Varlýk Görünümü",
  712 + "type-entity-views": "Varlýk Görünümleri",
  713 + "list-of-entity-views": "{count, çoðul, 1 {Bir varlýk görünümü} diðer {# varlýk görüntüleme}} listesi",
  714 + "entity-view-name-starts-with": "Adý {{önek}} ile baþlayan varlýk görünümleri",
  715 + "type-rule": "Kural",
  716 + "type-rules": "Kurallar",
  717 + "list-of-rules": "{count, çoðul, 1 {Bir kural} diðer {# kurallarýn}} listesi",
  718 + "rule-name-starts-with": "Ýsimleri {{prefix}} ile baþlayan kurallar",
  719 + "type-plugin": "Eklenti",
  720 + "type-plugins": "Eklentiler",
  721 + "list-of-plugins": "{count, çoðul, 1 {Bir eklenti} diðer {# eklenti listesi}}",
  722 + "plugin-name-starts-with": "Ýsimleri {{prefix}} ile baþlayan eklentiler",
  723 + "type-tenant": "Kiracý",
  724 + "type-tenants": "Kiracýlar",
  725 + "list-of-tenants": "{count, çoðul, 1 {Bir kiracý} diðer {# kiracýlarýn listesi}}",
  726 + "tenant-name-starts-with": "Ýsimleri {{prefix}} ile baþlayan kiracýlar, ",
  727 + "type-customer": "Müþteri",
  728 + "type-customers": "Müþteriler",
  729 + "list-of-customers": "{count, çoðul, 1 {Bir müþteri} diðer {# müþteri}} listesi",
  730 + "customer-name-starts-with": "Ýsimleri {{prefix}} ile baþlayan müþteriler,",
  731 + "type-user": "Kullanýcý",
  732 + "type-users": "Kullanýcýlar",
  733 + "list-of-users": "{count, çoðul, 1 {Bir kullanýcý} diðer {# user}} listesi",
  734 + "user-name-starts-with": "Ýsimleri {{prefix}} ile baþlayan kullanýcýlar",
  735 + "type-dashboard": "Pano",
  736 + "type-dashboards": "Gösterge tablolarý",
  737 + "list-of-dashboards": "{count, çoðul, 1 {Bir pano} Diðer {# panolarýn}}} listesi",
  738 + "dashboard-name-starts-with": "Ýsimleri {{prefix}} ile baþlayan panolar",
  739 + "type-alarm": "Alarm",
  740 + "type-alarms": "Alarmlar",
  741 + "list-of-alarms": "{count, çoðul, 1 {Bir alarm} diðer {{# alarm}} listesi",
  742 + "alarm-name-starts-with": "Ýsimleri {{prefix}} ile baþlayan alarmlar",
  743 + "type-rulechain": "Kural zinciri",
  744 + "type-rulechains": "Kural zincirleri",
  745 + "list-of-rulechains": "{count, çoðul, 1 {Bir kural zinciri} diðer {# kural zincirinin listesi}}",
  746 + "rulechain-name-starts-with": "Ýsimleri {{prefix}} ile baþlayan kural zincirleri",
  747 + "type-rulenode": "Kural düðümü",
  748 + "type-rulenodes": "Kural düðümleri",
  749 + "list-of-rulenodes": "{count, çoðul, 1 {Bir kural node} diðer {# kural düðümünün listesi}}",
  750 + "rulenode-name-starts-with": "Ýsimleri '{{prefix}} ile baþlayan kural düðümleri",
  751 + "type-current-customer": "Mevcut Müþteri",
  752 + "search": "Varlýklar ara",
  753 + "selected-entities": "{count, çoðul, 1 {1 varlýk} diðer {# entities}} seçildi",
  754 + "entity-name": "Varlýk adý",
  755 + "details": "Varlýk ayrýntýlarý",
  756 + "no-entities-prompt": "Hiçbir varlýk bulunamadý",
  757 + "no-data": "Gösterilecek bilgi yok"
  758 + },
  759 + "entity-view": {
  760 + "entity-view": "Varlýk Görünümü",
  761 + "entity-views": "Varlýk Görünümleri",
  762 + "management": "Varlýk Görünümü yönetimi",
  763 + "view-entity-views": "Varlýk Görünümlerini Görüntüle",
  764 + "entity-view-alias": "Varlýk Görünümü takma adý",
  765 + "aliases": "Varlýk Görünümü takma adlarý",
  766 + "no-alias-matching": "'{{alias}} bulunamadý. ",
  767 + "no-aliases-found": "Takma ad bulunamadý",
  768 + "no-key-matching": "'{{anahtar bulunamadý.",
  769 + "no-keys-found": "Anahtar bulunamadý.",
  770 + "create-new-alias": "Yeni bir tane oluþtur!",
  771 + "create-new-key": "Yeni bir tane oluþtur!",
  772 + "duplicate-alias-error": "Yinelenen takma ad bulundu {{alias}} '.. Entity View diðer adlar, gösterge panosunda benzersiz olmalýdýr. ",
  773 + "configure-alias": "Yapýlandýrma {{alias}} takma ad",
  774 + "no-entity-views-matching": "{{Entity}} ile eþleþen hiçbir varlýk yorumu bulunamadý. ",
  775 + "alias": "Alias",
  776 + "alias-required": "Varlýk Görünümü takma adý gerekiyor.",
  777 + "remove-alias": "Varlýk görünümü takma adýný kaldýr",
  778 + "add-alias": "Varlýk görünümü takma adý ekle",
  779 + "name-starts-with": "Varlýk Görünümü adý ile baþlýyor",
  780 + "entity-view-list": "Varlýk Görünümü listesi",
  781 + "use-entity-view-name-filter": "Filtre kullan",
  782 + "entity-view-list-empty": "Hiçbir varlýk görüþü seçilmedi.",
  783 + "entity-view-name-filter-required": "Varlýk görünüm adý filtresi gerekli.",
  784 + "entity-view-name-filter-no-entity-view-matched": "{{EntityView}} ile baþlayan hiçbir varlýk sayýsý bulunamadý.",
  785 + "add": "Varlýk Görünümü Ekle",
  786 + "assign-to-customer": "Müþteriye atama",
  787 + "assign-entity-view-to-customer": "Varlýk Görünümlerini Müþteriye Atama",
  788 + "assign-entity-view-to-customer-text": "Lütfen müþteriye atamak için varlýk görünümlerini seçin",
  789 + "no-entity-views-text": "Varlýk görüþü bulunamadý",
  790 + "assign-to-customer-text": "Lütfen varlýk görünümlerini atamak için müþteriyi seçin",
  791 + "entity-view-details": "Varlýk görünümü ayrýntýlarý",
  792 + "add-entity-view-text": "Yeni varlýk görünümü ekle",
  793 + "delete": "Varlýk görünümünü sil",
  794 + "assign-entity-views": "Varlýk görünümleri atama",
  795 + "assign-entity-views-text": "Müþteriye {count, çoðul, 1 {1 entityView} diðer {# entityViews}} atayýn ",
  796 + "delete-entity-views": "Varlýk görünümlerini sil",
  797 + "unassign-from-customer": "Müþteriden atama",
  798 + "unassign-entity-views": "Varlýk görünümlerini atama",
  799 + "unassign-entity-views-action-title": "Müþteriden atama sayým, çoðul, 1 {1 entityView} diðer {# entityViews}}",
  800 + "assign-new-entity-view": "Yeni varlýk görünümü atama",
  801 + "delete-entity-view-title": "Varlýk görünümünü silmek istediðinizden emin misiniz?, {{EntityViewName}} '? ",
  802 + "delete-entity-view-text": "Dikkatli olun, onaylandýktan sonra varlýk görünümü ve ilgili tüm veriler kurtarýlamayacak.",
  803 + "delete-entity-views-title": "{Count, çoðul, 1 {1 entityView} diðer {# entityViews}} varlýk görünümüne sahip olmak istediðinizden emin misiniz?",
  804 + "delete-entity-views-action-title": "Sil {count, çoðul, 1 {1 entityView} diðer {# entityViews}}",
  805 + "delete-entity-views-text": "Dikkatli olun, onaylandýktan sonra tüm seçilen görünümler kaldýrýlacak ve ilgili tüm veriler kurtarýlamayacaktýr.",
  806 + "unassign-entity-view-title": "Varlýk görünümünün atamasýný kaldýrmak istediðinizden emin misiniz? {{EntityViewName}} '? ",
  807 + "unassign-entity-view-text": "Onaydan sonra varlýk görünümü atanmamýþ olacak ve müþteri tarafýndan eriþilemeyecektir.",
  808 + "unassign-entity-view": "Varlýk görünümünün atamasýný kaldýr",
  809 + "unassign-entity-views-title": "Sayým, çoðul, 1 {1 entityView} diðer {# entityViews}} hesabýnýn atamasýný kaldýrmak istediðinizden emin misiniz?",
  810 + "unassign-entity-views-text": "Onaylandýktan sonra, seçilen tüm öðe görünümleri atamadan kaldýrýlacak ve müþteri tarafýndan eriþilemeyecektir.",
  811 + "entity-view-type": "Varlýk Görünümü türü",
  812 + "entity-view-type-required": "Varlýk Görünümü türü gerekli.",
  813 + "select-entity-view-type": "Varlýk görüntüleme türünü seç",
  814 + "enter-entity-view-type": "Varlýk görüntüleme türünü girin",
  815 + "any-entity-view": "Herhangi bir varlýk görünümü",
  816 + "no-entity-view-types-matching": "{{EntitySubtype}} ile eþleþen hiçbir varlýk görüntüleme türü bulunamadý. ",
  817 + "entity-view-type-list-empty": "Hiçbir varlýk görünümü türü seçilmemiþ.",
  818 + "entity-view-types": "Varlýk Görünümü türleri",
  819 + "name": "Ad",
  820 + "name-required": "Ýsim gerekli.",
  821 + "description": "Açýklama",
  822 + "events": "Etkinlikler",
  823 + "details": "Ayrýntýlar",
  824 + "copyId": "Varlýk görüntüleme kimliðini kopyala",
  825 + "assignedToCustomer": "Müþteriye atandý",
  826 + "unable-entity-view-device-alias-title": "Varlýk görünümü takma adý silinemiyor",
  827 + "unable-entity-view-device-alias-text": "Cihaz takma adý {{entityViewAlias}} ', aþaðýdaki widget (lar) tarafýndan kullanýldýðý þekliyle silinemez: <br/> {{widgetsList}} ",
  828 + "select-entity-view": "Varlýk görünümünü seç",
  829 + "make-public": "Varlýðý herkese görünür yap",
  830 + "start-ts": "Ts",
  831 + "end-ts": "End ts"
  832 + },
  833 + "event": {
  834 + "event-type": "Etkinlik tipi",
  835 + "type-error": "Hata",
  836 + "type-lc-event": "Yaþam döngüsü etkinliði",
  837 + "type-stats": "Ýstatistik",
  838 + "type-debug-rule-node": "Hata ayýklama",
  839 + "type-debug-rule-chain": "Hata ayýklama",
  840 + "no-events-prompt": "Etkinlik bulunamadý",
  841 + "error": "Hata",
  842 + "alarm": "Alarm",
  843 + "event-time": "Etkinlik zamaný",
  844 + "server": "Sunucu",
  845 + "body": "Vücut",
  846 + "method": "Yöntem",
  847 + "type": "Tür",
  848 + "entity": "Varlýk",
  849 + "message-id": "Mesaj Kimliði",
  850 + "message-type": "Mesaj tipi",
  851 + "data-type": "Veri tipi",
  852 + "relation-type": "Ýliþki Türü",
  853 + "metadata": "Meta veri",
  854 + "data": "Veri",
  855 + "event": "Etkinlik",
  856 + "status": "Durum",
  857 + "success": "Baþarý",
  858 + "failed": "Baþarýsýz oldu",
  859 + "messages-processed": "Mesajlar iþlendi",
  860 + "errors-occurred": "Hatalar oluþtu"
  861 + },
  862 + "extension": {
  863 + "extensions": "Uzantýlar",
  864 + "selected-extensions": "{count, çoðul, 1 {1 uzantý} diðer {# extensions}} seçildi",
  865 + "type": "Tür",
  866 + "key": "Anahtar",
  867 + "value": "Deðer",
  868 + "id": "ÝD",
  869 + "extension-id": "Uzantý kimliði",
  870 + "extension-type": "Uzatma tipi",
  871 + "transformer-json": "JSON *",
  872 + "unique-id-required": "Mevcut uzantý kimliði zaten mevcut.",
  873 + "delete": "Uzantýyý sil",
  874 + "add": "Uzantý eklemek",
  875 + "edit": "Uzantýyý düzenle",
  876 + "delete-extension-title": "{{ExtensionId}} uzantýsýný silmek istediðinizden emin misiniz? ",
  877 + "delete-extension-text": "Dikkatli olun, onaylamadan sonra uzantý ve ilgili tüm veriler kurtarýlamaz.",
  878 + "delete-extensions-title": "{Count, çoðul, 1 {1 uzantý} diðer {# extensions}} silmek istediðinizden emin misiniz?",
  879 + "delete-extensions-text": "Dikkatli olun, onaylandýktan sonra tüm seçilen uzantýlar kaldýrýlacak.",
  880 + "converters": "Dönüþtürücü",
  881 + "converter-id": "Dönüþtürücü kimliði",
  882 + "configuration": "Yapýlandýrma",
  883 + "converter-configurations": "Dönüþtürücü yapýlandýrmalarý",
  884 + "token": "Güvenlik belirteci",
  885 + "add-converter": "Dönüþtürücü ekle",
  886 + "add-config": "Dönüþtürücü yapýlandýrmasý ekle",
  887 + "device-name-expression": "Cihaz adý ifadesi",
  888 + "device-type-expression": "Cihaz tipi ifadesi",
  889 + "custom": "Özel",
  890 + "to-double": "Çifte",
  891 + "transformer": "Transformer",
  892 + "json-required": "Trafo jsonu gerekli.",
  893 + "json-parse": "Trafo json ayrýþtýrýlamýyor.",
  894 + "attributes": "Öznitellikler",
  895 + "add-attribute": "Özellik ekle",
  896 + "add-map": "Eþleme elemaný ekle",
  897 + "timeseries": "Zaman serisi",
  898 + "add-timeseries": "Zaman çizelgeleri ekle",
  899 + "field-required": "Alan gereklidir",
  900 + "brokers": "Komisyoncular",
  901 + "add-broker": "Broker ekle",
  902 + "host": "Host",
  903 + "port": "Liman",
  904 + "port-range": "Liman 1'den 65535'e kadar olmalýdýr.",
  905 + "ssl": "SSL",
  906 + "credentials": "Kimlik bilgileri",
  907 + "username": "Kullanýcý adý",
  908 + "password": "Parola",
  909 + "retry-interval": "Milisaniye cinsinden tekrar deneme aralýðý",
  910 + "anonymous": "Anonim",
  911 + "basic": "Temel",
  912 + "pem": "PEM",
  913 + "ca-cert": "CA sertifika dosyasý *",
  914 + "private-key": "Özel anahtar dosya *",
  915 + "cert": "Sertifika dosyasý *",
  916 + "no-file": "Dosya seçilmedi.",
  917 + "drop-file": "Bir dosya býrakýn veya yüklenecek bir dosya seçmek için týklayýn.",
  918 + "mapping": "Mapping",
  919 + "topic-filter": "Konu filtresi",
  920 + "converter-type": "Dönüþtürücü tipi",
  921 + "converter-json": "Json",
  922 + "json-name-expression": "Cihaz adý json ifadesi",
  923 + "topic-name-expression": "Cihaz adý konu ifadesi",
  924 + "json-type-expression": "Cihaz tipi json ifadesi",
  925 + "topic-type-expression": "Cihaz tipi konu ifadesi",
  926 + "attribute-key-expression": "Öznitelik anahtar ifadesi",
  927 + "attr-json-key-expression": "Öznitelik anahtar json ifadesi",
  928 + "attr-topic-key-expression": "Öznitelik anahtar konu ifadesi",
  929 + "request-id-expression": "Kimlik ifadesi iste",
  930 + "request-id-json-expression": "Kimlik json ifadesi iste",
  931 + "request-id-topic-expression": "Kimlik konu ifadesini isteyin",
  932 + "response-topic-expression": "Yanýt konusu ifadesi",
  933 + "value-expression": "Deðer ifadesi",
  934 + "topic": "Konu",
  935 + "timeout": "Zaman aþýmý milisaniye cinsinden",
  936 + "converter-json-required": "Dönüþtürücü json gerekli.",
  937 + "converter-json-parse": "Dönüþtürücü json ayrýþtýrýlamýyor.",
  938 + "filter-expression": "Filtre ifadesi",
  939 + "connect-requests": "Ýstekleri baðla",
  940 + "add-connect-request": "Baðlantý talebi ekle",
  941 + "disconnect-requests": "Ýstekleri kes",
  942 + "add-disconnect-request": "Baðlantýyý kes isteði ekle",
  943 + "attribute-requests": "Özellik istekleri",
  944 + "add-attribute-request": "Özellik isteði ekle",
  945 + "attribute-updates": "Öznitelik güncellemeleri",
  946 + "add-attribute-update": "Özellik güncellemesi ekle",
  947 + "server-side-rpc": "Sunucu tarafý RPC",
  948 + "add-server-side-rpc-request": "Sunucu tarafý RPC isteði ekle",
  949 + "device-name-filter": "Cihaz adý filtresi",
  950 + "attribute-filter": "Özellik filtresi",
  951 + "method-filter": "Yöntem filtresi",
  952 + "request-topic-expression": "Konu ifadesi iste",
  953 + "response-timeout": "Milisaniye cinsinden yanýt zaman aþýmý",
  954 + "topic-expression": "Konu ifadesi",
  955 + "client-scope": "Müþteri kapsamý",
  956 + "add-device": "Cihaz ekle",
  957 + "opc-server": "Sunucular",
  958 + "opc-add-server": "Sunucu ekle",
  959 + "opc-add-server-prompt": "Lütfen sunucu ekle",
  960 + "opc-application-name": "Uygulama Adý",
  961 + "opc-application-uri": "Uygulama uri",
  962 + "opc-scan-period-in-seconds": "Saniyeler içinde tarama süresi",
  963 + "opc-security": "Güvenlik",
  964 + "opc-identity": "Kimlik",
  965 + "opc-keystore": "Keystore",
  966 + "opc-type": "Tür",
  967 + "opc-keystore-type": "Tür",
  968 + "opc-keystore-location": "Yer *",
  969 + "opc-keystore-password": "Parola",
  970 + "opc-keystore-alias": "Alias",
  971 + "opc-keystore-key-password": "Anahtar þifre",
  972 + "opc-device-node-pattern": "Cihaz düðümü modeli",
  973 + "opc-device-name-pattern": "Cihaz adý deseni",
  974 + "modbus-server": "Sunucular / köle",
  975 + "modbus-add-server": "Sunucu ekle / köle",
  976 + "modbus-add-server-prompt": "Lütfen sunucu / slave ekle",
  977 + "modbus-transport": "Taþýma",
  978 + "modbus-port-name": "Seri port adý",
  979 + "modbus-encoding": "Kodlama",
  980 + "modbus-parity": "Parite",
  981 + "modbus-baudrate": "Baud hýzý",
  982 + "modbus-databits": "Veri bitleri",
  983 + "modbus-stopbits": "Bitleri durdur",
  984 + "modbus-databits-range": "Veri bitleri 7 ila 8 arasýnda olmalýdýr",
  985 + "modbus-stopbits-range": "Durma bitleri 1'den 2'ye kadar olmalýdýr.",
  986 + "modbus-unit-id": "Birim Kimliði",
  987 + "modbus-unit-id-range": "Birim numarasý 1 ile 247 arasýnda olmalýdýr.",
  988 + "modbus-device-name": "Cihaz adý",
  989 + "modbus-poll-period": "Anket dönemi (ms)",
  990 + "modbus-attributes-poll-period": "Nitelikler yoklama süresi (ms)",
  991 + "modbus-timeseries-poll-period": "Timeseries anket süresi (ms)",
  992 + "modbus-poll-period-range": "Anket dönemi pozitif deðer olmalý",
  993 + "modbus-tag": "Etiket",
  994 + "modbus-function": "Ýþlev",
  995 + "modbus-register-address": "Kayýt adresi",
  996 + "modbus-register-address-range": "Kayýt adresi 0 ile 65535 arasýnda olmalýdýr.",
  997 + "modbus-register-bit-index": "Bit endeksi",
  998 + "modbus-register-bit-index-range": "Bit endeksi 0 ile 15 arasýnda olmalýdýr",
  999 + "modbus-register-count": "Kayýt sayýsý",
  1000 + "modbus-register-count-range": "Kayýt sayýsý pozitif bir deðer olmalýdýr.",
  1001 + "modbus-byte-order": "Bayt sýrasý",
  1002 + "sync": {
  1003 + "status": "Durum",
  1004 + "sync": "Senkronizasyon",
  1005 + "not-sync": "Eþitleme",
  1006 + "last-sync-time": "Son senkronizasyon zamaný",
  1007 + "not-available": "Müsait deðil"
  1008 + },
  1009 + "export-extensions-configuration": "Ýhracat uzantýlarý yapýlandýrmasý",
  1010 + "import-extensions-configuration": "Uzantýlarýný içe aktarma yapýlandýrmasý",
  1011 + "import-extensions": "Uzantýlarý içe aktar",
  1012 + "import-extension": "Uzantý içe aktar",
  1013 + "export-extension": "Ýhracat uzantýsý",
  1014 + "file": "Uzantýlar dosyasý",
  1015 + "invalid-file-error": "Geçersiz uzantý dosyasý"
  1016 + },
  1017 + "fullscreen": {
  1018 + "expand": "Tam ekrana geniþlet",
  1019 + "exit": "Tam ekrandan çýk",
  1020 + "toggle": "Tam ekran modunu deðiþtir",
  1021 + "fullscreen": "Tam ekran"
  1022 + },
  1023 + "function": {
  1024 + "function": "Ýþlev"
  1025 + },
  1026 + "grid": {
  1027 + "delete-item-title": "Bu maddeyi silmek istediðinden emin misin?",
  1028 + "delete-item-text": "Dikkatli olun, onaylandýktan sonra bu öðe ve ilgili tüm veriler kurtarýlamaz.",
  1029 + "delete-items-title": "{Count, çoðul, 1 {1 item} diðer {# items}} silmek istediðinizden emin misiniz?",
  1030 + "delete-items-action-title": "Sil {count, çoðul, 1 {1 öðe} diðer {# items}}",
  1031 + "delete-items-text": "Dikkatli olun, onaylandýktan sonra tüm seçilen öðeler silinecek ve ilgili tüm veriler kurtarýlamayacaktýr.",
  1032 + "add-item-text": "Yeni öðe ekle",
  1033 + "no-items-text": "Hiç bir öðe bulunamadý",
  1034 + "item-details": "Ürün detaylarý",
  1035 + "delete-item": "Öðeyi silmek",
  1036 + "delete-items": "Ürünleri sil",
  1037 + "scroll-to-top": "Baþa kaydýr"
  1038 + },
  1039 + "help": {
  1040 + "goto-help-page": "Yardým sayfasýna git"
  1041 + },
  1042 + "home": {
  1043 + "home": "Ev",
  1044 + "profile": "Profil",
  1045 + "logout": "Çýkýþ Yap",
  1046 + "menu": "Menü",
  1047 + "avatar": "Avatar",
  1048 + "open-user-menu": "Kullanýcý menüsünü aç"
  1049 + },
  1050 + "import": {
  1051 + "no-file": "Dosya seçilmedi",
  1052 + "drop-file": "Bir JSON dosyasýný indirin veya yüklemek için bir dosya seçmek için týklayýn."
  1053 + },
  1054 + "item": {
  1055 + "selected": "Seçilmiþ"
  1056 + },
  1057 + "js-func": {
  1058 + "no-return-error": "Ýþlev deðer döndürmeli!",
  1059 + "return-type-mismatch": "Ýþlev '{{type}}' tipinin deðerini döndürmelidir!",
  1060 + "tidy": "Düzenli"
  1061 + },
  1062 + "key-val": {
  1063 + "key": "Anahtar",
  1064 + "value": "Deðer",
  1065 + "remove-entry": "Giriþi kaldýr",
  1066 + "add-entry": "Giriþ ekle",
  1067 + "no-data": "Giriþ yok"
  1068 + },
  1069 + "layout": {
  1070 + "layout": "Düzen",
  1071 + "manage": "Düzenleri yönet",
  1072 + "settings": "Yerleþim ayarlarý",
  1073 + "color": "Renk",
  1074 + "main": "Ana",
  1075 + "right": "Sað",
  1076 + "select": "Hedef yerleþimini seç"
  1077 + },
  1078 + "legend": {
  1079 + "position": "Efsane pozisyonu",
  1080 + "show-max": "Maksimum deðeri göster",
  1081 + "show-min": "Min deðerini göster",
  1082 + "show-avg": "Ortalama deðeri göster",
  1083 + "show-total": "Toplam deðeri göster",
  1084 + "settings": "Açýklama ayarlarý",
  1085 + "min": "min",
  1086 + "max": "max",
  1087 + "avg": "avg",
  1088 + "total": "Genel Toplam"
  1089 + },
  1090 + "login": {
  1091 + "login": "Oturum aç",
  1092 + "request-password-reset": "Þifre sýfýrlama isteði",
  1093 + "reset-password": "Þifreyi yenile",
  1094 + "create-password": "Þifre oluþtur",
  1095 + "passwords-mismatch-error": "Girilen þifreler ayný olmalýdýr!",
  1096 + "password-again": "Þifre Tekrar",
  1097 + "sign-in": "Lütfen giriþ yapýn",
  1098 + "username": "Kullanýcý adý (email)",
  1099 + "remember-me": "Beni hatýrla",
  1100 + "forgot-password": "Parolanýzý mý unuttunuz?",
  1101 + "password-reset": "Parola sýfýrlama",
  1102 + "new-password": "Yeni Þifre",
  1103 + "new-password-again": "Yeni Þifre Tekrar",
  1104 + "password-link-sent-message": "Þifre sýfýrlama baðlantýsý baþarýyla gönderildi!",
  1105 + "email": "E-posta"
  1106 + },
  1107 + "position": {
  1108 + "top": "Üst",
  1109 + "bottom": "Alt",
  1110 + "left": "Ayrýldý",
  1111 + "right": "Sað"
  1112 + },
  1113 + "profile": {
  1114 + "profile": "Profil",
  1115 + "change-password": "Þifre deðiþtir",
  1116 + "current-password": "Þimdiki Þifre"
  1117 + },
  1118 + "relation": {
  1119 + "relations": "Ýliþkiler",
  1120 + "direction": "Yön",
  1121 + "search-direction": {
  1122 + "FROM": "Kimden",
  1123 + "TO": "Alýcý"
  1124 + },
  1125 + "direction-type": {
  1126 + "FROM": "den",
  1127 + "TO": "to"
  1128 + },
  1129 + "from-relations": "Giden iliþkiler",
  1130 + "to-relations": "Gelen iliþkiler",
  1131 + "selected-relations": "{count, çoðul, 1 {1 iliþki} diðer {# relations}} seçildi",
  1132 + "type": "Tür",
  1133 + "to-entity-type": "Varlýk türüne",
  1134 + "to-entity-name": "Varlýk adýna",
  1135 + "from-entity-type": "Varlýk türünden",
  1136 + "from-entity-name": "Varlýk adýndan",
  1137 + "to-entity": "Varlýk",
  1138 + "from-entity": "Varlýktan",
  1139 + "delete": "Ýliþkisi sil",
  1140 + "relation-type": "Ýliþki türü",
  1141 + "relation-type-required": "Ýliþki türü gerekli.",
  1142 + "any-relation-type": "Her hangi bir tür",
  1143 + "add": "Ýliþki ekle",
  1144 + "edit": "Ýliþkisi düzenle",
  1145 + "delete-to-relation-title": "{{EntityName}} varlýðýna iliþkin iliþkiyi silmek istediðinizden emin misiniz? ",
  1146 + "delete-to-relation-text": "Doðrulamadan sonra, varlýðýn {{entityName}} öðesinin mevcut varlýktan alakasýz olacaðýndan emin olun. ",
  1147 + "delete-to-relations-title": "{Count, çoðul, 1 {1 iliþki} diðer {# relations}} silmek istediðinizden emin misiniz?",
  1148 + "delete-to-relations-text": "Dikkatli olun, onaylandýktan sonra tüm seçilen iliþkiler kaldýrýlacak ve ilgili varlýklar mevcut varlýk ile ilgisiz olacaktýr.",
  1149 + "delete-from-relation-title": "{{EntityName}} varlýðýndan iliþkiyi silmek istediðinizden emin misiniz? ",
  1150 + "delete-from-relation-text": "Doðrulama varlýðýnýn, {{entityName}} kuruluþundan alakasýz olacaðýndan emin olun.",
  1151 + "delete-from-relations-title": "{Count, çoðul, 1 {1 iliþki} diðer {# relations}} silmek istediðinizden emin misiniz?",
  1152 + "delete-from-relations-text": "Dikkatli olun, onaylandýktan sonra tüm seçilen iliþkiler kaldýrýlacak ve mevcut varlýk ilgili varlýklardan ilgisiz olacaktýr.",
  1153 + "remove-relation-filter": "Ýliþki filtresini kaldýr",
  1154 + "add-relation-filter": "Ýliþki filtresi ekle",
  1155 + "any-relation": "Herhangi bir iliþki",
  1156 + "relation-filters": "Ýliþki filtreleri",
  1157 + "additional-info": "Ek bilgi (JSON)",
  1158 + "invalid-additional-info": "Ek bilgi json ayrýþtýrýlamýyor."
  1159 + },
  1160 + "rulechain": {
  1161 + "rulechain": "Kural zinciri",
  1162 + "rulechains": "Kural zincirleri",
  1163 + "root": "Kök",
  1164 + "delete": "Kural zincirini sil",
  1165 + "name": "Ad",
  1166 + "name-required": "Ýsim gerekli.",
  1167 + "description": "Açýklama",
  1168 + "add": "Kural Zinciri Ekleme",
  1169 + "set-root": "Kural zincirinin kökü yap",
  1170 + "set-root-rulechain-title": "Kural zincirini {{ruleChainName}} root? Yapmak istediðinizden emin misiniz?",
  1171 + "set-root-rulechain-text": "Onaydan sonra kural zinciri kökleþecek ve gelen tüm iletilerle ilgilenecek.",
  1172 + "delete-rulechain-title": "Kural zincirini {{ruleChainName}} silmek istediðinizden emin misiniz?",
  1173 + "delete-rulechain-text": "Dikkatli olun, onaylamadan sonra kural zinciri ve ilgili tüm veriler kurtarýlamaz.",
  1174 + "delete-rulechains-title": "{Count, çoðul, 1 {1 kural zinciri} diðer {# kural zincirleri}} silmek istediðinizden emin misiniz?",
  1175 + "delete-rulechains-action-title": "Sil {count, çoðul, 1 {1 kural zinciri} diðer {# kural zincirleri}}",
  1176 + "delete-rulechains-text": "Dikkatli olun, onaylandýktan sonra seçilen tüm kural zincirleri silinecek ve ilgili tüm veriler kurtarýlamayacaktýr.",
  1177 + "add-rulechain-text": "Yeni kural zinciri ekle",
  1178 + "no-rulechains-text": "Kural zinciri bulunamadý",
  1179 + "rulechain-details": "Kural zinciri detaylarý",
  1180 + "details": "Ayrýntýlar",
  1181 + "events": "Etkinlikler",
  1182 + "system": "Sistem",
  1183 + "import": "Kural zincirini içe aktar",
  1184 + "export": "Kural zinciri dýþa aktar",
  1185 + "export-failed-error": "Kural zinciri dýþa aktarýlamadý: {{error}}",
  1186 + "create-new-rulechain": "Yeni kural zinciri oluþtur",
  1187 + "rulechain-file": "Kural zinciri dosyasý",
  1188 + "invalid-rulechain-file-error": "Kural zinciri içe aktarýlamýyor: Geçersiz kural zinciri veri yapýsý.",
  1189 + "copyId": "Kural zinciri kimliðini kopyala",
  1190 + "idCopiedMessage": "Kural zinciri kimliði panoya kopyalandý",
  1191 + "select-rulechain": "Kural zincirini seç",
  1192 + "no-rulechains-matching": "{{Entity}} ile eþleþen kural zinciri bulunamadý. ",
  1193 + "rulechain-required": "Kural zinciri gerekli",
  1194 + "management": "Kural yönetimi",
  1195 + "debug-mode": "Hata ayýklama modu"
  1196 + },
  1197 + "rulenode": {
  1198 + "details": "Ayrýntýlar",
  1199 + "events": "Etkinlikler",
  1200 + "search": "Arama düðümleri",
  1201 + "open-node-library": "Düðüm kütüphanesini aç",
  1202 + "add": "Kural düðümü ekle",
  1203 + "name": "Ad",
  1204 + "name-required": "Ýsim gerekli.",
  1205 + "type": "Tür",
  1206 + "description": "Açýklama",
  1207 + "delete": "Kural düðümünü sil",
  1208 + "select-all-objects": "Tüm düðümleri ve baðlantýlarý seç",
  1209 + "deselect-all-objects": "Tüm düðümlerin ve baðlantýlarýn seçimini kaldýrýn",
  1210 + "delete-selected-objects": "Seçilen düðümleri ve baðlantýlarý sil",
  1211 + "delete-selected": "Silme seçildi",
  1212 + "select-all": "Hepsini seç",
  1213 + "copy-selected": "Seçilenleri kopyala",
  1214 + "deselect-all": "Hiçbirini seçme",
  1215 + "rulenode-details": "Kural düðümü ayrýntýlarý",
  1216 + "debug-mode": "Hata ayýklama modu",
  1217 + "configuration": "Yapýlandýrma",
  1218 + "link": "Baðlantý",
  1219 + "link-details": "Kural düðüm baðlantý detaylarý",
  1220 + "add-link": "Link ekle",
  1221 + "link-label": "Baðlantý etiketi",
  1222 + "link-label-required": "Baðlantý etiketi gerekli.",
  1223 + "custom-link-label": "Özel baðlantý etiketi",
  1224 + "custom-link-label-required": "Özel baðlantý etiketi gerekli.",
  1225 + "link-labels": "Link etiketleri",
  1226 + "link-labels-required": "Link etiketleri gerekli.",
  1227 + "no-link-labels-found": "Baðlantý etiketi bulunamadý",
  1228 + "no-link-label-matching": "{{label}} bulunamadý. ",
  1229 + "create-new-link-label": "Yeni bir tane oluþtur!",
  1230 + "type-filter": "Filtre",
  1231 + "type-filter-details": "Gelen iletileri yapýlandýrýlmýþ koþullara göre filtrele",
  1232 + "type-enrichment": "Zenginleþtirme",
  1233 + "type-enrichment-details": "Mesaj Meta Verilerine ek bilgi",
  1234 + "type-transformation": "Dönüþüm",
  1235 + "type-transformation-details": "Mesaj yükünü ve Meta Verileri Deðiþtir",
  1236 + "type-action": "Aksiyon",
  1237 + "type-action-details": "Özel eylem gerçekleþtir",
  1238 + "type-external": "Dýþ",
  1239 + "type-external-details": "Dýþ sistemle etkileþir",
  1240 + "type-rule-chain": "Kural Zinciri",
  1241 + "type-rule-chain-details": "Belirtilen Kural Zincirine gelen mesajlarý ilet",
  1242 + "type-input": "Giriþ",
  1243 + "type-input-details": "Kural Zinciri'nin mantýksal girdisi, bir sonraki ilgili Kural Düðümüne gelen iletileri iletme",
  1244 + "type-unknown": "Bilinmeyen",
  1245 + "type-unknown-details": "Çözümlenmemiþ Kural Düðümü",
  1246 + "directive-is-not-loaded": "Tanýmlanmýþ yapýlandýrma yönergesi {{directiveName}} 'mevcut deðil. ",
  1247 + "ui-resources-load-error": "Yapýlandýrma kullanýcý arayüzü kaynaklarý yüklenemedi.",
  1248 + "invalid-target-rulechain": "Hedef kural zinciri çözülemiyor!",
  1249 + "test-script-function": "Test komut dosyasý iþlevi",
  1250 + "message": "Mesaj",
  1251 + "message-type": "Mesaj tipi",
  1252 + "select-message-type": "Mesaj tipini seç",
  1253 + "message-type-required": "Mesaj türü gerekli",
  1254 + "metadata": "Meta veri",
  1255 + "metadata-required": "Meta veri giriþleri boþ býrakýlamaz.",
  1256 + "output": "Çýktý",
  1257 + "test": "Ölçek",
  1258 + "help": "Yardým et"
  1259 + },
  1260 + "tenant": {
  1261 + "tenant": "Kiracý",
  1262 + "tenants": "Kiracýlar",
  1263 + "management": "Kiracý yönetimi",
  1264 + "add": "Kiracý ekle",
  1265 + "admins": "Yöneticiler",
  1266 + "manage-tenant-admins": "Kiracý yöneticileri yönet",
  1267 + "delete": "Kiracýyý sil",
  1268 + "add-tenant-text": "Yeni kiracý ekle",
  1269 + "no-tenants-text": "Kiracý bulunamadý",
  1270 + "tenant-details": "Kiracý detaylarý",
  1271 + "delete-tenant-title": "Kiracýyý silmek istediðinizden emin misiniz? {{TenantTitle}}? ",
  1272 + "delete-tenant-text": "Dikkatli olun, onayýndan sonra kiracý ve ilgili tüm veriler kurtarýlamaz.",
  1273 + "delete-tenants-title": "{Count, çoðul, 1 {1 kiracý} diðer {# kiracý}} silmek istediðinizden emin misiniz?",
  1274 + "delete-tenants-action-title": "Sil {count, çoðul, 1 {1 kiracý} diðer {# kiracý}}",
  1275 + "delete-tenants-text": "Dikkatli olun, onaylandýktan sonra tüm kiracýlar silinecek ve ilgili tüm veriler kurtarýlamayacaktýr.",
  1276 + "title": "Baþlýk",
  1277 + "title-required": "Baþlýk gerekli.",
  1278 + "description": "Açýklama",
  1279 + "details": "Ayrýntýlar",
  1280 + "events": "Etkinlikler",
  1281 + "copyId": "Kiracý Kimliði Kimliði",
  1282 + "idCopiedMessage": "Kiracý Kimliði panoya kopyalandý",
  1283 + "select-tenant": "Kiracý seç",
  1284 + "no-tenants-matching": "{{Entity}} ile eþleþen kiracý bulunamadý. ",
  1285 + "tenant-required": "Kiracý gerekli"
  1286 + },
  1287 + "timeinterval": {
  1288 + "seconds-interval": "{saniye, çoðul, 1 {1 saniye} diðer {# seconds}}",
  1289 + "minutes-interval": "{dakika, çoðul, 1 {1 dakika} diðer {# dakika}}",
  1290 + "hours-interval": "{saat, çoðul, 1 {1 saat} diðer {# hours}}",
  1291 + "days-interval": "{günler, çoðul, 1 {1 gün} diðer {# gün}}",
  1292 + "days": "Gün",
  1293 + "hours": "Saatler",
  1294 + "minutes": "Dakika",
  1295 + "seconds": "Saniye",
  1296 + "advanced": "Ýleri"
  1297 + },
  1298 + "timewindow": {
  1299 + "days": "{günler, çoðul, 1 {gün} diðer {# gün}}",
  1300 + "hours": "{saat, çoðul, 0 {saat} 1 {1 saat} diðer {# hours}}",
  1301 + "minutes": "{dakika, çoðul, 0 {dakika} 1 {1 dakika} diðer {# dakika}}",
  1302 + "seconds": "{saniye, çoðul, 0 {saniye} 1 {1 saniye} diðer {# seconds}}",
  1303 + "realtime": "Gerçek zaman",
  1304 + "history": "Tarihçe",
  1305 + "last-prefix": "son",
  1306 + "period": "{{startTime}} - {{endTime}} arasýnda ",
  1307 + "edit": "Zaman penceresini düzenle",
  1308 + "date-range": "Tarih aralýðý",
  1309 + "last": "Son",
  1310 + "time-period": "Zaman dilimi"
  1311 + },
  1312 + "user": {
  1313 + "user": "Kullanýcý",
  1314 + "users": "Kullanýcýlar",
  1315 + "customer-users": "Müþteri Kullanýcýlarý",
  1316 + "tenant-admins": "Kiracý Admins",
  1317 + "sys-admin": "Sistem yöneticisi",
  1318 + "tenant-admin": "Kiracý yöneticisi",
  1319 + "customer": "Müþteri",
  1320 + "anonymous": "Anonim",
  1321 + "add": "Kullanýcý Ekle",
  1322 + "delete": "Kullanýcýyý sil",
  1323 + "add-user-text": "Yeni kullanýcý Ekle",
  1324 + "no-users-text": "Kullanýcý bulunamadý",
  1325 + "user-details": "Kullanýcý detaylarý",
  1326 + "delete-user-title": "Kullanýcýyý silmek istediðinizden emin misiniz? {{UserEmail}} '? ",
  1327 + "delete-user-text": "Dikkatli olun, onaylandýktan sonra kullanýcý ve ilgili tüm veriler kurtarýlamayacaktýr.",
  1328 + "delete-users-title": "{Count, çoðul, 1 {1 kullanýcý} diðer {# users}} silmek istediðinizden emin misiniz?",
  1329 + "delete-users-action-title": "Sil {count, çoðul, 1 {1 kullanýcý} diðer {# users}}",
  1330 + "delete-users-text": "Dikkatli olun, onaylandýktan sonra tüm seçilen kullanýcýlar silinecek ve ilgili tüm veriler kurtarýlamayacaktýr.",
  1331 + "activation-email-sent-message": "Etkinleþtirme e-postasý baþarýyla gönderildi!",
  1332 + "resend-activation": "Etkinleþtirmeyi tekrar gönder",
  1333 + "email": "E-posta",
  1334 + "email-required": "Email gereklidir.",
  1335 + "invalid-email-format": "Geçersiz e-posta formatý.",
  1336 + "first-name": "Ýsim",
  1337 + "last-name": "Soyadý",
  1338 + "description": "Açýklama",
  1339 + "default-dashboard": "Varsayýlan gösterge paneli",
  1340 + "always-fullscreen": "Her zaman tam ekran",
  1341 + "select-user": "Kullanýcý seç",
  1342 + "no-users-matching": "{{Entity}} ile eþleþen kullanýcý bulunamadý. ",
  1343 + "user-required": "Kullanýcý gerekli",
  1344 + "activation-method": "Aktivasyon yöntemi",
  1345 + "display-activation-link": "Etkinleþtirme baðlantýsý göster",
  1346 + "send-activation-mail": "Etkinleþtirme postasý gönder",
  1347 + "activation-link": "Kullanýcý aktivasyon linki",
  1348 + "activation-link-text": "Kullanýcýyý aktif hale getirmek için aþaðýdaki <a href='{{activationLink}} target='_blank'> aktivasyon linki </a> kullanýn.",
  1349 + "copy-activation-link": "Etkinleþtirme baðlantýsýný kopyala",
  1350 + "activation-link-copied-message": "Kullanýcý aktivasyon linki panoya kopyalandý",
  1351 + "details": "Ayrýntýlar",
  1352 + "login-as-tenant-admin": "Kiracý Yönetici Giriþi",
  1353 + "login-as-customer-user": "Müþteri olarak giriþ yap"
  1354 + },
  1355 + "value": {
  1356 + "type": "Deðer türü",
  1357 + "string": "Dize",
  1358 + "string-value": "Dize deðeri",
  1359 + "integer": "Integer",
  1360 + "integer-value": "Tamsayý deðeri",
  1361 + "invalid-integer-value": "Geçersiz tam sayý",
  1362 + "double": "Çift",
  1363 + "double-value": "Çift deðer",
  1364 + "boolean": "Boole",
  1365 + "boolean-value": "Boole deðeri",
  1366 + "false": "Yanlýþ",
  1367 + "true": "Doðru",
  1368 + "long": "Uzun"
  1369 + },
  1370 + "widget": {
  1371 + "widget-library": "Widgets Kitaplýðý",
  1372 + "widget-bundle": "Widget Paketi",
  1373 + "select-widgets-bundle": "Widget paketini seç",
  1374 + "management": "Widget yönetimi",
  1375 + "editor": "Widget Düzenleyici",
  1376 + "widget-type-not-found": "Sorun yükleme pencere öðesi yapýlandýrmasý. <br> Muhtemelen iliþkili /n pencere öðesi türü kaldýrýldý.",
  1377 + "widget-type-load-error": "Widget, aþaðýdaki hatalardan dolayý yüklenmedi:",
  1378 + "remove": "Widget'ý kaldýr",
  1379 + "edit": "Widget düzenle",
  1380 + "remove-widget-title": "Widget'ý kaldýrmak istediðinizden emin misiniz? {{WidgetTitle}}? ",
  1381 + "remove-widget-text": "Onaydan sonra widget ve ilgili tüm veriler kurtarýlamaz.",
  1382 + "timeseries": "Zaman serisi",
  1383 + "search-data": "Arama verileri",
  1384 + "no-data-found": "Veri bulunamadý",
  1385 + "latest-values": "Son deðerler",
  1386 + "rpc": "Kontrol aracý",
  1387 + "alarm": "Alarm gereci",
  1388 + "static": "Statik pencere öðesi",
  1389 + "select-widget-type": "Widget tipi seç",
  1390 + "missing-widget-title-error": "Widget baþlýðý belirtilmelidir!",
  1391 + "widget-saved": "Widget kaydedildi",
  1392 + "unable-to-save-widget-error": "Widget kaydedilemiyor! Widget'ýn hatalarý var!",
  1393 + "save": "Widget kaydet",
  1394 + "saveAs": "Widget olarak kaydet",
  1395 + "save-widget-type-as": "Widget türünü kaydet",
  1396 + "save-widget-type-as-text": "Lütfen yeni pencere öðesi baþlýðýný girin ve / veya hedef widget'larý seçin",
  1397 + "toggle-fullscreen": "Tam ekrana geç",
  1398 + "run": "Widget'ý çalýþtýr",
  1399 + "title": "Widget Baþlýðý",
  1400 + "title-required": "Widget baþlýðý gerekiyor.",
  1401 + "type": "Widget türü",
  1402 + "resources": "Kaynaklar",
  1403 + "resource-url": "JavaScript / CSS URL",
  1404 + "remove-resource": "Kaynaðý kaldýr",
  1405 + "add-resource": "Kaynak ekle",
  1406 + "html": "HTML",
  1407 + "tidy": "Düzenli",
  1408 + "css": "CSS",
  1409 + "settings-schema": "Ayarlar þemasý",
  1410 + "datakey-settings-schema": "Veri anahtarý ayarlarý þemasý",
  1411 + "javascript": "JavaScript",
  1412 + "remove-widget-type-title": "'{{WidgetName}}' widget türünü kaldýrmak istediðinizden emin misiniz?",
  1413 + "remove-widget-type-text": "Onaydan sonra widget tipi ve ilgili tüm veriler kurtarýlamayacak.",
  1414 + "remove-widget-type": "Widget türünü kaldýr",
  1415 + "add-widget-type": "Yeni widget türü ekle",
  1416 + "widget-type-load-failed-error": "Widget türü yüklenemedi!",
  1417 + "widget-template-load-failed-error": "Widget þablonu yüklenemedi!",
  1418 + "add": "Widget Ekle",
  1419 + "undo": "Widget deðiþikliklerini geri al",
  1420 + "export": "Widget'ý dýþa aktar"
  1421 + },
  1422 + "widget-action": {
  1423 + "header-button": "Widget baþlýk düðmesi",
  1424 + "open-dashboard-state": "Yeni gösterge panosuna git",
  1425 + "update-dashboard-state": "Mevcut kontrol paneli durumunu güncelle",
  1426 + "open-dashboard": "Diðer kontrol paneline git",
  1427 + "custom": "Özel eylem",
  1428 + "target-dashboard-state": "Hedef gösterge panosu durumu",
  1429 + "target-dashboard-state-required": "Hedef gösterge tablosu gerekli",
  1430 + "set-entity-from-widget": "Öðeyi pencere öðesinden ayarla",
  1431 + "target-dashboard": "Hedef gösterge panosu",
  1432 + "open-right-layout": "Saðdaki gösterge tablosu düzeni (mobil görünüm)"
  1433 + },
  1434 + "widgets-bundle": {
  1435 + "current": "Mevcut paket",
  1436 + "widgets-bundles": "Widget Paketleri",
  1437 + "add": "Widget Paketi Ekle",
  1438 + "delete": "Widget paketini sil",
  1439 + "title": "Baþlýk",
  1440 + "title-required": "Baþlýk gerekli.",
  1441 + "add-widgets-bundle-text": "Yeni widget'lar paketi ekle",
  1442 + "no-widgets-bundles-text": "Widget paketi bulunamadý",
  1443 + "empty": "Widget'ler paketi boþ",
  1444 + "details": "Ayrýntýlar",
  1445 + "widgets-bundle-details": "Widget'lar paket ayrýntýlarýný",
  1446 + "delete-widgets-bundle-title": "Widget paketini {{widgetsBundleTitle}} silmek istediðinizden emin misiniz?",
  1447 + "delete-widgets-bundle-text": "Dikkatli olun, onaylandýktan sonra widget'lar paketi ve tüm ilgili veriler kurtarýlamayacak.",
  1448 + "delete-widgets-bundles-title": "{Count, çoðul, 1 {1 widgets bundle} diðer {# widgets bundles}} silmek istediðinizden emin misiniz?",
  1449 + "delete-widgets-bundles-action-title": "Sil {count, çoðul, 1 {1 widgets bundle} diðer {# widgets bundles}}",
  1450 + "delete-widgets-bundles-text": "Dikkatli olun, onaylandýktan sonra tüm seçilen widget'lar paketler kaldýrýlacak ve ilgili tüm veriler kurtarýlamayacaktýr.",
  1451 + "no-widgets-bundles-matching": "{{WidgetsBundle}} ile eþleþen hiçbir widget grubu bulunamadý. ",
  1452 + "widgets-bundle-required": "Widget paketi gerekli.",
  1453 + "system": "Sistem",
  1454 + "import": "Widget paketlerini içe aktar",
  1455 + "export": "Widget paketini dýþa aktar",
  1456 + "export-failed-error": "Widget grubu dýþa aktarýlamadý: {{error}}",
  1457 + "create-new-widgets-bundle": "Yeni widget'lar paketi oluþtur",
  1458 + "widgets-bundle-file": "Widgets paket dosyasý",
  1459 + "invalid-widgets-bundle-file-error": "Widget grubu içe aktarýlamýyor: Geçersiz widget'lar veri yapýsýný paketliyor."
  1460 + },
  1461 + "widget-config": {
  1462 + "data": "Veri",
  1463 + "settings": "Ayarlar",
  1464 + "advanced": "Ýleri",
  1465 + "title": "Baþlýk",
  1466 + "general-settings": "Genel Ayarlar",
  1467 + "display-title": "Baþlýk",
  1468 + "drop-shadow": "Düþen gölge",
  1469 + "enable-fullscreen": "Tam ekraný etkinleþtir",
  1470 + "background-color": "Arka plan rengi",
  1471 + "text-color": "Metin rengi",
  1472 + "padding": "Dolgu malzemesi",
  1473 + "margin": "Kenar",
  1474 + "widget-style": "Widget stili",
  1475 + "title-style": "Baþlýk stili",
  1476 + "mobile-mode-settings": "Mobil mod ayarlarý",
  1477 + "order": "Sipariþ",
  1478 + "height": "Yükseklik",
  1479 + "units": "Deðerin yanýnda gösterilecek özel sembol",
  1480 + "decimals": "Kayan noktadan sonraki basamak sayýsý",
  1481 + "timewindow": "Timewindow",
  1482 + "use-dashboard-timewindow": "Gösterge panosu zaman tüneli",
  1483 + "display-legend": "Gösterge efsanesi",
  1484 + "datasources": "Veri kaynaklarý",
  1485 + "maximum-datasources": "Maksimum {sayým, çoðul, 1 {1 veri kaynaðý izinli.} Diðer {# veri kaynaklarýna izin verilir}}",
  1486 + "datasource-type": "Tür",
  1487 + "datasource-parameters": "Parametreler",
  1488 + "remove-datasource": "Veri kaynaðýný kaldýr",
  1489 + "add-datasource": "Veri kaynaðý ekle",
  1490 + "target-device": "Hedef cihaz",
  1491 + "alarm-source": "Alarm kaynaðý",
  1492 + "actions": "Ýþlemler",
  1493 + "action": "Aksiyon",
  1494 + "add-action": "Eylem ekle",
  1495 + "search-actions": "Arama iþlemleri",
  1496 + "action-source": "Eylem kaynaðý",
  1497 + "action-source-required": "Eylem kaynaðý gerekli.",
  1498 + "action-name": "Ad",
  1499 + "action-name-required": "Eylem adý gerekli.",
  1500 + "action-name-not-unique": "Ayný ada sahip baþka bir eylem zaten var. <br/> Eylem adý, ayný eylem kaynaðý içinde benzersiz olmalýdýr.",
  1501 + "action-icon": "Simge",
  1502 + "action-type": "Tür",
  1503 + "action-type-required": "Eylem türü gerekli",
  1504 + "edit-action": "Eylemi düzenle",
  1505 + "delete-action": "Eylemi sil",
  1506 + "delete-action-title": "Widget eylemini sil",
  1507 + "delete-action-text": "Widget eylemini {{actionName}} adýyla silmek istediðinizden emin misiniz?"
  1508 + },
  1509 + "widget-type": {
  1510 + "import": "Widget türü içe aktar",
  1511 + "export": "Widget türünü dýþa aktar",
  1512 + "export-failed-error": "Widget türü verilemiyor: {{error}}",
  1513 + "create-new-widget-type": "Yeni pencere öðesi türü oluþtur",
  1514 + "widget-type-file": "Widget tipi dosya",
  1515 + "invalid-widget-type-file-error": "Widget türü içe aktarýlamýyor: Geçersiz pencere öðesi türü yapýsý."
  1516 + },
  1517 + "icon": {
  1518 + "icon": "Simge",
  1519 + "select-icon": "Simge seç",
  1520 + "material-icons": "Malzeme simgeleri",
  1521 + "show-all": "Tüm simgeleri göster"
  1522 + },
  1523 + "custom": {
  1524 + "widget-action": {
  1525 + "action-cell-button": "Eylem hücresi düðmesi",
  1526 + "row-click": "Satýr týklamasý",
  1527 + "marker-click": "Ýþaretçi týklamasýnda ",
  1528 + "tooltip-tag-action": "Araç ipucu etiketi iþlemi"
  1529 + }
  1530 + },
  1531 + "language": {
  1532 + "language": "Dil",
  1533 + "locales": {
  1534 + "fr_FR": "Fransýzca",
  1535 + "zh_CN": "Çince",
  1536 + "en_US": "Ýngilizce",
  1537 + "it_IT": "Ýtalyan",
  1538 + "ko_KR": "Koreli",
  1539 + "ru_RU": "Rusça",
  1540 + "es_ES": "Ýspanyol",
  1541 + "ja_JA": "Japonca",
  1542 + "TR": "Türk"
  1543 + }
  1544 + }
  1545 +}
... ...
... ... @@ -158,12 +158,17 @@
158 158 "filter-type-device-type": "Device type",
159 159 "filter-type-device-type-description": "Devices of type '{{deviceType}}'",
160 160 "filter-type-device-type-and-name-description": "Devices of type '{{deviceType}}' and with name starting with '{{prefix}}'",
  161 + "filter-type-entity-view-type": "Entity View type",
  162 + "filter-type-entity-view-type-description": "Entity Views of type '{{entityView}}'",
  163 + "filter-type-entity-view-type-and-name-description": "Entity Views of type '{{entityView}}' and with name starting with '{{prefix}}'",
161 164 "filter-type-relations-query": "Relations query",
162 165 "filter-type-relations-query-description": "{{entities}} that have {{relationType}} relation {{direction}} {{rootEntity}}",
163 166 "filter-type-asset-search-query": "Asset search query",
164 167 "filter-type-asset-search-query-description": "Assets with types {{assetTypes}} that have {{relationType}} relation {{direction}} {{rootEntity}}",
165 168 "filter-type-device-search-query": "Device search query",
166 169 "filter-type-device-search-query-description": "Devices with types {{deviceTypes}} that have {{relationType}} relation {{direction}} {{rootEntity}}",
  170 + "filter-type-entity-view-search-query": "Entity view search query",
  171 + "filter-type-entity-view-search-query-description": "Entity views with types {{entityViewTypes}} that have {{relationType}} relation {{direction}} {{rootEntity}}",
167 172 "entity-filter": "Entity filter",
168 173 "resolve-multiple": "Resolve as multiple entities",
169 174 "filter-type": "Filter type",
... ... @@ -839,7 +844,8 @@
839 844 "client-attributes": "Client attributes",
840 845 "shared-attributes": "Shared attributes",
841 846 "server-attributes": "Server attributes",
842   - "latest-timeseries": "Latest timeseries"
  847 + "latest-timeseries": "Latest timeseries",
  848 + "related-entity": "Related entity"
843 849 },
844 850 "event": {
845 851 "event-type": "Event type",
... ... @@ -1551,7 +1557,8 @@
1551 1557 "ko_KR": "Korean",
1552 1558 "ru_RU": "Russian",
1553 1559 "es_ES": "Spanish",
1554   - "ja_JA": "Japanese"
  1560 + "ja_JA": "Japanese",
  1561 + "TR": "Turkish"
1555 1562 }
1556 1563 }
1557 1564 }
... ...