121fcef82SMichael Große<?php 221fcef82SMichael Große 321fcef82SMichael Großenamespace dokuwiki\Ui; 421fcef82SMichael Große 5427ed988SMichael Großeuse \dokuwiki\Form\Form; 6427ed988SMichael Große 721fcef82SMichael Großeclass Search extends Ui 821fcef82SMichael Große{ 921fcef82SMichael Große protected $query; 1021fcef82SMichael Große protected $pageLookupResults = array(); 1121fcef82SMichael Große protected $fullTextResults = array(); 1221fcef82SMichael Große protected $highlight = array(); 1321fcef82SMichael Große 1421fcef82SMichael Große /** 1521fcef82SMichael Große * Search constructor. 1621fcef82SMichael Große * 1721fcef82SMichael Große * @param string $query the search query 1821fcef82SMichael Große */ 1921fcef82SMichael Große public function __construct($query) 2021fcef82SMichael Große { 2121fcef82SMichael Große $this->query = $query; 2221fcef82SMichael Große } 2321fcef82SMichael Große 2421fcef82SMichael Große /** 2521fcef82SMichael Große * run the search 2621fcef82SMichael Große */ 2721fcef82SMichael Große public function execute() 2821fcef82SMichael Große { 2921fcef82SMichael Große $this->pageLookupResults = ft_pageLookup($this->query, true, useHeading('navigation')); 3021fcef82SMichael Große $this->fullTextResults = ft_pageSearch($this->query, $highlight); 3121fcef82SMichael Große $this->highlight = $highlight; 3221fcef82SMichael Große } 3321fcef82SMichael Große 3421fcef82SMichael Große /** 3521fcef82SMichael Große * display the search result 3621fcef82SMichael Große * 3721fcef82SMichael Große * @return void 3821fcef82SMichael Große */ 3921fcef82SMichael Große public function show() 4021fcef82SMichael Große { 4121fcef82SMichael Große $searchHTML = ''; 4221fcef82SMichael Große 43427ed988SMichael Große $searchHTML .= $this->getSearchFormHTML($this->query); 44427ed988SMichael Große 4521fcef82SMichael Große $searchHTML .= $this->getSearchIntroHTML($this->query); 4621fcef82SMichael Große 4721fcef82SMichael Große $searchHTML .= $this->getPageLookupHTML($this->pageLookupResults); 4821fcef82SMichael Große 4921fcef82SMichael Große $searchHTML .= $this->getFulltextResultsHTML($this->fullTextResults, $this->highlight); 5021fcef82SMichael Große 5121fcef82SMichael Große echo $searchHTML; 5221fcef82SMichael Große } 5321fcef82SMichael Große 5421fcef82SMichael Große /** 55427ed988SMichael Große * Get a form which can be used to adjust/refine the search 56427ed988SMichael Große * 57427ed988SMichael Große * @param string $query 58427ed988SMichael Große * 59427ed988SMichael Große * @return string 60427ed988SMichael Große */ 61427ed988SMichael Große protected function getSearchFormHTML($query) 62427ed988SMichael Große { 63427ed988SMichael Große global $lang; 64427ed988SMichael Große 65bb8ef867SMichael Große $Indexer = idx_get_indexer(); 66bb8ef867SMichael Große $parsedQuery = ft_queryParser($Indexer, $query); 67427ed988SMichael Große 68bb8ef867SMichael Große $searchForm = (new Form())->attrs(['method' => 'get'])->addClass('search-results-form'); 69bb8ef867SMichael Große $searchForm->setHiddenField('do', 'search'); 70bb8ef867SMichael Große $searchForm->addFieldsetOpen()->addClass('search-results-form__fieldset'); 71bb8ef867SMichael Große $searchForm->addTextInput('id')->val($query); 72427ed988SMichael Große $searchForm->addButton('', $lang['btn_search'])->attr('type', 'submit'); 73bb8ef867SMichael Große 74bb8ef867SMichael Große if ($this->isSearchAssistanceAvailable($parsedQuery)) { 75bb8ef867SMichael Große $this->addSearchAssistanceElements($searchForm, $parsedQuery); 76bb8ef867SMichael Große } else { 77bb8ef867SMichael Große $searchForm->addClass('search-results-form--no-assistance'); 78bb8ef867SMichael Große $searchForm->addTagOpen('span')->addClass('search-results-form__no-assistance-message'); 79bb8ef867SMichael Große $searchForm->addHTML('FIXME Your query is too complex. Search assistance is unavailable. See <a href="https://doku.wiki/search">doku.wiki/search</a> for more help.'); 80bb8ef867SMichael Große $searchForm->addTagClose('span'); 81bb8ef867SMichael Große } 82bb8ef867SMichael Große 83427ed988SMichael Große $searchForm->addFieldsetClose(); 84427ed988SMichael Große 85427ed988SMichael Große return $searchForm->toHTML(); 86427ed988SMichael Große } 87427ed988SMichael Große 88427ed988SMichael Große /** 89bb8ef867SMichael Große * Decide if the given query is simple enough to provide search assistance 90bb8ef867SMichael Große * 91bb8ef867SMichael Große * @param array $parsedQuery 92bb8ef867SMichael Große * 93bb8ef867SMichael Große * @return bool 94bb8ef867SMichael Große */ 95bb8ef867SMichael Große protected function isSearchAssistanceAvailable(array $parsedQuery) 96bb8ef867SMichael Große { 97bb8ef867SMichael Große if (count($parsedQuery['words']) > 1) { 98bb8ef867SMichael Große return false; 99bb8ef867SMichael Große } 100bb8ef867SMichael Große if (!empty($parsedQuery['not'])) { 101bb8ef867SMichael Große return false; 102bb8ef867SMichael Große } 103bb8ef867SMichael Große 104bb8ef867SMichael Große if (!empty($parsedQuery['phrases'])) { 105bb8ef867SMichael Große return false; 106bb8ef867SMichael Große } 107bb8ef867SMichael Große 108bb8ef867SMichael Große if (!empty($parsedQuery['notns'])) { 109bb8ef867SMichael Große return false; 110bb8ef867SMichael Große } 111bb8ef867SMichael Große if (count($parsedQuery['ns']) > 1) { 112bb8ef867SMichael Große return false; 113bb8ef867SMichael Große } 114bb8ef867SMichael Große 115bb8ef867SMichael Große return true; 116bb8ef867SMichael Große } 117bb8ef867SMichael Große 118bb8ef867SMichael Große /** 119bb8ef867SMichael Große * Add the elements to be used for search assistance 120bb8ef867SMichael Große * 121bb8ef867SMichael Große * @param Form $searchForm 122bb8ef867SMichael Große * @param array $parsedQuery 123bb8ef867SMichael Große */ 124bb8ef867SMichael Große protected function addSearchAssistanceElements(Form $searchForm, array $parsedQuery) 125bb8ef867SMichael Große { 126bb8ef867SMichael Große $matchType = ''; 127bb8ef867SMichael Große $searchTerm = null; 128bb8ef867SMichael Große if (count($parsedQuery['words']) === 1) { 129bb8ef867SMichael Große $searchTerm = $parsedQuery['words'][0]; 130bb8ef867SMichael Große $firstChar = $searchTerm[0]; 131bb8ef867SMichael Große $lastChar = substr($searchTerm, -1); 132bb8ef867SMichael Große $matchType = 'exact'; 133bb8ef867SMichael Große 134bb8ef867SMichael Große if ($firstChar === '*') { 135bb8ef867SMichael Große $matchType = 'starts'; 136bb8ef867SMichael Große } 137bb8ef867SMichael Große if ($lastChar === '*') { 138bb8ef867SMichael Große $matchType = 'ends'; 139bb8ef867SMichael Große } 140bb8ef867SMichael Große if ($firstChar === '*' && $lastChar === '*') { 141bb8ef867SMichael Große $matchType = 'contains'; 142bb8ef867SMichael Große } 143bb8ef867SMichael Große $searchTerm = trim($searchTerm, '*'); 144bb8ef867SMichael Große } 145bb8ef867SMichael Große 146bb8ef867SMichael Große $searchForm->addTextInput( 147bb8ef867SMichael Große 'searchTerm', 148bb8ef867SMichael Große '', 149bb8ef867SMichael Große $searchForm->findPositionByAttribute('type', 'submit') 150bb8ef867SMichael Große ) 151bb8ef867SMichael Große ->val($searchTerm) 152bb8ef867SMichael Große ->attr('style', 'display: none;'); 153bb8ef867SMichael Große $searchForm->addButton('toggleAssistant', 'toggle search assistant') 154bb8ef867SMichael Große ->attr('type', 'button') 155bb8ef867SMichael Große ->id('search-results-form__show-assistance-button') 156bb8ef867SMichael Große ->addClass('search-results-form__show-assistance-button'); 157bb8ef867SMichael Große 158bb8ef867SMichael Große $searchForm->addTagOpen('div') 159bb8ef867SMichael Große ->addClass('js-advancedSearchOptions') 160bb8ef867SMichael Große ->attr('style', 'display: none;'); 161bb8ef867SMichael Große 162bb8ef867SMichael Große $searchForm->addTagOpen('div')->addClass('search-results-form__subwrapper'); 163bb8ef867SMichael Große $searchForm->addRadioButton('matchType', 'exact Match FM')->val('exact')->attr('checked', 164bb8ef867SMichael Große $matchType === 'exact' ?: null); 165bb8ef867SMichael Große $searchForm->addRadioButton('matchType', 'starts with FM')->val('starts')->attr('checked', 166bb8ef867SMichael Große $matchType === 'starts' ?: null); 167bb8ef867SMichael Große $searchForm->addRadioButton('matchType', 'ends with FM')->val('ends')->attr('checked', 168bb8ef867SMichael Große $matchType === 'ends' ?: null); 169bb8ef867SMichael Große $searchForm->addRadioButton('matchType', 'contains FM')->val('contains')->attr('checked', 170bb8ef867SMichael Große $matchType === 'contains' ?: null); 171bb8ef867SMichael Große $searchForm->addTagClose('div'); 172bb8ef867SMichael Große 173bb8ef867SMichael Große $this->addNamespaceSelector($searchForm, $parsedQuery); 174bb8ef867SMichael Große 175bb8ef867SMichael Große $searchForm->addTagClose('div'); 176bb8ef867SMichael Große } 177bb8ef867SMichael Große 178bb8ef867SMichael Große /** 179bb8ef867SMichael Große * Add the elements for the namespace selector 180bb8ef867SMichael Große * 181bb8ef867SMichael Große * @param Form $searchForm 182bb8ef867SMichael Große * @param array $parsedQuery 183bb8ef867SMichael Große */ 184bb8ef867SMichael Große protected function addNamespaceSelector(Form $searchForm, array $parsedQuery) 185bb8ef867SMichael Große { 186bb8ef867SMichael Große $baseNS = empty($parsedQuery['ns']) ? '' : $parsedQuery['ns'][0]; 187bb8ef867SMichael Große $namespaces = []; 188bb8ef867SMichael Große $searchForm->addTagOpen('div')->addClass('search-results-form__subwrapper'); 189bb8ef867SMichael Große if ($baseNS) { 190bb8ef867SMichael Große $searchForm->addRadioButton('namespace', '(no namespace FIXME)')->val(''); 191bb8ef867SMichael Große $parts = [$baseNS => count($this->fullTextResults)]; 192bb8ef867SMichael Große $upperNameSpace = $baseNS; 193bb8ef867SMichael Große while ($upperNameSpace = getNS($upperNameSpace)) { 194bb8ef867SMichael Große $parts[$upperNameSpace] = 0; 195bb8ef867SMichael Große } 196bb8ef867SMichael Große $namespaces = array_reverse($parts); 197bb8ef867SMichael Große }; 198bb8ef867SMichael Große 199bb8ef867SMichael Große $namespaces = array_merge($namespaces, $this->getAdditionalNamespacesFromResults($baseNS)); 200bb8ef867SMichael Große 201bb8ef867SMichael Große foreach ($namespaces as $extraNS => $count) { 202bb8ef867SMichael Große $label = $extraNS . ($count ? " ($count)" : ''); 203bb8ef867SMichael Große $namespaceCB = $searchForm->addRadioButton('namespace', $label)->val($extraNS); 204bb8ef867SMichael Große if ($extraNS === $baseNS) { 205bb8ef867SMichael Große $namespaceCB->attr('checked', true); 206bb8ef867SMichael Große } 207bb8ef867SMichael Große } 208bb8ef867SMichael Große 209bb8ef867SMichael Große $searchForm->addTagClose('div'); 210bb8ef867SMichael Große } 211bb8ef867SMichael Große 212bb8ef867SMichael Große /** 213bb8ef867SMichael Große * Parse the full text results for their top namespaces below the given base namespace 214bb8ef867SMichael Große * 215bb8ef867SMichael Große * @param string $baseNS the namespace within which was searched, empty string for root namespace 216bb8ef867SMichael Große * 217bb8ef867SMichael Große * @return array an associative array with namespace => #number of found pages, sorted descending 218bb8ef867SMichael Große */ 219bb8ef867SMichael Große protected function getAdditionalNamespacesFromResults($baseNS) 220bb8ef867SMichael Große { 221bb8ef867SMichael Große $namespaces = []; 222bb8ef867SMichael Große $baseNSLength = strlen($baseNS); 223bb8ef867SMichael Große foreach ($this->fullTextResults as $page => $numberOfHits) { 224bb8ef867SMichael Große $namespace = getNS($page); 225bb8ef867SMichael Große if (!$namespace) { 226bb8ef867SMichael Große continue; 227bb8ef867SMichael Große } 228bb8ef867SMichael Große if ($namespace === $baseNS) { 229bb8ef867SMichael Große continue; 230bb8ef867SMichael Große } 231bb8ef867SMichael Große $firstColon = strpos((string)$namespace, ':', $baseNSLength + 1) ?: strlen($namespace); 232bb8ef867SMichael Große $subtopNS = substr($namespace, 0, $firstColon); 233bb8ef867SMichael Große if (empty($namespaces[$subtopNS])) { 234bb8ef867SMichael Große $namespaces[$subtopNS] = 0; 235bb8ef867SMichael Große } 236bb8ef867SMichael Große $namespaces[$subtopNS] += 1; 237bb8ef867SMichael Große } 238bb8ef867SMichael Große arsort($namespaces); 239bb8ef867SMichael Große return $namespaces; 240bb8ef867SMichael Große } 241bb8ef867SMichael Große 242bb8ef867SMichael Große /** 24321fcef82SMichael Große * Build the intro text for the search page 24421fcef82SMichael Große * 24521fcef82SMichael Große * @param string $query the search query 24621fcef82SMichael Große * 24721fcef82SMichael Große * @return string 24821fcef82SMichael Große */ 24921fcef82SMichael Große protected function getSearchIntroHTML($query) 25021fcef82SMichael Große { 25121fcef82SMichael Große global $ID, $lang; 25221fcef82SMichael Große 25321fcef82SMichael Große $intro = p_locale_xhtml('searchpage'); 25421fcef82SMichael Große // allow use of placeholder in search intro 25521fcef82SMichael Große $pagecreateinfo = (auth_quickaclcheck($ID) >= AUTH_CREATE) ? $lang['searchcreatepage'] : ''; 25621fcef82SMichael Große $intro = str_replace( 25721fcef82SMichael Große array('@QUERY@', '@SEARCH@', '@CREATEPAGEINFO@'), 25821fcef82SMichael Große array(hsc(rawurlencode($query)), hsc($query), $pagecreateinfo), 25921fcef82SMichael Große $intro 26021fcef82SMichael Große ); 26121fcef82SMichael Große return $intro; 26221fcef82SMichael Große } 26321fcef82SMichael Große 26421fcef82SMichael Große /** 26521fcef82SMichael Große * Build HTML for a list of pages with matching pagenames 26621fcef82SMichael Große * 26721fcef82SMichael Große * @param array $data search results 26821fcef82SMichael Große * 26921fcef82SMichael Große * @return string 27021fcef82SMichael Große */ 27121fcef82SMichael Große protected function getPageLookupHTML($data) 27221fcef82SMichael Große { 27321fcef82SMichael Große if (empty($data)) { 27421fcef82SMichael Große return ''; 27521fcef82SMichael Große } 27621fcef82SMichael Große 27721fcef82SMichael Große global $lang; 27821fcef82SMichael Große 27921fcef82SMichael Große $html = '<div class="search_quickresult">'; 28021fcef82SMichael Große $html .= '<h3>' . $lang['quickhits'] . ':</h3>'; 28121fcef82SMichael Große $html .= '<ul class="search_quickhits">'; 28221fcef82SMichael Große foreach ($data as $id => $title) { 283*4eab6f7cSMichael Große $link = html_wikilink(':' . $id); 284*4eab6f7cSMichael Große $eventData = [ 285*4eab6f7cSMichael Große 'listItemContent' => [$link], 286*4eab6f7cSMichael Große 'page' => $id, 287*4eab6f7cSMichael Große ]; 288*4eab6f7cSMichael Große trigger_event('SEARCH_RESULT_PAGELOOKUP', $eventData); 289*4eab6f7cSMichael Große $html .= '<li>' . implode('', $eventData['listItemContent']) . '</li>'; 29021fcef82SMichael Große } 29121fcef82SMichael Große $html .= '</ul> '; 29221fcef82SMichael Große //clear float (see http://www.complexspiral.com/publications/containing-floats/) 29321fcef82SMichael Große $html .= '<div class="clearer"></div>'; 29421fcef82SMichael Große $html .= '</div>'; 29521fcef82SMichael Große 29621fcef82SMichael Große return $html; 29721fcef82SMichael Große } 29821fcef82SMichael Große 29921fcef82SMichael Große /** 30021fcef82SMichael Große * Build HTML for fulltext search results or "no results" message 30121fcef82SMichael Große * 30221fcef82SMichael Große * @param array $data the results of the fulltext search 30321fcef82SMichael Große * @param array $highlight the terms to be highlighted in the results 30421fcef82SMichael Große * 30521fcef82SMichael Große * @return string 30621fcef82SMichael Große */ 30721fcef82SMichael Große protected function getFulltextResultsHTML($data, $highlight) 30821fcef82SMichael Große { 30921fcef82SMichael Große global $lang; 31021fcef82SMichael Große 31121fcef82SMichael Große if (empty($data)) { 31221fcef82SMichael Große return '<div class="nothing">' . $lang['nothingfound'] . '</div>'; 31321fcef82SMichael Große } 31421fcef82SMichael Große 31521fcef82SMichael Große $html = ''; 31621fcef82SMichael Große $html .= '<dl class="search_results">'; 31721fcef82SMichael Große $num = 1; 31821fcef82SMichael Große foreach ($data as $id => $cnt) { 319*4eab6f7cSMichael Große $resultLink = html_wikilink(':' . $id, null, $highlight); 320*4eab6f7cSMichael Große $hits = ''; 321*4eab6f7cSMichael Große $snippet = ''; 32221fcef82SMichael Große if ($cnt !== 0) { 323*4eab6f7cSMichael Große $hits = $cnt . ' ' . $lang['hits']; 32421fcef82SMichael Große if ($num < FT_SNIPPET_NUMBER) { // create snippets for the first number of matches only 325*4eab6f7cSMichael Große $snippet = '<dd>' . ft_snippet($id, $highlight) . '</dd>'; 32621fcef82SMichael Große } 32721fcef82SMichael Große $num++; 32821fcef82SMichael Große } 329*4eab6f7cSMichael Große 330*4eab6f7cSMichael Große $eventData = [ 331*4eab6f7cSMichael Große 'resultHeader' => [$resultLink, $hits], 332*4eab6f7cSMichael Große 'resultBody' => [$snippet], 333*4eab6f7cSMichael Große 'page' => $id, 334*4eab6f7cSMichael Große ]; 335*4eab6f7cSMichael Große trigger_event('SEARCH_RESULT_FULLPAGE', $eventData); 336*4eab6f7cSMichael Große $html .= '<div class="search_fullpage_result">'; 337*4eab6f7cSMichael Große $html .= '<dt>' . implode(' ', $eventData['resultHeader']) . '</dt>'; 338*4eab6f7cSMichael Große $html .= implode('', $eventData['resultBody']); 339*4eab6f7cSMichael Große $html .= '</div>'; 34021fcef82SMichael Große } 34121fcef82SMichael Große $html .= '</dl>'; 34221fcef82SMichael Große 34321fcef82SMichael Große return $html; 34421fcef82SMichael Große } 34521fcef82SMichael Große} 346