Merge pull request #6722 from open-sausages/pulls/4.0/requirements-html-cleanup

Better HTML generation behaviour for Requirements_Backend
This commit is contained in:
Chris Joe 2017-06-16 13:52:06 +12:00 committed by GitHub
commit 102eaed36c
23 changed files with 244 additions and 189 deletions

View File

@ -2,7 +2,9 @@
Name: corehtml Name: corehtml
--- ---
SilverStripe\Core\Injector\Injector: SilverStripe\Core\Injector\Injector:
HTMLValue: SilverStripe\View\Parsers\HTMLValue:
class: SilverStripe\View\Parsers\HTML4Value class: SilverStripe\View\Parsers\HTML4Value
# Shorthand
HTMLValue: '%$SilverStripe\View\Parsers\HTMLValue'
SilverStripe\Forms\HTMLEditor\HTMLEditorConfig: SilverStripe\Forms\HTMLEditor\HTMLEditorConfig:
class: SilverStripe\Forms\HTMLEditor\TinyMCEConfig class: SilverStripe\Forms\HTMLEditor\TinyMCEConfig

View File

@ -1695,6 +1695,9 @@ The following filesystem synchronisation methods and tasks are also removed
* Added method `FormField::setSubmittedValue($value, $data)` to process input submitted from form * Added method `FormField::setSubmittedValue($value, $data)` to process input submitted from form
submission, in contrast to `FormField::setValue($value, $data)` which is intended to load its submission, in contrast to `FormField::setValue($value, $data)` which is intended to load its
value from the ORM. The second argument to setValue() has been added. value from the ORM. The second argument to setValue() has been added.
* `HTMLValue` service name is now fully qualified `SilverStripe\View\Parsers\HTMLValue`.
* FormField::create_tag() has been moved to `HTML` class and renamed to createTag. Invoke with
`HTML::createTag()`.
The following methods and properties on `Requirements_Backend` have been renamed: The following methods and properties on `Requirements_Backend` have been renamed:

View File

@ -315,61 +315,6 @@ class FormField extends RequestHandler
return preg_replace('/([a-z]+)([A-Z])/', '$1 $2', $label); return preg_replace('/([a-z]+)([A-Z])/', '$1 $2', $label);
} }
/**
* Construct and return HTML tag.
*
* @param string $tag
* @param array $attributes
* @param null|string $content
*
* @return string
*/
public static function create_tag($tag, $attributes, $content = null)
{
$preparedAttributes = '';
foreach ($attributes as $attributeKey => $attributeValue) {
if (!empty($attributeValue) || $attributeValue === '0' || ($attributeKey == 'value' && $attributeValue !== null)) {
$preparedAttributes .= sprintf(
' %s="%s"',
$attributeKey,
Convert::raw2att($attributeValue)
);
}
}
if ($content || !in_array($tag, [
'area',
'base',
'br',
'col',
'embed',
'hr',
'img',
'input',
'link',
'meta',
'param',
'source',
'track',
'wbr',
])) {
return sprintf(
'<%s%s>%s</%s>',
$tag,
$preparedAttributes,
$content,
$tag
);
}
return sprintf(
'<%s%s />',
$tag,
$preparedAttributes
);
}
/** /**
* Creates a new field. * Creates a new field.
* *

View File

@ -19,6 +19,7 @@ use SilverStripe\Forms\FormField;
use SilverStripe\Forms\Form; use SilverStripe\Forms\Form;
use LogicException; use LogicException;
use InvalidArgumentException; use InvalidArgumentException;
use SilverStripe\View\HTML;
/** /**
* Displays a {@link SS_List} in a grid format. * Displays a {@link SS_List} in a grid format.
@ -466,9 +467,8 @@ class GridField extends FormField
} }
// Display a message when the grid field is empty. // Display a message when the grid field is empty.
if (empty($content['body'])) { if (empty($content['body'])) {
$cell = FormField::create_tag( $cell = HTML::createTag(
'td', 'td',
array( array(
'colspan' => count($columns), 'colspan' => count($columns),
@ -476,7 +476,7 @@ class GridField extends FormField
_t('SilverStripe\\Forms\\GridField\\GridField.NoItemsFound', 'No items found') _t('SilverStripe\\Forms\\GridField\\GridField.NoItemsFound', 'No items found')
); );
$row = FormField::create_tag( $row = HTML::createTag(
'tr', 'tr',
array( array(
'class' => 'ss-gridfield-item ss-gridfield-no-items', 'class' => 'ss-gridfield-item ss-gridfield-no-items',
@ -518,20 +518,20 @@ class GridField extends FormField
); );
if ($this->getDescription()) { if ($this->getDescription()) {
$content['after'] .= FormField::create_tag( $content['after'] .= HTML::createTag(
'span', 'span',
array('class' => 'description'), array('class' => 'description'),
$this->getDescription() $this->getDescription()
); );
} }
$table = FormField::create_tag( $table = HTML::createTag(
'table', 'table',
$tableAttributes, $tableAttributes,
$header . "\n" . $footer . "\n" . $body $header . "\n" . $footer . "\n" . $body
); );
return FormField::create_tag( return HTML::createTag(
'fieldset', 'fieldset',
$fieldsetAttributes, $fieldsetAttributes,
$content['before'] . $table . $content['after'] $content['before'] . $table . $content['after']
@ -549,7 +549,7 @@ class GridField extends FormField
*/ */
protected function newCell($total, $index, $record, $attributes, $content) protected function newCell($total, $index, $record, $attributes, $content)
{ {
return FormField::create_tag( return HTML::createTag(
'td', 'td',
$attributes, $attributes,
$content $content
@ -567,7 +567,7 @@ class GridField extends FormField
*/ */
protected function newRow($total, $index, $record, $attributes, $content) protected function newRow($total, $index, $record, $attributes, $content)
{ {
return FormField::create_tag( return HTML::createTag(
'tr', 'tr',
$attributes, $attributes,
$content $content
@ -973,9 +973,7 @@ class GridField extends FormField
* *
* @param HTTPRequest $request * @param HTTPRequest $request
* @param DataModel $model * @param DataModel $model
* * @return array|RequestHandler|HTTPResponse|string
* @return array|RequestHandler|HTTPResponse|string|void
*
* @throws HTTPResponse_Exception * @throws HTTPResponse_Exception
*/ */
public function handleRequest(HTTPRequest $request, DataModel $model) public function handleRequest(HTTPRequest $request, DataModel $model)
@ -1091,7 +1089,7 @@ class GridField extends FormField
protected function getOptionalTableHeader(array $content) protected function getOptionalTableHeader(array $content)
{ {
if ($content['header']) { if ($content['header']) {
return FormField::create_tag( return HTML::createTag(
'thead', 'thead',
array(), array(),
$content['header'] $content['header']
@ -1109,7 +1107,7 @@ class GridField extends FormField
protected function getOptionalTableBody(array $content) protected function getOptionalTableBody(array $content)
{ {
if ($content['body']) { if ($content['body']) {
return FormField::create_tag( return HTML::createTag(
'tbody', 'tbody',
array('class' => 'ss-gridfield-items'), array('class' => 'ss-gridfield-items'),
$content['body'] $content['body']
@ -1127,7 +1125,7 @@ class GridField extends FormField
protected function getOptionalTableFooter($content) protected function getOptionalTableFooter($content)
{ {
if ($content['footer']) { if ($content['footer']) {
return FormField::create_tag( return HTML::createTag(
'tfoot', 'tfoot',
array(), array(),
$content['footer'] $content['footer']

View File

@ -3,11 +3,11 @@
namespace SilverStripe\Forms\GridField; namespace SilverStripe\Forms\GridField;
use SilverStripe\Core\Injector\Injectable; use SilverStripe\Core\Injector\Injectable;
use SilverStripe\Forms\FormField;
use SilverStripe\ORM\DataObject; use SilverStripe\ORM\DataObject;
use SilverStripe\ORM\FieldType\DBField; use SilverStripe\ORM\FieldType\DBField;
use SilverStripe\ORM\Hierarchy\Hierarchy; use SilverStripe\ORM\Hierarchy\Hierarchy;
use SilverStripe\View\ArrayData; use SilverStripe\View\ArrayData;
use SilverStripe\View\HTML;
use SilverStripe\View\SSViewer; use SilverStripe\View\SSViewer;
/** /**
@ -79,7 +79,7 @@ class GridFieldLevelup implements GridField_HTMLProvider
'href' => sprintf($this->linkSpec, $parentID), 'href' => sprintf($this->linkSpec, $parentID),
'class' => 'cms-panel-link ss-ui-button font-icon-level-up no-text grid-levelup' 'class' => 'cms-panel-link ss-ui-button font-icon-level-up no-text grid-levelup'
)); ));
$linkTag = FormField::create_tag('a', $attrs); $linkTag = HTML::createTag('a', $attrs);
$forTemplate = new ArrayData(array( $forTemplate = new ArrayData(array(
'UpLink' => DBField::create_field('HTMLFragment', $linkTag) 'UpLink' => DBField::create_field('HTMLFragment', $linkTag)

View File

@ -4,7 +4,7 @@ namespace SilverStripe\Forms\HtmlEditor;
use SilverStripe\Core\Convert; use SilverStripe\Core\Convert;
use SilverStripe\Core\Injector\Injector; use SilverStripe\Core\Injector\Injector;
use SilverStripe\Forms\FormField; use SilverStripe\View\HTML;
use SilverStripe\View\Parsers\ShortcodeHandler; use SilverStripe\View\Parsers\ShortcodeHandler;
use Embed\Adapters\Adapter; use Embed\Adapters\Adapter;
use Embed\Embed; use Embed\Embed;
@ -132,7 +132,7 @@ class EmbedShortcodeProvider implements ShortcodeHandler
unset($arguments['height']); unset($arguments['height']);
unset($arguments['url']); unset($arguments['url']);
unset($arguments['caption']); unset($arguments['caption']);
return FormField::create_tag('div', $arguments, $content); return HTML::createTag('div', $arguments, $content);
} }
/** /**
@ -151,7 +151,7 @@ class EmbedShortcodeProvider implements ShortcodeHandler
unset($arguments['height']); unset($arguments['height']);
unset($arguments['url']); unset($arguments['url']);
$arguments['href'] = $href; $arguments['href'] = $href;
return Formfield::create_tag('a', $arguments, Convert::raw2xml($title)); return HTML::createTag('a', $arguments, Convert::raw2xml($title));
} }
/** /**
@ -165,6 +165,6 @@ class EmbedShortcodeProvider implements ShortcodeHandler
{ {
$arguments['src'] = $src; $arguments['src'] = $src;
unset($arguments['url']); unset($arguments['url']);
return FormField::create_tag('img', $arguments); return HTML::createTag('img', $arguments);
} }
} }

View File

@ -3,11 +3,11 @@
namespace SilverStripe\Forms\HTMLEditor; namespace SilverStripe\Forms\HTMLEditor;
use SilverStripe\Assets\Image; use SilverStripe\Assets\Image;
use SilverStripe\Core\Injector\Injector;
use SilverStripe\Forms\TextareaField; use SilverStripe\Forms\TextareaField;
use SilverStripe\ORM\DataObject; use SilverStripe\ORM\DataObject;
use SilverStripe\ORM\DataObjectInterface; use SilverStripe\ORM\DataObjectInterface;
use Exception; use Exception;
use SilverStripe\View\Parsers\HTMLValue;
/** /**
* A TinyMCE-powered WYSIWYG HTML editor field with image and link insertion and tracking capabilities. Editor fields * A TinyMCE-powered WYSIWYG HTML editor field with image and link insertion and tracking capabilities. Editor fields
@ -129,7 +129,7 @@ class HTMLEditorField extends TextareaField
} }
// Sanitise if requested // Sanitise if requested
$htmlValue = Injector::inst()->create('HTMLValue', $this->Value()); $htmlValue = HTMLValue::create($this->Value());
if (HTMLEditorField::config()->sanitise_server_side) { if (HTMLEditorField::config()->sanitise_server_side) {
$santiser = HTMLEditorSanitiser::create(HTMLEditorConfig::get_active()); $santiser = HTMLEditorSanitiser::create(HTMLEditorConfig::get_active());
$santiser->sanitise($htmlValue); $santiser->sanitise($htmlValue);

View File

@ -4,7 +4,7 @@ namespace SilverStripe\Forms;
use SilverStripe\ORM\ArrayList; use SilverStripe\ORM\ArrayList;
use SilverStripe\ORM\FieldType\DBField; use SilverStripe\ORM\FieldType\DBField;
use SilverStripe\View\Requirements; use SilverStripe\View\HTML;
/** /**
* Represents a number of fields which are selectable by a radio * Represents a number of fields which are selectable by a radio
@ -92,7 +92,7 @@ class SelectionGroup extends CompositeField
$itemID = $this->ID() . '_' . (++$count); $itemID = $this->ID() . '_' . (++$count);
// @todo Move into SelectionGroup_Item.ss template at some point. // @todo Move into SelectionGroup_Item.ss template at some point.
$extra = array( $extra = array(
"RadioButton" => DBField::create_field('HTMLFragment', FormField::create_tag( "RadioButton" => DBField::create_field('HTMLFragment', HTML::createTag(
'input', 'input',
array( array(
'class' => 'selector', 'class' => 'selector',

View File

@ -7,6 +7,7 @@ use SilverStripe\Core\Injector\Injector;
use SilverStripe\Control\HTTP; use SilverStripe\Control\HTTP;
use SilverStripe\Forms\TextField; use SilverStripe\Forms\TextField;
use SilverStripe\Forms\HTMLEditor\HTMLEditorField; use SilverStripe\Forms\HTMLEditor\HTMLEditorField;
use SilverStripe\View\Parsers\HTMLValue;
use SilverStripe\View\Parsers\ShortcodeParser; use SilverStripe\View\Parsers\ShortcodeParser;
/** /**
@ -177,7 +178,7 @@ class DBHTMLText extends DBText
public function whitelistContent($value) public function whitelistContent($value)
{ {
if ($this->whitelist) { if ($this->whitelist) {
$dom = Injector::inst()->create('HTMLValue', $value); $dom = HTMLValue::create($value);
$query = array(); $query = array();
$textFilter = ' | //body/text()'; $textFilter = ' | //body/text()';

78
src/View/HTML.php Normal file
View File

@ -0,0 +1,78 @@
<?php
namespace SilverStripe\View;
use InvalidArgumentException;
use SilverStripe\Core\Config\Configurable;
use SilverStripe\Core\Convert;
/**
* HTML Helper class
*/
class HTML
{
use Configurable;
/**
* List of HTML5 void elements
*
* @see https://www.w3.org/TR/html51/syntax.html#void-elements
* @config
* @var array
*/
private static $void_elements = [
'area',
'base',
'br',
'col',
'embed',
'hr',
'img',
'input',
'keygen',
'link',
'menuitem',
'meta',
'param',
'source',
'track',
'wbr'
];
/**
* Construct and return HTML tag.
*
* @param string $tag
* @param array $attributes
* @param string $content Content to use between two tags. Not valid for void elements (e.g. link)
* @return string
*/
public static function createTag($tag, $attributes, $content = null)
{
$tag = strtolower($tag);
// Build list of arguments
$preparedAttributes = '';
foreach ($attributes as $attributeKey => $attributeValue) {
// Only set non-empty strings (ensures strlen(0) > 0)
if (strlen($attributeValue) > 0) {
$preparedAttributes .= sprintf(
' %s="%s"',
$attributeKey,
Convert::raw2att($attributeValue)
);
}
}
// Check void element type
if (in_array($tag, static::config()->get('void_elements'))) {
if ($content) {
throw new InvalidArgumentException("Void element \"{$tag}\" cannot have content");
}
return "<{$tag}{$preparedAttributes} />";
}
// Closed tag type
return "<{$tag}{$preparedAttributes}>{$content}</{$tag}>";
}
}

View File

@ -42,7 +42,7 @@ class Diff extends \Diff
$content = $cleaner->cleanHTML($content); $content = $cleaner->cleanHTML($content);
} else { } else {
// At most basic level of cleaning, use DOMDocument to save valid XML. // At most basic level of cleaning, use DOMDocument to save valid XML.
$doc = Injector::inst()->create('HTMLValue', $content); $doc = HTMLValue::create($content);
$content = $doc->getContent(); $content = $doc->getContent();
} }

View File

@ -2,7 +2,6 @@
namespace SilverStripe\View\Parsers; namespace SilverStripe\View\Parsers;
use SilverStripe\Core\Injector\Injector;
use HTMLPurifier; use HTMLPurifier;
/** /**
@ -14,7 +13,7 @@ class PurifierHTMLCleaner extends HTMLCleaner
public function cleanHTML($content) public function cleanHTML($content)
{ {
$html = new HTMLPurifier(); $html = new HTMLPurifier();
$doc = Injector::inst()->create('HTMLValue', $html->purify($content)); $doc = HTMLValue::create($html->purify($content));
return $doc->getContent(); return $doc->getContent();
} }
} }

View File

@ -624,7 +624,7 @@ class ShortcodeParser
if ($content) { if ($content) {
/** @var HTMLValue $parsed */ /** @var HTMLValue $parsed */
$parsed = Injector::inst()->create('HTMLValue', $content); $parsed = HTMLValue::create($content);
$body = $parsed->getBody(); $body = $parsed->getBody();
if ($body) { if ($body) {
$this->insertListAfter($body->childNodes, $node); $this->insertListAfter($body->childNodes, $node);
@ -669,7 +669,7 @@ class ShortcodeParser
// Now parse the result into a DOM // Now parse the result into a DOM
if (!$htmlvalue->isValid()) { if (!$htmlvalue->isValid()) {
if (self::$error_behavior == self::ERROR) { if (self::$error_behavior == self::ERROR) {
user_error('Couldn\'t decode HTML when processing short codes', E_USER_ERRROR); user_error('Couldn\'t decode HTML when processing short codes', E_USER_ERROR);
} else { } else {
$continue = false; $continue = false;
} }

View File

@ -778,38 +778,55 @@ class Requirements_Backend
// Combine files - updates $this->javascript and $this->css // Combine files - updates $this->javascript and $this->css
$this->processCombinedFiles(); $this->processCombinedFiles();
// Script tags for js links
foreach ($this->getJavascript() as $file => $attributes) { foreach ($this->getJavascript() as $file => $attributes) {
$async = (isset($attributes['async']) && $attributes['async'] == true) ? " async" : ""; // Build html attributes
$defer = (isset($attributes['defer']) && $attributes['defer'] == true) ? " defer" : ""; $htmlAttributes = [
$type = Convert::raw2att(isset($attributes['type']) ? $attributes['type'] : "application/javascript"); 'type' => isset($attributes['type']) ? $attributes['type'] : "application/javascript",
$path = Convert::raw2att($this->pathForFile($file)); 'src' => $this->pathForFile($file),
if ($path) { ];
$jsRequirements .= "<script type=\"{$type}\" src=\"{$path}\"{$async}{$defer}></script>"; if (!empty($attributes['async'])) {
$htmlAttributes['async'] = 'async';
} }
if (!empty($attributes['defer'])) {
$htmlAttributes['defer'] = 'defer';
}
$jsRequirements .= HTML::createTag('script', $htmlAttributes);
$jsRequirements .= "\n";
} }
// Add all inline JavaScript *after* including external files they might rely on // Add all inline JavaScript *after* including external files they might rely on
foreach ($this->getCustomScripts() as $script) { foreach ($this->getCustomScripts() as $script) {
$jsRequirements .= "<script type=\"application/javascript\">//<![CDATA[\n"; $jsRequirements .= HTML::createTag(
$jsRequirements .= "$script\n"; 'script',
$jsRequirements .= "//]]></script>"; [ 'type' => 'application/javascript' ],
"//<![CDATA[\n{$script}\n//]]>"
);
$jsRequirements .= "\n";
} }
// CSS file links
foreach ($this->getCSS() as $file => $params) { foreach ($this->getCSS() as $file => $params) {
$path = Convert::raw2att($this->pathForFile($file)); $htmlAttributes = [
if ($path) { 'rel' => 'stylesheet',
$media = (isset($params['media']) && !empty($params['media'])) 'type' => 'text/css',
? " media=\"{$params['media']}\"" : ""; 'href' => $this->pathForFile($file),
$requirements .= "<link rel=\"stylesheet\" type=\"text/css\" {$media} href=\"$path\" />\n"; ];
if (!empty($params['media'])) {
$htmlAttributes['media'] = $params['media'];
} }
$requirements .= HTML::createTag('link', $htmlAttributes);
$requirements .= "\n";
} }
// Literal custom CSS content
foreach ($this->getCustomCSS() as $css) { foreach ($this->getCustomCSS() as $css) {
$requirements .= "<style type=\"text/css\">\n$css\n</style>\n"; $requirements .= HTML::createTag('style', ['type' => 'text/css'], "\n{$css}\n");
$requirements .= "\n";
} }
foreach ($this->getCustomHeadTags() as $customHeadTag) { foreach ($this->getCustomHeadTags() as $customHeadTag) {
$requirements .= "$customHeadTag\n"; $requirements .= "{$customHeadTag}\n";
} }
// Inject CSS into body // Inject CSS into body
@ -971,8 +988,8 @@ class Requirements_Backend
$candidates = array( $candidates = array(
'en.js', 'en.js',
'en_US.js', 'en_US.js',
i18n::getData()->langFromLocale(i18n::config()->default_locale) . '.js', i18n::getData()->langFromLocale(i18n::config()->get('default_locale')) . '.js',
i18n::config()->default_locale . '.js', i18n::config()->get('default_locale') . '.js',
i18n::getData()->langFromLocale(i18n::get_locale()) . '.js', i18n::getData()->langFromLocale(i18n::get_locale()) . '.js',
i18n::get_locale() . '.js', i18n::get_locale() . '.js',
); );

View File

@ -1,4 +1,4 @@
# features/login.feature @retry
Feature: Log in Feature: Log in
As an site owner As an site owner
I want to access to the CMS to be secure I want to access to the CMS to be secure

View File

@ -1,4 +1,4 @@
@todo @retry
Feature: Lost Password Feature: Lost Password
As a site owner As a site owner
I want to be able to reset my password I want to be able to reset my password

View File

@ -1,4 +1,4 @@
@javascript @javascript @retry
Feature: Manage users Feature: Manage users
As a site administrator As a site administrator
I want to create and manage user accounts on my site I want to create and manage user accounts on my site

View File

@ -1,3 +1,4 @@
@retry
Feature: Manage my own settings Feature: Manage my own settings
As a CMS user As a CMS user
I want to be able to change personal settings I want to be able to change personal settings

View File

@ -1,4 +1,4 @@
@javascript @javascript @retry
Feature: Manage Security Permissions for Groups Feature: Manage Security Permissions for Groups
As a site administrator As a site administrator
I want to control my user's security permissions in an intuitive way I want to control my user's security permissions in an intuitive way

View File

@ -5,7 +5,6 @@ namespace SilverStripe\Forms\Tests;
use SilverStripe\Core\Config\Config; use SilverStripe\Core\Config\Config;
use SilverStripe\Core\ClassInfo; use SilverStripe\Core\ClassInfo;
use SilverStripe\Dev\SapphireTest; use SilverStripe\Dev\SapphireTest;
use SilverStripe\Control\Controller;
use SilverStripe\Forms\FormField; use SilverStripe\Forms\FormField;
use SilverStripe\Forms\Tests\FormFieldTest\TestExtension; use SilverStripe\Forms\Tests\FormFieldTest\TestExtension;
use SilverStripe\Forms\TextField; use SilverStripe\Forms\TextField;
@ -13,7 +12,6 @@ use SilverStripe\Forms\RequiredFields;
use SilverStripe\Forms\FieldList; use SilverStripe\Forms\FieldList;
use SilverStripe\Forms\Form; use SilverStripe\Forms\Form;
use ReflectionClass; use ReflectionClass;
use SilverStripe\ORM\FieldType\DBField;
class FormFieldTest extends SapphireTest class FormFieldTest extends SapphireTest
{ {
@ -371,14 +369,4 @@ class FormFieldTest extends SapphireTest
$schema['message']['value'] $schema['message']['value']
); );
} }
public function testCreateVoidTag()
{
$tag = FormField::create_tag('meta', [
'name' => 'description',
'content' => 'test tag',
]);
$this->assertNotContains('</meta>', $tag);
$this->assertRegexp('#/>$#', $tag);
}
} }

View File

@ -6,6 +6,7 @@ use SilverStripe\Core\Injector\Injector;
use SilverStripe\Dev\FunctionalTest; use SilverStripe\Dev\FunctionalTest;
use SilverStripe\Forms\HTMLEditor\HTMLEditorConfig; use SilverStripe\Forms\HTMLEditor\HTMLEditorConfig;
use SilverStripe\Forms\HTMLEditor\HTMLEditorSanitiser; use SilverStripe\Forms\HTMLEditor\HTMLEditorSanitiser;
use SilverStripe\View\Parsers\HTMLValue;
class HTMLEditorSanitiserTest extends FunctionalTest class HTMLEditorSanitiserTest extends FunctionalTest
{ {
@ -53,7 +54,7 @@ class HTMLEditorSanitiserTest extends FunctionalTest
$config->setOptions(array('valid_elements' => $validElements)); $config->setOptions(array('valid_elements' => $validElements));
$sanitiser = new HtmlEditorSanitiser($config); $sanitiser = new HtmlEditorSanitiser($config);
$htmlValue = Injector::inst()->create('HTMLValue', $input); $htmlValue = HTMLValue::create($input);
$sanitiser->sanitise($htmlValue); $sanitiser->sanitise($htmlValue);
$this->assertEquals($output, $htmlValue->getContent(), $desc); $this->assertEquals($output, $htmlValue->getContent(), $desc);

View File

@ -0,0 +1,58 @@
<?php
namespace SilverStripe\View\Tests;
use InvalidArgumentException;
use SilverStripe\Dev\SapphireTest;
use SilverStripe\View\HTML;
class HTMLTest extends SapphireTest
{
public function testCreateVoidTag()
{
$tag = HTML::createTag('meta', [
'name' => 'description',
'content' => 'test tag',
]);
$this->assertEquals('<meta name="description" content="test tag" />', $tag);
}
public function testEmptyAttributes()
{
$tag = HTML::createTag('meta', [
'value' => 0,
'content' => '',
'max' => 3,
'details' => null,
'disabled' => false,
'readonly' => true,
]);
$this->assertEquals('<meta value="0" max="3" readonly="1" />', $tag);
}
public function testNormalTag()
{
$tag = HTML::createTag('a', [
'title' => 'Some link',
'nullattr' => null,
]);
$this->assertEquals('<a title="Some link"></a>', $tag);
$tag = HTML::createTag('a', [
'title' => 'HTML & Text',
'nullattr' => null,
], 'Some <strong>content!</strong>');
$this->assertEquals('<a title="HTML &amp; Text">Some <strong>content!</strong></a>', $tag);
}
public function testVoidContentError()
{
$this->expectException(InvalidArgumentException::class);
$this->expectExceptionMessage("Void element \"link\" cannot have content");
HTML::createTag('link', [
'title' => 'HTML & Text',
'nullattr' => null,
], 'Some <strong>content!</strong>');
}
}

View File

@ -2,6 +2,7 @@
namespace SilverStripe\View\Tests; namespace SilverStripe\View\Tests;
use InvalidArgumentException;
use SilverStripe\Control\Controller; use SilverStripe\Control\Controller;
use SilverStripe\Core\Injector\Injector; use SilverStripe\Core\Injector\Injector;
use SilverStripe\Dev\SapphireTest; use SilverStripe\Dev\SapphireTest;
@ -34,9 +35,7 @@ class RequirementsTest extends SapphireTest
public function testExternalUrls() public function testExternalUrls()
{ {
/** /** @var Requirements_Backend $backend */
* @var Requirements_Backend $backend
*/
$backend = Injector::inst()->create(Requirements_Backend::class); $backend = Injector::inst()->create(Requirements_Backend::class);
$backend->setCombinedFilesEnabled(true); $backend->setCombinedFilesEnabled(true);
@ -166,9 +165,7 @@ class RequirementsTest extends SapphireTest
public function testCustomType() public function testCustomType()
{ {
/** /** @var Requirements_Backend $backend */
* @var Requirements_Backend $backend
*/
$backend = Injector::inst()->create(Requirements_Backend::class); $backend = Injector::inst()->create(Requirements_Backend::class);
$basePath = $this->getThemeRoot(); $basePath = $this->getThemeRoot();
$this->setupRequirements($backend); $this->setupRequirements($backend);
@ -194,9 +191,7 @@ class RequirementsTest extends SapphireTest
public function testCombinedJavascript() public function testCombinedJavascript()
{ {
/** /** @var Requirements_Backend $backend */
* @var Requirements_Backend $backend
*/
$backend = Injector::inst()->create(Requirements_Backend::class); $backend = Injector::inst()->create(Requirements_Backend::class);
$this->setupCombinedRequirements($backend); $this->setupCombinedRequirements($backend);
@ -249,9 +244,7 @@ class RequirementsTest extends SapphireTest
// Then do it again, this time not requiring the files beforehand // Then do it again, this time not requiring the files beforehand
unlink($combinedFilePath); unlink($combinedFilePath);
/** /** @var Requirements_Backend $backend */
* @var Requirements_Backend $backend
*/
$backend = Injector::inst()->create(Requirements_Backend::class); $backend = Injector::inst()->create(Requirements_Backend::class);
$this->setupCombinedNonrequiredRequirements($backend); $this->setupCombinedNonrequiredRequirements($backend);
$html = $backend->includeInHTML(self::$html_template); $html = $backend->includeInHTML(self::$html_template);
@ -294,9 +287,7 @@ class RequirementsTest extends SapphireTest
public function testCombinedJavascriptAsyncDefer() public function testCombinedJavascriptAsyncDefer()
{ {
/** /** @var Requirements_Backend $backend */
* @var Requirements_Backend $backend
*/
$backend = Injector::inst()->create(Requirements_Backend::class); $backend = Injector::inst()->create(Requirements_Backend::class);
$this->setupCombinedRequirementsJavascriptAsyncDefer($backend, true, false); $this->setupCombinedRequirementsJavascriptAsyncDefer($backend, true, false);
@ -371,9 +362,7 @@ class RequirementsTest extends SapphireTest
// setup again for testing defer // setup again for testing defer
unlink($combinedFilePath); unlink($combinedFilePath);
/** /** @var Requirements_Backend $backend */
* @var Requirements_Backend $backend
*/
$backend = Injector::inst()->create(Requirements_Backend::class); $backend = Injector::inst()->create(Requirements_Backend::class);
$this->setupCombinedRequirementsJavascriptAsyncDefer($backend, false, true); $this->setupCombinedRequirementsJavascriptAsyncDefer($backend, false, true);
@ -445,9 +434,7 @@ class RequirementsTest extends SapphireTest
// setup again for testing async and defer // setup again for testing async and defer
unlink($combinedFilePath); unlink($combinedFilePath);
/** /** @var Requirements_Backend $backend */
* @var Requirements_Backend $backend
*/
$backend = Injector::inst()->create(Requirements_Backend::class); $backend = Injector::inst()->create(Requirements_Backend::class);
$this->setupCombinedRequirementsJavascriptAsyncDefer($backend, true, true); $this->setupCombinedRequirementsJavascriptAsyncDefer($backend, true, true);
@ -456,7 +443,7 @@ class RequirementsTest extends SapphireTest
/* ASYNC/DEFER IS INCLUDED IN SCRIPT TAG */ /* ASYNC/DEFER IS INCLUDED IN SCRIPT TAG */
$this->assertRegExp( $this->assertRegExp(
'/src=".*' . preg_quote($combinedFileName, '/') . '" async defer/', '/src=".*' . preg_quote($combinedFileName, '/') . '" async="async" defer="defer"/',
$html, $html,
'async and defer are included in script tag' 'async and defer are included in script tag'
); );
@ -520,9 +507,7 @@ class RequirementsTest extends SapphireTest
public function testCombinedCss() public function testCombinedCss()
{ {
$basePath = $this->getThemeRoot(); $basePath = $this->getThemeRoot();
/** /** @var Requirements_Backend $backend */
* @var Requirements_Backend $backend
*/
$backend = Injector::inst()->create(Requirements_Backend::class); $backend = Injector::inst()->create(Requirements_Backend::class);
$this->setupRequirements($backend); $this->setupRequirements($backend);
@ -551,9 +536,7 @@ class RequirementsTest extends SapphireTest
); );
// Test that combining a file multiple times doesn't trigger an error // Test that combining a file multiple times doesn't trigger an error
/** /** @var Requirements_Backend $backend */
* @var Requirements_Backend $backend
*/
$backend = Injector::inst()->create(Requirements_Backend::class); $backend = Injector::inst()->create(Requirements_Backend::class);
$this->setupRequirements($backend); $this->setupRequirements($backend);
$backend->combineFiles( $backend->combineFiles(
@ -582,9 +565,7 @@ class RequirementsTest extends SapphireTest
public function testBlockedCombinedJavascript() public function testBlockedCombinedJavascript()
{ {
$basePath = $this->getThemeRoot(); $basePath = $this->getThemeRoot();
/** /** @var Requirements_Backend $backend */
* @var Requirements_Backend $backend
*/
$backend = Injector::inst()->create(Requirements_Backend::class); $backend = Injector::inst()->create(Requirements_Backend::class);
$this->setupCombinedRequirements($backend); $this->setupCombinedRequirements($backend);
$combinedFileName = '/_combinedfiles/RequirementsTest_bc-2a55d56.js'; $combinedFileName = '/_combinedfiles/RequirementsTest_bc-2a55d56.js';
@ -622,14 +603,12 @@ class RequirementsTest extends SapphireTest
clearstatcache(); // needed to get accurate file_exists() results clearstatcache(); // needed to get accurate file_exists() results
// Exception generated from including invalid file // Exception generated from including invalid file
$this->setExpectedException( $this->expectException(InvalidArgumentException::class);
'InvalidArgumentException', $this->expectExceptionMessage(sprintf(
sprintf( "Requirements_Backend::combine_files(): Already included file(s) %s in combined file '%s'",
"Requirements_Backend::combine_files(): Already included file(s) %s in combined file '%s'", $basePath . '/javascript/RequirementsTest_c.js',
$basePath . '/javascript/RequirementsTest_c.js', 'RequirementsTest_bc.js'
'RequirementsTest_bc.js' ));
)
);
$backend->combineFiles( $backend->combineFiles(
'RequirementsTest_ac.js', 'RequirementsTest_ac.js',
array( array(
@ -643,9 +622,7 @@ class RequirementsTest extends SapphireTest
{ {
$basePath = $this->getThemeRoot(); $basePath = $this->getThemeRoot();
/** /** @var Requirements_Backend $backend */
* @var Requirements_Backend $backend
*/
$backend = Injector::inst()->create(Requirements_Backend::class); $backend = Injector::inst()->create(Requirements_Backend::class);
$this->setupRequirements($backend); $this->setupRequirements($backend);
@ -672,9 +649,7 @@ class RequirementsTest extends SapphireTest
{ {
$basePath = $this->getThemeRoot(); $basePath = $this->getThemeRoot();
/** /** @var Requirements_Backend $backend */
* @var Requirements_Backend $backend
*/
$backend = Injector::inst()->create(Requirements_Backend::class); $backend = Injector::inst()->create(Requirements_Backend::class);
$this->setupRequirements($backend); $this->setupRequirements($backend);
$backend->javascript($basePath . '/a.js'); $backend->javascript($basePath . '/a.js');
@ -733,9 +708,7 @@ class RequirementsTest extends SapphireTest
{ {
$testPath = $this->getThemeRoot(); $testPath = $this->getThemeRoot();
/** /** @var Requirements_Backend $backend */
* @var Requirements_Backend $backend
*/
$backend = Injector::inst()->create(Requirements_Backend::class); $backend = Injector::inst()->create(Requirements_Backend::class);
$this->setupRequirements($backend); $this->setupRequirements($backend);
$holder = Requirements::backend(); $holder = Requirements::backend();
@ -796,9 +769,7 @@ class RequirementsTest extends SapphireTest
public function testJsWriteToBody() public function testJsWriteToBody()
{ {
/** /** @var Requirements_Backend $backend */
* @var Requirements_Backend $backend
*/
$backend = Injector::inst()->create(Requirements_Backend::class); $backend = Injector::inst()->create(Requirements_Backend::class);
$this->setupRequirements($backend); $this->setupRequirements($backend);
$backend->javascript('http://www.mydomain.com/test.js'); $backend->javascript('http://www.mydomain.com/test.js');
@ -813,15 +784,13 @@ class RequirementsTest extends SapphireTest
$backend->setWriteJavascriptToBody(true); $backend->setWriteJavascriptToBody(true);
$html = $backend->includeInHTML($template); $html = $backend->includeInHTML($template);
$this->assertNotContains('<head><script', $html); $this->assertNotContains('<head><script', $html);
$this->assertContains('</script></body>', $html); $this->assertContains("</script>\n</body>", $html);
} }
public function testIncludedJsIsNotCommentedOut() public function testIncludedJsIsNotCommentedOut()
{ {
$template = '<html><head></head><body><!--<script>alert("commented out");</script>--></body></html>'; $template = '<html><head></head><body><!--<script>alert("commented out");</script>--></body></html>';
/** /** @var Requirements_Backend $backend */
* @var Requirements_Backend $backend
*/
$backend = Injector::inst()->create(Requirements_Backend::class); $backend = Injector::inst()->create(Requirements_Backend::class);
$this->setupRequirements($backend); $this->setupRequirements($backend);
$backend->javascript($this->getThemeRoot() . '/javascript/RequirementsTest_a.js'); $backend->javascript($this->getThemeRoot() . '/javascript/RequirementsTest_a.js');
@ -847,16 +816,15 @@ class RequirementsTest extends SapphireTest
$html = $backend->includeInHTML($template); $html = $backend->includeInHTML($template);
$this->assertEquals( $this->assertEquals(
'<html><head></head><body><!--<script>alert("commented out");</script>-->' '<html><head></head><body><!--<script>alert("commented out");</script>-->'
. '<h1>more content</h1><script type="application/javascript" src="' . $urlSrc . '"></script></body></html>', . '<h1>more content</h1><script type="application/javascript" src="' . $urlSrc
. "\"></script>\n</body></html>",
$html $html
); );
} }
public function testForceJsToBottom() public function testForceJsToBottom()
{ {
/** /** @var Requirements_Backend $backend */
* @var Requirements_Backend $backend
*/
$backend = Injector::inst()->create(Requirements_Backend::class); $backend = Injector::inst()->create(Requirements_Backend::class);
$this->setupRequirements($backend); $this->setupRequirements($backend);
$backend->javascript('http://www.mydomain.com/test.js'); $backend->javascript('http://www.mydomain.com/test.js');
@ -872,10 +840,10 @@ EOS
$template = '<html><head></head><body><header>My header</header><p>Body<script></script></p></body></html>'; $template = '<html><head></head><body><header>My header</header><p>Body<script></script></p></body></html>';
// The expected outputs // The expected outputs
$expectedScripts = "<script type=\"application/javascript\" src=\"http://www.mydomain.com/test.js\">" $expectedScripts = "<script type=\"application/javascript\" src=\"http://www.mydomain.com/test.js\"></script>\n"
. "</script><script type=\"application/javascript\">//<![CDATA[\n" . "<script type=\"application/javascript\">//<![CDATA[\n"
. "var globalvar = {\n\tpattern: '\\\\\$custom\\\\1'\n};\n" . "var globalvar = {\n\tpattern: '\\\\\$custom\\\\1'\n};\n"
. "//]]></script>"; . "//]]></script>\n";
$JsInHead = "<html><head>$expectedScripts</head><body><header>My header</header><p>Body<script></script></p></body></html>"; $JsInHead = "<html><head>$expectedScripts</head><body><header>My header</header><p>Body<script></script></p></body></html>";
$JsInBody = "<html><head></head><body><header>My header</header><p>Body$expectedScripts<script></script></p></body></html>"; $JsInBody = "<html><head></head><body><header>My header</header><p>Body$expectedScripts<script></script></p></body></html>";
$JsAtEnd = "<html><head></head><body><header>My header</header><p>Body<script></script></p>$expectedScripts</body></html>"; $JsAtEnd = "<html><head></head><body><header>My header</header><p>Body<script></script></p>$expectedScripts</body></html>";
@ -922,9 +890,7 @@ EOS
$template = '<html><head></head><body><header>My header</header><p>Body</p></body></html>'; $template = '<html><head></head><body><header>My header</header><p>Body</p></body></html>';
$basePath = $this->getThemeRoot(); $basePath = $this->getThemeRoot();
/** /** @var Requirements_Backend $backend */
* @var Requirements_Backend $backend
*/
$backend = Injector::inst()->create(Requirements_Backend::class); $backend = Injector::inst()->create(Requirements_Backend::class);
$this->setupRequirements($backend); $this->setupRequirements($backend);
@ -954,9 +920,7 @@ EOS
*/ */
public function testProvidedFiles() public function testProvidedFiles()
{ {
/** /** @var Requirements_Backend $backend */
* @var Requirements_Backend $backend
*/
$template = '<html><head></head><body><header>My header</header><p>Body</p></body></html>'; $template = '<html><head></head><body><header>My header</header><p>Body</p></body></html>';
$basePath = $this->getThemeRoot(); $basePath = $this->getThemeRoot();