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 74549a0837SAndreas Gohr /** 75549a0837SAndreas Gohr * Render xhtml output or metadata 76549a0837SAndreas Gohr * 77549a0837SAndreas Gohr * @param string $mode Renderer mode (supported modes: xhtml) 78549a0837SAndreas Gohr * @param Doku_Renderer $renderer The renderer 79549a0837SAndreas Gohr * @param array $data The data from the handler() function 80549a0837SAndreas Gohr * @return bool If rendering was successful. 81549a0837SAndreas Gohr */ 82ab466032SAndreas Gohr public function render($mode, Doku_Renderer $renderer, $data) { 83549a0837SAndreas Gohr if($mode != 'xhtml') return false; 845511bd5bSAndreas Gohr if(!$data) return false; 8529877279SMichael Große 8629877279SMichael Große //reset counters 8729877279SMichael Große $this->sums = array(); 8829877279SMichael Große 8915929be2SAndreas Gohr try { 905511bd5bSAndreas Gohr $search = new SearchConfig($data); 911a07b696SMichael Große $data = $search->getConf(); 9229877279SMichael Große $rows = $search->execute(); 93f87e1c10SMichael Große $cnt = $search->getCount(); 945511bd5bSAndreas Gohr 9529877279SMichael Große if ($cnt === 0) { 96f87e1c10SMichael Große $this->nullList($data, $mode, $renderer); 97f87e1c10SMichael Große return true; 9829877279SMichael Große } 995511bd5bSAndreas Gohr 100f87e1c10SMichael Große $this->renderPreTable($mode, $renderer, $data); 10129877279SMichael Große $this->renderRows($mode, $renderer, $data, $rows); 10229877279SMichael Große $this->renderPostTable($mode, $renderer, $data, $cnt); 1035511bd5bSAndreas Gohr } catch (StructException $e) { 10415929be2SAndreas Gohr msg($e->getMessage(), -1, $e->getLine(), $e->getFile()); 10515929be2SAndreas Gohr } 10615929be2SAndreas Gohr 107549a0837SAndreas Gohr return true; 108549a0837SAndreas Gohr } 10929877279SMichael Große 11029877279SMichael Große /** 11129877279SMichael Große * create the pretext to the actual table rows 11229877279SMichael Große * 11329877279SMichael Große * @param $mode 11429877279SMichael Große * @param Doku_Renderer $renderer 11529877279SMichael Große * @param $data 11629877279SMichael Große */ 117f87e1c10SMichael Große protected function renderPreTable($mode, Doku_Renderer $renderer, $data) { 118eed15625SMichael Große $this->startScope($mode, $renderer, md5(serialize($data))); 119fc8e2563SMichael Große $this->showActiveFilters($mode, $renderer); 12029877279SMichael Große $this->startTable($mode, $renderer); 12129877279SMichael Große $renderer->tablethead_open(); 122f87e1c10SMichael Große $this->buildColumnHeaders($mode, $renderer, $data); 1231a07b696SMichael Große $this->addDynamicFilters($mode, $renderer, $data); 12429877279SMichael Große $renderer->tablethead_close(); 12529877279SMichael Große } 12629877279SMichael Große 12729877279SMichael Große /** 12829877279SMichael Große * @param array $data 12929877279SMichael Große * @param int $rowcnt 13029877279SMichael Große * 13129877279SMichael Große * @return string 13229877279SMichael Große */ 13329877279SMichael Große private function renderPostTable($mode, Doku_Renderer $renderer, $data, $rowcnt) { 13429877279SMichael Große $this->summarize($mode, $renderer, $data, $this->sums); 13529877279SMichael Große $this->addLimitControls($mode, $renderer, $data, $rowcnt); 13629877279SMichael Große $this->finishTableAndScope($mode, $renderer); 13729877279SMichael Große } 13829877279SMichael Große 13929877279SMichael Große /** 14029877279SMichael Große * if limit was set, add control 14129877279SMichael Große * 142f87e1c10SMichael Große * @param string $mode the mode of the renderer 143f87e1c10SMichael Große * @param Doku_Renderer $renderer the renderer 144f87e1c10SMichael Große * @param array $data the configuration of the table/search 14529877279SMichael Große * @param $rowcnt 14629877279SMichael Große */ 14729877279SMichael Große protected function addLimitControls($mode, Doku_Renderer $renderer, $data, $rowcnt) { 1481a07b696SMichael Große global $ID; 14929877279SMichael Große 15029877279SMichael Große if($data['limit']) { 15129877279SMichael Große $renderer->tablerow_open(); 15229877279SMichael Große $renderer->tableheader_open((count($data['cols']) + ($data['rownumbers'] ? 1 : 0))); 15329877279SMichael Große $offset = (int) $_REQUEST['dataofs']; 154fc8e2563SMichael Große 155fc8e2563SMichael Große // keep url params 156fc8e2563SMichael Große $params = array(); 1571a07b696SMichael Große if (!empty($data['current_params']['dataflt'])) {$params['dataflt'] = $data['current_params']['dataflt'];} 1581a07b696SMichael Große if (!empty($data['current_params']['datasrt'])) {$params['datasrt'] = $data['current_params']['datasrt'];} 159fc8e2563SMichael Große 16029877279SMichael Große if($offset) { 16129877279SMichael Große $prev = $offset - $data['limit']; 16229877279SMichael Große if($prev < 0) { 16329877279SMichael Große $prev = 0; 16429877279SMichael Große } 16529877279SMichael Große $params['dataofs'] = $prev; 166fc8e2563SMichael Große $renderer->internallink($ID . '?' . http_build_query($params), $this->getLang('prev')); 16729877279SMichael Große } 16829877279SMichael Große 16993485d71SMichael Große if($rowcnt > $offset + $data['limit']) { 17029877279SMichael Große $next = $offset + $data['limit']; 17129877279SMichael Große $params['dataofs'] = $next; 172fc8e2563SMichael Große $renderer->internallink($ID . '?' . http_build_query($params), $this->getLang('next')); 17329877279SMichael Große } 17429877279SMichael Große $renderer->tableheader_close(); 17529877279SMichael Große $renderer->tablerow_close(); 17629877279SMichael Große } 17729877279SMichael Große } 17829877279SMichael Große 17929877279SMichael Große /** 180f87e1c10SMichael Große * @param string $mode the mode of the renderer 181f87e1c10SMichael Große * @param Doku_Renderer $renderer the renderer 18229877279SMichael Große */ 183fc8e2563SMichael Große protected function showActiveFilters($mode, Doku_Renderer $renderer) { 1841a07b696SMichael Große global $ID; 18529877279SMichael Große 1861a07b696SMichael Große if($mode == 'xhtml' && !empty($data['current_params']['dataflt'])) { 1871a07b696SMichael Große $filters = $data['current_params']['dataflt']; 18829877279SMichael Große $confHelper = $this->loadHelper('struct_config'); 18929877279SMichael Große $fltrs = array(); 19029877279SMichael Große foreach($filters as $colcomp => $filter) { 19129877279SMichael Große $filter = $confHelper->parseFilterLine('', $colcomp.$filter); 19229877279SMichael Große if(strpos($filter[1], '~') !== false) { 19329877279SMichael Große if(strpos($filter[1], '!~') !== false) { 19429877279SMichael Große $comparator_value = '!~' . str_replace('%', '*', $filter[2]); 19529877279SMichael Große } else { 19629877279SMichael Große $comparator_value = '~' . str_replace('%', '', $filter[2]); 19729877279SMichael Große } 19829877279SMichael Große $fltrs[] = $filter[0] . $comparator_value; 19929877279SMichael Große } else { 20029877279SMichael Große $fltrs[] = $filter[0] . $filter[1] . $filter[2]; 20129877279SMichael Große } 20229877279SMichael Große } 20329877279SMichael Große 20429877279SMichael Große $renderer->doc .= '<div class="filter">'; 20529877279SMichael Große $renderer->doc .= '<h4>' . sprintf($this->getLang('tablefilteredby'), hsc(implode(' & ', $fltrs))) . '</h4>'; 20629877279SMichael Große $renderer->doc .= '<div class="resetfilter">'; 20729877279SMichael Große $renderer->internallink($ID, $this->getLang('tableresetfilter')); 20829877279SMichael Große $renderer->doc .= '</div>'; 20929877279SMichael Große $renderer->doc .= '</div>'; 21029877279SMichael Große } 21129877279SMichael Große } 21229877279SMichael Große 21329877279SMichael Große /** 214f87e1c10SMichael Große * @param string $mode the mode of the renderer 215f87e1c10SMichael Große * @param Doku_Renderer $renderer the renderer 216f87e1c10SMichael Große * @param array $data the configuration of the table/search 21729877279SMichael Große */ 2181a07b696SMichael Große protected function addDynamicFilters($mode, Doku_Renderer $renderer, $data) { 21929877279SMichael Große if ($mode != 'xhtml') return; 22029877279SMichael Große 22129877279SMichael Große global $conf, $ID; 22229877279SMichael Große 2231a07b696SMichael Große $cur_params = $data['current_params']; 22429877279SMichael Große $html = ''; 22529877279SMichael Große if($data['dynfilters']) { 22629877279SMichael Große $html .= '<tr class="dataflt">'; 22729877279SMichael Große 22829877279SMichael Große if($data['rownumbers']) { 22929877279SMichael Große $html .= '<th></th>'; 23029877279SMichael Große } 23129877279SMichael Große 23229877279SMichael Große foreach($data['headers'] as $num => $head) { 23329877279SMichael Große $html .= '<th>'; 23429877279SMichael Große $form = new Doku_Form(array('method' => 'GET',)); 23529877279SMichael Große $form->_hidden = array(); 23629877279SMichael Große if(!$conf['userewrite']) { 23729877279SMichael Große $form->addHidden('id', $ID); 23829877279SMichael Große } 23929877279SMichael Große 240fc8e2563SMichael Große $key = $data['cols'][$num] . '*~'; 241fc8e2563SMichael Große $val = isset($cur_params['dataflt'][$key]) ? $cur_params['dataflt'][$key] : ''; 24229877279SMichael Große 24329877279SMichael Große // Add current request params 244fc8e2563SMichael Große if (!empty($cur_params['datasrt'])) { 245fc8e2563SMichael Große $form->addHidden('datasrt', $cur_params['datasrt']); 246fc8e2563SMichael Große } 247fc8e2563SMichael Große if (!empty($cur_params['dataofs'])) { 248fc8e2563SMichael Große $form->addHidden('dataofs', $cur_params['dataofs']); 249fc8e2563SMichael Große } 250fc8e2563SMichael Große foreach($cur_params['dataflt'] as $c_key => $c_val) { 25129877279SMichael Große if($c_val !== '' && $c_key !== $key) { 252fc8e2563SMichael Große $form->addHidden('dataflt[' . $c_key . ']', $c_val); 25329877279SMichael Große } 25429877279SMichael Große } 25529877279SMichael Große 256fc8e2563SMichael Große $form->addElement(form_makeField('text', 'dataflt[' . $key . ']', $val, '')); 25729877279SMichael Große $html .= $form->getForm(); 25829877279SMichael Große $html .= '</th>'; 25929877279SMichael Große } 26029877279SMichael Große $html .= '</tr>'; 26129877279SMichael Große $renderer->doc .= $html; 26229877279SMichael Große } 26329877279SMichael Große } 26429877279SMichael Große 26529877279SMichael Große /** 266f87e1c10SMichael Große * @param string $mode the mode of the renderer 267f87e1c10SMichael Große * @param Doku_Renderer $renderer the renderer 26829877279SMichael Große */ 26929877279SMichael Große private function startTable($mode, Doku_Renderer $renderer) { 27029877279SMichael Große $renderer->table_open(); 27129877279SMichael Große } 27229877279SMichael Große 27329877279SMichael Große /** 274f87e1c10SMichael Große * @param string $mode the mode of the renderer 275f87e1c10SMichael Große * @param Doku_Renderer $renderer the renderer 276f87e1c10SMichael Große * @param array $data the configuration of the table/search 27729877279SMichael Große * 27829877279SMichael Große */ 279f87e1c10SMichael Große protected function buildColumnHeaders($mode, Doku_Renderer $renderer, $data) { 28029877279SMichael Große global $ID; 28129877279SMichael Große 28229877279SMichael Große $renderer->tablerow_open(); 28329877279SMichael Große 28429877279SMichael Große if($data['rownumbers']) { 28529877279SMichael Große $renderer->tableheader_open(); 28629877279SMichael Große $renderer->cdata('#'); 28729877279SMichael Große $renderer->tableheader_close(); 28829877279SMichael Große } 28929877279SMichael Große 29029877279SMichael Große foreach($data['headers'] as $num => $head) { 291f87e1c10SMichael Große $ckey = $data['cols'][$num]; 29229877279SMichael Große 29329877279SMichael Große $width = ''; 29429877279SMichael Große if(isset($data['widths'][$num]) AND $data['widths'][$num] != '-') { 29529877279SMichael Große $width = ' style="width: ' . $data['widths'][$num] . ';"'; 29629877279SMichael Große } 29729877279SMichael Große if ($mode == 'xhmtl') { 29829877279SMichael Große $renderer->doc .= '<th' . $width . '>'; 29929877279SMichael Große } else { 30029877279SMichael Große $renderer->tableheader_open(); 30129877279SMichael Große } 30229877279SMichael Große 30329877279SMichael Große // add sort arrow 30429877279SMichael Große if ($mode == 'xhtml') { 30529877279SMichael Große if(isset($data['sort']) && $ckey == $data['sort'][0]) { 30629877279SMichael Große if($data['sort'][1] == 'ASC') { 30729877279SMichael Große $renderer->doc .= '<span>↓</span> '; 30829877279SMichael Große $ckey = '^' . $ckey; 30929877279SMichael Große } else { 31029877279SMichael Große $renderer->doc .= '<span>↑</span> '; 31129877279SMichael Große } 31229877279SMichael Große } 31329877279SMichael Große } 3141a07b696SMichael Große $renderer->internallink($ID . "?" . http_build_query(array('datasrt' => $ckey,) + $data['current_params']), hsc($head)); 31529877279SMichael Große $renderer->tableheader_close(); 31629877279SMichael Große } 31729877279SMichael Große $renderer->tablerow_close(); 31829877279SMichael Große } 31929877279SMichael Große 320f87e1c10SMichael Große /** 321f87e1c10SMichael Große * @param string $mode the mode of the renderer 322f87e1c10SMichael Große * @param Doku_Renderer $renderer the renderer 323eed15625SMichael Große * @param string $hash hash to identify the table and group images in gallery 324f87e1c10SMichael Große */ 325eed15625SMichael Große protected function startScope($mode, \Doku_Renderer $renderer, $hash) { 32629877279SMichael Große if ($mode == 'xhtml') { 327*08e4e96eSMichael Große $renderer->doc .= "<div class=\"table structaggregation\">"; 328*08e4e96eSMichael Große $renderer->info['struct_table_hash'] = $hash; 32929877279SMichael Große } 33029877279SMichael Große } 33129877279SMichael Große 33229877279SMichael Große /** 33329877279SMichael Große * if summarize was set, add sums 33429877279SMichael Große * 335f87e1c10SMichael Große * @param string $mode the mode of the renderer 336f87e1c10SMichael Große * @param Doku_Renderer $renderer the renderer 337f87e1c10SMichael Große * @param array $data the configuration of the table/search 338f87e1c10SMichael Große * @param array $sums the summarized output of the numerical fields 33929877279SMichael Große */ 34029877279SMichael Große private function summarize($mode, \Doku_Renderer $renderer, $data, $sums) { 34129877279SMichael Große if($data['summarize']) { 34229877279SMichael Große $renderer->tablerow_open(); 34329877279SMichael Große $len = count($data['cols']); 34429877279SMichael Große 34529877279SMichael Große if($data['rownumbers']) { 34629877279SMichael Große $renderer->tablecell_open(); 34729877279SMichael Große $renderer->tablecell_close(); 34829877279SMichael Große } 34929877279SMichael Große 35029877279SMichael Große for($i = 0; $i < $len; $i++) { 35129877279SMichael Große $renderer->tablecell_open(1, $data['align'][$i]); 35229877279SMichael Große if(!empty($sums[$i])) { 35329877279SMichael Große $renderer->cdata('∑ ' . $sums[$i]); 35429877279SMichael Große } else { 35529877279SMichael Große if ($mode == 'xhtml') { 35629877279SMichael Große $renderer->doc .= ' '; 35729877279SMichael Große } 35829877279SMichael Große } 35929877279SMichael Große $renderer->tablecell_close(); 36029877279SMichael Große } 36129877279SMichael Große $renderer->tablerow_close(); 36229877279SMichael Große } 36329877279SMichael Große } 36429877279SMichael Große 36529877279SMichael Große /** 366f87e1c10SMichael Große * @param string $mode the mode of the renderer 367f87e1c10SMichael Große * @param Doku_Renderer $renderer the renderer 36829877279SMichael Große * 36929877279SMichael Große */ 37029877279SMichael Große private function finishTableAndScope($mode, Doku_Renderer $renderer) { 37129877279SMichael Große $renderer->table_close(); 37229877279SMichael Große if ($mode == 'xhmtl') { 37329877279SMichael Große $renderer->doc .= '</div>'; 374*08e4e96eSMichael Große unset($renderer->info['struct_table_hash']); 37529877279SMichael Große } 37629877279SMichael Große } 37729877279SMichael Große 37829877279SMichael Große /** 379f87e1c10SMichael Große * @param string $mode the mode of the renderer 380f87e1c10SMichael Große * @param Doku_Renderer $renderer the renderer 381f87e1c10SMichael Große * @param array $data the configuration of the table/search 38229877279SMichael Große * @param $rows 38329877279SMichael Große * 38429877279SMichael Große */ 38529877279SMichael Große private function renderRows($mode, Doku_Renderer $renderer, $data, $rows) { 38629877279SMichael Große $renderer->tabletbody_open(); 38729877279SMichael Große foreach($rows as $rownum => $row) { 38829877279SMichael Große $renderer->tablerow_open(); 38929877279SMichael Große 39029877279SMichael Große if($data['rownumbers']) { 39129877279SMichael Große $renderer->tablecell_open(); 39229877279SMichael Große $renderer->doc .= $rownum + 1; 39329877279SMichael Große $renderer->tablecell_close(); 39429877279SMichael Große } 39529877279SMichael Große 39629877279SMichael Große /** @var plugin\struct\meta\Value $value */ 39729877279SMichael Große foreach($row as $colnum => $value) { 39829877279SMichael Große $renderer->tablecell_open(); 39929877279SMichael Große $value->render($renderer, $mode); 40029877279SMichael Große $renderer->tablecell_close(); 40129877279SMichael Große 40229877279SMichael Große // summarize 40329877279SMichael Große if($data['summarize'] && is_numeric($value->getValue())) { 40429877279SMichael Große if(!isset($this->sums[$colnum])) { 40529877279SMichael Große $this->sums[$colnum] = 0; 40629877279SMichael Große } 40729877279SMichael Große $this->sums[$colnum] += $value->getValue(); 40829877279SMichael Große } 40929877279SMichael Große } 41029877279SMichael Große $renderer->tablerow_close(); 41129877279SMichael Große } 41229877279SMichael Große $renderer->tabletbody_close(); 41329877279SMichael Große } 414f87e1c10SMichael Große 415f87e1c10SMichael Große /** 416f87e1c10SMichael Große * @param array $data the configuration of the table/search 417f87e1c10SMichael Große * @param string $mode the mode of the renderer 418f87e1c10SMichael Große * @param Doku_Renderer $renderer the renderer 419f87e1c10SMichael Große */ 420f87e1c10SMichael Große private function nullList($data, $mode, Doku_Renderer $renderer) { 421f87e1c10SMichael Große $this->renderPreTable($mode, $renderer, $data); 422f87e1c10SMichael Große $renderer->tablerow_open(); 423f87e1c10SMichael Große $renderer->tablecell_open(count($data['cols']) + $data['rownumbers'], 'center'); 424f87e1c10SMichael Große $renderer->cdata($this->getLang('none')); 425f87e1c10SMichael Große $renderer->tablecell_close(); 426f87e1c10SMichael Große $renderer->tablerow_close(); 427f87e1c10SMichael Große $renderer->table_close(); 428f87e1c10SMichael Große } 429549a0837SAndreas Gohr} 430549a0837SAndreas Gohr 431549a0837SAndreas Gohr// vim:ts=4:sw=4:et: 432