Grouping
Grouping in the RadDiagram is a way of organizing shapes and connections in logical groups. Groups are non-visual, logical collections of shapes and/or connections and/or other groups which can be created in code behind or with Commands.
Grouping API
IGroup, IShape and IConnection implement the IGroupable interface.
IGroup provides a collection of IGroupable Items. This practically means that you can have groups of shapes, connections and other groups. This allows you to have unlimited levels of nested logical groups.
The IGroupable interface has two members:
ParentGroup- of type IGroup.
ParentGroupChanged event- of type EventHandler.
The IGroup has the following members
IsSelected - of type boolean.
Items- IList.
Name- string.
ItemsChanged- of type NotifyCollectionChangedEventHandler.
SelectionChanged- of type EventHandler.
The RadDiagram provides the following "grouping" members:
Group([string groupName], params IGroupable[] items)) - creates an IGroup from a given Array of IGroupable items.
Ungroup(params IGroup[] items)- removes the logical groups previously created.
Groups- IEnumerable. The list of root-level Groups created in the RadDiagram.
GroupsChanged- of type EventHandler. Fires when grouping or ungrouping is performed. Does not fire when an Item's ParentGroup is changed.
GroupStyle- Style that applies to the Selection Rectangle around the logical groups.
The DiagramCommands class has Group and Ungroup which are both RoutedUICommands. The Group Command groups the selected IGroupable-s and the Ungroup command ungroups the selected IGroups.
Grouping In Code Behind
In the next example we will create 20 shapes with contents - "1", "2",... "20" and we will group them by their parity.
Let's first create some random shapes:
Random random = new Random();
for (int i = 0; i < 21; i++)
{
RadDiagramShape s = new RadDiagramShape() { Width = 30, Height = 30, Content = i };
s.Shape = new Telerik.WinControls.RoundRectShape(5);
s.BackColor = System.Drawing.Color.CadetBlue;
s.Position = new Telerik.Windows.Diagrams.Core.Point(random.Next(0, 900), random.Next(0, 200));
this.radDiagram1.Items.Add(s);
}
Dim random As New Random()
For i As Integer = 0 To 20
Dim s As New RadDiagramShape() With { _
.Width = 30, _
.Height = 30, _
.Content = i _
}
s.Shape = New Telerik.WinControls.RoundRectShape(5)
s.BackColor = Color.CadetBlue
s.Position = New Telerik.Windows.Diagrams.Core.Point(random.[Next](0, 900), random.[Next](0, 200))
Me.RadDiagram1.Items.Add(s)
Next
Now let's implement some grouping logic:
IShape[] evenShapes = this.radDiagram1.Shapes.Where(x => int.Parse(x.Content.ToString()) % 2 == 0).ToArray<IShape>();
IShape[] oddShapes = this.radDiagram1.Shapes.Where(x => int.Parse(x.Content.ToString()) % 2 == 1).ToArray<IShape>();
this.radDiagram1.Group("Even", evenShapes);
this.radDiagram1.Group("Odd", oddShapes);
Dim evenShapes As IShape() = Me.RadDiagram1.Shapes.Where(Function(x) Integer.Parse(x.Content.ToString()) Mod 2 = 0).ToArray()
Dim oddShapes As IShape() = Me.RadDiagram1.Shapes.Where(Function(x) Integer.Parse(x.Content.ToString()) Mod 2 = 1).ToArray()
Me.RadDiagram1.Group("Even", evenShapes)
Me.RadDiagram1.Group("Odd", oddShapes)
Below you can see how the grouping works:
The shapes' positions are not changed after grouping. If you click shape 3, this selects every odd shape. Dragging a single shape from group, drags the whole group.
Now let's play with grouping and upgrouping. Below are some code examples and the result of the code execution:
this.radDiagram1.Group("123", this.radDiagram1.Shapes[1], this.radDiagram1.Shapes[2], this.radDiagram1.Shapes[3]);
this.radDiagram1.Group("345", this.radDiagram1.Shapes[3], this.radDiagram1.Shapes[4], this.radDiagram1.Shapes[5]);
Me.RadDiagram1.Group("123", Me.RadDiagram1.Shapes(1), Me.RadDiagram1.Shapes(2), Me.RadDiagram1.Shapes(3))
Me.RadDiagram1.Group("345", Me.RadDiagram1.Shapes(3), Me.RadDiagram1.Shapes(4), Me.RadDiagram1.Shapes(5))
This makes group "123" with items {Shapes[1],Shapes[2]} and "345" with items {Shapes[3], Shapes[4], Shapes[5]}. The creation of the second group excludes Shapes[3] from group "123".
Alternatively if we use one name in the method:
this.radDiagram1.Group("123", this.radDiagram1.Shapes[1], this.radDiagram1.Shapes[2], this.radDiagram1.Shapes[3]);
this.radDiagram1.Group("123", this.radDiagram1.Shapes[3], this.radDiagram1.Shapes[4], this.radDiagram1.Shapes[5]);
Me.RadDiagram1.Group("123", Me.RadDiagram1.Shapes(1), Me.RadDiagram1.Shapes(2), Me.RadDiagram1.Shapes(3))
Me.RadDiagram1.Group("123", Me.RadDiagram1.Shapes(3), Me.RadDiagram1.Shapes(4), Me.RadDiagram1.Shapes(5))
This produces a single group "123" with the 5 elements Shapes[1]-Shapes[5].
How To Create SubGroups
Subgrouping must be done from subgroups to parent groups. In other words, creating a parent group, then creating its subgroup is not possible.
Below is an example of creating a subgroups and a parent group.
IGroup groupA = this.radDiagram1.Group("1-2", this.radDiagram1.Shapes[1], this.radDiagram1.Shapes[2]);
IGroup groupB = this.radDiagram1.Group("3-4", this.radDiagram1.Shapes[3], this.radDiagram1.Shapes[4]);
IGroup parentGroup = this.radDiagram1.Group("1-2-3-4", groupA, groupB);
Dim groupA As IGroup = Me.RadDiagram1.Group("1-2", Me.RadDiagram1.Shapes(1), Me.RadDiagram1.Shapes(2))
Dim groupB As IGroup = Me.RadDiagram1.Group("3-4", Me.RadDiagram1.Shapes(3), Me.RadDiagram1.Shapes(4))
Dim parentGroup As IGroup = Me.RadDiagram1.Group("1-2-3-4", groupA, groupB)
Below you can see how consecutive clicks on a shape reflects the selection of groups. First click selects the outermost (the biggest group), second click selects the smaller group and the third selects only the shape.The forth click will select the biggest group and so on.
If you need to select particular items without selecting the whole group, you can use Rectangular Selection.
What will happen if we try to create parent group then create subgroups?
this.radDiagram1.Group("1-2-3-4", this.radDiagram1.Shapes[1], this.radDiagram1.Shapes[2], this.radDiagram1.Shapes[3], this.radDiagram1.Shapes[4]);
this.radDiagram1.Group("1-2", this.radDiagram1.Shapes[1], this.radDiagram1.Shapes[2]);
Me.RadDiagram1.Group("1-2-3-4", Me.RadDiagram1.Shapes(1), Me.RadDiagram1.Shapes(2), Me.RadDiagram1.Shapes(3), Me.RadDiagram1.Shapes(4))
Me.RadDiagram1.Group("1-2", Me.RadDiagram1.Shapes(1), Me.RadDiagram1.Shapes(2))
This will create two separate groups - "1-2-3-4" with items {Shapes[3], Shapes[4]} and "1-2" with items {Shapes[1], Shapes[2]}.
Creating a parent group then creating its subgroup is not possible.
Ungrouping
You can Ungroup one or several groups with the Ungroup method of RadDiagram:
this.radDiagram1.Ungroup(groupA, groupB);
this.radDiagram1.Ungroup(this.radDiagram1.Groups.ToArray());
Me.RadDiagram1.Ungroup(groupA, groupB)
Me.RadDiagram1.Ungroup(Me.RadDiagram1.Groups.ToArray())
Grouping With Commands
Using the DiagramCommands Group and Ungroup is straightforward. Group applies to the selected IGroupables and Ungroup applies to the selected IGroups.
this.radDiagram1.DiagramElement.ServiceLocator.GetService<ISelectionService<IDiagramItem>>().SelectItem(this.radDiagram1.Shapes[1],true);
this.radDiagram1.DiagramElement.ServiceLocator.GetService<ISelectionService<IDiagramItem>>().SelectItem(this.radDiagram1.Shapes[5],true);
this.radDiagram1.DiagramElement.TryExecuteCommand(Telerik.WinControls.UI.Diagrams.DiagramCommands.Group);
Me.RadDiagram1.DiagramElement.ServiceLocator.GetService(Of ISelectionService(Of IDiagramItem))().SelectItem(Me.RadDiagram1.Shapes(1), True)
Me.RadDiagram1.DiagramElement.ServiceLocator.GetService(Of ISelectionService(Of IDiagramItem))().SelectItem(Me.RadDiagram1.Shapes(5), True)
Me.RadDiagram1.DiagramElement.TryExecuteCommand(Telerik.WinControls.UI.Diagrams.DiagramCommands.Group)