xref: /plugin/struct/meta/AggregationTable.php (revision 2dbe71f867df0982fb08a9a8f3cd3d027278de03)
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,
123844a4f01SFrieder Schrempf            'search' => $this->searchConfig,
124844a4f01SFrieder Schrempf            'columns' => $this->columns,
125844a4f01SFrieder Schrempf            'data' => $this->result
126844a4f01SFrieder Schrempf        );
127844a4f01SFrieder Schrempf
128*2dbe71f8SAnna Dabrowska        $event = new \Doku_Event(
129844a4f01SFrieder Schrempf            'PLUGIN_STRUCT_RENDER_AGGREGATION_TABLE',
130*2dbe71f8SAnna Dabrowska            $rendercontext
131844a4f01SFrieder Schrempf        );
132*2dbe71f8SAnna Dabrowska        $event->trigger([$this, 'renderTable']);
133844a4f01SFrieder Schrempf
134844a4f01SFrieder Schrempf        // export handle
135844a4f01SFrieder Schrempf        $this->renderExportControls();
136844a4f01SFrieder Schrempf        $this->finishScope();
137844a4f01SFrieder Schrempf    }
138844a4f01SFrieder Schrempf
139844a4f01SFrieder Schrempf    /**
140844a4f01SFrieder Schrempf     * Render the default aggregation table
141844a4f01SFrieder Schrempf     */
142844a4f01SFrieder Schrempf    public function renderTable($rendercontext)
143844a4f01SFrieder Schrempf    {
14407993756SAndreas Gohr        $this->renderer->table_open();
14507993756SAndreas Gohr
14607993756SAndreas Gohr        // header
14707993756SAndreas Gohr        $this->renderer->tablethead_open();
148986ab7e6SAndreas Gohr        $this->renderColumnHeaders();
149986ab7e6SAndreas Gohr        $this->renderDynamicFilters();
15007993756SAndreas Gohr        $this->renderer->tablethead_close();
15107993756SAndreas Gohr
15207993756SAndreas Gohr        if ($this->resultCount) {
15307993756SAndreas Gohr            // actual data
154a9fd81f9SAndreas Gohr            $this->renderer->tabletbody_open();
155986ab7e6SAndreas Gohr            $this->renderResult();
156a9fd81f9SAndreas Gohr            $this->renderer->tabletbody_close();
15707993756SAndreas Gohr
158a9fd81f9SAndreas Gohr            // footer (tfoot is develonly currently)
159a9fd81f9SAndreas Gohr            if (method_exists($this->renderer, 'tabletfoot_open')) $this->renderer->tabletfoot_open();
160986ab7e6SAndreas Gohr            $this->renderSums();
161986ab7e6SAndreas Gohr            $this->renderPagingControls();
162a9fd81f9SAndreas Gohr            if (method_exists($this->renderer, 'tabletfoot_close')) $this->renderer->tabletfoot_close();
16307993756SAndreas Gohr        } else {
16407993756SAndreas Gohr            // nothing found
165986ab7e6SAndreas Gohr            $this->renderEmptyResult();
16607993756SAndreas Gohr        }
16707993756SAndreas Gohr
16807993756SAndreas Gohr        // table close
16907993756SAndreas Gohr        $this->renderer->table_close();
17007993756SAndreas Gohr    }
17107993756SAndreas Gohr
17207993756SAndreas Gohr    /**
17307993756SAndreas Gohr     * Adds additional info to document and renderer in XHTML mode
17407993756SAndreas Gohr     *
17507993756SAndreas Gohr     * @see finishScope()
17607993756SAndreas Gohr     */
177d6d97f60SAnna Dabrowska    protected function startScope()
178d6d97f60SAnna Dabrowska    {
17907993756SAndreas Gohr        // unique identifier for this aggregation
18007993756SAndreas Gohr        $this->renderer->info['struct_table_hash'] = md5(var_export($this->data, true));
18109dd691aSAndreas Gohr
18209dd691aSAndreas Gohr        // wrapping div
18309dd691aSAndreas Gohr        if ($this->mode != 'xhtml') return;
18409dd691aSAndreas Gohr        $this->renderer->doc .= "<div class=\"structaggregation\">";
18507993756SAndreas Gohr    }
18607993756SAndreas Gohr
18707993756SAndreas Gohr    /**
18807993756SAndreas Gohr     * Closes the table and anything opened in startScope()
18907993756SAndreas Gohr     *
19007993756SAndreas Gohr     * @see startScope()
19107993756SAndreas Gohr     */
192d6d97f60SAnna Dabrowska    protected function finishScope()
193d6d97f60SAnna Dabrowska    {
19407993756SAndreas Gohr        // remove identifier from renderer again
19507993756SAndreas Gohr        if (isset($this->renderer->info['struct_table_hash'])) {
19607993756SAndreas Gohr            unset($this->renderer->info['struct_table_hash']);
19707993756SAndreas Gohr        }
19809dd691aSAndreas Gohr
19909dd691aSAndreas Gohr        // wrapping div
20009dd691aSAndreas Gohr        if ($this->mode != 'xhtml') return;
20109dd691aSAndreas Gohr        $this->renderer->doc .= '</div>';
20207993756SAndreas Gohr    }
20307993756SAndreas Gohr
20407993756SAndreas Gohr    /**
20507993756SAndreas Gohr     * Displays info about the currently applied filters
20607993756SAndreas Gohr     */
207d6d97f60SAnna Dabrowska    protected function renderActiveFilters()
208d6d97f60SAnna Dabrowska    {
20907993756SAndreas Gohr        if ($this->mode != 'xhtml') return;
21007993756SAndreas Gohr        $dynamic = $this->searchConfig->getDynamicParameters();
21107993756SAndreas Gohr        $filters = $dynamic->getFilters();
21207993756SAndreas Gohr        if (!$filters) return;
21307993756SAndreas Gohr
21407993756SAndreas Gohr        $fltrs = array();
21507993756SAndreas Gohr        foreach ($filters as $column => $filter) {
21607993756SAndreas Gohr            list($comp, $value) = $filter;
21704ec4785SAnna Dabrowska
21804ec4785SAnna Dabrowska            // display the filters in a human readable format
21904ec4785SAnna Dabrowska            foreach ($this->columns as $col) {
22004ec4785SAnna Dabrowska                if ($column === $col->getFullQualifiedLabel()) {
22104ec4785SAnna Dabrowska                    $column = $col->getTranslatedLabel();
22204ec4785SAnna Dabrowska                }
22304ec4785SAnna Dabrowska            }
2241f075418SAnna Dabrowska            $fltrs[] = sprintf('"%s" %s "%s"', $column, $this->helper->getLang("comparator $comp"), $value);
22507993756SAndreas Gohr        }
22607993756SAndreas Gohr
22707993756SAndreas Gohr        $this->renderer->doc .= '<div class="filter">';
22807993756SAndreas Gohr        $this->renderer->doc .= '<h4>' . sprintf($this->helper->getLang('tablefilteredby'), hsc(implode(' & ', $fltrs))) . '</h4>';
22907993756SAndreas Gohr        $this->renderer->doc .= '<div class="resetfilter">';
23007993756SAndreas Gohr        $this->renderer->internallink($this->id, $this->helper->getLang('tableresetfilter'));
23107993756SAndreas Gohr        $this->renderer->doc .= '</div>';
23207993756SAndreas Gohr        $this->renderer->doc .= '</div>';
23307993756SAndreas Gohr    }
23407993756SAndreas Gohr
23507993756SAndreas Gohr    /**
23607993756SAndreas Gohr     * Shows the column headers with links to sort by column
23707993756SAndreas Gohr     */
238d6d97f60SAnna Dabrowska    protected function renderColumnHeaders()
239d6d97f60SAnna Dabrowska    {
24007993756SAndreas Gohr        $this->renderer->tablerow_open();
24107993756SAndreas Gohr
24207993756SAndreas Gohr        // additional column for row numbers
24334ea6e10SAnna Dabrowska        if (!empty($this->data['rownumbers'])) {
24407993756SAndreas Gohr            $this->renderer->tableheader_open();
24507993756SAndreas Gohr            $this->renderer->cdata('#');
24607993756SAndreas Gohr            $this->renderer->tableheader_close();
24707993756SAndreas Gohr        }
24807993756SAndreas Gohr
24907993756SAndreas Gohr        // show all headers
2508c4ee9beSAndreas Gohr        foreach ($this->columns as $num => $column) {
2518c4ee9beSAndreas Gohr            $header = '';
2528c4ee9beSAndreas Gohr            if (isset($this->data['headers'][$num])) {
2538c4ee9beSAndreas Gohr                $header = $this->data['headers'][$num];
2548c4ee9beSAndreas Gohr            }
25507993756SAndreas Gohr
25607993756SAndreas Gohr            // use field label if no header was set
25707993756SAndreas Gohr            if (blank($header)) {
25801f8b845SAndreas Gohr                if (is_a($column, 'dokuwiki\plugin\struct\meta\Column')) {
25907993756SAndreas Gohr                    $header = $column->getTranslatedLabel();
26007993756SAndreas Gohr                } else {
26107993756SAndreas Gohr                    $header = 'column ' . $num; // this should never happen
26207993756SAndreas Gohr                }
26307993756SAndreas Gohr            }
26407993756SAndreas Gohr
26507993756SAndreas Gohr            // simple mode first
26607993756SAndreas Gohr            if ($this->mode != 'xhtml') {
26707993756SAndreas Gohr                $this->renderer->tableheader_open();
26807993756SAndreas Gohr                $this->renderer->cdata($header);
26907993756SAndreas Gohr                $this->renderer->tableheader_close();
27007993756SAndreas Gohr                continue;
27107993756SAndreas Gohr            }
27207993756SAndreas Gohr
27307993756SAndreas Gohr            // still here? create custom header for more flexibility
27407993756SAndreas Gohr
2759113d04aSAndreas Gohr            // width setting, widths are prevalidated, no escape needed
27607993756SAndreas Gohr            $width = '';
2779113d04aSAndreas Gohr            if (isset($this->data['widths'][$num]) && $this->data['widths'][$num] != '-') {
2789113d04aSAndreas Gohr                $width = ' style="min-width: ' . $this->data['widths'][$num] . ';' .
2799113d04aSAndreas Gohr                         'max-width: ' . $this->data['widths'][$num] . ';"';
28007993756SAndreas Gohr            }
28107993756SAndreas Gohr
282d4b5a17cSAndreas Gohr            // prepare data attribute for inline edits
283d6d97f60SAnna Dabrowska            if (
284d6d97f60SAnna Dabrowska                !is_a($column, '\dokuwiki\plugin\struct\meta\PageColumn') &&
285d4b5a17cSAndreas Gohr                !is_a($column, '\dokuwiki\plugin\struct\meta\RevisionColumn')
286d4b5a17cSAndreas Gohr            ) {
287d4b5a17cSAndreas Gohr                $data = 'data-field="' . hsc($column->getFullQualifiedLabel()) . '"';
288d4b5a17cSAndreas Gohr            } else {
289d4b5a17cSAndreas Gohr                $data = '';
290d4b5a17cSAndreas Gohr            }
291d4b5a17cSAndreas Gohr
29207993756SAndreas Gohr            // sort indicator and link
29307993756SAndreas Gohr            $sortclass = '';
29407993756SAndreas Gohr            $sorts = $this->searchConfig->getSorts();
29507993756SAndreas Gohr            $dynamic = $this->searchConfig->getDynamicParameters();
296aa124708SAndreas Gohr            $dynamic->setSort($column, true);
29707993756SAndreas Gohr            if (isset($sorts[$column->getFullQualifiedLabel()])) {
298aa124708SAndreas Gohr                list(/*colname*/, $currentSort) = $sorts[$column->getFullQualifiedLabel()];
299aa124708SAndreas Gohr                if ($currentSort) {
30007993756SAndreas Gohr                    $sortclass = 'sort-down';
30107993756SAndreas Gohr                    $dynamic->setSort($column, false);
30207993756SAndreas Gohr                } else {
30307993756SAndreas Gohr                    $sortclass = 'sort-up';
30407993756SAndreas Gohr                }
30507993756SAndreas Gohr            }
30607993756SAndreas Gohr            $link = wl($this->id, $dynamic->getURLParameters());
30707993756SAndreas Gohr
30807993756SAndreas Gohr            // output XHTML header
309d4b5a17cSAndreas Gohr            $this->renderer->doc .= "<th $width $data>";
31007993756SAndreas Gohr            $this->renderer->doc .= '<a href="' . $link . '" class="' . $sortclass . '" title="' . $this->helper->getLang('sort') . '">' . hsc($header) . '</a>';
31107993756SAndreas Gohr            $this->renderer->doc .= '</th>';
31207993756SAndreas Gohr        }
31307993756SAndreas Gohr
31407993756SAndreas Gohr        $this->renderer->tablerow_close();
31507993756SAndreas Gohr    }
31607993756SAndreas Gohr
31707993756SAndreas Gohr    /**
318b7e1d73bSAndreas Gohr     * Is the result set currently dynamically filtered?
319b7e1d73bSAndreas Gohr     * @return bool
320b7e1d73bSAndreas Gohr     */
321d6d97f60SAnna Dabrowska    protected function isDynamicallyFiltered()
322d6d97f60SAnna Dabrowska    {
323b7e1d73bSAndreas Gohr        if ($this->mode != 'xhtml') return false;
324b7e1d73bSAndreas Gohr        if (!$this->data['dynfilters']) return false;
325b7e1d73bSAndreas Gohr
326b7e1d73bSAndreas Gohr        $dynamic = $this->searchConfig->getDynamicParameters();
327b7e1d73bSAndreas Gohr        return (bool) $dynamic->getFilters();
328b7e1d73bSAndreas Gohr    }
329b7e1d73bSAndreas Gohr
330b7e1d73bSAndreas Gohr    /**
33107993756SAndreas Gohr     * Add input fields for dynamic filtering
33207993756SAndreas Gohr     */
333d6d97f60SAnna Dabrowska    protected function renderDynamicFilters()
334d6d97f60SAnna Dabrowska    {
33507993756SAndreas Gohr        if ($this->mode != 'xhtml') return;
33607993756SAndreas Gohr        if (!$this->data['dynfilters']) return;
3371bc467a4SMichael Grosse        if (is_a($this->renderer, 'renderer_plugin_dw2pdf')) {
338e6ae02ecSMichael Grosse            return;
339e6ae02ecSMichael Grosse        }
3401bc467a4SMichael Grosse        global $conf;
34107993756SAndreas Gohr
34207993756SAndreas Gohr        $this->renderer->doc .= '<tr class="dataflt">';
34307993756SAndreas Gohr
34407993756SAndreas Gohr        // add extra column for row numbers
34507993756SAndreas Gohr        if ($this->data['rownumbers']) {
34607993756SAndreas Gohr            $this->renderer->doc .= '<th></th>';
34707993756SAndreas Gohr        }
34807993756SAndreas Gohr
34907993756SAndreas Gohr        // each column gets a form
35007993756SAndreas Gohr        foreach ($this->columns as $column) {
35107993756SAndreas Gohr            $this->renderer->doc .= '<th>';
35207993756SAndreas Gohr            {
35307993756SAndreas Gohr                $form = new \Doku_Form(array('method' => 'GET', 'action' => wl($this->id)));
35476195677SAndreas Gohr                unset($form->_hidden['sectok']); // we don't need it here
35576195677SAndreas Gohr                if (!$conf['userewrite']) $form->addHidden('id', $this->id);
35607993756SAndreas Gohr
35707993756SAndreas Gohr                // current value
35807993756SAndreas Gohr                $dynamic = $this->searchConfig->getDynamicParameters();
35907993756SAndreas Gohr                $filters = $dynamic->getFilters();
36007993756SAndreas Gohr            if (isset($filters[$column->getFullQualifiedLabel()])) {
36107993756SAndreas Gohr                list(, $current) = $filters[$column->getFullQualifiedLabel()];
36207993756SAndreas Gohr                $dynamic->removeFilter($column);
36307993756SAndreas Gohr            } else {
36407993756SAndreas Gohr                $current = '';
36507993756SAndreas Gohr            }
36607993756SAndreas Gohr
36707993756SAndreas Gohr                // Add current request params
36807993756SAndreas Gohr                $params = $dynamic->getURLParameters();
36907993756SAndreas Gohr            foreach ($params as $key => $val) {
37007993756SAndreas Gohr                $form->addHidden($key, $val);
37107993756SAndreas Gohr            }
37207993756SAndreas Gohr
37307993756SAndreas Gohr                // add input field
374db9b8745SAndreas Gohr                $key = $column->getFullQualifiedLabel() . $column->getType()->getDefaultComparator();
375d60f71efSAndreas Gohr                $form->addElement(form_makeField('text', SearchConfigParameters::$PARAM_FILTER . '[' . $key . ']', $current, ''));
37607993756SAndreas Gohr                $this->renderer->doc .= $form->getForm();
37707993756SAndreas Gohr            }
37807993756SAndreas Gohr            $this->renderer->doc .= '</th>';
37907993756SAndreas Gohr        }
38007993756SAndreas Gohr        $this->renderer->doc .= '</tr>';
38107993756SAndreas Gohr    }
38207993756SAndreas Gohr
38307993756SAndreas Gohr    /**
38407993756SAndreas Gohr     * Display the actual table data
38507993756SAndreas Gohr     */
386d6d97f60SAnna Dabrowska    protected function renderResult()
387d6d97f60SAnna Dabrowska    {
38807993756SAndreas Gohr        foreach ($this->result as $rownum => $row) {
38947eb8cceSSzymon Olewniczak            $data = array(
39047eb8cceSSzymon Olewniczak                'id' => $this->id,
39147eb8cceSSzymon Olewniczak                'mode' => $this->mode,
39247eb8cceSSzymon Olewniczak                'renderer' => $this->renderer,
39347eb8cceSSzymon Olewniczak                'searchConfig' => $this->searchConfig,
39447eb8cceSSzymon Olewniczak                'data' => $this->data,
39547eb8cceSSzymon Olewniczak                'rownum' => &$rownum,
39647eb8cceSSzymon Olewniczak                'row' => &$row,
39747eb8cceSSzymon Olewniczak            );
39847eb8cceSSzymon Olewniczak            $evt = new \Doku_Event('PLUGIN_STRUCT_AGGREGATIONTABLE_RENDERRESULTROW', $data);
39947eb8cceSSzymon Olewniczak            if ($evt->advise_before()) {
400f107f479SAndreas Gohr                $this->renderResultRow($rownum, $row);
401f107f479SAndreas Gohr            }
40247eb8cceSSzymon Olewniczak            $evt->advise_after();
40347eb8cceSSzymon Olewniczak        }
404f107f479SAndreas Gohr    }
405f107f479SAndreas Gohr
406f107f479SAndreas Gohr    /**
407f107f479SAndreas Gohr     * Render a single result row
408f107f479SAndreas Gohr     *
409f107f479SAndreas Gohr     * @param int $rownum
410f107f479SAndreas Gohr     * @param array $row
411f107f479SAndreas Gohr     */
412d6d97f60SAnna Dabrowska    protected function renderResultRow($rownum, $row)
413d6d97f60SAnna Dabrowska    {
41407993756SAndreas Gohr        $this->renderer->tablerow_open();
41507993756SAndreas Gohr
416d4b5a17cSAndreas Gohr        // add data attribute for inline edit
417d4b5a17cSAndreas Gohr        if ($this->mode == 'xhtml') {
418d4b5a17cSAndreas Gohr            $pid = $this->resultPIDs[$rownum];
4190ceefd5cSAnna Dabrowska            $rid = $this->resultRids[$rownum];
4206fd73b4bSAnna Dabrowska            $rev = $this->resultRevs[$rownum];
421d4b5a17cSAndreas Gohr            $this->renderer->doc = substr(rtrim($this->renderer->doc), 0, -1); // remove closing '>'
4226fd73b4bSAnna Dabrowska            $this->renderer->doc .= ' data-pid="' . hsc($pid) . '" data-rev="' . $rev . '" data-rid="' . $rid . '">';
423d4b5a17cSAndreas Gohr        }
424d4b5a17cSAndreas Gohr
42507993756SAndreas Gohr        // row number column
42634ea6e10SAnna Dabrowska        if (!empty($this->data['rownumbers'])) {
42707993756SAndreas Gohr            $this->renderer->tablecell_open();
4283215aebfSSzymon Olewniczak            $searchConfigConf = $this->searchConfig->getConf();
4293215aebfSSzymon Olewniczak            $this->renderer->cdata($rownum + $searchConfigConf['offset'] + 1);
43007993756SAndreas Gohr            $this->renderer->tablecell_close();
43107993756SAndreas Gohr        }
43207993756SAndreas Gohr
43307993756SAndreas Gohr        /** @var Value $value */
43407993756SAndreas Gohr        foreach ($row as $colnum => $value) {
43534ea6e10SAnna Dabrowska            $align = isset($this->data['align'][$colnum]) ?  $this->data['align'][$colnum] : null;
43634ea6e10SAnna Dabrowska            $this->renderer->tablecell_open(1, $align);
43707993756SAndreas Gohr            $value->render($this->renderer, $this->mode);
43807993756SAndreas Gohr            $this->renderer->tablecell_close();
43907993756SAndreas Gohr
44007993756SAndreas Gohr            // summarize
44107993756SAndreas Gohr            if ($this->data['summarize'] && is_numeric($value->getValue())) {
44207993756SAndreas Gohr                if (!isset($this->sums[$colnum])) {
44307993756SAndreas Gohr                    $this->sums[$colnum] = 0;
44407993756SAndreas Gohr                }
44507993756SAndreas Gohr                $this->sums[$colnum] += $value->getValue();
44607993756SAndreas Gohr            }
44707993756SAndreas Gohr        }
44807993756SAndreas Gohr        $this->renderer->tablerow_close();
44907993756SAndreas Gohr    }
45007993756SAndreas Gohr
45107993756SAndreas Gohr    /**
45207993756SAndreas Gohr     * Renders an information row for when no results were found
45307993756SAndreas Gohr     */
454d6d97f60SAnna Dabrowska    protected function renderEmptyResult()
455d6d97f60SAnna Dabrowska    {
45607993756SAndreas Gohr        $this->renderer->tablerow_open();
45770cf6339SAndreas Gohr        $this->renderer->tablecell_open(count($this->columns) + $this->data['rownumbers'], 'center');
45807993756SAndreas Gohr        $this->renderer->cdata($this->helper->getLang('none'));
45907993756SAndreas Gohr        $this->renderer->tablecell_close();
46007993756SAndreas Gohr        $this->renderer->tablerow_close();
46107993756SAndreas Gohr    }
46207993756SAndreas Gohr
46307993756SAndreas Gohr    /**
46407993756SAndreas Gohr     * Add sums if wanted
46507993756SAndreas Gohr     */
466d6d97f60SAnna Dabrowska    protected function renderSums()
467d6d97f60SAnna Dabrowska    {
468d18090e8SAndreas Gohr        if (empty($this->data['summarize'])) return;
46907993756SAndreas Gohr
470a0bf8bb2SAndreas Gohr        $this->renderer->info['struct_table_meta'] = true;
4718925ba29SAndreas Gohr        if ($this->mode == 'xhtml') {
4728925ba29SAndreas Gohr            /** @noinspection PhpMethodParametersCountMismatchInspection */
4738925ba29SAndreas Gohr            $this->renderer->tablerow_open('summarize');
4748925ba29SAndreas Gohr        } else {
47507993756SAndreas Gohr            $this->renderer->tablerow_open();
4768925ba29SAndreas Gohr        }
47707993756SAndreas Gohr
47807993756SAndreas Gohr        if ($this->data['rownumbers']) {
4798925ba29SAndreas Gohr            $this->renderer->tableheader_open();
4808925ba29SAndreas Gohr            $this->renderer->tableheader_close();
48107993756SAndreas Gohr        }
48207993756SAndreas Gohr
483aee4116bSAndreas Gohr        $len = count($this->columns);
48407993756SAndreas Gohr        for ($i = 0; $i < $len; $i++) {
4858925ba29SAndreas Gohr            $this->renderer->tableheader_open(1, $this->data['align'][$i]);
486aee4116bSAndreas Gohr            if (!empty($this->sums[$i])) {
4879b97e610SAndreas Gohr                $this->renderer->cdata('∑ ');
4889b97e610SAndreas Gohr                $this->columns[$i]->getType()->renderValue($this->sums[$i], $this->renderer, $this->mode);
48907993756SAndreas Gohr            } else {
49007993756SAndreas Gohr                if ($this->mode == 'xhtml') {
49107993756SAndreas Gohr                    $this->renderer->doc .= '&nbsp;';
49207993756SAndreas Gohr                }
49307993756SAndreas Gohr            }
4948925ba29SAndreas Gohr            $this->renderer->tableheader_close();
49507993756SAndreas Gohr        }
49607993756SAndreas Gohr        $this->renderer->tablerow_close();
497a0bf8bb2SAndreas Gohr        $this->renderer->info['struct_table_meta'] = false;
49807993756SAndreas Gohr    }
49907993756SAndreas Gohr
50007993756SAndreas Gohr    /**
501986ab7e6SAndreas Gohr     * Adds paging controls to the table
50207993756SAndreas Gohr     */
503d6d97f60SAnna Dabrowska    protected function renderPagingControls()
504d6d97f60SAnna Dabrowska    {
50507993756SAndreas Gohr        if (empty($this->data['limit'])) return;
506a0bf8bb2SAndreas Gohr        if ($this->mode != 'xhtml') return;
50707993756SAndreas Gohr
508a0bf8bb2SAndreas Gohr        $this->renderer->info['struct_table_meta'] = true;
50907993756SAndreas Gohr        $this->renderer->tablerow_open();
5104bc1074dSMichael Grosse        $this->renderer->tableheader_open((count($this->columns) + ($this->data['rownumbers'] ? 1 : 0)));
51107993756SAndreas Gohr        $offset = $this->data['offset'];
51207993756SAndreas Gohr
51307993756SAndreas Gohr        // prev link
51407993756SAndreas Gohr        if ($offset) {
51507993756SAndreas Gohr            $prev = $offset - $this->data['limit'];
51607993756SAndreas Gohr            if ($prev < 0) {
51707993756SAndreas Gohr                $prev = 0;
51807993756SAndreas Gohr            }
51907993756SAndreas Gohr
52007993756SAndreas Gohr            $dynamic = $this->searchConfig->getDynamicParameters();
52107993756SAndreas Gohr            $dynamic->setOffset($prev);
52207993756SAndreas Gohr            $link = wl($this->id, $dynamic->getURLParameters());
52307993756SAndreas Gohr            $this->renderer->doc .= '<a href="' . $link . '" class="prev">' . $this->helper->getLang('prev') . '</a>';
52407993756SAndreas Gohr        }
52507993756SAndreas Gohr
52607993756SAndreas Gohr        // next link
52707993756SAndreas Gohr        if ($this->resultCount > $offset + $this->data['limit']) {
52807993756SAndreas Gohr            $next = $offset + $this->data['limit'];
52907993756SAndreas Gohr            $dynamic = $this->searchConfig->getDynamicParameters();
53007993756SAndreas Gohr            $dynamic->setOffset($next);
53107993756SAndreas Gohr            $link = wl($this->id, $dynamic->getURLParameters());
53207993756SAndreas Gohr            $this->renderer->doc .= '<a href="' . $link . '" class="next">' . $this->helper->getLang('next') . '</a>';
53307993756SAndreas Gohr        }
53407993756SAndreas Gohr
53507993756SAndreas Gohr        $this->renderer->tableheader_close();
53607993756SAndreas Gohr        $this->renderer->tablerow_close();
537a0bf8bb2SAndreas Gohr        $this->renderer->info['struct_table_meta'] = true;
53807993756SAndreas Gohr    }
53909dd691aSAndreas Gohr
54009dd691aSAndreas Gohr    /**
54109dd691aSAndreas Gohr     * Adds CSV export controls
54209dd691aSAndreas Gohr     */
543d6d97f60SAnna Dabrowska    protected function renderExportControls()
544d6d97f60SAnna Dabrowska    {
54509dd691aSAndreas Gohr        if ($this->mode != 'xhtml') return;
5467b240ca8SAndreas Gohr        if (empty($this->data['csv'])) return;
54709dd691aSAndreas Gohr        if (!$this->resultCount) return;
54809dd691aSAndreas Gohr
549c8ccdaf8SAndreas Gohr        $dynamic = $this->searchConfig->getDynamicParameters();
550c8ccdaf8SAndreas Gohr        $params = $dynamic->getURLParameters();
551c8ccdaf8SAndreas Gohr        $params['hash'] = $this->renderer->info['struct_table_hash'];
552c8ccdaf8SAndreas Gohr
55309dd691aSAndreas Gohr        // FIXME apply dynamic filters
554eafc109fSAndreas Gohr        $link = exportlink($this->id, 'struct_csv', $params);
55509dd691aSAndreas Gohr
55609dd691aSAndreas Gohr        $this->renderer->doc .= '<a href="' . $link . '" class="export mediafile mf_csv">' . $this->helper->getLang('csvexport') . '</a>';
55709dd691aSAndreas Gohr    }
55807993756SAndreas Gohr}
559