* Implement documentation search Signed-off-by: jolheiser <john.olheiser@gmail.com> Co-Authored-By: guillep2k <18600385+guillep2k@users.noreply.github.com>master
@@ -1,3 +1,4 @@ | |||
public/ | |||
templates/swagger/v1_json.tmpl | |||
themes/ | |||
resources/ |
@@ -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 | |||
version: 1.9.5 | |||
outputs: | |||
home: | |||
- HTML | |||
- RSS | |||
- JSON | |||
menu: | |||
page: | |||
- name: Website | |||
@@ -2,12 +2,12 @@ | |||
date: "2017-01-20T15:00:00+08:00" | |||
title: "Help" | |||
slug: "help" | |||
weight: 50 | |||
weight: 5 | |||
toc: false | |||
draft: false | |||
menu: | |||
sidebar: | |||
name: "Help" | |||
weight: 50 | |||
weight: 5 | |||
identifier: "help" | |||
--- |
@@ -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" | |||
title: "帮助" | |||
slug: "help" | |||
weight: 50 | |||
weight: 5 | |||
toc: false | |||
draft: false | |||
menu: | |||
sidebar: | |||
name: "帮助" | |||
weight: 50 | |||
weight: 5 | |||
identifier: "help" | |||
--- |
@@ -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" | |||
--- |
@@ -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. | |||
@@ -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. | |||
@@ -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. | |||
@@ -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. | |||
@@ -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 -}} |
@@ -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" . }} |