summaryrefslogtreecommitdiff
path: root/platform/www/inc/Ui/Search.php
diff options
context:
space:
mode:
Diffstat (limited to 'platform/www/inc/Ui/Search.php')
-rw-r--r--platform/www/inc/Ui/Search.php647
1 files changed, 647 insertions, 0 deletions
diff --git a/platform/www/inc/Ui/Search.php b/platform/www/inc/Ui/Search.php
new file mode 100644
index 0000000..e4eef67
--- /dev/null
+++ b/platform/www/inc/Ui/Search.php
@@ -0,0 +1,647 @@
+<?php
+
+namespace dokuwiki\Ui;
+
+use dokuwiki\Extension\Event;
+use dokuwiki\Form\Form;
+
+class Search extends Ui
+{
+ protected $query;
+ protected $parsedQuery;
+ protected $searchState;
+ protected $pageLookupResults = array();
+ protected $fullTextResults = array();
+ protected $highlight = array();
+
+ /**
+ * Search constructor.
+ *
+ * @param array $pageLookupResults pagename lookup results in the form [pagename => pagetitle]
+ * @param array $fullTextResults fulltext search results in the form [pagename => #hits]
+ * @param array $highlight array of strings to be highlighted
+ */
+ public function __construct(array $pageLookupResults, array $fullTextResults, $highlight)
+ {
+ global $QUERY;
+ $Indexer = idx_get_indexer();
+
+ $this->query = $QUERY;
+ $this->parsedQuery = ft_queryParser($Indexer, $QUERY);
+ $this->searchState = new SearchState($this->parsedQuery);
+
+ $this->pageLookupResults = $pageLookupResults;
+ $this->fullTextResults = $fullTextResults;
+ $this->highlight = $highlight;
+ }
+
+ /**
+ * display the search result
+ *
+ * @return void
+ */
+ public function show()
+ {
+ $searchHTML = '';
+
+ $searchHTML .= $this->getSearchIntroHTML($this->query);
+
+ $searchHTML .= $this->getSearchFormHTML($this->query);
+
+ $searchHTML .= $this->getPageLookupHTML($this->pageLookupResults);
+
+ $searchHTML .= $this->getFulltextResultsHTML($this->fullTextResults, $this->highlight);
+
+ echo $searchHTML;
+ }
+
+ /**
+ * Get a form which can be used to adjust/refine the search
+ *
+ * @param string $query
+ *
+ * @return string
+ */
+ protected function getSearchFormHTML($query)
+ {
+ global $lang, $ID, $INPUT;
+
+ $searchForm = (new Form(['method' => 'get'], true))->addClass('search-results-form');
+ $searchForm->setHiddenField('do', 'search');
+ $searchForm->setHiddenField('id', $ID);
+ $searchForm->setHiddenField('sf', '1');
+ if ($INPUT->has('min')) {
+ $searchForm->setHiddenField('min', $INPUT->str('min'));
+ }
+ if ($INPUT->has('max')) {
+ $searchForm->setHiddenField('max', $INPUT->str('max'));
+ }
+ if ($INPUT->has('srt')) {
+ $searchForm->setHiddenField('srt', $INPUT->str('srt'));
+ }
+ $searchForm->addFieldsetOpen()->addClass('search-form');
+ $searchForm->addTextInput('q')->val($query)->useInput(false);
+ $searchForm->addButton('', $lang['btn_search'])->attr('type', 'submit');
+
+ $this->addSearchAssistanceElements($searchForm);
+
+ $searchForm->addFieldsetClose();
+
+ Event::createAndTrigger('FORM_SEARCH_OUTPUT', $searchForm);
+
+ return $searchForm->toHTML();
+ }
+
+ /**
+ * Add elements to adjust how the results are sorted
+ *
+ * @param Form $searchForm
+ */
+ protected function addSortTool(Form $searchForm)
+ {
+ global $INPUT, $lang;
+
+ $options = [
+ 'hits' => [
+ 'label' => $lang['search_sort_by_hits'],
+ 'sort' => '',
+ ],
+ 'mtime' => [
+ 'label' => $lang['search_sort_by_mtime'],
+ 'sort' => 'mtime',
+ ],
+ ];
+ $activeOption = 'hits';
+
+ if ($INPUT->str('srt') === 'mtime') {
+ $activeOption = 'mtime';
+ }
+
+ $searchForm->addTagOpen('div')->addClass('toggle')->attr('aria-haspopup', 'true');
+ // render current
+ $currentWrapper = $searchForm->addTagOpen('div')->addClass('current');
+ if ($activeOption !== 'hits') {
+ $currentWrapper->addClass('changed');
+ }
+ $searchForm->addHTML($options[$activeOption]['label']);
+ $searchForm->addTagClose('div');
+
+ // render options list
+ $searchForm->addTagOpen('ul')->attr('aria-expanded', 'false');
+
+ foreach ($options as $key => $option) {
+ $listItem = $searchForm->addTagOpen('li');
+
+ if ($key === $activeOption) {
+ $listItem->addClass('active');
+ $searchForm->addHTML($option['label']);
+ } else {
+ $link = $this->searchState->withSorting($option['sort'])->getSearchLink($option['label']);
+ $searchForm->addHTML($link);
+ }
+ $searchForm->addTagClose('li');
+ }
+ $searchForm->addTagClose('ul');
+
+ $searchForm->addTagClose('div');
+
+ }
+
+ /**
+ * Check if the query is simple enough to modify its namespace limitations without breaking the rest of the query
+ *
+ * @param array $parsedQuery
+ *
+ * @return bool
+ */
+ protected function isNamespaceAssistanceAvailable(array $parsedQuery) {
+ if (preg_match('/[\(\)\|]/', $parsedQuery['query']) === 1) {
+ return false;
+ }
+
+ return true;
+ }
+
+ /**
+ * Check if the query is simple enough to modify the fragment search behavior without breaking the rest of the query
+ *
+ * @param array $parsedQuery
+ *
+ * @return bool
+ */
+ protected function isFragmentAssistanceAvailable(array $parsedQuery) {
+ if (preg_match('/[\(\)\|]/', $parsedQuery['query']) === 1) {
+ return false;
+ }
+
+ if (!empty($parsedQuery['phrases'])) {
+ return false;
+ }
+
+ return true;
+ }
+
+ /**
+ * Add the elements to be used for search assistance
+ *
+ * @param Form $searchForm
+ */
+ protected function addSearchAssistanceElements(Form $searchForm)
+ {
+ $searchForm->addTagOpen('div')
+ ->addClass('advancedOptions')
+ ->attr('style', 'display: none;')
+ ->attr('aria-hidden', 'true');
+
+ $this->addFragmentBehaviorLinks($searchForm);
+ $this->addNamespaceSelector($searchForm);
+ $this->addDateSelector($searchForm);
+ $this->addSortTool($searchForm);
+
+ $searchForm->addTagClose('div');
+ }
+
+ /**
+ * Add the elements to adjust the fragment search behavior
+ *
+ * @param Form $searchForm
+ */
+ protected function addFragmentBehaviorLinks(Form $searchForm)
+ {
+ if (!$this->isFragmentAssistanceAvailable($this->parsedQuery)) {
+ return;
+ }
+ global $lang;
+
+ $options = [
+ 'exact' => [
+ 'label' => $lang['search_exact_match'],
+ 'and' => array_map(function ($term) {
+ return trim($term, '*');
+ }, $this->parsedQuery['and']),
+ 'not' => array_map(function ($term) {
+ return trim($term, '*');
+ }, $this->parsedQuery['not']),
+ ],
+ 'starts' => [
+ 'label' => $lang['search_starts_with'],
+ 'and' => array_map(function ($term) {
+ return trim($term, '*') . '*';
+ }, $this->parsedQuery['and']),
+ 'not' => array_map(function ($term) {
+ return trim($term, '*') . '*';
+ }, $this->parsedQuery['not']),
+ ],
+ 'ends' => [
+ 'label' => $lang['search_ends_with'],
+ 'and' => array_map(function ($term) {
+ return '*' . trim($term, '*');
+ }, $this->parsedQuery['and']),
+ 'not' => array_map(function ($term) {
+ return '*' . trim($term, '*');
+ }, $this->parsedQuery['not']),
+ ],
+ 'contains' => [
+ 'label' => $lang['search_contains'],
+ 'and' => array_map(function ($term) {
+ return '*' . trim($term, '*') . '*';
+ }, $this->parsedQuery['and']),
+ 'not' => array_map(function ($term) {
+ return '*' . trim($term, '*') . '*';
+ }, $this->parsedQuery['not']),
+ ]
+ ];
+
+ // detect current
+ $activeOption = 'custom';
+ foreach ($options as $key => $option) {
+ if ($this->parsedQuery['and'] === $option['and']) {
+ $activeOption = $key;
+ }
+ }
+ if ($activeOption === 'custom') {
+ $options = array_merge(['custom' => [
+ 'label' => $lang['search_custom_match'],
+ ]], $options);
+ }
+
+ $searchForm->addTagOpen('div')->addClass('toggle')->attr('aria-haspopup', 'true');
+ // render current
+ $currentWrapper = $searchForm->addTagOpen('div')->addClass('current');
+ if ($activeOption !== 'exact') {
+ $currentWrapper->addClass('changed');
+ }
+ $searchForm->addHTML($options[$activeOption]['label']);
+ $searchForm->addTagClose('div');
+
+ // render options list
+ $searchForm->addTagOpen('ul')->attr('aria-expanded', 'false');
+
+ foreach ($options as $key => $option) {
+ $listItem = $searchForm->addTagOpen('li');
+
+ if ($key === $activeOption) {
+ $listItem->addClass('active');
+ $searchForm->addHTML($option['label']);
+ } else {
+ $link = $this->searchState
+ ->withFragments($option['and'], $option['not'])
+ ->getSearchLink($option['label'])
+ ;
+ $searchForm->addHTML($link);
+ }
+ $searchForm->addTagClose('li');
+ }
+ $searchForm->addTagClose('ul');
+
+ $searchForm->addTagClose('div');
+
+ // render options list
+ }
+
+ /**
+ * Add the elements for the namespace selector
+ *
+ * @param Form $searchForm
+ */
+ protected function addNamespaceSelector(Form $searchForm)
+ {
+ if (!$this->isNamespaceAssistanceAvailable($this->parsedQuery)) {
+ return;
+ }
+
+ global $lang;
+
+ $baseNS = empty($this->parsedQuery['ns']) ? '' : $this->parsedQuery['ns'][0];
+ $extraNS = $this->getAdditionalNamespacesFromResults($baseNS);
+
+ $searchForm->addTagOpen('div')->addClass('toggle')->attr('aria-haspopup', 'true');
+ // render current
+ $currentWrapper = $searchForm->addTagOpen('div')->addClass('current');
+ if ($baseNS) {
+ $currentWrapper->addClass('changed');
+ $searchForm->addHTML('@' . $baseNS);
+ } else {
+ $searchForm->addHTML($lang['search_any_ns']);
+ }
+ $searchForm->addTagClose('div');
+
+ // render options list
+ $searchForm->addTagOpen('ul')->attr('aria-expanded', 'false');
+
+ $listItem = $searchForm->addTagOpen('li');
+ if ($baseNS) {
+ $listItem->addClass('active');
+ $link = $this->searchState->withNamespace('')->getSearchLink($lang['search_any_ns']);
+ $searchForm->addHTML($link);
+ } else {
+ $searchForm->addHTML($lang['search_any_ns']);
+ }
+ $searchForm->addTagClose('li');
+
+ foreach ($extraNS as $ns => $count) {
+ $listItem = $searchForm->addTagOpen('li');
+ $label = $ns . ($count ? " <bdi>($count)</bdi>" : '');
+
+ if ($ns === $baseNS) {
+ $listItem->addClass('active');
+ $searchForm->addHTML($label);
+ } else {
+ $link = $this->searchState->withNamespace($ns)->getSearchLink($label);
+ $searchForm->addHTML($link);
+ }
+ $searchForm->addTagClose('li');
+ }
+ $searchForm->addTagClose('ul');
+
+ $searchForm->addTagClose('div');
+
+ }
+
+ /**
+ * Parse the full text results for their top namespaces below the given base namespace
+ *
+ * @param string $baseNS the namespace within which was searched, empty string for root namespace
+ *
+ * @return array an associative array with namespace => #number of found pages, sorted descending
+ */
+ protected function getAdditionalNamespacesFromResults($baseNS)
+ {
+ $namespaces = [];
+ $baseNSLength = strlen($baseNS);
+ foreach ($this->fullTextResults as $page => $numberOfHits) {
+ $namespace = getNS($page);
+ if (!$namespace) {
+ continue;
+ }
+ if ($namespace === $baseNS) {
+ continue;
+ }
+ $firstColon = strpos((string)$namespace, ':', $baseNSLength + 1) ?: strlen($namespace);
+ $subtopNS = substr($namespace, 0, $firstColon);
+ if (empty($namespaces[$subtopNS])) {
+ $namespaces[$subtopNS] = 0;
+ }
+ $namespaces[$subtopNS] += 1;
+ }
+ ksort($namespaces);
+ arsort($namespaces);
+ return $namespaces;
+ }
+
+ /**
+ * @ToDo: custom date input
+ *
+ * @param Form $searchForm
+ */
+ protected function addDateSelector(Form $searchForm)
+ {
+ global $INPUT, $lang;
+
+ $options = [
+ 'any' => [
+ 'before' => false,
+ 'after' => false,
+ 'label' => $lang['search_any_time'],
+ ],
+ 'week' => [
+ 'before' => false,
+ 'after' => '1 week ago',
+ 'label' => $lang['search_past_7_days'],
+ ],
+ 'month' => [
+ 'before' => false,
+ 'after' => '1 month ago',
+ 'label' => $lang['search_past_month'],
+ ],
+ 'year' => [
+ 'before' => false,
+ 'after' => '1 year ago',
+ 'label' => $lang['search_past_year'],
+ ],
+ ];
+ $activeOption = 'any';
+ foreach ($options as $key => $option) {
+ if ($INPUT->str('min') === $option['after']) {
+ $activeOption = $key;
+ break;
+ }
+ }
+
+ $searchForm->addTagOpen('div')->addClass('toggle')->attr('aria-haspopup', 'true');
+ // render current
+ $currentWrapper = $searchForm->addTagOpen('div')->addClass('current');
+ if ($INPUT->has('max') || $INPUT->has('min')) {
+ $currentWrapper->addClass('changed');
+ }
+ $searchForm->addHTML($options[$activeOption]['label']);
+ $searchForm->addTagClose('div');
+
+ // render options list
+ $searchForm->addTagOpen('ul')->attr('aria-expanded', 'false');
+
+ foreach ($options as $key => $option) {
+ $listItem = $searchForm->addTagOpen('li');
+
+ if ($key === $activeOption) {
+ $listItem->addClass('active');
+ $searchForm->addHTML($option['label']);
+ } else {
+ $link = $this->searchState
+ ->withTimeLimitations($option['after'], $option['before'])
+ ->getSearchLink($option['label'])
+ ;
+ $searchForm->addHTML($link);
+ }
+ $searchForm->addTagClose('li');
+ }
+ $searchForm->addTagClose('ul');
+
+ $searchForm->addTagClose('div');
+ }
+
+
+ /**
+ * Build the intro text for the search page
+ *
+ * @param string $query the search query
+ *
+ * @return string
+ */
+ protected function getSearchIntroHTML($query)
+ {
+ global $lang;
+
+ $intro = p_locale_xhtml('searchpage');
+
+ $queryPagename = $this->createPagenameFromQuery($this->parsedQuery);
+ $createQueryPageLink = html_wikilink($queryPagename . '?do=edit', $queryPagename);
+
+ $pagecreateinfo = '';
+ if (auth_quickaclcheck($queryPagename) >= AUTH_CREATE) {
+ $pagecreateinfo = sprintf($lang['searchcreatepage'], $createQueryPageLink);
+ }
+ $intro = str_replace(
+ array('@QUERY@', '@SEARCH@', '@CREATEPAGEINFO@'),
+ array(hsc(rawurlencode($query)), hsc($query), $pagecreateinfo),
+ $intro
+ );
+
+ return $intro;
+ }
+
+ /**
+ * Create a pagename based the parsed search query
+ *
+ * @param array $parsedQuery
+ *
+ * @return string pagename constructed from the parsed query
+ */
+ public function createPagenameFromQuery($parsedQuery)
+ {
+ $cleanedQuery = cleanID($parsedQuery['query']); // already strtolowered
+ if ($cleanedQuery === \dokuwiki\Utf8\PhpString::strtolower($parsedQuery['query'])) {
+ return ':' . $cleanedQuery;
+ }
+ $pagename = '';
+ if (!empty($parsedQuery['ns'])) {
+ $pagename .= ':' . cleanID($parsedQuery['ns'][0]);
+ }
+ $pagename .= ':' . cleanID(implode(' ' , $parsedQuery['highlight']));
+ return $pagename;
+ }
+
+ /**
+ * Build HTML for a list of pages with matching pagenames
+ *
+ * @param array $data search results
+ *
+ * @return string
+ */
+ protected function getPageLookupHTML($data)
+ {
+ if (empty($data)) {
+ return '';
+ }
+
+ global $lang;
+
+ $html = '<div class="search_quickresult">';
+ $html .= '<h2>' . $lang['quickhits'] . ':</h2>';
+ $html .= '<ul class="search_quickhits">';
+ foreach ($data as $id => $title) {
+ $name = null;
+ if (!useHeading('navigation') && $ns = getNS($id)) {
+ $name = shorten(noNS($id), ' (' . $ns . ')', 30);
+ }
+ $link = html_wikilink(':' . $id, $name);
+ $eventData = [
+ 'listItemContent' => [$link],
+ 'page' => $id,
+ ];
+ Event::createAndTrigger('SEARCH_RESULT_PAGELOOKUP', $eventData);
+ $html .= '<li>' . implode('', $eventData['listItemContent']) . '</li>';
+ }
+ $html .= '</ul> ';
+ //clear float (see http://www.complexspiral.com/publications/containing-floats/)
+ $html .= '<div class="clearer"></div>';
+ $html .= '</div>';
+
+ return $html;
+ }
+
+ /**
+ * Build HTML for fulltext search results or "no results" message
+ *
+ * @param array $data the results of the fulltext search
+ * @param array $highlight the terms to be highlighted in the results
+ *
+ * @return string
+ */
+ protected function getFulltextResultsHTML($data, $highlight)
+ {
+ global $lang;
+
+ if (empty($data)) {
+ return '<div class="nothing">' . $lang['nothingfound'] . '</div>';
+ }
+
+ $html = '<div class="search_fulltextresult">';
+ $html .= '<h2>' . $lang['search_fullresults'] . ':</h2>';
+
+ $html .= '<dl class="search_results">';
+ $num = 0;
+ $position = 0;
+
+ foreach ($data as $id => $cnt) {
+ $position += 1;
+ $resultLink = html_wikilink(':' . $id, null, $highlight);
+
+ $resultHeader = [$resultLink];
+
+
+ $restrictQueryToNSLink = $this->restrictQueryToNSLink(getNS($id));
+ if ($restrictQueryToNSLink) {
+ $resultHeader[] = $restrictQueryToNSLink;
+ }
+
+ $resultBody = [];
+ $mtime = filemtime(wikiFN($id));
+ $lastMod = '<span class="lastmod">' . $lang['lastmod'] . '</span> ';
+ $lastMod .= '<time datetime="' . date_iso8601($mtime) . '" title="' . dformat($mtime) . '">' .
+ dformat($mtime, '%f') .
+ '</time>';
+ $resultBody['meta'] = $lastMod;
+ if ($cnt !== 0) {
+ $num++;
+ $hits = '<span class="hits">' . $cnt . ' ' . $lang['hits'] . '</span>, ';
+ $resultBody['meta'] = $hits . $resultBody['meta'];
+ if ($num <= FT_SNIPPET_NUMBER) { // create snippets for the first number of matches only
+ $resultBody['snippet'] = ft_snippet($id, $highlight);
+ }
+ }
+
+ $eventData = [
+ 'resultHeader' => $resultHeader,
+ 'resultBody' => $resultBody,
+ 'page' => $id,
+ 'position' => $position,
+ ];
+ Event::createAndTrigger('SEARCH_RESULT_FULLPAGE', $eventData);
+ $html .= '<div class="search_fullpage_result">';
+ $html .= '<dt>' . implode(' ', $eventData['resultHeader']) . '</dt>';
+ foreach ($eventData['resultBody'] as $class => $htmlContent) {
+ $html .= "<dd class=\"$class\">$htmlContent</dd>";
+ }
+ $html .= '</div>';
+ }
+ $html .= '</dl>';
+
+ $html .= '</div>';
+
+ return $html;
+ }
+
+ /**
+ * create a link to restrict the current query to a namespace
+ *
+ * @param false|string $ns the namespace to which to restrict the query
+ *
+ * @return false|string
+ */
+ protected function restrictQueryToNSLink($ns)
+ {
+ if (!$ns) {
+ return false;
+ }
+ if (!$this->isNamespaceAssistanceAvailable($this->parsedQuery)) {
+ return false;
+ }
+ if (!empty($this->parsedQuery['ns']) && $this->parsedQuery['ns'][0] === $ns) {
+ return false;
+ }
+
+ $name = '@' . $ns;
+ return $this->searchState->withNamespace($ns)->getSearchLink($name);
+ }
+}