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>↓</span> '; 31629877279SMichael Große $ckey = '^' . $ckey; 31729877279SMichael Große } else { 31829877279SMichael Große $renderer->doc .= '<span>↑</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 .= ' '; 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