xref: /plugin/struct/syntax/table.php (revision 1a07b69644a74cb7c512c56dff6a8a500d61eef7)
1549a0837SAndreas Gohr<?php
2549a0837SAndreas Gohr/**
3549a0837SAndreas Gohr * DokuWiki Plugin struct (Syntax Component)
4549a0837SAndreas Gohr *
5549a0837SAndreas Gohr * @license GPL 2 http://www.gnu.org/licenses/gpl-2.0.html
6549a0837SAndreas Gohr * @author  Andreas Gohr, Michael Große <dokuwiki@cosmocode.de>
7549a0837SAndreas Gohr */
8549a0837SAndreas Gohr
9549a0837SAndreas Gohr// must be run within Dokuwiki
105511bd5bSAndreas Gohruse plugin\struct\meta\ConfigParser;
1115929be2SAndreas Gohruse plugin\struct\meta\Search;
125511bd5bSAndreas Gohruse plugin\struct\meta\SearchConfig;
1315929be2SAndreas Gohruse plugin\struct\meta\SearchException;
145511bd5bSAndreas Gohruse plugin\struct\meta\StructException;
1515929be2SAndreas Gohr
16549a0837SAndreas Gohrif (!defined('DOKU_INC')) die();
17549a0837SAndreas Gohr
18549a0837SAndreas Gohrclass syntax_plugin_struct_table extends DokuWiki_Syntax_Plugin {
19549a0837SAndreas Gohr    /**
20549a0837SAndreas Gohr     * @return string Syntax mode type
21549a0837SAndreas Gohr     */
22549a0837SAndreas Gohr    public function getType() {
2315929be2SAndreas Gohr        return 'substition';
24549a0837SAndreas Gohr    }
25549a0837SAndreas Gohr    /**
26549a0837SAndreas Gohr     * @return string Paragraph type
27549a0837SAndreas Gohr     */
28549a0837SAndreas Gohr    public function getPType() {
2915929be2SAndreas Gohr        return 'block';
30549a0837SAndreas Gohr    }
31549a0837SAndreas Gohr    /**
32549a0837SAndreas Gohr     * @return int Sort order - Low numbers go before high numbers
33549a0837SAndreas Gohr     */
34549a0837SAndreas Gohr    public function getSort() {
355511bd5bSAndreas Gohr        return 155;
36549a0837SAndreas Gohr    }
37549a0837SAndreas Gohr
38549a0837SAndreas Gohr    /**
39549a0837SAndreas Gohr     * Connect lookup pattern to lexer.
40549a0837SAndreas Gohr     *
41549a0837SAndreas Gohr     * @param string $mode Parser mode
42549a0837SAndreas Gohr     */
43549a0837SAndreas Gohr    public function connectTo($mode) {
445511bd5bSAndreas Gohr        $this->Lexer->addSpecialPattern('----+ *struct table *-+\n.*?\n----+', $mode, 'plugin_struct_table');
45549a0837SAndreas Gohr    }
46549a0837SAndreas Gohr
47549a0837SAndreas Gohr
48549a0837SAndreas Gohr    /**
49549a0837SAndreas Gohr     * Handle matches of the struct syntax
50549a0837SAndreas Gohr     *
51549a0837SAndreas Gohr     * @param string $match The match of the syntax
52549a0837SAndreas Gohr     * @param int    $state The state of the handler
53549a0837SAndreas Gohr     * @param int    $pos The position in the document
54549a0837SAndreas Gohr     * @param Doku_Handler    $handler The handler
55549a0837SAndreas Gohr     * @return array Data for the renderer
56549a0837SAndreas Gohr     */
57ab466032SAndreas Gohr    public function handle($match, $state, $pos, Doku_Handler $handler){
58549a0837SAndreas Gohr
595511bd5bSAndreas Gohr        $lines = explode("\n", $match);
605511bd5bSAndreas Gohr        array_shift($lines);
615511bd5bSAndreas Gohr        array_pop($lines);
625511bd5bSAndreas Gohr
635511bd5bSAndreas Gohr        try {
645511bd5bSAndreas Gohr            $parser = new ConfigParser($lines);
655511bd5bSAndreas Gohr            return  $parser->getConfig();
665511bd5bSAndreas Gohr        } catch (StructException $e) {
675511bd5bSAndreas Gohr            msg($e->getMessage(), -1, $e->getLine(), $e->getFile());
685511bd5bSAndreas Gohr            return null;
695511bd5bSAndreas Gohr        }
70549a0837SAndreas Gohr    }
71549a0837SAndreas Gohr
7229877279SMichael Große    protected $sums = array();
7329877279SMichael Große
7429877279SMichael Große    /** @var helper_plugin_struct_aggregation $dthlp  */
7529877279SMichael Große    protected $dthlp = null;
7629877279SMichael Große
77549a0837SAndreas Gohr    /**
78549a0837SAndreas Gohr     * Render xhtml output or metadata
79549a0837SAndreas Gohr     *
80549a0837SAndreas Gohr     * @param string         $mode      Renderer mode (supported modes: xhtml)
81549a0837SAndreas Gohr     * @param Doku_Renderer  $renderer  The renderer
82549a0837SAndreas Gohr     * @param array          $data      The data from the handler() function
83549a0837SAndreas Gohr     * @return bool If rendering was successful.
84549a0837SAndreas Gohr     */
85ab466032SAndreas Gohr    public function render($mode, Doku_Renderer $renderer, $data) {
86549a0837SAndreas Gohr        if($mode != 'xhtml') return false;
875511bd5bSAndreas Gohr        if(!$data) return false;
8829877279SMichael Große        $this->dthlp = $this->loadHelper('struct_aggregation');
8929877279SMichael Große
9029877279SMichael Große        $clist = $data['cols'];
9129877279SMichael Große
9229877279SMichael Große        //reset counters
9329877279SMichael Große        $this->sums = array();
9429877279SMichael Große
9515929be2SAndreas Gohr        try {
965511bd5bSAndreas Gohr            $search = new SearchConfig($data);
97*1a07b696SMichael Große            $data = $search->getConf();
9829877279SMichael Große            $rows = $search->execute();
9929877279SMichael Große            $cnt = count($rows);
1005511bd5bSAndreas Gohr
10129877279SMichael Große            if ($cnt === 0) {
10229877279SMichael Große                //$this->nullList($data, $clist, $R);
10329877279SMichael Große                //return true;
10429877279SMichael Große            }
1055511bd5bSAndreas Gohr
10629877279SMichael Große            $this->renderPreTable($mode, $renderer, $clist, $data);
10729877279SMichael Große            $this->renderRows($mode, $renderer, $data, $rows);
10829877279SMichael Große            $this->renderPostTable($mode, $renderer, $data, $cnt);
1095511bd5bSAndreas Gohr        } catch (StructException $e) {
11015929be2SAndreas Gohr            msg($e->getMessage(), -1, $e->getLine(), $e->getFile());
11115929be2SAndreas Gohr        }
11215929be2SAndreas Gohr
113549a0837SAndreas Gohr        return true;
114549a0837SAndreas Gohr    }
11529877279SMichael Große
11629877279SMichael Große    /**
11729877279SMichael Große     * create the pretext to the actual table rows
11829877279SMichael Große     *
11929877279SMichael Große     * @param               $mode
12029877279SMichael Große     * @param Doku_Renderer $renderer
12129877279SMichael Große     * @param               $clist
12229877279SMichael Große     * @param               $data
12329877279SMichael Große     */
12429877279SMichael Große    protected function renderPreTable($mode, Doku_Renderer $renderer, $clist, $data) {
12529877279SMichael Große        $this->startScope($mode, $renderer);
126fc8e2563SMichael Große        $this->showActiveFilters($mode, $renderer);
12729877279SMichael Große        $this->startTable($mode, $renderer);
12829877279SMichael Große        $renderer->tablethead_open();
129*1a07b696SMichael Große        $this->buildColumnHeaders($mode, $renderer, $clist, $data);
130*1a07b696SMichael Große        $this->addDynamicFilters($mode, $renderer, $data);
13129877279SMichael Große        $renderer->tablethead_close();
13229877279SMichael Große    }
13329877279SMichael Große
13429877279SMichael Große    /**
13529877279SMichael Große     * @param array $data
13629877279SMichael Große     * @param int $rowcnt
13729877279SMichael Große     *
13829877279SMichael Große     * @return string
13929877279SMichael Große     */
14029877279SMichael Große    private function renderPostTable($mode, Doku_Renderer $renderer, $data, $rowcnt) {
14129877279SMichael Große        $this->summarize($mode, $renderer, $data, $this->sums);
14229877279SMichael Große        $this->addLimitControls($mode, $renderer, $data, $rowcnt);
14329877279SMichael Große        $this->finishTableAndScope($mode, $renderer);
14429877279SMichael Große    }
14529877279SMichael Große
14629877279SMichael Große    /**
14729877279SMichael Große     * if limit was set, add control
14829877279SMichael Große     *
14993485d71SMichael Große     * @param               $mode
15093485d71SMichael Große     * @param Doku_Renderer $renderer
15129877279SMichael Große     * @param               $data
15229877279SMichael Große     * @param               $rowcnt
15329877279SMichael Große     */
15429877279SMichael Große    protected function addLimitControls($mode, Doku_Renderer $renderer, $data, $rowcnt) {
155*1a07b696SMichael Große        global $ID;
15629877279SMichael Große
15729877279SMichael Große        if($data['limit']) {
15829877279SMichael Große            $renderer->tablerow_open();
15929877279SMichael Große            $renderer->tableheader_open((count($data['cols']) + ($data['rownumbers'] ? 1 : 0)));
16029877279SMichael Große            $offset = (int) $_REQUEST['dataofs'];
161fc8e2563SMichael Große
162fc8e2563SMichael Große            // keep url params
163fc8e2563SMichael Große            $params = array();
164*1a07b696SMichael Große            if (!empty($data['current_params']['dataflt'])) {$params['dataflt'] = $data['current_params']['dataflt'];}
165*1a07b696SMichael Große            if (!empty($data['current_params']['datasrt'])) {$params['datasrt'] = $data['current_params']['datasrt'];}
166fc8e2563SMichael Große
16729877279SMichael Große            if($offset) {
16829877279SMichael Große                $prev = $offset - $data['limit'];
16929877279SMichael Große                if($prev < 0) {
17029877279SMichael Große                    $prev = 0;
17129877279SMichael Große                }
17229877279SMichael Große                $params['dataofs'] = $prev;
173fc8e2563SMichael Große                $renderer->internallink($ID . '?' . http_build_query($params), $this->getLang('prev'));
17429877279SMichael Große            }
17529877279SMichael Große
17693485d71SMichael Große            if($rowcnt > $offset + $data['limit']) {
17729877279SMichael Große                $next = $offset + $data['limit'];
17829877279SMichael Große                $params['dataofs'] = $next;
179fc8e2563SMichael Große                $renderer->internallink($ID . '?' . http_build_query($params), $this->getLang('next'));
18029877279SMichael Große            }
18129877279SMichael Große            $renderer->tableheader_close();
18229877279SMichael Große            $renderer->tablerow_close();
18329877279SMichael Große        }
18429877279SMichael Große    }
18529877279SMichael Große
18629877279SMichael Große    /**
18729877279SMichael Große     * @param               $mode
18829877279SMichael Große     * @param Doku_Renderer $renderer
18929877279SMichael Große     */
190fc8e2563SMichael Große    protected function showActiveFilters($mode, Doku_Renderer $renderer) {
191*1a07b696SMichael Große        global $ID;
19229877279SMichael Große
193*1a07b696SMichael Große        if($mode == 'xhtml' && !empty($data['current_params']['dataflt'])) {
194*1a07b696SMichael Große            $filters = $data['current_params']['dataflt'];
19529877279SMichael Große            $confHelper = $this->loadHelper('struct_config');
19629877279SMichael Große            $fltrs = array();
19729877279SMichael Große            foreach($filters as $colcomp => $filter) {
19829877279SMichael Große                $filter = $confHelper->parseFilterLine('', $colcomp.$filter);
19929877279SMichael Große                if(strpos($filter[1], '~') !== false) {
20029877279SMichael Große                    if(strpos($filter[1], '!~') !== false) {
20129877279SMichael Große                        $comparator_value = '!~' . str_replace('%', '*', $filter[2]);
20229877279SMichael Große                    } else {
20329877279SMichael Große                        $comparator_value = '~' . str_replace('%', '', $filter[2]);
20429877279SMichael Große                    }
20529877279SMichael Große                    $fltrs[] = $filter[0] . $comparator_value;
20629877279SMichael Große                } else {
20729877279SMichael Große                    $fltrs[] = $filter[0] . $filter[1] . $filter[2];
20829877279SMichael Große                }
20929877279SMichael Große            }
21029877279SMichael Große
21129877279SMichael Große            $renderer->doc .= '<div class="filter">';
21229877279SMichael Große            $renderer->doc .= '<h4>' . sprintf($this->getLang('tablefilteredby'), hsc(implode(' & ', $fltrs))) . '</h4>';
21329877279SMichael Große            $renderer->doc .= '<div class="resetfilter">';
21429877279SMichael Große            $renderer->internallink($ID, $this->getLang('tableresetfilter'));
21529877279SMichael Große            $renderer->doc .=  '</div>';
21629877279SMichael Große            $renderer->doc .= '</div>';
21729877279SMichael Große        }
21829877279SMichael Große    }
21929877279SMichael Große
22029877279SMichael Große    /**
22129877279SMichael Große     * @param               $mode
22229877279SMichael Große     * @param Doku_Renderer $renderer
22329877279SMichael Große     * @param               $data
22429877279SMichael Große     */
225*1a07b696SMichael Große    protected function addDynamicFilters($mode, Doku_Renderer $renderer, $data) {
22629877279SMichael Große        if ($mode != 'xhtml') return;
22729877279SMichael Große
22829877279SMichael Große        global $conf, $ID;
22929877279SMichael Große
230*1a07b696SMichael Große        $cur_params = $data['current_params'];
23129877279SMichael Große        $html = '';
23229877279SMichael Große        if($data['dynfilters']) {
23329877279SMichael Große            $html .= '<tr class="dataflt">';
23429877279SMichael Große
23529877279SMichael Große            if($data['rownumbers']) {
23629877279SMichael Große                $html .= '<th></th>';
23729877279SMichael Große            }
23829877279SMichael Große
23929877279SMichael Große            foreach($data['headers'] as $num => $head) {
24029877279SMichael Große                $html .= '<th>';
24129877279SMichael Große                $form = new Doku_Form(array('method' => 'GET',));
24229877279SMichael Große                $form->_hidden = array();
24329877279SMichael Große                if(!$conf['userewrite']) {
24429877279SMichael Große                    $form->addHidden('id', $ID);
24529877279SMichael Große                }
24629877279SMichael Große
247fc8e2563SMichael Große                $key = $data['cols'][$num] . '*~';
248fc8e2563SMichael Große                $val = isset($cur_params['dataflt'][$key]) ? $cur_params['dataflt'][$key] : '';
24929877279SMichael Große
25029877279SMichael Große                // Add current request params
251fc8e2563SMichael Große                if (!empty($cur_params['datasrt'])) {
252fc8e2563SMichael Große                    $form->addHidden('datasrt', $cur_params['datasrt']);
253fc8e2563SMichael Große                }
254fc8e2563SMichael Große                if (!empty($cur_params['dataofs'])) {
255fc8e2563SMichael Große                    $form->addHidden('dataofs', $cur_params['dataofs']);
256fc8e2563SMichael Große                }
257fc8e2563SMichael Große                foreach($cur_params['dataflt'] as $c_key => $c_val) {
25829877279SMichael Große                    if($c_val !== '' && $c_key !== $key) {
259fc8e2563SMichael Große                        $form->addHidden('dataflt[' . $c_key . ']', $c_val);
26029877279SMichael Große                    }
26129877279SMichael Große                }
26229877279SMichael Große
263fc8e2563SMichael Große                $form->addElement(form_makeField('text', 'dataflt[' . $key . ']', $val, ''));
26429877279SMichael Große                $html .= $form->getForm();
26529877279SMichael Große                $html .= '</th>';
26629877279SMichael Große            }
26729877279SMichael Große            $html .= '</tr>';
26829877279SMichael Große            $renderer->doc .= $html;
26929877279SMichael Große        }
27029877279SMichael Große    }
27129877279SMichael Große
27229877279SMichael Große    /**
27329877279SMichael Große     * @param               $mode
27429877279SMichael Große     * @param Doku_Renderer $renderer
27529877279SMichael Große     */
27629877279SMichael Große    private function startTable($mode, Doku_Renderer $renderer) {
27729877279SMichael Große        $renderer->table_open();
27829877279SMichael Große    }
27929877279SMichael Große
28029877279SMichael Große    /**
28129877279SMichael Große     * @param               $mode
28229877279SMichael Große     * @param Doku_Renderer $renderer
28329877279SMichael Große     * @param               $clist
28429877279SMichael Große     * @param               $data
28529877279SMichael Große     *
28629877279SMichael Große     */
287*1a07b696SMichael Große    protected function buildColumnHeaders($mode, Doku_Renderer $renderer, $clist, $data) {
28829877279SMichael Große        global $ID;
28929877279SMichael Große
29029877279SMichael Große        $renderer->tablerow_open();
29129877279SMichael Große
29229877279SMichael Große        if($data['rownumbers']) {
29329877279SMichael Große            $renderer->tableheader_open();
29429877279SMichael Große            $renderer->cdata('#');
29529877279SMichael Große            $renderer->tableheader_close();
29629877279SMichael Große        }
29729877279SMichael Große
29829877279SMichael Große        foreach($data['headers'] as $num => $head) {
29929877279SMichael Große            $ckey = $clist[$num];
30029877279SMichael Große
30129877279SMichael Große            $width = '';
30229877279SMichael Große            if(isset($data['widths'][$num]) AND $data['widths'][$num] != '-') {
30329877279SMichael Große                $width = ' style="width: ' . $data['widths'][$num] . ';"';
30429877279SMichael Große            }
30529877279SMichael Große            if ($mode == 'xhmtl') {
30629877279SMichael Große                $renderer->doc .= '<th' . $width . '>';
30729877279SMichael Große            } else {
30829877279SMichael Große                $renderer->tableheader_open();
30929877279SMichael Große            }
31029877279SMichael Große
31129877279SMichael Große            // add sort arrow
31229877279SMichael Große            if ($mode == 'xhtml') {
31329877279SMichael Große                if(isset($data['sort']) && $ckey == $data['sort'][0]) {
31429877279SMichael Große                    if($data['sort'][1] == 'ASC') {
31529877279SMichael Große                        $renderer->doc .= '<span>&darr;</span> ';
31629877279SMichael Große                        $ckey = '^' . $ckey;
31729877279SMichael Große                    } else {
31829877279SMichael Große                        $renderer->doc .= '<span>&uarr;</span> ';
31929877279SMichael Große                    }
32029877279SMichael Große                }
32129877279SMichael Große            }
322*1a07b696SMichael Große            $renderer->internallink($ID . "?" . http_build_query(array('datasrt' => $ckey,) + $data['current_params']), hsc($head));
32329877279SMichael Große            $renderer->tableheader_close();
32429877279SMichael Große        }
32529877279SMichael Große        $renderer->tablerow_close();
32629877279SMichael Große    }
32729877279SMichael Große
32829877279SMichael Große
32929877279SMichael Große    protected function startScope($mode, \Doku_Renderer $renderer) {
33029877279SMichael Große        if ($mode == 'xhtml') {
33129877279SMichael Große            $renderer->doc .= '<div class="table structaggegation">';
33229877279SMichael Große        }
33329877279SMichael Große    }
33429877279SMichael Große
33529877279SMichael Große    /**
33629877279SMichael Große     * if summarize was set, add sums
33729877279SMichael Große     *
33829877279SMichael Große     * @param               $mode
33929877279SMichael Große     * @param Doku_Renderer $renderer
34029877279SMichael Große     * @param               $data
34129877279SMichael Große     * @param               $sums
34229877279SMichael Große     */
34329877279SMichael Große    private function summarize($mode, \Doku_Renderer $renderer, $data, $sums) {
34429877279SMichael Große        if($data['summarize']) {
34529877279SMichael Große            $renderer->tablerow_open();
34629877279SMichael Große            $len = count($data['cols']);
34729877279SMichael Große
34829877279SMichael Große            if($data['rownumbers']) {
34929877279SMichael Große                $renderer->tablecell_open();
35029877279SMichael Große                $renderer->tablecell_close();
35129877279SMichael Große            }
35229877279SMichael Große
35329877279SMichael Große            for($i = 0; $i < $len; $i++) {
35429877279SMichael Große                $renderer->tablecell_open(1, $data['align'][$i]);
35529877279SMichael Große                if(!empty($sums[$i])) {
35629877279SMichael Große                    $renderer->cdata('∑ ' . $sums[$i]);
35729877279SMichael Große                } else {
35829877279SMichael Große                    if ($mode == 'xhtml') {
35929877279SMichael Große                        $renderer->doc .= '&nbsp;';
36029877279SMichael Große                    }
36129877279SMichael Große                }
36229877279SMichael Große                $renderer->tablecell_close();
36329877279SMichael Große            }
36429877279SMichael Große            $renderer->tablerow_close();
36529877279SMichael Große        }
36629877279SMichael Große    }
36729877279SMichael Große
36829877279SMichael Große    /**
36929877279SMichael Große     * @param               $mode
37029877279SMichael Große     * @param Doku_Renderer $renderer
37129877279SMichael Große     *
37229877279SMichael Große     */
37329877279SMichael Große    private function finishTableAndScope($mode, Doku_Renderer $renderer) {
37429877279SMichael Große        $renderer->table_close();
37529877279SMichael Große        if ($mode == 'xhmtl') {
37629877279SMichael Große            $renderer->doc .= '</div>';
37729877279SMichael Große        }
37829877279SMichael Große    }
37929877279SMichael Große
38029877279SMichael Große    /**
38129877279SMichael Große     * @param               $mode
38229877279SMichael Große     * @param Doku_Renderer $renderer
38329877279SMichael Große     * @param               $data
38429877279SMichael Große     * @param               $rows
38529877279SMichael Große     *
38629877279SMichael Große     */
38729877279SMichael Große    private function renderRows($mode, Doku_Renderer $renderer, $data, $rows) {
38829877279SMichael Große        $renderer->tabletbody_open();
38929877279SMichael Große        foreach($rows as $rownum => $row) {
39029877279SMichael Große            $renderer->tablerow_open();
39129877279SMichael Große
39229877279SMichael Große            if($data['rownumbers']) {
39329877279SMichael Große                $renderer->tablecell_open();
39429877279SMichael Große                $renderer->doc .= $rownum + 1;
39529877279SMichael Große                $renderer->tablecell_close();
39629877279SMichael Große            }
39729877279SMichael Große
39829877279SMichael Große            /** @var plugin\struct\meta\Value $value */
39929877279SMichael Große            foreach($row as $colnum => $value) {
40029877279SMichael Große                $renderer->tablecell_open();
40129877279SMichael Große                $value->render($renderer, $mode);
40229877279SMichael Große                $renderer->tablecell_close();
40329877279SMichael Große
40429877279SMichael Große                // summarize
40529877279SMichael Große                if($data['summarize'] && is_numeric($value->getValue())) {
40629877279SMichael Große                    if(!isset($this->sums[$colnum])) {
40729877279SMichael Große                        $this->sums[$colnum] = 0;
40829877279SMichael Große                    }
40929877279SMichael Große                    $this->sums[$colnum] += $value->getValue();
41029877279SMichael Große                }
41129877279SMichael Große            }
41229877279SMichael Große            $renderer->tablerow_close();
41329877279SMichael Große        }
41429877279SMichael Große        $renderer->tabletbody_close();
41529877279SMichael Große    }
416549a0837SAndreas Gohr}
417549a0837SAndreas Gohr
418549a0837SAndreas Gohr// vim:ts=4:sw=4:et:
419