12e6b7ea4Sfiwswe<?php 22e6b7ea4Sfiwswe/** 32e6b7ea4Sfiwswe * DokuWiki Plugin embeddedphp (Syntax Component) 42e6b7ea4Sfiwswe * 514c252b2Sfiwswe * @license GPL 2 http://www.gnu.org/licenses/gpl-2.0.html 62e6b7ea4Sfiwswe * @author fiwswe <dwplugin@fwml.de> 72e6b7ea4Sfiwswe */ 869545d01Sfiwswe 92e6b7ea4Sfiwsweclass syntax_plugin_embeddedphp_phpinline extends \dokuwiki\Extension\SyntaxPlugin 102e6b7ea4Sfiwswe{ 112e6b7ea4Sfiwswe /** 122e6b7ea4Sfiwswe * Return the tag this plugin instance reacts to 132e6b7ea4Sfiwswe * 142e6b7ea4Sfiwswe * @return string 152e6b7ea4Sfiwswe */ 1669545d01Sfiwswe protected function GetTag(): string 172e6b7ea4Sfiwswe { 182e6b7ea4Sfiwswe return 'php'; 192e6b7ea4Sfiwswe } 202e6b7ea4Sfiwswe 212e6b7ea4Sfiwswe /** @inheritDoc */ 222e6b7ea4Sfiwswe public function getType() 232e6b7ea4Sfiwswe { 2469545d01Sfiwswe return 'protected'; 252e6b7ea4Sfiwswe } 262e6b7ea4Sfiwswe 272e6b7ea4Sfiwswe /** @inheritDoc */ 282e6b7ea4Sfiwswe public function getSort() 292e6b7ea4Sfiwswe { 302e6b7ea4Sfiwswe // The default <php>/<PHP> handler up to "Igor" has priority 180. By setting a 312e6b7ea4Sfiwswe // lower priority we override the built-in functionality. 322e6b7ea4Sfiwswe return 179; 332e6b7ea4Sfiwswe } 342e6b7ea4Sfiwswe 35*144f9a0aSfiwswe 362e6b7ea4Sfiwswe /* 372e6b7ea4Sfiwswe * Return the plugin Lexer mode 38*144f9a0aSfiwswe * This works fine for most trivial cases. But some plugins 39*144f9a0aSfiwswe * may need to override this method. 402e6b7ea4Sfiwswe * 412e6b7ea4Sfiwswe * @return string 422e6b7ea4Sfiwswe */ 4369545d01Sfiwswe protected function getPluginModeName(): string 442e6b7ea4Sfiwswe { 45*144f9a0aSfiwswe $x = ['plugin', 46*144f9a0aSfiwswe $this->getPluginName(), 47*144f9a0aSfiwswe $this->getPluginComponent()]; // If component is empty it will be filtered later. 48*144f9a0aSfiwswe 49*144f9a0aSfiwswe return implode('_', array_filter($x)); 502e6b7ea4Sfiwswe } 512e6b7ea4Sfiwswe 522e6b7ea4Sfiwswe /** @inheritDoc */ 532e6b7ea4Sfiwswe public function connectTo($mode) 542e6b7ea4Sfiwswe { 5569545d01Sfiwswe $p = '<'.$this->GetTag().'\b>(?=.*?</'.$this->GetTag().'>)'; 562e6b7ea4Sfiwswe $m = $this->getPluginModeName(); 572e6b7ea4Sfiwswe $this->Lexer->addEntryPattern($p, $mode, $m); 582e6b7ea4Sfiwswe } 592e6b7ea4Sfiwswe 602e6b7ea4Sfiwswe /** @inheritDoc */ 612e6b7ea4Sfiwswe public function postConnect() 622e6b7ea4Sfiwswe { 632e6b7ea4Sfiwswe $p = '</'.$this->GetTag().'>'; 642e6b7ea4Sfiwswe $m = $this->getPluginModeName(); 652e6b7ea4Sfiwswe $this->Lexer->addExitPattern($p, $m); 662e6b7ea4Sfiwswe } 672e6b7ea4Sfiwswe 682e6b7ea4Sfiwswe /** @inheritDoc */ 692e6b7ea4Sfiwswe public function handle($match, $state, $pos, Doku_Handler $handler) 702e6b7ea4Sfiwswe { 7169545d01Sfiwswe global $INPUT; 7269545d01Sfiwswe 732e6b7ea4Sfiwswe // If we are parsing a submitted comment. Executing embedded PHP in comments is 742e6b7ea4Sfiwswe // not a good idea! 7569545d01Sfiwswe if ($INPUT->has('comment')) { 762e6b7ea4Sfiwswe return false; 772e6b7ea4Sfiwswe } 782e6b7ea4Sfiwswe 792e6b7ea4Sfiwswe switch($state) { 802e6b7ea4Sfiwswe case DOKU_LEXER_UNMATCHED : 812e6b7ea4Sfiwswe // Return the data needed in $this->render() as an array: 822e6b7ea4Sfiwswe return [$state, $match]; 832e6b7ea4Sfiwswe } 842e6b7ea4Sfiwswe 852e6b7ea4Sfiwswe return false; 862e6b7ea4Sfiwswe } 872e6b7ea4Sfiwswe 882e6b7ea4Sfiwswe /** @inheritDoc */ 892e6b7ea4Sfiwswe public function render($mode, Doku_Renderer $renderer, $data) 902e6b7ea4Sfiwswe { 912e6b7ea4Sfiwswe if ($mode === 'xhtml') { 922e6b7ea4Sfiwswe if (is_array($data) && (count($data) > 1)) { 932e6b7ea4Sfiwswe $this->php($data[1], $renderer); 942e6b7ea4Sfiwswe 952e6b7ea4Sfiwswe return true; 962e6b7ea4Sfiwswe } 972e6b7ea4Sfiwswe } 982e6b7ea4Sfiwswe 992e6b7ea4Sfiwswe return false; 1002e6b7ea4Sfiwswe } 1012e6b7ea4Sfiwswe 1022e6b7ea4Sfiwswe /** 1032e6b7ea4Sfiwswe * Determine whether embedding PHP code is allowed 1042e6b7ea4Sfiwswe * 1052e6b7ea4Sfiwswe * @return bool true if executing embedded PHP code is allowed 1062e6b7ea4Sfiwswe */ 10769545d01Sfiwswe protected function allowEmbedding(): bool 1082e6b7ea4Sfiwswe { 1092e6b7ea4Sfiwswe $allow = ($this->getConf('embedphpok') == 1) && 1102e6b7ea4Sfiwswe ($this->getConf('privatewiki') == 1); 1112e6b7ea4Sfiwswe 1122e6b7ea4Sfiwswe return $allow; 1132e6b7ea4Sfiwswe } 1142e6b7ea4Sfiwswe 1152e6b7ea4Sfiwswe /** 1162e6b7ea4Sfiwswe * Execute PHP code if allowed 1172e6b7ea4Sfiwswe * 1182e6b7ea4Sfiwswe * @param string $text PHP code that is either executed or printed 1192e6b7ea4Sfiwswe * @param Doku_Renderer $renderer Renderer used for output 1202e6b7ea4Sfiwswe */ 12169545d01Sfiwswe protected function php($text, Doku_Renderer $renderer): void 12269545d01Sfiwswe { 1232e6b7ea4Sfiwswe if ($this->allowEmbedding()) { 1242e6b7ea4Sfiwswe ob_start(); 1252e6b7ea4Sfiwswe eval($text); 1262e6b7ea4Sfiwswe $o = ob_get_contents(); 1272e6b7ea4Sfiwswe if (!empty($o)) { 1282e6b7ea4Sfiwswe if ($this->isBlockElement()) { 1292e6b7ea4Sfiwswe $renderer->doc .= '<div class="embeddedphp">'.$o.'</div>'; 1302e6b7ea4Sfiwswe } else { 1312e6b7ea4Sfiwswe $renderer->doc .= '<span class="embeddedphp">'.$o.'</span>'; 1322e6b7ea4Sfiwswe } 1332e6b7ea4Sfiwswe } 1342e6b7ea4Sfiwswe ob_end_clean(); 1352e6b7ea4Sfiwswe } else { 13669545d01Sfiwswe $wrapper = $this->isBlockElement() ? 'pre' : 'code'; 1372e6b7ea4Sfiwswe $renderer->doc .= /*'###.get_class($this)'.*/p_xhtml_cached_geshi($text, 'php', $wrapper); 1382e6b7ea4Sfiwswe } 1392e6b7ea4Sfiwswe } 1402e6b7ea4Sfiwswe 1412e6b7ea4Sfiwswe /** 1422e6b7ea4Sfiwswe * Generic test to differentiate between inline and block modes 1432e6b7ea4Sfiwswe * 1442e6b7ea4Sfiwswe * @return bool true if this generates a block element, false otherwise. 1452e6b7ea4Sfiwswe */ 14669545d01Sfiwswe protected function isBlockElement(): bool 1472e6b7ea4Sfiwswe { 1482e6b7ea4Sfiwswe return false; 1492e6b7ea4Sfiwswe } 1502e6b7ea4Sfiwswe} 1512e6b7ea4Sfiwswe 152