This blog post is [Updatable] - I use it to store snippets at the same place just for a case.
I know for most of experienced Sitecore geeks my post may seem sort of obvious, however even you may find something interesting below or suggest me to add anything you consider important to append.
While working with Sitecore for years, I have met so many attempts to re-invent the wheel in sense of ugly implementation of functions that are part of Sitecore API, so I decided to write this post, placing most frequent but sometimes less known usages here within one post. So, here we go:
- Standard API - create, read, edit, save, rename,
- Media library
- Publishing
- Presentation / placeholders
- Indexing
- DMS / xDB
- Other
1. Standard API - create, read, edit, save, rename, move etc.
1.1. To start with the basics - how to get the item.
1 2 3 4 5 6 | Item item = Sitecore.Context.Database.GetItem( "/sitecore/content/home/products/tv" );
if (item != null )
{
}
|
Avoid using explicit database by name like below, get database out of Context, rather than by name explicitly.
1 2 | var database = Database().GetDatabase( "master" ));
|
1.2. Accessing the fields - multiple ways:
1 2 | string value1 = someItem[fieldName];
|
1 2 | string value2 = someField.Value;
|
1 2 3 | bool allowStandardValue = false ;
string value3 = someField.GetValue(allowStandardValue);
|
1 2 3 | bool allowStandardValue = false ;
bool allowDefaultValue = false ;
string value5 = someField.GetValue(allowStandardValue, allowDefaultValue);
|
The order usually follows: if actual value exists, it is returned, otherwise if the item is clone - it returns clone's value, if not - returns standard value, and in case standard value not set - the default value (empty string, otherwise - null).
1.3. Read all fields.
For the sake of performance, Sitecore will not give you all fields in the FieldCollection in the following code, only fields with explicit values on item level, including empty string.
1 | Sitecore.Context.Item.Fields.ReadAll();
|
1.4. Checking if a Sitecore field has a value
1 | Sitecore.Context.Item.Fields[ "fieldName" ].HasValue
|
or 1 | Sitecore.Context.Item.Fields[ "fieldName" ].Value != ""
|
1.5. When rendering fields from within the code, always use FieldRenderer class, in order to ensure that resulting output HTML is Page Editor friendly.
1 | FieldRenderer.Render(item, "field name" )
|
1.6. Editing the item - straightforward way.
1 2 3 | item.Editing.BeginEdit();
item.Fields[ "Title" ].Value = "My New Title" ;
item.Editing.EndEdit();
|
1.7. More elegant way of editing the item, using IDisposable EditContext
1 2 3 4 | using ( new EditContext(item))
{
item[ "Title" ] = "My New Title" ;
}
|
More about EditContext and its options please read in API - IDisposable Usings article.
1.8. Miltilist field. I have evidenced several attempts of manipulation multi-list items by physically stored pipe-separated GUIDs, all clumsy and not best practice code with bad smell. Don't do that - use Sitecore API instead:
1 2 | Sitecore.Data.Fields.MultilistField multilistField = Sitecore.Context.Item.Fields[ "myMultilistField" ];
Sitecore.Data.Items.Item[] items = multilistField.GetItems();
|
1.10. Create item programmatically
Minimum code: 1 2 3 4 | Item newItem = parent.Add(itemName, template);
newItem.Editing.BeginEdit();
newItem.Fields[ "fieldName" ].Value = "fieldValue" ;
newItem.Editing.EndEdit();
|
Full ready-to-use-code: 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 | public static void CreateItem()
{
using ( new Sitecore.SecurityModel. SecurityDisabler())
{
Sitecore.Data. Database master = Sitecore.Data.Database.GetDatabase( "master" );
TemplateItem template = master.GetItem( "/sitecore/templates/Ecosystem/Concrete/MenuPage" );
Item parentItem = master.GetItem( "/sitecore/content/Ecosystem" );
Item newItem = parentItem.Add( "My New Page" , template);
newItem.Editing.BeginEdit();
try
{
newItem.Fields[ "Title" ].Value = "My New Page - Title" ;
newItem.Editing.EndEdit();
}
catch (System. Exception ex)
{
newItem.Editing.CancelEdit();
}
}
}
|
1.11. Item propose name - will suggest valid item name according with current settings
1 | ItemUtil.ProposeValidItemName( "some-name" );
|
1.12. Item has plenty of helpful methods, like CopyTo() and MoveTo() and others you may see in the context menu.
1 | item.MoveTo (parentItem);
|
2.1. Determining item type.
1 2 | Sitecore.Context.Item.Paths.IsContentItem;
|
1 2 | Sitecore.Context.Item.Paths.IsMediaItem
|
2.2. The example below is anti-pattern. At the first glance it seems to be an elegant way of inline resolving media path right in MVC Razor view. So what is wrong with that?
1 2 | < img src="@MediaManager.GetMediaUrl(
Sitecore.Context.Database.GetItem(Html.Sitecore().CurrentRendering.DataSource))" />
|
First and most important - if datasource is not set (or set wrongly) then GetItem will return null being passed into GetMedaiUrl as the parameter, so it will fire NullReferenceException and the whole page won't load. You must validate parameters and do null-checks, so the best way out is to extract this logic into helper static (or extensions) class.
But the best option for sure is to pass this resolved URL into a view as a public property of strongly-typed viewmodel object.
3. Publishing
3.1. Publish item programmatically
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 | private void PublishItem(Sitecore.Data.Items.Item item)
{
Sitecore.Publishing.PublishOptions publishOptions =
new Sitecore.Publishing.PublishOptions(item.Database,
Database.GetDatabase( "web" ),
Sitecore.Publishing.PublishMode.SingleItem,
item.Language,
System.DateTime.Now);
Sitecore.Publishing.Publisher publisher = new Sitecore.Publishing.Publisher(publishOptions);
publisher.Options.RootItem = item;
publisher.Options.Deep = true ;
publisher.Publish();
}
|
4. Presentation / placeholders
4.1. Get count of renderings in a particular placeholder:
1 2 3 4 | public static int GetRenderingsCount( string placeholderKey)
{
return Sitecore.Context.Item.Visualization.GetRenderings(Sitecore.Context.Device, false ).Count(p => p.Placeholder.Equals(placeholderKey));
}
|
4.2. Get position index of a rendering within a particular placeholder:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 | public static string GetComponentPositionOnPlaceHolder(System.Web.UI.Control webControl)
{
if (webControl == null ) return "0" ;
var split = webControl.ID.Split( '_' );
if (split.Length > 1)
{
int index = 0;
if ( int .TryParse(split.Last(), out index))
{
return (index + 1).ToString(CultureInfo.InvariantCulture);
}
}
return "0" ;
}
|
5. Indexing
5.1. Simple Linq query
1 2 3 4 5 6 7 | using ( var context = new ContentSearchManager.GetIndex( "indexname" ).CreateSearchContext())
{
var query = context.GetQueryable<sometype>.Where(i => i.Name.StartsWith( "Something" ));
}
</sometype>
|
5.2. Add single item to index
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 | public static void AddItemToIndex(Item item, string indexName)
{
var tempItem = (SitecoreIndexableItem)item;
ContentSearchManager.GetIndex(indexName).Refresh(tempItem);
}
public static void UpdateItemInIndex(Item item, string indexName)
{
var tempItem = (SitecoreIndexableItem)item;
ContentSearchManager.GetIndex(indexName).Update(tempItem.UniqueId);
}
public static void DeleteItemInIndex(Item item, string indexName)
{
var tempItem = (SitecoreIndexableItem)item;
ContentSearchManager.GetIndex(indexName).Delete(tempItem.UniqueId);
}
|
5.3. Queryable extensions
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 | public static class QueryableExtensions
{
public static IQueryable<t> IsContentItem<t>( this IQueryable<t> query) where T : SearchResultItem
{
return query.IsDescendantOf(ItemIDs.ContentRoot);
}
public static IQueryable<t> IsContextLanguage<t>( this IQueryable<t> query) where T : SearchResultItem
{
return query.Where(searchResultItem => searchResultItem.Language.Equals(Context.Language.Name));
}
public static IQueryable<t> IsLatestVersion<t>( this IQueryable<t> query) where T : SearchResultItem
{
return query.Where(searchResultItem => searchResultItem[ "_latestversion" ].Equals( "1" ));
}
public static IQueryable<t> HasBaseTemplate<t>( this IQueryable<t> query, ID templateId) where T : SearchResultItem
{
var id = templateId.ToShortID().ToString().ToLowerInvariant();
return query.Where(searchResultItem => searchResultItem[ "_basetemplates" ].Contains(id));
}
public static IQueryable<t> IsDescendantOf<t>( this IQueryable<t> query, ID parentId) where T : SearchResultItem
{
return query.Where(searchResultItem => searchResultItem.Paths.Any(ancestorId => ancestorId == parentId));
}
}
</t></t></t></t></t></t></t></t></t></t></t></t></t></t></t>
|
6. DMS / xDB
6.1. DMS - hit the goal programmatically.
1 2 3 4 5 6 7 8 9 10 11 12 13 | if (Sitecore.Analytics.Tracker.IsActive && Sitecore.Analytics.Tracker.CurrentPage != null )
{
Item goalItem = Sitecore.Context.Database.GetItem(&quot;{xxxxx-xxx-xxx-xxxxx}&quot;);
PageEventItem goal = new PageEventItem(goalItem);
VisitorDataSet.PageEventsRow pageEventsRw = Sitecore.Analytics.Tracker.CurrentPage.Register(goal);
pageEventsRw.Data = "custom text" ;
Sitecore.Analytics.Tracker.Submit();
}
|
7. Other
7.1. Jumping between the sites of current Sitecore instance
1 | Sitecore.Context.SetActiveSite(currentSiteName);
|