2016-01-28 23:27:15 +01:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2016 Martin Donath <martin.donath@squidfunk.com>
|
|
|
|
*
|
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
* of this software and associated documentation files (the "Software"), to
|
|
|
|
* deal in the Software without restriction, including without limitation the
|
|
|
|
* rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
|
|
|
|
* sell copies of the Software, and to permit persons to whom the Software is
|
|
|
|
* furnished to do so, subject to the following conditions:
|
|
|
|
*
|
|
|
|
* The above copyright notice and this permission notice shall be included in
|
|
|
|
* all copies or substantial portions of the Software.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
|
|
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
|
|
|
|
* IN THE SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
2016-08-07 18:01:56 +02:00
|
|
|
/* ----------------------------------------------------------------------------
|
|
|
|
* Imports
|
|
|
|
* ------------------------------------------------------------------------- */
|
2016-01-28 23:27:15 +01:00
|
|
|
|
2016-09-30 13:29:45 +02:00
|
|
|
import FastClick from "fastclick"
|
|
|
|
import lunr from "lunr"
|
|
|
|
|
|
|
|
// import Expander from "./components/expander"
|
2016-10-06 12:14:33 +02:00
|
|
|
|
2016-10-23 19:35:41 +10:00
|
|
|
import GithubSourceFacts from "./components/GithubSourceFacts"
|
2016-10-06 12:14:33 +02:00
|
|
|
import Material from "./components/Material"
|
2016-09-30 13:29:45 +02:00
|
|
|
|
|
|
|
// import Search from './components/search';
|
2016-01-28 23:27:15 +01:00
|
|
|
|
|
|
|
/* ----------------------------------------------------------------------------
|
2016-08-07 18:01:56 +02:00
|
|
|
* Application
|
2016-01-28 23:27:15 +01:00
|
|
|
* ------------------------------------------------------------------------- */
|
|
|
|
|
2016-10-06 12:14:33 +02:00
|
|
|
class Application {
|
|
|
|
|
2016-10-23 19:35:41 +10:00
|
|
|
/**
|
|
|
|
* Constructor.
|
|
|
|
*
|
|
|
|
* @constructor
|
|
|
|
* @param {object} config Configuration object
|
|
|
|
* @return {void}
|
|
|
|
*/
|
|
|
|
constructor(config) {
|
|
|
|
this.config = config
|
|
|
|
}
|
|
|
|
|
2016-10-06 12:14:33 +02:00
|
|
|
/**
|
|
|
|
* @return {void}
|
|
|
|
*/
|
|
|
|
initialize() {
|
|
|
|
const material = new Material()
|
|
|
|
material.initialize()
|
2016-10-23 19:35:41 +10:00
|
|
|
|
|
|
|
if (this.hasGithubRepo()) {
|
|
|
|
const githubSource = new GithubSourceFacts(
|
|
|
|
this.config.storage,
|
|
|
|
this.config.repo.url
|
|
|
|
)
|
|
|
|
githubSource.initialize()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Is this application about a Github repository?
|
|
|
|
*
|
|
|
|
* @return {bool} - true if `repo.icon` or `repo.url` contains 'github'
|
|
|
|
*/
|
|
|
|
hasGithubRepo() {
|
|
|
|
return this.config.repo.icon === "github"
|
|
|
|
|| this.config.repo.url.includes("github")
|
2016-10-06 12:14:33 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export default Application
|
|
|
|
|
|
|
|
// TODO: wrap in function call
|
|
|
|
// application module export
|
|
|
|
|
2016-01-28 23:27:15 +01:00
|
|
|
/* Initialize application upon DOM ready */
|
2016-09-30 13:29:45 +02:00
|
|
|
document.addEventListener("DOMContentLoaded", () => {
|
2016-01-28 23:27:15 +01:00
|
|
|
|
|
|
|
/* Test for iOS */
|
2016-09-30 13:29:45 +02:00
|
|
|
Modernizr.addTest("ios", () => {
|
|
|
|
return !!navigator.userAgent.match(/(iPad|iPhone|iPod)/g)
|
|
|
|
})
|
2016-01-28 23:27:15 +01:00
|
|
|
|
|
|
|
/* Test for web application context */
|
2016-09-30 13:29:45 +02:00
|
|
|
Modernizr.addTest("standalone", () => {
|
|
|
|
return !!navigator.standalone
|
|
|
|
})
|
2016-01-28 23:27:15 +01:00
|
|
|
|
|
|
|
/* Attack FastClick to mitigate 300ms delay on touch devices */
|
2016-09-30 13:29:45 +02:00
|
|
|
FastClick.attach(document.body)
|
2016-01-28 23:27:15 +01:00
|
|
|
|
2016-09-30 13:29:45 +02:00
|
|
|
const query = document.getElementById("query")
|
|
|
|
query.addEventListener("focus", () => {
|
2016-10-23 10:21:37 +02:00
|
|
|
document.querySelector(".md-search").dataset.mdLocked = ""
|
2016-09-30 13:29:45 +02:00
|
|
|
})
|
2016-09-02 00:33:45 +02:00
|
|
|
|
|
|
|
/* Intercept click on search mode toggle */
|
2016-09-30 13:29:45 +02:00
|
|
|
let offset = 0
|
|
|
|
const toggle = document.getElementById("search")
|
2016-10-23 10:21:37 +02:00
|
|
|
toggle.addEventListener("click", () => { // TODO: click may be the wrong event...
|
|
|
|
const list = document.body // classList md bla
|
2016-09-30 13:29:45 +02:00
|
|
|
const lock = !matchMedia("only screen and (min-width: 960px)").matches
|
2016-09-02 00:33:45 +02:00
|
|
|
|
|
|
|
/* Exiting search mode */
|
2016-10-23 10:21:37 +02:00
|
|
|
if (list.dataset.mdLocked) {
|
|
|
|
delete list.dataset.mdLocked
|
2016-09-02 00:33:45 +02:00
|
|
|
|
|
|
|
/* Scroll to former position, but wait for 100ms to prevent flashes
|
|
|
|
on iOS. A short timeout seems to do the trick */
|
|
|
|
if (lock)
|
2016-09-30 13:29:45 +02:00
|
|
|
setTimeout(() => {
|
|
|
|
window.scrollTo(0, offset)
|
|
|
|
}, 100)
|
2016-09-02 00:33:45 +02:00
|
|
|
|
|
|
|
/* Entering search mode */
|
|
|
|
} else {
|
2016-09-30 13:29:45 +02:00
|
|
|
offset = window.scrollY
|
2016-09-02 00:33:45 +02:00
|
|
|
|
|
|
|
/* First timeout: scroll to top after transition, to omit flickering */
|
|
|
|
if (lock)
|
2016-09-30 13:29:45 +02:00
|
|
|
setTimeout(() => {
|
|
|
|
window.scrollTo(0, 0)
|
|
|
|
}, 400)
|
2016-09-02 00:33:45 +02:00
|
|
|
|
|
|
|
/* Second timeout: Lock body after finishing transition and scrolling to
|
|
|
|
top and focus input field. Sadly, the focus event is not dispatched
|
|
|
|
on iOS Safari and there's nothing we can do about it. */
|
2016-09-30 13:29:45 +02:00
|
|
|
setTimeout(() => {
|
2016-09-02 00:33:45 +02:00
|
|
|
|
|
|
|
/* This additional check is necessary to handle fast subsequent clicks
|
|
|
|
on the toggle and the timeout to lock the body must be cancelled */
|
2016-10-23 10:21:37 +02:00
|
|
|
// if (ev.target.checked) {
|
|
|
|
if (lock)
|
|
|
|
list.dataset.mdLocked = ""
|
|
|
|
setTimeout(() => {
|
|
|
|
document.getElementById("query").focus()
|
|
|
|
}, 200)
|
|
|
|
// }
|
2016-09-30 13:29:45 +02:00
|
|
|
}, 450)
|
2016-09-02 00:33:45 +02:00
|
|
|
}
|
2016-09-30 13:29:45 +02:00
|
|
|
})
|
2016-09-02 00:33:45 +02:00
|
|
|
|
2016-10-23 10:21:37 +02:00
|
|
|
// TODO: only do this on MOBILE and TABLET
|
|
|
|
const toggleSearchClose = document.querySelector(".md-search__icon")
|
|
|
|
toggleSearchClose.setAttribute("for", "search") // TODO: override query with search, when on mobile!!!
|
|
|
|
// toggleSearchClose.addEventListener("click", ev => {
|
|
|
|
// ev.preventDefault()
|
|
|
|
// // ev.target
|
|
|
|
//
|
|
|
|
// const search = document.getElementById("search")
|
|
|
|
// search.checked = false
|
|
|
|
// })
|
|
|
|
|
2016-09-24 18:52:37 +02:00
|
|
|
// var toc = new Sidebar('.md-sidebar--secondary');
|
|
|
|
// toc.listen();
|
|
|
|
|
2016-09-30 13:29:45 +02:00
|
|
|
const toggles =
|
|
|
|
document.querySelectorAll(".md-nav__item--nested > .md-nav__link");
|
|
|
|
[].forEach.call(toggles, togglex => {
|
|
|
|
const nav = togglex.nextElementSibling
|
2016-09-24 18:52:37 +02:00
|
|
|
|
|
|
|
// 1.
|
|
|
|
|
2016-09-30 13:29:45 +02:00
|
|
|
nav.style.maxHeight = `${nav.getBoundingClientRect().height}px`
|
2016-09-24 18:52:37 +02:00
|
|
|
|
2016-09-30 13:29:45 +02:00
|
|
|
togglex.addEventListener("click", () => {
|
|
|
|
const first = nav.getBoundingClientRect().height
|
2016-09-24 18:52:37 +02:00
|
|
|
if (first) {
|
2016-09-30 13:29:45 +02:00
|
|
|
// console.log('closing');
|
|
|
|
nav.style.maxHeight = `${first}px` // reset!
|
2016-09-24 18:52:37 +02:00
|
|
|
requestAnimationFrame(() => {
|
|
|
|
|
2016-09-30 13:29:45 +02:00
|
|
|
nav.classList.add("md-nav--transitioning")
|
|
|
|
nav.style.maxHeight = "0px"
|
|
|
|
})
|
2016-09-24 18:52:37 +02:00
|
|
|
} else {
|
2016-09-30 13:29:45 +02:00
|
|
|
// console.log('opening');
|
2016-09-24 18:52:37 +02:00
|
|
|
|
|
|
|
/* Toggle and read height */
|
2016-09-30 13:29:45 +02:00
|
|
|
nav.style.maxHeight = ""
|
2016-09-24 18:52:37 +02:00
|
|
|
|
2016-09-30 13:29:45 +02:00
|
|
|
nav.classList.add("md-nav--toggled")
|
|
|
|
const last = nav.getBoundingClientRect().height
|
|
|
|
nav.classList.remove("md-nav--toggled")
|
2016-09-24 18:52:37 +02:00
|
|
|
|
|
|
|
// Initial state
|
2016-09-30 13:29:45 +02:00
|
|
|
nav.style.maxHeight = "0px"
|
2016-09-24 18:52:37 +02:00
|
|
|
|
|
|
|
/* Enable animations */
|
|
|
|
requestAnimationFrame(() => {
|
2016-09-30 13:29:45 +02:00
|
|
|
nav.classList.add("md-nav--transitioning")
|
|
|
|
nav.style.maxHeight = `${last}px`
|
|
|
|
})
|
2016-09-24 18:52:37 +02:00
|
|
|
}
|
|
|
|
|
2016-09-30 13:29:45 +02:00
|
|
|
})
|
2016-09-24 18:52:37 +02:00
|
|
|
|
|
|
|
// Capture the end with transitionend
|
2016-09-30 13:29:45 +02:00
|
|
|
nav.addEventListener("transitionend", ev => {
|
|
|
|
ev.target.classList.remove("md-nav--transitioning")
|
|
|
|
if (ev.target.getBoundingClientRect().height > 0) {
|
|
|
|
ev.target.style.maxHeight = "100%"
|
2016-09-24 18:52:37 +02:00
|
|
|
}
|
2016-09-30 13:29:45 +02:00
|
|
|
})
|
|
|
|
})
|
2016-09-23 20:26:27 +02:00
|
|
|
|
2016-09-23 11:56:25 +02:00
|
|
|
// setTimeout(function() {
|
2016-09-30 13:29:45 +02:00
|
|
|
fetch("/mkdocs/search_index.json") // TODO: prepend BASE URL!!!
|
|
|
|
.then(response => {
|
|
|
|
return response.json()
|
|
|
|
})
|
|
|
|
.then(data => {
|
|
|
|
// console.log(data)
|
|
|
|
|
|
|
|
/* Create index */
|
2016-10-23 10:21:37 +02:00
|
|
|
const index = lunr(function() {
|
2016-09-30 13:29:45 +02:00
|
|
|
/* eslint-disable no-invalid-this, lines-around-comment */
|
2016-10-06 12:14:33 +02:00
|
|
|
this.field("title", { boost: 10 })
|
2016-09-30 13:29:45 +02:00
|
|
|
this.field("text")
|
|
|
|
this.ref("location")
|
|
|
|
/* eslint-enable no-invalid-this, lines-around-comment */
|
|
|
|
})
|
2016-09-23 17:46:16 +02:00
|
|
|
|
2016-09-30 13:29:45 +02:00
|
|
|
/* Index articles */
|
|
|
|
const articles = {}
|
|
|
|
data.docs.forEach(article => {
|
|
|
|
|
|
|
|
// TODO: match for two whitespaces, then replace unnecessary whitespace after string
|
|
|
|
article.text = article.text.replace(/\s(\.,\:)\s/gi, (string, g1) => {
|
|
|
|
return `${g1} `
|
|
|
|
})
|
|
|
|
// TODO: window.baseUrl sucks...
|
|
|
|
article.location = window.baseUrl + article.location
|
|
|
|
articles[article.location] = article
|
|
|
|
index.add(article)
|
2016-09-23 17:46:16 +02:00
|
|
|
})
|
2016-09-23 11:56:25 +02:00
|
|
|
|
2016-09-30 13:29:45 +02:00
|
|
|
/* Register keyhandler to execute search on key up */
|
|
|
|
const queryx = document.getElementById("query")
|
|
|
|
queryx.addEventListener("keyup", () => {
|
|
|
|
const container = document.querySelector(".md-search-result__list")
|
|
|
|
while (container.firstChild)
|
|
|
|
container.removeChild(container.firstChild)
|
|
|
|
|
|
|
|
// /* Abort, if the query is empty */
|
|
|
|
// var bar = document.querySelector('.bar.search');
|
|
|
|
// if (!query.value.length) {
|
|
|
|
// while (meta.firstChild)
|
|
|
|
// meta.removeChild(meta.firstChild);
|
|
|
|
//
|
|
|
|
// /* Restore state */
|
|
|
|
// bar.classList.remove('non-empty');
|
|
|
|
// return;
|
|
|
|
// }
|
|
|
|
|
|
|
|
/* Show reset button */
|
|
|
|
// bar.classList.add('non-empty');
|
|
|
|
|
|
|
|
/* Execute search */
|
|
|
|
const results = index.search(query.value)
|
|
|
|
results.forEach(result => {
|
|
|
|
const article = articles[result.ref]
|
|
|
|
|
|
|
|
/* Create a link referring to the article */
|
|
|
|
const link = document.createElement("a")
|
|
|
|
link.classList.add("md-search-result__link")
|
|
|
|
link.href = article.location
|
|
|
|
|
|
|
|
// /* Create article container */
|
|
|
|
const li = document.createElement("li")
|
|
|
|
li.classList.add("md-search-result__item")
|
|
|
|
li.appendChild(link)
|
|
|
|
|
|
|
|
/* Create title element */
|
|
|
|
const title = document.createElement("div")
|
|
|
|
title.classList.add("md-search-result__title")
|
|
|
|
|
|
|
|
// article.title.split(//)
|
|
|
|
|
|
|
|
title.innerHTML = article.title
|
|
|
|
link.appendChild(title)
|
|
|
|
|
2016-10-23 10:21:37 +02:00
|
|
|
/* Truncate a string after the given number of characters */
|
|
|
|
const truncate = function(string, n) {
|
|
|
|
let i = n
|
|
|
|
if (string.length > i) {
|
|
|
|
while (string[i] !== " " && --i > 0);
|
|
|
|
return `${string.substring(0, i)}…`
|
|
|
|
}
|
|
|
|
return string
|
|
|
|
}
|
|
|
|
|
2016-09-30 13:29:45 +02:00
|
|
|
/* Create text element */
|
|
|
|
const text = document.createElement("p")
|
|
|
|
text.classList.add("md-search-result__description")
|
2016-10-23 10:21:37 +02:00
|
|
|
text.innerHTML = truncate(article.text) // .truncate(140);
|
|
|
|
text.innerHTML = truncate(article.text, 140) // .truncate(140);
|
2016-09-30 13:29:45 +02:00
|
|
|
link.appendChild(text)
|
|
|
|
|
|
|
|
container.appendChild(li)
|
|
|
|
})
|
|
|
|
|
|
|
|
/* Show number of search results */
|
|
|
|
// var number = document.createElement('strong');
|
|
|
|
|
|
|
|
const meta = document.querySelector(".md-search-result__meta")
|
|
|
|
meta.innerHTML = `${results.length} search result${
|
|
|
|
results.length !== 1
|
|
|
|
? "s"
|
|
|
|
: ""}`
|
|
|
|
|
|
|
|
/* Update number */
|
|
|
|
// while (meta.firstChild)
|
|
|
|
// meta.removeChild(meta.firstChild);
|
|
|
|
// meta.appendChild(number);
|
|
|
|
})
|
2016-09-23 11:56:25 +02:00
|
|
|
|
2016-09-30 13:29:45 +02:00
|
|
|
// setTimeout(function() {
|
|
|
|
// li.classList.remove('md-source__fact--hidden');
|
|
|
|
// }, 100);
|
2016-09-23 11:56:25 +02:00
|
|
|
|
2016-09-30 13:29:45 +02:00
|
|
|
})
|
|
|
|
.catch(() => {
|
|
|
|
// console.log("parsing failed", ex)
|
|
|
|
})
|
2016-09-23 11:56:25 +02:00
|
|
|
// }, 1000);
|
2016-09-02 00:33:45 +02:00
|
|
|
|
2016-10-23 10:21:37 +02:00
|
|
|
fetch(
|
|
|
|
"https://api.github.com/repos/squidfunk/mkdocs-material/releases/latest")
|
|
|
|
.then(response => {
|
|
|
|
return response.json()
|
|
|
|
})
|
|
|
|
// .then(data => {
|
|
|
|
// // console.log(data)
|
|
|
|
// })
|
|
|
|
|
2016-09-30 13:29:45 +02:00
|
|
|
})
|