Add External Rule Node types. Rule Chain UI improvements.
This commit is contained in:
parent
5e88b02f1e
commit
e7ae82b104
@ -118,6 +118,7 @@ public class AnnotationComponentDiscoveryService implements ComponentDiscoverySe
|
||||
case FILTER:
|
||||
case TRANSFORMATION:
|
||||
case ACTION:
|
||||
case EXTERNAL:
|
||||
RuleNode ruleNodeAnnotation = clazz.getAnnotation(RuleNode.class);
|
||||
scannedComponent.setName(ruleNodeAnnotation.name());
|
||||
scannedComponent.setScope(ruleNodeAnnotation.scope());
|
||||
@ -194,6 +195,8 @@ public class AnnotationComponentDiscoveryService implements ComponentDiscoverySe
|
||||
nodeDefinition.setDefaultConfiguration(mapper.valueToTree(defaultConfiguration));
|
||||
nodeDefinition.setUiResources(nodeAnnotation.uiResources());
|
||||
nodeDefinition.setConfigDirective(nodeAnnotation.configDirective());
|
||||
nodeDefinition.setIcon(nodeAnnotation.icon());
|
||||
nodeDefinition.setIconUrl(nodeAnnotation.iconUrl());
|
||||
return nodeDefinition;
|
||||
}
|
||||
|
||||
|
||||
@ -20,6 +20,6 @@ package org.thingsboard.server.common.data.plugin;
|
||||
*/
|
||||
public enum ComponentType {
|
||||
|
||||
ENRICHMENT, FILTER, TRANSFORMATION, ACTION, OLD_ACTION, PLUGIN
|
||||
ENRICHMENT, FILTER, TRANSFORMATION, ACTION, EXTERNAL, OLD_ACTION, PLUGIN
|
||||
|
||||
}
|
||||
|
||||
@ -31,5 +31,7 @@ public class NodeDefinition {
|
||||
JsonNode defaultConfiguration;
|
||||
String[] uiResources;
|
||||
String configDirective;
|
||||
String icon;
|
||||
String iconUrl;
|
||||
|
||||
}
|
||||
|
||||
@ -49,6 +49,10 @@ public @interface RuleNode {
|
||||
|
||||
String configDirective() default "";
|
||||
|
||||
String icon() default "";
|
||||
|
||||
String iconUrl() default "";
|
||||
|
||||
boolean customRelations() default false;
|
||||
|
||||
}
|
||||
|
||||
@ -47,7 +47,9 @@ import static org.thingsboard.rule.engine.DonAsynchron.withCallback;
|
||||
"Message payload can be accessed via <code>msg</code> property. For example <code>'temperature = ' + msg.temperature ;</code>" +
|
||||
"Message metadata can be accessed via <code>metadata</code> property. For example <code>'name = ' + metadata.customerName;</code>",
|
||||
uiResources = {"static/rulenode/rulenode-core-config.js"},
|
||||
configDirective = "tbActionNodeAlarmConfig")
|
||||
configDirective = "tbActionNodeAlarmConfig",
|
||||
icon = "notifications_active"
|
||||
)
|
||||
|
||||
public class TbAlarmNode implements TbNode {
|
||||
|
||||
|
||||
@ -33,7 +33,9 @@ import static org.thingsboard.rule.engine.DonAsynchron.withCallback;
|
||||
"Message payload can be accessed via <code>msg</code> property. For example <code>'temperature = ' + msg.temperature ;</code>" +
|
||||
"Message metadata can be accessed via <code>metadata</code> property. For example <code>'name = ' + metadata.customerName;</code>",
|
||||
uiResources = {"static/rulenode/rulenode-core-config.js"},
|
||||
configDirective = "tbActionNodeLogConfig")
|
||||
configDirective = "tbActionNodeLogConfig",
|
||||
icon = "menu"
|
||||
)
|
||||
|
||||
public class TbLogNode implements TbNode {
|
||||
|
||||
|
||||
@ -38,13 +38,14 @@ import static org.thingsboard.rule.engine.DonAsynchron.withCallback;
|
||||
|
||||
@Slf4j
|
||||
@RuleNode(
|
||||
type = ComponentType.ACTION,
|
||||
type = ComponentType.EXTERNAL,
|
||||
name = "aws sns",
|
||||
configClazz = TbSnsNodeConfiguration.class,
|
||||
nodeDescription = "Publish messages to AWS SNS",
|
||||
nodeDetails = "Expects messages with any message type. Will publish message to AWS SNS topic.",
|
||||
uiResources = {"static/rulenode/rulenode-core-config.js"},
|
||||
configDirective = "tbActionNodeSnsConfig"
|
||||
configDirective = "tbActionNodeSnsConfig",
|
||||
iconUrl = ""
|
||||
)
|
||||
public class TbSnsNode implements TbNode {
|
||||
|
||||
|
||||
@ -41,13 +41,14 @@ import static org.thingsboard.rule.engine.DonAsynchron.withCallback;
|
||||
|
||||
@Slf4j
|
||||
@RuleNode(
|
||||
type = ComponentType.ACTION,
|
||||
type = ComponentType.EXTERNAL,
|
||||
name = "aws sqs",
|
||||
configClazz = TbSqsNodeConfiguration.class,
|
||||
nodeDescription = "Publish messages to AWS SQS",
|
||||
nodeDetails = "Expects messages with any message type. Will publish message to AWS SQS queue.",
|
||||
uiResources = {"static/rulenode/rulenode-core-config.js"},
|
||||
configDirective = "tbActionNodeSqsConfig"
|
||||
configDirective = "tbActionNodeSqsConfig",
|
||||
iconUrl = ""
|
||||
)
|
||||
public class TbSqsNode implements TbNode {
|
||||
|
||||
|
||||
@ -40,7 +40,8 @@ import static org.thingsboard.rule.engine.DonAsynchron.withCallback;
|
||||
nodeDetails = "Generates messages with configurable period. ",
|
||||
inEnabled = false,
|
||||
uiResources = {"static/rulenode/rulenode-core-config.js", "static/rulenode/rulenode-core-config.css"},
|
||||
configDirective = "tbActionNodeGeneratorConfig"
|
||||
configDirective = "tbActionNodeGeneratorConfig",
|
||||
icon = "repeat"
|
||||
)
|
||||
|
||||
public class TbMsgGeneratorNode implements TbNode {
|
||||
|
||||
@ -28,13 +28,14 @@ import java.util.concurrent.ExecutionException;
|
||||
|
||||
@Slf4j
|
||||
@RuleNode(
|
||||
type = ComponentType.ACTION,
|
||||
type = ComponentType.EXTERNAL,
|
||||
name = "kafka",
|
||||
configClazz = TbKafkaNodeConfiguration.class,
|
||||
nodeDescription = "Publish messages to Kafka server",
|
||||
nodeDetails = "Expects messages with any message type. Will send record via Kafka producer to Kafka server.",
|
||||
uiResources = {"static/rulenode/rulenode-core-config.js"},
|
||||
configDirective = "tbActionNodeKafkaConfig"
|
||||
configDirective = "tbActionNodeKafkaConfig",
|
||||
iconUrl = ""
|
||||
)
|
||||
public class TbKafkaNode implements TbNode {
|
||||
|
||||
|
||||
@ -38,7 +38,9 @@ import static org.thingsboard.rule.engine.mail.TbSendEmailNode.SEND_EMAIL_TYPE;
|
||||
nodeDetails = "Related Entity found using configured relation direction and Relation Type. " +
|
||||
"If multiple Related Entities are found, only first Entity is used as new Originator, other entities are discarded. ",
|
||||
uiResources = {"static/rulenode/rulenode-core-config.js"},
|
||||
configDirective = "tbTransformationNodeToEmailConfig")
|
||||
configDirective = "tbTransformationNodeToEmailConfig",
|
||||
icon = "email"
|
||||
)
|
||||
public class TbMsgToEmailNode implements TbNode {
|
||||
|
||||
private static final ObjectMapper MAPPER = new ObjectMapper();
|
||||
|
||||
@ -35,7 +35,7 @@ import static org.thingsboard.rule.engine.DonAsynchron.withCallback;
|
||||
|
||||
@Slf4j
|
||||
@RuleNode(
|
||||
type = ComponentType.ACTION,
|
||||
type = ComponentType.EXTERNAL,
|
||||
name = "send email",
|
||||
configClazz = TbSendEmailNodeConfiguration.class,
|
||||
nodeDescription = "Log incoming messages using JS script for transformation Message into String",
|
||||
@ -43,7 +43,8 @@ import static org.thingsboard.rule.engine.DonAsynchron.withCallback;
|
||||
"Message payload can be accessed via <code>msg</code> property. For example <code>'temperature = ' + msg.temperature ;</code>" +
|
||||
"Message metadata can be accessed via <code>metadata</code> property. For example <code>'name = ' + metadata.customerName;</code>",
|
||||
uiResources = {"static/rulenode/rulenode-core-config.js"},
|
||||
configDirective = "tbActionNodeSendEmailConfig"
|
||||
configDirective = "tbActionNodeSendEmailConfig",
|
||||
icon = "send"
|
||||
)
|
||||
public class TbSendEmailNode implements TbNode {
|
||||
|
||||
|
||||
@ -43,13 +43,14 @@ import java.util.concurrent.TimeoutException;
|
||||
|
||||
@Slf4j
|
||||
@RuleNode(
|
||||
type = ComponentType.ACTION,
|
||||
type = ComponentType.EXTERNAL,
|
||||
name = "mqtt",
|
||||
configClazz = TbMqttNodeConfiguration.class,
|
||||
nodeDescription = "Publish messages to MQTT broker",
|
||||
nodeDetails = "Expects messages with any message type. Will publish message to MQTT broker.",
|
||||
uiResources = {"static/rulenode/rulenode-core-config.js", "static/rulenode/rulenode-core-config.css"},
|
||||
configDirective = "tbActionNodeMqttConfig"
|
||||
configDirective = "tbActionNodeMqttConfig",
|
||||
icon = "call_split"
|
||||
)
|
||||
public class TbMqttNode implements TbNode {
|
||||
|
||||
|
||||
@ -33,13 +33,14 @@ import static org.thingsboard.rule.engine.DonAsynchron.withCallback;
|
||||
|
||||
@Slf4j
|
||||
@RuleNode(
|
||||
type = ComponentType.ACTION,
|
||||
type = ComponentType.EXTERNAL,
|
||||
name = "rabbitmq",
|
||||
configClazz = TbRabbitMqNodeConfiguration.class,
|
||||
nodeDescription = "Publish messages to RabbitMQ",
|
||||
nodeDetails = "Expects messages with any message type. Will publish message to RabbitMQ queue.",
|
||||
uiResources = {"static/rulenode/rulenode-core-config.js"},
|
||||
configDirective = "tbActionNodeRabbitMqConfig"
|
||||
configDirective = "tbActionNodeRabbitMqConfig",
|
||||
iconUrl = ""
|
||||
)
|
||||
public class TbRabbitMqNode implements TbNode {
|
||||
|
||||
|
||||
@ -26,7 +26,8 @@ import org.springframework.http.ResponseEntity;
|
||||
import org.springframework.http.client.Netty4ClientHttpRequestFactory;
|
||||
import org.springframework.util.concurrent.ListenableFuture;
|
||||
import org.springframework.util.concurrent.ListenableFutureCallback;
|
||||
import org.springframework.web.client.*;
|
||||
import org.springframework.web.client.AsyncRestTemplate;
|
||||
import org.springframework.web.client.HttpClientErrorException;
|
||||
import org.thingsboard.rule.engine.TbNodeUtils;
|
||||
import org.thingsboard.rule.engine.api.*;
|
||||
import org.thingsboard.server.common.data.plugin.ComponentType;
|
||||
@ -34,19 +35,19 @@ import org.thingsboard.server.common.msg.TbMsg;
|
||||
import org.thingsboard.server.common.msg.TbMsgMetaData;
|
||||
|
||||
import javax.net.ssl.SSLException;
|
||||
import java.util.Map;
|
||||
import java.util.concurrent.ExecutionException;
|
||||
import java.util.concurrent.TimeUnit;
|
||||
|
||||
@Slf4j
|
||||
@RuleNode(
|
||||
type = ComponentType.ACTION,
|
||||
type = ComponentType.EXTERNAL,
|
||||
name = "rest api call",
|
||||
configClazz = TbRestApiCallNodeConfiguration.class,
|
||||
nodeDescription = "Invoke REST API calls to external REST server",
|
||||
nodeDetails = "Expects messages with any message type. Will invoke REST API call to external REST server.",
|
||||
uiResources = {"static/rulenode/rulenode-core-config.js"},
|
||||
configDirective = "tbActionNodeRestApiCallConfig"
|
||||
configDirective = "tbActionNodeRestApiCallConfig",
|
||||
iconUrl = ""
|
||||
)
|
||||
public class TbRestApiCallNode implements TbNode {
|
||||
|
||||
|
||||
@ -36,7 +36,8 @@ import org.thingsboard.server.common.msg.TbMsg;
|
||||
nodeDescription = "Sends reply to the RPC call from device",
|
||||
nodeDetails = "Expects messages with any message type. Will forward message body to the device.",
|
||||
uiResources = {"static/rulenode/rulenode-core-config.js"},
|
||||
configDirective = "tbActionNodeRpcReplyConfig"
|
||||
configDirective = "tbActionNodeRpcReplyConfig",
|
||||
icon = "call_merge"
|
||||
)
|
||||
public class TbSendRPCReplyNode implements TbNode {
|
||||
|
||||
|
||||
@ -43,7 +43,8 @@ import java.util.concurrent.TimeUnit;
|
||||
nodeDescription = "Sends one-way RPC call to device",
|
||||
nodeDetails = "Expects messages with \"method\" and \"params\". Will forward response from device to next nodes.",
|
||||
uiResources = {"static/rulenode/rulenode-core-config.js"},
|
||||
configDirective = "tbActionNodeRpcRequestConfig"
|
||||
configDirective = "tbActionNodeRpcRequestConfig",
|
||||
icon = "call_made"
|
||||
)
|
||||
public class TbSendRPCRequestNode implements TbNode {
|
||||
|
||||
|
||||
@ -48,7 +48,8 @@ import java.util.Set;
|
||||
nodeDescription = "Saves attributes data",
|
||||
nodeDetails = "Saves entity attributes based on configurable scope parameter. Expects messages with 'POST_ATTRIBUTES_REQUEST' message type",
|
||||
uiResources = {"static/rulenode/rulenode-core-config.js", "static/rulenode/rulenode-core-config.css"},
|
||||
configDirective = "tbActionNodeAttributesConfig"
|
||||
configDirective = "tbActionNodeAttributesConfig",
|
||||
icon = "file_upload"
|
||||
)
|
||||
public class TbMsgAttributesNode implements TbNode {
|
||||
|
||||
|
||||
@ -45,7 +45,8 @@ import java.util.Map;
|
||||
nodeDescription = "Saves timeseries data",
|
||||
nodeDetails = "Saves timeseries telemetry data based on configurable TTL parameter. Expects messages with 'POST_TELEMETRY_REQUEST' message type",
|
||||
uiResources = {"static/rulenode/rulenode-core-config.js", "static/rulenode/rulenode-core-config.css"},
|
||||
configDirective = "tbActionNodeTimeseriesConfig"
|
||||
configDirective = "tbActionNodeTimeseriesConfig",
|
||||
icon = "file_upload"
|
||||
)
|
||||
public class TbMsgTimeseriesNode implements TbNode {
|
||||
|
||||
|
||||
@ -40,7 +40,9 @@ import java.util.HashSet;
|
||||
nodeDetails = "Related Entity found using configured relation direction and Relation Type. " +
|
||||
"If multiple Related Entities are found, only first Entity is used as new Originator, other entities are discarded. ",
|
||||
uiResources = {"static/rulenode/rulenode-core-config.js", "static/rulenode/rulenode-core-config.css"},
|
||||
configDirective = "tbTransformationNodeChangeOriginatorConfig")
|
||||
configDirective = "tbTransformationNodeChangeOriginatorConfig",
|
||||
icon = "find_replace"
|
||||
)
|
||||
public class TbChangeOriginatorNode extends TbAbstractTransformNode {
|
||||
|
||||
protected static final String CUSTOMER_SOURCE = "CUSTOMER";
|
||||
|
||||
File diff suppressed because one or more lines are too long
@ -485,7 +485,7 @@ export default angular.module('thingsboard.types', [])
|
||||
clientSide: false
|
||||
}
|
||||
},
|
||||
ruleNodeTypeComponentTypes: ["FILTER", "ENRICHMENT", "TRANSFORMATION", "ACTION"],
|
||||
ruleNodeTypeComponentTypes: ["FILTER", "ENRICHMENT", "TRANSFORMATION", "ACTION", "EXTERNAL"],
|
||||
ruleChainNodeComponent: {
|
||||
type: 'RULE_CHAIN',
|
||||
name: 'rule chain',
|
||||
@ -536,6 +536,13 @@ export default angular.module('thingsboard.types', [])
|
||||
nodeClass: "tb-action-type",
|
||||
icon: "flash_on"
|
||||
},
|
||||
EXTERNAL: {
|
||||
value: "EXTERNAL",
|
||||
name: "rulenode.type-external",
|
||||
details: "rulenode.type-external-details",
|
||||
nodeClass: "tb-external-type",
|
||||
icon: "cloud_upload"
|
||||
},
|
||||
RULE_CHAIN: {
|
||||
value: "RULE_CHAIN",
|
||||
name: "rulenode.type-rule-chain",
|
||||
|
||||
@ -1228,6 +1228,8 @@ export default angular.module('thingsboard.locale', [])
|
||||
"type-transformation-details": "Change Message payload and Metadata",
|
||||
"type-action": "Action",
|
||||
"type-action-details": "Perform special action",
|
||||
"type-external": "External",
|
||||
"type-external-details": "Interacts with external system",
|
||||
"type-rule-chain": "Rule Chain",
|
||||
"type-rule-chain-details": "Forwards incoming messages to specified Rule Chain",
|
||||
"directive-is-not-loaded": "Defined configuration directive '{{directiveName}}' is not available.",
|
||||
|
||||
@ -246,6 +246,7 @@ export function RuleChainController($state, $scope, $compile, $q, $mdUtil, $time
|
||||
var contextInfo = {
|
||||
headerClass: node.nodeClass,
|
||||
icon: node.icon,
|
||||
iconUrl: node.iconUrl,
|
||||
title: node.name,
|
||||
subtitle: node.component.name
|
||||
};
|
||||
@ -805,12 +806,21 @@ export function RuleChainController($state, $scope, $compile, $q, $mdUtil, $time
|
||||
var ruleNodeComponent = ruleNodeComponents[i];
|
||||
componentType = ruleNodeComponent.type;
|
||||
var model = vm.ruleNodeTypesModel[componentType].model;
|
||||
var icon = vm.types.ruleNodeType[componentType].icon;
|
||||
var iconUrl = null;
|
||||
if (ruleNodeComponent.configurationDescriptor.nodeDefinition.icon) {
|
||||
icon = ruleNodeComponent.configurationDescriptor.nodeDefinition.icon;
|
||||
}
|
||||
if (ruleNodeComponent.configurationDescriptor.nodeDefinition.iconUrl) {
|
||||
iconUrl = ruleNodeComponent.configurationDescriptor.nodeDefinition.iconUrl;
|
||||
}
|
||||
var node = {
|
||||
id: 'node-lib-' + componentType + '-' + model.nodes.length,
|
||||
component: ruleNodeComponent,
|
||||
name: '',
|
||||
nodeClass: vm.types.ruleNodeType[componentType].nodeClass,
|
||||
icon: vm.types.ruleNodeType[componentType].icon,
|
||||
icon: icon,
|
||||
iconUrl: iconUrl,
|
||||
x: 30,
|
||||
y: 10+50*model.nodes.length,
|
||||
connectors: []
|
||||
@ -904,6 +914,14 @@ export function RuleChainController($state, $scope, $compile, $q, $mdUtil, $time
|
||||
var ruleNode = vm.ruleChainMetaData.nodes[i];
|
||||
var component = ruleChainService.getRuleNodeComponentByClazz(ruleNode.type);
|
||||
if (component) {
|
||||
var icon = vm.types.ruleNodeType[component.type].icon;
|
||||
var iconUrl = null;
|
||||
if (component.configurationDescriptor.nodeDefinition.icon) {
|
||||
icon = component.configurationDescriptor.nodeDefinition.icon;
|
||||
}
|
||||
if (component.configurationDescriptor.nodeDefinition.iconUrl) {
|
||||
iconUrl = component.configurationDescriptor.nodeDefinition.iconUrl;
|
||||
}
|
||||
var node = {
|
||||
id: 'rule-chain-node-' + vm.nextNodeID++,
|
||||
ruleNodeId: ruleNode.id,
|
||||
@ -915,7 +933,8 @@ export function RuleChainController($state, $scope, $compile, $q, $mdUtil, $time
|
||||
component: component,
|
||||
name: ruleNode.name,
|
||||
nodeClass: vm.types.ruleNodeType[component.type].nodeClass,
|
||||
icon: vm.types.ruleNodeType[component.type].icon,
|
||||
icon: icon,
|
||||
iconUrl: iconUrl,
|
||||
connectors: []
|
||||
};
|
||||
if (component.configurationDescriptor.nodeDefinition.inEnabled) {
|
||||
|
||||
@ -76,7 +76,7 @@
|
||||
-moz-user-select: none;
|
||||
-ms-user-select: none;
|
||||
user-select: none;
|
||||
min-width: 180px;
|
||||
min-width: 150px;
|
||||
}
|
||||
.fc-canvas {
|
||||
background: #f9f9f9;
|
||||
@ -161,6 +161,9 @@
|
||||
&.tb-action-type {
|
||||
background-color: #f1928f;
|
||||
}
|
||||
&.tb-external-type {
|
||||
background-color: #fbc766;
|
||||
}
|
||||
&.tb-rule-chain-type {
|
||||
background-color: #d6c4f1;
|
||||
}
|
||||
|
||||
@ -75,6 +75,8 @@
|
||||
<md-expansion-panel md-component-id="{{typeId}}" id="{{typeId}}" ng-repeat="(typeId, typeModel) in vm.ruleNodeTypesModel">
|
||||
<md-expansion-panel-collapsed ng-mouseenter="vm.typeHeaderMouseEnter($event, typeId)"
|
||||
ng-mouseleave="vm.destroyTooltips()">
|
||||
<md-icon aria-label="node-type-icon"
|
||||
class="material-icons" style="margin-right: 8px;">{{vm.types.ruleNodeType[typeId].icon}}</md-icon>
|
||||
<div class="tb-panel-title" translate>{{vm.types.ruleNodeType[typeId].name}}</div>
|
||||
<md-expansion-panel-icon></md-expansion-panel-icon>
|
||||
</md-expansion-panel-collapsed>
|
||||
@ -82,6 +84,8 @@
|
||||
<md-expansion-panel-header ng-mouseenter="vm.typeHeaderMouseEnter($event, typeId)"
|
||||
ng-mouseleave="vm.destroyTooltips()"
|
||||
ng-click="vm.$mdExpansionPanel(typeId).collapse()">
|
||||
<md-icon aria-label="node-type-icon"
|
||||
class="material-icons" style="margin-right: 8px;">{{vm.types.ruleNodeType[typeId].icon}}</md-icon>
|
||||
<div class="tb-panel-title" translate>{{vm.types.ruleNodeType[typeId].name}}</div>
|
||||
<md-expansion-panel-icon></md-expansion-panel-icon>
|
||||
</md-expansion-panel-header>
|
||||
@ -114,8 +118,10 @@
|
||||
</div>
|
||||
<md-menu-content id="tb-rule-chain-context-menu" width="4" ng-mouseleave="$mdCloseMousepointMenu()">
|
||||
<div class="tb-context-menu-header {{vm.contextInfo.headerClass}}">
|
||||
<md-icon aria-label="node-type-icon"
|
||||
<md-icon ng-if="!vm.contextInfo.iconUrl" aria-label="node-type-icon"
|
||||
class="material-icons">{{vm.contextInfo.icon}}</md-icon>
|
||||
<md-icon ng-if="vm.contextInfo.iconUrl" aria-label="node-type-icon"
|
||||
md-svg-icon="{{vm.contextInfo.iconUrl}}"></md-icon>
|
||||
<div flex>
|
||||
<div class="tb-context-menu-title">{{vm.contextInfo.title}}</div>
|
||||
<div class="tb-context-menu-subtitle">{{vm.contextInfo.subtitle}}</div>
|
||||
|
||||
@ -24,8 +24,10 @@
|
||||
ng-mouseleave="callbacks.mouseLeave($event, node)">
|
||||
<div class="{{flowchartConstants.nodeOverlayClass}}"></div>
|
||||
<div class="tb-rule-node {{node.nodeClass}}" ng-class="{'tb-rule-node-highlighted' : node.highlighted, 'tb-rule-node-invalid': node.error }">
|
||||
<md-icon aria-label="node-type-icon" flex="15"
|
||||
<md-icon ng-if="!node.iconUrl" aria-label="node-type-icon" flex="15"
|
||||
class="material-icons">{{node.icon}}</md-icon>
|
||||
<md-icon ng-if="node.iconUrl" aria-label="node-type-icon" flex="15"
|
||||
md-svg-icon="{{node.iconUrl}}"></md-icon>
|
||||
<div layout="column" flex="85" layout-align="center">
|
||||
<span class="tb-node-type">{{ node.component.name }}</span>
|
||||
<span class="tb-node-title" ng-if="node.name">{{ node.name }}</span>
|
||||
|
||||
Loading…
x
Reference in New Issue
Block a user