xref: /plugin/struct/syntax/output.php (revision 3f594db8bf8e43ac461d652e6f9892b886db0456)
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
10f411d872SAndreas Gohruse dokuwiki\plugin\struct\meta\AccessTable;
11ba766201SAndreas Gohruse dokuwiki\plugin\struct\meta\Assignments;
127cbcfbdbSAndreas Gohruse dokuwiki\plugin\struct\meta\StructException;
13257dd7f8SAndreas Gohr
14d6d97f60SAnna Dabrowskaclass syntax_plugin_struct_output extends DokuWiki_Syntax_Plugin
15d6d97f60SAnna Dabrowska{
1687050b53SMichael Grosse    protected $hasBeenRendered = false;
1787050b53SMichael Grosse
1817a3a578SAndreas Gohr    protected const XHTML_OPEN = '<div id="plugin__struct_output">';
1917a3a578SAndreas Gohr    protected const XHTML_CLOSE = '</div>';
207938ca48SAndreas Gohr
21257dd7f8SAndreas Gohr    /**
22979b16f8SAndreas Gohr     * Regexp to check on which actions the struct data may be rendered
23979b16f8SAndreas Gohr     */
2417a3a578SAndreas Gohr    protected const WHITELIST_ACTIONS = '/^(show|export_.*)$/';
25979b16f8SAndreas Gohr
26979b16f8SAndreas Gohr    /**
27257dd7f8SAndreas Gohr     * @return string Syntax mode type
28257dd7f8SAndreas Gohr     */
29d6d97f60SAnna Dabrowska    public function getType()
30d6d97f60SAnna Dabrowska    {
31257dd7f8SAndreas Gohr        return 'substition';
32257dd7f8SAndreas Gohr    }
33da30fdd3SAndreas Gohr
34257dd7f8SAndreas Gohr    /**
35257dd7f8SAndreas Gohr     * @return string Paragraph type
36257dd7f8SAndreas Gohr     */
37d6d97f60SAnna Dabrowska    public function getPType()
38d6d97f60SAnna Dabrowska    {
39257dd7f8SAndreas Gohr        return 'block';
40257dd7f8SAndreas Gohr    }
41da30fdd3SAndreas Gohr
42257dd7f8SAndreas Gohr    /**
43257dd7f8SAndreas Gohr     * @return int Sort order - Low numbers go before high numbers
44257dd7f8SAndreas Gohr     */
45d6d97f60SAnna Dabrowska    public function getSort()
46d6d97f60SAnna Dabrowska    {
47257dd7f8SAndreas Gohr        return 155;
48257dd7f8SAndreas Gohr    }
49257dd7f8SAndreas Gohr
50257dd7f8SAndreas Gohr    /**
51257dd7f8SAndreas Gohr     * Connect lookup pattern to lexer.
52257dd7f8SAndreas Gohr     *
5382c064c1SAndreas Gohr     * We do not connect any pattern here, because the call to this plugin is not
5482c064c1SAndreas Gohr     * triggered from syntax but our action component
5582c064c1SAndreas Gohr     *
5682c064c1SAndreas Gohr     * @asee action_plugin_struct_output
57257dd7f8SAndreas Gohr     * @param string $mode Parser mode
58257dd7f8SAndreas Gohr     */
59d6d97f60SAnna Dabrowska    public function connectTo($mode)
60d6d97f60SAnna Dabrowska    {
61257dd7f8SAndreas Gohr    }
62257dd7f8SAndreas Gohr
63257dd7f8SAndreas Gohr    /**
64257dd7f8SAndreas Gohr     * Handle matches of the struct syntax
65257dd7f8SAndreas Gohr     *
66257dd7f8SAndreas Gohr     * @param string $match The match of the syntax
67257dd7f8SAndreas Gohr     * @param int $state The state of the handler
68257dd7f8SAndreas Gohr     * @param int $pos The position in the document
69257dd7f8SAndreas Gohr     * @param Doku_Handler $handler The handler
70257dd7f8SAndreas Gohr     * @return array Data for the renderer
71257dd7f8SAndreas Gohr     */
72d6d97f60SAnna Dabrowska    public function handle($match, $state, $pos, Doku_Handler $handler)
73d6d97f60SAnna Dabrowska    {
7482c064c1SAndreas Gohr        // this is never called
7582c064c1SAndreas Gohr        return array();
76257dd7f8SAndreas Gohr    }
77257dd7f8SAndreas Gohr
78257dd7f8SAndreas Gohr    /**
7982c064c1SAndreas Gohr     * Render schema data
80257dd7f8SAndreas Gohr     *
81564e138bSAnna Dabrowska     * Currently completely renderer agnostic
8282c064c1SAndreas Gohr     *
8334ea6e10SAnna Dabrowska     * @param string $format Renderer format
8434ea6e10SAnna Dabrowska     * @param Doku_Renderer $renderer The renderer
85257dd7f8SAndreas Gohr     * @param array $data The data from the handler() function
86257dd7f8SAndreas Gohr     * @return bool If rendering was successful.
87257dd7f8SAndreas Gohr     */
8834ea6e10SAnna Dabrowska    public function render($format, Doku_Renderer $renderer, $data)
89d6d97f60SAnna Dabrowska    {
900e4a3e7cSMichael Große        global $ACT;
91257dd7f8SAndreas Gohr        global $ID;
9282c064c1SAndreas Gohr        global $INFO;
93257dd7f8SAndreas Gohr        global $REV;
94bdefb930SAnna Dabrowska
9534ea6e10SAnna Dabrowska        foreach (helper_plugin_struct::BLACKLIST_RENDERER as $blacklisted) {
9634ea6e10SAnna Dabrowska            if ($renderer instanceof $blacklisted) {
970e4a3e7cSMichael Große                return true;
980e4a3e7cSMichael Große            }
99bdefb930SAnna Dabrowska        }
10082c064c1SAndreas Gohr        if ($ID != $INFO['id']) return true;
1012f1a213bSAndreas Gohr        if (!$INFO['exists']) return true;
10287050b53SMichael Grosse        if ($this->hasBeenRendered) return true;
103979b16f8SAndreas Gohr        if (!preg_match(self::WHITELIST_ACTIONS, act_clean($ACT))) return true;
10487050b53SMichael Grosse
10587050b53SMichael Grosse        // do not render the output twice on the same page, e.g. when another page has been included
10687050b53SMichael Grosse        $this->hasBeenRendered = true;
1077cbcfbdbSAndreas Gohr        try {
108025cb9daSAndreas Gohr            $assignments = Assignments::getInstance();
1097cbcfbdbSAndreas Gohr        } catch (StructException $e) {
1107cbcfbdbSAndreas Gohr            return false;
1117cbcfbdbSAndreas Gohr        }
112257dd7f8SAndreas Gohr        $tables = $assignments->getPageAssignments($ID);
113257dd7f8SAndreas Gohr        if (!$tables) return true;
114257dd7f8SAndreas Gohr
11534ea6e10SAnna Dabrowska        if ($format == 'xhtml') $renderer->doc .= self::XHTML_OPEN;
1165a1eab78SAndreas Gohr
1177938ca48SAndreas Gohr        $hasdata = false;
118257dd7f8SAndreas Gohr        foreach ($tables as $table) {
119a28d6152SAndreas Gohr            try {
1204cd5cc28SAnna Dabrowska                $schemadata = AccessTable::getPageAccess($table, $ID, (int)$REV);
121a28d6152SAndreas Gohr            } catch (StructException $ignored) {
122a28d6152SAndreas Gohr                continue; // no such schema at this revision
123a28d6152SAndreas Gohr            }
124b1ab837dSFrieder Schrempf
125b1ab837dSFrieder Schrempf            $rendercontext = array(
126b1ab837dSFrieder Schrempf                'renderer' => $renderer,
127b1ab837dSFrieder Schrempf                'format' => $format,
128b1ab837dSFrieder Schrempf                'meta' => p_get_metadata($ID),
129b1ab837dSFrieder Schrempf                'schemadata' => $schemadata,
130b1ab837dSFrieder Schrempf                'hasdata' => &$hasdata
131b1ab837dSFrieder Schrempf            );
132b1ab837dSFrieder Schrempf
1332dbe71f8SAnna Dabrowska            $event = new \Doku_Event(
134b1ab837dSFrieder Schrempf                'PLUGIN_STRUCT_RENDER_SCHEMA_DATA',
1352dbe71f8SAnna Dabrowska                $rendercontext
136b1ab837dSFrieder Schrempf            );
1372dbe71f8SAnna Dabrowska            $event->trigger([$this, 'renderSchemaData']);
138b1ab837dSFrieder Schrempf        }
139b1ab837dSFrieder Schrempf
140b1ab837dSFrieder Schrempf        if ($format == 'xhtml') $renderer->doc .= self::XHTML_CLOSE;
141b1ab837dSFrieder Schrempf
142b1ab837dSFrieder Schrempf        // if no data has been output, remove empty wrapper again
143b1ab837dSFrieder Schrempf        if ($format == 'xhtml' && !$hasdata) {
144b1ab837dSFrieder Schrempf            $renderer->doc = substr($renderer->doc, 0, -1 * strlen(self::XHTML_OPEN . self::XHTML_CLOSE));
145b1ab837dSFrieder Schrempf        }
146b1ab837dSFrieder Schrempf
147b1ab837dSFrieder Schrempf        return true;
148b1ab837dSFrieder Schrempf    }
149b1ab837dSFrieder Schrempf
150b1ab837dSFrieder Schrempf    /**
151b1ab837dSFrieder Schrempf     * Default schema data rendering (simple table view)
152b1ab837dSFrieder Schrempf     *
153b1ab837dSFrieder Schrempf     * @param array The render context including renderer and data
154b1ab837dSFrieder Schrempf     */
155b1ab837dSFrieder Schrempf    public function renderSchemaData($rendercontext)
156b1ab837dSFrieder Schrempf    {
157b1ab837dSFrieder Schrempf        $schemadata = $rendercontext['schemadata'];
158b1ab837dSFrieder Schrempf        $renderer = $rendercontext['renderer'];
159b1ab837dSFrieder Schrempf        $format = $rendercontext['format'];
160b1ab837dSFrieder Schrempf
1610dd23cefSAndreas Gohr        $schemadata->optionSkipEmpty(true);
1620dd23cefSAndreas Gohr        $data = $schemadata->getData();
163b1ab837dSFrieder Schrempf        if (!count($data))
164b1ab837dSFrieder Schrempf            return;
165b1ab837dSFrieder Schrempf
166b1ab837dSFrieder Schrempf        $rendercontext['hasdata'] = true;
167257dd7f8SAndreas Gohr
168*3f594db8SAndreas Gohr        if($format == 'xhtml') {
169*3f594db8SAndreas Gohr            $renderer->doc .= '<div class="struct_output_'.$schemadata->getSchema()->getTable().'">';
170*3f594db8SAndreas Gohr        }
171*3f594db8SAndreas Gohr
17234ea6e10SAnna Dabrowska        $renderer->table_open();
17334ea6e10SAnna Dabrowska        $renderer->tablethead_open();
17434ea6e10SAnna Dabrowska        $renderer->tablerow_open();
17534ea6e10SAnna Dabrowska        $renderer->tableheader_open(2);
17634ea6e10SAnna Dabrowska        $renderer->cdata($schemadata->getSchema()->getTranslatedLabel());
17734ea6e10SAnna Dabrowska        $renderer->tableheader_close();
17834ea6e10SAnna Dabrowska        $renderer->tablerow_close();
17934ea6e10SAnna Dabrowska        $renderer->tablethead_close();
180da30fdd3SAndreas Gohr
18134ea6e10SAnna Dabrowska        $renderer->tabletbody_open();
182257dd7f8SAndreas Gohr        foreach ($data as $field) {
18334ea6e10SAnna Dabrowska            $renderer->tablerow_open();
18434ea6e10SAnna Dabrowska            $renderer->tableheader_open();
18534ea6e10SAnna Dabrowska            $renderer->cdata($field->getColumn()->getTranslatedLabel());
18634ea6e10SAnna Dabrowska            $renderer->tableheader_close();
18734ea6e10SAnna Dabrowska            $renderer->tablecell_open();
18834ea6e10SAnna Dabrowska            if ($format == 'xhtml') {
18934ea6e10SAnna Dabrowska                $renderer->doc = substr($renderer->doc, 0, -1) .
190748e747fSAnna Dabrowska                    ' data-struct="' . hsc($field->getColumn()->getFullQualifiedLabel()) .
191748e747fSAnna Dabrowska                    '">';
19225852712SAndreas Gohr            }
19334ea6e10SAnna Dabrowska            $field->render($renderer, $format);
19434ea6e10SAnna Dabrowska            $renderer->tablecell_close();
19534ea6e10SAnna Dabrowska            $renderer->tablerow_close();
196257dd7f8SAndreas Gohr        }
19734ea6e10SAnna Dabrowska        $renderer->tabletbody_close();
19834ea6e10SAnna Dabrowska        $renderer->table_close();
199*3f594db8SAndreas Gohr
200*3f594db8SAndreas Gohr        if($format == 'xhtml') {
201*3f594db8SAndreas Gohr            $renderer->doc .= '</div>';
202*3f594db8SAndreas Gohr        }
203da30fdd3SAndreas Gohr    }
204257dd7f8SAndreas Gohr}
205257dd7f8SAndreas Gohr
206257dd7f8SAndreas Gohr// vim:ts=4:sw=4:et:
207