187 lines
		
	
	
		
			6.9 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			187 lines
		
	
	
		
			6.9 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| 
 | |
| namespace dokuwiki\Parsing\Handler;
 | |
| 
 | |
| class Lists extends AbstractRewriter
 | |
| {
 | |
|     protected $listCalls = array();
 | |
|     protected $listStack = array();
 | |
| 
 | |
|     protected $initialDepth = 0;
 | |
| 
 | |
|     const NODE = 1;
 | |
| 
 | |
|     /** @inheritdoc */
 | |
|     public function finalise()
 | |
|     {
 | |
|         $last_call = end($this->calls);
 | |
|         $this->writeCall(array('list_close',array(), $last_call[2]));
 | |
| 
 | |
|         $this->process();
 | |
|         $this->callWriter->finalise();
 | |
|         unset($this->callWriter);
 | |
|     }
 | |
| 
 | |
|     /** @inheritdoc */
 | |
|     public function process()
 | |
|     {
 | |
| 
 | |
|         foreach ($this->calls as $call) {
 | |
|             switch ($call[0]) {
 | |
|                 case 'list_item':
 | |
|                     $this->listOpen($call);
 | |
|                     break;
 | |
|                 case 'list_open':
 | |
|                     $this->listStart($call);
 | |
|                     break;
 | |
|                 case 'list_close':
 | |
|                     $this->listEnd($call);
 | |
|                     break;
 | |
|                 default:
 | |
|                     $this->listContent($call);
 | |
|                     break;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         $this->callWriter->writeCalls($this->listCalls);
 | |
|         return $this->callWriter;
 | |
|     }
 | |
| 
 | |
|     protected function listStart($call)
 | |
|     {
 | |
|         $depth = $this->interpretSyntax($call[1][0], $listType);
 | |
| 
 | |
|         $this->initialDepth = $depth;
 | |
|         //                   array(list type, current depth, index of current listitem_open)
 | |
|         $this->listStack[] = array($listType, $depth, 1);
 | |
| 
 | |
|         $this->listCalls[] = array('list'.$listType.'_open',array(),$call[2]);
 | |
|         $this->listCalls[] = array('listitem_open',array(1),$call[2]);
 | |
|         $this->listCalls[] = array('listcontent_open',array(),$call[2]);
 | |
|     }
 | |
| 
 | |
| 
 | |
|     protected function listEnd($call)
 | |
|     {
 | |
|         $closeContent = true;
 | |
| 
 | |
|         while ($list = array_pop($this->listStack)) {
 | |
|             if ($closeContent) {
 | |
|                 $this->listCalls[] = array('listcontent_close',array(),$call[2]);
 | |
|                 $closeContent = false;
 | |
|             }
 | |
|             $this->listCalls[] = array('listitem_close',array(),$call[2]);
 | |
|             $this->listCalls[] = array('list'.$list[0].'_close', array(), $call[2]);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     protected function listOpen($call)
 | |
|     {
 | |
|         $depth = $this->interpretSyntax($call[1][0], $listType);
 | |
|         $end = end($this->listStack);
 | |
|         $key = key($this->listStack);
 | |
| 
 | |
|         // Not allowed to be shallower than initialDepth
 | |
|         if ($depth < $this->initialDepth) {
 | |
|             $depth = $this->initialDepth;
 | |
|         }
 | |
| 
 | |
|         if ($depth == $end[1]) {
 | |
|             // Just another item in the list...
 | |
|             if ($listType == $end[0]) {
 | |
|                 $this->listCalls[] = array('listcontent_close',array(),$call[2]);
 | |
|                 $this->listCalls[] = array('listitem_close',array(),$call[2]);
 | |
|                 $this->listCalls[] = array('listitem_open',array($depth-1),$call[2]);
 | |
|                 $this->listCalls[] = array('listcontent_open',array(),$call[2]);
 | |
| 
 | |
|                 // new list item, update list stack's index into current listitem_open
 | |
|                 $this->listStack[$key][2] = count($this->listCalls) - 2;
 | |
| 
 | |
|                 // Switched list type...
 | |
|             } else {
 | |
|                 $this->listCalls[] = array('listcontent_close',array(),$call[2]);
 | |
|                 $this->listCalls[] = array('listitem_close',array(),$call[2]);
 | |
|                 $this->listCalls[] = array('list'.$end[0].'_close', array(), $call[2]);
 | |
|                 $this->listCalls[] = array('list'.$listType.'_open', array(), $call[2]);
 | |
|                 $this->listCalls[] = array('listitem_open', array($depth-1), $call[2]);
 | |
|                 $this->listCalls[] = array('listcontent_open',array(),$call[2]);
 | |
| 
 | |
|                 array_pop($this->listStack);
 | |
|                 $this->listStack[] = array($listType, $depth, count($this->listCalls) - 2);
 | |
|             }
 | |
|         } elseif ($depth > $end[1]) { // Getting deeper...
 | |
|             $this->listCalls[] = array('listcontent_close',array(),$call[2]);
 | |
|             $this->listCalls[] = array('list'.$listType.'_open', array(), $call[2]);
 | |
|             $this->listCalls[] = array('listitem_open', array($depth-1), $call[2]);
 | |
|             $this->listCalls[] = array('listcontent_open',array(),$call[2]);
 | |
| 
 | |
|             // set the node/leaf state of this item's parent listitem_open to NODE
 | |
|             $this->listCalls[$this->listStack[$key][2]][1][1] = self::NODE;
 | |
| 
 | |
|             $this->listStack[] = array($listType, $depth, count($this->listCalls) - 2);
 | |
|         } else { // Getting shallower ( $depth < $end[1] )
 | |
|             $this->listCalls[] = array('listcontent_close',array(),$call[2]);
 | |
|             $this->listCalls[] = array('listitem_close',array(),$call[2]);
 | |
|             $this->listCalls[] = array('list'.$end[0].'_close',array(),$call[2]);
 | |
| 
 | |
|             // Throw away the end - done
 | |
|             array_pop($this->listStack);
 | |
| 
 | |
|             while (1) {
 | |
|                 $end = end($this->listStack);
 | |
|                 $key = key($this->listStack);
 | |
| 
 | |
|                 if ($end[1] <= $depth) {
 | |
|                     // Normalize depths
 | |
|                     $depth = $end[1];
 | |
| 
 | |
|                     $this->listCalls[] = array('listitem_close',array(),$call[2]);
 | |
| 
 | |
|                     if ($end[0] == $listType) {
 | |
|                         $this->listCalls[] = array('listitem_open',array($depth-1),$call[2]);
 | |
|                         $this->listCalls[] = array('listcontent_open',array(),$call[2]);
 | |
| 
 | |
|                         // new list item, update list stack's index into current listitem_open
 | |
|                         $this->listStack[$key][2] = count($this->listCalls) - 2;
 | |
|                     } else {
 | |
|                         // Switching list type...
 | |
|                         $this->listCalls[] = array('list'.$end[0].'_close', array(), $call[2]);
 | |
|                         $this->listCalls[] = array('list'.$listType.'_open', array(), $call[2]);
 | |
|                         $this->listCalls[] = array('listitem_open', array($depth-1), $call[2]);
 | |
|                         $this->listCalls[] = array('listcontent_open',array(),$call[2]);
 | |
| 
 | |
|                         array_pop($this->listStack);
 | |
|                         $this->listStack[] = array($listType, $depth, count($this->listCalls) - 2);
 | |
|                     }
 | |
| 
 | |
|                     break;
 | |
| 
 | |
|                     // Haven't dropped down far enough yet.... ( $end[1] > $depth )
 | |
|                 } else {
 | |
|                     $this->listCalls[] = array('listitem_close',array(),$call[2]);
 | |
|                     $this->listCalls[] = array('list'.$end[0].'_close',array(),$call[2]);
 | |
| 
 | |
|                     array_pop($this->listStack);
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     protected function listContent($call)
 | |
|     {
 | |
|         $this->listCalls[] = $call;
 | |
|     }
 | |
| 
 | |
|     protected function interpretSyntax($match, & $type)
 | |
|     {
 | |
|         if (substr($match, -1) == '*') {
 | |
|             $type = 'u';
 | |
|         } else {
 | |
|             $type = 'o';
 | |
|         }
 | |
|         // Is the +1 needed? It used to be count(explode(...))
 | |
|         // but I don't think the number is seen outside this handler
 | |
|         return substr_count(str_replace("\t", '  ', $match), '  ') + 1;
 | |
|     }
 | |
| }
 |