mirror of
https://codeberg.org/forgejo/forgejo.git
synced 2024-12-26 01:40:36 +00:00
Implement documentation search (#8937)
* Implement documentation search Signed-off-by: jolheiser <john.olheiser@gmail.com> Co-Authored-By: guillep2k <18600385+guillep2k@users.noreply.github.com>
This commit is contained in:
parent
afe50873a5
commit
3b0303a4fc
13 changed files with 362 additions and 4 deletions
1
docs/.gitignore
vendored
1
docs/.gitignore
vendored
|
@ -1,3 +1,4 @@
|
||||||
public/
|
public/
|
||||||
templates/swagger/v1_json.tmpl
|
templates/swagger/v1_json.tmpl
|
||||||
themes/
|
themes/
|
||||||
|
resources/
|
||||||
|
|
176
docs/assets/js/search.js
Normal file
176
docs/assets/js/search.js
Normal file
|
@ -0,0 +1,176 @@
|
||||||
|
function ready(fn) {
|
||||||
|
if (document.readyState != 'loading') {
|
||||||
|
fn();
|
||||||
|
} else {
|
||||||
|
document.addEventListener('DOMContentLoaded', fn);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ready(doSearch);
|
||||||
|
|
||||||
|
const summaryInclude = 60;
|
||||||
|
const fuseOptions = {
|
||||||
|
shouldSort: true,
|
||||||
|
includeMatches: true,
|
||||||
|
matchAllTokens: true,
|
||||||
|
threshold: 0.0, // for parsing diacritics
|
||||||
|
tokenize: true,
|
||||||
|
location: 0,
|
||||||
|
distance: 100,
|
||||||
|
maxPatternLength: 32,
|
||||||
|
minMatchCharLength: 1,
|
||||||
|
keys: [{
|
||||||
|
name: "title",
|
||||||
|
weight: 0.8
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "contents",
|
||||||
|
weight: 0.5
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "tags",
|
||||||
|
weight: 0.3
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "categories",
|
||||||
|
weight: 0.3
|
||||||
|
}
|
||||||
|
]
|
||||||
|
};
|
||||||
|
|
||||||
|
function param(name) {
|
||||||
|
return decodeURIComponent((location.search.split(name + '=')[1] || '').split('&')[0]).replace(/\+/g, ' ');
|
||||||
|
}
|
||||||
|
|
||||||
|
let searchQuery = param("s");
|
||||||
|
|
||||||
|
function doSearch() {
|
||||||
|
if (searchQuery) {
|
||||||
|
document.getElementById("search-query").value = searchQuery;
|
||||||
|
executeSearch(searchQuery);
|
||||||
|
} else {
|
||||||
|
const para = document.createElement("P");
|
||||||
|
para.innerText = "Please enter a word or phrase above";
|
||||||
|
document.getElementById("search-results").appendChild(para);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function getJSON(url, fn) {
|
||||||
|
const request = new XMLHttpRequest();
|
||||||
|
request.open('GET', url, true);
|
||||||
|
request.onload = function () {
|
||||||
|
if (request.status >= 200 && request.status < 400) {
|
||||||
|
const data = JSON.parse(request.responseText);
|
||||||
|
fn(data);
|
||||||
|
} else {
|
||||||
|
console.log("Target reached on " + url + " with error " + request.status);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
request.onerror = function () {
|
||||||
|
console.log("Connection error " + request.status);
|
||||||
|
};
|
||||||
|
request.send();
|
||||||
|
}
|
||||||
|
|
||||||
|
function executeSearch(searchQuery) {
|
||||||
|
getJSON("/" + document.LANG + "/index.json", function (data) {
|
||||||
|
const pages = data;
|
||||||
|
const fuse = new Fuse(pages, fuseOptions);
|
||||||
|
const result = fuse.search(searchQuery);
|
||||||
|
console.log({
|
||||||
|
"matches": result
|
||||||
|
});
|
||||||
|
document.getElementById("search-results").innerHTML = "";
|
||||||
|
if (result.length > 0) {
|
||||||
|
populateResults(result);
|
||||||
|
} else {
|
||||||
|
const para = document.createElement("P");
|
||||||
|
para.innerText = "No matches found";
|
||||||
|
document.getElementById("search-results").appendChild(para);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function populateResults(result) {
|
||||||
|
result.forEach(function (value, key) {
|
||||||
|
const content = value.item.contents;
|
||||||
|
let snippet = "";
|
||||||
|
const snippetHighlights = [];
|
||||||
|
if (fuseOptions.tokenize) {
|
||||||
|
snippetHighlights.push(searchQuery);
|
||||||
|
value.matches.forEach(function (mvalue) {
|
||||||
|
if (mvalue.key === "tags" || mvalue.key === "categories") {
|
||||||
|
snippetHighlights.push(mvalue.value);
|
||||||
|
} else if (mvalue.key === "contents") {
|
||||||
|
const ind = content.toLowerCase().indexOf(searchQuery.toLowerCase());
|
||||||
|
const start = ind - summaryInclude > 0 ? ind - summaryInclude : 0;
|
||||||
|
const end = ind + searchQuery.length + summaryInclude < content.length ? ind + searchQuery.length + summaryInclude : content.length;
|
||||||
|
snippet += content.substring(start, end);
|
||||||
|
if (ind > -1) {
|
||||||
|
snippetHighlights.push(content.substring(ind, ind + searchQuery.length))
|
||||||
|
} else {
|
||||||
|
snippetHighlights.push(mvalue.value.substring(mvalue.indices[0][0], mvalue.indices[0][1] - mvalue.indices[0][0] + 1));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (snippet.length < 1) {
|
||||||
|
snippet += content.substring(0, summaryInclude * 2);
|
||||||
|
}
|
||||||
|
//pull template from hugo templarte definition
|
||||||
|
const templateDefinition = document.getElementById("search-result-template").innerHTML;
|
||||||
|
//replace values
|
||||||
|
const output = render(templateDefinition, {
|
||||||
|
key: key,
|
||||||
|
title: value.item.title,
|
||||||
|
link: value.item.permalink,
|
||||||
|
tags: value.item.tags,
|
||||||
|
categories: value.item.categories,
|
||||||
|
snippet: snippet
|
||||||
|
});
|
||||||
|
document.getElementById("search-results").appendChild(htmlToElement(output));
|
||||||
|
|
||||||
|
snippetHighlights.forEach(function (snipvalue) {
|
||||||
|
new Mark(document.getElementById("summary-" + key)).mark(snipvalue);
|
||||||
|
});
|
||||||
|
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function render(templateString, data) {
|
||||||
|
let conditionalMatches, copy;
|
||||||
|
const conditionalPattern = /\$\{\s*isset ([a-zA-Z]*) \s*\}(.*)\$\{\s*end\s*}/g;
|
||||||
|
//since loop below depends on re.lastInxdex, we use a copy to capture any manipulations whilst inside the loop
|
||||||
|
copy = templateString;
|
||||||
|
while ((conditionalMatches = conditionalPattern.exec(templateString)) !== null) {
|
||||||
|
if (data[conditionalMatches[1]]) {
|
||||||
|
//valid key, remove conditionals, leave content.
|
||||||
|
copy = copy.replace(conditionalMatches[0], conditionalMatches[2]);
|
||||||
|
} else {
|
||||||
|
//not valid, remove entire section
|
||||||
|
copy = copy.replace(conditionalMatches[0], '');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
templateString = copy;
|
||||||
|
//now any conditionals removed we can do simple substitution
|
||||||
|
let key, find, re;
|
||||||
|
for (key in data) {
|
||||||
|
find = '\\$\\{\\s*' + key + '\\s*\\}';
|
||||||
|
re = new RegExp(find, 'g');
|
||||||
|
templateString = templateString.replace(re, data[key]);
|
||||||
|
}
|
||||||
|
return templateString;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* By Mark Amery: https://stackoverflow.com/a/35385518
|
||||||
|
* @param {String} HTML representing a single element
|
||||||
|
* @return {Element}
|
||||||
|
*/
|
||||||
|
function htmlToElement(html) {
|
||||||
|
const template = document.createElement('template');
|
||||||
|
html = html.trim(); // Never return a text node of whitespace as the result
|
||||||
|
template.innerHTML = html;
|
||||||
|
return template.content.firstChild;
|
||||||
|
}
|
|
@ -20,6 +20,12 @@ params:
|
||||||
website: https://docs.gitea.io
|
website: https://docs.gitea.io
|
||||||
version: 1.9.5
|
version: 1.9.5
|
||||||
|
|
||||||
|
outputs:
|
||||||
|
home:
|
||||||
|
- HTML
|
||||||
|
- RSS
|
||||||
|
- JSON
|
||||||
|
|
||||||
menu:
|
menu:
|
||||||
page:
|
page:
|
||||||
- name: Website
|
- name: Website
|
||||||
|
|
|
@ -2,12 +2,12 @@
|
||||||
date: "2017-01-20T15:00:00+08:00"
|
date: "2017-01-20T15:00:00+08:00"
|
||||||
title: "Help"
|
title: "Help"
|
||||||
slug: "help"
|
slug: "help"
|
||||||
weight: 50
|
weight: 5
|
||||||
toc: false
|
toc: false
|
||||||
draft: false
|
draft: false
|
||||||
menu:
|
menu:
|
||||||
sidebar:
|
sidebar:
|
||||||
name: "Help"
|
name: "Help"
|
||||||
weight: 50
|
weight: 5
|
||||||
identifier: "help"
|
identifier: "help"
|
||||||
---
|
---
|
||||||
|
|
13
docs/content/doc/help.fr-fr.md
Normal file
13
docs/content/doc/help.fr-fr.md
Normal file
|
@ -0,0 +1,13 @@
|
||||||
|
---
|
||||||
|
date: "2017-01-20T15:00:00+08:00"
|
||||||
|
title: "Aide"
|
||||||
|
slug: "help"
|
||||||
|
weight: 5
|
||||||
|
toc: false
|
||||||
|
draft: false
|
||||||
|
menu:
|
||||||
|
sidebar:
|
||||||
|
name: "Aide"
|
||||||
|
weight: 5
|
||||||
|
identifier: "help"
|
||||||
|
---
|
|
@ -2,12 +2,12 @@
|
||||||
date: "2017-01-20T15:00:00+08:00"
|
date: "2017-01-20T15:00:00+08:00"
|
||||||
title: "帮助"
|
title: "帮助"
|
||||||
slug: "help"
|
slug: "help"
|
||||||
weight: 50
|
weight: 5
|
||||||
toc: false
|
toc: false
|
||||||
draft: false
|
draft: false
|
||||||
menu:
|
menu:
|
||||||
sidebar:
|
sidebar:
|
||||||
name: "帮助"
|
name: "帮助"
|
||||||
weight: 50
|
weight: 5
|
||||||
identifier: "help"
|
identifier: "help"
|
||||||
---
|
---
|
||||||
|
|
13
docs/content/doc/help.zh-tw.md
Normal file
13
docs/content/doc/help.zh-tw.md
Normal file
|
@ -0,0 +1,13 @@
|
||||||
|
---
|
||||||
|
date: "2017-01-20T15:00:00+08:00"
|
||||||
|
title: "救命"
|
||||||
|
slug: "help"
|
||||||
|
weight: 5
|
||||||
|
toc: false
|
||||||
|
draft: false
|
||||||
|
menu:
|
||||||
|
sidebar:
|
||||||
|
name: "救命"
|
||||||
|
weight: 5
|
||||||
|
identifier: "help"
|
||||||
|
---
|
25
docs/content/doc/help/search.en-us.md
Normal file
25
docs/content/doc/help/search.en-us.md
Normal file
|
@ -0,0 +1,25 @@
|
||||||
|
---
|
||||||
|
date: "2019-11-12T16:00:00+02:00"
|
||||||
|
title: "Search"
|
||||||
|
slug: "search"
|
||||||
|
weight: 4
|
||||||
|
toc: true
|
||||||
|
draft: false
|
||||||
|
menu:
|
||||||
|
sidebar:
|
||||||
|
parent: "help"
|
||||||
|
name: "Search"
|
||||||
|
weight: 4
|
||||||
|
identifier: "search"
|
||||||
|
sitemap:
|
||||||
|
priority : 0.1
|
||||||
|
layout: "search"
|
||||||
|
---
|
||||||
|
|
||||||
|
|
||||||
|
This file exists solely to respond to /search URL with the related `search` layout template.
|
||||||
|
|
||||||
|
No content shown here is rendered, all content is based in the template layouts/doc/search.html
|
||||||
|
|
||||||
|
Setting a very low sitemap priority will tell search engines this is not important content.
|
||||||
|
|
25
docs/content/doc/help/search.fr-fr.md
Normal file
25
docs/content/doc/help/search.fr-fr.md
Normal file
|
@ -0,0 +1,25 @@
|
||||||
|
---
|
||||||
|
date: "2019-11-12T16:00:00+02:00"
|
||||||
|
title: "Chercher"
|
||||||
|
slug: "search"
|
||||||
|
weight: 4
|
||||||
|
toc: true
|
||||||
|
draft: false
|
||||||
|
menu:
|
||||||
|
sidebar:
|
||||||
|
parent: "help"
|
||||||
|
name: "Chercher"
|
||||||
|
weight: 4
|
||||||
|
identifier: "search"
|
||||||
|
sitemap:
|
||||||
|
priority : 0.1
|
||||||
|
layout: "search"
|
||||||
|
---
|
||||||
|
|
||||||
|
|
||||||
|
This file exists solely to respond to /search URL with the related `search` layout template.
|
||||||
|
|
||||||
|
No content shown here is rendered, all content is based in the template layouts/doc/search.html
|
||||||
|
|
||||||
|
Setting a very low sitemap priority will tell search engines this is not important content.
|
||||||
|
|
25
docs/content/doc/help/search.zh-cn.md
Normal file
25
docs/content/doc/help/search.zh-cn.md
Normal file
|
@ -0,0 +1,25 @@
|
||||||
|
---
|
||||||
|
date: "2019-11-12T16:00:00+02:00"
|
||||||
|
title: "搜索"
|
||||||
|
slug: "search"
|
||||||
|
weight: 4
|
||||||
|
toc: true
|
||||||
|
draft: false
|
||||||
|
menu:
|
||||||
|
sidebar:
|
||||||
|
parent: "help"
|
||||||
|
name: "搜索"
|
||||||
|
weight: 4
|
||||||
|
identifier: "search"
|
||||||
|
sitemap:
|
||||||
|
priority : 0.1
|
||||||
|
layout: "search"
|
||||||
|
---
|
||||||
|
|
||||||
|
|
||||||
|
This file exists solely to respond to /search URL with the related `search` layout template.
|
||||||
|
|
||||||
|
No content shown here is rendered, all content is based in the template layouts/doc/search.html
|
||||||
|
|
||||||
|
Setting a very low sitemap priority will tell search engines this is not important content.
|
||||||
|
|
25
docs/content/doc/help/search.zh-tw.md
Normal file
25
docs/content/doc/help/search.zh-tw.md
Normal file
|
@ -0,0 +1,25 @@
|
||||||
|
---
|
||||||
|
date: "2019-11-12T16:00:00+02:00"
|
||||||
|
title: "搜索"
|
||||||
|
slug: "search"
|
||||||
|
weight: 4
|
||||||
|
toc: true
|
||||||
|
draft: false
|
||||||
|
menu:
|
||||||
|
sidebar:
|
||||||
|
parent: "help"
|
||||||
|
name: "搜索"
|
||||||
|
weight: 4
|
||||||
|
identifier: "search"
|
||||||
|
sitemap:
|
||||||
|
priority : 0.1
|
||||||
|
layout: "search"
|
||||||
|
---
|
||||||
|
|
||||||
|
|
||||||
|
This file exists solely to respond to /search URL with the related `search` layout template.
|
||||||
|
|
||||||
|
No content shown here is rendered, all content is based in the template layouts/doc/search.html
|
||||||
|
|
||||||
|
Setting a very low sitemap priority will tell search engines this is not important content.
|
||||||
|
|
5
docs/layouts/_default/index.json
Normal file
5
docs/layouts/_default/index.json
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
{{- $.Scratch.Add "index" slice -}}
|
||||||
|
{{- range .Site.RegularPages -}}
|
||||||
|
{{- $.Scratch.Add "index" (dict "title" .Title "tags" .Params.tags "categories" .Params.categories "contents" .Plain "permalink" .Permalink) -}}
|
||||||
|
{{- end -}}
|
||||||
|
{{- $.Scratch.Get "index" | jsonify -}}
|
44
docs/layouts/doc/search.html
Normal file
44
docs/layouts/doc/search.html
Normal file
|
@ -0,0 +1,44 @@
|
||||||
|
{{ partial "header.html" . }}
|
||||||
|
{{ partial "navbar.html" . }}
|
||||||
|
|
||||||
|
<section class="section">
|
||||||
|
<div class="container is-centered page">
|
||||||
|
<div class="columns">
|
||||||
|
<div class="column is-one-quarter">
|
||||||
|
{{ partial "menu" . }}
|
||||||
|
</div>
|
||||||
|
<div class="column">
|
||||||
|
<div class=" content">
|
||||||
|
<section class="resume-section p-3 p-lg-5 d-flex flex-column">
|
||||||
|
<div class="my-auto" >
|
||||||
|
<form action="{{ "search" | absLangURL }}">
|
||||||
|
<label>Search:
|
||||||
|
<input id="search-query" name="s"/>
|
||||||
|
</label>
|
||||||
|
</form>
|
||||||
|
<br/>
|
||||||
|
<div id="search-results"></div>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
<!-- this template is sucked in by search.js and appended to the search-results div above. So editing here will adjust style -->
|
||||||
|
<script id="search-result-template" type="text/x-js-template">
|
||||||
|
<div id="summary-${key}">
|
||||||
|
<h4><a href="${link}">${title}</a></h4>
|
||||||
|
<p>${snippet}</p>
|
||||||
|
${ isset tags }<p>Tags: ${tags}</p>${ end }
|
||||||
|
${ isset categories }<p>Categories: ${categories}</p>${ end }
|
||||||
|
<hr/>
|
||||||
|
</div>
|
||||||
|
</script>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<script src="https://cdnjs.cloudflare.com/ajax/libs/fuse.js/3.4.5/fuse.min.js"></script>
|
||||||
|
<script src="https://cdnjs.cloudflare.com/ajax/libs/mark.js/8.11.1/mark.min.js"></script>
|
||||||
|
<script>document.LANG = "{{ .Language.Lang }}";</script>
|
||||||
|
{{ $script := resources.Get "js/search.js" | minify | fingerprint -}}
|
||||||
|
<script src="{{ $script.Permalink }}" {{ printf "integrity=%q" $script.Data.Integrity | safeHTMLAttr }}></script>
|
||||||
|
{{ partial "footer.html" . }}
|
Loading…
Reference in a new issue