mirror of
https://github.com/silverstripe/silverstripe-framework
synced 2024-10-22 14:05:37 +02:00
API: Pass extra context information to shortcode handlers.
This allows shortcodes to perform more complex actions on the element which contains them. For example, the element reference can be used to add extra classes or attributes to links which provide additional metadata.
This commit is contained in:
parent
f0ccdeb9fc
commit
a339687493
@ -64,6 +64,9 @@ These parameters are passed to the callback:
|
|||||||
will not have been parsed, and can optionally be fed back into the parser.
|
will not have been parsed, and can optionally be fed back into the parser.
|
||||||
- The ShortcodeParser instance used to parse the content.
|
- The ShortcodeParser instance used to parse the content.
|
||||||
- The shortcode tag name that was matched within the parsed content.
|
- The shortcode tag name that was matched within the parsed content.
|
||||||
|
- An associative array of extra information about the shortcode being parsed. For example, if the shortcode is
|
||||||
|
is inside an attribute, the `element` key contains a reference to the parent `DOMElement`, and the `node`
|
||||||
|
key the attribute's `DOMNode`.
|
||||||
|
|
||||||
## Example: Google Maps Iframe by Address
|
## Example: Google Maps Iframe by Address
|
||||||
|
|
||||||
|
@ -68,6 +68,7 @@ class ShortcodeParser {
|
|||||||
* this will not have been parsed, and can optionally be fed back into the parser.
|
* this will not have been parsed, and can optionally be fed back into the parser.
|
||||||
* - The {@link ShortcodeParser} instance used to parse the content.
|
* - The {@link ShortcodeParser} instance used to parse the content.
|
||||||
* - The shortcode tag name that was matched within the parsed content.
|
* - The shortcode tag name that was matched within the parsed content.
|
||||||
|
* - An associative array of extra information about the shortcode being parsed.
|
||||||
*
|
*
|
||||||
* @param string $shortcode The shortcode tag to map to the callback - normally in lowercase_underscore format.
|
* @param string $shortcode The shortcode tag to map to the callback - normally in lowercase_underscore format.
|
||||||
* @param callback $callback The callback to replace the shortcode with.
|
* @param callback $callback The callback to replace the shortcode with.
|
||||||
@ -102,9 +103,9 @@ class ShortcodeParser {
|
|||||||
$this->shortcodes = array();
|
$this->shortcodes = array();
|
||||||
}
|
}
|
||||||
|
|
||||||
public function callShortcode($tag, $attributes, $content) {
|
public function callShortcode($tag, $attributes, $content, $extra = array()) {
|
||||||
if (!isset($this->shortcodes[$tag])) return false;
|
if (!isset($this->shortcodes[$tag])) return false;
|
||||||
return call_user_func($this->shortcodes[$tag], $attributes, $content, $this, $tag);
|
return call_user_func($this->shortcodes[$tag], $attributes, $content, $this, $tag, $extra);
|
||||||
}
|
}
|
||||||
|
|
||||||
// --------------------------------------------------------------------------------------------------------------
|
// --------------------------------------------------------------------------------------------------------------
|
||||||
@ -332,11 +333,12 @@ class ShortcodeParser {
|
|||||||
for($i = 0; $i < $attributes->length; $i++) {
|
for($i = 0; $i < $attributes->length; $i++) {
|
||||||
$node = $attributes->item($i);
|
$node = $attributes->item($i);
|
||||||
$tags = $this->extractTags($node->nodeValue);
|
$tags = $this->extractTags($node->nodeValue);
|
||||||
|
$extra = array('node' => $node, 'element' => $node->ownerElement);
|
||||||
|
|
||||||
if($tags) {
|
if($tags) {
|
||||||
$node->nodeValue = $this->replaceTagsWithText($node->nodeValue, $tags,
|
$node->nodeValue = $this->replaceTagsWithText($node->nodeValue, $tags,
|
||||||
function($idx, $tag) use ($parser){
|
function($idx, $tag) use ($parser, $extra){
|
||||||
$content = $parser->callShortcode($tag['open'], $tag['attrs'], $tag['content']);
|
$content = $parser->callShortcode($tag['open'], $tag['attrs'], $tag['content'], $extra);
|
||||||
|
|
||||||
if ($content === false) {
|
if ($content === false) {
|
||||||
if(ShortcodeParser::$error_behavior == ShortcodeParser::ERROR) {
|
if(ShortcodeParser::$error_behavior == ShortcodeParser::ERROR) {
|
||||||
|
@ -6,6 +6,7 @@
|
|||||||
class ShortcodeParserTest extends SapphireTest {
|
class ShortcodeParserTest extends SapphireTest {
|
||||||
|
|
||||||
protected $arguments, $contents, $tagName, $parser;
|
protected $arguments, $contents, $tagName, $parser;
|
||||||
|
protected $extra = array();
|
||||||
|
|
||||||
public function setUp() {
|
public function setUp() {
|
||||||
ShortcodeParser::get('test')->register('test_shortcode', array($this, 'shortcodeSaver'));
|
ShortcodeParser::get('test')->register('test_shortcode', array($this, 'shortcodeSaver'));
|
||||||
@ -210,15 +211,24 @@ class ShortcodeParserTest extends SapphireTest {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function testExtraContext() {
|
||||||
|
$this->parser->parse('<a href="[test_shortcode]">Test</a>');
|
||||||
|
|
||||||
|
$this->assertInstanceOf('DOMNode', $this->extra['node']);
|
||||||
|
$this->assertInstanceOf('DOMElement', $this->extra['element']);
|
||||||
|
$this->assertEquals($this->extra['element']->tagName, 'a');
|
||||||
|
}
|
||||||
|
|
||||||
// -----------------------------------------------------------------------------------------------------------------
|
// -----------------------------------------------------------------------------------------------------------------
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Stores the result of a shortcode parse in object properties for easy testing access.
|
* Stores the result of a shortcode parse in object properties for easy testing access.
|
||||||
*/
|
*/
|
||||||
public function shortcodeSaver($arguments, $content = null, $parser, $tagName = null) {
|
public function shortcodeSaver($arguments, $content, $parser, $tagName, $extra) {
|
||||||
$this->arguments = $arguments;
|
$this->arguments = $arguments;
|
||||||
$this->contents = $content;
|
$this->contents = $content;
|
||||||
$this->tagName = $tagName;
|
$this->tagName = $tagName;
|
||||||
|
$this->extra = $extra;
|
||||||
|
|
||||||
return $content;
|
return $content;
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user