One of the easiest ways to understand ChameleonForms is to see an equivalent sample of something you are already familiar with (e.g. out-of-the-box ASP.NET MVC) alongside the equivalent with ChameleonForms. With this in mind we have created an example comparison in the Example project that you can run and inspect to illustrate how to implement an example form using:
- Built-in Tag helpers
- Built-in HTML helpers
- Abstracted and templated Editor Templates
- ChameleonForms - Tag Helpers
- ChameleonForms - HTML Helpers
The example in question is an arbitrary signup form against the SignupViewModel
in the ComparisonController
:
public class SignupViewModel
{
[Required]
public string FirstName { get; set; }
[Required]
public string LastName { get; set; }
[DisplayFormat(DataFormatString = "{0:d/M/yyyy}", ApplyFormatInEditMode = true)]
[Required]
public DateTime DateOfBirth { get; set; }
[Required]
[EmailAddress]
public string EmailAddress { get; set; }
[Required]
[DataType(DataType.Password)]
public string Password { get; set; }
[Required]
public MembershipType MembershipType { get; set; }
[Url]
public Uri Homepage { get; set; }
[DataType(DataType.MultilineText)]
public string Bio { get; set; }
[Required]
public bool TermsAndConditions { get; set; }
}
public enum MembershipType
{
Standard,
Bonze,
Silver,
Gold,
Platinum
}
The sample implementation of the form has been split up as follows:
- Form
- Message (Information)
- Instructions to fill in the form to sign up for an account
- Section to enter details
- Fields for first name, last name and date of birth
- Section to enter account details
- Fields for email address, password, and drop down for membership type
- Section to enter additional details
- Field for homepage and text area for bio
- Message (Action)
- Link to terms and conditions and checkbox for user to confirm
- Navigation
- Submit button
- Message (Information)
Here is a comparison of the different implementations of the same form:
<form asp-action="TagHelpers" asp-controller="Comparison" method="post" id="signup-form" novalidate="novalidate">
<div class="information_message">
<h3>Signup for an account</h3>
<div class="message">
<p>Please fill in your information below to signup for an account.</p>
</div>
</div>
<fieldset>
<legend>Your details</legend>
<dl>
<dt><label asp-for="FirstName"></label> <em class="required">*</em></dt>
<dd><input asp-for="FirstName" required="required" /> <span asp-validation-for="FirstName"></span></dd>
<dt><label asp-for="LastName"></label> <em class="required">*</em></dt>
<dd><input asp-for="LastName" required="required" /> <span asp-validation-for="LastName"></span></dd>
<dt><label asp-for="DateOfBirth"></label> <em class="required">*</em></dt>
<dd><input asp-for="DateOfBirth" required="required" type="text" data-val-format="d/M/yyyy" aria-describedby="DateOfBirth--Hint" /><div class="hint" id="DateOfBirth--Hint">DD/MM/YYYY</div> <span asp-validation-for="DateOfBirth"></span></dd>
</dl>
</fieldset>
<fieldset>
<legend>Account details</legend>
<dl>
<dt><label asp-for="EmailAddress"></label> <em class="required">*</em></dt>
<dd><input asp-for="EmailAddress" required="required" aria-describedby="EmailAddress--Hint" /><div class="hint" id="EmailAddress--Hint">An email will be sent to this address to confirm you own it</div> <span asp-validation-for="EmailAddress"></span></dd>
<dt><label asp-for="Password"></label> <em class="required">*</em></dt>
<dd><input asp-for="Password" required="required" /> <span asp-validation-for="Password"></span></dd>
<dt><label asp-for="MembershipType"></label> <em class="required">*</em></dt>
<dd><select asp-for="MembershipType" required="required" asp-items="Html.GetEnumSelectList<ComparisonController.MembershipType>()"></select> <span asp-validation-for="MembershipType"></span></dd>
</dl>
</fieldset>
<fieldset>
<legend>Additional details</legend>
<dl>
<dt><label asp-for="Bio"></label></dt>
<dd><textarea asp-for="Bio" cols="20" rows="2"></textarea> <span asp-validation-for="Bio"></span></dd>
<dt><label asp-for="Homepage"></label></dt>
<dd><input asp-for="Homepage" placeholder="https://" /> <span asp-validation-for="Homepage"></span></dd>
</dl>
</fieldset>
<div class="action_message">
<h3>Confirm the Terms & Conditions</h3>
<div class="message">
<p>Please <a href="#">read the terms and conditions</a>.</p>
<input asp-for="TermsAndConditions" /> <label asp-for="TermsAndConditions">I agree to the terms and conditions</label>
</div>
</div>
<div class="form_navigation">
<button type="submit" class="btn btn-primary">Signup</button>
</div>
</form>
@using (Html.BeginForm("HtmlHelpers", "Comparison", FormMethod.Post, new {id = "signup-form", novalidate = "novalidate"}))
{
<div class="information_message">
<h3>Signup for an account</h3>
<div class="message">
<p>Please fill in your information below to signup for an account.</p>
</div>
</div>
<fieldset>
<legend>Your details</legend>
<dl>
<dt>@Html.LabelFor(m => m.FirstName) <em class="required">*</em></dt>
<dd>@Html.TextBoxFor(m => m.FirstName, new { required = "required" }) @Html.ValidationMessageFor(m => m.FirstName)</dd>
<dt>@Html.LabelFor(m => m.LastName) <em class="required">*</em></dt>
<dd>@Html.TextBoxFor(m => m.LastName, new { required = "required" }) @Html.ValidationMessageFor(m => m.LastName)</dd>
<dt>@Html.LabelFor(m => m.DateOfBirth) <em class="required">*</em></dt>
<dd>@Html.TextBoxFor(m => m.DateOfBirth, new{data_val_format = "d/M/yyyy", aria_describedby = "DateOfBirth--Hint", required = "required"})<div class="hint" id="DateOfBirth--Hint">DD/MM/YYYY</div> @Html.ValidationMessageFor(m => m.DateOfBirth)</dd>
</dl>
</fieldset>
<fieldset>
<legend>Account details</legend>
<dl>
<dt>@Html.LabelFor(m => m.EmailAddress) <em class="required">*</em></dt>
<dd>@Html.TextBoxFor(m => m.EmailAddress, new{aria_describedby = "EmailAddress--Hint", required = "required"})<div class="hint" id="EmailAddress--Hint">An email will be sent to this address to confirm you own it</div> @Html.ValidationMessageFor(m => m.EmailAddress)</dd>
<dt>@Html.LabelFor(m => m.Password) <em class="required">*</em></dt>
<dd>@Html.PasswordFor(m => m.Password, new { required = "required" }) @Html.ValidationMessageFor(m => m.Password)</dd>
<dt>@Html.LabelFor(m => m.MembershipType) <em class="required">*</em></dt>
<dd>@Html.DropDownListFor(m => m.MembershipType,
Enum.GetValues(typeof(ComparisonController.MembershipType))
.Cast<ComparisonController.MembershipType>()
.Select(m => new SelectListItem{Text = m.Humanize(), Value = m.ToString()}), new { required = "required" }
) @Html.ValidationMessageFor(m => m.MembershipType)
</dd>
</dl>
</fieldset>
<fieldset>
<legend>Additional details</legend>
<dl>
<dt>@Html.LabelFor(m => m.Bio)</dt>
<dd>@Html.TextAreaFor(m => m.Bio) @Html.ValidationMessageFor(m => m.Bio)</dd>
<dt>@Html.LabelFor(m => m.Homepage)</dt>
<dd>@Html.TextBoxFor(m => m.Homepage, new{placeholder = "https://", type = "url"}) @Html.ValidationMessageFor(m => m.Homepage)</dd>
</dl>
</fieldset>
<div class="action_message">
<h3>Confirm the Terms & Conditions</h3>
<div class="message">
<p>Please <a href="#">read the terms and conditions</a>.</p>
@Html.CheckBoxFor(m => m.TermsAndConditions) @Html.LabelFor(m => m.TermsAndConditions, "I agree to the terms and conditions")
</div>
</div>
<div class="form_navigation">
<button type="submit" class="btn btn-primary">Signup</button>
</div>
}
Note: for this to work there are a lot of extra files you need to implement.
@using (Html.BeginForm("EditorTemplates", "Comparison", FormMethod.Post, new {id = "signup-form", novalidate = "novalidate"}))
{
<div class="information_message">
<h3>Signup for an account</h3>
<div class="message">
<p>Please fill in your information below to signup for an account.</p>
</div>
</div>
using (Html.BeginSection("Your details"))
{
@Html.EditorFor(m => m.FirstName)
@Html.EditorFor(m => m.LastName)
@Html.EditorFor(m => m.DateOfBirth, new {hint = "DD/MM/YYYY"})
}
using (Html.BeginSection("Account details"))
{
@Html.EditorFor(m => m.EmailAddress, new {hint = "An email will be sent to this address to confirm you own it"})
@Html.EditorFor(m => m.Password)
@Html.EditorFor(m => m.MembershipType)
}
using (Html.BeginSection("Additional details"))
{
@Html.EditorFor(m => m.Bio)
@Html.EditorFor(m => m.Homepage, new {placeholder = "https://"})
}
<div class="action_message">
<h3>Confirm the Terms & Conditions</h3>
<div class="message">
<p>Please <a href="#">read the terms and conditions</a>.</p>
@Html.CheckBoxFor(m => m.TermsAndConditions) @Html.LabelFor(m => m.TermsAndConditions, "I agree to the terms and conditions")
</div>
</div>
<div class="form_navigation">
<button type="submit" class="btn btn-primary">Signup</button>
</div>
}
<chameleon-form attr-id="signup-form">
<form-message type="Information" heading="Signup for an account">
<message-paragraph>Please fill in your information below to signup for an account.</message-paragraph>
</form-message>
<form-section heading="Your details">
<field for="FirstName" />
<field for="LastName" />
<field for="DateOfBirth" hint="DD/MM/YYYY" />
</form-section>
<form-section heading="Account details">
<field for="EmailAddress" hint="An email will be sent to this address to confirm you own it" />
<field for="Password" />
<field for="MembershipType" />
</form-section>
<form-section heading="Additional details">
<field for="Bio" />
<field for="Homepage" placeholder="http://" />
</form-section>
<form-message type="Action" heading="Confirm the Terms & Conditions">
<message-paragraph>Please <a href="#">read the terms and conditions</a></message-paragraph>
<field-element for="TermsAndConditions" inline-label="I agree to the terms and conditions" />
</form-message>
<form-navigation>
<submit-button label="Signup" emphasis-style="Primary" />
</form-navigation>
</chameleon-form>
@using (var f = Html.BeginChameleonForm(htmlAttributes: Html.Attrs().Id("signup-form")))
{
using (var m = f.BeginMessage(MessageType.Information, "Signup for an account"))
{
@m.Paragraph("Please fill in your information below to signup for an account.")
}
using (var s = f.BeginSection("Your details"))
{
@s.FieldFor(m => m.FirstName)
@s.FieldFor(m => m.LastName)
@s.FieldFor(m => m.DateOfBirth).WithHint("DD/MM/YYYY")
}
using (var s = f.BeginSection("Account details"))
{
@s.FieldFor(m => m.EmailAddress).WithHint("An email will be sent to this address to confirm you own it")
@s.FieldFor(m => m.Password)
@s.FieldFor(m => m.MembershipType)
}
using (var s = f.BeginSection("Additional details"))
{
@s.FieldFor(m => m.Bio)
@s.FieldFor(m => m.Homepage).Placeholder("http://")
}
using (var m = f.BeginMessage(MessageType.Action, "Confirm the Terms & Conditions"))
{
@m.Paragraph(@<text>Please <a href="#">read the terms and conditions</a></text>)
@f.FieldElementFor(mm => mm.TermsAndConditions).InlineLabel("I agree to the terms and conditions")
}
using (var n = f.BeginNavigation())
{
@n.Submit("Signup")
}
}
All of the above items will have HTML output that is similar to this. There will be a few small differences between each of the implementations based on the functionality they support e.g. built-in tag helpers outputs enum <option>
's as integer values rather than strings and attribute order will render differently etc. This specific HTML output was taken from the ChameleonForms - tag helpers implementation.
<form action="" enctype="application/x-www-form-urlencoded" id="signup-form" method="post" novalidate="novalidate">
<div class="information_message">
<h3>Signup for an account</h3>
<div class="message">
<p>
Please fill in your information below to signup for an account.
</p>
</div>
</div>
<fieldset>
<legend>Your details</legend>
<dl>
<dt><label for="FirstName">First name</label> <em class="required">*</em></dt>
<dd>
<input data-val="true" data-val-required="The First name field is required." id="FirstName" name="FirstName" required="required" type="text" value="" /> <span class="field-validation-valid" data-valmsg-for="FirstName" data-valmsg-replace="true"></span>
</dd>
<dt><label for="LastName">Last name</label> <em class="required">*</em></dt>
<dd>
<input data-val="true" data-val-required="The Last name field is required." id="LastName" name="LastName" required="required" type="text" value="" /> <span class="field-validation-valid" data-valmsg-for="LastName" data-valmsg-replace="true"></span>
</dd>
<dt><label for="DateOfBirth">Date of birth</label> <em class="required">*</em></dt>
<dd>
<input aria-describedby="DateOfBirth--Hint" data-val="true" data-val-date="The field Date of birth must be a date with format d/M/yyyy." data-val-format="d/M/yyyy" data-val-required="The Date of birth field is required." id="DateOfBirth" name="DateOfBirth" required="required" type="text" value="" />
<div class="hint" id="DateOfBirth--Hint">DD/MM/YYYY</div>
<span class="field-validation-valid" data-valmsg-for="DateOfBirth" data-valmsg-replace="true"></span>
</dd>
</dl>
</fieldset>
<fieldset>
<legend>Account details</legend>
<dl>
<dt><label for="EmailAddress">Email address</label> <em class="required">*</em></dt>
<dd>
<input aria-describedby="EmailAddress--Hint" data-val="true" data-val-email="The Email address field is not a valid e-mail address." data-val-required="The Email address field is required." id="EmailAddress" name="EmailAddress" required="required" type="email" value="" />
<div class="hint" id="EmailAddress--Hint">An email will be sent to this address to confirm you own it</div>
<span class="field-validation-valid" data-valmsg-for="EmailAddress" data-valmsg-replace="true"></span>
</dd>
<dt><label for="Password">Password</label> <em class="required">*</em></dt>
<dd>
<input data-val="true" data-val-required="The Password field is required." id="Password" name="Password" required="required" type="password" /> <span class="field-validation-valid" data-valmsg-for="Password" data-valmsg-replace="true"></span>
</dd>
<dt><label for="MembershipType">Membership type</label> <em class="required">*</em></dt>
<dd>
<select data-val="true" data-val-required="The Membership type field is required." id="MembershipType" name="MembershipType" required="required">
<option selected="selected" value="Standard">Standard</option>
<option value="Bonze">Bonze</option>
<option value="Silver">Silver</option>
<option value="Gold">Gold</option>
<option value="Platinum">Platinum</option>
</select>
<span class="field-validation-valid" data-valmsg-for="MembershipType" data-valmsg-replace="true"></span>
</dd>
</dl>
</fieldset>
<fieldset>
<legend>Additional details</legend>
<dl>
<dt><label for="Bio">Bio</label></dt>
<dd>
<textarea cols="20" id="Bio" name="Bio" rows="2">
</textarea> <span class="field-validation-valid" data-valmsg-for="Bio" data-valmsg-replace="true"></span>
</dd>
<dt><label for="Homepage">Homepage</label></dt>
<dd>
<input id="Homepage" name="Homepage" placeholder="https://" type="url" value="" /> <span class="field-validation-valid" data-valmsg-for="Homepage" data-valmsg-replace="true"></span>
</dd>
</dl>
</fieldset>
<div class="action_message">
<h3>Confirm the Terms & Conditions</h3>
<div class="message">
<p>
Please <a href="#">read the terms and conditions</a>
</p>
<input data-val="true" data-val-required="The Terms and conditions field is required." id="TermsAndConditions" name="TermsAndConditions" required="required" type="checkbox" value="true" /> <label for="TermsAndConditions">I agree to the terms and conditions</label>
</div>
</div>
<div class="form_navigation">
<button class="btn-primary" type="submit">Signup</button>
</div>
</form>
There is an immediately discernible difference between the Tag Helpers and Html Helpers examples as compared to Editor Templates and the ChameleonForms examples, as well as some more subtle differences between the Editor Templates and ChameleonForms.
When you are writing forms with this approach you need to think about the HTML output of the form since you have to write so much of it; in contrast to ChameleonForms and Editor Templates where you are instead concentrating on the structure of the form. If you have consistent HTML for the forms in your site then we think this shift in mindset is an advantage and will lead to better designed forms that are easier to change. This also means that when you need to change the HTML structure of your forms you have a big job ahead of you rewriting large portions of your application (and testing every page thoroughly since it's so easy to make a mistake).
In addition to the above, you can't easily add conventions across your forms based on the type of field or data:
- You have to explicitly specify the type of field you want (e.g.
Html.TextBoxFor
vsHtml.PasswordFor
vsHtml.TextAreaFor
etc.). - You have to manually add the required indicator and aria attributes.
- You have to manually specify the select list items for each drop-down.
You have to also specify the label, field and validation HTML separately and in the right order. It's easy to make copy/paste errors and not remember to update one of the lambda expressions.
Overall, the form is harder to read / more cluttered, harder to maintain, slower to develop and easier to get wrong.
On first glance the syntax of the Editor Templates example and the ChameleonForms example are very similar. There are a number of differences, however:
- The additional ad-hoc per-field configurations (e.g. adding a hint, adding a placeholder, etc.) are added via magic strings (adding them to an anonymous object is still a magic string in our books), which isn't type-safe and you don't get intellisense.
- e.g.
@Html.EditorFor(m => m.SomeField, new { hint = "hint..." })
vs<field for="SomeField" hint="hint..." />
or@s.FieldFor(m => m.SomeField).Hint("hint...")
.
- e.g.
- Furthermore, the additional configurations aren't able to be chained fluently / intellisensed, which we feel makes them much quicker / easier to write and discover.
- e.g.
<field for="SomeField" hint="A hint..." placeholder="http://" add-class="urlfield" />
or@s.FieldFor(m => m.SomeField).Hint("A hint...").Placeholder("http://").AddClass("urlfield")
.
- e.g.
- If you want an enum to show up as a drop-down you have to create an editor template for every enum in your application.
- ChameleonForms automatically outputs an enum field as a dropdown.
- If you want to change from a drop-down to a list of radio buttons for a particular field then you will need to do it inline every time or add extra code to every editor template that you want to do it for (or provide some sort of parent editor template that all of them use).
- You can use
.AsRadioList()
to specify that the field should be a list of radio buttons instead, e.g.@s.FieldFor(m => m.EnumField).AsRadioList()
.
- You can use
- In order to get it working with similar functionality to ChameleonForms for the elements required in this form we had to add 27 files with a total of 252 lines of code (big thanks to Dan Malcolm for converting the standard Editor Templates to Razor, which formed the basis of the code we implemented) - if you wanted to provide test coverage of the form output then you would have even more files and lines of code.
- With ChameleonForms you simply
Install-Package ChameleonForms
using NuGet Package Manager Console and you are good to go with much more functionality.
- With ChameleonForms you simply
The tag helper syntax is immensely more clean and less cluttered than the HTML Helpers syntax and it's nice that you have familiar HTML for the whole form - it's very clear what the output will be at a glance. However, it has it's own issues compared to ChameleonForms:
- You have to specify the different control elements (e.g.
<input>
vs<select>
vs<textarea>
) and you need to know if each of them is self-closing or not. - You have the same problem as HTML Helpers where you have to manually add the required indicator and hint and the required and aria attributes etc. and make sure to use the same
asp-for
value across multiple controls (label vs control vs validation), which is easy to get wrong while copy/pasting. - You have to override the
type
field on theDateTime
to stop the built-in browser behaviour from thedatetime-local
default type that tag the in-built tag helpers add, which doesn't really work. - You have to also specify the label, field and validation HTML separately and in the right order. It's easy to make copy/paste errors.
- You have to specify the boilerplate template structure around each of your form fields, which means you are baking in yourthe current template structure and it's a big job to make a wholesale change in your app.