Blazor Diagram Shapes
The Shape is the main building block of the Telerik Diagram for Blazor. It represents a single node in the graph. This article describes all Diagram Shape types and customization options.
Basics
The fundamental settings of the Telerik Diagram Shapes include:
- The Shape
Type
determines the overall Shape appearance. - The Shape
Id
is required to define connections between related Shapes. Text
defines the Shape label andTextWrap
defines the label wrapping behavior. Set these parameters to the child<DiagramShapeContent>
tag.Width
andHeight
determine the Shape size in pixels. The default values are100
.
Using basic Shape parameters
<DiagramShape Id="shape1"
Type="@DiagramShapeType.Rectangle"
Width="200"
Height="50">
<DiagramShapeContent Text="Shape 1" />
</DiagramShape>
In addition to the above, you can use the following Shape parameters:
X
andY
to define the exact Shape position coordinates. These parameters have effect only when a predefined Diagram layout is not set.DataItem
to provide an object with additional values to be used in a visual function.Path
to define a custom Shape form.
Shape Types
The available Diagram Shape types are the members of the DiagramShapeType
enum. The default Shape type is Rectangle
.
Some Shape types are designed for flowcharts, also known as workflow or process diagrams. However, all Shape types can be used in any scenario.
The Shape Path
parameter allows you to manually define a custom Shape form with multiple straight or curved lines that doesn't match any of the predefined Shape types.
All Diagram Shape types except Image and Text
<TelerikDiagram Height="440px" Zoom="0.5">
<DiagramShapeDefaults Width="150" Height="150">
</DiagramShapeDefaults>
<DiagramShapes>
@foreach (DiagramShapeType shapeType in AllDiagramShapeTypes)
{
<DiagramShape Type="@shapeType"
Id="@( $"id-{shapeType}" )"
X="@GetShapeX(shapeType)"
Y="@GetShapeY(shapeType)"
Width="@GetShapeWidth(shapeType)">
<DiagramShapeContent Text="@shapeType.ToString()"
Color="@GetShapeContentColor(shapeType)" />
</DiagramShape>
}
</DiagramShapes>
</TelerikDiagram>
@code {
private readonly List<DiagramShapeType> AllDiagramShapeTypes = new() {
DiagramShapeType.Circle,
DiagramShapeType.Collate,
DiagramShapeType.Database,
DiagramShapeType.DataInputOutput,
DiagramShapeType.DataStorage,
DiagramShapeType.Decision,
DiagramShapeType.Delay,
DiagramShapeType.DirectAccessStorage,
DiagramShapeType.Display,
DiagramShapeType.Document,
DiagramShapeType.Extract,
//DiagramShapeType.Image,
DiagramShapeType.InternalStorage,
DiagramShapeType.LogicalOr,
DiagramShapeType.ManualInputOutput,
DiagramShapeType.ManualOperation,
DiagramShapeType.Merge,
DiagramShapeType.MultipleDocuments,
DiagramShapeType.OffPageConnector,
DiagramShapeType.OnPageConnector,
DiagramShapeType.PredefinedProcess,
DiagramShapeType.Preparation,
DiagramShapeType.Process,
DiagramShapeType.Rectangle,
DiagramShapeType.Sort,
DiagramShapeType.SummingJunction,
//DiagramShapeType.Text,
DiagramShapeType.Terminator
};
private double GetShapeX(DiagramShapeType shapeType)
{
return AllDiagramShapeTypes.IndexOf(shapeType) % 7 * 200;
}
private double GetShapeY(DiagramShapeType shapeType)
{
return AllDiagramShapeTypes.IndexOf(shapeType) / 7 * 200;
}
public string GetShapeContentColor(DiagramShapeType shapeType)
{
List<DiagramShapeType> shapesWithBlackColor = new() {
DiagramShapeType.Collate, DiagramShapeType.Sort, DiagramShapeType.Text
};
return shapesWithBlackColor.Contains(shapeType) ? "#000" : string.Empty;
}
public int GetShapeWidth(DiagramShapeType shapeType)
{
return shapeType == DiagramShapeType.Terminator ? 300 : 150;
}
}
Type-Specific Shape Features
Some Shape types provide unique behavior or settings:
- The
Circle
Shape can look like an ellipse if you set differentWidth
andHeight
values. Generally, all Shapes adjust their form and proportions, based on the set dimensions. - All Shape types support text labels, but only the
Image
Shape can display a graphic. Use the<DiagramShape>
Source
parameter to set an image URL or a data URI.RAZOR<DiagramShape Type="@DiagramShapeType.Image" Source="https://www.domain.com/image.gif" /> <DiagramShape Type="@DiagramShapeType.Image" Source="data:image/...;base64,....." />
- The
Terminator
Shape normally requires aWidth
that is larger than theHeight
. - The
Text
Shape has no borders and background. It occupies the minimum required amount of space to enclose the text content. To display text Shapes with some empty space around the content, use transparent Shapes of another type. - All shapes, except
Circle
,Image
,Rectangle
, andText
can display with rounded corners. SeeCornerRadius
in the Styling section.
Using transparent Rectangle shapes instead of Text shapes
<TelerikDiagram>
<DiagramLayout Type="@DiagramLayoutType.Tree" />
<DiagramShapeDefaults Type="@DiagramShapeType.Rectangle"
Height="50">
<DiagramShapeDefaultsContent Color="#000" />
<DiagramShapeDefaultsFill Color="transparent" />
<DiagramShapeDefaultsStroke Width="0" />
</DiagramShapeDefaults>
<DiagramShapes>
<DiagramShape Id="shape1">
<DiagramShapeContent Text="Shape 1" />
</DiagramShape>
<DiagramShape Id="shape2">
<DiagramShapeContent Text="Shape 2" />
</DiagramShape>
<DiagramShape Id="shape3">
<DiagramShapeContent Text="Shape 3" />
</DiagramShape>
</DiagramShapes>
<DiagramConnections>
<DiagramConnection FromId="shape1" ToId="shape2" />
<DiagramConnection FromId="shape1" ToId="shape3" />
</DiagramConnections>
</TelerikDiagram>
Connectors
Connectors are the 5 dots that appear on the Shape boundaries and center on hover. Users can grab a connector and drag it to another shape to create a new connection. If the user grabs the center connector, the Diagram can create a connection from any side of the Shape. If the user grabs a connector on the Shape's boundary, the Diagram will create a connection from that specific side of the Shape.
You can customize connectors globally or per shape. Connectors settings are part of the shape settings. As a result:
<DiagramShapeDefaultsConnectorDefaults>
must be a child of<DiagramShapeDefaults>
.<DiagramShapeConnectorDefaults>
must be a child of<DiagramShape>
.
Configure connectors globally and per Shape
<TelerikDiagram>
<DiagramShapeDefaults>
<DiagramShapeDefaultsConnectorDefaults Width="20" Height="20">
<DiagramShapeDefaultsConnectorDefaultsFill Color="yellow" Opacity="0.8" />
<DiagramShapeDefaultsConnectorDefaultsStroke Color="blue" Width="3" DashType="@DashType.Dash" />
<DiagramShapeDefaultsConnectorDefaultsHover>
<DiagramShapeDefaultsConnectorDefaultsHoverFill Color="orange" Opacity="1" />
</DiagramShapeDefaultsConnectorDefaultsHover>
</DiagramShapeDefaultsConnectorDefaults>
</DiagramShapeDefaults>
<DiagramShapes>
<DiagramShape>
<DiagramShapeConnectorDefaults Width="15" Height="15">
<DiagramShapeConnectorDefaultsFill Color="lime" Opacity="0.6" />
<DiagramShapeConnectorDefaultsStroke Color="green" Width="3" DashType="@DashType.Dot" />
<DiagramShapeConnectorDefaultsHover>
<DiagramShapeConnectorDefaultsHoverFill Color="blue" Opacity="0.9" />
</DiagramShapeConnectorDefaultsHover>
</DiagramShapeConnectorDefaults>
</DiagramShape>
</DiagramShapes>
</TelerikDiagram>
Styling
The following Shape styling options are available in child tags of <DiagramShapeDefaults>
and <DiagramShape>
:
- Text color and font properties
- Background color (fill) and opacity for the default and hover states
- Rotation angle
- Border (stroke) color, type, width, and opacity
In addition to the above:
<DiagramShapeDefaults>
and<DiagramShape>
have aCornerRadius
parameter that rounds both the border and the background at the Shape corners.<DiagramShapeDefaultsContent>
and<DiagramShapeContent>
have aRelativePadding
parameter that adds padding as a ratio of the Shape width. For example,RelativePadding="0.1"
applies a 10% padding. In some cases, you can increase the padding to force the Shape text to wrap.
Setting global and Shape-specific color styles
<TelerikDiagram>
<DiagramShapeDefaults CornerRadius="4">
<DiagramShapeDefaultsContent Color="white" FontSize="16" RelativePadding="0.1" />
<DiagramShapeDefaultsFill Color="purple" />
<DiagramShapeDefaultsHover>
<DiagramShapeDefaultsHoverFill Color="blue" />
</DiagramShapeDefaultsHover>
</DiagramShapeDefaults>
<DiagramShapes>
<DiagramShape CornerRadius="6">
<DiagramShapeContent Color="#3d3d3d" FontSize="24" RelativePadding="0.2" />
<DiagramShapeFill Color="#e0e0e0" />
<DiagramShapeHover>
<DiagramShapeHoverFill Color="#d6d6d6" />
</DiagramShapeHover>
</DiagramShape>
</DiagramShapes>
</TelerikDiagram>
Editability
By default, the Diagram allows users to:
- Connect one Shape to other Shapes.
- Drag a Shape to new coordinates.
- Remove the selected Shape(s).
To restrict these operations globally for all Shapes, use the parameters of the <DiagramShapeDefaultsEditable>
tag.
To restrict or enable operations for a specific Shape, use the parameters of the <DiagramShapeEditable>
tag.
Setting global and Shape-specific editing options
<TelerikDiagram>
<DiagramShapeDefaults>
<DiagramShapeDefaultsEditable Connect="true" Drag="false" Remove="false" />
</DiagramShapeDefaults>
<DiagramShapes>
<DiagramShape>
<DiagramShapeEditable Connect="false" />
</DiagramShape>
</DiagramShapes>
</TelerikDiagram>
Example
The following configuration is not using a prefefined Diagram layout. However, you can remove all Shape X
and Y
parameters and set a layout though the <DiagramLayout>
tag.
Customize Diagram Shapes
<TelerikDiagram Height="300px">
<DiagramShapeDefaults Selectable="true"
Type="@DiagramShapeType.Rectangle">
<DiagramShapeDefaultsConnectorDefaults Width="10" Height="10">
<DiagramShapeDefaultsConnectorDefaultsFill Color="lime" Opacity="0.8" />
<DiagramShapeDefaultsConnectorDefaultsStroke Color="green" Width="3" DashType="@DashType.Solid" />
<DiagramShapeDefaultsConnectorDefaultsHover>
<DiagramShapeDefaultsConnectorDefaultsHoverFill Color="orange" Opacity="1" />
</DiagramShapeDefaultsConnectorDefaultsHover>
</DiagramShapeDefaultsConnectorDefaults>
<DiagramShapeDefaultsContent Color="white"
FontFamily="arial"
FontSize="16"
FontStyle="normal"
FontWeight="normal"
Text="Default Text"
TextWrap="@DiagramShapesContentTextWrap.Wrap" />
<DiagramShapeDefaultsFill Color="purple" Opacity="0.8" />
<DiagramShapeDefaultsHover>
<DiagramShapeDefaultsHoverFill Color="blue" Opacity="1" />
</DiagramShapeDefaultsHover>
<DiagramShapeDefaultsEditable Connect="true" Drag="true" Remove="false" />
<DiagramShapeDefaultsRotation Angle="0" />
<DiagramShapeDefaultsStroke Color="black" DashType="@DashType.Dot" Width="2" />
</DiagramShapeDefaults>
<DiagramShapes>
<DiagramShape Height="150"
Id="shape1"
Type="@DiagramShapeType.Circle"
Width="100"
X="160"
Y="20">
<DiagramShapeContent Text="Circle" />
<DiagramShapeEditable Enabled="false" Connect="false" />
</DiagramShape>
<DiagramShape Id="shape2"
Source="@Base64SvgImage"
Type="@DiagramShapeType.Image"
X="20"
Y="50">
<DiagramShapeConnectorDefaults Width="15" Height="15">
<DiagramShapeConnectorDefaultsFill Color="orange" />
<DiagramShapeConnectorDefaultsStroke Color="blue" />
<DiagramShapeConnectorDefaultsHover>
<DiagramShapeConnectorDefaultsHoverFill Color="purple" />
</DiagramShapeConnectorDefaultsHover>
</DiagramShapeConnectorDefaults>
<DiagramShapeContent Text="Image" Color="#000" FontSize="20" FontWeight="bold" />
</DiagramShape>
<DiagramShape CornerRadius="6"
Height="80"
Id="shape3"
Width="160"
X="350"
Y="50">
<DiagramShapeContent Color="#3d3d3d"
FontSize="18"
FontStyle="italic"
FontWeight="bold"
RelativePadding="0.1"
Text="Rounded Rectangle" />
<DiagramShapeFill Color="#e0e0e0" />
<DiagramShapeHover>
<DiagramShapeHoverFill Color="#d6d6d6" />
</DiagramShapeHover>
<DiagramShapeRotation Angle="20" />
<DiagramShapeStroke DashType="@DashType.LongDash" Color="blue" Width="2" />
</DiagramShape>
<DiagramShape Id="shape4"
Type="@DiagramShapeType.Text"
MinHeight="40"
MinWidth="100"
X="150"
Y="250">
<DiagramShapeContent Text="Text Shape" Color="black" />
</DiagramShape>
<DiagramShape Height="75"
Id="shape5"
Path="M 2 13 A 1.42 1.42 0 0 1 6 13"
Width="150"
X="250"
Y="150">
<DiagramShapeContent Text="Custom Path" />
<DiagramShapeStroke Color="transparent" DashType="@DashType.Solid" Width="0" />
</DiagramShape>
<DiagramShape Height="50"
Id="shape6"
Width="120"
X="550"
Y="100">
<DiagramShapeContent Color="#ccc" />
</DiagramShape>
</DiagramShapes>
<DiagramConnections>
<DiagramConnection FromId="shape1" ToId="shape2" />
<DiagramConnection FromId="shape1" ToId="shape3" />
<DiagramConnection FromId="shape1" ToId="shape4" />
<DiagramConnection FromId="shape3" ToId="shape5" />
<DiagramConnection FromId="shape3" ToId="shape6" />
</DiagramConnections>
</TelerikDiagram>
@code {
private readonly string Base64SvgImage = "";
}
Visual Function
You can draw Shapes and display their content by using the API of the Diagram's JavaScript rendering engine. This is an advanced scenario that is recommended if the desired result cannot be achieved in another way.
The visual function allows a single Shape to render:
- Multiple pieces of data with different styles and positions. Without a visual function, each Shape can display one image and one text label.
- Multiple ovals, polygons, and lines. Without a visual function, each Shape can have a predefined form, or display a custom form through the
Path
parameter.
To use a visual function:
- Get familiar with the related JavaScript API and available visual primitives.
- Implement a JavaScript function that returns a
TelerikBlazor.DiagramCommon.Group
JavaScript object. TheGroup
can contain any number of other primitives likeCircle
,Image
,Line
,Rectangle
,TextBlock
, and others. - Set the
Visual
parameter of<DiagramShapeDefaults>
or<DiagramShape>
to the JavaScript function name. This will either affect all Shapes or a specific Shape. - Position each primitive with the
x
andy
properties of its JavaScript object. Otherwise the primitive renders at the top-left corner of theGroup
. - To align or center primitives automatically, use a
Layout
primitive as a parent container. Make sure toreflow()
theLayout
object after adding children. - Each new primitive element displays on top of the previous ones.
- (optional) Retrieve the Shape parameter values from the the function argument. It is a JavaScript object that contains all Shape settings, including the global ones in
<DiagramShapeDefaults>
. - (optional) Set the Shape
DataItem
parameter to a JSON-serializable object. Retrieve the object property values from thedataItem
property of the function argument.
When using a visual function, the Diagram ignores all Shape parameters in the Razor markup, but you can still consume these settings from the visual function argument. The only exception is the Text
parameter of <DiagramShapeContent />
and <DiagramShapeDefaultsContent />
, which is always rendered if set.
This section links to the documentation of Kendo UI for jQuery. The Telerik Diagram for Blazor is not a wrapper of the Kendo UI Diagram. However, both components use the same client-side rendering engine. When the Kendo UI documentation mentions the
kendo.dataviz.diagram
JavaScript namespace, you must useTelerikBlazor.DiagramCommon
instead.
In addition to the following example, also check this Blazor Diagram demo, which uses a visual function.
Using Diagram Shape visual function
<TelerikDiagram>
<DiagramLayout Type="@DiagramLayoutType.Tree" />
<DiagramShapeDefaults Width="240" Height="120" Visual="shapeVisualFunction">
<DiagramShapeDefaultsContent Color="white" FontSize="20" FontWeight="bold" />
<DiagramShapeDefaultsStroke Width="3" />
</DiagramShapeDefaults>
<DiagramShapes>
<DiagramShape Id="shape1" DataItem="@ShapeDataItem1">
<DiagramShapeStroke Color="brown" />
</DiagramShape>
<DiagramShape Id="shape2" DataItem="@ShapeDataItem2">
<DiagramShapeStroke Color="purple" />
</DiagramShape>
</DiagramShapes>
<DiagramConnections>
<DiagramConnection FromId="shape1" ToId="shape2" />
</DiagramConnections>
</TelerikDiagram>
@* Move JavaScript code to an external JS file *@
<script suppress-error="BL9992">
function shapeVisualFunction(context) {
let diagramNS = TelerikBlazor.DiagramCommon;
let shapeGroup = new diagramNS.Group({
autoSize: true
});
let outerCircle = new diagramNS.Circle({
width: context.width,
height: context.height,
fill: "orange",
stroke: {
color: context.stroke.color,
width: context.stroke.width
}
});
shapeGroup.append(outerCircle);
let innerCirleSpacing = 20;
let innerCirle = new diagramNS.Circle({
width: context.width - innerCirleSpacing * 2,
height: context.height - innerCirleSpacing * 2,
fill: "red",
x: innerCirleSpacing,
y: innerCirleSpacing
});
shapeGroup.append(innerCirle);
let textRect = new diagramNS.Rect(0, 0, context.width, context.height);
let textLayout = new diagramNS.Layout(textRect, {
alignContent: "center",
alignItems: "center",
justifyContent: "center",
orientation: "vertical",
spacing: 4
});
shapeGroup.append(textLayout);
let title = new diagramNS.TextBlock({
text: context.dataItem.Title,
fontSize: context.content.fontSize,
fontWeight: context.content.fontWeight,
color: context.content.color
});
let subTitle = new diagramNS.TextBlock({
text: context.dataItem.SubTitle,
fontSize: 14,
color: "yellow"
});
textLayout.append(title);
textLayout.append(subTitle);
textLayout.reflow();
return shapeGroup;
}
</script>
@code {
private ShapeModel ShapeDataItem1 { get; set; } = new()
{
Title = "First Shape",
SubTitle = "New Line and Styles"
};
private ShapeModel ShapeDataItem2 { get; set; } = new()
{
Title = "Second Shape",
SubTitle = "Centered Text"
};
public class ShapeModel
{
public string Title { get; set; } = string.Empty;
public string SubTitle { get; set; } = string.Empty;
}
}