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 9529877279SMichael Große /** @var \helper_plugin_struct_config $confHlp */ 9629877279SMichael Große $confHlp = plugin_load('helper','struct_config'); 97549a0837SAndreas Gohr 9815929be2SAndreas Gohr try { 9929877279SMichael Große global $INPUT; 10029877279SMichael Große 10129877279SMichael Große $datasrt = $INPUT->str('datasrt'); 10229877279SMichael Große if ($datasrt) { 10329877279SMichael Große $data['sort'] = $confHlp->parseSort($datasrt); 10429877279SMichael Große } 10529877279SMichael Große $dataflt = $INPUT->arr('dataflt'); 10629877279SMichael Große if ($dataflt) { 10729877279SMichael Große foreach ($dataflt as $colcomp => $filter) { 10829877279SMichael Große $data['filter'][] = $confHlp->parseFilterLine('AND', $colcomp . $filter); 10929877279SMichael Große } 11029877279SMichael Große } 1115511bd5bSAndreas Gohr $search = new SearchConfig($data); 11229877279SMichael Große $rows = $search->execute(); 11329877279SMichael Große $cnt = count($rows); 1145511bd5bSAndreas Gohr 11529877279SMichael Große if ($cnt === 0) { 11629877279SMichael Große //$this->nullList($data, $clist, $R); 11729877279SMichael Große //return true; 11829877279SMichael Große } 1195511bd5bSAndreas Gohr 12093485d71SMichael Große $dataofs = $INPUT->has('dataofs') ? $INPUT->int('dataofs') : 0; 12129877279SMichael Große if ($data['limit'] && $cnt > $data['limit']) { 12293485d71SMichael Große $rows = array_slice($rows, $dataofs, $data['limit']); 12329877279SMichael Große } 12415929be2SAndreas Gohr 12529877279SMichael Große $this->renderPreTable($mode, $renderer, $clist, $data); 12629877279SMichael Große $this->renderRows($mode, $renderer, $data, $rows); 12729877279SMichael Große $this->renderPostTable($mode, $renderer, $data, $cnt); 12829877279SMichael Große 12929877279SMichael Große $renderer->doc .= ''; 1305511bd5bSAndreas Gohr } catch (StructException $e) { 13115929be2SAndreas Gohr msg($e->getMessage(), -1, $e->getLine(), $e->getFile()); 13215929be2SAndreas Gohr } 13315929be2SAndreas Gohr 134549a0837SAndreas Gohr return true; 135549a0837SAndreas Gohr } 13629877279SMichael Große 13729877279SMichael Große /** 13829877279SMichael Große * create the pretext to the actual table rows 13929877279SMichael Große * 14029877279SMichael Große * @param $mode 14129877279SMichael Große * @param Doku_Renderer $renderer 14229877279SMichael Große * @param $clist 14329877279SMichael Große * @param $data 14429877279SMichael Große */ 14529877279SMichael Große protected function renderPreTable($mode, Doku_Renderer $renderer, $clist, $data) { 14629877279SMichael Große // Save current request params to not loose them 14729877279SMichael Große $cur_params = $this->dthlp->_get_current_param(); 14829877279SMichael Große 14929877279SMichael Große $this->startScope($mode, $renderer); 150*fc8e2563SMichael Große $this->showActiveFilters($mode, $renderer); 15129877279SMichael Große $this->startTable($mode, $renderer); 15229877279SMichael Große $renderer->tablethead_open(); 15329877279SMichael Große $this->buildColumnHeaders($mode, $renderer, $clist, $data, $cur_params); 15429877279SMichael Große $this->addDynamicFilters($mode, $renderer, $data, $cur_params); 15529877279SMichael Große $renderer->tablethead_close(); 15629877279SMichael Große } 15729877279SMichael Große 15829877279SMichael Große /** 15929877279SMichael Große * @param array $data 16029877279SMichael Große * @param int $rowcnt 16129877279SMichael Große * 16229877279SMichael Große * @return string 16329877279SMichael Große */ 16429877279SMichael Große private function renderPostTable($mode, Doku_Renderer $renderer, $data, $rowcnt) { 16529877279SMichael Große $this->summarize($mode, $renderer, $data, $this->sums); 16629877279SMichael Große $this->addLimitControls($mode, $renderer, $data, $rowcnt); 16729877279SMichael Große $this->finishTableAndScope($mode, $renderer); 16829877279SMichael Große } 16929877279SMichael Große 17029877279SMichael Große /** 17129877279SMichael Große * if limit was set, add control 17229877279SMichael Große * 17393485d71SMichael Große * @param $mode 17493485d71SMichael Große * @param Doku_Renderer $renderer 17529877279SMichael Große * @param $data 17629877279SMichael Große * @param $rowcnt 17729877279SMichael Große */ 17829877279SMichael Große protected function addLimitControls($mode, Doku_Renderer $renderer, $data, $rowcnt) { 179*fc8e2563SMichael Große global $ID, $INPUT; 18029877279SMichael Große 18129877279SMichael Große if($data['limit']) { 18229877279SMichael Große $renderer->tablerow_open(); 18329877279SMichael Große $renderer->tableheader_open((count($data['cols']) + ($data['rownumbers'] ? 1 : 0))); 18429877279SMichael Große $offset = (int) $_REQUEST['dataofs']; 185*fc8e2563SMichael Große 186*fc8e2563SMichael Große // keep url params 187*fc8e2563SMichael Große $params = array(); 188*fc8e2563SMichael Große $params['dataflt'] = $INPUT->arr('dataflt'); 189*fc8e2563SMichael Große if ($INPUT->has('datasrt')) {$params['datasrt'] = $INPUT->str('datasrt');} 190*fc8e2563SMichael Große 19129877279SMichael Große if($offset) { 19229877279SMichael Große $prev = $offset - $data['limit']; 19329877279SMichael Große if($prev < 0) { 19429877279SMichael Große $prev = 0; 19529877279SMichael Große } 19629877279SMichael Große $params['dataofs'] = $prev; 197*fc8e2563SMichael Große $renderer->internallink($ID . '?' . http_build_query($params), $this->getLang('prev')); 19829877279SMichael Große } 19929877279SMichael Große 20093485d71SMichael Große if($rowcnt > $offset + $data['limit']) { 20129877279SMichael Große $next = $offset + $data['limit']; 20229877279SMichael Große $params['dataofs'] = $next; 203*fc8e2563SMichael Große $renderer->internallink($ID . '?' . http_build_query($params), $this->getLang('next')); 20429877279SMichael Große } 20529877279SMichael Große $renderer->tableheader_close(); 20629877279SMichael Große $renderer->tablerow_close(); 20729877279SMichael Große } 20829877279SMichael Große } 20929877279SMichael Große 21029877279SMichael Große /** 21129877279SMichael Große * @param $mode 21229877279SMichael Große * @param Doku_Renderer $renderer 21329877279SMichael Große */ 214*fc8e2563SMichael Große protected function showActiveFilters($mode, Doku_Renderer $renderer) { 21529877279SMichael Große global $ID, $INPUT; 21629877279SMichael Große 21729877279SMichael Große if($mode == 'xhtml' && $INPUT->has('dataflt')) { 21829877279SMichael Große $filters = $INPUT->arr('dataflt'); 21929877279SMichael Große $confHelper = $this->loadHelper('struct_config'); 22029877279SMichael Große $fltrs = array(); 22129877279SMichael Große foreach($filters as $colcomp => $filter) { 22229877279SMichael Große $filter = $confHelper->parseFilterLine('', $colcomp.$filter); 22329877279SMichael Große if(strpos($filter[1], '~') !== false) { 22429877279SMichael Große if(strpos($filter[1], '!~') !== false) { 22529877279SMichael Große $comparator_value = '!~' . str_replace('%', '*', $filter[2]); 22629877279SMichael Große } else { 22729877279SMichael Große $comparator_value = '~' . str_replace('%', '', $filter[2]); 22829877279SMichael Große } 22929877279SMichael Große $fltrs[] = $filter[0] . $comparator_value; 23029877279SMichael Große } else { 23129877279SMichael Große $fltrs[] = $filter[0] . $filter[1] . $filter[2]; 23229877279SMichael Große } 23329877279SMichael Große } 23429877279SMichael Große 23529877279SMichael Große $renderer->doc .= '<div class="filter">'; 23629877279SMichael Große $renderer->doc .= '<h4>' . sprintf($this->getLang('tablefilteredby'), hsc(implode(' & ', $fltrs))) . '</h4>'; 23729877279SMichael Große $renderer->doc .= '<div class="resetfilter">'; 23829877279SMichael Große $renderer->internallink($ID, $this->getLang('tableresetfilter')); 23929877279SMichael Große $renderer->doc .= '</div>'; 24029877279SMichael Große $renderer->doc .= '</div>'; 24129877279SMichael Große } 24229877279SMichael Große } 24329877279SMichael Große 24429877279SMichael Große /** 24529877279SMichael Große * @param $mode 24629877279SMichael Große * @param Doku_Renderer $renderer 24729877279SMichael Große * @param $data 24829877279SMichael Große * @param $cur_params 24929877279SMichael Große */ 25029877279SMichael Große protected function addDynamicFilters($mode, Doku_Renderer $renderer, $data, $cur_params) { 25129877279SMichael Große if ($mode != 'xhtml') return; 25229877279SMichael Große 25329877279SMichael Große global $conf, $ID; 25429877279SMichael Große 25529877279SMichael Große $html = ''; 25629877279SMichael Große if($data['dynfilters']) { 25729877279SMichael Große $html .= '<tr class="dataflt">'; 25829877279SMichael Große 25929877279SMichael Große if($data['rownumbers']) { 26029877279SMichael Große $html .= '<th></th>'; 26129877279SMichael Große } 26229877279SMichael Große 26329877279SMichael Große foreach($data['headers'] as $num => $head) { 26429877279SMichael Große $html .= '<th>'; 26529877279SMichael Große $form = new Doku_Form(array('method' => 'GET',)); 26629877279SMichael Große $form->_hidden = array(); 26729877279SMichael Große if(!$conf['userewrite']) { 26829877279SMichael Große $form->addHidden('id', $ID); 26929877279SMichael Große } 27029877279SMichael Große 271*fc8e2563SMichael Große $key = $data['cols'][$num] . '*~'; 272*fc8e2563SMichael Große $val = isset($cur_params['dataflt'][$key]) ? $cur_params['dataflt'][$key] : ''; 27329877279SMichael Große 27429877279SMichael Große // Add current request params 275*fc8e2563SMichael Große if (!empty($cur_params['datasrt'])) { 276*fc8e2563SMichael Große $form->addHidden('datasrt', $cur_params['datasrt']); 277*fc8e2563SMichael Große } 278*fc8e2563SMichael Große if (!empty($cur_params['dataofs'])) { 279*fc8e2563SMichael Große $form->addHidden('dataofs', $cur_params['dataofs']); 280*fc8e2563SMichael Große } 281*fc8e2563SMichael Große foreach($cur_params['dataflt'] as $c_key => $c_val) { 28229877279SMichael Große if($c_val !== '' && $c_key !== $key) { 283*fc8e2563SMichael Große $form->addHidden('dataflt[' . $c_key . ']', $c_val); 28429877279SMichael Große } 28529877279SMichael Große } 28629877279SMichael Große 287*fc8e2563SMichael Große $form->addElement(form_makeField('text', 'dataflt[' . $key . ']', $val, '')); 28829877279SMichael Große $html .= $form->getForm(); 28929877279SMichael Große $html .= '</th>'; 29029877279SMichael Große } 29129877279SMichael Große $html .= '</tr>'; 29229877279SMichael Große $renderer->doc .= $html; 29329877279SMichael Große } 29429877279SMichael Große } 29529877279SMichael Große 29629877279SMichael Große /** 29729877279SMichael Große * @param $mode 29829877279SMichael Große * @param Doku_Renderer $renderer 29929877279SMichael Große */ 30029877279SMichael Große private function startTable($mode, Doku_Renderer $renderer) { 30129877279SMichael Große $renderer->table_open(); 30229877279SMichael Große } 30329877279SMichael Große 30429877279SMichael Große /** 30529877279SMichael Große * @param $mode 30629877279SMichael Große * @param Doku_Renderer $renderer 30729877279SMichael Große * @param $clist 30829877279SMichael Große * @param $data 30929877279SMichael Große * @param $cur_params 31029877279SMichael Große * 31129877279SMichael Große */ 31229877279SMichael Große protected function buildColumnHeaders($mode, Doku_Renderer $renderer, $clist, $data, $cur_params) { 31329877279SMichael Große global $ID; 31429877279SMichael Große 31529877279SMichael Große $renderer->tablerow_open(); 31629877279SMichael Große 31729877279SMichael Große if($data['rownumbers']) { 31829877279SMichael Große $renderer->tableheader_open(); 31929877279SMichael Große $renderer->cdata('#'); 32029877279SMichael Große $renderer->tableheader_close(); 32129877279SMichael Große } 32229877279SMichael Große 32329877279SMichael Große foreach($data['headers'] as $num => $head) { 32429877279SMichael Große $ckey = $clist[$num]; 32529877279SMichael Große 32629877279SMichael Große $width = ''; 32729877279SMichael Große if(isset($data['widths'][$num]) AND $data['widths'][$num] != '-') { 32829877279SMichael Große $width = ' style="width: ' . $data['widths'][$num] . ';"'; 32929877279SMichael Große } 33029877279SMichael Große if ($mode == 'xhmtl') { 33129877279SMichael Große $renderer->doc .= '<th' . $width . '>'; 33229877279SMichael Große } else { 33329877279SMichael Große $renderer->tableheader_open(); 33429877279SMichael Große } 33529877279SMichael Große 33629877279SMichael Große // add sort arrow 33729877279SMichael Große if ($mode == 'xhtml') { 33829877279SMichael Große if(isset($data['sort']) && $ckey == $data['sort'][0]) { 33929877279SMichael Große if($data['sort'][1] == 'ASC') { 34029877279SMichael Große $renderer->doc .= '<span>↓</span> '; 34129877279SMichael Große $ckey = '^' . $ckey; 34229877279SMichael Große } else { 34329877279SMichael Große $renderer->doc .= '<span>↑</span> '; 34429877279SMichael Große } 34529877279SMichael Große } 34629877279SMichael Große } 347*fc8e2563SMichael Große $renderer->internallink($ID . "?" . http_build_query(array('datasrt' => $ckey,) + $cur_params), hsc($head)); 34829877279SMichael Große $renderer->tableheader_close(); 34929877279SMichael Große } 35029877279SMichael Große $renderer->tablerow_close(); 35129877279SMichael Große } 35229877279SMichael Große 35329877279SMichael Große 35429877279SMichael Große protected function startScope($mode, \Doku_Renderer $renderer) { 35529877279SMichael Große if ($mode == 'xhtml') { 35629877279SMichael Große $renderer->doc .= '<div class="table structaggegation">'; 35729877279SMichael Große } 35829877279SMichael Große } 35929877279SMichael Große 36029877279SMichael Große /** 36129877279SMichael Große * if summarize was set, add sums 36229877279SMichael Große * 36329877279SMichael Große * @param $mode 36429877279SMichael Große * @param Doku_Renderer $renderer 36529877279SMichael Große * @param $data 36629877279SMichael Große * @param $sums 36729877279SMichael Große */ 36829877279SMichael Große private function summarize($mode, \Doku_Renderer $renderer, $data, $sums) { 36929877279SMichael Große if($data['summarize']) { 37029877279SMichael Große $renderer->tablerow_open(); 37129877279SMichael Große $len = count($data['cols']); 37229877279SMichael Große 37329877279SMichael Große if($data['rownumbers']) { 37429877279SMichael Große $renderer->tablecell_open(); 37529877279SMichael Große $renderer->tablecell_close(); 37629877279SMichael Große } 37729877279SMichael Große 37829877279SMichael Große for($i = 0; $i < $len; $i++) { 37929877279SMichael Große $renderer->tablecell_open(1, $data['align'][$i]); 38029877279SMichael Große if(!empty($sums[$i])) { 38129877279SMichael Große $renderer->cdata('∑ ' . $sums[$i]); 38229877279SMichael Große } else { 38329877279SMichael Große if ($mode == 'xhtml') { 38429877279SMichael Große $renderer->doc .= ' '; 38529877279SMichael Große } 38629877279SMichael Große } 38729877279SMichael Große $renderer->tablecell_close(); 38829877279SMichael Große } 38929877279SMichael Große $renderer->tablerow_close(); 39029877279SMichael Große } 39129877279SMichael Große } 39229877279SMichael Große 39329877279SMichael Große /** 39429877279SMichael Große * @param $mode 39529877279SMichael Große * @param Doku_Renderer $renderer 39629877279SMichael Große * 39729877279SMichael Große */ 39829877279SMichael Große private function finishTableAndScope($mode, Doku_Renderer $renderer) { 39929877279SMichael Große $renderer->table_close(); 40029877279SMichael Große if ($mode == 'xhmtl') { 40129877279SMichael Große $renderer->doc .= '</div>'; 40229877279SMichael Große } 40329877279SMichael Große } 40429877279SMichael Große 40529877279SMichael Große /** 40629877279SMichael Große * @param $mode 40729877279SMichael Große * @param Doku_Renderer $renderer 40829877279SMichael Große * @param $data 40929877279SMichael Große * @param $rows 41029877279SMichael Große * 41129877279SMichael Große */ 41229877279SMichael Große private function renderRows($mode, Doku_Renderer $renderer, $data, $rows) { 41329877279SMichael Große $renderer->tabletbody_open(); 41429877279SMichael Große foreach($rows as $rownum => $row) { 41529877279SMichael Große $renderer->tablerow_open(); 41629877279SMichael Große 41729877279SMichael Große if($data['rownumbers']) { 41829877279SMichael Große $renderer->tablecell_open(); 41929877279SMichael Große $renderer->doc .= $rownum + 1; 42029877279SMichael Große $renderer->tablecell_close(); 42129877279SMichael Große } 42229877279SMichael Große 42329877279SMichael Große /** @var plugin\struct\meta\Value $value */ 42429877279SMichael Große foreach($row as $colnum => $value) { 42529877279SMichael Große $renderer->tablecell_open(); 42629877279SMichael Große $value->render($renderer, $mode); 42729877279SMichael Große $renderer->tablecell_close(); 42829877279SMichael Große 42929877279SMichael Große // summarize 43029877279SMichael Große if($data['summarize'] && is_numeric($value->getValue())) { 43129877279SMichael Große if(!isset($this->sums[$colnum])) { 43229877279SMichael Große $this->sums[$colnum] = 0; 43329877279SMichael Große } 43429877279SMichael Große $this->sums[$colnum] += $value->getValue(); 43529877279SMichael Große } 43629877279SMichael Große } 43729877279SMichael Große $renderer->tablerow_close(); 43829877279SMichael Große } 43929877279SMichael Große $renderer->tabletbody_close(); 44029877279SMichael Große } 441549a0837SAndreas Gohr} 442549a0837SAndreas Gohr 443549a0837SAndreas Gohr// vim:ts=4:sw=4:et: 444