Formatting, casting, and escaping variable content#
All objects that are being rendered in a template should be a ViewableData instance such as DataObject,
DBField or Controller. From these objects, the template can include any method from the object in scope.
Casting#
The templating system casts variables and the result of method calls into one of the various DBField
classes before outputting them in the final rendered markup. Those DBField classes provide methods that developers can use to format data in
the template (e.g. choosing how to format a date), as well as defining whether HTML markup in that data will be escaped, stripped, or included
directly in the rendered result.
Methods which return data to the template should either return an explicit object instance describing the type of
content that method sends back, or provide a type (using the same format as the $db array described in
Data types and casting) in the $casting array for the object. When rendering that method
to a template, Silverstripe CMS will ensure that the object is wrapped in the correct type. This provides you with all of the methods available in
that class, and ensures and values are safely escaped.
namespace App\Data;
use SilverStripe\View\ViewableData;
class MyTemplatedObject extends ViewableData
{
private static $casting = [
'Header' => 'HTMLText',
];
public function getHeader()
{
return '<h1>This is my header</h1>';
}
}
When calling $MyCustomMethod on the above object from a template, Silverstripe CMS now has the context that this method contains HTML, so it
won't escape the value.
For every field used in templates, a casting helper will be applied. This will first check for any
casting helper on your model specific to that field, and will fall back to the default_cast config
in case none are specified.
NOTE
By default, all content without a type explicitly defined in a $casting array will use the ViewableData.default_cast configuration. By default,
that configuration is set to Text, so HTML characters are escaped.
Common casting types#
Values can be cast as any DBField instance, but these tend to be the most common:
TextWhich is a plain text string, and will be safely encoded viahtmlspecialchars()when placed into a template.Varcharwhich is the same asTextbut for single-line text that should not have line breaks.HTMLFragmentis a block of raw HTML, which should not be escaped. Take care to sanitise any HTML value saved into the database.HTMLTextis aHTMLFragment, but has shortcodes enabled. This should only be used for content that is modified via a TinyMCE editor, which will insert shortcodes.Intfor integers.Decimalfor floating point values.BooleanFor boolean values.Datetimefor date and time.
Formatting#
As has been mentioned in a few sections of documentation already, in a template you have access to all of the public properties and methods of any
object in scope. For instance, if we provide a DBHtmlText instance to the template (either directly
by returning it from a method, or by declaring it in the $db configuration for a DataObject model, or by declaring it in the $casting configuration),
we can use $MyField.FirstParagraph in the template. This will
output the result of the DBHtmlText::FirstParagraph() method to the template.
<%-- app/src/Page.ss --%>
<%-- prints the result of DBHtmlText::FirstParagragh() --%>
$Content.FirstParagraph
<%-- prints the result of DBDatetime::Format("d/m/Y") --%>
$LastEdited.Format("d/m/Y")
Any public method from the object in scope can be called within the template. If that method returns another
ViewableData instance, you can chain the method calls.
<%-- prints the first paragraph of content for the first item in the list --%>
$MyList.First.Content.FirstParagraph
<%-- prints "Copyright 2023" --%>
<p>Copyright {$Now.Year}</p>
<%-- prints <div class="about-us"> --%>
<div class="$URLSegment.LowerCase">
Commonly useful formatting methods#
All DBField instances share the following useful methods for formatting their values:
RAW()- outputs the raw value into the template with no escaping.XML()(and its aliasHTML()) - encodes the value (usinghtmlspecialchars()) before outputting it.CDATA- formats the value safely for insertion as a literal string in an XML file.- e.g.
<element>$Field.CDATA</element>will ensure that the<element>body is safely escaped as a string.
- e.g.
JS()- ensures that text is properly escaped for use in JavaScript.- e.g.
var fieldVal = '$Field.JS';can be used in JavaScript defined in templates to encode values safely.
- e.g.
ATT()(and its aliasHTMLATT()) - formats the value appropriate for an HTML attribute string- e.g.
<div data-my-field="$MyField.HTMLATT"></div>
- e.g.
RAWURLATT()- encodes strings for use in URLs viarawurlencode()URLATT()- encodes strings for use in URLs viaurlencode()JSON()- encodes the value as a JSON string viajson_encode()
NOTE
See the API documentation for all the formatting methods available to you for the various field types.
Escaping HTML values in templates#
WARNING
For specific security advice related to escaping values, see the Security documentation.
The concept of escaping values in templates is ultimately just a combination of formatting and casting.
Values are typically escaped (i.e. the special HTML characters are encoded) in templates by either not
declaring a casting type, or by defaulting to the Text casting type defined on ViewableData.
See the casting section above for instructions on configuring your model to declare casting types for fields, and how some of the more common casting types affect escaping.
NOTE
In addition to escaping via casting, DBField instances have an escape_type configuration property which is
either set to "xml" or "raw". This configuration tells you whether XML content will be escaped or not, but does
not actually directly affect the casting of the value in templates. That is determined by what is returned from
the forTemplate() method (or any method explicitly called from within the template).
Escape methods in templates#
Within the template, fields can have their encoding customised at a certain level with format methods.
See the formatting section above for some of the more common formatting methods available and how they affect escaping.
TIP
If you are unsure of whether the field has been cast to HTMLText but you know
it contains safe HTML content, you can use .RAW to ensure the HTML is not escaped.
But be careful using .RAW on non HTML field types - if the value being formatted includes content provided
by the user you could be introducing attack vectors for cross-site scripting attacks.
Cast summary methods#
Certain subclasses of DBField also have additional summary or manipulations methods, each of which can be chained in order to perform more complicated manipulations.
For instance, The following class methods can be used in templates for the below types:
Text / HTMLText methods:
Plain()Will convert any HTML to plain text version. For example, could be used for plain-text version of emails.LimitSentences(<num>)- limits output to the first<num>sentences in the content. This method internally callsPlain(), converting HTML content into plain text.