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 65*bb8ef867SMichael Große $Indexer = idx_get_indexer(); 66*bb8ef867SMichael Große $parsedQuery = ft_queryParser($Indexer, $query); 67427ed988SMichael Große 68*bb8ef867SMichael Große $searchForm = (new Form())->attrs(['method' => 'get'])->addClass('search-results-form'); 69*bb8ef867SMichael Große $searchForm->setHiddenField('do', 'search'); 70*bb8ef867SMichael Große $searchForm->addFieldsetOpen()->addClass('search-results-form__fieldset'); 71*bb8ef867SMichael Große $searchForm->addTextInput('id')->val($query); 72427ed988SMichael Große $searchForm->addButton('', $lang['btn_search'])->attr('type', 'submit'); 73*bb8ef867SMichael Große 74*bb8ef867SMichael Große if ($this->isSearchAssistanceAvailable($parsedQuery)) { 75*bb8ef867SMichael Große $this->addSearchAssistanceElements($searchForm, $parsedQuery); 76*bb8ef867SMichael Große } else { 77*bb8ef867SMichael Große $searchForm->addClass('search-results-form--no-assistance'); 78*bb8ef867SMichael Große $searchForm->addTagOpen('span')->addClass('search-results-form__no-assistance-message'); 79*bb8ef867SMichael 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.'); 80*bb8ef867SMichael Große $searchForm->addTagClose('span'); 81*bb8ef867SMichael Große } 82*bb8ef867SMichael 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 /** 89*bb8ef867SMichael Große * Decide if the given query is simple enough to provide search assistance 90*bb8ef867SMichael Große * 91*bb8ef867SMichael Große * @param array $parsedQuery 92*bb8ef867SMichael Große * 93*bb8ef867SMichael Große * @return bool 94*bb8ef867SMichael Große */ 95*bb8ef867SMichael Große protected function isSearchAssistanceAvailable(array $parsedQuery) 96*bb8ef867SMichael Große { 97*bb8ef867SMichael Große if (count($parsedQuery['words']) > 1) { 98*bb8ef867SMichael Große return false; 99*bb8ef867SMichael Große } 100*bb8ef867SMichael Große if (!empty($parsedQuery['not'])) { 101*bb8ef867SMichael Große return false; 102*bb8ef867SMichael Große } 103*bb8ef867SMichael Große 104*bb8ef867SMichael Große if (!empty($parsedQuery['phrases'])) { 105*bb8ef867SMichael Große return false; 106*bb8ef867SMichael Große } 107*bb8ef867SMichael Große 108*bb8ef867SMichael Große if (!empty($parsedQuery['notns'])) { 109*bb8ef867SMichael Große return false; 110*bb8ef867SMichael Große } 111*bb8ef867SMichael Große if (count($parsedQuery['ns']) > 1) { 112*bb8ef867SMichael Große return false; 113*bb8ef867SMichael Große } 114*bb8ef867SMichael Große 115*bb8ef867SMichael Große return true; 116*bb8ef867SMichael Große } 117*bb8ef867SMichael Große 118*bb8ef867SMichael Große /** 119*bb8ef867SMichael Große * Add the elements to be used for search assistance 120*bb8ef867SMichael Große * 121*bb8ef867SMichael Große * @param Form $searchForm 122*bb8ef867SMichael Große * @param array $parsedQuery 123*bb8ef867SMichael Große */ 124*bb8ef867SMichael Große protected function addSearchAssistanceElements(Form $searchForm, array $parsedQuery) 125*bb8ef867SMichael Große { 126*bb8ef867SMichael Große $matchType = ''; 127*bb8ef867SMichael Große $searchTerm = null; 128*bb8ef867SMichael Große if (count($parsedQuery['words']) === 1) { 129*bb8ef867SMichael Große $searchTerm = $parsedQuery['words'][0]; 130*bb8ef867SMichael Große $firstChar = $searchTerm[0]; 131*bb8ef867SMichael Große $lastChar = substr($searchTerm, -1); 132*bb8ef867SMichael Große $matchType = 'exact'; 133*bb8ef867SMichael Große 134*bb8ef867SMichael Große if ($firstChar === '*') { 135*bb8ef867SMichael Große $matchType = 'starts'; 136*bb8ef867SMichael Große } 137*bb8ef867SMichael Große if ($lastChar === '*') { 138*bb8ef867SMichael Große $matchType = 'ends'; 139*bb8ef867SMichael Große } 140*bb8ef867SMichael Große if ($firstChar === '*' && $lastChar === '*') { 141*bb8ef867SMichael Große $matchType = 'contains'; 142*bb8ef867SMichael Große } 143*bb8ef867SMichael Große $searchTerm = trim($searchTerm, '*'); 144*bb8ef867SMichael Große } 145*bb8ef867SMichael Große 146*bb8ef867SMichael Große $searchForm->addTextInput( 147*bb8ef867SMichael Große 'searchTerm', 148*bb8ef867SMichael Große '', 149*bb8ef867SMichael Große $searchForm->findPositionByAttribute('type', 'submit') 150*bb8ef867SMichael Große ) 151*bb8ef867SMichael Große ->val($searchTerm) 152*bb8ef867SMichael Große ->attr('style', 'display: none;'); 153*bb8ef867SMichael Große $searchForm->addButton('toggleAssistant', 'toggle search assistant') 154*bb8ef867SMichael Große ->attr('type', 'button') 155*bb8ef867SMichael Große ->id('search-results-form__show-assistance-button') 156*bb8ef867SMichael Große ->addClass('search-results-form__show-assistance-button'); 157*bb8ef867SMichael Große 158*bb8ef867SMichael Große $searchForm->addTagOpen('div') 159*bb8ef867SMichael Große ->addClass('js-advancedSearchOptions') 160*bb8ef867SMichael Große ->attr('style', 'display: none;'); 161*bb8ef867SMichael Große 162*bb8ef867SMichael Große $searchForm->addTagOpen('div')->addClass('search-results-form__subwrapper'); 163*bb8ef867SMichael Große $searchForm->addRadioButton('matchType', 'exact Match FM')->val('exact')->attr('checked', 164*bb8ef867SMichael Große $matchType === 'exact' ?: null); 165*bb8ef867SMichael Große $searchForm->addRadioButton('matchType', 'starts with FM')->val('starts')->attr('checked', 166*bb8ef867SMichael Große $matchType === 'starts' ?: null); 167*bb8ef867SMichael Große $searchForm->addRadioButton('matchType', 'ends with FM')->val('ends')->attr('checked', 168*bb8ef867SMichael Große $matchType === 'ends' ?: null); 169*bb8ef867SMichael Große $searchForm->addRadioButton('matchType', 'contains FM')->val('contains')->attr('checked', 170*bb8ef867SMichael Große $matchType === 'contains' ?: null); 171*bb8ef867SMichael Große $searchForm->addTagClose('div'); 172*bb8ef867SMichael Große 173*bb8ef867SMichael Große $this->addNamespaceSelector($searchForm, $parsedQuery); 174*bb8ef867SMichael Große 175*bb8ef867SMichael Große $searchForm->addTagClose('div'); 176*bb8ef867SMichael Große } 177*bb8ef867SMichael Große 178*bb8ef867SMichael Große /** 179*bb8ef867SMichael Große * Add the elements for the namespace selector 180*bb8ef867SMichael Große * 181*bb8ef867SMichael Große * @param Form $searchForm 182*bb8ef867SMichael Große * @param array $parsedQuery 183*bb8ef867SMichael Große */ 184*bb8ef867SMichael Große protected function addNamespaceSelector(Form $searchForm, array $parsedQuery) 185*bb8ef867SMichael Große { 186*bb8ef867SMichael Große $baseNS = empty($parsedQuery['ns']) ? '' : $parsedQuery['ns'][0]; 187*bb8ef867SMichael Große $namespaces = []; 188*bb8ef867SMichael Große $searchForm->addTagOpen('div')->addClass('search-results-form__subwrapper'); 189*bb8ef867SMichael Große if ($baseNS) { 190*bb8ef867SMichael Große $searchForm->addRadioButton('namespace', '(no namespace FIXME)')->val(''); 191*bb8ef867SMichael Große $parts = [$baseNS => count($this->fullTextResults)]; 192*bb8ef867SMichael Große $upperNameSpace = $baseNS; 193*bb8ef867SMichael Große while ($upperNameSpace = getNS($upperNameSpace)) { 194*bb8ef867SMichael Große $parts[$upperNameSpace] = 0; 195*bb8ef867SMichael Große } 196*bb8ef867SMichael Große $namespaces = array_reverse($parts); 197*bb8ef867SMichael Große }; 198*bb8ef867SMichael Große 199*bb8ef867SMichael Große $namespaces = array_merge($namespaces, $this->getAdditionalNamespacesFromResults($baseNS)); 200*bb8ef867SMichael Große 201*bb8ef867SMichael Große foreach ($namespaces as $extraNS => $count) { 202*bb8ef867SMichael Große $label = $extraNS . ($count ? " ($count)" : ''); 203*bb8ef867SMichael Große $namespaceCB = $searchForm->addRadioButton('namespace', $label)->val($extraNS); 204*bb8ef867SMichael Große if ($extraNS === $baseNS) { 205*bb8ef867SMichael Große $namespaceCB->attr('checked', true); 206*bb8ef867SMichael Große } 207*bb8ef867SMichael Große } 208*bb8ef867SMichael Große 209*bb8ef867SMichael Große $searchForm->addTagClose('div'); 210*bb8ef867SMichael Große } 211*bb8ef867SMichael Große 212*bb8ef867SMichael Große /** 213*bb8ef867SMichael Große * Parse the full text results for their top namespaces below the given base namespace 214*bb8ef867SMichael Große * 215*bb8ef867SMichael Große * @param string $baseNS the namespace within which was searched, empty string for root namespace 216*bb8ef867SMichael Große * 217*bb8ef867SMichael Große * @return array an associative array with namespace => #number of found pages, sorted descending 218*bb8ef867SMichael Große */ 219*bb8ef867SMichael Große protected function getAdditionalNamespacesFromResults($baseNS) 220*bb8ef867SMichael Große { 221*bb8ef867SMichael Große $namespaces = []; 222*bb8ef867SMichael Große $baseNSLength = strlen($baseNS); 223*bb8ef867SMichael Große foreach ($this->fullTextResults as $page => $numberOfHits) { 224*bb8ef867SMichael Große $namespace = getNS($page); 225*bb8ef867SMichael Große if (!$namespace) { 226*bb8ef867SMichael Große continue; 227*bb8ef867SMichael Große } 228*bb8ef867SMichael Große if ($namespace === $baseNS) { 229*bb8ef867SMichael Große continue; 230*bb8ef867SMichael Große } 231*bb8ef867SMichael Große $firstColon = strpos((string)$namespace, ':', $baseNSLength + 1) ?: strlen($namespace); 232*bb8ef867SMichael Große $subtopNS = substr($namespace, 0, $firstColon); 233*bb8ef867SMichael Große if (empty($namespaces[$subtopNS])) { 234*bb8ef867SMichael Große $namespaces[$subtopNS] = 0; 235*bb8ef867SMichael Große } 236*bb8ef867SMichael Große $namespaces[$subtopNS] += 1; 237*bb8ef867SMichael Große } 238*bb8ef867SMichael Große arsort($namespaces); 239*bb8ef867SMichael Große return $namespaces; 240*bb8ef867SMichael Große } 241*bb8ef867SMichael Große 242*bb8ef867SMichael 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) { 28321fcef82SMichael Große $html .= '<li> '; 28421fcef82SMichael Große if (useHeading('navigation')) { 28521fcef82SMichael Große $name = $title; 28621fcef82SMichael Große } else { 28721fcef82SMichael Große $ns = getNS($id); 28821fcef82SMichael Große if ($ns) { 28921fcef82SMichael Große $name = shorten(noNS($id), ' (' . $ns . ')', 30); 29021fcef82SMichael Große } else { 29121fcef82SMichael Große $name = $id; 29221fcef82SMichael Große } 29321fcef82SMichael Große } 29421fcef82SMichael Große $html .= html_wikilink(':' . $id, $name); 29521fcef82SMichael Große $html .= '</li> '; 29621fcef82SMichael Große } 29721fcef82SMichael Große $html .= '</ul> '; 29821fcef82SMichael Große //clear float (see http://www.complexspiral.com/publications/containing-floats/) 29921fcef82SMichael Große $html .= '<div class="clearer"></div>'; 30021fcef82SMichael Große $html .= '</div>'; 30121fcef82SMichael Große 30221fcef82SMichael Große return $html; 30321fcef82SMichael Große } 30421fcef82SMichael Große 30521fcef82SMichael Große /** 30621fcef82SMichael Große * Build HTML for fulltext search results or "no results" message 30721fcef82SMichael Große * 30821fcef82SMichael Große * @param array $data the results of the fulltext search 30921fcef82SMichael Große * @param array $highlight the terms to be highlighted in the results 31021fcef82SMichael Große * 31121fcef82SMichael Große * @return string 31221fcef82SMichael Große */ 31321fcef82SMichael Große protected function getFulltextResultsHTML($data, $highlight) 31421fcef82SMichael Große { 31521fcef82SMichael Große global $lang; 31621fcef82SMichael Große 31721fcef82SMichael Große if (empty($data)) { 31821fcef82SMichael Große return '<div class="nothing">' . $lang['nothingfound'] . '</div>'; 31921fcef82SMichael Große } 32021fcef82SMichael Große 32121fcef82SMichael Große $html = ''; 32221fcef82SMichael Große $html .= '<dl class="search_results">'; 32321fcef82SMichael Große $num = 1; 32421fcef82SMichael Große foreach ($data as $id => $cnt) { 32521fcef82SMichael Große $html .= '<dt>'; 32621fcef82SMichael Große $html .= html_wikilink(':' . $id, useHeading('navigation') ? null : $id, $highlight); 32721fcef82SMichael Große if ($cnt !== 0) { 32821fcef82SMichael Große $html .= ': ' . $cnt . ' ' . $lang['hits'] . ''; 32921fcef82SMichael Große } 33021fcef82SMichael Große $html .= '</dt>'; 33121fcef82SMichael Große if ($cnt !== 0) { 33221fcef82SMichael Große if ($num < FT_SNIPPET_NUMBER) { // create snippets for the first number of matches only 33321fcef82SMichael Große $html .= '<dd>' . ft_snippet($id, $highlight) . '</dd>'; 33421fcef82SMichael Große } 33521fcef82SMichael Große $num++; 33621fcef82SMichael Große } 33721fcef82SMichael Große } 33821fcef82SMichael Große $html .= '</dl>'; 33921fcef82SMichael Große 34021fcef82SMichael Große return $html; 34121fcef82SMichael Große } 34221fcef82SMichael Große} 343