xref: /dokuwiki/inc/Ui/Search.php (revision 4eab6f7c40ab5e2fbbb5e0efb20d930d4b1730fe)
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