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 85*81a0edd9SMichael Große trigger_event('SEARCH_FORM_DISPLAY', $searchForm); 86*81a0edd9SMichael Große 87427ed988SMichael Große return $searchForm->toHTML(); 88427ed988SMichael Große } 89427ed988SMichael Große 90427ed988SMichael Große /** 91bb8ef867SMichael Große * Decide if the given query is simple enough to provide search assistance 92bb8ef867SMichael Große * 93bb8ef867SMichael Große * @param array $parsedQuery 94bb8ef867SMichael Große * 95bb8ef867SMichael Große * @return bool 96bb8ef867SMichael Große */ 97bb8ef867SMichael Große protected function isSearchAssistanceAvailable(array $parsedQuery) 98bb8ef867SMichael Große { 99bb8ef867SMichael Große if (count($parsedQuery['words']) > 1) { 100bb8ef867SMichael Große return false; 101bb8ef867SMichael Große } 102bb8ef867SMichael Große if (!empty($parsedQuery['not'])) { 103bb8ef867SMichael Große return false; 104bb8ef867SMichael Große } 105bb8ef867SMichael Große 106bb8ef867SMichael Große if (!empty($parsedQuery['phrases'])) { 107bb8ef867SMichael Große return false; 108bb8ef867SMichael Große } 109bb8ef867SMichael Große 110bb8ef867SMichael Große if (!empty($parsedQuery['notns'])) { 111bb8ef867SMichael Große return false; 112bb8ef867SMichael Große } 113bb8ef867SMichael Große if (count($parsedQuery['ns']) > 1) { 114bb8ef867SMichael Große return false; 115bb8ef867SMichael Große } 116bb8ef867SMichael Große 117bb8ef867SMichael Große return true; 118bb8ef867SMichael Große } 119bb8ef867SMichael Große 120bb8ef867SMichael Große /** 121bb8ef867SMichael Große * Add the elements to be used for search assistance 122bb8ef867SMichael Große * 123bb8ef867SMichael Große * @param Form $searchForm 124bb8ef867SMichael Große * @param array $parsedQuery 125bb8ef867SMichael Große */ 126bb8ef867SMichael Große protected function addSearchAssistanceElements(Form $searchForm, array $parsedQuery) 127bb8ef867SMichael Große { 128bb8ef867SMichael Große $matchType = ''; 129bb8ef867SMichael Große $searchTerm = null; 130bb8ef867SMichael Große if (count($parsedQuery['words']) === 1) { 131bb8ef867SMichael Große $searchTerm = $parsedQuery['words'][0]; 132bb8ef867SMichael Große $firstChar = $searchTerm[0]; 133bb8ef867SMichael Große $lastChar = substr($searchTerm, -1); 134bb8ef867SMichael Große $matchType = 'exact'; 135bb8ef867SMichael Große 136bb8ef867SMichael Große if ($firstChar === '*') { 137bb8ef867SMichael Große $matchType = 'starts'; 138bb8ef867SMichael Große } 139bb8ef867SMichael Große if ($lastChar === '*') { 140bb8ef867SMichael Große $matchType = 'ends'; 141bb8ef867SMichael Große } 142bb8ef867SMichael Große if ($firstChar === '*' && $lastChar === '*') { 143bb8ef867SMichael Große $matchType = 'contains'; 144bb8ef867SMichael Große } 145bb8ef867SMichael Große $searchTerm = trim($searchTerm, '*'); 146bb8ef867SMichael Große } 147bb8ef867SMichael Große 148bb8ef867SMichael Große $searchForm->addTextInput( 149bb8ef867SMichael Große 'searchTerm', 150bb8ef867SMichael Große '', 151bb8ef867SMichael Große $searchForm->findPositionByAttribute('type', 'submit') 152bb8ef867SMichael Große ) 153bb8ef867SMichael Große ->val($searchTerm) 154bb8ef867SMichael Große ->attr('style', 'display: none;'); 155bb8ef867SMichael Große $searchForm->addButton('toggleAssistant', 'toggle search assistant') 156bb8ef867SMichael Große ->attr('type', 'button') 157bb8ef867SMichael Große ->id('search-results-form__show-assistance-button') 158bb8ef867SMichael Große ->addClass('search-results-form__show-assistance-button'); 159bb8ef867SMichael Große 160bb8ef867SMichael Große $searchForm->addTagOpen('div') 161bb8ef867SMichael Große ->addClass('js-advancedSearchOptions') 162bb8ef867SMichael Große ->attr('style', 'display: none;'); 163bb8ef867SMichael Große 164bb8ef867SMichael Große $searchForm->addTagOpen('div')->addClass('search-results-form__subwrapper'); 165bb8ef867SMichael Große $searchForm->addRadioButton('matchType', 'exact Match FM')->val('exact')->attr('checked', 166bb8ef867SMichael Große $matchType === 'exact' ?: null); 167bb8ef867SMichael Große $searchForm->addRadioButton('matchType', 'starts with FM')->val('starts')->attr('checked', 168bb8ef867SMichael Große $matchType === 'starts' ?: null); 169bb8ef867SMichael Große $searchForm->addRadioButton('matchType', 'ends with FM')->val('ends')->attr('checked', 170bb8ef867SMichael Große $matchType === 'ends' ?: null); 171bb8ef867SMichael Große $searchForm->addRadioButton('matchType', 'contains FM')->val('contains')->attr('checked', 172bb8ef867SMichael Große $matchType === 'contains' ?: null); 173bb8ef867SMichael Große $searchForm->addTagClose('div'); 174bb8ef867SMichael Große 175bb8ef867SMichael Große $this->addNamespaceSelector($searchForm, $parsedQuery); 176bb8ef867SMichael Große 177bb8ef867SMichael Große $searchForm->addTagClose('div'); 178bb8ef867SMichael Große } 179bb8ef867SMichael Große 180bb8ef867SMichael Große /** 181bb8ef867SMichael Große * Add the elements for the namespace selector 182bb8ef867SMichael Große * 183bb8ef867SMichael Große * @param Form $searchForm 184bb8ef867SMichael Große * @param array $parsedQuery 185bb8ef867SMichael Große */ 186bb8ef867SMichael Große protected function addNamespaceSelector(Form $searchForm, array $parsedQuery) 187bb8ef867SMichael Große { 188bb8ef867SMichael Große $baseNS = empty($parsedQuery['ns']) ? '' : $parsedQuery['ns'][0]; 189bb8ef867SMichael Große $namespaces = []; 190bb8ef867SMichael Große $searchForm->addTagOpen('div')->addClass('search-results-form__subwrapper'); 191bb8ef867SMichael Große if ($baseNS) { 192bb8ef867SMichael Große $searchForm->addRadioButton('namespace', '(no namespace FIXME)')->val(''); 193bb8ef867SMichael Große $parts = [$baseNS => count($this->fullTextResults)]; 194bb8ef867SMichael Große $upperNameSpace = $baseNS; 195bb8ef867SMichael Große while ($upperNameSpace = getNS($upperNameSpace)) { 196bb8ef867SMichael Große $parts[$upperNameSpace] = 0; 197bb8ef867SMichael Große } 198bb8ef867SMichael Große $namespaces = array_reverse($parts); 199bb8ef867SMichael Große }; 200bb8ef867SMichael Große 201bb8ef867SMichael Große $namespaces = array_merge($namespaces, $this->getAdditionalNamespacesFromResults($baseNS)); 202bb8ef867SMichael Große 203bb8ef867SMichael Große foreach ($namespaces as $extraNS => $count) { 204bb8ef867SMichael Große $label = $extraNS . ($count ? " ($count)" : ''); 205bb8ef867SMichael Große $namespaceCB = $searchForm->addRadioButton('namespace', $label)->val($extraNS); 206bb8ef867SMichael Große if ($extraNS === $baseNS) { 207bb8ef867SMichael Große $namespaceCB->attr('checked', true); 208bb8ef867SMichael Große } 209bb8ef867SMichael Große } 210bb8ef867SMichael Große 211bb8ef867SMichael Große $searchForm->addTagClose('div'); 212bb8ef867SMichael Große } 213bb8ef867SMichael Große 214bb8ef867SMichael Große /** 215bb8ef867SMichael Große * Parse the full text results for their top namespaces below the given base namespace 216bb8ef867SMichael Große * 217bb8ef867SMichael Große * @param string $baseNS the namespace within which was searched, empty string for root namespace 218bb8ef867SMichael Große * 219bb8ef867SMichael Große * @return array an associative array with namespace => #number of found pages, sorted descending 220bb8ef867SMichael Große */ 221bb8ef867SMichael Große protected function getAdditionalNamespacesFromResults($baseNS) 222bb8ef867SMichael Große { 223bb8ef867SMichael Große $namespaces = []; 224bb8ef867SMichael Große $baseNSLength = strlen($baseNS); 225bb8ef867SMichael Große foreach ($this->fullTextResults as $page => $numberOfHits) { 226bb8ef867SMichael Große $namespace = getNS($page); 227bb8ef867SMichael Große if (!$namespace) { 228bb8ef867SMichael Große continue; 229bb8ef867SMichael Große } 230bb8ef867SMichael Große if ($namespace === $baseNS) { 231bb8ef867SMichael Große continue; 232bb8ef867SMichael Große } 233bb8ef867SMichael Große $firstColon = strpos((string)$namespace, ':', $baseNSLength + 1) ?: strlen($namespace); 234bb8ef867SMichael Große $subtopNS = substr($namespace, 0, $firstColon); 235bb8ef867SMichael Große if (empty($namespaces[$subtopNS])) { 236bb8ef867SMichael Große $namespaces[$subtopNS] = 0; 237bb8ef867SMichael Große } 238bb8ef867SMichael Große $namespaces[$subtopNS] += 1; 239bb8ef867SMichael Große } 240bb8ef867SMichael Große arsort($namespaces); 241bb8ef867SMichael Große return $namespaces; 242bb8ef867SMichael Große } 243bb8ef867SMichael Große 244bb8ef867SMichael Große /** 24521fcef82SMichael Große * Build the intro text for the search page 24621fcef82SMichael Große * 24721fcef82SMichael Große * @param string $query the search query 24821fcef82SMichael Große * 24921fcef82SMichael Große * @return string 25021fcef82SMichael Große */ 25121fcef82SMichael Große protected function getSearchIntroHTML($query) 25221fcef82SMichael Große { 25321fcef82SMichael Große global $ID, $lang; 25421fcef82SMichael Große 25521fcef82SMichael Große $intro = p_locale_xhtml('searchpage'); 25621fcef82SMichael Große // allow use of placeholder in search intro 25721fcef82SMichael Große $pagecreateinfo = (auth_quickaclcheck($ID) >= AUTH_CREATE) ? $lang['searchcreatepage'] : ''; 25821fcef82SMichael Große $intro = str_replace( 25921fcef82SMichael Große array('@QUERY@', '@SEARCH@', '@CREATEPAGEINFO@'), 26021fcef82SMichael Große array(hsc(rawurlencode($query)), hsc($query), $pagecreateinfo), 26121fcef82SMichael Große $intro 26221fcef82SMichael Große ); 26321fcef82SMichael Große return $intro; 26421fcef82SMichael Große } 26521fcef82SMichael Große 26621fcef82SMichael Große /** 26721fcef82SMichael Große * Build HTML for a list of pages with matching pagenames 26821fcef82SMichael Große * 26921fcef82SMichael Große * @param array $data search results 27021fcef82SMichael Große * 27121fcef82SMichael Große * @return string 27221fcef82SMichael Große */ 27321fcef82SMichael Große protected function getPageLookupHTML($data) 27421fcef82SMichael Große { 27521fcef82SMichael Große if (empty($data)) { 27621fcef82SMichael Große return ''; 27721fcef82SMichael Große } 27821fcef82SMichael Große 27921fcef82SMichael Große global $lang; 28021fcef82SMichael Große 28121fcef82SMichael Große $html = '<div class="search_quickresult">'; 28221fcef82SMichael Große $html .= '<h3>' . $lang['quickhits'] . ':</h3>'; 28321fcef82SMichael Große $html .= '<ul class="search_quickhits">'; 28421fcef82SMichael Große foreach ($data as $id => $title) { 2854eab6f7cSMichael Große $link = html_wikilink(':' . $id); 2864eab6f7cSMichael Große $eventData = [ 2874eab6f7cSMichael Große 'listItemContent' => [$link], 2884eab6f7cSMichael Große 'page' => $id, 2894eab6f7cSMichael Große ]; 2904eab6f7cSMichael Große trigger_event('SEARCH_RESULT_PAGELOOKUP', $eventData); 2914eab6f7cSMichael Große $html .= '<li>' . implode('', $eventData['listItemContent']) . '</li>'; 29221fcef82SMichael Große } 29321fcef82SMichael Große $html .= '</ul> '; 29421fcef82SMichael Große //clear float (see http://www.complexspiral.com/publications/containing-floats/) 29521fcef82SMichael Große $html .= '<div class="clearer"></div>'; 29621fcef82SMichael Große $html .= '</div>'; 29721fcef82SMichael Große 29821fcef82SMichael Große return $html; 29921fcef82SMichael Große } 30021fcef82SMichael Große 30121fcef82SMichael Große /** 30221fcef82SMichael Große * Build HTML for fulltext search results or "no results" message 30321fcef82SMichael Große * 30421fcef82SMichael Große * @param array $data the results of the fulltext search 30521fcef82SMichael Große * @param array $highlight the terms to be highlighted in the results 30621fcef82SMichael Große * 30721fcef82SMichael Große * @return string 30821fcef82SMichael Große */ 30921fcef82SMichael Große protected function getFulltextResultsHTML($data, $highlight) 31021fcef82SMichael Große { 31121fcef82SMichael Große global $lang; 31221fcef82SMichael Große 31321fcef82SMichael Große if (empty($data)) { 31421fcef82SMichael Große return '<div class="nothing">' . $lang['nothingfound'] . '</div>'; 31521fcef82SMichael Große } 31621fcef82SMichael Große 31721fcef82SMichael Große $html = ''; 31821fcef82SMichael Große $html .= '<dl class="search_results">'; 31921fcef82SMichael Große $num = 1; 32021fcef82SMichael Große foreach ($data as $id => $cnt) { 3214eab6f7cSMichael Große $resultLink = html_wikilink(':' . $id, null, $highlight); 3224eab6f7cSMichael Große $hits = ''; 3234eab6f7cSMichael Große $snippet = ''; 32421fcef82SMichael Große if ($cnt !== 0) { 3254eab6f7cSMichael Große $hits = $cnt . ' ' . $lang['hits']; 32621fcef82SMichael Große if ($num < FT_SNIPPET_NUMBER) { // create snippets for the first number of matches only 3274eab6f7cSMichael Große $snippet = '<dd>' . ft_snippet($id, $highlight) . '</dd>'; 32821fcef82SMichael Große } 32921fcef82SMichael Große $num++; 33021fcef82SMichael Große } 3314eab6f7cSMichael Große 3324eab6f7cSMichael Große $eventData = [ 3334eab6f7cSMichael Große 'resultHeader' => [$resultLink, $hits], 3344eab6f7cSMichael Große 'resultBody' => [$snippet], 3354eab6f7cSMichael Große 'page' => $id, 3364eab6f7cSMichael Große ]; 3374eab6f7cSMichael Große trigger_event('SEARCH_RESULT_FULLPAGE', $eventData); 3384eab6f7cSMichael Große $html .= '<div class="search_fullpage_result">'; 3394eab6f7cSMichael Große $html .= '<dt>' . implode(' ', $eventData['resultHeader']) . '</dt>'; 3404eab6f7cSMichael Große $html .= implode('', $eventData['resultBody']); 3414eab6f7cSMichael Große $html .= '</div>'; 34221fcef82SMichael Große } 34321fcef82SMichael Große $html .= '</dl>'; 34421fcef82SMichael Große 34521fcef82SMichael Große return $html; 34621fcef82SMichael Große } 34721fcef82SMichael Große} 348