10cecf9d5Sandi<?php 20cecf9d5Sandiif(!defined('DOKU_INC')) define('DOKU_INC',realpath(dirname(__FILE__).'/../../').'/'); 30cecf9d5Sandi 40cecf9d5Sandiclass Doku_Handler { 50cecf9d5Sandi 60cecf9d5Sandi var $Renderer = NULL; 70cecf9d5Sandi 80cecf9d5Sandi var $CallWriter = NULL; 90cecf9d5Sandi 100cecf9d5Sandi var $calls = array(); 110cecf9d5Sandi 120cecf9d5Sandi var $meta = array( 130cecf9d5Sandi 'section' => FALSE, 140cecf9d5Sandi 'toc' => TRUE, 159dc2c2afSandi 'cache' => TRUE, 160cecf9d5Sandi ); 170cecf9d5Sandi 18b7c441b9SHarry Fuecks var $rewriteBlocks = TRUE; 19b7c441b9SHarry Fuecks 200cecf9d5Sandi function Doku_Handler() { 210cecf9d5Sandi $this->CallWriter = & new Doku_Handler_CallWriter($this); 220cecf9d5Sandi } 230cecf9d5Sandi 24433bef32Sandi function _addCall($handler, $args, $pos) { 250cecf9d5Sandi $call = array($handler,$args, $pos); 260cecf9d5Sandi $this->CallWriter->writeCall($call); 270cecf9d5Sandi } 280cecf9d5Sandi 29433bef32Sandi function _finalize(){ 300cecf9d5Sandi if ( $this->meta['section'] ) { 310cecf9d5Sandi $S = & new Doku_Handler_Section(); 320cecf9d5Sandi $this->calls = $S->process($this->calls); 330cecf9d5Sandi } 340cecf9d5Sandi 35b7c441b9SHarry Fuecks if ( $this->rewriteBlocks ) { 360cecf9d5Sandi $B = & new Doku_Handler_Block(); 370cecf9d5Sandi $this->calls = $B->process($this->calls); 38b7c441b9SHarry Fuecks } 390cecf9d5Sandi 400cecf9d5Sandi if ( $this->meta['toc'] ) { 410cecf9d5Sandi $T = & new Doku_Handler_Toc(); 420cecf9d5Sandi $this->calls = $T->process($this->calls); 430cecf9d5Sandi } 440cecf9d5Sandi 450cecf9d5Sandi array_unshift($this->calls,array('document_start',array(),0)); 460cecf9d5Sandi $last_call = end($this->calls); 470cecf9d5Sandi array_push($this->calls,array('document_end',array(),$last_call[2])); 480cecf9d5Sandi } 490cecf9d5Sandi 500cecf9d5Sandi function fetch() { 510cecf9d5Sandi $call = each($this->calls); 520cecf9d5Sandi if ( $call ) { 530cecf9d5Sandi return $call['value']; 540cecf9d5Sandi } 550cecf9d5Sandi return FALSE; 560cecf9d5Sandi } 57*ee20e7d1Sandi 58*ee20e7d1Sandi 59*ee20e7d1Sandi /** 60*ee20e7d1Sandi * Special plugin handler 61*ee20e7d1Sandi * 62*ee20e7d1Sandi * This handler is called for all modes starting with 'plugin_'. 63*ee20e7d1Sandi * An additional parameter with the plugin name is passed 64*ee20e7d1Sandi * 65*ee20e7d1Sandi * @author Andreas Gohr <andi@splitbrain.org> 66*ee20e7d1Sandi */ 67*ee20e7d1Sandi function plugin($match, $state, $pos, $pluginname){ 68*ee20e7d1Sandi $data = array($match); 69*ee20e7d1Sandi $plugin = null; 70*ee20e7d1Sandi if(plugin_load('syntax',$pluginname,$plugin)){ 71*ee20e7d1Sandi $data = $plugin->handle($match, $state, $pos, $handler); 72*ee20e7d1Sandi } 73*ee20e7d1Sandi $this->_addCall('plugin',array($pluginname,$data,$pos),$pos); 74*ee20e7d1Sandi return TRUE; 75*ee20e7d1Sandi } 760cecf9d5Sandi 770cecf9d5Sandi function base($match, $state, $pos) { 780cecf9d5Sandi switch ( $state ) { 790cecf9d5Sandi case DOKU_LEXER_UNMATCHED: 80433bef32Sandi $this->_addCall('cdata',array($match), $pos); 810cecf9d5Sandi return TRUE; 820cecf9d5Sandi break; 830cecf9d5Sandi 840cecf9d5Sandi } 850cecf9d5Sandi } 860cecf9d5Sandi 870cecf9d5Sandi function header($match, $state, $pos) { 880cecf9d5Sandi $match = trim($match); 890cecf9d5Sandi $levels = array( 900cecf9d5Sandi '======'=>1, 910cecf9d5Sandi '====='=>2, 920cecf9d5Sandi '===='=>3, 930cecf9d5Sandi '==='=>4, 940cecf9d5Sandi '=='=>5, 950cecf9d5Sandi ); 960cecf9d5Sandi $hsplit = preg_split( '/(={2,})/u', $match,-1, 970cecf9d5Sandi PREG_SPLIT_DELIM_CAPTURE | PREG_SPLIT_NO_EMPTY ); 980cecf9d5Sandi 990cecf9d5Sandi // Locate the level - default to level 1 if no match (title contains == signs) 1000cecf9d5Sandi if ( isset($hsplit[0]) && array_key_exists($hsplit[0], $levels) ) { 1010cecf9d5Sandi $level = $levels[$hsplit[0]]; 1020cecf9d5Sandi } else { 1030cecf9d5Sandi $level = 1; 1040cecf9d5Sandi } 1050cecf9d5Sandi 1060cecf9d5Sandi // Strip of the marker for the header, based on the level - the rest is the title 1070cecf9d5Sandi $iLevels = array_flip($levels); 1080cecf9d5Sandi $markerLen = strlen($iLevels[$level]); 1090cecf9d5Sandi $title = substr($match, $markerLen, strlen($match)-($markerLen*2)); 1100cecf9d5Sandi 111433bef32Sandi $this->_addCall('header',array($title,$level,$pos), $pos); 1120cecf9d5Sandi $this->meta['section'] = TRUE; 1130cecf9d5Sandi return TRUE; 1140cecf9d5Sandi } 1150cecf9d5Sandi 1160cecf9d5Sandi function notoc($match, $state, $pos) { 1170cecf9d5Sandi $this->meta['toc'] = FALSE; 1180cecf9d5Sandi return TRUE; 1190cecf9d5Sandi } 1200cecf9d5Sandi 1219dc2c2afSandi function nocache($match, $state, $pos) { 1229dc2c2afSandi $this->_addCall('nocache',array(),$pos); 1239dc2c2afSandi return TRUE; 1249dc2c2afSandi } 1259dc2c2afSandi 1260cecf9d5Sandi function linebreak($match, $state, $pos) { 127433bef32Sandi $this->_addCall('linebreak',array(),$pos); 1280cecf9d5Sandi return TRUE; 1290cecf9d5Sandi } 1300cecf9d5Sandi 1310cecf9d5Sandi function eol($match, $state, $pos) { 132433bef32Sandi $this->_addCall('eol',array(),$pos); 1330cecf9d5Sandi return TRUE; 1340cecf9d5Sandi } 1350cecf9d5Sandi 1360cecf9d5Sandi function hr($match, $state, $pos) { 137433bef32Sandi $this->_addCall('hr',array(),$pos); 1380cecf9d5Sandi return TRUE; 1390cecf9d5Sandi } 1400cecf9d5Sandi 141433bef32Sandi function _nestingTag($match, $state, $pos, $name) { 1420cecf9d5Sandi switch ( $state ) { 1430cecf9d5Sandi case DOKU_LEXER_ENTER: 144433bef32Sandi $this->_addCall($name.'_open', array(), $pos); 1450cecf9d5Sandi break; 1460cecf9d5Sandi case DOKU_LEXER_EXIT: 147433bef32Sandi $this->_addCall($name.'_close', array(), $pos); 1480cecf9d5Sandi break; 1490cecf9d5Sandi case DOKU_LEXER_UNMATCHED: 150433bef32Sandi $this->_addCall('cdata',array($match), $pos); 1510cecf9d5Sandi break; 1520cecf9d5Sandi } 1530cecf9d5Sandi } 1540cecf9d5Sandi 1550cecf9d5Sandi function strong($match, $state, $pos) { 156433bef32Sandi $this->_nestingTag($match, $state, $pos, 'strong'); 1570cecf9d5Sandi return TRUE; 1580cecf9d5Sandi } 1590cecf9d5Sandi 1600cecf9d5Sandi function emphasis($match, $state, $pos) { 161433bef32Sandi $this->_nestingTag($match, $state, $pos, 'emphasis'); 1620cecf9d5Sandi return TRUE; 1630cecf9d5Sandi } 1640cecf9d5Sandi 1650cecf9d5Sandi function underline($match, $state, $pos) { 166433bef32Sandi $this->_nestingTag($match, $state, $pos, 'underline'); 1670cecf9d5Sandi return TRUE; 1680cecf9d5Sandi } 1690cecf9d5Sandi 1700cecf9d5Sandi function monospace($match, $state, $pos) { 171433bef32Sandi $this->_nestingTag($match, $state, $pos, 'monospace'); 1720cecf9d5Sandi return TRUE; 1730cecf9d5Sandi } 1740cecf9d5Sandi 1750cecf9d5Sandi function subscript($match, $state, $pos) { 176433bef32Sandi $this->_nestingTag($match, $state, $pos, 'subscript'); 1770cecf9d5Sandi return TRUE; 1780cecf9d5Sandi } 1790cecf9d5Sandi 1800cecf9d5Sandi function superscript($match, $state, $pos) { 181433bef32Sandi $this->_nestingTag($match, $state, $pos, 'superscript'); 1820cecf9d5Sandi return TRUE; 1830cecf9d5Sandi } 1840cecf9d5Sandi 1850cecf9d5Sandi function deleted($match, $state, $pos) { 186433bef32Sandi $this->_nestingTag($match, $state, $pos, 'deleted'); 1870cecf9d5Sandi return TRUE; 1880cecf9d5Sandi } 1890cecf9d5Sandi 1900cecf9d5Sandi 1910cecf9d5Sandi function footnote($match, $state, $pos) { 192433bef32Sandi $this->_nestingTag($match, $state, $pos, 'footnote'); 1930cecf9d5Sandi return TRUE; 1940cecf9d5Sandi } 1950cecf9d5Sandi 1960cecf9d5Sandi function listblock($match, $state, $pos) { 1970cecf9d5Sandi switch ( $state ) { 1980cecf9d5Sandi case DOKU_LEXER_ENTER: 1990cecf9d5Sandi $ReWriter = & new Doku_Handler_List($this->CallWriter); 2000cecf9d5Sandi $this->CallWriter = & $ReWriter; 201433bef32Sandi $this->_addCall('list_open', array($match), $pos); 2020cecf9d5Sandi break; 2030cecf9d5Sandi case DOKU_LEXER_EXIT: 204433bef32Sandi $this->_addCall('list_close', array(), $pos); 2050cecf9d5Sandi $this->CallWriter->process(); 2060cecf9d5Sandi $ReWriter = & $this->CallWriter; 2070cecf9d5Sandi $this->CallWriter = & $ReWriter->CallWriter; 2080cecf9d5Sandi break; 2090cecf9d5Sandi case DOKU_LEXER_MATCHED: 210433bef32Sandi $this->_addCall('list_item', array($match), $pos); 2110cecf9d5Sandi break; 2120cecf9d5Sandi case DOKU_LEXER_UNMATCHED: 213433bef32Sandi $this->_addCall('cdata', array($match), $pos); 2140cecf9d5Sandi break; 2150cecf9d5Sandi } 2160cecf9d5Sandi return TRUE; 2170cecf9d5Sandi } 2180cecf9d5Sandi 2190cecf9d5Sandi function unformatted($match, $state, $pos) { 2200cecf9d5Sandi if ( $state == DOKU_LEXER_UNMATCHED ) { 221433bef32Sandi $this->_addCall('unformatted',array($match), $pos); 2220cecf9d5Sandi } 2230cecf9d5Sandi return TRUE; 2240cecf9d5Sandi } 2250cecf9d5Sandi 2260cecf9d5Sandi function php($match, $state, $pos) { 2270cecf9d5Sandi if ( $state == DOKU_LEXER_UNMATCHED ) { 228433bef32Sandi $this->_addCall('php',array($match), $pos); 2290cecf9d5Sandi } 2300cecf9d5Sandi return TRUE; 2310cecf9d5Sandi } 2320cecf9d5Sandi 2330cecf9d5Sandi function html($match, $state, $pos) { 2340cecf9d5Sandi if ( $state == DOKU_LEXER_UNMATCHED ) { 235433bef32Sandi $this->_addCall('html',array($match), $pos); 2360cecf9d5Sandi } 2370cecf9d5Sandi return TRUE; 2380cecf9d5Sandi } 2390cecf9d5Sandi 2400cecf9d5Sandi function preformatted($match, $state, $pos) { 2410cecf9d5Sandi switch ( $state ) { 2420cecf9d5Sandi case DOKU_LEXER_ENTER: 2430cecf9d5Sandi $ReWriter = & new Doku_Handler_Preformatted($this->CallWriter); 2440cecf9d5Sandi $this->CallWriter = & $ReWriter; 245433bef32Sandi $this->_addCall('preformatted_start',array(), $pos); 2460cecf9d5Sandi break; 2470cecf9d5Sandi case DOKU_LEXER_EXIT: 248433bef32Sandi $this->_addCall('preformatted_end',array(), $pos); 2490cecf9d5Sandi $this->CallWriter->process(); 2500cecf9d5Sandi $ReWriter = & $this->CallWriter; 2510cecf9d5Sandi $this->CallWriter = & $ReWriter->CallWriter; 2520cecf9d5Sandi break; 2530cecf9d5Sandi case DOKU_LEXER_MATCHED: 254433bef32Sandi $this->_addCall('preformatted_newline',array(), $pos); 2550cecf9d5Sandi break; 2560cecf9d5Sandi case DOKU_LEXER_UNMATCHED: 257433bef32Sandi $this->_addCall('preformatted_content',array($match), $pos); 2580cecf9d5Sandi break; 2590cecf9d5Sandi } 2600cecf9d5Sandi 2610cecf9d5Sandi return TRUE; 2620cecf9d5Sandi } 2630cecf9d5Sandi 2640cecf9d5Sandi function file($match, $state, $pos) { 2650cecf9d5Sandi if ( $state == DOKU_LEXER_UNMATCHED ) { 266433bef32Sandi $this->_addCall('file',array($match), $pos); 2670cecf9d5Sandi } 2680cecf9d5Sandi return TRUE; 2690cecf9d5Sandi } 2700cecf9d5Sandi 2710cecf9d5Sandi function quote($match, $state, $pos) { 2720cecf9d5Sandi 2730cecf9d5Sandi switch ( $state ) { 2740cecf9d5Sandi 2750cecf9d5Sandi case DOKU_LEXER_ENTER: 2760cecf9d5Sandi $ReWriter = & new Doku_Handler_Quote($this->CallWriter); 2770cecf9d5Sandi $this->CallWriter = & $ReWriter; 278433bef32Sandi $this->_addCall('quote_start',array($match), $pos); 2790cecf9d5Sandi break; 2800cecf9d5Sandi 2810cecf9d5Sandi case DOKU_LEXER_EXIT: 282433bef32Sandi $this->_addCall('quote_end',array(), $pos); 2830cecf9d5Sandi $this->CallWriter->process(); 2840cecf9d5Sandi $ReWriter = & $this->CallWriter; 2850cecf9d5Sandi $this->CallWriter = & $ReWriter->CallWriter; 2860cecf9d5Sandi break; 2870cecf9d5Sandi 2880cecf9d5Sandi case DOKU_LEXER_MATCHED: 289433bef32Sandi $this->_addCall('quote_newline',array($match), $pos); 2900cecf9d5Sandi break; 2910cecf9d5Sandi 2920cecf9d5Sandi case DOKU_LEXER_UNMATCHED: 293433bef32Sandi $this->_addCall('cdata',array($match), $pos); 2940cecf9d5Sandi break; 2950cecf9d5Sandi 2960cecf9d5Sandi } 2970cecf9d5Sandi 2980cecf9d5Sandi return TRUE; 2990cecf9d5Sandi } 3000cecf9d5Sandi 3010cecf9d5Sandi function code($match, $state, $pos) { 3020cecf9d5Sandi switch ( $state ) { 3030cecf9d5Sandi case DOKU_LEXER_UNMATCHED: 3040cecf9d5Sandi $matches = preg_split('/>/u',$match,2); 3050cecf9d5Sandi $matches[0] = trim($matches[0]); 3060cecf9d5Sandi if ( trim($matches[0]) == '' ) { 3070cecf9d5Sandi $matches[0] = NULL; 3080cecf9d5Sandi } 3090cecf9d5Sandi # $matches[0] contains name of programming language 3100cecf9d5Sandi # if available 311433bef32Sandi $this->_addCall( 3120cecf9d5Sandi 'code', 3130cecf9d5Sandi array($matches[1],$matches[0]), 3140cecf9d5Sandi $pos 3150cecf9d5Sandi ); 3160cecf9d5Sandi break; 3170cecf9d5Sandi } 3180cecf9d5Sandi return TRUE; 3190cecf9d5Sandi } 3200cecf9d5Sandi 3210cecf9d5Sandi function acronym($match, $state, $pos) { 322433bef32Sandi $this->_addCall('acronym',array($match), $pos); 3230cecf9d5Sandi return TRUE; 3240cecf9d5Sandi } 3250cecf9d5Sandi 3260cecf9d5Sandi function smiley($match, $state, $pos) { 327433bef32Sandi $this->_addCall('smiley',array($match), $pos); 3280cecf9d5Sandi return TRUE; 3290cecf9d5Sandi } 3300cecf9d5Sandi 3310cecf9d5Sandi function wordblock($match, $state, $pos) { 332433bef32Sandi $this->_addCall('wordblock',array($match), $pos); 3330cecf9d5Sandi return TRUE; 3340cecf9d5Sandi } 3350cecf9d5Sandi 3360cecf9d5Sandi function entity($match, $state, $pos) { 337433bef32Sandi $this->_addCall('entity',array($match), $pos); 3380cecf9d5Sandi return TRUE; 3390cecf9d5Sandi } 3400cecf9d5Sandi 3410cecf9d5Sandi function multiplyentity($match, $state, $pos) { 3420cecf9d5Sandi preg_match_all('/\d+/',$match,$matches); 343433bef32Sandi $this->_addCall('multiplyentity',array($matches[0][0],$matches[0][1]), $pos); 3440cecf9d5Sandi return TRUE; 3450cecf9d5Sandi } 3460cecf9d5Sandi 3470cecf9d5Sandi function singlequoteopening($match, $state, $pos) { 348433bef32Sandi $this->_addCall('singlequoteopening',array(), $pos); 3490cecf9d5Sandi return TRUE; 3500cecf9d5Sandi } 3510cecf9d5Sandi 3520cecf9d5Sandi function singlequoteclosing($match, $state, $pos) { 353433bef32Sandi $this->_addCall('singlequoteclosing',array(), $pos); 3540cecf9d5Sandi return TRUE; 3550cecf9d5Sandi } 3560cecf9d5Sandi 3570cecf9d5Sandi function doublequoteopening($match, $state, $pos) { 358433bef32Sandi $this->_addCall('doublequoteopening',array(), $pos); 3590cecf9d5Sandi return TRUE; 3600cecf9d5Sandi } 3610cecf9d5Sandi 3620cecf9d5Sandi function doublequoteclosing($match, $state, $pos) { 363433bef32Sandi $this->_addCall('doublequoteclosing',array(), $pos); 3640cecf9d5Sandi return TRUE; 3650cecf9d5Sandi } 3660cecf9d5Sandi 3670cecf9d5Sandi function camelcaselink($match, $state, $pos) { 368433bef32Sandi $this->_addCall('camelcaselink',array($match), $pos); 3690cecf9d5Sandi return TRUE; 3700cecf9d5Sandi } 3710cecf9d5Sandi 3720cecf9d5Sandi /* 3730cecf9d5Sandi */ 3740cecf9d5Sandi function internallink($match, $state, $pos) { 3750cecf9d5Sandi // Strip the opening and closing markup 3760cecf9d5Sandi $link = preg_replace(array('/^\[\[/','/\]\]$/u'),'',$match); 3770cecf9d5Sandi 3780cecf9d5Sandi // Split title from URL 3790cecf9d5Sandi $link = preg_split('/\|/u',$link,2); 3800cecf9d5Sandi if ( !isset($link[1]) ) { 3810cecf9d5Sandi $link[1] = NULL; 3820cecf9d5Sandi } else if ( preg_match('/^\{\{[^\}]+\}\}$/',$link[1]) ) { 3835578eb8fSandi // If the title is an image, convert it to an array containing the image details 384b625487dSandi $link[1] = Doku_Handler_Parse_Media($link[1]); 3850cecf9d5Sandi } 3860b7c14c2Sandi $link[0] = trim($link[0]); 3870cecf9d5Sandi 3880e1c636eSandi //decide which kind of link it is 3890e1c636eSandi 3905578eb8fSandi if ( preg_match('/^[a-zA-Z]+>{1}.+$/u',$link[0]) ) { 3910e1c636eSandi // Interwiki 3920cecf9d5Sandi $interwiki = preg_split('/>/u',$link[0]); 393433bef32Sandi $this->_addCall( 3940cecf9d5Sandi 'interwikilink', 3950cecf9d5Sandi array($link[0],$link[1],strtolower($interwiki[0]),$interwiki[1]), 3960cecf9d5Sandi $pos 3970cecf9d5Sandi ); 3980b7c14c2Sandi }elseif ( preg_match('/^\\\\\\\\[\w.:?\-;,]+?\\\\/u',$link[0]) ) { 3990e1c636eSandi // Windows Share 400433bef32Sandi $this->_addCall( 4010cecf9d5Sandi 'windowssharelink', 4020cecf9d5Sandi array($link[0],$link[1]), 4030cecf9d5Sandi $pos 4040cecf9d5Sandi ); 4050e1c636eSandi }elseif ( preg_match('#([a-z0-9\-_.]+?)@([\w\-]+\.([\w\-\.]+\.)*[\w]+)#i',$link[0]) ) { 4060e1c636eSandi // E-Mail 407433bef32Sandi $this->_addCall( 4080e1c636eSandi 'emaillink', 4090e1c636eSandi array($link[0],$link[1]), 4100e1c636eSandi $pos 4110e1c636eSandi ); 4120e1c636eSandi }elseif ( preg_match('#^([a-z0-9]+?)://#i',$link[0]) ) { 4130e1c636eSandi // external link (accepts all protocols) 414433bef32Sandi $this->_addCall( 4150cecf9d5Sandi 'externallink', 4160cecf9d5Sandi array($link[0],$link[1]), 4170cecf9d5Sandi $pos 4180cecf9d5Sandi ); 4190b7c14c2Sandi }elseif ( preg_match('!^#.+!',$link[0]) ){ 4200b7c14c2Sandi // local link 4210b7c14c2Sandi $this->_addCall( 4220b7c14c2Sandi 'locallink', 4230b7c14c2Sandi array(substr($link[0],1),$link[1]), 4240b7c14c2Sandi $pos 4250b7c14c2Sandi ); 4260e1c636eSandi }else{ 4270e1c636eSandi // internal link 428433bef32Sandi $this->_addCall( 4290e1c636eSandi 'internallink', 4300e1c636eSandi array($link[0],$link[1]), 4310e1c636eSandi $pos 4320e1c636eSandi ); 4330cecf9d5Sandi } 4340e1c636eSandi 4350cecf9d5Sandi return TRUE; 4360cecf9d5Sandi } 4370cecf9d5Sandi 4380cecf9d5Sandi function filelink($match, $state, $pos) { 439433bef32Sandi $this->_addCall('filelink',array($match, NULL), $pos); 4400cecf9d5Sandi return TRUE; 4410cecf9d5Sandi } 4420cecf9d5Sandi 4430cecf9d5Sandi function windowssharelink($match, $state, $pos) { 444433bef32Sandi $this->_addCall('windowssharelink',array($match, NULL), $pos); 4450cecf9d5Sandi return TRUE; 4460cecf9d5Sandi } 4470cecf9d5Sandi 4480cecf9d5Sandi function media($match, $state, $pos) { 4490cecf9d5Sandi $p = Doku_Handler_Parse_Media($match); 4500cecf9d5Sandi 451433bef32Sandi $this->_addCall( 4520cecf9d5Sandi $p['type'], 4530cecf9d5Sandi array($p['src'], $p['title'], $p['align'], $p['width'], $p['height'], $p['cache']), 4540cecf9d5Sandi $pos 4550cecf9d5Sandi ); 4560cecf9d5Sandi return TRUE; 4570cecf9d5Sandi } 4580cecf9d5Sandi 459b625487dSandi function rss($match, $state, $pos) { 460b625487dSandi $link = preg_replace(array('/^\{\{rss>/','/\}\}$/'),'',$match); 461433bef32Sandi $this->_addCall('rss',array($link),$pos); 462f1cf0e10Sandi return TRUE; 463b625487dSandi } 464b625487dSandi 4650cecf9d5Sandi function externallink($match, $state, $pos) { 4660cecf9d5Sandi // Prevent use of multibyte strings in URLs 4670cecf9d5Sandi // See: http://www.boingboing.net/2005/02/06/shmoo_group_exploit_.html 4680cecf9d5Sandi // Not worried about other charsets so long as page is output as UTF-8 4690cecf9d5Sandi /*if ( strlen($match) != utf8_strlen($match) ) { 470433bef32Sandi $this->_addCall('cdata',array($match), $pos); 4710cecf9d5Sandi } else {*/ 4720cecf9d5Sandi 473433bef32Sandi $this->_addCall('externallink',array($match, NULL), $pos); 4740cecf9d5Sandi //} 4750cecf9d5Sandi return TRUE; 4760cecf9d5Sandi } 4770cecf9d5Sandi 47871352defSandi function emaillink($match, $state, $pos) { 4790cecf9d5Sandi $email = preg_replace(array('/^</','/>$/'),'',$match); 480433bef32Sandi $this->_addCall('emaillink',array($email, NULL), $pos); 4810cecf9d5Sandi return TRUE; 4820cecf9d5Sandi } 4830cecf9d5Sandi 4840cecf9d5Sandi function table($match, $state, $pos) { 4850cecf9d5Sandi switch ( $state ) { 4860cecf9d5Sandi 4870cecf9d5Sandi case DOKU_LEXER_ENTER: 4880cecf9d5Sandi 4890cecf9d5Sandi $ReWriter = & new Doku_Handler_Table($this->CallWriter); 4900cecf9d5Sandi $this->CallWriter = & $ReWriter; 4910cecf9d5Sandi 492433bef32Sandi $this->_addCall('table_start', array(), $pos); 493433bef32Sandi //$this->_addCall('table_row', array(), $pos); 4940cecf9d5Sandi if ( trim($match) == '^' ) { 495433bef32Sandi $this->_addCall('tableheader', array(), $pos); 4960cecf9d5Sandi } else { 497433bef32Sandi $this->_addCall('tablecell', array(), $pos); 4980cecf9d5Sandi } 4990cecf9d5Sandi break; 5000cecf9d5Sandi 5010cecf9d5Sandi case DOKU_LEXER_EXIT: 502433bef32Sandi $this->_addCall('table_end', array(), $pos); 5030cecf9d5Sandi $this->CallWriter->process(); 5040cecf9d5Sandi $ReWriter = & $this->CallWriter; 5050cecf9d5Sandi $this->CallWriter = & $ReWriter->CallWriter; 5060cecf9d5Sandi break; 5070cecf9d5Sandi 5080cecf9d5Sandi case DOKU_LEXER_UNMATCHED: 5090cecf9d5Sandi if ( trim($match) != '' ) { 510433bef32Sandi $this->_addCall('cdata',array($match), $pos); 5110cecf9d5Sandi } 5120cecf9d5Sandi break; 5130cecf9d5Sandi 5140cecf9d5Sandi case DOKU_LEXER_MATCHED: 5150cecf9d5Sandi if ( preg_match('/.{2}/',$match) ) { 516433bef32Sandi $this->_addCall('table_align', array($match), $pos); 5170cecf9d5Sandi } else if ( $match == "\n|" ) { 518433bef32Sandi $this->_addCall('table_row', array(), $pos); 519433bef32Sandi $this->_addCall('tablecell', array(), $pos); 5200cecf9d5Sandi } else if ( $match == "\n^" ) { 521433bef32Sandi $this->_addCall('table_row', array(), $pos); 522433bef32Sandi $this->_addCall('tableheader', array(), $pos); 5230cecf9d5Sandi } else if ( $match == '|' ) { 524433bef32Sandi $this->_addCall('tablecell', array(), $pos); 5250cecf9d5Sandi } else if ( $match == '^' ) { 526433bef32Sandi $this->_addCall('tableheader', array(), $pos); 5270cecf9d5Sandi } 5280cecf9d5Sandi break; 5290cecf9d5Sandi } 5300cecf9d5Sandi return TRUE; 5310cecf9d5Sandi } 5320cecf9d5Sandi} 5330cecf9d5Sandi 5340cecf9d5Sandi//------------------------------------------------------------------------ 5350cecf9d5Sandifunction Doku_Handler_Parse_Media($match) { 5360cecf9d5Sandi 5370cecf9d5Sandi // Strip the opening and closing markup 5380cecf9d5Sandi $link = preg_replace(array('/^\{\{/','/\}\}$/u'),'',$match); 5390cecf9d5Sandi 5400cecf9d5Sandi // Split title from URL 5410cecf9d5Sandi $link = preg_split('/\|/u',$link,2); 5420cecf9d5Sandi 5430cecf9d5Sandi 5440cecf9d5Sandi // Check alignment 5450cecf9d5Sandi $ralign = (bool)preg_match('/^ /',$link[0]); 5460cecf9d5Sandi $lalign = (bool)preg_match('/ $/',$link[0]); 5470cecf9d5Sandi 5480cecf9d5Sandi // Logic = what's that ;)... 5490cecf9d5Sandi if ( $lalign & $ralign ) { 5500cecf9d5Sandi $align = 'center'; 5510cecf9d5Sandi } else if ( $ralign ) { 5520cecf9d5Sandi $align = 'right'; 5530cecf9d5Sandi } else if ( $lalign ) { 5540cecf9d5Sandi $align = 'left'; 5550cecf9d5Sandi } else { 5560cecf9d5Sandi $align = NULL; 5570cecf9d5Sandi } 5580cecf9d5Sandi 5590cecf9d5Sandi // The title... 5600cecf9d5Sandi if ( !isset($link[1]) ) { 5610cecf9d5Sandi $link[1] = NULL; 5620cecf9d5Sandi } 5630cecf9d5Sandi 5644826ab45Sandi //remove aligning spaces 5654826ab45Sandi $link[0] = trim($link[0]); 5660cecf9d5Sandi 5674826ab45Sandi //split into src and parameters (using the very last questionmark) 5684826ab45Sandi $pos = strrpos($link[0], '?'); 5694826ab45Sandi if($pos !== false){ 5704826ab45Sandi $src = substr($link[0],0,$pos); 5714826ab45Sandi $param = substr($link[0],$pos+1); 5720cecf9d5Sandi }else{ 5734826ab45Sandi $src = $link[0]; 5744826ab45Sandi $param = ''; 5750cecf9d5Sandi } 5760cecf9d5Sandi 5774826ab45Sandi //parse width and height 5784826ab45Sandi if(preg_match('#(\d+)(x(\d+))?#i',$param,$size)){ 5794826ab45Sandi ($size[1]) ? $w = $size[1] : $w = NULL; 5804826ab45Sandi ($size[3]) ? $h = $size[3] : $h = NULL; 5810cecf9d5Sandi } 5820cecf9d5Sandi 5834826ab45Sandi //get caching command 5844826ab45Sandi if (preg_match('/(nocache|recache)/i',$param,$cachemode)){ 5854826ab45Sandi $cache = $cachemode[1]; 5860cecf9d5Sandi }else{ 5874826ab45Sandi $cache = 'cache'; 5880cecf9d5Sandi } 5890cecf9d5Sandi 5900cecf9d5Sandi // Check whether this is a local or remote image 5914826ab45Sandi if ( preg_match('#^(https?|ftp)#i',$src) ) { 5924826ab45Sandi $call = 'externalmedia'; 5930cecf9d5Sandi } else { 5944826ab45Sandi $call = 'internalmedia'; 5950cecf9d5Sandi } 5960cecf9d5Sandi 5970cecf9d5Sandi $params = array( 5980cecf9d5Sandi 'type'=>$call, 5994826ab45Sandi 'src'=>$src, 6000cecf9d5Sandi 'title'=>$link[1], 6010cecf9d5Sandi 'align'=>$align, 6024826ab45Sandi 'width'=>$w, 6034826ab45Sandi 'height'=>$h, 6040cecf9d5Sandi 'cache'=>$cache, 6050cecf9d5Sandi ); 6060cecf9d5Sandi 6070cecf9d5Sandi return $params; 6080cecf9d5Sandi} 6090cecf9d5Sandi 6100cecf9d5Sandi//------------------------------------------------------------------------ 6110cecf9d5Sandiclass Doku_Handler_CallWriter { 6120cecf9d5Sandi 6130cecf9d5Sandi var $Handler; 6140cecf9d5Sandi 6150cecf9d5Sandi function Doku_Handler_CallWriter(& $Handler) { 6160cecf9d5Sandi $this->Handler = & $Handler; 6170cecf9d5Sandi } 6180cecf9d5Sandi 6190cecf9d5Sandi function writeCall($call) { 6200cecf9d5Sandi $this->Handler->calls[] = $call; 6210cecf9d5Sandi } 6220cecf9d5Sandi 6230cecf9d5Sandi function writeCalls($calls) { 6240cecf9d5Sandi $this->Handler->calls = array_merge($this->Handler->calls, $calls); 6250cecf9d5Sandi } 6260cecf9d5Sandi} 6270cecf9d5Sandi 6280cecf9d5Sandi//------------------------------------------------------------------------ 6290cecf9d5Sandiclass Doku_Handler_List { 6300cecf9d5Sandi 6310cecf9d5Sandi var $CallWriter; 6320cecf9d5Sandi 6330cecf9d5Sandi var $calls = array(); 6340cecf9d5Sandi var $listCalls = array(); 6350cecf9d5Sandi var $listStack = array(); 6360cecf9d5Sandi 6370cecf9d5Sandi function Doku_Handler_List(& $CallWriter) { 6380cecf9d5Sandi $this->CallWriter = & $CallWriter; 6390cecf9d5Sandi } 6400cecf9d5Sandi 6410cecf9d5Sandi function writeCall($call) { 6420cecf9d5Sandi $this->calls[] = $call; 6430cecf9d5Sandi } 6440cecf9d5Sandi 6450cecf9d5Sandi // Probably not needed but just in case... 6460cecf9d5Sandi function writeCalls($calls) { 6470cecf9d5Sandi $this->calls = array_merge($this->calls, $calls); 6480cecf9d5Sandi $this->CallWriter->writeCalls($this->calls); 6490cecf9d5Sandi } 6500cecf9d5Sandi 6510cecf9d5Sandi //------------------------------------------------------------------------ 6520cecf9d5Sandi function process() { 6530cecf9d5Sandi foreach ( $this->calls as $call ) { 6540cecf9d5Sandi switch ($call[0]) { 6550cecf9d5Sandi case 'list_item': 6560cecf9d5Sandi $this->listOpen($call); 6570cecf9d5Sandi break; 6580cecf9d5Sandi case 'list_open': 6590cecf9d5Sandi $this->listStart($call); 6600cecf9d5Sandi break; 6610cecf9d5Sandi case 'list_close': 6620cecf9d5Sandi $this->listEnd($call); 6630cecf9d5Sandi break; 6640cecf9d5Sandi default: 6650cecf9d5Sandi $this->listContent($call); 6660cecf9d5Sandi break; 6670cecf9d5Sandi } 6680cecf9d5Sandi } 6690cecf9d5Sandi 6700cecf9d5Sandi $this->CallWriter->writeCalls($this->listCalls); 6710cecf9d5Sandi } 6720cecf9d5Sandi 6730cecf9d5Sandi //------------------------------------------------------------------------ 6740cecf9d5Sandi function listStart($call) { 6750cecf9d5Sandi $depth = $this->interpretSyntax($call[1][0], $listType); 6760cecf9d5Sandi 6770cecf9d5Sandi $this->initialDepth = $depth; 6780cecf9d5Sandi $this->listStack[] = array($listType, $depth); 6790cecf9d5Sandi 6800cecf9d5Sandi $this->listCalls[] = array('list'.$listType.'_open',array(),$call[2]); 6810cecf9d5Sandi $this->listCalls[] = array('listitem_open',array(1),$call[2]); 6820cecf9d5Sandi $this->listCalls[] = array('listcontent_open',array(),$call[2]); 6830cecf9d5Sandi } 6840cecf9d5Sandi 6850cecf9d5Sandi //------------------------------------------------------------------------ 6860cecf9d5Sandi function listEnd($call) { 6870cecf9d5Sandi $closeContent = TRUE; 6880cecf9d5Sandi 6890cecf9d5Sandi while ( $list = array_pop($this->listStack) ) { 6900cecf9d5Sandi if ( $closeContent ) { 6910cecf9d5Sandi $this->listCalls[] = array('listcontent_close',array(),$call[2]); 6920cecf9d5Sandi $closeContent = FALSE; 6930cecf9d5Sandi } 6940cecf9d5Sandi $this->listCalls[] = array('listitem_close',array(),$call[2]); 6950cecf9d5Sandi $this->listCalls[] = array('list'.$list[0].'_close', array(), $call[2]); 6960cecf9d5Sandi } 6970cecf9d5Sandi } 6980cecf9d5Sandi 6990cecf9d5Sandi //------------------------------------------------------------------------ 7000cecf9d5Sandi function listOpen($call) { 7010cecf9d5Sandi $depth = $this->interpretSyntax($call[1][0], $listType); 7020cecf9d5Sandi $end = end($this->listStack); 7030cecf9d5Sandi 7040cecf9d5Sandi // Not allowed to be shallower than initialDepth 7050cecf9d5Sandi if ( $depth < $this->initialDepth ) { 7060cecf9d5Sandi $depth = $this->initialDepth; 7070cecf9d5Sandi } 7080cecf9d5Sandi 7090cecf9d5Sandi //------------------------------------------------------------------------ 7100cecf9d5Sandi if ( $depth == $end[1] ) { 7110cecf9d5Sandi 7120cecf9d5Sandi // Just another item in the list... 7130cecf9d5Sandi if ( $listType == $end[0] ) { 7140cecf9d5Sandi $this->listCalls[] = array('listcontent_close',array(),$call[2]); 7150cecf9d5Sandi $this->listCalls[] = array('listitem_close',array(),$call[2]); 7160cecf9d5Sandi $this->listCalls[] = array('listitem_open',array($depth-1),$call[2]); 7170cecf9d5Sandi $this->listCalls[] = array('listcontent_open',array(),$call[2]); 7180cecf9d5Sandi 7190cecf9d5Sandi // Switched list type... 7200cecf9d5Sandi } else { 7210cecf9d5Sandi 7220cecf9d5Sandi $this->listCalls[] = array('listcontent_close',array(),$call[2]); 7230cecf9d5Sandi $this->listCalls[] = array('listitem_close',array(),$call[2]); 7240cecf9d5Sandi $this->listCalls[] = array('list'.$end[0].'_close', array(), $call[2]); 7250cecf9d5Sandi $this->listCalls[] = array('list'.$listType.'_open', array(), $call[2]); 7260cecf9d5Sandi $this->listCalls[] = array('listitem_open', array($depth-1), $call[2]); 7270cecf9d5Sandi $this->listCalls[] = array('listcontent_open',array(),$call[2]); 7280cecf9d5Sandi 7290cecf9d5Sandi array_pop($this->listStack); 7300cecf9d5Sandi $this->listStack[] = array($listType, $depth); 7310cecf9d5Sandi } 7320cecf9d5Sandi 7330cecf9d5Sandi //------------------------------------------------------------------------ 7340cecf9d5Sandi // Getting deeper... 7350cecf9d5Sandi } else if ( $depth > $end[1] ) { 7360cecf9d5Sandi 7370cecf9d5Sandi $this->listCalls[] = array('listcontent_close',array(),$call[2]); 7380cecf9d5Sandi $this->listCalls[] = array('list'.$listType.'_open', array(), $call[2]); 7390cecf9d5Sandi $this->listCalls[] = array('listitem_open', array($depth-1), $call[2]); 7400cecf9d5Sandi $this->listCalls[] = array('listcontent_open',array(),$call[2]); 7410cecf9d5Sandi 7420cecf9d5Sandi $this->listStack[] = array($listType, $depth); 7430cecf9d5Sandi 7440cecf9d5Sandi //------------------------------------------------------------------------ 7450cecf9d5Sandi // Getting shallower ( $depth < $end[1] ) 7460cecf9d5Sandi } else { 7470cecf9d5Sandi $this->listCalls[] = array('listcontent_close',array(),$call[2]); 7480cecf9d5Sandi $this->listCalls[] = array('listitem_close',array(),$call[2]); 7490cecf9d5Sandi $this->listCalls[] = array('list'.$end[0].'_close',array(),$call[2]); 7500cecf9d5Sandi 7510cecf9d5Sandi // Throw away the end - done 7520cecf9d5Sandi array_pop($this->listStack); 7530cecf9d5Sandi 7540cecf9d5Sandi while (1) { 7550cecf9d5Sandi $end = end($this->listStack); 7560cecf9d5Sandi 7570cecf9d5Sandi if ( $end[1] <= $depth ) { 7580cecf9d5Sandi 7590cecf9d5Sandi // Normalize depths 7600cecf9d5Sandi $depth = $end[1]; 7610cecf9d5Sandi 7620cecf9d5Sandi $this->listCalls[] = array('listitem_close',array(),$call[2]); 7630cecf9d5Sandi 7640cecf9d5Sandi if ( $end[0] == $listType ) { 7650cecf9d5Sandi $this->listCalls[] = array('listitem_open',array($depth-1),$call[2]); 7660cecf9d5Sandi $this->listCalls[] = array('listcontent_open',array(),$call[2]); 7670cecf9d5Sandi 7680cecf9d5Sandi } else { 7690cecf9d5Sandi // Switching list type... 7700cecf9d5Sandi $this->listCalls[] = array('list'.$end[0].'_close', array(), $call[2]); 7710cecf9d5Sandi $this->listCalls[] = array('list'.$listType.'_open', array(), $call[2]); 7720cecf9d5Sandi $this->listCalls[] = array('listitem_open', array($depth-1), $call[2]); 7730cecf9d5Sandi $this->listCalls[] = array('listcontent_open',array(),$call[2]); 7740cecf9d5Sandi 7750cecf9d5Sandi array_pop($this->listStack); 7760cecf9d5Sandi $this->listStack[] = array($listType, $depth); 7770cecf9d5Sandi } 7780cecf9d5Sandi 7790cecf9d5Sandi break; 7800cecf9d5Sandi 7810cecf9d5Sandi // Haven't dropped down far enough yet.... ( $end[1] > $depth ) 7820cecf9d5Sandi } else { 7830cecf9d5Sandi 7840cecf9d5Sandi $this->listCalls[] = array('listitem_close',array(),$call[2]); 7850cecf9d5Sandi $this->listCalls[] = array('list'.$end[0].'_close',array(),$call[2]); 7860cecf9d5Sandi 7870cecf9d5Sandi array_pop($this->listStack); 7880cecf9d5Sandi 7890cecf9d5Sandi } 7900cecf9d5Sandi 7910cecf9d5Sandi } 7920cecf9d5Sandi 7930cecf9d5Sandi } 7940cecf9d5Sandi } 7950cecf9d5Sandi 7960cecf9d5Sandi //------------------------------------------------------------------------ 7970cecf9d5Sandi function listContent($call) { 7980cecf9d5Sandi $this->listCalls[] = $call; 7990cecf9d5Sandi } 8000cecf9d5Sandi 8010cecf9d5Sandi //------------------------------------------------------------------------ 8020cecf9d5Sandi function interpretSyntax($match, & $type) { 8030cecf9d5Sandi if ( substr($match,-1) == '*' ) { 8040cecf9d5Sandi $type = 'u'; 8050cecf9d5Sandi } else { 8060cecf9d5Sandi $type = 'o'; 8070cecf9d5Sandi } 8080cecf9d5Sandi return count(explode(' ',str_replace("\t",' ',$match))); 8090cecf9d5Sandi } 8100cecf9d5Sandi} 8110cecf9d5Sandi 8120cecf9d5Sandi//------------------------------------------------------------------------ 8130cecf9d5Sandiclass Doku_Handler_Preformatted { 8140cecf9d5Sandi 8150cecf9d5Sandi var $CallWriter; 8160cecf9d5Sandi 8170cecf9d5Sandi var $calls = array(); 8180cecf9d5Sandi var $pos; 8190cecf9d5Sandi var $text =''; 8200cecf9d5Sandi 8210cecf9d5Sandi 8220cecf9d5Sandi 8230cecf9d5Sandi function Doku_Handler_Preformatted(& $CallWriter) { 8240cecf9d5Sandi $this->CallWriter = & $CallWriter; 8250cecf9d5Sandi } 8260cecf9d5Sandi 8270cecf9d5Sandi function writeCall($call) { 8280cecf9d5Sandi $this->calls[] = $call; 8290cecf9d5Sandi } 8300cecf9d5Sandi 8310cecf9d5Sandi // Probably not needed but just in case... 8320cecf9d5Sandi function writeCalls($calls) { 8330cecf9d5Sandi $this->calls = array_merge($this->calls, $calls); 8340cecf9d5Sandi $this->CallWriter->writeCalls($this->calls); 8350cecf9d5Sandi } 8360cecf9d5Sandi 8370cecf9d5Sandi function process() { 8380cecf9d5Sandi foreach ( $this->calls as $call ) { 8390cecf9d5Sandi switch ($call[0]) { 8400cecf9d5Sandi case 'preformatted_start': 8410cecf9d5Sandi $this->pos = $call[2]; 8420cecf9d5Sandi break; 8430cecf9d5Sandi case 'preformatted_newline': 8440cecf9d5Sandi $this->text .= "\n"; 8450cecf9d5Sandi break; 8460cecf9d5Sandi case 'preformatted_content': 8470cecf9d5Sandi $this->text .= $call[1][0]; 8480cecf9d5Sandi break; 8490cecf9d5Sandi case 'preformatted_end': 8500cecf9d5Sandi $this->CallWriter->writeCall(array('preformatted',array($this->text),$this->pos)); 8510cecf9d5Sandi break; 8520cecf9d5Sandi } 8530cecf9d5Sandi } 8540cecf9d5Sandi } 8550cecf9d5Sandi} 8560cecf9d5Sandi 8570cecf9d5Sandi//------------------------------------------------------------------------ 8580cecf9d5Sandiclass Doku_Handler_Quote { 8590cecf9d5Sandi 8600cecf9d5Sandi var $CallWriter; 8610cecf9d5Sandi 8620cecf9d5Sandi var $calls = array(); 8630cecf9d5Sandi 8640cecf9d5Sandi var $quoteCalls = array(); 8650cecf9d5Sandi 8660cecf9d5Sandi function Doku_Handler_Quote(& $CallWriter) { 8670cecf9d5Sandi $this->CallWriter = & $CallWriter; 8680cecf9d5Sandi } 8690cecf9d5Sandi 8700cecf9d5Sandi function writeCall($call) { 8710cecf9d5Sandi $this->calls[] = $call; 8720cecf9d5Sandi } 8730cecf9d5Sandi 8740cecf9d5Sandi // Probably not needed but just in case... 8750cecf9d5Sandi function writeCalls($calls) { 8760cecf9d5Sandi $this->calls = array_merge($this->calls, $calls); 8770cecf9d5Sandi $this->CallWriter->writeCalls($this->calls); 8780cecf9d5Sandi } 8790cecf9d5Sandi 8800cecf9d5Sandi function process() { 8810cecf9d5Sandi 8820cecf9d5Sandi $quoteDepth = 1; 8830cecf9d5Sandi 8840cecf9d5Sandi foreach ( $this->calls as $call ) { 8850cecf9d5Sandi switch ($call[0]) { 8860cecf9d5Sandi 8870cecf9d5Sandi case 'quote_start': 8880cecf9d5Sandi 8890cecf9d5Sandi $this->quoteCalls[] = array('quote_open',array(),$call[2]); 8900cecf9d5Sandi 8910cecf9d5Sandi case 'quote_newline': 8920cecf9d5Sandi 8930cecf9d5Sandi $quoteLength = $this->getDepth($call[1][0]); 8940cecf9d5Sandi 8950cecf9d5Sandi if ( $quoteLength > $quoteDepth ) { 8960cecf9d5Sandi $quoteDiff = $quoteLength - $quoteDepth; 8970cecf9d5Sandi for ( $i = 1; $i <= $quoteDiff; $i++ ) { 8980cecf9d5Sandi $this->quoteCalls[] = array('quote_open',array(),$call[2]); 8990cecf9d5Sandi } 9000cecf9d5Sandi } else if ( $quoteLength < $quoteDepth ) { 9010cecf9d5Sandi $quoteDiff = $quoteDepth - $quoteLength; 9020cecf9d5Sandi for ( $i = 1; $i <= $quoteDiff; $i++ ) { 9030cecf9d5Sandi $this->quoteCalls[] = array('quote_close',array(),$call[2]); 9040cecf9d5Sandi } 9050cecf9d5Sandi } 9060cecf9d5Sandi 9070cecf9d5Sandi $quoteDepth = $quoteLength; 9080cecf9d5Sandi 9090cecf9d5Sandi break; 9100cecf9d5Sandi 9110cecf9d5Sandi case 'quote_end': 9120cecf9d5Sandi 9130cecf9d5Sandi if ( $quoteDepth > 1 ) { 9140cecf9d5Sandi $quoteDiff = $quoteDepth - 1; 9150cecf9d5Sandi for ( $i = 1; $i <= $quoteDiff; $i++ ) { 9160cecf9d5Sandi $this->quoteCalls[] = array('quote_close',array(),$call[2]); 9170cecf9d5Sandi } 9180cecf9d5Sandi } 9190cecf9d5Sandi 9200cecf9d5Sandi $this->quoteCalls[] = array('quote_close',array(),$call[2]); 9210cecf9d5Sandi 9220cecf9d5Sandi $this->CallWriter->writeCalls($this->quoteCalls); 9230cecf9d5Sandi break; 9240cecf9d5Sandi 9250cecf9d5Sandi default: 9260cecf9d5Sandi $this->quoteCalls[] = $call; 9270cecf9d5Sandi break; 9280cecf9d5Sandi } 9290cecf9d5Sandi } 9300cecf9d5Sandi } 9310cecf9d5Sandi 9320cecf9d5Sandi function getDepth($marker) { 9330cecf9d5Sandi preg_match('/>{1,}/', $marker, $matches); 9340cecf9d5Sandi $quoteLength = strlen($matches[0]); 9350cecf9d5Sandi return $quoteLength; 9360cecf9d5Sandi } 9370cecf9d5Sandi} 9380cecf9d5Sandi 9390cecf9d5Sandi//------------------------------------------------------------------------ 9400cecf9d5Sandiclass Doku_Handler_Table { 9410cecf9d5Sandi 9420cecf9d5Sandi var $CallWriter; 9430cecf9d5Sandi 9440cecf9d5Sandi var $calls = array(); 9450cecf9d5Sandi var $tableCalls = array(); 9460cecf9d5Sandi var $maxCols = 0; 9470cecf9d5Sandi var $maxRows = 1; 9480cecf9d5Sandi var $currentCols = 0; 9490cecf9d5Sandi var $firstCell = FALSE; 9500cecf9d5Sandi var $lastCellType = 'tablecell'; 9510cecf9d5Sandi 9520cecf9d5Sandi function Doku_Handler_Table(& $CallWriter) { 9530cecf9d5Sandi $this->CallWriter = & $CallWriter; 9540cecf9d5Sandi } 9550cecf9d5Sandi 9560cecf9d5Sandi function writeCall($call) { 9570cecf9d5Sandi $this->calls[] = $call; 9580cecf9d5Sandi } 9590cecf9d5Sandi 9600cecf9d5Sandi // Probably not needed but just in case... 9610cecf9d5Sandi function writeCalls($calls) { 9620cecf9d5Sandi $this->calls = array_merge($this->calls, $calls); 9630cecf9d5Sandi $this->CallWriter->writeCalls($this->calls); 9640cecf9d5Sandi } 9650cecf9d5Sandi 9660cecf9d5Sandi //------------------------------------------------------------------------ 9670cecf9d5Sandi function process() { 9680cecf9d5Sandi foreach ( $this->calls as $call ) { 9690cecf9d5Sandi switch ( $call[0] ) { 9700cecf9d5Sandi case 'table_start': 9710cecf9d5Sandi $this->tableStart($call); 9720cecf9d5Sandi break; 9730cecf9d5Sandi case 'table_row': 9740cecf9d5Sandi $this->tableRowClose(array('tablerow_close',$call[1],$call[2])); 9750cecf9d5Sandi $this->tableRowOpen(array('tablerow_open',$call[1],$call[2])); 9760cecf9d5Sandi break; 9770cecf9d5Sandi case 'tableheader': 9780cecf9d5Sandi case 'tablecell': 9790cecf9d5Sandi $this->tableCell($call); 9800cecf9d5Sandi break; 9810cecf9d5Sandi case 'table_end': 9820cecf9d5Sandi $this->tableRowClose(array('tablerow_close',$call[1],$call[2])); 9830cecf9d5Sandi $this->tableEnd($call); 9840cecf9d5Sandi break; 9850cecf9d5Sandi default: 9860cecf9d5Sandi $this->tableDefault($call); 9870cecf9d5Sandi break; 9880cecf9d5Sandi } 9890cecf9d5Sandi } 9900cecf9d5Sandi $this->CallWriter->writeCalls($this->tableCalls); 9910cecf9d5Sandi } 9920cecf9d5Sandi 9930cecf9d5Sandi function tableStart($call) { 9940cecf9d5Sandi $this->tableCalls[] = array('table_open',array(),$call[2]); 9950cecf9d5Sandi $this->tableCalls[] = array('tablerow_open',array(),$call[2]); 9960cecf9d5Sandi $this->firstCell = TRUE; 9970cecf9d5Sandi } 9980cecf9d5Sandi 9990cecf9d5Sandi function tableEnd($call) { 10000cecf9d5Sandi $this->tableCalls[] = array('table_close',array(),$call[2]); 10010cecf9d5Sandi $this->finalizeTable(); 10020cecf9d5Sandi } 10030cecf9d5Sandi 10040cecf9d5Sandi function tableRowOpen($call) { 10050cecf9d5Sandi $this->tableCalls[] = $call; 10060cecf9d5Sandi $this->currentCols = 0; 10070cecf9d5Sandi $this->firstCell = TRUE; 10080cecf9d5Sandi $this->lastCellType = 'tablecell'; 10090cecf9d5Sandi $this->maxRows++; 10100cecf9d5Sandi } 10110cecf9d5Sandi 10120cecf9d5Sandi function tableRowClose($call) { 10130cecf9d5Sandi // Strip off final cell opening and anything after it 10140cecf9d5Sandi while ( $discard = array_pop($this->tableCalls ) ) { 10150cecf9d5Sandi 10160cecf9d5Sandi if ( $discard[0] == 'tablecell_open' || $discard[0] == 'tableheader_open') { 10170cecf9d5Sandi 10180cecf9d5Sandi // Its a spanning element - put it back and close it 10190cecf9d5Sandi if ( $discard[1][0] > 1 ) { 10200cecf9d5Sandi 10210cecf9d5Sandi $this->tableCalls[] = $discard; 10220cecf9d5Sandi if ( strstr($discard[0],'cell') ) { 10230cecf9d5Sandi $name = 'tablecell'; 10240cecf9d5Sandi } else { 10250cecf9d5Sandi $name = 'tableheader'; 10260cecf9d5Sandi } 10270cecf9d5Sandi $this->tableCalls[] = array($name.'_close',array(),$call[2]); 10280cecf9d5Sandi } 10290cecf9d5Sandi 10300cecf9d5Sandi break; 10310cecf9d5Sandi } 10320cecf9d5Sandi } 10330cecf9d5Sandi $this->tableCalls[] = $call; 10340cecf9d5Sandi 10350cecf9d5Sandi if ( $this->currentCols > $this->maxCols ) { 10360cecf9d5Sandi $this->maxCols = $this->currentCols; 10370cecf9d5Sandi } 10380cecf9d5Sandi } 10390cecf9d5Sandi 10400cecf9d5Sandi function tableCell($call) { 10410cecf9d5Sandi if ( !$this->firstCell ) { 10420cecf9d5Sandi 10430cecf9d5Sandi // Increase the span 10440cecf9d5Sandi $lastCall = end($this->tableCalls); 10450cecf9d5Sandi 10460cecf9d5Sandi // A cell call which follows an open cell means an empty cell so span 10470cecf9d5Sandi if ( $lastCall[0] == 'tablecell_open' || $lastCall[0] == 'tableheader_open' ) { 10480cecf9d5Sandi $this->tableCalls[] = array('colspan',array(),$call[2]); 10490cecf9d5Sandi 10500cecf9d5Sandi } 10510cecf9d5Sandi 10520cecf9d5Sandi $this->tableCalls[] = array($this->lastCellType.'_close',array(),$call[2]); 10530cecf9d5Sandi $this->tableCalls[] = array($call[0].'_open',array(1,NULL),$call[2]); 10540cecf9d5Sandi $this->lastCellType = $call[0]; 10550cecf9d5Sandi 10560cecf9d5Sandi } else { 10570cecf9d5Sandi 10580cecf9d5Sandi $this->tableCalls[] = array($call[0].'_open',array(1,NULL),$call[2]); 10590cecf9d5Sandi $this->lastCellType = $call[0]; 10600cecf9d5Sandi $this->firstCell = FALSE; 10610cecf9d5Sandi 10620cecf9d5Sandi } 10630cecf9d5Sandi 10640cecf9d5Sandi $this->currentCols++; 10650cecf9d5Sandi } 10660cecf9d5Sandi 10670cecf9d5Sandi function tableDefault($call) { 10680cecf9d5Sandi $this->tableCalls[] = $call; 10690cecf9d5Sandi } 10700cecf9d5Sandi 10710cecf9d5Sandi function finalizeTable() { 10720cecf9d5Sandi 10730cecf9d5Sandi // Add the max cols and rows to the table opening 10740cecf9d5Sandi if ( $this->tableCalls[0][0] == 'table_open' ) { 10750cecf9d5Sandi // Adjust to num cols not num col delimeters 10760cecf9d5Sandi $this->tableCalls[0][1][] = $this->maxCols - 1; 10770cecf9d5Sandi $this->tableCalls[0][1][] = $this->maxRows; 10780cecf9d5Sandi } else { 10790cecf9d5Sandi trigger_error('First element in table call list is not table_open'); 10800cecf9d5Sandi } 10810cecf9d5Sandi 10820cecf9d5Sandi $lastRow = 0; 10830cecf9d5Sandi $lastCell = 0; 10840cecf9d5Sandi $toDelete = array(); 10850cecf9d5Sandi 10860cecf9d5Sandi // Look for the colspan elements and increment the colspan on the 10870cecf9d5Sandi // previous non-empty opening cell. Once done, delete all the cells 10880cecf9d5Sandi // that contain colspans 10890cecf9d5Sandi foreach ( $this->tableCalls as $key => $call ) { 10900cecf9d5Sandi 10910cecf9d5Sandi if ( $call[0] == 'tablerow_open' ) { 10920cecf9d5Sandi 10930cecf9d5Sandi $lastRow = $key; 10940cecf9d5Sandi 10950cecf9d5Sandi } else if ( $call[0] == 'tablecell_open' || $call[0] == 'tableheader_open' ) { 10960cecf9d5Sandi 10970cecf9d5Sandi $lastCell = $key; 10980cecf9d5Sandi 10990cecf9d5Sandi } else if ( $call[0] == 'table_align' ) { 11000cecf9d5Sandi 11010cecf9d5Sandi // If the previous element was a cell open, align right 11020cecf9d5Sandi if ( $this->tableCalls[$key-1][0] == 'tablecell_open' || $this->tableCalls[$key-1][0] == 'tableheader_open' ) { 11030cecf9d5Sandi $this->tableCalls[$key-1][1][1] = 'right'; 11040cecf9d5Sandi 11050cecf9d5Sandi // If the next element if the close of an element, align either center or left 11060cecf9d5Sandi } else if ( $this->tableCalls[$key+1][0] == 'tablecell_close' || $this->tableCalls[$key+1][0] == 'tableheader_close' ) { 11070cecf9d5Sandi if ( $this->tableCalls[$lastCell][1][1] == 'right' ) { 11080cecf9d5Sandi $this->tableCalls[$lastCell][1][1] = 'center'; 11090cecf9d5Sandi } else { 11100cecf9d5Sandi $this->tableCalls[$lastCell][1][1] = 'left'; 11110cecf9d5Sandi } 11120cecf9d5Sandi 11130cecf9d5Sandi } 11140cecf9d5Sandi 11150cecf9d5Sandi // Now convert the whitespace back to cdata 11160cecf9d5Sandi $this->tableCalls[$key][0] = 'cdata'; 11170cecf9d5Sandi 11180cecf9d5Sandi } else if ( $call[0] == 'colspan' ) { 11190cecf9d5Sandi 11200cecf9d5Sandi $this->tableCalls[$key-1][1][0] = FALSE; 11210cecf9d5Sandi 11220cecf9d5Sandi for($i = $key-2; $i > $lastRow; $i--) { 11230cecf9d5Sandi 11240cecf9d5Sandi if ( $this->tableCalls[$i][0] == 'tablecell_open' || $this->tableCalls[$i][0] == 'tableheader_open' ) { 11250cecf9d5Sandi 11260cecf9d5Sandi if ( FALSE !== $this->tableCalls[$i][1][0] ) { 11270cecf9d5Sandi $this->tableCalls[$i][1][0]++; 11280cecf9d5Sandi break; 11290cecf9d5Sandi } 11300cecf9d5Sandi 11310cecf9d5Sandi 11320cecf9d5Sandi } 11330cecf9d5Sandi } 11340cecf9d5Sandi 11350cecf9d5Sandi $toDelete[] = $key-1; 11360cecf9d5Sandi $toDelete[] = $key; 11370cecf9d5Sandi $toDelete[] = $key+1; 11380cecf9d5Sandi } 11390cecf9d5Sandi } 11400cecf9d5Sandi 11410cecf9d5Sandi foreach ( $toDelete as $delete ) { 11420cecf9d5Sandi unset($this->tableCalls[$delete]); 11430cecf9d5Sandi } 11440cecf9d5Sandi 11450cecf9d5Sandi $this->tableCalls = array_values($this->tableCalls); 11460cecf9d5Sandi } 11470cecf9d5Sandi} 11480cecf9d5Sandi 11490cecf9d5Sandi//------------------------------------------------------------------------ 11500cecf9d5Sandiclass Doku_Handler_Section { 11510cecf9d5Sandi 11520cecf9d5Sandi function process($calls) { 11530cecf9d5Sandi 11540cecf9d5Sandi $sectionCalls = array(); 11550cecf9d5Sandi $inSection = FALSE; 11560cecf9d5Sandi 11570cecf9d5Sandi foreach ( $calls as $call ) { 11580cecf9d5Sandi 11590cecf9d5Sandi if ( $call[0] == 'header' ) { 11600cecf9d5Sandi 11610cecf9d5Sandi if ( $inSection ) { 11620cecf9d5Sandi $sectionCalls[] = array('section_close',array(), $call[2]); 11630cecf9d5Sandi } 11640cecf9d5Sandi 11650cecf9d5Sandi $sectionCalls[] = $call; 11660cecf9d5Sandi $sectionCalls[] = array('section_open',array($call[1][1]), $call[2]); 11670cecf9d5Sandi $inSection = TRUE; 11680cecf9d5Sandi 11690cecf9d5Sandi } else { 11700cecf9d5Sandi $sectionCalls[] = $call; 11710cecf9d5Sandi } 11720cecf9d5Sandi } 11730cecf9d5Sandi 11740cecf9d5Sandi if ( $inSection ) { 11750cecf9d5Sandi $sectionCalls[] = array('section_close',array(), $call[2]); 11760cecf9d5Sandi } 11770cecf9d5Sandi 11780cecf9d5Sandi return $sectionCalls; 11790cecf9d5Sandi } 11800cecf9d5Sandi 11810cecf9d5Sandi} 11820cecf9d5Sandi 11832a27e99aSandi/** 11842a27e99aSandi * Handler for paragraphs 11852a27e99aSandi * 11860b7c14c2Sandi * @author Harry Fuecks <hfuecks@gmail.com> 11872a27e99aSandi */ 11880cecf9d5Sandiclass Doku_Handler_Block { 11890cecf9d5Sandi 11900cecf9d5Sandi var $calls = array(); 11910cecf9d5Sandi 11920cecf9d5Sandi var $blockStack = array(); 11930cecf9d5Sandi 11940cecf9d5Sandi var $inParagraph = FALSE; 11950cecf9d5Sandi var $atStart = TRUE; 119658b56c06Sandi var $skipEolKey = -1; 11970cecf9d5Sandi 11980cecf9d5Sandi // Blocks don't contain linefeeds 11990cecf9d5Sandi var $blockOpen = array( 12000cecf9d5Sandi 'header', 12010cecf9d5Sandi 'listu_open','listo_open','listitem_open', 12020cecf9d5Sandi 'table_open','tablerow_open','tablecell_open','tableheader_open', 12030cecf9d5Sandi 'quote_open', 12040cecf9d5Sandi 'section_open', // Needed to prevent p_open between header and section_open 12050cecf9d5Sandi 'code','file','php','html','hr','preformatted', 12060cecf9d5Sandi ); 12070cecf9d5Sandi 12080cecf9d5Sandi var $blockClose = array( 12090cecf9d5Sandi 'header', 12100cecf9d5Sandi 'listu_close','listo_close','listitem_close', 12110cecf9d5Sandi 'table_close','tablerow_close','tablecell_close','tableheader_close', 12120cecf9d5Sandi 'quote_close', 12130cecf9d5Sandi 'section_close', // Needed to prevent p_close after section_close 12140cecf9d5Sandi 'code','file','php','html','hr','preformatted', 12150cecf9d5Sandi ); 12160cecf9d5Sandi 12170cecf9d5Sandi // Stacks can contain linefeeds 12180cecf9d5Sandi var $stackOpen = array( 12190cecf9d5Sandi 'footnote_open','section_open', 12200cecf9d5Sandi ); 12210cecf9d5Sandi 12220cecf9d5Sandi var $stackClose = array( 12230cecf9d5Sandi 'footnote_close','section_close', 12240cecf9d5Sandi ); 12250cecf9d5Sandi 12262a27e99aSandi /** 12272a27e99aSandi * Close a paragraph if needed 12282a27e99aSandi * 12292a27e99aSandi * This function makes sure there are no empty paragraphs on the stack 12302a27e99aSandi * 12312a27e99aSandi * @author Andreas Gohr <andi@splitbrain.org> 12322a27e99aSandi */ 1233506ae684Sandi function closeParagraph($pos){ 1234506ae684Sandi // look back if there was any content - we don't want empty paragraphs 1235506ae684Sandi $content = ''; 1236506ae684Sandi for($i=count($this->calls)-1; $i>=0; $i--){ 1237506ae684Sandi if($this->calls[$i][0] == 'p_open'){ 1238506ae684Sandi break; 1239506ae684Sandi }elseif($this->calls[$i][0] == 'cdata'){ 1240506ae684Sandi $content .= $this->calls[$i][1][0]; 1241506ae684Sandi }else{ 1242506ae684Sandi $content = 'found markup'; 1243506ae684Sandi break; 1244506ae684Sandi } 1245506ae684Sandi } 1246506ae684Sandi 1247506ae684Sandi if(trim($content)==''){ 1248506ae684Sandi //remove the whole paragraph 1249506ae684Sandi array_splice($this->calls,$i); 1250506ae684Sandi }else{ 1251506ae684Sandi $this->calls[] = array('p_close',array(), $pos); 1252506ae684Sandi } 1253506ae684Sandi } 1254506ae684Sandi 12552a27e99aSandi /** 12562a27e99aSandi * Processes the whole instruction stack to open and close paragraphs 12572a27e99aSandi * 12580b7c14c2Sandi * @author Harry Fuecks <hfuecks@gmail.com> 12592a27e99aSandi * @author Andreas Gohr <andi@splitbrain.org> 12602a27e99aSandi * @todo This thing is really messy and should be rewritten 12612a27e99aSandi */ 12620cecf9d5Sandi function process($calls) { 12630cecf9d5Sandi foreach ( $calls as $key => $call ) { 12640cecf9d5Sandi 12650cecf9d5Sandi // Process blocks which are stack like... (contain linefeeds) 12660cecf9d5Sandi if ( in_array($call[0],$this->stackOpen ) ) { 12670cecf9d5Sandi /* 12680cecf9d5Sandi if ( $this->atStart ) { 12690cecf9d5Sandi $this->calls[] = array('p_open',array(), $call[2]); 12700cecf9d5Sandi $this->atStart = FALSE; 12710cecf9d5Sandi $this->inParagraph = TRUE; 12720cecf9d5Sandi } 12730cecf9d5Sandi */ 12740cecf9d5Sandi $this->calls[] = $call; 12750cecf9d5Sandi 12760cecf9d5Sandi // Hack - footnotes shouldn't immediately contain a p_open 12770cecf9d5Sandi if ( $call[0] != 'footnote_open' ) { 12780cecf9d5Sandi $this->addToStack(); 12790cecf9d5Sandi } else { 12800cecf9d5Sandi $this->addToStack(FALSE); 12810cecf9d5Sandi } 12820cecf9d5Sandi continue; 12830cecf9d5Sandi } 12840cecf9d5Sandi 12850cecf9d5Sandi if ( in_array($call[0],$this->stackClose ) ) { 12860cecf9d5Sandi 12870cecf9d5Sandi if ( $this->inParagraph ) { 1288506ae684Sandi //$this->calls[] = array('p_close',array(), $call[2]); 1289506ae684Sandi $this->closeParagraph($call[2]); 12900cecf9d5Sandi } 12910cecf9d5Sandi $this->calls[] = $call; 12920cecf9d5Sandi $this->removeFromStack(); 12930cecf9d5Sandi continue; 12940cecf9d5Sandi } 12950cecf9d5Sandi 12960cecf9d5Sandi if ( !$this->atStart ) { 12970cecf9d5Sandi 12980cecf9d5Sandi if ( $call[0] == 'eol' ) { 12990cecf9d5Sandi 130058b56c06Sandi 130158b56c06Sandi /* XXX 13020cecf9d5Sandi if ( $this->inParagraph ) { 13030cecf9d5Sandi $this->calls[] = array('p_close',array(), $call[2]); 13040cecf9d5Sandi } 13050cecf9d5Sandi $this->calls[] = array('p_open',array(), $call[2]); 13060cecf9d5Sandi $this->inParagraph = TRUE; 130758b56c06Sandi */ 130858b56c06Sandi 130958b56c06Sandi # Check this isn't an eol instruction to skip... 131058b56c06Sandi if ( $this->skipEolKey != $key ) { 131158b56c06Sandi # Look to see if the next instruction is an EOL 131258b56c06Sandi if ( isset($calls[$key+1]) && $calls[$key+1][0] == 'eol' ) { 131358b56c06Sandi 131458b56c06Sandi if ( $this->inParagraph ) { 1315506ae684Sandi //$this->calls[] = array('p_close',array(), $call[2]); 1316506ae684Sandi $this->closeParagraph($call[2]); 131758b56c06Sandi } 131858b56c06Sandi 131958b56c06Sandi $this->calls[] = array('p_open',array(), $call[2]); 132058b56c06Sandi $this->inParagraph = TRUE; 132158b56c06Sandi 132258b56c06Sandi 132358b56c06Sandi # Mark the next instruction for skipping 132458b56c06Sandi $this->skipEolKey = $key+1; 132558b56c06Sandi 132658b56c06Sandi }else{ 132758b56c06Sandi //if this is just a single eol make a space from it 132858b56c06Sandi $this->calls[] = array('cdata',array(" "), $call[2]); 132958b56c06Sandi } 133058b56c06Sandi } 133158b56c06Sandi 13320cecf9d5Sandi 13330cecf9d5Sandi } else { 13340cecf9d5Sandi 13350cecf9d5Sandi $storeCall = TRUE; 13360cecf9d5Sandi 13370cecf9d5Sandi if ( $this->inParagraph && in_array($call[0], $this->blockOpen) ) { 1338506ae684Sandi //$this->calls[] = array('p_close',array(), $call[2]); 1339506ae684Sandi $this->closeParagraph($call[2]); 13400cecf9d5Sandi $this->inParagraph = FALSE; 13410cecf9d5Sandi $this->calls[] = $call; 13420cecf9d5Sandi $storeCall = FALSE; 13430cecf9d5Sandi } 13440cecf9d5Sandi 13450cecf9d5Sandi if ( in_array($call[0], $this->blockClose) ) { 13460cecf9d5Sandi if ( $this->inParagraph ) { 1347506ae684Sandi //$this->calls[] = array('p_close',array(), $call[2]); 1348506ae684Sandi $this->closeParagraph($call[2]); 13490cecf9d5Sandi $this->inParagraph = FALSE; 13500cecf9d5Sandi } 13510cecf9d5Sandi if ( $storeCall ) { 13520cecf9d5Sandi $this->calls[] = $call; 13530cecf9d5Sandi $storeCall = FALSE; 13540cecf9d5Sandi } 13550cecf9d5Sandi 13560cecf9d5Sandi // This really sucks and suggests this whole class sucks but... 13570cecf9d5Sandi if ( isset($calls[$key+1]) 13580cecf9d5Sandi && 13590cecf9d5Sandi !in_array($calls[$key+1][0], $this->blockOpen) 13600cecf9d5Sandi && 13610cecf9d5Sandi !in_array($calls[$key+1][0], $this->blockClose) 13620cecf9d5Sandi ) { 13630cecf9d5Sandi 13640cecf9d5Sandi $this->calls[] = array('p_open',array(), $call[2]); 13650cecf9d5Sandi $this->inParagraph = TRUE; 13660cecf9d5Sandi } 13670cecf9d5Sandi } 13680cecf9d5Sandi 13690cecf9d5Sandi if ( $storeCall ) { 13700cecf9d5Sandi $this->calls[] = $call; 13710cecf9d5Sandi } 13720cecf9d5Sandi 13730cecf9d5Sandi } 13740cecf9d5Sandi 13750cecf9d5Sandi 13760cecf9d5Sandi } else { 13770cecf9d5Sandi 13780cecf9d5Sandi // Unless there's already a block at the start, start a paragraph 13790cecf9d5Sandi if ( !in_array($call[0],$this->blockOpen) ) { 13800cecf9d5Sandi $this->calls[] = array('p_open',array(), $call[2]); 13810cecf9d5Sandi if ( $call[0] != 'eol' ) { 13820cecf9d5Sandi $this->calls[] = $call; 13830cecf9d5Sandi } 13840cecf9d5Sandi $this->atStart = FALSE; 13850cecf9d5Sandi $this->inParagraph = TRUE; 13860cecf9d5Sandi } else { 13870cecf9d5Sandi $this->calls[] = $call; 13880cecf9d5Sandi $this->atStart = FALSE; 13890cecf9d5Sandi } 13900cecf9d5Sandi 13910cecf9d5Sandi } 13920cecf9d5Sandi 13930cecf9d5Sandi } 13940cecf9d5Sandi 13950cecf9d5Sandi if ( $this->inParagraph ) { 13960cecf9d5Sandi if ( $call[0] == 'p_open' ) { 13970cecf9d5Sandi // Ditch the last call 13980cecf9d5Sandi array_pop($this->calls); 13990cecf9d5Sandi } else if ( !in_array($call[0], $this->blockClose) ) { 1400506ae684Sandi //$this->calls[] = array('p_close',array(), $call[2]); 1401506ae684Sandi $this->closeParagraph($call[2]); 14020cecf9d5Sandi } else { 14030cecf9d5Sandi $last_call = array_pop($this->calls); 1404506ae684Sandi //$this->calls[] = array('p_close',array(), $call[2]); 1405506ae684Sandi $this->closeParagraph($call[2]); 14060cecf9d5Sandi $this->calls[] = $last_call; 14070cecf9d5Sandi } 14080cecf9d5Sandi } 14090cecf9d5Sandi 14100cecf9d5Sandi return $this->calls; 14110cecf9d5Sandi } 14120cecf9d5Sandi 14130cecf9d5Sandi function addToStack($newStart = TRUE) { 14140cecf9d5Sandi $this->blockStack[] = array($this->atStart, $this->inParagraph); 14150cecf9d5Sandi $this->atStart = $newStart; 14160cecf9d5Sandi $this->inParagraph = FALSE; 14170cecf9d5Sandi } 14180cecf9d5Sandi 14190cecf9d5Sandi function removeFromStack() { 14200cecf9d5Sandi $state = array_pop($this->blockStack); 14210cecf9d5Sandi $this->atStart = $state[0]; 14220cecf9d5Sandi $this->inParagraph = $state[1]; 14230cecf9d5Sandi } 14240cecf9d5Sandi} 14252a27e99aSandi 14260cecf9d5Sandi//------------------------------------------------------------------------ 14270cecf9d5Sandidefine('DOKU_TOC_OPEN',1); 14280cecf9d5Sandidefine('DOKU_TOCBRANCH_OPEN',2); 14290cecf9d5Sandidefine('DOKU_TOCITEM_OPEN',3); 14300cecf9d5Sandidefine('DOKU_TOC_ELEMENT',4); 14310cecf9d5Sandidefine('DOKU_TOCITEM_CLOSE',5); 14320cecf9d5Sandidefine('DOKU_TOCBRANCH_CLOSE',6); 14330cecf9d5Sandidefine('DOKU_TOC_CLOSE',7); 14340cecf9d5Sandi 14350cecf9d5Sandiclass Doku_Handler_Toc { 14360cecf9d5Sandi 14370cecf9d5Sandi var $calls = array(); 14380cecf9d5Sandi var $tocStack = array(); 14390cecf9d5Sandi var $toc = array(); 14400cecf9d5Sandi var $numHeaders = 0; 14410cecf9d5Sandi 14420cecf9d5Sandi function process($calls) { 1443af587fa8Sandi #FIXME can this be done better? 1444af587fa8Sandi global $conf; 14450cecf9d5Sandi 14460cecf9d5Sandi foreach ( $calls as $call ) { 1447af587fa8Sandi if ( $call[0] == 'header' && $call[1][1] <= $conf['maxtoclevel'] ) { 14480cecf9d5Sandi $this->numHeaders++; 14490cecf9d5Sandi $this->addToToc($call); 14500cecf9d5Sandi } 14510cecf9d5Sandi $this->calls[] = $call; 14520cecf9d5Sandi } 14530cecf9d5Sandi 14540cecf9d5Sandi // Complete the table of contents then prepend to the calls 14550cecf9d5Sandi $this->finalizeToc($call); 14560cecf9d5Sandi return $this->calls; 14570cecf9d5Sandi } 14580cecf9d5Sandi 14590cecf9d5Sandi function addToToc($call) { 14600cecf9d5Sandi 14610cecf9d5Sandi $depth = $call[1][1]; 14620cecf9d5Sandi 14630cecf9d5Sandi // If it's the opening item... 14640cecf9d5Sandi if ( count ( $this->toc) == 0 ) { 14650cecf9d5Sandi 14660cecf9d5Sandi $this->addTocCall($call, DOKU_TOC_OPEN); 14670cecf9d5Sandi 14680cecf9d5Sandi for ( $i = 1; $i <= $depth; $i++ ) { 14690cecf9d5Sandi 14700cecf9d5Sandi $this->addTocCall(array($call[0],array($call[1][0],$i),$call[2]), DOKU_TOCBRANCH_OPEN); 14710cecf9d5Sandi 14720cecf9d5Sandi if ( $i != $depth ) { 14733926d381Sandi $this->addTocCall(array($call[0],array($call[1][0], $i, '', TRUE),$call[2]), DOKU_TOCITEM_OPEN); 14740cecf9d5Sandi } else { 14750cecf9d5Sandi $this->addTocCall(array($call[0],array($call[1][0], $i),$call[2]), DOKU_TOCITEM_OPEN); 14760cecf9d5Sandi $this->addTocCall(array($call[0],array($call[1][0], $i),$call[2]), DOKU_TOC_ELEMENT); 14770cecf9d5Sandi } 14780cecf9d5Sandi 14790cecf9d5Sandi $this->tocStack[] = $i; 14800cecf9d5Sandi 14810cecf9d5Sandi } 14820cecf9d5Sandi return; 14830cecf9d5Sandi } 14840cecf9d5Sandi 14850cecf9d5Sandi $currentDepth = end($this->tocStack); 14860cecf9d5Sandi $initialDepth = $currentDepth; 14870cecf9d5Sandi 14880cecf9d5Sandi // Create new branches as needed 14890cecf9d5Sandi if ( $depth > $currentDepth ) { 14900cecf9d5Sandi 14910cecf9d5Sandi for ($i = $currentDepth+1; $i <= $depth; $i++ ) { 14920cecf9d5Sandi $this->addTocCall(array($call[0],array($call[1][0],$i),$call[2]), DOKU_TOCBRANCH_OPEN); 14930cecf9d5Sandi // It's just a filler 14940cecf9d5Sandi if ( $i != $depth ) { 14953926d381Sandi $this->addTocCall(array($call[0],array($call[1][0], $i, '', TRUE),$call[2]), DOKU_TOCITEM_OPEN); 14960cecf9d5Sandi } else { 14970cecf9d5Sandi $this->addTocCall(array($call[0],array($call[1][0], $i),$call[2]), DOKU_TOCITEM_OPEN); 14980cecf9d5Sandi } 14990cecf9d5Sandi $this->tocStack[] = $i; 15000cecf9d5Sandi } 15010cecf9d5Sandi 15020cecf9d5Sandi $currentDepth = $i-1; 15030cecf9d5Sandi 15040cecf9d5Sandi } 15050cecf9d5Sandi 15060cecf9d5Sandi // Going down 15070cecf9d5Sandi if ( $depth < $currentDepth ) { 15080cecf9d5Sandi for ( $i = $currentDepth; $i >= $depth; $i-- ) { 15090cecf9d5Sandi if ( $i != $depth ) { 15100cecf9d5Sandi array_pop($this->tocStack); 15110cecf9d5Sandi $this->addTocCall(array($call[0],array($call[1][0],$i),$call[2]), DOKU_TOCITEM_CLOSE); 15120cecf9d5Sandi $this->addTocCall(array($call[0],array($call[1][0],$i),$call[2]), DOKU_TOCBRANCH_CLOSE); 15130cecf9d5Sandi } else { 15140cecf9d5Sandi $this->addTocCall(array($call[0],array($call[1][0],$i),$call[2]), DOKU_TOCITEM_CLOSE); 15150cecf9d5Sandi $this->addTocCall(array($call[0],array($call[1][0],$i),$call[2]), DOKU_TOCITEM_OPEN); 15160cecf9d5Sandi $this->addTocCall($call, DOKU_TOC_ELEMENT); 15170cecf9d5Sandi return; 15180cecf9d5Sandi } 15190cecf9d5Sandi } 15200cecf9d5Sandi } 15210cecf9d5Sandi 15220cecf9d5Sandi if ( $depth == $initialDepth ) { 15230cecf9d5Sandi $this->addTocCall($call, DOKU_TOCITEM_CLOSE); 15240cecf9d5Sandi $this->addTocCall($call, DOKU_TOCITEM_OPEN); 15250cecf9d5Sandi } 15260cecf9d5Sandi 15270cecf9d5Sandi $this->addTocCall($call, DOKU_TOC_ELEMENT); 15280cecf9d5Sandi 15290cecf9d5Sandi 15300cecf9d5Sandi } 15310cecf9d5Sandi 15320cecf9d5Sandi function addTocCall($call, $type) { 15330cecf9d5Sandi switch ( $type ) { 15340cecf9d5Sandi case DOKU_TOC_OPEN: 15350cecf9d5Sandi $this->toc[] = array('toc_open',array(),$call[2]); 15360cecf9d5Sandi break; 15370cecf9d5Sandi 15380cecf9d5Sandi case DOKU_TOCBRANCH_OPEN: 15390cecf9d5Sandi $this->toc[] = array('tocbranch_open',array($call[1][1]),$call[2]); 15400cecf9d5Sandi break; 15410cecf9d5Sandi 15420cecf9d5Sandi case DOKU_TOCITEM_OPEN: 15433926d381Sandi if ( isset( $call[1][3] ) ) { 15440cecf9d5Sandi $this->toc[] = array('tocitem_open',array($call[1][1], TRUE),$call[2]); 15450cecf9d5Sandi } else { 15460cecf9d5Sandi $this->toc[] = array('tocitem_open',array($call[1][1]),$call[2]); 15470cecf9d5Sandi } 15480cecf9d5Sandi break; 15490cecf9d5Sandi 15500cecf9d5Sandi case DOKU_TOC_ELEMENT: 15510cecf9d5Sandi $this->toc[] = array('tocelement',array($call[1][1],$call[1][0]),$call[2]); 15520cecf9d5Sandi break; 15530cecf9d5Sandi 15540cecf9d5Sandi case DOKU_TOCITEM_CLOSE: 15550cecf9d5Sandi $this->toc[] = array('tocitem_close',array($call[1][1]),$call[2]); 15560cecf9d5Sandi break; 15570cecf9d5Sandi 15580cecf9d5Sandi case DOKU_TOCBRANCH_CLOSE: 15590cecf9d5Sandi $this->toc[] = array('tocbranch_close',array($call[1][1]),$call[2]); 15600cecf9d5Sandi break; 15610cecf9d5Sandi 15620cecf9d5Sandi case DOKU_TOC_CLOSE: 15630cecf9d5Sandi if ( count($this->toc) > 0 ) { 15640cecf9d5Sandi $this->toc[] = array('toc_close',array(),$call[2]); 15650cecf9d5Sandi } 15660cecf9d5Sandi break; 15670cecf9d5Sandi } 15680cecf9d5Sandi } 15690cecf9d5Sandi 15700cecf9d5Sandi function finalizeToc($call) { 15713926d381Sandi global $conf; 15723926d381Sandi if ( $this->numHeaders < $conf['maxtoclevel'] ) { 15730cecf9d5Sandi return; 15740cecf9d5Sandi } 15750cecf9d5Sandi if ( count ($this->tocStack) > 0 ) { 15760cecf9d5Sandi while ( NULL !== ($toc = array_pop($this->tocStack)) ) { 15770cecf9d5Sandi $this->addTocCall(array($call[0],array('',$toc),$call[2]), DOKU_TOCITEM_CLOSE); 15780cecf9d5Sandi $this->addTocCall(array($call[0],array('',$toc),$call[2]), DOKU_TOCBRANCH_CLOSE); 15790cecf9d5Sandi } 15800cecf9d5Sandi } 15810cecf9d5Sandi $this->addTocCall($call, DOKU_TOC_CLOSE); 15820cecf9d5Sandi $this->calls = array_merge($this->toc, $this->calls); 15830cecf9d5Sandi } 15840cecf9d5Sandi 15850cecf9d5Sandi} 15860cecf9d5Sandi 1587340756e4Sandi 15884826ab45Sandi//Setup VIM: ex: et ts=4 enc=utf-8 : 1589