xref: /dokuwiki/inc/Ui/Search.php (revision 4d0cb6e1cdd7281b9ab09540927bc3c447fcff17)
1<?php
2
3namespace dokuwiki\Ui;
4
5use \dokuwiki\Form\Form;
6
7class Search extends Ui
8{
9    protected $query;
10    protected $parsedQuery;
11    protected $pageLookupResults = array();
12    protected $fullTextResults = array();
13    protected $highlight = array();
14
15    /**
16     * Search constructor.
17     *
18     * @param string $query the search query
19     */
20    public function __construct()
21    {
22        global $QUERY;
23
24        $Indexer = idx_get_indexer();
25        $parsedQuery = ft_queryParser($Indexer, $QUERY);
26
27        $this->query = $QUERY;
28        $this->parsedQuery = $parsedQuery;
29    }
30
31    /**
32     * run the search
33     */
34    public function execute()
35    {
36        $this->pageLookupResults = ft_pageLookup($this->query, true, useHeading('navigation'));
37        $this->fullTextResults = ft_pageSearch($this->query, $highlight);
38        $this->highlight = $highlight;
39    }
40
41    /**
42     * display the search result
43     *
44     * @return void
45     */
46    public function show()
47    {
48        $searchHTML = '';
49
50        $searchHTML .= $this->getSearchFormHTML($this->query);
51
52        $searchHTML .= $this->getSearchIntroHTML($this->query);
53
54        $searchHTML .= $this->getPageLookupHTML($this->pageLookupResults);
55
56        $searchHTML .= $this->getFulltextResultsHTML($this->fullTextResults, $this->highlight);
57
58        echo $searchHTML;
59    }
60
61    /**
62     * Get a form which can be used to adjust/refine the search
63     *
64     * @param string $query
65     *
66     * @return string
67     */
68    protected function getSearchFormHTML($query)
69    {
70        global $lang, $ID;
71
72        $searchForm = (new Form())->attrs(['method' => 'get'])->addClass('search-results-form');
73        $searchForm->setHiddenField('do', 'search');
74        $searchForm->setHiddenField('from', $ID);
75        $searchForm->setHiddenField('searchPageForm', '1');
76        $searchForm->addFieldsetOpen()->addClass('search-results-form__fieldset');
77        $searchForm->addTextInput('id')->val($query)->useInput(false);
78        $searchForm->addButton('', $lang['btn_search'])->attr('type', 'submit');
79
80        if ($this->isSearchAssistanceAvailable($this->parsedQuery)) {
81            $this->addSearchAssistanceElements($searchForm, $this->parsedQuery);
82        } else {
83            $searchForm->addClass('search-results-form--no-assistance');
84            $searchForm->addTagOpen('span')->addClass('search-results-form__no-assistance-message');
85            $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.');
86            $searchForm->addTagClose('span');
87        }
88
89        $searchForm->addFieldsetClose();
90
91        trigger_event('SEARCH_FORM_DISPLAY', $searchForm);
92
93        return $searchForm->toHTML();
94    }
95
96    /**
97     * Decide if the given query is simple enough to provide search assistance
98     *
99     * @param array $parsedQuery
100     *
101     * @return bool
102     */
103    protected function isSearchAssistanceAvailable(array $parsedQuery)
104    {
105        if (count($parsedQuery['words']) > 1) {
106            return false;
107        }
108        if (!empty($parsedQuery['not'])) {
109            return false;
110        }
111
112        if (!empty($parsedQuery['phrases'])) {
113            return false;
114        }
115
116        if (!empty($parsedQuery['notns'])) {
117            return false;
118        }
119        if (count($parsedQuery['ns']) > 1) {
120            return false;
121        }
122
123        return true;
124    }
125
126    /**
127     * Add the elements to be used for search assistance
128     *
129     * @param Form  $searchForm
130     * @param array $parsedQuery
131     */
132    protected function addSearchAssistanceElements(Form $searchForm, array $parsedQuery)
133    {
134        $searchForm->addButton('toggleAssistant', 'toggle search assistant')
135            ->attr('type', 'button')
136            ->id('search-results-form__show-assistance-button')
137            ->addClass('search-results-form__show-assistance-button');
138
139        $searchForm->addTagOpen('div')
140            ->addClass('js-advancedSearchOptions')
141            ->attr('style', 'display: none;');
142
143        $this->addFragmentBehaviorLinks($searchForm, $parsedQuery);
144        $this->addNamespaceSelector($searchForm, $parsedQuery);
145
146        $searchForm->addTagClose('div');
147    }
148
149    protected function addFragmentBehaviorLinks(Form $searchForm, array $parsedQuery)
150    {
151        $searchForm->addTagOpen('div')->addClass('search-results-form__subwrapper');
152
153        $this->addSearchLink(
154            $searchForm,
155            'exact Match',
156            array_map(function($term){return trim($term, '*');},$this->parsedQuery['and']),
157            $this->parsedQuery['ns']
158        );
159
160        $searchForm->addHTML(' ');
161
162        $this->addSearchLink(
163            $searchForm,
164            'starts with',
165            array_map(function($term){return trim($term, '*') . '*';},$this->parsedQuery['and']),
166            $this->parsedQuery['ns']
167        );
168
169        $searchForm->addHTML(' ');
170
171        $this->addSearchLink(
172            $searchForm,
173            'ends with',
174            array_map(function($term){return '*' . trim($term, '*');},$this->parsedQuery['and']),
175            $this->parsedQuery['ns']
176        );
177
178        $searchForm->addHTML(' ');
179
180        $this->addSearchLink(
181            $searchForm,
182            'contains',
183            array_map(function($term){return '*' . trim($term, '*') . '*';},$this->parsedQuery['and']),
184            $this->parsedQuery['ns']
185        );
186
187        $searchForm->addTagClose('div');
188    }
189
190    protected function addSearchLink(Form $searchForm, $label, $and, $ns) {
191        $newQuery = ft_queryUnparser_simple(
192            $and,
193            [],
194            [],
195            $ns,
196            []
197        );
198        $searchForm->addTagOpen('a')
199            ->attrs([
200                'href' => wl($newQuery, ['do' => 'search', 'searchPageForm' => '1'], false, '&')
201            ])
202        ;
203        $searchForm->addHTML($label);
204        $searchForm->addTagClose('a');
205    }
206
207    /**
208     * Add the elements for the namespace selector
209     *
210     * @param Form  $searchForm
211     * @param array $parsedQuery
212     */
213    protected function addNamespaceSelector(Form $searchForm, array $parsedQuery)
214    {
215        $baseNS = empty($parsedQuery['ns']) ? '' : $parsedQuery['ns'][0];
216        $searchForm->addTagOpen('div')->addClass('search-results-form__subwrapper');
217
218        if ($baseNS) {
219            $searchForm->addTagOpen('div');
220
221            $this->addSearchLink(
222                $searchForm,
223                'remove current namespace restriction',
224                $this->parsedQuery['and'],
225                []
226            );
227
228            $searchForm->addTagClose('div');
229        }
230
231        $extraNS = $this->getAdditionalNamespacesFromResults($baseNS);
232        if (!empty($extraNS)) {
233            $searchForm->addTagOpen('div');
234            $searchForm->addHTML('first level ns below current: ');
235
236            foreach ($extraNS as $extraNS => $count) {
237                $searchForm->addHTML(' ');
238                $label = $extraNS . ($count ? " ($count)" : '');
239
240                $this->addSearchLink($searchForm, $label, $this->parsedQuery['and'], [$extraNS]);
241            }
242            $searchForm->addTagClose('div');
243        }
244
245        $searchForm->addTagClose('div');
246    }
247
248    /**
249     * Parse the full text results for their top namespaces below the given base namespace
250     *
251     * @param string $baseNS the namespace within which was searched, empty string for root namespace
252     *
253     * @return array an associative array with namespace => #number of found pages, sorted descending
254     */
255    protected function getAdditionalNamespacesFromResults($baseNS)
256    {
257        $namespaces = [];
258        $baseNSLength = strlen($baseNS);
259        foreach ($this->fullTextResults as $page => $numberOfHits) {
260            $namespace = getNS($page);
261            if (!$namespace) {
262                continue;
263            }
264            if ($namespace === $baseNS) {
265                continue;
266            }
267            $firstColon = strpos((string)$namespace, ':', $baseNSLength + 1) ?: strlen($namespace);
268            $subtopNS = substr($namespace, 0, $firstColon);
269            if (empty($namespaces[$subtopNS])) {
270                $namespaces[$subtopNS] = 0;
271            }
272            $namespaces[$subtopNS] += 1;
273        }
274        arsort($namespaces);
275        return $namespaces;
276    }
277
278    /**
279     * Build the intro text for the search page
280     *
281     * @param string $query the search query
282     *
283     * @return string
284     */
285    protected function getSearchIntroHTML($query)
286    {
287        global $ID, $lang;
288
289        $intro = p_locale_xhtml('searchpage');
290        // allow use of placeholder in search intro
291        $pagecreateinfo = (auth_quickaclcheck($ID) >= AUTH_CREATE) ? $lang['searchcreatepage'] : '';
292        $intro = str_replace(
293            array('@QUERY@', '@SEARCH@', '@CREATEPAGEINFO@'),
294            array(hsc(rawurlencode($query)), hsc($query), $pagecreateinfo),
295            $intro
296        );
297        return $intro;
298    }
299
300    /**
301     * Build HTML for a list of pages with matching pagenames
302     *
303     * @param array $data search results
304     *
305     * @return string
306     */
307    protected function getPageLookupHTML($data)
308    {
309        if (empty($data)) {
310            return '';
311        }
312
313        global $lang;
314
315        $html = '<div class="search_quickresult">';
316        $html .= '<h3>' . $lang['quickhits'] . ':</h3>';
317        $html .= '<ul class="search_quickhits">';
318        foreach ($data as $id => $title) {
319            $link = html_wikilink(':' . $id);
320            $eventData = [
321                'listItemContent' => [$link],
322                'page' => $id,
323            ];
324            trigger_event('SEARCH_RESULT_PAGELOOKUP', $eventData);
325            $html .= '<li>' . implode('', $eventData['listItemContent']) . '</li>';
326        }
327        $html .= '</ul> ';
328        //clear float (see http://www.complexspiral.com/publications/containing-floats/)
329        $html .= '<div class="clearer"></div>';
330        $html .= '</div>';
331
332        return $html;
333    }
334
335    /**
336     * Build HTML for fulltext search results or "no results" message
337     *
338     * @param array $data      the results of the fulltext search
339     * @param array $highlight the terms to be highlighted in the results
340     *
341     * @return string
342     */
343    protected function getFulltextResultsHTML($data, $highlight)
344    {
345        global $lang;
346
347        if (empty($data)) {
348            return '<div class="nothing">' . $lang['nothingfound'] . '</div>';
349        }
350
351        $html = '';
352        $html .= '<dl class="search_results">';
353        $num = 1;
354
355        foreach ($data as $id => $cnt) {
356            $resultLink = html_wikilink(':' . $id, null, $highlight);
357
358            $resultHeader = [$resultLink];
359
360            $snippet = '';
361            if ($cnt !== 0) {
362                $resultHeader[] = $cnt . ' ' . $lang['hits'];
363                if ($num < FT_SNIPPET_NUMBER) { // create snippets for the first number of matches only
364                    $snippet = '<dd>' . ft_snippet($id, $highlight) . '</dd>';
365                }
366                $num++;
367            }
368
369            $restrictQueryToNSLink = $this->restrictQueryToNSLink(getNS($id));
370            if ($restrictQueryToNSLink) {
371                $resultHeader[] = $restrictQueryToNSLink;
372            }
373
374            $eventData = [
375                'resultHeader' => $resultHeader,
376                'resultBody' => [$snippet],
377                'page' => $id,
378            ];
379            trigger_event('SEARCH_RESULT_FULLPAGE', $eventData);
380            $html .= '<div class="search_fullpage_result">';
381            $html .= '<dt>' . implode(' ', $eventData['resultHeader']) . '</dt>';
382            $html .= implode('', $eventData['resultBody']);
383            $html .= '</div>';
384        }
385        $html .= '</dl>';
386
387        return $html;
388    }
389
390    /**
391     * create a link to restrict the current query to a namespace
392     *
393     * @param bool|string $ns the namespace to which to restrict the query
394     *
395     * @return bool|string
396     */
397    protected function restrictQueryToNSLink($ns)
398    {
399        if (!$ns) {
400            return false;
401        }
402        if (!$this->isSearchAssistanceAvailable($this->parsedQuery)) {
403            return false;
404        }
405        if (!empty($this->parsedQuery['ns']) && $this->parsedQuery['ns'][0] === $ns) {
406            return false;
407        }
408
409        $newQuery = ft_queryUnparser_simple(
410            $this->parsedQuery['and'],
411            [],
412            [],
413            [$ns],
414            []
415        );
416        $href = wl($newQuery, ['do' => 'search', 'searchPageForm' => '1']);
417        $attributes = buildAttributes([
418            'rel' => 'nofollow',
419            'class' => 'search_namespace_link',
420        ]);
421        $name = '@' . $ns;
422        return "<a href=\"$href\" $attributes>$name</a>";
423    }
424}
425