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;
 | |
|     }
 | |
| }
 |