*/ // must be run within Dokuwiki use plugin\struct\meta\ConfigParser; use plugin\struct\meta\Search; use plugin\struct\meta\SearchConfig; use plugin\struct\meta\SearchException; use plugin\struct\meta\StructException; if (!defined('DOKU_INC')) die(); class syntax_plugin_struct_table extends DokuWiki_Syntax_Plugin { /** * @return string Syntax mode type */ public function getType() { return 'substition'; } /** * @return string Paragraph type */ public function getPType() { return 'block'; } /** * @return int Sort order - Low numbers go before high numbers */ public function getSort() { return 155; } /** * Connect lookup pattern to lexer. * * @param string $mode Parser mode */ public function connectTo($mode) { $this->Lexer->addSpecialPattern('----+ *struct table *-+\n.*?\n----+', $mode, 'plugin_struct_table'); } /** * Handle matches of the struct syntax * * @param string $match The match of the syntax * @param int $state The state of the handler * @param int $pos The position in the document * @param Doku_Handler $handler The handler * @return array Data for the renderer */ public function handle($match, $state, $pos, Doku_Handler $handler){ $lines = explode("\n", $match); array_shift($lines); array_pop($lines); try { $parser = new ConfigParser($lines); return $parser->getConfig(); } catch (StructException $e) { msg($e->getMessage(), -1, $e->getLine(), $e->getFile()); return null; } } protected $sums = array(); /** @var helper_plugin_struct_aggregation $dthlp */ protected $dthlp = null; /** * Render xhtml output or metadata * * @param string $mode Renderer mode (supported modes: xhtml) * @param Doku_Renderer $renderer The renderer * @param array $data The data from the handler() function * @return bool If rendering was successful. */ public function render($mode, Doku_Renderer $renderer, $data) { if($mode != 'xhtml') return false; if(!$data) return false; $this->dthlp = $this->loadHelper('struct_aggregation'); $clist = $data['cols']; //reset counters $this->sums = array(); /** @var \helper_plugin_struct_config $confHlp */ $confHlp = plugin_load('helper','struct_config'); try { global $INPUT; $datasrt = $INPUT->str('datasrt'); if ($datasrt) { $data['sort'] = $confHlp->parseSort($datasrt); } $dataflt = $INPUT->arr('dataflt'); if ($dataflt) { foreach ($dataflt as $colcomp => $filter) { $data['filter'][] = $confHlp->parseFilterLine('AND', $colcomp . $filter); } } $search = new SearchConfig($data); $rows = $search->execute(); $cnt = count($rows); if ($cnt === 0) { //$this->nullList($data, $clist, $R); //return true; } $dataofs = $INPUT->has('dataofs') ? $INPUT->int('dataofs') : 0; if ($data['limit'] && $cnt > $data['limit']) { $rows = array_slice($rows, $dataofs, $data['limit']); } $this->renderPreTable($mode, $renderer, $clist, $data); $this->renderRows($mode, $renderer, $data, $rows); $this->renderPostTable($mode, $renderer, $data, $cnt); $renderer->doc .= ''; } catch (StructException $e) { msg($e->getMessage(), -1, $e->getLine(), $e->getFile()); } return true; } /** * create the pretext to the actual table rows * * @param $mode * @param Doku_Renderer $renderer * @param $clist * @param $data */ protected function renderPreTable($mode, Doku_Renderer $renderer, $clist, $data) { // Save current request params to not loose them $cur_params = $this->dthlp->_get_current_param(); $this->startScope($mode, $renderer); $this->showActiveFilters($mode, $renderer, $cur_params); $this->startTable($mode, $renderer); $renderer->tablethead_open(); $this->buildColumnHeaders($mode, $renderer, $clist, $data, $cur_params); $this->addDynamicFilters($mode, $renderer, $data, $cur_params); $renderer->tablethead_close(); } /** * @param array $data * @param int $rowcnt * * @return string */ private function renderPostTable($mode, Doku_Renderer $renderer, $data, $rowcnt) { $this->summarize($mode, $renderer, $data, $this->sums); $this->addLimitControls($mode, $renderer, $data, $rowcnt); $this->finishTableAndScope($mode, $renderer); } /** * if limit was set, add control * * @param $mode * @param Doku_Renderer $renderer * @param $data * @param $rowcnt */ protected function addLimitControls($mode, Doku_Renderer $renderer, $data, $rowcnt) { global $ID; if($data['limit']) { $renderer->tablerow_open(); $renderer->tableheader_open((count($data['cols']) + ($data['rownumbers'] ? 1 : 0))); $offset = (int) $_REQUEST['dataofs']; if($offset) { $prev = $offset - $data['limit']; if($prev < 0) { $prev = 0; } // keep url params $params = $this->dthlp->_a2ua('dataflt', $_REQUEST['dataflt']); if(isset($_REQUEST['datasrt'])) { $params['datasrt'] = $_REQUEST['datasrt']; } $params['dataofs'] = $prev; if ($mode == 'xhtml') { $renderer->doc .= '' . $this->getLang('prev') . ' '; } else { $renderer->internallink($ID,$this->getLang('prev')); } } if($rowcnt > $offset + $data['limit']) { $next = $offset + $data['limit']; // keep url params $params = $this->dthlp->_a2ua('dataflt', $_REQUEST['dataflt']); if(isset($_REQUEST['datasrt'])) { $params['datasrt'] = $_REQUEST['datasrt']; } $params['dataofs'] = $next; if ($mode == 'xhtml') { $renderer->doc .= '' . $this->getLang('next') . ''; } else { $renderer->internallink($ID,$this->getLang('next')); } } $renderer->tableheader_close(); $renderer->tablerow_close(); } } /** * @param $mode * @param Doku_Renderer $renderer * @param $cur_params */ protected function showActiveFilters($mode, Doku_Renderer $renderer, $cur_params) { global $ID, $INPUT; if($mode == 'xhtml' && $INPUT->has('dataflt')) { $filters = $INPUT->arr('dataflt'); $confHelper = $this->loadHelper('struct_config'); $fltrs = array(); foreach($filters as $colcomp => $filter) { $filter = $confHelper->parseFilterLine('', $colcomp.$filter); if(strpos($filter[1], '~') !== false) { if(strpos($filter[1], '!~') !== false) { $comparator_value = '!~' . str_replace('%', '*', $filter[2]); } else { $comparator_value = '~' . str_replace('%', '', $filter[2]); } $fltrs[] = $filter[0] . $comparator_value; } else { $fltrs[] = $filter[0] . $filter[1] . $filter[2]; } } $renderer->doc .= '