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