UniView
The UniView is a templated listing control that supports displaying of hierarchical data from grouped data sources.
Grouped data sources
Grouped data sources are represented by the GroupedDataSource class. A grouped data source takes a standard data source, such as a DataSet, and categorizes the data items into hierarchy levels according to the values of specified columns.
You can use the UniView with any bindable data source — not only Kentico data and objects.
Tip: If you want to display documents from Kentico, you can use the CMSUniView control, which has built-in support for loading documents.
Getting started
The following is a step-by-step tutorial that shows how to display all pages (CMS.MenuItem documents) from the sample Corporate Site using the UniView control:
Create a new Web form in your web project.
Drag the UniView control from the toolbox onto the form.
Add the following code marked by the UniView templates comments between the <cms:UniView> tags:
<cms:UniView ID="UniView1" runat="server"> <%-- UniView templates ----------------------------------------------------------------- --%> <ItemTemplate> <li> <%# HTMLHelper.HTMLEncode(Convert.ToString(Eval("NodeName"))) %> </li> </ItemTemplate> <AlternatingItemTemplate> <li> <font color="#999999"> <%# HTMLHelper.HTMLEncode(Convert.ToString(Eval("NodeName"))) %> </font> </li> </AlternatingItemTemplate> <HeaderTemplate> <ul> </HeaderTemplate> <FooterTemplate> </ul> </FooterTemplate> <%-- UniView templates ------------------------------------------------------------------ --%> </cms:UniView>
This sets the templates used when displaying the menu items. The control dynamically replaces the <%# … %> inline code with values of the currently displayed record. This is then repeated for every record in the data source.
Add the following references to the beginning of the web form’s code behind:
using System.Data; using CMS.DocumentEngine; using CMS.Helpers;
Add the following code to the Page_Load method:
// Creates a DataSet containing all menu item documents in the system DataSet ds = DocumentHelper.GetDocuments("CMS.MenuItem").Path("/", PathTypeEnum.Children).OrderBy("NodeLevel, NodeOrder"); // Checks that the DataSet isn't empty if (!DataHelper.DataSourceIsEmpty(ds)) { // Binds the DataSet to the UniView control this.UniView1.DataSource = ds; this.UniView1.DataBind(); }
This code reads documents from the database and provides them to the UniView control as a DataSet.
Save the changes to the web form and its code behind file.
Right-click the web form in the Solution explorer and select View in Browser.
The control displays all page (menu item) documents in the system.
Using grouped data sources
A flat list is not ideal for displaying pages that are actually organized in a hierarchy. The following steps show how to set up the UniView to display a hierarchical structure:
Add another reference to the beginning of the web form’s code behind:
using CMS.Base;
Modify the code of the Page_Load method according to the following:
// Creates a DataSet containing all menu item documents in the system DataSet ds = DocumentHelper.GetDocuments("CMS.MenuItem").Path("/", PathTypeEnum.Children).OrderBy("NodeLevel, NodeOrder"); // Checks that the DataSet isn't empty if (!DataHelper.DataSourceIsEmpty(ds)) { // Creates a GroupedDataSource from the DataSet // You need to specify the columns that the data uses to identify parent items, and each item's level in the hierarchy GroupedDataSource gpd = new GroupedDataSource(ds, "NodeParentID", "NodeLevel"); // Specifies the column that the data uses as an identifier (to determine parent-child relationships) this.UniView1.RelationColumnID = "NodeID"; // Binds the DataSet to the UniView control this.UniView1.DataSource = gpd; this.UniView1.DataBind(); }
This code takes the DataSet containing the documents, groups them according to the NodeID of their parent document and determines their level in the hierarchy according to their NodeLevel.
Switch to the web form’s markup, and add the SubLevelPlaceHolder control into the UniView’s ItemTemplate and AlternatingItemTemplate:
<ItemTemplate> <li> <%# HTMLHelper.HTMLEncode(Convert.ToString(Eval("NodeName"))) %> <cms:SubLevelPlaceHolder runat="server" ID="plcSub" /> </li> </ItemTemplate> <AlternatingItemTemplate> <li> <font color="#999999"> <%# HTMLHelper.HTMLEncode(Convert.ToString(Eval("NodeName"))) %> </font> <cms:SubLevelPlaceHolder runat="server" ID="plcSub" /> </li> </AlternatingItemTemplate>
The SubLevelPlaceHolder specifies where exactly the control inserts child levels in the output code. For items that have descendants in the hierarchy, the control renders the child level instead of the placeholder (including the header and footer template for the new level).
Save the changes.
When you refresh the web form’s output, the control displays the pages in a hierarchical structure:
Configuration
You can set the following properties for the UniView control:
Property name |
Description |
Sample value |
AlternatingRange |
Indicates how often the AlternatingItemTemplate should be used. |
|
AlternatingStartPosition |
Indicates the item number from which the AlternatingItemTemplate should start being used. |
|
DataSource |
The object from which the list of data items is retrieved. |
|
HideHeaderAndFooterForSingleItem |
If enabled, the UniView does not render the content of the HeaderTemplate and FooterTemplate for levels that only contain a single item. |
|
HierarchicalDisplayMode |
Sets the hierarchical display mode. Inner generates sub-levels inside the level above, Separate generates sub-levels outside of the upper levels. |
“Inner” |
OuterData |
Sets the data generated in the HeaderTemplate and FooterTemplate. |
|
PagerDataItem |
Gets or sets the pager data item object. |
|
PagerForceNumberOfResults |
If set, the DataSet containing paged items is not modified by the pager, but the pager itself behaves as if the amount of paged items were identical to this value. The value must be set to -1 for the property to be disabled. |
|
RelationColumnID |
Specifies the name of the column that the source data uses as an identifier (to determine parent-child relationships). |
“NodeID” |
SelectedItemColumnName |
The name of the column that should be used for to find out which item is currently selected. |
“DocumentID” |
SelectedItemValue |
The item whose column specified by the SelectedItemColumn property matches the value of this property will be designated as the currently selected item. Typically, you will need to insert a Macro expression in order to dynamically retrieve the appropriate value from the current context. |
“{%currentpageinfo.documentid%}” |
Transformations |
Allows you to assign a HierarchicalTransformations object representing a hierarchical transformation. The UniView renders the source data according to the hierarchical transformation (instead of the ItemTemplates). Note: To use a hierarchical transformation, you need to set the Transformations property before calling the UniView’s DataBind method. See also: UniView |
|
UseNearestItemForHeaderAndFooter |
Indicates whether the control provides data to the item templates (or transformations) that display the header and footer content. You can work with the data inside the code of the templates.
The control ignores this property if the OuterData property is set. |
Appearance and styling
You can use two different approaches to define the output format of the UniView control:
- Item templates
- A hierarchical transformation assigned through the Transformations property in the API (see the UniView section)
You can define the following item templates within the UniView tags:
Template name |
Description |
Sample value |
AlternatingItemTemplate |
Template used for alternating items. |
|
FirstItemTemplate |
Template for the first item on every level in the hierarchy. Only applied to levels that contain more than one item. |
|
FooterTemplate |
Template rendered at the end of every level (after the last item on the level). Can be used to close encapsulating elements from the HeaderTemplate. |
|
HeaderTemplate |
Template rendered at the beginning of every level (before the first item on the level). Allows you to visually separate or style individual levels. |
|
ItemTemplate |
Template used for all standard items, that are not covered by a specialized template (e.g. alternating items, first items). |
|
LastItemTemplate |
Template for the last item on every level in the hierarchy. Only applied to levels that contain more than one item. |
|
SeparatorTemplate |
Template rendered between items on the same level. The UniView does not place the separator between items on different hierarchy levels (i.e. between a parent item and its child). |
|
SingleItemTemplate |
Template applied in cases where there is only one item on a level in the hierarchy. |
Setting the location of sublevels
When displaying hierarchical data, you can add a placeholder that specifies the position of sublevels inside the code of item templates:
<cms:SubLevelPlaceHolder runat="server" ID="plcSub" />
For items that have descendants in the hierarchy, the placeholder is replaced by the child level under the given item (including the header and footer for the new level). If you do not add the sublevel placeholder, the system automatically renders child levels after the code of parent items.
You can add the placeholder into any type of item template (Item, AlternatingItem, FirstItem, LastItem, SingleItem).
Note: To use the sublevel placeholder, the HierarchicalDisplayMode property of the control must be set to Inner (this is the default state).
Displaying data using hierarchical transformations
The following example demonstrates how to use hierarchical transformations with the UniView control. The sample scenario displays a hierarchy of forum posts from the Kentico database, based on the forum thread structure. You can use the same approach for any type of hierarchical data.
Tip: The CMSUniView and QueryUniView controls allow you to use hierarchical transformations without writing custom code. However, the CMS controls can only display documents or other hierarchical Kentico data. The approach described below is primarily intended for displaying:
- External hierarchical data
- Customized or composite Kentico data sources
Create a new Web form in your web project.
Add the UniView control onto the form:
<cms:UniView ID="UniViewForumPosts" runat="server" />
Switch to the web form’s code behind file and add the following references:
using System.Data; using CMS.Forums; using CMS.Helpers; using CMS.Base; using CMS.Controls; using CMS.PortalEngine;
Set up the UniView control in the web form’s Page_Load method:
protected void Page_Load(object sender, EventArgs e) { // Loads a DataSet containing all forum posts, ordered according to the PostLevel and PostTime DataSet forumPosts = ForumPostInfoProvider.GetForumPosts().OrderBy("PostLevel, PostTime").Columns("PostID, PostForumID, PostParentID, PostIDPath, PostLevel, PostSubject, PostText, PostTime"); // Checks that the DataSet isn't empty if (!DataHelper.DataSourceIsEmpty(forumPosts)) { // Creates a GroupedDataSource from the forumPosts DataSet GroupedDataSource groupedForumPosts = new GroupedDataSource(forumPosts, "PostParentID", "PostLevel"); // Assigns the grouped data source to the UniView UniViewForumPosts.DataSource = groupedForumPosts; // Sets the display mode of the UniView UniViewForumPosts.HierarchicalDisplayMode = HierarchicalDisplayModeEnum.Inner; // Specifies the column that the data uses as an identifier (to determine parent-child relationships) UniViewForumPosts.RelationColumnID = "PostID"; // Gets the hierarchical transformation from the system TransformationInfo ti = TransformationInfoProvider.GetTransformation("CMS.Root.ForumPosts"); // Checks that the transformation exists if (ti != null) { // Checks that the transformation is hierarchical if (ti.TransformationIsHierarchical) { // Stores the structure of the hierarchical transformation into a HierarchicalTransformations object HierarchicalTransformations transformation = new HierarchicalTransformations("PostID"); transformation.LoadFromXML(ti.TransformationHierarchicalXMLDocument); // Assigns the hierarchical transformation to the UniView control UniViewForumPosts.Transformations = transformation; } } // Binds the UniView's data UniViewForumPosts.DataBind(); } }
Save the changes to the web form and its code behind file.
Define the hierarchical transformation in the Kentico administration interface (named CMS.Root.ForumPosts).
Creating the hierarchical transformation
For example, you can create the following transformation for this scenario:
- Open the Document types application.
- Edit the Root document type.
- Open the Transformations tab.
- Click New hierarchical transformation.
- Name the transformation ForumPosts.
- Click Save.Add 4 sub-transformations:
Transformation type: Item transformation
Document types: All (empty)
Level: 0
ASCX transformation code:
<div style="background-color:#FFFBE8;"> <a href="<%# ForumFunctions.GetPostURL(Eval("PostIDPath"), Eval("PostForumID")) %>"><%# Eval("PostSubject", true) %></a> <p><%# HTMLEncode(LimitLength(StripTags(RemoveDynamicControls(RemoveDiscussionMacros(Eval("PostText")))), 400)) %></p> </div>
Transformation type: Item transformation
Document types: All (empty)
Level: 1
- Apply to sublevels: yes (checked)
ASCX transformation code:
<div> <a href="<%# ForumFunctions.GetPostURL(Eval("PostIDPath"), Eval("PostForumID")) %>"><%# Eval("PostSubject", true) %></a> <p><%# HTMLEncode(LimitLength(StripTags(RemoveDynamicControls(RemoveDiscussionMacros(Eval("PostText")))), 400)) %></p> </div>
Transformation type: Header transformation
Level: All (empty)
- Apply to sublevels: yes (checked)
Transformation code:
<div style="margin-left:30px">
Transformation type: Footer transformation
Level: All (empty)
- Apply to sublevels: yes (checked)
Transformation code:
</div>
Return to Visual Studio, right-click the web form in the Solution explorer and select View in Browser.
The page displays the forum posts according to the hierarchical transformation: