Propeller MVC is a small library created to improve the developer workflow when building Sitecore solutions. Sitecore provides standard rendering of Sitecore items but in the real world, you often need a more fine grained access to the data properties. Secondly, more and more data is flowing between the server and browser via asynchronous calls which means we need the raw data instead of the HTML presentation.
This section covers the basic process of using the framework and consists of four steps.
- Define a ViewModel
- Map Sitecore template
- instantiate ViewModel
- Render ViewModel
The ViewModel contains the UI properties and must inherit from PropellerViewModel
.
The base ViewModel requires a generic type which should be its self. The reason for this is to provide static typing in the razor-view (more on that later).
public class CharacterViewModel : PropellerViewModel<CharacterViewModel>
{
public string Species { get; set; }
public string Gender { get; set; }
public string Occupation { get; set; }
public string Affiliation { get; set; }
public string Homeworld { get; set; }
}
Propeller MVC uses guids to map Sitecore Items to code. To lighting the burden of mapping, the framework uses a code-first approach which provides static typing. Each ViewModel is mapped using a ConfigurationMap
class as shown where.
public class CharacterMap : ConfigurationMap<CharacterViewModel>
{
public CharacterMap()
{
SetProperty(p => p.Species ).Map(new ID("{FF80B838-B0E4-4266-9E6A-2918585C4EB7}"));
SetProperty(p => p.Gender ).Map(new ID("{7079E001-680A-460C-BB86-91E31C3EA2A5}"));
SetProperty(p => p.Occupation ).Map(new ID("{73FF41AA-2ACF-45F8-9FA3-8C4F6374F217}"));
SetProperty(p => p.Affiliation).Map(new ID("{EE444D4A-7FDF-4D94-8ABA-6054A898E721}"));
SetProperty(p => p.Homeworld ).Map(new ID("{E8D236D1-E473-4816-89BD-EE4CCD613972}"));
}
}
The Controller part is the simplest step. Basically the Controller has two tasks, 1) instantiate the ViewModel 2) provide the Sitecore item for the ViewModel. Generally this item ( regardless of it being the page item or rendering item) is accessible via the RenderingContext and the ViewModel will automatically resolve this.
NOTE
If you arn't using the presentasion package there is a ModelFactory you can use instead of the ViewModelFactory which works the exact same way.
public class CharacterController : Controller
{
public ActionResult Index()
{
var vmFactory = new ViewModelFactory();
var characterViewModel = vmFactory.Create<CharacterViewModel>(RenderingContext.Current.Rendering);
return View(characterViewModel);
}
}
Use the ViewModel as the @model
in the razor view. In order to support Sitecores Experience editor the ViewModel provides a Render
method which uses an expression for selecting the property to render.
<form class="form-horizontal">
<div class="form-group">
<label class="col-sm-2 control-label">Name</label>
<div class="col-sm-10">
<p class="form-control-static">@Model.DisplayName</p>
</div>
<label class="col-sm-2 control-label">Species</label>
<div class="col-sm-10">
<p class="form-control-static">@Model.Render(p => p.Species)</p>
</div>
<label class="col-sm-2 control-label">Gender</label>
<div class="col-sm-10">
<p class="form-control-static">@Model.Render(p => p.Gender)</p>
</div>
</div>
</form>
Configuration is mainly about mapping the Model with the Sitecore Item. While it is possible to do this mapping by using the template field name and the ViewModels property name (thus using a convension based mapping), a decision was made not to use this approch. The reason being that renaming any property (in code or Sitecore) would break the mapping. Secondly it would make it harder to debug, especially with identical property names in different parts of the solution. Using item guids is a much more solid solution and makes it easy to trace down the mapping.
The actual task of mapping is done by using a ConfigurationMap
which takes a generic type defining the ViewModel to map and must be done for every ViewModel.
public class CharacterMap : ConfigurationMap<CharacterViewModel>
{
public CharacterMap()
{
SetProperty(p => p.Species ).Map(new ID("{FF80B838-B0E4-4266-9E6A-2918585C4EB7}"));
SetProperty(p => p.Gender ).Map(new ID("{7079E001-680A-460C-BB86-91E31C3EA2A5}"));
SetProperty(p => p.Occupation ).Map(new ID("{73FF41AA-2ACF-45F8-9FA3-8C4F6374F217}"));
SetProperty(p => p.Affiliation).Map(new ID("{EE444D4A-7FDF-4D94-8ABA-6054A898E721}"));
SetProperty(p => p.Homeworld ).Map(new ID("{E8D236D1-E473-4816-89BD-EE4CCD613972}"));
}
}
The first configuration method is the Map()
method and is the only required configuration. It simply maps the property with the Sitecore template field Id, like this:
SetProperty(p => p.Affiliation).Map(new ID("{EE444D4A-7FDF-4D94-8ABA-6054A898E721}"));
This mapping only registers the field Id together with the ViewModel property. This means that the ViewModel properties remain null/empty but you can use the ViewModels Render()
in the Razor view to output the content as html.
If you want to populate the ViewModel with actual data, the configuration system provides at method automating this. The configuration method is called Include()
.
SetProperty(p => p.Affiliation).Map(new ID("{EE444D4A-7FDF-4D94-8ABA-6054A898E721}")).Include();
The ViewModel contains a Commit()
method which saves edited properties. It will however only work on properties which have been configured with the Editable()
method (Editable configuration implicitly triggers the Include configuration).
SetProperty(p => p.Affiliation).Map(new ID("{EE444D4A-7FDF-4D94-8ABA-6054A898E721}")).Editable();
ViewModels should not inherit from other ViewModels. If a base class is needed one should use an Interface
to define the base properties. In order to only map the base class properties once, configuration is applied to the Interface and not the ViewModel implmenting the Interface.
public class BaseViewModelMap : ConfigurationMap<ICharacterViewModel>
{
public CharacterMap()
{
SetProperty(p => p.Species ).Map(new ID("{FF80B838-B0E4-4266-9E6A-2918585C4EB7}"));
SetProperty(p => p.Gender ).Map(new ID("{7079E001-680A-460C-BB86-91E31C3EA2A5}"));
SetProperty(p => p.Occupation ).Map(new ID("{73FF41AA-2ACF-45F8-9FA3-8C4F6374F217}"));
SetProperty(p => p.Affiliation).Map(new ID("{EE444D4A-7FDF-4D94-8ABA-6054A898E721}"));
SetProperty(p => p.Homeworld ).Map(new ID("{E8D236D1-E473-4816-89BD-EE4CCD613972}"));
}
}
This ViewModel can render the data in a number of ways. We will cover these methods with a series of use cases.
Without doing anything you have access to the ItemName, DisplayName and url.
The ViewModel has a Render method which uses Sitecores FieldRenderer.Render method. This means you get the standard rendering provided by Sitecore which also supports the Experience editor for editing content. The “problem” with Sitecores render method is that you need to provide it the field Id for the property you want to render or the property name (in the configuration chapter we discussed why we avoid field name as an identifier). The ViewModel hides the field identification away by providing an expression with a generic type of its self. This enables static typing of the property to render. Because the ConfigurationMap has mapped the property name (fully qualified propertyname) with the field, the ViewModel is able to lookup the field Id based on the expression. No more magic Ids or Constant files lying around. The ViewModel provides a tight mapping to the Sitecore item which cannot ‘leak’ outside the scope of the
ViewModel.Render(p => p.Species)
The ViewModels property type has no infuence on the rendering the data when using the ViewModels Render
method. It simply provides static typing and provices a way of resolving Sitecores field id.
Basic rendering works great with simple data types but are not very useful when dealing with referenced data items. If you were to Render a DropLink or TreeList, the result be the Id of referenced item(s), not particular useful. In order to deal with referenced items, the property type must be a PropellerModel or an IEnumerable. That way you can just access the properties on the referenced item directly. Circurlar references are autotomaticly detected and managed when constructing the object graph, so no worries there.
Web services (web api's) are becomming increasenly popular and is such case, you are not interrested in the HTML output from Sitecore. What you are want is the raw values. Now you could manually fetch the raw values from the Sitecore item, but this is tedious work and genereally not something we want to spend our time with. Luckily the framework provides a way to automate this work and is done at the configuration level. As mentioned in the Configuration section, you can chain configuration for each property. You can tell the framework to automatically populate the ViewModels properties by configuring them with the Include()
method like this.
SetProperty(p => p.Species).Map(new ID("{FF80B838-B0E4-4266-9E6A-2918585C4EB7}")).Include();
IMPORTANT: For this feature to work, it is important that the ViewModel properties are defined with the currect data type e.g string
for text, bool
for Checkbox, double
for number, DateTime
for date etc. Complex types e.i other ViewModels will also be extracted if the correct ViewModel is defined at the property type.
Sitecore has its own complex types like Image and Generel link. Extracting raw values from these types is done using a FieldAdapter
which must implement the IFieldAdapter
. The framework comes with set of FieldAdapters but their main purpose is to provide a plugable system for customizing the data extracting. Here is an example of the FieldAdapter for Sitecores Image type.
public class Image : IFieldAdapter
{
public string Url { get; set; }
public string Alt { get; set; }
public void InitAdapter(Item item, ID propId)
{
ImageField image = item.Fields[propId];
var mediaItem = image.MediaDatabase.GetItem(image.MediaID);
Url = Sitecore.Resources.Media.MediaManager.GetMediaUrl(mediaItem);
Alt = image.Alt;
}
}
The IFieldAdapter
contains only the InitAdapter
method which provide the Sitecore item and the target field id. From here you can extract any data you wish and provide it via properties. The ModelFactory
will automatially run the InitAdapter
method when it detects a property which type implements IFieldAdapter
. Now you can access your own properties anywhere. In the example below the Photo property uses the ImageFieldAdapter which provides two properties; Url
and Alt
.
<img src="@Model.Photo.Url"
alt="@Model.Photo.Alt"
style="max-width: 100%;height: auto"/>
The framework supports the use of ASP.NET helper Template (in the razor view) and can take any object type, typically a type provided by the ViewModel.
@helper VehicleTemplate(Propeller.Mvc.Model.Adapters.Image img)
{
<div style="background-image:url('@img.Url)');">
<strong style="color: white;">@img.Alt</strong>
</div>
}
And is used by Template method which is an extension method for the IPropellerTemplate
interface.
@Model.Template(vm => VehicleTemplate(vm.Photo))
The template is obviously beneficial when you have 'complex' renderings several places on the page because you only defined it once. But maybe more beneficial is, that it provides a combinetion of custom rendering and supporting the Experience editor. The mechanics of the Template
method is very simple. When in page edit mode, the template method uses the the Sitecores built in Render method as described in 2.2 When not in page edit mode it uses the template (helper method).
As the extension method hinted, the ViewModel needs ot implement the IPropellerTemplate interface where T is the ViewModel type. The interface only hads one method TempateArg
which is very trivial to implement.
public VehicleViewModel TemplateArg()
{
return this;
}