Commit c78823332b5d016095557a7248c7ea96794f8cd5
Merge branch 'master' of github.com:thingsboard/thingsboard into edge/refactoring
Showing
61 changed files
with
1103 additions
and
335 deletions
... | ... | @@ -455,6 +455,24 @@ |
455 | 455 | "dataKeySettingsSchema": "{}\n", |
456 | 456 | "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\":\"Photo camera input\",\"showTitleIcon\":false,\"titleIcon\":\"more_horiz\",\"iconColor\":\"rgba(0, 0, 0, 0.87)\",\"iconSize\":\"24px\",\"titleTooltip\":\"\",\"dropShadow\":true,\"enableFullscreen\":false,\"widgetStyle\":{},\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"useDashboardTimewindow\":true,\"displayTimewindow\":true,\"showLegend\":false,\"actions\":{}}" |
457 | 457 | } |
458 | + }, | |
459 | + { | |
460 | + "alias": "update_json_attribute", | |
461 | + "name": "Update JSON attribute", | |
462 | + "image": "", | |
463 | + "description": "Simple form to input new JSON value for pre-defined attribute/timeseries key.", | |
464 | + "descriptor": { | |
465 | + "type": "latest", | |
466 | + "sizeX": 7.5, | |
467 | + "sizeY": 3, | |
468 | + "resources": [], | |
469 | + "templateHtml": "<tb-json-input-widget \n [ctx]=\"ctx\">\n</tb-json-input-widget>", | |
470 | + "templateCss": ".attribute-update-form {\n overflow: hidden;\n height: 100%;\n display: flex;\n flex-direction: column;\n}\n\n.attribute-update-form__grid {\n display: flex;\n}\n.grid__element:first-child {\n flex: 1;\n}\n.grid__element:last-child {\n margin-top: 19px;\n margin-left: 7px;\n}\n.grid__element {\n display: flex;\n}\n\n.attribute-update-form .mat-button.mat-icon-button {\n width: 32px;\n min-width: 32px;\n height: 32px;\n min-height: 32px;\n padding: 0 !important;\n margin: 0 !important;\n line-height: 20px;\n}\n\n.attribute-update-form .mat-icon-button mat-icon {\n width: 20px;\n min-width: 20px;\n height: 20px;\n min-height: 20px;\n font-size: 20px;\n}\n\n.tb-toast {\n font-size: 14px!important;\n}", | |
471 | + "controllerScript": "self.onInit = function() {\n}\n\nself.onDataUpdated = function() {\n self.ctx.$scope.jsonInputWidget.onDataUpdated();\n}\n\nself.onResize = function() {\n}\n\nself.typeParameters = function() {\n return {\n maxDatasources: 1,\n maxDataKeys: 1,\n singleEntity: true\n }\n}\n\nself.onDestroy = function() {\n}", | |
472 | + "settingsSchema": "{\n \"schema\": {\n \"type\": \"object\",\n \"title\": \"AdvancedSettings\",\n \"properties\": {\n \"widgetTitle\": {\n \"title\": \"Widget title\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"widgetMode\": {\n \"title\": \"Widget mode\",\n \"type\": \"string\",\n \"default\": \"ATTRIBUTE\"\n },\n \"attributeScope\": {\n \"title\": \"Attribute scope\",\n \"type\": \"string\",\n \"default\": \"SERVER_SCOPE\"\n },\n \"showLabel\":{\n \"title\": \"Show label\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"labelValue\": {\n \"title\": \"Label\",\n \"type\": \"string\",\n \"default\": \"\"\n },\n \"attributeRequired\": {\n \"title\": \"Value required\",\n \"type\": \"boolean\",\n \"default\": true\n },\n \"showResultMessage\": {\n \"title\": \"Show result message\",\n \"type\": \"boolean\",\n \"default\": true\n }\n },\n \"required\": []\n },\n \"form\": [\n \"widgetTitle\",\n {\n \"key\": \"widgetMode\",\n \"type\": \"rc-select\",\n \"multiple\": false,\n \"items\": [\n {\n \"value\": \"ATTRIBUTE\",\n \"label\": \"Update attribute\"\n },\n {\n \"value\": \"TIME_SERIES\",\n \"label\": \"Update timeseries\"\n }\n ]\n },\n {\n \"key\": \"attributeScope\",\n \"type\": \"rc-select\",\n \"multiple\": false,\n \"condition\": \"model.widgetMode === 'ATTRIBUTE'\",\n \"items\": [\n {\n \"value\": \"SERVER_SCOPE\",\n \"label\": \"Server attribute\"\n },\n {\n \"value\": \"SHARED_SCOPE\",\n \"label\": \"Shared attribute\"\n }\n ]\n },\n \"showLabel\",\n {\n \"key\": \"labelValue\",\n \"condition\": \"model.showLabel\"\n },\n \"attributeRequired\",\n \"showResultMessage\"\n ]\n}", | |
473 | + "dataKeySettingsSchema": "{}", | |
474 | + "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\":{\"attributeScope\":\"SERVER_SCOPE\",\"showLabel\":true,\"attributeRequired\":true,\"showResultMessage\":true},\"title\":\"Update JSON attribute\",\"showTitleIcon\":false,\"iconColor\":\"rgba(0, 0, 0, 0.87)\",\"iconSize\":\"24px\",\"titleTooltip\":\"\",\"dropShadow\":true,\"enableFullscreen\":false,\"widgetStyle\":{},\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"showLegend\":false}" | |
475 | + } | |
458 | 476 | } |
459 | 477 | ] |
460 | 478 | } | ... | ... |
... | ... | @@ -21,7 +21,7 @@ import org.springframework.web.bind.annotation.RequestMapping; |
21 | 21 | @Controller |
22 | 22 | public class WebConfig { |
23 | 23 | |
24 | - @RequestMapping(value = "/{path:^(?!api$)(?!assets$)(?!static$)(?!webjars$)[^\\.]*}/**") | |
24 | + @RequestMapping(value = {"/assets", "/assets/", "/{path:^(?!api$)(?!assets$)(?!static$)(?!webjars$)[^\\.]*}/**"}) | |
25 | 25 | public String redirect() { |
26 | 26 | return "forward:/index.html"; |
27 | 27 | } | ... | ... |
application/src/main/java/org/thingsboard/server/service/install/DefaultSystemDataLoaderService.java
... | ... | @@ -17,8 +17,10 @@ package org.thingsboard.server.service.install; |
17 | 17 | |
18 | 18 | import com.fasterxml.jackson.databind.ObjectMapper; |
19 | 19 | import com.fasterxml.jackson.databind.node.ObjectNode; |
20 | +import lombok.Getter; | |
20 | 21 | import lombok.extern.slf4j.Slf4j; |
21 | 22 | import org.springframework.beans.factory.annotation.Autowired; |
23 | +import org.springframework.beans.factory.annotation.Value; | |
22 | 24 | import org.springframework.context.annotation.Bean; |
23 | 25 | import org.springframework.context.annotation.Profile; |
24 | 26 | import org.springframework.security.crypto.bcrypt.BCryptPasswordEncoder; | ... | ... |
... | ... | @@ -40,7 +40,7 @@ import org.thingsboard.server.transport.lwm2m.utils.TypeServer; |
40 | 40 | |
41 | 41 | import java.io.IOException; |
42 | 42 | import java.security.GeneralSecurityException; |
43 | -import java.util.Arrays; | |
43 | +import java.util.Collections; | |
44 | 44 | import java.util.List; |
45 | 45 | import java.util.UUID; |
46 | 46 | |
... | ... | @@ -70,8 +70,7 @@ public class LwM2MBootstrapSecurityStore implements BootstrapSecurityStore { |
70 | 70 | |
71 | 71 | @Override |
72 | 72 | public List<SecurityInfo> getAllByEndpoint(String endPoint) { |
73 | - String endPointKey = endPoint; | |
74 | - ReadResultSecurityStore store = lwM2MCredentialsSecurityInfoValidator.createAndValidateCredentialsSecurityInfo(endPointKey, TypeServer.BOOTSTRAP); | |
73 | + ReadResultSecurityStore store = lwM2MCredentialsSecurityInfoValidator.createAndValidateCredentialsSecurityInfo(endPoint, TypeServer.BOOTSTRAP); | |
75 | 74 | if (store.getBootstrapJsonCredential() != null && store.getSecurityMode() < LwM2MSecurityMode.DEFAULT_MODE.code) { |
76 | 75 | /** add value to store from BootstrapJson */ |
77 | 76 | this.setBootstrapConfigScurityInfo(store); |
... | ... | @@ -87,7 +86,7 @@ public class LwM2MBootstrapSecurityStore implements BootstrapSecurityStore { |
87 | 86 | } catch (InvalidConfigurationException e) { |
88 | 87 | log.error("", e); |
89 | 88 | } |
90 | - return store.getSecurityInfo() == null ? null : Arrays.asList(store.getSecurityInfo()); | |
89 | + return store.getSecurityInfo() == null ? null : Collections.singletonList(store.getSecurityInfo()); | |
91 | 90 | } |
92 | 91 | } |
93 | 92 | return null; | ... | ... |
common/transport/lwm2m/src/main/java/org/thingsboard/server/transport/lwm2m/server/LwM2mQueuedRequest.java
renamed from
ui-ngx/src/app/shared/components/time/quick-time-interval.component.scss
... | ... | @@ -13,7 +13,8 @@ |
13 | 13 | * See the License for the specific language governing permissions and |
14 | 14 | * limitations under the License. |
15 | 15 | */ |
16 | +package org.thingsboard.server.transport.lwm2m.server; | |
16 | 17 | |
17 | -:host { | |
18 | - min-width: 364px; | |
18 | +public interface LwM2mQueuedRequest { | |
19 | + void send(); | |
19 | 20 | } | ... | ... |
... | ... | @@ -16,6 +16,7 @@ |
16 | 16 | package org.thingsboard.server.transport.lwm2m.server; |
17 | 17 | |
18 | 18 | import lombok.extern.slf4j.Slf4j; |
19 | +import org.eclipse.californium.core.coap.CoAP; | |
19 | 20 | import org.eclipse.californium.core.coap.Response; |
20 | 21 | import org.eclipse.leshan.core.attributes.Attribute; |
21 | 22 | import org.eclipse.leshan.core.attributes.AttributeSet; |
... | ... | @@ -34,6 +35,7 @@ import org.eclipse.leshan.core.request.ObserveRequest; |
34 | 35 | import org.eclipse.leshan.core.request.ReadRequest; |
35 | 36 | import org.eclipse.leshan.core.request.WriteAttributesRequest; |
36 | 37 | import org.eclipse.leshan.core.request.WriteRequest; |
38 | +import org.eclipse.leshan.core.request.exception.ClientSleepingException; | |
37 | 39 | import org.eclipse.leshan.core.response.CancelObservationResponse; |
38 | 40 | import org.eclipse.leshan.core.response.DeleteResponse; |
39 | 41 | import org.eclipse.leshan.core.response.DiscoverResponse; |
... | ... | @@ -58,7 +60,6 @@ import java.util.Date; |
58 | 60 | import java.util.concurrent.ExecutorService; |
59 | 61 | import java.util.concurrent.Executors; |
60 | 62 | |
61 | -import static org.eclipse.californium.core.coap.CoAP.ResponseCode.isSuccess; | |
62 | 63 | import static org.eclipse.leshan.core.attributes.Attribute.MINIMUM_PERIOD; |
63 | 64 | import static org.thingsboard.server.transport.lwm2m.server.LwM2mTransportHandler.DEFAULT_TIMEOUT; |
64 | 65 | import static org.thingsboard.server.transport.lwm2m.server.LwM2mTransportHandler.GET_TYPE_OPER_DISCOVER; |
... | ... | @@ -215,13 +216,19 @@ public class LwM2mTransportRequest { |
215 | 216 | request = new WriteAttributesRequest(resultIds.getObjectId(), attrSet); |
216 | 217 | } |
217 | 218 | break; |
218 | - default: | |
219 | 219 | } |
220 | 220 | |
221 | 221 | if (request != null) { |
222 | - this.sendRequest(registration, lwM2MClient, request, timeoutInMs); | |
223 | - } | |
224 | - else { | |
222 | + try { | |
223 | + this.sendRequest(registration, lwM2MClient, request, timeoutInMs); | |
224 | + } catch (ClientSleepingException e) { | |
225 | + DownlinkRequest finalRequest = request; | |
226 | + long finalTimeoutInMs = timeoutInMs; | |
227 | + lwM2MClient.getQueuedRequests().add(() -> sendRequest(registration, lwM2MClient, finalRequest, finalTimeoutInMs)); | |
228 | + } catch (Exception e) { | |
229 | + log.error("[{}] [{}] [{}] Failed to send downlink.", registration.getEndpoint(), targetIdVer, typeOper, e); | |
230 | + } | |
231 | + } else { | |
225 | 232 | log.error("[{}], [{}] - [{}] error SendRequest", registration.getEndpoint(), typeOper, targetIdVer); |
226 | 233 | } |
227 | 234 | } |
... | ... | @@ -240,7 +247,7 @@ public class LwM2mTransportRequest { |
240 | 247 | if (!lwM2MClient.isInit()) { |
241 | 248 | lwM2MClient.initValue(this.serviceImpl, convertToIdVerFromObjectId(request.getPath().toString(), registration)); |
242 | 249 | } |
243 | - if (isSuccess(((Response) response.getCoapResponse()).getCode())) { | |
250 | + if (CoAP.ResponseCode.isSuccess(((Response) response.getCoapResponse()).getCode())) { | |
244 | 251 | this.handleResponse(registration, request.getPath().toString(), response, request); |
245 | 252 | if (request instanceof WriteRequest && ((WriteRequest) request).isReplaceRequest()) { |
246 | 253 | String msg = String.format("%s: sendRequest Replace: CoapCde - %s Lwm2m code - %d name - %s Resource path - %s value - %s SendRequest to Client", |
... | ... | @@ -283,7 +290,7 @@ public class LwM2mTransportRequest { |
283 | 290 | case FLOAT: // Double |
284 | 291 | return (contentFormat == null) ? new WriteRequest(objectId, instanceId, resourceId, Double.parseDouble(value.toString())) : new WriteRequest(contentFormat, objectId, instanceId, resourceId, Double.parseDouble(value.toString())); |
285 | 292 | case TIME: // Date |
286 | - Date date = new Date(Long.decode(value.toString())); | |
293 | + Date date = new Date(Long.decode(value.toString())); | |
287 | 294 | return (contentFormat == null) ? new WriteRequest(objectId, instanceId, resourceId, date) : new WriteRequest(contentFormat, objectId, instanceId, resourceId, date); |
288 | 295 | case OPAQUE: // byte[] value, base64 |
289 | 296 | return (contentFormat == null) ? new WriteRequest(objectId, instanceId, resourceId, Hex.decodeHex(value.toString().toCharArray())) : new WriteRequest(contentFormat, objectId, instanceId, resourceId, Hex.decodeHex(value.toString().toCharArray())); | ... | ... |
... | ... | @@ -146,9 +146,9 @@ public class LwM2mTransportServiceImpl implements LwM2mTransportService { |
146 | 146 | * Next -> Create new LwM2MClient for current session -> setModelClient... |
147 | 147 | * |
148 | 148 | * @param registration - Registration LwM2M Client |
149 | - * @param previousObsersations - may be null | |
149 | + * @param previousObservations - may be null | |
150 | 150 | */ |
151 | - public void onRegistered(Registration registration, Collection<Observation> previousObsersations) { | |
151 | + public void onRegistered(Registration registration, Collection<Observation> previousObservations) { | |
152 | 152 | executorRegistered.submit(() -> { |
153 | 153 | try { |
154 | 154 | log.warn("[{}] [{{}] Client: create after Registration", registration.getEndpoint(), registration.getId()); |
... | ... | @@ -188,6 +188,13 @@ public class LwM2mTransportServiceImpl implements LwM2mTransportService { |
188 | 188 | LwM2mClient lwM2MClient = this.lwM2mClientContext.getLwM2MClient(sessionInfo); |
189 | 189 | if (lwM2MClient.getDeviceId() == null && lwM2MClient.getProfileId() == null) { |
190 | 190 | initLwM2mClient(lwM2MClient, sessionInfo); |
191 | + } else { | |
192 | + if (registration.getBindingMode().useQueueMode()) { | |
193 | + LwM2mQueuedRequest request; | |
194 | + while ((request = lwM2MClient.getQueuedRequests().poll()) != null) { | |
195 | + request.send(); | |
196 | + } | |
197 | + } | |
191 | 198 | } |
192 | 199 | |
193 | 200 | log.info("Client: [{}] updatedReg [{}] name [{}] profile ", registration.getId(), registration.getEndpoint(), sessionInfo.getDeviceType()); |
... | ... | @@ -206,7 +213,7 @@ public class LwM2mTransportServiceImpl implements LwM2mTransportService { |
206 | 213 | * !!! Warn: if have not finishing unReg, then this operation will be finished on next Client`s connect |
207 | 214 | */ |
208 | 215 | public void unReg(Registration registration, Collection<Observation> observations) { |
209 | - executorUnRegistered.submit(() -> { | |
216 | + executorUnRegistered.submit(() -> { | |
210 | 217 | try { |
211 | 218 | this.setCancelObservations(registration); |
212 | 219 | this.sendLogsToThingsboard(LOG_LW2M_INFO + ": Client unRegistration", registration); |
... | ... | @@ -239,8 +246,11 @@ public class LwM2mTransportServiceImpl implements LwM2mTransportService { |
239 | 246 | } |
240 | 247 | } |
241 | 248 | |
249 | + @Override | |
242 | 250 | public void onSleepingDev(Registration registration) { |
243 | 251 | log.info("[{}] [{}] Received endpoint Sleeping version event", registration.getId(), registration.getEndpoint()); |
252 | + this.sendLogsToThingsboard(LOG_LW2M_INFO + ": Client is sleeping!", registration); | |
253 | + | |
244 | 254 | //TODO: associate endpointId with device information. |
245 | 255 | } |
246 | 256 | |
... | ... | @@ -417,6 +427,7 @@ public class LwM2mTransportServiceImpl implements LwM2mTransportService { |
417 | 427 | */ |
418 | 428 | protected void onAwakeDev(Registration registration) { |
419 | 429 | log.info("[{}] [{}] Received endpoint Awake version event", registration.getId(), registration.getEndpoint()); |
430 | + this.sendLogsToThingsboard(LOG_LW2M_INFO + ": Client is awake!", registration); | |
420 | 431 | //TODO: associate endpointId with device information. |
421 | 432 | } |
422 | 433 | |
... | ... | @@ -612,7 +623,7 @@ public class LwM2mTransportServiceImpl implements LwM2mTransportService { |
612 | 623 | if (GET_TYPE_OPER_READ.equals(typeOper)) { |
613 | 624 | result = JacksonUtil.fromString(lwM2MClientProfile.getPostAttributeProfile().toString(), new TypeReference<>() { |
614 | 625 | }); |
615 | - result.addAll(JacksonUtil.fromString(lwM2MClientProfile.getPostTelemetryProfile().toString(), new TypeReference<>() { | |
626 | + result.addAll(JacksonUtil.convertValue(lwM2MClientProfile.getPostTelemetryProfile().toString(), new TypeReference<>() { | |
616 | 627 | })); |
617 | 628 | } else { |
618 | 629 | result = JacksonUtil.fromString(lwM2MClientProfile.getPostObserveProfile().toString(), new TypeReference<>() { | ... | ... |
... | ... | @@ -25,13 +25,16 @@ import org.eclipse.leshan.server.registration.Registration; |
25 | 25 | import org.eclipse.leshan.server.security.SecurityInfo; |
26 | 26 | import org.thingsboard.server.gen.transport.TransportProtos; |
27 | 27 | import org.thingsboard.server.gen.transport.TransportProtos.ValidateDeviceCredentialsResponseMsg; |
28 | +import org.thingsboard.server.transport.lwm2m.server.LwM2mQueuedRequest; | |
28 | 29 | import org.thingsboard.server.transport.lwm2m.server.LwM2mTransportServiceImpl; |
29 | 30 | |
30 | 31 | import java.util.List; |
31 | 32 | import java.util.Map; |
33 | +import java.util.Queue; | |
32 | 34 | import java.util.Set; |
33 | 35 | import java.util.UUID; |
34 | 36 | import java.util.concurrent.ConcurrentHashMap; |
37 | +import java.util.concurrent.ConcurrentLinkedQueue; | |
35 | 38 | import java.util.concurrent.CopyOnWriteArrayList; |
36 | 39 | import java.util.stream.Collectors; |
37 | 40 | |
... | ... | @@ -54,6 +57,7 @@ public class LwM2mClient implements Cloneable { |
54 | 57 | private final Map<String, ResourceValue> resources; |
55 | 58 | private final Map<String, TransportProtos.TsKvProto> delayedRequests; |
56 | 59 | private final List<String> pendingRequests; |
60 | + private final Queue<LwM2mQueuedRequest> queuedRequests; | |
57 | 61 | private boolean init; |
58 | 62 | |
59 | 63 | public Object clone() throws CloneNotSupportedException { |
... | ... | @@ -71,6 +75,7 @@ public class LwM2mClient implements Cloneable { |
71 | 75 | this.profileId = profileId; |
72 | 76 | this.sessionId = sessionId; |
73 | 77 | this.init = false; |
78 | + this.queuedRequests = new ConcurrentLinkedQueue<>(); | |
74 | 79 | } |
75 | 80 | |
76 | 81 | public boolean saveResourceValue(String pathRez, LwM2mResource rez, LwM2mModelProvider modelProvider) { | ... | ... |
... | ... | @@ -51,11 +51,11 @@ public abstract class AbstractComponentDescriptorInsertRepository implements Com |
51 | 51 | TransactionStatus transaction = getTransactionStatus(TransactionDefinition.PROPAGATION_REQUIRES_NEW); |
52 | 52 | try { |
53 | 53 | componentDescriptorEntity = processSaveOrUpdate(entity, insertOrUpdateOnUniqueKeyConflict); |
54 | + transactionManager.commit(transaction); | |
54 | 55 | } catch (Throwable th) { |
55 | 56 | log.trace("Could not execute the update statement for Component Descriptor with id {}, name {} and entityType {}", entity.getUuid(), entity.getName(), entity.getType()); |
56 | 57 | transactionManager.rollback(transaction); |
57 | 58 | } |
58 | - transactionManager.commit(transaction); | |
59 | 59 | } else { |
60 | 60 | log.trace("Could not execute the insert statement for Component Descriptor with id {}, name {} and entityType {}", entity.getUuid(), entity.getName(), entity.getType()); |
61 | 61 | } | ... | ... |
... | ... | @@ -51,11 +51,11 @@ public abstract class AbstractEventInsertRepository implements EventInsertReposi |
51 | 51 | TransactionStatus transaction = getTransactionStatus(TransactionDefinition.PROPAGATION_REQUIRES_NEW); |
52 | 52 | try { |
53 | 53 | eventEntity = processSaveOrUpdate(entity, insertOrUpdateOnUniqueKeyConflict); |
54 | + transactionManager.commit(transaction); | |
54 | 55 | } catch (Throwable th) { |
55 | 56 | log.trace("Could not execute the update statement for Entity with entityId {} and entityType {}", entity.getEventUid(), entity.getEventType()); |
56 | 57 | transactionManager.rollback(transaction); |
57 | 58 | } |
58 | - transactionManager.commit(transaction); | |
59 | 59 | } else { |
60 | 60 | log.trace("Could not execute the insert statement for Entity with entityId {} and entityType {}", entity.getEventUid(), entity.getEventType()); |
61 | 61 | } | ... | ... |
... | ... | @@ -62,6 +62,8 @@ import java.util.Collections; |
62 | 62 | import java.util.List; |
63 | 63 | import java.util.Optional; |
64 | 64 | import java.util.concurrent.TimeUnit; |
65 | +import java.util.concurrent.locks.Lock; | |
66 | +import java.util.concurrent.locks.ReentrantLock; | |
65 | 67 | import java.util.stream.Collectors; |
66 | 68 | |
67 | 69 | import static com.datastax.oss.driver.api.querybuilder.QueryBuilder.literal; |
... | ... | @@ -107,6 +109,7 @@ public class CassandraBaseTimeseriesDao extends AbstractCassandraBaseTimeseriesD |
107 | 109 | private PreparedStatement[] fetchStmtsDesc; |
108 | 110 | private PreparedStatement deleteStmt; |
109 | 111 | private PreparedStatement deletePartitionStmt; |
112 | + private final Lock stmtCreationLock = new ReentrantLock(); | |
110 | 113 | |
111 | 114 | private boolean isInstall() { |
112 | 115 | return environment.acceptsProfiles(Profiles.of("install")); |
... | ... | @@ -545,13 +548,20 @@ public class CassandraBaseTimeseriesDao extends AbstractCassandraBaseTimeseriesD |
545 | 548 | |
546 | 549 | private PreparedStatement getDeleteStmt() { |
547 | 550 | if (deleteStmt == null) { |
548 | - deleteStmt = prepare("DELETE FROM " + ModelConstants.TS_KV_CF + | |
549 | - " WHERE " + ModelConstants.ENTITY_TYPE_COLUMN + EQUALS_PARAM | |
550 | - + "AND " + ModelConstants.ENTITY_ID_COLUMN + EQUALS_PARAM | |
551 | - + "AND " + ModelConstants.KEY_COLUMN + EQUALS_PARAM | |
552 | - + "AND " + ModelConstants.PARTITION_COLUMN + EQUALS_PARAM | |
553 | - + "AND " + ModelConstants.TS_COLUMN + " >= ? " | |
554 | - + "AND " + ModelConstants.TS_COLUMN + " < ?"); | |
551 | + stmtCreationLock.lock(); | |
552 | + try { | |
553 | + if (deleteStmt == null) { | |
554 | + deleteStmt = prepare("DELETE FROM " + ModelConstants.TS_KV_CF + | |
555 | + " WHERE " + ModelConstants.ENTITY_TYPE_COLUMN + EQUALS_PARAM | |
556 | + + "AND " + ModelConstants.ENTITY_ID_COLUMN + EQUALS_PARAM | |
557 | + + "AND " + ModelConstants.KEY_COLUMN + EQUALS_PARAM | |
558 | + + "AND " + ModelConstants.PARTITION_COLUMN + EQUALS_PARAM | |
559 | + + "AND " + ModelConstants.TS_COLUMN + " >= ? " | |
560 | + + "AND " + ModelConstants.TS_COLUMN + " < ?"); | |
561 | + } | |
562 | + } finally { | |
563 | + stmtCreationLock.unlock(); | |
564 | + } | |
555 | 565 | } |
556 | 566 | return deleteStmt; |
557 | 567 | } |
... | ... | @@ -585,27 +595,41 @@ public class CassandraBaseTimeseriesDao extends AbstractCassandraBaseTimeseriesD |
585 | 595 | |
586 | 596 | private PreparedStatement getDeletePartitionStmt() { |
587 | 597 | if (deletePartitionStmt == null) { |
588 | - deletePartitionStmt = prepare("DELETE FROM " + ModelConstants.TS_KV_PARTITIONS_CF + | |
589 | - " WHERE " + ModelConstants.ENTITY_TYPE_COLUMN + EQUALS_PARAM | |
590 | - + "AND " + ModelConstants.ENTITY_ID_COLUMN + EQUALS_PARAM | |
591 | - + "AND " + ModelConstants.PARTITION_COLUMN + EQUALS_PARAM | |
592 | - + "AND " + ModelConstants.KEY_COLUMN + EQUALS_PARAM); | |
598 | + stmtCreationLock.lock(); | |
599 | + try { | |
600 | + if (deletePartitionStmt == null) { | |
601 | + deletePartitionStmt = prepare("DELETE FROM " + ModelConstants.TS_KV_PARTITIONS_CF + | |
602 | + " WHERE " + ModelConstants.ENTITY_TYPE_COLUMN + EQUALS_PARAM | |
603 | + + "AND " + ModelConstants.ENTITY_ID_COLUMN + EQUALS_PARAM | |
604 | + + "AND " + ModelConstants.PARTITION_COLUMN + EQUALS_PARAM | |
605 | + + "AND " + ModelConstants.KEY_COLUMN + EQUALS_PARAM); | |
606 | + } | |
607 | + } finally { | |
608 | + stmtCreationLock.unlock(); | |
609 | + } | |
593 | 610 | } |
594 | 611 | return deletePartitionStmt; |
595 | 612 | } |
596 | 613 | |
597 | 614 | private PreparedStatement getSaveStmt(DataType dataType) { |
598 | 615 | if (saveStmts == null) { |
599 | - saveStmts = new PreparedStatement[DataType.values().length]; | |
600 | - for (DataType type : DataType.values()) { | |
601 | - saveStmts[type.ordinal()] = prepare(INSERT_INTO + ModelConstants.TS_KV_CF + | |
602 | - "(" + ModelConstants.ENTITY_TYPE_COLUMN + | |
603 | - "," + ModelConstants.ENTITY_ID_COLUMN + | |
604 | - "," + ModelConstants.KEY_COLUMN + | |
605 | - "," + ModelConstants.PARTITION_COLUMN + | |
606 | - "," + ModelConstants.TS_COLUMN + | |
607 | - "," + getColumnName(type) + ")" + | |
608 | - " VALUES(?, ?, ?, ?, ?, ?)"); | |
616 | + stmtCreationLock.lock(); | |
617 | + try { | |
618 | + if (saveStmts == null) { | |
619 | + saveStmts = new PreparedStatement[DataType.values().length]; | |
620 | + for (DataType type : DataType.values()) { | |
621 | + saveStmts[type.ordinal()] = prepare(INSERT_INTO + ModelConstants.TS_KV_CF + | |
622 | + "(" + ModelConstants.ENTITY_TYPE_COLUMN + | |
623 | + "," + ModelConstants.ENTITY_ID_COLUMN + | |
624 | + "," + ModelConstants.KEY_COLUMN + | |
625 | + "," + ModelConstants.PARTITION_COLUMN + | |
626 | + "," + ModelConstants.TS_COLUMN + | |
627 | + "," + getColumnName(type) + ")" + | |
628 | + " VALUES(?, ?, ?, ?, ?, ?)"); | |
629 | + } | |
630 | + } | |
631 | + } finally { | |
632 | + stmtCreationLock.unlock(); | |
609 | 633 | } |
610 | 634 | } |
611 | 635 | return saveStmts[dataType.ordinal()]; |
... | ... | @@ -613,16 +637,23 @@ public class CassandraBaseTimeseriesDao extends AbstractCassandraBaseTimeseriesD |
613 | 637 | |
614 | 638 | private PreparedStatement getSaveTtlStmt(DataType dataType) { |
615 | 639 | if (saveTtlStmts == null) { |
616 | - saveTtlStmts = new PreparedStatement[DataType.values().length]; | |
617 | - for (DataType type : DataType.values()) { | |
618 | - saveTtlStmts[type.ordinal()] = prepare(INSERT_INTO + ModelConstants.TS_KV_CF + | |
619 | - "(" + ModelConstants.ENTITY_TYPE_COLUMN + | |
620 | - "," + ModelConstants.ENTITY_ID_COLUMN + | |
621 | - "," + ModelConstants.KEY_COLUMN + | |
622 | - "," + ModelConstants.PARTITION_COLUMN + | |
623 | - "," + ModelConstants.TS_COLUMN + | |
624 | - "," + getColumnName(type) + ")" + | |
625 | - " VALUES(?, ?, ?, ?, ?, ?) USING TTL ?"); | |
640 | + stmtCreationLock.lock(); | |
641 | + try { | |
642 | + if (saveTtlStmts == null) { | |
643 | + saveTtlStmts = new PreparedStatement[DataType.values().length]; | |
644 | + for (DataType type : DataType.values()) { | |
645 | + saveTtlStmts[type.ordinal()] = prepare(INSERT_INTO + ModelConstants.TS_KV_CF + | |
646 | + "(" + ModelConstants.ENTITY_TYPE_COLUMN + | |
647 | + "," + ModelConstants.ENTITY_ID_COLUMN + | |
648 | + "," + ModelConstants.KEY_COLUMN + | |
649 | + "," + ModelConstants.PARTITION_COLUMN + | |
650 | + "," + ModelConstants.TS_COLUMN + | |
651 | + "," + getColumnName(type) + ")" + | |
652 | + " VALUES(?, ?, ?, ?, ?, ?) USING TTL ?"); | |
653 | + } | |
654 | + } | |
655 | + } finally { | |
656 | + stmtCreationLock.unlock(); | |
626 | 657 | } |
627 | 658 | } |
628 | 659 | return saveTtlStmts[dataType.ordinal()]; |
... | ... | @@ -630,24 +661,38 @@ public class CassandraBaseTimeseriesDao extends AbstractCassandraBaseTimeseriesD |
630 | 661 | |
631 | 662 | private PreparedStatement getPartitionInsertStmt() { |
632 | 663 | if (partitionInsertStmt == null) { |
633 | - partitionInsertStmt = prepare(INSERT_INTO + ModelConstants.TS_KV_PARTITIONS_CF + | |
634 | - "(" + ModelConstants.ENTITY_TYPE_COLUMN + | |
635 | - "," + ModelConstants.ENTITY_ID_COLUMN + | |
636 | - "," + ModelConstants.PARTITION_COLUMN + | |
637 | - "," + ModelConstants.KEY_COLUMN + ")" + | |
638 | - " VALUES(?, ?, ?, ?)"); | |
664 | + stmtCreationLock.lock(); | |
665 | + try { | |
666 | + if (partitionInsertStmt == null) { | |
667 | + partitionInsertStmt = prepare(INSERT_INTO + ModelConstants.TS_KV_PARTITIONS_CF + | |
668 | + "(" + ModelConstants.ENTITY_TYPE_COLUMN + | |
669 | + "," + ModelConstants.ENTITY_ID_COLUMN + | |
670 | + "," + ModelConstants.PARTITION_COLUMN + | |
671 | + "," + ModelConstants.KEY_COLUMN + ")" + | |
672 | + " VALUES(?, ?, ?, ?)"); | |
673 | + } | |
674 | + } finally { | |
675 | + stmtCreationLock.unlock(); | |
676 | + } | |
639 | 677 | } |
640 | 678 | return partitionInsertStmt; |
641 | 679 | } |
642 | 680 | |
643 | 681 | private PreparedStatement getPartitionInsertTtlStmt() { |
644 | 682 | if (partitionInsertTtlStmt == null) { |
645 | - partitionInsertTtlStmt = prepare(INSERT_INTO + ModelConstants.TS_KV_PARTITIONS_CF + | |
646 | - "(" + ModelConstants.ENTITY_TYPE_COLUMN + | |
647 | - "," + ModelConstants.ENTITY_ID_COLUMN + | |
648 | - "," + ModelConstants.PARTITION_COLUMN + | |
649 | - "," + ModelConstants.KEY_COLUMN + ")" + | |
650 | - " VALUES(?, ?, ?, ?) USING TTL ?"); | |
683 | + stmtCreationLock.lock(); | |
684 | + try { | |
685 | + if (partitionInsertTtlStmt == null) { | |
686 | + partitionInsertTtlStmt = prepare(INSERT_INTO + ModelConstants.TS_KV_PARTITIONS_CF + | |
687 | + "(" + ModelConstants.ENTITY_TYPE_COLUMN + | |
688 | + "," + ModelConstants.ENTITY_ID_COLUMN + | |
689 | + "," + ModelConstants.PARTITION_COLUMN + | |
690 | + "," + ModelConstants.KEY_COLUMN + ")" + | |
691 | + " VALUES(?, ?, ?, ?) USING TTL ?"); | |
692 | + } | |
693 | + } finally { | |
694 | + stmtCreationLock.unlock(); | |
695 | + } | |
651 | 696 | } |
652 | 697 | return partitionInsertTtlStmt; |
653 | 698 | } |
... | ... | @@ -713,12 +758,26 @@ public class CassandraBaseTimeseriesDao extends AbstractCassandraBaseTimeseriesD |
713 | 758 | switch (orderBy) { |
714 | 759 | case ASC_ORDER: |
715 | 760 | if (fetchStmtsAsc == null) { |
716 | - fetchStmtsAsc = initFetchStmt(orderBy); | |
761 | + stmtCreationLock.lock(); | |
762 | + try { | |
763 | + if (fetchStmtsAsc == null) { | |
764 | + fetchStmtsAsc = initFetchStmt(orderBy); | |
765 | + } | |
766 | + } finally { | |
767 | + stmtCreationLock.unlock(); | |
768 | + } | |
717 | 769 | } |
718 | 770 | return fetchStmtsAsc[aggType.ordinal()]; |
719 | 771 | case DESC_ORDER: |
720 | 772 | if (fetchStmtsDesc == null) { |
721 | - fetchStmtsDesc = initFetchStmt(orderBy); | |
773 | + stmtCreationLock.lock(); | |
774 | + try { | |
775 | + if (fetchStmtsDesc == null) { | |
776 | + fetchStmtsDesc = initFetchStmt(orderBy); | |
777 | + } | |
778 | + } finally { | |
779 | + stmtCreationLock.unlock(); | |
780 | + } | |
722 | 781 | } |
723 | 782 | return fetchStmtsDesc[aggType.ordinal()]; |
724 | 783 | default: | ... | ... |
... | ... | @@ -10,6 +10,7 @@ To run the black box tests with using Docker, the local Docker images of Thingsb |
10 | 10 | As result, in REPOSITORY column, next images should be present: |
11 | 11 | |
12 | 12 | thingsboard/tb-coap-transport |
13 | + thingsboard/tb-lwm2m-transport | |
13 | 14 | thingsboard/tb-http-transport |
14 | 15 | thingsboard/tb-mqtt-transport |
15 | 16 | thingsboard/tb-node | ... | ... |
... | ... | @@ -17,15 +17,14 @@ |
17 | 17 | FROM thingsboard/openjdk11 |
18 | 18 | |
19 | 19 | RUN apt-get update |
20 | -RUN apt-get install -y curl nmap procps | |
21 | -RUN echo 'deb http://ftp.us.debian.org/debian sid main' | tee --append /etc/apt/sources.list.d/debian.list > /dev/null | |
22 | -RUN echo 'deb http://apt.postgresql.org/pub/repos/apt/ sid-pgdg main' | tee --append /etc/apt/sources.list.d/pgdg.list > /dev/null | |
20 | +RUN apt-get install -y curl nmap procps gnupg2 | |
21 | +RUN echo 'deb http://apt.postgresql.org/pub/repos/apt/ buster-pgdg main' | tee --append /etc/apt/sources.list.d/pgdg.list > /dev/null | |
23 | 22 | RUN curl -L https://www.postgresql.org/media/keys/ACCC4CF8.asc | apt-key add - |
24 | 23 | RUN echo 'deb http://downloads.apache.org/cassandra/debian 40x main' | tee --append /etc/apt/sources.list.d/cassandra.list > /dev/null |
25 | 24 | RUN curl -L https://downloads.apache.org/cassandra/KEYS | apt-key add - |
26 | -ENV PG_MAJOR=11 | |
25 | +ENV PG_MAJOR=12 | |
27 | 26 | RUN apt-get update |
28 | -RUN apt-get install -y cassandra cassandra-tools postgresql-11 | |
27 | +RUN apt-get install -y cassandra cassandra-tools postgresql-12 | |
29 | 28 | RUN update-rc.d cassandra disable |
30 | 29 | RUN update-rc.d postgresql disable |
31 | 30 | RUN sed -i.old '/ulimit/d' /etc/init.d/cassandra | ... | ... |
... | ... | @@ -17,13 +17,12 @@ |
17 | 17 | FROM thingsboard/openjdk11 |
18 | 18 | |
19 | 19 | RUN apt-get update |
20 | -RUN apt-get install -y curl | |
21 | -RUN echo 'deb http://ftp.us.debian.org/debian sid main' | tee --append /etc/apt/sources.list.d/debian.list > /dev/null | |
22 | -RUN echo 'deb http://apt.postgresql.org/pub/repos/apt/ sid-pgdg main' | tee --append /etc/apt/sources.list.d/pgdg.list > /dev/null | |
20 | +RUN apt-get install -y curl gnupg2 | |
21 | +RUN echo 'deb http://apt.postgresql.org/pub/repos/apt/ buster-pgdg main' | tee --append /etc/apt/sources.list.d/pgdg.list > /dev/null | |
23 | 22 | RUN curl -L https://www.postgresql.org/media/keys/ACCC4CF8.asc | apt-key add - |
24 | -ENV PG_MAJOR 11 | |
23 | +ENV PG_MAJOR 12 | |
25 | 24 | RUN apt-get update |
26 | -RUN apt-get install -y postgresql-11 | |
25 | +RUN apt-get install -y postgresql-12 | |
27 | 26 | RUN update-rc.d postgresql disable |
28 | 27 | |
29 | 28 | COPY logback.xml ${pkg.name}.conf start-db.sh stop-db.sh start-tb.sh upgrade-tb.sh install-tb.sh ${pkg.name}.deb /tmp/ | ... | ... |
... | ... | @@ -39,7 +39,7 @@ |
39 | 39 | <tb-postgres.docker.name>tb-postgres</tb-postgres.docker.name> |
40 | 40 | <tb-cassandra.docker.name>tb-cassandra</tb-cassandra.docker.name> |
41 | 41 | <pkg.installFolder>/usr/share/${pkg.name}</pkg.installFolder> |
42 | - <pkg.upgradeVersion>2.4.2</pkg.upgradeVersion> | |
42 | + <pkg.upgradeVersion>3.3.0</pkg.upgradeVersion> | |
43 | 43 | </properties> |
44 | 44 | |
45 | 45 | <dependencies> | ... | ... |
... | ... | @@ -99,7 +99,7 @@ |
99 | 99 | org/thingsboard/server/extensions/core/plugin/telemetry/gen/**/* |
100 | 100 | </sonar.exclusions> |
101 | 101 | <elasticsearch.version>5.0.2</elasticsearch.version> |
102 | - <delight-nashorn-sandbox.version>0.1.31</delight-nashorn-sandbox.version> | |
102 | + <delight-nashorn-sandbox.version>0.1.16</delight-nashorn-sandbox.version> | |
103 | 103 | <kafka.version>2.6.0</kafka.version> |
104 | 104 | <bucket4j.version>4.1.1</bucket4j.version> |
105 | 105 | <fst.version>2.57</fst.version> | ... | ... |
... | ... | @@ -1338,7 +1338,7 @@ public class RestClient implements ClientHttpRequestInterceptor, Closeable { |
1338 | 1338 | HttpEntity.EMPTY, DeviceProfile.class, deviceProfileId).getBody(); |
1339 | 1339 | } |
1340 | 1340 | |
1341 | - public PageData<DeviceProfile> getTenantDevices(PageLink pageLink) { | |
1341 | + public PageData<DeviceProfile> getDeviceProfiles(PageLink pageLink) { | |
1342 | 1342 | Map<String, String> params = new HashMap<>(); |
1343 | 1343 | addPageLinkToParam(params, pageLink); |
1344 | 1344 | return restTemplate.exchange( | ... | ... |
... | ... | @@ -18,10 +18,10 @@ import { Injectable } from '@angular/core'; |
18 | 18 | import { HttpClient } from '@angular/common/http'; |
19 | 19 | import { PageLink } from '@shared/models/page/page-link'; |
20 | 20 | import { defaultHttpOptionsFromConfig, RequestConfig } from '@core/http/http-utils'; |
21 | -import { Observable } from 'rxjs'; | |
21 | +import { forkJoin, Observable, of } from 'rxjs'; | |
22 | 22 | import { PageData } from '@shared/models/page/page-data'; |
23 | 23 | import { Resource, ResourceInfo } from '@shared/models/resource.models'; |
24 | -import { map } from 'rxjs/operators'; | |
24 | +import { catchError, map, mergeMap } from 'rxjs/operators'; | |
25 | 25 | |
26 | 26 | @Injectable({ |
27 | 27 | providedIn: 'root' |
... | ... | @@ -70,6 +70,25 @@ export class ResourceService { |
70 | 70 | ); |
71 | 71 | } |
72 | 72 | |
73 | + public saveResources(resources: Resource[], config?: RequestConfig): Observable<Resource[]> { | |
74 | + let partSize = 100; | |
75 | + partSize = resources.length > partSize ? partSize : resources.length; | |
76 | + const resourceObservables: Observable<Resource>[] = []; | |
77 | + for (let i = 0; i < partSize; i++) { | |
78 | + resourceObservables.push(this.saveResource(resources[i], config).pipe(catchError(() => of({} as Resource)))); | |
79 | + } | |
80 | + return forkJoin(resourceObservables).pipe( | |
81 | + mergeMap((resource) => { | |
82 | + resources.splice(0, partSize); | |
83 | + if (resources.length) { | |
84 | + return this.saveResources(resources, config); | |
85 | + } else { | |
86 | + return of(resource); | |
87 | + } | |
88 | + }) | |
89 | + ); | |
90 | + } | |
91 | + | |
73 | 92 | public saveResource(resource: Resource, config?: RequestConfig): Observable<Resource> { |
74 | 93 | return this.http.post<Resource>('/api/resource', resource, defaultHttpOptionsFromConfig(config)); |
75 | 94 | } | ... | ... |
... | ... | @@ -79,7 +79,7 @@ export class TimeService { |
79 | 79 | |
80 | 80 | public boundMinInterval(min: number): number { |
81 | 81 | if (isDefined(min)) { |
82 | - min = Math.floor(min / 1000) * 1000; | |
82 | + min = Math.ceil(min / 1000) * 1000; | |
83 | 83 | } |
84 | 84 | return this.toBound(min, MIN_INTERVAL, MAX_INTERVAL, MIN_INTERVAL); |
85 | 85 | } | ... | ... |
... | ... | @@ -127,6 +127,10 @@ export function isEmpty(obj: any): boolean { |
127 | 127 | return true; |
128 | 128 | } |
129 | 129 | |
130 | +export function isLiteralObject(value: any) { | |
131 | + return (!!value) && (value.constructor === Object); | |
132 | +} | |
133 | + | |
130 | 134 | export function formatValue(value: any, dec?: number, units?: string, showZeroDecimals?: boolean): string | undefined { |
131 | 135 | if (isDefinedAndNotNull(value) && isNumeric(value) && |
132 | 136 | (isDefinedAndNotNull(dec) || isDefinedAndNotNull(units) || Number(value).toString() === value)) { | ... | ... |
... | ... | @@ -90,7 +90,6 @@ import { |
90 | 90 | } from '@home/components/alias/entity-aliases-dialog.component'; |
91 | 91 | import { EntityAliases } from '@app/shared/models/alias.models'; |
92 | 92 | import { EditWidgetComponent } from '@home/components/dashboard-page/edit-widget.component'; |
93 | -import { WidgetsBundle } from '@shared/models/widgets-bundle.model'; | |
94 | 93 | import { |
95 | 94 | AddWidgetDialogComponent, |
96 | 95 | AddWidgetDialogData |
... | ... | @@ -118,8 +117,7 @@ import { ComponentPortal } from '@angular/cdk/portal'; |
118 | 117 | import { |
119 | 118 | DISPLAY_WIDGET_TYPES_PANEL_DATA, |
120 | 119 | DisplayWidgetTypesPanelComponent, |
121 | - DisplayWidgetTypesPanelData, | |
122 | - WidgetTypes | |
120 | + DisplayWidgetTypesPanelData | |
123 | 121 | } from '@home/components/dashboard-page/widget-types-panel.component'; |
124 | 122 | import { DashboardWidgetSelectComponent } from '@home/components/dashboard-page/dashboard-widget-select.component'; |
125 | 123 | import {AliasEntityType, EntityType} from "@shared/models/entity-type.models"; |
... | ... | @@ -1189,13 +1187,16 @@ export class DashboardPageComponent extends PageComponent implements IDashboardC |
1189 | 1187 | overlayRef.dispose(); |
1190 | 1188 | }); |
1191 | 1189 | |
1190 | + const filterWidgetTypes = this.dashboardWidgetSelectComponent.filterWidgetTypes; | |
1191 | + const widgetTypesList = Array.from(this.dashboardWidgetSelectComponent.widgetTypes.values()).map(type => { | |
1192 | + return {type, display: filterWidgetTypes === null ? true : filterWidgetTypes.includes(type)}; | |
1193 | + }); | |
1194 | + | |
1192 | 1195 | const providers: StaticProvider[] = [ |
1193 | 1196 | { |
1194 | 1197 | provide: DISPLAY_WIDGET_TYPES_PANEL_DATA, |
1195 | 1198 | useValue: { |
1196 | - types: Array.from(this.dashboardWidgetSelectComponent.widgetTypes.values()).map(type => { | |
1197 | - return {type, display: true}; | |
1198 | - }), | |
1199 | + types: widgetTypesList, | |
1199 | 1200 | typesUpdated: (newTypes) => { |
1200 | 1201 | this.filterWidgetTypes = newTypes.filter(type => type.display).map(type => type.type); |
1201 | 1202 | } | ... | ... |
... | ... | @@ -77,6 +77,10 @@ export class DashboardWidgetSelectComponent implements OnInit { |
77 | 77 | this.filterWidgetTypes$.next(widgetTypes); |
78 | 78 | } |
79 | 79 | |
80 | + get filterWidgetTypes(): Array<widgetType> { | |
81 | + return this.filterWidgetTypes$.value; | |
82 | + } | |
83 | + | |
80 | 84 | @Output() |
81 | 85 | widgetSelected: EventEmitter<WidgetInfo> = new EventEmitter<WidgetInfo>(); |
82 | 86 | ... | ... |
... | ... | @@ -80,7 +80,7 @@ |
80 | 80 | (mousedown)="widgetMouseDown($event, widget)" |
81 | 81 | (click)="widgetClicked($event, widget)" |
82 | 82 | (contextmenu)="openWidgetContextMenu($event, widget)"> |
83 | - <div *ngIf="widgetComponent.widgetContext?.inited" fxLayout="row" fxLayoutAlign="space-between start"> | |
83 | + <div *ngIf="!!widgetComponent.widgetContext?.inited" fxLayout="row" fxLayoutAlign="space-between start"> | |
84 | 84 | <div class="tb-widget-title" fxLayout="column" fxLayoutAlign="center start" *ngIf="widget.showWidgetTitlePanel"> |
85 | 85 | <span *ngIf="widget.showTitle" |
86 | 86 | [ngClass]="{'single-row': widget.hasTimewindow}" | ... | ... |
... | ... | @@ -15,11 +15,21 @@ |
15 | 15 | /// |
16 | 16 | |
17 | 17 | import { Component, forwardRef, Input, OnInit } from '@angular/core'; |
18 | -import { ControlValueAccessor, FormBuilder, FormGroup, NG_VALUE_ACCESSOR, Validators } from '@angular/forms'; | |
18 | +import { | |
19 | + ControlValueAccessor, | |
20 | + FormBuilder, | |
21 | + FormGroup, | |
22 | + NG_VALIDATORS, | |
23 | + NG_VALUE_ACCESSOR, | |
24 | + ValidationErrors, | |
25 | + Validator, | |
26 | + Validators | |
27 | +} from '@angular/forms'; | |
19 | 28 | import { |
20 | 29 | BooleanFilterPredicate, |
21 | 30 | BooleanOperation, |
22 | - booleanOperationTranslationMap, EntityKeyValueType, | |
31 | + booleanOperationTranslationMap, | |
32 | + EntityKeyValueType, | |
23 | 33 | FilterPredicateType |
24 | 34 | } from '@shared/models/query/query.models'; |
25 | 35 | |
... | ... | @@ -32,10 +42,15 @@ import { |
32 | 42 | provide: NG_VALUE_ACCESSOR, |
33 | 43 | useExisting: forwardRef(() => BooleanFilterPredicateComponent), |
34 | 44 | multi: true |
45 | + }, | |
46 | + { | |
47 | + provide: NG_VALIDATORS, | |
48 | + useExisting: forwardRef(() => BooleanFilterPredicateComponent), | |
49 | + multi: true | |
35 | 50 | } |
36 | 51 | ] |
37 | 52 | }) |
38 | -export class BooleanFilterPredicateComponent implements ControlValueAccessor, OnInit { | |
53 | +export class BooleanFilterPredicateComponent implements ControlValueAccessor, Validator, OnInit { | |
39 | 54 | |
40 | 55 | @Input() disabled: boolean; |
41 | 56 | |
... | ... | @@ -73,7 +88,7 @@ export class BooleanFilterPredicateComponent implements ControlValueAccessor, On |
73 | 88 | registerOnTouched(fn: any): void { |
74 | 89 | } |
75 | 90 | |
76 | - setDisabledState?(isDisabled: boolean): void { | |
91 | + setDisabledState(isDisabled: boolean): void { | |
77 | 92 | this.disabled = isDisabled; |
78 | 93 | if (this.disabled) { |
79 | 94 | this.booleanFilterPredicateFormGroup.disable({emitEvent: false}); |
... | ... | @@ -82,17 +97,20 @@ export class BooleanFilterPredicateComponent implements ControlValueAccessor, On |
82 | 97 | } |
83 | 98 | } |
84 | 99 | |
100 | + validate(): ValidationErrors | null { | |
101 | + return this.booleanFilterPredicateFormGroup ? null : { | |
102 | + booleanFilterPredicate: {valid: false} | |
103 | + }; | |
104 | + } | |
105 | + | |
85 | 106 | writeValue(predicate: BooleanFilterPredicate): void { |
86 | 107 | this.booleanFilterPredicateFormGroup.get('operation').patchValue(predicate.operation, {emitEvent: false}); |
87 | 108 | this.booleanFilterPredicateFormGroup.get('value').patchValue(predicate.value, {emitEvent: false}); |
88 | 109 | } |
89 | 110 | |
90 | 111 | private updateModel() { |
91 | - let predicate: BooleanFilterPredicate = null; | |
92 | - if (this.booleanFilterPredicateFormGroup.valid) { | |
93 | - predicate = this.booleanFilterPredicateFormGroup.getRawValue(); | |
94 | - predicate.type = FilterPredicateType.BOOLEAN; | |
95 | - } | |
112 | + const predicate: BooleanFilterPredicate = this.booleanFilterPredicateFormGroup.getRawValue(); | |
113 | + predicate.type = FilterPredicateType.BOOLEAN; | |
96 | 114 | this.propagateChange(predicate); |
97 | 115 | } |
98 | 116 | ... | ... |
... | ... | @@ -16,11 +16,7 @@ |
16 | 16 | |
17 | 17 | import { Component, forwardRef, Input, OnInit } from '@angular/core'; |
18 | 18 | import { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms'; |
19 | -import { | |
20 | - ComplexFilterPredicate, | |
21 | - ComplexFilterPredicateInfo, | |
22 | - EntityKeyValueType | |
23 | -} from '@shared/models/query/query.models'; | |
19 | +import { ComplexFilterPredicateInfo, EntityKeyValueType } from '@shared/models/query/query.models'; | |
24 | 20 | import { MatDialog } from '@angular/material/dialog'; |
25 | 21 | import { |
26 | 22 | ComplexFilterPredicateDialogComponent, |
... | ... | @@ -71,7 +67,7 @@ export class ComplexFilterPredicateComponent implements ControlValueAccessor, On |
71 | 67 | registerOnTouched(fn: any): void { |
72 | 68 | } |
73 | 69 | |
74 | - setDisabledState?(isDisabled: boolean): void { | |
70 | + setDisabledState(isDisabled: boolean): void { | |
75 | 71 | this.disabled = isDisabled; |
76 | 72 | } |
77 | 73 | ... | ... |
... | ... | @@ -21,7 +21,10 @@ import { |
21 | 21 | FormArray, |
22 | 22 | FormBuilder, |
23 | 23 | FormGroup, |
24 | + NG_VALIDATORS, | |
24 | 25 | NG_VALUE_ACCESSOR, |
26 | + ValidationErrors, | |
27 | + Validator, | |
25 | 28 | Validators |
26 | 29 | } from '@angular/forms'; |
27 | 30 | import { Observable, of, Subscription } from 'rxjs'; |
... | ... | @@ -49,10 +52,15 @@ import { map } from 'rxjs/operators'; |
49 | 52 | provide: NG_VALUE_ACCESSOR, |
50 | 53 | useExisting: forwardRef(() => FilterPredicateListComponent), |
51 | 54 | multi: true |
55 | + }, | |
56 | + { | |
57 | + provide: NG_VALIDATORS, | |
58 | + useExisting: forwardRef(() => FilterPredicateListComponent), | |
59 | + multi: true | |
52 | 60 | } |
53 | 61 | ] |
54 | 62 | }) |
55 | -export class FilterPredicateListComponent implements ControlValueAccessor, OnInit { | |
63 | +export class FilterPredicateListComponent implements ControlValueAccessor, Validator, OnInit { | |
56 | 64 | |
57 | 65 | @Input() disabled: boolean; |
58 | 66 | |
... | ... | @@ -108,6 +116,12 @@ export class FilterPredicateListComponent implements ControlValueAccessor, OnIni |
108 | 116 | } |
109 | 117 | } |
110 | 118 | |
119 | + validate(control: AbstractControl): ValidationErrors | null { | |
120 | + return this.filterListFormGroup.valid ? null : { | |
121 | + filterList: {valid: false} | |
122 | + }; | |
123 | + } | |
124 | + | |
111 | 125 | writeValue(predicates: Array<KeyFilterPredicateInfo>): void { |
112 | 126 | if (this.valueChangeSubscription) { |
113 | 127 | this.valueChangeSubscription.unsubscribe(); |
... | ... | @@ -178,7 +192,7 @@ export class FilterPredicateListComponent implements ControlValueAccessor, OnIni |
178 | 192 | |
179 | 193 | private updateModel() { |
180 | 194 | const predicates: Array<KeyFilterPredicateInfo> = this.filterListFormGroup.getRawValue().predicates; |
181 | - if (this.filterListFormGroup.valid && predicates.length) { | |
195 | + if (predicates.length) { | |
182 | 196 | this.propagateChange(predicates); |
183 | 197 | } else { |
184 | 198 | this.propagateChange(null); | ... | ... |
... | ... | @@ -19,7 +19,10 @@ import { |
19 | 19 | ControlValueAccessor, |
20 | 20 | FormBuilder, |
21 | 21 | FormGroup, |
22 | + NG_VALIDATORS, | |
22 | 23 | NG_VALUE_ACCESSOR, |
24 | + ValidationErrors, | |
25 | + Validator, | |
23 | 26 | ValidatorFn, |
24 | 27 | Validators |
25 | 28 | } from '@angular/forms'; |
... | ... | @@ -39,10 +42,15 @@ import { |
39 | 42 | provide: NG_VALUE_ACCESSOR, |
40 | 43 | useExisting: forwardRef(() => FilterPredicateValueComponent), |
41 | 44 | multi: true |
45 | + }, | |
46 | + { | |
47 | + provide: NG_VALIDATORS, | |
48 | + useExisting: forwardRef(() => FilterPredicateValueComponent), | |
49 | + multi: true | |
42 | 50 | } |
43 | 51 | ] |
44 | 52 | }) |
45 | -export class FilterPredicateValueComponent implements ControlValueAccessor, OnInit { | |
53 | +export class FilterPredicateValueComponent implements ControlValueAccessor, Validator, OnInit { | |
46 | 54 | |
47 | 55 | private readonly inheritModeForSources: DynamicValueSourceType[] = [ |
48 | 56 | DynamicValueSourceType.CURRENT_CUSTOMER, |
... | ... | @@ -62,7 +70,22 @@ export class FilterPredicateValueComponent implements ControlValueAccessor, OnIn |
62 | 70 | } |
63 | 71 | } |
64 | 72 | |
65 | - @Input() onlyUserDynamicSource = false; | |
73 | + private onlyUserDynamicSourceValue = false; | |
74 | + | |
75 | + @Input() | |
76 | + set onlyUserDynamicSource(dynamicMode: boolean) { | |
77 | + this.onlyUserDynamicSourceValue = dynamicMode; | |
78 | + if (this.filterPredicateValueFormGroup) { | |
79 | + this.updateValidationDynamicMode(); | |
80 | + setTimeout(() => { | |
81 | + this.updateModel(); | |
82 | + }, 0); | |
83 | + } | |
84 | + } | |
85 | + | |
86 | + get onlyUserDynamicSource(): boolean { | |
87 | + return this.onlyUserDynamicSourceValue; | |
88 | + } | |
66 | 89 | |
67 | 90 | @Input() |
68 | 91 | valueType: EntityKeyValueType; |
... | ... | @@ -83,6 +106,7 @@ export class FilterPredicateValueComponent implements ControlValueAccessor, OnIn |
83 | 106 | allow = true; |
84 | 107 | |
85 | 108 | private propagateChange = null; |
109 | + private propagateChangePending = false; | |
86 | 110 | |
87 | 111 | constructor(private fb: FormBuilder) { |
88 | 112 | } |
... | ... | @@ -126,6 +150,7 @@ export class FilterPredicateValueComponent implements ControlValueAccessor, OnIn |
126 | 150 | this.updateShowInheritMode(sourceType); |
127 | 151 | } |
128 | 152 | ); |
153 | + this.updateValidationDynamicMode(); | |
129 | 154 | this.filterPredicateValueFormGroup.valueChanges.subscribe(() => { |
130 | 155 | this.updateModel(); |
131 | 156 | }); |
... | ... | @@ -133,12 +158,18 @@ export class FilterPredicateValueComponent implements ControlValueAccessor, OnIn |
133 | 158 | |
134 | 159 | registerOnChange(fn: any): void { |
135 | 160 | this.propagateChange = fn; |
161 | + if (this.propagateChangePending) { | |
162 | + this.propagateChangePending = false; | |
163 | + setTimeout(() => { | |
164 | + this.updateModel(); | |
165 | + }, 0); | |
166 | + } | |
136 | 167 | } |
137 | 168 | |
138 | 169 | registerOnTouched(fn: any): void { |
139 | 170 | } |
140 | 171 | |
141 | - setDisabledState?(isDisabled: boolean): void { | |
172 | + setDisabledState(isDisabled: boolean): void { | |
142 | 173 | this.disabled = isDisabled; |
143 | 174 | if (this.disabled) { |
144 | 175 | this.filterPredicateValueFormGroup.disable({emitEvent: false}); |
... | ... | @@ -147,28 +178,35 @@ export class FilterPredicateValueComponent implements ControlValueAccessor, OnIn |
147 | 178 | } |
148 | 179 | } |
149 | 180 | |
181 | + validate(): ValidationErrors | null { | |
182 | + return this.filterPredicateValueFormGroup.valid ? null : { | |
183 | + filterPredicateValue: {valid: false} | |
184 | + }; | |
185 | + } | |
186 | + | |
150 | 187 | writeValue(predicateValue: FilterPredicateValue<string | number | boolean>): void { |
188 | + this.propagateChangePending = false; | |
151 | 189 | this.filterPredicateValueFormGroup.get('defaultValue').patchValue(predicateValue.defaultValue, {emitEvent: false}); |
152 | - this.filterPredicateValueFormGroup.get('dynamicValue.sourceType').patchValue(predicateValue.dynamicValue ? | |
153 | - predicateValue.dynamicValue.sourceType : null, {emitEvent: false}); | |
154 | - this.filterPredicateValueFormGroup.get('dynamicValue.sourceAttribute').patchValue(predicateValue.dynamicValue ? | |
155 | - predicateValue.dynamicValue.sourceAttribute : null, {emitEvent: false}); | |
156 | - this.filterPredicateValueFormGroup.get('dynamicValue.inherit').patchValue(predicateValue.dynamicValue ? | |
157 | - predicateValue.dynamicValue.inherit : false, {emitEvent: false}); | |
190 | + this.filterPredicateValueFormGroup.get('dynamicValue').patchValue({ | |
191 | + sourceType: predicateValue.dynamicValue ? predicateValue.dynamicValue.sourceType : null, | |
192 | + sourceAttribute: predicateValue.dynamicValue ? predicateValue.dynamicValue.sourceAttribute : null, | |
193 | + inherit: predicateValue.dynamicValue ? predicateValue.dynamicValue.inherit : false | |
194 | + }, {emitEvent: this.onlyUserDynamicSource}); | |
158 | 195 | this.updateShowInheritMode(predicateValue?.dynamicValue?.sourceType); |
159 | 196 | } |
160 | 197 | |
161 | 198 | private updateModel() { |
162 | - let predicateValue: FilterPredicateValue<string | number | boolean> = null; | |
163 | - if (this.filterPredicateValueFormGroup.valid) { | |
164 | - predicateValue = this.filterPredicateValueFormGroup.getRawValue(); | |
165 | - if (predicateValue.dynamicValue) { | |
166 | - if (!predicateValue.dynamicValue.sourceType || !predicateValue.dynamicValue.sourceAttribute) { | |
167 | - predicateValue.dynamicValue = null; | |
168 | - } | |
199 | + const predicateValue: FilterPredicateValue<string | number | boolean> = this.filterPredicateValueFormGroup.getRawValue(); | |
200 | + if (predicateValue.dynamicValue) { | |
201 | + if (!predicateValue.dynamicValue.sourceType || !predicateValue.dynamicValue.sourceAttribute) { | |
202 | + predicateValue.dynamicValue = null; | |
169 | 203 | } |
170 | 204 | } |
171 | - this.propagateChange(predicateValue); | |
205 | + if (this.propagateChange) { | |
206 | + this.propagateChange(predicateValue); | |
207 | + } else { | |
208 | + this.propagateChangePending = true; | |
209 | + } | |
172 | 210 | } |
173 | 211 | |
174 | 212 | private updateShowInheritMode(sourceType: DynamicValueSourceType) { |
... | ... | @@ -179,4 +217,16 @@ export class FilterPredicateValueComponent implements ControlValueAccessor, OnIn |
179 | 217 | this.inheritMode = false; |
180 | 218 | } |
181 | 219 | } |
220 | + | |
221 | + private updateValidationDynamicMode() { | |
222 | + if (this.onlyUserDynamicSource) { | |
223 | + this.filterPredicateValueFormGroup.get('dynamicValue.sourceType').setValidators(Validators.required); | |
224 | + this.filterPredicateValueFormGroup.get('dynamicValue.sourceAttribute').setValidators(Validators.required); | |
225 | + } else { | |
226 | + this.filterPredicateValueFormGroup.get('dynamicValue.sourceType').clearValidators(); | |
227 | + this.filterPredicateValueFormGroup.get('dynamicValue.sourceAttribute').clearValidators(); | |
228 | + } | |
229 | + this.filterPredicateValueFormGroup.get('dynamicValue.sourceType').updateValueAndValidity({emitEvent: false}); | |
230 | + this.filterPredicateValueFormGroup.get('dynamicValue.sourceAttribute').updateValueAndValidity({emitEvent: false}); | |
231 | + } | |
182 | 232 | } | ... | ... |
... | ... | @@ -15,11 +15,17 @@ |
15 | 15 | /// |
16 | 16 | |
17 | 17 | import { Component, forwardRef, Input, OnInit } from '@angular/core'; |
18 | -import { ControlValueAccessor, FormBuilder, FormGroup, NG_VALUE_ACCESSOR, Validators } from '@angular/forms'; | |
19 | 18 | import { |
20 | - EntityKeyValueType, | |
21 | - FilterPredicateType, KeyFilterPredicate, KeyFilterPredicateInfo | |
22 | -} from '@shared/models/query/query.models'; | |
19 | + ControlValueAccessor, | |
20 | + FormBuilder, | |
21 | + FormGroup, | |
22 | + NG_VALIDATORS, | |
23 | + NG_VALUE_ACCESSOR, | |
24 | + ValidationErrors, | |
25 | + Validator, | |
26 | + Validators | |
27 | +} from '@angular/forms'; | |
28 | +import { EntityKeyValueType, FilterPredicateType, KeyFilterPredicateInfo } from '@shared/models/query/query.models'; | |
23 | 29 | |
24 | 30 | @Component({ |
25 | 31 | selector: 'tb-filter-predicate', |
... | ... | @@ -30,10 +36,15 @@ import { |
30 | 36 | provide: NG_VALUE_ACCESSOR, |
31 | 37 | useExisting: forwardRef(() => FilterPredicateComponent), |
32 | 38 | multi: true |
39 | + }, | |
40 | + { | |
41 | + provide: NG_VALIDATORS, | |
42 | + useExisting: forwardRef(() => FilterPredicateComponent), | |
43 | + multi: true | |
33 | 44 | } |
34 | 45 | ] |
35 | 46 | }) |
36 | -export class FilterPredicateComponent implements ControlValueAccessor, OnInit { | |
47 | +export class FilterPredicateComponent implements ControlValueAccessor, Validator, OnInit { | |
37 | 48 | |
38 | 49 | @Input() disabled: boolean; |
39 | 50 | |
... | ... | @@ -75,7 +86,7 @@ export class FilterPredicateComponent implements ControlValueAccessor, OnInit { |
75 | 86 | registerOnTouched(fn: any): void { |
76 | 87 | } |
77 | 88 | |
78 | - setDisabledState?(isDisabled: boolean): void { | |
89 | + setDisabledState(isDisabled: boolean): void { | |
79 | 90 | this.disabled = isDisabled; |
80 | 91 | if (this.disabled) { |
81 | 92 | this.filterPredicateFormGroup.disable({emitEvent: false}); |
... | ... | @@ -84,6 +95,12 @@ export class FilterPredicateComponent implements ControlValueAccessor, OnInit { |
84 | 95 | } |
85 | 96 | } |
86 | 97 | |
98 | + validate(): ValidationErrors | null { | |
99 | + return this.filterPredicateFormGroup.valid ? null : { | |
100 | + filterPredicate: {valid: false} | |
101 | + }; | |
102 | + } | |
103 | + | |
87 | 104 | writeValue(predicate: KeyFilterPredicateInfo): void { |
88 | 105 | this.type = predicate.keyFilterPredicate.type; |
89 | 106 | this.filterPredicateFormGroup.get('predicate').patchValue(predicate.keyFilterPredicate, {emitEvent: false}); | ... | ... |
... | ... | @@ -22,7 +22,10 @@ import { |
22 | 22 | FormBuilder, |
23 | 23 | FormControl, |
24 | 24 | FormGroup, |
25 | + NG_VALIDATORS, | |
25 | 26 | NG_VALUE_ACCESSOR, |
27 | + ValidationErrors, | |
28 | + Validator, | |
26 | 29 | Validators |
27 | 30 | } from '@angular/forms'; |
28 | 31 | import { Observable, Subscription } from 'rxjs'; |
... | ... | @@ -46,10 +49,15 @@ import { EntityId } from '@shared/models/id/entity-id'; |
46 | 49 | provide: NG_VALUE_ACCESSOR, |
47 | 50 | useExisting: forwardRef(() => KeyFilterListComponent), |
48 | 51 | multi: true |
52 | + }, | |
53 | + { | |
54 | + provide: NG_VALIDATORS, | |
55 | + useExisting: forwardRef(() => KeyFilterListComponent), | |
56 | + multi: true | |
49 | 57 | } |
50 | 58 | ] |
51 | 59 | }) |
52 | -export class KeyFilterListComponent implements ControlValueAccessor, OnInit { | |
60 | +export class KeyFilterListComponent implements ControlValueAccessor, Validator, OnInit { | |
53 | 61 | |
54 | 62 | @Input() disabled: boolean; |
55 | 63 | |
... | ... | @@ -104,6 +112,12 @@ export class KeyFilterListComponent implements ControlValueAccessor, OnInit { |
104 | 112 | } |
105 | 113 | } |
106 | 114 | |
115 | + validate(): ValidationErrors | null { | |
116 | + return this.keyFilterListFormGroup.valid && this.keyFiltersControl.valid ? null : { | |
117 | + keyFilterList: {valid: false} | |
118 | + }; | |
119 | + } | |
120 | + | |
107 | 121 | writeValue(keyFilters: Array<KeyFilterInfo>): void { |
108 | 122 | if (this.valueChangeSubscription) { |
109 | 123 | this.valueChangeSubscription.unsubscribe(); | ... | ... |
... | ... | @@ -15,7 +15,16 @@ |
15 | 15 | /// |
16 | 16 | |
17 | 17 | import { Component, forwardRef, Input, OnInit } from '@angular/core'; |
18 | -import { ControlValueAccessor, FormBuilder, FormGroup, NG_VALUE_ACCESSOR, Validators } from '@angular/forms'; | |
18 | +import { | |
19 | + ControlValueAccessor, | |
20 | + FormBuilder, | |
21 | + FormGroup, | |
22 | + NG_VALIDATORS, | |
23 | + NG_VALUE_ACCESSOR, | |
24 | + ValidationErrors, | |
25 | + Validator, | |
26 | + Validators | |
27 | +} from '@angular/forms'; | |
19 | 28 | import { |
20 | 29 | EntityKeyValueType, |
21 | 30 | FilterPredicateType, |
... | ... | @@ -33,10 +42,15 @@ import { |
33 | 42 | provide: NG_VALUE_ACCESSOR, |
34 | 43 | useExisting: forwardRef(() => NumericFilterPredicateComponent), |
35 | 44 | multi: true |
45 | + }, | |
46 | + { | |
47 | + provide: NG_VALIDATORS, | |
48 | + useExisting: forwardRef(() => NumericFilterPredicateComponent), | |
49 | + multi: true | |
36 | 50 | } |
37 | 51 | ] |
38 | 52 | }) |
39 | -export class NumericFilterPredicateComponent implements ControlValueAccessor, OnInit { | |
53 | +export class NumericFilterPredicateComponent implements ControlValueAccessor, Validator, OnInit { | |
40 | 54 | |
41 | 55 | @Input() disabled: boolean; |
42 | 56 | |
... | ... | @@ -76,7 +90,7 @@ export class NumericFilterPredicateComponent implements ControlValueAccessor, On |
76 | 90 | registerOnTouched(fn: any): void { |
77 | 91 | } |
78 | 92 | |
79 | - setDisabledState?(isDisabled: boolean): void { | |
93 | + setDisabledState(isDisabled: boolean): void { | |
80 | 94 | this.disabled = isDisabled; |
81 | 95 | if (this.disabled) { |
82 | 96 | this.numericFilterPredicateFormGroup.disable({emitEvent: false}); |
... | ... | @@ -85,17 +99,20 @@ export class NumericFilterPredicateComponent implements ControlValueAccessor, On |
85 | 99 | } |
86 | 100 | } |
87 | 101 | |
102 | + validate(): ValidationErrors | null { | |
103 | + return this.numericFilterPredicateFormGroup.valid ? null : { | |
104 | + numericFilterPredicate: {valid: false} | |
105 | + }; | |
106 | + } | |
107 | + | |
88 | 108 | writeValue(predicate: NumericFilterPredicate): void { |
89 | 109 | this.numericFilterPredicateFormGroup.get('operation').patchValue(predicate.operation, {emitEvent: false}); |
90 | 110 | this.numericFilterPredicateFormGroup.get('value').patchValue(predicate.value, {emitEvent: false}); |
91 | 111 | } |
92 | 112 | |
93 | 113 | private updateModel() { |
94 | - let predicate: NumericFilterPredicate = null; | |
95 | - if (this.numericFilterPredicateFormGroup.valid) { | |
96 | - predicate = this.numericFilterPredicateFormGroup.getRawValue(); | |
97 | - predicate.type = FilterPredicateType.NUMERIC; | |
98 | - } | |
114 | + const predicate: NumericFilterPredicate = this.numericFilterPredicateFormGroup.getRawValue(); | |
115 | + predicate.type = FilterPredicateType.NUMERIC; | |
99 | 116 | this.propagateChange(predicate); |
100 | 117 | } |
101 | 118 | ... | ... |
... | ... | @@ -15,7 +15,16 @@ |
15 | 15 | /// |
16 | 16 | |
17 | 17 | import { Component, forwardRef, Input, OnInit } from '@angular/core'; |
18 | -import { ControlValueAccessor, FormBuilder, FormGroup, NG_VALUE_ACCESSOR, Validators } from '@angular/forms'; | |
18 | +import { | |
19 | + ControlValueAccessor, | |
20 | + FormBuilder, | |
21 | + FormGroup, | |
22 | + NG_VALIDATORS, | |
23 | + NG_VALUE_ACCESSOR, | |
24 | + ValidationErrors, | |
25 | + Validator, | |
26 | + Validators | |
27 | +} from '@angular/forms'; | |
19 | 28 | import { |
20 | 29 | EntityKeyValueType, |
21 | 30 | FilterPredicateType, |
... | ... | @@ -33,10 +42,15 @@ import { |
33 | 42 | provide: NG_VALUE_ACCESSOR, |
34 | 43 | useExisting: forwardRef(() => StringFilterPredicateComponent), |
35 | 44 | multi: true |
45 | + }, | |
46 | + { | |
47 | + provide: NG_VALIDATORS, | |
48 | + useExisting: forwardRef(() => StringFilterPredicateComponent), | |
49 | + multi: true | |
36 | 50 | } |
37 | 51 | ] |
38 | 52 | }) |
39 | -export class StringFilterPredicateComponent implements ControlValueAccessor, OnInit { | |
53 | +export class StringFilterPredicateComponent implements ControlValueAccessor, Validator, OnInit { | |
40 | 54 | |
41 | 55 | @Input() disabled: boolean; |
42 | 56 | |
... | ... | @@ -90,12 +104,15 @@ export class StringFilterPredicateComponent implements ControlValueAccessor, OnI |
90 | 104 | this.stringFilterPredicateFormGroup.get('ignoreCase').patchValue(predicate.ignoreCase, {emitEvent: false}); |
91 | 105 | } |
92 | 106 | |
107 | + validate(c): ValidationErrors { | |
108 | + return this.stringFilterPredicateFormGroup.valid ? null : { | |
109 | + stringFilterPredicate: {valid: false} | |
110 | + }; | |
111 | + } | |
112 | + | |
93 | 113 | private updateModel() { |
94 | - let predicate: StringFilterPredicate = null; | |
95 | - if (this.stringFilterPredicateFormGroup.valid) { | |
96 | - predicate = this.stringFilterPredicateFormGroup.getRawValue(); | |
97 | - predicate.type = FilterPredicateType.STRING; | |
98 | - } | |
114 | + const predicate: StringFilterPredicate = this.stringFilterPredicateFormGroup.getRawValue(); | |
115 | + predicate.type = FilterPredicateType.STRING; | |
99 | 116 | this.propagateChange(predicate); |
100 | 117 | } |
101 | 118 | ... | ... |
1 | +<!-- | |
2 | + | |
3 | + Copyright © 2016-2021 The Thingsboard Authors | |
4 | + | |
5 | + Licensed under the Apache License, Version 2.0 (the "License"); | |
6 | + you may not use this file except in compliance with the License. | |
7 | + You may obtain a copy of the License at | |
8 | + | |
9 | + http://www.apache.org/licenses/LICENSE-2.0 | |
10 | + | |
11 | + Unless required by applicable law or agreed to in writing, software | |
12 | + distributed under the License is distributed on an "AS IS" BASIS, | |
13 | + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | |
14 | + See the License for the specific language governing permissions and | |
15 | + limitations under the License. | |
16 | + | |
17 | +--> | |
18 | + | |
19 | +<div class="tb-json-input" tb-toast toastTarget="{{ toastTargetId }}"> | |
20 | + <form *ngIf="attributeUpdateFormGroup" | |
21 | + fxLayout="column" | |
22 | + class="tb-json-input__form" | |
23 | + [formGroup]="attributeUpdateFormGroup" | |
24 | + (ngSubmit)="save()"> | |
25 | + <div fxLayout="column" fxLayoutGap="10px" fxFlex *ngIf="entityDetected && isValidParameter && dataKeyDetected"> | |
26 | + <fieldset fxFlex> | |
27 | + <tb-json-object-edit | |
28 | + [editorStyle]="{minHeight: '100px'}" | |
29 | + fillHeight="true" | |
30 | + [required]="settings.attributeRequired" | |
31 | + label="{{ settings.showLabel ? labelValue : '' }}" | |
32 | + formControlName="currentValue" | |
33 | + (focusin)="isFocused = true;" | |
34 | + (focusout)="isFocused = false;" | |
35 | + ></tb-json-object-edit> | |
36 | + </fieldset> | |
37 | + <div class="tb-json-input-form__actions" fxLayout="row" fxLayoutAlign="end center" fxLayoutGap="20px"> | |
38 | + <button mat-button color="primary" | |
39 | + type="button" | |
40 | + [disabled]="!attributeUpdateFormGroup.dirty" | |
41 | + (click)="discard()" | |
42 | + matTooltip="{{ 'widgets.input-widgets.discard-changes' | translate }}" | |
43 | + matTooltipPosition="above"> | |
44 | + {{ "action.undo" | translate }} | |
45 | + </button> | |
46 | + <button mat-button mat-raised-button color="primary" | |
47 | + type="submit" | |
48 | + [disabled]="attributeUpdateFormGroup.invalid || !attributeUpdateFormGroup.dirty"> | |
49 | + {{ "action.save" | translate }} | |
50 | + </button> | |
51 | + </div> | |
52 | + </div> | |
53 | + | |
54 | + <div fxLayout="column" fxLayoutAlign="center center" fxFlex *ngIf="!entityDetected || !dataKeyDetected || !isValidParameter"> | |
55 | + <div class="tb-json-input__error" | |
56 | + *ngIf="!entityDetected"> | |
57 | + {{ 'widgets.input-widgets.no-entity-selected' | translate }} | |
58 | + </div> | |
59 | + <div class="tb-json-input__error" | |
60 | + *ngIf="entityDetected && !dataKeyDetected"> | |
61 | + {{ 'widgets.input-widgets.no-datakey-selected' | translate }} | |
62 | + </div> | |
63 | + <div class="tb-json-input__error" | |
64 | + *ngIf="dataKeyDetected && !isValidParameter"> | |
65 | + {{ errorMessage | translate }} | |
66 | + </div> | |
67 | + </div> | |
68 | + </form> | |
69 | +</div> | ... | ... |
1 | +/** | |
2 | + * Copyright © 2016-2021 The Thingsboard Authors | |
3 | + * | |
4 | + * Licensed under the Apache License, Version 2.0 (the "License"); | |
5 | + * you may not use this file except in compliance with the License. | |
6 | + * You may obtain a copy of the License at | |
7 | + * | |
8 | + * http://www.apache.org/licenses/LICENSE-2.0 | |
9 | + * | |
10 | + * Unless required by applicable law or agreed to in writing, software | |
11 | + * distributed under the License is distributed on an "AS IS" BASIS, | |
12 | + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | |
13 | + * See the License for the specific language governing permissions and | |
14 | + * limitations under the License. | |
15 | + */ | |
16 | + | |
17 | +.tb-json-input { | |
18 | + width: 100%; | |
19 | + height: 100%; | |
20 | + padding: 5px; | |
21 | + | |
22 | + &__form { | |
23 | + overflow: auto; | |
24 | + height: 100%; | |
25 | + } | |
26 | + | |
27 | + &__error { | |
28 | + text-align: center; | |
29 | + font-size: 18px; | |
30 | + color: #a0a0a0; | |
31 | + } | |
32 | +} | |
33 | + | |
34 | +.tb-toast { | |
35 | + font-size: 14px!important; | |
36 | +} | ... | ... |
1 | +/// | |
2 | +/// Copyright © 2016-2021 The Thingsboard Authors | |
3 | +/// | |
4 | +/// Licensed under the Apache License, Version 2.0 (the "License"); | |
5 | +/// you may not use this file except in compliance with the License. | |
6 | +/// You may obtain a copy of the License at | |
7 | +/// | |
8 | +/// http://www.apache.org/licenses/LICENSE-2.0 | |
9 | +/// | |
10 | +/// Unless required by applicable law or agreed to in writing, software | |
11 | +/// distributed under the License is distributed on an "AS IS" BASIS, | |
12 | +/// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | |
13 | +/// See the License for the specific language governing permissions and | |
14 | +/// limitations under the License. | |
15 | +/// | |
16 | + | |
17 | +import { Component, Input, OnInit } from '@angular/core'; | |
18 | +import { PageComponent } from '@shared/components/page.component'; | |
19 | +import { WidgetContext } from '@home/models/widget-component.models'; | |
20 | +import { Store } from '@ngrx/store'; | |
21 | +import { AppState } from '@core/core.state'; | |
22 | +import { UtilsService } from '@core/services/utils.service'; | |
23 | +import { TranslateService } from '@ngx-translate/core'; | |
24 | +import { Datasource, DatasourceData, DatasourceType, WidgetConfig } from '@shared/models/widget.models'; | |
25 | +import { IWidgetSubscription } from '@core/api/widget-api.models'; | |
26 | +import { FormBuilder, FormGroup, ValidatorFn, Validators } from '@angular/forms'; | |
27 | +import { AttributeService } from '@core/http/attribute.service'; | |
28 | +import { AttributeData, AttributeScope, DataKeyType, LatestTelemetry } from '@shared/models/telemetry/telemetry.models'; | |
29 | +import { EntityId } from '@shared/models/id/entity-id'; | |
30 | +import { EntityType } from '@shared/models/entity-type.models'; | |
31 | +import { createLabelFromDatasource } from '@core/utils'; | |
32 | +import { Observable } from 'rxjs'; | |
33 | + | |
34 | +enum JsonInputWidgetMode { | |
35 | + ATTRIBUTE = 'ATTRIBUTE', | |
36 | + TIME_SERIES = 'TIME_SERIES', | |
37 | +} | |
38 | + | |
39 | +interface JsonInputWidgetSettings { | |
40 | + widgetTitle: string; | |
41 | + widgetMode: JsonInputWidgetMode; | |
42 | + attributeScope?: AttributeScope; | |
43 | + showLabel: boolean; | |
44 | + labelValue?: string; | |
45 | + attributeRequired: boolean; | |
46 | + showResultMessage: boolean; | |
47 | +} | |
48 | + | |
49 | +@Component({ | |
50 | + selector: 'tb-json-input-widget ', | |
51 | + templateUrl: './json-input-widget.component.html', | |
52 | + styleUrls: ['./json-input-widget.component.scss'] | |
53 | +}) | |
54 | +export class JsonInputWidgetComponent extends PageComponent implements OnInit { | |
55 | + | |
56 | + @Input() | |
57 | + ctx: WidgetContext; | |
58 | + | |
59 | + public settings: JsonInputWidgetSettings; | |
60 | + private widgetConfig: WidgetConfig; | |
61 | + private subscription: IWidgetSubscription; | |
62 | + private datasource: Datasource; | |
63 | + | |
64 | + labelValue: string; | |
65 | + | |
66 | + entityDetected = false; | |
67 | + dataKeyDetected = false; | |
68 | + isValidParameter = false; | |
69 | + errorMessage: string; | |
70 | + | |
71 | + isFocused: boolean; | |
72 | + originalValue: any; | |
73 | + attributeUpdateFormGroup: FormGroup; | |
74 | + | |
75 | + toastTargetId = 'json-input-widget' + this.utils.guid(); | |
76 | + | |
77 | + constructor(protected store: Store<AppState>, | |
78 | + private utils: UtilsService, | |
79 | + private fb: FormBuilder, | |
80 | + private attributeService: AttributeService, | |
81 | + private translate: TranslateService) { | |
82 | + super(store); | |
83 | + } | |
84 | + | |
85 | + ngOnInit(): void { | |
86 | + this.ctx.$scope.jsonInputWidget = this; | |
87 | + this.settings = this.ctx.settings; | |
88 | + this.widgetConfig = this.ctx.widgetConfig; | |
89 | + this.subscription = this.ctx.defaultSubscription; | |
90 | + this.datasource = this.subscription.datasources[0]; | |
91 | + this.initializeConfig(); | |
92 | + this.validateDatasources(); | |
93 | + this.buildForm(); | |
94 | + this.ctx.updateWidgetParams(); | |
95 | + } | |
96 | + | |
97 | + private initializeConfig() { | |
98 | + if (this.settings.widgetTitle && this.settings.widgetTitle.length) { | |
99 | + const title = createLabelFromDatasource(this.datasource, this.settings.widgetTitle); | |
100 | + this.ctx.widgetTitle = this.utils.customTranslation(title, title); | |
101 | + } else { | |
102 | + this.ctx.widgetTitle = this.ctx.widgetConfig.title; | |
103 | + } | |
104 | + | |
105 | + if (this.settings.labelValue && this.settings.labelValue.length) { | |
106 | + const label = createLabelFromDatasource(this.datasource, this.settings.labelValue); | |
107 | + this.labelValue = this.utils.customTranslation(label, label); | |
108 | + } else { | |
109 | + this.labelValue = this.translate.instant('widgets.input-widgets.value'); | |
110 | + } | |
111 | + } | |
112 | + | |
113 | + private validateDatasources() { | |
114 | + if (this.datasource?.type === DatasourceType.entity) { | |
115 | + this.entityDetected = true; | |
116 | + if (this.datasource.dataKeys.length) { | |
117 | + this.dataKeyDetected = true; | |
118 | + | |
119 | + if (this.settings.widgetMode === JsonInputWidgetMode.ATTRIBUTE) { | |
120 | + if (this.datasource.dataKeys[0].type === DataKeyType.attribute) { | |
121 | + if (this.settings.attributeScope === AttributeScope.SERVER_SCOPE || this.datasource.entityType === EntityType.DEVICE) { | |
122 | + this.isValidParameter = true; | |
123 | + } else { | |
124 | + this.errorMessage = 'widgets.input-widgets.not-allowed-entity'; | |
125 | + } | |
126 | + } else { | |
127 | + this.errorMessage = 'widgets.input-widgets.no-attribute-selected'; | |
128 | + } | |
129 | + } else { | |
130 | + if (this.datasource.dataKeys[0].type === DataKeyType.timeseries) { | |
131 | + this.isValidParameter = true; | |
132 | + } else { | |
133 | + this.errorMessage = 'widgets.input-widgets.no-timeseries-selected'; | |
134 | + } | |
135 | + } | |
136 | + | |
137 | + } | |
138 | + } | |
139 | + } | |
140 | + | |
141 | + private buildForm() { | |
142 | + const validators: ValidatorFn[] = []; | |
143 | + if (this.settings.attributeRequired) { | |
144 | + validators.push(Validators.required); | |
145 | + } | |
146 | + this.attributeUpdateFormGroup = this.fb.group({ | |
147 | + currentValue: [{}, validators] | |
148 | + }); | |
149 | + this.attributeUpdateFormGroup.valueChanges.subscribe( () => { | |
150 | + this.ctx.detectChanges(); | |
151 | + }); | |
152 | + } | |
153 | + | |
154 | + private updateWidgetData(data: Array<DatasourceData>) { | |
155 | + if (this.isValidParameter) { | |
156 | + let value = {}; | |
157 | + if (data[0].data[0][1] !== '') { | |
158 | + try { | |
159 | + value = JSON.parse(data[0].data[0][1]); | |
160 | + } catch (e) { | |
161 | + value = data[0].data[0][1]; | |
162 | + } | |
163 | + } | |
164 | + this.originalValue = value; | |
165 | + if (!this.isFocused) { | |
166 | + this.attributeUpdateFormGroup.get('currentValue').patchValue(this.originalValue); | |
167 | + this.ctx.detectChanges(); | |
168 | + } | |
169 | + } | |
170 | + } | |
171 | + | |
172 | + public onDataUpdated() { | |
173 | + this.updateWidgetData(this.subscription.data); | |
174 | + } | |
175 | + | |
176 | + public save() { | |
177 | + this.isFocused = false; | |
178 | + | |
179 | + const attributeToSave: AttributeData = { | |
180 | + key: this.datasource.dataKeys[0].name, | |
181 | + value: this.attributeUpdateFormGroup.get('currentValue').value | |
182 | + }; | |
183 | + | |
184 | + const entityId: EntityId = { | |
185 | + entityType: this.datasource.entityType, | |
186 | + id: this.datasource.entityId | |
187 | + }; | |
188 | + | |
189 | + let saveAttributeObservable: Observable<any>; | |
190 | + if (this.settings.widgetMode === JsonInputWidgetMode.ATTRIBUTE) { | |
191 | + saveAttributeObservable = this.attributeService.saveEntityAttributes( | |
192 | + entityId, | |
193 | + this.settings.attributeScope, | |
194 | + [ attributeToSave ], | |
195 | + {} | |
196 | + ); | |
197 | + } else { | |
198 | + saveAttributeObservable = this.attributeService.saveEntityTimeseries( | |
199 | + entityId, | |
200 | + LatestTelemetry.LATEST_TELEMETRY, | |
201 | + [ attributeToSave ], | |
202 | + {} | |
203 | + ); | |
204 | + } | |
205 | + saveAttributeObservable.subscribe( | |
206 | + () => { | |
207 | + this.attributeUpdateFormGroup.markAsPristine(); | |
208 | + this.ctx.detectChanges(); | |
209 | + if (this.settings.showResultMessage) { | |
210 | + this.ctx.showSuccessToast(this.translate.instant('widgets.input-widgets.update-successful'), | |
211 | + 1000, 'bottom', 'left', this.toastTargetId); | |
212 | + } | |
213 | + }, | |
214 | + () => { | |
215 | + if (this.settings.showResultMessage) { | |
216 | + this.ctx.showErrorToast(this.translate.instant('widgets.input-widgets.update-failed'), | |
217 | + 'bottom', 'left', this.toastTargetId); | |
218 | + } | |
219 | + }); | |
220 | + } | |
221 | + | |
222 | + public discard() { | |
223 | + this.attributeUpdateFormGroup.reset({currentValue: this.originalValue}, {emitEvent: false}); | |
224 | + this.attributeUpdateFormGroup.markAsPristine(); | |
225 | + this.isFocused = false; | |
226 | + } | |
227 | +} | ... | ... |
... | ... | @@ -19,7 +19,8 @@ import { |
19 | 19 | createLabelFromDatasource, |
20 | 20 | hashCode, |
21 | 21 | isDefined, |
22 | - isDefinedAndNotNull, isFunction, | |
22 | + isDefinedAndNotNull, | |
23 | + isFunction, | |
23 | 24 | isNumber, |
24 | 25 | isUndefined, |
25 | 26 | padValue |
... | ... | @@ -30,7 +31,7 @@ import { Datasource, DatasourceData } from '@shared/models/widget.models'; |
30 | 31 | import _ from 'lodash'; |
31 | 32 | import { mapProviderSchema, providerSets } from '@home/components/widget/lib/maps/schemes'; |
32 | 33 | import { addCondition, mergeSchemes } from '@core/schema-utils'; |
33 | -import L, {Projection} from "leaflet"; | |
34 | +import L from 'leaflet'; | |
34 | 35 | |
35 | 36 | export function getProviderSchema(mapProvider: MapProviders, ignoreImageMap = false) { |
36 | 37 | const providerSchema = _.cloneDeep(mapProviderSchema); |
... | ... | @@ -318,22 +319,24 @@ export const parseWithTranslation = { |
318 | 319 | }; |
319 | 320 | |
320 | 321 | export function parseData(input: DatasourceData[]): FormattedData[] { |
321 | - return _(input).groupBy(el => el?.datasource?.entityName) | |
322 | + return _(input).groupBy(el => el?.datasource.entityId + el?.datasource.entityType) | |
322 | 323 | .values().value().map((entityArray, i) => { |
323 | 324 | const obj: FormattedData = { |
324 | 325 | entityName: entityArray[0]?.datasource?.entityName, |
325 | - entityId: entityArray[0]?.datasource?.entityId, | |
326 | - entityType: entityArray[0]?.datasource?.entityType, | |
327 | - $datasource: entityArray[0]?.datasource, | |
326 | + entityId: entityArray[0].datasource.entityId, | |
327 | + entityType: entityArray[0].datasource.entityType, | |
328 | + $datasource: entityArray[0].datasource, | |
328 | 329 | dsIndex: i, |
329 | 330 | deviceType: null |
330 | 331 | }; |
331 | 332 | entityArray.filter(el => el.data.length).forEach(el => { |
332 | - const indexDate = el?.data?.length ? el.data.length - 1 : 0; | |
333 | - obj[el?.dataKey?.label] = el?.data[indexDate][1]; | |
334 | - obj[el?.dataKey?.label + '|ts'] = el?.data[indexDate][0]; | |
335 | - if (el?.dataKey?.label === 'type') { | |
336 | - obj.deviceType = el?.data[indexDate][1]; | |
333 | + const indexDate = el.data.length ? el.data.length - 1 : 0; | |
334 | + if (!obj.hasOwnProperty(el.dataKey.label) || el.data[indexDate][1] !== '') { | |
335 | + obj[el.dataKey.label] = el.data[indexDate][1]; | |
336 | + obj[el.dataKey.label + '|ts'] = el.data[indexDate][0]; | |
337 | + if (el.dataKey.label === 'type') { | |
338 | + obj.deviceType = el.data[indexDate][1]; | |
339 | + } | |
337 | 340 | } |
338 | 341 | }); |
339 | 342 | return obj; | ... | ... |
... | ... | @@ -121,7 +121,7 @@ export class SwitchComponent extends PageComponent implements OnInit, OnDestroy |
121 | 121 | |
122 | 122 | this.switchResize$ = new ResizeObserver(() => { |
123 | 123 | this.resize(); |
124 | - }) | |
124 | + }); | |
125 | 125 | this.switchResize$.observe(this.switchContainerRef.nativeElement); |
126 | 126 | this.init(); |
127 | 127 | } |
... | ... | @@ -202,13 +202,13 @@ export class SwitchComponent extends PageComponent implements OnInit, OnDestroy |
202 | 202 | let width = this.switchContainer.width(); |
203 | 203 | let height = this.switchContainer.height(); |
204 | 204 | if (this.showOnOffLabels) { |
205 | - height = height*2/3; | |
205 | + height = height * 2 / 3; | |
206 | 206 | } |
207 | - const ratio = width/height; | |
207 | + const ratio = width / height; | |
208 | 208 | if (ratio > switchAspectRation) { |
209 | - width = height*switchAspectRation; | |
209 | + width = height * switchAspectRation; | |
210 | 210 | } else { |
211 | - height = width/switchAspectRation; | |
211 | + height = width / switchAspectRation; | |
212 | 212 | } |
213 | 213 | this.switchElement.css({width, height}); |
214 | 214 | this.matSlideToggle.css({width, height, minWidth: width}); |
... | ... | @@ -232,11 +232,11 @@ export class SwitchComponent extends PageComponent implements OnInit, OnDestroy |
232 | 232 | fontSize--; |
233 | 233 | textWidth = this.measureTextWidth(text, fontSize); |
234 | 234 | } |
235 | - element.css({fontSize: fontSize+'px', lineHeight: fontSize+'px'}); | |
235 | + element.css({fontSize: fontSize + 'px', lineHeight: fontSize + 'px'}); | |
236 | 236 | } |
237 | 237 | |
238 | 238 | private measureTextWidth(text: string, fontSize: number): number { |
239 | - this.textMeasure.css({fontSize: fontSize+'px', lineHeight: fontSize+'px'}); | |
239 | + this.textMeasure.css({fontSize: fontSize + 'px', lineHeight: fontSize + 'px'}); | |
240 | 240 | this.textMeasure.text(text); |
241 | 241 | return this.textMeasure.width(); |
242 | 242 | } |
... | ... | @@ -260,6 +260,7 @@ export class SwitchComponent extends PageComponent implements OnInit, OnDestroy |
260 | 260 | this.ctx.controlApi.sendTwoWayCommand(this.getValueMethod, null, this.requestTimeout).subscribe( |
261 | 261 | (responseBody) => { |
262 | 262 | this.setValue(this.parseValueFunction(responseBody)); |
263 | + this.ctx.detectChanges(); | |
263 | 264 | }, |
264 | 265 | () => { |
265 | 266 | const errorText = this.ctx.defaultSubscription.rpcErrorText; | ... | ... |
... | ... | @@ -38,6 +38,7 @@ import { ImportExportService } from '@home/components/import-export/import-expor |
38 | 38 | import { NavigationCardsWidgetComponent } from '@home/components/widget/lib/navigation-cards-widget.component'; |
39 | 39 | import { NavigationCardWidgetComponent } from '@home/components/widget/lib/navigation-card-widget.component'; |
40 | 40 | import { EdgesOverviewWidgetComponent } from '@home/components/widget/lib/edges-overview-widget.component'; |
41 | +import { JsonInputWidgetComponent } from '@home/components/widget/lib/json-input-widget.component'; | |
41 | 42 | |
42 | 43 | @NgModule({ |
43 | 44 | declarations: |
... | ... | @@ -51,6 +52,7 @@ import { EdgesOverviewWidgetComponent } from '@home/components/widget/lib/edges- |
51 | 52 | EdgesOverviewWidgetComponent, |
52 | 53 | DateRangeNavigatorWidgetComponent, |
53 | 54 | DateRangeNavigatorPanelComponent, |
55 | + JsonInputWidgetComponent, | |
54 | 56 | MultipleInputWidgetComponent, |
55 | 57 | TripAnimationComponent, |
56 | 58 | PhotoCameraInputWidgetComponent, |
... | ... | @@ -72,6 +74,7 @@ import { EdgesOverviewWidgetComponent } from '@home/components/widget/lib/edges- |
72 | 74 | EdgesOverviewWidgetComponent, |
73 | 75 | RpcWidgetsModule, |
74 | 76 | DateRangeNavigatorWidgetComponent, |
77 | + JsonInputWidgetComponent, | |
75 | 78 | MultipleInputWidgetComponent, |
76 | 79 | TripAnimationComponent, |
77 | 80 | PhotoCameraInputWidgetComponent, | ... | ... |
... | ... | @@ -104,7 +104,7 @@ export class AssetsTableConfigResolver implements Resolve<EntityTableConfig<Asse |
104 | 104 | )); |
105 | 105 | }; |
106 | 106 | this.config.onEntityAction = action => this.onAssetAction(action); |
107 | - this.config.detailsReadonly = () => this.config.componentsData.assetScope === 'customer_user'; | |
107 | + this.config.detailsReadonly = () => (this.config.componentsData.assetScope === 'customer_user' || this.config.componentsData.assetScope === 'edge_customer_user'); | |
108 | 108 | |
109 | 109 | this.config.headerComponent = AssetTableHeaderComponent; |
110 | 110 | |
... | ... | @@ -151,7 +151,7 @@ export class AssetsTableConfigResolver implements Resolve<EntityTableConfig<Asse |
151 | 151 | this.config.cellActionDescriptors = this.configureCellActions(this.config.componentsData.assetScope); |
152 | 152 | this.config.groupActionDescriptors = this.configureGroupActions(this.config.componentsData.assetScope); |
153 | 153 | this.config.addActionDescriptors = this.configureAddActions(this.config.componentsData.assetScope); |
154 | - this.config.addEnabled = this.config.componentsData.assetScope !== 'customer_user'; | |
154 | + this.config.addEnabled = !(this.config.componentsData.assetScope === 'customer_user' || this.config.componentsData.assetScope === 'edge_customer_user'); | |
155 | 155 | this.config.entitiesDeleteEnabled = this.config.componentsData.assetScope === 'tenant'; |
156 | 156 | this.config.deleteEnabled = () => this.config.componentsData.assetScope === 'tenant'; |
157 | 157 | return this.config; | ... | ... |
... | ... | @@ -103,7 +103,7 @@ export class DashboardsTableConfigResolver implements Resolve<EntityTableConfig< |
103 | 103 | return this.dashboardService.saveDashboard(dashboard as Dashboard); |
104 | 104 | }; |
105 | 105 | this.config.onEntityAction = action => this.onDashboardAction(action); |
106 | - this.config.detailsReadonly = () => this.config.componentsData.dashboardScope === 'customer_user'; | |
106 | + this.config.detailsReadonly = () => (this.config.componentsData.dashboardScope === 'customer_user' || this.config.componentsData.dashboardScope === 'edge_customer_user'); | |
107 | 107 | } |
108 | 108 | |
109 | 109 | resolve(route: ActivatedRouteSnapshot): Observable<EntityTableConfig<DashboardInfo | Dashboard>> { |
... | ... | @@ -147,7 +147,7 @@ export class DashboardsTableConfigResolver implements Resolve<EntityTableConfig< |
147 | 147 | this.config.cellActionDescriptors = this.configureCellActions(this.config.componentsData.dashboardScope); |
148 | 148 | this.config.groupActionDescriptors = this.configureGroupActions(this.config.componentsData.dashboardScope); |
149 | 149 | this.config.addActionDescriptors = this.configureAddActions(this.config.componentsData.dashboardScope); |
150 | - this.config.addEnabled = this.config.componentsData.dashboardScope !== 'customer_user'; | |
150 | + this.config.addEnabled = !(this.config.componentsData.dashboardScope === 'customer_user' || this.config.componentsData.dashboardScope === 'edge_customer_user'); | |
151 | 151 | this.config.entitiesDeleteEnabled = this.config.componentsData.dashboardScope === 'tenant'; |
152 | 152 | this.config.deleteEnabled = () => this.config.componentsData.dashboardScope === 'tenant'; |
153 | 153 | return this.config; | ... | ... |
... | ... | @@ -112,7 +112,7 @@ export class DevicesTableConfigResolver implements Resolve<EntityTableConfig<Dev |
112 | 112 | )); |
113 | 113 | }; |
114 | 114 | this.config.onEntityAction = action => this.onDeviceAction(action); |
115 | - this.config.detailsReadonly = () => this.config.componentsData.deviceScope === 'customer_user'; | |
115 | + this.config.detailsReadonly = () => (this.config.componentsData.deviceScope === 'customer_user' || this.config.componentsData.deviceScope === 'edge_customer_user'); | |
116 | 116 | |
117 | 117 | this.config.headerComponent = DeviceTableHeaderComponent; |
118 | 118 | |
... | ... | @@ -161,7 +161,7 @@ export class DevicesTableConfigResolver implements Resolve<EntityTableConfig<Dev |
161 | 161 | this.config.cellActionDescriptors = this.configureCellActions(this.config.componentsData.deviceScope); |
162 | 162 | this.config.groupActionDescriptors = this.configureGroupActions(this.config.componentsData.deviceScope); |
163 | 163 | this.config.addActionDescriptors = this.configureAddActions(this.config.componentsData.deviceScope); |
164 | - this.config.addEnabled = this.config.componentsData.deviceScope !== 'customer_user'; | |
164 | + this.config.addEnabled = !(this.config.componentsData.deviceScope === 'customer_user' || this.config.componentsData.deviceScope === 'edge_customer_user'); | |
165 | 165 | this.config.entitiesDeleteEnabled = this.config.componentsData.deviceScope === 'tenant'; |
166 | 166 | this.config.deleteEnabled = () => this.config.componentsData.deviceScope === 'tenant'; |
167 | 167 | return this.config; | ... | ... |
... | ... | @@ -59,21 +59,6 @@ const routes: Routes = [ |
59 | 59 | } |
60 | 60 | }, |
61 | 61 | { |
62 | - path: ':edgeId/ruleChains', | |
63 | - component: EntitiesTableComponent, | |
64 | - data: { | |
65 | - auth: [Authority.TENANT_ADMIN], | |
66 | - ruleChainsType: 'edge', | |
67 | - breadcrumb: { | |
68 | - label: 'edge.edge-rulechains', | |
69 | - icon: 'settings_ethernet' | |
70 | - }, | |
71 | - }, | |
72 | - resolve: { | |
73 | - entitiesTableConfig: RuleChainsTableConfigResolver | |
74 | - } | |
75 | - }, | |
76 | - { | |
77 | 62 | path: ':edgeId/assets', |
78 | 63 | component: EntitiesTableComponent, |
79 | 64 | data: { |
... | ... | @@ -157,6 +142,50 @@ const routes: Routes = [ |
157 | 142 | ] |
158 | 143 | }, |
159 | 144 | { |
145 | + path: ':edgeId/ruleChains', | |
146 | + data: { | |
147 | + breadcrumb: { | |
148 | + label: 'edge.edge-rulechains', | |
149 | + icon: 'settings_ethernet' | |
150 | + } | |
151 | + }, | |
152 | + children: [ | |
153 | + { | |
154 | + path: '', | |
155 | + component: EntitiesTableComponent, | |
156 | + data: { | |
157 | + auth: [Authority.TENANT_ADMIN], | |
158 | + title: 'edge.rulechains', | |
159 | + ruleChainsType: 'edge' | |
160 | + }, | |
161 | + resolve: { | |
162 | + entitiesTableConfig: RuleChainsTableConfigResolver | |
163 | + } | |
164 | + }, | |
165 | + { | |
166 | + path: ':ruleChainId', | |
167 | + component: RuleChainPageComponent, | |
168 | + canDeactivate: [ConfirmOnExitGuard], | |
169 | + data: { | |
170 | + breadcrumb: { | |
171 | + labelFunction: ruleChainBreadcumbLabelFunction, | |
172 | + icon: 'settings_ethernet' | |
173 | + } as BreadCrumbConfig<RuleChainPageComponent>, | |
174 | + auth: [Authority.TENANT_ADMIN], | |
175 | + title: 'rulechain.edge-rulechain', | |
176 | + import: false, | |
177 | + ruleChainType: RuleChainType.EDGE | |
178 | + }, | |
179 | + resolve: { | |
180 | + ruleChain: RuleChainResolver, | |
181 | + ruleChainMetaData: ResolvedRuleChainMetaDataResolver, | |
182 | + ruleNodeComponents: RuleNodeComponentsResolver, | |
183 | + tooltipster: TooltipsterResolver | |
184 | + } | |
185 | + } | |
186 | + ] | |
187 | + }, | |
188 | + { | |
160 | 189 | path: 'ruleChains', |
161 | 190 | data: { |
162 | 191 | breadcrumb: { | ... | ... |
... | ... | @@ -44,31 +44,31 @@ |
44 | 44 | <button mat-raised-button color="primary" |
45 | 45 | [disabled]="(isLoading$ | async)" |
46 | 46 | (click)="onEntityAction($event, 'openEdgeAssets')" |
47 | - [fxShow]="!isEdit"> | |
47 | + [fxShow]="!isEdit && edgeScope !== 'customer'"> | |
48 | 48 | {{'edge.manage-edge-assets' | translate }} |
49 | 49 | </button> |
50 | 50 | <button mat-raised-button color="primary" |
51 | 51 | [disabled]="(isLoading$ | async)" |
52 | 52 | (click)="onEntityAction($event, 'openEdgeDevices')" |
53 | - [fxShow]="!isEdit"> | |
53 | + [fxShow]="!isEdit && edgeScope !== 'customer'"> | |
54 | 54 | {{'edge.manage-edge-devices' | translate }} |
55 | 55 | </button> |
56 | 56 | <button mat-raised-button color="primary" |
57 | 57 | [disabled]="(isLoading$ | async)" |
58 | 58 | (click)="onEntityAction($event, 'openEdgeEntityViews')" |
59 | - [fxShow]="!isEdit"> | |
59 | + [fxShow]="!isEdit && edgeScope !== 'customer'"> | |
60 | 60 | {{'edge.manage-edge-entity-views' | translate }} |
61 | 61 | </button> |
62 | 62 | <button mat-raised-button color="primary" |
63 | 63 | [disabled]="(isLoading$ | async)" |
64 | 64 | (click)="onEntityAction($event, 'openEdgeDashboards')" |
65 | - [fxShow]="!isEdit"> | |
65 | + [fxShow]="!isEdit && edgeScope !== 'customer'"> | |
66 | 66 | {{'edge.manage-edge-dashboards' | translate }} |
67 | 67 | </button> |
68 | 68 | <button mat-raised-button color="primary" |
69 | 69 | [disabled]="(isLoading$ | async)" |
70 | 70 | (click)="onEntityAction($event, 'openEdgeRuleChains')" |
71 | - [fxShow]="!isEdit && (edgeScope === 'tenant' || edgeScope === 'customer')"> | |
71 | + [fxShow]="!isEdit && edgeScope === 'tenant'"> | |
72 | 72 | {{'edge.manage-edge-rulechains' | translate }} |
73 | 73 | </button> |
74 | 74 | </div> |
... | ... | @@ -85,7 +85,7 @@ |
85 | 85 | ngxClipboard |
86 | 86 | (cbOnSuccess)="onEdgeInfoCopied('key')" |
87 | 87 | [cbContent]="entity?.routingKey" |
88 | - [fxShow]="!isEdit && (edgeScope === 'tenant' || edgeScope === 'customer')"> | |
88 | + [fxShow]="!isEdit && edgeScope !== 'customer_user'"> | |
89 | 89 | <mat-icon svgIcon="mdi:clipboard-arrow-left"></mat-icon> |
90 | 90 | <span translate>edge.copy-edge-key</span> |
91 | 91 | </button> |
... | ... | @@ -93,7 +93,7 @@ |
93 | 93 | ngxClipboard |
94 | 94 | (cbOnSuccess)="onEdgeInfoCopied('secret')" |
95 | 95 | [cbContent]="entity?.secret" |
96 | - [fxShow]="!isEdit && (edgeScope === 'tenant' || edgeScope === 'customer')"> | |
96 | + [fxShow]="!isEdit && edgeScope !== 'customer_user'"> | |
97 | 97 | <mat-icon svgIcon="mdi:clipboard-arrow-left"></mat-icon> |
98 | 98 | <span translate>edge.copy-edge-secret</span> |
99 | 99 | </button> |
... | ... | @@ -101,7 +101,7 @@ |
101 | 101 | ngxClipboard |
102 | 102 | [disabled]="(isLoading$ | async)" |
103 | 103 | (click)="onEntityAction($event, 'syncEdge')" |
104 | - [fxShow]="!isEdit && (edgeScope === 'tenant' || edgeScope === 'customer')"> | |
104 | + [fxShow]="!isEdit && edgeScope !== 'customer_user'"> | |
105 | 105 | <mat-icon svgIcon="mdi:sync"></mat-icon> |
106 | 106 | <span translate>edge.sync</span> |
107 | 107 | </button> |
... | ... | @@ -132,7 +132,7 @@ |
132 | 132 | [required]="true" |
133 | 133 | [entityType]="entityType.EDGE"> |
134 | 134 | </tb-entity-subtype-autocomplete> |
135 | - <div [fxShow]="(edgeScope === 'tenant' || edgeScope === 'customer')"> | |
135 | + <div [fxShow]="edgeScope !== 'customer_user'"> | |
136 | 136 | <div class="tb-hint" [innerHTML]="'edge.edge-license-key-hint' | translate"></div> |
137 | 137 | <mat-form-field class="mat-block"> |
138 | 138 | <mat-label translate>edge.edge-license-key</mat-label> |
... | ... | @@ -142,7 +142,7 @@ |
142 | 142 | </mat-error> |
143 | 143 | </mat-form-field> |
144 | 144 | </div> |
145 | - <div [fxShow]="(edgeScope === 'tenant' || edgeScope === 'customer')"> | |
145 | + <div [fxShow]="edgeScope !== 'customer_user'"> | |
146 | 146 | <div translate class="tb-hint">edge.cloud-endpoint-hint</div> |
147 | 147 | <mat-form-field class="mat-block"> |
148 | 148 | <mat-label translate>edge.cloud-endpoint</mat-label> |
... | ... | @@ -153,7 +153,7 @@ |
153 | 153 | </mat-form-field> |
154 | 154 | </div> |
155 | 155 | </fieldset> |
156 | - <div fxLayout="row" [fxShow]="(edgeScope === 'tenant' || edgeScope === 'customer')"> | |
156 | + <div fxLayout="row" [fxShow]="edgeScope !== 'customer_user'"> | |
157 | 157 | <mat-form-field class="mat-block" fxFlex> |
158 | 158 | <mat-label translate>edge.edge-key</mat-label> |
159 | 159 | <input matInput formControlName="routingKey"> |
... | ... | @@ -164,7 +164,7 @@ |
164 | 164 | <mat-icon svgIcon="mdi:clipboard-arrow-left"></mat-icon> |
165 | 165 | </button> |
166 | 166 | </div> |
167 | - <div fxLayout="row" [fxShow]="(edgeScope === 'tenant' || edgeScope === 'customer')"> | |
167 | + <div fxLayout="row" [fxShow]="edgeScope !== 'customer_user'"> | |
168 | 168 | <mat-form-field class="mat-block" fxFlex> |
169 | 169 | <mat-label translate>edge.edge-secret</mat-label> |
170 | 170 | <input matInput formControlName="secret"> | ... | ... |
... | ... | @@ -477,8 +477,8 @@ export class EdgesTableConfigResolver implements Resolve<EntityTableConfig<EdgeI |
477 | 477 | $event.stopPropagation(); |
478 | 478 | } |
479 | 479 | this.dialogService.confirm( |
480 | - this.translate.instant('edge.unassign-edge-title', {count: edges.length}), | |
481 | - this.translate.instant('edge.unassign-edge-text'), | |
480 | + this.translate.instant('edge.unassign-edges-title', {count: edges.length}), | |
481 | + this.translate.instant('edge.unassign-edges-text'), | |
482 | 482 | this.translate.instant('action.no'), |
483 | 483 | this.translate.instant('action.yes'), |
484 | 484 | true | ... | ... |
... | ... | @@ -104,7 +104,7 @@ export class EntityViewsTableConfigResolver implements Resolve<EntityTableConfig |
104 | 104 | )); |
105 | 105 | }; |
106 | 106 | this.config.onEntityAction = action => this.onEntityViewAction(action); |
107 | - this.config.detailsReadonly = () => this.config.componentsData.entityViewScope === 'customer_user'; | |
107 | + this.config.detailsReadonly = () => (this.config.componentsData.entityViewScope === 'customer_user' || this.config.componentsData.entityViewScope === 'edge_customer_user'); | |
108 | 108 | |
109 | 109 | this.config.headerComponent = EntityViewTableHeaderComponent; |
110 | 110 | |
... | ... | @@ -152,7 +152,7 @@ export class EntityViewsTableConfigResolver implements Resolve<EntityTableConfig |
152 | 152 | this.config.cellActionDescriptors = this.configureCellActions(this.config.componentsData.entityViewScope); |
153 | 153 | this.config.groupActionDescriptors = this.configureGroupActions(this.config.componentsData.entityViewScope); |
154 | 154 | this.config.addActionDescriptors = this.configureAddActions(this.config.componentsData.entityViewScope); |
155 | - this.config.addEnabled = this.config.componentsData.entityViewScope !== 'customer_user'; | |
155 | + this.config.addEnabled = !(this.config.componentsData.entityViewScope === 'customer_user' || this.config.componentsData.entityViewScope !== 'edge_customer_user'); | |
156 | 156 | this.config.entitiesDeleteEnabled = this.config.componentsData.entityViewScope === 'tenant'; |
157 | 157 | this.config.deleteEnabled = () => this.config.componentsData.entityViewScope === 'tenant'; |
158 | 158 | return this.config; | ... | ... |
... | ... | @@ -57,13 +57,13 @@ |
57 | 57 | </mat-form-field> |
58 | 58 | <mat-form-field class="mat-block"> |
59 | 59 | <mat-label translate>language.language</mat-label> |
60 | - <mat-select matInput formControlName="language"> | |
60 | + <mat-select formControlName="language"> | |
61 | 61 | <mat-option *ngFor="let lang of languageList" [value]="lang"> |
62 | 62 | {{ lang ? ('language.locales.' + lang | translate) : ''}} |
63 | 63 | </mat-option> |
64 | 64 | </mat-select> |
65 | 65 | </mat-form-field> |
66 | - <section class="tb-home-dashboard" fxFlex fxLayout="column" fxLayout.gt-sm="row"> | |
66 | + <section class="tb-home-dashboard" fxFlex fxLayout="column" fxLayout.gt-sm="row" *ngIf="!isSysAdmin()"> | |
67 | 67 | <tb-dashboard-autocomplete |
68 | 68 | fxFlex |
69 | 69 | placeholder="{{ 'dashboard.home-dashboard' | translate }}" | ... | ... |
... | ... | @@ -16,7 +16,7 @@ |
16 | 16 | |
17 | 17 | import { Component, OnInit } from '@angular/core'; |
18 | 18 | import { UserService } from '@core/http/user.service'; |
19 | -import { User } from '@shared/models/user.model'; | |
19 | +import { AuthUser, User } from '@shared/models/user.model'; | |
20 | 20 | import { Authority } from '@shared/models/authority.enum'; |
21 | 21 | import { PageComponent } from '@shared/components/page.component'; |
22 | 22 | import { Store } from '@ngrx/store'; |
... | ... | @@ -33,6 +33,7 @@ import { DialogService } from '@core/services/dialog.service'; |
33 | 33 | import { AuthService } from '@core/auth/auth.service'; |
34 | 34 | import { ActivatedRoute } from '@angular/router'; |
35 | 35 | import { isDefinedAndNotNull } from '@core/utils'; |
36 | +import { getCurrentAuthUser } from '@core/auth/auth.selectors'; | |
36 | 37 | |
37 | 38 | @Component({ |
38 | 39 | selector: 'tb-profile', |
... | ... | @@ -45,6 +46,7 @@ export class ProfileComponent extends PageComponent implements OnInit, HasConfir |
45 | 46 | profile: FormGroup; |
46 | 47 | user: User; |
47 | 48 | languageList = env.supportedLangs; |
49 | + private readonly authUser: AuthUser; | |
48 | 50 | |
49 | 51 | constructor(protected store: Store<AppState>, |
50 | 52 | private route: ActivatedRoute, |
... | ... | @@ -55,6 +57,7 @@ export class ProfileComponent extends PageComponent implements OnInit, HasConfir |
55 | 57 | public dialogService: DialogService, |
56 | 58 | public fb: FormBuilder) { |
57 | 59 | super(store); |
60 | + this.authUser = getCurrentAuthUser(this.store); | |
58 | 61 | } |
59 | 62 | |
60 | 63 | ngOnInit() { |
... | ... | @@ -134,4 +137,8 @@ export class ProfileComponent extends PageComponent implements OnInit, HasConfir |
134 | 137 | return this.profile; |
135 | 138 | } |
136 | 139 | |
140 | + isSysAdmin(): boolean { | |
141 | + return this.authUser.authority === Authority.SYS_ADMIN; | |
142 | + } | |
143 | + | |
137 | 144 | } | ... | ... |
... | ... | @@ -24,7 +24,6 @@ import { |
24 | 24 | import { Resolve } from '@angular/router'; |
25 | 25 | import { Resource, ResourceInfo, ResourceTypeTranslationMap } from '@shared/models/resource.models'; |
26 | 26 | import { EntityType, entityTypeResources, entityTypeTranslations } from '@shared/models/entity-type.models'; |
27 | -import { Direction } from '@shared/models/page/sort-order'; | |
28 | 27 | import { NULL_UUID } from '@shared/models/id/has-uuid'; |
29 | 28 | import { DatePipe } from '@angular/common'; |
30 | 29 | import { TranslateService } from '@ngx-translate/core'; |
... | ... | @@ -34,13 +33,14 @@ import { Store } from '@ngrx/store'; |
34 | 33 | import { AppState } from '@core/core.state'; |
35 | 34 | import { Authority } from '@shared/models/authority.enum'; |
36 | 35 | import { ResourcesLibraryComponent } from '@home/pages/resource/resources-library.component'; |
37 | -import { Observable } from 'rxjs/internal/Observable'; | |
38 | -import { PageData } from '@shared/models/page/page-data'; | |
36 | +import { PageLink } from '@shared/models/page/page-link'; | |
37 | +import { EntityAction } from '@home/models/entity/entity-component.models'; | |
38 | +import { map } from 'rxjs/operators'; | |
39 | 39 | |
40 | 40 | @Injectable() |
41 | -export class ResourcesLibraryTableConfigResolver implements Resolve<EntityTableConfig<Resource>> { | |
41 | +export class ResourcesLibraryTableConfigResolver implements Resolve<EntityTableConfig<Resource, PageLink, ResourceInfo>> { | |
42 | 42 | |
43 | - private readonly config: EntityTableConfig<Resource> = new EntityTableConfig<Resource>(); | |
43 | + private readonly config: EntityTableConfig<Resource, PageLink, ResourceInfo> = new EntityTableConfig<Resource, PageLink, ResourceInfo>(); | |
44 | 44 | private readonly resourceTypesTranslationMap = ResourceTypeTranslationMap; |
45 | 45 | |
46 | 46 | constructor(private store: Store<AppState>, |
... | ... | @@ -52,17 +52,16 @@ export class ResourcesLibraryTableConfigResolver implements Resolve<EntityTableC |
52 | 52 | this.config.entityComponent = ResourcesLibraryComponent; |
53 | 53 | this.config.entityTranslations = entityTypeTranslations.get(EntityType.TB_RESOURCE); |
54 | 54 | this.config.entityResources = entityTypeResources.get(EntityType.TB_RESOURCE); |
55 | - this.config.defaultSortOrder = {property: 'title', direction: Direction.ASC}; | |
56 | 55 | |
57 | 56 | this.config.entityTitle = (resource) => resource ? |
58 | 57 | resource.title : ''; |
59 | 58 | |
60 | 59 | this.config.columns.push( |
61 | 60 | new DateEntityTableColumn<ResourceInfo>('createdTime', 'common.created-time', this.datePipe, '150px'), |
62 | - new EntityTableColumn<ResourceInfo>('title', 'widgets-bundle.title', '60%'), | |
61 | + new EntityTableColumn<ResourceInfo>('title', 'resource.title', '60%'), | |
63 | 62 | new EntityTableColumn<ResourceInfo>('resourceType', 'resource.resource-type', '40%', |
64 | 63 | entity => this.resourceTypesTranslationMap.get(entity.resourceType)), |
65 | - new EntityTableColumn<ResourceInfo>('tenantId', 'widgets-bundle.system', '60px', | |
64 | + new EntityTableColumn<ResourceInfo>('tenantId', 'resource.system', '60px', | |
66 | 65 | entity => { |
67 | 66 | return checkBoxCell(entity.tenantId.id === NULL_UUID); |
68 | 67 | }), |
... | ... | @@ -83,13 +82,34 @@ export class ResourcesLibraryTableConfigResolver implements Resolve<EntityTableC |
83 | 82 | this.config.deleteEntitiesTitle = count => this.translate.instant('resource.delete-resources-title', {count}); |
84 | 83 | this.config.deleteEntitiesContent = () => this.translate.instant('resource.delete-resources-text'); |
85 | 84 | |
86 | - this.config.entitiesFetchFunction = pageLink => this.resourceService.getResources(pageLink) as Observable<PageData<Resource>>; | |
85 | + this.config.entitiesFetchFunction = pageLink => this.resourceService.getResources(pageLink); | |
87 | 86 | this.config.loadEntity = id => this.resourceService.getResource(id.id); |
88 | - this.config.saveEntity = resource => this.resourceService.saveResource(resource); | |
87 | + this.config.saveEntity = resource => this.saveResource(resource); | |
89 | 88 | this.config.deleteEntity = id => this.resourceService.deleteResource(id.id); |
89 | + | |
90 | + this.config.onEntityAction = action => this.onResourceAction(action); | |
91 | + } | |
92 | + | |
93 | + saveResource(resource) { | |
94 | + if (Array.isArray(resource.data)) { | |
95 | + const resources = []; | |
96 | + resource.data.forEach((data, index) => { | |
97 | + resources.push({ | |
98 | + resourceType: resource.resourceType, | |
99 | + data, | |
100 | + fileName: resource.fileName[index], | |
101 | + title: resource.title | |
102 | + }); | |
103 | + }); | |
104 | + return this.resourceService.saveResources(resources, {resendRequest: true}).pipe( | |
105 | + map((response) => response[0]) | |
106 | + ); | |
107 | + } else { | |
108 | + return this.resourceService.saveResource(resource); | |
109 | + } | |
90 | 110 | } |
91 | 111 | |
92 | - resolve(): EntityTableConfig<Resource> { | |
112 | + resolve(): EntityTableConfig<Resource, PageLink, ResourceInfo> { | |
93 | 113 | this.config.tableTitle = this.translate.instant('resource.resources-library'); |
94 | 114 | const authUser = getCurrentAuthUser(this.store); |
95 | 115 | this.config.deleteEnabled = (resource) => this.isResourceEditable(resource, authUser.authority); |
... | ... | @@ -105,7 +125,16 @@ export class ResourcesLibraryTableConfigResolver implements Resolve<EntityTableC |
105 | 125 | this.resourceService.downloadResource(resource.id.id).subscribe(); |
106 | 126 | } |
107 | 127 | |
108 | - private isResourceEditable(resource: Resource, authority: Authority): boolean { | |
128 | + onResourceAction(action: EntityAction<ResourceInfo>): boolean { | |
129 | + switch (action.action) { | |
130 | + case 'uploadResource': | |
131 | + this.exportResource(action.event, action.entity); | |
132 | + return true; | |
133 | + } | |
134 | + return false; | |
135 | + } | |
136 | + | |
137 | + private isResourceEditable(resource: ResourceInfo, authority: Authority): boolean { | |
109 | 138 | if (authority === Authority.TENANT_ADMIN) { |
110 | 139 | return resource && resource.tenantId && resource.tenantId.id !== NULL_UUID; |
111 | 140 | } else { | ... | ... |
... | ... | @@ -18,6 +18,12 @@ |
18 | 18 | <div class="tb-details-buttons" fxLayout.xs="column"> |
19 | 19 | <button mat-raised-button color="primary" fxFlex.xs |
20 | 20 | [disabled]="(isLoading$ | async)" |
21 | + (click)="onEntityAction($event, 'uploadResource')" | |
22 | + [fxShow]="!isEdit"> | |
23 | + {{'resource.export' | translate }} | |
24 | + </button> | |
25 | + <button mat-raised-button color="primary" fxFlex.xs | |
26 | + [disabled]="(isLoading$ | async)" | |
21 | 27 | (click)="onEntityAction($event, 'delete')" |
22 | 28 | [fxShow]="!hideDelete() && !isEdit"> |
23 | 29 | {{'resource.delete' | translate }} |
... | ... | @@ -44,9 +50,11 @@ |
44 | 50 | <tb-file-input |
45 | 51 | formControlName="data" |
46 | 52 | required |
47 | - [convertToBase64]="true" | |
53 | + [readAsBinary]="true" | |
48 | 54 | [allowedExtensions]="getAllowedExtensions()" |
55 | + [contentConvertFunction]="convertToBase64File" | |
49 | 56 | [accept]="getAcceptType()" |
57 | + [multipleFile]="entityForm.get('resourceType').value === resourceType.LWM2M_MODEL" | |
50 | 58 | dropLabel="{{'resource.drop-file' | translate}}" |
51 | 59 | [existingFileName]="entityForm.get('fileName')?.value" |
52 | 60 | (fileNameChanged)="entityForm?.get('fileName').patchValue($event)"> | ... | ... |
... | ... | @@ -29,7 +29,7 @@ import { |
29 | 29 | ResourceTypeMIMETypes, |
30 | 30 | ResourceTypeTranslationMap |
31 | 31 | } from '@shared/models/resource.models'; |
32 | -import { distinctUntilChanged, takeUntil } from 'rxjs/operators'; | |
32 | +import { pairwise, startWith, takeUntil } from 'rxjs/operators'; | |
33 | 33 | |
34 | 34 | @Component({ |
35 | 35 | selector: 'tb-resources-library', |
... | ... | @@ -54,15 +54,22 @@ export class ResourcesLibraryComponent extends EntityComponent<Resource> impleme |
54 | 54 | ngOnInit() { |
55 | 55 | super.ngOnInit(); |
56 | 56 | this.entityForm.get('resourceType').valueChanges.pipe( |
57 | - distinctUntilChanged((oldValue, newValue) => [oldValue, newValue].includes(this.resourceType.LWM2M_MODEL)), | |
57 | + startWith(ResourceType.LWM2M_MODEL), | |
58 | + pairwise(), | |
58 | 59 | takeUntil(this.destroy$) |
59 | - ).subscribe((type) => { | |
60 | + ).subscribe(([previousType, type]) => { | |
61 | + if (previousType === this.resourceType.LWM2M_MODEL) { | |
62 | + this.entityForm.get('title').setValidators(Validators.required); | |
63 | + this.entityForm.get('title').updateValueAndValidity({emitEvent: false}); | |
64 | + } | |
60 | 65 | if (type === this.resourceType.LWM2M_MODEL) { |
61 | 66 | this.entityForm.get('title').clearValidators(); |
62 | - } else { | |
63 | - this.entityForm.get('title').setValidators(Validators.required); | |
67 | + this.entityForm.get('title').updateValueAndValidity({emitEvent: false}); | |
64 | 68 | } |
65 | - this.entityForm.get('title').updateValueAndValidity({emitEvent: false}); | |
69 | + this.entityForm.patchValue({ | |
70 | + data: null, | |
71 | + fileName: null | |
72 | + }, {emitEvent: false}); | |
66 | 73 | }); |
67 | 74 | } |
68 | 75 | |
... | ... | @@ -119,4 +126,8 @@ export class ResourcesLibraryComponent extends EntityComponent<Resource> impleme |
119 | 126 | return '*/*'; |
120 | 127 | } |
121 | 128 | } |
129 | + | |
130 | + convertToBase64File(data: string): string { | |
131 | + return window.btoa(data); | |
132 | + } | |
122 | 133 | } | ... | ... |
... | ... | @@ -19,19 +19,19 @@ |
19 | 19 | <button mat-raised-button color="primary" |
20 | 20 | [disabled]="(isLoading$ | async)" |
21 | 21 | (click)="onEntityAction($event, 'open')" |
22 | - [fxShow]="!isEdit && (ruleChainScope === 'tenant' || ruleChainScope === 'edges')"> | |
22 | + [fxShow]="!isEdit"> | |
23 | 23 | {{'rulechain.open-rulechain' | translate }} |
24 | 24 | </button> |
25 | 25 | <button mat-raised-button color="primary" |
26 | 26 | [disabled]="(isLoading$ | async)" |
27 | 27 | (click)="onEntityAction($event, 'export')" |
28 | - [fxShow]="!isEdit && (ruleChainScope === 'tenant' || ruleChainScope === 'edges')"> | |
28 | + [fxShow]="!isEdit"> | |
29 | 29 | {{'rulechain.export' | translate }} |
30 | 30 | </button> |
31 | 31 | <button mat-raised-button color="primary" |
32 | 32 | [disabled]="(isLoading$ | async)" |
33 | 33 | (click)="onEntityAction($event, 'setRoot')" |
34 | - [fxShow]="!isEdit && !entity?.root && ruleChainScope === 'tenant'"> | |
34 | + [fxShow]="!isEdit && ((!entity?.root && ruleChainScope === 'tenant') || (!isEdgeRootRuleChain() && ruleChainScope === 'edge'))"> | |
35 | 35 | {{'rulechain.set-root' | translate }} |
36 | 36 | </button> |
37 | 37 | <button mat-raised-button color="primary" |
... | ... | @@ -54,12 +54,6 @@ |
54 | 54 | </button> |
55 | 55 | <button mat-raised-button color="primary" |
56 | 56 | [disabled]="(isLoading$ | async)" |
57 | - (click)="onEntityAction($event, 'setRoot')" | |
58 | - [fxShow]="!isEdit && !isEdgeRootRuleChain() && ruleChainScope === 'edge'"> | |
59 | - {{'rulechain.set-root' | translate }} | |
60 | - </button> | |
61 | - <button mat-raised-button color="primary" | |
62 | - [disabled]="(isLoading$ | async)" | |
63 | 57 | (click)="onEntityAction($event, 'unassignFromEdge')" |
64 | 58 | [fxShow]="!isEdit && !isEdgeRootRuleChain() && ruleChainScope === 'edge'"> |
65 | 59 | {{'edge.unassign-from-edge' | translate }} |
... | ... | @@ -67,7 +61,7 @@ |
67 | 61 | <button mat-raised-button color="primary" |
68 | 62 | [disabled]="(isLoading$ | async)" |
69 | 63 | (click)="onEntityAction($event, 'delete')" |
70 | - [fxShow]="!hideDelete() && !isEdit && (ruleChainScope === 'tenant' || ruleChainScope === 'edges')"> | |
64 | + [fxShow]="!hideDelete() && !isEdit && ruleChainScope !== 'edge'"> | |
71 | 65 | {{'rulechain.delete' | translate }} |
72 | 66 | </button> |
73 | 67 | <div fxLayout="row"> | ... | ... |
... | ... | @@ -211,53 +211,51 @@ export class RuleChainsTableConfigResolver implements Resolve<EntityTableConfig< |
211 | 211 | |
212 | 212 | configureCellActions(ruleChainScope: string): Array<CellActionDescriptor<RuleChain>> { |
213 | 213 | const actions: Array<CellActionDescriptor<RuleChain>> = []; |
214 | - if (ruleChainScope === 'tenant' || ruleChainScope === 'edges') { | |
214 | + actions.push( | |
215 | + { | |
216 | + name: this.translate.instant('rulechain.open-rulechain'), | |
217 | + icon: 'settings_ethernet', | |
218 | + isEnabled: () => true, | |
219 | + onAction: ($event, entity) => this.openRuleChain($event, entity) | |
220 | + }, | |
221 | + { | |
222 | + name: this.translate.instant('rulechain.export'), | |
223 | + icon: 'file_download', | |
224 | + isEnabled: () => true, | |
225 | + onAction: ($event, entity) => this.exportRuleChain($event, entity) | |
226 | + } | |
227 | + ); | |
228 | + if (ruleChainScope === 'tenant') { | |
215 | 229 | actions.push( |
216 | 230 | { |
217 | - name: this.translate.instant('rulechain.open-rulechain'), | |
218 | - icon: 'settings_ethernet', | |
219 | - isEnabled: () => true, | |
220 | - onAction: ($event, entity) => this.openRuleChain($event, entity) | |
231 | + name: this.translate.instant('rulechain.set-root'), | |
232 | + icon: 'flag', | |
233 | + isEnabled: (entity) => this.isNonRootRuleChain(entity), | |
234 | + onAction: ($event, entity) => this.setRootRuleChain($event, entity) | |
235 | + } | |
236 | + ); | |
237 | + } | |
238 | + if (ruleChainScope === 'edges') { | |
239 | + actions.push( | |
240 | + { | |
241 | + name: this.translate.instant('rulechain.set-edge-template-root-rulechain'), | |
242 | + icon: 'flag', | |
243 | + isEnabled: (entity) => this.isNonRootRuleChain(entity), | |
244 | + onAction: ($event, entity) => this.setEdgeTemplateRootRuleChain($event, entity) | |
221 | 245 | }, |
222 | 246 | { |
223 | - name: this.translate.instant('rulechain.export'), | |
224 | - icon: 'file_download', | |
225 | - isEnabled: () => true, | |
226 | - onAction: ($event, entity) => this.exportRuleChain($event, entity) | |
247 | + name: this.translate.instant('rulechain.set-auto-assign-to-edge'), | |
248 | + icon: 'bookmark_outline', | |
249 | + isEnabled: (entity) => this.isNotAutoAssignToEdgeRuleChain(entity), | |
250 | + onAction: ($event, entity) => this.setAutoAssignToEdgeRuleChain($event, entity) | |
251 | + }, | |
252 | + { | |
253 | + name: this.translate.instant('rulechain.unset-auto-assign-to-edge'), | |
254 | + icon: 'bookmark', | |
255 | + isEnabled: (entity) => this.isAutoAssignToEdgeRuleChain(entity), | |
256 | + onAction: ($event, entity) => this.unsetAutoAssignToEdgeRuleChain($event, entity) | |
227 | 257 | } |
228 | 258 | ); |
229 | - if (ruleChainScope === 'tenant') { | |
230 | - actions.push( | |
231 | - { | |
232 | - name: this.translate.instant('rulechain.set-root'), | |
233 | - icon: 'flag', | |
234 | - isEnabled: (entity) => this.isNonRootRuleChain(entity), | |
235 | - onAction: ($event, entity) => this.setRootRuleChain($event, entity) | |
236 | - } | |
237 | - ); | |
238 | - } | |
239 | - if (ruleChainScope === 'edges') { | |
240 | - actions.push( | |
241 | - { | |
242 | - name: this.translate.instant('rulechain.set-edge-template-root-rulechain'), | |
243 | - icon: 'flag', | |
244 | - isEnabled: (entity) => this.isNonRootRuleChain(entity), | |
245 | - onAction: ($event, entity) => this.setEdgeTemplateRootRuleChain($event, entity) | |
246 | - }, | |
247 | - { | |
248 | - name: this.translate.instant('rulechain.set-auto-assign-to-edge'), | |
249 | - icon: 'bookmark_outline', | |
250 | - isEnabled: (entity) => this.isNotAutoAssignToEdgeRuleChain(entity), | |
251 | - onAction: ($event, entity) => this.setAutoAssignToEdgeRuleChain($event, entity) | |
252 | - }, | |
253 | - { | |
254 | - name: this.translate.instant('rulechain.unset-auto-assign-to-edge'), | |
255 | - icon: 'bookmark', | |
256 | - isEnabled: (entity) => this.isAutoAssignToEdgeRuleChain(entity), | |
257 | - onAction: ($event, entity) => this.unsetAutoAssignToEdgeRuleChain($event, entity) | |
258 | - } | |
259 | - ); | |
260 | - } | |
261 | 259 | } |
262 | 260 | if (ruleChainScope === 'edge') { |
263 | 261 | actions.push( |
... | ... | @@ -301,6 +299,8 @@ export class RuleChainsTableConfigResolver implements Resolve<EntityTableConfig< |
301 | 299 | } |
302 | 300 | if (this.config.componentsData.ruleChainScope === 'edges') { |
303 | 301 | this.router.navigateByUrl(`edges/ruleChains/${ruleChain.id.id}`); |
302 | + } else if (this.config.componentsData.ruleChainScope === 'edge') { | |
303 | + this.router.navigateByUrl(`edges/${this.config.componentsData.edgeId}/ruleChains/${ruleChain.id.id}`); | |
304 | 304 | } else { |
305 | 305 | this.router.navigateByUrl(`ruleChains/${ruleChain.id.id}`); |
306 | 306 | } | ... | ... |
... | ... | @@ -18,7 +18,7 @@ |
18 | 18 | <div class="tb-container"> |
19 | 19 | <label class="tb-title">{{ label }}</label> |
20 | 20 | <ng-container #flow="flow" |
21 | - [flowConfig]="{singleFile: true, allowDuplicateUploads: true}"> | |
21 | + [flowConfig]="{allowDuplicateUploads: true}"> | |
22 | 22 | <div class="tb-file-select-container"> |
23 | 23 | <div class="tb-file-clear-container"> |
24 | 24 | <button mat-button mat-icon-button color="primary" |
... | ... | @@ -34,7 +34,7 @@ |
34 | 34 | flowDrop |
35 | 35 | [flow]="flow.flowJs"> |
36 | 36 | <label for="{{inputId}}">{{ dropLabel }}</label> |
37 | - <input class="file-input" flowButton type="file" [flow]="flow.flowJs" [flowAttributes]="{accept: accept}" id="{{inputId}}"> | |
37 | + <input class="file-input" flowButton #flowInput type="file" [flow]="flow.flowJs" [flowAttributes]="{accept: accept}" id="{{inputId}}"> | |
38 | 38 | </div> |
39 | 39 | </div> |
40 | 40 | </ng-container> | ... | ... |
... | ... | @@ -17,6 +17,7 @@ |
17 | 17 | import { |
18 | 18 | AfterViewInit, |
19 | 19 | Component, |
20 | + ElementRef, | |
20 | 21 | EventEmitter, |
21 | 22 | forwardRef, |
22 | 23 | Input, |
... | ... | @@ -102,17 +103,34 @@ export class FileInputComponent extends PageComponent implements AfterViewInit, |
102 | 103 | existingFileName: string; |
103 | 104 | |
104 | 105 | @Input() |
105 | - convertToBase64 = false; | |
106 | + readAsBinary = false; | |
107 | + | |
108 | + private multipleFileValue = false; | |
109 | + | |
110 | + @Input() | |
111 | + set multipleFile(value: boolean) { | |
112 | + this.multipleFileValue = value; | |
113 | + if (this.flow?.flowJs) { | |
114 | + this.updateMultipleFileMode(this.multipleFile); | |
115 | + } | |
116 | + } | |
117 | + | |
118 | + get multipleFile(): boolean { | |
119 | + return this.multipleFileValue; | |
120 | + } | |
106 | 121 | |
107 | 122 | @Output() |
108 | - fileNameChanged = new EventEmitter<string>(); | |
123 | + fileNameChanged = new EventEmitter<string|string[]>(); | |
109 | 124 | |
110 | - fileName: string; | |
125 | + fileName: string | string[]; | |
111 | 126 | fileContent: any; |
112 | 127 | |
113 | 128 | @ViewChild('flow', {static: true}) |
114 | 129 | flow: FlowDirective; |
115 | 130 | |
131 | + @ViewChild('flowInput', {static: true}) | |
132 | + flowInput: ElementRef; | |
133 | + | |
116 | 134 | autoUploadSubscription: Subscription; |
117 | 135 | |
118 | 136 | private propagateChange = null; |
... | ... | @@ -125,34 +143,60 @@ export class FileInputComponent extends PageComponent implements AfterViewInit, |
125 | 143 | |
126 | 144 | ngAfterViewInit() { |
127 | 145 | this.autoUploadSubscription = this.flow.events$.subscribe(event => { |
128 | - if (event.type === 'fileAdded') { | |
129 | - const file = event.event[0] as flowjs.FlowFile; | |
130 | - if (this.filterFile(file)) { | |
131 | - const reader = new FileReader(); | |
132 | - reader.onload = (loadEvent) => { | |
133 | - if (typeof reader.result === 'string') { | |
134 | - const fileContent = this.convertToBase64 ? window.btoa(reader.result) : reader.result; | |
135 | - if (fileContent && fileContent.length > 0) { | |
136 | - if (this.contentConvertFunction) { | |
137 | - this.fileContent = this.contentConvertFunction(fileContent); | |
138 | - } else { | |
139 | - this.fileContent = fileContent; | |
140 | - } | |
141 | - if (this.fileContent) { | |
142 | - this.fileName = file.name; | |
143 | - } else { | |
144 | - this.fileName = null; | |
145 | - } | |
146 | - this.updateModel(); | |
147 | - } | |
146 | + if (event.type === 'filesAdded') { | |
147 | + const readers = []; | |
148 | + (event.event[0] as flowjs.FlowFile[]).forEach(file => { | |
149 | + if (this.filterFile(file)) { | |
150 | + readers.push(this.readerAsFile(file)); | |
151 | + } | |
152 | + }); | |
153 | + if (readers.length) { | |
154 | + Promise.all(readers).then((filesContent) => { | |
155 | + filesContent = filesContent.filter(content => content.fileContent != null); | |
156 | + if (filesContent.length === 1) { | |
157 | + this.fileContent = filesContent[0].fileContent; | |
158 | + this.fileName = filesContent[0].fileName; | |
159 | + this.updateModel(); | |
160 | + } else if (filesContent.length > 1) { | |
161 | + this.fileContent = filesContent.map(content => content.fileContent); | |
162 | + this.fileName = filesContent.map(content => content.fileName); | |
163 | + this.updateModel(); | |
164 | + } | |
165 | + }); | |
166 | + } | |
167 | + } | |
168 | + }); | |
169 | + if (!this.multipleFile) { | |
170 | + this.updateMultipleFileMode(this.multipleFile); | |
171 | + } | |
172 | + } | |
173 | + | |
174 | + private readerAsFile(file: flowjs.FlowFile): Promise<any> { | |
175 | + return new Promise((resolve) => { | |
176 | + const reader = new FileReader(); | |
177 | + reader.onload = () => { | |
178 | + let fileName = null; | |
179 | + let fileContent = null; | |
180 | + if (typeof reader.result === 'string') { | |
181 | + fileContent = reader.result; | |
182 | + if (fileContent && fileContent.length > 0) { | |
183 | + if (this.contentConvertFunction) { | |
184 | + fileContent = this.contentConvertFunction(fileContent); | |
185 | + } | |
186 | + if (fileContent) { | |
187 | + fileName = file.name; | |
148 | 188 | } |
149 | - }; | |
150 | - if (this.convertToBase64) { | |
151 | - reader.readAsBinaryString(file.file); | |
152 | - } else { | |
153 | - reader.readAsText(file.file); | |
154 | 189 | } |
155 | 190 | } |
191 | + resolve({fileContent, fileName}); | |
192 | + }; | |
193 | + reader.onerror = () => { | |
194 | + resolve({fileContent: null, fileName: null}); | |
195 | + }; | |
196 | + if (this.readAsBinary) { | |
197 | + reader.readAsBinaryString(file.file); | |
198 | + } else { | |
199 | + reader.readAsText(file.file); | |
156 | 200 | } |
157 | 201 | }); |
158 | 202 | } |
... | ... | @@ -207,4 +251,11 @@ export class FileInputComponent extends PageComponent implements AfterViewInit, |
207 | 251 | this.fileContent = null; |
208 | 252 | this.updateModel(); |
209 | 253 | } |
254 | + | |
255 | + private updateMultipleFileMode(multiple: boolean) { | |
256 | + this.flow.flowJs.opts.singleFile = !multiple; | |
257 | + if (!multiple) { | |
258 | + this.flowInput.nativeElement.removeAttribute('multiple'); | |
259 | + } | |
260 | + } | |
210 | 261 | } | ... | ... |
... | ... | @@ -22,7 +22,7 @@ import { ActionNotificationHide, ActionNotificationShow } from '@core/notificati |
22 | 22 | import { Store } from '@ngrx/store'; |
23 | 23 | import { AppState } from '@core/core.state'; |
24 | 24 | import { CancelAnimationFrame, RafService } from '@core/services/raf.service'; |
25 | -import { guid, isUndefined } from '@core/utils'; | |
25 | +import { guid, isUndefined, isDefinedAndNotNull, isLiteralObject } from '@core/utils'; | |
26 | 26 | import { ResizeObserver } from '@juggle/resize-observer'; |
27 | 27 | import { getAce } from '@shared/models/ace/ace.models'; |
28 | 28 | |
... | ... | @@ -230,8 +230,7 @@ export class JsonObjectEditComponent implements OnInit, ControlValueAccessor, Va |
230 | 230 | this.contentValue = ''; |
231 | 231 | this.objectValid = false; |
232 | 232 | try { |
233 | - | |
234 | - if (this.modelValue) { | |
233 | + if (isDefinedAndNotNull(this.modelValue)) { | |
235 | 234 | this.contentValue = JSON.stringify(this.modelValue, isUndefined(this.sort) ? undefined : |
236 | 235 | (key, objectValue) => { |
237 | 236 | return this.sort(key, objectValue); |
... | ... | @@ -260,6 +259,9 @@ export class JsonObjectEditComponent implements OnInit, ControlValueAccessor, Va |
260 | 259 | if (this.contentValue && this.contentValue.length > 0) { |
261 | 260 | try { |
262 | 261 | data = JSON.parse(this.contentValue); |
262 | + if (!isLiteralObject(data)) { | |
263 | + throw new TypeError(`Value is not a valid JSON`); | |
264 | + } | |
263 | 265 | this.objectValid = true; |
264 | 266 | this.validationError = ''; |
265 | 267 | } catch (ex) { | ... | ... |
... | ... | @@ -15,8 +15,9 @@ |
15 | 15 | limitations under the License. |
16 | 16 | |
17 | 17 | --> |
18 | -<section fxLayout="column" fxLayoutAlign="start start"> | |
19 | - <section fxLayout="row" fxLayout.xs="row wrap" fxLayoutAlign="start start" fxLayoutGap="16px"> | |
18 | +<section fxLayout="column" fxLayoutAlign="start stretch"> | |
19 | + <section fxLayout="row" fxLayoutAlign="start start" fxLayoutGap="16px" | |
20 | + fxLayout.xs="column" fxLayoutAlign.xs="start stretch" fxLayoutGap.xs="0"> | |
20 | 21 | <mat-form-field> |
21 | 22 | <mat-placeholder translate>datetime.date-from</mat-placeholder> |
22 | 23 | <mat-datetimepicker-toggle [for]="startDatePicker" matPrefix></mat-datetimepicker-toggle> |
... | ... | @@ -30,7 +31,8 @@ |
30 | 31 | <input matInput [disabled]="disabled" [(ngModel)]="startDate" [matDatetimepicker]="startTimePicker" (ngModelChange)="onStartDateChange()"> |
31 | 32 | </mat-form-field> |
32 | 33 | </section> |
33 | - <section fxLayout="row" fxLayout.xs="row wrap" fxLayoutAlign="start start" fxLayoutGap="16px"> | |
34 | + <section fxLayout="row" fxLayoutAlign="start start" fxLayoutGap="16px" | |
35 | + fxLayout.xs="column" fxLayoutAlign.xs="start stretch" fxLayoutGap.xs="0"> | |
34 | 36 | <mat-form-field> |
35 | 37 | <mat-placeholder translate>datetime.date-to</mat-placeholder> |
36 | 38 | <mat-datetimepicker-toggle [for]="endDatePicker" matPrefix></mat-datetimepicker-toggle> | ... | ... |
... | ... | @@ -21,7 +21,6 @@ import { QuickTimeInterval, QuickTimeIntervalTranslationMap } from '@shared/mode |
21 | 21 | @Component({ |
22 | 22 | selector: 'tb-quick-time-interval', |
23 | 23 | templateUrl: './quick-time-interval.component.html', |
24 | - styleUrls: ['./quick-time-interval.component.scss'], | |
25 | 24 | providers: [ |
26 | 25 | { |
27 | 26 | provide: NG_VALUE_ACCESSOR, | ... | ... |
... | ... | @@ -51,7 +51,7 @@ |
51 | 51 | [fxShow]="timewindowForm.get('realtime.realtimeType').value === realtimeTypes.INTERVAL" |
52 | 52 | [required]="timewindow.selectedTab === timewindowTypes.REALTIME && |
53 | 53 | timewindowForm.get('realtime.realtimeType').value === realtimeTypes.INTERVAL" |
54 | - style="padding-top: 8px; min-width: 364px"></tb-quick-time-interval> | |
54 | + style="padding-top: 8px"></tb-quick-time-interval> | |
55 | 55 | </section> |
56 | 56 | </mat-radio-button> |
57 | 57 | </mat-radio-group> |
... | ... | @@ -75,6 +75,7 @@ |
75 | 75 | <tb-timeinterval |
76 | 76 | formControlName="timewindowMs" |
77 | 77 | predefinedName="timewindow.last" |
78 | + class="history-time-input" | |
78 | 79 | [fxShow]="timewindowForm.get('history.historyType').value === historyTypes.LAST_INTERVAL" |
79 | 80 | [required]="timewindow.selectedTab === timewindowTypes.HISTORY && |
80 | 81 | timewindowForm.get('history.historyType').value === historyTypes.LAST_INTERVAL" |
... | ... | @@ -86,6 +87,7 @@ |
86 | 87 | <span translate>timewindow.time-period</span> |
87 | 88 | <tb-datetime-period |
88 | 89 | formControlName="fixedTimewindow" |
90 | + class="history-time-input" | |
89 | 91 | [fxShow]="timewindowForm.get('history.historyType').value === historyTypes.FIXED" |
90 | 92 | [required]="timewindow.selectedTab === timewindowTypes.HISTORY && |
91 | 93 | timewindowForm.get('history.historyType').value === historyTypes.FIXED" |
... | ... | @@ -97,10 +99,11 @@ |
97 | 99 | <span translate>timewindow.interval</span> |
98 | 100 | <tb-quick-time-interval |
99 | 101 | formControlName="quickInterval" |
102 | + class="history-time-input" | |
100 | 103 | [fxShow]="timewindowForm.get('history.historyType').value === historyTypes.INTERVAL" |
101 | 104 | [required]="timewindow.selectedTab === timewindowTypes.HISTORY && |
102 | 105 | timewindowForm.get('history.historyType').value === historyTypes.INTERVAL" |
103 | - style="padding-top: 8px; min-width: 364px"></tb-quick-time-interval> | |
106 | + style="padding-top: 8px"></tb-quick-time-interval> | |
104 | 107 | </section> |
105 | 108 | </mat-radio-button> |
106 | 109 | </mat-radio-group> |
... | ... | @@ -134,21 +137,23 @@ |
134 | 137 | (ngModelChange)="onHideAggIntervalChanged()"></mat-checkbox> |
135 | 138 | </section> |
136 | 139 | <section fxLayout="column" fxFlex [fxShow]="isEdit || !timewindow.hideAggInterval"> |
137 | - <div class="limit-slider-container" | |
138 | - fxLayout="row" fxLayoutAlign="start center"> | |
139 | - <span translate>aggregation.limit</span> | |
140 | - <mat-slider fxFlex formControlName="limit" | |
141 | - thumbLabel | |
142 | - [value]="timewindowForm.get('aggregation.limit').value" | |
143 | - min="{{minDatapointsLimit()}}" | |
144 | - max="{{maxDatapointsLimit()}}"> | |
145 | - </mat-slider> | |
146 | - <mat-form-field style="max-width: 80px;"> | |
147 | - <input matInput formControlName="limit" type="number" step="1" | |
148 | - [value]="timewindowForm.get('aggregation.limit').value" | |
149 | - min="{{minDatapointsLimit()}}" | |
150 | - max="{{maxDatapointsLimit()}}"/> | |
151 | - </mat-form-field> | |
140 | + <div class="limit-slider-container" fxLayout="row" fxLayoutAlign="start center" | |
141 | + fxLayout.xs="column" fxLayoutAlign.xs="stretch"> | |
142 | + <label translate>aggregation.limit</label> | |
143 | + <div fxLayout="row" fxLayoutAlign="start center" fxFlex> | |
144 | + <mat-slider fxFlex formControlName="limit" | |
145 | + thumbLabel | |
146 | + [value]="timewindowForm.get('aggregation.limit').value" | |
147 | + min="{{minDatapointsLimit()}}" | |
148 | + max="{{maxDatapointsLimit()}}"> | |
149 | + </mat-slider> | |
150 | + <mat-form-field class="limit-slider-value"> | |
151 | + <input matInput formControlName="limit" type="number" step="1" | |
152 | + [value]="timewindowForm.get('aggregation.limit').value" | |
153 | + min="{{minDatapointsLimit()}}" | |
154 | + max="{{maxDatapointsLimit()}}"/> | |
155 | + </mat-form-field> | |
156 | + </div> | |
152 | 157 | </div> |
153 | 158 | </section> |
154 | 159 | </section> | ... | ... |
... | ... | @@ -13,6 +13,8 @@ |
13 | 13 | * See the License for the specific language governing permissions and |
14 | 14 | * limitations under the License. |
15 | 15 | */ |
16 | +@import "../../../../scss/constants"; | |
17 | + | |
16 | 18 | :host { |
17 | 19 | width: 100%; |
18 | 20 | height: 100%; |
... | ... | @@ -40,21 +42,29 @@ |
40 | 42 | } |
41 | 43 | |
42 | 44 | .limit-slider-container { |
43 | - >:first-child { | |
44 | - margin-right: 16px; | |
45 | - } | |
46 | - >:last-child { | |
45 | + .limit-slider-value { | |
47 | 46 | margin-left: 16px; |
48 | - } | |
49 | - >:first-child, >:last-child { | |
50 | 47 | min-width: 25px; |
51 | - max-width: 42px; | |
48 | + max-width: 80px; | |
52 | 49 | } |
53 | 50 | mat-form-field input[type=number] { |
54 | 51 | text-align: center; |
55 | 52 | } |
56 | 53 | } |
57 | 54 | |
55 | + @media #{$mat-gt-sm} { | |
56 | + .history-time-input { | |
57 | + min-width: 364px; | |
58 | + } | |
59 | + .limit-slider-container { | |
60 | + > label { | |
61 | + margin-right: 16px; | |
62 | + width: min-content; | |
63 | + max-width: 40%; | |
64 | + } | |
65 | + } | |
66 | + } | |
67 | + | |
58 | 68 | } |
59 | 69 | |
60 | 70 | :host ::ng-deep { | ... | ... |
... | ... | @@ -141,6 +141,7 @@ import { FileSizePipe } from '@shared/pipe/file-size.pipe'; |
141 | 141 | import { WidgetsBundleSearchComponent } from '@shared/components/widgets-bundle-search.component'; |
142 | 142 | import { SelectableColumnsPipe } from '@shared/pipe/selectable-columns.pipe'; |
143 | 143 | import { QuickTimeIntervalComponent } from '@shared/components/time/quick-time-interval.component'; |
144 | +import { MAT_DATE_LOCALE } from '@angular/material/core'; | |
144 | 145 | |
145 | 146 | @NgModule({ |
146 | 147 | providers: [ |
... | ... | @@ -154,6 +155,10 @@ import { QuickTimeIntervalComponent } from '@shared/components/time/quick-time-i |
154 | 155 | { |
155 | 156 | provide: FlowInjectionToken, |
156 | 157 | useValue: Flow |
158 | + }, | |
159 | + { | |
160 | + provide: MAT_DATE_LOCALE, | |
161 | + useValue: 'en-GB' | |
157 | 162 | } |
158 | 163 | ], |
159 | 164 | declarations: [ | ... | ... |
... | ... | @@ -1943,7 +1943,7 @@ |
1943 | 1943 | "selected-rulechains": "已选择 { count, plural, 1 {# 个规则链} other {# 个规则链} }", |
1944 | 1944 | "set-root": "设置为根规则链", |
1945 | 1945 | "set-root-rulechain-text": "确认之后,规则链将变为根规格链,并将处理所有传入的传输消息。", |
1946 | - "set-root-rulechain-title": "您确定要生成规则链'{{RuleChainName}}'根吗?", | |
1946 | + "set-root-rulechain-title": "您确定要生成规则链'{{ruleChainName}}'根吗?", | |
1947 | 1947 | "system": "系统" |
1948 | 1948 | }, |
1949 | 1949 | "rulenode": { | ... | ... |