1257dd7f8SAndreas Gohr<?php 2d6d97f60SAnna Dabrowska 3257dd7f8SAndreas Gohr/** 4257dd7f8SAndreas Gohr * DokuWiki Plugin struct (Syntax Component) 5257dd7f8SAndreas Gohr * 6257dd7f8SAndreas Gohr * @license GPL 2 http://www.gnu.org/licenses/gpl-2.0.html 7257dd7f8SAndreas Gohr * @author Andreas Gohr, Michael Große <dokuwiki@cosmocode.de> 8257dd7f8SAndreas Gohr */ 9257dd7f8SAndreas Gohr 107234bfb1Ssplitbrainuse dokuwiki\Extension\SyntaxPlugin; 117234bfb1Ssplitbrainuse dokuwiki\Extension\Event; 12f411d872SAndreas Gohruse dokuwiki\plugin\struct\meta\AccessTable; 13ba766201SAndreas Gohruse dokuwiki\plugin\struct\meta\Assignments; 147cbcfbdbSAndreas Gohruse dokuwiki\plugin\struct\meta\StructException; 15257dd7f8SAndreas Gohr 167234bfb1Ssplitbrainclass syntax_plugin_struct_output extends SyntaxPlugin 17d6d97f60SAnna Dabrowska{ 1887050b53SMichael Grosse protected $hasBeenRendered = false; 1987050b53SMichael Grosse 2017a3a578SAndreas Gohr protected const XHTML_OPEN = '<div id="plugin__struct_output">'; 2117a3a578SAndreas Gohr protected const XHTML_CLOSE = '</div>'; 227938ca48SAndreas Gohr 23257dd7f8SAndreas Gohr /** 24979b16f8SAndreas Gohr * Regexp to check on which actions the struct data may be rendered 25979b16f8SAndreas Gohr */ 2617a3a578SAndreas Gohr protected const WHITELIST_ACTIONS = '/^(show|export_.*)$/'; 27979b16f8SAndreas Gohr 28979b16f8SAndreas Gohr /** 29257dd7f8SAndreas Gohr * @return string Syntax mode type 30257dd7f8SAndreas Gohr */ 31d6d97f60SAnna Dabrowska public function getType() 32d6d97f60SAnna Dabrowska { 33257dd7f8SAndreas Gohr return 'substition'; 34257dd7f8SAndreas Gohr } 35da30fdd3SAndreas Gohr 36257dd7f8SAndreas Gohr /** 37257dd7f8SAndreas Gohr * @return string Paragraph type 38257dd7f8SAndreas Gohr */ 39d6d97f60SAnna Dabrowska public function getPType() 40d6d97f60SAnna Dabrowska { 41257dd7f8SAndreas Gohr return 'block'; 42257dd7f8SAndreas Gohr } 43da30fdd3SAndreas Gohr 44257dd7f8SAndreas Gohr /** 45257dd7f8SAndreas Gohr * @return int Sort order - Low numbers go before high numbers 46257dd7f8SAndreas Gohr */ 47d6d97f60SAnna Dabrowska public function getSort() 48d6d97f60SAnna Dabrowska { 49257dd7f8SAndreas Gohr return 155; 50257dd7f8SAndreas Gohr } 51257dd7f8SAndreas Gohr 52257dd7f8SAndreas Gohr /** 53257dd7f8SAndreas Gohr * Connect lookup pattern to lexer. 54257dd7f8SAndreas Gohr * 5582c064c1SAndreas Gohr * We do not connect any pattern here, because the call to this plugin is not 5682c064c1SAndreas Gohr * triggered from syntax but our action component 5782c064c1SAndreas Gohr * 5882c064c1SAndreas Gohr * @asee action_plugin_struct_output 59257dd7f8SAndreas Gohr * @param string $mode Parser mode 60257dd7f8SAndreas Gohr */ 61d6d97f60SAnna Dabrowska public function connectTo($mode) 62d6d97f60SAnna Dabrowska { 63257dd7f8SAndreas Gohr } 64257dd7f8SAndreas Gohr 65257dd7f8SAndreas Gohr /** 66257dd7f8SAndreas Gohr * Handle matches of the struct syntax 67257dd7f8SAndreas Gohr * 68257dd7f8SAndreas Gohr * @param string $match The match of the syntax 69257dd7f8SAndreas Gohr * @param int $state The state of the handler 70257dd7f8SAndreas Gohr * @param int $pos The position in the document 71257dd7f8SAndreas Gohr * @param Doku_Handler $handler The handler 72257dd7f8SAndreas Gohr * @return array Data for the renderer 73257dd7f8SAndreas Gohr */ 74d6d97f60SAnna Dabrowska public function handle($match, $state, $pos, Doku_Handler $handler) 75d6d97f60SAnna Dabrowska { 7682c064c1SAndreas Gohr // this is never called 777234bfb1Ssplitbrain return []; 78257dd7f8SAndreas Gohr } 79257dd7f8SAndreas Gohr 80257dd7f8SAndreas Gohr /** 8182c064c1SAndreas Gohr * Render schema data 82257dd7f8SAndreas Gohr * 83564e138bSAnna Dabrowska * Currently completely renderer agnostic 8482c064c1SAndreas Gohr * 8534ea6e10SAnna Dabrowska * @param string $format Renderer format 8634ea6e10SAnna Dabrowska * @param Doku_Renderer $renderer The renderer 87257dd7f8SAndreas Gohr * @param array $data The data from the handler() function 88257dd7f8SAndreas Gohr * @return bool If rendering was successful. 89257dd7f8SAndreas Gohr */ 9034ea6e10SAnna Dabrowska public function render($format, Doku_Renderer $renderer, $data) 91d6d97f60SAnna Dabrowska { 920e4a3e7cSMichael Große global $ACT; 93257dd7f8SAndreas Gohr global $ID; 9482c064c1SAndreas Gohr global $INFO; 95257dd7f8SAndreas Gohr global $REV; 96bdefb930SAnna Dabrowska 9734ea6e10SAnna Dabrowska foreach (helper_plugin_struct::BLACKLIST_RENDERER as $blacklisted) { 9834ea6e10SAnna Dabrowska if ($renderer instanceof $blacklisted) { 990e4a3e7cSMichael Große return true; 1000e4a3e7cSMichael Große } 101bdefb930SAnna Dabrowska } 102be4dde77SAndreas Gohr if (!isset($INFO['id']) || ($ID != $INFO['id'])) return true; 1032f1a213bSAndreas Gohr if (!$INFO['exists']) return true; 10487050b53SMichael Grosse if ($this->hasBeenRendered) return true; 105979b16f8SAndreas Gohr if (!preg_match(self::WHITELIST_ACTIONS, act_clean($ACT))) return true; 10687050b53SMichael Grosse 10787050b53SMichael Grosse // do not render the output twice on the same page, e.g. when another page has been included 10887050b53SMichael Grosse $this->hasBeenRendered = true; 1097cbcfbdbSAndreas Gohr try { 110025cb9daSAndreas Gohr $assignments = Assignments::getInstance(); 1117cbcfbdbSAndreas Gohr } catch (StructException $e) { 1127cbcfbdbSAndreas Gohr return false; 1137cbcfbdbSAndreas Gohr } 114257dd7f8SAndreas Gohr $tables = $assignments->getPageAssignments($ID); 115257dd7f8SAndreas Gohr if (!$tables) return true; 116257dd7f8SAndreas Gohr 11734ea6e10SAnna Dabrowska if ($format == 'xhtml') $renderer->doc .= self::XHTML_OPEN; 1185a1eab78SAndreas Gohr 1197938ca48SAndreas Gohr $hasdata = false; 120257dd7f8SAndreas Gohr foreach ($tables as $table) { 121a28d6152SAndreas Gohr try { 1224cd5cc28SAnna Dabrowska $schemadata = AccessTable::getPageAccess($table, $ID, (int)$REV); 123a28d6152SAndreas Gohr } catch (StructException $ignored) { 124a28d6152SAndreas Gohr continue; // no such schema at this revision 125a28d6152SAndreas Gohr } 126b1ab837dSFrieder Schrempf 127*7fe2cdf2SAndreas Gohr $rendercontext = [ 128*7fe2cdf2SAndreas Gohr 'renderer' => $renderer, 129*7fe2cdf2SAndreas Gohr 'format' => $format, 130*7fe2cdf2SAndreas Gohr 'meta' => p_get_metadata($ID), 131*7fe2cdf2SAndreas Gohr 'schemadata' => $schemadata, 132*7fe2cdf2SAndreas Gohr 'hasdata' => &$hasdata 133*7fe2cdf2SAndreas Gohr ]; 134b1ab837dSFrieder Schrempf 1357234bfb1Ssplitbrain $event = new Event( 136b1ab837dSFrieder Schrempf 'PLUGIN_STRUCT_RENDER_SCHEMA_DATA', 1372dbe71f8SAnna Dabrowska $rendercontext 138b1ab837dSFrieder Schrempf ); 1392dbe71f8SAnna Dabrowska $event->trigger([$this, 'renderSchemaData']); 140b1ab837dSFrieder Schrempf } 141b1ab837dSFrieder Schrempf 142b1ab837dSFrieder Schrempf if ($format == 'xhtml') $renderer->doc .= self::XHTML_CLOSE; 143b1ab837dSFrieder Schrempf 144b1ab837dSFrieder Schrempf // if no data has been output, remove empty wrapper again 145b1ab837dSFrieder Schrempf if ($format == 'xhtml' && !$hasdata) { 146b1ab837dSFrieder Schrempf $renderer->doc = substr($renderer->doc, 0, -1 * strlen(self::XHTML_OPEN . self::XHTML_CLOSE)); 147b1ab837dSFrieder Schrempf } 148b1ab837dSFrieder Schrempf 149b1ab837dSFrieder Schrempf return true; 150b1ab837dSFrieder Schrempf } 151b1ab837dSFrieder Schrempf 152b1ab837dSFrieder Schrempf /** 153b1ab837dSFrieder Schrempf * Default schema data rendering (simple table view) 154b1ab837dSFrieder Schrempf * 155b1ab837dSFrieder Schrempf * @param array The render context including renderer and data 156b1ab837dSFrieder Schrempf */ 157b1ab837dSFrieder Schrempf public function renderSchemaData($rendercontext) 158b1ab837dSFrieder Schrempf { 159b1ab837dSFrieder Schrempf $schemadata = $rendercontext['schemadata']; 160b1ab837dSFrieder Schrempf $renderer = $rendercontext['renderer']; 161b1ab837dSFrieder Schrempf $format = $rendercontext['format']; 162b1ab837dSFrieder Schrempf 1630dd23cefSAndreas Gohr $schemadata->optionSkipEmpty(true); 1640dd23cefSAndreas Gohr $data = $schemadata->getData(); 165b1ab837dSFrieder Schrempf if (!count($data)) 166b1ab837dSFrieder Schrempf return; 167b1ab837dSFrieder Schrempf 168b1ab837dSFrieder Schrempf $rendercontext['hasdata'] = true; 169257dd7f8SAndreas Gohr 1703f594db8SAndreas Gohr if ($format == 'xhtml') { 1713f594db8SAndreas Gohr $renderer->doc .= '<div class="struct_output_' . $schemadata->getSchema()->getTable() . '">'; 1723f594db8SAndreas Gohr } 1733f594db8SAndreas Gohr 17434ea6e10SAnna Dabrowska $renderer->table_open(); 17534ea6e10SAnna Dabrowska $renderer->tablethead_open(); 17634ea6e10SAnna Dabrowska $renderer->tablerow_open(); 17734ea6e10SAnna Dabrowska $renderer->tableheader_open(2); 17834ea6e10SAnna Dabrowska $renderer->cdata($schemadata->getSchema()->getTranslatedLabel()); 17934ea6e10SAnna Dabrowska $renderer->tableheader_close(); 18034ea6e10SAnna Dabrowska $renderer->tablerow_close(); 18134ea6e10SAnna Dabrowska $renderer->tablethead_close(); 182da30fdd3SAndreas Gohr 18334ea6e10SAnna Dabrowska $renderer->tabletbody_open(); 184257dd7f8SAndreas Gohr foreach ($data as $field) { 18534ea6e10SAnna Dabrowska $renderer->tablerow_open(); 18634ea6e10SAnna Dabrowska $renderer->tableheader_open(); 18734ea6e10SAnna Dabrowska $renderer->cdata($field->getColumn()->getTranslatedLabel()); 18834ea6e10SAnna Dabrowska $renderer->tableheader_close(); 18934ea6e10SAnna Dabrowska $renderer->tablecell_open(); 19034ea6e10SAnna Dabrowska if ($format == 'xhtml') { 19134ea6e10SAnna Dabrowska $renderer->doc = substr($renderer->doc, 0, -1) . 192748e747fSAnna Dabrowska ' data-struct="' . hsc($field->getColumn()->getFullQualifiedLabel()) . 193748e747fSAnna Dabrowska '">'; 19425852712SAndreas Gohr } 19534ea6e10SAnna Dabrowska $field->render($renderer, $format); 19634ea6e10SAnna Dabrowska $renderer->tablecell_close(); 19734ea6e10SAnna Dabrowska $renderer->tablerow_close(); 198257dd7f8SAndreas Gohr } 19934ea6e10SAnna Dabrowska $renderer->tabletbody_close(); 20034ea6e10SAnna Dabrowska $renderer->table_close(); 2013f594db8SAndreas Gohr 2023f594db8SAndreas Gohr if ($format == 'xhtml') { 2033f594db8SAndreas Gohr $renderer->doc .= '</div>'; 2043f594db8SAndreas Gohr } 205da30fdd3SAndreas Gohr } 206257dd7f8SAndreas Gohr} 207257dd7f8SAndreas Gohr 208257dd7f8SAndreas Gohr// vim:ts=4:sw=4:et: 209