212 lines
6.8 KiB
PHP
212 lines
6.8 KiB
PHP
<?php
|
|
|
|
namespace dokuwiki\Parsing\Handler;
|
|
|
|
/**
|
|
* Handler for paragraphs
|
|
*
|
|
* @author Harry Fuecks <hfuecks@gmail.com>
|
|
*/
|
|
class Block
|
|
{
|
|
protected $calls = array();
|
|
protected $skipEol = false;
|
|
protected $inParagraph = false;
|
|
|
|
// Blocks these should not be inside paragraphs
|
|
protected $blockOpen = array(
|
|
'header',
|
|
'listu_open','listo_open','listitem_open','listcontent_open',
|
|
'table_open','tablerow_open','tablecell_open','tableheader_open','tablethead_open',
|
|
'quote_open',
|
|
'code','file','hr','preformatted','rss',
|
|
'htmlblock','phpblock',
|
|
'footnote_open',
|
|
);
|
|
|
|
protected $blockClose = array(
|
|
'header',
|
|
'listu_close','listo_close','listitem_close','listcontent_close',
|
|
'table_close','tablerow_close','tablecell_close','tableheader_close','tablethead_close',
|
|
'quote_close',
|
|
'code','file','hr','preformatted','rss',
|
|
'htmlblock','phpblock',
|
|
'footnote_close',
|
|
);
|
|
|
|
// Stacks can contain paragraphs
|
|
protected $stackOpen = array(
|
|
'section_open',
|
|
);
|
|
|
|
protected $stackClose = array(
|
|
'section_close',
|
|
);
|
|
|
|
|
|
/**
|
|
* Constructor. Adds loaded syntax plugins to the block and stack
|
|
* arrays
|
|
*
|
|
* @author Andreas Gohr <andi@splitbrain.org>
|
|
*/
|
|
public function __construct()
|
|
{
|
|
global $DOKU_PLUGINS;
|
|
//check if syntax plugins were loaded
|
|
if (empty($DOKU_PLUGINS['syntax'])) return;
|
|
foreach ($DOKU_PLUGINS['syntax'] as $n => $p) {
|
|
$ptype = $p->getPType();
|
|
if ($ptype == 'block') {
|
|
$this->blockOpen[] = 'plugin_'.$n;
|
|
$this->blockClose[] = 'plugin_'.$n;
|
|
} elseif ($ptype == 'stack') {
|
|
$this->stackOpen[] = 'plugin_'.$n;
|
|
$this->stackClose[] = 'plugin_'.$n;
|
|
}
|
|
}
|
|
}
|
|
|
|
protected function openParagraph($pos)
|
|
{
|
|
if ($this->inParagraph) return;
|
|
$this->calls[] = array('p_open',array(), $pos);
|
|
$this->inParagraph = true;
|
|
$this->skipEol = true;
|
|
}
|
|
|
|
/**
|
|
* Close a paragraph if needed
|
|
*
|
|
* This function makes sure there are no empty paragraphs on the stack
|
|
*
|
|
* @author Andreas Gohr <andi@splitbrain.org>
|
|
*
|
|
* @param string|integer $pos
|
|
*/
|
|
protected function closeParagraph($pos)
|
|
{
|
|
if (!$this->inParagraph) return;
|
|
// look back if there was any content - we don't want empty paragraphs
|
|
$content = '';
|
|
$ccount = count($this->calls);
|
|
for ($i=$ccount-1; $i>=0; $i--) {
|
|
if ($this->calls[$i][0] == 'p_open') {
|
|
break;
|
|
} elseif ($this->calls[$i][0] == 'cdata') {
|
|
$content .= $this->calls[$i][1][0];
|
|
} else {
|
|
$content = 'found markup';
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (trim($content)=='') {
|
|
//remove the whole paragraph
|
|
//array_splice($this->calls,$i); // <- this is much slower than the loop below
|
|
for ($x=$ccount; $x>$i;
|
|
$x--) array_pop($this->calls);
|
|
} else {
|
|
// remove ending linebreaks in the paragraph
|
|
$i=count($this->calls)-1;
|
|
if ($this->calls[$i][0] == 'cdata') $this->calls[$i][1][0] = rtrim($this->calls[$i][1][0], "\n");
|
|
$this->calls[] = array('p_close',array(), $pos);
|
|
}
|
|
|
|
$this->inParagraph = false;
|
|
$this->skipEol = true;
|
|
}
|
|
|
|
protected function addCall($call)
|
|
{
|
|
$key = count($this->calls);
|
|
if ($key and ($call[0] == 'cdata') and ($this->calls[$key-1][0] == 'cdata')) {
|
|
$this->calls[$key-1][1][0] .= $call[1][0];
|
|
} else {
|
|
$this->calls[] = $call;
|
|
}
|
|
}
|
|
|
|
// simple version of addCall, without checking cdata
|
|
protected function storeCall($call)
|
|
{
|
|
$this->calls[] = $call;
|
|
}
|
|
|
|
/**
|
|
* Processes the whole instruction stack to open and close paragraphs
|
|
*
|
|
* @author Harry Fuecks <hfuecks@gmail.com>
|
|
* @author Andreas Gohr <andi@splitbrain.org>
|
|
*
|
|
* @param array $calls
|
|
*
|
|
* @return array
|
|
*/
|
|
public function process($calls)
|
|
{
|
|
// open first paragraph
|
|
$this->openParagraph(0);
|
|
foreach ($calls as $key => $call) {
|
|
$cname = $call[0];
|
|
if ($cname == 'plugin') {
|
|
$cname='plugin_'.$call[1][0];
|
|
$plugin = true;
|
|
$plugin_open = (($call[1][2] == DOKU_LEXER_ENTER) || ($call[1][2] == DOKU_LEXER_SPECIAL));
|
|
$plugin_close = (($call[1][2] == DOKU_LEXER_EXIT) || ($call[1][2] == DOKU_LEXER_SPECIAL));
|
|
} else {
|
|
$plugin = false;
|
|
}
|
|
/* stack */
|
|
if (in_array($cname, $this->stackClose) && (!$plugin || $plugin_close)) {
|
|
$this->closeParagraph($call[2]);
|
|
$this->storeCall($call);
|
|
$this->openParagraph($call[2]);
|
|
continue;
|
|
}
|
|
if (in_array($cname, $this->stackOpen) && (!$plugin || $plugin_open)) {
|
|
$this->closeParagraph($call[2]);
|
|
$this->storeCall($call);
|
|
$this->openParagraph($call[2]);
|
|
continue;
|
|
}
|
|
/* block */
|
|
// If it's a substition it opens and closes at the same call.
|
|
// To make sure next paragraph is correctly started, let close go first.
|
|
if (in_array($cname, $this->blockClose) && (!$plugin || $plugin_close)) {
|
|
$this->closeParagraph($call[2]);
|
|
$this->storeCall($call);
|
|
$this->openParagraph($call[2]);
|
|
continue;
|
|
}
|
|
if (in_array($cname, $this->blockOpen) && (!$plugin || $plugin_open)) {
|
|
$this->closeParagraph($call[2]);
|
|
$this->storeCall($call);
|
|
continue;
|
|
}
|
|
/* eol */
|
|
if ($cname == 'eol') {
|
|
// Check this isn't an eol instruction to skip...
|
|
if (!$this->skipEol) {
|
|
// Next is EOL => double eol => mark as paragraph
|
|
if (isset($calls[$key+1]) && $calls[$key+1][0] == 'eol') {
|
|
$this->closeParagraph($call[2]);
|
|
$this->openParagraph($call[2]);
|
|
} else {
|
|
//if this is just a single eol make a space from it
|
|
$this->addCall(array('cdata',array("\n"), $call[2]));
|
|
}
|
|
}
|
|
continue;
|
|
}
|
|
/* normal */
|
|
$this->addCall($call);
|
|
$this->skipEol = false;
|
|
}
|
|
// close last paragraph
|
|
$call = end($this->calls);
|
|
$this->closeParagraph($call[2]);
|
|
return $this->calls;
|
|
}
|
|
}
|