xref: /plugin/struct/meta/AggregationTable.php (revision 3f640228fdb9065bb71e558f8f86b81f0dcc1d2f)
107993756SAndreas Gohr<?php
207993756SAndreas Gohr
3ba766201SAndreas Gohrnamespace dokuwiki\plugin\struct\meta;
407993756SAndreas Gohr
5d60f71efSAndreas Gohr/**
6d60f71efSAndreas Gohr * Creates the table aggregation output
7d60f71efSAndreas Gohr *
8ba766201SAndreas Gohr * @package dokuwiki\plugin\struct\meta
9d60f71efSAndreas Gohr */
10d6d97f60SAnna Dabrowskaclass AggregationTable
11d6d97f60SAnna Dabrowska{
1207993756SAndreas Gohr
1307993756SAndreas Gohr    /**
1407993756SAndreas Gohr     * @var string the page id of the page this is rendered to
1507993756SAndreas Gohr     */
1607993756SAndreas Gohr    protected $id;
1707993756SAndreas Gohr    /**
1807993756SAndreas Gohr     * @var string the Type of renderer used
1907993756SAndreas Gohr     */
2007993756SAndreas Gohr    protected $mode;
2107993756SAndreas Gohr    /**
2207993756SAndreas Gohr     * @var \Doku_Renderer the DokuWiki renderer used to create the output
2307993756SAndreas Gohr     */
2407993756SAndreas Gohr    protected $renderer;
2507993756SAndreas Gohr    /**
2607993756SAndreas Gohr     * @var SearchConfig the configured search - gives access to columns etc.
2707993756SAndreas Gohr     */
2807993756SAndreas Gohr    protected $searchConfig;
2907993756SAndreas Gohr
3007993756SAndreas Gohr    /**
3107993756SAndreas Gohr     * @var Column[] the list of columns to be displayed
3207993756SAndreas Gohr     */
3307993756SAndreas Gohr    protected $columns;
3407993756SAndreas Gohr
3507993756SAndreas Gohr    /**
3607993756SAndreas Gohr     * @var  Value[][] the search result
3707993756SAndreas Gohr     */
3807993756SAndreas Gohr    protected $result;
3907993756SAndreas Gohr
4007993756SAndreas Gohr    /**
4107993756SAndreas Gohr     * @var int number of all results
4207993756SAndreas Gohr     */
4307993756SAndreas Gohr    protected $resultCount;
4407993756SAndreas Gohr
4507993756SAndreas Gohr    /**
46d4b5a17cSAndreas Gohr     * @var string[] the result PIDs for each row
47d4b5a17cSAndreas Gohr     */
48d4b5a17cSAndreas Gohr    protected $resultPIDs;
490ceefd5cSAnna Dabrowska    protected $resultRids;
506fd73b4bSAnna Dabrowska    protected $resultRevs;
51d4b5a17cSAndreas Gohr
52d4b5a17cSAndreas Gohr    /**
5307993756SAndreas Gohr     * @var array for summing up columns
5407993756SAndreas Gohr     */
5507993756SAndreas Gohr    protected $sums;
5607993756SAndreas Gohr
5707993756SAndreas Gohr    /**
586b5e52fdSAndreas Gohr     * @var bool skip full table when no results found
596b5e52fdSAndreas Gohr     */
606b5e52fdSAndreas Gohr    protected $simplenone = true;
616b5e52fdSAndreas Gohr
626b5e52fdSAndreas Gohr    /**
6307993756SAndreas Gohr     * @todo we might be able to get rid of this helper and move this to SearchConfig
6407993756SAndreas Gohr     * @var \helper_plugin_struct_config
6507993756SAndreas Gohr     */
6607993756SAndreas Gohr    protected $helper;
6707993756SAndreas Gohr
6807993756SAndreas Gohr    /**
6907993756SAndreas Gohr     * Initialize the Aggregation renderer and executes the search
7007993756SAndreas Gohr     *
7107993756SAndreas Gohr     * You need to call @see render() on the resulting object.
7207993756SAndreas Gohr     *
7307993756SAndreas Gohr     * @param string $id
7407993756SAndreas Gohr     * @param string $mode
7507993756SAndreas Gohr     * @param \Doku_Renderer $renderer
7607993756SAndreas Gohr     * @param SearchConfig $searchConfig
7707993756SAndreas Gohr     */
788ce43f5aSAnna Dabrowska    public function __construct($id, $mode, \Doku_Renderer $renderer, SearchConfig $searchConfig)
79d6d97f60SAnna Dabrowska    {
8007993756SAndreas Gohr        $this->id = $id;
8107993756SAndreas Gohr        $this->mode = $mode;
8207993756SAndreas Gohr        $this->renderer = $renderer;
8307993756SAndreas Gohr        $this->searchConfig = $searchConfig;
8407993756SAndreas Gohr        $this->data = $searchConfig->getConf();
8507993756SAndreas Gohr        $this->columns = $searchConfig->getColumns();
8607993756SAndreas Gohr
878ce43f5aSAnna Dabrowska        $this->result = $this->searchConfig->execute();
8807993756SAndreas Gohr        $this->resultCount = $this->searchConfig->getCount();
89d4b5a17cSAndreas Gohr        $this->resultPIDs = $this->searchConfig->getPids();
900ceefd5cSAnna Dabrowska        $this->resultRids = $this->searchConfig->getRids();
916fd73b4bSAnna Dabrowska        $this->resultRevs = $this->searchConfig->getRevs();
9207993756SAndreas Gohr        $this->helper = plugin_load('helper', 'struct_config');
9307993756SAndreas Gohr    }
9407993756SAndreas Gohr
9507993756SAndreas Gohr    /**
96844a4f01SFrieder Schrempf     * Returns the page id for the table
97844a4f01SFrieder Schrempf     */
98844a4f01SFrieder Schrempf    public function getID()
99844a4f01SFrieder Schrempf    {
100844a4f01SFrieder Schrempf        return $this->id;
101844a4f01SFrieder Schrempf    }
102844a4f01SFrieder Schrempf
103844a4f01SFrieder Schrempf    /**
10407993756SAndreas Gohr     * Create the table on the renderer
10507993756SAndreas Gohr     */
106d6d97f60SAnna Dabrowska    public function render()
107d6d97f60SAnna Dabrowska    {
108b7e1d73bSAndreas Gohr
109b7e1d73bSAndreas Gohr        // abort early if there are no results at all (not filtered)
1106b5e52fdSAndreas Gohr        if (!$this->resultCount && !$this->isDynamicallyFiltered() && $this->simplenone) {
111b7e1d73bSAndreas Gohr            $this->startScope();
112b7e1d73bSAndreas Gohr            $this->renderer->cdata($this->helper->getLang('none'));
113b7e1d73bSAndreas Gohr            $this->finishScope();
114b7e1d73bSAndreas Gohr            return;
115b7e1d73bSAndreas Gohr        }
116b7e1d73bSAndreas Gohr
11707993756SAndreas Gohr        $this->startScope();
118986ab7e6SAndreas Gohr        $this->renderActiveFilters();
119844a4f01SFrieder Schrempf
120844a4f01SFrieder Schrempf        $rendercontext = array(
121844a4f01SFrieder Schrempf            'table' => $this,
122844a4f01SFrieder Schrempf            'renderer' => $this->renderer,
123*3f640228SFrieder Schrempf            'format' => $this->mode,
124844a4f01SFrieder Schrempf            'search' => $this->searchConfig,
125844a4f01SFrieder Schrempf            'columns' => $this->columns,
126844a4f01SFrieder Schrempf            'data' => $this->result
127844a4f01SFrieder Schrempf        );
128844a4f01SFrieder Schrempf
1292dbe71f8SAnna Dabrowska        $event = new \Doku_Event(
130844a4f01SFrieder Schrempf            'PLUGIN_STRUCT_RENDER_AGGREGATION_TABLE',
1312dbe71f8SAnna Dabrowska            $rendercontext
132844a4f01SFrieder Schrempf        );
1332dbe71f8SAnna Dabrowska        $event->trigger([$this, 'renderTable']);
134844a4f01SFrieder Schrempf
135844a4f01SFrieder Schrempf        // export handle
136844a4f01SFrieder Schrempf        $this->renderExportControls();
137844a4f01SFrieder Schrempf        $this->finishScope();
138844a4f01SFrieder Schrempf    }
139844a4f01SFrieder Schrempf
140844a4f01SFrieder Schrempf    /**
141844a4f01SFrieder Schrempf     * Render the default aggregation table
142844a4f01SFrieder Schrempf     */
143844a4f01SFrieder Schrempf    public function renderTable($rendercontext)
144844a4f01SFrieder Schrempf    {
14507993756SAndreas Gohr        $this->renderer->table_open();
14607993756SAndreas Gohr
14707993756SAndreas Gohr        // header
14807993756SAndreas Gohr        $this->renderer->tablethead_open();
149986ab7e6SAndreas Gohr        $this->renderColumnHeaders();
150986ab7e6SAndreas Gohr        $this->renderDynamicFilters();
15107993756SAndreas Gohr        $this->renderer->tablethead_close();
15207993756SAndreas Gohr
15307993756SAndreas Gohr        if ($this->resultCount) {
15407993756SAndreas Gohr            // actual data
155a9fd81f9SAndreas Gohr            $this->renderer->tabletbody_open();
156986ab7e6SAndreas Gohr            $this->renderResult();
157a9fd81f9SAndreas Gohr            $this->renderer->tabletbody_close();
15807993756SAndreas Gohr
159a9fd81f9SAndreas Gohr            // footer (tfoot is develonly currently)
160a9fd81f9SAndreas Gohr            if (method_exists($this->renderer, 'tabletfoot_open')) $this->renderer->tabletfoot_open();
161986ab7e6SAndreas Gohr            $this->renderSums();
162986ab7e6SAndreas Gohr            $this->renderPagingControls();
163a9fd81f9SAndreas Gohr            if (method_exists($this->renderer, 'tabletfoot_close')) $this->renderer->tabletfoot_close();
16407993756SAndreas Gohr        } else {
16507993756SAndreas Gohr            // nothing found
166986ab7e6SAndreas Gohr            $this->renderEmptyResult();
16707993756SAndreas Gohr        }
16807993756SAndreas Gohr
16907993756SAndreas Gohr        // table close
17007993756SAndreas Gohr        $this->renderer->table_close();
17107993756SAndreas Gohr    }
17207993756SAndreas Gohr
17307993756SAndreas Gohr    /**
17407993756SAndreas Gohr     * Adds additional info to document and renderer in XHTML mode
17507993756SAndreas Gohr     *
17607993756SAndreas Gohr     * @see finishScope()
17707993756SAndreas Gohr     */
178d6d97f60SAnna Dabrowska    protected function startScope()
179d6d97f60SAnna Dabrowska    {
18007993756SAndreas Gohr        // unique identifier for this aggregation
18107993756SAndreas Gohr        $this->renderer->info['struct_table_hash'] = md5(var_export($this->data, true));
18209dd691aSAndreas Gohr
18309dd691aSAndreas Gohr        // wrapping div
18409dd691aSAndreas Gohr        if ($this->mode != 'xhtml') return;
18509dd691aSAndreas Gohr        $this->renderer->doc .= "<div class=\"structaggregation\">";
18607993756SAndreas Gohr    }
18707993756SAndreas Gohr
18807993756SAndreas Gohr    /**
18907993756SAndreas Gohr     * Closes the table and anything opened in startScope()
19007993756SAndreas Gohr     *
19107993756SAndreas Gohr     * @see startScope()
19207993756SAndreas Gohr     */
193d6d97f60SAnna Dabrowska    protected function finishScope()
194d6d97f60SAnna Dabrowska    {
19507993756SAndreas Gohr        // remove identifier from renderer again
19607993756SAndreas Gohr        if (isset($this->renderer->info['struct_table_hash'])) {
19707993756SAndreas Gohr            unset($this->renderer->info['struct_table_hash']);
19807993756SAndreas Gohr        }
19909dd691aSAndreas Gohr
20009dd691aSAndreas Gohr        // wrapping div
20109dd691aSAndreas Gohr        if ($this->mode != 'xhtml') return;
20209dd691aSAndreas Gohr        $this->renderer->doc .= '</div>';
20307993756SAndreas Gohr    }
20407993756SAndreas Gohr
20507993756SAndreas Gohr    /**
20607993756SAndreas Gohr     * Displays info about the currently applied filters
20707993756SAndreas Gohr     */
208d6d97f60SAnna Dabrowska    protected function renderActiveFilters()
209d6d97f60SAnna Dabrowska    {
21007993756SAndreas Gohr        if ($this->mode != 'xhtml') return;
21107993756SAndreas Gohr        $dynamic = $this->searchConfig->getDynamicParameters();
21207993756SAndreas Gohr        $filters = $dynamic->getFilters();
21307993756SAndreas Gohr        if (!$filters) return;
21407993756SAndreas Gohr
21507993756SAndreas Gohr        $fltrs = array();
21607993756SAndreas Gohr        foreach ($filters as $column => $filter) {
21707993756SAndreas Gohr            list($comp, $value) = $filter;
21804ec4785SAnna Dabrowska
21904ec4785SAnna Dabrowska            // display the filters in a human readable format
22004ec4785SAnna Dabrowska            foreach ($this->columns as $col) {
22104ec4785SAnna Dabrowska                if ($column === $col->getFullQualifiedLabel()) {
22204ec4785SAnna Dabrowska                    $column = $col->getTranslatedLabel();
22304ec4785SAnna Dabrowska                }
22404ec4785SAnna Dabrowska            }
2251f075418SAnna Dabrowska            $fltrs[] = sprintf('"%s" %s "%s"', $column, $this->helper->getLang("comparator $comp"), $value);
22607993756SAndreas Gohr        }
22707993756SAndreas Gohr
22807993756SAndreas Gohr        $this->renderer->doc .= '<div class="filter">';
22907993756SAndreas Gohr        $this->renderer->doc .= '<h4>' . sprintf($this->helper->getLang('tablefilteredby'), hsc(implode(' & ', $fltrs))) . '</h4>';
23007993756SAndreas Gohr        $this->renderer->doc .= '<div class="resetfilter">';
23107993756SAndreas Gohr        $this->renderer->internallink($this->id, $this->helper->getLang('tableresetfilter'));
23207993756SAndreas Gohr        $this->renderer->doc .= '</div>';
23307993756SAndreas Gohr        $this->renderer->doc .= '</div>';
23407993756SAndreas Gohr    }
23507993756SAndreas Gohr
23607993756SAndreas Gohr    /**
23707993756SAndreas Gohr     * Shows the column headers with links to sort by column
23807993756SAndreas Gohr     */
239d6d97f60SAnna Dabrowska    protected function renderColumnHeaders()
240d6d97f60SAnna Dabrowska    {
24107993756SAndreas Gohr        $this->renderer->tablerow_open();
24207993756SAndreas Gohr
24307993756SAndreas Gohr        // additional column for row numbers
24434ea6e10SAnna Dabrowska        if (!empty($this->data['rownumbers'])) {
24507993756SAndreas Gohr            $this->renderer->tableheader_open();
24607993756SAndreas Gohr            $this->renderer->cdata('#');
24707993756SAndreas Gohr            $this->renderer->tableheader_close();
24807993756SAndreas Gohr        }
24907993756SAndreas Gohr
25007993756SAndreas Gohr        // show all headers
2518c4ee9beSAndreas Gohr        foreach ($this->columns as $num => $column) {
2528c4ee9beSAndreas Gohr            $header = '';
2538c4ee9beSAndreas Gohr            if (isset($this->data['headers'][$num])) {
2548c4ee9beSAndreas Gohr                $header = $this->data['headers'][$num];
2558c4ee9beSAndreas Gohr            }
25607993756SAndreas Gohr
25707993756SAndreas Gohr            // use field label if no header was set
25807993756SAndreas Gohr            if (blank($header)) {
25901f8b845SAndreas Gohr                if (is_a($column, 'dokuwiki\plugin\struct\meta\Column')) {
26007993756SAndreas Gohr                    $header = $column->getTranslatedLabel();
26107993756SAndreas Gohr                } else {
26207993756SAndreas Gohr                    $header = 'column ' . $num; // this should never happen
26307993756SAndreas Gohr                }
26407993756SAndreas Gohr            }
26507993756SAndreas Gohr
26607993756SAndreas Gohr            // simple mode first
26707993756SAndreas Gohr            if ($this->mode != 'xhtml') {
26807993756SAndreas Gohr                $this->renderer->tableheader_open();
26907993756SAndreas Gohr                $this->renderer->cdata($header);
27007993756SAndreas Gohr                $this->renderer->tableheader_close();
27107993756SAndreas Gohr                continue;
27207993756SAndreas Gohr            }
27307993756SAndreas Gohr
27407993756SAndreas Gohr            // still here? create custom header for more flexibility
27507993756SAndreas Gohr
2769113d04aSAndreas Gohr            // width setting, widths are prevalidated, no escape needed
27707993756SAndreas Gohr            $width = '';
2789113d04aSAndreas Gohr            if (isset($this->data['widths'][$num]) && $this->data['widths'][$num] != '-') {
2799113d04aSAndreas Gohr                $width = ' style="min-width: ' . $this->data['widths'][$num] . ';' .
2809113d04aSAndreas Gohr                         'max-width: ' . $this->data['widths'][$num] . ';"';
28107993756SAndreas Gohr            }
28207993756SAndreas Gohr
283d4b5a17cSAndreas Gohr            // prepare data attribute for inline edits
284d6d97f60SAnna Dabrowska            if (
285d6d97f60SAnna Dabrowska                !is_a($column, '\dokuwiki\plugin\struct\meta\PageColumn') &&
286d4b5a17cSAndreas Gohr                !is_a($column, '\dokuwiki\plugin\struct\meta\RevisionColumn')
287d4b5a17cSAndreas Gohr            ) {
288d4b5a17cSAndreas Gohr                $data = 'data-field="' . hsc($column->getFullQualifiedLabel()) . '"';
289d4b5a17cSAndreas Gohr            } else {
290d4b5a17cSAndreas Gohr                $data = '';
291d4b5a17cSAndreas Gohr            }
292d4b5a17cSAndreas Gohr
29307993756SAndreas Gohr            // sort indicator and link
29407993756SAndreas Gohr            $sortclass = '';
29507993756SAndreas Gohr            $sorts = $this->searchConfig->getSorts();
29607993756SAndreas Gohr            $dynamic = $this->searchConfig->getDynamicParameters();
297aa124708SAndreas Gohr            $dynamic->setSort($column, true);
29807993756SAndreas Gohr            if (isset($sorts[$column->getFullQualifiedLabel()])) {
299aa124708SAndreas Gohr                list(/*colname*/, $currentSort) = $sorts[$column->getFullQualifiedLabel()];
300aa124708SAndreas Gohr                if ($currentSort) {
30107993756SAndreas Gohr                    $sortclass = 'sort-down';
30207993756SAndreas Gohr                    $dynamic->setSort($column, false);
30307993756SAndreas Gohr                } else {
30407993756SAndreas Gohr                    $sortclass = 'sort-up';
30507993756SAndreas Gohr                }
30607993756SAndreas Gohr            }
30707993756SAndreas Gohr            $link = wl($this->id, $dynamic->getURLParameters());
30807993756SAndreas Gohr
30907993756SAndreas Gohr            // output XHTML header
310d4b5a17cSAndreas Gohr            $this->renderer->doc .= "<th $width $data>";
31107993756SAndreas Gohr            $this->renderer->doc .= '<a href="' . $link . '" class="' . $sortclass . '" title="' . $this->helper->getLang('sort') . '">' . hsc($header) . '</a>';
31207993756SAndreas Gohr            $this->renderer->doc .= '</th>';
31307993756SAndreas Gohr        }
31407993756SAndreas Gohr
31507993756SAndreas Gohr        $this->renderer->tablerow_close();
31607993756SAndreas Gohr    }
31707993756SAndreas Gohr
31807993756SAndreas Gohr    /**
319b7e1d73bSAndreas Gohr     * Is the result set currently dynamically filtered?
320b7e1d73bSAndreas Gohr     * @return bool
321b7e1d73bSAndreas Gohr     */
322d6d97f60SAnna Dabrowska    protected function isDynamicallyFiltered()
323d6d97f60SAnna Dabrowska    {
324b7e1d73bSAndreas Gohr        if ($this->mode != 'xhtml') return false;
325b7e1d73bSAndreas Gohr        if (!$this->data['dynfilters']) return false;
326b7e1d73bSAndreas Gohr
327b7e1d73bSAndreas Gohr        $dynamic = $this->searchConfig->getDynamicParameters();
328b7e1d73bSAndreas Gohr        return (bool) $dynamic->getFilters();
329b7e1d73bSAndreas Gohr    }
330b7e1d73bSAndreas Gohr
331b7e1d73bSAndreas Gohr    /**
33207993756SAndreas Gohr     * Add input fields for dynamic filtering
33307993756SAndreas Gohr     */
334d6d97f60SAnna Dabrowska    protected function renderDynamicFilters()
335d6d97f60SAnna Dabrowska    {
33607993756SAndreas Gohr        if ($this->mode != 'xhtml') return;
33707993756SAndreas Gohr        if (!$this->data['dynfilters']) return;
3381bc467a4SMichael Grosse        if (is_a($this->renderer, 'renderer_plugin_dw2pdf')) {
339e6ae02ecSMichael Grosse            return;
340e6ae02ecSMichael Grosse        }
3411bc467a4SMichael Grosse        global $conf;
34207993756SAndreas Gohr
34307993756SAndreas Gohr        $this->renderer->doc .= '<tr class="dataflt">';
34407993756SAndreas Gohr
34507993756SAndreas Gohr        // add extra column for row numbers
34607993756SAndreas Gohr        if ($this->data['rownumbers']) {
34707993756SAndreas Gohr            $this->renderer->doc .= '<th></th>';
34807993756SAndreas Gohr        }
34907993756SAndreas Gohr
35007993756SAndreas Gohr        // each column gets a form
35107993756SAndreas Gohr        foreach ($this->columns as $column) {
35207993756SAndreas Gohr            $this->renderer->doc .= '<th>';
35307993756SAndreas Gohr            {
35407993756SAndreas Gohr                $form = new \Doku_Form(array('method' => 'GET', 'action' => wl($this->id)));
35576195677SAndreas Gohr                unset($form->_hidden['sectok']); // we don't need it here
35676195677SAndreas Gohr                if (!$conf['userewrite']) $form->addHidden('id', $this->id);
35707993756SAndreas Gohr
35807993756SAndreas Gohr                // current value
35907993756SAndreas Gohr                $dynamic = $this->searchConfig->getDynamicParameters();
36007993756SAndreas Gohr                $filters = $dynamic->getFilters();
36107993756SAndreas Gohr            if (isset($filters[$column->getFullQualifiedLabel()])) {
36207993756SAndreas Gohr                list(, $current) = $filters[$column->getFullQualifiedLabel()];
36307993756SAndreas Gohr                $dynamic->removeFilter($column);
36407993756SAndreas Gohr            } else {
36507993756SAndreas Gohr                $current = '';
36607993756SAndreas Gohr            }
36707993756SAndreas Gohr
36807993756SAndreas Gohr                // Add current request params
36907993756SAndreas Gohr                $params = $dynamic->getURLParameters();
37007993756SAndreas Gohr            foreach ($params as $key => $val) {
37107993756SAndreas Gohr                $form->addHidden($key, $val);
37207993756SAndreas Gohr            }
37307993756SAndreas Gohr
37407993756SAndreas Gohr                // add input field
375db9b8745SAndreas Gohr                $key = $column->getFullQualifiedLabel() . $column->getType()->getDefaultComparator();
376d60f71efSAndreas Gohr                $form->addElement(form_makeField('text', SearchConfigParameters::$PARAM_FILTER . '[' . $key . ']', $current, ''));
37707993756SAndreas Gohr                $this->renderer->doc .= $form->getForm();
37807993756SAndreas Gohr            }
37907993756SAndreas Gohr            $this->renderer->doc .= '</th>';
38007993756SAndreas Gohr        }
38107993756SAndreas Gohr        $this->renderer->doc .= '</tr>';
38207993756SAndreas Gohr    }
38307993756SAndreas Gohr
38407993756SAndreas Gohr    /**
38507993756SAndreas Gohr     * Display the actual table data
38607993756SAndreas Gohr     */
387d6d97f60SAnna Dabrowska    protected function renderResult()
388d6d97f60SAnna Dabrowska    {
38907993756SAndreas Gohr        foreach ($this->result as $rownum => $row) {
39047eb8cceSSzymon Olewniczak            $data = array(
39147eb8cceSSzymon Olewniczak                'id' => $this->id,
39247eb8cceSSzymon Olewniczak                'mode' => $this->mode,
39347eb8cceSSzymon Olewniczak                'renderer' => $this->renderer,
39447eb8cceSSzymon Olewniczak                'searchConfig' => $this->searchConfig,
39547eb8cceSSzymon Olewniczak                'data' => $this->data,
39647eb8cceSSzymon Olewniczak                'rownum' => &$rownum,
39747eb8cceSSzymon Olewniczak                'row' => &$row,
39847eb8cceSSzymon Olewniczak            );
39947eb8cceSSzymon Olewniczak            $evt = new \Doku_Event('PLUGIN_STRUCT_AGGREGATIONTABLE_RENDERRESULTROW', $data);
40047eb8cceSSzymon Olewniczak            if ($evt->advise_before()) {
401f107f479SAndreas Gohr                $this->renderResultRow($rownum, $row);
402f107f479SAndreas Gohr            }
40347eb8cceSSzymon Olewniczak            $evt->advise_after();
40447eb8cceSSzymon Olewniczak        }
405f107f479SAndreas Gohr    }
406f107f479SAndreas Gohr
407f107f479SAndreas Gohr    /**
408f107f479SAndreas Gohr     * Render a single result row
409f107f479SAndreas Gohr     *
410f107f479SAndreas Gohr     * @param int $rownum
411f107f479SAndreas Gohr     * @param array $row
412f107f479SAndreas Gohr     */
413d6d97f60SAnna Dabrowska    protected function renderResultRow($rownum, $row)
414d6d97f60SAnna Dabrowska    {
41507993756SAndreas Gohr        $this->renderer->tablerow_open();
41607993756SAndreas Gohr
417d4b5a17cSAndreas Gohr        // add data attribute for inline edit
418d4b5a17cSAndreas Gohr        if ($this->mode == 'xhtml') {
419d4b5a17cSAndreas Gohr            $pid = $this->resultPIDs[$rownum];
4200ceefd5cSAnna Dabrowska            $rid = $this->resultRids[$rownum];
4216fd73b4bSAnna Dabrowska            $rev = $this->resultRevs[$rownum];
422d4b5a17cSAndreas Gohr            $this->renderer->doc = substr(rtrim($this->renderer->doc), 0, -1); // remove closing '>'
4236fd73b4bSAnna Dabrowska            $this->renderer->doc .= ' data-pid="' . hsc($pid) . '" data-rev="' . $rev . '" data-rid="' . $rid . '">';
424d4b5a17cSAndreas Gohr        }
425d4b5a17cSAndreas Gohr
42607993756SAndreas Gohr        // row number column
42734ea6e10SAnna Dabrowska        if (!empty($this->data['rownumbers'])) {
42807993756SAndreas Gohr            $this->renderer->tablecell_open();
4293215aebfSSzymon Olewniczak            $searchConfigConf = $this->searchConfig->getConf();
4303215aebfSSzymon Olewniczak            $this->renderer->cdata($rownum + $searchConfigConf['offset'] + 1);
43107993756SAndreas Gohr            $this->renderer->tablecell_close();
43207993756SAndreas Gohr        }
43307993756SAndreas Gohr
43407993756SAndreas Gohr        /** @var Value $value */
43507993756SAndreas Gohr        foreach ($row as $colnum => $value) {
43634ea6e10SAnna Dabrowska            $align = isset($this->data['align'][$colnum]) ?  $this->data['align'][$colnum] : null;
43734ea6e10SAnna Dabrowska            $this->renderer->tablecell_open(1, $align);
43807993756SAndreas Gohr            $value->render($this->renderer, $this->mode);
43907993756SAndreas Gohr            $this->renderer->tablecell_close();
44007993756SAndreas Gohr
44107993756SAndreas Gohr            // summarize
44207993756SAndreas Gohr            if ($this->data['summarize'] && is_numeric($value->getValue())) {
44307993756SAndreas Gohr                if (!isset($this->sums[$colnum])) {
44407993756SAndreas Gohr                    $this->sums[$colnum] = 0;
44507993756SAndreas Gohr                }
44607993756SAndreas Gohr                $this->sums[$colnum] += $value->getValue();
44707993756SAndreas Gohr            }
44807993756SAndreas Gohr        }
44907993756SAndreas Gohr        $this->renderer->tablerow_close();
45007993756SAndreas Gohr    }
45107993756SAndreas Gohr
45207993756SAndreas Gohr    /**
45307993756SAndreas Gohr     * Renders an information row for when no results were found
45407993756SAndreas Gohr     */
455d6d97f60SAnna Dabrowska    protected function renderEmptyResult()
456d6d97f60SAnna Dabrowska    {
45707993756SAndreas Gohr        $this->renderer->tablerow_open();
45870cf6339SAndreas Gohr        $this->renderer->tablecell_open(count($this->columns) + $this->data['rownumbers'], 'center');
45907993756SAndreas Gohr        $this->renderer->cdata($this->helper->getLang('none'));
46007993756SAndreas Gohr        $this->renderer->tablecell_close();
46107993756SAndreas Gohr        $this->renderer->tablerow_close();
46207993756SAndreas Gohr    }
46307993756SAndreas Gohr
46407993756SAndreas Gohr    /**
46507993756SAndreas Gohr     * Add sums if wanted
46607993756SAndreas Gohr     */
467d6d97f60SAnna Dabrowska    protected function renderSums()
468d6d97f60SAnna Dabrowska    {
469d18090e8SAndreas Gohr        if (empty($this->data['summarize'])) return;
47007993756SAndreas Gohr
471a0bf8bb2SAndreas Gohr        $this->renderer->info['struct_table_meta'] = true;
4728925ba29SAndreas Gohr        if ($this->mode == 'xhtml') {
4738925ba29SAndreas Gohr            /** @noinspection PhpMethodParametersCountMismatchInspection */
4748925ba29SAndreas Gohr            $this->renderer->tablerow_open('summarize');
4758925ba29SAndreas Gohr        } else {
47607993756SAndreas Gohr            $this->renderer->tablerow_open();
4778925ba29SAndreas Gohr        }
47807993756SAndreas Gohr
47907993756SAndreas Gohr        if ($this->data['rownumbers']) {
4808925ba29SAndreas Gohr            $this->renderer->tableheader_open();
4818925ba29SAndreas Gohr            $this->renderer->tableheader_close();
48207993756SAndreas Gohr        }
48307993756SAndreas Gohr
484aee4116bSAndreas Gohr        $len = count($this->columns);
48507993756SAndreas Gohr        for ($i = 0; $i < $len; $i++) {
4868925ba29SAndreas Gohr            $this->renderer->tableheader_open(1, $this->data['align'][$i]);
487aee4116bSAndreas Gohr            if (!empty($this->sums[$i])) {
4889b97e610SAndreas Gohr                $this->renderer->cdata('∑ ');
4899b97e610SAndreas Gohr                $this->columns[$i]->getType()->renderValue($this->sums[$i], $this->renderer, $this->mode);
49007993756SAndreas Gohr            } else {
49107993756SAndreas Gohr                if ($this->mode == 'xhtml') {
49207993756SAndreas Gohr                    $this->renderer->doc .= '&nbsp;';
49307993756SAndreas Gohr                }
49407993756SAndreas Gohr            }
4958925ba29SAndreas Gohr            $this->renderer->tableheader_close();
49607993756SAndreas Gohr        }
49707993756SAndreas Gohr        $this->renderer->tablerow_close();
498a0bf8bb2SAndreas Gohr        $this->renderer->info['struct_table_meta'] = false;
49907993756SAndreas Gohr    }
50007993756SAndreas Gohr
50107993756SAndreas Gohr    /**
502986ab7e6SAndreas Gohr     * Adds paging controls to the table
50307993756SAndreas Gohr     */
504d6d97f60SAnna Dabrowska    protected function renderPagingControls()
505d6d97f60SAnna Dabrowska    {
50607993756SAndreas Gohr        if (empty($this->data['limit'])) return;
507a0bf8bb2SAndreas Gohr        if ($this->mode != 'xhtml') return;
50807993756SAndreas Gohr
509a0bf8bb2SAndreas Gohr        $this->renderer->info['struct_table_meta'] = true;
51007993756SAndreas Gohr        $this->renderer->tablerow_open();
5114bc1074dSMichael Grosse        $this->renderer->tableheader_open((count($this->columns) + ($this->data['rownumbers'] ? 1 : 0)));
51207993756SAndreas Gohr        $offset = $this->data['offset'];
51307993756SAndreas Gohr
51407993756SAndreas Gohr        // prev link
51507993756SAndreas Gohr        if ($offset) {
51607993756SAndreas Gohr            $prev = $offset - $this->data['limit'];
51707993756SAndreas Gohr            if ($prev < 0) {
51807993756SAndreas Gohr                $prev = 0;
51907993756SAndreas Gohr            }
52007993756SAndreas Gohr
52107993756SAndreas Gohr            $dynamic = $this->searchConfig->getDynamicParameters();
52207993756SAndreas Gohr            $dynamic->setOffset($prev);
52307993756SAndreas Gohr            $link = wl($this->id, $dynamic->getURLParameters());
52407993756SAndreas Gohr            $this->renderer->doc .= '<a href="' . $link . '" class="prev">' . $this->helper->getLang('prev') . '</a>';
52507993756SAndreas Gohr        }
52607993756SAndreas Gohr
52707993756SAndreas Gohr        // next link
52807993756SAndreas Gohr        if ($this->resultCount > $offset + $this->data['limit']) {
52907993756SAndreas Gohr            $next = $offset + $this->data['limit'];
53007993756SAndreas Gohr            $dynamic = $this->searchConfig->getDynamicParameters();
53107993756SAndreas Gohr            $dynamic->setOffset($next);
53207993756SAndreas Gohr            $link = wl($this->id, $dynamic->getURLParameters());
53307993756SAndreas Gohr            $this->renderer->doc .= '<a href="' . $link . '" class="next">' . $this->helper->getLang('next') . '</a>';
53407993756SAndreas Gohr        }
53507993756SAndreas Gohr
53607993756SAndreas Gohr        $this->renderer->tableheader_close();
53707993756SAndreas Gohr        $this->renderer->tablerow_close();
538a0bf8bb2SAndreas Gohr        $this->renderer->info['struct_table_meta'] = true;
53907993756SAndreas Gohr    }
54009dd691aSAndreas Gohr
54109dd691aSAndreas Gohr    /**
54209dd691aSAndreas Gohr     * Adds CSV export controls
54309dd691aSAndreas Gohr     */
544d6d97f60SAnna Dabrowska    protected function renderExportControls()
545d6d97f60SAnna Dabrowska    {
54609dd691aSAndreas Gohr        if ($this->mode != 'xhtml') return;
5477b240ca8SAndreas Gohr        if (empty($this->data['csv'])) return;
54809dd691aSAndreas Gohr        if (!$this->resultCount) return;
54909dd691aSAndreas Gohr
550c8ccdaf8SAndreas Gohr        $dynamic = $this->searchConfig->getDynamicParameters();
551c8ccdaf8SAndreas Gohr        $params = $dynamic->getURLParameters();
552c8ccdaf8SAndreas Gohr        $params['hash'] = $this->renderer->info['struct_table_hash'];
553c8ccdaf8SAndreas Gohr
55409dd691aSAndreas Gohr        // FIXME apply dynamic filters
555eafc109fSAndreas Gohr        $link = exportlink($this->id, 'struct_csv', $params);
55609dd691aSAndreas Gohr
55709dd691aSAndreas Gohr        $this->renderer->doc .= '<a href="' . $link . '" class="export mediafile mf_csv">' . $this->helper->getLang('csvexport') . '</a>';
55809dd691aSAndreas Gohr    }
55907993756SAndreas Gohr}
560